"""
Calculate Dietary Reference Intakes (DRIs) based on Health Canada's DRI Calculator.
Reference: https://health-infobase.canada.ca/nutrition/dietary-reference-intakes-calculator/
"""
from __future__ import annotations

import csv
import io
import math
from pathlib import Path
from functools import lru_cache
from typing import TYPE_CHECKING, Any, Dict, List, Optional

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


BASE_DIR = Path(__file__).resolve().parent



def _resolve_dri_reference_dir() -> Path:
	# Current layout: code/utils/*.py with the reference tables in sibling code/datasets/dri_reference.
	return BASE_DIR.parent / "datasets" / "dri_reference"


DRI_REFERENCE_DIR = _resolve_dri_reference_dir()
MACRO_CSV_PATH = DRI_REFERENCE_DIR / "Macronutrients_table-en.csv"

ACTIVITY_LEVELS = {"inactive", "low active", "active", "very active"}
SEX_VALUES = {"male", "female"}
LACTATION_STATUS_VALUES = {"none", "0-6 months postpartum", "7-12 months postpartum"}

FIBRE_CONST = 14 / 1000
PREGNANT_SECOND_TRIMESTER_START = 13

PROTEIN_INTAKES_G_PER_KG: Dict[str, Any] = {
	"0-6 months": {"M": 1.52, "F": 1.52},
	"7-12 months": {"M": 1.2, "F": 1.2},
	"1-3 years": {"M": 1.05, "F": 1.05},
	"4-8 years": {"M": 0.95, "F": 0.95},
	"9-13 years": {"M": 0.95, "F": 0.95},
	"14-18 years": {"M": 0.85, "F": 0.85},
	"19-30 years": {"M": 0.80, "F": 0.80},
	"31-50 years": {"M": 0.80, "F": 0.80},
	"51-70 years": {"M": 0.80, "F": 0.80},
	"71 years and over": {"M": 0.80, "F": 0.80},
	"Pregnant": lambda gestation_weeks: 0.8 if gestation_weeks < 20 else 1.1,
	"Lactation": 1.3,
}

AMDR_PERCENTS = {
	"19 years and over": {
		"Carbohydrates": {"lower": 0.45, "upper": 0.65},
		"Protein": {"lower": 0.10, "upper": 0.35},
		"Fats": {"lower": 0.20, "upper": 0.35},
	},
	"4-8 years": {
		"Carbohydrates": {"lower": 0.45, "upper": 0.65},
		"Protein": {"lower": 0.10, "upper": 0.30},
		"Fats": {"lower": 0.25, "upper": 0.35},
	},
	"1-3 years": {
		"Carbohydrates": {"lower": 0.45, "upper": 0.65},
		"Protein": {"lower": 0.05, "upper": 0.20},
		"Fats": {"lower": 0.30, "upper": 0.40},
	},
}

PREFERRED_MACRO_ORDER = [
	"Carbohydrates",
	"Total protein",
	"Total Fat",
	"Linoleic Acid",
	"α-linolenic Acid",
	"Total fibre",
]


def _clean_csv_value(value: Optional[str]) -> str:
	if value is None:
		return ""
	return str(value).strip().lower()


@lru_cache(maxsize=4)
def _load_csv_rows(file_path: str) -> List[Dict[str, str]]:
	try:
		raw = Path(file_path).read_text(encoding="utf-8-sig")
	except OSError as exc:
		raise ValueError(f"Unable to read DRI reference table: {file_path}") from exc

	reader = csv.DictReader(io.StringIO(raw))
	rows: List[Dict[str, str]] = []
	for row in reader:
		rows.append({k: _clean_csv_value(v) for k, v in row.items()})
	return rows


def _sex_abbrev(sex: str) -> str:
	return "M" if sex == "male" else "F"


def _months_to_years(months: float) -> float:
	return months / 12.0


def _pounds_to_kg(pounds: float) -> float:
	return pounds * 0.45359


def _feet_inches_to_cm(feet: float, inches: float) -> float:
	return (feet + (inches / 12.0)) * 30.48


