- Migration 035: add sensory_tags column to recipes (default '{}')
- scripts/tag_sensory_profiles.py: batch tagger using ingredient names,
direction keywords, and ingredient_profiles texture data
- app/services/recipe/sensory.py: SensoryExclude frozen dataclass,
build_sensory_exclude(), passes_sensory_filter() with graceful degradation
(untagged recipes always pass; malformed JSON always passes)
- store.browse_recipes and _browse_by_match: accept SensoryExclude, apply
filter in recipe-building loop (default path) and scoring loop (match sort)
- recipe_engine.suggest: load sensory_preferences from settings, apply
passes_sensory_filter() after exclude_set check in the rows loop
- settings endpoint: add sensory_preferences to _ALLOWED_KEYS
- Frontend: SensoryPreferences types in api.ts; sensoryPreferences state and
saveSensory() action in settings store; Sensory section in SettingsView with
texture avoid pills, smell/noise tolerance scale pills with ok/limit/neutral
color coding
- 66 new tests (29 classification + 13 sensory service + 2 settings); 281 total
46 lines
1.4 KiB
Python
46 lines
1.4 KiB
Python
"""User settings endpoints."""
|
|
from __future__ import annotations
|
|
|
|
from fastapi import APIRouter, Depends, HTTPException
|
|
from pydantic import BaseModel
|
|
|
|
from app.cloud_session import CloudUser, get_session
|
|
from app.db.session import get_store
|
|
from app.db.store import Store
|
|
|
|
router = APIRouter()
|
|
|
|
_ALLOWED_KEYS = frozenset({"cooking_equipment", "unit_system", "shopping_locale", "sensory_preferences"})
|
|
|
|
|
|
class SettingBody(BaseModel):
|
|
value: str
|
|
|
|
|
|
@router.get("/{key}")
|
|
async def get_setting(
|
|
key: str,
|
|
session: CloudUser = Depends(get_session),
|
|
store: Store = Depends(get_store),
|
|
) -> dict:
|
|
"""Return the stored value for a settings key."""
|
|
if key not in _ALLOWED_KEYS:
|
|
raise HTTPException(status_code=422, detail=f"Unknown settings key: '{key}'.")
|
|
value = store.get_setting(key)
|
|
if value is None:
|
|
raise HTTPException(status_code=404, detail=f"Setting '{key}' not found.")
|
|
return {"key": key, "value": value}
|
|
|
|
|
|
@router.put("/{key}")
|
|
async def set_setting(
|
|
key: str,
|
|
body: SettingBody,
|
|
session: CloudUser = Depends(get_session),
|
|
store: Store = Depends(get_store),
|
|
) -> dict:
|
|
"""Upsert a settings key-value pair."""
|
|
if key not in _ALLOWED_KEYS:
|
|
raise HTTPException(status_code=422, detail=f"Unknown settings key: '{key}'.")
|
|
store.set_setting(key, body.value)
|
|
return {"key": key, "value": body.value}
|