import json
from datetime import datetime, timedelta
import os
import yaml
from geopy.distance import geodesic
from fuzzywuzzy import fuzz


city_center_coords= {
    "Beijing": {
      "lon": 116.407387,
      "lat": 39.904179
    },
    "Changchun": {
      "lon": 125.323643,
      "lat": 43.816996
    },
    "Changsha": {
      "lon": 112.938882,
      "lat": 28.228304
    },
    "Chengdu": {
      "lon": 104.066301,
      "lat": 30.572961
    },
    "Chongqing": {
      "lon": 106.551787,
      "lat": 29.56268
    },
    "Dalian": {
      "lon": 121.614786,
      "lat": 38.913962
    },
    "Fuzhou": {
      "lon": 119.296411,
      "lat": 26.074286
    },
    "Guangzhou": {
      "lon": 113.264499,
      "lat": 23.130061
    },
    "Guilin": {
      "lon": 110.179752,
      "lat": 25.235615
    },
    "Guiyang": {
      "lon": 106.628201,
      "lat": 26.646694
    },
    "Haikou": {
      "lon": 110.200162,
      "lat": 20.046316
    },
    "Hangzhou": {
      "lon": 120.209903,
      "lat": 30.246566
    },
    "Harbin": {
      "lon": 126.53505,
      "lat": 45.802981
    },
    "Hong Kong": {
      "lon": 114.170714,
      "lat": 22.278354
    },
    "Jinan": {
      "lon": 117.120128,
      "lat": 36.652069
    },
    "Kaifeng": {
      "lon": 114.314278,
      "lat": 34.798083
    },
    "Kunming": {
      "lon": 102.833669,
      "lat": 24.88149
    },
    "Lijiang": {
      "lon": 100.225936,
      "lat": 26.855165
    },
    "Luoyang": {
      "lon": 112.453895,
      "lat": 34.619702
    },
    "Nanchang": {
      "lon": 115.857972,
      "lat": 28.682976
    },
    "Nanjing": {
      "lon": 118.796624,
      "lat": 32.059344
    },
    "Nanning": {
      "lon": 108.366407,
      "lat": 22.8177
    },
    "Ningbo": {
      "lon": 121.62454,
      "lat": 29.860258
    },
    "Qingdao": {
      "lon": 120.382665,
      "lat": 36.066938
    },
    "Sanya": {
      "lon": 109.511709,
      "lat": 18.252865
    },
    "Shanghai": {
      "lon": 121.473667,
      "lat": 31.230525
    },
    "Shenyang": {
      "lon": 123.464675,
      "lat": 41.677576
    },
    "Shenzhen": {
      "lon": 114.057939,
      "lat": 22.543527
    },
    "Suzhou": {
      "lon": 120.585294,
      "lat": 31.299758
    },
    "Taiyuan": {
      "lon": 112.549656,
      "lat": 37.870451
    },
    "Tianjin": {
      "lon": 117.201509,
      "lat": 39.085318
    },
    "Weihai": {
      "lon": 122.120519,
      "lat": 37.513315
    },
    "Wuhan": {
      "lon": 114.304569,
      "lat": 30.593354
    },
    "Wuxi": {
      "lon": 120.311889,
      "lat": 31.491064
    },
    "Xi'an": {
      "lon": 108.939645,
      "lat": 34.343207
    },
    "Xiamen": {
      "lon": 118.08891,
      "lat": 24.479627
    },
    "Xishuangbanna": {
      "lon": 100.797002,
      "lat": 22.009037
    },
    "Yantai": {
      "lon": 121.447755,
      "lat": 37.464551
    },
    "Zhengzhou": {
      "lon": 113.625351,
      "lat": 34.746303
    },
    "Zhuhai": {
      "lon": 113.576892,
      "lat": 22.271644
    }
  }

city_center_coords = {key.lower(): value for key, value in city_center_coords.items()}

