import json
import random
from inspect import signature

import math
from datetime import datetime, timedelta
from collections import defaultdict
from geopy.distance import geodesic, distance
import os
random.seed(1120)
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
    }
  }
CITY_CENTER_COORDS = {key.lower(): value for key, value in CITY_CENTER_COORDS.items()}

# ------------------------------
# 注册装饰器（定义在类外部）
# ------------------------------
def register_func(*names):
    """
    注册装饰器：可用于类方法，自动写入类属性 func_map。
    """
    def decorator(func):
        # 把注册名暂存在函数属性上，类创建后再填入 func_map
        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

class BaseHotelEvaluator:
    def __init__(self, hotels_path):
        """
        hotel_path: 酒店 JSON 文件路径
        """
        self.hotels_path = hotels_path
        with open(self.hotels_path, "r", encoding="utf-8") as f:
            self.hotels = json.load(f)
        
        self.city_dict = {}
        for hotel in self.hotels:
            hotel["all_product_id"] = [p["product_id"] for p in hotel.get("products", [])]
            city = self._norm_city(hotel["real_city"])
            if city not in self.city_dict:
                self.city_dict[city] = []
            self.city_dict[city].append(hotel)
            
        

    # ------------------------------
    # 通用工具函数
    # ------------------------------

    def get_product_from_product_id(self, product_id):
        for hotel in self.hotels:
            for product in hotel["products"]:
                if product_id == product["product_id"]:
                    return product
        return None
    
    def get_hotel_from_product_id(self, product_id):
        for hotel in self.hotels:
            for product in hotel["products"]:
                if product_id == product["product_id"]:
                    return hotel
        return None
    
    def get_hotel_from_hotel_id(self, hotel_id):
        for hotel in self.hotels:
            if hotel_id == hotel["Hotel_id"]:
                return hotel
        return None

    def get_total_date_from_product_list(self, product_list): 
        unique_dates = {item["date"] for item in product_list if item.get("date")}
        return len(unique_dates)

    def get_total_price_from_product_list(self, product_list):
        total = 0
        for daily in product_list:
            product = self.get_product_from_product_id(daily["product_id"])
            total += product[daily["date"]] * daily["room_num"]
        return total
    
    def get_stay_nights(self, in_date, out_date): # 入住时长不包括checkout那一天
        in_dt = datetime.strptime(in_date, "%Y-%m-%d")
        out_dt = datetime.strptime(out_date, "%Y-%m-%d")
        return (out_dt - in_dt).days
    
    def get_total_nights_from_dict(self, parsed):
        total = 0
        for item in parsed:
            total += self.get_stay_nights(item["in_date"], item["out_date"])
        return total

    def get_avg_price_hotel(self, in_date, out_date, hotel): # 入住时长不包括checkout那一天
        checkin_date = datetime.strptime(in_date, "%Y-%m-%d")
        checkout_date = datetime.strptime(out_date, "%Y-%m-%d")
        total_price = 0
        num_days = 0

        current_date = checkin_date
        while current_date < checkout_date:  # 不含 checkout
            weekday = current_date.strftime("%a")  # 得到 Mon, Tue, ...
            total_price += hotel[f"{weekday}_lowest_price"]
            num_days += 1
            current_date += timedelta(days=1)
        return total_price / num_days if num_days > 0 else 0


    def convert_to_date_dict(self, product_list):
        result = defaultdict(list)
        
        for item in product_list:
            date = item["date"]
            pid = item["product_id"]
            
            result[date].extend([pid] * int(item["room_num"]))
        
        return dict(result)

    def convert_weekstr_to_weekday(self, weekstr):
        weekday = datetime.strptime(weekstr, "%Y-%m-%d").strftime("%a")  # %a = Mon/Tue/Wed...
        return weekday
    
    
    def _norm_city(self, s: str) -> str:
        """统一城市字符串：去空格并小写"""
        return (s or "").strip().lower()

    def parse_input_dict(self, input_dict):
        if len(input_dict) == 1:
            d = input_dict[0]
            return [{
                "city": d["to"],
                "in_date": d["depart_date"],
                "out_date": d["return_date"],
                "person_num": d["number_of_people"],
            }]
        elif len(input_dict) == 2:
            d1, d2 = input_dict
            return [{
                "city": d1["to"],
                "in_date": d1["depart_date"],
                "out_date": d2["depart_date"],
                "person_num": d1["number_of_people"],
            },
            {
                "city": d2["to"],
                "in_date": d2["depart_date"],
                "out_date": d2["return_date"],
                "person_num": d2["number_of_people"],
            }]
        else:
            raise ValueError("输入数据格式错误")
        
    # ------------------------------
    # 执行入口
    # ------------------------------
    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)


