from typing import TYPE_CHECKING, Any, Dict, List, Optional, Tuple
import ast
import json
import re

if TYPE_CHECKING:
    from google.adk.tools.tool_context import ToolContext
else:
    ToolContext = Any
from ortools.linear_solver import pywraplp


_MACRO_KEY_ALIASES = {
    "protein": "protein",
    "proteins": "protein",
    "carb": "carbohydrates",
    "carbs": "carbohydrates",
    "carbohydrate": "carbohydrates",
    "carbohydrates": "carbohydrates",
    "fat": "total_fat",
    "fats": "total_fat",
    "total_fat": "total_fat",
    "fibre": "total_fibre",
    "fiber": "total_fibre",
    "total_fibre": "total_fibre",
    "total_fiber": "total_fibre",
}

_TARGET_KEY_ALIASES = {
    **_MACRO_KEY_ALIASES,
    "calorie": "calories",
    "calories": "calories",
    "kcal": "calories",
    "energy": "calories",
    "cost": "cost",
    "price": "cost",
    "dietary_fiber": "total_fibre",
    "dietary_fibre": "total_fibre",
}

_NUTRITION_KEY_CANDIDATES = {
    "protein": ["protein", "proteins"],
    "carbohydrates": ["carbohydrates", "carbs", "carbohydrate"],
    "total_fat": ["total_fat", "fat", "fats", "lipid"],
    "total_fibre": ["total_fibre", "total_fiber", "fiber", "fibre", "dietary_fiber", "dietary_fibre"],
    "calories": ["calories", "kcal", "energy_kcal", "energy"],
}

_MACRO_DAILY_TARGET_KEYS = {"protein", "carbohydrates", "total_fat", "total_fibre", "calories"}
_OPTIMIZER_TARGET_KEYS = {"protein", "carbohydrates", "total_fat", "total_fibre", "calories"}
_OPTIMIZER_RANGE_TARGET_KEYS = {"protein", "carbohydrates", "total_fat"}

_TARGET_PRIORITY_WEIGHTS = {
    "calories": 100.0,
    "protein": 25.0,
    "carbohydrates": 3.0,
    "total_fat": 3.0,
    "total_fibre": 20.0,
}

_CALORIE_DIFFERENCE_TOLERANCE = 10.0
_FIBRE_DIFFERENCE_TOLERANCE = 5.0
_RANGE_DEVIATION_PRIORITY_MULTIPLIER = 1000.0
_CALORIE_DIFFERENCE_KEY = "daily_calorie_difference"
_FIBRE_DIFFERENCE_KEY = "daily_total_fibre_difference"
_RANGE_DEVIATION_KEYS = {
    "protein": "daily_protein_range_deviation",
    "carbohydrates": "daily_carbohydrate_range_deviation",
    "total_fat": "daily_total_fat_range_deviation",
}


def _to_float(value: Any, default: float = 0.0) -> float:
    try:
        if value is None:
            return default
        return float(value)
    except (TypeError, ValueError):
        return default


def _to_dict(value: Any) -> Dict[str, Any]:
    if isinstance(value, dict):
        return value
    if isinstance(value, str):
        try:
            parsed = json.loads(value)
            return parsed if isinstance(parsed, dict) else {}
        except json.JSONDecodeError:
            return {}
    return {}


def _is_product_record(value: Any) -> bool:
    if not isinstance(value, dict):
        return False
    if "index" not in value:
        return False
    return "name" in value or "nutrition" in value or "nutrition_100g" in value or "cost" in value


def _collect_products(value: Any, out: Dict[int, Dict[str, Any]]) -> None:
    if _is_product_record(value):
        try:
            idx = int(value.get("index"))
        except (TypeError, ValueError):
            idx = None
        if idx is not None:
            out[idx] = value
        return

    if isinstance(value, list):
        for item in value:
            _collect_products(item, out)
        return

    if isinstance(value, dict):
        for nested in value.values():
            _collect_products(nested, out)


def _parse_quantity_range(range_spec: Any) -> Tuple[float, float]:
    default_min, default_max = 0.0, 1000.0
    if not isinstance(range_spec, dict):
        return default_min, default_max

    min_value = None
    max_value = None

    for key in ("min", "min_g", "min_qty", "min_quantity", "low"):
        if key in range_spec:
            min_value = _to_float(range_spec.get(key), default_min)
            break

    for key in ("max", "max_g", "max_qty", "max_quantity", "high"):
        if key in range_spec:
            max_value = _to_float(range_spec.get(key), default_max)
            break

    if min_value is None and isinstance(range_spec.get("range"), list) and len(range_spec["range"]) >= 2:
        min_value = _to_float(range_spec["range"][0], default_min)
        max_value = _to_float(range_spec["range"][1], default_max)

    if min_value is None and isinstance(range_spec.get("quantity_range"), list) and len(range_spec["quantity_range"]) >= 2:
        min_value = _to_float(range_spec["quantity_range"][0], default_min)
        max_value = _to_float(range_spec["quantity_range"][1], default_max)

    min_value = default_min if min_value is None else max(0.0, min_value)
    max_value = default_max if max_value is None else max(0.0, max_value)

    if min_value > max_value:
        min_value, max_value = max_value, min_value
    return min_value, max_value


def _extract_products_from_history(tool_context: ToolContext) -> Dict[int, Dict[str, Any]]:
    products_by_index: Dict[int, Dict[str, Any]] = {}
    session = tool_context._invocation_context.session
    for event in session.events:
        content = getattr(event, "content", None)
        parts = getattr(content, "parts", None) if content else None
        if not parts:
            continue

        for part in parts:
            function_response = getattr(part, "function_response", None)
            if not function_response:
                continue
            if getattr(function_response, "name", "") != "find_ingredient":
                continue

            payload = getattr(function_response, "response", None)
            if not isinstance(payload, dict):
                continue
            result = payload.get("result")
            _collect_products(result, products_by_index)

    return products_by_index


def _extract_latest_function_response_from_history(
    tool_context: ToolContext,
    function_name: str,
) -> Dict[str, Any]:
    latest_payload: Dict[str, Any] = {}
    session = tool_context._invocation_context.session
    for event in session.events:
        content = getattr(event, "content", None)
        parts = getattr(content, "parts", None) if content else None
        if not parts:
            continue

        for part in parts:
            function_response = getattr(part, "function_response", None)
            if not function_response:
                continue
            if getattr(function_response, "name", "") != function_name:
                continue

            payload = getattr(function_response, "response", None)
            if isinstance(payload, dict):
                latest_payload = payload

    return latest_payload


