from __future__ import annotations

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

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

from utils.optimizer import _collect_function_responses_from_history
from utils.optimizer import _build_dummy_tool_context_for_local_run
from utils.optimizer import _extract_latest_function_response_from_history
from utils.optimizer import _extract_products_from_history
from utils.optimizer import _filter_optimizer_targets
from utils.optimizer import _float_maps_match
from utils.optimizer import _macro_nutrients_from_calculated
from utils.optimizer import _optimize_meal_options_schema
from utils.optimizer import _parse_json_input


_MFP_CALORIE_TOLERANCE = 10.0
_MFP_MACRO_TOLERANCES = {
    "protein": 5.0,
    "carbohydrates": 2.0,
    "total_fat": 5.0,
}
_MFP_DETAIL_KEYS = {
    "calories": "daily_calorie_difference",
    "protein": "daily_protein_difference",
    "carbohydrates": "daily_carbohydrate_difference",
    "total_fat": "daily_total_fat_difference",
}
_MFP_REASON_KEYS = {
    "calories": "calorie_difference_exceeds_tolerance",
    "protein": "protein_difference_exceeds_tolerance",
    "carbohydrates": "carbohydrate_difference_exceeds_tolerance",
    "total_fat": "total_fat_difference_exceeds_tolerance",
}


def _to_float(value: Any) -> Optional[float]:
    try:
        if value is None:
            return None
        return float(value)
    except (TypeError, ValueError):
        return None


def _unwrap_result_payload(payload: Dict[str, Any]) -> Dict[str, Any]:
    nested = payload.get("result") if isinstance(payload, dict) else None
    if isinstance(nested, dict):
        return nested
    return payload if isinstance(payload, dict) else {}


def _build_target_bundle_from_mfp_payload(payload: Dict[str, Any]) -> Dict[str, float]:
    result_payload = _unwrap_result_payload(payload)

    targets: Dict[str, float] = {}

    calories = _to_float(result_payload.get("target_calories_kcal"))
    if calories is not None:
        targets["calories"] = calories

    macros = result_payload.get("macros_g_per_day") if isinstance(result_payload.get("macros_g_per_day"), dict) else {}

    protein = _to_float(macros.get("protein"))
    if protein is not None:
        targets["protein"] = protein

    carbohydrates = _to_float(macros.get("carbohydrates"))
    if carbohydrates is not None:
        targets["carbohydrates"] = carbohydrates

    total_fat = None
    for fat_key in ("total_fat", "fat", "fats"):
        total_fat = _to_float(macros.get(fat_key))
        if total_fat is not None:
            break
    if total_fat is not None:
        targets["total_fat"] = total_fat

    return _filter_optimizer_targets(targets)


def _build_mfp_infeasibility_details(
    achieved_targets: Dict[str, float],
    targets: Dict[str, float],
) -> Dict[str, float]:
    details: Dict[str, float] = {}
    for target_name, target_value in targets.items():
        detail_key = _MFP_DETAIL_KEYS.get(target_name)
        if not detail_key:
            continue
        achieved_value = float(achieved_targets.get(target_name, 0.0))
        details[detail_key] = round(achieved_value - float(target_value), 4)
    return details


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

    calorie_difference = infeasibility_details.get(_MFP_DETAIL_KEYS["calories"])
    if calorie_difference is not None and abs(float(calorie_difference)) > _MFP_CALORIE_TOLERANCE:
        reasons.append(_MFP_REASON_KEYS["calories"])

    for target_name, tolerance in _MFP_MACRO_TOLERANCES.items():
        difference = infeasibility_details.get(_MFP_DETAIL_KEYS[target_name])
        if difference is not None and abs(float(difference)) > tolerance:
            reasons.append(_MFP_REASON_KEYS[target_name])

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

    return reasons


def _dedupe_reasons(reasons: List[str]) -> List[str]:
    deduped: List[str] = []
    for reason in reasons:
        if reason not in deduped:
            deduped.append(reason)
    return deduped