@collect_registered_funcs
class HotelEvaluator(BaseHotelEvaluator):
    def __init__(self,hotel_path="data_final/hotels.json"):
        # 取出目录部分
        base_dir = os.path.dirname(hotel_path)

        # 在同目录下自动生成 train_path 和 flight_path
        train_path = os.path.join(base_dir, "trains.json")
        flight_path = os.path.join(base_dir, "flights.json")

        super().__init__(hotel_path)
        with open(train_path, "r") as f:
            self.trains = json.load(f)
        with open(flight_path, "r") as f:
            self.flights = json.load(f)

    """
    酒店价格计算与匹配系统
    """

    # ------------------------------
    # 注册的计算函数
    # ------------------------------

    def get_less(self, stay_nights, person_num):
        less = {
            f"{50 * stay_nights * person_num}": [0, 50 * stay_nights * person_num],
            f"{100 * stay_nights * person_num}": [0, 100 * stay_nights * person_num],
            f"{150 * stay_nights * person_num}": [0, 150 * stay_nights * person_num],
            f"{200 * stay_nights * person_num}": [0, 200 * stay_nights * person_num],
            f"{250 * stay_nights * person_num}": [0, 250 * stay_nights * person_num],
            f"{300 * stay_nights * person_num}": [0, 300 * stay_nights * person_num],
            f"{400 * stay_nights * person_num}": [0, 400 * stay_nights * person_num],
            f"{500 * stay_nights * person_num}": [0, 500 * stay_nights * person_num],
            f"{600 * stay_nights * person_num}": [0, 600 * stay_nights * person_num],
            f"{800 * stay_nights * person_num}": [0, 800 * stay_nights * person_num],
            f"{1000 * stay_nights * person_num}": [0, 1000 * stay_nights * person_num],
        }
        return less

    def get_more(self, stay_nights, person_num):
        more = {
            f"{50 * stay_nights * person_num}": [50 * stay_nights * person_num, 999999],
            f"{100 * stay_nights * person_num}": [100 * stay_nights * person_num, 999999],
            f"{150 * stay_nights * person_num}": [150 * stay_nights * person_num, 999999],
            f"{200 * stay_nights * person_num}": [200 * stay_nights * person_num, 999999],
            f"{250 * stay_nights * person_num}": [250 * stay_nights * person_num, 999999],
            f"{300 * stay_nights * person_num}": [300 * stay_nights * person_num, 999999],
            f"{400 * stay_nights * person_num}": [400 * stay_nights * person_num, 999999],
            f"{500 * stay_nights * person_num}": [500 * stay_nights * person_num, 999999],
            f"{600 * stay_nights * person_num}": [600 * stay_nights * person_num, 999999],
            f"{800 * stay_nights * person_num}": [800 * stay_nights * person_num, 999999],
            f"{1000 * stay_nights * person_num}": [1000 * stay_nights * person_num, 999999],
        }
        return more
    
    def get_around(self, stay_nights, person_num):
        around = {
            f"around {50 * stay_nights * person_num}": [0, 100 * stay_nights * person_num],
            f"around {100 * stay_nights * person_num}": [50 * stay_nights * person_num, 150 * stay_nights * person_num],
            f"around {150 * stay_nights * person_num}": [100 * stay_nights * person_num,200 * stay_nights * person_num],
            f"around {200 * stay_nights * person_num}": [150 * stay_nights * person_num,250 * stay_nights * person_num],
            f"around {250 * stay_nights * person_num}": [200 * stay_nights * person_num,300 * stay_nights * person_num],
            f"around {300 * stay_nights * person_num}": [200 * stay_nights * person_num,400 * stay_nights * person_num],
            f"around {400 * stay_nights * person_num}": [300 * stay_nights * person_num,500 * stay_nights * person_num],
            f"around {500 * stay_nights * person_num}": [350 * stay_nights * person_num,650 * stay_nights * person_num],
            f"around {600 * stay_nights * person_num}": [450 * stay_nights * person_num,750 * stay_nights * person_num],
            f"around {800 * stay_nights * person_num}": [600 * stay_nights * person_num,1000 * stay_nights * person_num],
            f"around {1000 * stay_nights * person_num}": [800 * stay_nights * person_num,1200 * stay_nights * person_num],
        }
        return around
    
    def get_between(self, stay_nights, person_num):
        between = {
            f"between {50 * stay_nights * person_num} and {150 * stay_nights * person_num}": [50* stay_nights * person_num,150* stay_nights * person_num],
            f"between {100 * stay_nights * person_num} and {200 * stay_nights * person_num}": [100* stay_nights * person_num,200* stay_nights * person_num],
            f"between {150 * stay_nights * person_num} and {250 * stay_nights * person_num}": [150* stay_nights * person_num,250* stay_nights * person_num],
            f"between {200 * stay_nights * person_num} and {300 * stay_nights * person_num}": [200* stay_nights * person_num,300* stay_nights * person_num],
            f"between {250 * stay_nights * person_num} and {350 * stay_nights * person_num}": [250* stay_nights * person_num,350* stay_nights * person_num],
            f"between {300 * stay_nights * person_num} and {400 * stay_nights * person_num}": [300* stay_nights * person_num,400* stay_nights * person_num],
            f"between {400 * stay_nights * person_num} and {600 * stay_nights * person_num}": [400* stay_nights * person_num,600* stay_nights * person_num],
            f"between {500 * stay_nights * person_num} and {700 * stay_nights * person_num}": [500* stay_nights * person_num,700* stay_nights * person_num],
            f"between {600 * stay_nights * person_num} and {800 * stay_nights * person_num}": [600* stay_nights * person_num,800* stay_nights * person_num],
            f"between {800 * stay_nights * person_num} and {1000 * stay_nights * person_num}": [800* stay_nights * person_num,1000* stay_nights * person_num],
            }
        return between


    # # ─────────────── 每晚每房（均） (Per night per room) ───────────────（直接判断每个房间每天的价钱，和总人数无关）

    def get_valid_per_night_per_room(self, parsed, constraints):

        bucket = {
            label: {
                "product_ids": set(),
                "range": cost,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, cost in constraints.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]
            in_date = q["in_date"]
            out_date = q["out_date"]

            for hotel in self.city_dict[self._norm_city(city)]:

                per_night_per_room = self.get_avg_price_hotel(in_date, out_date, hotel)
                    # 把产品放入对应区间
                for label, info in bucket.items():
                    low, high = info["range"]
                    if low < per_night_per_room < high:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["range"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

    @register_func(
        "generate_cost_less_per_night_per_room"
    )
    def generate_cost_less_per_night_per_room(self, input_dict, generate_params = None):
        """
        生成 每晚每房开销低于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_less_per_night_per_room = self.get_less(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_room(parsed, cost_less_per_night_per_room)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_more_per_night_per_room"
    )
    def generate_cost_more_per_night_per_room(self, input_dict, generate_params = None):
        """
        生成 每晚每房开销高于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_more_per_night_per_room = self.get_more(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_room(parsed, cost_more_per_night_per_room)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_around_per_night_per_room"
    )
    def generate_cost_around_per_night_per_room(self, input_dict, generate_params = None):
        """
        生成 每晚每房开销大约
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_around_per_night_per_room = self.get_around(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_room(parsed, cost_around_per_night_per_room)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
        
    @register_func(
        "generate_cost_between_per_night_per_room"
    )
    def generate_cost_between_per_night_per_room(self, input_dict, generate_params = None):
        """
        生成 每晚每房开销在某个区间
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_range_per_night_per_room = self.get_between(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_room(parsed, cost_range_per_night_per_room)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_cost_per_night_per_room"
    )
    def validate_cost_per_night_per_room(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        summ = 0
        room_num = 0
        for item in product_list:
            weekday = self.convert_weekstr_to_weekday(item["date"])
            product = self.get_product_from_product_id(item["product_id"])
            summ += float(product[weekday]) * int(item["room_num"])
            room_num += int(item["room_num"])
        avg = summ / room_num if room_num != 0 else 0
        return float(constraint[0]) <= float(avg) <= float(constraint[1])

    # # ─────────────── 每人每晚（均） (Per night per person) ───────────────（直接判断每个房间的人均价，生成时和总人数无关，验证时和总人数有关）

    def get_valid_per_night_per_person(self, parsed, constraints):


        bucket = {
            label: {
                "product_ids": set(),
                "range": cost,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, cost in constraints.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]
            in_date = q["in_date"]
            out_date = q["out_date"]

            for hotel in self.city_dict[self._norm_city(city)]:

                per_night_per_person = self.get_avg_price_hotel(in_date, out_date, hotel)
                    # 把产品放入对应区间
                for label, info in bucket.items():
                    low, high = info["range"]
                    if low < per_night_per_person  < high or low < per_night_per_person*0.75 < high or low < per_night_per_person*0.6 < high:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["range"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

  
    @register_func(
        "generate_cost_less_per_night_per_person"
    )
    def generate_cost_less_per_night_per_person(self, input_dict, generate_params = None):
        """
        生成 每晚每人开销低于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_less_per_night_per_person = self.get_less(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_person(parsed, cost_less_per_night_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_more_per_night_per_person"
    )
    def generate_cost_more_per_night_per_person(self, input_dict, generate_params = None):
        """
        生成 每晚每人开销高于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_more_per_night_per_person = self.get_more(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_person(parsed, cost_more_per_night_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
        
    @register_func(
        "generate_cost_around_per_night_per_person"
    )
    def generate_cost_around_per_night_per_person(self, input_dict, generate_params = None):
        """
        生成 每晚每人开销大约
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_around_per_night_per_person = self.get_around(1,1)


        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_person(parsed, cost_around_per_night_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_between_per_night_per_person"
    )
    def generate_cost_between_per_night_per_person(self, input_dict, generate_params = None):
        """
        生成 每晚每人开销在某个区间
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        cost_range_per_night_per_person = self.get_between(1,1)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_per_person(parsed, cost_range_per_night_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_cost_per_night_per_person"
    )
    def validate_cost_per_night_per_person(self, product_list, validation_params):
        date_dict = self.convert_to_date_dict(product_list)
        date_num = len(date_dict)
        constraint = list(validation_params.values())[0]
        summ = 0
        person_num = int(product_list[0]["total_person_num"])
        for item in product_list:
            weekday = self.convert_weekstr_to_weekday(item["date"])
            product = self.get_product_from_product_id(item["product_id"])
            summ += float(product[weekday]) * int(item["room_num"])
        avg = summ / person_num / date_num  if person_num != 0 and date_num != 0 else 0
        return float(constraint[0]) <= float(avg) <= float(constraint[1])

    # # ─────────────── 每晚总价（均） (per night total) ───────────────（房间价钱*间数，生成时和总人数有关，验证时和总人数无关）
    
    def get_valid_per_night_total(self, parsed, constraints):

        bucket = {
            label: {
                "product_ids": set(),
                "range": cost,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, cost in constraints.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]
            in_date = q["in_date"]
            out_date = q["out_date"]
            person_num = q["person_num"]

            for hotel in self.city_dict[self._norm_city(city)]:

                per_night_total = self.get_avg_price_hotel(in_date, out_date, hotel)
                    # 把产品放入对应区间
                for label, info in bucket.items():
                    low, high = info["range"]
                    if low < per_night_total*person_num  < high or low < per_night_total*1.5 * math.ceil(person_num/2) < high or low < per_night_total*1.8 * math.ceil(person_num/3) < high:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["range"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

    @register_func(
        "generate_cost_less_per_night_total"
    )
    def generate_cost_less_per_night_total(self, input_dict, generate_params = None):
        """
        生成 每晚总价低于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        cost_less_per_night_total = self.get_less(1, person_num)


        valid = self.get_valid_per_night_total(parsed, cost_less_per_night_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_more_per_night_total"
    )
    def generate_cost_more_per_night_total(self, input_dict, generate_params = None):
        """
        生成 每晚总价高于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        cost_more_per_night_total = self.get_less(1, person_num)


        valid = self.get_valid_per_night_total(parsed, cost_more_per_night_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_around_per_night_total"
    )
    def generate_cost_around_per_night_total(self, input_dict, generate_params = None):
        """
        生成 每晚总价大约
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        cost_around_per_night_total = self.get_around(1,person_num)


        valid = self.get_valid_per_night_total(parsed, cost_around_per_night_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_between_per_night_total"
    )
    def generate_cost_between_per_night_total(self, input_dict, generate_params = None):
        """
        生成 每晚总价在某区间
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        cost_between_per_night_total = self.get_between(1,person_num)

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_per_night_total(parsed, cost_between_per_night_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_cost_per_night_total"
    )
    def validate_cost_per_night_total(self, product_list, validation_params):
        date_dict = self.convert_to_date_dict(product_list)
        date_num = len(date_dict)
        constraint = list(validation_params.values())[0]
        summ = 0
        for item in product_list:
            weekday = self.convert_weekstr_to_weekday(item["date"])
            product = self.get_product_from_product_id(item["product_id"])
            summ += float(product[weekday]) * int(item["room_num"])
        avg = summ / date_num if date_num != 0 else 0
        return float(constraint[0]) <= float(avg) <= float(constraint[1])

    # # ─────────────── 每人总价 (per person total) ───────────────（每个房间人均价*总天数，生成时和总人数无关，验证时和总人数有关）

    def get_valid_per_person_total(self, parsed, constraints):

        bucket = {
            label: {
                "product_ids": set(),
                "range": cost,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, cost in constraints.items()
        }
        stay_nights = self.get_total_nights_from_dict(parsed)

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]
            in_date = q["in_date"]
            out_date = q["out_date"]
            person_num = q["person_num"]
            for hotel in self.city_dict[self._norm_city(city)]:

                per_person_total = self.get_avg_price_hotel(in_date, out_date, hotel)
                    # 把产品放入对应区间
                for label, info in bucket.items():
                    low, high = info["range"]
                    if low < per_person_total*stay_nights  < high or low < per_person_total*stay_nights*0.6 < high or low < per_person_total*stay_nights * 0.75< high:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["range"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

    @register_func(
        "generate_cost_less_per_person_total"
    )
    def generate_cost_less_per_person_total(self, input_dict, generate_params = None):
        """
        生成 每人总价低于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_less_total_per_person = self.get_less(stay_nights, 1)



        valid = self.get_valid_per_person_total(parsed, cost_less_total_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_more_per_person_total"
    )
    def generate_cost_more_per_person_total(self, input_dict, generate_params = None):
        """
        生成 每人总价高于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_more_total_per_person = self.get_more(stay_nights,1)




        valid = self.get_valid_per_person_total(parsed, cost_more_total_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_around_per_person_total"
    )
    def generate_cost_around_per_person_total(self, input_dict, generate_params = None):
        """
        生成 每人总价大约
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_around_total_per_person = self.get_around(stay_nights, 1)



        valid = self.get_valid_per_person_total(parsed, cost_around_total_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_between_per_person_total"
    )
    def generate_cost_between_per_person_total(self, input_dict, generate_params = None):
        """
        生成 每人总价在某区间
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_range_total_per_person = self.get_between(stay_nights, 1)



        valid = self.get_valid_per_person_total(parsed, cost_range_total_per_person)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_cost_per_person_total"
    )
    def validate_cost_per_person_total(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        summ = 0
        person_num = int(product_list[0]["total_person_num"])
        for item in product_list:
            weekday = self.convert_weekstr_to_weekday(item["date"])
            product = self.get_product_from_product_id(item["product_id"])
            summ += float(product[weekday]) * int(item["room_num"])
        avg = summ / person_num if person_num != 0 else 0
        return float(constraint[0]) <= float(avg) <= float(constraint[1])

    # # ─────────────── 总价 (Total) ───────────────（每个房间价格*间数*总天数，和总人数有关）

    def get_valid_total(self, parsed, constraints):

        bucket = {
            label: {
                "product_ids": set(),
                "range": cost,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, cost in constraints.items()
        }
        stay_nights = self.get_total_nights_from_dict(parsed)
        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]
            in_date = q["in_date"]
            out_date = q["out_date"]
            person_num = q["person_num"]

            for hotel in self.city_dict[self._norm_city(city)]:

                total = self.get_avg_price_hotel(in_date, out_date, hotel)
                    # 把产品放入对应区间
                for label, info in bucket.items():
                    low, high = info["range"]
                    if low < total*stay_nights*person_num < high or low < total*stay_nights*math.ceil(person_num / 2) * 0.75 < high or low < total*stay_nights * math.ceil(person_num/3) * 0.6< high:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["range"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

    @register_func(
        "generate_cost_less_total"
    )
    def generate_cost_less_total(self, input_dict, generate_params = None):
        """
        生成 总价低于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        stay_nights = self.get_total_nights_from_dict(parsed)

        cost_less_total = self.get_less(stay_nights, person_num)



        valid = self.get_valid_total(parsed, cost_less_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_more_total"
    )
    def generate_cost_more_total(self, input_dict, generate_params = None):
        """
        生成 总价高于
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_more_total = self.get_more(stay_nights, person_num)


        valid = self.get_valid_total(parsed, cost_more_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_around_total"
    )
    def generate_cost_around_total(self, input_dict, generate_params = None):
        """
        生成 总价大约
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_around_total = self.get_around(stay_nights, person_num)


        valid = self.get_valid_total(parsed, cost_around_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_cost_between_total"
    )
    def generate_cost_between_total(self, input_dict, generate_params = None):
        """
        生成 总价在某区间
        输入参数: input_dict, generate_params = None
        in_date格式形如"2025-11-12"
        """

        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]
        stay_nights = self.get_total_nights_from_dict(parsed)
        cost_range_total = self.get_between(stay_nights, person_num)

        valid = self.get_valid_total(parsed, cost_range_total)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        # print(all_labels_and_ranges)
        if generate_params:
            param = list(generate_params)[0]
            # print(param)
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_cost_total"
    )
    def validate_cost_total(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        summ = 0
        for item in product_list:
            weekday = self.convert_weekstr_to_weekday(item["date"])
            product = self.get_product_from_product_id(item["product_id"])
            summ += float(product[weekday]) * int(item["room_num"])
        return float(constraint[0]) <= float(summ) <= float(constraint[1])


    def get_valid_hotel_type(self, parsed, constraints):

        bucket = {
            label: {
                "product_ids": set(),
                "hotel_type": hotel_type,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, hotel_type in constraints.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]


            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["hotel_type"] in info["hotel_type"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["hotel_type"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

    @register_func(
        "generate_hotel_type_include"
    )
    def generate_hotel_type_include(self, input_dict, generate_params = None):
        specify_type = {
            "Economy": ["Economy"],
            'Midscale': ['Midscale'],
            'Upscale': ['Upscale'],
            'Luxury': ['Luxury'],
            'Economy and Midscale': ['Economy', 'Midscale'],
            'Midscale and Upscale':['Midscale','Upscale'],
            'Upscale and Luxury':['Upscale', 'Luxury'],
        }
        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_hotel_type(parsed, specify_type)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
    
    @register_func(
        "generate_hotel_type_exclude"
    )    
    def generate_hotel_type_exclude(self, input_dict, generate_params = None):
        exclude_type = {
            "Economy": ["Midscale", "Upscale", "Luxury"],
            'Midscale':["Economy", "Upscale", "Luxury"],
            'Upscale': ["Economy","Midscale", "Luxury"],
            'Luxury': ["Economy","Midscale", "Upscale"],
            'Economy and Luxury': ["Midscale", "Upscale"],
        }

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_hotel_type(parsed, exclude_type)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
           
    @register_func(
        "validate_hotel_type"
    )
    def validate_hotel_type(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_hotel_from_product_id(item["product_id"])["hotel_type"] not in constraint:
                return False
        return True

    @register_func(
        "generate_review_count"
    )
    def generate_review_count(self, input_dict, generate_params = None):
        review_count= {
            "30": 30,
            "50": 50,
            "100": 100,
            "200": 200,
            "300": 300,
            "400": 400,
            "500": 500,

        }


        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "count": count,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, count in review_count.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["review_count"] >= info["count"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["count"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
       
    @register_func(
        "validate_review_count"
    )
    def validate_review_count(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_hotel_from_product_id(item["product_id"])["review_count"] < constraint:
                return False
        return True

    @register_func(
        "generate_good_rate"
    )
    def generate_good_rate(self, input_dict, generate_params = None):
        good_rate = {
            "0.75": 0.75,
            "0.8": 0.8,
            "0.85": 0.85,
            "0.9": 0.9,
        }

        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "good_rate": g_rate,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, g_rate in good_rate.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["good_remarks_rate"] >= info["good_rate"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["good_rate"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })
        
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
        
    @register_func(
        "validate_good_rate"
    )
    def validate_good_rate(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_hotel_from_product_id(item["product_id"])["good_remarks_rate"] < constraint:
                return False
        return True

    @register_func(
        "generate_star"
    )
    def generate_star(self, input_dict, generate_params = None):
        star_l = {
            "3": 3,
            "3.5": 3.5,
            "4": 4,
            "4.5": 4.5,
        }
        
        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "stars": stars,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, stars in star_l.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["stars"] >= info["stars"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["stars"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_star"
    )
    def validate_star(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_hotel_from_product_id(item["product_id"])["stars"] < constraint:
                return False
        return True

    def get_valid_aspect_rating(self, parsed, constraints):
        bucket = {
            label: {
                "product_ids": set(),
                "rating": rating,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, rating in constraints.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:

                    # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["product_rating"] >= info["rating"][0] and hotel["environment_rating"] >= info["rating"][1] and hotel["service_rating"] >= info["rating"][2]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        results = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                results.append({
                    "selected_description": label,
                    "validation_params": info["rating"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })

        return results

    @register_func(
        "generate_aspect_rating_all"
    )
    def generate_aspect_rating_all(self, input_dict, generate_params = None):
        all_rating = {
            "8": [8,8,8],
            "7.5": [7.5,7.5,7.5],
            "7": [7,7,7],
            "6.5": [6.5,6.5,6.5],
            "6": [6,6,6],
        }
    
        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_aspect_rating(parsed, all_rating)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result
        
    @register_func(
        "generate_aspect_rating_product"
    )
    def generate_aspect_rating_product(self, input_dict, generate_params = None):
        product_rating = {
            "8.5": [8.5,0,0],
            "8": [8,0,0],
            "7.5": [7.5,0,0],
            "7": [7,0,0],
        }
    
        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_aspect_rating(parsed, product_rating)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_aspect_rating_environment"
    )
    def generate_aspect_rating_environment(self, input_dict, generate_params = None):
        environment_rating = {
            "8.5": [0,8.5,0],
            "8": [0,8,0],
            "7.5": [0,7.5,0],
            "7": [0,7,0],
        }
        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_aspect_rating(parsed, environment_rating)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "generate_aspect_rating_service"
    )
    def generate_aspect_rating_service(self, input_dict, generate_params = None):
        service_rating = {
            "8.5": [0,0,8.5],
            "8": [0,0,8],
            "7.5": [0,0,7.5],
            "7": [0,0,7],
        }

        parsed = self.parse_input_dict(input_dict)

        valid = self.get_valid_aspect_rating(parsed, service_rating)
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_aspect_rating"
    )
    def validate_aspect_rating(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            hotel = self.get_hotel_from_product_id(item["product_id"])
            if hotel["product_rating"] < constraint[0] or hotel["environment_rating"] < constraint[1] or hotel["service_rating"] < constraint[2]:
                return False
        return True

    @register_func(
        "generate_cancel_policy"
    )
    def generate_cancel_policy(self, input_dict, generate_params = None):
        cancel_policy = {
            "12:00 PM on check-in day": 0,
            "1 day before check-in day": 1,
            "2 days before check-in day": 1,
            "3 days before check-in day": 2,
            "7 days before check-in day": 2,
        }

        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "cancel_policy": policy,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, policy in cancel_policy.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["cancel_policy"] <= info["cancel_policy"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["cancel_policy"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_cancel_policy"
    )
    def validate_cancel_policy(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_product_from_product_id(item["product_id"])["cancel_policy"] > constraint:
                return False
        return True

    @register_func(
        "generate_pet_friendly"
    )
    def generate_pet_friendly(self, input_dict, generate_params = None):
        pet_friendly = {
            "true": True
        }

        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "pet_friendly": friendly,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, friendly in pet_friendly.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    if hotel["is_pet_friendly"] == info["pet_friendly"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["pet_friendly"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_pet_friendly"
    )
    def validate_pet_friendly(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_hotel_from_product_id(item["product_id"])["is_pet_friendly"] != constraint:
                return False
        return True
       
    @register_func(
        "generate_breakfast_number"
    )
    def generate_breakfast_number(self, input_dict, generate_params = None): # breakfast_num无法generate，只能验证
        parsed = self.parse_input_dict(input_dict)
        person_num = parsed[0]["person_num"]

        # breakfast_num_e = {
        #     "0": 0,
        #     "": person_num 
        # }
        breakfast_num_e = {}
        for i in range(0, person_num+1):
            breakfast_num_e[f"{i}"] = i


        bucket = {
            label: {
                "product_ids": set(),
                "breakfast_number": breakfast_number,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, breakfast_number in breakfast_num_e.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():

                    info["product_ids"].update(hotel["all_product_id"])

                    # 标记该城市至少有一个产品命中
                    info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["breakfast_number"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_breakfast_number_equal"
    )
    def validate_breakfast_number_equal(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        date_dict = self.convert_to_date_dict(product_list)
        for date, product_ids in date_dict.items():
            total_breakfast = 0
            for product_id in product_ids:
                product = self.get_product_from_product_id(product_id)
                if product:
                    total_breakfast += product["breakfast_num"]
            if total_breakfast != constraint:
                return False
        return True

    @register_func(
        "validate_breakfast_number_more"
    )
    def validate_breakfast_number_more(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        date_dict = self.convert_to_date_dict(product_list)
        for date, product_ids in date_dict.items():
            total_breakfast = 0
            for product_id in product_ids:
                product = self.get_product_from_product_id(product_id)
                if product:
                    total_breakfast += product["breakfast_num"]
            if total_breakfast < constraint:
                return False
        return True

    @register_func(
        "generate_has_window"
    )
    def generate_has_window(self, input_dict, generate_params = None):
        has_window = {
            "true": True
        }

        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "has_window": window,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, window in has_window.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                # 把产品放入对应区间
                for label, info in bucket.items():
                    info["product_ids"].update(hotel["all_product_id"])

                    # 标记该城市至少有一个产品命中
                    info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["has_window"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_has_window"
    )
    def validate_has_window(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            if self.get_product_from_product_id(item["product_id"])["has_window"] != constraint:
                return False
        return True
    

    def getcity(self, city):
        info = CITY_CENTER_COORDS[self._norm_city(city)]
        return (info["lat"], info["lon"])
    

    @register_func(
        "generate_city_center"
    )
    def generate_city_center(self, input_dict, generate_params = None):
        city_center = {
            "3": 3,
            "5": 5,
            "7": 7,
            "10": 10
        }

        parsed = self.parse_input_dict(input_dict)
        bucket = {
            label: {
                "product_ids": set(),
                "distance": distance,
                "city_hit": {q["city"]: False for q in parsed}  # 每个城市都要 hit
            }
            for label, distance in city_center.items()
        }

        # 一次遍历所有 query、所有 hotel、所有 product
        for q in parsed:
            city = q["city"]

            for hotel in self.city_dict[self._norm_city(city)]:
                p1 = (hotel["latitude"], hotel["longitude"])
                p2 = self.getcity(city) ### 查找城市中心点的经纬度
                distance = geodesic(p1, p2).km
                # 把产品放入对应区间
                for label, info in bucket.items():
                    if  distance <= info["distance"]:
                        info["product_ids"].update(hotel["all_product_id"])

                        # 标记该城市至少有一个产品命中
                        info["city_hit"][city] = True

        # 输出结果：必须满足所有城市 city_hit == True
        valid = []
        for label, info in bucket.items():
            if all(info["city_hit"].values()) and info["product_ids"]:
                valid.append({
                    "selected_description": label,
                    "validation_params": info["distance"],
                    "candidate_product_ids": sorted(info["product_ids"])
                })
        all_labels_and_ranges = {
                x["selected_description"]:x["validation_params"]
                for x in valid
            }
        if generate_params:
            param = list(generate_params)[0]
            if param in all_labels_and_ranges:
                result =  [x for x in valid if x["selected_description"] == param][0]
                result["all_labels_and_ranges"] = all_labels_and_ranges
                return result
            else:
                print("selected generate param is not valid.")
        else:
            result = random.choice(valid)
            result["all_labels_and_ranges"] = all_labels_and_ranges
            return result

    @register_func(
        "validate_city_center"
    )
    def validate_city_center(self, product_list, validation_params):
        constraint = list(validation_params.values())[0]
        for item in product_list:
            hotel = self.get_hotel_from_product_id(item["product_id"])
            city = hotel["real_city"]
            p1 = (hotel["latitude"], hotel["longitude"])
            p2 = self.getcity(city) ### 查找城市中心点的经纬度
            if geodesic(p1, p2).km > constraint:
                return False
        return True

    @register_func("validate_last_night")
    def validate_last_night(self, product_list, validation_params):
        is_city_center = self.validate_city_center(product_list, validation_params)
        if not is_city_center:
            return False
        constraint = list(validation_params.values())[0]

        last_item = product_list[-1]
        last_hotel = self.get_hotel_from_product_id(last_item["product_id"])
        transport_id = last_item["transport_id"]

        # --- 找 transport 信息 ---
        if transport_id.startswith("Train"):
            tr = next((x for x in self.trains if x["Train_id"] == transport_id), None)
            if tr is None:
                return False  # 或 raise ValueError(...)
            p_trans = (tr["Departure Station Latitude"], tr["Departure Station Longitude"])

        else:
            fl = next((x for x in self.flights if x["Flight_id"] == transport_id), None)
            if fl is None:
                return False
            p_trans = (fl["Departure Airport Latitude"], fl["Departure Airport Longitude"])

        # --- distance check ---
        p_last_hotel = (last_hotel["latitude"], last_hotel["longitude"])

        return geodesic(p_last_hotel, p_trans).km <= 10






# 靠近景点机场火车站由于需要跟别的表联动，暂时没写


# ================================
# 使用示例
# ================================
if __name__ == "__main__":
    evaluator = HotelEvaluator("/data2/wzy/trip/hotel_database_with_product.json")


    input_dict = [
      {
        "from": "Chengdu",
        "to": "Beijing",
        "distance_km": 1524.79,
        "depart_date": "2025-10-26",
        "stay_days": 3,
        "return_date": "2025-10-28",
        "number_of_people": 1
      },
      {
        "from": "Beijing",
        "to": "Changchun",
        "distance_km": 857.75,
        "depart_date": "2025-10-29",
        "stay_days": 3,
        "return_date": "2025-10-31",
        "number_of_people": 1
      }
    ]


    import time
    start = time.time()
    # res_hotel_generation = evaluator.execute("generate_cost_less_per_night_per_room",input_dict, generate_params = None)
    # res_hotel_generation = evaluator.execute("generate_cost_more_per_night_per_person", input_dict, generate_params=None)
    # res_hotel_generation = evaluator.execute("generate_cost_around_per_night_total", input_dict, generate_params=None)
    res_hotel_generation = evaluator.execute("generate_cost_more_per_person_total", input_dict, generate_params=None)
    # res_hotel_generation = evaluator.execute("generate_cost_more_total", input_dict, generate_params=None)
    # res_hotel_generation = evaluator.execute("generate_hotel_type_include", input_dict, generate_params={'Upscale and Luxury':['Upscale', 'Luxury']})
    # res_hotel_generation = evaluator.execute("generate_review_count", input_dict, generate_params={"500": 500})
    # res_hotel_generation = evaluator.execute("generate_good_rate", input_dict, generate_params={"0.95": 0.95})
    # res_hotel_generation = evaluator.execute("generate_star", input_dict, generate_params={"4.5": 4.5})
    # res_hotel_generation = evaluator.execute("generate_aspect_rating_all", input_dict, generate_params={ "8": [8,8,8]})

    # res_hotel_generation = evaluator.execute("generate_cancel_policy", input_dict, generate_params={"12:00 PM on check-in day": 0})
    # res_hotel_generation = evaluator.execute("generate_pet_friendly", input_dict, generate_params={"true": True})
    # res_hotel_generation = evaluator.execute("generate_breakfast_number", input_dict, generate_params={"no": 0})
    # res_hotel_generation = evaluator.execute("generate_has_window", input_dict, generate_params={"true": True})
    end = time.time()
    print(end - start)
    # with open("/data2/wzy/trip/result.json", "w+") as f:
    #     json.dump(res_hotel_generation,f, indent=2, ensure_ascii=False)

    product_list_sample = [
        {
            "product_id": "P_H_00023581",
            "date":"2025-11-13",
            "total_person_num":"4",
            "room_num":"2",
        },
        {
            "product_id": "P_H_00023640",
            "date":"2025-11-14",
            "total_person_num":"4",
            "room_num":"2",
        }
    ]

    # 11-13 1312.79*2 11-14 496.0*2

    # res_hotel_validation = evaluator.execute("validate_cost_per_night_per_room", product_list_sample, {"400": [400,999999]})
    # res_hotel_validation = evaluator.execute("validate_cost_per_night_per_person", product_list_sample, {"700": [0,700]})
    # res_hotel_validation = evaluator.execute("validate_cost_per_night_total", product_list_sample, {"600": [600, 999999]})
    # res_hotel_validation = evaluator.execute("validate_cost_per_person_total", product_list_sample, {"around 1000": [850,1150]})
    # res_hotel_validation = evaluator.execute("validate_cost_total", product_list_sample, {"around 3500": [3300,3700]})
    # res_hotel_validation = evaluator.execute("validate_hotel_type", product_list_sample, {'Upscale': ["Economy","Midscale", "Luxury"]})
    # res_hotel_validation = evaluator.execute("validate_review_count", product_list_sample, {'80': 80})
    # res_hotel_validation = evaluator.execute("validate_good_rate", product_list_sample, {'0.75': 0.75})
    # res_hotel_validation = evaluator.execute("validate_star", product_list_sample, {'2': 2}) #实际上constraint并没有这个item
    # res_hotel_validation = evaluator.execute("validate_aspect_rating", product_list_sample, {'6.0': [6.0,6.0,6.0]}) #实际上constraint并没有这个item
    # res_hotel_validation = evaluator.execute("validate_cancel_policy", product_list_sample, {"2": 2})
    # res_hotel_validation = evaluator.execute("validate_pet_friendly", product_list_sample, {"true": True}) #结果为False
    # res_hotel_validation = evaluator.execute("validate_breakfast_number_equal", product_list_sample, {"3": 3}) #结果为False
    # res_hotel_validation = evaluator.execute("validate_breakfast_number_more", product_list_sample, {"no": 0})
    # res_hotel_validation = evaluator.execute("validate_has_window", product_list_sample, {"true": True}) #结果为False

    # print(res_hotel_validation)
