from __future__ import annotations

from typing import Any, Dict, List, Optional, Tuple


DEFAULT_DAY_COUNT = 3
DEFAULT_MEALS_PER_DAY = 3

CUISINE_QUERY_LIBRARY: Dict[str, List[Dict[str, Any]]] = {
    "american": [
        {"label": "oats", "name_query": "oats|oatmeal", "max_results": 3},
        {"label": "eggs", "name_query": "egg|eggs", "max_results": 3},
        {"label": "bread", "name_query": "whole.*wheat.*bread|bread", "max_results": 3},
        {"label": "chicken", "name_query": "chicken.*breast|chicken", "max_results": 3},
        {"label": "turkey", "name_query": "turkey", "max_results": 3},
        {"label": "rice", "name_query": "brown.*rice|white.*rice|rice", "max_results": 3},
        {"label": "potato", "name_query": "potato|sweet.*potato", "max_results": 3},
        {"label": "beans", "name_query": "black.*beans|kidney.*beans|beans", "max_results": 3},
        {"label": "broccoli", "name_query": "broccoli", "max_results": 3},
        {"label": "banana", "name_query": "banana", "max_results": 3},
        {"label": "apple", "name_query": "apple", "max_results": 3},
        {"label": "milk", "name_query": "milk", "max_results": 3},
    ],
    "chinese": [
        {"label": "rice", "name_query": "jasmine.*rice|rice", "max_results": 3},
        {"label": "noodles", "name_query": "noodles|wheat.*noodles|rice.*noodles", "max_results": 3},
        {"label": "tofu", "name_query": "firm.*tofu|tofu", "max_results": 3},
        {"label": "chicken", "name_query": "chicken.*breast|chicken", "max_results": 3},
        {"label": "pork", "name_query": "lean.*pork|pork", "max_results": 3},
        {"label": "bok choy", "name_query": "bok.*choy|pak.*choi", "max_results": 3},
        {"label": "broccoli", "name_query": "broccoli|gai.*lan", "max_results": 3},
        {"label": "mushrooms", "name_query": "shiitake|mushroom", "max_results": 3},
        {"label": "edamame", "name_query": "edamame|soybeans", "max_results": 3},
        {"label": "egg", "name_query": "egg|eggs", "max_results": 3},
        {"label": "cabbage", "name_query": "napa.*cabbage|cabbage", "max_results": 3},
        {"label": "orange", "name_query": "mandarin|orange", "max_results": 3},
    ],
    "indian": [
        {"label": "rice", "name_query": "basmati.*rice|rice", "max_results": 3},
        {"label": "lentils", "name_query": "masoor.*dal|moong.*dal|lentils|dal", "max_results": 3},
        {"label": "chickpeas", "name_query": "chickpeas|chana", "max_results": 3},
        {"label": "paneer", "name_query": "paneer|cottage.*cheese", "max_results": 3},
        {"label": "yogurt", "name_query": "plain.*yogurt|curd|yogurt", "max_results": 3},
        {"label": "chicken", "name_query": "chicken", "max_results": 3},
        {"label": "spinach", "name_query": "spinach|palak", "max_results": 3},
        {"label": "cauliflower", "name_query": "cauliflower|gobi", "max_results": 3},
        {"label": "peas", "name_query": "green.*peas|peas", "max_results": 3},
        {"label": "potato", "name_query": "potato|aloo", "max_results": 3},
        {"label": "oats", "name_query": "oats|oatmeal", "max_results": 3},
        {"label": "banana", "name_query": "banana", "max_results": 3},
    ],
    "japanese": [
        {"label": "rice", "name_query": "sushi.*rice|rice", "max_results": 3},
        {"label": "udon", "name_query": "udon|noodles", "max_results": 3},
        {"label": "salmon", "name_query": "salmon", "max_results": 3},
        {"label": "tuna", "name_query": "tuna", "max_results": 3},
        {"label": "tofu", "name_query": "tofu", "max_results": 3},
        {"label": "edamame", "name_query": "edamame|soybeans", "max_results": 3},
        {"label": "egg", "name_query": "egg|eggs", "max_results": 3},
        {"label": "seaweed", "name_query": "seaweed|nori|wakame", "max_results": 3},
        {"label": "mushrooms", "name_query": "shiitake|mushroom", "max_results": 3},
        {"label": "spinach", "name_query": "spinach", "max_results": 3},
        {"label": "sweet potato", "name_query": "sweet.*potato", "max_results": 3},
        {"label": "banana", "name_query": "banana", "max_results": 3},
    ],
    "thai": [
        {"label": "rice", "name_query": "jasmine.*rice|rice", "max_results": 3},
        {"label": "rice noodles", "name_query": "rice.*noodles|noodles", "max_results": 3},
        {"label": "chicken", "name_query": "chicken", "max_results": 3},
        {"label": "shrimp", "name_query": "shrimp|prawns", "max_results": 3},
        {"label": "tofu", "name_query": "tofu", "max_results": 3},
        {"label": "egg", "name_query": "egg|eggs", "max_results": 3},
        {"label": "broccoli", "name_query": "broccoli", "max_results": 3},
        {"label": "carrot", "name_query": "carrot", "max_results": 3},
        {"label": "green beans", "name_query": "green.*beans|beans", "max_results": 3},
        {"label": "mango", "name_query": "mango", "max_results": 3},
        {"label": "pineapple", "name_query": "pineapple", "max_results": 3},
        {"label": "coconut milk", "name_query": "light.*coconut.*milk|coconut.*milk", "max_results": 3},
    ],
    "middle eastern": [
        {"label": "chickpeas", "name_query": "chickpeas|garbanzo", "max_results": 3},
        {"label": "lentils", "name_query": "lentils", "max_results": 3},
        {"label": "bulgur", "name_query": "bulgur", "max_results": 3},
        {"label": "rice", "name_query": "rice", "max_results": 3},
        {"label": "chicken", "name_query": "chicken", "max_results": 3},
        {"label": "beef", "name_query": "lean.*beef|beef", "max_results": 3},
        {"label": "yogurt", "name_query": "plain.*yogurt|yogurt", "max_results": 3},
        {"label": "cucumber", "name_query": "cucumber", "max_results": 3},
        {"label": "tomato", "name_query": "tomato", "max_results": 3},
        {"label": "eggplant", "name_query": "eggplant", "max_results": 3},
        {"label": "spinach", "name_query": "spinach", "max_results": 3},
        {"label": "dates", "name_query": "dates|date", "max_results": 3},
    ],
    "korean": [
        {"label": "rice", "name_query": "short.*grain.*rice|rice", "max_results": 3},
        {"label": "rice cakes", "name_query": "rice.*cake|tteok", "max_results": 3},
        {"label": "sweet potato noodles", "name_query": "sweet.*potato.*noodles|dangmyeon|glass.*noodles", "max_results": 3},
        {"label": "chicken", "name_query": "chicken", "max_results": 3},
        {"label": "beef", "name_query": "lean.*beef|beef", "max_results": 3},
        {"label": "tofu", "name_query": "firm.*tofu|tofu", "max_results": 3},
        {"label": "egg", "name_query": "egg|eggs", "max_results": 3},
        {"label": "kimchi", "name_query": "kimchi", "max_results": 3},
        {"label": "spinach", "name_query": "spinach", "max_results": 3},
        {"label": "mushrooms", "name_query": "shiitake|oyster.*mushroom|mushroom", "max_results": 3},
        {"label": "cabbage", "name_query": "napa.*cabbage|cabbage", "max_results": 3},
        {"label": "seaweed", "name_query": "seaweed|gim|nori", "max_results": 3},
    ],
}