def _collect_function_responses_from_history(
    tool_context: ToolContext,
) -> List[Tuple[str, Dict[str, Any]]]:
    responses: List[Tuple[str, Dict[str, Any]]] = []
    session = getattr(getattr(tool_context, "_invocation_context", None), "session", None)
    events = getattr(session, "events", []) if session is not None else []

    for event in events:
        content = getattr(event, "content", None)
        parts = getattr(content, "parts", None) if content else None
        if not parts:
            continue

        for part in parts:
            function_response = getattr(part, "function_response", None)
            if not function_response:
                continue

            name = str(getattr(function_response, "name", ""))
            payload = getattr(function_response, "response", None)
            responses.append((name, payload if isinstance(payload, dict) else {}))

    return responses


def _float_maps_match(left: Dict[str, float], right: Dict[str, float], tolerance: float = 1e-6) -> bool:
    if set(left.keys()) != set(right.keys()):
        return False
    for key, left_value in left.items():
        right_value = right.get(key)
        if right_value is None:
            return False
        if abs(float(left_value) - float(right_value)) > tolerance:
            return False
    return True


def _range_maps_match(
    left: Dict[str, Dict[str, float]],
    right: Dict[str, Dict[str, float]],
    tolerance: float = 1e-6,
) -> bool:
    if set(left.keys()) != set(right.keys()):
        return False

    for key, left_bounds in left.items():
        right_bounds = right.get(key)
        if not isinstance(right_bounds, dict):
            return False
        left_lower = _to_float(left_bounds.get("lower"), 0.0)
        left_upper = _to_float(left_bounds.get("upper"), 0.0)
        right_lower = _to_float(right_bounds.get("lower"), 0.0)
        right_upper = _to_float(right_bounds.get("upper"), 0.0)
        if abs(left_lower - right_lower) > tolerance:
            return False
        if abs(left_upper - right_upper) > tolerance:
            return False

    return True


def _should_skip_repeated_optimizer_call(
    tool_context: ToolContext,
    current_targets: Dict[str, float],
    current_target_ranges: Dict[str, Dict[str, float]],
) -> bool:
    responses = _collect_function_responses_from_history(tool_context)
    if not responses:
        return False

    last_name, last_payload = responses[-1]
    if last_name != "optimize_quantity":
        return False

    # Function responses may be wrapped under a top-level "result" field.
    last_result_payload = _to_dict(last_payload.get("result")) if isinstance(last_payload, dict) else {}
    if not last_result_payload:
        last_result_payload = last_payload

    last_status = str(last_result_payload.get("status", "")).lower()
    was_previously_skipped = bool(last_result_payload.get("converged_in_previous_tool_call", False))

    if last_status != "feasible" and not was_previously_skipped:
        return False

    previous_dri_payload: Optional[Dict[str, Any]] = None
    for response_name, response_payload in reversed(responses[:-1]):
        if response_name == "calculate_health_canada_dri" and isinstance(response_payload, dict):
            previous_dri_payload = response_payload
            break

    if previous_dri_payload is None:
        return False

    previous_targets, previous_target_ranges = _build_target_bundle_from_dri_payload(previous_dri_payload)
    previous_optimizer_targets = _filter_optimizer_targets(previous_targets)
    previous_optimizer_target_ranges = _filter_optimizer_target_ranges(previous_target_ranges)

    return _float_maps_match(previous_optimizer_targets, current_targets) and _range_maps_match(
        previous_optimizer_target_ranges,
        current_target_ranges,
    )


def _cost_per_gram(product: Dict[str, Any]) -> Optional[float]:
    cost = _to_dict(product.get("cost"))
    if not cost:
        return None

    unit = str(cost.get("unit", "")).strip().lower()
    price = _to_float(cost.get("price"), -1.0)
    quantity = _to_float(cost.get("quantity"), 1.0)
    if price < 0 or quantity <= 0:
        return None

    if unit in {"kg", "kilogram", "kilograms"}:
        return price / (quantity * 1000.0)
    if unit in {"g", "gram", "grams"}:
        return price / quantity
    if unit in {"lb", "lbs", "pound", "pounds"}:
        return price / (quantity * 453.59237)

    if unit in {"each", "unit", "piece"}:
        serving = _to_dict(product.get("serving"))
        metric = serving.get("metric") if isinstance(serving.get("metric"), dict) else {}
        grams_per_piece = _to_float(metric.get("quantity"), 0.0)
        metric_unit = str(metric.get("unit", "")).strip().lower()
        if grams_per_piece > 0 and metric_unit in {"g", "gram", "grams"}:
            return price / (quantity * grams_per_piece)
        return None

    return None


def _format_number(value: float) -> str:
    if float(value).is_integer():
        return str(int(value))
    return f"{value:.4f}".rstrip("0").rstrip(".")


def _standard_serving_basis(product: Dict[str, Any]) -> Optional[Tuple[float, str]]:
    serving = _to_dict(product.get("serving"))
    if not serving:
        return None

    metric = serving.get("metric") if isinstance(serving.get("metric"), dict) else {}
    grams_per_serving = _to_float(metric.get("quantity"), 0.0)
    metric_unit = str(metric.get("unit", "")).strip().lower()
    if grams_per_serving <= 0 or metric_unit not in {"g", "gram", "grams"}:
        return None

    common = serving.get("common") if isinstance(serving.get("common"), dict) else {}
    common_quantity = _to_float(common.get("quantity"), 0.0)
    common_unit = str(common.get("unit", "")).strip()
    if common_quantity > 0 and common_unit:
        serving_label = f"{_format_number(common_quantity)} {common_unit}"
    else:
        serving_label = f"{_format_number(grams_per_serving)} g"

    return grams_per_serving, serving_label


