Files
aitbc/apps/exchange/exchange_api.py
aitbc 9db720add8
Some checks failed
Documentation Validation / validate-docs (push) Has been cancelled
Python Tests / test-python (push) Has been cancelled
API Endpoint Tests / test-api-endpoints (push) Has been cancelled
CLI Tests / test-cli (push) Has been cancelled
Integration Tests / test-service-integration (push) Has been cancelled
Package Tests / test-python-packages (map[name:aitbc-agent-sdk path:packages/py/aitbc-agent-sdk]) (push) Has been cancelled
Package Tests / test-python-packages (map[name:aitbc-core path:packages/py/aitbc-core]) (push) Has been cancelled
Package Tests / test-python-packages (map[name:aitbc-crypto path:packages/py/aitbc-crypto]) (push) Has been cancelled
Package Tests / test-python-packages (map[name:aitbc-sdk path:packages/py/aitbc-sdk]) (push) Has been cancelled
Package Tests / test-javascript-packages (map[name:aitbc-sdk-js path:packages/js/aitbc-sdk]) (push) Has been cancelled
Package Tests / test-javascript-packages (map[name:aitbc-token path:packages/solidity/aitbc-token]) (push) Has been cancelled
Security Scanning / security-scan (push) Has been cancelled
Systemd Sync / sync-systemd (push) Has been cancelled
docs: add code quality and type checking workflows to master index
- Add Code Quality Module section with pre-commit hooks and quality checks
- Add Type Checking CI/CD Module section with MyPy workflow and coverage
- Update README with code quality achievements and project structure
- Migrate FastAPI apps from deprecated on_event to lifespan context manager
- Update pyproject.toml files to reference consolidated dependencies
- Remove unused app.py import in coordinator-api
- Add type hints to agent
2026-03-31 21:45:43 +02:00

362 lines
10 KiB
Python
Executable File

