import json
import math
import random
import datetime as dt
import os
from itertools import combinations
# random.seed(42)
# ------------------------------
# 注册系统：与 AttractionEvaluator 一致
# ------------------------------
def register_func(*names):
    """
    注册装饰器：可用于类方法，自动写入类属性 func_map。
    """
    def decorator(func):
        func._register_names = names
        return func
    return decorator


def collect_registered_funcs(cls):
    """
    类装饰器：在类定义完成后，自动收集所有带 _register_names 的函数。
    """
    cls.func_map = {}
    for attr_name, attr_value in cls.__dict__.items():
        if callable(attr_value) and hasattr(attr_value, "_register_names"):
            for name in attr_value._register_names:
                cls.func_map[name] = attr_value
    return cls


# ------------------------------
# 主类定义
# ------------------------------
@collect_registered_funcs
class TransportationEvaluator:
    def __init__(self, flights_path, trains_path=None):
        """
        flights_path: 航班 JSON 文件路径
        trains_path: 火车 JSON 文件路径，可选
        """
        self.flights_path = flights_path
        self.trains_path = trains_path or os.path.join(os.path.dirname(flights_path), 'trains.json')

        # 初始化时直接读取航班数据和火车数据
        with open(self.flights_path, "r", encoding="utf-8") as f:
            self.flights = json.load(f)
        
        with open(self.trains_path, "r", encoding="utf-8") as f:
            self.trains = json.load(f)

    # ---------------------------------
    # 调用执行入口
    # ---------------------------------
    def execute(self, rubric_key, *args, **kwargs):
        func = self.func_map.get(rubric_key)
        if not func:
            raise ValueError(f"未注册的函数: {rubric_key}")
        return func(self, *args, **kwargs)

    # ------------------------------
    # 工具函数
    # ------------------------------

    def parse_time_to_minutes(self,t_str):
        """辅助函数：把 'HH:MM' 转成分钟数"""
        try:
            h, m = map(int, t_str.split(":"))
            return h * 60 + m
        except:
            return None

    def generate_hour_ranges(self):
        """生成每两小时一段的时间范围映射，label为中间时间+平均"""
        ranges = {}
        base = dt.datetime.strptime("00:00", "%H:%M")
        for i in range(5,22):
            start = base + dt.timedelta(hours=1 * i)
            end = start + dt.timedelta(hours=2)
            mid = start + (end - start) / 2  # 取中点时间
            label = f"around {mid.strftime('%H:%M')}"
            period = f"{start.strftime('%H:%M')}-{end.strftime('%H:%M')}"
            ranges[label] = period
            # print(label, period)
        return ranges


    def _weekday_abbr(self, date_str: str) -> str:
        """将 YYYY-MM-DD 转为 Mon/Tue/..."""
        date_obj = dt.datetime.strptime(date_str, "%Y-%m-%d").date()
        week_abbr = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]
        return week_abbr[date_obj.weekday()]

    def _price_for_day(self, flight: dict, week_abbr: str):
        """获取当天价格"""
        key = f"{week_abbr}_lowest_price"
        return flight.get(key) if key in flight else flight.get("Price")



        #routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str,number_of_people的字典
    
    
    #示例
    # routes_info1 = [
    #     {
    #         "departure_city": "Beijing",
    #         "arrival_city": "Shanghai",
    #         "date_str": "2024-01-15", 
    #         "number_of_people": 5
    #     },
    #             {
    #         "departure_city": "Shanghai",
    #         "arrival_city": "Beijing",
    #         "date_str": "2024-01-16",  
    #         "number_of_people": 5
    #     }
    # ]




    # ------------------------------
    # Depart Time相关函数
    # ------------------------------

    @register_func("generate_depart_within_specific_range")
    def generate_depart_within_specific_range(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        """

        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")
    
        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']

        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
    
        # 时间段映射（每半小时）
        time_ranges = self.generate_hour_ranges()
    
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
    
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            ids = []
            for transport in transports:
                dep_time = self.parse_time_to_minutes(str(transport.get("Departure Time")))
                if dep_time is not None and start_min <= dep_time < end_min:
                    if transport['type'] == 'flight':
                        ids.append(transport.get("Flight_id"))
                    else:
                        ids.append(transport.get("Train_id"))
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
    
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
    

        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }


    @register_func("generate_depart_within_general_period")
    def generate_depart_within_general_period(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        特别处理：跨越00:00的时间段
        """

        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")
    
        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']

        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
        
        # 使用预定义的broad_time_periods
        time_ranges = {
            "Early dawn / Before dawn": "03:00-06:00",
            "Early morning": "05:00-08:00",
            "Morning": "07:00-10:00",
            "Late morning / Forenoon": "08:00-12:00",
            "Noon / Midday": "11:00-14:00",
            "Early afternoon": "12:00-16:00",
            "Afternoon": "13:00-18:00",
            "Evening (early) / Dusk": "17:00-20:00",
            "Evening / Night (early)": "19:00-22:00",
            "Night": "21:00-01:00",
            "Late night / Midnight": "23:00-04:00",
        }
        
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
        
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            
            if None in (start_min, end_min):
                continue
            
            ids = []
            for transport in transports:
                dep_time = self.parse_time_to_minutes(str(transport.get("Departure Time")))
                if dep_time is None:
                    continue
                
                # 处理跨越00:00的时间段
                # 例如：Night "21:00-01:00"，表示21:00到次日01:00
                if start_min > end_min:
                    # 跨越午夜的情况：时间在start_min到24:00 或 00:00到end_min
                    if dep_time >= start_min or dep_time < end_min:
                        if transport['type'] == 'flight':
                            ids.append(transport.get("Flight_id"))
                        else:
                            ids.append(transport.get("Train_id"))
                else:
                    # 普通时间段：时间在start_min到end_min之间
                    if start_min <= dep_time < end_min:
                        if transport['type'] == 'flight':
                            ids.append(transport.get("Flight_id"))
                        else:
                            ids.append(transport.get("Train_id"))
            
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
        
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        

        selected_label = random.choice(list(valid_periods.keys()))

        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }
       

    @register_func("generate_depart_before_time")
    def generate_depart_before_time(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        时间段范围：从00:00到指定小时（每小时一个间隔）
        """

        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")
    

        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']

        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
    
        # 生成从00:00到每个整点的时间范围映射
        time_ranges = {}
        for hour in range(6, 23):  # 从6:00到22:00，每小时一个间隔
            end_time = f"{hour:02d}:00"
            label = f"before {end_time}"
            period = f"00:00-{end_time}"
            time_ranges[label] = period
    
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
    
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            ids = []
            for transport in transports:
                dep_time = self.parse_time_to_minutes(str(transport.get("Departure Time")))
                if dep_time is not None and start_min <= dep_time < end_min:
                    if transport['type'] == 'flight':
                        ids.append(transport.get("Flight_id"))
                    else:
                        ids.append(transport.get("Train_id"))
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
    
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
    
        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }


    @register_func("generate_depart_after_time")
    def generate_depart_after_time(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        时间段范围：从指定小时到23:59（每小时一个间隔）
        """
        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")
        
        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']


        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
    
        # 生成从每个整点到23:59的时间范围映射
        time_ranges = {}
        for hour in range(6, 23):  # 从00:00到22:00，每小时一个间隔
            start_time = f"{hour:02d}:00"
            label = f"after {start_time}"
            period = f"{start_time}-23:59"
            time_ranges[label] = period
    
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
    
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            ids = []
            for transport in transports:
                dep_time = self.parse_time_to_minutes(str(transport.get("Departure Time")))
                if dep_time is not None and start_min <= dep_time <= end_min:
                    if transport['type'] == 'flight':
                        ids.append(transport.get("Flight_id"))
                    else:
                        ids.append(transport.get("Train_id"))
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
    
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
    

        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }



    # ---------------------------------
    # Arrival Time相关函数
    # ---------------------------------
    
    @register_func("generate_arrive_within_specific_range")
    def generate_arrive_within_specific_range(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        """

        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")

        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']

        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
    
        # 时间段映射（每半小时）
        time_ranges = self.generate_hour_ranges()
    
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
    
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            ids = []
            for transport in transports:
                arr_time = self.parse_time_to_minutes(str(transport.get("Arrival Time")))
                if arr_time is not None and start_min <= arr_time < end_min:
                    if transport['type'] == 'flight':
                        ids.append(transport.get("Flight_id"))
                    else:
                        ids.append(transport.get("Train_id"))
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
    
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
    
 
        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }



    @register_func("generate_arrive_within_general_period")
    def generate_arrive_within_general_period(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        特别处理：跨越00:00的时间段
        """

        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")

        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']


        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
        
        # 使用预定义的broad_time_periods
        time_ranges = {
            "Early dawn / Before dawn": "03:00-06:00",
            "Early morning": "05:00-08:00",
            "Morning": "07:00-10:00",
            "Late morning / Forenoon": "08:00-12:00",
            "Noon / Midday": "11:00-14:00",
            "Early afternoon": "12:00-16:00",
            "Afternoon": "13:00-18:00",
            "Evening (early) / Dusk": "17:00-20:00",
            "Evening / Night (early)": "19:00-22:00",
            "Night": "21:00-01:00",
            "Late night / Midnight": "23:00-04:00",
        }
        
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
        
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            
            if None in (start_min, end_min):
                continue
            
            ids = []
            for transport in transports:
                arr_time = self.parse_time_to_minutes(str(transport.get("Arrival Time")))
                if arr_time is None:
                    continue
                
                # 处理跨越00:00的时间段
                # 例如：Night "21:00-01:00"，表示21:00到次日01:00
                if start_min > end_min:
                    # 跨越午夜的情况：时间在start_min到24:00 或 00:00到end_min
                    if arr_time >= start_min or arr_time < end_min:
                        if transport['type'] == 'flight':
                            ids.append(transport.get("Flight_id"))
                        else:
                            ids.append(transport.get("Train_id"))
                else:
                    # 普通时间段：时间在start_min到end_min之间
                    if start_min <= arr_time < end_min:
                        if transport['type'] == 'flight':
                            ids.append(transport.get("Flight_id"))
                        else:
                            ids.append(transport.get("Train_id"))
            
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
        
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        

        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }
    


    @register_func("generate_arrive_before_time")
    def generate_arrive_before_time(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        时间段范围：从00:00到指定小时（每小时一个间隔）
        """

        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")
        
        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']

        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
    
        # 生成从00:00到每个整点的时间范围映射
        time_ranges = {}
        for hour in range(6, 23):  # 从6:00到22:00，每小时一个间隔
            end_time = f"{hour:02d}:00"
            label = f"before {end_time}"
            period = f"00:00-{end_time}"
            time_ranges[label] = period
    
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
    
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            ids = []
            for transport in transports:
                arr_time = self.parse_time_to_minutes(str(transport.get("Arrival Time")))
                if arr_time is not None and start_min <= arr_time < end_min:
                    if transport['type'] == 'flight':
                        ids.append(transport.get("Flight_id"))
                    else:
                        ids.append(transport.get("Train_id"))
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
    
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
    

        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }



    @register_func("generate_arrive_after_time")
    def generate_arrive_after_time(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: routes_info, transport_type='all', generate=False
        输出: 随机选中的时间段 + 候选航班/火车 ID
        当generate=True时，返回所有的label和对应的range
        时间段范围：从指定小时到23:59（每小时一个间隔）
        """


        # --- 统一处理 routes_info ---
        # 如果是 list，则取第一项（你的使用方式看起来只会处理单条）
        if isinstance(routes_info, list):
            if len(routes_info) == 0:
                raise ValueError("routes_info 列表为空")
            routes_info = routes_info[0]

        # 必须确保是 dict
        if not isinstance(routes_info, dict):
            raise TypeError("routes_info 必须是 dict 或 list")
        

        departure_city = routes_info['departure_city']
        arrival_city = routes_info['arrival_city']
        date_str = routes_info['date_str']


        week_abbr = self._weekday_abbr(date_str)
        available_key = f"{week_abbr}_available"
    
        # 生成从每个整点到23:59的时间范围映射
        time_ranges = {}
        for hour in range(6, 23):  # 从6:00到22:00，每小时一个间隔
            start_time = f"{hour:02d}:00"
            label = f"after {start_time}"
            period = f"{start_time}-23:59"
            time_ranges[label] = period
    
        # 筛选交通方式（城市 + 可用性）
        transports = []
        if transport_type in ['flight', 'all']:
            flights = [
                f for f in self.flights
                if f.get("Departure City") == departure_city
                and f.get("Arrival City") == arrival_city
                and (available_key not in f or f.get(available_key) == 1)
            ]
            # 为航班添加类型标识
            for flight in flights:
                flight['type'] = 'flight'
            transports.extend(flights)
            
        if transport_type in ['train', 'all']:
            trains = [
                t for t in self.trains
                if t.get("Departure City") == departure_city
                and t.get("Arrival City") == arrival_city
            ]
            # 为火车添加类型标识
            for train in trains:
                train['type'] = 'train'
            transports.extend(trains)
    
        valid_periods = {}
        for label, period in time_ranges.items():
            if generate_params and label not in generate_params:
                continue
            start_str, end_str = period.split("-")
            start_min = self.parse_time_to_minutes(start_str)
            end_min = self.parse_time_to_minutes(end_str)
            ids = []
            for transport in transports:
                arr_time = self.parse_time_to_minutes(str(transport.get("Arrival Time")))
                if arr_time is not None and start_min <= arr_time <= end_min:
                    if transport['type'] == 'flight':
                        ids.append(transport.get("Flight_id"))
                    else:
                        ids.append(transport.get("Train_id"))
            if ids:
                valid_periods[label] = {"range": period, "transports": ids}
    
        if not valid_periods:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
    

        selected_label = random.choice(list(valid_periods.keys()))
        selected_info = valid_periods[selected_label]
        
        # 获取所有候选交通方式的product_ids
        candidate_product_ids = []
        for transport_id in selected_info["transports"]:
            # 检查是航班还是火车
            found = False
            if transport_id.startswith('Flight_'):
                for transport in self.flights:
                    if transport.get("Flight_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
            else:
                for transport in self.trains:
                    if transport.get("Train_id") == transport_id:
                        # 提取product_ids
                        products = transport.get("Product", [])
                        for product in products:
                            if "product_id" in product:
                                candidate_product_ids.append(product["product_id"])
                        found = True
                        break
        
        # 构建所有label和range的映射
        all_labels_and_ranges = {label: info["range"] for label, info in valid_periods.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_info["range"],
            "candidate_ids": selected_info["transports"],
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }
        


    # ---------------------------------
    # 价格相关函数
    # ---------------------------------

    @register_func("generate_transport_cost_less_per_person_one_way")
    def generate_transport_cost_less_per_person_one_way(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate: 是否返回所有可能的价格范围，默认为False
        输出: 随机选中的价格上限 + 符合条件的航班/火车ID和产品ID
        当generate=True时，返回所有的label和对应的price
        """
        # 为每条路线生成价格范围
        all_valid_prices = {}  
        route_results = []
        
        # 首先遍历所有路线和交通方式，找到最高价格+100作为上限
        max_price_all = 0
        # 新增：存储每条路线的最小价格
        route_min_prices = []
        
        for route in routes_info:
            departure_city = route.get("departure_city")
            arrival_city = route.get("arrival_city")
            date_str = route.get("date_str")
            
            if not (departure_city and arrival_city and date_str):
                continue
            
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 获取所有交通方式
            transports = []
            if transport_type in ['flight', 'all']:
                flights = [
                    f for f in self.flights
                    if f.get("Departure City") == departure_city
                    and f.get("Arrival City") == arrival_city
                    and (available_key not in f or f.get(available_key) == 1)
                ]
                for flight in flights:
                    flight['type'] = 'flight'
                transports.extend(flights)
            
            if transport_type in ['train', 'all']:
                trains = [
                    t for t in self.trains
                    if t.get("Departure City") == departure_city
                    and t.get("Arrival City") == arrival_city
                ]
                for train in trains:
                    train['type'] = 'train'
                transports.extend(trains)
            
            # 查找当前路线的最高价格和最低价格
            route_min_price = float('inf')  # 初始化当前路线的最低价格为无穷大
            for transport in transports:
                products = transport.get("Product", [])
                
                # 根据交通类型筛选产品
                if transport['type'] == 'train':
                    filtered_products = [
                        p for p in products 
                        if p.get("seat_type") in ["Second class", "First class"]
                    ]
                else:
                    filtered_products = [
                        p for p in products 
                        if p.get("seat_type") == "Economy class"
                    ]
                
                # 查找最高价格和最低价格
                for product in filtered_products:
                    try:
                        day_price_key = week_abbr
                        price = product.get(day_price_key) or product.get("price") or product.get("Price")
                        if price and isinstance(price, (int, float)):
                            if price > max_price_all:
                                max_price_all = price
                            if price < route_min_price:
                                route_min_price = price
                    except (ValueError, TypeError):
                        continue
            
            # 如果当前路线有有效价格，记录其最低价格
            if route_min_price != float('inf'):
                route_min_prices.append(route_min_price)
        
        # 计算上限价格 (最高价格+100)，并转换为整数
        upper_limit = min(int(max_price_all + 100), 2000)  # 确保不超过2000并转换为整数
        
        # 计算下限价格，即所有路线最低价格中的最大值
        if route_min_prices:
            lower_limit = math.ceil(max(route_min_prices) / 100) * 100
        else:
            lower_limit = 100  # 如果没有找到任何有效价格，设置默认下限为100
        # print(f"最大价格: {max_price_all}")
        # print(f"所有路线最低价格: {route_min_prices}")
        # print(f"上限价格: {upper_limit}")
        # print(f"下限价格: {lower_limit}")
        # 价格范围从下限到上限每隔100生成一个
        price_ranges = {}
        for price in range(lower_limit, upper_limit + 100, 100):
            if price <= 2000:  # 确保不超过2000
                price_ranges[f"{price}"] = price

        # 处理每条路线
        for route in routes_info:
            departure_city = route.get("departure_city")
            arrival_city = route.get("arrival_city")
            date_str = route.get("date_str")
            
            if not (departure_city and arrival_city and date_str):
                continue
            
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 筛选交通方式（城市 + 可用性）
            transports = []
            if transport_type in ['flight', 'all']:
                flights = [
                    f for f in self.flights
                    if f.get("Departure City") == departure_city
                    and f.get("Arrival City") == arrival_city
                    and (available_key not in f or f.get(available_key) == 1)
                ]
                # 为航班添加类型标识
                for flight in flights:
                    flight['type'] = 'flight'
                transports.extend(flights)
            
            if transport_type in ['train', 'all']:
                trains = [
                    t for t in self.trains
                    if t.get("Departure City") == departure_city
                    and t.get("Arrival City") == arrival_city
                ]
                # 为火车添加类型标识
                for train in trains:
                    train['type'] = 'train'
                transports.extend(trains)
            
            # 为这条路线找出所有有效的价格区间
            valid_prices_for_route = {}

            
            # 按价格从低到高检查每个区间
            for label, max_price in sorted(price_ranges.items(), key=lambda x: x[1]):
                if generate_params and label not in generate_params:
                    continue
                ids = []
                product_ids = []
                
                for transport in transports:
                    # 获取该交通方式的ID
                    transport_id = transport.get("Flight_id") if transport['type'] == 'flight' else transport.get("Train_id")
                    
                    # 获取产品
                    products = transport.get("Product", [])
                    
                    # 筛选符合条件的产品
                    filtered_products = []
                    if transport['type'] == 'train':
                        # 火车默认只找Second class和First class
                        filtered_products = [
                            p for p in products 
                            if p.get("seat_type") in ["Second class", "First class"]
                        ]
                    else:
                        filtered_products = [
                            p for p in products 
                            if p.get("seat_type") == "Economy class"
                        ]
                    
                    for product in filtered_products:
                        # 获取当天的价格
                        try:
                            # 尝试获取基于星期几的价格
                            day_price_key = week_abbr  # 注意这里直接使用星期几缩写作为键
                            price = product.get(day_price_key) or product.get("price") or product.get("Price")
                            
                            # 检查价格是否低于当前区间的上限
                            if price < max_price:
                                if transport_id not in ids:
                                    ids.append(transport_id)
                                product_id = product.get("product_id")
                                if product_id and product_id not in product_ids:
                                    product_ids.append(product_id)
                        except (ValueError, TypeError):
                            continue
                
                if ids:
                    valid_prices_for_route[label] = {
                        "price": max_price,
                        "transports": ids,
                        "products": product_ids
                    }
            
            if valid_prices_for_route:
                all_valid_prices.update(valid_prices_for_route)
                route_results.append({
                    "route": route,
                    "valid_prices": valid_prices_for_route
                })
            else:
                # 如果这条路线没有符合条件的价格区间，使用默认值
                route_results.append({
                    "route": route,
                    "valid_prices": {}
                })
        # print("\n=== 测试结果: all_valid_prices ===")
        # print(all_valid_prices)
        # print("\n=== 测试结果: route_results ===")
        # print(route_results)
        # 如果没有任何路线有符合条件的价格区间
        if not all_valid_prices:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        
        # 选择一个价格区间作为最终结果
        # 为了简单起见，我们为所有路线选择相同的价格区间
        selected_label = random.choice(list(all_valid_prices.keys()))
        selected_max_price = all_valid_prices[selected_label]["price"]
        
        # 收集所有路线中符合所选价格区间的交通方式和产品
        selected_candidate_ids = []
        selected_candidate_product_ids = []
        
        for route_result in route_results:
            valid_prices = route_result["valid_prices"]
            # 找出该路线中不大于所选价格的最高价格区间
            best_price_label = None
            best_price_value = -1
            
            for label, info in valid_prices.items():
                price = info["price"]
                if price <= selected_max_price and price > best_price_value:
                    best_price_value = price
                    best_price_label = label
            
            if best_price_label:
                selected_candidate_ids.extend(valid_prices[best_price_label]["transports"])
                selected_candidate_product_ids.extend(valid_prices[best_price_label]["products"])
        
        # 去重
        selected_candidate_ids = list(set(selected_candidate_ids))
        selected_candidate_product_ids = list(set(selected_candidate_product_ids))
        
     
        # 构建所有label和price的映射
        all_labels_and_ranges = {label: info["price"] for label, info in all_valid_prices.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_max_price,
            "candidate_ids": selected_candidate_ids,
            "candidate_product_ids": selected_candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges,
            # "route_results": route_results
        }
      

    @register_func("generate_transport_cost_less_per_person_round_trip")
    def generate_transport_cost_less_per_person_round_trip(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate: 是否返回所有可能的价格范围，默认为False
        输出: 随机选中的往返价格上限(单程价格的2倍) + 符合条件的航班/火车ID和产品ID
        当generate=True时，返回所有的label和对应的price
        """

        # 人数倍数
        multiplier = 2

        # ---------- 1️⃣ generate_params 先除回去 ----------
        processed_generate_params = None
        if generate_params:
            processed_generate_params = {}
            for label, price in generate_params.items():
                # 提取数字，如 "3000" -> 3000
                num = int(label)
                raw_num = num // multiplier
                raw_price = price // multiplier
                processed_generate_params[f"{raw_num}"] = raw_price

        # ---------- 2️⃣ 调用 one-way 函数（使用除回去的参数） ----------
        one_way_result = self.generate_transport_cost_less_per_person_one_way(
            routes_info, 
            transport_type, 
            processed_generate_params
        )

        # 检查结果是否有效且包含validation_params
        if one_way_result and one_way_result.get("validation_params"):
            # 单程价格
            one_way_price = one_way_result["validation_params"]
            round_trip_price = one_way_price * 2
            one_way_result["validation_params"] = round_trip_price

            # ---- 处理 selected_description：把  XXX 的数字翻倍 ----
            if "selected_description" in one_way_result:
                desc = one_way_result["selected_description"]
                num = int(desc)
                new_num = num * 2
                one_way_result["selected_description"] = f"{new_num}"

            # ---- 更新 all_labels_and_ranges：key 和 value 都需翻倍 ----
            if one_way_result.get("all_labels_and_ranges"):
                new_dict = {}
                for label, price in one_way_result["all_labels_and_ranges"].items():
                    num = int(label)
                    new_label = f"{num * 2}"
                    new_dict[new_label] = price * 2

                one_way_result["all_labels_and_ranges"] = new_dict

        return one_way_result


    @register_func("generate_transport_cost_less_total")
    def generate_transport_cost_less_total(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate: 是否返回所有可能的价格范围，默认为False
        输出: 随机选中的往返价格上限(单程价格的2倍) + 符合条件的航班/火车ID和产品ID
        当generate=True时，返回所有的label和对应的price
        """

        # 人数倍数
        multiplier = 2 * routes_info[0]["number_of_people"]

        # ---------- 1️⃣ generate_params 先除回去 ----------
        processed_generate_params = None
        if generate_params:
            processed_generate_params = {}
            for label, price in generate_params.items():
                # 提取数字，如 "3000" -> 3000
                num = int(label)
                raw_num = num // multiplier
                raw_price = price // multiplier
                processed_generate_params[f"{raw_num}"] = raw_price

        # ---------- 2️⃣ 调用 one-way 函数（使用除回去的参数） ----------
        one_way_result = self.generate_transport_cost_less_per_person_one_way(
            routes_info, 
            transport_type, 
            processed_generate_params
        )
        
        # 检查结果是否有效且包含validation_params
        if one_way_result and one_way_result.get("validation_params"):
            # 人数倍数
            multiplier = 2 * routes_info[0]["number_of_people"]

            # ------ 1. 更新 validation_params ------
            one_way_price = one_way_result["validation_params"]
            one_way_result["validation_params"] = one_way_price * multiplier

            # ------ 2. 更新 all_labels_and_ranges ------
            if one_way_result.get("all_labels_and_ranges"):
                new_dict = {}
                for label, price in one_way_result["all_labels_and_ranges"].items():
                    # 提取 label 中的数字，例如 "400" -> 400
                    num = int(label)
                    new_num = num * multiplier
                    new_label = f"{new_num}"
                    new_dict[new_label] = price * multiplier

                one_way_result["all_labels_and_ranges"] = new_dict

            # ------ 3. 更新 selected_description ------
            selected_desc = one_way_result.get("selected_description")
            num = int(selected_desc)
            new_num = num * multiplier
            one_way_result["selected_description"] = f"{new_num}"

        return one_way_result



    # ---------------------------------
    # 平台相关函数
    # ---------------------------------

    @register_func("generate_platform_specify")
    def generate_platform_specify(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate_params: 生成参数，可选
        输出: 随机选中的平台组合 + 符合条件的航班/火车ID和产品ID
        """
        # 定义所有可用的平台
        all_platforms = ["ctrip", "alitrip", "qunar", "direct"]
        
        # 为每条路线生成有效的平台组合
        all_valid_platforms = {}
        route_results = []
        
        # 处理每条路线
        for route in routes_info:
            departure_city = route.get("departure_city")
            arrival_city = route.get("arrival_city")
            date_str = route.get("date_str")
            
            if not (departure_city and arrival_city and date_str):
                continue
            
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 筛选交通方式（城市 + 可用性）
            transports = []
            if transport_type in ['flight', 'all']:
                flights = [
                    f for f in self.flights
                    if f.get("Departure City") == departure_city
                    and f.get("Arrival City") == arrival_city
                    and (available_key not in f or f.get(available_key) == 1)
                ]
                for flight in flights:
                    flight['type'] = 'flight'
                transports.extend(flights)
            
            if transport_type in ['train', 'all']:
                trains = [
                    t for t in self.trains
                    if t.get("Departure City") == departure_city
                    and t.get("Arrival City") == arrival_city
                ]
                for train in trains:
                    train['type'] = 'train'
                transports.extend(trains)
            # print(transports)
            # 为这条路线找出所有有效的平台组合
            valid_platforms_for_route = {}
            
            # 生成1-2个平台的组合
            platform_combinations = []
            # 单个平台的组合
            for platform in all_platforms:
                platform_combinations.append([platform])
            # 两个不同平台的组合
            for i in range(len(all_platforms)):
                for j in range(i + 1, len(all_platforms)):
                    platform_combinations.append([all_platforms[i], all_platforms[j]])
            # print(platform_combinations)
            # 检查每个平台组合是否有效
            for platform_combination in platform_combinations:
                # 生成组合的标签（用逗号连接）
                platform_label = ", ".join(platform_combination)
                
                if generate_params and platform_label not in generate_params:
                    continue
                
                ids = []
                product_ids = []
                
                for transport in transports:
                    # 获取该交通方式的ID
                    transport_id = transport.get("Flight_id") if transport['type'] == 'flight' else transport.get("Train_id")
                    
                    # 获取产品
                    products = transport.get("Product", [])
                    
                    # 筛选符合平台条件的产品
                    for product in products:
                        # 检查产品是否在指定平台上
                        product_platform = product.get("source_platform")
                        if product_platform and product_platform in platform_combination:
                            if transport_id not in ids:
                                ids.append(transport_id)
                            product_id = product.get("product_id")
                            if product_id and product_id not in product_ids:
                                product_ids.append(product_id)
                
                if ids:
                    valid_platforms_for_route[platform_label] = {
                        "platforms": platform_combination,
                        "transports": ids,
                        "products": product_ids
                    }
            
            if valid_platforms_for_route:
                # 更新所有有效的平台组合
                for platform_label, info in valid_platforms_for_route.items():
                    if platform_label not in all_valid_platforms:
                        all_valid_platforms[platform_label] = info
                route_results.append({
                    "route": route,
                    "valid_platforms": valid_platforms_for_route
                })
            else:
                # 如果这条路线没有符合条件的平台组合，使用默认值
                route_results.append({
                    "route": route,
                    "valid_platforms": {}
                })
        
        # 如果没有任何路线有符合条件的平台组合
        if not all_valid_platforms:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        
        # 选择一个平台组合作为最终结果
        selected_label = random.choice(list(all_valid_platforms.keys()))
        selected_platforms = all_valid_platforms[selected_label]["platforms"]
        
        # 收集所有路线中符合所选平台组合的交通方式和产品
        selected_candidate_ids = []
        selected_candidate_product_ids = []
        
        for route_result in route_results:
            valid_platforms = route_result["valid_platforms"]
            # 检查该路线是否有使用所选平台组合的交通方式
            if selected_label in valid_platforms:
                selected_candidate_ids.extend(valid_platforms[selected_label]["transports"])
                selected_candidate_product_ids.extend(valid_platforms[selected_label]["products"])
        
        # 去重
        selected_candidate_ids = list(set(selected_candidate_ids))
        selected_candidate_product_ids = list(set(selected_candidate_product_ids))
        
        # 构建所有label和platforms的映射
        all_labels_and_ranges = {label: info["platforms"] for label, info in all_valid_platforms.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_platforms,
            "candidate_ids": selected_candidate_ids,
            "candidate_product_ids": selected_candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }


    @register_func("generate_platform_exclude")
    def generate_platform_exclude(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数:
        routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
        transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
        generate_params: 生成参数，可选，包含要排除的平台列表

        输出:
        随机选中的不包含排除平台的平台组合 + 符合条件的航班/火车ID和产品ID
        """

        # 定义所有可用的平台
        all_platforms = ["ctrip", "alitrip", "qunar", "direct"]

        # 获取要排除的平台列表（新版逻辑）
        if generate_params:
            # 如果有传入，直接使用
            excluded_platforms_list = generate_params
        else:
            # 否则列举所有可能的 1-2 平台组合（不再随机）
            excluded_platforms_list = []
            for r in [1, 2]:
                excluded_platforms_list.extend([list(c) for c in combinations(all_platforms, r)])

        # 所有有效的平台组合（这里指针对每个排除项）
        all_valid_platforms = {}
        route_results = []

        # 对每一种排除情况执行一次完整流程
        for excluded_platforms in excluded_platforms_list:

            # 计算剩余可用平台
            available_platforms = [p for p in all_platforms if p not in excluded_platforms]
        

            # 用于存储每个排除项的总体结果
            tmp_all_transports = []
            tmp_all_products = []

            # 为每条路线生成有效的平台组合（新版：不再生成 1-2 平台组合，直接使用剩余平台）
            for route in routes_info:
                departure_city = route.get("departure_city")
                arrival_city = route.get("arrival_city")
                date_str = route.get("date_str")
                if not (departure_city and arrival_city and date_str):
                    continue

                week_abbr = self._weekday_abbr(date_str)
                available_key = f"{week_abbr}_available"

                transports = []

                # 筛选航班
                if transport_type in ['flight', 'all']:
                    flights = [
                        f for f in self.flights
                        if f.get("Departure City") == departure_city
                        and f.get("Arrival City") == arrival_city
                        and (available_key not in f or f.get(available_key) == 1)
                    ]
                    for flight in flights:
                        flight['type'] = 'flight'
                    transports.extend(flights)

                # 筛选火车
                if transport_type in ['train', 'all']:
                    trains = [
                        t for t in self.trains
                        if t.get("Departure City") == departure_city
                        and t.get("Arrival City") == arrival_city
                    ]
                    for train in trains:
                        train['type'] = 'train'
                    transports.extend(trains)

                # 筛选产品（只要求产品的平台在剩余平台列表内）
                for transport in transports:
                    transport_id = transport.get("Flight_id") if transport['type'] == 'flight' else transport.get("Train_id")
                    products = transport.get("Product", [])

                    for product in products:
                        product_platform = product.get("source_platform")

                        if product_platform and product_platform in available_platforms:
                            tmp_all_transports.append(transport_id)

                            product_id = product.get("product_id")
                            if product_id:
                                tmp_all_products.append(product_id)

            # label 用排除项来命名
            label = f"{', '.join(excluded_platforms)}"

            # 保存该排除项的结果
            all_valid_platforms[label] = {
                "platforms": available_platforms,
                "transports": list(set(tmp_all_transports)),
                "products": list(set(tmp_all_products))
            }

        # 如果没有任何有效结果
        if not all_valid_platforms:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result

        # 从所有排除组合结果中随机选择一个
        selected_label = random.choice(list(all_valid_platforms.keys()))
        selected_platforms = all_valid_platforms[selected_label]["platforms"]

        selected_candidate_ids = all_valid_platforms[selected_label]["transports"]
        selected_candidate_product_ids = all_valid_platforms[selected_label]["products"]

        # 构建 all_labels_and_ranges（与原版本保持一致）
        all_labels_and_ranges = {
            label: info["platforms"] for label, info in all_valid_platforms.items()
        }

        return {
            "selected_description": selected_label,
            "validation_params": selected_platforms,
            "candidate_ids": selected_candidate_ids,
            "candidate_product_ids": selected_candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }


    # ---------------------------------
    # 准点率相关函数
    # ---------------------------------
    
    @register_func("generate_minimum_ontime_rate")
    def generate_minimum_ontime_rate(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate_params: 生成参数，可选
        输出: 随机选中的准点率 + 符合条件的航班/火车ID和产品ID
        """
        # 定义所有可能的准点率值（0.8到1.0，间隔0.05）
        all_ontime_rates = [round(0.8 + i * 0.05, 2) for i in range(5)]
        
        # 为每条路线生成有效的准点率组合
        all_valid_ontime_rates = {}
        route_results = []
        
        # 处理每条路线
        for route in routes_info:
            departure_city = route.get("departure_city")
            arrival_city = route.get("arrival_city")
            date_str = route.get("date_str")
            
            if not (departure_city and arrival_city and date_str):
                continue
            
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 筛选交通方式（城市 + 可用性）
            transports = []
            if transport_type in ['flight', 'all']:
                flights = [
                    f for f in self.flights
                    if f.get("Departure City") == departure_city
                    and f.get("Arrival City") == arrival_city
                    and (available_key not in f or f.get(available_key) == 1)
                ]
                for flight in flights:
                    flight['type'] = 'flight'
                transports.extend(flights)
            
            if transport_type in ['train', 'all']:
                trains = [
                    t for t in self.trains
                    if t.get("Departure City") == departure_city
                    and t.get("Arrival City") == arrival_city
                ]
                for train in trains:
                    train['type'] = 'train'
                transports.extend(trains)
            
            # 为这条路线找出所有有效的准点率
            valid_ontime_rates_for_route = {}
            
            # 检查每个准点率是否有效
            for ontime_rate in all_ontime_rates:
                # 生成准点率的标签（百分比形式）
                ontime_rate_label = f"{int(ontime_rate * 100)}"
                # print(ontime_rate_label)
                
                if generate_params and ontime_rate_label not in generate_params:
                    continue
                
                ids = []
                product_ids = []
                
                for transport in transports:
                    # 对于火车，默认全选
                    if transport['type'] == 'train':
                        transport_id = transport.get("Train_id")
                        if transport_id not in ids:
                            ids.append(transport_id)
                        # 获取产品
                        products = transport.get("Product", [])
                        for product in products:
                            product_id = product.get("product_id")
                            if product_id and product_id not in product_ids:
                                product_ids.append(product_id)
                    # 对于飞机，根据准点率筛选
                    elif transport['type'] == 'flight':
                        flight_ontime_perf = transport.get("On-Time Performance", 0)
                        if flight_ontime_perf >= ontime_rate:
                            transport_id = transport.get("Flight_id")
                            if transport_id not in ids:
                                ids.append(transport_id)
                            # 获取产品
                            products = transport.get("Product", [])
                            for product in products:
                                product_id = product.get("product_id")
                                if product_id and product_id not in product_ids:
                                    product_ids.append(product_id)
                
                if ids:
                    valid_ontime_rates_for_route[ontime_rate_label] = {
                        "ontime_rate": ontime_rate,
                        "transports": ids,
                        "products": product_ids
                    }
            
            if valid_ontime_rates_for_route:
                # 更新所有有效的准点率
                for ontime_rate_label, info in valid_ontime_rates_for_route.items():
                    if ontime_rate_label not in all_valid_ontime_rates:
                        all_valid_ontime_rates[ontime_rate_label] = info
                route_results.append({
                    "route": route,
                    "valid_ontime_rates": valid_ontime_rates_for_route
                })
            else:
                # 如果这条路线没有符合条件的准点率，使用默认值
                route_results.append({
                    "route": route,
                    "valid_ontime_rates": {}
                })
        
        # 如果没有任何路线有符合条件的准点率
        if not all_valid_ontime_rates:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        
        # 选择一个准点率作为最终结果
        selected_label = random.choice(list(all_valid_ontime_rates.keys()))
        selected_ontime_rate = all_valid_ontime_rates[selected_label]["ontime_rate"]
        
        # 收集所有路线中符合所选准点率的交通方式和产品
        selected_candidate_ids = []
        selected_candidate_product_ids = []
        
        for route_result in route_results:
            valid_ontime_rates = route_result["valid_ontime_rates"]
            # 检查该路线是否有使用所选准点率的交通方式
            if selected_label in valid_ontime_rates:
                selected_candidate_ids.extend(valid_ontime_rates[selected_label]["transports"])
                selected_candidate_product_ids.extend(valid_ontime_rates[selected_label]["products"])
        
        # 去重
        selected_candidate_ids = list(set(selected_candidate_ids))
        selected_candidate_product_ids = list(set(selected_candidate_product_ids))
        
        # 构建所有label和ontime_rate的映射
        all_labels_and_ranges = {label: info["ontime_rate"] for label, info in all_valid_ontime_rates.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": [selected_ontime_rate],
            "candidate_ids": selected_candidate_ids,
            "candidate_product_ids": selected_candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges
        }


    @register_func("generate_max_delay")
    def generate_max_delay(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate_params: 生成参数，可选
        输出: 随机选中的最大延误时间 + 符合条件的航班/火车ID和产品ID
        """
        # 定义所有可能的最大延误时间值（0到30，间隔5分钟）
        all_max_delays = [i * 5 for i in range(7)]  # [0, 5, 10, 15, 20, 25, 30]
        
        # 为每条路线生成有效的最大延误时间组合
        all_valid_max_delays = {}
        route_results = []
        
        # 处理每条路线
        for route in routes_info:
            departure_city = route.get("departure_city")
            arrival_city = route.get("arrival_city")
            date_str = route.get("date_str")
            
            if not (departure_city and arrival_city and date_str):
                continue
            
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 筛选交通方式（城市 + 可用性）
            transports = []
            if transport_type in ['flight', 'all']:
                flights = [
                    f for f in self.flights
                    if f.get("Departure City") == departure_city
                    and f.get("Arrival City") == arrival_city
                    and (available_key not in f or f.get(available_key) == 1)
                ]
                for flight in flights:
                    flight['type'] = 'flight'
                transports.extend(flights)
            
            if transport_type in ['train', 'all']:
                trains = [
                    t for t in self.trains
                    if t.get("Departure City") == departure_city
                    and t.get("Arrival City") == arrival_city
                ]
                for train in trains:
                    train['type'] = 'train'
                transports.extend(trains)
            
            # 为这条路线找出所有有效的最大延误时间
            valid_max_delays_for_route = {}
            
            # 检查每个最大延误时间是否有效
            for max_delay in all_max_delays:
                # 生成最大延误时间的标签（分钟形式）
                max_delay_label = f"{max_delay}"
                
                if generate_params and max_delay_label not in generate_params:
                    continue
                
                ids = []
                product_ids = []
                
                for transport in transports:
                    # 对于火车，默认全选
                    if transport['type'] == 'train':
                        transport_id = transport.get("Train_id")
                        if transport_id not in ids:
                            ids.append(transport_id)
                        # 获取产品
                        products = transport.get("Product", [])
                        for product in products:
                            product_id = product.get("product_id")
                            if product_id and product_id not in product_ids:
                                product_ids.append(product_id)
                    # 对于飞机，根据最大延误时间筛选
                    elif transport['type'] == 'flight':
                        flight_avg_delay = transport.get("Average Delay (minutes)", 0)
                        if flight_avg_delay <= max_delay:
                            transport_id = transport.get("Flight_id")
                            if transport_id not in ids:
                                ids.append(transport_id)
                            # 获取产品
                            products = transport.get("Product", [])
                            for product in products:
                                product_id = product.get("product_id")
                                if product_id and product_id not in product_ids:
                                    product_ids.append(product_id)
                
                if ids:
                    valid_max_delays_for_route[max_delay_label] = {
                        "max_delay": max_delay,
                        "transports": ids,
                        "products": product_ids
                    }
            
            if valid_max_delays_for_route:
                route_results.append({
                    "route": route,
                    "valid_max_delays": valid_max_delays_for_route
                })
            else:
                # 如果这条路线没有符合条件的最大延误时间，使用默认值
                route_results.append({
                    "route": route,
                    "valid_max_delays": {}
                })
        
        # 合并所有路线的结果
        # 首先为每个延误时间级别收集所有路线的交通方式和产品
        delays_consolidated = {}
        for max_delay_label in [str(d) for d in all_max_delays]:
            all_ids = set()
            all_product_ids = set()
            
            for route_result in route_results:
                valid_delays = route_result.get("valid_max_delays", {})
                if max_delay_label in valid_delays:
                    all_ids.update(valid_delays[max_delay_label]["transports"])
                    all_product_ids.update(valid_delays[max_delay_label]["products"])
            
            if all_ids:
                delays_consolidated[max_delay_label] = {
                    "max_delay": int(max_delay_label),
                    "transports": list(all_ids),
                    "products": list(all_product_ids)
                }
        
        # 过滤掉与较小延误时间级别结果相同的较大延误时间级别
        filtered_delays = {}
        prev_transports = set()
        prev_products = set()
        
        # 按照延误时间从小到大检查
        for max_delay in all_max_delays:
            max_delay_label = f"{max_delay}"
            
            if max_delay_label not in delays_consolidated:
                continue
            
            current_info = delays_consolidated[max_delay_label]
            current_transports = set(current_info["transports"])
            current_products = set(current_info["products"])

            
            # 检查当前延误时间级别的结果是否与之前的不同
            # 对于0分钟，直接添加
            # 对于其他分钟数，确保与上一个级别有不同的结果
            if max_delay == 0 or current_transports != prev_transports:
                # print(max_delay_label, current_transports, prev_transports)
                filtered_delays[max_delay_label] = current_info
                prev_transports = current_transports
                prev_products = current_products
        
        # 如果没有任何有效的延误时间级别
        if not filtered_delays:
            result = {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        
        # 选择一个有效的延误时间作为最终结果
        # print(filtered_delays.keys())
        selected_label = random.choice(list(filtered_delays.keys()))
        selected_max_delay = filtered_delays[selected_label]["max_delay"]
        
        # 构建所有label和max_delay的映射
        all_labels_and_ranges = {label: info["max_delay"] for label, info in filtered_delays.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": [selected_max_delay],
            "candidate_ids": filtered_delays[selected_label]["transports"],
            "candidate_product_ids": filtered_delays[selected_label]["products"],
            "all_labels_and_ranges": all_labels_and_ranges
        }


    # ---------------------------------
    # 航司相关函数
    # ---------------------------------
    @register_func("generate_airline_exclude")
    def generate_airline_exclude(self, routes_info, transport_type='all', generate_params=None):
        """
        输入参数: 
            routes_info: 路线信息列表，每个元素是包含departure_city, arrival_city, date_str的字典
            transport_type: 交通类型，可选'flight', 'train', 'all'，默认为'all'
            generate_params: 生成参数，可选
        输出: 随机选中的排除航空公司名称 + 符合条件的航班/火车ID和产品ID
        """
        # 获取所有唯一的航空公司
        all_airlines = set()
        if transport_type in ['flight', 'all']:
            for flight in self.flights:
                airline = flight.get("Airline")
                if airline:
                    all_airlines.add(airline)
        all_airlines = sorted(list(all_airlines))
        
        # 生成所有可能的排除航空公司选项（单个航空公司）
        all_exclude_options = all_airlines  # 只考虑单个航空公司的排除
        
        # 为每条路线生成有效的排除航空公司选项
        all_valid_options = {}
        route_results = []
        
        # 处理每条路线
        for route in routes_info:
            departure_city = route.get("departure_city")
            arrival_city = route.get("arrival_city")
            date_str = route.get("date_str")
            
            if not (departure_city and arrival_city and date_str):
                continue
            
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 筛选交通方式（城市 + 可用性）
            transports = []
            if transport_type in ['flight', 'all']:
                flights = [
                    f for f in self.flights
                    if f.get("Departure City") == departure_city
                    and f.get("Arrival City") == arrival_city
                    and (available_key not in f or f.get(available_key) == 1)
                ]
                for flight in flights:
                    flight['type'] = 'flight'
                transports.extend(flights)
            
            if transport_type in ['train', 'all']:
                trains = [
                    t for t in self.trains
                    if t.get("Departure City") == departure_city
                    and t.get("Arrival City") == arrival_city
                ]
                for train in trains:
                    train['type'] = 'train'
                transports.extend(trains)
            
            # 为这条路线找出所有有效的排除航空公司选项
            valid_options_for_route = {}
            
            # 检查每个排除航空公司是否有效
            for exclude_airline in all_exclude_options:
                # 生成排除航空公司的标签
                exclude_label = exclude_airline
                
                if generate_params and exclude_label not in generate_params:
                    continue
                
                ids = []
                product_ids = []
                
                for transport in transports:
                    # 对于火车，默认全选
                    if transport['type'] == 'train':
                        transport_id = transport.get("Train_id")
                        if transport_id not in ids:
                            ids.append(transport_id)
                        # 获取产品
                        products = transport.get("Product", [])
                        for product in products:
                            product_id = product.get("product_id")
                            if product_id and product_id not in product_ids:
                                product_ids.append(product_id)
                    # 对于飞机，根据排除航空公司筛选
                    elif transport['type'] == 'flight':
                        airline = transport.get("Airline")
                        if airline != exclude_airline:
                            transport_id = transport.get("Flight_id")
                            if transport_id not in ids:
                                ids.append(transport_id)
                            # 获取产品
                            products = transport.get("Product", [])
                            for product in products:
                                product_id = product.get("product_id")
                                if product_id and product_id not in product_ids:
                                    product_ids.append(product_id)
                
                if ids:
                    valid_options_for_route[exclude_label] = {
                        "exclude_airline": exclude_airline,
                        "transports": ids,
                        "products": product_ids
                    }
            
            if valid_options_for_route:
                route_results.append({
                    "route": route,
                    "valid_options": valid_options_for_route
                })
            else:
                # 如果这条路线没有符合条件的选项，使用默认值
                route_results.append({
                    "route": route,
                    "valid_options": {}
                })
        
        # 合并所有路线的结果
        options_consolidated = {}
        for exclude_label in all_exclude_options:
            all_ids = set()
            all_product_ids = set()
            
            for route_result in route_results:
                valid_options = route_result.get("valid_options", {})
                if exclude_label in valid_options:
                    all_ids.update(valid_options[exclude_label]["transports"])
                    all_product_ids.update(valid_options[exclude_label]["products"])
            
            if all_ids:
                options_consolidated[exclude_label] = {
                    "exclude_airline": exclude_label,
                    "transports": list(all_ids),
                    "products": list(all_product_ids)
                }
        
        # 过滤掉结果相同的选项
        filtered_options = {}
        seen_results = set()
        
        for exclude_label, info in options_consolidated.items():
            result_signature = tuple(sorted(info["transports"]))
            if result_signature not in seen_results:
                filtered_options[exclude_label] = info
                seen_results.add(result_signature)
        
        # 如果没有任何有效的选项
        if not filtered_options:
            result = {
                "selected_description": "",
                "validation_params": [""],
                "candidate_ids": [],
                "candidate_product_ids": []
            }
            result["all_labels_and_ranges"] = {}
            return result
        
        # 选择一个有效的选项作为最终结果
        selected_label = random.choice(list(filtered_options.keys()))
        
        # 构建所有label和exclude_airline的映射
        all_labels_and_ranges = {label: info["exclude_airline"] for label, info in filtered_options.items()}
        
        return {
            "selected_description": selected_label,
            "validation_params": [selected_label],
            "candidate_ids": filtered_options[selected_label]["transports"],
            "candidate_product_ids": filtered_options[selected_label]["products"],
            "all_labels_and_ranges": all_labels_and_ranges
        }


    # 校验函数
    # ---------------------------------

    #transportation_info示例
    # [{"id": "Flight_123", "product_id": "P123", "date": "2025-11-10"}]
    #validation_params示例
    # ["08:00-10:00"]

    # @register_func("validate_depart_time_condition")
    # def validate_depart_time_condition(self, transportation_info, validation_params):
    #     """
    #     校验航班/火车出发时间是否在给定时间段内
    #     transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id和date
    #     """
    #     # 处理输入：既可能是单个dict，也可能是dict的列表
    #     transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
    #     # 解析时间范围
    #     if not validation_params:
    #         return False
        
    #     start_time, end_time = validation_params.split("-")
    #     start_min = self.parse_time_to_minutes(start_time)
    #     end_min = self.parse_time_to_minutes(end_time)
        
    #     if None in (start_min, end_min):
    #         return False
        
    #     # 检查每个交通信息
    #     for info in transport_infos:
    #         # 获取交通方式ID、产品ID和日期
    #         transport_id = info.get("id")
    #         product_id = info.get("product_id")
    #         date_str = info.get("date")
            
    #         if not transport_id or not date_str:
    #             return False
            
    #         # 根据ID前缀判断是航班还是火车
    #         dep_time_str = None
    #         if transport_id.startswith('Flight_'):
    #             # 计算星期几缩写
    #             week_abbr = self._weekday_abbr(date_str)
    #             available_key = f"{week_abbr}_available"
                
    #             # 在航班数据中查找
    #             for flight in self.flights:
    #                 if flight.get("Flight_id") == transport_id:
    #                     # 检查该日期航班是否可用
    #                     if available_key in flight and flight.get(available_key) != 1:
    #                         return False  # 航班在该日期不可用
                        
    #                     # 检查产品ID是否匹配（如果提供了）
    #                     if product_id:
    #                         products = flight.get("Product", [])
    #                         product_matched = any(p.get("product_id") == product_id for p in products)
    #                         if not product_matched:
    #                             continue
    #                     dep_time_str = flight.get("Departure Time")
    #                     break
    #         else:
    #             # 在火车数据中查找
    #             for train in self.trains:
    #                 if train.get("Train_id") == transport_id:
    #                     # 检查产品ID是否匹配（如果提供了）
    #                     if product_id:
    #                         products = train.get("Product", [])
    #                         product_matched = any(p.get("product_id") == product_id for p in products)
    #                         if not product_matched:
    #                             continue
    #                     dep_time_str = train.get("Departure Time")
    #                     break
            
    #         if not dep_time_str:
    #             return False
            
    #         dep_min = self.parse_time_to_minutes(dep_time_str)
    #         if dep_min is None:
    #             return False
            
    #         # 如果有任一信息的时间不在范围内，则返回False
    #         # 处理跨天情况：如果start_min > end_min，说明是跨天时间段
    #         if start_min <= end_min:
    #             # 同一天时间段
    #             if not (start_min <= dep_min < end_min):
    #                 return False
    #         else:
    #             # 跨天时间段
    #             if not (dep_min >= start_min or dep_min < end_min):
    #                 return False
        
    #     return True


    # @register_func("validate_arrive_time_condition")
    # def validate_arrive_time_condition(self, transportation_info, validation_params):
    #     """
    #     校验航班/火车到达时间是否在给定时间段内
    #     transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id和date
    #     """
    #     # 处理输入：既可能是单个dict，也可能是dict的列表
    #     transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
    #     # 解析时间范围
    #     if not validation_params:
    #         return False
        
    #     start_time, end_time = validation_params.split("-")
    #     start_min = self.parse_time_to_minutes(start_time)
    #     end_min = self.parse_time_to_minutes(end_time)
        
    #     if None in (start_min, end_min):
    #         return False
        
    #     # 检查每个交通信息
    #     for info in transport_infos:
    #         # 获取交通方式ID、产品ID和日期
    #         transport_id = info.get("id")
    #         product_id = info.get("product_id")
    #         date_str = info.get("date")
            
    #         if not transport_id or not date_str:
    #             return False
            
    #         # 根据ID前缀判断是航班还是火车
    #         arr_time_str = None
    #         if transport_id.startswith('Flight_'):
    #             # 计算星期几缩写
    #             week_abbr = self._weekday_abbr(date_str)
    #             available_key = f"{week_abbr}_available"
                
    #             # 在航班数据中查找
    #             for flight in self.flights:
    #                 if flight.get("Flight_id") == transport_id:
    #                     # 检查该日期航班是否可用
    #                     if available_key in flight and flight.get(available_key) != 1:
    #                         return False  # 航班在该日期不可用
                        
    #                     # 检查产品ID是否匹配（如果提供了）
    #                     if product_id:
    #                         products = flight.get("Product", [])
    #                         product_matched = any(p.get("product_id") == product_id for p in products)
    #                         if not product_matched:
    #                             continue
    #                     arr_time_str = flight.get("Arrival Time")
    #                     break
    #         else:
    #             # 在火车数据中查找
    #             for train in self.trains:
    #                 if train.get("Train_id") == transport_id:
    #                     # 检查产品ID是否匹配（如果提供了）
    #                     if product_id:
    #                         products = train.get("Product", [])
    #                         product_matched = any(p.get("product_id") == product_id for p in products)
    #                         if not product_matched:
    #                             continue
    #                     arr_time_str = train.get("Arrival Time")
    #                     break
            
    #         if not arr_time_str:
    #             return False
            
    #         arr_min = self.parse_time_to_minutes(arr_time_str)
    #         if arr_min is None:
    #             return False
            
    #         # 如果有任一信息的时间不在范围内，则返回False
    #         # 处理跨天情况：如果start_min > end_min，说明是跨天时间段
    #         if start_min <= end_min:
    #             # 同一天时间段
    #             if not (start_min <= arr_min < end_min):
    #                 return False
    #         else:
    #             # 跨天时间段
    #             if not (arr_min >= start_min or arr_min < end_min):
    #                 return False
        
    #     return True


    #     #transportation_info示例


    @register_func("validate_transport_cost_less_per_person_one_way")
    def validate_transport_cost_less_per_person_one_way(self, transportation_info, validation_params):
        """
        校验航班/火车价格是否低于指定的人均单程价格上限
        transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id、date等信息
        validation_params: 包含价格上限的列表，如[400]
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
        # 解析价格上限
        if not validation_params:
            return False
        
        max_price = validation_params
        
        # 检查每个交通信息
        for info in transport_infos:
            # 获取交通方式ID、产品ID和日期
            transport_id = info.get("id")
            product_id = info.get("product_id")
            date_str = info.get("date")
            
            if not transport_id or not date_str:
                return False
            
            # 计算星期几缩写
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 根据ID前缀判断是航班还是火车
            found = False
            price = None
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查该日期航班是否可用
                        if available_key in flight and flight.get(available_key) != 1:
                            return False  # 航班在该日期不可用
                        
                        # 查找对应的产品
                        products = flight.get("Product", [])
                        if product_id:
                            # 如果提供了product_id，查找特定产品
                            for product in products:
                                if product.get("product_id") == product_id:
                                    # 尝试获取基于星期几的价格
                                    price = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    found = True
                                    break
                        else:
                            # 如果没有提供product_id，查找所有经济舱产品的最低价格
                            min_price = float('inf')
                            for product in products:
                                if product.get("seat_type") == "Economy class":
                                    p = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    if p and isinstance(p, (int, float)) and p < min_price:
                                        min_price = p
                            if min_price != float('inf'):
                                price = min_price
                                found = True
                        break
            else:
                # 在火车数据中查找
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        # 查找对应的产品
                        products = train.get("Product", [])
                        if product_id:
                            # 如果提供了product_id，查找特定产品
                            for product in products:
                                if product.get("product_id") == product_id:
                                    # 尝试获取基于星期几的价格
                                    price = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    found = True
                                    break
                        else:
                            # 如果没有提供product_id，查找二等座或一等座的最低价格
                            min_price = float('inf')
                            for product in products:
                                if product.get("seat_type") in ["Second class", "First class"]:
                                    p = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    if p and isinstance(p, (int, float)) and p < min_price:
                                        min_price = p
                            if min_price != float('inf'):
                                price = min_price
                                found = True
                        break
            
            # 检查是否找到有效的价格
            if not found or price is None:
                return False
            
            # 检查价格是否低于上限
            if not isinstance(price, (int, float)) or price > max_price:
                return False
        
        # 所有交通信息都满足价格条件
        return True


    @register_func("validate_transport_cost_less_per_person_round_trip")
    def validate_transport_cost_less_per_person_round_trip(self, transportation_info, validation_params):
        """
        校验航班/火车往返价格是否低于指定的人均往返价格上限
        transportation_info: 可以是包含两个dict的列表（去程和返程），每个dict包含id、product_id、date等信息
        validation_params: 包含价格上限的列表，如[800]
        """
        # 处理输入：必须是包含至少两个dict的列表（去程和返程）
        if isinstance(transportation_info, dict):
            return False  # 单个交通信息不能构成往返
        
        transport_infos = transportation_info
        if len(transport_infos) < 2:
            return False
        
        # 解析价格上限
        if not validation_params:
            return False
        
        max_total_price = validation_params
        total_price = 0
        
        # 检查每个交通信息（去程和返程）
        for info in transport_infos:
            # 获取交通方式ID、产品ID和日期
            transport_id = info.get("id")
            product_id = info.get("product_id")
            date_str = info.get("date")
            
            if not transport_id or not date_str:
                return False
            
            # 计算星期几缩写
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 根据ID前缀判断是航班还是火车
            found = False
            price = None
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查该日期航班是否可用
                        if available_key in flight and flight.get(available_key) != 1:
                            return False  # 航班在该日期不可用
                        
                        # 查找对应的产品
                        products = flight.get("Product", [])
                        if product_id:
                            # 如果提供了product_id，查找特定产品
                            for product in products:
                                if product.get("product_id") == product_id:
                                    # 尝试获取基于星期几的价格
                                    price = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    found = True
                                    break
                        else:
                            # 如果没有提供product_id，查找所有经济舱产品的最低价格
                            min_price = float('inf')
                            for product in products:
                                if product.get("seat_type") == "Economy class":
                                    p = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    if p and isinstance(p, (int, float)) and p < min_price:
                                        min_price = p
                            if min_price != float('inf'):
                                price = min_price
                                found = True
                        break
            else:
                # 在火车数据中查找
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        # 查找对应的产品
                        products = train.get("Product", [])
                        if product_id:
                            # 如果提供了product_id，查找特定产品
                            for product in products:
                                if product.get("product_id") == product_id:
                                    # 尝试获取基于星期几的价格
                                    price = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    found = True
                                    break
                        else:
                            # 如果没有提供product_id，查找二等座或一等座的最低价格
                            min_price = float('inf')
                            for product in products:
                                if product.get("seat_type") in ["Second class", "First class"]:
                                    p = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    if p and isinstance(p, (int, float)) and p < min_price:
                                        min_price = p
                            if min_price != float('inf'):
                                price = min_price
                                found = True
                        break
            
            # 检查是否找到有效的价格
            if not found or price is None:
                return False
            
            # 累加价格
            if isinstance(price, (int, float)):
                total_price += price
            else:
                return False
        # print(total_price)
        # 检查总价格是否低于上限
        return total_price <= max_total_price


    @register_func("validate_transport_cost_less_total")
    def validate_transport_cost_less_total(self, transportation_info, validation_params):
        """
        校验航班/火车总价格（可能包含多个交通项）是否低于指定的总价格上限
        transportation_info: 交通信息列表，每个dict包含id、product_id、date、quantity等信息
        validation_params: 包含价格上限的列表，如[300]
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
        # 解析价格上限
        if not validation_params:
            return False
        
        max_total_price = validation_params
        total_price = 0
        
        # 检查每个交通信息
        for info in transport_infos:
            # 获取交通方式ID、产品ID、日期和数量
            transport_id = info.get("id")
            product_id = info.get("product_id")
            date_str = info.get("date")
            quantity = info.get("quantity", 1)  # 默认数量为1
            
            # 检查数量是否有效
            if not isinstance(quantity, int) or quantity < 1:
                quantity = 1
            
            if not transport_id or not date_str:
                return False
            
            # 计算星期几缩写
            week_abbr = self._weekday_abbr(date_str)
            available_key = f"{week_abbr}_available"
            
            # 根据ID前缀判断是航班还是火车
            found = False
            price = None
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查该日期航班是否可用
                        if available_key in flight and flight.get(available_key) != 1:
                            return False  # 航班在该日期不可用
                        
                        # 查找对应的产品
                        products = flight.get("Product", [])
                        if product_id:
                            # 如果提供了product_id，查找特定产品
                            for product in products:
                                if product.get("product_id") == product_id:
                                    # 尝试获取基于星期几的价格
                                    price = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    found = True
                                    break
                        else:
                            # 如果没有提供product_id，查找所有经济舱产品的最低价格
                            min_price = float('inf')
                            for product in products:
                                if product.get("seat_type") == "Economy class":
                                    p = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    if p and isinstance(p, (int, float)) and p < min_price:
                                        min_price = p
                            if min_price != float('inf'):
                                price = min_price
                                found = True
                        break
            else:
                # 在火车数据中查找
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        # 查找对应的产品
                        products = train.get("Product", [])
                        if product_id:
                            # 如果提供了product_id，查找特定产品
                            for product in products:
                                if product.get("product_id") == product_id:
                                    # 尝试获取基于星期几的价格
                                    price = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    found = True
                                    break
                        else:
                            # 如果没有提供product_id，查找二等座或一等座的最低价格
                            min_price = float('inf')
                            for product in products:
                                if product.get("seat_type") in ["Second class", "First class"]:
                                    p = product.get(week_abbr) or product.get("price") or product.get("Price")
                                    if p and isinstance(p, (int, float)) and p < min_price:
                                        min_price = p
                            if min_price != float('inf'):
                                price = min_price
                                found = True
                        break
            
            # 检查是否找到有效的价格
            if not found or price is None:
                return False
            
            # 累加价格，并考虑数量
            if isinstance(price, (int, float)):
                total_price += price * quantity
            else:
                return False
        
        # 检查总价格是否低于上限
        # print(total_price)
        return total_price <= max_total_price

    #transportation_info示例
    # [
    #     {"id": "Flight_123", "product_id": "P123", "date": "2025-11-10", "quantity": 1},
    #     {"id": "Train_456", "product_id": "P456", "date": "2025-11-10", "quantity": 1},

    # ]
    #validation_params示例
    # [300]


    @register_func("validate_platform")
    def validate_platform(self, transportation_info, validation_params):
        """
        校验航班/火车产品是否在指定的平台上
        transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id和date
        validation_params: 包含要指定的平台列表，如["alitrip", "direct"]
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
        # 检查validation_params是否有效
        if not validation_params or not isinstance(validation_params, list):
            return False
        
        # 检查每个交通信息
        for info in transport_infos:
            # 获取交通方式ID、产品ID和日期
            transport_id = info.get("id")
            product_id = info.get("product_id")
            date_str = info.get("date")
            
            if not transport_id or not product_id:
                return False
            
            # 计算星期几缩写
            week_abbr = self._weekday_abbr(date_str) if date_str else None
            available_key = f"{week_abbr}_available" if week_abbr else None
            
            # 根据ID前缀判断是航班还是火车
            found = False
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查该日期航班是否可用
                        if available_key and available_key in flight and flight.get(available_key) != 1:
                            return False  # 航班在该日期不可用
                        
                        # 查找对应的产品
                        products = flight.get("Product", [])
                        for product in products:
                            if product.get("product_id") == product_id:
                                # 检查产品平台是否在指定平台列表中
                                product_platform = product.get("source_platform")
                                if product_platform and product_platform in validation_params:
                                    found = True
                                    break
                        break
            else:
                # 在火车数据中查找
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        # 查找对应的产品
                        products = train.get("Product", [])
                        for product in products:
                            if product.get("product_id") == product_id:
                                # 检查产品平台是否在指定平台列表中
                                product_platform = product.get("source_platform")
                                if product_platform and product_platform in validation_params:
                                    found = True
                                    break
                        break
            
            # 如果没有找到符合条件的产品，则返回False
            if not found:
                return False
        
        # 所有交通信息都满足平台条件
        return True


    @register_func("validate_minimum_ontime_rate")
    def validate_minimum_ontime_rate(self, transportation_info, validation_params):
        """
        校验航班/火车是否满足最低准点率要求
        transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id和date
        validation_params: 包含最低准点率阈值的列表，如0.85
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
        # 检查validation_params是否有效
        if not validation_params:
            return False
        
        # 获取最低准点率阈值
        min_ontime_rate = validation_params
        
        # 检查每个交通信息
        for info in transport_infos:
            # 获取交通方式ID
            transport_id = info.get("id")
            
            if not transport_id:
                return False
            
            # 根据ID前缀判断是航班还是火车
            found = False
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查航班准点率是否满足要求
                        flight_ontime_perf = flight.get("On-Time Performance", 0)
                        if flight_ontime_perf >= min_ontime_rate:
                            found = True
                        break
            else:
                # 对于火车，默认通过验证
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        found = True
                        break
            
            # 如果没有找到符合条件的交通方式，则返回False
            if not found:
                return False
        
        # 所有交通信息都满足准点率条件
        return True


    @register_func("validate_max_delay")
    def validate_max_delay(self, transportation_info, validation_params):
        """
        校验航班/火车是否满足最大允许延误时间要求
        transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id和date
        validation_params: 包含最大允许延误时间（分钟）的列表，如[15]
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
        # 检查validation_params是否有效
        if not validation_params:
            return False
        
        # 获取最大允许延误时间
        max_delay = validation_params
        
        # 检查每个交通信息
        for info in transport_infos:
            # 获取交通方式ID
            transport_id = info.get("id")
            
            if not transport_id:
                return False
            
            # 根据ID前缀判断是航班还是火车
            found = False
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查航班平均延误时间是否小于等于最大允许延误时间
                        flight_avg_delay = flight.get("Average Delay (minutes)", 0)
                        if flight_avg_delay <= max_delay:
                            found = True
                        break
            else:
                # 对于火车，默认通过验证
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        found = True
                        break
            
            # 如果没有找到符合条件的交通方式，则返回False
            if not found:
                return False
        
        # 所有交通信息都满足最大延误时间条件
        return True


    @register_func("validate_airline_exclude")
    def validate_airline_exclude(self, transportation_info, validation_params):
        """
        校验航班/火车是否不在排除的航空公司列表中
        transportation_info: 可以是单个dict或dict的列表，每个dict包含id、product_id和date
        validation_params: 包含排除航空公司名称的列表，如["Hainan Airlines"]
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else transportation_info
        
        # 检查validation_params是否有效
        if not validation_params:
            return False
        
        # 获取排除航空公司名称
        exclude_airline = validation_params
        
        # 检查每个交通信息
        for info in transport_infos:
            # 获取交通方式ID
            transport_id = info.get("id")
            
            if not transport_id:
                return False
            
            # 根据ID前缀判断是航班还是火车
            found = False
            
            if transport_id.startswith('Flight_'):
                # 在航班数据中查找
                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 检查航班所属航空公司是否不是排除的航空公司
                        flight_airline = flight.get("Airline")
                        if flight_airline != exclude_airline:
                            found = True
                        break
            else:
                # 对于火车，默认通过验证
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        found = True
                        break
            
            # 如果没有找到符合条件的交通方式，则返回False
            if not found:
                return False
        
        # 所有交通信息都满足条件
        return True

    @register_func("validate_arrive_time_condition")
    def validate_arrive_time_condition(self, transportation_info, validation_params):
        """
        校验航班/火车到达时间是否在给定时间段内

        transportation_info: dict 或 list[dict]，每个 dict 可能包含:
            - id, date
            - 可选 product_id（会被忽略）
            - 可选 type: "d_a"/"r_a"/... 或 list[str]
        validation_params: "HH:MM-HH:MM"

        规则：
        - 只校验 type 结尾匹配 "_a" 的交通信息；没有匹配的则跳过校验（返回 True）
        - 到达时间落在 [start, end) 区间内才算通过（支持跨天：如 "23:00-02:00"）
        - 航班额外校验：若存在 "{Mon|Tue|...}_available" 且 != 1，则失败
        - 不依赖 product_id（即使传了也忽略），避免误伤
        """
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else (transportation_info or [])

        if not validation_params:
            return False
        if not isinstance(validation_params, str) or "-" not in validation_params:
            return False

        # 解析时间范围
        start_time, end_time = validation_params.split("-", 1)
        start_min = self.parse_time_to_minutes(start_time)
        end_min = self.parse_time_to_minutes(end_time)
        if None in (start_min, end_min):
            return False

        # === 只挑 type 结尾匹配 "_a" 的进行校验；没有的跳过 ===
        def _type_endswith(info, suffix: str) -> bool:
            t = info.get("type")
            if isinstance(t, list):
                return any(isinstance(x, str) and x.endswith(suffix) for x in t)
            return isinstance(t, str) and t.endswith(suffix)

        to_check = [info for info in transport_infos if isinstance(info, dict) and _type_endswith(info, "_a")]
        if not to_check:
            return True  # 没有对应 type 的交通信息 → 跳过校验

        # 检查每个交通信息
        for info in to_check:
            transport_id = info.get("id")
            date_str = info.get("date")

            if not transport_id or not date_str:
                return False

            arr_time_str = None
            if isinstance(transport_id, str) and transport_id.startswith("Flight_"):
                week_abbr = self._weekday_abbr(date_str)
                if not week_abbr:
                    return False

                available_key = f"{week_abbr}_available"

                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 若存在可用性字段则必须为 1；不存在则兼容旧数据不校验
                        if available_key in flight and flight.get(available_key) != 1:
                            return False

                        arr_time_str = flight.get("Arrival Time")
                        break
            else:
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        arr_time_str = train.get("Arrival Time")
                        break

            if not arr_time_str:
                return False

            arr_min = self.parse_time_to_minutes(arr_time_str)
            if arr_min is None:
                return False

            # 时间段判断（支持跨天），区间为 [start, end)
            if start_min <= end_min:
                if not (start_min <= arr_min < end_min):
                    return False
            else:
                if not (arr_min >= start_min or arr_min < end_min):
                    return False

        return True

    @register_func("validate_depart_time_condition")
    def validate_depart_time_condition(self, transportation_info, validation_params):
        """
        校验航班/火车出发时间是否在给定时间段内

        transportation_info: dict 或 list[dict]，每个 dict 可能包含:
            - id, date
            - 可选 product_id（会被忽略）
            - 可选 type: "d_d"/"r_d"/... 或 list[str]
        validation_params: "HH:MM-HH:MM"

        规则：
        - 只校验 type 结尾匹配 "_d" 的交通信息；没有匹配的则跳过校验（返回 True）
        - 出发时间落在 [start, end) 区间内才算通过（支持跨天：如 "23:00-02:00"）
        - 航班额外校验：若存在 "{Mon|Tue|...}_available" 且 != 1，则失败
        - 不依赖 product_id（即使传了也忽略），避免误伤
        """
        transport_infos = [transportation_info] if isinstance(transportation_info, dict) else (transportation_info or [])

        if not validation_params:
            return False
        if not isinstance(validation_params, str) or "-" not in validation_params:
            return False

        # 解析时间范围
        start_time, end_time = validation_params.split("-", 1)
        start_min = self.parse_time_to_minutes(start_time)
        end_min = self.parse_time_to_minutes(end_time)
        if None in (start_min, end_min):
            return False

        # === 只挑 type 结尾匹配 "_d" 的进行校验；没有的跳过 ===
        def _type_endswith(info, suffix: str) -> bool:
            t = info.get("type")
            if isinstance(t, list):
                return any(isinstance(x, str) and x.endswith(suffix) for x in t)
            return isinstance(t, str) and t.endswith(suffix)

        to_check = [info for info in transport_infos if isinstance(info, dict) and _type_endswith(info, "_d")]
        if not to_check:
            return True  # 没有对应 type 的交通信息 → 跳过校验

        # 检查每个交通信息
        for info in to_check:
            transport_id = info.get("id")
            date_str = info.get("date")

            if not transport_id or not date_str:
                return False

            dep_time_str = None
            if isinstance(transport_id, str) and transport_id.startswith("Flight_"):
                week_abbr = self._weekday_abbr(date_str)
                if not week_abbr:
                    return False

                available_key = f"{week_abbr}_available"

                for flight in self.flights:
                    if flight.get("Flight_id") == transport_id:
                        # 若存在可用性字段则必须为 1；不存在则兼容旧数据不校验
                        if available_key in flight and flight.get(available_key) != 1:
                            return False

                        dep_time_str = flight.get("Departure Time")
                        break
            else:
                for train in self.trains:
                    if train.get("Train_id") == transport_id:
                        dep_time_str = train.get("Departure Time")
                        break

            if not dep_time_str:
                return False

            dep_min = self.parse_time_to_minutes(dep_time_str)
            if dep_min is None:
                return False

            # 时间段判断（支持跨天），区间为 [start, end)
            if start_min <= end_min:
                if not (start_min <= dep_min < end_min):
                    return False
            else:
                if not (dep_min >= start_min or dep_min < end_min):
                    return False

        return True



if __name__ == "__main__":
    print("\n测试用例1：单个路线，交通类型为'flight'")
    routes_info1 = [
        {
            "departure_city": "Beijing",
            # "arrival_city": "Changchun",
            "arrival_city": "Wuxi",
            "date_str": "2024-01-15",  # 周一
            "number_of_people": 5
        },
        # {
        #     "departure_city": "Shanghai",
        #     "arrival_city": "Beijing",
        #     "date_str": "2024-01-16",  # 周一
        #     "number_of_people": 5
        # }
    ]
    # 初始化评估器
    te = TransportationEvaluator("data_final/flights.json")
    transportation_info = [
        {"id":"Train_00000001","product_id": "P_T_0000001", "date": "2025-11-10", "quantity": 1, "type": "d_d"},
        {"id": "Flight_00000675", "product_id": "P_F_00005394", "date": "2025-11-10", "quantity": 1, "type": "d_a"}
    ]
    res = te.validate_airline_exclude(transportation_info,validation_params="9 Air")
    print(res)

    res = te.validate_minimum_ontime_rate(transportation_info,validation_params=0.89)
    print(res)

    res = te.validate_depart_time_condition(transportation_info,validation_params="05:00-12:00")
    print(res)
    

    # result1 = te.generate_platform_specify(routes_info1, transport_type='flight',generate_params={"alitrip": ["alitrip"]})
    # print(json.dumps(result1, ensure_ascii=False, indent=2))

    # result2 = te.generate_platform_exclude(routes_info1, transport_type='flight',generate_params={"alitrip": ["alitrip"]})
    # # result2 = te.generate_platform_exclude(routes_info1)
    # print(json.dumps(result2, ensure_ascii=False, indent=2))


    # # result3 = te.generate_minimum_ontime_rate(routes_info1, transport_type='flight',generate_params={"95": 0.95})
    # # result3 = te.generate_minimum_ontime_rate(routes_info1, transport_type='flight')
    # result3 = te.generate_minimum_ontime_rate(routes_info1, transport_type='flight')
    # print(json.dumps(result3, ensure_ascii=False, indent=2))


    # result4 = te.generate_max_delay(routes_info1)
    # print(json.dumps(result4, ensure_ascii=False, indent=2))

    # result5 = te.generate_airline_exclude(routes_info1)
    # print(json.dumps(result5, ensure_ascii=False, indent=2))

        # 读取输入数据
    # input_file = 'Evaluation/Transportation/query_syn/city_trips_2_all_sample_with_people.json'
    # with open(input_file, 'r', encoding='utf-8') as f:
    #     city_trips = json.load(f)
    
    # # 结果存储
    # results = []
    
    # # 遍历每个trip
    # for trip in city_trips:
    #     trip_id = trip['id']
    #     route = trip['route'][0]  # 假设每个trip只有一条route
        
    #     # 准备routes_info格式
    #     routes_info_1 = {
    #         'departure_city': route['from'],
    #         'arrival_city': route['to'],
    #         'date_str': route['depart_date'],
    #         'number_of_people': route['number_of_people']
    #     }

    #     routes_info_2 = {
    #         'departure_city': route['to'],
    #         'arrival_city': route['from'],
    #         'date_str': route['return_date'],
    #         'number_of_people': route['number_of_people']
    #     }

    #     routes_info = [routes_info_1, routes_info_2]
    #     result = te.generate_transport_cost_less_per_person_round_trip(routes_info)
    #     print(result)
    #     break

        




    