def _age_to_dri_age_range(age_years: float, is_pregnant: bool, is_breastfeeding: bool) -> str:
	if (is_pregnant or is_breastfeeding) and age_years <= 18:
		return "≤ 18 years"
	if age_years >= 71:
		return "71 years and over"
	if age_years >= 51:
		return "51-70 years"
	if age_years >= 31:
		return "31-50 years"
	if age_years >= 19:
		return "19-30 years"
	if age_years >= 14:
		return "14-18 years"
	if age_years >= 9:
		return "9-13 years"
	if age_years >= 4:
		return "4-8 years"
	if age_years >= 1:
		return "1-3 years"
	if age_years >= 7 / 12:
		return "7-12 months"
	return "0-6 months"


def _age_to_amdr_range(age_years: float) -> str:
	if age_years >= 19:
		return "19 years and over"
	if age_years >= 4:
		return "4-8 years"
	if age_years >= 1:
		return "1-3 years"
	return "under 1 year"


def _age_to_eer_group(age_years: float) -> str:
	if age_years >= 19:
		return "19+ years"
	if age_years >= 14:
		return "14 to 18.99 years"
	if age_years >= 3:
		return "3 to 13.99 years"
	if age_years >= 0.5:
		return "6 months to 2.99 years"
	if age_years >= 0.25:
		return "3 to 5.99 months"
	return "0 to 2.99 months"


def _get_dri_sex(sex: str, age_years: float, is_pregnant: bool, is_breastfeeding: bool) -> str:
	if is_pregnant:
		return "pregnant"
	if is_breastfeeding:
		return "lactation"
	if age_years < 9:
		return "both"
	return sex


def _get_bmi(height_cm: float, weight_kg: float) -> float:
	return weight_kg / math.pow(height_cm / 100.0, 2)


def _get_prepregnancy_weight_status(age_years: float, bmi: float) -> str:
	age_int = int(age_years)
	if age_int == 14:
		if bmi < 15.6:
			return "UW"
		if bmi <= 22.9:
			return "NW"
		if bmi <= 26.6:
			return "OW"
		return "OB"
	if age_int == 15:
		if bmi < 16:
			return "UW"
		if bmi <= 23.6:
			return "NW"
		if bmi <= 27.6:
			return "OW"
		return "OB"
	if age_int == 16:
		if bmi < 16.4:
			return "UW"
		if bmi <= 24.2:
			return "NW"
		if bmi <= 28.2:
			return "OW"
		return "OB"
	if age_int == 17:
		if bmi < 16.6:
			return "UW"
		if bmi <= 24.6:
			return "NW"
		if bmi <= 28.6:
			return "OW"
		return "OB"
	if age_int == 18:
		if bmi < 16.7:
			return "UW"
		if bmi <= 24.9:
			return "NW"
		if bmi <= 28.8:
			return "OW"
		return "OB"

	if age_years >= 19 and age_years < 51:
		if bmi < 18.5:
			return "UW"
		if bmi < 25:
			return "NW"
		if bmi < 30:
			return "OW"
		return "OB"

	return "NW"


def _energy_deposition(weight_status: str) -> float:
	if weight_status == "UW":
		return 300
	if weight_status == "NW":
		return 200
	if weight_status == "OW":
		return 150
	if weight_status == "OB":
		return -50
	return 200