def _build_mfp_optimization_message(status: str, reasons: List[str]) -> str:
    if status == "feasible":
        return "Optimization satisfied the MyFitnessPal calorie and macro targets."
    if "solver_failed" in reasons:
        return "Optimizer could not find a feasible solution for the current meal options and MyFitnessPal targets."
    if "missing_nutrient_data" in reasons:
        return "Optimization is infeasible and some required nutrient data is unavailable."
    return "Optimization is infeasible for the current MyFitnessPal calorie or macro targets."


def _collect_day_meal_options(
    meals: Dict[str, Any],
) -> List[Tuple[str, Dict[str, Dict[str, List[Dict[str, Any]]]]]]:
    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
                    if item.get("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))

    return days_to_optimize


def _should_skip_repeated_mfp_optimizer_call(
    tool_context: ToolContext,
    current_targets: 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_for_mfp_targets":
        return False

    last_result_payload = _unwrap_result_payload(last_payload if isinstance(last_payload, dict) else {})
    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_mfp_payload: Optional[Dict[str, Any]] = None
    for response_name, response_payload in reversed(responses[:-1]):
        if response_name == "calculate_mfp_macros" and isinstance(response_payload, dict):
            previous_mfp_payload = response_payload
            break

    if previous_mfp_payload is None:
        return False

    previous_targets = _build_target_bundle_from_mfp_payload(previous_mfp_payload)
    return _float_maps_match(previous_targets, current_targets)


async def _optimize_quantity_for_mfp_targets_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_mfp_macros")
    if not history_target_json:
        raise ValueError("No prior calculate_mfp_macros tool results found in conversation history")

    optimizer_targets = _build_target_bundle_from_mfp_payload(history_target_json)
    if not optimizer_targets:
        raise ValueError("No calorie or macro targets available in latest calculate_mfp_macros output")

    if _should_skip_repeated_mfp_optimizer_call(tool_context, optimizer_targets):
        return {
            "status": "feasible",
            "converged_in_previous_tool_call": True,
            "message": "Optimizer converged in previous tool call with the same MyFitnessPal targets; do not call this tool again unless targets or meal options change.",
        }

    days_to_optimize = _collect_day_meal_options(meals)
    if not days_to_optimize:
        raise ValueError("meals_json must contain at least one day with valid 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={},
            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]
    )

    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": []}
    collected_infeasible_reasons: List[str] = []
    averaged_day_count = 0

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

        if not isinstance(day_result, dict):
            all_feasible = False
            collected_infeasible_reasons.append("solver_failed")
            continue

        if day_result.get("status") == "infeasible":
            all_feasible = False
            collected_infeasible_reasons.extend(list(day_result.get("infeasible_reasons", [])))

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

        unavailable_targets = day_result.get("unavailable_targets")
        if isinstance(unavailable_targets, dict):
            if "point_targets" in unavailable_targets:
                combined_unavailable_targets["point_targets"].extend(unavailable_targets["point_targets"])
            if "range_targets" in unavailable_targets:
                combined_unavailable_targets["range_targets"].extend(unavailable_targets["range_targets"])

        day_nutrition = day_result.get("average_macro_nutrient_from_calculated_quantity_per_day", {})
        if isinstance(day_nutrition, dict) and day_nutrition:
            averaged_day_count += 1
            for nutrient, value in day_nutrition.items():
                average_nutrition[nutrient] = average_nutrition.get(nutrient, 0.0) + float(value)

    if averaged_day_count > 0:
        for nutrient in average_nutrition:
            average_nutrition[nutrient] = round(average_nutrition[nutrient] / averaged_day_count, 4)

    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"]

    achieved_targets = _macro_nutrients_from_calculated(average_nutrition)
    infeasibility_details = _build_mfp_infeasibility_details(achieved_targets, optimizer_targets)
    reasons = _dedupe_reasons(
        collected_infeasible_reasons + _build_mfp_infeasible_reasons(infeasibility_details, combined_unavailable_targets)
    )
    status = "feasible" if all_feasible and not reasons else "infeasible"

    response: Dict[str, Any] = {
        "status": status,
        "message": _build_mfp_optimization_message(status, reasons),
        "calculated_quantity_per_day": {
            day: result.get("average_calculated_quantity_per_day", {})
            for day, result in daily_results.items()
            if isinstance(result, dict) and result.get("average_calculated_quantity_per_day")
        },
        "nutrition_per_meal": {
            day: result.get("nutrition_per_meal", {})
            for day, result in daily_results.items()
            if isinstance(result, dict) and result.get("nutrition_per_meal")
        },
        "achieved_targets_per_day": achieved_targets,
    }

    if all_ignored_items:
        response["ignored_items"] = all_ignored_items

    if combined_unavailable_targets:
        response["unavailable_targets"] = combined_unavailable_targets

    if status == "infeasible":
        response["infeasibility_details"] = infeasibility_details
        response["infeasible_reasons"] = reasons

    return response


async def optimize_quantity_for_mfp_targets(
    meals_json: str,
    number_of_days: int = 1,
    tool_context: Optional[ToolContext] = None,
) -> Any:
    """
    Optimize ingredient quantities for candidate meals using targets from the latest `calculate_mfp_macros` result.

    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 compatibility parameter. Day-wise optimization is driven by the
            day keys present in `meals_json`.
        tool_context: ADK tool context used to read prior tool responses.

    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 meal
            - achieved_targets_per_day: daily macro totals

        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 if a relaxed primary solve succeeded
            - achieved_targets_per_day: achieved daily macro totals from the optimizer result
            - infeasibility_details: calorie and macro differences relative to MyFitnessPal targets
            - infeasible_reasons: machine-readable explanation keys
    """
    try:
        return await _optimize_quantity_for_mfp_targets_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_mfp(mfp_response: Optional[Dict[str, Any]] = None) -> Any:
    from types import SimpleNamespace

    tool_context = _build_dummy_tool_context_for_local_run()
    if not isinstance(mfp_response, dict):
        return tool_context

    mfp_function_response = SimpleNamespace(name="calculate_mfp_macros", response=mfp_response)
    mfp_part = SimpleNamespace(function_response=mfp_function_response)
    mfp_content = SimpleNamespace(parts=[mfp_part])
    tool_context._invocation_context.session.events.insert(0, SimpleNamespace(content=mfp_content))
    return tool_context


async def _run_local_demo_case(
    title: str,
    meals_json: Dict[str, Any],
    mfp_target_json: Dict[str, Any],
    number_of_days: int = 1,
) -> None:
    import json

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

    result = await optimize_quantity_for_mfp_targets(
        meals_json=meals_json,
        number_of_days=number_of_days,
        tool_context=_build_dummy_tool_context_for_local_run_mfp(mfp_response=mfp_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": 10, "min_qty": 0, "max_qty": 500},
                ],
                "option_2": [
                    {"index": 7, "min_qty": 0, "max_qty": 1200},
                    {"index": 4, "min_qty": 0, "max_qty": 900},
                ],
            },
        },
    }

    feasible_target_json = {
        "target_calories_kcal": 2214.0,
        "macros_g_per_day": {
            "protein": 171.0455,
            "carbohydrates": 286.9,
            "fat": 44.0782,
        },
        "goal": "maintain",
        "notes": ["Demo target aligned with the dummy ingredient set so the optimizer returns a feasible solution."],
    }

    infeasible_target_json = {
        "target_calories_kcal": 1400.0,
        "macros_g_per_day": {
            "protein": 220.0,
            "carbohydrates": 60.0,
            "fat": 20.0,
        },
        "goal": "lose weight",
        "notes": ["Demo target intentionally aggressive to show infeasible output."],
    }

    await _run_local_demo_case(
        title="Feasible-looking MyFitnessPal target",
        meals_json=feasible_meals_json,
        mfp_target_json=feasible_target_json,
        number_of_days=number_of_days,
    )

    await _run_local_demo_case(
        title="Likely infeasible MyFitnessPal target",
        meals_json=feasible_meals_json,
        mfp_target_json=infeasible_target_json,
        number_of_days=number_of_days,
    )


if __name__ == "__main__":
    import asyncio

    asyncio.run(_run_local_demo())