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.Attraction.val_attractions import AttractionEvaluator

# ================= 1. 严格复用你提供的基础辅助函数 =================

def find_generate_func(rubric_detail_dict, target_id):
    """在 rubric_detail_dict 中根据 label 的 _id 找到对应 description 的 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", {})
        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:
                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", {})
        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:
                validate_func = desc_val.get("validate_func")
                value = label_dict.get("value")
                return validate_func, value
    return None, None

def build_generate_params(rubric_detail_dict, picked_ids):
    """根据 picked_ids 构建 evaluator 需要的 generate_params"""
    generate_params = {}
    for desc_key, desc_val in rubric_detail_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:
                generate_params[label_key] = label_obj["value"]
    return generate_params

# ================= 2. 核心动态验证逻辑 (严格遵循景点版流程) =================

SPECIAL_RUBRICS = {
    "RUBRIC_INCLUDE_ATTRACTIONS",
    "RUBRIC_INCLUDE_CATEGORIES",
    "RUBRIC_SIGHT_LEVEL",
    "RUBRIC_CATEGORY_PRIORITY",
}

def check_feasibility_attraction(trip_data, evaluator):
    """
    严格按照你提供的逻辑判定可行性。
    1. 计算普通约束交集。
    2. 若有普通约束：校验 (数量 >= stay_total * 2) AND (特殊校验通过)。
    3. 若无普通约束：直接可行，数量 10000。
    """
    route = trip_data.get("route", [])
    city_list = [c["to"] for c in route]
    stay_days = [c["stay_days"] for c in route]
    stay_total = sum(stay_days)

    applied_chains = trip_data.get("applied_modification_chains", {})
    original_rubrics = trip_data.get("rubric_results", {})

    normal_candidate_sets = []
    special_rubric_info = {}

    # --- 模拟遍历所选维度并执行 evaluator ---
    for rubric_name, picked_ids in applied_chains.items():
        if not picked_ids:
            continue
        
        last_id = picked_ids[-1]
        rubric_detail_dict = original_rubrics.get(rubric_name, {})
        
        # 寻找 generate_func
        generate_func_name = find_generate_func(rubric_detail_dict, last_id)
        if not generate_func_name:
            continue

        # 构造 generate_params
        generate_params = build_generate_params(rubric_detail_dict, [last_id])

        # 执行 evaluator.execute (生成 candidate_ids)
        try:
            result = evaluator.execute(
                generate_func_name,
                city_list,
                stay_days,
                generate_params
            )
            candidate_ids = set(result.get("candidate_ids", []))
        except:
            return False, 0

        # 分类
        if rubric_name in SPECIAL_RUBRICS:
            special_rubric_info[rubric_name] = {
                "rubric_block": rubric_detail_dict,
                "picked_ids": picked_ids
            }
        else:
            normal_candidate_sets.append(candidate_ids)

    # --- 最终判定流程 (严格复制你提供的 if-else 分支) ---
    if normal_candidate_sets:
        # 求普通 rubric 的交集
        intersection = set.intersection(*normal_candidate_sets)
        feasible_normal = len(intersection) >= stay_total * 2

        # 准备校验用的 attraction_info
        attraction_info = [{"poiId": poi_id, "visiting_time": 60} for poi_id in intersection]

        # 特殊 rubric 验证
        special_validation_pass = True
        for rubric_name, info in special_rubric_info.items():
            rubric_detail = info["rubric_block"]
            picked_ids = info["picked_ids"]
            last_id = picked_ids[-1]

            validate_func_name, validation_params = find_validate_func_and_value(rubric_detail, last_id)
            if not validate_func_name:
                continue

            # 你的逻辑：RUBRIC_CATEGORY_PRIORITY 强制指向 validate_include_categories
            if rubric_name == "RUBRIC_CATEGORY_PRIORITY":
                validate_func_name = "validate_include_categories"

            # 执行校验
            valid_result = evaluator.execute(
                validate_func_name,
                attraction_info,
                validation_params
            )
            if not valid_result:
                special_validation_pass = False
                break

        feasible = feasible_normal and special_validation_pass
        return feasible

    else:
        # ！！！这是你要求的关键点：没有普通约束，直接 True，数量 10000
        return True

# ================= 3. 使用示例 =================

if __name__ == "__main__":
    # 配置文件路径
    attraction_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/attractions.json"

    # 初始化 Evaluator
    with open(attraction_path, 'r', encoding='utf-8') as f:
        attractions_data = json.load(f)
    evaluator = AttractionEvaluator(attractions_data)


    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_INCLUDE_ATTRACTIONS": {
        "Must include specific attractions": {
          "probability": 1,
          "description": "The itinerary must include the following attractions: {slot}.",
          "violation_description": "The itinerary does not include the required attractions: {slot}.",
          "generate_func": "generate_include_attractions",
          "validate_func": "validate_include_attractions",
          "result": {
            "selected_description": "Flag Raising Ceremony",
            "validation_params": [
              10559683
            ],
            "all_labels_and_ranges": {
              "Flag Raising Ceremony": {
                "value": [
                  10559683
                ],
                "_id": "attraction_3_1_10"
              }
            }
          },
          "_id": "attraction_3_1"
        }
      },
      "RUBRIC_INCLUDE_CATEGORIES": {
        "Must include attractions of specific categories": {
          "probability": 1,
          "description": "The itinerary must include attractions from the following categories: {slot}.",
          "violation_description": "The itinerary does not include attractions from the required categories: {slot}.",
          "generate_func": "generate_include_categories",
          "validate_func": "validate_include_categories",
          "result": {
            "selected_description": "Arts & Museums, Photo Spots & Instagrammable",
            "validation_params": [
              "Arts & Museums",
              "Photo Spots & Instagrammable"
            ],
            "all_labels_and_ranges": {
              "Arts & Museums, Historical & Cultural Heritage": {
                "value": [
                  "Arts & Museums",
                  "Historical & Cultural Heritage"
                ],
                "_id": "attraction_1_1_7"
              }
            }
          },
          "_id": "attraction_1_1"
        }
      }
    },
    "applied_modification_chains": {
      "RUBRIC_INCLUDE_ATTRACTIONS": [
        "attraction_3_1_10"
      ],
      "RUBRIC_INCLUDE_CATEGORIES": [
        "attraction_1_1_7"
      ]
    },
    "intersection_count": 10000,
    "feasible": True,
    "special_validation_pass": True,
    "SAMPLE_RUBRIC_COUNT": 2,
    "TOTAL_CHAIN_LENGTH": 2
  }
    is_ok = check_feasibility_attraction(test_item, evaluator)
    print(f"Trip ID: {test_item.get('trip_id')}")
    print(f"Is Feasible: {is_ok}")