import sqlite3
import os
import json
from typing import List, Dict, Optional
from datetime import datetime, timedelta
import difflib

# Import shared functions from tasks/solver_utils.py
import sys
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..', '..', 'tasks'))
from solver_utils import calculate_rooms_needed, group_offers_by_configuration

class Accommodations:
    """A class to handle hotel and accommodation-related queries."""

    def __init__(self, db_path="api_data/accommodations.sqlite"):
        """Initializes the tool, pointing to the SQLite database."""
        project_root = os.path.join(os.path.dirname(__file__), '..', '..', '..')
        self.db_path = os.path.join(project_root, db_path)
        if not os.path.exists(self.db_path):
            raise FileNotFoundError(f"Database not found at {self.db_path}. Please run 'api_data/process_raw_data.py' first.")
        
        # Hardcoded location lists (static data from database)
        self.national_parks = [
            "Acadia National Park", "Arches National Park", "Bryce Canyon National Park",
            "Death Valley National Park", "Everglades National Park", "Glacier National Park",
            "Grand Canyon", "Grand Teton National Park", "Great Smoky Mountains National Park",
            "Hawaii Volcanoes National Park", "Joshua Tree National Park", "Mammoth Cave National Park",
            "Mount Rainier National Park", "Olympic National Park", "Rocky Mountain National Park",
            "Sequoia National Park", "Shenandoah National Park", "Yellowstone National Park",
            "Yosemite National Park", "Zion National Park"
        ]
        
        self.cities = [
            "Amargosa Valley", "Ash Fork", "Ashford", "Bar Harbor", "Beatty", "Bigfork",
            "Brownsville", "Bryce Canyon", "Bryson City", "Cannonville", "Cardston County",
            "Cathedral City", "Cave City", "Charlottesville", "Cherokee", "Columbia Falls",
            "Coral Springs", "Crystal Mountain (Washington)", "Culpeper", "Curry Village",
            "Death Valley", "Death Valley Junction", "Doral", "East Coram (Montana)",
            "East Glacier Park", "Eatonville (Washington)", "El Portal", "Enumclaw",
            "Essex (Montana)", "Estes Park", "Etlan", "Flagstaff", "Forks", "Front Royal",
            "Gatlinburg", "Gordonsville (Virginia)", "Grand Lake", "Harrisonburg", "Henrieville",
            "Hildale", "Hilldale (Utah)", "Homestead", "Independence", "Indian Wells", "Indio",
            "Jackson", "Joshua Tree", "Kalispell", "Kanab", "Kendall", "Kernville",
            "La Quinta", "La Verkin", "Lindsay (California)", "Locust Dale", "Lone Pine",
            "Luray", "Mammoth Cave", "Massanutten", "Meadview  (Arizona)", "Miramar", "Moab",
            "North Wawona (California)", "Olancha", "Packwood", "Pahrump", "Palm Desert",
            "Palm Springs", "Park City", "Peach Springs", "Pembroke Pines", "Pigeon Forge",
            "Pioneertown (California)", "Port Angeles", "Porterville", "Puyallup", "Quinault",
            "Rancho Mirage", "Seligman", "Sequim", "Sequoia", "Shenandoah", "Springdale",
            "Sunrise", "Syria (Virginia)", "Three Rivers", "Townsend", "Tropic", "Tuba City",
            "Tusayan", "Twentynine Palms", "Valle", "Virgin", "Volcano", "Waterton Park",
            "Wawona", "West Glacier", "West Yellowstone", "Weston", "Whitefish", "Widtsoe",
            "Williams", "Wofford Heights", "Woodstock", "Yosemite West", "Yucca Valley"
        ]
        
        self.all_locations = self.national_parks + self.cities

    def _calculate_location_matches(self, query: str, min_confidence: float = 0.2, max_results: int = 10) -> List[Dict]:
        """
        Shared method to calculate fuzzy location matches with confidence scores.
        
        Args:
            query: The location search query
            min_confidence: Minimum confidence threshold for matches
            max_results: Maximum number of results to return
            
        Returns:
            List of matches with location_name, confidence_score, and location_type
        """
        if not query or not query.strip():
            return []
        
        query_lower = query.strip().lower()
        results = []
        
        for location in self.all_locations:
            location_lower = location.lower()
            
            # Calculate different similarity metrics
            seq_similarity = difflib.SequenceMatcher(None, query_lower, location_lower).ratio()
            
            # Boost score for substring matches
            if query_lower in location_lower or location_lower in query_lower:
                substring_bonus = 0.3
            else:
                substring_bonus = 0.0
            
            # Boost score for word-level matches (handles "Smoky" -> "Great Smoky Mountains")
            query_words = query_lower.split()
            location_words = location_lower.split()
            word_matches = sum(1 for word in query_words if any(word in loc_word for loc_word in location_words))
            word_bonus = (word_matches / len(query_words)) * 0.4 if query_words else 0
            
            # Final confidence score
            confidence = seq_similarity + substring_bonus + word_bonus
            confidence = min(confidence, 1.0)  # Cap at 1.0
            
            # Only include if confidence is above threshold
            if confidence >= min_confidence:
                location_type = "national_park" if location in self.national_parks else "city"
                results.append({
                    "location_name": location,
                    "confidence_score": round(confidence, 3),
                    "location_type": location_type
                })
        
        # Sort by confidence score (highest first) and limit results
        results.sort(key=lambda x: x["confidence_score"], reverse=True)
        return results[:max_results]

    def search_location_name(self, query: str) -> Dict:
        """
        Search for location names using fuzzy matching to help find the correct location name.
        Args:
            query: The location search query (e.g., "Acadia", "Smoky Mountains", "Yellowstone")
        Returns:
            Dict with matches, containing location names and confidence scores
        """
        if not query or not query.strip():
            return {"error": "Location query cannot be empty."}
        
        # Use shared matching logic
        results = self._calculate_location_matches(query, min_confidence=0.2, max_results=10)
        
        if not results:
            return {
                "matches": [],
                "message": f"No location matches found for '{query}'. Try a different search term.",
                "available_examples": {
                    "national_parks": self.national_parks[:5],  # Show first 5 as examples
                    "cities": self.cities[:5]
                }
            }
        
        # If we have an exact match, put it first
        query_lower = query.strip().lower()
        exact_matches = [r for r in results if r["location_name"].lower() == query_lower]
        if exact_matches:
            # Move exact match to front
            results = exact_matches + [r for r in results if r not in exact_matches]
        
        return {
            "matches": results,
            "message": f"Found {len(results)} location match(es) for '{query}'."
        }

    def _find_best_location_match(self, query: str) -> Optional[str]:
        """
        Helper method to find the best fuzzy location match with high confidence.
        Returns None if no good match found (confidence < 0.7).
        """
        # Use shared matching logic with high confidence threshold
        results = self._calculate_location_matches(query, min_confidence=0.7, max_results=1)
        
        # Return the best match if one exists, otherwise None
        return results[0]["location_name"] if results else None

    def search_hotels(
        self,
        location: str,
        start_date: str,
        end_date: str,
        num_guests: int = 1,
        min_price_nightly: Optional[float] = None,
        max_price_nightly: Optional[float] = None,
        min_review_score: Optional[float] = None,
        free_cancellation: Optional[bool] = None,
        no_prepayment_needed: Optional[bool] = None,
        has_parking: Optional[bool] = None,
        breakfast_included: Optional[bool] = None,
        is_pet_friendly: Optional[bool] = None
    ) -> Dict:
        """
        Search for hotels and return group booking packages for a given location and date range.
        min_price_nightly and max_price_nightly refer to the per room per night price cap (in USD).
        """
        try:
            start_dt = datetime.fromisoformat(start_date)
            end_dt = datetime.fromisoformat(end_date)
            num_nights = (end_dt - start_dt).days
            if num_nights <= 0:
                return {"error": "End date must be after the start date."}
        except ValueError:
            return {"error": "Invalid date format. Please use YYYY-MM-DD."}

        if num_guests <= 0:
            return {"error": "Number of guests must be at least 1."}

        # Query to get all offers for the date range and location
        query = """
        SELECT
            h.hotel_id,
            h.name AS hotel_name,
            h.star_rating,
            h.review_score,
            h.distance_to_park_km,
            h.national_park,
            h.city,
            r.room_id,
            r.room_name,
            r.room_type,
            o.offer_config_id,
            o.cancellation_policy,
            o.meal_plan,
            o.max_occupancy,
            o.price,
            o.date,
            o.rooms_available,
            o.is_prepayment_needed
        FROM offers o
        JOIN rooms r ON o.room_id = r.room_id
        JOIN hotels h ON r.hotel_id = h.hotel_id
        """
        
        params = []
        where_clauses = [
            "o.date >= ?",
            "o.date < ?",
            "(h.city = ? OR h.national_park = ?)"
        ]
        params.extend([start_date, end_date, location, location])

        # Add basic filters (keep limited for agent tool)
        if min_review_score:
            where_clauses.append("h.review_score >= ?")
            params.append(min_review_score)
        if has_parking:
            where_clauses.append("h.has_parking = 1")
        if is_pet_friendly:
            where_clauses.append("h.is_pet_friendly = 1")
        if breakfast_included:
            where_clauses.append("o.meal_plan LIKE '%Breakfast%'")
        if free_cancellation:
            where_clauses.append("o.cancellation_policy NOT LIKE '%Non-refundable%'")
        if no_prepayment_needed:
            where_clauses.append("o.is_prepayment_needed = 0")
            
        if where_clauses:
            query += " WHERE " + " AND ".join(where_clauses)
            
        query += " ORDER BY h.hotel_id, o.offer_config_id, o.date;"

        try:
            conn = sqlite3.connect(self.db_path)
            conn.row_factory = sqlite3.Row
            cursor = conn.cursor()
            results = cursor.execute(query, tuple(params)).fetchall()

            if not results:
                # Try fuzzy location matching as fallback
                best_location_match = self._find_best_location_match(location)
                
                if best_location_match and best_location_match != location:
                    # Try the search again with the corrected location
                    corrected_params = params.copy()
                    corrected_params[2] = best_location_match  # Update both location references
                    corrected_params[3] = best_location_match
                    
                    corrected_results = cursor.execute(query, tuple(corrected_params)).fetchall()
                    conn.close()
                    
                    if corrected_results:
                        hotels_with_packages = self._generate_hotel_packages(corrected_results, num_nights, num_guests, min_price_nightly, max_price_nightly, start_date, end_date)
                        return {
                            "hotels": hotels_with_packages,
                            "location_correction": {
                                "original_location": location,
                                "corrected_to": best_location_match,
                                "message": f"No results found for '{location}'. Showing results for '{best_location_match}' instead."
                            }
                        }
                else:
                    conn.close()
                
                return {"hotels": [], "message": "No offers found matching the specified criteria."}

            # Close connection and process results
            conn.close()
            
            # Group offers and generate packages
            hotels_with_packages = self._generate_hotel_packages(results, num_nights, num_guests, min_price_nightly, max_price_nightly, start_date, end_date)
            
            return {"hotels": hotels_with_packages}

        except Exception as e:
            return {"error": f"Could not perform search due to a database issue: {e}"}

    def _generate_hotel_packages(self, offers: List, num_nights: int, num_guests: int, min_price_nightly: Optional[float], max_price_nightly: Optional[float], start_date: str, end_date: str) -> List[Dict]:
        """
        Generate group booking packages from raw offers using the same logic as ground truth solver.
        min_price_nightly and max_price_nightly refer to the per room per night price cap (in USD).
        """
        # Group offers by configuration using shared function
        offer_configs = group_offers_by_configuration(offers, num_nights)
        
        # Group by hotel
        hotels_dict = {}
        
        for config in offer_configs:
            offer_data = config['offer_data']
            hotel_id = offer_data['hotel_id']
            
            # Calculate rooms needed for this configuration using shared function
            rooms_needed = calculate_rooms_needed(num_guests, offer_data['max_occupancy'])
            
            # Check if enough rooms available
            if config['rooms_available'] < rooms_needed:
                continue
            
            # Calculate total cost for all rooms
            total_cost = config['total_price'] * rooms_needed
            
            # Apply price filters to total cost per room per night
            if min_price_nightly and config['avg_price_per_night'] < min_price_nightly:
                continue
            if max_price_nightly and config['avg_price_per_night'] > max_price_nightly:
                continue
            
            # Create hotel entry if it doesn't exist
            if hotel_id not in hotels_dict:
                description = "N/A"
                if offer_data['distance_to_park_km'] and offer_data['national_park']:
                    description = f"{offer_data['distance_to_park_km']} km from {offer_data['national_park']}"
                elif offer_data['city']:
                    description = f"Located in {offer_data['city']}"
                
                hotels_dict[hotel_id] = {
                    'hotel_id': hotel_id,
                    'hotel_name': offer_data['hotel_name'],
                    'star_rating': offer_data['star_rating'],
                    'review_score': offer_data['review_score'],
                    'description': description,
                    'group_packages': []
                }
            
            # Create package ID with config, date range, and room count for uniqueness
            # Extract day from dates (since all dates are in Aug 2025)
            start_day = start_date.split('-')[2]  # Extract day from YYYY-MM-DD
            end_day = end_date.split('-')[2]
            package_id = f"pkg_{offer_data['offer_config_id']}_{start_day}_{end_day}_{rooms_needed}"
            
            package = {
                'package_id': package_id,
                'description': f"{num_guests} guests in {rooms_needed} {offer_data['room_name']}",
                'rooms_needed': rooms_needed,
                'room_configuration': {
                    'room_id': offer_data['room_id'],
                    'room_type': offer_data['room_type'],
                    'room_name': offer_data['room_name'],
                    'max_occupancy_per_room': offer_data['max_occupancy']
                },
                'pricing': {
                    'total_cost': round(total_cost, 2)
                },
                'policies': {
                    'cancellation_policy': offer_data['cancellation_policy'],
                    'meal_plan': offer_data['meal_plan'],
                    'prepayment_needed': bool(offer_data.get('is_prepayment_needed', False))
                }
            }
            
            hotels_dict[hotel_id]['group_packages'].append(package)
        
        # Convert to list and filter out hotels with no packages
        hotels_list = [hotel for hotel in hotels_dict.values() if hotel['group_packages']]
        
        return hotels_list



    def get_hotel_details(self, hotel_identifier: str) -> Dict:
        """
        Retrieves full details for a specific hotel by its ID or name (supports fuzzy matching).
        Args:
            hotel_identifier: Either hotel_id or hotel_name (supports partial matches)
        """
        # Try to find by hotel_id first (exact match)
        query1 = "SELECT * FROM hotels WHERE hotel_id = ?"
        
        # If not found, try by exact hotel name (case-insensitive)
        query2 = "SELECT * FROM hotels WHERE LOWER(name) = LOWER(?)"
        
        # If still not found, try partial matching
        query3 = "SELECT * FROM hotels WHERE LOWER(name) LIKE LOWER(?)"
        
        try:
            conn = sqlite3.connect(self.db_path)
            conn.row_factory = sqlite3.Row
            cursor = conn.cursor()
            
            # First try by ID
            result = cursor.execute(query1, (hotel_identifier,)).fetchone()
            
            # If not found by ID, try exact name match
            if not result:
                result = cursor.execute(query2, (hotel_identifier,)).fetchone()
            
            # If still not found, try partial matching
            if not result:
                fuzzy_pattern = f"%{hotel_identifier}%"
                results = cursor.execute(query3, (fuzzy_pattern,)).fetchall()
                
                if len(results) == 1:
                    result = results[0]
                elif len(results) > 1:
                    # Multiple matches - return list of options
                    conn.close()
                    matches = [{"hotel_id": row["hotel_id"], "name": row["name"], "star_rating": row["star_rating"]} for row in results]
                    return {
                        "error": f"Multiple hotels found matching '{hotel_identifier}'. Please be more specific:",
                        "matches": matches
                    }
            
            conn.close()

            if not result:
                return {"error": f"Hotel with identifier '{hotel_identifier}' not found. Please check the hotel name or ID."}
            
            hotel_details = dict(result)
            
            # Convert integer boolean fields to proper Python booleans
            boolean_fields = [
                'has_pool', 'has_gym', 'has_spa', 'has_bar', 'has_restaurant', 
                'has_wifi', 'has_parking', 'is_pet_friendly', 'has_airport_shuttle'
            ]
            for field in boolean_fields:
                if field in hotel_details and hotel_details[field] is not None:
                    hotel_details[field] = bool(hotel_details[field])
            
            return hotel_details

        except Exception as e:
            return {"error": f"Could not retrieve hotel details due to a database issue: {e}"}

    def get_room_details(self, package_id: str) -> Dict:
        """
        Retrieves full details for a specific room using a package_id.
        Args:
            package_id: Package ID in format "pkg_{offer_config_id}_{start_day}_{end_day}_{rooms_needed}"
        """
        # Extract offer_config_id from package_id
        if not package_id.startswith("pkg_"):
            return {"error": f"Invalid package_id format: '{package_id}'. Expected format: pkg_{{offer_config_id}}_{{start_day}}_{{end_day}}_{{rooms_needed}}"}
        
        try:
            # Parse package_id: pkg_{offer_config_id}_{start_day}_{end_day}_{rooms_needed}
            parts = package_id.split("_")
            if len(parts) != 5:
                return {"error": f"Invalid package_id format: '{package_id}'. Expected 5 parts separated by underscores."}
            
            offer_config_id = parts[1]  # Extract offer_config_id
            
            # Query to get room details using offer_config_id
            query = """
            SELECT r.*
            FROM rooms r
            JOIN offers o ON r.room_id = o.room_id
            WHERE o.offer_config_id = ?
            LIMIT 1
            """
            
            conn = sqlite3.connect(self.db_path)
            conn.row_factory = sqlite3.Row
            cursor = conn.cursor()
            
            result = cursor.execute(query, (offer_config_id,)).fetchone()
            conn.close()

            if not result:
                return {"error": f"No room found for package_id '{package_id}'. The offer_config_id '{offer_config_id}' may not exist."}
            
            room_details = dict(result)
            
            # Convert integer boolean fields to proper Python booleans
            boolean_fields = [
                'has_fridge', 'has_kitchenette', 'has_air_conditioning', 
                'allow_children', 'has_crib'
            ]
            for field in boolean_fields:
                if field in room_details and room_details[field] is not None:
                    room_details[field] = bool(room_details[field])
            
            return room_details

        except Exception as e:
            return {"error": f"Could not retrieve room details due to a database issue: {e}"}

