Files
aitbc/apps/trade-exchange/exchange_api.py
oib c984a1e052 chore: enhance security configuration across applications
- Add root-level *.json to .gitignore to prevent wallet backup leaks
- Replace wildcard CORS origins with explicit localhost URLs across all apps
- Add OPTIONS method to CORS allowed methods for preflight requests
- Update coordinator database to use absolute path in data/ directory to prevent duplicates
- Add JWT secret validation in coordinator config (must be set via environment)
- Replace deprecated get_session dependency with Session
2026-02-13 16:07:03 +01:00

357 lines
10 KiB
Python

#!/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 database import init_db, get_db_session
from models import User, Order, Trade, Balance
# Initialize FastAPI app
app = FastAPI(title="AITBC Trade Exchange API", version="1.0.0")
# 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]
# Initialize database on startup
@app.on_event("startup")
async def startup_event():
init_db()
# 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,
db: Session = Depends(get_db_session),
user_id: UserDep
):
"""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)