class HotelTool:
    """
    优化版酒店搜索工具
    - 城市 fuzzy 匹配提前计算
    - 搜索经纬度提前计算
    - 日期 weekday 列表提前生成
    - cancel_policy 初始化时就转 int
    """

    # -------------------------
    # 初始化
    # -------------------------
    def __init__(self, config_path=None):
        current_dir = os.path.dirname(__file__)
        if config_path is None:
            config_path = os.path.join(os.path.dirname(current_dir), "config.yaml")

        with open(config_path, "r", encoding="utf-8") as f:
            cfg = yaml.safe_load(f)

        hotel_path = cfg.get("data_path", {}).get("hotel")
        if not hotel_path:
            raise ValueError("Missing hotel path in config.yaml (data_path.hotel)")

        with open(hotel_path, "r", encoding="utf-8") as f:
            self.hotels = json.load(f)

        # 预处理 cancel_policy（非常重要的性能优化）
        for h in self.hotels:
            try:
                h["cancel_policy"] = int(h.get("cancel_policy", -1))
            except:
                h["cancel_policy"] = -1

        # 构建 index
        self.rebuild_index()

    def _norm_city(self, s: str) -> str:
        return (s or "").strip().lower()

    def _weekday_abbr(self, date_str: str) -> str:
        date_obj = datetime.strptime(date_str, "%Y-%m-%d").date()
        week_abbr = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]
        return week_abbr[date_obj.weekday()]

    def rebuild_index(self):
        index = {}
        index_hotel = {}
        for hotel in self.hotels:
            city = self._norm_city(hotel.get("real_city", ""))
            id = str(hotel.get("Hotel_id"))
            index_hotel[id] = hotel
            if city:
                index.setdefault(city, []).append(hotel)
        self.city_index = index
        self.index_hotel = index_hotel

    # -------------------------
    # 工具函数
    # -------------------------
    def _calculate_distance(self, lat1, lon1, lat2, lon2):
        return geodesic((lat1, lon1), (lat2, lon2)).km

    def _fuzzy_match_city(self, city_key: str):
        """ fuzzy 匹配仅调用一次 """
        best_city = None
        best_score = 0
        for cname in city_center_coords:
            score = fuzz.ratio(city_key.lower(), cname.lower())
            if score > best_score:
                best_score = score
                best_city = cname
        return best_city if best_score >= 60 else None

    # -------------------------
    # 公开接口
    # -------------------------
    def search_hotels(self, city, check_in_date, check_out_date,
                      price_min=0.0, price_max=9999999.0,
                      longitude=None, latitude=None, distance_threshold=10.0,
                      hotel_type=["Economy", "Midscale", "Upscale", "Luxury"],
                      stars=0.0, review_count=0, good_remarks_rate=0.0,
                      product_rating=0.0, environment_rating=0.0, service_rating=0.0,
                      room_types=[],
                      cancel_policy=3,
                      is_pet_friendly=None, has_breakfast=None,
                      sort_key="price", sort_order="default",
                      page=1, page_size=10):
        
        # --------- 兼容字符串输入：数值/布尔/JSON列表 ---------

        # 数值（允许 "123"）
        try: price_min = float(price_min)
        except: price_min = 0.0
        try: price_max = float(price_max)
        except: price_max = 9999999.0

        if isinstance(longitude, str):
            s = longitude.strip().lower()
            longitude = None if s in ("", "none", "null") else float(longitude)
        if isinstance(latitude, str):
            s = latitude.strip().lower()
            latitude = None if s in ("", "none", "null") else float(latitude)
        if isinstance(distance_threshold, str):
            s = distance_threshold.strip().lower()
            distance_threshold = None if s in ("", "none", "null") else float(distance_threshold)

        try: stars = float(stars)
        except: stars = 0.0
        try: review_count = int(float(review_count))
        except: review_count = 0
        try: good_remarks_rate = float(good_remarks_rate)
        except: good_remarks_rate = 0.0
        try: product_rating = float(product_rating)
        except: product_rating = 0.0
        try: environment_rating = float(environment_rating)
        except: environment_rating = 0.0
        try: service_rating = float(service_rating)
        except: service_rating = 0.0

        try: cancel_policy = int(float(cancel_policy))
        except: cancel_policy = 3
        cancel_policy = max(0, min(3, cancel_policy))

        # 布尔：只支持 "true"/"false"
        if isinstance(is_pet_friendly, str):
            s = is_pet_friendly.strip().lower()
            if s == "true":
                is_pet_friendly = True
            elif s == "false":
                is_pet_friendly = False
            else:
                is_pet_friendly = None

        if isinstance(has_breakfast, str):
            s = has_breakfast.strip().lower()
            if s == "true":
                has_breakfast = True
            elif s == "false":
                has_breakfast = False
            else:
                has_breakfast = None

        # 列表：只支持 JSON 字符串
        if isinstance(hotel_type, str):
            try:
                hotel_type = json.loads(hotel_type)
            except:
                hotel_type = ["Economy", "Midscale", "Upscale", "Luxury"]
        if not isinstance(hotel_type, list):
            hotel_type = ["Economy", "Midscale", "Upscale", "Luxury"]

        if isinstance(room_types, str):
            try:
                room_types = json.loads(room_types)
            except:
                room_types = []
        if room_types and isinstance(room_types, list):
            room_types = [str(x).strip().lower() for x in room_types if str(x).strip()]
        else:
            room_types = []

        #0:当天前可取消 1:1天前可取消 2:3天前可取消 3:不可取消
        # 转换为列表格式 满足严格取消政策
        cancel_policy_list = []
        for i in range(cancel_policy+1):
            cancel_policy_list.append(i)
        cancel_policy = cancel_policy_list

        # ["Single", "Large", "Double", "Triple", "Family"]
        # ROOM_TYPE_FIELD_MAP = {
        #     "Single": "has_single",
        #     "Large": "has_large",
        #     "Double": "has_double",
        #     "Triple": "has_triple",
        #     "Family": "has_family",
        # }
        ROOM_TYPE_FIELD_MAP = {
            "single": "has_single",
            "large": "has_large",
            "double": "has_double",
            "triple": "has_triple",
            "family": "has_family",
        }


        # -------------- 分页参数清理 --------------
        try:
            page = max(1, int(page))
        except:
            page = 1

        try:
            page_size = max(1, int(page_size))
        except:
            page_size = 10

        # -------------- 日期解析 --------------
        check_in = datetime.strptime(check_in_date, "%Y-%m-%d")
        check_out = datetime.strptime(check_out_date, "%Y-%m-%d")
        days = (check_out - check_in).days
        if days <= 0:
            return "Invalid date range: check-out date must be after check-in date."

        # === 生成日期 weekday 列表 ===（仅执行一次）
        weekday_list = []
        tmp_date = check_in
        while tmp_date < check_out:
            weekday_list.append(self._weekday_abbr(tmp_date.strftime("%Y-%m-%d")))
            tmp_date += timedelta(days=1)

        # -------------- 城市索引 --------------
        city_norm = self._norm_city(city)
        # 修改：如果城市不在索引中，直接返回无效城市名
        if city_norm not in self.city_index:
            return "Invalid city name."
        candidate_hotels = self.city_index[city_norm]

        # -------------- 搜索经纬度提前计算（含 fuzzy） --------------
        search_lon = search_lat = None

        if longitude is not None and latitude is not None:
            search_lon, search_lat = longitude, latitude
        else:
            if city:
                city_key = city.strip().title()
                match_city = self._fuzzy_match_city(city_key)
                if match_city:
                    search_lon = city_center_coords[match_city]["lon"]
                    search_lat = city_center_coords[match_city]["lat"]

        # 是否需要计算距离
        need_distance = (search_lon is not None and search_lat is not None)

        # -------------- 主循环：过滤酒店 --------------
        rows = []
        for hotel in candidate_hotels:

            # 快速 get
            get = hotel.get

            # --- 快速过滤 ---
            if get("hotel_type") not in hotel_type:
                continue
            if get("stars", 0) < stars:
                continue
            if get("review_count", 0) < review_count:
                continue
            if get("good_remarks_rate", 0) < good_remarks_rate:
                continue
            if get("product_rating", 0) < product_rating:
                continue
            if get("environment_rating", 0) < environment_rating:
                continue
            if get("service_rating", 0) < service_rating:
                continue

            # ---- 房型过滤：room_types=[] 表示不筛选 ----
            # 用户传入 room_types=["Single","Large"] → 酒店必须同时包含这些房型
            if room_types:
                # AND 逻辑：所有指定房型都必须具备
                all_ok = True
                for rt in room_types:
                    field = ROOM_TYPE_FIELD_MAP.get(rt)
                    if field and not get(field, False):
                        all_ok = False
                        break
                if not all_ok:
                    continue


            # cancel_policy 现在已经是 int，不再需要 int() 转换
            if get("cancel_policy") not in cancel_policy:
                continue

            if is_pet_friendly is not None and get("is_pet_friendly") != is_pet_friendly:
                continue
            if has_breakfast is not None and get("has_breakfast") != has_breakfast:
                continue

            # ----------- 计算价格（使用 weekday_list） -----------
            sum_price = 0
            valid_price = True

            for wd in weekday_list:
                p = get(f"{wd}_lowest_price")
                if p is None:
                    valid_price = False
                    break
                sum_price += p

            if not valid_price:
                continue

            avg_price = sum_price / days
            if not (price_min <= avg_price <= price_max):
                continue

            # ----------- 计算距离（仅当 need_distance=True） -----------
            distance = float("inf")
            if need_distance:
                hotel_lat = get("latitude")
                hotel_lon = get("longitude")
                if hotel_lat and hotel_lon:
                    distance = self._calculate_distance(search_lat, search_lon, hotel_lat, hotel_lon)
                    if distance_threshold is not None and distance > distance_threshold:
                        continue

            # ----------- 保存 ----------- 
            rows.append({
                "_price": avg_price,
                "_stars": get("stars", 0),
                "_review_count": get("review_count", 0),
                # "_good_remarks": get("good_remarks_rate", 0),
                "min_avg_price_per_night": round(avg_price, 2),
                "distance": round(distance, 2),
                "data": hotel
            })

        if not rows:
            return "No matching hotels found."

        # -------------- 排序 --------------
        # 智能选择排序方式：当 sort_order 为 "default" 时
        if sort_order.lower() == "default":
            # price 和 distance 默认升序
            if sort_key in ["price", "distance"]:
                reverse = False
            # stars 和 review_count 默认降序
            elif sort_key in ["stars", "review_count"]:
                reverse = True
            else:
                # 默认情况下使用升序（价格排序）
                reverse = False
        else:
            # 其他情况按照用户指定的排序顺序
            reverse = (sort_order.lower() == "desc")

        if sort_key == "price":
            rows.sort(key=lambda r: r["_price"], reverse=reverse)
        elif sort_key == "stars":
            rows.sort(key=lambda r: r["_stars"], reverse=reverse)
        elif sort_key == "review_count":
            rows.sort(key=lambda r: r["_review_count"], reverse=reverse)
        elif sort_key == "distance":
            rows.sort(key=lambda r: r["distance"], reverse=reverse)
        else:
            rows.sort(key=lambda r: r["_price"], reverse=reverse)

        # -------------- 分页 --------------
        total = len(rows)
        start = (page - 1) * page_size
        end = min(start + page_size, total)
        page_rows = rows[start:end]

        formatted = [
            f"Showing {start + 1}-{end} of {total} results."
        ]

        for r in page_rows:
            h = r["data"]
            # formatted.append(
            #     f"Hotel_id: {h.get('Hotel_id')} | {h.get('real_city')} | {h.get('name')} | "
            #     f"type: {h.get('hotel_type')} | price: {r['min_avg_price_per_night']}+ | "
            #     f"stars: {h.get('stars')}/5 | comments: {int(h.get('review_count', 0))} | "
            #     f"distance: {r['distance']}km"
            # )
            formatted.append(
                f"Hotel_id: {h.get('Hotel_id')} | {h.get('name')} | "
                f"type: {h.get('hotel_type')} | price: {r['min_avg_price_per_night']}+ | "
                f"stars: {h.get('stars')}/5 | review_count: {int(h.get('review_count', 0))} | "
                f"longitude: {h.get('longitude')}, latitude: {h.get('latitude')} | "
                f"distance: {r['distance']}km"
            )

        return "\n".join(formatted)

    def get_hotel_detail_with_products(self, hotel_id, check_in_date, check_out_date, room_num = 1, person_num=1,
                                        room_type=[], min_breakfast_per_room=0,
                                        cancel_policy=3, has_window=None,
                                        # sort_key="default", sort_order="asc",
                                        page=1, page_size=10
                                    ):
        """
        获取酒店详细信息及其产品

        参数:
            hotel_id: 酒店ID，对应酒店数据中的 `Hotel_id` 字段。
            check_in_date: 入住日期 (YYYY-MM-DD)。
            check_out_date: 退房日期 (YYYY-MM-DD)。
            room_num: 需要预订的房间数量。
            person_num: 总入住人数，用于判断在给定 room_num 下，每个产品是否“住得下”。
                        住得下的产品会排在前面，住不下的会排在后面。
            room_type: 可选，房型过滤，例如 ["Single", "Large", "Double", "Triple", "Family"]。
                       传空列表表示不过滤房型。
            min_breakfast_per_room: 每间房最少包含的早餐数量（小于该值的产品会被过滤）。
            cancel_policy: 可接受的“最严格”取消政策上限：
                           0: 只接受 0（当天12点前可取消）
                           1: 接受 0/1
                           2: 接受 0/1/2
                           3: 接受 0/1/2/3（包含不可取消，相当于不限制取消政策）
            has_window: 是否有窗户。True/False 表示强制过滤，有窗/无窗；None 表示不过滤。
            page: 页码，从 1 开始。
            page_size: 每页返回的产品数量。

        返回:
            一个多行字符串:
              - 第一行: 酒店摘要信息（Hotel_id、城市、类型、星级、评分、经纬度、是否宠物友好等）。
              - 第二行: 产品分页信息，例如 "Showing 1-10 of 23 products."
              - 后续若干行: 每个符合条件的产品信息。
                对于不满足 room_num 和 person_num 组合的产品，前面会插入一行:
                "The following rooms do not meet the room-count and occupancy criteria:"
        """

        # --------- 兼容字符串输入：数值/布尔/JSON列表 ---------
        hotel_id = str(hotel_id)

        try: room_num = int(float(room_num))
        except: room_num = 1
        room_num = max(1, room_num)

        try: person_num = int(float(person_num))
        except: person_num = 1
        person_num = max(1, person_num)

        try: min_breakfast_per_room = int(float(min_breakfast_per_room))
        except: min_breakfast_per_room = 0
        min_breakfast_per_room = max(0, min_breakfast_per_room)

        try: cancel_policy = int(float(cancel_policy))
        except: cancel_policy = 3
        cancel_policy = max(0, min(3, cancel_policy))

        # 布尔：只支持 "true"/"false"
        if isinstance(has_window, str):
            s = has_window.strip().lower()
            if s == "true":
                has_window = True
            elif s == "false":
                has_window = False
            else:
                has_window = None

        # 列表：只支持 JSON 字符串
        if isinstance(room_type, str):
            try:
                room_type = json.loads(room_type)
            except:
                room_type = []
        if room_type and isinstance(room_type, list):
            room_type = [str(x).strip().lower() for x in room_type if str(x).strip()]
        else:
            room_type = []

        try: page = max(1, int(float(page)))
        except: page = 1
        try: page_size = max(1, int(float(page_size)))
        except: page_size = 10


        # 查找酒店
        hotel = self.index_hotel.get(str(hotel_id))
        if not hotel:
            return f"Hotel not found: {hotel_id}"
        
        # 日期解析
        check_in = datetime.strptime(check_in_date, "%Y-%m-%d")
        check_out = datetime.strptime(check_out_date, "%Y-%m-%d")
        days = (check_out - check_in).days
        if days <= 0:
            return "Invalid date range: check-out date must be after check-in date."
        
        # 生成日期 weekday 列表
        weekday_list = []
        tmp_date = check_in
        while tmp_date < check_out:
            weekday_list.append(self._weekday_abbr(tmp_date.strftime("%Y-%m-%d")))
            tmp_date += timedelta(days=1)
        
        # 转换取消政策为列表格式（兼容search_hotels的逻辑）
        cancel_policy_list = []
        for i in range(cancel_policy+1):
            cancel_policy_list.append(i)
        cancel_policy = cancel_policy_list
        
        # 酒店摘要信息
        # summary_line = (
        #     f"Hotel_id: {hotel.get('Hotel_id')} | {hotel.get('name')} | {hotel.get('real_city')} | "
        #     f"type: {hotel.get('hotel_type')} | stars: {hotel.get('stars')}/5 | "
        #     f"longitude: {hotel.get('longitude')} | latitude: {hotel.get('latitude')} | "
        #     f"product_rating: {hotel.get('product_rating')} | environment_rating: {hotel.get('environment_rating')} | "
        #     f"service_rating: {hotel.get('service_rating')} | review_count: {hotel.get('review_count')} | "
        #     f"pet_friendly: {hotel.get('is_pet_friendly')}"
        # )
        summary_line = (
            f"Hotel_id: {hotel.get('Hotel_id')} | {hotel.get('real_city')} | "
            f"type: {hotel.get('hotel_type')} | stars: {hotel.get('stars')}/5 | review_count: {hotel.get('review_count')} | "
            f"longitude: {hotel.get('longitude')} | latitude: {hotel.get('latitude')} | "
            f"product_rating: {hotel.get('product_rating')} | environment_rating: {hotel.get('environment_rating')} | service_rating: {hotel.get('service_rating')} |"
            f"pet_friendly: {hotel.get('is_pet_friendly')}"
        )
        
        # 筛选产品
        def _norm(x): return (str(x) if x is not None else "").strip().lower()
        
        products = hotel.get("products", []) or []
        filtered_products = []
        
        for p in products:
            # 房型过滤
            if room_type and _norm(p.get("room_type")) not in [_norm(rt) for rt in room_type]:
                continue
            
            # 早餐数量过滤
            if p.get("breakfast_num", 0) < min_breakfast_per_room:
                continue
            
            # 取消政策过滤
            if p.get("cancel_policy") not in cancel_policy:
                continue
            
            # 窗户过滤
            if has_window is not None and p.get("has_window") != has_window:
                continue
            
            # 计算产品价格（参考search_hotels的逻辑）
            sum_price = 0
            valid_price = True
            
            for wd in weekday_list:
                day_price = p.get(wd)
                if day_price is None:
                    valid_price = False
                    break
                sum_price += day_price
            
            if not valid_price:
                continue
            
            # 计算总价和均价
            avg_price = sum_price / days
            total_price = sum_price * room_num  # 总价乘以房间数
            
            # 添加价格信息到产品
            p_with_price = p.copy()
            p_with_price["_avg_price"] = avg_price
            p_with_price["_total_price"] = total_price
            filtered_products.append(p_with_price)
        
        # 排序
        # reverse = (sort_order.lower() == "desc")

        # # 定义房型排序顺序
        # room_type_order = ["single", "large", "double", "triple", "family"]

        # filtered_products.sort(key=lambda p: (
        #     room_type_order.index(p.get("room_type", "")) if p.get("room_type", "") in room_type_order else len(room_type_order),
        #     p["_total_price"]
        # ))

                # -----------------------
       
                # 排序逻辑改造：
        # 1）先根据 room_num 和 person_num 判断某个 product（房型）能不能住得下
        # 2）对于“住得下”的产品：
        #    - 按 room_type 分组
        #    - 计算每个 room_type 的最低价
        #    - 先按房型最低价对房型排序，再在每个房型内部按价格从低到高排序
        # 3）对于“住不下”的产品，重复 2）中的流程，排在“住得下”之后

        room_type_order = ["single", "large", "double", "triple", "family"]

        def can_accommodate(p):
            """判断该产品在给定 room_num 和 person_num 下是否住得下"""
            max_person_per_room = p.get("person_num") or 0
            return max_person_per_room * room_num >= person_num

        from collections import defaultdict

        def sort_group(products_group):
            """
            对一组产品（要么全是住得下，要么全是住不下）按房型->价格排序：
            1. 按 room_type 分组
            2. 每个房型算一个最低价
            3. 房型按最低价升序排序（价格相同再按预设 room_type_order）
            4. 每个房型内部按总价升序
            """
            # 1. 按房型分组
            by_type = defaultdict(list)
            for p in products_group:
                rt = p.get("room_type") or ""
                by_type[rt].append(p)

            # 2. 计算每个房型的最低价（这里用 _total_price，_avg_price 也可以，排序结果等价）
            type_min_price = {}
            for rt, plist in by_type.items():
                type_min_price[rt] = min(pp["_total_price"] for pp in plist)

            # 3. 先对房型排序
            def room_type_sort_key(rt):
                # 如果在标准房型列表里，用其索引做次级排序，否则放到最后
                base_idx = room_type_order.index(rt) if rt in room_type_order else len(room_type_order)
                return (
                    type_min_price[rt],  # 先按该房型的最低价
                    base_idx,            # 再按预设房型顺序
                    rt                   # 最后按房型名稳定排序，避免不确定性
                )

            sorted_room_types = sorted(by_type.keys(), key=room_type_sort_key)

            # 4. 在每个房型内部按总价升序，再拼接
            result = []
            for rt in sorted_room_types:
                plist = by_type[rt]
                plist_sorted = sorted(plist, key=lambda x: x["_total_price"])
                result.extend(plist_sorted)
            return result

        # 拆成“住得下”和“住不下”两组
        can_live_products = []
        cannot_live_products = []
        for p in filtered_products:
            if can_accommodate(p):
                can_live_products.append(p)
            else:
                cannot_live_products.append(p)

        # 各自按规则排序，再合并：住得下的在前，住不下的在后
        sorted_can_live = sort_group(can_live_products) if can_live_products else []
        sorted_cannot_live = sort_group(cannot_live_products) if cannot_live_products else []
        filtered_products = sorted_can_live + sorted_cannot_live

        
        # 分页
        try:
            page = max(1, int(page))
        except:
            page = 1
        
        try:
            page_size = max(1, int(page_size))
        except:
            page_size = 10
        
        total = len(filtered_products)
        start = (page - 1) * page_size
        end = min(start + page_size, total)
        paged_products = filtered_products[start:end]
                 
        # 格式化产品信息
        # 添加取消政策映射字典
        cancel_policy_map = {
            0: "Free cancellation before 12:00 PM on the check-in day",
            1: "Free cancellation up to 1 day before check-in",
            2: "Free cancellation up to 3 days before check-in",
            3: "Non-cancellable"
        }
        
        # product_lines = []
        # for p in paged_products:
        #     cancel_policy_num = p.get('cancel_policy')
        #     cancel_policy_desc = cancel_policy_map.get(cancel_policy_num, f"Unknown ({cancel_policy_num})")
        #     product_lines.append(
        #         f"product_id: {p.get('product_id')} | room_type: {p.get('room_type')} | "
        #         f"maximum occupancy per room: {p.get('person_num')} | breakfast_num: {p.get('breakfast_num')} | "
        #         f"cancel_policy: {cancel_policy_num} ({cancel_policy_desc}) | has_window: {p.get('has_window')} | "
        #         f"avg_price_per_night_per_room: {round(p['_avg_price'], 2)} | "
        #         # f"total_price({days} nights, {room_num} room): {round(p['_total_price'], 2)}"
        #     )
        product_lines = []
        printed_cannot_header = False   # “住不下的产品”标题是否已打印

        for p in paged_products:
            # 根据 room_num 和 person_num 判断当前产品是否住得下
            is_can_accommodate = can_accommodate(p)

            # 只在每个组第一次出现时加标题
            if (not is_can_accommodate) and not printed_cannot_header:
                product_lines.append("The following rooms do not meet the room-count and occupancy criteria:")
                printed_cannot_header = True

            cancel_policy_num = p.get('cancel_policy')
            cancel_policy_desc = cancel_policy_map.get(cancel_policy_num, f"Unknown ({cancel_policy_num})")

            product_lines.append(
                f"product_id: {p.get('product_id')} | room_type: {p.get('room_type')} | "
                f"maximum occupancy per room: {p.get('person_num')} | breakfast_num: {p.get('breakfast_num')} | "
                f"cancel_policy: {cancel_policy_num} ({cancel_policy_desc}) | has_window: {p.get('has_window')} | "
                f"avg_price_per_night_per_room: {round(p['_avg_price'], 2)} | "
                # f"total_price({days} nights, {room_num} room): {round(p['_total_price'], 2)}"
            )

        
        if not product_lines:
            return f"No products available for {hotel_id} from {check_in_date} to {check_out_date} with the specified filters."
        
        # 添加分页信息
        pagination_info = f"Showing {start + 1}-{end} of {total} products."
        
        return "\n".join([summary_line, pagination_info] + product_lines)
    
    def get_hotel_coordinates(self, hotel_id):
        """
        根据酒店ID获取经纬度坐标
        
        参数:
            hotel_id: 酒店ID
            
        返回:
            包含酒店ID、名称、经度和纬度的字符串
        """
        if not hotel_id:
            return "No hotel ID provided."
        
        # 根据酒店ID查找
        # hotel = self.index_hotel.get(hotel_id)
        hotel = self.index_hotel.get(str(hotel_id))

        if not hotel:
            return "No hotel found with the provided ID."
        
        return (    
                    f"Hotel ID: {hotel.get('Hotel_id')} | "
                    f"Name: {hotel.get('name')} | "
                    f"latitude: {hotel.get('latitude')} | longitude: {hotel.get('longitude')}"
                )