def _parse_json_input(value: Any, expected: type, field_name: str) -> Any:
    if isinstance(value, expected):
        return value

    def _strip_code_fences(text: str) -> str:
        stripped = text.strip()
        if not stripped.startswith("```"):
            return stripped

        lines = stripped.splitlines()
        if len(lines) >= 2 and lines[0].startswith("```") and lines[-1].strip().startswith("```"):
            return "\n".join(lines[1:-1]).strip()
        return stripped

    def _extract_balanced_json_like(text: str) -> str:
        start_obj = text.find("{")
        start_arr = text.find("[")
        starts = [pos for pos in (start_obj, start_arr) if pos >= 0]
        if not starts:
            return text

        start = min(starts)
        opener = text[start]
        closer = "}" if opener == "{" else "]"

        depth = 0
        in_string = False
        string_quote = ""
        escaped = False

        for idx in range(start, len(text)):
            ch = text[idx]
            if in_string:
                if escaped:
                    escaped = False
                    continue
                if ch == "\\":
                    escaped = True
                    continue
                if ch == string_quote:
                    in_string = False
                    continue
                continue

            if ch in {'"', "'"}:
                in_string = True
                string_quote = ch
                continue

            if ch == opener:
                depth += 1
                continue

            if ch == closer:
                depth -= 1
                if depth == 0:
                    return text[start : idx + 1]

        return text[start:]

    def _remove_trailing_commas(text: str) -> str:
        previous = None
        current = text
        while previous != current:
            previous = current
            current = re.sub(r",\s*([}\]])", r"\1", current)
        return current

    def _try_parse_candidates(raw_text: str) -> Any:
        candidates = []
        normalized = _strip_code_fences(raw_text)
        if normalized:
            candidates.append(normalized)

        extracted = _extract_balanced_json_like(normalized)
        if extracted and extracted not in candidates:
            candidates.append(extracted)

        for candidate in candidates:
            for text_variant in (candidate, _remove_trailing_commas(candidate)):
                try:
                    parsed = json.loads(text_variant)
                    if isinstance(parsed, expected):
                        return parsed
                except json.JSONDecodeError:
                    pass

                # Accept Python literal style structures (single quotes/None/True/False).
                try:
                    parsed_literal = ast.literal_eval(text_variant)
                except (SyntaxError, ValueError):
                    parsed_literal = None
                if isinstance(parsed_literal, expected):
                    return parsed_literal

        return None

    if isinstance(value, str):
        parsed = _try_parse_candidates(value)
        if parsed is not None:
            return parsed
        raise ValueError(f"{field_name} must be valid JSON")

    if isinstance(expected, tuple):
        names = ", ".join(t.__name__ for t in expected)
        raise ValueError(f"{field_name} must be one of ({names}) or JSON-encoded equivalent")
    raise ValueError(f"{field_name} must be a {expected.__name__} or JSON-encoded {expected.__name__}")


def _canonical_target_key(raw_key: str) -> str:
    key = str(raw_key).strip().lower()
    return _TARGET_KEY_ALIASES.get(key, key)


def _objective_weight(target_name: str, target_value: float) -> float:
    canonical_key = _canonical_target_key(target_name)
    priority = _TARGET_PRIORITY_WEIGHTS.get(canonical_key, 1.0)
    return priority / max(abs(float(target_value)), 1.0)


def _range_deviation(value: float, bounds: Dict[str, float]) -> Optional[float]:
    parsed = _extract_range_bounds(bounds)
    if parsed is None:
        return None

    lower, upper = float(parsed[0]), float(parsed[1])
    if value < lower:
        return lower - value
    if value > upper:
        return value - upper
    return 0.0


def _build_infeasibility_details(
    achieved_targets: Dict[str, float],
    point_targets: Dict[str, float],
    range_targets: Dict[str, Dict[str, float]],
) -> Dict[str, float]:
    details: Dict[str, float] = {}

    if "calories" in point_targets:
        calorie_difference = float(achieved_targets.get("calories", 0.0)) - float(point_targets["calories"])
        details[_CALORIE_DIFFERENCE_KEY] = round(calorie_difference, 4)

    if "total_fibre" in point_targets:
        fibre_difference = float(achieved_targets.get("total_fibre", 0.0)) - float(point_targets["total_fibre"])
        details[_FIBRE_DIFFERENCE_KEY] = round(fibre_difference, 4)

    for target_name, output_key in _RANGE_DEVIATION_KEYS.items():
        bounds = range_targets.get(target_name)
        if not isinstance(bounds, dict):
            continue
        deviation = _range_deviation(float(achieved_targets.get(target_name, 0.0)), bounds)
        if deviation is None:
            continue
        details[output_key] = round(deviation, 4)

    return details


def _build_unavailable_target_payload(
    point_targets: List[str],
    range_targets: List[str],
) -> Dict[str, List[str]]:
    payload: Dict[str, List[str]] = {}
    if point_targets:
        payload["point_targets"] = sorted(set(point_targets))
    if range_targets:
        payload["range_targets"] = sorted(set(range_targets))
    return payload


def _build_infeasible_reasons(
    infeasibility_details: Dict[str, float],
    unavailable_targets: Dict[str, List[str]],
) -> List[str]:
    reasons: List[str] = []

    calorie_difference = infeasibility_details.get(_CALORIE_DIFFERENCE_KEY)
    if calorie_difference is not None and abs(float(calorie_difference)) > _CALORIE_DIFFERENCE_TOLERANCE:
        reasons.append("calorie_difference_exceeds_tolerance")

    fibre_difference = infeasibility_details.get(_FIBRE_DIFFERENCE_KEY)
    if fibre_difference is not None and abs(float(fibre_difference)) > _FIBRE_DIFFERENCE_TOLERANCE:
        reasons.append("fibre_difference_exceeds_tolerance")

    for output_key in _RANGE_DEVIATION_KEYS.values():
        if float(infeasibility_details.get(output_key, 0.0)) > 1e-6:
            reasons.append(output_key)

    if unavailable_targets.get("point_targets") or unavailable_targets.get("range_targets"):
        reasons.append("missing_nutrient_data")

    return reasons


def _build_optimization_message(status: str, reasons: List[str]) -> str:
    if status == "feasible":
        return "Optimization satisfied calorie/fibre tolerances and macro ranges."
    if "missing_nutrient_data" in reasons:
        return "Optimization is infeasible and some required nutrient data is unavailable."
    return "Optimization is infeasible for the current calorie/fibre tolerances or macro ranges."


def _extract_number(value: Any) -> Optional[float]:
    if value is None:
        return None
    if isinstance(value, (int, float)):
        return float(value)
    text = str(value).replace(",", "")
    match = re.search(r"-?\d+(?:\.\d+)?", text)
    if not match:
        return None
    try:
        return float(match.group(0))
    except ValueError:
        return None


def _extract_first_number_from_mapping(
    data: Dict[str, Any],
    explicit_keys: List[str],
    contains_tokens: Optional[List[str]] = None,
) -> Optional[float]:
    if not isinstance(data, dict):
        return None

    normalized: Dict[str, Any] = {}
    for raw_key, raw_value in data.items():
        normalized[str(raw_key).strip().lower()] = raw_value

    for key in explicit_keys:
        candidate_key = str(key).strip().lower()
        if candidate_key in normalized:
            number = _extract_number(normalized.get(candidate_key))
            if number is not None:
                return number

    if contains_tokens:
        required = [token.strip().lower() for token in contains_tokens if token.strip()]
        for key, value in normalized.items():
            if all(token in key for token in required):
                number = _extract_number(value)
                if number is not None:
                    return number

    return None