def _calculate_eer(
	sex: str,
	age_years: float,
	height_cm: float,
	weight_kg: float,
	activity_level: Optional[str],
	is_pregnant: bool,
	gestation_weeks: Optional[float],
	prepregnancy_weight_kg: Optional[float],
	lactation_status: str,
) -> int:
	if sex == "female" and is_pregnant and gestation_weeks is not None and gestation_weeks >= PREGNANT_SECOND_TRIMESTER_START:
		if activity_level is None:
			raise ValueError("activity_level is required for pregnant second/third trimester calculations")
		if prepregnancy_weight_kg is None:
			raise ValueError("prepregnancy_weight is required for pregnant second/third trimester calculations")
		bmi = _get_bmi(height_cm, prepregnancy_weight_kg)
		status = _get_prepregnancy_weight_status(age_years, bmi)
		ed = _energy_deposition(status)
		if activity_level == "inactive":
			value = 1131.20 - (2.04 * age_years) + (0.34 * height_cm) + (12.15 * weight_kg) + (9.16 * gestation_weeks) + ed
		elif activity_level == "low active":
			value = 693.35 - (2.04 * age_years) + (5.73 * height_cm) + (10.20 * weight_kg) + (9.16 * gestation_weeks) + ed
		elif activity_level == "active":
			value = -223.84 - (2.04 * age_years) + (13.23 * height_cm) + (8.15 * weight_kg) + (9.16 * gestation_weeks) + ed
		else:
			value = -779.72 - (2.04 * age_years) + (18.45 * height_cm) + (8.73 * weight_kg) + (9.16 * gestation_weeks) + ed
		return round(value)

	if sex == "female" and lactation_status != "none":
		if activity_level is None:
			raise ValueError("activity_level is required for lactation calculations")
		milk_cost = 540 if lactation_status == "0-6 months postpartum" else 380
		mobilization = 140 if lactation_status == "0-6 months postpartum" else 0
		if age_years >= 19:
			if activity_level == "inactive":
				value = 584.90 - (7.01 * age_years) + (5.72 * height_cm) + (11.71 * weight_kg) + milk_cost - mobilization
			elif activity_level == "low active":
				value = 575.77 - (7.01 * age_years) + (6.60 * height_cm) + (12.14 * weight_kg) + milk_cost - mobilization
			elif activity_level == "active":
				value = 710.25 - (7.01 * age_years) + (6.54 * height_cm) + (12.34 * weight_kg) + milk_cost - mobilization
			else:
				value = 511.83 - (7.01 * age_years) + (9.07 * height_cm) + (12.56 * weight_kg) + milk_cost - mobilization
		else:
			if activity_level == "inactive":
				value = 55.59 - (22.25 * age_years) + (8.43 * height_cm) + (17.07 * weight_kg) + milk_cost - mobilization
			elif activity_level == "low active":
				value = -297.54 - (22.25 * age_years) + (12.77 * height_cm) + (14.73 * weight_kg) + milk_cost - mobilization
			elif activity_level == "active":
				value = -189.55 - (22.25 * age_years) + (11.74 * height_cm) + (18.34 * weight_kg) + milk_cost - mobilization
			else:
				value = -709.59 - (22.25 * age_years) + (18.22 * height_cm) + (14.25 * weight_kg) + milk_cost - mobilization
		return round(value)

	sex_abbrev = _sex_abbrev(sex)
	age_group = _age_to_eer_group(age_years)

	if age_group == "0 to 2.99 months":
		if sex_abbrev == "M":
			value = (-716.45) - (1.00 * age_years) + (17.82 * height_cm) + (15.06 * weight_kg) + 200
		else:
			value = -69.15 + (80.0 * age_years) + (2.65 * height_cm) + (54.15 * weight_kg) + 180
		return round(value)

	if age_group == "3 to 5.99 months":
		if sex_abbrev == "M":
			value = -716.45 - (1.00 * age_years) + (17.82 * height_cm) + (15.06 * weight_kg) + 50
		else:
			value = -69.15 + (80.0 * age_years) + (2.65 * height_cm) + (54.15 * weight_kg) + 60
		return round(value)

	if age_group == "6 months to 2.99 years":
		if sex_abbrev == "M":
			value = -716.45 - (1.00 * age_years) + (17.82 * height_cm) + (15.06 * weight_kg) + 20
		else:
			a = 20 if age_years < 1 else 15
			value = -69.15 + (80.0 * age_years) + (2.65 * height_cm) + (54.15 * weight_kg) + a
		return round(value)

	if activity_level is None:
		raise ValueError("activity_level is required for age 3+ years")

	if age_group == "3 to 13.99 years":
		if sex_abbrev == "M":
			b = 20 if age_years < 4 else (25 if age_years > 8 else 15)
			if activity_level == "inactive":
				value = -447.51 + (3.68 * age_years) + (13.01 * height_cm) + (13.15 * weight_kg) + b
			elif activity_level == "low active":
				value = 19.12 + (3.68 * age_years) + (8.62 * height_cm) + (20.28 * weight_kg) + b
			elif activity_level == "active":
				value = -388.19 + (3.68 * age_years) + (12.66 * height_cm) + (20.46 * weight_kg) + b
			else:
				value = -671.75 + (3.68 * age_years) + (15.38 * height_cm) + (23.25 * weight_kg) + b
		else:
			c = 15 if age_years < 4 else (30 if age_years > 8 else 15)
			if activity_level == "inactive":
				value = 55.59 - (22.25 * age_years) + (8.43 * height_cm) + (17.07 * weight_kg) + c
			elif activity_level == "low active":
				value = -297.54 - (22.25 * age_years) + (12.77 * height_cm) + (14.73 * weight_kg) + c
			elif activity_level == "active":
				value = -189.55 - (22.25 * age_years) + (11.74 * height_cm) + (18.34 * weight_kg) + c
			else:
				value = -709.59 - (22.25 * age_years) + (18.22 * height_cm) + (14.25 * weight_kg) + c
		return round(value)

	if age_group == "14 to 18.99 years":
		if sex_abbrev == "M":
			if activity_level == "inactive":
				value = -447.51 + (3.68 * age_years) + (13.01 * height_cm) + (13.15 * weight_kg) + 20
			elif activity_level == "low active":
				value = 19.12 + (3.68 * age_years) + (8.62 * height_cm) + (20.28 * weight_kg) + 20
			elif activity_level == "active":
				value = -388.19 + (3.68 * age_years) + (12.66 * height_cm) + (20.46 * weight_kg) + 20
			else:
				value = -671.75 + (3.68 * age_years) + (15.38 * height_cm) + (23.25 * weight_kg) + 20
		else:
			if activity_level == "inactive":
				value = 55.59 - (22.25 * age_years) + (8.43 * height_cm) + (17.07 * weight_kg) + 20
			elif activity_level == "low active":
				value = -297.54 - (22.25 * age_years) + (12.77 * height_cm) + (14.73 * weight_kg) + 20
			elif activity_level == "active":
				value = -189.55 - (22.25 * age_years) + (11.74 * height_cm) + (18.34 * weight_kg) + 20
			else:
				value = -709.59 - (22.25 * age_years) + (18.22 * height_cm) + (14.25 * weight_kg) + 20
		return round(value)

	if sex_abbrev == "M":
		if activity_level == "inactive":
			value = 753.07 - (10.83 * age_years) + (6.50 * height_cm) + (14.10 * weight_kg)
		elif activity_level == "low active":
			value = 581.47 - (10.83 * age_years) + (8.30 * height_cm) + (14.94 * weight_kg)
		elif activity_level == "active":
			value = 1004.82 - (10.83 * age_years) + (6.52 * height_cm) + (15.91 * weight_kg)
		else:
			value = -517.88 - (10.83 * age_years) + (15.61 * height_cm) + (19.11 * weight_kg)
	else:
		if activity_level == "inactive":
			value = 584.90 - (7.01 * age_years) + (5.72 * height_cm) + (11.71 * weight_kg)
		elif activity_level == "low active":
			value = 575.77 - (7.01 * age_years) + (6.60 * height_cm) + (12.14 * weight_kg)
		elif activity_level == "active":
			value = 710.25 - (7.01 * age_years) + (6.54 * height_cm) + (12.34 * weight_kg)
		else:
			value = 511.83 - (7.01 * age_years) + (9.07 * height_cm) + (12.56 * weight_kg)

	return round(value)
