import json
import os
import sys
from copy import deepcopy
import random
random.seed(42)
# 添加项目根目录到Python路径
sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..','..')))

from Evaluation.Restaurant.val_restaurants import RestaurantEvaluator  
# ================= 1. 严格复用并适配餐厅版的辅助函数 =================

def find_generate_func(rubric_detail_dict, target_id):
    """根据 label 的 _id 在 rubric_detail_dict 中找到对应的 generate_func"""
    for desc_key, desc_val in rubric_detail_dict.items():
        if desc_key == "_id" or not isinstance(desc_val, dict):
            continue
        res = desc_val.get("result")
        if not isinstance(res, dict):
            continue
        all_labels = res.get("all_labels_and_ranges", {})
        for label_dict in all_labels.values():
            if isinstance(label_dict, dict) and label_dict.get("_id") == target_id:
                return desc_val.get("generate_func")
    return None

def find_validate_func_and_value(rubric_detail, target_id):
    """根据 label 的 _id 找到对应 description 的 validate_func 和 value"""
    for desc_key, desc_val in rubric_detail.items():
        if desc_key == "_id" or not isinstance(desc_val, dict):
            continue
        res = desc_val.get("result")
        if not isinstance(res, dict):
            continue
        all_labels = res.get("all_labels_and_ranges", {})
        for label_dict in all_labels.values():
            if isinstance(label_dict, dict) and label_dict.get("_id") == target_id:
                return desc_val.get("validate_func"), label_dict.get("value")
    return None, None

def build_generate_params(rubric_result_dict, picked_ids):
    """构建餐厅版需要的参数格式：{'all_labels_and_ranges': {label_key: value}}"""
    inner_params = {}
    for desc_key, desc_val in rubric_result_dict.items():
        if not isinstance(desc_val, dict) or "result" not in desc_val:
            continue
        all_labels = desc_val["result"]["all_labels_and_ranges"]
        for label_key, label_obj in all_labels.items():
            if label_obj["_id"] in picked_ids:
                inner_params[label_key] = label_obj["value"]
    return {"all_labels_and_ranges": inner_params}

def build_restaurant_info_from_ids(ids):
    """餐厅版校验输入的结构：[{'id': rid}]"""
    return [{"id": rid} for rid in ids]

# ================= 2. 核心动态验证函数 (餐厅版) =================

# 定义特殊 Rubrics（不参与交集，而是在交集后做 validate）
SPECIAL_RUBRICS = {
    "RUBRIC_INCLUDE_CUISINE",
}

def check_feasibility_restaurant(trip_data, evaluator):
    """
    完全基于 applied_modification_chains 动态调用餐厅评估器验证。
    """
    # 1. 提取基础参数（严格对应你 process_single_trip 的逻辑）
    route = trip_data.get("route", [])
    city_list = sorted({seg["to"].lower() for seg in route})
    
    days = trip_data.get("days")
    if days is None:
        days = sum(seg.get("stay_days", 0) for seg in route)
        if days <= 0: days = len(route)
    
    anchor_info = {"num_people": route[0].get("number_of_people", 2)}
    
    applied_chains = trip_data.get("applied_modification_chains", {})
    original_rubrics = trip_data.get("rubric_results", {})
    
    normal_candidate_sets = []
    special_rubric_tasks = []

    # 2. 动态生成各约束的候选集合
    for rubric_name, picked_ids in applied_chains.items():
        if not picked_ids:
            continue
            
        last_id = picked_ids[-1]
        rubric_detail = original_rubrics.get(rubric_name, {})
        
        # 寻找函数名和构造动态参数
        gen_func = find_generate_func(rubric_detail, last_id)
        if not gen_func: continue
        
        gen_params = build_generate_params(rubric_detail, [last_id])

        # 执行动态计算
        try:
            result = evaluator.execute(
                gen_func,
                city_list,
                days,
                anchor_info=anchor_info,
                generate_params=gen_params
            )
            candidate_ids = set(result.get("candidate_ids", []))
        except Exception as e:
            print(f"Dynamic eval error in {gen_func}: {e}")
            return False, 0

        # 分类：特殊还是普通
        if rubric_name in SPECIAL_RUBRICS:
            special_rubric_tasks.append({
                "name": rubric_name,
                "rubric_detail": rubric_detail,
                "last_id": last_id
            })
        else:
            normal_candidate_sets.append(candidate_ids)

    # 3. 计算普通约束的动态交集
    if normal_candidate_sets:
        intersection = set.intersection(*normal_candidate_sets)
        # 餐厅版阈值：days * 10
        feasible_normal = len(intersection) >= (days * 10)
    else:
        feasible_normal = True
        intersection = set()

    # 4. 执行特殊约束动态校验
    special_validation_pass = True
    if feasible_normal:
        # 构造用于 validate_xxx 的餐厅数据结构
        restaurant_info = build_restaurant_info_from_ids(intersection)
        
        for task in special_rubric_tasks:
            val_func, val_params = find_validate_func_and_value(task["rubric_detail"], task["last_id"])
            if not val_func: continue
            
            # 特判：如果需要重定向方法名可以在此处理
            # if task["name"] == "RUBRIC_INCLUDE_CUISINE": ...

            # 实时校验交集是否满足要求
            is_valid = evaluator.execute(
                val_func,
                restaurant_info,
                val_params
            )
            if not is_valid:
                special_validation_pass = False
                break
    else:
        special_validation_pass = False

    return (feasible_normal and special_validation_pass)

