
import os
import sys
import json


sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..')))
from interact.check_feasibility import check_feasible_hotel, check_feasible_restaurant, check_feasible_transport, check_feasible_attraction


# from  check_feasible_attraction import check_feasibility_attraction
# from  check_feasible_hotel import check_feasibility_hotel
# from  check_feasible_restaurant import check_feasibility_restaurant
# from  check_feasible_transport import check_feasibility_transport




def check_feasibility(meta_dict, id_list, attraction_evaluator, hotel_evaluator, restaurant_evaluator, transport_evaluator):
    
    """
    根据 id_list 提取各领域约束。
    特判：若同一个 Rubric 有多个 ID 存在于 id_list，只保留在原 Chain 中 Index 最大的那个。
    """

    # 要是id list里面没有要drop的，直接认为可行

    rollbacks = meta_dict.get("rollback_actions", [])
    dropped_ids = {item["dropped_label_id"] for item in rollbacks}
    flag = False
    for dropped_id in dropped_ids:
        if dropped_id in id_list:
            flag = True
    if not flag:
      return []





    # 1. 建立双向索引，用于快速查找 ID 所属 Rubric 及其在 Chain 中的位置
    # id_info 结构: { "hotel_1_1_1": {"r_key": "RUBRIC_COST", "index": 0}, ... }
    id_info = {}
    full_chains = meta_dict.get("applied_modification_chains", {}) or {}
    for r_key, c_ids in full_chains.items():
        for idx, cid in enumerate(c_ids):
            id_info[cid] = {"r_key": r_key, "index": idx}

    # 2. 筛选每个 Rubric 对应的“最靠后” ID
    # 使用字典记录：{ "RUBRIC_COST": {"id": "hotel_1_1_1", "index": 2}, ... }
    latest_active_ids = {}
    for cid in id_list:
        if cid not in id_info:
            continue
            
        info = id_info[cid]
        r_key = info["r_key"]
        current_idx = info["index"]
        
        # 如果该 Rubric 还没记录过，或者当前 ID 的索引比记录的更大，则更新
        if r_key not in latest_active_ids or current_idx > latest_active_ids[r_key]["index"]:
            latest_active_ids[r_key] = {"id": cid, "index": current_idx}

    # 3. 按领域（Domain）重组数据
    domain_mapping = {
        "hotel": {"rubrics": {}, "chains": {}},
        "attraction": {"rubrics": {}, "chains": {}},
        "restaurant": {"rubrics": {}, "chains": {}},
        "transport": {"rubrics": {}, "chains": {}}
    }

    for r_key, data in latest_active_ids.items():
        target_id = data["id"]
        
        # 判定领域前缀
        domain = None
        if target_id.startswith("hotel_"): domain = "hotel"
        elif target_id.startswith("attraction_"): domain = "attraction"
        elif target_id.startswith("restaurant_"): domain = "restaurant"
        elif target_id.startswith("transport_"): domain = "transport"
        
        if domain:
            # 填充该领域专用的 chains (只包含最靠后的那一个 ID)
            domain_mapping[domain]["chains"][r_key] = [target_id]
            # 填充该领域专用的 rubric 详情
            domain_mapping[domain]["rubrics"][r_key] = meta_dict["rubric_results"].get(r_key, {})
    
    # 4. 执行各领域校验
    def get_domain_feasible(domain_name, module, evaluator):
        # 如果该领域没有任何约束，视为可行
        if not domain_mapping[domain_name]["chains"]:
            return True
        
        # 构造各领域验证器期望的输入格式
        domain_meta = {
            "trip_id": meta_dict.get("trip_id"),
            "route": meta_dict.get("route", []),
            "rubric_results": domain_mapping[domain_name]["rubrics"],
            "applied_modification_chains": domain_mapping[domain_name]["chains"]
        }

        # print(json.dumps(domain_meta, indent=2))
        
        # 动态调用对应领域的 check_feasibility 函数
        # 假设函数名为 check_feasibility_hotel, check_feasibility_attraction 等
        check_func = getattr(module, f"check_feasibility_{domain_name}")
        is_ok = check_func(domain_meta, evaluator)
        return is_ok

    results = [
        get_domain_feasible("attraction", check_feasible_attraction, attraction_evaluator),
        get_domain_feasible("hotel", check_feasible_hotel, hotel_evaluator),
        get_domain_feasible("restaurant", check_feasible_restaurant, restaurant_evaluator),
        get_domain_feasible("transport", check_feasible_transport, transport_evaluator)
    ]

    feasible =  all(results)
    bad_ids = []
    if not feasible:
        rollbacks = meta_dict.get("rollback_actions", [])
        dropped_ids = {item["dropped_label_id"] for item in rollbacks}
        for dropped_id in dropped_ids:
            if dropped_id in id_list:
              bad_ids.append(dropped_id)
    return bad_ids
                