def _calculate_total_protein(
	age_years: float,
	sex: str,
	weight_kg: float,
	is_pregnant: bool,
	is_breastfeeding: bool,
	gestation_weeks: Optional[float],
) -> int:
	if is_pregnant:
		intake = PROTEIN_INTAKES_G_PER_KG["Pregnant"](gestation_weeks or 0)
	elif is_breastfeeding:
		intake = PROTEIN_INTAKES_G_PER_KG["Lactation"]
	else:
		age_range = _age_to_dri_age_range(age_years, False, False)
		intake = PROTEIN_INTAKES_G_PER_KG[age_range][_sex_abbrev(sex)]
	return round(intake * weight_kg)


def _calculate_amdr(age_years: float, eer: int) -> Dict[str, Dict[str, Dict[str, int]]]:
	age_range = _age_to_amdr_range(age_years)
	if age_range == "under 1 year":
		return {"percent": {}, "grams": {}}

	percents = AMDR_PERCENTS[age_range]
	percent_out: Dict[str, Dict[str, int]] = {}
	grams_out: Dict[str, Dict[str, int]] = {}

	for nutrient, bounds in percents.items():
		percent_out[nutrient] = {
			"lower": round(bounds["lower"] * 100),
			"upper": round(bounds["upper"] * 100),
		}
		kcal_divisor = 9 if nutrient == "Fats" else 4
		grams_out[nutrient] = {
			"lower": round((bounds["lower"] * eer) / kcal_divisor),
			"upper": round((bounds["upper"] * eer) / kcal_divisor),
		}

	return {"percent": percent_out, "grams": grams_out}


