import json
from geopy.distance import distance as geopy_distance
import random
from typing import Dict, Any, List, Optional, Iterable, Tuple

random.seed(42)

# 至少要有这么多餐厅满足某个阈值，才认为这个阈值是“可用的”
MIN_CANDIDATE_RESTAURANTS = 50

# 价格区间的预定义（人均，单位：，更“好看”的常见组合）
# 用于 between / around / less / more 等各种 price rubrics 的底层区间
PRICE_BUCKET_RANGES = [
    (50, 100),
    (50, 150),
    (100,150),
    (100, 200),
    (150, 200),
    (150, 300),
    (200, 400),
    (300, 600),
]

PRICE_BUCKET_LESS_RANGES = [50,100,150,200,300,400,600]
PRICE_BUCKET_MORE_RANGES = [50,100,150,200,300,400,600]

# review_count 阈值的预定义集合
REVIEW_COUNT_THRESHOLDS = [20, 50, 100, 200, 300, 500, 800, 1000]

def register_func(*names):
    def decorator(func):
        func._register_names = names
        return func
    return decorator


def collect_registered_funcs(cls):
    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

def _parse_price_mode(rubric_key: Optional[str]) -> str:
    """根据 rubric 的 level-two 名称解析是 less / more / around / between."""
    if not rubric_key:
        return "between"
    k = rubric_key.lower()
    if "less than" in k:
        return "less"
    if "more than" in k:
        return "more"
    if "around" in k:
        return "around"
    if "between" in k:
        return "between"
    return "between"


def _per_person_scale_factor(rubric_key: Optional[str], days: int) -> float:
    """
    对于 per-person 系列：
    - 单顿：factor = 1
    - 每天：factor = 2（默认两顿）
    - 全程：factor = 2 * days
    """
    if not rubric_key:
        return 1.0
    k = rubric_key.lower()
    meals_per_day = 2
    if "per-day" in k:
        return float(meals_per_day)
    if "whole trip" in k:
        return float(meals_per_day * max(days, 1))
    # 默认当作单顿
    return 1.0


def _group_scale_factor(rubric_key: Optional[str], days: int, num_people: int) -> float:
    """
    对于 whole group 系列：
    - 每顿：factor = num_people
    - 每天：factor = num_people * 2
    - 全程：factor = num_people * 2 * days
    """
    meals_per_day = 2
    np = max(num_people, 1)
    if not rubric_key:
        return float(np)
    k = rubric_key.lower()
    if "per-meal" in k:
        return float(np)
    if "per-day" in k:
        return float(np * meals_per_day)
    if "whole group and the whole trip" in k or "whole trip" in k:
        return float(np * meals_per_day * max(days, 1))
    return float(np)