if __name__ == '__main__':
    accommodations_tool = Accommodations()
    
    # --- Test Case 1: Search for group packages for a small party ---
    print("\n--- Test Case 1: Searching for packages for 2 people in Yellowstone National Park ---")
    search_results = accommodations_tool.search_hotels(
        location="Yellowstone National Park",
        start_date="2025-08-05",
        end_date="2025-08-06",
        num_guests=2
    )
    print("original search results: ", json.dumps(search_results['hotels'][0], indent=4))
    
    if isinstance(search_results, dict) and search_results.get("hotels"):
        print(f"Found {len(search_results['hotels'])} total hotels with packages.")
        
        # Show first few hotels with their packages
        for hotel_data in search_results['hotels'][:3]:  # Limit to first 3 hotels for readability
            print(f"\n--- Hotel: {hotel_data['hotel_name']} ---")
            print(f"  - Hotel ID: {hotel_data['hotel_id']}")
            print(f"  - Star Rating: {hotel_data['star_rating']}")
            print(f"  - Review Score: {hotel_data['review_score']}")
            print(f"  - Description: {hotel_data['description']}")
            print(f"  - Group Packages: {len(hotel_data['group_packages'])}")
            
            # Show first 2 packages for each hotel
            for i, package in enumerate(hotel_data['group_packages'][:2], 1):
                print(f"    Package {i}:")
                print(f"      - Package ID: {package.get('package_id')}")
                print(f"      - Description: {package.get('description')}")
                print(f"      - Total Cost: ${package.get('pricing', {}).get('total_cost')}")
                print(f"      - Rooms Needed: {package.get('rooms_needed')}")
                print(f"      - Room ID: {package.get('room_configuration', {}).get('room_id')}")
                print(f"      - Room Name: {package.get('room_configuration', {}).get('room_name')}")
                print(f"      - Cancellation: {package.get('policies', {}).get('cancellation_policy')}")
                print(f"      - Meal Plan: {package.get('policies', {}).get('meal_plan')}")
            
            if len(hotel_data['group_packages']) > 2:
                print(f"    ... and {len(hotel_data['group_packages']) - 2} more packages")

    else:
        print(f"Search returned: {search_results}")
    print("*"*100)

    # --- Test Case 2: Search for larger group with filters ---
    print("\n\n--- Test Case 2: Searching for packages for 6 people with filters ---")
    search_results_filtered = accommodations_tool.search_hotels(
        location="Yellowstone National Park",
        start_date="2025-08-15",
        end_date="2025-08-20",
        num_guests=6,
        free_cancellation=True,
        breakfast_included=True,
        max_price_nightly=300
    )
    print("original search results: ", json.dumps(search_results['hotels'][0], indent=4))

    if isinstance(search_results_filtered, dict) and search_results_filtered.get("hotels"):
        print(f"Found {len(search_results_filtered['hotels'])} hotels with packages for 6 people.")
        
        # Show all filtered results
        for hotel_data in search_results_filtered['hotels']:
            print(f"\n--- Hotel: {hotel_data['hotel_name']} ---")
            print(f"  - Description: {hotel_data['description']}")
            print(f"  - Group Packages: {len(hotel_data['group_packages'])}")
            
            for i, package in enumerate(hotel_data['group_packages'], 1):
                print(f"    Package {i}:")
                print(f"      - Package ID: {package.get('package_id')}")
                print(f"      - Description: {package.get('description')}")
                print(f"      - Total Cost: ${package.get('pricing', {}).get('total_cost')}")
                print(f"      - Room ID: {package.get('room_configuration', {}).get('room_id')}")
                print(f"      - Cancellation: {package.get('policies', {}).get('cancellation_policy')}")
                print(f"      - Meal Plan: {package.get('policies', {}).get('meal_plan')}")
    else:
        print(f"Search returned: {search_results_filtered}")
    print("*"*100)

    # --- Test Case 3: Test get_hotel_details function ---
    print("\n\n--- Test Case 3: Testing get_hotel_details function ---")
    if isinstance(search_results, dict) and search_results.get("hotels"):
        # Get details for the first hotel from our search
        first_hotel = search_results['hotels'][0]
        hotel_id = first_hotel['hotel_id']
        
        print(f"Getting detailed information for hotel ID: {hotel_id}")
        hotel_details = accommodations_tool.get_hotel_details(hotel_id)
        
        if isinstance(hotel_details, dict) and not hotel_details.get("error"):
            print(f"Hotel Details: {hotel_details}")
        else:
            print(f"Error getting hotel details: {hotel_details}")
    print("*"*100)

    # --- Test Case 4: Test get_room_details function ---
    print("\n\n--- Test Case 4: Testing get_room_details function ---")
    if isinstance(search_results, dict) and search_results.get("hotels"):
        # Get room details from the first package
        first_hotel = search_results['hotels'][0]
        if first_hotel['group_packages']:
            # Extract room_id from package_id (assuming format: pkg_hotelid_roomconfigid_partysizepax)
            first_package = first_hotel['group_packages'][0]
            package_id = first_package['package_id']
            
            # For demo purposes, let's use the get_hotel_details to show detailed hotel info
            print(f"Getting detailed information for package: {package_id}")
            print(f"Package details:")
            print(f"  - Description: {first_package.get('description')}")
            print(f"  - Rooms Needed: {first_package.get('rooms_needed')}")
            print(f"  - Room Configuration: {first_package.get('room_configuration')}")
            print(f"  - Pricing: {first_package.get('pricing')}")
            print(f"  - Policies: {first_package.get('policies')}")
            
            # Extract room_id from room_configuration and test get_room_details
            room_id = first_package.get('room_configuration', {}).get('room_id')
            if room_id:
                print(f"  - Getting detailed room information for room ID: {room_id}")
                # Use hotel_id from the first hotel in our search results
                hotel_id = first_hotel['hotel_id']
                room_details = accommodations_tool.get_room_details(first_package['package_id'])
                if not room_details.get('error'):
                    print(f"    Room Details: {room_details}")
                else:
                    print(f"    Error getting room details: {room_details.get('error')}")
    print("*"*100)
    
    # --- Test Case 5: Test package-specific functionality ---
    print("\n\n--- Test Case 5: Testing package generation for different group sizes ---")
    
    # Test small group (1 person)
    small_group = accommodations_tool.search_hotels(
        location="Yellowstone National Park",
        start_date="2025-08-05",
        end_date="2025-08-06",
        num_guests=1
    )
    print(f"Small group (1 person): {len(small_group.get('hotels', []))} hotels found")
    
    # Test large group (8 people)
    large_group = accommodations_tool.search_hotels(
        location="Yellowstone National Park",
        start_date="2025-08-05",
        end_date="2025-08-06",
        num_guests=8
    )
    print(f"Large group (8 people): {len(large_group.get('hotels', []))} hotels found")
    
    # Show example of room calculation
    if large_group.get('hotels') and large_group['hotels'][0]['group_packages']:
        example_package = large_group['hotels'][0]['group_packages'][0]
        print(f"Example large group package:")
        print(f"  - Description: {example_package.get('description')}")
        print(f"  - Rooms needed: {example_package.get('rooms_needed')}")
        print(f"  - Total cost: ${example_package.get('pricing', {}).get('total_cost')}")
    print("*"*100)
    
    # --- Test Case 6: Test error handling ---
    print("\n\n--- Test Case 6: Testing error handling ---")
    print("Testing invalid number of guests:")
    invalid_guests = accommodations_tool.search_hotels(
        location="Yellowstone National Park",
        start_date="2025-08-05",
        end_date="2025-08-06",
        num_guests=0
    )
    print(f"Result: {invalid_guests}")
    
    print("\nTesting invalid hotel ID:")
    invalid_hotel = accommodations_tool.get_hotel_details("INVALID_ID")
    print(f"Result: {invalid_hotel}")
    
    print("\nTesting invalid date range:")
    invalid_search = accommodations_tool.search_hotels(
        location="Yellowstone National Park",
        start_date="2025-08-20",
        end_date="2025-08-15",  # End before start
        num_guests=2
    )
    print(f"Result: {invalid_search}")
    print("*"*100)