#!/usr/bin/env python3
"""
FastAPI backend for the AITBC Trade Exchange
"""
from datetime import datetime, timedelta
from typing import List, Optional
from fastapi import FastAPI, Depends, HTTPException, status, Header
from fastapi.middleware.cors import CORSMiddleware
from pydantic import BaseModel
from sqlalchemy import desc, func, and_
from sqlalchemy.orm import Session
import hashlib
import time
from typing import Annotated
from contextlib import asynccontextmanager
from database import init_db, get_db_session
from models import User, Order, Trade, Balance
@asynccontextmanager
async def lifespan(app: FastAPI):
# Startup
init_db()
yield
# Shutdown (cleanup if needed)
pass
# Initialize FastAPI app
app = FastAPI(title="AITBC Trade Exchange API", version="1.0.0", lifespan=lifespan)
# In-memory session storage (use Redis in production)
user_sessions = {}
def verify_session_token(token: str = Header(..., alias="Authorization")) -> int:
"""Verify session token and return user_id"""
# Remove "Bearer " prefix if present
if token.startswith("Bearer "):
token = token[7:]
if token not in user_sessions:
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail="Invalid or expired token"
)
session = user_sessions[token]
# Check if expired
if int(time.time()) > session["expires_at"]:
del user_sessions[token]
raise HTTPException(
status_code=status.HTTP_401_UNAUTHORIZED,
detail="Token expired"
)
return session["user_id"]
def optional_auth(token: Optional[str] = Header(None, alias="Authorization")) -> Optional[int]:
"""Optional authentication - returns user_id if token is valid, None otherwise"""
if not token:
return None
try:
return verify_session_token(token)
except HTTPException:
return None
# Type annotations for dependencies
UserDep = Annotated[int, Depends(verify_session_token)]
OptionalUserDep = Annotated[Optional[int], Depends(optional_auth)]
# Add CORS middleware
app.add_middleware(
CORSMiddleware,
allow_origins=[
"http://localhost:3000",
"http://localhost:8080",
"http://localhost:8000",
"http://localhost:3003"
],
allow_credentials=True,
allow_methods=["GET", "POST", "PUT", "DELETE", "OPTIONS"],
allow_headers=["*"], # Allow all headers for auth tokens
)
# Pydantic models
class OrderCreate(BaseModel):
order_type: str # 'BUY' or 'SELL'
amount: float
price: float
class OrderResponse(BaseModel):
id: int
order_type: str
amount: float
price: float
total: float
filled: float
remaining: float
status: str
created_at: datetime
class Config:
from_attributes = True
class TradeResponse(BaseModel):
id: int
amount: float
price: float
total: float
created_at: datetime
class Config:
from_attributes = True
class OrderBookResponse(BaseModel):
buys: List[OrderResponse]
sells: List[OrderResponse]
# Create mock data if database is empty
db = get_db_session()
try:
# Check if we have any trades
if db.query(Trade).count() == 0:
create_mock_trades(db)
finally:
db.close()
def create_mock_trades(db: Session):
"""Create some mock trades for demonstration"""
import random
# Create mock trades over the last hour
now = datetime.utcnow()
trades = []
for i in range(20):
# Generate random trade data
amount = random.uniform(10, 500)
price = random.uniform(0.000009, 0.000012)
total = amount * price
trade = Trade(
buyer_id=1, # Mock user ID
seller_id=2, # Mock user ID
order_id=1, # Mock order ID
amount=amount,
price=price,
total=total,
trade_hash=f"mock_tx_{i:04d}",
created_at=now - timedelta(minutes=random.randint(0, 60))
)
trades.append(trade)
db.add_all(trades)
db.commit()
print(f"Created {len(trades)} mock trades")
@app.get("/api/trades/recent", response_model=List[TradeResponse])
def get_recent_trades(limit: int = 20, db: Session = Depends(get_db_session)):
"""Get recent trades"""
trades = db.query(Trade).order_by(desc(Trade.created_at)).limit(limit).all()
return trades
@app.get("/api/orders", response_model=List[OrderResponse])
def get_orders(
status_filter: Optional[str] = None,
user_only: bool = False,
db: Session = Depends(get_db_session),
user_id: OptionalUserDep = None
):
"""Get all orders with optional status filter"""
query = db.query(Order)
# Filter by user if requested and authenticated
if user_only and user_id:
query = query.filter(Order.user_id == user_id)
if status_filter:
query = query.filter(Order.status == status_filter.upper())
orders = query.order_by(Order.created_at.desc()).all()
return orders
@app.get("/api/my/orders", response_model=List[OrderResponse])
def get_my_orders(
user_id: UserDep,
status_filter: Optional[str] = None,
db: Session = Depends(get_db_session)
):
"""Get current user's orders"""
query = db.query(Order).filter(Order.user_id == user_id)
if status_filter:
query = query.filter(Order.status == status_filter.upper())
orders = query.order_by(Order.created_at.desc()).all()
return orders
@app.get("/api/orders/orderbook", response_model=OrderBookResponse)
def get_orderbook(db: Session = Depends(get_db_session)):
"""Get current order book"""
# Get open buy orders (sorted by price descending)
buys = db.query(Order).filter(
and_(Order.order_type == 'BUY', Order.status == 'OPEN')
).order_by(desc(Order.price)).limit(20).all()
# Get open sell orders (sorted by price ascending)
sells = db.query(Order).filter(
and_(Order.order_type == 'SELL', Order.status == 'OPEN')
).order_by(Order.price).limit(20).all()
return OrderBookResponse(buys=buys, sells=sells)
@app.post("/api/orders", response_model=OrderResponse)
def create_order(
order: OrderCreate,
user_id: UserDep,
db: Session = Depends(get_db_session)
):
"""Create a new order"""
# Validate order type
if order.order_type not in ['BUY', 'SELL']:
raise HTTPException(
status_code=status.HTTP_400_BAD_REQUEST,
detail="Order type must be 'BUY' or 'SELL'"
)
# Create order
total = order.amount * order.price
db_order = Order(
user_id=user_id, # Use authenticated user_id
order_type=order.order_type,
amount=order.amount,
price=order.price,
total=total,
remaining=order.amount
)
db.add(db_order)
db.commit()
db.refresh(db_order)
# Try to match the order
try_match_order(db_order, db)
return db_order
def try_match_order(order: Order, db: Session):
"""Try to match an order with existing orders"""
if order.order_type == 'BUY':
# Match with sell orders at same or lower price
matching_orders = db.query(Order).filter(
and_(
Order.order_type == 'SELL',
Order.status == 'OPEN',
Order.price <= order.price
)
).order_by(Order.price).all()
else:
# Match with buy orders at same or higher price
matching_orders = db.query(Order).filter(
and_(
Order.order_type == 'BUY',
Order.status == 'OPEN',
Order.price >= order.price
)
).order_by(desc(Order.price)).all()
for match in matching_orders:
if order.remaining <= 0:
break
# Calculate trade amount
trade_amount = min(order.remaining, match.remaining)
trade_total = trade_amount * match.price
# Create trade record
trade = Trade(
buyer_id=order.user_id if order.order_type == 'BUY' else match.user_id,
seller_id=match.user_id if order.order_type == 'BUY' else order.user_id,
order_id=order.id,
amount=trade_amount,
price=match.price,
total=trade_total,
trade_hash=f"trade_{datetime.utcnow().timestamp()}"
)
db.add(trade)
# Update orders
order.filled += trade_amount
order.remaining -= trade_amount
match.filled += trade_amount
match.remaining -= trade_amount
# Update order statuses
if order.remaining <= 0:
order.status = 'FILLED'
else:
order.status = 'PARTIALLY_FILLED'
if match.remaining <= 0:
match.status = 'FILLED'
else:
match.status = 'PARTIALLY_FILLED'
db.commit()
@app.post("/api/auth/login")
def login_user(wallet_address: str, db: Session = Depends(get_db_session)):
"""Login with wallet address"""
# Find or create user
user = db.query(User).filter(User.wallet_address == wallet_address).first()
if not user:
user = User(
wallet_address=wallet_address,
email=f"{wallet_address}@aitbc.local",
is_active=True
)
db.add(user)
db.commit()
db.refresh(user)
# Create session token
token_data = f"{user.id}:{int(time.time())}"
token = hashlib.sha256(token_data.encode()).hexdigest()
# Store session
user_sessions[token] = {
"user_id": user.id,
"created_at": int(time.time()),
"expires_at": int(time.time()) + 86400 # 24 hours
}
return {"token": token, "user_id": user.id}
@app.post("/api/auth/logout")
def logout_user(token: str = Header(..., alias="Authorization")):
"""Logout user"""
if token.startswith("Bearer "):
token = token[7:]
if token in user_sessions:
del user_sessions[token]
return {"message": "Logged out successfully"}
@app.get("/api/health")
def health_check():
"""Health check endpoint"""
return {"status": "ok", "timestamp": datetime.utcnow()}
if __name__ == "__main__":
import uvicorn
uvicorn.run(app, host="0.0.0.0", port=3003)