# ======================
# 主类
# ======================
@collect_registered_funcs
class RestaurantEvaluator:
    def __init__(self, restaurants_path: str):
        self.restaurants_path = restaurants_path

        with open(restaurants_path, "r", encoding="utf-8") as f:
            self.restaurants: List[Dict[str, Any]] = json.load(f)

        # 建立城市索引
        self.restaurants_by_city: Dict[str, List[Dict[str, Any]]] = {}
        
        # 建立 ID 索引（新增）
        self.restaurants_by_id: Dict[str, Dict[str, Any]] = {}

        for r in self.restaurants:
            # 城市索引
            city = str(r.get("real_city", "")).strip().lower()
            if city:
                self.restaurants_by_city.setdefault(city, []).append(r)

            # ID 索引
            rid = str(r.get("id")).strip()
            if rid:
                self.restaurants_by_id[rid] = r

    # ------------------
    # 调度入口
    # ------------------
    def execute(self, func_name: str, *args, **kwargs) -> Any:
        func = self.func_map.get(func_name)
        if func is None:
            raise ValueError(f"Unregistered function: {func_name}")
        return func(self, *args, **kwargs)

    # ------------------
    # 工具函数：基础索引
    # ------------------
    def _filter_by_cities(self, city_list: Iterable[str]) -> List[Dict[str, Any]]:
        """
        返回这些城市内的所有餐馆（不考虑距离）
        """
        city_set = {str(c).strip().lower() for c in city_list}
        result = []
        for city in city_set:
            result.extend(self.restaurants_by_city.get(city, []))
        return result

    def _get_restaurant_by_id(self, rest_id: str) -> Optional[Dict[str, Any]]:
        return self.restaurants_by_id.get(str(rest_id).strip())

    # ========= 判断餐厅是否“必须预约” =========
    @staticmethod
    def _is_must_reserve(r: Dict[str, Any]) -> bool:
        """
        尽量鲁棒地判断一个餐厅是否“必须预约才能去”。

        约定优先使用字段：
        - need_reservation / must_reserve: bool 或 0/1
        - 其次尝试从字符串字段中匹配“must/required/need reservation/预约”等字样

        如果你的真实字段名不一样，只需要改这里的取值逻辑即可，其他代码不需要动。
        """
        # 常见布尔 / 数值字段
        for key in ["need_reservation", "must_reserve", "must_reservation"]:
            v = r.get(key)
            if isinstance(v, bool):
                return v
            if isinstance(v, (int, float)):
                return v > 0

        # 尝试从字符串 tag 中推断
        for key in ["reservation_note", "reservation_type", "reservation_policy"]:
            v = r.get(key)
            if isinstance(v, str):
                s = v.strip().lower()
                if not s:
                    continue
                # 英文 + 简单中文匹配都兜一下
                patterns = [
                    "must reserve", "must-reserve", "reservation only",
                    "need reservation", "required reservation",
                    "only by reservation",
                    "仅限预约", "只接受预约", "必须预约", "预约制"
                ]
                if any(p in s for p in patterns):
                    return True

        return False


    @staticmethod
    def _product_ids_for_restaurants(restaurants: List[Dict[str, Any]]) -> List[str]:
        ids: List[str] = []
        for r in restaurants:
            for p in r.get("products", []):
                pid = p.get("product_id")
                if pid:
                    ids.append(str(pid))
        return ids

    # ======================
    # generate_* 系列
    # ======================

    @register_func("generate_price_per_person_range")
    def generate_price_per_person_range(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“人均消费区间”的 rubric 参数（基于 avg_price）。
        所有 per-person 的价格约束都复用这里的候选集合：
        - 候选过滤：avg_price ∈ [low_pp, high_pp]
        - selected_description：根据 rubric_key + days，将 [low_pp, high_pp]
          转换成自然语言的 {slot}（某个价格 / 一段区间）。
        - validation_params：始终是 [low_pp, high_pp]（per-person per-meal），
          方便各种价格 rubrics 统一复用验证逻辑。
        """
        # 二次调用：通过 all_labels_and_ranges 直接还原
        if generate_params and "all_labels_and_ranges" in generate_params:
            label_map = dict(generate_params["all_labels_and_ranges"])
            if not label_map:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }
            # 一般只有一个
            selected_label, vals = next(iter(label_map.items()))
            if not isinstance(vals, list) or len(vals) < 2:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }
            low_pp, high_pp = float(vals[0]), float(vals[1])

            restaurants = self._filter_by_cities(city_list)
            if not restaurants:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }

            candidate_restaurants: List[Dict[str, Any]] = []
            for r in restaurants:
                p = r.get("avg_price")
                if isinstance(p, (int, float)) and low_pp <= float(p) <= high_pp:
                    candidate_restaurants.append(r)

            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }

            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]

            candidate_product_ids: List[str] = []
            for r in candidate_restaurants:
                for prod in r.get("products", []):
                    pid = prod.get("product_id")
                    people = prod.get("people")
                    price = prod.get("price")
                    if not pid or not isinstance(people, (int, float)) or people <= 0:
                        continue
                    if not isinstance(price, (int, float)):
                        continue
                    per_person = float(price) / float(people)
                    if low_pp <= per_person <= high_pp:
                        candidate_product_ids.append(str(pid))

            return {
                "selected_description": selected_label,
                "validation_params": [low_pp, high_pp],
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
                "all_labels_and_ranges": {selected_label: [low_pp, high_pp]},
            }

                # ---------- 第一次调用：根据 city / days + rubric_key 找一个 per-person 区间 ----------
        restaurants = self._filter_by_cities(city_list)

        prices = []
        for r in restaurants:
            p = r.get("avg_price")
            if isinstance(p, (int, float)):
                prices.append(float(p))
        if not prices or not restaurants:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        rubric_key = None
        if generate_params:
            rubric_key = generate_params.get("rubric_key")

        mode = _parse_price_mode(rubric_key)
        factor = _per_person_scale_factor(rubric_key, days)

        viable: Dict[str, Dict[str, Any]] = {}

        # ---------- 根据 mode 选择不同的价格 bucket ----------
        if mode == "less":
            # less：从 PRICE_BUCKET_LESS_RANGES 生成 (0, x) 区间
            bucket_ranges = [(0, float(x)) for x in PRICE_BUCKET_LESS_RANGES if x > 0]
        elif mode == "more":
            # more：从 PRICE_BUCKET_MORE_RANGES 生成 (x, BIG) 区间
            BIG = 1e9
            bucket_ranges = [(float(x), BIG) for x in PRICE_BUCKET_MORE_RANGES if x > 0]
        else:
            # 默认使用原始 PRICE_BUCKET_RANGES
            bucket_ranges = [(float(a), float(b)) for a, b in PRICE_BUCKET_RANGES]

        # ---------- 主循环：遍历 bucket_ranges ----------
        for low_pp, high_pp in bucket_ranges:
            if low_pp >= high_pp:
                continue

            # 先看有没有足够多餐厅落在这个 per-person 区间
            candidate_restaurants: List[Dict[str, Any]] = []
            for r in restaurants:
                p = r.get("avg_price")
                if isinstance(p, (int, float)) and low_pp <= float(p) <= high_pp:
                    candidate_restaurants.append(r)

            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue

            # 计算展示区间
            disp_low = low_pp * factor
            disp_high = high_pp * factor

            if mode == "less":
                # “低于某价格”：展示上界
                disp_val = disp_high
                slot_str = f"{int(disp_val)}"
            elif mode == "more":
                # “高于某价格”：展示下界
                disp_val = disp_low
                slot_str = f"{int(disp_val)}"
            elif mode == "around":
                # “大约某价格”：中点取整到 25 元
                center = (disp_low + disp_high) / 2.0
                step = 25
                disp_val = round(center / step) * step
                slot_str = f"{int(disp_val)}"
            else:  # between / default
                slot_str = f"{int(disp_low)}-{int(disp_high)}"

            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]

            candidate_product_ids: List[str] = []
            for r in candidate_restaurants:
                for prod in r.get("products", []):
                    pid = prod.get("product_id")
                    people = prod.get("people")
                    price = prod.get("price")
                    if not pid or not isinstance(people, (int, float)) or people <= 0:
                        continue
                    if not isinstance(price, (int, float)):
                        continue
                    per_person = float(price) / float(people)
                    if low_pp <= per_person <= high_pp:
                        candidate_product_ids.append(str(pid))

            viable[slot_str] = {
                "range": [low_pp, high_pp],  # validation 使用 per-person 区间
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }


        if not viable:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        selected_slot = random.choice(list(viable.keys()))
        info = viable[selected_slot]

        all_labels_and_ranges = {
            lbl: v["range"] for lbl, v in viable.items()
        }

        return {
            "selected_description": selected_slot,         # 用来填 {slot}，句子就自然了
            "validation_params": info["range"],            # [low_pp, high_pp]
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    # === 已有：基于人均价格 + 人数生成「whole group」约束 ===
    @register_func("generate_total_budget_for_group")
    def generate_total_budget_for_group(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        基于 per-person 区间，生成“整团总价”类约束。
        - 候选 & 验证逻辑：仍然依赖 per-person per-meal 的 [low_pp, high_pp]
        - 但 selected_description 会把这个区间乘上人数 / 每天 2 顿 / 天数，
          变成 “X-Y for the whole group per meal / per day / whole trip” 这样的自然语言。
        """

        # 二次调用：完全按传入的 all_labels_and_ranges 还原
        if generate_params and "all_labels_and_ranges" in generate_params:
            label_map = dict(generate_params["all_labels_and_ranges"])
            if not label_map:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }
            selected_label, vals = next(iter(label_map.items()))
            if not isinstance(vals, list) or len(vals) < 3:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }
            low_pp, high_pp, np = float(vals[0]), float(vals[1]), int(vals[2])

            restaurants = self._filter_by_cities(city_list)
            if not restaurants:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }

            candidate_restaurants: List[Dict[str, Any]] = []
            for r in restaurants:
                p = r.get("avg_price")
                if isinstance(p, (int, float)) and low_pp <= float(p) <= high_pp:
                    candidate_restaurants.append(r)

            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }

            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]

            candidate_product_ids: List[str] = []
            for r in candidate_restaurants:
                for prod in r.get("products", []):
                    pid = prod.get("product_id")
                    people = prod.get("people")
                    price = prod.get("price")
                    if not pid or not isinstance(people, (int, float)) or people <= 0:
                        continue
                    if not isinstance(price, (int, float)):
                        continue
                    per_person = float(price) / float(people)
                    if low_pp <= per_person <= high_pp:
                        candidate_product_ids.append(str(pid))

            return {
                "selected_description": selected_label,
                "validation_params": [low_pp, high_pp, np],
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
                "all_labels_and_ranges": {selected_label: [low_pp, high_pp, np]},
            }

        # ---------- 第一次调用 ----------
        num_people = 1
        if isinstance(anchor_info, dict):
            v = anchor_info.get("num_people")
            if isinstance(v, (int, float)) and v > 0:
                num_people = int(v)

        rubric_key = None
        if generate_params:
            rubric_key = generate_params.get("rubric_key")

        factor = _group_scale_factor(rubric_key, days, num_people)

        # 先用 per-person generator 选一个 bucket
        base = self.generate_price_per_person_range(
            city_list, days, anchor_info, generate_params=None
        )

        per_selected_label = base.get("selected_description") or ""
        per_range = base.get("validation_params") or []
        per_all = base.get("all_labels_and_ranges") or {}

        if not per_selected_label or len(per_range) != 2:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        group_all: Dict[str, List[float]] = {}
        selected_group_label = None

        for _, prange in per_all.items():
            if not isinstance(prange, list) or len(prange) != 2:
                continue
            plow, phigh = float(prange[0]), float(prange[1])

            disp_low = plow * factor
            disp_high = phigh * factor
            slot_str = f"{int(disp_low)}-{int(disp_high)}"

            label_suffix = ""
            if rubric_key:
                k = rubric_key.lower()
                if "per-meal" in k:
                    label_suffix = f" for {num_people} people per meal"
                elif "per-day" in k:
                    label_suffix = f" for {num_people} people per day"
                elif "whole group and the whole trip" in k or "whole trip" in k:
                    label_suffix = f" for {num_people} people for the whole trip"
            if label_suffix:
                slot_full = slot_str + label_suffix
            else:
                slot_full = slot_str

            group_all[slot_full] = [plow, phigh, num_people]

        # 选中的是 base 里那一条（per_selected_label 对应的 per_range）
        low_pp, high_pp = float(per_range[0]), float(per_range[1])
        disp_low_sel = low_pp * factor
        disp_high_sel = high_pp * factor
        slot_sel = f"{int(disp_low_sel)}-{int(disp_high_sel)}"
        label_suffix = ""
        if rubric_key:
            k = rubric_key.lower()
            if "per-meal" in k:
                label_suffix = f" for {num_people} people per meal"
            elif "per-day" in k:
                label_suffix = f" for {num_people} people per day"
            elif "whole group and the whole trip" in k or "whole trip" in k:
                label_suffix = f" for {num_people} people for the whole trip"
        if label_suffix:
            selected_label = slot_sel + label_suffix
        else:
            selected_label = slot_sel

        group_all[selected_label] = [low_pp, high_pp, num_people]

        return {
            "selected_description": selected_label,
            "validation_params": [low_pp, high_pp, num_people],
            "candidate_ids": base.get("candidate_ids", []),
            "candidate_product_ids": base.get("candidate_product_ids", []),
            "all_labels_and_ranges": group_all,
        }


    @register_func("generate_min_overall_rating")
    def generate_min_overall_rating(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“最低总体评分”的 rubric 参数（stars 字段）。
        """
        restaurants = self._filter_by_cities(city_list)
        ratings = [
            float(r.get("stars"))
            for r in restaurants
            if isinstance(r.get("stars"), (int, float))
        ]
        if not ratings:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        if generate_params and "all_labels_and_ranges" in generate_params:
            thresholds_map = dict(generate_params["all_labels_and_ranges"])
        else:
            max_rating = max(ratings)
            min_rating = min(ratings)
            base_thresholds = [3.5, 4.0, 4.5]
            thresholds_map: Dict[str, List[float]] = {}
            for t in base_thresholds:
                if t <= max_rating:
                    label = f"{t:.1f}"
                    thresholds_map[label] = [t]
            if not thresholds_map:
                t = round((min_rating + max_rating) / 2.0, 1)
                label = f"{t:.1f}"
                thresholds_map[label] = [t]

        viable_labels: Dict[str, Dict[str, Any]] = {}
        for label, vals in thresholds_map.items():
            t = float(vals[0])
            candidate_restaurants = [
                r for r in restaurants
                if isinstance(r.get("stars"), (int, float)) and float(r["stars"]) >= t
            ]
            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue
            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
            candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)
            viable_labels[label] = {
                "range": [t],
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }

        if not viable_labels:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        selected_label = random.choice(list(viable_labels.keys()))
        info = viable_labels[selected_label]
        all_labels_and_ranges = {lbl: v["range"] for lbl, v in viable_labels.items()}

        return {
            "selected_description": selected_label,
            "validation_params": info["range"],
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    @register_func("generate_min_review_count")
    def generate_min_review_count(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“最少评价数量”的 rubric 参数。

        使用字段：review_count
        - 使用预定义 REVIEW_COUNT_THRESHOLDS
        - 只保留“仍有足够餐厅 review_count >= t”的阈值
        """
        restaurants = self._filter_by_cities(city_list)
        counts = [
            float(r.get("review_count"))
            for r in restaurants
            if isinstance(r.get("review_count"), (int, float))
        ]
        if not counts:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        if generate_params and "all_labels_and_ranges" in generate_params:
            thresholds_map = dict(generate_params["all_labels_and_ranges"])
        else:
            thresholds_map: Dict[str, List[int]] = {}
            max_count = max(counts)
            # 只保留 <= max_count 的阈值
            for t in REVIEW_COUNT_THRESHOLDS:
                if t <= max_count:
                    thresholds_map[str(t)] = [t]
            if not thresholds_map:
                # 回退到中位数
                sorted_counts = sorted(counts)
                mid = int(sorted_counts[len(sorted_counts) // 2])
                thresholds_map[str(mid)] = [mid]

        viable_labels: Dict[str, Dict[str, Any]] = {}
        for label, vals in thresholds_map.items():
            t = int(vals[0])
            candidate_restaurants = [
                r for r in restaurants
                if isinstance(r.get("review_count"), (int, float)) and int(r["review_count"]) >= t
            ]
            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue
            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
            candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)
            viable_labels[label] = {
                "range": [t],
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }

        if not viable_labels:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        selected_label = random.choice(list(viable_labels.keys()))
        info = viable_labels[selected_label]
        all_labels_and_ranges = {lbl: v["range"] for lbl, v in viable_labels.items()}

        return {
            "selected_description": selected_label,
            "validation_params": info["range"],
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    @register_func("generate_include_cuisines")
    def generate_include_cuisines(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“需要包含的菜系”（多样性约束）。

        用 small_cate 作为菜系标签。
        - 根据 days 决定组合长度范围
        - candidate_ids：至少属于这些菜系之一的餐厅
        - candidate_product_ids：这些餐厅的全部产品
        """
        restaurants = self._filter_by_cities(city_list)

        cuisine_counter: Dict[str, int] = {}
        for r in restaurants:
            cate_raw = str(r.get("small_cate", "")).strip()
            if not cate_raw:
                continue
            # 过滤掉“Other Delicacies”“Other Chinese Cuisine”等带 other 的类别
            if "other" in cate_raw.lower():
                continue
            cuisine_counter[cate_raw] = cuisine_counter.get(cate_raw, 0) + 1

        if not cuisine_counter:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        if generate_params and "all_labels_and_ranges" in generate_params:
            combo_map = dict(generate_params["all_labels_and_ranges"])
        else:
            cuisine_names = list(cuisine_counter.keys())

            if days <= 1:
                k_min, k_max = 1, 1
            elif 2 <= days <= 3:
                k_min, k_max = 1, 2
            elif 4 <= days <= 5:
                k_min, k_max = 1, 3
            else:
                k_min, k_max = 2, 4

            k_max = min(k_max, len(cuisine_names))
            if k_max < 1:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }
            if k_min > k_max:
                k_min = 1

            combo_map: Dict[str, List[str]] = {}
            seen: set = set()
            max_combos = 15
            max_attempts = 100
            attempts = 0

            while len(combo_map) < max_combos and attempts < max_attempts:
                attempts += 1
                k = random.randint(k_min, k_max)
                k = min(k, len(cuisine_names))
                if k <= 0:
                    break
                sample = random.sample(cuisine_names, k=k)
                key_tuple = tuple(sorted(sample))
                if key_tuple in seen:
                    continue
                seen.add(key_tuple)
                label = ", ".join(key_tuple)
                combo_map[label] = list(key_tuple)

            if not combo_map:
                only = random.choice(cuisine_names)
                combo_map[only] = [only]

        viable_labels: Dict[str, Dict[str, Any]] = {}
        for label, cu_list in combo_map.items():
            # 任一菜系命中的餐厅都算候选
            cu_lower = {c.lower() for c in cu_list}
            candidate_restaurants = []
            for r in restaurants:
                cate = str(r.get("small_cate", "")).strip().lower()
                if cate and cate in cu_lower:
                    candidate_restaurants.append(r)

            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue

            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
            candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)
            viable_labels[label] = {
                "range": cu_list,
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }

        if not viable_labels:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        selected_label = random.choice(list(viable_labels.keys()))
        info = viable_labels[selected_label]
        all_labels_and_ranges = {lbl: v["range"] for lbl, v in viable_labels.items()}

        return {
            "selected_description": selected_label,
            "validation_params": info["range"],
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    @register_func("generate_exclude_cuisines")
    def generate_exclude_cuisines(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“需要避免的菜系”。

        对于每个菜系 cname：
        - candidate_ids：所有 small_cate != cname 的餐厅
        - candidate_product_ids：这些餐厅的全部产品
        """
        restaurants = self._filter_by_cities(city_list)
        if not restaurants:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        cuisine_counter: Dict[str, int] = {}
        for r in restaurants:
            cate_raw = str(r.get("small_cate", "")).strip()
            if not cate_raw:
                continue
            # 过滤掉“Other ...”类目，不用作约束选项
            if "other" in cate_raw.lower():
                continue
            cuisine_counter[cate_raw] = cuisine_counter.get(cate_raw, 0) + 1

        if not cuisine_counter:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        if generate_params and "all_labels_and_ranges" in generate_params:
            cuisine_map = dict(generate_params["all_labels_and_ranges"])
        else:
            # 按频率从低到高遍历
            sorted_items = sorted(cuisine_counter.items(), key=lambda x: x[1])
            cuisine_map: Dict[str, List[str]] = {}
            for cname, _ in sorted_items:
                cuisine_map[cname] = [cname]

        viable_labels: Dict[str, Dict[str, Any]] = {}
        for cname, val in cuisine_map.items():
            cname_lower = cname.lower()
            candidate_restaurants = [
                r for r in restaurants
                if str(r.get("small_cate", "")).strip().lower() != cname_lower
            ]
            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue
            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
            candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)
            viable_labels[cname] = {
                "range": val,
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }

        if not viable_labels:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        selected_label = random.choice(list(viable_labels.keys()))
        info = viable_labels[selected_label]
        all_labels_and_ranges = {lbl: v["range"] for lbl, v in viable_labels.items()}

        return {
            "selected_description": selected_label,
            "validation_params": info["range"],
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }


    # ---------- Reservable ----------


    @register_func("generate_reservable")
    def generate_reservable(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“必须可预订”的约束。
        """
        restaurants = self._filter_by_cities(city_list)
        candidate_restaurants = [
            r for r in restaurants if bool(r.get("reservable"))
        ]
        if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
        candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)

        all_labels_and_ranges = {"reservable_true": [True]}

        return {
            "selected_description": "",
            "validation_params": [True],
            "candidate_ids": candidate_ids,
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    @register_func("generate_exclude_cuisines")
    def generate_exclude_cuisines(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“需要避免的菜系”（支持多样化组合，与 include 的生成逻辑保持一致）。

        small_cate 作为菜系标签。
        - 根据 days 决定组合长度范围
        - candidate_ids：small_cate 不属于这些被排除菜系的餐厅
        - candidate_product_ids：这些餐厅的全部产品
        """
        restaurants = self._filter_by_cities(city_list)

        cuisine_counter: Dict[str, int] = {}
        for r in restaurants:
            cate_raw = str(r.get("small_cate", "")).strip()
            if not cate_raw:
                continue
            if "other" in cate_raw.lower():
                continue
            cuisine_counter[cate_raw] = cuisine_counter.get(cate_raw, 0) + 1

        if not cuisine_counter:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        # ---------------------
        # 复用旧参数（如果 generate_params 存在）
        # ---------------------
        if generate_params and "all_labels_and_ranges" in generate_params:
            combo_map = dict(generate_params["all_labels_and_ranges"])

        else:
            cuisine_names = list(cuisine_counter.keys())

            # 和 include 一样，根据 days 生成组合长度
            k_min, k_max = 1, 3

            k_max = min(k_max, len(cuisine_names))
            if k_max < 1:
                return {
                    "selected_description": "",
                    "validation_params": [],
                    "candidate_ids": [],
                    "candidate_product_ids": [],
                    "all_labels_and_ranges": {},
                }
            if k_min > k_max:
                k_min = 1

            # 随机生成组合，与 include 保持一致
            combo_map: Dict[str, List[str]] = {}
            seen = set()
            max_combos = 10
            max_attempts = 100
            attempts = 0

            while len(combo_map) < max_combos and attempts < max_attempts:
                attempts += 1
                k = random.randint(k_min, k_max)
                sample = random.sample(cuisine_names, k=k)
                key_tuple = tuple(sorted(sample))
                if key_tuple in seen:
                    continue
                seen.add(key_tuple)

                label = ", ".join(key_tuple)
                combo_map[label] = list(key_tuple)

            if not combo_map:
                only = random.choice(cuisine_names)
                combo_map[only] = [only]

        # ---------------------
        # 组合评估：排除这些 small_cate
        # ---------------------
        viable_labels: Dict[str, Dict[str, Any]] = {}

        for label, cu_list in combo_map.items():
            exclude_lower = {c.lower() for c in cu_list}

            candidate_restaurants = [
                r for r in restaurants
                if str(r.get("small_cate", "")).strip().lower() not in exclude_lower
            ]

            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue

            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
            candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)

            viable_labels[label] = {
                "range": cu_list,
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }

        if not viable_labels:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        # 随机选择一个标签，保持与 include 一致
        selected_label = random.choice(list(viable_labels.keys()))
        info = viable_labels[selected_label]
        all_labels_and_ranges = {lbl: v["range"] for lbl, v in viable_labels.items()}

        return {
            "selected_description": selected_label,
            "validation_params": info["range"],
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }


    @register_func("generate_exclude_must_reserve")
    def generate_exclude_must_reserve(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        """
        生成“排除必须预约才能去的餐厅”的约束。

        逻辑：
        - 使用 _is_must_reserve(r) 判断餐厅是否为“must-reserve”
        - candidate_restaurants = 所有 not must-reserve 的餐厅
        - 约束本身没有区间/多档，所以 selected_description 设成空字符串，
          上层不会再触发带 generate_params 的二次调用（和 generate_reservable 行为保持一致）。
        """
        restaurants = self._filter_by_cities(city_list)
        if not restaurants:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        candidate_restaurants = [
            r for r in restaurants
            if not self._is_must_reserve(r)
        ]

        if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
            # 如果可用餐厅太少，就认为这个约束在该城市不可用，返回空
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
        candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)

        all_labels_and_ranges = {"exclude_must_reserve": [True]}

        return {
            "selected_description": "",
            "validation_params": [True],  # 只是一个布尔开关，不用多信息
            "candidate_ids": candidate_ids,
            "candidate_product_ids": candidate_product_ids,
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    # ---------- 三类 subrating（各自独立） ----------
    # 真实字段：
    #   product_rating      ~ 食物评分
    #   environment_rating  ~ 环境评分
    #   service_rating      ~ 服务评分

    def _generate_min_subrating(
        self,
        city_list: List[str],
        field_name: str,
        base_thresholds: List[float],
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        restaurants = self._filter_by_cities(city_list)
        ratings = [
            float(r.get(field_name))
            for r in restaurants
            if isinstance(r.get(field_name), (int, float))
        ]
        if not ratings:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        if generate_params and "all_labels_and_ranges" in generate_params:
            thresholds_map = dict(generate_params["all_labels_and_ranges"])
        else:
            max_rating = max(ratings)
            min_rating = min(ratings)
            thresholds_map: Dict[str, List[float]] = {}
            for t in base_thresholds:
                if t <= max_rating:
                    label = f"{t:.1f}"
                    thresholds_map[label] = [t]
            if not thresholds_map:
                t = round((min_rating + max_rating) / 2.0, 1)
                label = f"{t:.1f}"
                thresholds_map[label] = [t]

        viable_labels: Dict[str, Dict[str, Any]] = {}
        for label, vals in thresholds_map.items():
            t = float(vals[0])
            candidate_restaurants = [
                r for r in restaurants
                if isinstance(r.get(field_name), (int, float)) and float(r[field_name]) >= t
            ]
            if len(candidate_restaurants) < MIN_CANDIDATE_RESTAURANTS:
                continue
            candidate_ids = [str(r.get("id")) for r in candidate_restaurants if r.get("id")]
            candidate_product_ids = self._product_ids_for_restaurants(candidate_restaurants)
            viable_labels[label] = {
                "range": [t],
                "candidate_ids": candidate_ids,
                "candidate_product_ids": candidate_product_ids,
            }

        if not viable_labels:
            return {
                "selected_description": "",
                "validation_params": [],
                "candidate_ids": [],
                "candidate_product_ids": [],
                "all_labels_and_ranges": {},
            }

        selected_label = random.choice(list(viable_labels.keys()))
        info = viable_labels[selected_label]
        all_labels_and_ranges = {lbl: v["range"] for lbl, v in viable_labels.items()}

        return {
            "selected_description": selected_label,
            "validation_params": info["range"],
            "candidate_ids": info["candidate_ids"],
            "candidate_product_ids": info["candidate_product_ids"],
            "all_labels_and_ranges": all_labels_and_ranges,
        }

    @register_func("generate_min_food_rating")
    def generate_min_food_rating(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        return self._generate_min_subrating(
            city_list,
            field_name="product_rating",
            base_thresholds=[6.0, 7.0, 8.0],
            generate_params=generate_params,
        )

    @register_func("generate_min_environment_rating")
    def generate_min_environment_rating(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        return self._generate_min_subrating(
            city_list,
            field_name="environment_rating",
            base_thresholds=[6.0, 7.0, 8.0],
            generate_params=generate_params,
        )

    @register_func("generate_min_service_rating")
    def generate_min_service_rating(
        self,
        city_list: List[str],
        days: int,
        anchor_info: Optional[Dict[str, Any]] = None,
        generate_params: Optional[Dict[str, Any]] = None,
    ) -> Dict[str, Any]:
        return self._generate_min_subrating(
            city_list,
            field_name="service_rating",
            base_thresholds=[6.0, 7.0, 8.0],
            generate_params=generate_params,
        )

    # ======================
    # validate_* 系列（字段名已对齐真实数据）
    # ======================
    @register_func("validate_price_per_person_range")
    def validate_price_per_person_range(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查每家餐厅的人均是否在 [low, high] 区间内（avg_price 字段）。
        """
        if not validation_params or len(validation_params) != 2:
            return False

        low, high = validation_params
        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            price = r.get("avg_price")
            if not isinstance(price, (int, float)):
                return False
            if not (low <= price <= high):
                return False

        return True

    # === 新增：group 预算约束的验证（仍复用 per-person 逻辑） ===
    @register_func("validate_total_budget_for_group")
    def validate_total_budget_for_group(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        验证“整团总价区间”约束：
        - validation_params = [low_pp, high_pp, num_people]
        - 实际约束仍然是 per-person avg_price ∈ [low_pp, high_pp]
        - num_people 只是作为上下文信息存在，方便上层展示/调试
        """
        if not validation_params or len(validation_params) < 2:
            return False

        try:
            low_pp = float(validation_params[0])
            high_pp = float(validation_params[1])
        except Exception:
            return False

        # 直接复用人均价格验证逻辑
        return self.validate_price_per_person_range(
            restaurant_info,
            [low_pp, high_pp],
        )


    @register_func("validate_min_overall_rating")
    def validate_min_overall_rating(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查 stars 是否 >= 阈值。
        """
        if not validation_params:
            return False
        threshold = float(validation_params[0])

        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            rating = r.get("stars")
            if not isinstance(rating, (int, float)):
                return False
            if rating < threshold:
                return False

        return True

    @register_func("validate_min_review_count")
    def validate_min_review_count(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查 review_count 是否 >= 阈值。
        """
        if not validation_params:
            return False
        threshold = int(validation_params[0])

        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            cnt = r.get("review_count")
            if not isinstance(cnt, (int, float)):
                return False
            if cnt < threshold:
                return False

        return True

    @register_func("validate_include_cuisines")
    def validate_include_cuisines(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查推荐结果整体上“是否覆盖了这些菜系”（列表级约束）。
        """
        required_cuisines = [str(c).strip() for c in (validation_params or []) if str(c).strip()]
        if not required_cuisines:
            return False

        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        covered = {c: False for c in required_cuisines}
        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                continue
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                continue
            cate = str(r.get("small_cate", "")).strip().lower()
            for c in required_cuisines:
                if cate and c.lower() == cate:
                    covered[c] = True

        return all(covered.values())

    @register_func("validate_exclude_cuisines")
    def validate_exclude_cuisines(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查推荐结果是否“避免了这些菜系”（每家都不得包含）。
        """
        forbidden_cuisines = [str(c).strip() for c in (validation_params or []) if str(c).strip()]
        if not forbidden_cuisines:
            return False

        forbidden_lower = {c.lower() for c in forbidden_cuisines}

        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            cate = str(r.get("small_cate", "")).strip().lower()
            if cate and cate in forbidden_lower:
                return False

        return True


    @register_func("validate_reservable")
    def validate_reservable(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查所有餐厅是否都 reservable == True。
        """
        if not validation_params:
            return False

        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            if not bool(r.get("reservable")):
                return False

        return True

    @register_func("validate_exclude_must_reserve")
    def validate_exclude_must_reserve(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        """
        检查所有餐厅是否都不是“必须预约才能去”的类型。
        """
        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            if self._is_must_reserve(r):
                # print(f"Restaurant {rest_id} is must reserve")
                return False

        return True

    # ---------- 三类 subrating 验证 ----------
    def _validate_min_subrating(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
        field_name: str,
    ) -> bool:
        if not validation_params:
            return False
        threshold = float(validation_params[0])

        infos = restaurant_info if isinstance(restaurant_info, list) else [restaurant_info]

        for info in infos:
            rest_id = info.get("id")
            if not rest_id:
                return False
            r = self._get_restaurant_by_id(rest_id)
            if not r:
                return False
            rating = r.get(field_name)
            if not isinstance(rating, (int, float)):
                return False
            if rating < threshold:
                return False

        return True

    @register_func("validate_min_food_rating")
    def validate_min_food_rating(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        return self._validate_min_subrating(restaurant_info, validation_params, "product_rating")

    @register_func("validate_min_environment_rating")
    def validate_min_environment_rating(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        return self._validate_min_subrating(restaurant_info, validation_params, "environment_rating")

    @register_func("validate_min_service_rating")
    def validate_min_service_rating(
        self,
        restaurant_info: Any,
        validation_params: List[Any],
    ) -> bool:
        return self._validate_min_subrating(restaurant_info, validation_params, "service_rating")


if __name__ == "__main__":
    import pprint

    restaurants_path = "restaurants.json"

    re_eval = RestaurantEvaluator(restaurants_path)

    city_list = ["Sanya"]
    days = 3
    anchor_info = {}  # 可选：{"restaurant_radius_km": 5.0, "anchor_lat": ..., "anchor_lon": ...}
    params = [city_list, days, anchor_info]

    def test_generate(name: str):
        print(f"\n=== {name} ===")
        result = re_eval.execute(name, *params)
        pprint.pp(result)
        print(
            f"[STAT] selected_description={result.get('selected_description')!r}, "
            f"len(candidate_ids)={len(result.get('candidate_ids', []))}, "
            f"len(candidate_product_ids)={len(result.get('candidate_product_ids', []))}, "
            f"num_all_labels={len(result.get('all_labels_and_ranges', {}))}"
        )

    test_generate("generate_price_per_person_range")
    test_generate("generate_min_overall_rating")
    test_generate("generate_min_review_count")
    test_generate("generate_include_cuisines")
    test_generate("generate_exclude_cuisines")
    test_generate("generate_within_distance_from_anchor")
    test_generate("generate_reservable")
    test_generate("generate_min_food_rating")
    test_generate("generate_min_environment_rating")
    test_generate("generate_min_service_rating")