from pprint import pprint
# from rubrics_attraction import *
import json
import random
import math
from inspect import signature
from geopy.distance import geodesic
from typing import List, Dict, Any, Optional

# 定义一个很大的数字来代替 float('inf')，用于 JSON 序列化
# 使用 1e10 (10,000,000,000) 作为"无穷大"的替代值
INF = 1e9


##########
# 所有 validation 函数的输入 attraction_info 都是 list of dict，每个dict包含 poiId，visiting_time（计划游玩时间，分钟为单位）
# 示例：
# attraction_info = [
#     {"poiId": 77064, "visiting_time": 300},
#     {"poiId": 78223, "visiting_time": 200}
# ]



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 AttractionEvaluator:
    def __init__(self, attractions):
        self.attractions = attractions
        # 市中心坐标数据（从 Train_tools.py 复制）
        self.city_center_coords = {
            "Beijing": {"lon": 116.407387, "lat": 39.904179},
            "Changchun": {"lon": 125.323643, "lat": 43.816996},
            "Changsha": {"lon": 112.938882, "lat": 28.228304},
            "Chengdu": {"lon": 104.066301, "lat": 30.572961},
            "Chongqing": {"lon": 106.551787, "lat": 29.56268},
            "Dalian": {"lon": 121.614786, "lat": 38.913962},
            "Fuzhou": {"lon": 119.296411, "lat": 26.074286},
            "Guangzhou": {"lon": 113.264499, "lat": 23.130061},
            "Guilin": {"lon": 110.179752, "lat": 25.235615},
            "Guiyang": {"lon": 106.628201, "lat": 26.646694},
            "Haikou": {"lon": 110.200162, "lat": 20.046316},
            "Hangzhou": {"lon": 120.209903, "lat": 30.246566},
            "Harbin": {"lon": 126.53505, "lat": 45.802981},
            "Hong Kong": {"lon": 114.170714, "lat": 22.278354},
            "Jinan": {"lon": 117.120128, "lat": 36.652069},
            "Kaifeng": {"lon": 114.314278, "lat": 34.798083},
            "Kunming": {"lon": 102.833669, "lat": 24.88149},
            "Lijiang": {"lon": 100.225936, "lat": 26.855165},
            "Luoyang": {"lon": 112.453895, "lat": 34.619702},
            "Nanchang": {"lon": 115.857972, "lat": 28.682976},
            "Nanjing": {"lon": 118.796624, "lat": 32.059344},
            "Nanning": {"lon": 108.366407, "lat": 22.8177},
            "Ningbo": {"lon": 121.62454, "lat": 29.860258},
            "Qingdao": {"lon": 120.382665, "lat": 36.066938},
            "Sanya": {"lon": 109.511709, "lat": 18.252865},
            "Shanghai": {"lon": 121.473667, "lat": 31.230525},
            "Shenyang": {"lon": 123.464675, "lat": 41.677576},
            "Shenzhen": {"lon": 114.057939, "lat": 22.543527},
            "Suzhou": {"lon": 120.585294, "lat": 31.299758},
            "Taiyuan": {"lon": 112.549656, "lat": 37.870451},
            "Tianjin": {"lon": 117.201509, "lat": 39.085318},
            "Weihai": {"lon": 122.120519, "lat": 37.513315},
            "Wuhan": {"lon": 114.304569, "lat": 30.593354},
            "Wuxi": {"lon": 120.311889, "lat": 31.491064},
            "Xi'an": {"lon": 108.939645, "lat": 34.343207},
            "Xiamen": {"lon": 118.08891, "lat": 24.479627},
            "Xishuangbanna": {"lon": 100.797002, "lat": 22.009037},
            "Yantai": {"lon": 121.447755, "lat": 37.464551},
            "Zhengzhou": {"lon": 113.625351, "lat": 34.746303},
            "Zhuhai": {"lon": 113.576892, "lat": 22.271644}
        }

    # ---------------------------------
    # 调用执行入口
    # ---------------------------------
    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 get_attraction_by_id(self, poiId):
        for attraction in self.attractions:
            if attraction['poiId'] == poiId:
                return attraction
        return None

    def get_all_categories(self, attractions):
        categories = set()
        for attraction in attractions:
            for category in attraction.get('categories', []):
                categories.add(category)
        return categories
    
    def calculate_distance(self, coord1, coord2):
        return geodesic(coord1, coord2).km


    ##############################################
    ############ include_categories ##############
    ##############################################


    #跟天数有关 随机10个有解的返回
    #2-3：1-2
    #4-5：1-3
    #>=6：2-4
    @register_func("generate_include_categories")
    def generate_include_categories(self,
                                    city_list: List[str],
                                    stay_days: List[int],
                                    generate_params=None) -> Dict[str, Any]:
        """
        根据停留天数生成类别包含约束
        
        Args:
            city_list: 城市列表
            stay_days: 每个城市的停留天数列表
            generate_params: 生成参数（暂时未使用）
            
        Returns:
            包含选中类别和候选景点的字典
        """
        # 确定类别数量范围
        max_days = max(stay_days) if stay_days else 0
        if 2 <= max_days <= 3:
            category_count_range = (1, 2)
        elif 4 <= max_days <= 5:
            category_count_range = (1, 3)
        elif max_days >= 6:
            category_count_range = (2, 4)
        else:
            category_count_range = (1, 2)  # 默认
            
        # 筛选出在目标城市中存在的景点类别
        available_categories = set()
        candidate_attraction_ids = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                categories = attraction.get('categories', [])
                available_categories.update(categories)
                candidate_attraction_ids.append(attraction.get('poiId'))
        
        available_categories = list(available_categories)
        
        if not available_categories:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 生成10个解
        all_solutions = {}
        seen_content = set()  # 用于跟踪已生成的内容，避免重复
        
        for i in range(10):
            # 随机选择类别数量
            k = random.randint(category_count_range[0], category_count_range[1])
            k = min(k, len(available_categories))  # 确保不超过可用类别数
            
            # 随机选择k个类别
            selected_categories = random.sample(available_categories, k)
            
            # 使用排序后的元组作为唯一标识（因为顺序不重要）
            content_key = tuple(sorted(selected_categories))
            
            # 如果内容已存在，跳过这个解
            if content_key in seen_content:
                continue
            
            # 标记为已见过
            seen_content.add(content_key)
            
            # 创建标签
            if len(selected_categories) == 1:
                label = selected_categories[0]
            else:
                label = ', '.join(selected_categories)
            
            # 由于内容已保证不重复，label也应该是唯一的
            all_solutions[label] = selected_categories
        
        # 选择要使用的类别：如果有generate_params则使用它，否则随机选择一个解
        if generate_params:
            # 处理字典格式的generate_params：{selected_description: validation_params}
            if isinstance(generate_params, dict):
                # 如果是字典，提取值（validation_params，应该是列表）
                if generate_params:
                    selected_categories = list(generate_params.values())[0]
                else:
                    selected_categories = None
            else:
                # 如果不是字典，直接使用（可能是列表）
                selected_categories = generate_params
            if selected_categories and len(selected_categories) == 1:
                selected_label = selected_categories[0]
            elif selected_categories:
                selected_label = ', '.join(selected_categories)
            else:
                selected_label = None
        elif all_solutions:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(all_solutions.keys()))
            selected_categories = all_solutions[selected_label]
        else:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
            
        # 筛选包含选中类别的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                attraction_categories = set(attraction.get('categories', []))
                if attraction_categories.intersection(selected_categories):
                    matching_attraction_ids.append(attraction.get('poiId'))
                    # 收集该景点的所有 product_id
                    ticket_products = attraction.get('ticket_products', [])
                    for ticket in ticket_products:
                        product_id = ticket.get('product_id')
                        if product_id:
                            matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_categories,
            "candidate_ids": matching_attraction_ids,
            "candidate_product_ids": matching_ticket_ids,  # 景点的票务产品ID
            "all_labels_and_ranges": all_solutions
        }
    

    @register_func("validate_include_categories")
    def validate_include_categories(self, attraction_info, validation_params):
        """
        验证景点信息是否包含指定类别
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 需要包含的类别列表
            
        Returns:
            bool: 是否满足约束条件
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return False
        
        required_categories = set(validation_params)
        
        # 检查每个景点信息
        found_categories = set()
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    attraction_categories = set(attraction.get('categories', []))
                    found_categories.update(attraction_categories)
        # print(f"required_categories: {required_categories}")
        # print(f"found_categories: {found_categories}")
        # 检查是否包含了所有必需的类别
        return required_categories.issubset(found_categories)

    

    ##############################################
    ############ exclude_categories ##############
    ##############################################


    #统一roll 1-2个，随机10个解
    #和上面判断重复（这个判断不在这里做，在rubrics_attraction中，涉及到传入多个rubric type进行随机时做）
    @register_func("generate_exclude_categories")
    def generate_exclude_categories(self,
                                    city_list: List[str],
                                    stay_days: List[int] = None,
                                    generate_params=None) -> Dict[str, Any]:
        """
        生成类别排除约束
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定要排除的类别列表
            
        Returns:
            包含排除类别和候选景点的字典
        """
        # 筛选出在目标城市中存在的景点类别
        available_categories = set()
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                categories = attraction.get('categories', [])
                available_categories.update(categories)
        
        available_categories = list(available_categories)
        
        if not available_categories:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 生成10个解
        all_solutions = {}
        seen_content = set()  # 用于跟踪已生成的内容，避免重复
        
        for i in range(10):
            # 统一roll 1-2个要排除的类别数量
            k = random.randint(1, min(2, len(available_categories)))
            
            # 随机选择k个类别进行排除
            excluded_categories = random.sample(available_categories, k)
            
            # 使用排序后的元组作为唯一标识（因为顺序不重要）
            content_key = tuple(sorted(excluded_categories))
            
            # 如果内容已存在，跳过这个解
            if content_key in seen_content:
                continue
            
            # 标记为已见过
            seen_content.add(content_key)
            
            # 创建标签
            if len(excluded_categories) == 1:
                label = excluded_categories[0]
            else:
                label = ', '.join(excluded_categories)
            
            # 由于内容已保证不重复，label也应该是唯一的
            all_solutions[label] = excluded_categories
        
        # 选择要使用的类别：如果有generate_params则使用它，否则随机选择一个解
        if generate_params:
            # 处理字典格式的generate_params：{selected_description: validation_params}
            if isinstance(generate_params, dict):
                # 如果是字典，提取值（validation_params，应该是列表）
                if generate_params:
                    excluded_categories = list(generate_params.values())[0]
                else:
                    excluded_categories = None
            else:
                # 如果不是字典，直接使用（可能是列表）
                excluded_categories = generate_params
            if excluded_categories and len(excluded_categories) == 1:
                selected_label = excluded_categories[0]
            elif excluded_categories:
                selected_label = ', '.join(excluded_categories)
            else:
                selected_label = None
        elif all_solutions:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(all_solutions.keys()))
            excluded_categories = all_solutions[selected_label]
        else:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
            
        # 筛选不包含排除类别的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                attraction_categories = set(attraction.get('categories', []))
                excluded_categories_set = set(excluded_categories)
                # 如果景点的类别与排除类别没有交集，则包含该景点
                if not attraction_categories.intersection(excluded_categories_set):
                    matching_attraction_ids.append(attraction.get('poiId'))
                    # 收集该景点的所有 product_id
                    ticket_products = attraction.get('ticket_products', [])
                    for ticket in ticket_products:
                        product_id = ticket.get('product_id')
                        if product_id:
                            matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": excluded_categories,
            "candidate_ids": matching_attraction_ids,
            "candidate_product_ids": matching_ticket_ids,  # 景点的票务产品ID
            "all_labels_and_ranges": all_solutions
        }


    @register_func("validate_exclude_categories")
    def validate_exclude_categories(self, attraction_info, validation_params):
        """
        验证景点信息是否不包含指定的排除类别
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 需要排除的类别列表
            
        Returns:
            bool: 是否满足约束条件（不包含排除的类别）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return True  # 如果没有排除类别，则总是满足条件
        
        excluded_categories = set(validation_params)
        
        # 检查每个景点信息
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    attraction_categories = set(attraction.get('categories', []))
                    # 如果发现任何排除的类别，则不满足约束
                    if attraction_categories.intersection(excluded_categories):
                        return False
        
        # 如果没有发现任何排除的类别，则满足约束
        return True

    
    ##############################################
    ############ include_attractions #############
    ##############################################
    
    #跟天数有关 随机10个有解的返回
    #按照热度(heatScore)调一下随机概率
    #2-3：1-2
    #4-5：1-3
    #>=6：1-5
    @register_func("generate_include_attractions")
    def generate_include_attractions(self,
                                    city_list: List[str],
                                    stay_days: List[int],
                                    generate_params=None) -> Dict[str, Any]:
        """
        根据停留天数生成景点包含约束，按热度调整随机概率
        
        Args:
            city_list: 城市列表
            stay_days: 每个城市的停留天数列表
            generate_params: 生成参数，指定要包含的景点ID列表
            
        Returns:
            包含选中景点和候选景点的字典（已按要求修改 candidate 仅返回选中项目）
        """

        # 确定景点数量范围
        max_days = max(stay_days) if stay_days else 0
        if 2 <= max_days <= 3:
            attraction_count_range = (1, 2)
        elif 4 <= max_days <= 5:
            attraction_count_range = (1, 3)
        elif max_days >= 6:
            attraction_count_range = (1, 5)
        else:
            attraction_count_range = (1, 2)  # 默认
            
        # 筛选出在目标城市中的景点
        available_attractions = []
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        # 若没有可用景点
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 基于热度生成权重
        def get_weighted_attractions(attractions):
            weights = []
            for attraction in attractions:
                heat_score = attraction.get('heatScore', 0)
                weight = max(1, heat_score)
                weights.append(weight)
            return weights
        
        # 生成解空间
        all_solutions = {}
        seen_content = set()  # 用于去重
        
        for i in range(10):
            # 随机选择景点数量
            k = random.randint(attraction_count_range[0], attraction_count_range[1])
            k = min(k, len(available_attractions))
            
            # 按热度随机抽取
            weights = get_weighted_attractions(available_attractions)
            selected_attractions = random.choices(available_attractions, weights=weights, k=k)
            
            # 去重
            selected_attractions = list({attr['poiId']: attr for attr in selected_attractions}.values())
            
            # 数量不够则补齐
            while len(selected_attractions) < k:
                remaining = [attr for attr in available_attractions if attr not in selected_attractions]
                if not remaining:
                    break
                remaining_weights = get_weighted_attractions(remaining)
                extra = random.choices(remaining, weights=remaining_weights, k=1)[0]
                if extra not in selected_attractions:
                    selected_attractions.append(extra)
            
            selected_ids = [attr.get('poiId') for attr in selected_attractions]
            content_key = tuple(sorted(selected_ids))
            
            # 去重
            if content_key in seen_content:
                continue
            seen_content.add(content_key)
            
            # 生成 label
            sorted_attractions = sorted(selected_attractions, key=lambda x: x.get('poiId'))
            names = [attr.get('poiName') for attr in sorted_attractions]
            label = names[0] if len(names) == 1 else ", ".join(names)
            
            all_solutions[label] = selected_ids
        
        # 选择使用的最终解
        if generate_params:
            if isinstance(generate_params, dict):
                if generate_params:
                    selected_attraction_ids = list(generate_params.values())[0]
                else:
                    selected_attraction_ids = None
            else:
                selected_attraction_ids = generate_params
            
            if not selected_attraction_ids:
                if all_solutions:
                    selected_label = random.choice(list(all_solutions.keys()))
                    selected_attraction_ids = all_solutions[selected_label]
                else:
                    return {
                        "selected_description": [],
                        "validation_params": [],
                        "candidate_ids": [],
                        "candidate_product_ids": [],
                        "all_labels_and_ranges": {}
                    }
            else:
                selected_names = []
                for poi_id in selected_attraction_ids:
                    attraction = self.get_attraction_by_id(poi_id)
                    if attraction:
                        name = attraction.get('poiName') or f"Attraction {poi_id}"
                        selected_names.append(name)
                    else:
                        selected_names.append(f"Attraction {poi_id}")
                
                selected_label = selected_names[0] if len(selected_names) == 1 else ", ".join(selected_names)
        
        elif all_solutions:
            selected_label = random.choice(list(all_solutions.keys()))
            selected_attraction_ids = all_solutions[selected_label]
        
        else:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 🔥 关键修改：只返回选中景点对应的产品 ID
        selected_ticket_ids = []
        for poi_id in selected_attraction_ids:
            attraction = self.get_attraction_by_id(poi_id)
            if attraction:
                for ticket in attraction.get('ticket_products', []):
                    pid = ticket.get('product_id')
                    if pid:
                        selected_ticket_ids.append(pid)

        # 返回最终结果（已按需求修改）
        return {
            "selected_description": selected_label,
            "validation_params": selected_attraction_ids,
            "candidate_ids": selected_attraction_ids,      # ← 只返回选中景点 ID
            "candidate_product_ids": selected_ticket_ids,  # ← 只返回选中景点对应产品 ID
            "all_labels_and_ranges": all_solutions
        }


    @register_func("validate_include_attractions")
    def validate_include_attractions(self, attraction_info, validation_params):
        """
        验证景点信息是否包含指定的景点
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 需要包含的景点ID列表
            
        Returns:
            bool: 是否满足约束条件
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return False
        
        required_attraction_ids = set(validation_params)
        
        # 检查每个景点信息
        found_attraction_ids = set()
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                found_attraction_ids.add(attraction_id)
        
        # 检查是否包含了所有必需的景点
        return required_attraction_ids.issubset(found_attraction_ids)



    ##############################################
    ############ exclude_attractions #############
    ##############################################

    #统一roll 1-2个，随机10个解
    #和上面判断重复 (这个判断不在这里做，在rubrics_attraction中，涉及到传入多个rubric type进行随机时做）
    @register_func("generate_exclude_attractions")
    def generate_exclude_attractions(self,
                                     city_list: List[str],
                                     stay_days: List[int] = None,
                                     generate_params=None) -> Dict[str, Any]:
        """
        生成景点排除约束，统一roll 1-2个景点
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定要排除的景点ID列表
            
        Returns:
            包含排除景点和候选景点的字典
        """
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 生成10个解
        all_solutions = {}
        seen_content = set()  # 用于跟踪已生成的内容，避免重复
        
        for i in range(10):
            # 统一roll 1-2个要排除的景点数量
            k = random.randint(1, min(2, len(available_attractions)))
            
            # 随机选择k个景点进行排除
            excluded_attractions = random.sample(available_attractions, k)
            
            # 获取景点ID列表
            excluded_ids = [attr.get('poiId') for attr in excluded_attractions]
            
            # 使用排序后的元组作为唯一标识（因为顺序不重要）
            content_key = tuple(sorted(excluded_ids))
            
            # 如果内容已存在，跳过这个解
            if content_key in seen_content:
                continue
            
            # 标记为已见过
            seen_content.add(content_key)
            
            # 创建标签（按ID排序后的顺序，确保相同内容生成相同label）
            sorted_excluded = sorted(excluded_attractions, key=lambda x: x.get('poiId'))
            attraction_names = [attr.get('poiName') or f"Attraction {attr.get('poiId')}" for attr in sorted_excluded]
            if len(attraction_names) == 1:
                label = attraction_names[0]
            else:
                label = ', '.join(attraction_names)
            
            # 由于内容已保证不重复，label也应该是唯一的
            all_solutions[label] = excluded_ids
        
        # 选择要使用的景点：如果有generate_params则使用它，否则随机选择一个解
        if generate_params:
            # 处理字典格式的generate_params：{selected_description: validation_params}
            if isinstance(generate_params, dict):
                # 如果是字典，提取值（validation_params，应该是列表）
                if generate_params:
                    excluded_attraction_ids = list(generate_params.values())[0]
                else:
                    excluded_attraction_ids = None
            else:
                # 如果不是字典，直接使用（可能是列表）
                excluded_attraction_ids = generate_params
            
            if not excluded_attraction_ids:
                # 如果没有有效的参数，随机选择一个解
                if all_solutions:
                    selected_label = random.choice(list(all_solutions.keys()))
                    excluded_attraction_ids = all_solutions[selected_label]
                else:
                    return {
                        "selected_description": [],
                        "validation_params": [],
                        "candidate_ids": [attr.get('poiId') for attr in available_attractions],
                        "candidate_product_ids": [],
                        "all_labels_and_ranges": {}
                    }
            else:
                # 获取景点名称用于生成标签
                excluded_names = []
                for poi_id in excluded_attraction_ids:
                    attraction = self.get_attraction_by_id(poi_id)
                    if attraction:
                        # 优先使用poiName，如果没有则使用默认格式
                        name = attraction.get('poiName') or f"Attraction {poi_id}"
                        excluded_names.append(name)
                    else:
                        excluded_names.append(f"Attraction {poi_id}")
                
                if len(excluded_names) == 1:
                    selected_label = excluded_names[0]
                else:
                    selected_label = ', '.join(excluded_names)
        elif all_solutions:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(all_solutions.keys()))
            excluded_attraction_ids = all_solutions[selected_label]
        else:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [attr.get('poiId') for attr in available_attractions],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
            
        # 筛选不包含排除景点的景点和票务产品
        excluded_ids_set = set(excluded_attraction_ids)
        matching_attraction_ids = []
        matching_ticket_ids = []
        
        for attraction in available_attractions:
            attraction_id = attraction.get('poiId')
            # 如果景点不在排除列表中，则包含该景点
            if attraction_id not in excluded_ids_set:
                matching_attraction_ids.append(attraction_id)
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": excluded_attraction_ids,
            "candidate_ids": matching_attraction_ids,
            "candidate_product_ids": matching_ticket_ids,  # 非排除景点的票务产品ID
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_exclude_attractions")
    def validate_exclude_attractions(self, attraction_info, validation_params):
        """
        验证景点信息是否不包含指定的排除景点
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 需要排除的景点ID列表
            
        Returns:
            bool: 是否满足约束条件（不包含排除的景点）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return True  # 如果没有排除景点，则总是满足条件
        
        excluded_attraction_ids = set(validation_params)
        
        # 检查每个景点信息
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                # 如果发现任何排除的景点，则不满足约束
                if attraction_id in excluded_attraction_ids:
                    return False
        
        # 如果没有发现任何排除的景点，则满足约束
        return True
    
    
    ################################################
    ########### include_popularity ########
    ################################################

    #city相关 按照百分位来算
    @register_func("generate_include_popularity")
    def generate_include_popularity(self,
                                    city_list: List[str],
                                    stay_days: List[int] = None,
                                    generate_params=None) -> Dict[str, Any]:
        """
        根据城市景点热度生成热度包含约束，按百分位动态划分
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定热度类型
            
        Returns:
            包含热度约束和候选景点的字典
        """
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                heat_score = attraction.get('heatScore', 0)
                available_attractions.append({
                    'attraction': attraction,
                    'heat_score': heat_score
                })
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 按热度分数排序（降序）
        available_attractions.sort(key=lambda x: x['heat_score'], reverse=True)
        
        # 计算百分位分界点
        total_count = len(available_attractions)
        top25_count = max(1, int(total_count * 0.25))  # 前25%
        top50_count = max(1, int(total_count * 0.50))  # 前50%
        
        # 获取分界点的热度分数
        if total_count == 1:
            # 只有一个景点的情况
            top25_threshold = available_attractions[0]['heat_score']
            top50_threshold = available_attractions[0]['heat_score']
        else:
            top25_threshold = available_attractions[min(top25_count-1, total_count-1)]['heat_score']
            top50_threshold = available_attractions[min(top50_count-1, total_count-1)]['heat_score']
        
        # 定义热度类别及其范围
        popularity_categories = {
            "hot and must-visit": (top25_threshold, INF),      # 前25%
            "hidden and less crowded": (0, top50_threshold)             # 后50%
        }
        
        # 选择要使用的热度类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        heat_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"hot and must-visit": [5.7, inf]}
            # 遍历popularity_categories，找到在generate_params中的label
            for category, heat_range_item in popularity_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值，如果不存在则使用默认值
                    heat_range = generate_params[category]
                    # 确保heat_range是元组格式
                    if isinstance(heat_range, list):
                        heat_range = tuple(heat_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                heat_range = generate_params[selected_label]
                if isinstance(heat_range, list):
                    heat_range = tuple(heat_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(popularity_categories.keys()))
            heat_range = popularity_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, heat_range_item in popularity_categories.items():
            # # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = heat_range_item
        
        # 筛选符合热度条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_heat, max_heat = heat_range
        
        for attr_info in available_attractions:
            attraction = attr_info['attraction']
            heat_score = attr_info['heat_score']
            
            # 判断是否在热度范围内
            # 处理 INF 或 float('inf') 的情况
            is_inf = (max_heat >= INF) or (isinstance(max_heat, float) and math.isinf(max_heat))
            if min_heat <= heat_score < max_heat or (is_inf and heat_score >= min_heat):
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": heat_range,  # 只存储热度范围元组
            "candidate_ids": matching_attraction_ids,  # 只包含符合热度条件的景点
            "candidate_product_ids": matching_ticket_ids,
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_include_popularity")
    def validate_include_popularity(self, attraction_info, validation_params):
        """
        验证景点信息是否符合指定热度类别
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 热度范围元组 (min_heat, max_heat)
            
        Returns:
            bool: 是否满足约束条件（全部景点都必须符合条件）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return False
        
        min_heat, max_heat = validation_params
        
        # 检查每个景点信息 - 必须全部符合条件
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    heat_score = attraction.get('heatScore', 0)
                    # 如果任何一个景点不在范围内，则不满足约束
                    # 处理 INF 或 float('inf') 的情况
                    is_inf = (max_heat >= INF) or (isinstance(max_heat, float) and math.isinf(max_heat))
                    if not (min_heat <= heat_score < max_heat or (is_inf and heat_score >= min_heat)):
                        return False
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都符合条件
    

    ################################################
    ########### exclude_popularity ########
    ################################################


    @register_func("generate_exclude_popularity")
    def generate_exclude_popularity(self,
                                    city_list: List[str],
                                    stay_days: List[int] = None,
                                    generate_params=None) -> Dict[str, Any]:
        """
        根据城市景点热度生成热度排除约束，按百分位动态划分
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定要排除的热度类型
            
        Returns:
            包含热度排除约束和候选景点的字典
        """
        # 复用include_popularity的逻辑来获取热度分类
        include_result = self.generate_include_popularity(city_list, None)
        
        if not include_result.get('all_labels_and_ranges'):
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 重新计算热度分类（因为include_result的validation_params现在只是range）
        available_attractions = []
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                heat_score = attraction.get('heatScore', 0)
                available_attractions.append({
                    'attraction': attraction,
                    'heat_score': heat_score
                })
        
        # 按热度分数排序（降序）
        available_attractions.sort(key=lambda x: x['heat_score'], reverse=True)
        
        # 计算百分位分界点
        total_count = len(available_attractions)
        top25_count = max(1, int(total_count * 0.25))
        top50_count = max(1, int(total_count * 0.50))
        
        if total_count == 1:
            top25_threshold = available_attractions[0]['heat_score']
            top50_threshold = available_attractions[0]['heat_score']
        else:
            top25_threshold = available_attractions[min(top25_count-1, total_count-1)]['heat_score']
            top50_threshold = available_attractions[min(top50_count-1, total_count-1)]['heat_score']
        
        popularity_categories = {
            "hot and must-visit": (top25_threshold, INF),
            "hidden and less crowded": (0, top50_threshold)
        }
        
        # 选择要使用的排除热度类别
        selected_label = None
        excluded_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"hot and must-visit": [5.7, inf]}
            # 遍历popularity_categories，找到在generate_params中的label
            for category, heat_range_item in popularity_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值，如果不存在则使用默认值
                    excluded_range = generate_params[category]
                    # 确保excluded_range是元组格式
                    if isinstance(excluded_range, list):
                        excluded_range = tuple(excluded_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                excluded_range = generate_params[selected_label]
                if isinstance(excluded_range, list):
                    excluded_range = tuple(excluded_range)
        else:
            selected_label = random.choice(list(popularity_categories.keys()))
            excluded_range = popularity_categories[selected_label]
        
        # 构建所有可能的排除解（排除当前选择的类别）
        all_solutions = {}
        for category, heat_range_item in popularity_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = heat_range_item
        
        # 筛选不符合排除热度条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_heat, max_heat = excluded_range
        
        for attr_info in available_attractions:
            attraction = attr_info['attraction']
            heat_score = attr_info['heat_score']
            
            # 如果景点热度不在排除范围内，则保留
            # 处理 INF 或 float('inf') 的情况
            is_inf = (max_heat >= INF) or (isinstance(max_heat, float) and math.isinf(max_heat))
            if not (min_heat <= heat_score < max_heat or (is_inf and heat_score >= min_heat)):
                matching_attraction_ids.append(attraction.get('poiId'))
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": excluded_range,  # 只存储排除的热度范围元组
            "candidate_ids": matching_attraction_ids,  # 只包含不在排除范围内的景点
            "candidate_product_ids": matching_ticket_ids,
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_exclude_popularity")
    def validate_exclude_popularity(self, attraction_info, validation_params):
        """
        验证景点信息是否不包含指定的排除热度类别
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 要排除的热度范围元组 (min_heat, max_heat)
            
        Returns:
            bool: 是否满足约束条件（不包含排除的热度类别）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return True  # 如果没有排除热度，则总是满足条件
        
        min_heat, max_heat = validation_params
        
        # 检查每个景点信息
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    heat_score = attraction.get('heatScore', 0)
                    # 如果发现任何在排除热度范围内的景点，则不满足约束
                    # 处理 INF 或 float('inf') 的情况
                    is_inf = (max_heat >= INF) or (isinstance(max_heat, float) and math.isinf(max_heat))
                    if min_heat <= heat_score < max_heat or (is_inf and heat_score >= min_heat):
                        return False
        
        # 如果没有发现任何在排除热度范围内的景点，则满足约束
        return True


    ################################################
    ########### include_comment_score ########
    ################################################

    @register_func("generate_include_comment_score")
    def generate_include_comment_score(self,
                                       city_list: List[str],
                                       stay_days: List[int] = None,
                                       generate_params=None) -> Dict[str, Any]:
        """
        根据城市景点评论分数生成评分包含约束，按百分位动态划分
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定评分类型
            
        Returns:
            包含评分约束和候选景点的字典
        """
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                comment_score = attraction.get('commentScore', 0)
                available_attractions.append({
                    'attraction': attraction,
                    'comment_score': comment_score
                })
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 按评论分数排序（降序）
        available_attractions.sort(key=lambda x: x['comment_score'], reverse=True)
        
        # 计算百分位分界点
        total_count = len(available_attractions)
        top25_count = max(1, int(total_count * 0.25))  # 前25%
        top50_count = max(1, int(total_count * 0.50))  # 前50%
        
        # 获取分界点的评论分数
        if total_count == 1:
            # 只有一个景点的情况
            top25_threshold = available_attractions[0]['comment_score']
            top50_threshold = available_attractions[0]['comment_score']
        else:
            top25_threshold = available_attractions[min(top25_count-1, total_count-1)]['comment_score']
            top50_threshold = available_attractions[min(top50_count-1, total_count-1)]['comment_score']
        
        # 定义评分类别及其范围
        rating_categories = {
            "highly rated": (top25_threshold, INF),      # 前25%
            "well rated": (top50_threshold, INF)     
        }
        
        # 选择要使用的评分类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        score_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"highly rated": [4.5, inf]}
            # 遍历rating_categories，找到在generate_params中的label
            for category, score_range_item in rating_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    score_range = generate_params[category]
                    # 确保score_range是元组格式
                    if isinstance(score_range, list):
                        score_range = tuple(score_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                score_range = generate_params[selected_label]
                if isinstance(score_range, list):
                    score_range = tuple(score_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(rating_categories.keys()))
            score_range = rating_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, score_range_item in rating_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = score_range_item
        
        # 筛选符合评分条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_score, max_score = score_range
        
        for attr_info in available_attractions:
            attraction = attr_info['attraction']
            comment_score = attr_info['comment_score']
            
            # 判断是否在评分范围内
            # 处理 INF 或 float('inf') 的情况
            is_inf = (max_score >= INF) or (isinstance(max_score, float) and math.isinf(max_score))
            if min_score <= comment_score < max_score or (is_inf and comment_score >= min_score):
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": score_range,  # 只存储评分范围元组
            "candidate_ids": matching_attraction_ids,  # 只包含符合评分条件的景点
            "candidate_product_ids": matching_ticket_ids,
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_include_comment_score")
    def validate_include_comment_score(self, attraction_info, validation_params):
        """
        验证景点信息是否符合指定评分类别
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 评分范围元组 (min_score, max_score)
            
        Returns:
            bool: 是否满足约束条件（全部景点都必须符合条件）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return False
        
        min_score, max_score = validation_params
        
        # 检查每个景点信息 - 必须全部符合条件
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    comment_score = attraction.get('commentScore', 0)
                    # 如果任何一个景点不在范围内，则不满足约束
                    # 处理 INF 或 float('inf') 的情况
                    is_inf = (max_score >= INF) or (isinstance(max_score, float) and math.isinf(max_score))
                    if not (min_score <= comment_score < max_score or (is_inf and comment_score >= min_score)):
                        return False
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都符合条件


    ################################################
    ########### exclude_comment_score ########
    ################################################

    @register_func("generate_exclude_comment_score")
    def generate_exclude_comment_score(self,
                                       city_list: List[str],
                                       stay_days: List[int] = None,
                                       generate_params=None) -> Dict[str, Any]:
        """
        根据城市景点评论分数生成评分排除约束，排除低分景点
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数（固定为"low rated"）
            
        Returns:
            包含评分排除约束和候选景点的字典
        """
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                comment_score = attraction.get('commentScore', 0)
                available_attractions.append({
                    'attraction': attraction,
                    'comment_score': comment_score
                })
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 按评论分数排序（升序，找最低的30%）
        available_attractions.sort(key=lambda x: x['comment_score'])
        
        # 计算底部30%的分界点
        total_count = len(available_attractions)
        bottom30_count = max(1, int(total_count * 0.30))  # 底部30%
        
        # 获取底部30%的最高分数作为排除阈值
        if total_count == 1:
            # 只有一个景点的情况
            bottom30_threshold = available_attractions[0]['comment_score']
        else:
            bottom30_threshold = available_attractions[min(bottom30_count-1, total_count-1)]['comment_score']
        
        # 固定的排除类别：低评分景点
        selected_label = "low rated"
        excluded_range = (0, bottom30_threshold - 0.01)  # 包含底部30%的范围，稍微加一点确保包含边界值
        
        # 筛选不在排除范围内的景点（即非低评分景点）
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_score, max_score = excluded_range
        
        for attr_info in available_attractions:
            attraction = attr_info['attraction']
            comment_score = attr_info['comment_score']
            
            # 如果景点评分不在排除范围内（即不是低评分），则保留
            if not (min_score <= comment_score < max_score):
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": excluded_range,  # 存储排除的评分范围元组
            "candidate_ids": matching_attraction_ids,  # 只包含非低评分的景点
            "candidate_product_ids": matching_ticket_ids,
            "all_labels_and_ranges": {selected_label: excluded_range}  # 没有其他选项，所以为空
        }

    @register_func("validate_exclude_comment_score")
    def validate_exclude_comment_score(self, attraction_info, validation_params):
        """
        验证景点信息是否不包含指定的排除评分类别（低评分）
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 要排除的评分范围元组 (min_score, max_score)
            
        Returns:
            bool: 是否满足约束条件（不包含低评分景点）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return True  # 如果没有排除评分，则总是满足条件
        
        min_score, max_score = validation_params
        
        # 检查每个景点信息
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    comment_score = attraction.get('commentScore', 0)
                    # 如果发现任何在排除评分范围内的景点（低评分），则不满足约束
                    if min_score <= comment_score < max_score:
                        return False
        
        # 如果没有发现任何低评分景点，则满足约束
        return True
    

    ################################################
    ########### include_free_attractions ########
    ################################################

    @register_func("generate_include_free_attractions")
    def generate_include_free_attractions(self,
                                        city_list: List[str],
                                        stay_days: List[int] = None,
                                        generate_params=None) -> Dict[str, Any]:
        """
        生成包含免费景点的约束

        Args:
            city_list: 城市列表
            generate_params: 生成参数（固定为"free"）

        Returns:
            包含免费景点约束和候选景点的字典
        """
        free_attractions = []
        all_attractions_in_cities = []

        # 筛选目标城市中的免费景点
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                all_attractions_in_cities.append(attraction)
                price = attraction.get('price', 0)
                if price == 0 or price == 0.0:  # 判断是否为免费景点
                    free_attractions.append(attraction)

        # 固定类别：免费景点
        selected_label = "free"

        matching_attraction_ids = []
        matching_ticket_ids = []

        # 如果存在免费景点，收集其 ID 和免费 ticket 产品
        for attraction in free_attractions:
            matching_attraction_ids.append(attraction.get('poiId'))
            ticket_products = attraction.get('ticket_products', [])
            for ticket in ticket_products:
                product_id = ticket.get('product_id')
                if product_id:
                    ticket_price = ticket.get('price', 0)
                    if ticket_price == 0 or ticket_price == 0.0:
                        matching_ticket_ids.append(product_id)

        # ⬇⬇⬇ 关键修复：无论是否找到免费景点，都固定返回 free label
        return {
            "selected_description": selected_label,
            "validation_params": selected_label,
            "candidate_ids": matching_attraction_ids,        # 可能为空，但结构不会消失
            "candidate_product_ids": matching_ticket_ids,    # 可能为空，但结构不会消失
            "all_labels_and_ranges": {selected_label: "free"}    # 不再为空对象
        }

    @register_func("validate_include_free_attractions")
    def validate_include_free_attractions(self, attraction_info, validation_params):
        """
        验证景点信息是否包含免费景点
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 验证参数（应该是"free"）
            
        Returns:
            bool: 是否满足约束条件（全部景点都必须是免费的）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params or validation_params != "free":
            return False
        
        # 检查是否全部景点都是免费的
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    price = attraction.get('price', 0)
                    # 如果发现任何收费景点，则不满足约束
                    if not (price == 0 or price == 0.0):
                        return False
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都是免费的


    ################################################
    ########### less_than_certain_price ########
    ################################################

    @register_func("generate_less_than_certain_price")
    def generate_less_than_certain_price(self,
                                         city_list: List[str],
                                         stay_days: List[int] = None,
                                         generate_params=None) -> Dict[str, Any]:
        """
        生成价格小于特定值的约束
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定价格类型
            
        Returns:
            包含价格约束和候选景点的字典
        """
        # 定义价格类别及其范围
        price_categories = {
            'less than 30': [0, 30],
            'less than 60': [0, 60],
            'less than 100': [0, 100],
        }
        
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 选择要使用的价格类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        price_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"less than 100": [0, 100]}
            # 遍历price_categories，找到在generate_params中的label
            for category, price_range_item in price_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    price_range = generate_params[category]
                    # 确保price_range是列表格式
                    if isinstance(price_range, tuple):
                        price_range = list(price_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                price_range = generate_params[selected_label]
                if isinstance(price_range, tuple):
                    price_range = list(price_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(price_categories.keys()))
            price_range = price_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, price_range_item in price_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = price_range_item
        
        # 筛选符合价格条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_price, max_price = price_range
        
        for attraction in available_attractions:
            price = attraction.get('price', 0)
            
            # 判断是否在价格范围内
            if min_price <= price < max_price:
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有符合价格条件的 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    ticket_price = ticket.get('price', 0)
                    if product_id and min_price <= ticket_price < max_price:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": price_range,  # 存储价格范围
            "candidate_ids": matching_attraction_ids,  # 只包含符合价格条件的景点
            "candidate_product_ids": matching_ticket_ids,  # 只包含符合价格条件的票务产品
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_less_than_certain_price")
    def validate_less_than_certain_price(self, attraction_info, validation_params):
        """
        验证景点信息是否符合指定价格范围
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 价格范围 [min_price, max_price]
            
        Returns:
            bool: 是否满足约束条件
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params or len(validation_params) != 2:
            return False
        
        min_price, max_price = validation_params
        
        # 检查每个景点信息 - 必须全部符合条件
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    price = attraction.get('price', 0)
                    # 如果任何一个景点不在价格范围内，则不满足约束
                    if not (min_price <= price < max_price):
                        return False
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都符合条件
    
    ################################################
    ########### within_certain_distance ########
    ################################################

    @register_func("generate_within_certain_distance")
    def generate_within_certain_distance(self,
                                         city_list: List[str],
                                         stay_days: List[int] = None,
                                         generate_params=None) -> Dict[str, Any]:
        """
        生成距离市中心距离约束
        
        Args:
            city_list: 城市列表
            stay_days: 停留天数（此函数不使用，但保持接口一致性）
            generate_params: 生成参数，指定距离类型
            
        Returns:
            包含距离约束和候选景点的字典
        """
        # 定义距离类别及其范围
        distance_categories = {
            '20': [0, 20],
            '30': [0, 30],
            '40': [0, 40],
            '50': [0, 50],
        }
        
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 选择要使用的距离类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        distance_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"within 20 km from city center": [0, 20]}
            # 遍历distance_categories，找到在generate_params中的label
            for category, distance_range_item in distance_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    distance_range = generate_params[category]
                    # 确保distance_range是列表格式
                    if isinstance(distance_range, tuple):
                        distance_range = list(distance_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                distance_range = generate_params[selected_label]
                if isinstance(distance_range, tuple):
                    distance_range = list(distance_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(distance_categories.keys()))
            distance_range = distance_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, distance_range_item in distance_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = distance_range_item
        
        # 根据距离市中心筛选符合条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_distance, max_distance = distance_range
        
        for attraction in available_attractions:
            city = attraction.get('city')
            if city not in self.city_center_coords:
                continue  # 如果城市不在市中心坐标数据中，跳过
            
            # 获取景点坐标
            attr_lat = attraction.get('latitude')
            attr_lon = attraction.get('longitude')
            if attr_lat is None or attr_lon is None:
                continue  # 如果景点没有坐标信息，跳过
            
            # 获取市中心坐标
            city_center = self.city_center_coords[city]
            city_center_coord = (city_center['lat'], city_center['lon'])
            attr_coord = (attr_lat, attr_lon)
            
            # 计算距离
            distance = self.calculate_distance(city_center_coord, attr_coord)
            
            # 检查距离是否在范围内
            if min_distance <= distance <= max_distance:
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": distance_range,  # 存储距离范围
            "candidate_ids": matching_attraction_ids,  # 符合条件的景点ID
            "candidate_product_ids": matching_ticket_ids,  # 符合条件的票务产品
            "all_labels_and_ranges": all_solutions
        }


    # @register_func("validate_within_certain_distance")
    # def validate_within_certain_distance(self, attraction_info, validation_params, coordination: tuple = None):
    #     """
    #     验证景点信息是否符合指定距离范围
        
    #     Args:
    #         attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
    #         validation_params: 距离范围 [min_distance, max_distance]
    #         coordination: 参考点坐标（通常是酒店坐标）(latitude, longitude)
            
    #     Returns:
    #         bool: 是否满足约束条件
    #     """
    #     # 处理输入：既可能是单个dict，也可能是dict的列表
    #     attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
    #     if not validation_params or len(validation_params) != 2:
    #         return False
        
    #     if coordination is None:
    #         raise ValueError("需要提供参考点的经纬度数据")
        
    #     min_distance, max_distance = validation_params
        
    #     # 检查每个景点信息
    #     for info in attraction_infos:
    #         attraction_id = info.get('id') or info.get('poiId')
    #         if attraction_id:
    #             attraction = self.get_attraction_by_id(attraction_id)
    #             if attraction:
    #                 # 获取景点坐标
    #                 attr_coord = (attraction.get('latitude'), attraction.get('longitude'))
                    
    #                 # 检查坐标是否完整
    #                 if None in attr_coord:
    #                     return False  # 如果景点没有坐标信息，则不满足约束
                    
    #                 # 计算距离
    #                 distance = self.calculate_distance(coordination, attr_coord)
                    
    #                 # 检查距离是否在范围内
    #                 if min_distance <= distance <= max_distance:
    #                     return True  # 至少有一个景点符合条件就返回True
        
    #     return False

    @register_func("validate_within_certain_distance")
    def validate_within_certain_distance(self, attraction_info, validation_params):
        """
        验证景点信息是否符合距离市中心指定距离范围
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 距离范围 [min_distance, max_distance]
            
        Returns:
            bool: 是否满足约束条件（全部景点都必须符合条件）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params or len(validation_params) != 2:
            return False
        
        min_distance, max_distance = validation_params
        
        # 检查每个景点信息 - 必须全部符合条件
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    city = attraction.get('city')
                    if city not in self.city_center_coords:
                        return False  # 如果城市不在市中心坐标数据中，不满足约束
                    
                    # 获取景点坐标
                    attr_lat = attraction.get('latitude')
                    attr_lon = attraction.get('longitude')
                    if attr_lat is None or attr_lon is None:
                        return False  # 如果景点没有坐标信息，不满足约束
                    
                    # 获取市中心坐标
                    city_center = self.city_center_coords[city]
                    city_center_coord = (city_center['lat'], city_center['lon'])
                    attr_coord = (attr_lat, attr_lon)
                    
                    # 计算距离
                    distance = self.calculate_distance(city_center_coord, attr_coord)
                    
                    # 检查距离是否在范围内
                    if not (min_distance <= distance <= max_distance+10):
                        return False  # 如果任何一个景点不在范围内，则不满足约束
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都符合条件

    @register_func("generate_within_certain_distance_from_city_center")
    def generate_within_certain_distance_from_city_center(self,
                                                         city_list: List[str],
                                                         stay_days: List[int] = None,
                                                         generate_params=None) -> Dict[str, Any]:
        """
        生成距离市中心距离约束
        
        Args:
            city_list: 城市列表
            stay_days: 停留天数（此函数不使用，但保持接口一致性）
            generate_params: 生成参数，指定距离类型
            
        Returns:
            包含距离约束和候选景点的字典
        """
        # 定义距离类别及其范围
        distance_categories = {
            '20': [0, 20],
            '30': [0, 30],
            '40': [0, 40],
            '50': [0, 50],
        }
        
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 选择要使用的距离类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        distance_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"within 20 km from city center": [0, 20]}
            # 遍历distance_categories，找到在generate_params中的label
            for category, distance_range_item in distance_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    distance_range = generate_params[category]
                    # 确保distance_range是列表格式
                    if isinstance(distance_range, tuple):
                        distance_range = list(distance_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                distance_range = generate_params[selected_label]
                if isinstance(distance_range, tuple):
                    distance_range = list(distance_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(distance_categories.keys()))
            distance_range = distance_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, distance_range_item in distance_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = distance_range_item
        
        # 根据距离市中心筛选符合条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_distance, max_distance = distance_range
        
        for attraction in available_attractions:
            city = attraction.get('city')
            if city not in self.city_center_coords:
                continue  # 如果城市不在市中心坐标数据中，跳过
            
            # 获取景点坐标
            attr_lat = attraction.get('latitude')
            attr_lon = attraction.get('longitude')
            if attr_lat is None or attr_lon is None:
                continue  # 如果景点没有坐标信息，跳过
            
            # 获取市中心坐标
            city_center = self.city_center_coords[city]
            city_center_coord = (city_center['lat'], city_center['lon'])
            attr_coord = (attr_lat, attr_lon)
            
            # 计算距离
            distance = self.calculate_distance(city_center_coord, attr_coord)
            
            # 检查距离是否在范围内
            if min_distance <= distance <= max_distance:
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": distance_range,  # 存储距离范围
            "candidate_ids": matching_attraction_ids,  # 符合条件的景点ID
            "candidate_product_ids": matching_ticket_ids,  # 符合条件的票务产品
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_within_certain_distance_from_city_center")
    def validate_within_certain_distance_from_city_center(self, attraction_info, validation_params):
        """
        验证景点信息是否符合距离市中心指定距离范围
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 距离范围 [min_distance, max_distance]
            
        Returns:
            bool: 是否满足约束条件（全部景点都必须符合条件）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params or len(validation_params) != 2:
            return False
        
        min_distance, max_distance = validation_params
        
        # 检查每个景点信息 - 必须全部符合条件
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    city = attraction.get('city')
                    if city not in self.city_center_coords:
                        return False  # 如果城市不在市中心坐标数据中，不满足约束
                    
                    # 获取景点坐标
                    attr_lat = attraction.get('latitude')
                    attr_lon = attraction.get('longitude')
                    if attr_lat is None or attr_lon is None:
                        return False  # 如果景点没有坐标信息，不满足约束
                    
                    # 获取市中心坐标
                    city_center = self.city_center_coords[city]
                    city_center_coord = (city_center['lat'], city_center['lon'])
                    attr_coord = (attr_lat, attr_lon)
                    
                    # 计算距离
                    distance = self.calculate_distance(city_center_coord, attr_coord)
                    
                    # 检查距离是否在范围内
                    if not (min_distance <= distance <= max_distance):
                        return False  # 如果任何一个景点不在范围内，则不满足约束
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都符合条件


    ################################################
    ########### prioritize_category ########
    ################################################

    @register_func("generate_prioritize_category")
    def generate_prioritize_category(self,
                                     city_list: List[str],
                                     stay_days: List[int],
                                     generate_params=None) -> Dict[str, Any]:
        """
        根据停留天数生成类别优先级约束，返回列表中的顺序代表优先级
        
        Args:
            city_list: 城市列表
            stay_days: 每个城市的停留天数列表
            generate_params: 生成参数，指定优先级类别列表
            
        Returns:
            包含优先级类别和候选景点的字典
        """
        # 确定类别数量范围
        max_days = max(stay_days) if stay_days else 0
        if 2 <= max_days <= 3:
            category_count_range = (1, 1)
        elif 4 <= max_days <= 5:
            category_count_range = (1, 2)
        elif max_days >= 6:
            category_count_range = (1, 3)
        else:
            category_count_range = (1, 1)  # 默认
            
        # 筛选出在目标城市中存在的景点类别
        available_categories = set()
        candidate_attraction_ids = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                categories = attraction.get('categories', [])
                available_categories.update(categories)
                candidate_attraction_ids.append(attraction.get('poiId'))
        
        available_categories = list(available_categories)
        
        if not available_categories:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 生成10个解
        all_solutions = {}
        seen_content = set()  # 用于跟踪已生成的内容，避免重复
        
        for i in range(10):
            # 随机选择类别数量
            k = random.randint(category_count_range[0], category_count_range[1])
            k = min(k, len(available_categories))  # 确保不超过可用类别数
            
            # 随机选择k个类别
            selected_categories = random.sample(available_categories, k)
            
            # 验证是否有解：检查是否存在包含这些类别的景点
            has_solution = False
            for attraction in self.attractions:
                if attraction.get('city') in city_list:
                    attraction_categories = set(attraction.get('categories', []))
                    if attraction_categories.intersection(selected_categories):
                        has_solution = True
                        break
            
            if not has_solution:
                continue  # 如果没有解，跳过这个组合
            
            # 使用排序后的元组作为唯一标识，避免不同顺序的重复
            content_key = tuple(sorted(selected_categories))
            
            # 如果内容已存在，跳过这个解
            if content_key in seen_content:
                continue
            
            # 标记为已见过
            seen_content.add(content_key)
            
            # 创建标签（按排序后的顺序，确保相同内容生成相同label）
            sorted_categories = sorted(selected_categories)
            if len(sorted_categories) == 1:
                label = f"1. {sorted_categories[0]}"
            else:
                label = ', '.join([f"{i+1}. {cat}" for i, cat in enumerate(sorted_categories)])
            
            # 由于内容已保证不重复，label也应该是唯一的
            all_solutions[label] = sorted_categories
        
        # 如果没有生成任何有效解，生成一个单类别的解
        if not all_solutions and available_categories:
            # 随机选择一个类别作为默认解
            default_category = random.choice(available_categories)
            label = f"1. {default_category}"
            all_solutions[label] = [default_category]
        
        # 选择要使用的类别：如果有generate_params则使用它，否则随机选择一个解
        if generate_params:
            # 处理字典格式的generate_params：{selected_description: validation_params}
            if isinstance(generate_params, dict):
                # 如果是字典，提取值（validation_params，应该是列表）
                if generate_params:
                    selected_categories = list(generate_params.values())[0]
                else:
                    selected_categories = None
            else:
                # 如果不是字典，直接使用（可能是列表）
                selected_categories = generate_params
            
            # 对类别进行排序，保持一致性
            if selected_categories:
                selected_categories = sorted(selected_categories)
            
            if selected_categories and len(selected_categories) == 1:
                selected_label = f"1. {selected_categories[0]}"
            elif selected_categories:
                selected_label = ', '.join([f"{i+1}. {cat}" for i, cat in enumerate(selected_categories)])
            else:
                selected_label = None
        elif all_solutions:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(all_solutions.keys()))
            selected_categories = all_solutions[selected_label]
        else:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
            
        # 筛选包含选中类别的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                attraction_categories = set(attraction.get('categories', []))
                if attraction_categories.intersection(selected_categories):
                    matching_attraction_ids.append(attraction.get('poiId'))
                    # 收集该景点的所有 product_id
                    ticket_products = attraction.get('ticket_products', [])
                    for ticket in ticket_products:
                        product_id = ticket.get('product_id')
                        if product_id:
                            matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": selected_categories,  # 保持原有顺序作为优先级
            "candidate_ids": matching_attraction_ids,
            "candidate_product_ids": matching_ticket_ids,  # 景点的票务产品ID
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_prioritize_category")
    def validate_prioritize_category(self, attraction_info, validation_params):
        """
        验证景点信息是否符合类别优先级约束
        优先级评估规则：
        1. 优先级高的（constraint中index靠前）类别，需要在attractions_list中出现更多次数
        2. 优先级列表中的所有类别都必须在attractions_list中至少出现一次
        3. 优先级高的类别对应的景点总停留时间应该 >= 优先级低的类别对应的景点总停留时间
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含poiId和visiting_time（计划游玩时间，分钟为单位）
            validation_params: 优先级类别列表，顺序代表优先级
            
        Returns:
            bool: 是否满足约束条件
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params:
            return False
        
        priority_categories = validation_params
        
        # 统计每个优先级类别在景点列表中的出现次数
        constraint_hit_num = {cat: 0 for cat in priority_categories}
        # 统计每个优先级类别对应的景点总停留时间（分钟）
        constraint_total_time = {cat: 0.0 for cat in priority_categories}
        
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    # 获取计划游玩时间（从attraction_info中获取，而不是从景点的reference_time）
                    visiting_time = info.get('visiting_time', 0)
                    # 如果没有提供visiting_time，默认为0
                    if visiting_time is None:
                        visiting_time = 0
                    visiting_time = float(visiting_time)
                    
                    attraction_categories = attraction.get('categories', [])
                    for cat in attraction_categories:
                        if cat in constraint_hit_num:
                            constraint_hit_num[cat] += 1
                            constraint_total_time[cat] += visiting_time
        
        # 检查所有优先级类别是否都至少出现一次
        for category in priority_categories:
            if constraint_hit_num[category] == 0:
                return False
        
        # 检查优先级顺序是否正确：优先级高的类别出现次数应该 >= 优先级低的类别
        for i in range(len(priority_categories) - 1):
            if constraint_hit_num[priority_categories[i]] < constraint_hit_num[priority_categories[i + 1]]:
                return False
        
        # 检查停留时间优先级顺序：优先级高的类别总停留时间应该 >= 优先级低的类别总停留时间
        for i in range(len(priority_categories) - 1):
            if constraint_total_time[priority_categories[i]] < constraint_total_time[priority_categories[i + 1]]:
                return False
        
        return True
    

    ################################################
    ########### include_comment_count ########
    ################################################

    @register_func("generate_include_comment_count")
    def generate_include_comment_count(self,
                                       city_list: List[str],
                                       stay_days: List[int] = None,
                                       generate_params=None) -> Dict[str, Any]:
        """
        生成评论数量大于特定值的约束
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定评论数量类型
            
        Returns:
            包含评论数量约束和候选景点的字典
        """
        # 定义评论数量类别及其范围
        comment_count_categories = {
            "more than 10 comments": [10, INF],
            "more than 50 comments": [50, INF],
            "more than 100 comments": [100, INF],
            "more than 300 comments": [300, INF],
            "more than 500 comments": [500, INF],
        }
        
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 选择要使用的评论数量类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        comment_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"more than 500 comments": [500, inf]}
            # 遍历comment_count_categories，找到在generate_params中的label
            for category, comment_range_item in comment_count_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    comment_range = generate_params[category]
                    # 确保comment_range是列表格式
                    if isinstance(comment_range, tuple):
                        comment_range = list(comment_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                comment_range = generate_params[selected_label]
                if isinstance(comment_range, tuple):
                    comment_range = list(comment_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(comment_count_categories.keys()))
            comment_range = comment_count_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, comment_range_item in comment_count_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = comment_range_item
        
        # 筛选符合评论数量条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_count, max_count = comment_range
        
        for attraction in available_attractions:
            comment_count = attraction.get('commentCount', 0)
            
            # 判断是否在评论数量范围内
            # 处理 INF 或 float('inf') 的情况
            is_inf = (max_count >= INF) or (isinstance(max_count, float) and math.isinf(max_count))
            if min_count <= comment_count < max_count or (is_inf and comment_count >= min_count):
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": comment_range,  # 存储评论数量范围
            "candidate_ids": matching_attraction_ids,  # 只包含符合评论数量条件的景点
            "candidate_product_ids": matching_ticket_ids,  # 符合条件的票务产品
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_include_comment_count")
    def validate_include_comment_count(self, attraction_info, validation_params):
        """
        验证景点信息是否符合指定评论数量范围
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 评论数量范围 [min_count, max_count]
            
        Returns:
            bool: 是否满足约束条件
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params or len(validation_params) != 2:
            return False
        
        min_count, max_count = validation_params
        
        # 检查每个景点信息 - 必须全部符合条件
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    comment_count = attraction.get('commentCount', 0)
                    # 如果任何一个景点不在评论数量范围内，则不满足约束
                    # 处理 INF 或 float('inf') 的情况
                    is_inf = (max_count >= INF) or (isinstance(max_count, float) and math.isinf(max_count))
                    if not (min_count <= comment_count < max_count or (is_inf and comment_count >= min_count)):
                        return False
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点都符合条件


    ################################################
    ########### exclude_comment_count ########
    ################################################

    @register_func("generate_exclude_comment_count")
    def generate_exclude_comment_count(self,
                                      city_list: List[str],
                                      stay_days: List[int] = None,
                                      generate_params=None) -> Dict[str, Any]:
        """
        生成评论数量排除约束，排除评论数量低于特定值的景点
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定评论数量类型
            
        Returns:
            包含评论数量排除约束和候选景点的字典
        """
        # 定义评论数量类别及其范围（低于此值的景点将被排除）
        comment_count_categories = {
            "fewer than 10 comments": [0, 10],
            "fewer than 50 comments": [0, 50],
            "fewer than 100 comments": [0, 100],
            "fewer than 300 comments": [0, 300],
            "fewer than 500 comments": [0, 500],
        }
        
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 选择要使用的评论数量类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        excluded_range = None
        
        if generate_params:
            # generate_params是字典格式：{label: range}，如 {"fewer than 100 comments": [0, 100]}
            # 遍历comment_count_categories，找到在generate_params中的label
            for category, excluded_range_item in comment_count_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    excluded_range = generate_params[category]
                    # 确保excluded_range是列表格式
                    if isinstance(excluded_range, tuple):
                        excluded_range = list(excluded_range)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                excluded_range = generate_params[selected_label]
                if isinstance(excluded_range, tuple):
                    excluded_range = list(excluded_range)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(comment_count_categories.keys()))
            excluded_range = comment_count_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, excluded_range_item in comment_count_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = excluded_range_item
        
        # 筛选不在排除范围内的景点（即评论数量不在 [0, threshold) 范围内的景点）
        matching_attraction_ids = []
        matching_ticket_ids = []
        min_count, max_count = excluded_range
        
        for attraction in available_attractions:
            comment_count = attraction.get('commentCount', 0)
            
            # 判断是否不在排除范围内（即评论数量 >= max_count）
            if comment_count >= max_count:
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": excluded_range,  # 存储排除范围 [min_count, max_count]（评论数量在此范围内的景点将被排除）
            "candidate_ids": matching_attraction_ids,  # 只包含不在排除范围内的景点
            "candidate_product_ids": matching_ticket_ids,  # 符合条件的票务产品
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_exclude_comment_count")
    def validate_exclude_comment_count(self, attraction_info, validation_params):
        """
        验证景点信息是否不包含评论数量在指定排除范围内的景点
        
        Args:
            attraction_info: 可以是单个dict或dict的列表，每个dict包含id等信息
            validation_params: 排除范围 [min_count, max_count]（评论数量在此范围内的景点将被排除）
            
        Returns:
            bool: 是否满足约束条件（所有景点的评论数量都不在排除范围内）
        """
        # 处理输入：既可能是单个dict，也可能是dict的列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info
        
        if not validation_params or len(validation_params) != 2:
            return True  # 如果没有排除范围，则总是满足条件
        
        min_count, max_count = validation_params
        
        # 检查每个景点信息 - 必须全部不在排除范围内（即评论数量不在 [min_count, max_count) 范围内）
        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if attraction_id:
                attraction = self.get_attraction_by_id(attraction_id)
                if attraction:
                    comment_count = attraction.get('commentCount', 0)
                    # 如果发现任何评论数量在排除范围内的景点，则不满足约束
                    if min_count <= comment_count < max_count:
                        return False  # 如果任何一个景点在排除范围内，则不满足约束
                else:
                    return False  # 景点不存在，不满足约束
            else:
                return False  # 没有有效的景点ID，不满足约束
        
        return True  # 所有景点的评论数量都不在排除范围内


    ################################################
    ########### include_sight_level ########
    ################################################

    @register_func("generate_include_sight_level")
    def generate_include_sight_level(self,
                                     city_list: List[str],
                                     stay_days: List[int] = None,
                                     generate_params=None) -> Dict[str, Any]:
        """
        生成景点等级包含约束
        
        Args:
            city_list: 城市列表
            generate_params: 生成参数，指定景点等级类型
            
        Returns:
            包含景点等级约束和候选景点的字典
        """
        # 定义景点等级类别及其包含的等级
        sight_level_categories = {
            "5A": ["5A"],
            "at least 4A": ["4A", "5A"],
        }
        
        # 筛选出在目标城市中的景点
        available_attractions = []
        
        for attraction in self.attractions:
            if attraction.get('city') in city_list:
                available_attractions.append(attraction)
        
        if not available_attractions:
            return {
                "selected_description": [],
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {}
            }
        
        # 选择要使用的景点等级类别：如果有generate_params则使用它，否则随机选择一个解
        selected_label = None
        allowed_levels = None
        
        if generate_params:
            # generate_params是字典格式：{label: levels}，如 {"at least 4A": ["4A", "5A"]}
            # 遍历sight_level_categories，找到在generate_params中的label
            for category, levels_item in sight_level_categories.items():
                if category in generate_params:
                    selected_label = category
                    # 使用generate_params中对应的值
                    allowed_levels = generate_params[category]
                    # 确保allowed_levels是列表格式
                    if isinstance(allowed_levels, tuple):
                        allowed_levels = list(allowed_levels)
                    break
            # 如果没有找到匹配的label，使用第一个
            if selected_label is None and generate_params:
                selected_label = list(generate_params.keys())[0]
                allowed_levels = generate_params[selected_label]
                if isinstance(allowed_levels, tuple):
                    allowed_levels = list(allowed_levels)
        else:
            # 随机选择一个解作为主要结果
            selected_label = random.choice(list(sight_level_categories.keys()))
            allowed_levels = sight_level_categories[selected_label]
        
        # 构建所有可能的解（排除当前选择的类别）
        all_solutions = {}
        for category, levels in sight_level_categories.items():
            # 只包含非当前选择的类别
            # if category != selected_label:
            all_solutions[category] = levels
        
        # 筛选符合景点等级条件的景点
        matching_attraction_ids = []
        matching_ticket_ids = []
        
        for attraction in available_attractions:
            sight_level = attraction.get('sightLevelStr')
            
            # 判断是否在允许的等级范围内
            if sight_level in allowed_levels:
                matching_attraction_ids.append(attraction.get('poiId'))
                # 收集该景点的所有 product_id
                ticket_products = attraction.get('ticket_products', [])
                for ticket in ticket_products:
                    product_id = ticket.get('product_id')
                    if product_id:
                        matching_ticket_ids.append(product_id)
        
        return {
            "selected_description": selected_label,
            "validation_params": allowed_levels,  # 存储允许的景点等级列表
            "candidate_ids": matching_attraction_ids,  # 只包含符合景点等级条件的景点
            "candidate_product_ids": matching_ticket_ids,  # 符合条件的票务产品
            "all_labels_and_ranges": all_solutions
        }

    @register_func("validate_include_sight_level")
    def validate_include_sight_level(self, attraction_info, validation_params):
        """
        验证景点信息是否符合指定景点等级要求
        要求：至少有 1～2 个景点安排在指定等级（此版本默认至少 1 个）
        
        Args:
            attraction_info: dict 或 dict列表，每个包含 id 或 poiId
            validation_params: 允许的景点等级列表
            
        Returns:
            bool: 是否满足约束条件（至少一个符合条件）
        """
        # 将输入统一处理为列表
        attraction_infos = [attraction_info] if isinstance(attraction_info, dict) else attraction_info

        if not validation_params:
            return False

        allowed_levels = validation_params

        matching_count = 0

        for info in attraction_infos:
            attraction_id = info.get('id') or info.get('poiId')
            if not attraction_id:
                continue
            
            attraction = self.get_attraction_by_id(attraction_id)
            if not attraction:
                continue

            sight_level = attraction.get('sightLevelStr')
            if sight_level in allowed_levels:
                matching_count += 1

        # 至少需要 1 个景点命中（如需 2 个可改成 >= 2）
        return matching_count >= 1






