Backend .mdc

FastAPI Production Architecture

Cursor rules for FastAPI services with router/service/repository boundaries, typed provider adapters, bulkhead isolation, idempotency, and domain exceptions.

Nasıl kullanılır
  1. Kural içeriğini kopyala.
  2. Projenin root klasöründe şu dosyayı oluştur: .cursorrules veya .cursor/rules/fastapi-production-architecture.mdc
  3. İçeriği yapıştır ve kaydet.

FastAPI Production Architecture Rules

Principles for production-ready FastAPI services.

LAYER ARCHITECTURE (Principles A1-A8)

This codebase follows strict 4-layer architecture: Router → Service → Repository → ORM/HTTP/Storage. Imports flow downward only. Each layer has hard boundaries you must NOT cross.

Router rules (app/routers/**)

  • Handlers are THIN: ≤10 lines of executable code per handler
  • Allowed imports: fastapi, app.schemas., app.core.deps, app.services.
  • FORBIDDEN imports: sqlalchemy, httpx, boto3, app.models., app.repositories.
  • Every endpoint declares response_model= for OpenAPI fidelity
  • Every protected/business endpoint requires user_id: str = Depends(get_current_user_id)
  • Public endpoints (health checks, webhooks, callbacks) are exempt from auth
  • Business logic lives in services. Routers parse input, call one service method, return response.

GOOD: @router.post(“/wallet/charge”, response_model=WalletResponse, status_code=201) async def charge( req: ChargeRequest, user_id: str = Depends(get_current_user_id), svc: WalletUserService = Depends(get_wallet_service), ) -> WalletResponse: wallet = await svc.charge( user_id=user_id, amount=req.amount, idempotency_key=req.idempotency_key, ) return WalletResponse.from_domain(wallet)

BAD (business logic + SQL in router): @router.post(“/wallet/charge”) async def charge(req: ChargeRequest, db: Session = Depends(get_db)): wallet = db.query(Wallet).filter(Wallet.user_id == user_id).with_for_update().one() …

Service rules (app/services/**)

  • FORBIDDEN imports: sqlalchemy, httpx, boto3, redis, FastAPI Request/Response/HTTPException
  • Constructor injects Protocol-typed dependencies, not concrete classes
  • Raise domain exceptions (InsufficientFundsError), not HTTPException

GOOD: from app.repositories.protocols import WalletRepoProtocol class WalletUserService: def init(self, repo: WalletRepoProtocol): # Protocol, not SQLAlchemy Session self._repo = repo

BAD: from sqlalchemy.orm import Session class WalletUserService: def init(self, db: Session): … # Wrong — service depends on infrastructure

Repository rules (app/repositories/**)

  • ONLY layer allowed to import sqlalchemy
  • Implements Protocol from app/repositories/protocols.py
  • Returns domain objects, not ORM models
  • Every query scoped by user_id (multi-tenancy)

Provider rules (app/providers/**)

  • ONLY layer allowed to import httpx directly
  • Returns GenerateResult | ProviderError — NEVER raw dict
  • Uses per-provider httpx.AsyncClient (bulkhead pattern)

FILE SIZE RULES (Principle A1)

LOCStateAction
0–399GreenNone.
400–599YellowPlan split. Add TODO(decompose) header.
600+RedBLOCK merge. Decompose first.

Convert file to package when ANY is true:

  • Crosses 400 LOC and next change pushes past 500
  • Contains 2+ disjoint sub-domains (image vs video, user vs admin)
  • Mixes HTTP handlers with worker handlers
  • Has 2+ callers each importing only one symbol

Safe split pattern (atomic PR):

  1. Create /init.py (empty for now)
  2. Move pieces to sub-files (a.py, b.py, c.py)
  3. Re-export old public names from init.py
  4. Run tests — must pass without changes
  5. Follow-up PR to migrate callers off legacy alias

init.py pattern: from .user import WalletUserService from .admin import WalletAdminService WalletService = WalletUserService # backwards-compat alias all = [“WalletUserService”, “WalletAdminService”, “WalletService”]

EXTERNAL INTEGRATION RULES (Principles B1-B10)

Rule 1: Anti-Corruption Layer (ACL)

Providers return GenerateResult | ProviderError, never dict.

from dataclasses import dataclass from decimal import Decimal

@dataclass(frozen=True) class GenerateResult: url: str cost_usd: Decimal latency_ms: int provider_request_id: str

class ProviderError(Exception): def init(self, message: str, *, retryable: bool, code: str | None = None): super().init(message); self.retryable = retryable; self.code = code

class ProviderTimeout(ProviderError): def init(self, message: str): super().init(message, retryable=True, code=“timeout”)

Rule 2: Per-Provider Bulkhead

Each external provider has its OWN httpx.AsyncClient with its OWN Limits. NEVER share.

GOOD: FAL_HTTP = httpx.AsyncClient( base_url=settings.FAL_BASE_URL, timeout=httpx.Timeout(connect=5.0, read=60.0, write=10.0, pool=5.0), limits=httpx.Limits(max_connections=20, max_keepalive_connections=10), ) OPENAI_HTTP = httpx.AsyncClient( base_url=“https://api.openai.com/v1”, limits=httpx.Limits(max_connections=50, max_keepalive_connections=20), )

BAD: HTTP = httpx.AsyncClient() # shared across all providers — no bulkhead isolation

Shutdown cleanup — close all provider clients in FastAPI lifespan

from contextlib import asynccontextmanager

@asynccontextmanager async def lifespan(app): yield # app startup await FAL_HTTP.aclose() await OPENAI_HTTP.aclose()

app = FastAPI(lifespan=lifespan)

Or on_event(“shutdown”):

@app.on_event(“shutdown”)

async def close_http_clients() -> None:

await FAL_HTTP.aclose()

await OPENAI_HTTP.aclose()

Rule 3: Idempotency Keys

Every side-effect operation accepts an idempotency_key: UUID. Look up before retrying.

Rule 4: Structured Logging with contextvars

Use ContextVar to thread provider, user_id, request_id through async call stacks.

import contextvars

provider_var = contextvars.ContextVar[str | None](“provider”, default=None) user_id_var = contextvars.ContextVar[str | None](“user_id”, default=None) request_id_var = contextvars.ContextVar[str | None](“request_id”, default=None)

Read in logs: provider_var.get(), user_id_var.get(), request_id_var.get()

JSON formatter picks these up automatically via extra={} or ContextVar.get()

Rule 5: Single-Writer Principle (Principle B10)

For safety-critical state: exactly ONE service-layer module does the writing. Only the designated writer service for a domain may call repo.hold(). Routers and providers must NOT call repo.hold() directly. Admin services implemented in the service layer may call repo.hold() only if they are the designated writer for that domain. Enforce via: code-review grep check (grep -r "repo\.hold(" --include="*.py") and unit tests that assert call-origin of repo.hold().

ANTI-PATTERNS — REJECT ON SIGHT

  1. def some_method(self, db: Session, …) → use Protocol-typed repo
  2. from app.models.user import User inside a service → return domain types from repo
  3. httpx.AsyncClient() instantiated inside a function → use shared per-provider client
  4. raise HTTPException(…) inside a service → raise domain exception
  5. db.query(…) inside a router → move to service then repo
  6. async def call(self, req) -> dict: return resp.json() → ACL violation
  7. HTTP = httpx.AsyncClient() shared → bulkhead violation
  8. result[“vendor_field”][“nested”] in service code → ACL violation
  9. logger.info(f”{user_id} did X”) → use structured logging with extra={}
  10. Side-effect operation without idempotency_key → double-charge risk

DEPENDENCY INJECTION

Use FastAPI Depends() + factory functions in app/core/deps.py. Do NOT install dependency-injector, punq, or any DI container.

def get_wallet_service(db: Session = Depends(get_db)) -> WalletUserService: return WalletUserService(repo=SQLAlchemyWalletRepo(db))

DOMAIN EXCEPTIONS PATTERN

Services raise domain errors. Routers map to HTTP.

services/wallet/exceptions.py

class InsufficientFundsError(Exception): … class WalletNotFoundError(Exception): …

routers/wallet.py

try: wallet = await svc.charge(…) except InsufficientFundsError: raise HTTPException(402, detail=“insufficient funds”)

Benzer kurallar

Daha fazla: Backend →