# ================= 3. 运行示例 =================

if __name__ == "__main__":
    # 初始化
    restaurant_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/restaurants.json"
    evaluator = RestaurantEvaluator(restaurant_path)
    
    test_data =    {
    "trip_id": "c3_35835",
    "route": [
      {
        "from": "Guilin",
        "to": "Beijing",
        "distance_km": 1727.42,
        "depart_date": "2025-10-26",
        "stay_days": 3,
        "return_date": "2025-10-28",
        "number_of_people": 5
      },
      {
        "from": "Beijing",
        "to": "Tianjin",
        "distance_km": 113.72,
        "depart_date": "2025-10-29",
        "stay_days": 2,
        "return_date": "2025-10-30",
        "number_of_people": 5
      }
    ],
    "rubric_results": {
      "RUBRIC_PRICE": {
        "Per-person per-meal cost between a certain price range": {
          "probability": 1,
          "description": "Each selected restaurant has a per-person per-meal cost that falls within {slot}.",
          "violation_description": "Some recommended restaurants have a per-person per-meal cost outside {slot}.",
          "generate_func": "generate_price_per_person_range",
          "validate_func": "validate_price_per_person_range",
          "result": {
            "selected_description": "50-100",
            "validation_params": [
              50.0,
              100.0
            ],
            "all_labels_and_ranges": {
              "200-400": {
                "value": [
                  200.0,
                  400.0
                ],
                "_id": "restaurant_1_4_7"
              }
            }
          },
          "_id": "restaurant_1_4"
        }
      },
      "RUBRIC_RATING": {
        "Minimum overall star rating": {
          "probability": 1,
          "description": "Only recommend restaurants with an overall rating of at least {slot} stars.",
          "violation_description": "Some recommended restaurants have an overall rating below {slot} stars.",
          "generate_func": "generate_min_overall_rating",
          "validate_func": "validate_min_overall_rating",
          "result": {
            "selected_description": "4.5",
            "validation_params": [
              4.5
            ],
            "all_labels_and_ranges": {
              "4.0": {
                "value": [
                  4.0
                ],
                "_id": "restaurant_2_1_2"
              }
            }
          },
          "_id": "restaurant_2_1"
        }
      }
    },
    "applied_modification_chains": {
      "RUBRIC_PRICE": [
        "restaurant_1_4_7"
      ],
      "RUBRIC_RATING": [
        "restaurant_2_1_2"
      ]
    },
    "intersection_count": 1739,
    "feasible": True,
    "special_validation_pass": True,
    "SAMPLE_RUBRIC_COUNT": 2,
    "TOTAL_CHAIN_LENGTH": 2
  }



    # 假设 test_data 是你的 trip 数据字典
    is_feasible = check_feasibility_restaurant(test_data, evaluator)
    print(f"Result: {is_feasible}")