general_evaluator_description = """
| 方法名                                      | 输入              | 输出               | 检查内容               |
| ---------------------------------------- | --------------- | ---------------- | ------------------ |
| extract_last_json                        | text:str        | dict / None      | 从文本提取 JSON         |
| validate_trip_plan_json                  | meta, raw_text  | ok/msg/trip_plan | JSON 结构合法性         |
| evaluate_id_existence                    | meta, trip_plan | ok/msg           | id & product 合法性   |
| evaluate_city_consistency                | meta, trip_plan | ok/msg           | 城市是否匹配             |
| evaluate_meta                            | meta, trip_plan | ok/msg           | 人数、日期、城市序列、酒店、转场等  |
| evaluate_daily_timing                    | meta, trip_plan | ok/msg           | 每日时间合理性            |
| evaluate_opening_hours                   | meta, trip_plan | ok/msg           | 营业时间               |
| evaluate_activity_duration               | meta, trip_plan | ok/msg           | 餐厅/景点时长合理性         |
| evaluate_intercity_transportation        | meta, trip_plan | ok/msg           | 飞机/火车时长 + check-in |
| evaluate_local_transportation            | meta, trip_plan | ok/msg           | 小交通合理性             |
| evaluate_restaurant_location             | meta, trip_plan | ok/msg           | 餐厅地理合理性            |
| evaluate_attraction_restaurant_diversity | meta, trip_plan | ok/msg           | 去重检查               |
| evaluate_product_requirements            | meta, trip_plan | ok/msg           | 产品数量、合法性、酒店房间容量    |
"""

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