if __name__ == "__main__":
    from Evaluation.Transportation.val_transports import TransportationEvaluator
    from Evaluation.Attraction.val_attractions import AttractionEvaluator
    from Evaluation.Restaurant.val_restaurants import RestaurantEvaluator
    from Evaluation.Hotel.val_hotels import HotelEvaluator
    flight_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/flights.json"
    transport_evaluator = TransportationEvaluator(flight_path)
    hotel_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/hotels.json"
    hotel_evaluator = HotelEvaluator(hotel_path)
    attraction_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/attractions.json"
    with open(attraction_path, 'r', encoding='utf-8') as f:
        attractions_data = json.load(f)
    attraction_evaluator = AttractionEvaluator(attractions_data)
    restaurant_path = "/data/hzs/triptailor_v2_wzy/ttv2_up/data_final/restaurants.json"
    restaurant_evaluator = RestaurantEvaluator(restaurant_path)
    

    test_data = {
    "trip_id": "c2_10660",
    "route": [
      {
        "from": "Guangzhou",
        "to": "Shenzhen",
        "distance_km": 104.17,
        "depart_date": "2025-10-22",
        "stay_days": 2,
        "return_date": "2025-10-23",
        "number_of_people": 3
      }
    ],
    "rubric_results": {
      "RUBRIC_COST_TRANSPORT": {
        "Less than a certain transportation cost per person (one-way)": {
          "probability": 1,
          "description": "The one-way transportation cost per person is less than {slot}.",
          "violation_description": "The one-way transportation cost per person exceeds {slot}.",
          "generate_func": "generate_transport_cost_less_per_person_one_way",
          "validate_func": "validate_transport_cost_less_per_person_one_way",
          "result": {
            "selected_description": "100",
            "validation_params": 100,
            "all_labels_and_ranges": {
              "100": {
                "value": 100,
                "_id": "transport_3_1_1"
              },
              "200": {
                "value": 200,
                "_id": "transport_3_1_2"
              },
              "300": {
                "value": 300,
                "_id": "transport_3_1_3"
              }
            }
          },
          "_id": "transport_3_1"
        }
      },
      "RUBRIC_TIME_RETURN": {
        "Arrive after a specific time": {
          "probability": 1,
          "description": "For the return trip, arrive after {slot}.",
          "violation_description": "The return arrival time is not after {slot}.",
          "generate_func": "generate_arrive_after_time",
          "validate_func": "validate_arrive_time_condition",
          "result": {
            "selected_description": "after 14:00",
            "validation_params": "14:00-23:59",
            "all_labels_and_ranges": {
              "after 13:00": {
                "value": "13:00-23:59",
                "_id": "transport_2_8_8"
              }
            }
          },
          "_id": "transport_2_8"
        }
      },
      "RUBRIC_TIME_DEPART": {
        "Arrive after a specific time": {
          "probability": 1,
          "description": "For the outbound trip, arrive after {slot}.",
          "violation_description": "The outbound arrival time is not after {slot}.",
          "generate_func": "generate_arrive_after_time",
          "validate_func": "validate_arrive_time_condition",
          "result": {
            "selected_description": "after 19:00",
            "validation_params": "19:00-23:59",
            "all_labels_and_ranges": {
              "after 10:00": {
                "value": "10:00-23:59",
                "_id": "transport_1_8_5"
              },
              "after 12:00": {
                "value": "12:00-23:59",
                "_id": "transport_1_8_7"
              }
            }
          },
          "_id": "transport_1_8"
        }
      },
      "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": "3.5",
            "validation_params": [
              3.5
            ],
            "all_labels_and_ranges": {
              "4.5": {
                "value": [
                  4.5
                ],
                "_id": "restaurant_2_1_3"
              }
            }
          },
          "_id": "restaurant_2_1"
        }
      },
      "RUBRIC_PRICE": {
        "Per-person per-meal cost less than a certain price": {
          "probability": 1,
          "description": "Each selected restaurant has a per-person per-meal cost less than {slot}.",
          "violation_description": "Some recommended restaurants have a per-person per-meal cost equal to or higher than {slot}.",
          "generate_func": "generate_price_per_person_range",
          "validate_func": "validate_price_per_person_range",
          "result": {
            "selected_description": "600",
            "validation_params": [
              0,
              600.0
            ],
            "all_labels_and_ranges": {
              "300": {
                "value": [
                  0,
                  300.0
                ],
                "_id": "restaurant_1_1_5"
              }
            }
          },
          "_id": "restaurant_1_1"
        }
      },
      "RUBRIC_INCLUDE_CUISINE": {
        "Include certain cuisines": {
          "probability": 1,
          "description": "Make sure the plan includes restaurants serving these cuisines: {slot}.",
          "violation_description": "The plan does not include enough options for these cuisines: {slot}.",
          "generate_func": "generate_include_cuisines",
          "validate_func": "validate_include_cuisines",
          "result": {
            "selected_description": "Jiangxi Cuisine, Xinjiang Cuisine",
            "validation_params": [
              "Jiangxi Cuisine",
              "Xinjiang Cuisine"
            ],
            "all_labels_and_ranges": {
              "Chaoshan Cuisine, Wontons and Dumplings": {
                "value": [
                  "Chaoshan Cuisine",
                  "Wontons and Dumplings"
                ],
                "_id": "restaurant_4_1_9"
              }
            }
          },
          "_id": "restaurant_4_1"
        }
      },
      "RUBRIC_EXCLUDE_CUISINE": {
        "Exclude certain cuisines": {
          "probability": 1,
          "description": "Avoid restaurants that focus on these cuisines: {slot}.",
          "violation_description": "Some recommended restaurants still focus on these cuisines: {slot}.",
          "generate_func": "generate_exclude_cuisines",
          "validate_func": "validate_exclude_cuisines",
          "result": {
            "selected_description": "Taiwanese Cuisine",
            "validation_params": [
              "Taiwanese Cuisine"
            ],
            "all_labels_and_ranges": {
              "Southeast Asian Cuisine": {
                "value": [
                  "Southeast Asian Cuisine"
                ],
                "_id": "restaurant_5_1_3"
              }
            }
          },
          "_id": "restaurant_5_1"
        }
      },
      "RUBRIC_REVIEW_COUNT_HOTEL": {
        "Not less than a certain count of review": {
          "probability": 1,
          "description": "The hotel must have at least {slot} reviews.",
          "violation_description": "The hotel has fewer than {slot} reviews.",
          "generate_func": "generate_review_count",
          "validate_func": "validate_review_count",
          "result": {
            "selected_description": "200",
            "validation_params": 200,
            "all_labels_and_ranges": {
              "100": {
                "value": 100,
                "_id": "hotel_3_1_3"
              },
              "500": {
                "value": 500,
                "_id": "hotel_3_1_7"
              }
            }
          },
          "_id": "hotel_3_1"
        }
      },
      "RUBRIC_GOOD_RATE": {
        "Not less than a certain good comment rate": {
          "probability": 1,
          "description": "The hotel's positive review rate must be at least {slot}.",
          "violation_description": "The hotel's positive review rate is below {slot}.",
          "generate_func": "generate_good_rate",
          "validate_func": "validate_good_rate",
          "result": {
            "selected_description": "0.85",
            "validation_params": 0.85,
            "all_labels_and_ranges": {
              "0.75": {
                "value": 0.75,
                "_id": "hotel_4_1_1"
              },
              "0.8": {
                "value": 0.8,
                "_id": "hotel_4_1_2"
              },
              "0.85": {
                "value": 0.85,
                "_id": "hotel_4_1_3"
              }
            }
          },
          "_id": "hotel_4_1"
        }
      },
      "RUBRIC_ASPECT_RATING": {
        "All three aspect is not less than a certain rate": {
          "probability": 1,
          "description": "For the hotel, each of the product, environment, and service ratings must be at least {slot}.",
          "violation_description": "For the hotel, at least one of the product, environment, or service ratings is below {slot}.",
          "generate_func": "generate_aspect_rating_all",
          "validate_func": "validate_aspect_rating",
          "result": {
            "selected_description": "7",
            "validation_params": [
              7,
              7,
              7
            ],
            "all_labels_and_ranges": {
              "7.5": {
                "value": [
                  7.5,
                  7.5,
                  7.5
                ],
                "_id": "hotel_6_1_2"
              }
            }
          },
          "_id": "hotel_6_1"
        }
      },
      "RUBRIC_CATEGORY_PRIORITY": {
        "Prioritize attractions of specific category": {
          "probability": 1,
          "description": "The itinerary should prioritize attractions from the following categories in order of preference: {slot}.",
          "violation_description": "The itinerary does not prioritize attractions from the specified categories in the correct order: {slot}.",
          "generate_func": "generate_prioritize_category",
          "validate_func": "validate_prioritize_category",
          "result": {
            "selected_description": "1. Shopping & Food Experiences",
            "validation_params": [
              "Shopping & Food Experiences"
            ],
            "all_labels_and_ranges": {
              "1. Leisure, Wellness & Resort": {
                "value": [
                  "Leisure, Wellness & Resort"
                ],
                "_id": "attraction_9_1_5"
              }
            }
          },
          "_id": "attraction_9_1"
        }
      },
      "RUBRIC_COMMENT_COUNT": {
        "Include attractions that have more than a certain number of comments": {
          "probability": 1,
          "description": "All attractions included in the itinerary must be those that have {slot}.",
          "violation_description": "The itinerary includes attractions that do not have {slot}.",
          "generate_func": "generate_include_comment_count",
          "validate_func": "validate_include_comment_count",
          "result": {
            "selected_description": "more than 500 comments",
            "validation_params": [
              500,
              1000000000.0
            ],
            "all_labels_and_ranges": {
              "more than 100 comments": {
                "value": [
                  100,
                  1000000000.0
                ],
                "_id": "attraction_10_1_3"
              }
            }
          },
          "_id": "attraction_10_1"
        }
      },
      "RUBRIC_HEAT_SCORE": {
        "Include attractions with specific popularity": {
          "probability": 1,
          "description": "All attractions included in the itinerary must be those that are {slot}.",
          "violation_description": "The itinerary does not include all attractions that are {slot}.",
          "generate_func": "generate_include_popularity",
          "validate_func": "validate_include_popularity",
          "result": {
            "selected_description": "hot and must-visit",
            "validation_params": [
              5.2,
              1000000000.0
            ],
            "all_labels_and_ranges": {
              "hidden and less crowded": {
                "value": [
                  0,
                  3.7
                ],
                "_id": "attraction_5_1_2"
              }
            }
          },
          "_id": "attraction_5_1"
        }
      }
    },
    "applied_modification_chains": {
      "RUBRIC_COST_TRANSPORT": [
        "transport_3_1_3",
        "transport_3_1_2",
        "transport_3_1_1"
      ],
      "RUBRIC_TIME_RETURN": [
        "transport_2_8_8"
      ],
      "RUBRIC_TIME_DEPART": [
        "transport_1_8_5",
        "transport_1_8_7"
      ],
      "RUBRIC_RATING": [
        "restaurant_2_1_3"
      ],
      "RUBRIC_PRICE": [
        "restaurant_1_1_5"
      ],
      "RUBRIC_INCLUDE_CUISINE": [
        "restaurant_4_1_9"
      ],
      "RUBRIC_EXCLUDE_CUISINE": [
        "restaurant_5_1_3"
      ],
      "RUBRIC_REVIEW_COUNT_HOTEL": [
        "hotel_3_1_3",
        "hotel_3_1_7"
      ],
      "RUBRIC_GOOD_RATE": [
        "hotel_4_1_1",
        "hotel_4_1_2",
        "hotel_4_1_3"
      ],
      "RUBRIC_ASPECT_RATING": [
        "hotel_6_1_2"
      ],
      "RUBRIC_CATEGORY_PRIORITY": [
        "attraction_9_1_5"
      ],
      "RUBRIC_COMMENT_COUNT": [
        "attraction_10_1_3"
      ],
      "RUBRIC_HEAT_SCORE": [
        "attraction_5_1_2"
      ]
    },
    "rollback_actions": [
      {
        "rubric": "RUBRIC_TIME_RETURN",
        "dropped_label_id": "transport_2_8_8"
      },
      {
        "rubric": "RUBRIC_INCLUDE_CUISINE",
        "dropped_label_id": "restaurant_4_1_9"
      }
    ],
    "SAMPLE_RUBRIC_COUNT_transport": 3,
    "TOTAL_CHAIN_LENGTH_transport": 6,
    "SAMPLE_RUBRIC_COUNT_restaurant": 4,
    "TOTAL_CHAIN_LENGTH_restaurant": 4,
    "SAMPLE_RUBRIC_COUNT_hotel": 3,
    "TOTAL_CHAIN_LENGTH_hotel": 6,
    "SAMPLE_RUBRIC_COUNT_attraction": 3,
    "TOTAL_CHAIN_LENGTH_attraction": 3,
    "SAMPLE_RUBRIC_COUNT": 13,
    "TOTAL_CHAIN_LENGTH": 19
  }
    id_list =   [
        "transport_3_1_3",
        "transport_3_1_2",
        "transport_3_1_1",

        "transport_2_8_8",

        "transport_1_8_5",
        "transport_1_8_7",

        "restaurant_2_1_3",

        "restaurant_1_1_5",

        "restaurant_4_1_9",

        "restaurant_5_1_3",

        "hotel_3_1_3",
        "hotel_3_1_7",

        "hotel_4_1_1",
        "hotel_4_1_2",
        "hotel_4_1_3",

        "hotel_6_1_2",

        "attraction_9_1_5",

        "attraction_10_1_3",

        "attraction_5_1_2",
      ]

    result = check_feasibility(test_data, id_list, attraction_evaluator, hotel_evaluator, restaurant_evaluator, transport_evaluator)
    print(result)


  #   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_COST_TRANSPORT": {
  #       "Less than a certain transportation cost per person (one-way)": {
  #         "probability": 1,
  #         "description": "The one-way transportation cost per person is less than {slot}.",
  #         "violation_description": "The one-way transportation cost per person exceeds {slot}.",
  #         "generate_func": "generate_transport_cost_less_per_person_one_way",
  #         "validate_func": "validate_transport_cost_less_per_person_one_way",
  #         "result": {
  #           "selected_description": "1300",
  #           "validation_params": 1300,
  #           "all_labels_and_ranges": {
  #             "800": {
  #               "value": 800,
  #               "_id": "transport_3_1_1"
  #             }
  #           }
  #         },
  #         "_id": "transport_3_1"
  #       }
  #     },
  #     "RUBRIC_TIME_RETURN": {
  #       "Depart after a specific time": {
  #         "probability": 1,
  #         "description": "For the return trip, depart after {slot}.",
  #         "violation_description": "The return departure time is not after {slot}.",
  #         "generate_func": "generate_depart_after_time",
  #         "validate_func": "validate_depart_time_condition",
  #         "result": {
  #           "selected_description": "after 07:00",
  #           "validation_params": "07:00-23:59",
  #           "all_labels_and_ranges": {
  #             "after 07:00": {
  #               "value": "07:00-23:59",
  #               "_id": "transport_2_4_2"
  #             }
  #           }
  #         },
  #         "_id": "transport_2_4"
  #       }
  #     }
  #   },
  #   "applied_modification_chains": {
  #     "RUBRIC_COST_TRANSPORT": [
  #       "transport_3_1_1"
  #     ],
  #     "RUBRIC_TIME_RETURN": [
  #       "transport_2_4_2"
  #     ]
  #   },
  #   "intersection_count": 1,
  #   "feasible": True,
  #   "SAMPLE_RUBRIC_COUNT": 2,
  #   "TOTAL_CHAIN_LENGTH": 2
  # }
  #   print(check_feasibility_transport(test_data, transport_evaluator))