New feature: photograph a recipe card, cookbook page, or handwritten note and have it extracted into a structured, editable recipe. Backend: - POST /recipes/scan: accept 1-4 photos, run VLM extraction, return structured JSON for review (not auto-saved) - POST /recipes/scan/save: persist a reviewed/edited recipe - GET/DELETE /recipes/user: user-created recipe CRUD - Vision backend priority: cf-orch -> local Qwen2.5-VL -> Anthropic BYOK - 503 with clear config hint when no vision backend available - Multi-photo support: facing pages (ingredients/directions) sent together - Pantry cross-reference: marks which ingredients are already on hand - migration 041: user_recipes table (title, servings, cook_time, steps, ingredients JSON, source, pantry_match_pct) - Tier gate: recipe_scan -> paid, BYOK-unlockable Frontend: - "Scan" button in the Recipes tab bar (camera icon) - RecipeScanModal: upload step (drag-drop + file picker, up to 4 photos, live previews), processing step (spinner), review/edit step (all fields inline-editable before save), pantry match badge, warning banner for low-confidence or incomplete scans Tests: 35 new tests (23 unit + 12 API), 404 total passing
32 lines
2.5 KiB
Python
32 lines
2.5 KiB
Python
from fastapi import APIRouter
|
|
from app.api.endpoints import health, receipts, export, inventory, ocr, recipes, settings, staples, feedback, feedback_attach, household, saved_recipes, imitate, meal_plans, orch_usage, session, shopping
|
|
from app.api.endpoints.community import router as community_router
|
|
from app.api.endpoints.corrections import router as corrections_router
|
|
from app.api.endpoints.recipe_scan import router as recipe_scan_router
|
|
from app.api.endpoints.recipe_tags import router as recipe_tags_router
|
|
|
|
api_router = APIRouter()
|
|
|
|
api_router.include_router(session.router, prefix="/session", tags=["session"])
|
|
api_router.include_router(health.router, prefix="/health", tags=["health"])
|
|
api_router.include_router(receipts.router, prefix="/receipts", tags=["receipts"])
|
|
api_router.include_router(ocr.router, prefix="/receipts", tags=["ocr"])
|
|
api_router.include_router(export.router, tags=["export"])
|
|
api_router.include_router(inventory.router, prefix="/inventory", tags=["inventory"])
|
|
api_router.include_router(saved_recipes.router, prefix="/recipes/saved", tags=["saved-recipes"])
|
|
# recipe_scan_router registered BEFORE recipes.router so /recipes/scan and /recipes/user
|
|
# take priority over /recipes/{recipe_id} (which would otherwise match them as int IDs).
|
|
api_router.include_router(recipe_scan_router, prefix="/recipes", tags=["recipe-scan"])
|
|
api_router.include_router(recipes.router, prefix="/recipes", tags=["recipes"])
|
|
api_router.include_router(settings.router, prefix="/settings", tags=["settings"])
|
|
api_router.include_router(staples.router, prefix="/staples", tags=["staples"])
|
|
api_router.include_router(feedback.router, prefix="/feedback", tags=["feedback"])
|
|
api_router.include_router(feedback_attach.router, prefix="/feedback", tags=["feedback"])
|
|
api_router.include_router(household.router, prefix="/household", tags=["household"])
|
|
api_router.include_router(imitate.router, prefix="/imitate", tags=["imitate"])
|
|
api_router.include_router(meal_plans.router, prefix="/meal-plans", tags=["meal-plans"])
|
|
api_router.include_router(orch_usage.router, prefix="/orch-usage", tags=["orch-usage"])
|
|
api_router.include_router(shopping.router, prefix="/shopping", tags=["shopping"])
|
|
api_router.include_router(community_router)
|
|
api_router.include_router(recipe_tags_router)
|
|
api_router.include_router(corrections_router, prefix="/corrections", tags=["corrections"])
|