import json
import os
import sys
from copy import deepcopy

# 添加项目根目录到Python路径（请根据实际情况调整）
sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..')))

from Evaluation.Hotel.val_hotels import HotelEvaluator


def find_generate_func(rubric_detail_dict, target_id):
    """
    在 rubric_detail_dict 中根据 label 的 _id 找到它所在的 description，
    并返回对应 description 的 generate_func。
    """
    for desc_key, desc_val in rubric_detail_dict.items():
        # 跳过 _id 等非 description 字段
        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", {})
        if not isinstance(all_labels, dict):
            continue

        for label_dict in all_labels.values():
            if isinstance(label_dict, dict) and label_dict.get("_id") == target_id:
                # 找到了正确的 description → 返回它的 generate_func
                return desc_val.get("generate_func")

    return None

def build_generate_params(rubric_result_dict, picked_ids):
    """
    原版逻辑：构建参数字典
    """
    generate_params = {}

    for desc_key, desc_val in rubric_result_dict.items():
        if "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:
                generate_params[label_key] = label_obj["value"]

    return generate_params

# ================= 2. 核心动态验证函数 (严格遵循酒店逻辑) =================

def check_feasibility_hotel(trip_data, evaluator):
    """
    仿照餐厅版结构，但内部逻辑严格执行你提供的酒店版代码
    """
    # 提取基础数据
    route = trip_data.get("route")  # HotelEvaluator 的 input_dict = route
    applied_chains = trip_data.get("applied_modification_chains", {})
    original_rubrics = trip_data.get("rubric_results", {})
    
    candidate_sets = []

    # 遍历当前已应用的链条
    for rubric_name, picked_ids in applied_chains.items():
        if not picked_ids:
            continue

        # 酒店逻辑：取链路最后一个 ID
        last_id = picked_ids[-1]
        rubric_detail_dict = original_rubrics.get(rubric_name, {})

        # 1. 找 generate_func (严格复用你的函数)
        generate_func_name = find_generate_func(rubric_detail_dict, last_id)
        if not generate_func_name:
            continue

        # 2. 构造 generate_params (严格复用你的函数)
        generate_params = build_generate_params(rubric_detail_dict, [last_id])

        # 3. 调用 evaluator.execute (严格遵循酒店调用格式)
        try:
            result = evaluator.execute(
                generate_func_name,
                route,
                generate_params=generate_params
            )
            # 酒店版提取 candidate_product_ids
            candidate_ids = set(result.get("candidate_product_ids", []))
            candidate_sets.append(candidate_ids)
        except Exception as e:
            # 仿照你的错误处理逻辑
            print(f"Dynamic hotel eval error: {e}")
            return False, 0

    # 4. 求交集并判定
    if not candidate_sets:
        return True, 0

    intersection = set.intersection(*candidate_sets)
    intersection_count = len(intersection)
    
    # 酒店版可行判定：len >= 1
    feasible = intersection_count >= 1

    return feasible

if __name__ == "__main__":

    # 配置文件路径
    hotel_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/hotels.json"

    evaluator = HotelEvaluator(hotel_path)

    test_item =    {
    "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_REVIEW_COUNT_HOTEL": {
        "Not less than a certain count of review": {
          "probability": 1,
          "description": "Hotel comments is not less than {slot}",
          "violation_description": "Hotel comments is less than {slot}.",
          "generate_func": "generate_review_count",
          "validate_func": "validate_review_count",
          "result": {
            "selected_description": "200",
            "validation_params": 200,
            "all_labels_and_ranges": {
              "30": {
                "value": 30,
                "_id": "hotel_3_1_1"
              },
              "100": {
                "value": 100,
                "_id": "hotel_3_1_3"
              },
              "400": {
                "value": 400,
                "_id": "hotel_3_1_6"
              }
            }
          },
          "_id": "hotel_3_1"
        }
      },
      "RUBRIC_BREAKFAST_NUMBER": {
        "Must have breakfast more than a certain number": {
          "probability": 1,
          "description": "Breakfast number must more than {slot}.",
          "violation_description": "Breakfast number is not more than {slot}.",
          "generate_func": "generate_breakfast_number",
          "validate_func": "validate_breakfast_number_more",
          "result": {
            "selected_description": "3",
            "validation_params": 3,
            "all_labels_and_ranges": {
              "2": {
                "value": 2,
                "_id": "hotel_9_2_3"
              }
            }
          },
          "_id": "hotel_9_2"
        }
      },
      "RUBRIC_COST": {
        "Per night per room cost less than a certain price": {
          "probability": 1,
          "description": "The hotel cost per night per room must be less than {slot}.",
          "violation_description": "The hotel cost per night per room exceeds {slot}.",
          "generate_func": "generate_cost_less_per_night_per_room",
          "validate_func": "validate_cost_per_night_per_room",
          "result": {
            "selected_description": "300",
            "validation_params": [
              0,
              300
            ],
            "all_labels_and_ranges": {
              "50": {
                "value": [
                  0,
                  50
                ],
                "_id": "hotel_1_1_1"
              }
            }
          },
          "_id": "hotel_1_1"
        }
      },
      "RUBRIC_LOCATION": {
        "Must near city center": {
          "probability": 1,
          "description": "All hotel must be located within {slot} kilometers of the city center to ensure convenient access to the main attractions and urban facilities.",
          "violation_description": "One or more hotel nights (except the final night) are not located within {slot} kilometers of the city center.",
          "generate_func": "generate_city_center",
          "validate_func": "validate_city_center",
          "result": {
            "selected_description": "5",
            "validation_params": 5,
            "all_labels_and_ranges": {
              "5": {
                "value": 5,
                "_id": "hotel_11_1_2"
              },
              "10": {
                "value": 10,
                "_id": "hotel_11_1_4"
              }
            }
          },
          "_id": "hotel_11_1"
        }
      }
    },
    "applied_modification_chains": {
      "RUBRIC_REVIEW_COUNT_HOTEL": [
        "hotel_3_1_1",
        "hotel_3_1_3",
        "hotel_3_1_6"
      ],
      "RUBRIC_BREAKFAST_NUMBER": [
        "hotel_9_2_3"
      ],
    #   "RUBRIC_COST": [
    #     "hotel_1_1_1"
    #   ],
      "RUBRIC_LOCATION": [
        "hotel_11_1_4",
        "hotel_11_1_2"
      ]
    },
    "intersection_count": 0,
    "feasible": False,
    "SAMPLE_RUBRIC_COUNT": 4,
    "TOTAL_CHAIN_LENGTH": 7,
    "rollback_actions": [
      {
        "rubric": "RUBRIC_COST",
        "dropped_label_id": "hotel_1_1_1"
      }
    ]
  }
    is_ok = check_feasibility_hotel(test_item, evaluator)
    print(f"Trip ID: {test_item.get('trip_id')}")
    print(f"Is Feasible: {is_ok}")