GENERIC_QUERY_LIBRARY: List[Dict[str, Any]] = [
    {"label": "oats", "name_query": "oats|oatmeal", "max_results": 3},
    {"label": "rice", "name_query": "rice", "max_results": 3},
    {"label": "bread", "name_query": "whole.*wheat.*bread|bread", "max_results": 3},
    {"label": "potato", "name_query": "potato|sweet.*potato", "max_results": 3},
    {"label": "banana", "name_query": "banana", "max_results": 3},
    {"label": "apple", "name_query": "apple", "max_results": 3},
    {"label": "broccoli", "name_query": "broccoli", "max_results": 3},
    {"label": "spinach", "name_query": "spinach", "max_results": 3},
    {"label": "egg", "name_query": "egg|eggs", "max_results": 3},
    {"label": "milk", "name_query": "milk", "max_results": 3},
    {"label": "yogurt", "name_query": "plain.*yogurt|greek.*yogurt|yogurt", "max_results": 3},
    {"label": "chicken", "name_query": "chicken", "max_results": 3},
    {"label": "tofu", "name_query": "tofu", "max_results": 3},
    {"label": "beans", "name_query": "beans|lentils|chickpeas", "max_results": 3},
]

MEAL_TEMPLATE_SEQUENCE: List[Tuple[str, List[str]]] = [
    ("breakfast", ["oats", "bread", "yogurt", "egg", "milk", "banana", "apple"]),
    ("lunch", ["rice", "noodles", "bulgur", "potato", "chicken", "tofu", "beans", "broccoli", "spinach", "cabbage", "carrot"]),
    ("dinner", ["rice", "noodles", "potato", "chicken", "tofu", "beans", "broccoli", "spinach", "eggplant", "mushrooms", "tomato", "cucumber"]),
]