def _nutrition_value(nutrition: Dict[str, Any], key: str) -> float:
    normalized = _canonical_target_key(key)
    candidates = _NUTRITION_KEY_CANDIDATES.get(normalized, [normalized])
    for candidate in candidates:
        if candidate in nutrition:
            return _to_float(nutrition.get(candidate), 0.0)
    return 0.0


def _extract_range_bounds(value: Any) -> Optional[Tuple[float, float]]:
    if not isinstance(value, dict):
        return None
    lower = _extract_number(value.get("lower"))
    upper = _extract_number(value.get("upper"))
    if lower is None or upper is None:
        return None
    if lower > upper:
        lower, upper = upper, lower
    return lower, upper


def _build_target_bundle_from_dri_payload(
    dri: Dict[str, Any],
) -> Tuple[Dict[str, float], Dict[str, Dict[str, float]]]:
    if isinstance(dri.get("result"), dict):
        dri_result = dri.get("result")
        if isinstance(dri_result, dict):
            dri = dri_result

    targets: Dict[str, float] = {}
    range_targets: Dict[str, Dict[str, float]] = {}
    derived = dri.get("derived") if isinstance(dri.get("derived"), dict) else {}

    # Support both compact/new DRI schema and previous schema variants.
    calories_payload = dri.get("calories")
    if isinstance(calories_payload, dict):
        eer = _extract_number(calories_payload.get("eer_kcal"))
    else:
        eer = _extract_number(calories_payload)

    if eer is None:
        energy_payload = dri.get("energy")
        if isinstance(energy_payload, dict):
            eer = _extract_number(energy_payload.get("eer_kcal"))
        else:
            eer = _extract_number(energy_payload)
    if eer is None:
        recommended_per_day = dri.get("recommended_g_per_day", {})
        if isinstance(recommended_per_day, dict):
            eer = _extract_first_number_from_mapping(
                recommended_per_day,
                explicit_keys=[
                    "calories eer_kcal",
                    "calories_eer_kcal",
                    "eer_kcal",
                    "calories",
                    "energy_kcal",
                    "energy",
                ],
                contains_tokens=["calorie", "eer"],
            )
    if eer is not None:
        targets["calories"] = eer

    recommended_macros = dri.get("macronutrient_recommended_g_per_day", {})
    if not isinstance(recommended_macros, dict) or not recommended_macros:
        recommended_macros = dri.get("recommended_g_per_day", {})
    if not isinstance(recommended_macros, dict):
        recommended_macros = {}

    protein = _extract_first_number_from_mapping(
        recommended_macros,
        explicit_keys=["Total protein", "total protein", "protein"],
    )
    if protein is None:
        protein = _extract_number(derived.get("total_protein_g_day"))
    if protein is not None:
        targets["protein"] = protein

    fibre = _extract_first_number_from_mapping(
        recommended_macros,
        explicit_keys=[
            "total fibre",
            "Total fibre",
            "total fiber",
            "Total fiber",
            "fibre",
            "fiber",
        ],
    )
    if fibre is None:
        fibre = _extract_number(derived.get("fibre_g_day"))
    if fibre is not None:
        targets["total_fibre"] = fibre

    carbohydrates = _extract_first_number_from_mapping(
        recommended_macros,
        explicit_keys=["Carbohydrates", "carbohydrates", "carbs", "carbohydrate"],
    )
    if carbohydrates is not None:
        targets["carbohydrates"] = carbohydrates

    total_fat = _extract_first_number_from_mapping(
        recommended_macros,
        explicit_keys=["Total Fat", "total fat", "fats", "fat", "total_fat"],
    )
    if total_fat is not None:
        targets["total_fat"] = total_fat

    macros = dri.get("macronutrients", [])
    if isinstance(macros, list):
        for row in macros:
            if not isinstance(row, dict):
                continue
            nutrient = str(row.get("nutrient", "")).strip().lower()
            intake = _extract_number(row.get("recommended_intake"))
            if intake is None:
                continue
            if nutrient == "carbohydrates":
                targets["carbohydrates"] = intake
            elif nutrient in {"total fat", "fats", "fat"}:
                targets["total_fat"] = intake
            elif nutrient in {"protein", "total protein"}:
                targets["protein"] = intake

    amdr = dri.get("macronutrient_ranges_g_per_day", {})
    if not isinstance(amdr, dict) or not amdr:
        amdr = dri.get("recommended_macronutrient_ranges_g_per_day", {})
    if not isinstance(amdr, dict) or not amdr:
        amdr = dri.get("amdr", {}).get("grams", {}) if isinstance(dri.get("amdr"), dict) else {}
    if isinstance(amdr, dict):
        for raw_key, raw_bounds in amdr.items():
            canonical_key = _canonical_target_key(str(raw_key))
            if canonical_key not in _OPTIMIZER_RANGE_TARGET_KEYS:
                continue
            bounds = _extract_range_bounds(raw_bounds)
            if bounds is None:
                continue
            range_targets[canonical_key] = {
                "lower": float(bounds[0]),
                "upper": float(bounds[1]),
            }

    # If a macro range exists, treat it as a hard constraint and avoid also
    # optimizing that macro to a single point target.
    for key in range_targets:
        targets.pop(key, None)

    return targets, range_targets


def _filter_optimizer_targets(targets: Dict[str, float]) -> Dict[str, float]:
    filtered: Dict[str, float] = {}
    for key, value in targets.items():
        canonical_key = _canonical_target_key(key)
        if canonical_key in _OPTIMIZER_TARGET_KEYS:
            filtered[canonical_key] = float(value)
    return filtered


def _filter_optimizer_target_ranges(
    target_ranges: Dict[str, Dict[str, float]],
) -> Dict[str, Dict[str, float]]:
    filtered: Dict[str, Dict[str, float]] = {}
    for key, bounds in target_ranges.items():
        canonical_key = _canonical_target_key(key)
        if canonical_key not in _OPTIMIZER_RANGE_TARGET_KEYS:
            continue
        parsed = _extract_range_bounds(bounds)
        if parsed is None:
            continue
        filtered[canonical_key] = {
            "lower": float(parsed[0]),
            "upper": float(parsed[1]),
        }
    return filtered


