import os
import pickle
import re
import json
import yaml
from datetime import datetime
from typing import Optional, Any
from geopy.distance import geodesic

def load_config(config_path: str):
    with open(config_path, "r", encoding="utf-8") as f:
        return yaml.safe_load(f)


def load_json(path: str):
    with open(path, "r", encoding="utf-8") as f:
        return json.load(f)

# ------------------------------------------
#        Cache loader / builder
# ------------------------------------------

def load_or_build_cache(config_path: str, cache_path: str = "    """
    如果 cache.pkl 存在 → 直接加载
    如果不存在 → 读取所有 JSON + config → 写入缓存
    """

    # 如果已有缓存文件，直接加载
    if os.path.exists(cache_path):
        with open(cache_path, "rb") as f:
            return pickle.load(f)

    # 构建缓存
    config = load_config(config_path)
    dp = config["data_path"]

    data = {
        "config": config,
        "flights": load_json(dp["flight"]),
        "trains": load_json(dp["train"]),
        "hotels": load_json(dp["hotel"]),
        "attractions": load_json(dp["attraction"]),
        "restaurants": load_json(dp["restaurant"]),
    }

    # 保存缓存
    with open(cache_path, "wb") as f:
        pickle.dump(data, f)

    return data


class GeneralEvaluator:

    def __init__(self, data):
        self.config = data["config"]
        self.flights = data["flights"]
        self.trains = data["trains"]
        self.hotels = data["hotels"]
        self.attractions = data["attractions"]
        self.restaurants = data["restaurants"]
        # 建立快速 ID → item dict
        self.id_index = {
            "flight": {f["Flight_id"]: f for f in self.flights},
            "train": {t["Train_id"]: t for t in self.trains},
            "hotel": {h["Hotel_id"]: h for h in self.hotels},
            "restaurant": {r["id"]: r for r in self.restaurants},
            "attraction": {str(a["poiId"]): a for a in self.attractions},
        }

    @staticmethod
    def _time_to_minutes(t: str) -> int:
        h, m = map(int, t.split(":"))
        return h * 60 + m

    @staticmethod
    def _parse_time_range(time_range: str):
        """
        返回 (start_min, end_min)，已处理跨天
        """
        start, end = time_range.split("-")
        s = GeneralEvaluator._time_to_minutes(start)
        e = GeneralEvaluator._time_to_minutes(end)
        if e < s:
            e += 24 * 60
        return s, e


    @staticmethod
    def _duration_minutes_safe(time_range: str) -> int:
        """
        安全版 duration，支持跨天
        """
        s, e = GeneralEvaluator._parse_time_range(time_range)
        return e - s

    @staticmethod
    def _duration_minutes(time_range: str) -> int:
        return GeneralEvaluator._duration_minutes_safe(time_range)

    @staticmethod
    def _is_within(time_range: str, open_ranges: list) -> bool:
        """
        支持如下情况：
        - 正常时段:       08:00-18:00
        - 跨天营业:       17:00-00:30
        - close=24:00:    09:00-24:00 (视作到第二天 00:00)
        - 活动本身跨天:   23:00-01:00
        """
        start, end = time_range.split("-")
        s = GeneralEvaluator._time_to_minutes(start)
        e = GeneralEvaluator._time_to_minutes(end)

        # 活动若跨天，例如 23:00-01:00
        if e < s:
            e += 24 * 60

        for o_start, o_end in open_ranges:
            os_ = GeneralEvaluator._time_to_minutes(o_start)
            oe_ = GeneralEvaluator._time_to_minutes(o_end)

            # “24:00” → 1440 分钟
            if o_end == "24:00":
                oe_ = 24 * 60

            # 若 close <= open → 跨天
            if oe_ <= os_:
                oe_ += 24 * 60

            # 判断活动时间段是否完全包含于营业时间段
            if s >= os_ and e <= oe_:
                return True

        return False

    
    @staticmethod
    def _is_within_with_buffer(time_range: str, open_ranges: list, buffer_min: int = 30) -> bool:
        """
        判断 time_range 是否完全落在任意一个营业区间内，但营业区间前后各放宽 buffer_min 分钟。
        open_ranges: list[tuple[str,str]] e.g. [("09:00","18:00"), ("17:00","00:30")]
        time_range: "23:00-01:00" 等
        """

        def to_min(t: str) -> int:
            h, m = map(int, t.split(":"))
            return h * 60 + m

        # --- activity interval (support cross-day) ---
        start, end = time_range.split("-")
        s = to_min(start)
        e = to_min(end)
        if e < s:
            e += 1440

        # 为了处理“跨天/对齐到次日”的情况，给 activity 两个候选轴
        act_candidates = [(s, e), (s + 1440, e + 1440)]

        for o_start, o_end in open_ranges:
            os_ = to_min(o_start)
            oe_ = 1440 if o_end == "24:00" else to_min(o_end)

            # open interval cross-day
            if oe_ <= os_:
                oe_ += 1440

            # --- apply buffer ---
            os_b = os_ - buffer_min
            oe_b = oe_ + buffer_min

            # 同理给 open interval 两个候选轴（覆盖“前一天/后一天”）
            open_candidates = [(os_b, oe_b), (os_b + 1440, oe_b + 1440)]

            # contains check
            for (as_, ae_) in act_candidates:
                for (bs_, be_) in open_candidates:
                    if as_ >= bs_ and ae_ <= be_:
                        return True

        return False

    @staticmethod
    def estimate_travel_time_minutes(origin_lat, origin_lng, destination_lat, destination_lng):
        # 定义速度规则（区间上限 km, 速度 km/h）
        speed_rules = [
            (1, 5),       # 步行
            (10, 30),     # 城市道路
            (50, 60),     # 城市快速路
            (99999, 90)   # 高速
        ]

        # 计算直线距离
        distance_km = geodesic(
            (origin_lat, origin_lng),
            (destination_lat, destination_lng)
        ).km

        remaining = distance_km
        last_upper = 0
        total_hours = 0

        # 累进式分段时间计算
        for upper, speed in speed_rules:
            if remaining <= 0:
                break

            segment_length = min(remaining, upper - last_upper)
            total_hours += segment_length / speed

            remaining -= segment_length
            last_upper = upper

        return total_hours * 60  # 返回分钟


    # A1. 格式完整性（JSON Structural Validity）
    def extract_last_json(self, text: str) -> Optional[Any]:
        pattern = r"```json\s*(\{.*?\})\s*```"
        matches = re.findall(pattern, text, flags=re.DOTALL)

        if not matches:
            return None
        
        last_json_str = matches[-1].strip()
        try:
            return json.loads(last_json_str)
        except json.JSONDecodeError:
            return None

    def validate_trip_plan_json(self, meta: dict, raw_text: str) -> dict:
        """
        先调用 extract_last_json，再检查 JSON trip_plan 是否符合规范：
        1. 是否成功提取 JSON
        2. 顶层结构 trip_plan 是否存在且字段齐全
        3. start_date / end_date / number_of_people / daily_schedule 是否存在
        4. 每个 daily_schedule 是否有 date / cities / activities
          - hotel：除最后一天外必须存在；最后一天可缺省（若存在也需符合结构）
        5. hotel 字段检查（{"id": str, "products": [{"id": str, "room_num": int>=1}, ...]}）
        6. activities 数组结构完整，字段齐全
        7. type / time 格式是否合法
        """

        # ============================================================
        # Step 1. 调 extract_last_json
        # ============================================================
        trip_plan = self.extract_last_json(raw_text)
        if trip_plan is None:
            return {"ok": False, "msg": "No valid JSON detected via extract_last_json"}

        errors = []

        # ============================================================
        # Step 2. 顶层 trip_plan
        # ============================================================
        if "trip_plan" not in trip_plan:
            return {"ok": False, "msg": "Missing top-level key 'trip_plan'"}

        tp = trip_plan["trip_plan"]

        REQUIRED_TOP = ["start_date", "end_date", "number_of_people", "daily_schedule"]
        for key in REQUIRED_TOP:
            if key not in tp:
                errors.append(f"trip_plan missing required field: {key}")

        if errors:
            return {"ok": False, "msg": "; ".join(set(errors))}

        # ============================================================
        # Step 3. basic type check
        # ============================================================
        if not isinstance(tp["daily_schedule"], list):
            errors.append("daily_schedule must be a list")
            return {"ok": False, "msg": "; ".join(set(errors)), "trip_plan": trip_plan}

        daily_schedule = tp["daily_schedule"]
        total_days = len(daily_schedule)

        # ============================================================
        # Step 4. 检查 daily_schedule
        # ============================================================
        for di, day in enumerate(daily_schedule, start=1):
            if not isinstance(day, dict):
                errors.append(f"daily_schedule[{di}] must be an object")
                continue

            # ---- 4.1 date / cities / activities ----
            for field in ["date", "cities", "activities"]:
                if field not in day:
                    errors.append(f"daily_schedule[{di}] missing required field: {field}")

            # ---- 4.1.1 hotel 是否必须（除最后一天外必须）----
            is_last_day = (di == total_days)
            if not is_last_day:
                if "hotel" not in day:
                    errors.append(f"daily_schedule[{di}] missing required field: hotel")
            else:
                if "hotel" in day:
                    errors.append(f"daily_schedule[{di}] should NOT contain hotel on the last day")

            # ---- 4.2 hotel 结构校验：如果存在就校验（包括最后一天给了 hotel 的情况）----
            if "hotel" in day:
                hotel = day.get("hotel", None)
                if hotel is None:
                    # 前面天 hotel 必填：给了 None 也算不合法；最后一天给 None 也算不合法（因为“给了就要对”）
                    errors.append(f"daily_schedule[{di}].hotel must be an object (not null)")
                elif not isinstance(hotel, dict):
                    errors.append(f"daily_schedule[{di}].hotel must be an object")
                else:
                    hid = hotel.get("id")
                    if not isinstance(hid, str) or not hid.strip():
                        errors.append(f"daily_schedule[{di}].hotel.id missing or invalid")

                    if "products" not in hotel:
                        errors.append(f"daily_schedule[{di}].hotel missing required field: products")
                    else:
                        prods = hotel.get("products")
                        if not isinstance(prods, list):
                            errors.append(f"daily_schedule[{di}].hotel.products must be a list")
                        else:
                            for pi, p in enumerate(prods, start=1):
                                if not isinstance(p, dict):
                                    errors.append(
                                        f"daily_schedule[{di}].hotel.products[{pi}] must be an object"
                                    )
                                    continue

                                pid = p.get("id")
                                if not isinstance(pid, str) or not pid.strip():
                                    errors.append(
                                        f"daily_schedule[{di}].hotel.products[{pi}].id missing or invalid"
                                    )

                                if "room_num" not in p:
                                    errors.append(
                                        f"daily_schedule[{di}].hotel.products[{pi}] missing room_num"
                                    )
                                else:
                                    rn = p.get("room_num")
                                    if not isinstance(rn, int) or rn < 1:
                                        errors.append(
                                            f"daily_schedule[{di}].hotel.products[{pi}].room_num invalid: {rn}"
                                        )

            # ---- 4.3 activities ----
            acts = day.get("activities", [])
            if not isinstance(acts, list):
                errors.append(f"daily_schedule[{di}].activities must be a list")
                continue

            for ai, act in enumerate(acts, start=1):
                if not isinstance(act, dict):
                    errors.append(f"daily_schedule[{di}].activities[{ai}] must be an object")
                    continue

                # ----- 必须字段 -----
                for f in ["time", "type", "description"]:
                    if f not in act:
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] missing required field: {f}"
                        )

                # ----- time 格式 -----
                if "time" in act:
                    if not isinstance(act["time"], str) or "-" not in act["time"]:
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] invalid time format: {act.get('time')}"
                        )

                # ----- id 产品规则 -----
                act_type = act.get("type")
                act_id = act.get("id")

                REQUIRED_ID_TYPES = ["Intercity Transportation", "Attraction", "Restaurant"]
                FORBIDDEN_ID_TYPES = ["Flight Check-in", "Local Transportation", "Hotel Check-in"]

                if act_type in REQUIRED_ID_TYPES:
                    if not act_id:
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] type={act_type} missing required id"
                        )
                    if "products" not in act:
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] type={act_type} missing products"
                        )
                    elif not isinstance(act["products"], list):
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] products must be list"
                        )

                if act_type in FORBIDDEN_ID_TYPES:
                    if act_id is not None:
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] type={act_type} should NOT have id"
                        )
                    if "products" in act and act["products"]:
                        errors.append(
                            f"daily_schedule[{di}].activities[{ai}] type={act_type} should NOT contain products"
                        )

        # ============================================================
        # Step 5. return results
        # ============================================================
        if errors:
            return {"ok": False, "msg": "; ".join(set(errors)), "trip_plan": trip_plan}

        return {"ok": True, "msg": "", "trip_plan": trip_plan}


    # A2. 范围与位置有效性（POI Validity）
    def evaluate_id_existence(self, meta: dict, trip_plan: dict) -> dict:
        errors = []

        # helper: 根据 id 找对应 item
        def load_item(act_type, act_id):
            if act_type == "Intercity Transportation":
                if act_id.startswith("Flight_"):
                    return self.id_index["flight"].get(act_id)
                elif act_id.startswith("Train_"):
                    return self.id_index["train"].get(act_id)
                else:
                    errors.append(f"Intercity Transportation id format invalid: {act_id}")
                    return None

            elif act_type == "Attraction":
                return self.id_index["attraction"].get(str(act_id))

            elif act_type == "Restaurant":
                return self.id_index["restaurant"].get(act_id)

            return None  # 不需要 id 的类型

        # helper: 返回不同数据集的产品列表字段
        def get_product_list(item):
            if item is None:
                return []
            if "Product" in item:
                return item["Product"]               # flight / train
            if "products" in item:
                return item["products"]               # hotel / restaurant
            if "ticket_products" in item:
                return item["ticket_products"]        # attraction
            return []

        REQUIRED_ID_TYPES = {"Intercity Transportation", "Attraction", "Restaurant"}
        FORBIDDEN_ID_TYPES = {"Flight Check-in", "Local Transportation", "Hotel Check-in"}

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            for act in day["activities"]:

                act_type = act["type"]
                act_id = act.get("id")
                act_products = act.get("products")

                # 1. 必须有 id 但缺失
                if act_type in REQUIRED_ID_TYPES:
                    if not act_id:
                        errors.append(f"{act_type} missing required id")
                        continue  # 没有 id 无法继续检查

                    item = load_item(act_type, act_id)
                    if item is None:
                        errors.append(f"{act_type} id={act_id} not found in dataset")
                        continue

                    # 产品必须存在
                    if act_products is None:
                        errors.append(f"{act_type} id={act_id} requires products but missing")
                        continue

                    # 产品 id 全部存在
                    valid_pids = {p["product_id"] for p in get_product_list(item)}
                    # print(f"Valid pids for {act_type} id={act_id}: {valid_pids}")
                    for p in act_products:
                        pid = p["id"]
                        if pid not in valid_pids:
                            errors.append(
                                f"Product id={pid} not found under parent id={act_id}"
                            )

                # 2. 必须无 id 的类型，但如果出现 id 报错
                elif act_type in FORBIDDEN_ID_TYPES:
                    if act_id:
                        errors.append(f"{act_type} should not have id")

                    if act_products:
                        errors.append(f"{act_type} should not contain products")

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}

    def evaluate_city_consistency(self, meta: dict, trip_plan: dict) -> dict:
        errors = []

        for day in trip_plan["trip_plan"]["daily_schedule"]:

            # 解析当天城市
            city_raw = day["cities"].lower().replace(" ", "")
            if "->" in city_raw:
                city_from, city_to = city_raw.split("->")
            else:
                city_from = city_to = city_raw

            current_city = city_from

            # 1. 检查当天 hotel（只在这里检查）
            if "hotel" in day:
                h = day["hotel"]
                hotel_item = self.id_index["hotel"].get(h["id"])
                hotel_city = hotel_item["real_city"].lower()
                if hotel_city != city_to:  # 酒店永远属于 day 的结束城市
                    errors.append(
                        f"Hotel id={h['id']} city mismatch: expected {city_to}, got {hotel_city}"
                    )

            # 2. 检查 activities
            for act in day["activities"]:
                act_type = act["type"]
                act_id = act.get("id")

                # 城市切换事件
                if act_type == "Intercity Transportation":
                    current_city = city_to
                    continue

                # 以下三类才检查城市
                if act_type not in ["Attraction", "Restaurant"]:
                    continue

                # 加载数据 item
                if act_type == "Attraction":
                    item = self.id_index["attraction"].get(str(act_id))
                    dataset_city = item["city"].lower()

                elif act_type == "Restaurant":
                    item = self.id_index["restaurant"].get(act_id)
                    dataset_city = item["real_city"].lower()

                # 检查城市是否一致
                if dataset_city != current_city:
                    errors.append(
                        f"{act_type} id={act_id} wrong city: expected {current_city}, got {dataset_city}"
                    )

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}


    # A3. 信息完整性（Information Completeness）
    def evaluate_meta(self, meta: dict, trip_plan: dict) -> dict:
        """
        Evaluate meta and trip_plan consistency:
        1. number_of_people / depart_date / return_date
        2. city travel sequence consistency
        3. intercity transportation existence & correctness
        4. NEW: transfer date must occur on previous.return_date or current.depart_date
        5. NEW: schedule days == meta trip days
        6. NEW: non-transfer days must include >=1 attraction + >=1 restaurant
        7. NEW: all days except return day must include hotel
        """

        errors = []

        # =====================================================
        # 1. Number of people / dates check
        # =====================================================
        meta_people = meta["route"][0]["number_of_people"]
        meta_depart = meta["route"][0]["depart_date"]
        meta_return = meta["route"][-1]["return_date"]

        plan_people = trip_plan["trip_plan"]["number_of_people"]
        plan_start = trip_plan["trip_plan"]["start_date"]
        plan_end = trip_plan["trip_plan"]["end_date"]

        if meta_people != plan_people:
            errors.append(f"Number of people mismatch: meta={meta_people}, plan={plan_people}")

        if meta_depart != plan_start:
            errors.append(f"Departure date mismatch: meta={meta_depart}, plan={plan_start}")

        if meta_return != plan_end:
            errors.append(f"Return date mismatch: meta={meta_return}, plan={plan_end}")

        # =====================================================
        # 2. Extract city sequence from meta.route
        # =====================================================
        meta_city_seq = []
        for leg in meta["route"]:
            if not meta_city_seq:
                meta_city_seq.append(leg["from"])
            meta_city_seq.append(leg["to"])

        # =====================================================
        # 3. Extract city sequence from trip_plan daily_schedule
        # =====================================================
        plan_city_seq = []
        for day in trip_plan["trip_plan"]["daily_schedule"]:
            parts = [x.strip() for x in day["cities"].split("->")]
            for city in parts:
                if city not in plan_city_seq:
                    plan_city_seq.append(city)

        if meta_city_seq != plan_city_seq:
            errors.append(f"City sequence mismatch: meta={meta_city_seq}, plan={plan_city_seq}")

        # =====================================================
        # 4. NEW: Validate schedule day count matches meta days
        # =====================================================
        from datetime import datetime

        d0 = datetime.strptime(meta_depart, "%Y-%m-%d")
        d1 = datetime.strptime(meta_return, "%Y-%m-%d")
        required_days = (d1 - d0).days + 1

        actual_days = len(trip_plan["trip_plan"]["daily_schedule"])
        if actual_days != required_days:
            errors.append(
                f"Daily schedule days mismatch: required {required_days}, got {actual_days}"
            )

        # =====================================================
        # 5. Verify intercity transfers
        # =====================================================
        for i, leg in enumerate(meta["route"]):
            meta_from = leg["from"].lower()
            meta_to = leg["to"].lower()

            found_day = False

            for di, day in enumerate(trip_plan["trip_plan"]["daily_schedule"]):
                city_raw = day["cities"].lower().replace(" ", "")

                # must be transfer day
                if "->" not in city_raw:
                    continue

                c_from, c_to = [x.strip() for x in city_raw.split("->")]

                if c_from == meta_from and c_to == meta_to:
                    found_day = True

                    # -----------------------------------------------------
                    # 5.1 transfer date validation
                    # -----------------------------------------------------
                    day_date = day["date"]

                    allowed_dates = {leg["depart_date"]}  # current depart_date
                    if i > 0:
                        allowed_dates.add(meta["route"][i - 1]["return_date"])  # previous return_date

                    if day_date not in allowed_dates:
                        errors.append(
                            f"Transfer {meta_from}->{meta_to} occurs on {day_date}, "
                            f"allowed dates are {sorted(list(allowed_dates))}"
                        )

                    # -----------------------------------------------------
                    # 5.2 intercity transportation validation
                    # -----------------------------------------------------
                    transports = [
                        act for act in day["activities"]
                        if act["type"] == "Intercity Transportation"
                    ]

                    if len(transports) == 0:
                        errors.append(
                            f"Missing Intercity Transportation for {meta_from}->{meta_to} on day {di+1}"
                        )
                        break

                    for act in transports:
                        tid = act.get("id")
                        if tid is None:
                            errors.append(f"Intercity Transportation on day {di+1} missing id")
                            continue

                        # Load dataset item
                        if tid.startswith("Flight_"):
                            item = self.id_index["flight"].get(tid)
                        elif tid.startswith("Train_"):
                            item = self.id_index["train"].get(tid)
                        else:
                            item = None
                            errors.append(f"Invalid Intercity Transportation id format: {tid}")

                        if item is None:
                            errors.append(f"Transportation id={tid} not found in dataset")
                            continue

                        dep = item["Departure City"].lower()
                        arr = item["Arrival City"].lower()

                        if dep != meta_from or arr != meta_to:
                            errors.append(
                                f"Transportation id={tid} city mismatch: "
                                f"expected {meta_from}->{meta_to}, got {dep}->{arr}"
                            )

                    break

            if not found_day:
                errors.append(
                    f"No day in trip_plan matches city transfer {meta_from}->{meta_to}"
                )


        # =====================================================
        # 7. NEW: All days except return day must include Hotel
        # =====================================================
        return_day = trip_plan["trip_plan"]["end_date"]

        for di, day in enumerate(trip_plan["trip_plan"]["daily_schedule"]):
            day_date = day["date"]

            # return day can skip hotel
            if day_date == return_day:
                continue

            if "hotel" not in day:
                errors.append(f"Day {di+1} ({day_date}) missing a Hotel activity")

        # =====================================================
        # Final result
        # =====================================================
        if len(errors) == 0:
            return {"ok": True, "msg": ""}

        return {"ok": False, "msg": "; ".join(set(errors))}


    # B1. 时间合理性（Temporal Reasonableness）
    def evaluate_daily_timing(self, meta: dict, trip_plan: dict) -> dict:
        """
        评估每日时间安排的合理性：
        - 若城市只有 1 次转场（A->B），第一活动必须 <=11:00 开始
        - 最后一活动必须 >=16:00 结束
        - 所有活动时间不能重叠
        - 任意活动间隙不能 ≥2 小时
        """

        errors = []

        # =====================================================
        # 6. NEW: Non-transfer days must include Attraction + Restaurant
        # =====================================================
        for di, day in enumerate(trip_plan["trip_plan"]["daily_schedule"]):
            city_raw = day["cities"].lower()

            if "->" in city_raw:
                continue

            types = [act["type"] for act in day["activities"]]

            if "Attraction" not in types:
                errors.append(f"Day {di+1} missing an Attraction activity")

            if "Restaurant" not in types:
                errors.append(f"Day {di+1} missing a Restaurant activity")

        def parse(t):
            return self._time_to_minutes(t)

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            acts = day["activities"]
            if not acts:
                continue

            # ========= 解析所有活动时间段 =========
            intervals = []
            for act in acts:
                t = act["time"]
                start, end = t.split("-")
                s = parse(start)
                e = parse(end)
                if e < s:
                    e += 24 * 60   # 跨天活动
                intervals.append((s, e))

            # ========= sort by start =========
            intervals.sort(key=lambda x: x[0])

            # ========= 判断城市转场数量 =========
            city_segments = [x.strip() for x in day["cities"].split("->")]
            num_seg = len(city_segments)

            # ========= 1. 开始时间是否 <11:00（无多次转场才检查） =========
            if num_seg <= 1:
                first_start = intervals[0][0] % (24*60)
                if first_start > 11 * 60:
                    errors.append(
                        f"{day['date']} first activity starts too late: {first_start//60:02d}:{first_start%60:02d}"
                    )

                # ========= 2. 结束时间是否 >16:00 =========
                # last_end = intervals[-1][1] % (24*60)
                last_end = intervals[-1][1]
                if last_end < 16 * 60:
                    # errors.append(
                    #     f"{day['date']} last activity ends too early: {last_end//60:02d}:{last_end%60:02d}"
                    # )
                    errors.append(
                        f"{day['date']} last activity ends too early: "
                        f"{last_end//60:02d}:{last_end%60:02d} (expected after 16:00)"
                    )

            # ========= 3. 时间段重叠检查 =========
            for i in range(len(intervals)-1):
                if intervals[i][1] > intervals[i+1][0]:
                    errors.append(
                        f"{day['date']} activities have overlapping time ranges"
                    )

            # ========= 4. 间隙 ≥ 2 小时 =========
            if num_seg <= 1:
              for i in range(len(intervals)-1):
                  gap = intervals[i+1][0] - intervals[i][1]
                  if gap > 120:
                      errors.append(
                          f"{day['date']} has idle gap > 2 hours between activities {i} and {i+1}"
                      )

        return {"ok": len(errors)==0, "msg": "; ".join(set(errors))}

    def evaluate_opening_hours(self, meta: dict, trip_plan: dict) -> dict:
        errors = []

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            for act in day["activities"]:
                act_type = act["type"]
                if act_type not in ["Attraction", "Restaurant"]:
                    continue

                act_id = act["id"]
                time_range = act["time"]

                # ============================
                # 景点（也可能跨天）
                # ============================
                if act_type == "Attraction":
                    item = self.id_index["attraction"].get(str(act_id))
                    if item is None:
                        continue

                    oh = item.get("opening_hours")
                    if oh and "open" in oh and "close" in oh:
                        open_ranges = [(oh["open"], oh["close"])]
                        if not self._is_within_with_buffer(time_range, open_ranges):
                        # if not self._is_within(time_range, open_ranges):
                            errors.append(
                                f"Attraction id={act_id} time {time_range} "
                                f"outside business hours {oh} on {day['date']}"
                            )

                # ============================
                # 餐厅（天然可能跨天）
                # ============================
                elif act_type == "Restaurant":
                    item = self.id_index["restaurant"].get(act_id)
                    if item is None:
                        continue

                    open_ranges = item.get("open_hours", [])
                    if open_ranges:
                        if not self._is_within_with_buffer(time_range, open_ranges):
                        # if not self._is_within(time_range, open_ranges):
                            errors.append(
                                f"Restaurant id={act_id} time {time_range} "
                                f"outside open_hours={open_ranges} on {day['date']}"
                            )

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}

    def evaluate_activity_duration(self, meta: dict, trip_plan: dict) -> dict:
        errors = []

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            for act in day["activities"]:
                act_type = act["type"]
                if act_type not in ["Attraction", "Restaurant"]:
                    continue

                duration = self._duration_minutes(act["time"])

                # -------------------------
                # 1. 餐厅检查：45-90 分钟
                # -------------------------
                if act_type == "Restaurant":
                    if not (45 <= duration <= 90):
                        errors.append(
                            f"Restaurant id={act['id']} invalid duration={duration} minutes on {day['date']}"
                        )
                    continue

                # -------------------------
                # 2. 景点检查：根据 reference_time 容忍范围
                # -------------------------
                item = self.id_index["attraction"].get(str(act["id"]))
                if item is None:
                    errors.append(f"Attraction id={act['id']} not found in dataset")
                    continue

                ref = item.get("reference_time")
                if not ref or "min_minutes" not in ref or "max_minutes" not in ref:
                    continue  # 没有参考时长则跳过

                min_m = ref["min_minutes"]
                max_m = ref["max_minutes"]

                lower = max(30, min_m - 90)
                upper = max_m + 90

                if not (lower <= duration <= upper):
                    errors.append(
                        f"Attraction id={act['id']} duration={duration} not in valid range [{lower},{upper}] on {day['date']}"
                    )

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}

    def evaluate_intercity_transportation(self, meta: dict, trip_plan: dict) -> dict:
        """
        评估航班/火车前置流程是否符合规范：
        - 航班：活动时间必须覆盖起飞-落地；前面必须有 Flight Check-in；Check-in 与起飞差为 1.5~2.5h
        - 火车：活动时间必须覆盖发车-到站；前面必须有 >=15 min 间隙
        """

        errors = []

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            acts = day["activities"]

            for idx, act in enumerate(acts):
                if act["type"] != "Intercity Transportation":
                    continue

                act_id = act["id"]
                t_start_str, t_end_str = act["time"].split("-")
                # t_start = self._time_to_minutes(t_start_str)
                # t_end = self._time_to_minutes(t_end_str)
                t_start, t_end = self._parse_time_range(act["time"])


                # ====== 航班 ======
                if act_id.startswith("Flight_"):
                    flight = self.id_index["flight"].get(act_id)
                    if not flight:
                        errors.append(f"Flight id={act_id} not found")
                        continue

                    f_dep = self._time_to_minutes(flight["Departure Time"])
                    f_arr = self._time_to_minutes(flight["Arrival Time"])

                    # 1. 活动时间必须覆盖航班起降
                    if not (t_start <= f_dep and t_end >= f_arr):
                        errors.append(
                            f"Flight {act_id} time range does not cover flight period on {day['date']}"
                        )

                    # 2. 前面必须找到 Flight Check-in
                    checkin_act = None
                    for prev in acts[:idx]:
                        if prev["type"] == "Flight Check-in":
                            checkin_act = prev
                            break

                    if checkin_act is None:
                        errors.append(
                            f"Flight {act_id} missing Flight Check-in on {day['date']}"
                        )
                    else:
                        # 3. 检查 check-in
                        c_start_str, c_end_str = checkin_act["time"].split("-")
                        # c_start = self._time_to_minutes(c_start_str)
                        # c_end = self._time_to_minutes(c_end_str)
                        c_start, c_end = self._parse_time_range(checkin_act["time"])
                        diff =  c_end - c_start
                        if diff < 90 or diff > 150:
                            errors.append(
                                f"Flight {act_id} check-in offset invalid: {diff} minutes (expect 90~150)"
                            )

                # ====== 火车 ======
                elif act_id.startswith("Train_"):
                    train = self.id_index["train"].get(act_id)
                    if not train:
                        errors.append(f"Train id={act_id} not found")
                        continue

                    t_dep = self._time_to_minutes(train["Departure Time"])
                    t_arr = self._time_to_minutes(train["Arrival Time"])

                    # 1. 活动时间必须覆盖火车运行
                    if not (t_start <= t_dep and t_end >= t_arr):
                        errors.append(
                            f"Train {act_id} time range does not cover train period on {day['date']}"
                        )

                    # 2. 检查前面是否留出 >=15分钟
                    if idx > 0:
                        prev_act = acts[idx-1]
                        # p_start, p_end = [
                        #     self._time_to_minutes(x) for x in prev_act["time"].split("-")
                        # ]
                        p_start, p_end = self._parse_time_range(prev_act["time"])

                        gap = t_start - p_end
                        if gap < 15:
                            errors.append(
                                f"Train {act_id} has insufficient pre-departure gap ({gap} min, expected 15-30 min) on {day['date']}"
                            )
                    # else:
                    #     errors.append(
                    #         f"Train {act_id} requires at least 15 min prep time but this is first activity"
                    #     )

                else:
                    errors.append(f"Invalid Intercity Transportation id={act_id}")

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}

    def evaluate_local_transportation(self, meta: dict, trip_plan: dict) -> dict:
        simple_msg = set()
        errors = []

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            acts = day["activities"]

            for i, act in enumerate(acts):
                act_type = act["type"]
                act_id = act.get("id")

                # --------------------------------------------------------
                # ❶ 检查景点/餐厅是否有前后 Local Transportation
                # --------------------------------------------------------
                if act_type in ["Attraction", "Restaurant"]:
                    # 前一个必须是 Local Transportation
                    if i == 0 or acts[i-1]["type"] != "Local Transportation":
                        # errors.append(
                        #     f"{day['date']} {act_type} id={act_id} missing preceding Local Transportation"
                        # )
                        errors.append(
                            f"{day['date']} {act_type} id={act_id} missing preceding Local Transportation and should add a reasonable-duration Local Transportation (e.g. 30 min) before it."
                        )
                        simple_msg.add("There are no arrangements for local transportation between some of the locations")

                    # 后一个必须是 Local Transportation
                    if i == len(acts) - 1 or acts[i+1]["type"] != "Local Transportation":
                        # errors.append(
                        #     f"{day['date']} {act_type} id={act_id} missing following Local Transportation"
                        # )
                        errors.append(
                            f"{day['date']} {act_type} id={act_id} missing following Local Transportation and should add a reasonable-duration Local Transportation (e.g. 30 min) following it."
                        )
                        simple_msg.add("There are no arrangements for local transportation between some of the locations")

                # --------------------------------------------------------
                # ❷ 若当前是 Local Transportation，则检查耗时是否合理
                # --------------------------------------------------------
                if act_type == "Local Transportation" and 0 < i < len(acts) - 1:
                    prev_act = acts[i - 1]
                    next_act = acts[i + 1]

                    # 只接受 Attraction ↔ Restaurant 的组合
                    pair = (prev_act["type"], next_act["type"])

                    if pair == ("Attraction", "Restaurant"):
                        A = self.id_index["attraction"].get(str(prev_act.get("id")))
                        B = self.id_index["restaurant"].get(next_act.get("id"))

                    elif pair == ("Restaurant", "Attraction"):
                        A = self.id_index["restaurant"].get(prev_act.get("id"))
                        B = self.id_index["attraction"].get(str(next_act.get("id")))

                    else:
                        continue  # 不是合法模式就跳过

                    if not A or not B:
                        continue

                    # 预估行程时间
                    est = self.estimate_travel_time_minutes(
                        A["latitude"], A["longitude"],
                        B["latitude"], B["longitude"]
                    )

                    # 实际 Local Transportation 时间
                    s, e = act["time"].split("-")
                    # actual = self._time_to_minutes(e) - self._time_to_minutes(s)
                    actual = self._duration_minutes_safe(act["time"])

                    if abs(actual - est) > 20:
                        errors.append(
                            f"{day['date']} Local Transportation between "
                            f"{prev_act['type']}({prev_act.get('id')}) and "
                            f"{next_act['type']}({next_act.get('id')}) "
                            f"has unrealistic travel time: "
                            f"unrealistic_time={actual} min, expected={est} min"
                        )
                        simple_msg.add("The local transportation times between some locations are not reasonable")

       
        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}



    # B2. 地理合理性（Spatial Logic）
    def evaluate_restaurant_location(self, meta: dict, trip_plan: dict) -> dict:
        """
        检查 Restaurant 前后的小交通时间是否存在一个 < 45分钟
        如果没有（即两个都 >=45min 或缺失），说明餐厅位置选择不合理
        """
        errors = []

        for day in trip_plan["trip_plan"]["daily_schedule"]:
            acts = day["activities"]

            for i, act in enumerate(acts):
                if act["type"] != "Restaurant":
                    continue

                act_id = act["id"]

                has_short_travel = False  # 是否存在 <45min 的小交通

                # ========== 前一个 Local Transportation ==========
                if i > 0 and acts[i-1]["type"] == "Local Transportation":
                    prev = acts[i-1]
                    _, prev_end = prev["time"].split("-")
                    r_start, _ = act["time"].split("-")
                    gap = self._time_to_minutes(r_start) - self._time_to_minutes(prev_end)

                    if gap <= 40:
                        has_short_travel = True

                # ========== 后一个 Local Transportation ==========
                if i < len(acts)-1 and acts[i+1]["type"] == "Local Transportation":
                    next_act = acts[i+1]
                    r_end = act["time"].split("-")[1]
                    next_start = next_act["time"].split("-")[0]
                    gap = self._time_to_minutes(next_start) - self._time_to_minutes(r_end)

                    if gap <= 40:
                        has_short_travel = True

                # ========== 如果前后都没有 <45min 小交通 ==========
                if not has_short_travel:
                    errors.append(
                        f"{day['date']} Restaurant id={act_id}: "
                        f"too far from the previous or next destinations."
                    )

        return {"ok": len(errors)==0, "msg": "; ".join(set(errors))}


    # B3. 体验多样性（Experience Diversity）
    def evaluate_attraction_restaurant_diversity(self, meta: dict, trip_plan: dict) -> dict:
        errors = []
        seen = set()
        for day in trip_plan["trip_plan"]["daily_schedule"]:
            for act in day["activities"]:
                if act["type"] not in ["Attraction", "Restaurant"]:
                    continue
                
                act_id = act.get("id")
                key = (act["type"], act_id)
                if key in seen:
                    errors.append(
                        f"Duplicate {act['type']} id={act_id} on date {day['date']}"
                    )
                else:
                    seen.add(key)

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}
    

    # B4. 产品合理性（Product Consistency）
    def evaluate_product_requirements(self, meta: dict, trip_plan: dict) -> dict:
        """
        校验产品需求一致性：
        1. 景点：有 ticket_products 则必须在 activities[].products 中体现
        - 产品数量 >= number_of_people
        2. 餐厅：若 restaurant.products 存在（套餐），活动必须正确引用
        - 每个套餐 price / people / product_id 必须在活动产品中匹配
        - 产品数量 × people >= number_of_people
        3. 酒店：hotel.products 剩余数量必须 >= 入住人数
        - 每个 product 的 person_num × quantity ≥ number_of_people
        """

        errors = []
        n_people = meta["route"][0]["number_of_people"]

        # ------------------------------------------------------
        # 1. 景点门票校验
        # ------------------------------------------------------
        for day in trip_plan["trip_plan"]["daily_schedule"]:
            for act in day["activities"]:
                if act["type"] != "Attraction":
                    continue

                act_id = str(act["id"])
                item = self.id_index["attraction"].get(act_id)

                # attraction 有 ticket_products → 必须在 act.products 出现
                ticket_products = item.get("ticket_products", [])
                ticket_price = item.get("price", 100.0)
                if ticket_price == 0.0:
                    continue
                if ticket_products:
                    if "products" not in act or not act["products"]:
                        errors.append(
                            f"Attraction id={act_id} requires ticket products but activity.products missing"
                        )
                        continue

                    valid_pid = {p["product_id"] for p in ticket_products}

                    # 检查产品存在性 & 数量满足人数
                    total_capacity = 0
                    for p in act["products"]:
                        pid = p["id"]
                        qty = p.get("quantity", 0)

                        # if pid not in valid_pid:
                        #     errors.append(f"Attraction id={act_id} has invalid product id={pid}")
                        # else:
                        #     # 注意：景点 ticket_products 可能没有 people 字段，仅票数量
                        total_capacity += qty

                    if total_capacity < n_people:
                        errors.append(
                            f"Attraction id={act_id} product quantity {total_capacity} < required {n_people}"
                        )

        # ------------------------------------------------------
        # 3. 酒店入住人数校验
        # ------------------------------------------------------
        for day in trip_plan["trip_plan"]["daily_schedule"]:
            if "hotel" not in day:
                continue

            h = day["hotel"]
            hotel_id = h["id"]
            item = self.id_index["hotel"].get(hotel_id)
            product_list = item.get("products", [])

            valid_pid = {p["product_id"]: p for p in product_list}

            total_capacity = 0

            for p in h["products"]:
                pid = p["id"]
                qty = p.get("room_num", 0)

                if pid not in valid_pid:
                    errors.append(
                        f"Hotel id={hotel_id} has invalid product id={pid}"
                    )
                    continue

                room_people = valid_pid[pid].get("person_num", 1)
                total_capacity += room_people * qty

            if total_capacity < n_people:
                errors.append(
                    f"Hotel id={hotel_id} room capacity {total_capacity} < number_of_people={n_people}"
                )

        return {"ok": len(errors) == 0, "msg": "; ".join(set(errors))}