if __name__ == "__main__":
    # 测试代码
    tool = HotelTool()
    hotels = tool.get_hotel_detail_with_products(
        hotel_id="Hotel_00000001",
        check_in_date="2025-11-08",
        check_out_date="2025-11-11",
        room_num=2,
        person_num=6,
        room_type=["Large","Double","Triple","Family"],
        min_breakfast_per_room=0,
        cancel_policy=3,
        has_window=True,
        page=1,
        page_size=100
    )
    print(hotels)


    hotels = tool.search_hotels(
        city="Beijing",
        check_in_date="2025-11-08",
        check_out_date="2025-11-11",
        hotel_type=["Economy"],
        sort_key="review_count",
        sort_order="default",
        page=1,
        page_size=5
    )
    print(hotels)


    hotel = tool.get_hotel_coordinates(hotel_id="Hotel_00000001")
    print(hotel)

    # hotels = tool.search_hotels(
    #     city="Changchun",
    #     check_in_date="2025-11-08",
    #     check_out_date="2025-11-11",
    #     person_num=2,
    #     room_num=1,
    #     hotel_type=["Economy"],
    #     cancel_policy=2,
    #     sort_key="distance",
    #     sort_order="default",
    #     page=1,
    #     page_size=10
    # )
    # print(hotels)

    # hotels = tool.search_hotels(
    #     city="Changchun",
    #     check_in_date="2025-11-08",
    #     check_out_date="2025-11-11",
    #     person_num=2,
    #     room_num=1,
    #     hotel_type=["Economy"],
    #     cancel_policy=1,
    #     sort_key="distance",
    #     sort_order="default",
    #     page=1,
    #     page_size=10
    # )
    # print(hotels)

    # hotels = tool.search_hotels(
    #     city="Changchun",
    #     check_in_date="2025-11-08",
    #     check_out_date="2025-11-11",
    #     person_num=2,
    #     room_num=1,
    #     hotel_type=["Economy"],
    #     room_types=["Single","Large","Double","Triple","Family"],
    #     sort_key="distance",
    #     sort_order="default",
    #     page=1,
    #     page_size=10
    # )
    # print(hotels)