def _macro_nutrients_from_calculated(nutrients_per_day: Dict[str, float]) -> Dict[str, float]:
    macro_out: Dict[str, float] = {}
    for key, value in nutrients_per_day.items():
        canonical = _canonical_target_key(key)
        if canonical in _MACRO_DAILY_TARGET_KEYS:
            macro_out[canonical] = round(float(value), 4)

    # Keep output schema stable even when source nutrition lacks fibre values.
    if "total_fibre" not in macro_out:
        macro_out["total_fibre"] = 0.0

    return macro_out


def _create_mip_solver() -> Optional[pywraplp.Solver]:
    for solver_name in ("CBC_MIXED_INTEGER_PROGRAMMING", "SCIP", "SAT"):
        solver = pywraplp.Solver.CreateSolver(solver_name)
        if solver is not None:
            return solver
    return None


def _optimize_meal_options_schema(
    meal_options: Dict[str, Dict[str, List[Dict[str, Any]]]],
    targets: Dict[str, float],
    target_ranges: Dict[str, Dict[str, float]],
    products_by_index: Dict[int, Dict[str, Any]],
) -> Dict[str, Any]:
    if not meal_options:
        raise ValueError("meals_json contains no valid meals/options")
    if not targets and not target_ranges:
        raise ValueError("No targets available from latest calculate_health_canada_dri output")

    solver = _create_mip_solver()
    if solver is None:
        raise RuntimeError("Failed to initialize OR-Tools mixed-integer solver")

    quantity_vars: Dict[Tuple[str, str, int], Any] = {}
    choose_vars: Dict[Tuple[str, str], Any] = {}
    item_meta: Dict[Tuple[str, str, int], Dict[str, Any]] = {}
    ignored_items: List[Dict[str, Any]] = []

    for meal_name, options in meal_options.items():
        valid_option_keys: List[str] = []
        for option_name, items in options.items():
            choice_var = solver.BoolVar(f"choose__{meal_name}__{option_name}")
            choose_vars[(meal_name, option_name)] = choice_var
            added_any = False

            for entry_idx, item in enumerate(items):
                try:
                    index = int(item.get("index"))
                except (TypeError, ValueError):
                    ignored_items.append({"meal": meal_name, "option": option_name, "item": item, "reason": "invalid_index"})
                    continue

                product = products_by_index.get(index)
                if not product:
                    ignored_items.append({"meal": meal_name, "option": option_name, "item": item, "reason": "missing_product"})
                    continue

                min_qty, max_qty = _parse_quantity_range(item)
                q = solver.NumVar(0.0, max_qty, f"q__{meal_name}__{option_name}__{index}__{entry_idx}")

                max_constraint = solver.Constraint(-solver.infinity(), 0.0)
                max_constraint.SetCoefficient(q, 1.0)
                max_constraint.SetCoefficient(choice_var, -max_qty)

                min_constraint = solver.Constraint(0.0, solver.infinity())
                min_constraint.SetCoefficient(q, 1.0)
                min_constraint.SetCoefficient(choice_var, -min_qty)

                quantity_vars[(meal_name, option_name, entry_idx)] = q
                item_meta[(meal_name, option_name, entry_idx)] = {
                    "index": index,
                    "nutrition": (
                        _to_dict(product.get("nutrition_100g"))
                        or _to_dict(product.get("nutrition"))
                    ),
                    "cost_per_g": _to_float(_cost_per_gram(product), 0.0),
                }
                added_any = True

            if added_any:
                valid_option_keys.append(option_name)
            else:
                remove_constraint = solver.Constraint(0.0, 0.0)
                remove_constraint.SetCoefficient(choice_var, 1.0)

        if not valid_option_keys:
            raise ValueError(f"Meal '{meal_name}' has no valid options after matching ingredient indices")

        choose_constraint = solver.Constraint(1.0, 1.0)
        for option_name in valid_option_keys:
            choose_constraint.SetCoefficient(choose_vars[(meal_name, option_name)], 1.0)

    objective = solver.Objective()
    unavailable_point_targets: List[str] = []
    unavailable_range_targets: List[str] = []

    for target_name, bounds in target_ranges.items():
        lower = _to_float(bounds.get("lower"), 0.0)
        upper = _to_float(bounds.get("upper"), 0.0)
        if lower > upper:
            lower, upper = upper, lower

        coeff_by_key: Dict[Tuple[str, str, int], float] = {}
        for key, q in quantity_vars.items():
            coeff = _nutrition_value(item_meta[key].get("nutrition", {}), target_name) / 100.0
            if coeff != 0.0:
                coeff_by_key[key] = coeff

        if not coeff_by_key:
            if lower > 0.0:
                unavailable_range_targets.append(target_name)

        lower_violation = solver.NumVar(0.0, solver.infinity(), f"{target_name}_lower_violation")
        upper_violation = solver.NumVar(0.0, solver.infinity(), f"{target_name}_upper_violation")

        lower_constraint = solver.Constraint(float(lower), solver.infinity())
        upper_constraint = solver.Constraint(-solver.infinity(), float(upper))
        for key, coeff in coeff_by_key.items():
            lower_constraint.SetCoefficient(quantity_vars[key], coeff)
            upper_constraint.SetCoefficient(quantity_vars[key], coeff)
        lower_constraint.SetCoefficient(lower_violation, 1.0)
        upper_constraint.SetCoefficient(upper_violation, -1.0)

        range_anchor = max(abs(float(lower)), abs(float(upper)), 1.0)
        weight = _objective_weight(target_name, range_anchor) * _RANGE_DEVIATION_PRIORITY_MULTIPLIER
        objective.SetCoefficient(lower_violation, weight)
        objective.SetCoefficient(upper_violation, weight)

    for target_name, target_value in targets.items():
        coeff_by_key: Dict[Tuple[str, str, int], float] = {}
        for key, q in quantity_vars.items():
            coeff = _nutrition_value(item_meta[key].get("nutrition", {}), target_name) / 100.0
            if coeff != 0.0:
                coeff_by_key[key] = coeff

        if not coeff_by_key:
            if abs(float(target_value)) > 1e-9:
                unavailable_point_targets.append(target_name)

        pos = solver.NumVar(0.0, solver.infinity(), f"{target_name}_over")
        neg = solver.NumVar(0.0, solver.infinity(), f"{target_name}_under")

        target_constraint = solver.Constraint(float(target_value), float(target_value))
        for key, coeff in coeff_by_key.items():
            target_constraint.SetCoefficient(quantity_vars[key], coeff)
        target_constraint.SetCoefficient(pos, -1.0)
        target_constraint.SetCoefficient(neg, 1.0)

        weight = _objective_weight(target_name, target_value)
        objective.SetCoefficient(pos, weight)
        objective.SetCoefficient(neg, weight)

    objective.SetMinimization()
    status = solver.Solve()
    if status not in (pywraplp.Solver.OPTIMAL, pywraplp.Solver.FEASIBLE):
        response: Dict[str, Any] = {
            "status": "infeasible",
            "message": "Solver could not find a feasible solution",
            "ignored_items": ignored_items,
            "infeasible_reasons": ["solver_failed"],
        }
        unavailable_targets = _build_unavailable_target_payload(
            unavailable_point_targets,
            unavailable_range_targets,
        )
        if unavailable_targets:
            response["unavailable_targets"] = unavailable_targets
        return response

    average_calculated_quantity_per_day: Dict[str, Dict[str, List[Dict[str, float]]]] = {}

    for meal_name, options in meal_options.items():
        selected_option: Optional[str] = None
        for option_name in options.keys():
            chooser = choose_vars.get((meal_name, option_name))
            if chooser is not None and chooser.solution_value() > 0.5:
                selected_option = option_name
                break

        if selected_option is None:
            continue

        selected_items_daily: List[Dict[str, float]] = []
        for key, q in quantity_vars.items():
            meal_k, option_k, entry_idx = key
            if meal_k != meal_name or option_k != selected_option:
                continue
            qty = q.solution_value()
            if qty <= 1e-6:
                continue
            selected_items_daily.append(
                {
                    "index": item_meta[key]["index"],
                    "qty": round(float(qty), 4),
                }
            )
        average_calculated_quantity_per_day[meal_name] = {selected_option: selected_items_daily}

    calculated_macro_nutrients_per_day: Dict[str, float] = {}
    nutrition_per_meal: Dict[str, Dict[str, float]] = {}
    for key, q in quantity_vars.items():
        qty = q.solution_value()
        nutrition = item_meta[key].get("nutrition", {})
        if not isinstance(nutrition, dict):
            continue
        meal_k, option_k, entry_idx = key
        for nutrient_key, nutrient_value in nutrition.items():
            canonical_nutrient = _canonical_target_key(nutrient_key)
            if canonical_nutrient not in _MACRO_DAILY_TARGET_KEYS:
                continue
            nutrient_amount = qty * (_to_float(nutrient_value, 0.0) / 100.0)
            if nutrient_amount == 0.0:
                continue
            # Add to daily totals
            current = calculated_macro_nutrients_per_day.get(canonical_nutrient, 0.0)
            calculated_macro_nutrients_per_day[canonical_nutrient] = current + nutrient_amount
            # Add to per-meal totals
            if meal_k not in nutrition_per_meal:
                nutrition_per_meal[meal_k] = {}
            current_meal = nutrition_per_meal[meal_k].get(canonical_nutrient, 0.0)
            nutrition_per_meal[meal_k][canonical_nutrient] = current_meal + nutrient_amount

    average_macro_nutrients = _macro_nutrients_from_calculated(calculated_macro_nutrients_per_day)
    
    # Format nutrition per meal with standard keys
    nutrition_per_meal_formatted: Dict[str, Dict[str, float]] = {}
    for meal_name, meal_nutrition in nutrition_per_meal.items():
        nutrition_per_meal_formatted[meal_name] = _macro_nutrients_from_calculated(meal_nutrition)
    
    unavailable_targets = _build_unavailable_target_payload(
        unavailable_point_targets,
        unavailable_range_targets,
    )
    infeasibility_details = _build_infeasibility_details(
        average_macro_nutrients,
        targets,
        target_ranges,
    )
    infeasible_reasons = _build_infeasible_reasons(infeasibility_details, unavailable_targets)
    result_status = "infeasible" if infeasible_reasons else "feasible"

    response: Dict[str, Any] = {
        "status": result_status,
        "message": _build_optimization_message(result_status, infeasible_reasons),
        "average_calculated_quantity_per_day": average_calculated_quantity_per_day,
        "average_macro_nutrient_from_calculated_quantity_per_day": average_macro_nutrients,
        "nutrition_per_meal": nutrition_per_meal_formatted,
    }

    if ignored_items:
        response["ignored_items"] = ignored_items
    if unavailable_targets:
        response["unavailable_targets"] = unavailable_targets
    if result_status == "infeasible":
        response["infeasibility_details"] = infeasibility_details
        response["infeasible_reasons"] = infeasible_reasons

    return response