PROTEIN_LABELS = {
    "egg",
    "eggs",
    "chicken",
    "turkey",
    "beef",
    "pork",
    "salmon",
    "tuna",
    "shrimp",
    "tofu",
    "paneer",
    "yogurt",
    "milk",
    "lentils",
    "beans",
    "chickpeas",
    "edamame",
}

CARB_LABELS = {
    "oats",
    "bread",
    "rice",
    "noodles",
    "udon",
    "bulgur",
    "potato",
    "sweet potato",
    "banana",
    "apple",
    "mango",
    "pineapple",
    "dates",
}

VEG_LABELS = {
    "broccoli",
    "spinach",
    "bok choy",
    "cabbage",
    "mushrooms",
    "carrot",
    "green beans",
    "cucumber",
    "tomato",
    "eggplant",
    "cauliflower",
    "peas",
    "seaweed",
}


def normalize_cuisine_name(cuisine: str) -> str:
    normalized = str(cuisine or "").strip().lower()
    return normalized if normalized in CUISINE_QUERY_LIBRARY else "american"


def build_ingredient_queries(cuisine: str) -> List[Dict[str, Any]]:
    normalized = normalize_cuisine_name(cuisine)
    seen = set()
    merged: List[Dict[str, Any]] = []
    for item in CUISINE_QUERY_LIBRARY.get(normalized, []):
        label = str(item.get("label", "")).strip().lower()
        if label and label not in seen:
            merged.append(dict(item))
            seen.add(label)
    for item in GENERIC_QUERY_LIBRARY:
        label = str(item.get("label", "")).strip().lower()
        if label and label not in seen:
            merged.append(dict(item))
            seen.add(label)
    return merged


def infer_plan_days(query_text: str) -> int:
    text = str(query_text or "").lower()
    for candidate in (7, 5, 3, 2, 1):
        if f"{candidate} day" in text or f"{candidate}-day" in text:
            return candidate
    return DEFAULT_DAY_COUNT


def age_years_from_profile(profile: Dict[str, Any]) -> float:
    age = float(profile.get("age", 0.0))
    age_unit = str(profile.get("age_unit", "years")).strip().lower()
    if age_unit == "months":
        return age / 12.0
    return age


def select_meal_count(profile: Dict[str, Any]) -> int:
    age_years = age_years_from_profile(profile)
    if age_years < 2.0:
        return 2
    return DEFAULT_MEALS_PER_DAY


def quantity_range_for_label(label: str, meal_name: str, age_years: float, attempt: int) -> Tuple[float, float]:
    normalized = str(label or "").strip().lower()
    is_infant = age_years < 1.0
    is_toddler = age_years < 4.0

    if normalized in PROTEIN_LABELS:
        if is_infant:
            base = (15.0, 80.0)
        elif is_toddler:
            base = (25.0, 120.0)
        else:
            base = (40.0, 180.0)
    elif normalized in CARB_LABELS:
        if is_infant:
            base = (20.0, 90.0)
        elif is_toddler:
            base = (40.0, 160.0)
        else:
            base = (60.0, 260.0)
    else:
        if is_infant:
            base = (10.0, 60.0)
        elif is_toddler:
            base = (20.0, 100.0)
        else:
            base = (40.0, 160.0)

    if meal_name == "breakfast" and normalized in {"banana", "apple", "mango", "pineapple", "dates"}:
        base = (base[0], max(base[1], base[0] + 40.0))

    widen = float(attempt) * 0.2
    min_qty = max(0.0, round(base[0] * max(0.0, 1.0 - widen), 2))
    max_qty = round(base[1] * (1.0 + widen), 2)
    return min_qty, max_qty