def _normalized_name(name: str) -> str:
	return _clean_csv_value(name).replace("\u00a0", " ")


def _table_rows_to_output(rows: List[Dict[str, str]], include_upper: bool) -> List[Dict[str, Any]]:
	result: List[Dict[str, str]] = []
	for row in rows:
		nutrient = _normalized_name(row.get("Nutrient", ""))
		unit = _clean_csv_value(row.get("Unit", ""))
		rda = _clean_csv_value(row.get("RDA", "")).replace(",", "")
		ai = _clean_csv_value(row.get("AI", "")).replace(",", "")
		ul = _clean_csv_value(row.get("UL", "")).replace(",", "")
		cdrr = _clean_csv_value(row.get("CDRR", "")).replace(",", "")

		recommended = f"{rda} {unit}" if rda else (f"{ai} {unit}" if ai else "ND")
		new_row = {
			"nutrient": nutrient,
			"recommended_intake": recommended,
		}
		if include_upper:
			upper = f"{ul} {unit}" if ul else (f"{cdrr} {unit}" if cdrr else "ND")
			new_row["maximum_intake_per_day"] = upper
		result.append(new_row)
	return result


def _order_macros(rows: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
	order = {name: idx for idx, name in enumerate(PREFERRED_MACRO_ORDER)}
	return sorted(rows, key=lambda row: order.get(_normalized_name(row.get("nutrient", "")), 999))


def _extract_grams_from_recommended_intake(value: Any) -> Optional[float]:
	if value is None:
		return None
	text = str(value).strip().lower()
	if not text or text == "nd":
		return None
	parts = text.replace(",", "").split()
	if len(parts) < 2:
		return None
	if parts[1] not in {"g", "gram", "grams"}:
		return None
	try:
		return float(parts[0])
	except ValueError:
		return None


def _recommended_macros_grams(macronutrients: List[Dict[str, Any]]) -> Dict[str, float]:
	out: Dict[str, float] = {}
	for row in macronutrients:
		if not isinstance(row, dict):
			continue
		nutrient = _normalized_name(str(row.get("nutrient", "")))
		recommended_grams = _extract_grams_from_recommended_intake(row.get("recommended_intake"))
		if not nutrient or recommended_grams is None:
			continue
		out[nutrient] = round(recommended_grams, 4)
	return out


def _latest_conversation_event_kind(tool_context: Optional[ToolContext]) -> Optional[str]:
	if tool_context is None:
		return None

	session = getattr(getattr(tool_context, "_invocation_context", None), "session", None)
	events = getattr(session, "events", []) if session is not None else []

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

		for part in reversed(parts):
			function_call = getattr(part, "function_call", None)
			if function_call:
				call_name = str(getattr(function_call, "name", ""))
				# Skip the current tool invocation record when this tool is executing.
				if call_name == "calculate_health_canada_dri":
					continue
				if call_name == "optimize_quantity":
					return "optimizer_tool_call"
				return "tool_call"

			function_response = getattr(part, "function_response", None)
			if function_response:
				response_name = str(getattr(function_response, "name", ""))
				if response_name == "optimize_quantity":
					return "optimizer_tool_response"
				return "tool_response"

			text = getattr(part, "text", None)
			if text and str(text).strip():
				author = str(getattr(event, "author", "")).strip().lower()
				if author == "user":
					return "user_message"
				return "non_user_message"

	return None


def _requires_user_message_guard(tool_context: Optional[ToolContext]) -> Optional[str]:
	latest_kind = _latest_conversation_event_kind(tool_context)
	if latest_kind in {None, "user_message"}:
		return None
	if latest_kind in {"optimizer_tool_call", "optimizer_tool_response"}:
		return "This tool only runs after a user message."
	return "This tool only runs after a user message."


def _calculate_health_canada_dri_impl(
	sex: str,
	age: float,
	weight: float,
	height: float,
	activity_level: Optional[str] = None,
	age_unit: str = "years",
	weight_unit: str = "kg",
	height_unit: str = "cm",
	height_inches: Optional[float] = None,
	is_pregnant: bool = False,
	gestation_weeks: Optional[float] = None,
	prepregnancy_weight: Optional[float] = None,
	prepregnancy_weight_unit: str = "kg",
	lactation_status: str = "none",
) -> Dict[str, Any]:
	"""
	Calculate Dietary Reference Intake outputs using Health Canada's DRI calculator logic.

	Args:
		sex: "Male" or "Female".
		age: Age value in `age_unit`.
		weight: Body weight in `weight_unit`.
		height: Height in cm if `height_unit="cm"`, or feet if `height_unit="ft_in"`.
		activity_level: One of Inactive, Low active, Active, Very active.
		age_unit: "years" or "months".
		weight_unit: "kg" or "lb".
		height_unit: "cm" or "ft_in".
		height_inches: Required when `height_unit="ft_in"`.
		is_pregnant: Pregnancy flag.
		gestation_weeks: Required in second/third trimester pregnancy calculations.
		prepregnancy_weight: Required with second/third trimester pregnancy equations.
		prepregnancy_weight_unit: "kg" or "lb".
		lactation_status: none | 0-6 months postpartum | 7-12 months postpartum.

	Returns:
		Dictionary containing calories and macro targets only:
		- `recommended_macronutrient_ranges_g_per_day` (AMDR lower/upper grams)
		- `recommended_g_per_day` (recommended grams/day for total fibre and calories)
	"""
	sex = _clean_csv_value(sex)
	if sex not in SEX_VALUES:
		raise ValueError("sex must be 'Male' or 'Female'")

	age_unit = _clean_csv_value(age_unit).lower()
	if age_unit not in {"years", "months"}:
		raise ValueError("age_unit must be 'years' or 'months'")

	weight_unit = _clean_csv_value(weight_unit).lower()
	if weight_unit not in {"kg", "lb"}:
		raise ValueError("weight_unit must be 'kg' or 'lb'")

	height_unit = _clean_csv_value(height_unit).lower()
	if height_unit not in {"cm", "ft_in"}:
		raise ValueError("height_unit must be 'cm' or 'ft_in'")

	lactation_status = _clean_csv_value(lactation_status).lower()
	if lactation_status not in LACTATION_STATUS_VALUES:
		raise ValueError("lactation_status must be one of: none, 0-6 months postpartum, 7-12 months postpartum")

	age = float(age)
	weight = float(weight)
	height = float(height)
	if age < 0:
		raise ValueError("age must be non-negative")
	if weight <= 0:
		raise ValueError("weight must be > 0")
	if height <= 0:
		raise ValueError("height must be > 0")

	age_years = _months_to_years(age) if age_unit == "months" else age
	weight_kg = _pounds_to_kg(weight) if weight_unit == "lb" else weight

	if height_unit == "ft_in":
		if height_inches is None:
			raise ValueError("height_inches is required when height_unit='ft_in'")
		height_cm = _feet_inches_to_cm(height, float(height_inches))
	else:
		height_cm = height

	if is_pregnant and sex != "female":
		raise ValueError("pregnancy inputs are only valid for sex='Female'")
	if lactation_status != "none" and sex != "female":
		raise ValueError("lactation inputs are only valid for sex='Female'")
	if is_pregnant and lactation_status != "none":
		raise ValueError("A person cannot be both pregnant and breastfeeding in this calculator")

	prepregnancy_weight_kg: Optional[float] = None
	if prepregnancy_weight is not None:
		prepregnancy_weight = float(prepregnancy_weight)
		if prepregnancy_weight <= 0:
			raise ValueError("prepregnancy_weight must be > 0")
		if _clean_csv_value(prepregnancy_weight_unit).lower() == "lb":
			prepregnancy_weight_kg = _pounds_to_kg(prepregnancy_weight)
		else:
			prepregnancy_weight_kg = prepregnancy_weight

	if age_years >= 3:
		if activity_level is None:
			raise ValueError("activity_level is required for age 3+ years")
		activity_level = _clean_csv_value(activity_level)
		if activity_level not in ACTIVITY_LEVELS:
			raise ValueError("activity_level must be one of: Inactive, Low active, Active, Very active")
	else:
		activity_level = None

	if is_pregnant and gestation_weeks is not None:
		gestation_weeks = float(gestation_weeks)

	is_breastfeeding = lactation_status != "none"

	eer = _calculate_eer(
		sex=sex,
		age_years=age_years,
		height_cm=height_cm,
		weight_kg=weight_kg,
		activity_level=activity_level,
		is_pregnant=is_pregnant,
		gestation_weeks=gestation_weeks,
		prepregnancy_weight_kg=prepregnancy_weight_kg,
		lactation_status=lactation_status,
	)

	total_protein = _calculate_total_protein(
		age_years=age_years,
		sex=sex,
		weight_kg=weight_kg,
		is_pregnant=is_pregnant,
		is_breastfeeding=is_breastfeeding,
		gestation_weeks=gestation_weeks,
	)

	fibre: Any = "No data" if age_years <= 1 else round(FIBRE_CONST * eer)
	amdr = _calculate_amdr(age_years=age_years, eer=eer)

	dri_sex = _get_dri_sex(sex, age_years, is_pregnant, is_breastfeeding)
	dri_age_range = _age_to_dri_age_range(age_years, is_pregnant, is_breastfeeding)
	macro_rows = _load_csv_rows(str(MACRO_CSV_PATH))

	selected_macro = [r for r in macro_rows if r.get("Sex") == dri_sex and r.get("Age") == dri_age_range]

	macro_output = _table_rows_to_output(
		[r for r in selected_macro if _clean_csv_value(r.get("Nutrient", "")).strip() != "Protein"],
		include_upper=True,
	)
	macro_output.insert(0, {"nutrient": "Total protein", "recommended_intake": f"{total_protein} g"})
	macro_output.append({"nutrient": "Total fibre", "recommended_intake": f"{fibre} g" if isinstance(fibre, int) else "ND"})
	macro_output = _order_macros(macro_output)
	recommended_macros_grams = _recommended_macros_grams(macro_output)
	amdr_grams = amdr.get("grams", {}) if isinstance(amdr, dict) else {}

	# Filter to only include relevant AMDR nutrients (Carbohydrates, Protein, Fats)
	filtered_amdr_grams = {
		k: v for k, v in amdr_grams.items()
		if k in {"Carbohydrates", "Protein", "Fats"}
	}
	
	# Filter to only include Total fibre from recommended macros
	filtered_recommended_macros = {
		k: v for k, v in recommended_macros_grams.items()
		if k == "total fibre"
	}

	return {
		"recommended_macronutrient_ranges_g_per_day": filtered_amdr_grams,
		"recommended_g_per_day": {
			**filtered_recommended_macros,
			"calories eer_kcal": eer,
		},
	}


def calculate_health_canada_dri(
	sex: str,
	age: float,
	weight: float,
	height: float,
	activity_level: Optional[str] = None,
	age_unit: str = "years",
	weight_unit: str = "kg",
	height_unit: str = "cm",
	height_inches: Optional[float] = None,
	is_pregnant: bool = False,
	gestation_weeks: Optional[float] = None,
	prepregnancy_weight: Optional[float] = None,
	prepregnancy_weight_unit: str = "kg",
	lactation_status: str = "none",
	tool_context: Optional[ToolContext] = None,
) -> Any:
	"""Calculate Dietary Reference Intake (DRI) targets for calories and macronutrients.

	Uses Health Canada's DRI calculator equations to estimate energy requirements
	and macronutrient targets based on personal characteristics.

	Args:
		sex: Biological sex. Must be "Male" or "Female".
		age: Age value interpreted according to `age_unit`.
		weight: Body weight interpreted according to `weight_unit`.
		height: Height value. Interpreted as centimetres when `height_unit="cm"`,
			or as feet when `height_unit="ft_in"` (provide `height_inches` for
			the remaining inches).
		activity_level: Physical activity level. Required for age 3+ years.
			One of: "Inactive", "Low active", "Active", "Very active".
		age_unit: Unit for `age`. Either "years" (default) or "months".
		weight_unit: Unit for `weight`. Either "kg" (default) or "lb".
		height_unit: Unit for `height`. Either "cm" (default) or "ft_in".
		height_inches: Inches component of height. Required when
			`height_unit="ft_in"`.
		is_pregnant: Set to True if the individual is currently pregnant.
			Only valid when `sex="Female"`.
		gestation_weeks: Gestational age in weeks. Used in second/third
			trimester EER calculations when `is_pregnant=True`.
		prepregnancy_weight: Pre-pregnancy body weight. Required for second/
			third trimester EER calculations. Interpreted according to
			`prepregnancy_weight_unit`.
		prepregnancy_weight_unit: Unit for `prepregnancy_weight`. Either
			"kg" (default) or "lb".
		lactation_status: Breastfeeding status. One of: "none" (default),
			"0-6 months postpartum", "7-12 months postpartum". Only valid
			when `sex="Female"`.
		tool_context: ADK tool context used to validate invocation ordering.

	Returns:
		On success, a dict with the following keys:

		- ``recommended_macronutrient_ranges_g_per_day``: AMDR lower/upper
		  bounds in grams for Carbohydrates, Protein, and Fats.
		- ``recommended_g_per_day``: Recommended daily intake values including
		  total fibre (grams) and calories ``eer_kcal``.

		On validation error, returns a string starting with ``"Error: "``.
	"""
	guard_message = _requires_user_message_guard(tool_context)
	if guard_message:
		return guard_message

	try:
		return _calculate_health_canada_dri_impl(
			sex=sex,
			age=age,
			weight=weight,
			height=height,
			activity_level=activity_level,
			age_unit=age_unit,
			weight_unit=weight_unit,
			height_unit=height_unit,
			height_inches=height_inches,
			is_pregnant=is_pregnant,
			gestation_weeks=gestation_weeks,
			prepregnancy_weight=prepregnancy_weight,
			prepregnancy_weight_unit=prepregnancy_weight_unit,
			lactation_status=lactation_status,
		)
	except (TypeError, ValueError, RuntimeError) as exc:
		return f"Error: {exc}"


if __name__ == "__main__":
	sample_input = {
		"sex": "Female",
		"age": 29,
		"age_unit": "years",
		"weight": 62,
		"weight_unit": "kg",
		"height": 165,
		"height_unit": "cm",
		"activity_level": "Low active",
		"is_pregnant": False,
		"lactation_status": "none",
	}

	print("Input:", sample_input)
	print("-"*70)

	sample_result = calculate_health_canada_dri(**sample_input)

	print(sample_result)
