- Admin: video management with Google Maps match status, manual restaurant mapping, restaurant remap on name change - Admin: user management tab with favorites/reviews detail - Admin: channel deletion fix for IDs with slashes - Frontend: responsive mobile layout (map top, list bottom, 2-row header) - Frontend: channel-colored map markers with legend - Frontend: my reviews list, favorites toggle, visit counter overlay - Frontend: force light mode for dark theme devices - Backend: visit tracking (site_visits table), user reviews endpoint - Backend: bulk transcript/extract streaming, geocode key fixes - Nginx config for production deployment Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
76 lines
2.4 KiB
Python
76 lines
2.4 KiB
Python
"""Restaurant API routes."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from fastapi import APIRouter, HTTPException, Query
|
|
|
|
from core import restaurant
|
|
|
|
router = APIRouter()
|
|
|
|
|
|
@router.get("")
|
|
def list_restaurants(
|
|
limit: int = Query(100, le=500),
|
|
offset: int = Query(0, ge=0),
|
|
cuisine: str | None = None,
|
|
region: str | None = None,
|
|
channel: str | None = None,
|
|
):
|
|
return restaurant.get_all(limit=limit, offset=offset, cuisine=cuisine, region=region, channel=channel)
|
|
|
|
|
|
@router.get("/{restaurant_id}")
|
|
def get_restaurant(restaurant_id: str):
|
|
r = restaurant.get_by_id(restaurant_id)
|
|
if not r:
|
|
raise HTTPException(404, "Restaurant not found")
|
|
return r
|
|
|
|
|
|
@router.put("/{restaurant_id}")
|
|
def update_restaurant(restaurant_id: str, body: dict):
|
|
from core.db import conn
|
|
r = restaurant.get_by_id(restaurant_id)
|
|
if not r:
|
|
raise HTTPException(404, "Restaurant not found")
|
|
|
|
allowed = ("name", "address", "region", "cuisine_type", "price_range",
|
|
"phone", "website", "latitude", "longitude")
|
|
sets = []
|
|
params: dict = {"rid": restaurant_id}
|
|
for field in allowed:
|
|
if field in body:
|
|
sets.append(f"{field} = :{field}")
|
|
params[field] = body[field] if body[field] != "" else None
|
|
if not sets:
|
|
raise HTTPException(400, "No fields to update")
|
|
sets.append("updated_at = SYSTIMESTAMP")
|
|
sql = f"UPDATE restaurants SET {', '.join(sets)} WHERE id = :rid"
|
|
with conn() as c:
|
|
c.cursor().execute(sql, params)
|
|
return {"ok": True}
|
|
|
|
|
|
@router.delete("/{restaurant_id}")
|
|
def delete_restaurant(restaurant_id: str):
|
|
from core.db import conn
|
|
r = restaurant.get_by_id(restaurant_id)
|
|
if not r:
|
|
raise HTTPException(404, "Restaurant not found")
|
|
with conn() as c:
|
|
cur = c.cursor()
|
|
cur.execute("DELETE FROM restaurant_vectors WHERE restaurant_id = :rid", {"rid": restaurant_id})
|
|
cur.execute("DELETE FROM user_reviews WHERE restaurant_id = :rid", {"rid": restaurant_id})
|
|
cur.execute("DELETE FROM video_restaurants WHERE restaurant_id = :rid", {"rid": restaurant_id})
|
|
cur.execute("DELETE FROM restaurants WHERE id = :rid", {"rid": restaurant_id})
|
|
return {"ok": True}
|
|
|
|
|
|
@router.get("/{restaurant_id}/videos")
|
|
def get_restaurant_videos(restaurant_id: str):
|
|
r = restaurant.get_by_id(restaurant_id)
|
|
if not r:
|
|
raise HTTPException(404, "Restaurant not found")
|
|
return restaurant.get_video_links(restaurant_id)
|