def _rotated(sequence: List[str], offset: int) -> List[str]:
    if not sequence:
        return []
    normalized_offset = offset % len(sequence)
    return sequence[normalized_offset:] + sequence[:normalized_offset]


def _best_result_index_for_label(label: str, results_by_label: Dict[str, List[Dict[str, Any]]]) -> Optional[int]:
    results = results_by_label.get(label, [])
    if not results:
        return None
    best = results[0]
    try:
        return int(best.get("index"))
    except (TypeError, ValueError):
        return None


def build_search_space(
    profile: Dict[str, Any],
    cuisine: str,
    number_of_days: int,
    results_by_label: Dict[str, List[Dict[str, Any]]],
    attempt: int,
) -> Dict[str, Dict[str, Dict[str, List[Dict[str, float]]]]]:
    del cuisine

    meal_count = select_meal_count(profile)
    age_years = age_years_from_profile(profile)
    active_templates = MEAL_TEMPLATE_SEQUENCE[:meal_count]
    search_space: Dict[str, Dict[str, Dict[str, List[Dict[str, float]]]]] = {}

    for day_number in range(1, max(1, int(number_of_days)) + 1):
        day_key = f"day_{day_number}"
        search_space[day_key] = {}
        for meal_position, (meal_name, label_pool) in enumerate(active_templates, start=1):
            meal_key = f"meal_{meal_position}"
            option_payload: Dict[str, List[Dict[str, float]]] = {}

            protein_candidates = [label for label in label_pool if label in PROTEIN_LABELS and results_by_label.get(label)]
            carb_candidates = [label for label in label_pool if label in CARB_LABELS and results_by_label.get(label)]
            veg_candidates = [label for label in label_pool if label in VEG_LABELS and results_by_label.get(label)]

            if not protein_candidates:
                protein_candidates = [label for label in results_by_label.keys() if label in PROTEIN_LABELS and results_by_label.get(label)]
            if not carb_candidates:
                carb_candidates = [label for label in results_by_label.keys() if label in CARB_LABELS and results_by_label.get(label)]
            if not veg_candidates:
                veg_candidates = [label for label in results_by_label.keys() if label in VEG_LABELS and results_by_label.get(label)]

            day_offset = max(0, day_number - 1)
            meal_offset = max(0, meal_position - 1)
            protein_candidates = _rotated(protein_candidates, day_offset + meal_offset)
            carb_candidates = _rotated(carb_candidates, day_offset + (2 * meal_offset))
            veg_candidates = _rotated(veg_candidates, day_offset + (3 * meal_offset))

            option_label_sets: List[List[str]] = []
            if protein_candidates and carb_candidates:
                option_label_sets.append([protein_candidates[0], carb_candidates[0]] + veg_candidates[:1])
            if len(protein_candidates) > 1 and carb_candidates:
                option_label_sets.append([protein_candidates[1], carb_candidates[0]] + veg_candidates[1:2])
            elif protein_candidates and len(carb_candidates) > 1:
                option_label_sets.append([protein_candidates[0], carb_candidates[1]] + veg_candidates[1:2])

            cleaned_sets: List[List[str]] = []
            for labels in option_label_sets:
                deduped: List[str] = []
                for label in labels:
                    if label and label not in deduped and results_by_label.get(label):
                        deduped.append(label)
                if len(deduped) >= 2:
                    cleaned_sets.append(deduped)

            for option_position, labels in enumerate(cleaned_sets[:2], start=1):
                items: List[Dict[str, float]] = []
                for label in labels:
                    index = _best_result_index_for_label(label, results_by_label)
                    if index is None:
                        continue
                    min_qty, max_qty = quantity_range_for_label(label, meal_name, age_years, attempt)
                    items.append({"index": index, "min_qty": min_qty, "max_qty": max_qty})
                if len(items) >= 2:
                    option_payload[f"option_{option_position}"] = items

            if option_payload:
                search_space[day_key][meal_key] = option_payload

    return search_space