class TripPlanEvaluator:
    """
    总评估类：
    - 输入：meta_dict, raw_text(待评估文本), id_list(要评估的 rubric _id 列表)
    - 输出：
        {
            "general_results": [ {...}, ... ],   # GeneralEvaluator 的 true/false dict 列表
            "id_true_results": [ {...}, ... ],  # rubric id 对应的 true dict 列表
            "id_false_results": [ {...}, ... ], # rubric id 对应的 false dict 列表
            "error_text": "组合后的错误文本"
        }
    """

    GENERAL_FUNCS_ORDER = [
        "evaluate_id_existence",
        "evaluate_city_consistency",
        "evaluate_meta",
        "evaluate_daily_timing",
        "evaluate_opening_hours",
        "evaluate_activity_duration",
        "evaluate_intercity_transportation",
        "evaluate_local_transportation",
        "evaluate_restaurant_location",
        "evaluate_attraction_restaurant_diversity",
        "evaluate_product_requirements",
    ]

    def __init__(
        self,
        general_evaluator,
        attraction_evaluator,
        restaurant_evaluator,
        hotel_evaluator,
        transportation_evaluator,
    ):
        self.general_evaluator = general_evaluator
        self.attraction_evaluator = attraction_evaluator
        self.restaurant_evaluator = restaurant_evaluator
        self.hotel_evaluator = hotel_evaluator
        self.transportation_evaluator = transportation_evaluator

    # ======================
    # 对外主入口
    # ======================
    def evaluate(
        self,
        meta_dict: Dict[str, Any],
        raw_text: str,
        id_list: List[str],
    ) -> Dict[str, Any]:
        """
        整体评估入口
        """
        # 1. GeneralEvaluator：拿到 trip_plan，并跑所有 general 校验
        meta_info = {
            "id": meta_dict["trip_id"],   # 直接取 trip_id
            "route": meta_dict["route"]   # 直接复用已有 route 数据
        }
        trip_plan, general_results = self._run_general_evaluations(meta_info, raw_text)
        # print("trip_plan:", trip_plan)
        # print(general_results)

        # 如果 trip_plan 都没拿到，后面 rubric 校验也没法做，直接返回
        if trip_plan is None:
            error_text = self._build_error_text(general_results, [])
            return {
                "general_results": general_results,
                "id_true_results": [],
                "id_false_results": [],
                "error_text": error_text,
            }
        trip_plan = trip_plan.get("trip_plan", {})

        # 2. 从 trip_plan 中抽取各个 evaluator 需要的参数
        attraction_info = self._extract_attraction_info(trip_plan)
        restaurant_info = self._extract_restaurant_info(trip_plan)
        hotel_product_list = self._extract_hotel_product_list(trip_plan)
        # transportation_info = self._extract_transportation_info(trip_plan)
        transportation_info = self._extract_transportation_info(trip_plan, meta_dict)


        # print("attraction_info:", attraction_info)
        # print("restaurant_info:", restaurant_info)
        # print("hotel_product_list:", hotel_product_list)
        # print("transportation_info:", transportation_info)

        domain_inputs = {
            "attraction": attraction_info,
            "restaurant": restaurant_info,
            "hotel": hotel_product_list,
            "transportation": transportation_info,
        }

        # 3. 逐个 id 在 meta_dict 中找到 validate_func 和 value，并调用对应 evaluator
        id_true_results: List[Dict[str, Any]] = []
        id_false_results: List[Dict[str, Any]] = []
        violation_texts: List[str] = []

        rubric_results = meta_dict.get("rubric_results", {})

        for _id in id_list:
            ctx = self._find_rubric_context(rubric_results, _id)
            if ctx is None:
                # 找不到这个 _id，对你来说通常是配置问题，直接当成失败
                record = {
                    "id": _id,
                    "rubric_key": None,
                    "description_name": None,
                    "validate_func": None,
                    "domain": None,
                    "label_key": None,
                    "value": None,
                    "ok": False,
                    "msg": f"rubric_results 中未找到 id: {_id}",
                }
                id_false_results.append(record)
                violation_texts.append(f"[{_id}] rubric_results 中未找到该 id")
                continue
            
            # print("ctx:", ctx)
            (
                rubric_key,
                description_name,
                desc_val,
                label_key,
                label_obj,
            ) = ctx

            validate_func_name: Optional[str] = desc_val.get("validate_func")
            violation_desc_template: str = desc_val.get("violation_description", "")
            validation_params = label_obj.get("value")  # 统一使用 "value"
            # validation_params = {label_key: label_obj.get("value")}

            domain = self._infer_domain_from_id(label_obj.get("_id", _id))
            if domain == "hotel":
                validation_params = {label_key: label_obj.get("value")}

            # print("validation_params:", validation_params)

            evaluator, domain_input = self._get_evaluator_and_input(
                domain_inputs, domain
            )

            if evaluator is None or domain_input is None or not validate_func_name:
                msg = (
                    f"无法执行 {rubric_key}/{_id}："
                    f"evaluator={evaluator}, domain_input={domain_input}, "
                    f"validate_func_name={validate_func_name}"
                )
                record = {
                    "id": _id,
                    "rubric_key": rubric_key,
                    "description_name": description_name,
                    "validate_func": validate_func_name,
                    "domain": domain,
                    "label_key": label_key,
                    "value": validation_params,
                    "ok": False,
                    "msg": msg,
                }
                id_false_results.append(record)
                violation_texts.append(f"[{_id}] {msg}")
                continue

            # 真正调用 evaluator 对应的 validate_xxx
            validate_func = getattr(evaluator, validate_func_name, None)
            if validate_func is None:
                msg = f"在 {domain} evaluator 中找不到方法 {validate_func_name}"
                record = {
                    "id": _id,
                    "rubric_key": rubric_key,
                    "description_name": description_name,
                    "validate_func": validate_func_name,
                    "domain": domain,
                    "label_key": label_key,
                    "value": validation_params,
                    "ok": False,
                    "msg": msg,
                }
                id_false_results.append(record)
                violation_texts.append(f"[{_id}] {msg}")
                continue

            try:
                result = validate_func(domain_input, validation_params)
                # 兼容：
                #   - 仅返回 bool
                #   - 返回 (bool, msg)
                if isinstance(result, tuple) and len(result) >= 2 and isinstance(
                    result[0], bool
                ):
                    ok, detail_msg = result[0], result[1]
                elif isinstance(result, bool):
                    ok, detail_msg = result, ""
                else:
                    ok, detail_msg = bool(result), ""
            except Exception as e:
                ok, detail_msg = False, f"调用 {validate_func_name} 出错: {e}"

            record = {
                "id": _id,
                "rubric_key": rubric_key,
                "description_name": description_name,
                "validate_func": validate_func_name,
                "domain": domain,
                "label_key": label_key,
                "value": validation_params,
                "ok": ok,
                "msg": detail_msg,
            }

            if ok:
                id_true_results.append(record)
            else:
                id_false_results.append(record)
                violation_text = self._format_violation_text(
                    _id, violation_desc_template, label_key, detail_msg
                )
                violation_texts.append(violation_text)

        # 4. 组合错误文本：General 的 msg + 各个 id 的 violation_description
        error_text = self._build_error_text(general_results, violation_texts)

        return {
            "general_results": general_results,
            "id_true_results": id_true_results,
            "id_false_results": id_false_results,
            "error_text": error_text,
        }

    # ======================
    # GeneralEvaluator 相关
    # ======================
    def _run_general_evaluations(
        self, meta_info: Dict[str, Any], raw_text: str
    ) -> Tuple[Optional[Dict[str, Any]], List[Dict[str, Any]]]:
        """
        先用 GeneralEvaluator 拿到 trip_plan，再按顺序跑所有 evaluate_xxx。
        """
        general_results: List[Dict[str, Any]] = []

        # 1) validate_trip_plan_json：拿 trip_plan
        try:
            result = self.general_evaluator.validate_trip_plan_json(meta_info, raw_text)
        except Exception as e:
            result = {"ok": False, "msg": f"validate_trip_plan_json 异常: {e}", "trip_plan": None}

        # --- 正确解析 result dict，而不是 tuple 解包 ---
        ok = bool(result.get("ok"))
        msg = result.get("msg", "")
        trip_plan = result.get("trip_plan")

        general_results.append(
            {
                "name": "validate_trip_plan_json",
                "ok": ok,
                "msg": msg,
            }
        )

        # JSON 校验失败 → 后续 evaluator 无意义，直接返回
        if trip_plan is None:
            return trip_plan, general_results

        # 2) 其它 evaluate_xxx(meta, trip_plan)
        for func_name in self.GENERAL_FUNCS_ORDER:
            func = getattr(self.general_evaluator, func_name, None)
            if func is None:
                general_results.append(
                    {
                        "name": func_name,
                        "ok": False,
                        "msg": f"GeneralEvaluator 中不存在方法 {func_name}",
                    }
                )
                continue

            try:
                r = func(meta_info, trip_plan)
            except Exception as e:
                r = {"ok": False, "msg": f"{func_name} 调用异常: {e}"}

            # --- 解析 evaluator dict 返回结构 ---
            ok = bool(r.get("ok"))
            msg = r.get("msg", "")

            general_results.append(
                {
                    "name": func_name,
                    "ok": ok,
                    "msg": msg,
                }
            )

        return trip_plan, general_results

    # ======================
    # 从 meta_dict 里找某个 _id 的上下文
    # ======================
    def _find_rubric_context(
        self, rubric_results: Dict[str, Any], target_id: str
    ) -> Optional[Tuple[str, str, Dict[str, Any], str, Dict[str, Any]]]:
        """
        在 meta_dict['rubric_results'] 里找到目标 _id 对应的：
        - rubric_key           (例如 "RUBRIC_INCLUDE_CATEGORIES")
        - description_name     (例如 "Must include attractions of specific categories")
        - desc_val             (description 对应的 dict，含 validate_func, violation_description 等)
        - label_key            (all_labels_and_ranges 的那个 key，例如 "Night Views & Night Tours, Natural Scenery")
        - label_obj            ({ "value": ..., "_id": ... })
        """
        for rubric_key, rubric_detail in rubric_results.items():
            if not isinstance(rubric_detail, dict):
                continue

            for desc_key, desc_val in rubric_detail.items():
                if desc_key == "_id" or not isinstance(desc_val, dict):
                    continue

                res = desc_val.get("result")
                if not isinstance(res, dict):
                    continue

                all_labels = res.get("all_labels_and_ranges", {})
                if not isinstance(all_labels, dict):
                    continue

                for label_key, label_obj in all_labels.items():
                    if (
                        isinstance(label_obj, dict)
                        and label_obj.get("_id") == target_id
                    ):
                        return rubric_key, desc_key, desc_val, label_key, label_obj

        return None

    # ======================
    # 域判断 & 输入构造
    # ======================
    def _infer_domain_from_id(self, _id: str) -> Optional[str]:
        """
        根据 _id 前缀来推断，约定：
        - attraction_xxx -> "attraction"
        - restaurant_xxx -> "restaurant"
        - hotel_xxx -> "hotel"
        - transport_xxx / transportation_xxx -> "transportation"
        """
        if not _id:
            return None
        if _id.startswith("attraction_"):
            return "attraction"
        if _id.startswith("restaurant_"):
            return "restaurant"
        if _id.startswith("hotel_"):
            return "hotel"
        if _id.startswith("transport_") or _id.startswith("transportation_"):
            return "transportation"
        return None

    def _get_evaluator_and_input(
        self,
        domain_inputs: Dict[str, Any],
        domain: Optional[str],
    ):
        if domain is None:
            return None, None

        if domain == "attraction":
            return self.attraction_evaluator, domain_inputs["attraction"]
        if domain == "restaurant":
            return self.restaurant_evaluator, domain_inputs["restaurant"]
        if domain == "hotel":
            return self.hotel_evaluator, domain_inputs["hotel"]
        if domain == "transportation":
            return self.transportation_evaluator, domain_inputs["transportation"]

        return None, None

    # ======================
    # trip_plan → 各 evaluator 输入构造
    # ======================
    @staticmethod
    def _parse_duration_minutes(time_range: str) -> int:
        """
        "HH:MM-HH:MM" -> 持续分钟数
        """
        try:
            start, end = time_range.split("-")
            sh, sm = map(int, start.split(":"))
            eh, em = map(int, end.split(":"))
            duration = (eh * 60 + em) - (sh * 60 + sm)
            return max(duration, 0)
        except Exception:
            return 0

    def _extract_attraction_info(self, trip_plan: Dict[str, Any]) -> List[Dict[str, Any]]:
        """
        景点 evaluator 需要的形式：
        validate_include_attractions(test_attraction_info, validation_params)
        test_attraction_info = [
            {"poiId": 99446, "visiting_time": 120},
            ...
        ]
        """
        result = []
        for day in trip_plan.get("daily_schedule", []):
            for act in day.get("activities", []):
                if act.get("type") == "Attraction":
                    poi_id = act.get("id")
                    time_range = act.get("time", "")
                    visiting_time = self._parse_duration_minutes(time_range)
                    if isinstance(poi_id, str) and poi_id.isdigit():
                        poi_id_val: Any = int(poi_id)
                    else:
                        poi_id_val = poi_id
                    result.append(
                        {
                            "poiId": poi_id_val,
                            "visiting_time": visiting_time,
                        }
                    )
        return result

    def _get_time_rubric_type(self, meta_dict: Dict[str, Any], rubric_key: str) -> Optional[str]:
      """
      从 meta_dict['rubric_results'][rubric_key] 中提取 validate_func，
      并映射成 type 标记：
        - RUBRIC_TIME_DEPART: arrive->d_a, depart->d_d
        - RUBRIC_TIME_RETURN: arrive->r_a, depart->r_d
      """
      rubric_results = (meta_dict or {}).get("rubric_results", {})
      block = rubric_results.get(rubric_key)
      if not isinstance(block, dict):
          return None

      # 可能有多个 description 条目，收集所有 validate_func
      validate_funcs = []
      for _, desc_val in block.items():
          if not isinstance(desc_val, dict):
              continue
          vf = desc_val.get("validate_func")
          if isinstance(vf, str) and vf:
              validate_funcs.append(vf)

      if not validate_funcs:
          return None

      # 规则：优先 arrive，其次 depart（如果两种同时存在）
      has_arrive = "validate_arrive_time_condition" in validate_funcs
      has_depart = "validate_depart_time_condition" in validate_funcs

      if rubric_key == "RUBRIC_TIME_DEPART":
          if has_arrive:
              return "d_a"
          if has_depart:
              return "d_d"
      elif rubric_key == "RUBRIC_TIME_RETURN":
          if has_arrive:
              return "r_a"
          if has_depart:
              return "r_d"

      return None

    def _extract_transportation_info(
        self, trip_plan: Dict[str, Any], meta_dict: Optional[Dict[str, Any]] = None
    ) -> List[Dict[str, Any]]:
        """
        交通 evaluator：
        def validate_transport_cost_less_per_person_one_way(self, transportation_info, validation_params)
        transportation_info 形如：
        [
            {"id": "Flight_123", "product_id": "P123", "date": "2025-11-10", "quantity": 1, "type": "d_a"},
            ...
        ]

        若 meta_dict 中存在 RUBRIC_TIME_DEPART / RUBRIC_TIME_RETURN：
          - 给第一个交通 dict 加 type: d_a / d_d
          - 给最后一个交通 dict 加 type: r_a / r_d
        """
        result: List[Dict[str, Any]] = []

        for day in trip_plan.get("daily_schedule", []):
            date = day.get("date")
            for act in day.get("activities", []):
                if act.get("type") in ["Intercity Transportation", "Flight", "Train"]:
                    trans_id = act.get("id")
                    for prod in act.get("products", []):
                        result.append(
                            {
                                "id": trans_id,
                                "product_id": prod.get("id"),
                                "date": date,
                                "quantity": prod.get("quantity", 1),
                            }
                        )

        # ===== 根据 rubric 打标 =====
        if meta_dict and result:
            depart_type = self._get_time_rubric_type(meta_dict, "RUBRIC_TIME_DEPART")
            return_type = self._get_time_rubric_type(meta_dict, "RUBRIC_TIME_RETURN")

            def _attach_type(item: Dict[str, Any], t: str) -> None:
                if not t:
                    return
                if "type" not in item:
                    item["type"] = t
                # else:
                #     # 极端情况：只有一段交通同时被标记（first==last），避免覆盖
                #     if isinstance(item["type"], list):
                #         if t not in item["type"]:
                #             item["type"].append(t)
                #     else:
                #         if item["type"] != t:
                #             item["type"] = [item["type"], t]

            if depart_type:
                _attach_type(result[0], depart_type)
            if return_type:
                _attach_type(result[-1], return_type)

        return result

    def _extract_restaurant_info(
        self, trip_plan: Dict[str, Any]
    ) -> List[Dict[str, Any]]:
        """
        餐厅 evaluator：
        def validate_exclude_cuisines(self, restaurant_info, validation_params)
        restaurant_info = [{"id": "rest_1"}, {"id": "rest_2"}]
        """
        result = []
        for day in trip_plan.get("daily_schedule", []):
            for act in day.get("activities", []):
                if act.get("type") == "Restaurant":
                    rest_id = act.get("id")
                    if rest_id:
                        result.append({"id": rest_id})
        return result

    def _find_last_transport_id(self, trip_plan: Dict[str, Any]) -> Optional[str]:
        """
        transport_id 是最后一天交通的 id（遍历所有天，最后一个 Intercity/Flight/Train）
        """
        last_id = None
        for day in trip_plan.get("daily_schedule", []):
            for act in day.get("activities", []):
                if act.get("type") in ["Intercity Transportation", "Flight", "Train"]:
                    last_id = act.get("id")
        return last_id

    # def _extract_hotel_product_list(
    #     self, trip_plan: Dict[str, Any]
    # ) -> List[Dict[str, Any]]:
    #     """
    #     酒店 evaluator：
    #     def validate_last_night(self, product_list, validation_params)
    #     product_list 形如：
    #     [
    #         {
    #             "product_id": "P_H_00023581",
    #             "date":"2025-11-13",
    #             "total_person_num":"4",
    #             "room_num":"2",
    #             "transport_id": "Flight_xxx"
    #         },
    #         ...
    #     ]
    #     """
    #     result = []
    #     total_person_num = trip_plan.get("number_of_people")
    #     if not isinstance(total_person_num, str):
    #         total_person_num = str(total_person_num) if total_person_num is not None else ""

    #     last_transport_id = self._find_last_transport_id(trip_plan)

    #     for day in trip_plan.get("daily_schedule", []):
    #         date = day.get("date")
    #         hotel = day.get("hotel")
    #         if not hotel:
    #             continue
    #         for product in hotel.get("products", []):
    #             room_num = product.get("room_num")
    #             if not isinstance(room_num, str) and room_num is not None:
    #                 room_num = str(room_num)
    #             result.append(
    #                 {
    #                     "product_id": product.get("id"),
    #                     "date": date,
    #                     "total_person_num": total_person_num,
    #                     "room_num": room_num,
    #                     "transport_id": last_transport_id,
    #                 }
    #             )
        
    #     return result
    
    def _extract_hotel_product_list(
        self, trip_plan: Dict[str, Any]
    ) -> List[Dict[str, Any]]:
        """
        酒店 evaluator：
        def validate_last_night(self, product_list, validation_params)
        product_list 形如：
        [
            {
                "product_id": "P_H_00023581",
                "date":"2025-11-13",
                "total_person_num":"4",
                "room_num":"2",
                "transport_id": "Flight_xxx"
            },
            ...
        ]
        """
        result = []
        total_person_num = trip_plan.get("number_of_people")
        if not isinstance(total_person_num, str):
            total_person_num = str(total_person_num) if total_person_num is not None else ""

        last_transport_id = self._find_last_transport_id(trip_plan)

        for day in trip_plan.get("daily_schedule", []):
            date = day.get("date")
            hotel = day.get("hotel")
            if not hotel:
                continue

            for product in hotel.get("products", []):
                # ✅ 增加容错：product 可能不是 dict，避免 product.get 崩溃
                if isinstance(product, dict):
                    product_id = product.get("id")
                    room_num = product.get("room_num")
                elif isinstance(product, str):
                    # str 只表示 id，取不到 room_num
                    product_id = product
                    room_num = ""
                else:
                    continue

                if not isinstance(room_num, str) and room_num is not None:
                    room_num = str(room_num)

                result.append(
                    {
                        "product_id": product_id,
                        "date": date,
                        "total_person_num": total_person_num,
                        "room_num": room_num,
                        "transport_id": last_transport_id,
                    }
                )

        return result



    # def _extract_transportation_info(
    #     self, trip_plan: Dict[str, Any]
    # ) -> List[Dict[str, Any]]:
    #     """
    #     交通 evaluator：
    #     def validate_transport_cost_less_per_person_one_way(self, transportation_info, validation_params)
    #     transportation_info 形如：
    #     [
    #         {"id": "Flight_123", "product_id": "P123", "date": "2025-11-10", "quantity": 1},
    #         ...
    #     ]
    #     """
    #     result = []
    #     for day in trip_plan.get("daily_schedule", []):
    #         date = day.get("date")
    #         for act in day.get("activities", []):
    #             if act.get("type") in ["Intercity Transportation", "Flight", "Train"]:
    #                 trans_id = act.get("id")
    #                 for prod in act.get("products", []):
    #                     result.append(
    #                         {
    #                             "id": trans_id,
    #                             "product_id": prod.get("id"),
    #                             "date": date,
    #                             "quantity": prod.get("quantity", 1),
    #                         }
    #                     )
    #     return result

    # ======================
    # 错误文案组合
    # ======================
    @staticmethod
    def _format_violation_text(
        _id: str,
        template: str,
        label_key: str,
        detail_msg: str,
    ) -> str:
        """
        错误文本 = msg + violation_description(slot 替换成 label_key)
        """
        try:
            violation_text = template.format(slot=label_key)
        except Exception:
            # 如果模板里没有 {slot}，就直接拼接 label_key
            violation_text = violation_text

        if detail_msg:
            return f"[ID: PF_{_id}] {detail_msg}；{violation_text}"
        else:
            return f"[ID: PF_{_id}] {violation_text}"

    @staticmethod
    def _build_error_text(
        general_results: List[Dict[str, Any]], violation_texts: List[str]
    ) -> str:
        """
        总错误文本 = 所有 general 的 msg (失败的) + 所有 rubric 失败的 violation_text
        """
        general_msgs = [
            f"[ID: GF_{item['name']}] {item['msg']}"
            for item in general_results
            if not item.get("ok") and item.get("msg")
        ]

        parts = []
        if general_msgs:
            parts.append("General Failed:\n" + "\n".join(general_msgs))
        if violation_texts:
            parts.append("Preference Failed:\n" + "\n".join(violation_texts))

        return "\n\n".join(parts) if parts else ""



