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)

def _resolve_root(config_path: str) -> str:
    config_dir = os.path.dirname(os.path.abspath(config_path))
    return os.path.dirname(config_dir)

def _resolve_path(path_value: str, root_dir: str) -> str:
    if os.path.isabs(path_value):
        return path_value
    return os.path.normpath(os.path.join(root_dir, path_value))

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

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

    root_dir = _resolve_root(config_path)
    if cache_path is None:
        cache_path = os.path.join(root_dir, "data_final", "cache.pkl")
    else:
        cache_path = _resolve_path(cache_path, root_dir)

    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"]
    resolved_dp = {k: _resolve_path(v, root_dir) for k, v in dp.items()}

    data = {
        "config": config,
        "flights": load_json(resolved_dp["flight"]),
        "trains": load_json(resolved_dp["train"]),
        "hotels": load_json(resolved_dp["hotel"]),
        "attractions": load_json(resolved_dp["attraction"]),
        "restaurants": load_json(resolved_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 extract_last_json(self, text: str) -> Optional[Any]:
        # 先尝试直接解析 JSON 格式的数据
        try:
            # 尝试直接解析 text 是否是一个有效的 JSON 字符串
            return json.loads(text)
        except json.JSONDecodeError:
            pass  # 如果解析失败，继续执行下面的逻辑
        
        # 如果直接解析失败，尝试提取 Markdown 代码块中的 JSON
        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": "Invalid or missing trip plan JSON."}

        errors = []

        # ============================================================
        # Step 2. 顶层 trip_plan
        # ============================================================
        if "trip_plan" not in trip_plan:
            return {"ok": False, "msg": "Invalid or missing trip plan JSON structure."}

        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("Trip plan JSON is missing required fields.")

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

        # ============================================================
        # Step 3. basic type check
        # ============================================================
        if not isinstance(tp["daily_schedule"], list):
            errors.append("Trip plan JSON format is invalid.")
            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("Trip plan JSON format is invalid.")
                continue

            # ---- 4.1 date / cities / activities ----
            for field in ["date", "cities", "activities"]:
                if field not in day:
                    errors.append("Trip plan JSON is missing required fields.")

            # ---- 4.1.1 hotel 是否必须（除最后一天外必须）----
            is_last_day = (di == total_days)
            if not is_last_day:
                if "hotel" not in day:
                    errors.append("Trip plan JSON is missing required fields.")
            else:
                if "hotel" in day:
                    errors.append("Trip plan JSON contains unexpected fields.")

            # ---- 4.2 hotel 结构校验：如果存在就校验（包括最后一天给了 hotel 的情况）----
            if "hotel" in day:
                hotel = day.get("hotel", None)
                if hotel is None:
                    errors.append("Trip plan JSON format is invalid.")
                elif not isinstance(hotel, dict):
                    errors.append("Trip plan JSON format is invalid.")
                else:
                    hid = hotel.get("id")
                    if not isinstance(hid, str) or not hid.strip():
                        errors.append("Trip plan JSON format is invalid.")

                    if "products" not in hotel:
                        errors.append("Trip plan JSON is missing required fields.")
                    else:
                        prods = hotel.get("products")
                        if not isinstance(prods, list):
                            errors.append("Trip plan JSON format is invalid.")
                        else:
                            for pi, p in enumerate(prods, start=1):
                                if not isinstance(p, dict):
                                    errors.append("Trip plan JSON format is invalid.")
                                    continue

                                pid = p.get("id")
                                if not isinstance(pid, str) or not pid.strip():
                                    errors.append("Trip plan JSON format is invalid.")

                                if "room_num" not in p:
                                    errors.append("Trip plan JSON is missing required fields.")
                                else:
                                    rn = p.get("room_num")
                                    if not isinstance(rn, int) or rn < 1:
                                        errors.append("Trip plan JSON format is invalid.")

            # ---- 4.3 activities ----
            acts = day.get("activities", [])
            if not isinstance(acts, list):
                errors.append("Trip plan JSON format is invalid.")
                continue

            for ai, act in enumerate(acts, start=1):
                if not isinstance(act, dict):
                    errors.append("Trip plan JSON format is invalid.")
                    continue

                # ----- 必须字段 -----
                for f in ["time", "type", "description"]:
                    if f not in act:
                        errors.append("Trip plan JSON is missing required fields.")

                # ----- time 格式 -----
                if "time" in act:
                    if not isinstance(act["time"], str) or "-" not in act["time"]:
                        errors.append("Trip plan JSON contains invalid field formats.")

                # ----- 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("Trip plan JSON is missing required fields.")
                    if "products" not in act:
                        errors.append("Trip plan JSON is missing required fields.")
                    elif not isinstance(act["products"], list):
                        errors.append("Trip plan JSON format is invalid.")

                if act_type in FORBIDDEN_ID_TYPES:
                    if act_id is not None:
                        errors.append("Trip plan JSON contains unexpected fields.")
                    if "products" in act and act["products"]:
                        errors.append("Trip plan JSON contains unexpected fields.")

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

        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("Some intercity transportation entry may be invalid.")
                    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("Some activities may be missing required identifiers.")
                        continue  # 没有 id 无法继续检查

                    item = load_item(act_type, act_id)
                    if item is None:
                        errors.append("Some activities may not be found in the dataset.")
                        continue

                    # 产品必须存在
                    if act_products is None:
                        errors.append("Some activities may be missing required products.")
                        continue

                    # 产品 id 全部存在
                    valid_pids = {p["product_id"] for p in get_product_list(item)}
                    for p in act_products:
                        pid = p["id"]
                        if pid not in valid_pids:
                            errors.append("Some products may be invalid or mismatched.")

                # 2. 必须无 id 的类型，但如果出现 id 报错
                elif act_type in FORBIDDEN_ID_TYPES:
                    if act_id:
                        errors.append("Some activities may contain unexpected identifiers.")

                    if act_products:
                        errors.append("Some activities may contain unexpected 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("Some hotel city information may be inconsistent.")

            # 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("Some activity city information may be inconsistent.")

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


    # A3. 信息完整性（Information Completeness）
    def evaluate_meta(self, meta: dict, trip_plan: dict) -> dict:
        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("Some trip meta information may be inconsistent.")

        if meta_depart != plan_start:
            errors.append("Some trip meta information may be inconsistent.")

        if meta_return != plan_end:
            errors.append("Some trip meta information may be inconsistent.")

        # =====================================================
        # 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("Some city routing information may be inconsistent.")

        # =====================================================
        # 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("Some schedule structure information may be inconsistent.")

        # =====================================================
        # 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("Some intercity transfer information may be inconsistent.")

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

                    if len(transports) == 0:
                        errors.append("Some intercity transportation arrangement may be missing.")
                        break

                    for act in transports:
                        tid = act.get("id")
                        if tid is None:
                            errors.append("Some intercity transportation entry may be missing identifiers.")
                            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("Some intercity transportation entry may be invalid.")

                        if item is None:
                            errors.append("Some intercity transportation information may be missing.")
                            continue

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

                        if dep != meta_from or arr != meta_to:
                            errors.append("Some intercity transportation routing may be inconsistent.")

                    break

            if not found_day:
                errors.append("Some intercity transfer day may be missing in the plan.")

        # =====================================================
        # 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("Some day may be missing hotel arrangement.")

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

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

    # B1. 时间合理性（Temporal Reasonableness）
    # ===== 修改第二版的 evaluate_daily_timing：阈值/边界对齐第一版 =====
    def evaluate_daily_timing(self, meta: dict, trip_plan: dict) -> dict:
        errors = []

        # 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("Some day may be missing an attraction activity.")
            if "Restaurant" not in types:
                errors.append("Some day may be 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:
                start, end = act["time"].split("-")
                s = parse(start)
                e = parse(end)
                if e < s:
                    e += 24 * 60
                intervals.append((s, e))

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

            city_segments = [x.strip() for x in day["cities"].split("->")]
            num_seg = len(city_segments)

            # ✅ 对齐第一版：只有非转场日（num_seg<=1）才检查早晚
            if num_seg <= 1:
                first_start = intervals[0][0] % (24 * 60)
                # ✅ 第一版：> 11:00 才算晚（11:00 允许）
                if first_start > 11 * 60:
                    errors.append("Some day's schedule may start too late.")

                last_end = intervals[-1][1]
                # ✅ 第一版：< 16:00 才算早（16:00 允许）
                if last_end < 16 * 60:
                    errors.append("Some day's schedule may end too early.")

            # overlap（两版一致）
            for i in range(len(intervals) - 1):
                if intervals[i][1] > intervals[i + 1][0]:
                    errors.append("Some activities may have overlapping time ranges.")

            # gap（两版一致：>120）
            if num_seg <= 1:
                for i in range(len(intervals) - 1):
                    gap = intervals[i + 1][0] - intervals[i][1]
                    if gap > 120:
                        errors.append("Some day may have long idle gaps between activities.")

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


    # ===== 修改第二版的 evaluate_opening_hours：把 _is_within 改成 _is_within_with_buffer =====
    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"])]
                        # ✅ 对齐第一版：带 buffer
                        if not self._is_within_with_buffer(time_range, open_ranges):
                            errors.append("Some attraction timing may be outside opening hours.")

                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:
                        # ✅ 对齐第一版：带 buffer
                        if not self._is_within_with_buffer(time_range, open_ranges):
                            errors.append("Some restaurant timing may be outside opening hours.")

        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("Some restaurant activity duration may be unreasonable.")
                    continue

                # -------------------------
                # 2. 景点检查：根据 reference_time 容忍范围
                # -------------------------
                item = self.id_index["attraction"].get(str(act["id"]))
                if item is None:
                    errors.append("Some attraction information may be missing.")
                    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("Some attraction activity duration may be unreasonable.")

        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, 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("Some flight information may be missing.")
                        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("Some flight transportation timing may be inconsistent.")

                    # 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("Some flight check-in arrangement may be missing.")
                    else:
                        # 3. 检查 check-in
                        c_start_str, c_end_str = checkin_act["time"].split("-")
                        c_start, c_end = self._parse_time_range(checkin_act["time"])
                        diff = c_end - c_start
                        if diff < 90 or diff > 150:
                            errors.append("Some flight check-in timing may be unreasonable.")

                # ====== 火车 ======
                elif act_id.startswith("Train_"):
                    train = self.id_index["train"].get(act_id)
                    if not train:
                        errors.append("Some train information may be missing.")
                        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("Some train transportation timing may be inconsistent.")

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

                        gap = t_start - p_end
                        if gap < 15:
                            errors.append("Some train pre-departure buffer may be insufficient.")

                else:
                    errors.append("Some intercity transportation entry may be invalid.")

        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("Some locations may be missing preceding local transportation.")
                        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("Some locations may be missing following local transportation.")
                        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._duration_minutes_safe(act["time"])

                    if abs(actual - est) > 20:
                        errors.append("Some local transportation times may be unreasonable.")
                        simple_msg.add("The local transportation times between some locations are not reasonable")

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


    # 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"Some restaurant location may be unreasonable."
                    )

        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"Some experiences may be duplicated in itinerary."
                    )
                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"Some required attraction products may be 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"Some attraction product quantity may be insufficient."
                        )

        # ------------------------------------------------------
        # 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"Some hotel product configuration may be invalid."
                    )
                    continue

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

            if total_capacity < n_people:
                errors.append(
                    f"Some hotel capacity may be insufficient."
                )

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