# flake8: noqa

SYSTEM_PROMPT = """You are a helpful and proactive travel planning assistant. The current date is June 1st, 2025. Your goal is to help the user find the best accommodation by satisfying their constraints and maximizing their implicit preferences.

**CRITICAL: You MUST respond ONLY in valid JSON format using the exact schema below. Do NOT include any text outside the JSON structure.**

Your workflow is as follows:
1.  Engage in a natural, friendly conversation with the user to understand their needs.
2.  Use relevant tools to find available options based on the user's criteria. Ask clarifying questions if necessary.
3.  When you have found one or more suitable options, you MUST use the `recommend_hotel` tool to validate the package IDs before including them in your response.
    - Provide a list of package IDs to the `package_ids` parameter.
    - Explain your reasoning for the choice within the `reasoning` parameter.
4.  **OPTIONAL - NOTE TAKING**: You may use the `Notebook` tool as needed to take notes or use as a scratch pad for necessary information that might be helpful for follow-up questions or complex planning tasks.

**RESPONSE FORMAT**: You MUST respond with ONLY a valid JSON object using the following schema (no additional text before or after):

When making a recommendation (after validation):
{{{{
  "message": "Based on your requirements, I found some excellent options that perfectly match your criteria:
            1. Hotel A:.
            - Additional room or booking info
            - Additional room or booking info

            2. Hotel B.
            - Additional room or booking info
            - Additional room or booking info

            3. Hotel C:.
            - Additional room or booking info
            - Additional room or booking info",
  "formal_recommendation": {{{{
    "package_ids": ["pkg_ABC123_01_05_2", "pkg_XYZ789_01_05_1"],
    "reasoning": "These packages offer the best combination of location, amenities, and value for your 3-night stay. Both are within your budget and include the breakfast you requested."
  }}}}
}}}}

When NOT making a recommendation:
{{{{
  "message": "You are welcome! Hope you have a wonderful trip!",
  "formal_recommendation": "None"
}}}}

**IMPORTANT**: 
- Always include both "message" and "formal_recommendation" fields in your JSON response
- User will only see your "message" so please include necessary information there if you want user to know
- Your response must be ONLY the JSON object - no text before or after the JSON structure
- The entire response should be parseable as valid JSON
"""

