fastapi-2026
4
总安装量
1
周安装量
#48142
全站排名
安装命令
npx skills add https://github.com/krishamaze/skills --skill fastapi-2026
Agent 安装分布
amp
1
cline
1
opencode
1
cursor
1
kimi-cli
1
codex
1
Skill 文档
FastAPI 2026 â Async API Patterns
Version (2026)
- FastAPI:
>=0.115.0 - Uvicorn:
0.34.0+(stable, with uvloop + httptools) - Pydantic:
v2(mandatory â v1 syntax is deprecated) - Python:
3.12recommended
Project Structure (2026 Standard)
api/
âââ main.py # app entry, lifespan, middleware
âââ routes/
â âââ browser.py # browser control routes
â âââ drafts.py # content queue routes
âââ models/
â âââ schemas.py # Pydantic v2 models
âââ services/
â âââ browser.py # browser automation logic
âââ core/
âââ config.py # settings via pydantic-settings
App Entry with Lifespan (2026 Pattern)
from contextlib import asynccontextmanager
from fastapi import FastAPI
from fastapi.middleware.cors import CORSMiddleware
@asynccontextmanager
async def lifespan(app: FastAPI):
# Startup
await browser_service.start()
yield
# Shutdown
await browser_service.stop()
app = FastAPI(title="Threads Agent API", version="1.0.0", lifespan=lifespan)
app.add_middleware(
CORSMiddleware,
allow_origins=["http://localhost:3000"],
allow_credentials=True,
allow_methods=["*"],
allow_headers=["*"],
)
Pydantic v2 Models (2026)
from pydantic import BaseModel, Field
from typing import Literal
from datetime import datetime
class DraftPost(BaseModel):
content: str = Field(max_length=500)
media_urls: list[str] = []
status: Literal["pending", "approved", "rejected"] = "pending"
created_at: datetime = Field(default_factory=datetime.utcnow)
Route Patterns
from fastapi import APIRouter, HTTPException, BackgroundTasks
router = APIRouter(prefix="/browser", tags=["browser"])
@router.get("/feed")
async def get_feed() -> list[dict]:
try:
return await browser_service.read_feed()
except Exception as e:
raise HTTPException(status_code=500, detail=str(e))
@router.post("/act")
async def execute_action(action: BrowserAction, background_tasks: BackgroundTasks):
background_tasks.add_task(browser_service.execute, action)
return {"status": "queued"}
Background Tasks (for browser actions)
@router.post("/approve/{draft_id}")
async def approve_draft(draft_id: str, background_tasks: BackgroundTasks):
draft = await drafts_store.get(draft_id)
if not draft:
raise HTTPException(404, "Draft not found")
background_tasks.add_task(browser_service.post, draft.content)
await drafts_store.update(draft_id, status="approved")
return {"status": "posting", "draft_id": draft_id}
WebSocket (live feed updates to dashboard)
from fastapi import WebSocket, WebSocketDisconnect
active_connections: list[WebSocket] = []
@router.websocket("/ws/feed")
async def feed_websocket(websocket: WebSocket):
await websocket.accept()
active_connections.append(websocket)
try:
while True:
await websocket.receive_text()
except WebSocketDisconnect:
active_connections.remove(websocket)
async def broadcast(message: dict):
for ws in active_connections:
await ws.send_json(message)
Settings (pydantic-settings v2)
from pydantic_settings import BaseSettings
class Settings(BaseSettings):
session_dir: str = "/sessions"
display: str = ":99"
cors_origins: list[str] = ["http://localhost:3000"]
model_config = {"env_file": ".env"}
settings = Settings()
Production Server
# Dev
uv run uvicorn api.main:app --reload --host 0.0.0.0 --port 8000
# Production
uv run gunicorn api.main:app \
--workers 2 \
--worker-class uvicorn.workers.UvicornWorker \
--bind 0.0.0.0:8000
Key Rules for Our Stack
- Never put blocking Playwright/Camoufox sync calls in
async defâ blocks event loop - Always use
BackgroundTasksfor browser actions - One browser instance â keep in a service class, not per-request
- WebSocket for live updates to dashboard â not polling
Anti-Patterns
# â @app.on_event deprecated in 2026 â use lifespan
@app.on_event("startup")
# â Pydantic v1 Config class
class Post(BaseModel):
class Config:
orm_mode = True
# â
Pydantic v2
class Post(BaseModel):
model_config = {"from_attributes": True}
# â Sync call in async route
@app.get("/feed")
async def get_feed():
return browser.sync_read_feed() # blocks event loop