mirror of
https://github.com/wshobson/agents.git
synced 2026-03-18 09:37:15 +00:00
* feat: add security middleware to REST API template Adds `TrustedHostMiddleware` and `CORSMiddleware` to the FastAPI template to ensure basic security protections are in place. Includes comments guiding users on how to configure these for production. - Added TrustedHostMiddleware for Host header validation - Added CORSMiddleware for Cross-Origin Resource Sharing - Added TODOs for production configuration * feat: add security middleware to REST API template Adds `TrustedHostMiddleware` and `CORSMiddleware` to the FastAPI template to ensure basic security protections are in place. Includes comments guiding users on how to configure these for production. - Added TrustedHostMiddleware for Host header validation - Added CORSMiddleware for Cross-Origin Resource Sharing - Configured safe defaults (allow_credentials=False) for the template - Added TODOs for production configuration * feat: secure API template and fix Pydantic deprecations Enhances `rest-api-template.py` with standard security middleware and updates Pydantic usage to V2 standards. - Added `TrustedHostMiddleware` and `CORSMiddleware` with safe defaults - Updated Pydantic models to use `ConfigDict` and `model_dump()` to resolve deprecation warnings - Documented security learnings in sentinel journal --------- Co-authored-by: google-labs-jules[bot] <161369871+google-labs-jules[bot]@users.noreply.github.com>
183 lines
5.2 KiB
Python
183 lines
5.2 KiB
Python
"""
|
|
Production-ready REST API template using FastAPI.
|
|
Includes pagination, filtering, error handling, and best practices.
|
|
"""
|
|
|
|
from fastapi import FastAPI, HTTPException, Query, Path, Depends, status
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
from fastapi.middleware.trustedhost import TrustedHostMiddleware
|
|
from fastapi.responses import JSONResponse
|
|
from pydantic import BaseModel, Field, EmailStr, ConfigDict
|
|
from typing import Optional, List, Any
|
|
from datetime import datetime
|
|
from enum import Enum
|
|
|
|
app = FastAPI(
|
|
title="API Template",
|
|
version="1.0.0",
|
|
docs_url="/api/docs"
|
|
)
|
|
|
|
# Security Middleware
|
|
# Trusted Host: Prevents HTTP Host Header attacks
|
|
app.add_middleware(
|
|
TrustedHostMiddleware,
|
|
allowed_hosts=["*"] # TODO: Configure this in production, e.g. ["api.example.com"]
|
|
)
|
|
|
|
# CORS: Configures Cross-Origin Resource Sharing
|
|
app.add_middleware(
|
|
CORSMiddleware,
|
|
allow_origins=["*"], # TODO: Update this with specific origins in production
|
|
allow_credentials=False, # TODO: Set to True if you need cookies/auth headers, but restrict origins
|
|
allow_methods=["*"],
|
|
allow_headers=["*"],
|
|
)
|
|
|
|
# Models
|
|
class UserStatus(str, Enum):
|
|
ACTIVE = "active"
|
|
INACTIVE = "inactive"
|
|
SUSPENDED = "suspended"
|
|
|
|
class UserBase(BaseModel):
|
|
email: EmailStr
|
|
name: str = Field(..., min_length=1, max_length=100)
|
|
status: UserStatus = UserStatus.ACTIVE
|
|
|
|
class UserCreate(UserBase):
|
|
password: str = Field(..., min_length=8)
|
|
|
|
class UserUpdate(BaseModel):
|
|
email: Optional[EmailStr] = None
|
|
name: Optional[str] = Field(None, min_length=1, max_length=100)
|
|
status: Optional[UserStatus] = None
|
|
|
|
class User(UserBase):
|
|
id: str
|
|
created_at: datetime
|
|
updated_at: datetime
|
|
|
|
model_config = ConfigDict(from_attributes=True)
|
|
|
|
# Pagination
|
|
class PaginationParams(BaseModel):
|
|
page: int = Field(1, ge=1)
|
|
page_size: int = Field(20, ge=1, le=100)
|
|
|
|
class PaginatedResponse(BaseModel):
|
|
items: List[Any]
|
|
total: int
|
|
page: int
|
|
page_size: int
|
|
pages: int
|
|
|
|
# Error handling
|
|
class ErrorDetail(BaseModel):
|
|
field: Optional[str] = None
|
|
message: str
|
|
code: str
|
|
|
|
class ErrorResponse(BaseModel):
|
|
error: str
|
|
message: str
|
|
details: Optional[List[ErrorDetail]] = None
|
|
|
|
@app.exception_handler(HTTPException)
|
|
async def http_exception_handler(request, exc):
|
|
return JSONResponse(
|
|
status_code=exc.status_code,
|
|
content=ErrorResponse(
|
|
error=exc.__class__.__name__,
|
|
message=exc.detail if isinstance(exc.detail, str) else exc.detail.get("message", "Error"),
|
|
details=exc.detail.get("details") if isinstance(exc.detail, dict) else None
|
|
).model_dump()
|
|
)
|
|
|
|
# Endpoints
|
|
@app.get("/api/users", response_model=PaginatedResponse, tags=["Users"])
|
|
async def list_users(
|
|
page: int = Query(1, ge=1),
|
|
page_size: int = Query(20, ge=1, le=100),
|
|
status: Optional[UserStatus] = Query(None),
|
|
search: Optional[str] = Query(None)
|
|
):
|
|
"""List users with pagination and filtering."""
|
|
# Mock implementation
|
|
total = 100
|
|
items = [
|
|
User(
|
|
id=str(i),
|
|
email=f"user{i}@example.com",
|
|
name=f"User {i}",
|
|
status=UserStatus.ACTIVE,
|
|
created_at=datetime.now(),
|
|
updated_at=datetime.now()
|
|
).model_dump()
|
|
for i in range((page-1)*page_size, min(page*page_size, total))
|
|
]
|
|
|
|
return PaginatedResponse(
|
|
items=items,
|
|
total=total,
|
|
page=page,
|
|
page_size=page_size,
|
|
pages=(total + page_size - 1) // page_size
|
|
)
|
|
|
|
@app.post("/api/users", response_model=User, status_code=status.HTTP_201_CREATED, tags=["Users"])
|
|
async def create_user(user: UserCreate):
|
|
"""Create a new user."""
|
|
# Mock implementation
|
|
return User(
|
|
id="123",
|
|
email=user.email,
|
|
name=user.name,
|
|
status=user.status,
|
|
created_at=datetime.now(),
|
|
updated_at=datetime.now()
|
|
)
|
|
|
|
@app.get("/api/users/{user_id}", response_model=User, tags=["Users"])
|
|
async def get_user(user_id: str = Path(..., description="User ID")):
|
|
"""Get user by ID."""
|
|
# Mock: Check if exists
|
|
if user_id == "999":
|
|
raise HTTPException(
|
|
status_code=status.HTTP_404_NOT_FOUND,
|
|
detail={"message": "User not found", "details": {"id": user_id}}
|
|
)
|
|
|
|
return User(
|
|
id=user_id,
|
|
email="user@example.com",
|
|
name="User Name",
|
|
status=UserStatus.ACTIVE,
|
|
created_at=datetime.now(),
|
|
updated_at=datetime.now()
|
|
)
|
|
|
|
@app.patch("/api/users/{user_id}", response_model=User, tags=["Users"])
|
|
async def update_user(user_id: str, update: UserUpdate):
|
|
"""Partially update user."""
|
|
# Validate user exists
|
|
existing = await get_user(user_id)
|
|
|
|
# Apply updates
|
|
update_data = update.model_dump(exclude_unset=True)
|
|
for field, value in update_data.items():
|
|
setattr(existing, field, value)
|
|
|
|
existing.updated_at = datetime.now()
|
|
return existing
|
|
|
|
@app.delete("/api/users/{user_id}", status_code=status.HTTP_204_NO_CONTENT, tags=["Users"])
|
|
async def delete_user(user_id: str):
|
|
"""Delete user."""
|
|
await get_user(user_id) # Verify exists
|
|
return None
|
|
|
|
if __name__ == "__main__":
|
|
import uvicorn
|
|
uvicorn.run(app, host="0.0.0.0", port=8000)
|