if __name__ == "__main__":
    ae = AttractionEvaluator(json.load(open("", 'r', encoding='utf-8')))


    # 测试 generate_include_categories
    # city_list = ["Beijing", "Shanghai"]
    # stay_days = [4, 3]
    # generate_params = [
    #     "Ecology, Flora & Fauna Zones",
    #     "Theme Parks & Rides"
    # ]
    # results = ae.generate_include_categories(
    #     city_list=city_list,
    #     stay_days=stay_days,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)


    # 测试 generate_exclude_categories
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = [
    #     "Ecology, Flora & Fauna Zones",
    #     "Theme Parks & Rides"
    # ]
    # results = ae.generate_exclude_categories(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_include_attractions
    # city_list = ["Beijing", "Shanghai"]
    # stay_days = [4, 3]
    # generate_params = [
    #     75594
    # ]
    # results = ae.generate_include_attractions(
    #     city_list=city_list,
    #     stay_days=stay_days,
    #     # generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_exclude_attractions
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = [
    #     75594  # Tiananmen Square的poiId
    # ]
    # results = ae.generate_exclude_attractions(
    #     city_list=city_list,
    #     # generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_include_popularity
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "hot and must-visit"  # 可选: "moderately popular", "hidden and less crowded"
    # results = ae.generate_include_popularity(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_exclude_popularity
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "hot and must-visit"  # 可选: "moderately
    # results = ae.generate_exclude_popularity(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_include_comment_score
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "highly rated"  # 可选: "well rated"
    # results = ae.generate_include_comment_score(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_exclude_comment_score
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "low rated"
    # results = ae.generate_exclude_comment_score(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_include_free_attractions
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "free"
    # results = ae.generate_include_free_attractions(
    #     city_list=city_list,
    #     # generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_less_than_certain_price
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "less than 60"  # 可选: "less than 30", "less than 100"
    # results = ae.generate_less_than_certain_price(
    #     city_list=city_list,
    #     # generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_within_certain_distance
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "within 10 km"  # 可选: "within 5
    # results = ae.generate_within_certain_distance(
    #     city_list=city_list,
    #     # generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_within_certain_distance_from_city_center
    # city_list = ["Beijing", "Shanghai"]
    # results = ae.generate_within_certain_distance_from_city_center(
    #     city_list=city_list
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_prioritize_category
    # city_list = ["Beijing", "Shanghai"]
    # stay_days = [4, 3]
    # generate_params = [
    #     "Ecology, Flora & Fauna Zones",
    #     "Theme Parks & Rides"
    # ]
    # results = ae.generate_prioritize_category(
    #     city_list=city_list,
    #     stay_days=stay_days,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)


    # 测试 generate_include_comment_count
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "more than 300 comments"  # 可选: "more than
    # results = ae.generate_include_comment_count(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_exclude_comment_count
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "fewer than 100 comments"  # 可选: "fewer than 10/50/100/300/500 comments"
    # results = ae.generate_exclude_comment_count(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    # 测试 generate_include_sight_level
    # city_list = ["Beijing", "Shanghai"]
    # generate_params = "5A"  # 可选: "at least 4A"
    # results = ae.generate_include_sight_level(
    #     city_list=city_list,
    #     generate_params=generate_params
    # )
    # json.dump(results, open("./test.json", 'w', encoding='utf-8'), indent=4, ensure_ascii=False)

    
    # =====================================================
    # 测试所有 validate 函数
    # =====================================================
    
    # 测试 validate_include_categories
    test_attraction_info = [
        {"poiId": 77064},  
        {"poiId": 130886889}  
    ]
    validation_params = ["Historical & Cultural Heritage", "Shopping & Food Experiences"]
    result = ae.validate_include_categories(test_attraction_info, validation_params)
    print(f"validate_include_categories result: {result}")

    # 测试 validate_exclude_categories
    test_attraction_info = [
        {"poiId": 77064},  
        {"poiId": 130886889}  
    ]
    validation_params = ["Natural Scenery"]  # 排除的分类
    result = ae.validate_exclude_categories(test_attraction_info, validation_params)
    print(f"validate_exclude_categories result: {result}")

    # 测试 validate_include_attractions
    test_attraction_info = [
        {"poiId": 130886889},
        {"poiId": 77071}
    ]
    validation_params = [77064]  # 必须包含的景点ID
    result = ae.validate_include_attractions(test_attraction_info, validation_params)
    print(f"validate_include_attractions result: {result}")

    # 测试 validate_exclude_attractions
    test_attraction_info = [
        {"poiId": 77064},
        {"poiId": 99999}
    ]
    validation_params = [99999]  # 排除的景点ID（不存在于test_attraction_info中）
    result = ae.validate_exclude_attractions(test_attraction_info, validation_params)
    print(f"validate_exclude_attractions result: {result}")

    # 测试 validate_include_popularity
    test_attraction_info = [
        {"poiId": 77064},  # heatScore: 7.6
        {"poiId": 77071}   # heatScore: 7.2
    ]
    validation_params = (7.5, INF)  # 热度范围
    result = ae.validate_include_popularity(test_attraction_info, validation_params)
    print(f"validate_include_popularity result: {result}")

    # 测试 validate_exclude_popularity
    test_attraction_info = [
        {"poiId": 77064},  # heatScore: 7.6
        {"poiId": 77071}   # heatScore: 7.2
    ]
    validation_params = (7.5, INF)  # 排除的热度范围（高于test数据）
    result = ae.validate_exclude_popularity(test_attraction_info, validation_params)
    print(f"validate_exclude_popularity result: {result}")

    # 测试 validate_include_comment_score
    test_attraction_info = [
        {"poiId": 77064},  # commentScore: 4.6
        {"poiId": 77071}   # commentScore: 4.7
    ]
    validation_params = (4.7, INF)  # 评分范围
    result = ae.validate_include_comment_score(test_attraction_info, validation_params)
    print(f"validate_include_comment_score result: {result}")

    # 测试 validate_exclude_comment_score
    test_attraction_info = [
        {"poiId": 77064},  # commentScore: 4.6
        {"poiId": 77071}   # commentScore: 4.7
    ]
    validation_params = (0, 4.7)  # 排除的评分范围（低分）
    result = ae.validate_exclude_comment_score(test_attraction_info, validation_params)
    print(f"validate_exclude_comment_score result: {result}")

    # 测试 validate_include_free_attractions
    test_attraction_info = [
        {"poiId": 77071},  # price: 0.0 (免费)
        {"poiId": 77064}   # price: 20.0 (收费)
    ]
    validation_params = "free"
    result = ae.validate_include_free_attractions(test_attraction_info, validation_params)
    print(f"validate_include_free_attractions result: {result}")

    # 测试 validate_less_than_certain_price
    test_attraction_info = [
        {"poiId": 77064},  # price: 20.0
        {"poiId": 77071}   # price: 0.0
    ]
    validation_params = [0, 10]  # 价格范围：小于30
    result = ae.validate_less_than_certain_price(test_attraction_info, validation_params)
    print(f"validate_less_than_certain_price result: {result}")

    # 测试 validate_within_certain_distance
    test_attraction_info = [
        {"poiId": 75611},  # 外滩
        {"poiId": 75627}   # 假设有坐标
    ]
    validation_params = [0, 10]  # 10公里内
    coordination = (39.9042, 116.4074)  # 北京
    # coordination = (31.2304, 121.4737) # 上海
    result = ae.validate_within_certain_distance(test_attraction_info, validation_params, coordination)
    print(f"validate_within_certain_distance result: {result}")

    # 测试 generate_within_certain_distance_from_city_center
    city_list = ["Beijing", "Shanghai"]
    results = ae.generate_within_certain_distance_from_city_center(
        city_list=city_list
    )
    print(f"\ngenerate_within_certain_distance_from_city_center result:")
    print(f"Selected description: {results.get('selected_description')}")
    print(f"Validation params: {results.get('validation_params')}")
    print(f"Candidate IDs count: {len(results.get('candidate_ids', []))}")
    print(f"Candidate product IDs count: {len(results.get('candidate_product_ids', []))}")

    # 测试 validate_within_certain_distance_from_city_center
    test_attraction_info = [
        {"poiId": 77064},  # 哈尔滨圣索菲亚大教堂
        {"poiId": 77071}   # 假设有坐标的景点
    ]
    validation_params = [0, 20]  # 20公里内
    result = ae.validate_within_certain_distance_from_city_center(test_attraction_info, validation_params)
    print(f"\nvalidate_within_certain_distance_from_city_center result: {result}")

    # 测试 validate_prioritize_category
    test_attraction_info = [
        {"poiId": 99446, "visiting_time": 120},  # categories: ["Historical & Cultural Heritage", "Photo Spots & Instagrammable"]
        {"poiId": 77092, "visiting_time": 90},  # categories: ["Historical & Cultural Heritage", "Arts & Museums"]
        {"poiId": 78238, "visiting_time": 500}   # 只有 city sightseeing & landmarks
    ]
    validation_params = ["Historical & Cultural Heritage", "City Sightseeing & Landmarks"]  # 优先级顺序
    result = ae.validate_prioritize_category(test_attraction_info, validation_params)
    print(f"validate_prioritize_category result: {result}")

    # 测试 validate_include_comment_count
    test_attraction_info = [
        {"poiId": 77064},  # commentCount: 5356
        {"poiId": 77071}   # commentCount: 6905
    ]
    validation_params = [6900, INF]  # 评论数量范围：1000以上
    result = ae.validate_include_comment_count(test_attraction_info, validation_params)
    print(f"validate_include_comment_count result: {result}")

    # 测试 validate_exclude_comment_count
    test_attraction_info = [
        {"poiId": 77064},  # commentCount: 5356
        {"poiId": 78223}   # commentCount: 139
    ]
    validation_params = [0, 500]  # 排除范围：评论数量在 [0, 100) 范围内的景点将被排除
    result = ae.validate_exclude_comment_count(test_attraction_info, validation_params)
    print(f"validate_exclude_comment_count result: {result}")

    # 测试 validate_include_sight_level
    test_attraction_info = [
        {"poiId": 78221},  # sightLevelStr: 5a
        {"poiId": 18022433},  # sightLevelStr: 4a
        {"poiId": 18022433} 
    ]
    validation_params = ["5A"]  # 超过一半的景点应该是5A级景点
    result = ae.validate_include_sight_level(test_attraction_info, validation_params)
    print(f"validate_include_sight_level result: {result}")