async def _optimize_quantity_impl(
    meals_json: str,
    number_of_days: int = 1,
    tool_context: Optional[ToolContext] = None,
) -> Any:
    if tool_context is None:
        raise ValueError("tool_context is required")

    meals = _parse_json_input(meals_json, dict, "meals_json")

    if not meals:
        raise ValueError("meals_json must be a non-empty dict with day -> meal -> option -> items structure")

    products_by_index = _extract_products_from_history(tool_context)
    if not products_by_index:
        raise ValueError("No prior find_ingredient tool results found in conversation history")

    history_target_json = _extract_latest_function_response_from_history(tool_context, "calculate_health_canada_dri")
    if not history_target_json:
        raise ValueError("No prior calculate_health_canada_dri tool results found in conversation history")

    targets, target_ranges = _build_target_bundle_from_dri_payload(history_target_json)
    optimizer_targets = _filter_optimizer_targets(targets)
    optimizer_target_ranges = _filter_optimizer_target_ranges(target_ranges)

    if not optimizer_targets and not optimizer_target_ranges:
        raise ValueError("No macro/calorie targets available in latest calculate_health_canada_dri output")

    if _should_skip_repeated_optimizer_call(tool_context, optimizer_targets, optimizer_target_ranges):
        return {
            "status": "feasible",
            "converged_in_previous_tool_call": True,
            "message": "Optimizer converged in previous tool call with the same targets; Don't call this tool unless user preferences or targets have changed. Proceed to the next step in the workflow.",
        }

    # Build per-day meal options (cheap, synchronous)
    days_to_optimize: List[Tuple[str, Dict[str, Dict[str, List[Dict[str, Any]]]]]] = []
    for day_name, day_meals in meals.items():
        if not isinstance(day_meals, dict):
            continue
        day_meal_options: Dict[str, Dict[str, List[Dict[str, Any]]]] = {}
        for meal_name, meal_options in day_meals.items():
            if not isinstance(meal_options, dict):
                continue
            parsed_options: Dict[str, List[Dict[str, Any]]] = {}
            for option_name, option_items in meal_options.items():
                if not isinstance(option_items, list):
                    continue
                clean_items: List[Dict[str, Any]] = []
                for item in option_items:
                    if not isinstance(item, dict):
                        continue
                    index = item.get("index")
                    if index is None:
                        continue
                    clean_items.append(item)
                if clean_items:
                    parsed_options[str(option_name)] = clean_items
            if parsed_options:
                day_meal_options[meal_name] = parsed_options
        if day_meal_options:
            days_to_optimize.append((day_name, day_meal_options))

    import asyncio as _asyncio

    async def _optimize_day(
        day_meal_opts: Dict[str, Dict[str, List[Dict[str, Any]]]],
    ) -> Dict[str, Any]:
        return await _asyncio.to_thread(
            _optimize_meal_options_schema,
            meal_options=day_meal_opts,
            targets=optimizer_targets,
            target_ranges=optimizer_target_ranges,
            products_by_index=products_by_index,
        )

    day_results_list: List[Dict[str, Any]] = await _asyncio.gather(
        *[_optimize_day(day_meal_opts) for _, day_meal_opts in days_to_optimize]
    )

    # Aggregate results
    daily_results: Dict[str, Dict[str, Any]] = {}
    all_ignored_items: List[Dict[str, Any]] = []
    all_feasible = True
    average_nutrition: Dict[str, float] = {}
    combined_unavailable_targets: Dict[str, List[str]] = {"point_targets": [], "range_targets": []}

    for (day_name, _), day_result in zip(days_to_optimize, day_results_list):
        daily_results[day_name] = day_result

        if day_result.get("status") == "infeasible":
            all_feasible = False

        if "ignored_items" in day_result:
            all_ignored_items.extend(day_result["ignored_items"])

        if "unavailable_targets" in day_result:
            unavail = day_result["unavailable_targets"]
            if "point_targets" in unavail:
                combined_unavailable_targets["point_targets"].extend(unavail["point_targets"])
            if "range_targets" in unavail:
                combined_unavailable_targets["range_targets"].extend(unavail["range_targets"])

        day_nutrition = day_result.get("average_macro_nutrient_from_calculated_quantity_per_day", {})
        for nutrient, value in day_nutrition.items():
            average_nutrition[nutrient] = average_nutrition.get(nutrient, 0.0) + float(value)
    
    # Average the nutrition across days
    num_days = len(daily_results)
    if num_days > 0:
        for nutrient in average_nutrition:
            average_nutrition[nutrient] = round(average_nutrition[nutrient] / num_days, 4)
    
    # Deduplicate unavailable targets
    if combined_unavailable_targets["point_targets"]:
        combined_unavailable_targets["point_targets"] = sorted(set(combined_unavailable_targets["point_targets"]))
    else:
        del combined_unavailable_targets["point_targets"]
    
    if combined_unavailable_targets["range_targets"]:
        combined_unavailable_targets["range_targets"] = sorted(set(combined_unavailable_targets["range_targets"]))
    else:
        del combined_unavailable_targets["range_targets"]
    
    # Build response
    result_status = "feasible" if all_feasible else "infeasible"
    
    response: Dict[str, Any] = {
        "status": result_status,
        "message": _build_optimization_message(result_status, [] if all_feasible else ["day_infeasible"]),
        "calculated_quantity_per_day": {day: result.get("average_calculated_quantity_per_day", {}) for day, result in daily_results.items() if result.get("status") == "feasible"},
        "nutrition_per_meal": {
            day: _get_nutrition_per_meal_by_day(result) 
            for day, result in daily_results.items() 
            if result.get("status") == "feasible"
        },
        "achieved_targets_per_day": average_nutrition,
    }
    
    if all_ignored_items:
        response["ignored_items"] = all_ignored_items
    
    if combined_unavailable_targets:
        response["unavailable_targets"] = combined_unavailable_targets
    
    return response