if __name__ == "__main__":

    import json
    import yaml
    import os
    import sys
    # ====== PATH 修复 ======
    ROOT = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
    if ROOT not in sys.path:
        sys.path.insert(0, ROOT)

    print("Added ROOT:", ROOT)

    from Evaluation.Attraction.val_attractions import AttractionEvaluator
    from Evaluation.Restaurant.val_restaurants import RestaurantEvaluator
    from Evaluation.Hotel.val_hotels import HotelEvaluator
    from Evaluation.Transportation.val_transports import TransportationEvaluator
    from Evaluation.General.val_general import GeneralEvaluator,load_or_build_cache

    # 1. 读取 config.yaml
    with open("Tools/config.yaml", "r", encoding="utf-8") as f:
        config = yaml.safe_load(f)

    data_path = config["data_path"]

    # 2. 根据 config.yaml 提取路径
    attraction_path = data_path["attraction"]
    restaurant_path = data_path["restaurant"]
    hotel_path = data_path["hotel"]
    flight_path = data_path["flight"]
    train_path = data_path["train"]   # 如果你之后需要

    # 3. 初始化评估器
    with open(attraction_path, 'r', encoding='utf-8') as f:
        attractions_data = json.load(f)

    attraction_evaluator = AttractionEvaluator(attractions_data)
    restaurant_evaluator = RestaurantEvaluator(restaurants_path=restaurant_path)
    hotel_evaluator = HotelEvaluator(hotel_path=hotel_path)
    transportation_evaluator = TransportationEvaluator(flights_path=flight_path)

    # 4. GeneralEvaluator 使用完整的 config 数据
    data = load_or_build_cache("Tools/config.yaml")
    general_evaluator = GeneralEvaluator(data)


    tpe = TripPlanEvaluator(
        attraction_evaluator=attraction_evaluator,
        restaurant_evaluator=restaurant_evaluator,
        hotel_evaluator=hotel_evaluator,
        transportation_evaluator=transportation_evaluator,
        general_evaluator=general_evaluator,
    )

    meta_dict =   {
    "trip_id": "c2_10413",
    "route": [
      {
        "from": "Guangzhou",
        "to": "Nanjing",
        "distance_km": 1129.71,
        "depart_date": "2025-10-22",
        "stay_days": 3,
        "return_date": "2025-10-24",
        "number_of_people": 2
      }
    ],
    "rubric_results": {
      "RUBRIC_HEAT_SCORE": {
        "Exclude attractions with specific popularity": {
          "probability": 1,
          "description": "All attractions included in the itinerary must exclude those that are {slot}.",
          "violation_description": "The itinerary includes some attractions that are {slot}.",
          "generate_func": "generate_exclude_popularity",
          "validate_func": "validate_exclude_popularity",
          "result": {
            "all_labels_and_ranges": {
              "hidden and less crowded": {
                "value": [
                  0,
                  3.7
                ],
                "_id": "attraction_5_2_2"
              }
            }
          },
          "_id": "attraction_5_2"
        }
      },
      "RUBRIC_COMMENT_COUNT": {
        "Include attractions that have more than a certain number of comments": {
          "probability": 1,
          "description": "All attractions included in the itinerary must be those that have {slot}.",
          "violation_description": "The itinerary includes attractions that do not have {slot}.",
          "generate_func": "generate_include_comment_count",
          "validate_func": "validate_include_comment_count",
          "result": {
            "all_labels_and_ranges": {
              "more than 50 comments": {
                "value": [
                  50,
                  1000000000.0
                ],
                "_id": "attraction_10_1_2"
              },
              "more than 300 comments": {
                "value": [
                  300,
                  1000000000.0
                ],
                "_id": "attraction_10_1_4"
              }
            }
          },
          "_id": "attraction_10_1"
        }
      },
      "RUBRIC_OPEN": {
        "Exclude must-reservation restaurants": {
          "probability": 1,
          "description": "Exclude restaurants that require mandatory advance reservations.",
          "violation_description": "Some selected restaurants still require mandatory advance reservations.",
          "generate_func": "generate_exclude_must_reserve",
          "validate_func": "validate_exclude_must_reserve",
          "result": {
            "all_labels_and_ranges": {
              "exclude_must_reserve": {
                "value": [
                  True
                ],
                "_id": "restaurant_6_2_1"
              }
            }
          },
          "_id": "restaurant_6_2"
        }
      },
      "RUBRIC_PRICE": {
        "Per-person per-meal cost between a certain price range": {
          "probability": 1,
          "description": "Each selected restaurant has a per-person per-meal cost that falls within {slot}.",
          "violation_description": "Some recommended restaurants have a per-person per-meal cost outside {slot}.",
          "generate_func": "generate_price_per_person_range",
          "validate_func": "validate_price_per_person_range",
          "result": {
            "all_labels_and_ranges": {
              "50-150": {
                "value": [
                  50.0,
                  150.0
                ],
                "_id": "restaurant_1_4_2"
              },
              "100-150": {
                "value": [
                  100.0,
                  150.0
                ],
                "_id": "restaurant_1_4_3"
              }
            }
          },
          "_id": "restaurant_1_4"
        }
      },
      "RUBRIC_TIME_DEPART": {
        "Depart after a specific time": {
          "probability": 1,
          "description": "For the outbound trip, depart after {slot}.",
          "violation_description": "The outbound departure time is not after {slot}.",
          "generate_func": "generate_depart_after_time",
          "validate_func": "validate_depart_time_condition",
          "result": {
            "all_labels_and_ranges": {
              "after 15:00": {
                "value": "15:00-23:59",
                "_id": "transport_1_4_10"
              }
            }
          },
          "_id": "transport_1_4"
        }
      },
      "RUBRIC_BREAKFAST_NUMBER": {
        "Must have breakfast more than a certain number": {
          "probability": 1,
          "description": "Breakfast number must more than {slot}.",
          "violation_description": "Breakfast number is not more than {slot}.",
          "generate_func": "generate_breakfast_number",
          "validate_func": "validate_breakfast_number_more",
          "result": {
            "all_labels_and_ranges": {
              "1": {
                "value": 1,
                "_id": "hotel_9_2_2"
              }
            }
          },
          "_id": "hotel_9_2"
        }
      }
    },
    "applied_modification_chains": {
      "RUBRIC_HEAT_SCORE": [
        "attraction_5_2_2"
      ],
      "RUBRIC_COMMENT_COUNT": [
        "attraction_10_1_2",
        "attraction_10_1_4"
      ],
      "RUBRIC_OPEN": [
        "restaurant_6_2_1"
      ],
      "RUBRIC_PRICE": [
        "restaurant_1_4_2",
        "restaurant_1_4_3"
      ],
      "RUBRIC_TIME_DEPART": [
        "transport_1_4_10"
      ],
      "RUBRIC_BREAKFAST_NUMBER": [
        "hotel_9_2_2"
      ]
    },
    "SAMPLE_RUBRIC_COUNT_attraction": 2,
    "SAMPLE_RUBRIC_COUNT_hotel": 1,
    "SAMPLE_RUBRIC_COUNT_restaurant": 2,
    "SAMPLE_RUBRIC_COUNT_transport": 1,
    "TOTAL_CHAIN_LENGTH_attraction": 3,
    "TOTAL_CHAIN_LENGTH_hotel": 1,
    "TOTAL_CHAIN_LENGTH_restaurant": 3,
    "TOTAL_CHAIN_LENGTH_transport": 1,
    "SAMPLE_RUBRIC_COUNT": 6,
    "TOTAL_CHAIN_LENGTH": 8
  }
    
    trip_plan = {
  "trip_plan": {
    "start_date": "2025-10-24",
    "end_date": "2025-10-31",
    "number_of_people": 2,
    "daily_schedule": [
      {
        "date": "2025-10-24",
        "cities": "Guangzhou -> Shanghai",
        "hotel": {
          "id": "Hotel_00005783",
          "products": [
            {
              "id": "P_H_00108230",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "06:00-07:30",
            "type": "Local Transportation",
            "description": "Transfer from Guangzhou city center to Baiyun Airport"
          },
          {
            "time": "07:30-08:00",
            "type": "Flight Check-in",
            "description": "Check in for flight MU9302 at Baiyun Airport"
          },
          {
            "time": "08:00-10:20",
            "type": "Intercity Transportation",
            "id": "Flight_00002709",
            "products": [
              {
                "id": "P_F_00021668",
                "quantity": 2
              }
            ],
            "description": "Flight MU9302 from Guangzhou Baiyun to Shanghai Pudong"
          },
          {
            "time": "10:20-11:15",
            "type": "Local Transportation",
            "description": "Transfer from Pudong Airport to hotel in Shanghai city center"
          },
          {
            "time": "11:15-12:00",
            "type": "Hotel Check-in",
            "description": "Check in at Dulaimi Hotel Apartments. Have lunch nearby."
          },
          {
            "time": "12:00-13:30",
            "type": "Restaurant",
            "id": "restaurant_4882841",
            "products": [
              {
                "id": "Pf96d3492e32c",
                "quantity": 1
              }
            ],
            "description": "Lunch at Old Shanghai Scallion Pancake - authentic local cuisine"
          },
          {
            "time": "13:30-14:00",
            "type": "Local Transportation",
            "description": "Travel from hotel to People's Square"
          },
          {
            "time": "14:00-16:00",
            "type": "Attraction",
            "id": "91002",
            "products": [],
            "description": "Visit People's Square - Shanghai's landmark and transportation hub"
          },
          {
            "time": "16:00-16:30",
            "type": "Local Transportation",
            "description": "From People's Square to Lujiazui financial district"
          },
          {
            "time": "16:30-19:30",
            "type": "Attraction",
            "id": "24302219",
            "products": [],
            "description": "Explore Lujiazui financial center and enjoy city views"
          },
          {
            "time": "19:30-20:00",
            "type": "Local Transportation",
            "description": "Return from Lujiazui to hotel area"
          },
          {
            "time": "20:00-21:30",
            "type": "Restaurant",
            "id": "restaurant_7248651",
            "products": [
              {
                "id": "P9a9a501a2027",
                "quantity": 1
              }
            ],
            "description": "Dinner at Domino's Pizza - Western cuisine option"
          },
          {
            "time": "21:30-22:00",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-25",
        "cities": "Shanghai",
        "hotel": {
          "id": "Hotel_00005783",
          "products": [
            {
              "id": "P_H_00108230",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "08:00-09:00",
            "type": "Local Transportation",
            "description": "From hotel to Pudong Art Museum"
          },
          {
            "time": "09:00-11:00",
            "type": "Attraction",
            "id": "124908212",
            "products": [
              {
                "id": "124908212_1",
                "quantity": 2
              }
            ],
            "description": "Visit Pudong Art Museum - Lujiazui's cultural landmark"
          },
          {
            "time": "11:00-11:30",
            "type": "Local Transportation",
            "description": "Travel to lunch location"
          },
          {
            "time": "11:30-13:00",
            "type": "Restaurant",
            "id": "restaurant_4024363",
            "products": [
              {
                "id": "P2ad5bd4b60a9",
                "quantity": 1
              }
            ],
            "description": "Lunch at Yiteng Sushi - Japanese cuisine"
          },
          {
            "time": "13:00-13:30",
            "type": "Local Transportation",
            "description": "Travel to Shanghai Film Museum"
          },
          {
            "time": "13:30-16:00",
            "type": "Attraction",
            "id": "10524120",
            "products": [
              {
                "id": "product_10524120_1",
                "quantity": 2
              }
            ],
            "description": "Visit Shanghai Film Museum - comprehensive film history exhibit"
          },
          {
            "time": "16:00-16:30",
            "type": "Local Transportation",
            "description": "Travel to dinner location"
          },
          {
            "time": "16:30-18:00",
            "type": "Restaurant",
            "id": "restaurant_8425267",
            "products": [],
            "description": "Dinner at LOKAL - Western cuisine (order on site)"
          },
          {
            "time": "18:00-18:30",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-26",
        "cities": "Shanghai",
        "hotel": {
          "id": "Hotel_00005783",
          "products": [
            {
              "id": "P_H_00108230",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "08:00-09:00",
            "type": "Local Transportation",
            "description": "From hotel to World Expo Cultural Park"
          },
          {
            "time": "09:00-12:00",
            "type": "Attraction",
            "id": "131917111",
            "products": [],
            "description": "Explore World Expo Cultural Park - historical exhibition site"
          },
          {
            "time": "12:00-12:30",
            "type": "Local Transportation",
            "description": "Travel to lunch location"
          },
          {
            "time": "12:30-14:00",
            "type": "Restaurant",
            "id": "restaurant_4503589",
            "products": [],
            "description": "Lunch at Hengding Bento - Japanese cuisine (order on site)"
          },
          {
            "time": "14:00-14:30",
            "type": "Local Transportation",
            "description": "Travel to Aurora Museum"
          },
          {
            "time": "14:30-17:00",
            "type": "Attraction",
            "id": "10533320",
            "products": [
              {
                "id": "product_10533320_1",
                "quantity": 2
              }
            ],
            "description": "Visit Aurora Museum - contemporary art exhibition"
          },
          {
            "time": "17:00-17:30",
            "type": "Local Transportation",
            "description": "Travel to dinner location"
          },
          {
            "time": "17:30-19:00",
            "type": "Restaurant",
            "id": "restaurant_7826284",
            "products": [],
            "description": "Dinner at Hamana - Fast food option (order on site)"
          },
          {
            "time": "19:00-19:30",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-27",
        "cities": "Shanghai",
        "hotel": {
          "id": "Hotel_00005783",
          "products": [
            {
              "id": "P_H_00108230",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "08:00-09:00",
            "type": "Local Transportation",
            "description": "From hotel to 351m Space Capsule"
          },
          {
            "time": "09:00-10:00",
            "type": "Attraction",
            "id": "60697915",
            "products": [
              {
                "id": "product_60697915_1",
                "quantity": 2
              }
            ],
            "description": "Experience 351m Space Capsule - panoramic city views"
          },
          {
            "time": "10:00-10:30",
            "type": "Local Transportation",
            "description": "Travel to lunch location"
          },
          {
            "time": "10:30-12:00",
            "type": "Restaurant",
            "id": "restaurant_9775355",
            "products": [],
            "description": "Lunch at Noodle Canteen - local snacks (order on site)"
          },
          {
            "time": "12:00-12:30",
            "type": "Local Transportation",
            "description": "Travel to Sleep No More attraction"
          },
          {
            "time": "12:30-15:00",
            "type": "Attraction",
            "id": "56140154",
            "products": [
              {
                "id": "product_56140154_1",
                "quantity": 2
              }
            ],
            "description": "Experience Sleep No More - immersive theater performance"
          },
          {
            "time": "15:00-15:30",
            "type": "Local Transportation",
            "description": "Travel to dinner location"
          },
          {
            "time": "15:30-17:00",
            "type": "Restaurant",
            "id": "restaurant_7147887",
            "products": [],
            "description": "Dinner at Grinder Burger - Fast food (order on site)"
          },
          {
            "time": "17:00-17:30",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-28",
        "cities": "Shanghai",
        "hotel": {
          "id": "Hotel_00005783",
          "products": [
            {
              "id": "P_H_00108230",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "08:00-09:00",
            "type": "Local Transportation",
            "description": "From hotel to Shanghai city center for shopping and exploration"
          },
          {
            "time": "09:00-12:00",
            "type": "Attraction",
            "id": "91002",
            "products": [],
            "description": "Explore People's Square area - shopping and cultural experiences"
          },
          {
            "time": "12:00-12:30",
            "type": "Local Transportation",
            "description": "Travel to lunch location"
          },
          {
            "time": "12:30-14:00",
            "type": "Restaurant",
            "id": "restaurant_4882841",
            "products": [
              {
                "id": "Pf96d3492e32c",
                "quantity": 1
              }
            ],
            "description": "Lunch at Old Shanghai Scallion Pancake - local specialties"
          },
          {
            "time": "14:00-17:00",
            "type": "Attraction",
            "id": "24302219",
            "products": [],
            "description": "Free time in Lujiazui - shopping, photos, and city exploration"
          },
          {
            "time": "17:00-17:30",
            "type": "Local Transportation",
            "description": "Travel to farewell dinner location"
          },
          {
            "time": "17:30-19:00",
            "type": "Restaurant",
            "id": "restaurant_4024363",
            "products": [
              {
                "id": "P2ad5bd4b60a9",
                "quantity": 1
              }
            ],
            "description": "Farewell dinner at Yiteng Sushi - Japanese cuisine"
          },
          {
            "time": "19:00-19:30",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-29",
        "cities": "Shanghai -> Suzhou",
        "hotel": {
          "id": "Hotel_00065452",
          "products": [
            {
              "id": "P_H_01216635",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "07:00-07:30",
            "type": "Local Transportation",
            "description": "From hotel to Shanghai Hongqiao Station"
          },
          {
            "time": "07:30-08:00",
            "type": "Intercity Transportation",
            "id": "Train_00000730",
            "products": [
              {
                "id": "P_T_00000730_1",
                "quantity": 2
              }
            ],
            "description": "Take D2216/D2217 train from Shanghai Hongqiao to Suzhou Industrial Park"
          },
          {
            "time": "08:00-08:30",
            "type": "Local Transportation",
            "description": "Transfer from Suzhou station to hotel"
          },
          {
            "time": "08:30-09:15",
            "type": "Hotel Check-in",
            "description": "Check in at Yilai Hotel in Suzhou"
          },
          {
            "time": "09:15-10:00",
            "type": "Local Transportation",
            "description": "Travel to The Humble Administrator's Garden"
          },
          {
            "time": "10:00-13:00",
            "type": "Attraction",
            "id": "82473",
            "products": [
              {
                "id": "82473_1",
                "quantity": 2
              }
            ],
            "description": "Visit The Humble Administrator's Garden - UNESCO World Heritage site"
          },
          {
            "time": "13:00-13:30",
            "type": "Local Transportation",
            "description": "Travel to lunch location"
          },
          {
            "time": "13:30-15:00",
            "type": "Restaurant",
            "id": "restaurant_2832072",
            "products": [
              {
                "id": "Paa897937f351",
                "quantity": 1
              }
            ],
            "description": "Lunch at Night Fishing Bullfrog Hot Pot - local specialty"
          },
          {
            "time": "15:00-15:30",
            "type": "Local Transportation",
            "description": "Travel to Liuyuan Garden"
          },
          {
            "time": "15:30-17:30",
            "type": "Attraction",
            "id": "10558598",
            "products": [
              {
                "id": "10558598_1",
                "quantity": 2
              }
            ],
            "description": "Visit Liuyuan Garden - classical Suzhou garden"
          },
          {
            "time": "17:30-18:00",
            "type": "Local Transportation",
            "description": "Travel to dinner location"
          },
          {
            "time": "18:00-19:30",
            "type": "Restaurant",
            "id": "restaurant_4027659",
            "products": [
              {
                "id": "Pede808cefb33",
                "quantity": 1
              }
            ],
            "description": "Dinner at Shang Du Japanese Restaurant - sushi and Japanese cuisine"
          },
          {
            "time": "19:30-20:00",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-30",
        "cities": "Suzhou",
        "hotel": {
          "id": "Hotel_00065452",
          "products": [
            {
              "id": "P_H_01216635",
              "room_num": 1
            }
          ]
        },
        "activities": [
          {
            "time": "08:00-09:00",
            "type": "Local Transportation",
            "description": "From hotel to Tiger Hill (Huqiu)"
          },
          {
            "time": "09:00-13:00",
            "type": "Attraction",
            "id": "76159",
            "products": [
              {
                "id": "76159_1",
                "quantity": 2
              }
            ],
            "description": "Visit Tiger Hill - historical site with panoramic views"
          },
          {
            "time": "13:00-13:30",
            "type": "Local Transportation",
            "description": "Travel to lunch location"
          },
          {
            "time": "13:30-15:00",
            "type": "Restaurant",
            "id": "restaurant_7045471",
            "products": [
              {
                "id": "P5013b10b679",
                "quantity": 1
              }
            ],
            "description": "Lunch at Pan Pai Maocai - local snacks (reservation required)"
          },
          {
            "time": "15:00-15:30",
            "type": "Local Transportation",
            "description": "Travel to Dongyuan Park"
          },
          {
            "time": "15:30-17:30",
            "type": "Attraction",
            "id": "13693019",
            "products": [],
            "description": "Visit Dongyuan Park - local park with cultural activities"
          },
          {
            "time": "17:30-18:00",
            "type": "Local Transportation",
            "description": "Travel to farewell dinner"
          },
          {
            "time": "18:00-19:30",
            "type": "Restaurant",
            "id": "restaurant_6073477",
            "products": [],
            "description": "Farewell dinner at Raw Vegetable Light Food - Western cuisine (order on site)"
          },
          {
            "time": "19:30-20:00",
            "type": "Local Transportation",
            "description": "Return to hotel"
          }
        ]
      },
      {
        "date": "2025-10-31",
        "cities": "Suzhou -> Departure",
        "activities": [
          {
            "time": "08:00-09:00",
            "type": "Local Transportation",
            "description": "From hotel to Suzhou city center for final exploration"
          },
          {
            "time": "09:00-11:00",
            "type": "Attraction",
            "id": "82473",
            "products": [],
            "description": "Final visit to The Humble Administrator's Garden or local market"
          },
          {
            "time": "11:00-12:00",
            "type": "Restaurant",
            "id": "restaurant_3576469",
            "products": [],
            "description": "Lunch at Red-billed Gull Hot Pot - local specialty (order on site)"
          },
          {
            "time": "12:00-13:00",
            "type": "Local Transportation",
            "description": "Travel to Suzhou station for departure"
          }
        ]
      }
    ]
  }
}
    
    raw_text = "Now I have all the information needed to create your complete 7-day itinerary. Let me build a comprehensive travel plan with optimized transportation, hotels, attractions, and restaurants.\n\n## Summary of Your Trip:\n- **Guangzhou → Shanghai**: Oct 24-29 (5 days)\n- **Shanghai → Suzhou**: Oct 29-31 (2 days)  \n- **Travelers**: 2 people\n- **Total Duration**: 7 days\n\nHere's your complete itinerary:\n\n```json\n{\n  \"trip_plan\": {\n    \"start_date\": \"2025-10-24\",\n    \"end_date\": \"2025-10-31\",\n    \"number_of_people\": 2,\n    \"daily_schedule\": [\n      {\n        \"date\": \"2025-10-24\",\n        \"cities\": \"Guangzhou -> Shanghai\",\n        \"hotel\": {\n          \"id\": \"Hotel_00005783\",\n          \"products\": [\n            {\n              \"id\": \"P_H_00108230\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"06:00-07:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Transfer from Guangzhou city center to Baiyun Airport\"\n          },\n          {\n            \"time\": \"07:30-08:00\",\n            \"type\": \"Flight Check-in\",\n            \"description\": \"Check in for flight MU9302 at Baiyun Airport\"\n          },\n          {\n            \"time\": \"08:00-10:20\",\n            \"type\": \"Intercity Transportation\",\n            \"id\": \"Flight_00002709\",\n            \"products\": [\n              {\n                \"id\": \"P_F_00021668\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Flight MU9302 from Guangzhou Baiyun to Shanghai Pudong\"\n          },\n          {\n            \"time\": \"10:20-11:15\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Transfer from Pudong Airport to hotel in Shanghai city center\"\n          },\n          {\n            \"time\": \"11:15-12:00\",\n            \"type\": \"Hotel Check-in\",\n            \"description\": \"Check in at Dulaimi Hotel Apartments. Have lunch nearby.\"\n          },\n          {\n            \"time\": \"12:00-13:30\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_4882841\",\n            \"products\": [\n              {\n                \"id\": \"Pf96d3492e32c\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Lunch at Old Shanghai Scallion Pancake - authentic local cuisine\"\n          },\n          {\n            \"time\": \"13:30-14:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel from hotel to People's Square\"\n          },\n          {\n            \"time\": \"14:00-16:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"91002\",\n            \"products\": [],\n            \"description\": \"Visit People's Square - Shanghai's landmark and transportation hub\"\n          },\n          {\n            \"time\": \"16:00-16:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From People's Square to Lujiazui financial district\"\n          },\n          {\n            \"time\": \"16:30-19:30\",\n            \"type\": \"Attraction\",\n            \"id\": \"24302219\",\n            \"products\": [],\n            \"description\": \"Explore Lujiazui financial center and enjoy city views\"\n          },\n          {\n            \"time\": \"19:30-20:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return from Lujiazui to hotel area\"\n          },\n          {\n            \"time\": \"20:00-21:30\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_7248651\",\n            \"products\": [\n              {\n                \"id\": \"P9a9a501a2027\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Dinner at Domino's Pizza - Western cuisine option\"\n          },\n          {\n            \"time\": \"21:30-22:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-25\",\n        \"cities\": \"Shanghai\",\n        \"hotel\": {\n          \"id\": \"Hotel_00005783\",\n          \"products\": [\n            {\n              \"id\": \"P_H_00108230\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"08:00-09:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to Pudong Art Museum\"\n          },\n          {\n            \"time\": \"09:00-11:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"124908212\",\n            \"products\": [\n              {\n                \"id\": \"124908212_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Visit Pudong Art Museum - Lujiazui's cultural landmark\"\n          },\n          {\n            \"time\": \"11:00-11:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to lunch location\"\n          },\n          {\n            \"time\": \"11:30-13:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_4024363\",\n            \"products\": [\n              {\n                \"id\": \"P2ad5bd4b60a9\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Lunch at Yiteng Sushi - Japanese cuisine\"\n          },\n          {\n            \"time\": \"13:00-13:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to Shanghai Film Museum\"\n          },\n          {\n            \"time\": \"13:30-16:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"10524120\",\n            \"products\": [\n              {\n                \"id\": \"product_10524120_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Visit Shanghai Film Museum - comprehensive film history exhibit\"\n          },\n          {\n            \"time\": \"16:00-16:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to dinner location\"\n          },\n          {\n            \"time\": \"16:30-18:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_8425267\",\n            \"products\": [],\n            \"description\": \"Dinner at LOKAL - Western cuisine (order on site)\"\n          },\n          {\n            \"time\": \"18:00-18:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-26\",\n        \"cities\": \"Shanghai\",\n        \"hotel\": {\n          \"id\": \"Hotel_00005783\",\n          \"products\": [\n            {\n              \"id\": \"P_H_00108230\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"08:00-09:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to World Expo Cultural Park\"\n          },\n          {\n            \"time\": \"09:00-12:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"131917111\",\n            \"products\": [],\n            \"description\": \"Explore World Expo Cultural Park - historical exhibition site\"\n          },\n          {\n            \"time\": \"12:00-12:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to lunch location\"\n          },\n          {\n            \"time\": \"12:30-14:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_4503589\",\n            \"products\": [],\n            \"description\": \"Lunch at Hengding Bento - Japanese cuisine (order on site)\"\n          },\n          {\n            \"time\": \"14:00-14:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to Aurora Museum\"\n          },\n          {\n            \"time\": \"14:30-17:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"10533320\",\n            \"products\": [\n              {\n                \"id\": \"product_10533320_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Visit Aurora Museum - contemporary art exhibition\"\n          },\n          {\n            \"time\": \"17:00-17:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to dinner location\"\n          },\n          {\n            \"time\": \"17:30-19:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_7826284\",\n            \"products\": [],\n            \"description\": \"Dinner at Hamana - Fast food option (order on site)\"\n          },\n          {\n            \"time\": \"19:00-19:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-27\",\n        \"cities\": \"Shanghai\",\n        \"hotel\": {\n          \"id\": \"Hotel_00005783\",\n          \"products\": [\n            {\n              \"id\": \"P_H_00108230\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"08:00-09:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to 351m Space Capsule\"\n          },\n          {\n            \"time\": \"09:00-10:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"60697915\",\n            \"products\": [\n              {\n                \"id\": \"product_60697915_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Experience 351m Space Capsule - panoramic city views\"\n          },\n          {\n            \"time\": \"10:00-10:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to lunch location\"\n          },\n          {\n            \"time\": \"10:30-12:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_9775355\",\n            \"products\": [],\n            \"description\": \"Lunch at Noodle Canteen - local snacks (order on site)\"\n          },\n          {\n            \"time\": \"12:00-12:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to Sleep No More attraction\"\n          },\n          {\n            \"time\": \"12:30-15:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"56140154\",\n            \"products\": [\n              {\n                \"id\": \"product_56140154_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Experience Sleep No More - immersive theater performance\"\n          },\n          {\n            \"time\": \"15:00-15:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to dinner location\"\n          },\n          {\n            \"time\": \"15:30-17:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_7147887\",\n            \"products\": [],\n            \"description\": \"Dinner at Grinder Burger - Fast food (order on site)\"\n          },\n          {\n            \"time\": \"17:00-17:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-28\",\n        \"cities\": \"Shanghai\",\n        \"hotel\": {\n          \"id\": \"Hotel_00005783\",\n          \"products\": [\n            {\n              \"id\": \"P_H_00108230\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"08:00-09:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to Shanghai city center for shopping and exploration\"\n          },\n          {\n            \"time\": \"09:00-12:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"91002\",\n            \"products\": [],\n            \"description\": \"Explore People's Square area - shopping and cultural experiences\"\n          },\n          {\n            \"time\": \"12:00-12:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to lunch location\"\n          },\n          {\n            \"time\": \"12:30-14:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_4882841\",\n            \"products\": [\n              {\n                \"id\": \"Pf96d3492e32c\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Lunch at Old Shanghai Scallion Pancake - local specialties\"\n          },\n          {\n            \"time\": \"14:00-17:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"24302219\",\n            \"products\": [],\n            \"description\": \"Free time in Lujiazui - shopping, photos, and city exploration\"\n          },\n          {\n            \"time\": \"17:00-17:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to farewell dinner location\"\n          },\n          {\n            \"time\": \"17:30-19:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_4024363\",\n            \"products\": [\n              {\n                \"id\": \"P2ad5bd4b60a9\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Farewell dinner at Yiteng Sushi - Japanese cuisine\"\n          },\n          {\n            \"time\": \"19:00-19:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-29\",\n        \"cities\": \"Shanghai -> Suzhou\",\n        \"hotel\": {\n          \"id\": \"Hotel_00065452\",\n          \"products\": [\n            {\n              \"id\": \"P_H_01216635\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"07:00-07:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to Shanghai Hongqiao Station\"\n          },\n          {\n            \"time\": \"07:30-08:00\",\n            \"type\": \"Intercity Transportation\",\n            \"id\": \"Train_00000730\",\n            \"products\": [\n              {\n                \"id\": \"P_T_00000730_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Take D2216/D2217 train from Shanghai Hongqiao to Suzhou Industrial Park\"\n          },\n          {\n            \"time\": \"08:00-08:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Transfer from Suzhou station to hotel\"\n          },\n          {\n            \"time\": \"08:30-09:15\",\n            \"type\": \"Hotel Check-in\",\n            \"description\": \"Check in at Yilai Hotel in Suzhou\"\n          },\n          {\n            \"time\": \"09:15-10:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to The Humble Administrator's Garden\"\n          },\n          {\n            \"time\": \"10:00-13:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"82473\",\n            \"products\": [\n              {\n                \"id\": \"82473_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Visit The Humble Administrator's Garden - UNESCO World Heritage site\"\n          },\n          {\n            \"time\": \"13:00-13:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to lunch location\"\n          },\n          {\n            \"time\": \"13:30-15:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_2832072\",\n            \"products\": [\n              {\n                \"id\": \"Paa897937f351\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Lunch at Night Fishing Bullfrog Hot Pot - local specialty\"\n          },\n          {\n            \"time\": \"15:00-15:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to Liuyuan Garden\"\n          },\n          {\n            \"time\": \"15:30-17:30\",\n            \"type\": \"Attraction\",\n            \"id\": \"10558598\",\n            \"products\": [\n              {\n                \"id\": \"10558598_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Visit Liuyuan Garden - classical Suzhou garden\"\n          },\n          {\n            \"time\": \"17:30-18:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to dinner location\"\n          },\n          {\n            \"time\": \"18:00-19:30\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_4027659\",\n            \"products\": [\n              {\n                \"id\": \"Pede808cefb33\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Dinner at Shang Du Japanese Restaurant - sushi and Japanese cuisine\"\n          },\n          {\n            \"time\": \"19:30-20:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-30\",\n        \"cities\": \"Suzhou\",\n        \"hotel\": {\n          \"id\": \"Hotel_00065452\",\n          \"products\": [\n            {\n              \"id\": \"P_H_01216635\",\n              \"room_num\": 1\n            }\n          ]\n        },\n        \"activities\": [\n          {\n            \"time\": \"08:00-09:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to Tiger Hill (Huqiu)\"\n          },\n          {\n            \"time\": \"09:00-13:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"76159\",\n            \"products\": [\n              {\n                \"id\": \"76159_1\",\n                \"quantity\": 2\n              }\n            ],\n            \"description\": \"Visit Tiger Hill - historical site with panoramic views\"\n          },\n          {\n            \"time\": \"13:00-13:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to lunch location\"\n          },\n          {\n            \"time\": \"13:30-15:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_7045471\",\n            \"products\": [\n              {\n                \"id\": \"P5013b10b679\",\n                \"quantity\": 1\n              }\n            ],\n            \"description\": \"Lunch at Pan Pai Maocai - local snacks (reservation required)\"\n          },\n          {\n            \"time\": \"15:00-15:30\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to Dongyuan Park\"\n          },\n          {\n            \"time\": \"15:30-17:30\",\n            \"type\": \"Attraction\",\n            \"id\": \"13693019\",\n            \"products\": [],\n            \"description\": \"Visit Dongyuan Park - local park with cultural activities\"\n          },\n          {\n            \"time\": \"17:30-18:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to farewell dinner\"\n          },\n          {\n            \"time\": \"18:00-19:30\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_6073477\",\n            \"products\": [],\n            \"description\": \"Farewell dinner at Raw Vegetable Light Food - Western cuisine (order on site)\"\n          },\n          {\n            \"time\": \"19:30-20:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Return to hotel\"\n          }\n        ]\n      },\n      {\n        \"date\": \"2025-10-31\",\n        \"cities\": \"Suzhou -> Departure\",\n        \"activities\": [\n          {\n            \"time\": \"08:00-09:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"From hotel to Suzhou city center for final exploration\"\n          },\n          {\n            \"time\": \"09:00-11:00\",\n            \"type\": \"Attraction\",\n            \"id\": \"82473\",\n            \"products\": [],\n            \"description\": \"Final visit to The Humble Administrator's Garden or local market\"\n          },\n          {\n            \"time\": \"11:00-12:00\",\n            \"type\": \"Restaurant\",\n            \"id\": \"restaurant_3576469\",\n            \"products\": [],\n            \"description\": \"Lunch at Red-billed Gull Hot Pot - local specialty (order on site)\"\n          },\n          {\n            \"time\": \"12:00-13:00\",\n            \"type\": \"Local Transportation\",\n            \"description\": \"Travel to Suzhou station for departure\"\n          }\n        ]\n      }\n    ]\n  }\n}\n```"
    
    id_list = ["attraction_10_1_4","restaurant_6_2_1", "transport_1_4_10","hotel_9_2_2"]
    
    result = tpe.evaluate(meta_dict=meta_dict,raw_text=raw_text,id_list=id_list)

    print(json.dumps(result, ensure_ascii=False, indent=2))

    print(result["error_text"])

    