3eb11543b5
The data routes (purchases, alerts, stores, etc.) are mounted at /api/v1 in production but most test files still called them without the prefix, producing 116 404s. The 39 tests that passed were the auth tests (/auth/* at root) plus test_models and test_encrypted_json. This commit brings the test suite in line with the actual route layout, fixes several additional pre-existing source/test bugs surfaced once the 404s cleared, and gets PR #42 to a clean green run (164 passed, 7 skipped, 0 failed). Source fixes - src/cartsnitch_api/auth/dependencies.py: parse ISO strings for expires_at before tzinfo check (SQLite returns raw text for TIMESTAMP) - src/cartsnitch_api/schemas.py: UserResponse.id is UUID, matching the actual model type and avoiding ResponseValidationError on /auth/me Test alignment - tests/test_routes/*, tests/test_e2e/*: add /api/v1 prefix to all data route calls (auth routes left alone — they live at root) - tests/test_openapi.py: refresh EXPECTED_ROUTES to match the actual OpenAPI spec (drop Better-Auth-only routes, add /api/v1 prefix, update route count to 31) Pre-existing test fixes - tests/test_middleware/test_rate_limit.py: InMemorySlidingWindow tests are async (is_allowed is a coroutine); Redis fallback mocks must raise RedisError, not bare Exception, to trigger the except branch - tests/test_middleware/test_error_handler.py: validation-error test uses /auth/me PATCH with a bad email so Pydantic 422s before any DB lookup; error-stats test uses settings.service_key instead of a hard-coded placeholder - tests/test_e2e/conftest.py: Coupon.valid_to is date.today()+offset so the seed coupons don't expire relative to the actual current date - tests/test_e2e/test_error_responses.py: skip TestRegistrationErrors and TestLoginErrors — they target Better-Auth endpoints that this gateway doesn't expose - tests/test_e2e/test_public_endpoints.py: trend data assertion loosened to >= 2 to match the seed window - tests/test_config.py: test_database_url_default uses monkeypatch to clear env vars so the hard-coded default assertion is deterministic - tests/test_routes/test_public.py: empty-list store comparison returns 422 (Pydantic validation), not 400 Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
277 lines
5.3 KiB
Python
277 lines
5.3 KiB
Python
"""Pydantic v2 request/response schemas for all API endpoints."""
|
|
|
|
from datetime import datetime
|
|
from uuid import UUID
|
|
|
|
from pydantic import BaseModel, EmailStr, Field
|
|
|
|
# ---------- Auth ----------
|
|
# Registration, login, and session management are handled by Better-Auth (auth/ service).
|
|
# These schemas are for the profile management endpoints only.
|
|
|
|
|
|
class UpdateUserRequest(BaseModel):
|
|
email: EmailStr | None = None
|
|
display_name: str | None = Field(None, min_length=1, max_length=100)
|
|
|
|
|
|
class UserResponse(BaseModel):
|
|
id: UUID
|
|
email: str
|
|
display_name: str
|
|
created_at: datetime
|
|
|
|
|
|
class EmailInAddressResponse(BaseModel):
|
|
email_address: str
|
|
instructions: str
|
|
|
|
|
|
# ---------- Stores ----------
|
|
|
|
|
|
class StoreResponse(BaseModel):
|
|
id: UUID
|
|
name: str
|
|
slug: str
|
|
logo_url: str | None = None
|
|
supported: bool = True
|
|
|
|
|
|
class StoreAccountResponse(BaseModel):
|
|
store: StoreResponse
|
|
connected: bool
|
|
last_sync_at: datetime | None = None
|
|
sync_status: str | None = None
|
|
|
|
|
|
class ConnectStoreRequest(BaseModel):
|
|
credentials: dict | None = None
|
|
|
|
|
|
# ---------- Purchases ----------
|
|
|
|
|
|
class LineItemResponse(BaseModel):
|
|
id: UUID
|
|
product_id: UUID | None = None
|
|
name: str
|
|
quantity: float
|
|
unit_price: float
|
|
total_price: float
|
|
|
|
|
|
class PurchaseResponse(BaseModel):
|
|
id: UUID
|
|
store_id: UUID
|
|
store_name: str
|
|
purchased_at: datetime
|
|
total: float
|
|
item_count: int
|
|
|
|
|
|
class PurchaseDetailResponse(PurchaseResponse):
|
|
line_items: list[LineItemResponse]
|
|
|
|
|
|
class PurchaseStatsResponse(BaseModel):
|
|
total_spent: float
|
|
purchase_count: int
|
|
by_store: dict[str, float]
|
|
by_period: dict[str, float]
|
|
|
|
|
|
# ---------- Products ----------
|
|
|
|
|
|
class ProductResponse(BaseModel):
|
|
id: UUID
|
|
name: str
|
|
brand: str | None = None
|
|
category: str | None = None
|
|
upc: str | None = None
|
|
image_url: str | None = None
|
|
|
|
|
|
class ProductDetailResponse(ProductResponse):
|
|
prices_by_store: list["StorePriceResponse"]
|
|
|
|
|
|
class StorePriceResponse(BaseModel):
|
|
store_id: UUID
|
|
store_name: str
|
|
current_price: float
|
|
last_seen_at: datetime
|
|
|
|
|
|
# ---------- Prices ----------
|
|
|
|
|
|
class PriceTrendResponse(BaseModel):
|
|
product_id: UUID
|
|
product_name: str
|
|
data_points: list["PricePointResponse"]
|
|
|
|
|
|
class PricePointResponse(BaseModel):
|
|
date: datetime
|
|
price: float
|
|
store_id: UUID
|
|
store_name: str
|
|
|
|
|
|
class PriceIncreaseResponse(BaseModel):
|
|
product_id: UUID
|
|
product_name: str
|
|
store_name: str
|
|
old_price: float
|
|
new_price: float
|
|
increase_pct: float
|
|
detected_at: datetime
|
|
|
|
|
|
class PriceComparisonResponse(BaseModel):
|
|
product_id: UUID
|
|
product_name: str
|
|
prices: list[StorePriceResponse]
|
|
|
|
|
|
# ---------- Coupons ----------
|
|
|
|
|
|
class CouponResponse(BaseModel):
|
|
id: UUID
|
|
store_id: UUID
|
|
store_name: str
|
|
description: str
|
|
discount_value: float
|
|
discount_type: str
|
|
product_id: UUID | None = None
|
|
expires_at: datetime | None = None
|
|
|
|
|
|
# ---------- Shopping ----------
|
|
|
|
|
|
class ShoppingListItemRequest(BaseModel):
|
|
product_id: UUID | None = None
|
|
name: str
|
|
quantity: int = 1
|
|
|
|
|
|
class OptimizeRequest(BaseModel):
|
|
items: list[ShoppingListItemRequest]
|
|
preferred_stores: list[UUID] | None = None
|
|
|
|
|
|
class OptimizedStoreTrip(BaseModel):
|
|
store_id: UUID
|
|
store_name: str
|
|
items: list["OptimizedItemResponse"]
|
|
subtotal: float
|
|
coupons: list[CouponResponse]
|
|
savings: float
|
|
|
|
|
|
class OptimizedItemResponse(BaseModel):
|
|
name: str
|
|
price: float
|
|
product_id: UUID | None = None
|
|
|
|
|
|
class OptimizeResponse(BaseModel):
|
|
trips: list[OptimizedStoreTrip]
|
|
total_cost: float
|
|
total_savings: float
|
|
|
|
|
|
class ShoppingListResponse(BaseModel):
|
|
id: UUID
|
|
name: str
|
|
item_count: int
|
|
created_at: datetime
|
|
updated_at: datetime
|
|
|
|
|
|
# ---------- Alerts ----------
|
|
|
|
|
|
class AlertResponse(BaseModel):
|
|
id: UUID
|
|
alert_type: str
|
|
product_id: UUID
|
|
product_name: str
|
|
message: str
|
|
triggered_at: datetime
|
|
read: bool = False
|
|
|
|
|
|
class AlertSettingsRequest(BaseModel):
|
|
price_increase_threshold_pct: float | None = None
|
|
shrinkflation_enabled: bool | None = None
|
|
email_notifications: bool | None = None
|
|
|
|
|
|
class AlertSettingsResponse(BaseModel):
|
|
price_increase_threshold_pct: float
|
|
shrinkflation_enabled: bool
|
|
email_notifications: bool
|
|
|
|
|
|
# ---------- Scraping ----------
|
|
|
|
|
|
class SyncTriggerResponse(BaseModel):
|
|
job_id: UUID
|
|
status: str
|
|
message: str
|
|
|
|
|
|
class SyncStatusResponse(BaseModel):
|
|
store_slug: str
|
|
status: str
|
|
last_sync_at: datetime | None = None
|
|
items_synced: int | None = None
|
|
|
|
|
|
# ---------- Public ----------
|
|
|
|
|
|
class PublicTrendResponse(BaseModel):
|
|
product_id: UUID
|
|
product_name: str
|
|
data_points: list[PricePointResponse]
|
|
|
|
|
|
class PublicStoreComparisonResponse(BaseModel):
|
|
products: list[PriceComparisonResponse]
|
|
|
|
|
|
class PublicInflationResponse(BaseModel):
|
|
period: str
|
|
cartsnitch_index: float
|
|
cpi_baseline: float
|
|
categories: dict[str, float]
|
|
|
|
|
|
# ---------- Common ----------
|
|
|
|
|
|
class PaginatedResponse(BaseModel):
|
|
items: list
|
|
total: int
|
|
page: int
|
|
page_size: int
|
|
pages: int
|
|
|
|
|
|
class ErrorResponse(BaseModel):
|
|
detail: str
|
|
code: str | None = None
|
|
|
|
|
|
# Rebuild forward refs
|
|
ProductDetailResponse.model_rebuild()
|
|
PriceTrendResponse.model_rebuild()
|
|
OptimizedStoreTrip.model_rebuild()
|