def _get_nutrition_per_meal_by_day(day_result: Dict[str, Any]) -> Dict[str, Dict[str, float]]:
    """Extract nutrition per meal from a day's optimization result."""
    return day_result.get("nutrition_per_meal", {})


async def optimize_quantity(
    meals_json: str,
    number_of_days: int = 1,
    tool_context: Optional[ToolContext] = None,
) -> Any:
    """
    Optimize ingredient quantities for candidate meals using products returned by prior `find_ingredient` calls.
    
    Each day is optimized independently, then results are averaged across days.

    Args:
        meals_json: JSON string in day -> meal -> option -> items format:
            {
                "day_1": {
                    "meal_1": {
                        "option_1": [
                            {"index": 2, "min_qty": 100, "max_qty": 300},
                            {"index": 40, "min_qty": 50, "max_qty": 200}
                        ],
                        "option_2": [
                            {"index": 10, "min_qty": 60, "max_qty": 180}
                        ]
                    },
                    "meal_2": { ... }
                },
                "day_2": { ... }
            }
        
        number_of_days: Deprecated - each day is optimized independently.

    Returns:
        Success (status='feasible'): dict with:
            - status: 'feasible'
            - calculated_quantity_per_day: quantities per day/meal/option/items structure
            - nutrition_per_meal: nutrition breakdown (protein, carbohydrates, total_fat, calories, total_fibre) for each day/meal
            - achieved_targets_per_day: averaged daily macro totals across all days
        
        Infeasible cases (status='infeasible'): dict with:
            - status: 'infeasible'
            - message: reason for infeasibility
            - ignored_items: items that couldn't be matched
            - unavailable_targets: targets without nutrient data (if applicable)
            - calculated_quantity_per_day: optimizer quantities for feasible days
            - nutrition_per_meal: nutrition breakdown for feasible days
            - achieved_targets_per_day: averaged daily macro totals

    """
    try:
        return await _optimize_quantity_impl(
            meals_json=meals_json,
            number_of_days=number_of_days,
            tool_context=tool_context,
        )
    except (TypeError, ValueError, RuntimeError) as exc:
        return f"Error: {exc}"