TOOL_SCHEMAS = {
    "flights": {
        "name": "FlightSearch",
        "description": "Search for flights between two cities on a specific date.",
        "parameters": {
            "type": "object",
            "properties": {
                "origin": {"type": "string", "description": "The city of departure, e.g. 'St. Petersburg'"},
                "destination": {"type": "string", "description": "The city of arrival, e.g. 'Rockford'"},
                "departure_date": {"type": "string", "description": "The date of the flight in YYYY-MM-DD format, e.g. '2022-03-16'"},
            },
            "required": ["origin", "destination", "departure_date"],
        },
    },
    "attractions": {
        "name": "AttractionSearch",
        "description": "Search for tourist attractions in a given city.",
        "parameters": { "type": "object", "properties": {"city": {"type": "string", "description": "The city to search for attractions in."}}, "required": ["city"]},
    },
    "accommodations": {
        "name": "AccommodationSearch",
        "description": "Search for available accommodations by location (city or national park), date range, and various filters like price, amenities, and policies.",
        "parameters": {
            "type": "object",
            "properties": {
                "location": {"type": "string", "description": "The city or national park to search for accommodations in. Use search_location_name if location name is unclear or might need fuzzy matching."},
                "start_date": {"type": "string", "description": "The check-in date in YYYY-MM-DD format."},
                "end_date": {"type": "string", "description": "The check-out date in YYYY-MM-DD format."},
                "num_guests": {"type": "integer", "description": "Number of guests in the group."},
                "min_price_nightly": {"type": "number", "description": "Minimum price per room per night (USD, optional)."},
                "max_price_nightly": {"type": "number", "description": "Maximum price per room per night (USD, optional)."},
                "min_review_score": {"type": "number", "description": "Minimum review score (optional)."},
                "free_cancellation": {"type": "boolean", "description": "Filter for free cancellation policy (optional)."},
                "no_prepayment_needed": {"type": "boolean", "description": "Filter for no prepayment required (optional)."},
                "has_parking": {"type": "boolean", "description": "Filter for hotels with parking (optional)."},
                "breakfast_included": {"type": "boolean", "description": "Filter for hotels with breakfast included (optional)."},
                "is_pet_friendly": {"type": "boolean", "description": "Filter for pet-friendly hotels (optional)."},
            },
            "required": ["location", "start_date", "end_date", "num_guests"],
        },
        "returns": {
            "description": "Returns a list of hotels, each with a list of group booking packages. Each hotel includes: hotel_id, hotel_name, star_rating, review_score, description. Each package includes: package_id, description, rooms_needed, room_configuration (room_id, room_type, room_name, max_occupancy_per_room), pricing (total_cost), policies (cancellation_policy, meal_plan, prepayment_needed). Use HotelDetails or RoomDetails for more information."
        }
    },
    "hotel_details": {
        "name": "get_hotel_details",
        "description": "Get detailed information about a specific hotel including amenities, location, and policies. Supports fuzzy matching - you can use partial hotel names.",
        "parameters": {
            "type": "object",
            "properties": {
                "hotel_identifier": {"type": "string", "description": "Either the hotel ID or hotel name from a previous search. Supports partial names (e.g., 'Crosswinds' will match 'Crosswinds Inn')."},
            },
            "required": ["hotel_identifier"],
        },
        "returns": {
            "description": "Returns detailed hotel information including: hotel_id, name, brand, city, national_park, star_rating, review_score, review_count, distance_to_park_km, latitude, longitude, amenities (has_pool, has_gym, has_spa, has_bar, has_restaurant, has_wifi, has_parking, is_pet_friendly, has_airport_shuttle). Returns error message if hotel not found or multiple matches found."
        }
    },
    "room_details": {
        "name": "get_room_details", 
        "description": "Get detailed information about a specific room including bed configuration, amenities, and policies. Use the package_id from your notebook records or previous search results.",
        "parameters": {
            "type": "object",
            "properties": {
                "package_id": {"type": "string", "description": "The package ID from a previous search or notebook record (format: pkg_configid_startday_endday_rooms)."},
            },
            "required": ["package_id"],
        },
        "returns": {
            "description": "Returns detailed room information including: room_id, hotel_id, room_name, room_type, number_of_beds, bed_type, room amenities (has_fridge, has_kitchenette, has_air_conditioning), and policies (allow_children, has_crib). Returns error message if package_id is invalid or room not found."
        }
    },
    "location_search": {
        "name": "search_location_name",
        "description": "Search for location names using fuzzy matching to find the correct location name for accommodation searches. Useful when location names are unclear, partial, or informal.",
        "parameters": {
            "type": "object",
            "properties": {
                "query": {"type": "string", "description": "The location search query (e.g., 'Acadia', 'Smoky Mountains', 'Yellowstone'). Can be partial or informal names."},
            },
            "required": ["query"],
        },
        "returns": {
            "description": "Returns a list of up to 10 matching locations ranked by confidence score. Each match includes: location_name, confidence_score (0-1), location_type (national_park or city). Use the highest-scoring match for your AccommodationSearch."
        }
    },
    "get_weekday": {
        "name": "get_weekday",
        "description": "Given a date string in YYYY-MM-DD format, returns the weekday name (e.g., 'Monday').",
        "parameters": {
            "type": "object",
            "properties": {
                "date_str": {"type": "string", "description": "Date in YYYY-MM-DD format."}
            },
            "required": ["date_str"]
        }
    },
    "calculate": {
        "name": "calculate",
        "description": "Evaluate a basic arithmetic expression (e.g., '100 + 200 * 3').",
        "parameters": {
            "type": "object",
            "properties": {
                "expression": {"type": "string", "description": "Arithmetic expression to evaluate."}
            },
            "required": ["expression"]
        }
    },
    "notebook": {
        "name": "Notebook",
        "description": "Optional notebook tool for agents to take notes or use as a scratch pad for necessary information as needed. Single entry point via the 'notebook' method.",
        "parameters": {
            "type": "object",
            "properties": {
                "action": {"type": "string", "description": "Action to perform: 'write', 'read', 'delete', or 'list_all'."},
                "input_data": {"type": "string", "description": "Content to write (required for 'write')."},
                "index": {"type": "integer", "description": "Index for 'read' or 'delete' (required for those actions)."}
            },
            "required": ["action"]
        },
        "returns": {
            "description": "For 'write': returns confirmation and index. For 'read': returns content at index. For 'delete': returns confirmation. For 'list_all': returns a list of all entries with their indices and content. Returns error message for invalid actions or parameters."
        }
    },
    "restaurants": {
        "name": "RestaurantSearch",
        "description": "Search for restaurants in a given city.",
        "parameters": { "type": "object", "properties": {"city": {"type": "string", "description": "The city to search for restaurants in."}}, "required": ["city"]},
    },
    "googleDistanceMatrix": {
        "name": "GoogleDistanceMatrix",
        "description": "Get the travel distance and time between an origin and a destination by a specific mode of travel.",
        "parameters": {
            "type": "object",
            "properties": {
                "origin": {"type": "string", "description": "The starting point."},
                "destination": {"type": "string", "description": "The ending point."},
                "mode": {"type": "string", "description": "The mode of travel (e.g., 'self-driving', 'taxi')."},
            },
            "required": ["origin", "destination", "mode"],
        },
    },
    "cities": {
        "name": "CitySearch",
        "description": "Search for cities within a given state.",
        "parameters": { "type": "object", "properties": {"state": {"type": "string", "description": "The state to search for cities in."}}, "required": ["state"]},
    },
    "recommender": {
        "name": "recommend_hotel",
        "description": "MANDATORY validation tool to verify package IDs exist before including them in your formal_recommendation JSON response. You MUST call this tool before recommending any packages to ensure they are valid and exist in the system. This prevents hallucination and ensures accurate recommendations.",
        "parameters": {
            "type": "object",
            "properties": {
                "package_ids": {
                    "type": "array",
                    "items": {"type": "string"},
                    "description": "A list of package IDs (1 to 3 items) as returned by the package-based search API that you want to validate."
                },
                "reasoning": {
                    "type": "string",
                    "description": "A short explanation of why these specific packages are being considered for recommendation."
                }
            },
            "required": ["package_ids", "reasoning"]
        },
        "returns": {
            "description": "Returns validation results confirming whether the package IDs are valid and exist in the system. Only include validated package IDs in your JSON formal_recommendation."
        }
    }
}

# --- User Simulator Prompts ---
# This prompt guides the LLM to act as a user with hidden preferences.
# It uses a "conscience" (the briefing) to ensure its feedback is grounded in the task's ground truth. 