Replaces the flat JSONB-based recipe schema with a normalized relational model:
Schema (migrations consolidated to 001_initial_schema + 002_seed_data):
- New: dishes, dish_translations, dish_tags — canonical dish catalog
- New: cuisines, tags, dish_categories with _translations tables + full seed data
- New: recipe_ingredients, recipe_steps with _translations (replaces JSONB blobs)
- New: user_saved_recipes thin bookmark (drops saved_recipes + saved_recipe_translations)
- New: product_ingredients M2M table
- recipes: now a cooking variant of a dish (dish_id FK, no title/JSONB columns)
- recipe_translations: repurposed to per-language notes only
- products: mapping_id → primary_ingredient_id
- menu_items: recipe_id FK → recipes; adds dish_id
- meal_diary: adds dish_id, recipe_id → recipes, portion_g
Backend (Go):
- New packages: internal/cuisine, internal/tag, internal/dish (registry + handler + repo)
- New GET /cuisines, GET /tags (public), GET /dishes, GET /dishes/{id}, GET /recipes/{id}
- recipe, savedrecipe, menu, diary, product, ingredient packages updated for new schema
Flutter:
- New models: Cuisine, Tag; new providers: cuisineNamesProvider, tagNamesProvider
- recipe.dart: RecipeIngredient gains unit_code + effectiveUnit getter
- saved_recipe.dart: thin model, manual fromJson, computed nutrition getter
- diary_entry.dart: adds dishId, recipeId, portionG
- recipe_detail_screen.dart: localized cuisine/tag names via providers
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
97 lines
3.8 KiB
Go
97 lines
3.8 KiB
Go
package dish
|
|
|
|
import "time"
|
|
|
|
// Dish is a canonical dish record combining dish metadata with optional recipes.
|
|
type Dish struct {
|
|
ID string `json:"id"`
|
|
CuisineSlug *string `json:"cuisine_slug"`
|
|
CategorySlug *string `json:"category_slug"`
|
|
Name string `json:"name"`
|
|
Description *string `json:"description"`
|
|
ImageURL *string `json:"image_url"`
|
|
Tags []string `json:"tags"`
|
|
AvgRating float64 `json:"avg_rating"`
|
|
ReviewCount int `json:"review_count"`
|
|
Recipes []Recipe `json:"recipes,omitempty"`
|
|
CreatedAt time.Time `json:"created_at"`
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
}
|
|
|
|
// Recipe is a cooking variant attached to a Dish.
|
|
type Recipe struct {
|
|
ID string `json:"id"`
|
|
DishID string `json:"dish_id"`
|
|
Source string `json:"source"`
|
|
Difficulty *string `json:"difficulty"`
|
|
PrepTimeMin *int `json:"prep_time_min"`
|
|
CookTimeMin *int `json:"cook_time_min"`
|
|
Servings *int `json:"servings"`
|
|
CaloriesPerServing *float64 `json:"calories_per_serving"`
|
|
ProteinPerServing *float64 `json:"protein_per_serving"`
|
|
FatPerServing *float64 `json:"fat_per_serving"`
|
|
CarbsPerServing *float64 `json:"carbs_per_serving"`
|
|
FiberPerServing *float64 `json:"fiber_per_serving"`
|
|
Ingredients []RecipeIngredient `json:"ingredients"`
|
|
Steps []RecipeStep `json:"steps"`
|
|
Notes *string `json:"notes,omitempty"`
|
|
CreatedAt time.Time `json:"created_at"`
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
}
|
|
|
|
// RecipeIngredient is a single ingredient row from recipe_ingredients.
|
|
type RecipeIngredient struct {
|
|
ID string `json:"id"`
|
|
IngredientID *string `json:"ingredient_id"`
|
|
Name string `json:"name"`
|
|
Amount float64 `json:"amount"`
|
|
UnitCode *string `json:"unit_code"`
|
|
IsOptional bool `json:"is_optional"`
|
|
SortOrder int `json:"sort_order"`
|
|
}
|
|
|
|
// RecipeStep is a single step row from recipe_steps.
|
|
type RecipeStep struct {
|
|
ID string `json:"id"`
|
|
StepNumber int `json:"step_number"`
|
|
Description string `json:"description"`
|
|
TimerSeconds *int `json:"timer_seconds"`
|
|
ImageURL *string `json:"image_url"`
|
|
}
|
|
|
|
// CreateRequest is the body used to create a new dish + recipe at once.
|
|
// Used when saving a Gemini-generated recommendation.
|
|
type CreateRequest struct {
|
|
Name string `json:"name"`
|
|
Description string `json:"description"`
|
|
CuisineSlug string `json:"cuisine_slug"`
|
|
ImageURL string `json:"image_url"`
|
|
Tags []string `json:"tags"`
|
|
Source string `json:"source"`
|
|
Difficulty string `json:"difficulty"`
|
|
PrepTimeMin int `json:"prep_time_min"`
|
|
CookTimeMin int `json:"cook_time_min"`
|
|
Servings int `json:"servings"`
|
|
Calories float64 `json:"calories_per_serving"`
|
|
Protein float64 `json:"protein_per_serving"`
|
|
Fat float64 `json:"fat_per_serving"`
|
|
Carbs float64 `json:"carbs_per_serving"`
|
|
Ingredients []IngredientInput `json:"ingredients"`
|
|
Steps []StepInput `json:"steps"`
|
|
}
|
|
|
|
// IngredientInput is a single ingredient in the create request.
|
|
type IngredientInput struct {
|
|
Name string `json:"name"`
|
|
Amount float64 `json:"amount"`
|
|
Unit string `json:"unit"`
|
|
IsOptional bool `json:"is_optional"`
|
|
}
|
|
|
|
// StepInput is a single step in the create request.
|
|
type StepInput struct {
|
|
Number int `json:"number"`
|
|
Description string `json:"description"`
|
|
TimerSeconds *int `json:"timer_seconds"`
|
|
}
|