def _build_dummy_tool_context_for_local_run(dri_response: Optional[Dict[str, Any]] = None) -> Any:
    from types import SimpleNamespace

    dummy_products = [
        {
            "index": 2,
            "name": "Greek yogurt",
            "nutrition_100g": {
                "protein": 10.0,
                "carbohydrates": 3.6,
                "total_fat": 0.4,
                "total_fibre": 0.0,
                "calories": 59,
            },
            "cost": {"price": 2.2, "quantity": 500.0, "unit": "g"},
        },
        {
            "index": 4,
            "name": "Cooked rice",
            "nutrition_100g": {
                "protein": 2.7,
                "carbohydrates": 28.0,
                "total_fat": 0.3,
                "total_fibre": 0.4,
                "calories": 130,
            },
            "cost": {"price": 2.0, "quantity": 1.0, "unit": "kg"},
        },
        {
            "index": 5,
            "name": "Chicken breast",
            "nutrition_100g": {
                "protein": 31.0,
                "carbohydrates": 0.0,
                "total_fat": 3.6,
                "total_fibre": 0.0,
                "calories": 165,
            },
            "cost": {"price": 6.0, "quantity": 1.0, "unit": "kg"},
        },
        {
            "index": 7,
            "name": "Tofu",
            "nutrition_100g": {
                "protein": 8.1,
                "carbohydrates": 1.9,
                "total_fat": 4.8,
                "total_fibre": 0.3,
                "calories": 76,
            },
            "cost": {"price": 2.5, "quantity": 400.0, "unit": "g"},
        },
        {
            "index": 10,
            "name": "Whole wheat bread",
            "nutrition_100g": {
                "protein": 12.0,
                "carbohydrates": 43.0,
                "total_fat": 4.2,
                "total_fibre": 7.0,
                "calories": 247,
            },
            "cost": {"price": 1.8, "quantity": 500.0, "unit": "g"},
        },
        {
            "index": 40,
            "name": "Banana",
            "nutrition_100g": {
                "protein": 1.1,
                "carbohydrates": 23.0,
                "total_fat": 0.3,
                "total_fibre": 2.6,
                "calories": 96,
            },
            "cost": {"price": 2.0, "quantity": 1.0, "unit": "kg"},
        },
    ]

    events = []

    if isinstance(dri_response, dict):
        dri_function_response = SimpleNamespace(name="calculate_health_canada_dri", response=dri_response)
        dri_part = SimpleNamespace(function_response=dri_function_response)
        dri_content = SimpleNamespace(parts=[dri_part])
        events.append(SimpleNamespace(content=dri_content))

    function_response = SimpleNamespace(name="find_ingredient", response={"result": dummy_products})
    part = SimpleNamespace(function_response=function_response)
    content = SimpleNamespace(parts=[part])
    events.append(SimpleNamespace(content=content))

    session = SimpleNamespace(events=events)
    invocation_context = SimpleNamespace(session=session)
    return SimpleNamespace(_invocation_context=invocation_context)


async def _run_local_demo_case(
    title: str,
    meals_json: Dict[str, Any],
    target_json: Dict[str, Any],
    number_of_days: int = 1,
) -> None:
    print(f"\n=== {title} ===")
    meals_json = {
        **meals_json,
    }

    print("Input (meals_json):")
    print(json.dumps(meals_json, indent=2, ensure_ascii=False))
    print("\nInput (target_json):")
    print(json.dumps(target_json, indent=2, ensure_ascii=False))
    print(f"\nInput (number_of_days): {number_of_days}")

    result = await optimize_quantity(
        meals_json=meals_json,
        number_of_days=number_of_days,
        tool_context=_build_dummy_tool_context_for_local_run(dri_response=target_json),
    )

    print("\nOutput (optimizer_result):")
    print(json.dumps(result, indent=2, ensure_ascii=False))


async def _run_local_demo() -> None:
    number_of_days = 2

    feasible_meals_json = {
        "day_1": {
            "meal_1": {
                "option_1": [
                    {"index": 2, "min_qty": 0, "max_qty": 600},
                    {"index": 4, "min_qty": 0, "max_qty": 900},
                ],
                "option_2": [
                    {"index": 5, "min_qty": 0, "max_qty": 700},
                    {"index": 40, "min_qty": 0, "max_qty": 1300},
                ],
            },
            "meal_2": {
                "option_1": [
                    {"index": 2, "min_qty": 0, "max_qty": 600},
                    {"index": 4, "min_qty": 0, "max_qty": 900},
                ],
                "option_2": [
                    {"index": 7, "min_qty": 0, "max_qty": 1200},
                    {"index": 10, "min_qty": 0, "max_qty": 900},
                ],
            },
        },
    }

    feasible_target_json = {
        "calories": {
            "eer_kcal": 2214,
        },
        "macronutrient_ranges_g_per_day": {
            "Carbohydrates": {"lower": 249, "upper": 360},
            "Protein": {"lower": 55, "upper": 194},
            "Fats": {"lower": 49, "upper": 86},
        },
        "macronutrient_recommended_g_per_day": {
            "Total fibre": 31.0,
        },
    }

    await _run_local_demo_case(
        title="Feasible Optimization Demo",
        meals_json=feasible_meals_json,
        target_json=feasible_target_json,
        number_of_days=number_of_days,
    )


async def _run_local_demo_2() -> None:
    number_of_days = 2

    fallback_meals_json = {
        "day_1": {
            "meal_1": {
                "option_1": [
                    {"index": 2, "min_qty": 0, "max_qty": 600},
                    {"index": 4, "min_qty": 0, "max_qty": 900},
                ],
                "option_2": [
                    {"index": 5, "min_qty": 0, "max_qty": 700},
                    {"index": 40, "min_qty": 0, "max_qty": 1300},
                ],
            },
            "meal_2": {
                "option_1": [
                    {"index": 2, "min_qty": 0, "max_qty": 600},
                    {"index": 4, "min_qty": 0, "max_qty": 900},
                ],
                "option_2": [
                    {"index": 7, "min_qty": 0, "max_qty": 1200},
                    {"index": 10, "min_qty": 0, "max_qty": 900},
                ],
            },
        },
    }

    infeasible_target_json = {
        "calories": {
            "eer_kcal": 2214,
        },
        "macronutrient_ranges_g_per_day": {
            "Carbohydrates": {"lower": 249, "upper": 360},
            "Protein": {"lower": 55, "upper": 194},
            "Fats": {"lower": 200, "upper": 230},
        },
        "macronutrient_recommended_g_per_day": {
            "Total fibre": 31.0,
        },
    }

    await _run_local_demo_case(
        title="Infeasible Demo (strict fat range is intentionally violated)",
        meals_json=fallback_meals_json,
        target_json=infeasible_target_json,
        number_of_days=number_of_days,
    )


if __name__ == "__main__":
    import asyncio

    asyncio.run(_run_local_demo())

    print("\n\n" + "=" * 200 + "\n\n")

    asyncio.run(_run_local_demo_2())


