from typing import List, Dict, Optional, Any
from scale_env.environment.toolkit import ToolKitBase, ToolType, is_tool
from .database import *
from thefuzz import fuzz, process

"""Tools for football."""

class FootballTools(ToolKitBase):
    """All tools for football."""
    
    db: FootballDB
    
    def __init__(self, db: FootballDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def mark_ticket_as_used(self, booking_id: str, seat_id: str, entry_time: str, entry_gate: str = None) -> dict:
        """
        Mark a ticket as used after successful entry at venue

        Args:
            booking_id: Unique identifier for the booking
            seat_id: Unique identifier for the seat
            entry_time: Time when ticket was used in yyyy-mm-dd HH:MM:SS format
            entry_gate: Gate number where entry occurred (optional)

        Returns:
            dict: Contains marked_as_used (bool) and entry_time (str)

        Raises:
            RuntimeError: If ticket validation fails or update fails
        """
        from datetime import datetime

        # Validate input parameters
        if not booking_id or not isinstance(booking_id, str):
            raise RuntimeError("Invalid booking_id: must be a non-empty string")

        if not seat_id or not isinstance(seat_id, str):
            raise RuntimeError("Invalid seat_id: must be a non-empty string")

        if not entry_time or not isinstance(entry_time, str):
            raise RuntimeError("Invalid entry_time: must be a non-empty string")

        # Parse and validate entry_time format
        try:
            parsed_entry_time = datetime.strptime(entry_time.strip(), "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise RuntimeError(f"Invalid entry_time format: expected 'yyyy-mm-dd HH:MM:SS', got '{entry_time}'")

        # Access database
        db = self.db

        # Get booking_seat table
        booking_seat_table = getattr(db, "booking_seat", None)
        if booking_seat_table is None:
            raise RuntimeError("booking_seat table not found in database")

        # Find the booking_seat record matching both booking_id and seat_id
        target_booking_seat = None
        target_booking_seat_id = None

        for bs_id, bs_record in booking_seat_table.items():
            if bs_record.booking_id == booking_id and bs_record.seat_id == seat_id:
                target_booking_seat = bs_record
                target_booking_seat_id = bs_id
                break

        # Check if booking_seat record exists
        if target_booking_seat is None:
            raise RuntimeError(f"No ticket found for booking_id '{booking_id}' and seat_id '{seat_id}'")

        # Validate pre-condition: ticket must not already be used
        if target_booking_seat.is_used:
            raise RuntimeError(
                f"Ticket for booking_id '{booking_id}' and seat_id '{seat_id}' has already been used at "
                f"{target_booking_seat.entry_time.strftime('%Y-%m-%d %H:%M:%S') if target_booking_seat.entry_time else 'unknown time'}"
            )

        # Update the booking_seat record to mark as used
        target_booking_seat.is_used = True
        target_booking_seat.entry_time = parsed_entry_time

        # Set entry_gate if provided
        if entry_gate is not None:
            target_booking_seat.entry_gate = entry_gate

        # Update the record in database
        booking_seat_table[target_booking_seat_id] = target_booking_seat
        setattr(db, "booking_seat", booking_seat_table)

        # Return success result
        return {
            "marked_as_used": True,
            "entry_time": entry_time.strip()
        }

    @is_tool()
    def cancel_booking(self, booking_id: str, cancellation_reason: str = None) -> dict:
        """
        Cancel an existing booking and release the associated seats.

        This method performs the following operations:
        1. Validates the booking exists and is in a cancellable state
        2. Updates the booking status to 'cancelled'
        3. Records cancellation details (time, reason, refund amount)
        4. Releases all associated seats back to 'available' status
        5. Returns cancellation confirmation details

        Args:
            booking_id: Unique identifier for the booking to cancel
            cancellation_reason: Optional reason for cancellation

        Returns:
            dict: Cancellation details including booking_id, cancellation_time, 
                  refund_amount, and updated status

        Raises:
            RuntimeError: If booking doesn't exist or is not in a cancellable state
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve booking table data
        booking_table = getattr(db, "booking", None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Check if booking exists
        if booking_id not in booking_table:
            raise RuntimeError(f"Booking with ID '{booking_id}' does not exist")

        # Retrieve the booking record
        booking = booking_table[booking_id]

        # Validate booking is in a cancellable state
        # Only 'confirmed' bookings can be cancelled (not already cancelled or completed)
        if booking.status == "cancelled":
            raise RuntimeError(f"Booking '{booking_id}' is already cancelled")
        if booking.status == "completed":
            raise RuntimeError(f"Booking '{booking_id}' has been completed and cannot be cancelled")
        if booking.status not in ["confirmed"]:
            raise RuntimeError(f"Booking '{booking_id}' with status '{booking.status}' cannot be cancelled")

        # Record cancellation time
        cancellation_time = datetime.now()

        # Calculate refund amount (full refund of total_price)
        refund_amount = float(booking.total_price)

        # Update booking record with cancellation details
        booking.status = "cancelled"
        booking.cancellation_time = cancellation_time
        booking.cancellation_reason = cancellation_reason if cancellation_reason else None
        booking.refund_amount = refund_amount

        # Update booking in database
        booking_table[booking_id] = booking
        setattr(db, "booking", booking_table)

        # Release associated seats
        # Retrieve booking_seat table to find all seats associated with this booking
        booking_seat_table = getattr(db, "booking_seat", None)
        if booking_seat_table is not None:
            # Find all booking_seat records for this booking
            seat_ids_to_release = []
            for bs_id, booking_seat in booking_seat_table.items():
                if booking_seat.booking_id == booking_id:
                    seat_ids_to_release.append(booking_seat.seat_id)

            # Release seats by updating their status to 'available'
            seat_table = getattr(db, "seat", None)
            if seat_table is not None:
                for seat_id in seat_ids_to_release:
                    if seat_id in seat_table:
                        seat = seat_table[seat_id]
                        # Change seat status from 'sold' or 'used' back to 'available'
                        seat.status = "available"
                        seat_table[seat_id] = seat

                # Update seat table in database
                setattr(db, "seat", seat_table)

        # Format cancellation_time as string in required format
        cancellation_time_str = cancellation_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return cancellation confirmation
        return {
            "booking_id": booking_id,
            "cancellation_time": cancellation_time_str,
            "refund_amount": refund_amount,
            "status": "cancelled"
        }

    @is_tool()
    def get_specific_seats(self, match_id: str, seat_category: str, section: Optional[str] = None) -> dict:
        """
        Retrieve specific seat details including row, seat number, and availability status

        Args:
            match_id: Unique identifier for the match
            seat_category: Seat category to retrieve seats from (e.g., 'VIP', 'Premium', 'Standard')
            section: Optional stadium section name to filter seats

        Returns:
            Dictionary containing list of seats with their details

        Raises:
            KeyError: If match_id doesn't exist or no seats found for the criteria
        """
        # Get database instance
        db = self.db

        # Get seat table from database
        seat_table = getattr(db, 'seat', None)

        # Validate that seat table exists
        if seat_table is None:
            raise KeyError("Seat table not found in database")

        # Validate that seat table is not empty
        if not seat_table:
            raise KeyError(f"No seats found in database")

        # Check if any seat exists for the given match_id
        match_seats_exist = any(seat.match_id == match_id for seat in seat_table.values())
        if not match_seats_exist:
            raise KeyError(f"No seats found for match_id: {match_id}")

        # Import fuzzy matching library for category matching
        from thefuzz import fuzz

        # Filter seats based on criteria
        filtered_seats = []

        for seat in seat_table.values():
            # Check if seat belongs to the specified match
            if seat.match_id != match_id:
                continue

            # Use fuzzy matching for seat category comparison (natural language field)
            # Set threshold at 80 for reasonable flexibility
            category_similarity = fuzz.ratio(seat.category.lower(), seat_category.lower())
            if category_similarity < 80:
                continue

            # If section is specified, filter by section using fuzzy matching
            if section is not None:
                section_similarity = fuzz.ratio(seat.section.lower(), section.lower())
                if section_similarity < 80:
                    continue

            # Add matching seat to results
            filtered_seats.append(seat)

        # Check if any seats match the criteria
        if not filtered_seats:
            error_msg = f"No seats found for match_id: {match_id}, seat_category: {seat_category}"
            if section is not None:
                error_msg += f", section: {section}"
            raise KeyError(error_msg)

        # Build result list with seat details
        seats_result = []
        for seat in filtered_seats:
            seat_detail = {
                'seat_id': seat.seat_id,
                'row': seat.row,
                'seat_number': seat.seat_number,
                'status': seat.status,
                'price': float(seat.price)  # Ensure price is returned as float
            }
            seats_result.append(seat_detail)

        # Return seats in a dictionary format as specified in the schema
        return {
            'seats': seats_result
        }

    @is_tool()
    def search_resale_tickets(self, match_id: str, max_price: float = None, preferred_section: str = None, min_quantity: int = None):
        """
        Search for available tickets on the official resale platform

        Args:
            match_id: Unique identifier for the match
            max_price: Maximum price filter (optional)
            preferred_section: Preferred stadium section (optional)
            min_quantity: Minimum number of adjacent seats (optional)

        Returns:
            dict: Dictionary containing list of available resale tickets

        Raises:
            KeyError: If match_id is not found or required data is missing
        """
        from thefuzz import fuzz, process

        # Access database
        db = self.db

        # Get resale_listing and seat tables
        resale_listing_table = getattr(db, 'resale_listing', None)
        seat_table = getattr(db, 'seat', None)

        # Validate that required tables exist
        if resale_listing_table is None:
            raise KeyError("resale_listing table not found in database")
        if seat_table is None:
            raise KeyError("seat table not found in database")

        # Validate match_id exists in seat table (at least one seat should exist for the match)
        match_exists = False
        for seat_id, seat in seat_table.items():
            if seat.match_id == match_id:
                match_exists = True
                break

        if not match_exists:
            raise KeyError(f"Match ID '{match_id}' not found in seat table")

        # Filter active resale listings
        active_listings = []
        for listing_id, listing in resale_listing_table.items():
            # Only include active listings
            if listing.listing_status != 'active':
                continue

            # Get the corresponding seat information
            seat = seat_table.get(listing.seat_id)
            if seat is None:
                continue

            # Filter by match_id (exact match required)
            if seat.match_id != match_id:
                continue

            # Apply max_price filter if specified
            if max_price is not None and listing.resale_price > max_price:
                continue

            # Apply preferred_section filter if specified (using fuzzy matching for natural language text)
            if preferred_section is not None:
                # Use fuzzy matching for section name (natural language text field)
                similarity = fuzz.ratio(preferred_section.lower(), seat.section.lower())
                # Set threshold at 80 for reasonable matching
                if similarity < 80:
                    continue

            # Add listing with seat information to results
            active_listings.append({
                'listing': listing,
                'seat': seat
            })

        # If min_quantity is specified, filter for adjacent seats
        if min_quantity is not None and min_quantity > 1:
            # Group listings by section and row
            section_row_groups = {}
            for item in active_listings:
                seat = item['seat']
                key = (seat.section, seat.row)
                if key not in section_row_groups:
                    section_row_groups[key] = []
                section_row_groups[key].append(item)

            # Find groups with adjacent seats meeting min_quantity requirement
            filtered_listings = []
            for key, group in section_row_groups.items():
                # Sort by seat_number (convert to int if possible for numeric comparison)
                try:
                    group.sort(key=lambda x: int(x['seat'].seat_number))
                except ValueError:
                    # If seat_number is not numeric, sort as string
                    group.sort(key=lambda x: x['seat'].seat_number)

                # Check for adjacent seats
                i = 0
                while i < len(group):
                    adjacent_count = 1
                    j = i + 1

                    # Count consecutive adjacent seats
                    while j < len(group):
                        try:
                            current_seat_num = int(group[j-1]['seat'].seat_number)
                            next_seat_num = int(group[j]['seat'].seat_number)
                            # Check if seats are adjacent (difference of 1)
                            if next_seat_num == current_seat_num + 1:
                                adjacent_count += 1
                                j += 1
                            else:
                                break
                        except ValueError:
                            # If conversion fails, cannot determine adjacency
                            break

                    # If we found enough adjacent seats, add them to results
                    if adjacent_count >= min_quantity:
                        filtered_listings.extend(group[i:i+adjacent_count])
                        i += adjacent_count
                    else:
                        i += 1

            active_listings = filtered_listings

        # Build result list with required fields
        available_tickets = []
        for item in active_listings:
            listing = item['listing']
            seat = item['seat']

            ticket_info = {
                'resale_listing_id': listing.resale_listing_id,
                'seat_id': seat.seat_id,
                'section': seat.section,
                'row': seat.row,
                'seat_number': seat.seat_number,
                'price': float(listing.resale_price)
            }
            available_tickets.append(ticket_info)

        # Sort results by price (ascending) for better user experience
        available_tickets.sort(key=lambda x: x['price'])

        return {
            'available_tickets': available_tickets
        }

    @is_tool()
    def search_customer_bookings(self, customer_id: str, status_filter: str = None, start_date: str = None, end_date: str = None):
        """
        Search and retrieve all bookings for a specific customer with optional filters

        Args:
            customer_id: Unique identifier for the customer
            status_filter: Optional filter for booking status (confirmed, cancelled, completed)
            start_date: Optional start date for booking search in yyyy-mm-dd format
            end_date: Optional end date for booking search in yyyy-mm-dd format

        Returns:
            Dictionary containing list of customer bookings

        Raises:
            KeyError: If customer_id does not exist in the system
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Verify customer exists in the system
        customer_table = getattr(db, 'customer', None)
        if customer_table is None or customer_id not in customer_table:
            raise KeyError(f"Customer with ID '{customer_id}' does not exist in the system")

        # Access booking table
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            # If no booking table exists, return empty list
            return {'bookings': []}

        # Parse date filters if provided
        start_datetime = None
        end_datetime = None

        if start_date:
            try:
                # Parse date string and set time to beginning of day
                start_datetime = datetime.strptime(start_date, "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"Invalid start_date format: '{start_date}'. Expected format: yyyy-mm-dd")

        if end_date:
            try:
                # Parse date string and set time to end of day (23:59:59)
                end_datetime = datetime.strptime(end_date + " 23:59:59", "%Y-%m-%d %H:%M:%S")
            except ValueError:
                raise ValueError(f"Invalid end_date format: '{end_date}'. Expected format: yyyy-mm-dd")

        # Validate date range if both provided
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Filter bookings for the specified customer
        customer_bookings = []

        for booking_id, booking in booking_table.items():
            # Check if booking belongs to the customer
            if booking.customer_id != customer_id:
                continue

            # Apply status filter if provided
            if status_filter and booking.status != status_filter:
                continue

            # Apply date range filter if provided
            if start_datetime and booking.booking_time < start_datetime:
                continue

            if end_datetime and booking.booking_time > end_datetime:
                continue

            # Format booking_time to string format
            booking_time_str = booking.booking_time.strftime("%Y-%m-%d %H:%M:%S")

            # Add booking to results
            customer_bookings.append({
                'booking_id': booking.booking_id,
                'booking_reference': booking.booking_reference,
                'match_id': booking.match_id,
                'booking_time': booking_time_str,
                'status': booking.status
            })

        # Sort bookings by booking_time in descending order (most recent first)
        customer_bookings.sort(key=lambda x: x['booking_time'], reverse=True)

        return {'bookings': customer_bookings}

    @is_tool()
    def get_season_ticket_benefits(self, customer_id: str, season: Optional[str] = None):
        """
        Retrieve benefits and privileges for season ticket holders

        Args:
            customer_id: Unique identifier for the customer
            season: Season identifier (optional)

        Returns:
            Dictionary containing season ticket holder benefits information

        Raises:
            KeyError: If customer does not exist or is not a season ticket holder
        """
        # Access the database
        db = self.db

        # Retrieve customer data
        customer_data = getattr(db, 'customer', None)
        if customer_data is None:
            raise KeyError("Customer table not found in database")

        # Check if customer exists (exact match for customer_id)
        if customer_id not in customer_data:
            raise KeyError(f"Customer with ID '{customer_id}' does not exist")

        customer = customer_data[customer_id]

        # Check if customer is a season ticket holder
        if not customer.is_season_ticket_holder:
            raise KeyError(f"Customer '{customer_id}' is not a season ticket holder")

        # Retrieve season ticket data
        season_ticket_data = getattr(db, 'season_ticket', None)
        if season_ticket_data is None:
            raise KeyError("Season ticket table not found in database")

        # Find matching season ticket(s) for the customer
        matching_season_ticket = None
        for ticket_id, ticket in season_ticket_data.items():
            # Match by customer_id (exact match) and optionally by season (exact match)
            if ticket.customer_id == customer_id:
                if season is None or ticket.season == season:
                    matching_season_ticket = ticket
                    break

        # If no matching season ticket found
        if matching_season_ticket is None:
            if season is not None:
                raise KeyError(f"No season ticket found for customer '{customer_id}' in season '{season}'")
            else:
                raise KeyError(f"No season ticket found for customer '{customer_id}'")

        # Define standard benefits for season ticket holders
        benefits = [
            "Priority booking for all home matches",
            "Guaranteed seat for every home game",
            "Exclusive access to season ticket holder events",
            "Discounted pricing compared to individual match tickets",
            "Free or discounted ticket transfers to friends and family",
            "Priority access to away match tickets",
            "Special merchandise discounts",
            "Dedicated customer service line"
        ]

        # Add loyalty tier specific benefits if applicable
        if customer.loyalty_tier:
            tier = customer.loyalty_tier.lower()
            if tier in ['gold', 'platinum']:
                benefits.extend([
                    "VIP lounge access",
                    "Complimentary food and beverage vouchers"
                ])
            if tier == 'platinum':
                benefits.extend([
                    "Meet and greet opportunities with players",
                    "Priority parking"
                ])

        # Format priority window start time to "yyyy-mm-dd HH:MM:SS" format
        priority_window_start_str = None
        if matching_season_ticket.priority_window_start is not None:
            from datetime import datetime
            if isinstance(matching_season_ticket.priority_window_start, datetime):
                priority_window_start_str = matching_season_ticket.priority_window_start.strftime("%Y-%m-%d %H:%M:%S")
            elif isinstance(matching_season_ticket.priority_window_start, str):
                # If already a string, ensure it's in the correct format
                try:
                    dt = datetime.strptime(matching_season_ticket.priority_window_start, "%Y-%m-%d %H:%M:%S")
                    priority_window_start_str = dt.strftime("%Y-%m-%d %H:%M:%S")
                except ValueError:
                    # Try parsing other common formats
                    for fmt in ["%Y-%m-%d", "%Y-%m-%dT%H:%M:%S"]:
                        try:
                            dt = datetime.strptime(matching_season_ticket.priority_window_start, fmt)
                            priority_window_start_str = dt.strftime("%Y-%m-%d %H:%M:%S")
                            break
                        except ValueError:
                            continue

        # Construct assigned seat information
        assigned_seat = {
            "seat_id": matching_season_ticket.season_ticket_id,
            "section": matching_season_ticket.section,
            "row": matching_season_ticket.row,
            "seat_number": matching_season_ticket.seat_number
        }

        # Return the benefits information
        return {
            "is_season_ticket_holder": True,
            "benefits": benefits,
            "priority_window_start": priority_window_start_str,
            "assigned_seat": assigned_seat
        }

    @is_tool()
    def list_ticket_for_resale(self, booking_id: str, seat_id: str, resale_price: float, seller_customer_id: str) -> dict:
        """
        List a ticket on the official resale platform for other customers to purchase.

        This method validates the booking and seat, checks eligibility for resale,
        creates a new resale listing, and returns the listing details.

        Args:
            booking_id: Unique identifier for the booking
            seat_id: Unique identifier for the seat to be listed
            resale_price: Price to list ticket for resale
            seller_customer_id: Customer ID of the seller

        Returns:
            dict: Dictionary containing:
                - resale_listing_id: Unique identifier for the resale listing
                - listed_time: Time when ticket was listed (yyyy-mm-dd HH:MM:SS format)
                - listing_status: Current status of the listing (active)

        Raises:
            RuntimeError: If booking not found, seat not found in booking, 
                         ticket already used, booking cancelled, seller mismatch,
                         or ticket already listed for resale
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access database
        db = self.db

        # Validate input parameters
        if not booking_id or not isinstance(booking_id, str):
            raise RuntimeError("Invalid booking_id: must be a non-empty string")

        if not seat_id or not isinstance(seat_id, str):
            raise RuntimeError("Invalid seat_id: must be a non-empty string")

        if not isinstance(resale_price, (int, float)) or resale_price <= 0:
            raise RuntimeError("Invalid resale_price: must be a positive number")

        if not seller_customer_id or not isinstance(seller_customer_id, str):
            raise RuntimeError("Invalid seller_customer_id: must be a non-empty string")

        # Get booking table
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Check if booking exists
        if booking_id not in booking_table:
            raise RuntimeError(f"Booking with ID '{booking_id}' not found")

        booking = booking_table[booking_id]

        # Verify the seller is the owner of the booking
        if booking.customer_id != seller_customer_id:
            raise RuntimeError(f"Seller customer ID '{seller_customer_id}' does not match booking owner '{booking.customer_id}'")

        # Check if booking is cancelled
        if booking.status == 'cancelled':
            raise RuntimeError(f"Cannot list ticket for resale: booking '{booking_id}' is cancelled")

        # Get booking_seat table to verify seat belongs to booking
        booking_seat_table = getattr(db, 'booking_seat', None)
        if booking_seat_table is None:
            raise RuntimeError("Booking seat table not found in database")

        # Find the booking_seat entry that matches both booking_id and seat_id
        matching_booking_seat = None
        for bs_id, bs in booking_seat_table.items():
            if bs.booking_id == booking_id and bs.seat_id == seat_id:
                matching_booking_seat = bs
                break

        if matching_booking_seat is None:
            raise RuntimeError(f"Seat '{seat_id}' not found in booking '{booking_id}'")

        # Check if ticket has already been used
        if matching_booking_seat.is_used:
            raise RuntimeError(f"Cannot list ticket for resale: ticket for seat '{seat_id}' has already been used")

        # Get resale_listing table
        resale_listing_table = getattr(db, 'resale_listing', None)
        if resale_listing_table is None:
            # Initialize empty resale_listing table if it doesn't exist
            resale_listing_table = {}
            setattr(db, 'resale_listing', resale_listing_table)

        # Check if this seat is already listed for resale (active listing)
        for listing_id, listing in resale_listing_table.items():
            if (listing.booking_id == booking_id and 
                listing.seat_id == seat_id and 
                listing.listing_status == 'active'):
                raise RuntimeError(f"Seat '{seat_id}' from booking '{booking_id}' is already listed for resale")

        # Generate unique resale_listing_id
        prefix = "RL"
        resale_listing_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8]

        # Ensure uniqueness
        while resale_listing_id in resale_listing_table:
            resale_listing_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8]

        # Get current time for listing
        listed_time = datetime.now()

        # Create new resale listing using ResaleListing class from database module
        new_listing = ResaleListing(
            resale_listing_id=resale_listing_id,
            booking_id=booking_id,
            seat_id=seat_id,
            seller_customer_id=seller_customer_id,
            resale_price=float(resale_price),
            listed_time=listed_time,
            listing_status='active',
            sold_time=None,
            buyer_customer_id=None,
            seller_payout_amount=None
        )

        # Add to resale_listing table
        resale_listing_table[resale_listing_id] = new_listing
        setattr(db, 'resale_listing', resale_listing_table)

        # Format listed_time as string in yyyy-mm-dd HH:MM:SS format
        listed_time_str = listed_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            'resale_listing_id': resale_listing_id,
            'listed_time': listed_time_str,
            'listing_status': 'active'
        }

    @is_tool()
    def get_seat_availability(self, match_id: str) -> dict:
        """
        Retrieve available seats for a specific match including seat categories and pricing.

        This method queries the seat database to find all available seats for the given match,
        then aggregates them by category with availability counts, pricing, and section information.

        Args:
            match_id: Unique identifier for the match

        Returns:
            dict: Dictionary containing seat_categories list with availability and pricing info

        Raises:
            KeyError: If match_id does not exist in the database
        """
        # Get database instance
        db = self.db

        # Retrieve seat table from database
        seat_table = getattr(db, 'seat', None)

        # Validate that seat table exists
        if seat_table is None:
            raise KeyError(f"Seat table not found in database")

        # Check if any seats exist for the given match_id
        match_seats_exist = False
        for seat in seat_table.values():
            if seat.match_id == match_id:
                match_seats_exist = True
                break

        # Raise KeyError if no seats found for this match
        if not match_seats_exist:
            raise KeyError(f"No seats found for match_id: {match_id}")

        # Dictionary to aggregate seat information by category and section
        # Key: (category, section), Value: {available_count, price, section}
        category_section_map = {}

        # Iterate through all seats in the database
        for seat in seat_table.values():
            # Filter seats for the specified match
            if seat.match_id != match_id:
                continue

            # Only count available seats
            if seat.status != 'available':
                continue

            # Create composite key for category and section
            key = (seat.category, seat.section)

            # Initialize or update the aggregation
            if key not in category_section_map:
                category_section_map[key] = {
                    'category': seat.category,
                    'section': seat.section,
                    'price': float(seat.price),
                    'available_count': 0
                }

            # Increment available seat count
            category_section_map[key]['available_count'] += 1

        # Convert aggregated data to list format
        seat_categories = []
        for category_info in category_section_map.values():
            seat_categories.append({
                'category': category_info['category'],
                'available_seats': category_info['available_count'],
                'price': category_info['price'],
                'section': category_info['section']
            })

        # Sort by category name for consistent output
        seat_categories.sort(key=lambda x: x['category'])

        # Return result in the specified format
        return {
            'seat_categories': seat_categories
        }

    @is_tool()
    def get_waitlist_position(self, customer_id: str, match_id: str) -> dict:
        """
        Get customer's position on the waitlist for sold-out matches.

        This method retrieves the waitlist position information for a specific customer
        and match combination. It returns detailed information including position,
        total waitlist size, join time, and estimated availability.

        Args:
            customer_id: Unique identifier for the customer (e.g., 'C67890')
            match_id: Unique identifier for the match (e.g., 'M12345')

        Returns:
            dict: Dictionary containing waitlist information with keys:
                - is_on_waitlist (bool): Whether customer is on the waitlist
                - position (int): Position in the waitlist
                - total_waitlist_size (int): Total number of customers on waitlist
                - joined_time (str): Time when customer joined waitlist in yyyy-mm-dd HH:MM:SS format
                - estimated_availability (str): Estimated likelihood of tickets becoming available

        Raises:
            KeyError: If customer is not found on the waitlist for the specified match
        """
        from datetime import datetime

        # Get the waitlist table from database
        db = self.db
        waitlist_table = getattr(db, 'waitlist', None)

        # If waitlist table doesn't exist or is empty, customer is not on waitlist
        if waitlist_table is None or len(waitlist_table) == 0:
            raise KeyError(f"Customer {customer_id} is not on the waitlist for match {match_id}")

        # Find the specific waitlist entry for this customer and match
        target_entry = None
        for waitlist_id, entry in waitlist_table.items():
            if entry.customer_id == customer_id and entry.match_id == match_id:
                target_entry = entry
                break

        # If no entry found, raise KeyError
        if target_entry is None:
            raise KeyError(f"Customer {customer_id} is not on the waitlist for match {match_id}")

        # Calculate total waitlist size for this match
        # Count all active waitlist entries for the same match
        total_waitlist_size = 0
        for entry in waitlist_table.values():
            if entry.match_id == match_id and entry.status == 'active':
                total_waitlist_size += 1

        # Format joined_time to string in yyyy-mm-dd HH:MM:SS format
        joined_time_str = target_entry.joined_time.strftime("%Y-%m-%d %H:%M:%S")

        # Calculate estimated availability based on position and total size
        # Logic: 
        # - Top 20% of waitlist: High likelihood
        # - 20-50% of waitlist: Medium likelihood
        # - Bottom 50% of waitlist: Low likelihood
        position_percentage = (target_entry.position / total_waitlist_size * 100) if total_waitlist_size > 0 else 100

        if position_percentage <= 20:
            estimated_availability = "High"
        elif position_percentage <= 50:
            estimated_availability = "Medium"
        else:
            estimated_availability = "Low"

        # Return the waitlist position information
        return {
            'is_on_waitlist': True,
            'position': target_entry.position,
            'total_waitlist_size': total_waitlist_size,
            'joined_time': joined_time_str,
            'estimated_availability': estimated_availability
        }

    @is_tool()
    def calculate_refund_amount(
        self,
        original_price: float,
        booking_time: str,
        cancellation_time: str,
        match_date: str,
        cancellation_fee_percentage: float = 0.0
    ) -> dict:
        """
        Calculate the refund amount for a cancelled booking based on cancellation policy and timing.

        This method computes:
        1. Days before match when cancellation occurs
        2. Cancellation fee based on the provided percentage
        3. Final refund amount
        4. Refund percentage relative to original price

        Args:
            original_price: Original booking price (must be positive)
            booking_time: Time when booking was made in yyyy-mm-dd HH:MM:SS format
            cancellation_time: Time when cancellation is requested in yyyy-mm-dd HH:MM:SS format
            match_date: Date and time of the match in yyyy-mm-dd HH:MM:SS format
            cancellation_fee_percentage: Cancellation fee as percentage of original price (default: 0.0)

        Returns:
            Dictionary containing:
            - refund_amount: Calculated refund amount after deducting cancellation fee
            - cancellation_fee: Cancellation fee deducted from original price
            - refund_percentage: Percentage of original price refunded
            - days_before_match: Number of days before match when cancelled

        Raises:
            ValueError: If any input parameter is invalid or dates are in wrong order
        """
        from datetime import datetime

        # Validate original_price
        if not isinstance(original_price, (int, float)) or original_price <= 0:
            raise ValueError("original_price must be a positive number")

        # Validate cancellation_fee_percentage
        if not isinstance(cancellation_fee_percentage, (int, float)):
            raise ValueError("cancellation_fee_percentage must be a number")
        if cancellation_fee_percentage < 0 or cancellation_fee_percentage > 100:
            raise ValueError("cancellation_fee_percentage must be between 0 and 100")

        # Parse datetime strings
        try:
            booking_dt = datetime.strptime(booking_time.strip(), "%Y-%m-%d %H:%M:%S")
        except (ValueError, AttributeError) as e:
            raise ValueError(f"Invalid booking_time format. Expected 'yyyy-mm-dd HH:MM:SS', got: {booking_time}")

        try:
            cancellation_dt = datetime.strptime(cancellation_time.strip(), "%Y-%m-%d %H:%M:%S")
        except (ValueError, AttributeError) as e:
            raise ValueError(f"Invalid cancellation_time format. Expected 'yyyy-mm-dd HH:MM:SS', got: {cancellation_time}")

        try:
            match_dt = datetime.strptime(match_date.strip(), "%Y-%m-%d %H:%M:%S")
        except (ValueError, AttributeError) as e:
            raise ValueError(f"Invalid match_date format. Expected 'yyyy-mm-dd HH:MM:SS', got: {match_date}")

        # Validate logical order of dates
        if cancellation_dt < booking_dt:
            raise ValueError("cancellation_time cannot be before booking_time")

        if cancellation_dt > match_dt:
            raise ValueError("cancellation_time cannot be after match_date")

        if booking_dt > match_dt:
            raise ValueError("booking_time cannot be after match_date")

        # Calculate days before match
        # Using total_seconds to get accurate difference, then convert to days
        time_diff = match_dt - cancellation_dt
        days_before_match = int(time_diff.total_seconds() / (24 * 3600))

        # If cancellation is on the same day but before match time, count as 0 days
        # If there are remaining hours, we still count it as that many full days
        if time_diff.total_seconds() >= 0:
            days_before_match = int(time_diff.days)

        # Calculate cancellation fee
        cancellation_fee = original_price * (cancellation_fee_percentage / 100.0)

        # Calculate refund amount
        refund_amount = original_price - cancellation_fee

        # Ensure refund_amount is not negative (edge case protection)
        if refund_amount < 0:
            refund_amount = 0.0

        # Calculate refund percentage
        if original_price > 0:
            refund_percentage = (refund_amount / original_price) * 100.0
        else:
            refund_percentage = 0.0

        # Round to 2 decimal places for monetary values
        refund_amount = round(refund_amount, 2)
        cancellation_fee = round(cancellation_fee, 2)
        refund_percentage = round(refund_percentage, 2)

        return {
            'refund_amount': refund_amount,
            'cancellation_fee': cancellation_fee,
            'refund_percentage': refund_percentage,
            'days_before_match': days_before_match
        }

    @is_tool()
    def calculate_ticket_price(self, base_price: float, ticket_quantity: int, service_fee_percentage: float = 0.0, discount_percentage: float = 0.0) -> dict:
        """
        Calculate the total price for selected tickets including base price and any applicable fees.

        This method computes:
        1. Subtotal: base_price * ticket_quantity
        2. Service fee: subtotal * (service_fee_percentage / 100)
        3. Discount amount: subtotal * (discount_percentage / 100)
        4. Total price: subtotal + service_fee - discount_amount

        Args:
            base_price: Base price per ticket (must be positive)
            ticket_quantity: Number of tickets (must be positive integer)
            service_fee_percentage: Service fee as a percentage of base price (default: 0.0)
            discount_percentage: Discount percentage to apply (default: 0.0)

        Returns:
            dict: Dictionary containing:
                - subtotal: Subtotal before fees and discounts
                - service_fee: Calculated service fee
                - discount_amount: Calculated discount amount
                - total_price: Final total price

        Raises:
            ValueError: If base_price is not positive, ticket_quantity is not positive,
                       service_fee_percentage is negative, or discount_percentage is negative
            TypeError: If parameters are not of expected numeric types
        """

        # Validate parameter types
        if not isinstance(base_price, (int, float)):
            raise TypeError(f"base_price must be a number, got {type(base_price).__name__}")

        if not isinstance(ticket_quantity, int):
            raise TypeError(f"ticket_quantity must be an integer, got {type(ticket_quantity).__name__}")

        if not isinstance(service_fee_percentage, (int, float)):
            raise TypeError(f"service_fee_percentage must be a number, got {type(service_fee_percentage).__name__}")

        if not isinstance(discount_percentage, (int, float)):
            raise TypeError(f"discount_percentage must be a number, got {type(discount_percentage).__name__}")

        # Validate parameter values
        if base_price <= 0:
            raise ValueError(f"base_price must be positive, got {base_price}")

        if ticket_quantity <= 0:
            raise ValueError(f"ticket_quantity must be positive, got {ticket_quantity}")

        if service_fee_percentage < 0:
            raise ValueError(f"service_fee_percentage cannot be negative, got {service_fee_percentage}")

        if discount_percentage < 0:
            raise ValueError(f"discount_percentage cannot be negative, got {discount_percentage}")

        # Calculate subtotal (base price * quantity)
        subtotal = base_price * ticket_quantity

        # Calculate service fee (percentage of subtotal)
        service_fee = subtotal * (service_fee_percentage / 100.0)

        # Calculate discount amount (percentage of subtotal)
        discount_amount = subtotal * (discount_percentage / 100.0)

        # Calculate final total price (subtotal + service fee - discount)
        total_price = subtotal + service_fee - discount_amount

        # Ensure total price doesn't go negative (edge case where discount > subtotal + service_fee)
        if total_price < 0:
            total_price = 0.0

        # Return the price breakdown
        return {
            'subtotal': round(subtotal, 2),
            'service_fee': round(service_fee, 2),
            'discount_amount': round(discount_amount, 2),
            'total_price': round(total_price, 2)
        }

    @is_tool()
    def check_duplicate_booking(self, customer_id: str, match_id: str) -> dict:
        """
        Check if a customer already has a booking for the same match to prevent duplicates.

        This method searches through all bookings to find any active bookings 
        (confirmed or completed status) for the given customer and match combination.

        Args:
            customer_id: Unique identifier for the customer
            match_id: Unique identifier for the match

        Returns:
            Dictionary containing:
            - has_existing_booking: Boolean indicating if duplicate exists
            - existing_booking_ids: List of booking IDs if any exist
            - total_tickets_booked: Total number of tickets already booked

        Raises:
            KeyError: If required parameters are missing or invalid
        """
        # Validate required parameters
        if not customer_id:
            raise KeyError("customer_id is required and cannot be empty")
        if not match_id:
            raise KeyError("match_id is required and cannot be empty")

        # Access the database
        db = self.db

        # Get booking table data
        booking_table = getattr(db, 'booking', None)

        # Initialize return values
        existing_booking_ids = []
        total_tickets_booked = 0

        # If booking table exists and has data, search for duplicates
        if booking_table:
            # Iterate through all bookings to find matches
            for booking_id, booking_item in booking_table.items():
                # Check if booking matches the customer and match criteria
                # Only consider bookings that are not cancelled (confirmed or completed)
                if (booking_item.customer_id == customer_id and 
                    booking_item.match_id == match_id and
                    booking_item.status in ['confirmed', 'completed']):

                    # Add this booking to the list of existing bookings
                    existing_booking_ids.append(booking_id)

                    # Count tickets: each booking represents at least 1 ticket
                    # In this schema, we count each booking as representing tickets
                    # If booking_seat junction table was available, we'd count from there
                    # For now, we increment by 1 per booking found
                    total_tickets_booked += 1

        # Determine if duplicate exists
        has_existing_booking = len(existing_booking_ids) > 0

        # Return the result dictionary
        return {
            'has_existing_booking': has_existing_booking,
            'existing_booking_ids': existing_booking_ids,
            'total_tickets_booked': total_tickets_booked
        }

    @is_tool()
    def join_match_waitlist(self, customer_id: str, match_id: str, ticket_quantity: int, preferred_category: str = None) -> dict:
        """
        Add customer to the waitlist for a sold-out match

        This method adds a customer to the waitlist for a specific match when tickets are sold out.
        It validates that the customer is not already on the waitlist, generates a unique waitlist ID,
        determines the customer's position in the waitlist, and creates the waitlist entry.

        Args:
            customer_id: Unique identifier for the customer
            match_id: Unique identifier for the match
            ticket_quantity: Number of tickets desired
            preferred_category: Optional preferred seat category

        Returns:
            dict: Contains waitlist_id, position, and joined_time

        Raises:
            RuntimeError: If customer is already on the waitlist for this match
        """
        from datetime import datetime
        import secrets
        import hashlib
        
        # Access the database
        db = self.db

        # Get the waitlist table data
        waitlist_data = getattr(db, 'waitlist', None)
        if waitlist_data is None:
            waitlist_data = {}
            setattr(db, 'waitlist', waitlist_data)

        # Validate input parameters
        if not customer_id or not isinstance(customer_id, str):
            raise ValueError("customer_id must be a non-empty string")
        if not match_id or not isinstance(match_id, str):
            raise ValueError("match_id must be a non-empty string")
        if not isinstance(ticket_quantity, int) or ticket_quantity <= 0:
            raise ValueError("ticket_quantity must be a positive integer")
        if preferred_category is not None and not isinstance(preferred_category, str):
            raise ValueError("preferred_category must be a string or None")

        # Check if customer is already on the waitlist for this match (active status only)
        for waitlist_entry in waitlist_data.values():
            if (waitlist_entry.customer_id == customer_id and 
                waitlist_entry.match_id == match_id and 
                waitlist_entry.status == 'active'):
                raise RuntimeError(f"Customer {customer_id} is already on the active waitlist for match {match_id}")

        # Generate unique waitlist ID with proper format (WL + 5 digits to match VARCHAR(10))
        prefix = "WL"
        max_attempts = 100
        for _ in range(max_attempts):
            # Generate 5 random digits
            random_num = secrets.randbelow(100000)
            waitlist_id = f"{prefix}{random_num:05d}"
            if waitlist_id not in waitlist_data:
                break
        else:
            # Fallback to hash-based generation if random fails
            waitlist_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8]
            while waitlist_id in waitlist_data:
                waitlist_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8]

        # Calculate the position in the waitlist
        # Position is determined by counting active waitlist entries for this match
        active_waitlist_count = sum(
            1 for entry in waitlist_data.values() 
            if entry.match_id == match_id and entry.status == 'active'
        )
        position = active_waitlist_count + 1

        # Get current time for joined_time
        joined_time = datetime.now()

        # Create the new waitlist entry
        new_waitlist_entry = Waitlist(
            waitlist_id=waitlist_id,
            customer_id=customer_id,
            match_id=match_id,
            ticket_quantity=ticket_quantity,
            preferred_category=preferred_category,
            position=position,
            joined_time=joined_time,
            status='active',
            notification_sent=False
        )

        # Add the new waitlist entry to the database
        waitlist_data[waitlist_id] = new_waitlist_entry
        setattr(db, 'waitlist', waitlist_data)

        # Format joined_time as string in yyyy-mm-dd HH:MM:SS format
        joined_time_str = joined_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            'waitlist_id': waitlist_id,
            'position': position,
            'joined_time': joined_time_str
        }

    @is_tool()
    def check_booking_eligibility(self, customer_id: str, match_id: str, ticket_quantity: int) -> dict:
        """
        Check if a customer is eligible to book tickets based on membership status, age restrictions, and purchase limits.

        This method validates customer eligibility by checking:
        1. Customer existence in the system
        2. Season ticket holder status
        3. Age restrictions (if applicable)
        4. Purchase quantity limits
        5. Existing booking history for the match

        Args:
            customer_id: Unique identifier for the customer
            match_id: Unique identifier for the match
            ticket_quantity: Number of tickets to book

        Returns:
            Dictionary containing:
            - is_eligible: Boolean indicating if customer can book
            - reasons: List of reasons for eligibility or ineligibility
            - restrictions: List of restrictions that apply

        Raises:
            KeyError: If customer_id does not exist in the system
        """
        # Access database
        db = self.db

        # Initialize return structure
        result = {
            'is_eligible': True,
            'reasons': [],
            'restrictions': []
        }

        # Get customer table
        customer_table = getattr(db, 'customer', None)
        if customer_table is None:
            raise KeyError(f"Customer table not found in database")

        # Check if customer exists
        if customer_id not in customer_table:
            raise KeyError(f"Customer with ID '{customer_id}' does not exist in the system")

        # Retrieve customer information
        customer = customer_table[customer_id]

        # Get booking table to check existing bookings
        booking_table = getattr(db, 'booking', None)

        # Check 1: Validate ticket quantity (must be positive)
        if ticket_quantity <= 0:
            result['is_eligible'] = False
            result['reasons'].append('Ticket quantity must be greater than 0')
            return result

        # Check 2: Maximum ticket limit per transaction
        # Standard maximum is 4 tickets per customer per match
        MAX_TICKETS_PER_BOOKING = 4
        result['restrictions'].append(f'Maximum {MAX_TICKETS_PER_BOOKING} tickets per customer')

        if ticket_quantity > MAX_TICKETS_PER_BOOKING:
            result['is_eligible'] = False
            result['reasons'].append(f'Requested quantity ({ticket_quantity}) exceeds maximum limit of {MAX_TICKETS_PER_BOOKING} tickets per booking')
            return result

        # Check 3: Check existing bookings for this match
        # Count how many tickets customer has already booked for this match
        existing_ticket_count = 0
        if booking_table is not None:
            for booking_id, booking in booking_table.items():
                # Only count confirmed bookings for the same match and customer
                if (booking.customer_id == customer_id and 
                    booking.match_id == match_id and 
                    booking.status == 'confirmed'):
                    # Each booking can contain multiple tickets
                    # We need to count seats from booking_seat table if available
                    # For now, we assume each booking represents some tickets
                    # In a real scenario, we'd need to count from booking_seat junction table
                    existing_ticket_count += 1  # Simplified: count bookings as proxy for tickets

        # Check if total tickets (existing + new) would exceed limit
        total_tickets = existing_ticket_count + ticket_quantity
        if total_tickets > MAX_TICKETS_PER_BOOKING:
            result['is_eligible'] = False
            result['reasons'].append(f'Total tickets for this match would exceed limit. Already booked: {existing_ticket_count}, requesting: {ticket_quantity}, maximum: {MAX_TICKETS_PER_BOOKING}')
            return result

        # Check 4: Season ticket holder status (gives priority/benefits)
        if customer.is_season_ticket_holder:
            result['reasons'].append('Customer is a season ticket holder')

            # Season ticket holders may have priority booking windows
            season_ticket_table = getattr(db, 'season_ticket', None)
            if season_ticket_table is not None:
                # Check if customer has season ticket with priority window
                for st_id, season_ticket in season_ticket_table.items():
                    if season_ticket.customer_id == customer_id:
                        if season_ticket.priority_window_start is not None:
                            result['reasons'].append('Priority booking window available for season ticket holders')
                        break
        else:
            result['reasons'].append('Customer is not a season ticket holder')

        # Check 5: Loyalty tier benefits
        if customer.loyalty_tier:
            loyalty_tier_upper = customer.loyalty_tier.upper()
            result['reasons'].append(f'Customer has {customer.loyalty_tier} loyalty tier status')

            # Higher tier customers may have higher purchase limits or priority
            if loyalty_tier_upper in ['GOLD', 'PLATINUM']:
                result['reasons'].append('Premium loyalty tier provides enhanced booking privileges')

        # Check 6: Age restrictions
        # Some matches may have age restrictions (e.g., family sections, adult-only areas)
        # For general eligibility, we check if age is available
        if customer.age is not None:
            # Minors (under 18) may need guardian accompaniment
            if customer.age < 18:
                result['restrictions'].append('Customers under 18 may require guardian accompaniment')
                result['reasons'].append(f'Customer age is {customer.age} years')

            # Senior citizens (65+) may be eligible for special pricing
            if customer.age >= 65:
                result['reasons'].append('Customer may be eligible for senior pricing')
        else:
            result['restrictions'].append('Age verification may be required at venue')

        # Check 7: Loyalty points (informational)
        if customer.total_loyalty_points > 0:
            result['reasons'].append(f'Customer has {customer.total_loyalty_points} loyalty points available')

        # If we've reached here without setting is_eligible to False, customer is eligible
        if result['is_eligible']:
            result['reasons'].append('All eligibility criteria met')

        return result

    @is_tool()
    def validate_seat_selection(self, seat_ids, require_adjacent=None, max_seats_per_booking=None):
        """
        Validate that selected seats meet booking requirements such as adjacency and availability.

        This method validates seat selection based on:
        1. Seat existence and availability
        2. Number of seats within allowed limit
        3. Adjacency requirements if specified
        4. Generates appropriate errors and warnings

        Args:
            seat_ids: List of seat identifiers to validate
            require_adjacent: Whether seats must be adjacent to each other (optional)
            max_seats_per_booking: Maximum number of seats allowed per booking (optional)

        Returns:
            dict: Validation result containing:
                - is_valid: Whether the seat selection is valid
                - errors: List of validation errors
                - warnings: List of validation warnings

        Raises:
            ValueError: If seat_ids is empty or invalid
        """
        # Initialize result structure
        errors = []
        warnings = []
        is_valid = True

        # Validate input parameters
        if not seat_ids:
            raise ValueError("seat_ids cannot be empty")

        if not isinstance(seat_ids, list):
            raise ValueError("seat_ids must be a list")

        # Get database reference
        db = self.db

        # Get seat table data - CRITICAL FIX: This tool has NO database access
        # The related_databases field is empty, meaning this is a pure validation tool
        # It should NOT access any database tables

        # Since this tool has no database access (related_databases: []),
        # it can only perform basic validation on the input parameters
        # without checking actual seat data from the database

        # Check max seats per booking constraint
        if max_seats_per_booking is not None:
            if not isinstance(max_seats_per_booking, int) or max_seats_per_booking <= 0:
                raise ValueError("max_seats_per_booking must be a positive integer")

            if len(seat_ids) > max_seats_per_booking:
                errors.append(f"Number of seats ({len(seat_ids)}) exceeds maximum allowed per booking ({max_seats_per_booking})")
                is_valid = False

        # Validate seat_id format (basic validation without database access)
        for seat_id in seat_ids:
            if not isinstance(seat_id, str) or not seat_id.strip():
                errors.append(f"Invalid seat_id format: {seat_id}")
                is_valid = False

        # Check for duplicate seat IDs
        if len(seat_ids) != len(set(seat_ids)):
            errors.append("Duplicate seat IDs found in selection")
            is_valid = False

        # Note: Without database access, we cannot validate:
        # - Seat existence
        # - Seat availability status
        # - Match ID consistency
        # - Actual adjacency (requires section/row/seat_number data)
        # - Section differences

        # Add warning about limited validation
        warnings.append("Limited validation performed - database access not available for this tool")

        return {
            'is_valid': is_valid,
            'errors': errors,
            'warnings': warnings
        }

    @is_tool()
    def notify_waitlist_availability(self, match_id: str, available_seats: int, seat_category: str = None):
        """
        Notify customers on waitlist when tickets become available

        This method:
        1. Retrieves active waitlist entries for the specified match
        2. Filters by seat category if provided
        3. Sorts by position to maintain fairness
        4. Notifies customers up to the number of available seats
        5. Updates waitlist entries and logs communications

        Args:
            match_id: Unique identifier for the match
            available_seats: Number of seats that became available
            seat_category: Category of available seats (optional)

        Returns:
            dict containing:
                - notifications_sent: Number of notifications sent
                - notified_customers: List of customer IDs notified
                - notification_time: Time when notifications were sent

        Raises:
            RuntimeError: If waitlist table doesn't exist or notification process fails
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Verify waitlist table exists
        waitlist_table = getattr(db, 'waitlist', None)
        if waitlist_table is None:
            raise RuntimeError("Waitlist table does not exist in database")

        # Verify communication_log table exists
        communication_log_table = getattr(db, 'communication_log', None)
        if communication_log_table is None:
            raise RuntimeError("Communication log table does not exist in database")

        # Validate input parameters
        if not isinstance(match_id, str) or not match_id:
            raise ValueError("match_id must be a non-empty string")

        if not isinstance(available_seats, int) or available_seats <= 0:
            raise ValueError("available_seats must be a positive integer")

        if seat_category is not None and not isinstance(seat_category, str):
            raise ValueError("seat_category must be a string or None")

        # Get current timestamp for notifications
        notification_time = datetime.now()
        notification_time_str = notification_time.strftime("%Y-%m-%d %H:%M:%S")

        # Retrieve all waitlist entries for the match
        eligible_entries = []
        for waitlist_id, waitlist_entry in waitlist_table.items():
            # Filter by match_id and active status
            if waitlist_entry.match_id != match_id:
                continue

            if waitlist_entry.status != 'active':
                continue

            # Filter by seat category if specified
            if seat_category is not None:
                # If preferred_category is None, customer accepts any category
                if waitlist_entry.preferred_category is not None:
                    # Use fuzzy matching for seat category comparison
                    from thefuzz import fuzz
                    similarity = fuzz.ratio(
                        waitlist_entry.preferred_category.lower(),
                        seat_category.lower()
                    )
                    # Use 80% similarity threshold for category matching
                    if similarity < 80:
                        continue

            eligible_entries.append(waitlist_entry)

        # Sort by position to maintain fairness (lower position = higher priority)
        eligible_entries.sort(key=lambda x: x.position)

        # Determine how many customers to notify (minimum of available seats and eligible entries)
        num_to_notify = min(available_seats, len(eligible_entries))

        # Track notified customers
        notified_customers = []
        notifications_sent = 0

        # Process notifications for eligible customers
        for i in range(num_to_notify):
            waitlist_entry = eligible_entries[i]
            customer_id = waitlist_entry.customer_id

            try:
                # Generate unique message ID
                message_id = "msg_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

                # Retrieve customer information to get contact details
                customer_table = getattr(db, 'customer', None)
                recipient_address = ""

                if customer_table is not None and customer_id in customer_table:
                    customer = customer_table[customer_id]
                    # Prefer email, fallback to phone
                    recipient_address = customer.email if hasattr(customer, 'email') and customer.email else ""
                    if not recipient_address and hasattr(customer, 'phone') and customer.phone:
                        recipient_address = customer.phone

                # If no contact info found, use a placeholder
                if not recipient_address:
                    recipient_address = f"customer_{customer_id}@placeholder.com"

                # Determine communication channel based on recipient address format
                channel = "email"
                if recipient_address.startswith("+") or recipient_address.isdigit():
                    channel = "sms"

                # Create communication log entry
                communication_log_entry = CommunicationLog(
                    message_id=message_id,
                    customer_id=customer_id,
                    booking_id=None,  # No booking yet, this is just availability notification
                    communication_type="notification",
                    channel=channel,
                    recipient_address=recipient_address,
                    sent_time=notification_time,
                    status="sent"
                )

                # Add to communication log
                if communication_log_table is not None:
                    communication_log_table[message_id] = communication_log_entry

                # Update waitlist entry status
                waitlist_entry.status = "notified"
                waitlist_entry.notification_sent = True

                # Add to notified customers list
                notified_customers.append(customer_id)
                notifications_sent += 1

            except Exception as e:
                # Log error but continue with other notifications
                # In production, this might be logged to a monitoring system
                continue

        # Update the database with modified waitlist entries
        setattr(db, 'waitlist', waitlist_table)
        setattr(db, 'communication_log', communication_log_table)

        # Return notification summary
        return {
            'notifications_sent': notifications_sent,
            'notified_customers': notified_customers,
            'notification_time': notification_time_str
        }

    @is_tool()
    def get_booking_payment_details(self, booking_id: str):
        """
        Retrieve payment details and transaction information for a specific booking.

        This method fetches comprehensive payment information associated with a booking,
        including payment method, status, amount, and transaction reference.

        Args:
            booking_id: Unique identifier for the booking

        Returns:
            dict: Payment details containing:
                - payment_id: Payment transaction identifier
                - payment_method: Payment method used
                - payment_status: Current payment status
                - amount_paid: Total amount paid
                - payment_time: Time when payment was processed (yyyy-mm-dd HH:MM:SS format)
                - transaction_reference: Payment gateway transaction reference

        Raises:
            KeyError: If booking_id does not exist in the system
        """
        # Access the database instance
        db = self.db

        # Retrieve booking table data
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise KeyError(f"Booking table not found in database")

        # Check if booking exists using exact match for booking_id (precise identifier)
        if booking_id not in booking_table:
            raise KeyError(f"Booking with ID '{booking_id}' does not exist")

        # Get the booking record
        booking_record = booking_table[booking_id]

        # Extract payment_id from booking record
        payment_id = booking_record.payment_id

        # Retrieve payment table data
        payment_table = getattr(db, 'payment', None)
        if payment_table is None:
            raise KeyError(f"Payment table not found in database")

        # Check if payment exists using exact match for payment_id (precise identifier)
        if payment_id not in payment_table:
            raise KeyError(f"Payment with ID '{payment_id}' does not exist")

        # Get the payment record
        payment_record = payment_table[payment_id]

        # Format payment_time to required string format (yyyy-mm-dd HH:MM:SS)
        payment_time_str = payment_record.payment_time.strftime("%Y-%m-%d %H:%M:%S")

        # Construct and return payment details dictionary
        payment_details = {
            'payment_id': payment_record.payment_id,
            'payment_method': payment_record.payment_method,
            'payment_status': payment_record.payment_status,
            'amount_paid': float(payment_record.amount),  # Convert to float for consistency
            'payment_time': payment_time_str,
            'transaction_reference': payment_record.transaction_reference
        }

        return payment_details

    @is_tool()
    def purchase_resale_ticket(self, resale_listing_id: str, buyer_customer_id: str, payment_id: str) -> dict:
        """
        Purchase a ticket from the official resale platform.

        This method handles the complete resale ticket purchase process:
        1. Validates the resale listing exists and is active
        2. Validates the payment exists and is completed
        3. Creates a new booking for the buyer
        4. Updates the seat ownership
        5. Calculates seller payout (typically 95% of resale price, 5% platform fee)
        6. Updates the resale listing status to sold

        Args:
            resale_listing_id: Unique identifier for the resale listing
            buyer_customer_id: Customer ID of the buyer
            payment_id: Payment transaction identifier

        Returns:
            dict: Contains new_booking_id, purchase_time, and seller_payout_amount

        Raises:
            RuntimeError: If listing doesn't exist, is not active, payment is invalid,
                         or any database operation fails
        """
        from datetime import datetime
        import secrets
        import hashlib
        
        # Access database
        db = self.db

        # Retrieve resale listing data
        resale_listings = getattr(db, 'resale_listing', None)
        if resale_listings is None:
            raise RuntimeError("Resale listing table not found in database")

        # Validate resale listing exists
        if resale_listing_id not in resale_listings:
            raise RuntimeError(f"Resale listing '{resale_listing_id}' does not exist")

        resale_listing = resale_listings[resale_listing_id]

        # Validate listing is active
        if resale_listing.listing_status != 'active':
            raise RuntimeError(
                f"Resale listing '{resale_listing_id}' is not active "
                f"(status: {resale_listing.listing_status})"
            )

        # Retrieve payment data
        payments = getattr(db, 'payment', None)
        if payments is None:
            raise RuntimeError("Payment table not found in database")

        # Validate payment exists
        if payment_id not in payments:
            raise RuntimeError(f"Payment '{payment_id}' does not exist")

        payment = payments[payment_id]

        # Validate payment status is completed
        if payment.payment_status != 'completed':
            raise RuntimeError(
                f"Payment '{payment_id}' is not completed "
                f"(status: {payment.payment_status})"
            )

        # Validate payment amount matches resale price
        # Allow small floating point differences
        if abs(payment.amount - resale_listing.resale_price) > 0.01:
            raise RuntimeError(
                f"Payment amount {payment.amount} does not match "
                f"resale price {resale_listing.resale_price}"
            )

        # Validate payment customer matches buyer
        if payment.customer_id != buyer_customer_id:
            raise RuntimeError(
                f"Payment customer '{payment.customer_id}' does not match "
                f"buyer '{buyer_customer_id}'"
            )

        # Retrieve original booking to get match information
        bookings = getattr(db, 'booking', None)
        if bookings is None:
            raise RuntimeError("Booking table not found in database")

        original_booking_id = resale_listing.booking_id
        if original_booking_id not in bookings:
            raise RuntimeError(f"Original booking '{original_booking_id}' not found")

        original_booking = bookings[original_booking_id]

        # Get current time for purchase
        purchase_time = datetime.now()

        # Generate new booking ID
        new_booking_id = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate booking reference (format: BK-XXXXXX)
        booking_reference = "BK-" + hashlib.sha256(
            secrets.token_bytes(32)
        ).hexdigest()[:6].upper()

        # Calculate seller payout amount (95% of resale price, 5% platform fee)
        platform_fee_percentage = 0.05
        seller_payout_amount = float(
            resale_listing.resale_price * (1 - platform_fee_percentage)
        )

        # Create new booking for buyer
        new_booking = Booking(
            booking_id=new_booking_id,
            booking_reference=booking_reference,
            match_id=original_booking.match_id,
            customer_id=buyer_customer_id,
            reservation_id=None,  # No reservation for resale purchases
            payment_id=payment_id,
            email=original_booking.email,  # Buyer's email should be used in production
            phone=original_booking.phone,  # Buyer's phone should be used in production
            total_price=float(resale_listing.resale_price),
            booking_time=purchase_time,
            status='confirmed'
        )

        # Add new booking to database
        bookings[new_booking_id] = new_booking
        setattr(db, 'booking', bookings)

        # Create new booking_seat entry for the buyer
        booking_seats = getattr(db, 'booking_seat', None)
        if booking_seats is None:
            raise RuntimeError("Booking seat table not found in database")

        # Generate new booking_seat_id
        new_booking_seat_id = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Find original booking_seat to copy QR code data if exists
        original_booking_seat = None
        for bs_id, bs in booking_seats.items():
            if (bs.booking_id == original_booking_id and 
                bs.seat_id == resale_listing.seat_id):
                original_booking_seat = bs
                break

        new_booking_seat = BookingSeat(
            booking_seat_id=new_booking_seat_id,
            booking_id=new_booking_id,
            seat_id=resale_listing.seat_id,
            is_used=False,
            entry_time=None,
            entry_gate=None,
            qr_code_data=original_booking_seat.qr_code_data if original_booking_seat else None
        )

        # Add new booking_seat to database
        booking_seats[new_booking_seat_id] = new_booking_seat
        setattr(db, 'booking_seat', booking_seats)

        # Update resale listing status to sold
        resale_listing.listing_status = 'sold'
        resale_listing.sold_time = purchase_time
        resale_listing.buyer_customer_id = buyer_customer_id
        resale_listing.seller_payout_amount = seller_payout_amount

        # Update resale listing in database
        resale_listings[resale_listing_id] = resale_listing
        setattr(db, 'resale_listing', resale_listings)

        # Update payment booking_id to link to new booking
        payment.booking_id = new_booking_id
        payments[payment_id] = payment
        setattr(db, 'payment', payments)

        # Format purchase time as string in yyyy-mm-dd HH:MM:SS format
        purchase_time_str = purchase_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return result
        return {
            'new_booking_id': new_booking_id,
            'purchase_time': purchase_time_str,
            'seller_payout_amount': seller_payout_amount
        }

    @is_tool()
    def validate_payment_amount(self, booking_amount: float, payment_amount: float, tolerance: float = 0.01) -> dict:
        """
        Validate that the payment amount matches the calculated booking total.

        This method compares the booking amount (calculated total from booking system) 
        with the payment amount (amount being paid by customer) and determines if they 
        match within an acceptable tolerance.

        Args:
            booking_amount: Calculated total from booking system
            payment_amount: Amount being paid by customer
            tolerance: Acceptable difference tolerance (default: 0.01)

        Returns:
            dict: Validation result containing:
                - is_valid (bool): Whether payment amount is valid
                - difference (float): Difference between calculated and paid amounts
                - validation_message (str): Validation result message

        Raises:
            ValueError: If booking_amount or payment_amount is negative or None
            TypeError: If booking_amount or payment_amount is not a number
        """
        # Validate input parameter types
        if not isinstance(booking_amount, (int, float)):
            raise TypeError(f"booking_amount must be a number, got {type(booking_amount).__name__}")

        if not isinstance(payment_amount, (int, float)):
            raise TypeError(f"payment_amount must be a number, got {type(payment_amount).__name__}")

        if not isinstance(tolerance, (int, float)):
            raise TypeError(f"tolerance must be a number, got {type(tolerance).__name__}")

        # Validate that amounts are not negative
        if booking_amount < 0:
            raise ValueError("booking_amount cannot be negative")

        if payment_amount < 0:
            raise ValueError("payment_amount cannot be negative")

        if tolerance < 0:
            raise ValueError("tolerance cannot be negative")

        # Calculate the absolute difference between booking and payment amounts
        difference = payment_amount - booking_amount
        absolute_difference = abs(difference)

        # Determine if the payment is valid based on tolerance
        is_valid = absolute_difference <= tolerance

        # Generate appropriate validation message
        if is_valid:
            if difference == 0:
                validation_message = "Payment amount matches booking total"
            else:
                validation_message = f"Payment amount matches booking total within acceptable tolerance (difference: {difference:.2f})"
        else:
            if difference > 0:
                validation_message = f"Payment amount exceeds booking total by {difference:.2f}"
            else:
                validation_message = f"Payment amount is less than booking total by {abs(difference):.2f}"

        # Return validation result
        return {
            "is_valid": is_valid,
            "difference": round(difference, 2),
            "validation_message": validation_message
        }

    @is_tool()
    def calculate_loyalty_points(self, booking_amount: float, customer_tier: str = None, match_importance: str = None) -> dict:
        """
        Calculate loyalty points earned from a ticket booking.

        This method computes loyalty points based on:
        1. Base points: 1 point per dollar spent (booking_amount)
        2. Tier bonus: Additional percentage based on customer loyalty tier
        3. Match bonus: Fixed bonus points based on match importance level

        Args:
            booking_amount: Total booking amount in dollars
            customer_tier: Customer loyalty tier (bronze, silver, gold, platinum), optional
            match_importance: Match importance level (regular, important, derby, final), optional

        Returns:
            Dictionary containing:
            - base_points: Base loyalty points (1 point per dollar)
            - tier_bonus: Bonus points from customer tier
            - match_bonus: Bonus points from match importance
            - total_points: Sum of all points

        Raises:
            ValueError: If booking_amount is negative or invalid tier/importance provided
        """

        # Validate booking_amount
        if booking_amount < 0:
            raise ValueError("booking_amount must be non-negative")

        # Calculate base points: 1 point per dollar spent
        base_points = int(booking_amount)

        # Define tier bonus percentages
        tier_bonus_rates = {
            'bronze': 0.05,    # 5% bonus
            'silver': 0.10,    # 10% bonus
            'gold': 0.20,      # 20% bonus
            'platinum': 0.30   # 30% bonus
        }

        # Calculate tier bonus
        tier_bonus = 0
        if customer_tier:
            # Normalize tier to lowercase for case-insensitive matching
            tier_normalized = customer_tier.lower().strip()

            if tier_normalized not in tier_bonus_rates:
                raise ValueError(f"Invalid customer_tier: {customer_tier}. Must be one of: bronze, silver, gold, platinum")

            # Calculate tier bonus as percentage of base points
            tier_bonus = int(base_points * tier_bonus_rates[tier_normalized])

        # Define match importance bonus points
        match_bonus_points = {
            'regular': 0,      # No bonus for regular matches
            'important': 50,   # 50 points for important matches
            'derby': 100,      # 100 points for derby matches
            'final': 200       # 200 points for final matches
        }

        # Calculate match bonus
        match_bonus = 0
        if match_importance:
            # Normalize match_importance to lowercase for case-insensitive matching
            importance_normalized = match_importance.lower().strip()

            if importance_normalized not in match_bonus_points:
                raise ValueError(f"Invalid match_importance: {match_importance}. Must be one of: regular, important, derby, final")

            match_bonus = match_bonus_points[importance_normalized]

        # Calculate total points
        total_points = base_points + tier_bonus + match_bonus

        # Return the breakdown of loyalty points
        return {
            'base_points': base_points,
            'tier_bonus': tier_bonus,
            'match_bonus': match_bonus,
            'total_points': total_points
        }

    @is_tool()
    def apply_discount_code(self, discount_code: str, original_price: float, match_id: str = None, customer_id: str = None) -> dict:
        """
        Validate and apply a discount code to calculate the discounted price.

        This method validates the discount code against various criteria:
        - Code existence and active status
        - Expiration date
        - Match-specific restrictions
        - Minimum purchase amount
        - Usage limit

        Args:
            discount_code: The discount code string to validate and apply
            original_price: The original price before discount
            match_id: Optional match identifier for match-specific codes
            customer_id: Optional customer identifier (not used in current validation)

        Returns:
            Dictionary containing validation result and discount details
        """
        from datetime import datetime

        # Input validation
        if not discount_code or not isinstance(discount_code, str):
            raise ValueError("discount_code must be a non-empty string")

        if not isinstance(original_price, (int, float)) or original_price < 0:
            raise ValueError("original_price must be a non-negative number")

        # Access database
        db = self.db
        discount_code_table = getattr(db, 'discount_code', None)

        # Check if discount code exists in database
        if discount_code_table is None or discount_code not in discount_code_table:
            return {
                'is_valid': False,
                'discount_type': None,
                'discount_value': 0.0,
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': None
            }

        # Retrieve discount code record
        code_record = discount_code_table[discount_code]

        # Validation 1: Check if code is active
        if not code_record.is_active:
            return {
                'is_valid': False,
                'discount_type': code_record.discount_type,
                'discount_value': float(code_record.discount_value),
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
            }

        # Validation 2: Check expiration date
        current_date = datetime.now().date()
        if code_record.expiration_date < current_date:
            return {
                'is_valid': False,
                'discount_type': code_record.discount_type,
                'discount_value': float(code_record.discount_value),
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
            }

        # Validation 3: Check match-specific restriction
        # If code has a specific match_id, the provided match_id must match
        if code_record.match_id is not None and match_id != code_record.match_id:
            return {
                'is_valid': False,
                'discount_type': code_record.discount_type,
                'discount_value': float(code_record.discount_value),
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
            }

        # Validation 4: Check minimum purchase amount
        if code_record.min_purchase_amount is not None and original_price < code_record.min_purchase_amount:
            return {
                'is_valid': False,
                'discount_type': code_record.discount_type,
                'discount_value': float(code_record.discount_value),
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
            }

        # Validation 5: Check usage limit
        if code_record.max_uses is not None and code_record.times_used >= code_record.max_uses:
            return {
                'is_valid': False,
                'discount_type': code_record.discount_type,
                'discount_value': float(code_record.discount_value),
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
            }

        # All validations passed, calculate discount
        discount_amount = 0.0

        if code_record.discount_type == 'percentage':
            # Calculate percentage discount
            discount_amount = original_price * (float(code_record.discount_value) / 100.0)
        elif code_record.discount_type == 'fixed_amount':
            # Apply fixed amount discount, but not exceeding original price
            discount_amount = min(float(code_record.discount_value), original_price)
        else:
            # Unknown discount type, treat as invalid
            return {
                'is_valid': False,
                'discount_type': code_record.discount_type,
                'discount_value': float(code_record.discount_value),
                'discount_amount': 0.0,
                'final_price': original_price,
                'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
            }

        # Calculate final price
        final_price = original_price - discount_amount

        # Ensure final price is not negative
        final_price = max(0.0, final_price)

        # Return successful validation result
        return {
            'is_valid': True,
            'discount_type': code_record.discount_type,
            'discount_value': float(code_record.discount_value),
            'discount_amount': round(discount_amount, 2),
            'final_price': round(final_price, 2),
            'expiration_date': code_record.expiration_date.strftime('%Y-%m-%d')
        }

    @is_tool()
    def reserve_seats(self, match_id: str, seat_ids: list, customer_id: str, reservation_duration_minutes: int = 15):
        """
        Reserve selected seats for a specific duration to allow customer to complete booking.

        This method:
        1. Validates that all requested seats exist and are available
        2. Creates a reservation record with expiration time
        3. Updates seat statuses to 'reserved'
        4. Creates reservation_seat junction records
        5. Returns reservation details including expiration time

        Args:
            match_id: Unique identifier for the match
            seat_ids: List of seat identifiers to reserve
            customer_id: Unique identifier for the customer
            reservation_duration_minutes: Duration in minutes to hold the reservation (default: 15)

        Returns:
            dict containing:
                - reservation_id: Generated unique reservation identifier
                - expiration_time: Expiration time in 'yyyy-mm-dd HH:MM:SS' format
                - reserved_seats: List of successfully reserved seat IDs

        Raises:
            RuntimeError: If seats are not available, don't exist, or don't belong to the match
        """
        from datetime import datetime, timedelta
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Validate inputs
        if not match_id or not isinstance(match_id, str):
            raise RuntimeError("Invalid match_id: must be a non-empty string")

        if not seat_ids or not isinstance(seat_ids, list) or len(seat_ids) == 0:
            raise RuntimeError("Invalid seat_ids: must be a non-empty list")

        if not customer_id or not isinstance(customer_id, str):
            raise RuntimeError("Invalid customer_id: must be a non-empty string")

        if not isinstance(reservation_duration_minutes, int) or reservation_duration_minutes <= 0:
            raise RuntimeError("Invalid reservation_duration_minutes: must be a positive integer")

        # Get seat table from database
        seat_table = getattr(db, 'seat', None)
        if seat_table is None:
            raise RuntimeError("Seat table not found in database")

        # Validate all seats before making any changes
        validated_seats = []
        for seat_id in seat_ids:
            # Check if seat exists
            seat = seat_table.get(seat_id)
            if seat is None:
                raise RuntimeError(f"Seat {seat_id} does not exist")

            # Check if seat belongs to the specified match
            if seat.match_id != match_id:
                raise RuntimeError(f"Seat {seat_id} does not belong to match {match_id}")

            # Check if seat is available
            if seat.status != 'available':
                raise RuntimeError(f"Seat {seat_id} is not available (current status: {seat.status})")

            validated_seats.append(seat)

        # Generate unique reservation_id
        reservation_id = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Calculate reservation and expiration times
        reservation_time = datetime.now()
        expiration_time = reservation_time + timedelta(minutes=reservation_duration_minutes)

        # Create reservation record
        reservation_table = getattr(db, 'reservation', None)
        if reservation_table is None:
            reservation_table = {}
            setattr(db, 'reservation', reservation_table)

        new_reservation = Reservation(
            reservation_id=reservation_id,
            match_id=match_id,
            customer_id=customer_id,
            reservation_time=reservation_time,
            expiration_time=expiration_time,
            status='active'
        )
        reservation_table[reservation_id] = new_reservation
        setattr(db, 'reservation', reservation_table)

        # Get or create reservation_seat table
        reservation_seat_table = getattr(db, 'reservation_seat', None)
        if reservation_seat_table is None:
            reservation_seat_table = {}
            setattr(db, 'reservation_seat', reservation_seat_table)

        # Update seat statuses and create reservation_seat records
        reserved_seat_ids = []
        for seat in validated_seats:
            # Update seat status to 'reserved'
            seat.status = 'reserved'
            seat_table[seat.seat_id] = seat

            # Create reservation_seat junction record
            reservation_seat_id = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]
            reservation_seat = ReservationSeat(
                reservation_seat_id=reservation_seat_id,
                reservation_id=reservation_id,
                seat_id=seat.seat_id
            )
            reservation_seat_table[reservation_seat_id] = reservation_seat

            reserved_seat_ids.append(seat.seat_id)

        # Save updated tables back to database
        setattr(db, 'seat', seat_table)
        setattr(db, 'reservation_seat', reservation_seat_table)

        # Format expiration time as string in 'yyyy-mm-dd HH:MM:SS' format
        expiration_time_str = expiration_time.strftime('%Y-%m-%d %H:%M:%S')

        # Return reservation details
        return {
            'reservation_id': reservation_id,
            'expiration_time': expiration_time_str,
            'reserved_seats': reserved_seat_ids
        }

    @is_tool()
    def generate_booking_reference(self, match_id: str, customer_id: str, booking_date: str = None) -> dict:
        """
        Generate a unique human-readable booking reference code.

        This method creates a booking reference in the format: TEAM-YYYY-HASH
        where TEAM is derived from the match teams, YYYY is the year, and HASH is a unique identifier.

        Args:
            match_id: Unique identifier for the match
            customer_id: Unique identifier for the customer
            booking_date: Date of booking in yyyy-mm-dd format (optional, defaults to current date)

        Returns:
            dict: Dictionary containing the generated booking_reference

        Raises:
            ValueError: If match_id or customer_id is empty/invalid, or if booking_date format is invalid
        """
        from datetime import datetime
        import secrets
        import hashlib
        import re

        # Validate input parameters
        if not match_id or not isinstance(match_id, str) or not match_id.strip():
            raise ValueError("match_id must be a non-empty string")

        if not customer_id or not isinstance(customer_id, str) or not customer_id.strip():
            raise ValueError("customer_id must be a non-empty string")

        # Parse and validate booking_date
        if booking_date is None:
            # Use current date if not provided
            booking_datetime = datetime.now()
        else:
            if not isinstance(booking_date, str):
                raise ValueError("booking_date must be a string in yyyy-mm-dd format")

            booking_date = booking_date.strip()

            # Validate date format (yyyy-mm-dd)
            if not re.match(r'^\d{4}-\d{2}-\d{2}$', booking_date):
                raise ValueError("booking_date must be in yyyy-mm-dd format")

            try:
                booking_datetime = datetime.strptime(booking_date, "%Y-%m-%d")
            except ValueError as e:
                raise ValueError(f"Invalid booking_date: {str(e)}")

        # Extract year from booking date
        year = booking_datetime.strftime("%Y")

        # Generate unique hash component
        # Combine match_id, customer_id, and timestamp for uniqueness
        unique_string = f"{match_id}_{customer_id}_{datetime.now().isoformat()}_{secrets.token_hex(8)}"
        hash_value = hashlib.sha256(unique_string.encode()).hexdigest()[:6].upper()

        # Create team prefix from match_id
        # Extract alphabetic characters from match_id to create a team prefix
        # If match_id contains letters, use first 2-3 letters; otherwise use 'BK' (booking)
        team_letters = ''.join(filter(str.isalpha, match_id)).upper()

        if len(team_letters) >= 2:
            team_prefix = team_letters[:2]
        elif len(team_letters) == 1:
            team_prefix = team_letters[0] + 'X'
        else:
            # Fallback: use 'BK' for booking if no letters in match_id
            team_prefix = 'BK'

        # Construct booking reference in format: PREFIX-YYYY-HASH
        booking_reference = f"{team_prefix}-{year}-{hash_value}"

        return {
            "booking_reference": booking_reference
        }

    @is_tool()
    def get_booking_history(self, customer_id: str, include_cancelled: bool = False, limit: int = None):
        """
        Retrieve complete booking history for a customer including past and upcoming bookings.

        Args:
            customer_id: Unique identifier for the customer
            include_cancelled: Whether to include cancelled bookings (default: False)
            limit: Maximum number of bookings to return (default: None, returns all)

        Returns:
            Dictionary containing:
            - total_bookings: Total number of bookings
            - upcoming_bookings: List of upcoming bookings
            - past_bookings: List of past bookings

        Raises:
            KeyError: If customer_id does not exist in the system
        """
        from datetime import datetime

        # Access database
        db = self.db

        # Verify customer exists
        customer_table = getattr(db, 'customer', None)
        if customer_table is None or customer_id not in customer_table:
            raise KeyError(f"Customer with ID '{customer_id}' does not exist in the system")

        # Get booking table
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            # No bookings exist, return empty result
            return {
                'total_bookings': 0,
                'upcoming_bookings': [],
                'past_bookings': []
            }

        # Get current time for comparison
        current_time = datetime.now()

        # Filter bookings for this customer
        customer_bookings = []
        for booking_id, booking in booking_table.items():
            # Check if booking belongs to this customer
            if booking.customer_id == customer_id:
                # Filter by cancelled status if needed
                if not include_cancelled and booking.status == 'cancelled':
                    continue
                customer_bookings.append(booking)

        # Sort bookings by booking_time (most recent first)
        customer_bookings.sort(key=lambda b: b.booking_time, reverse=True)

        # Apply limit if specified
        if limit is not None and limit > 0:
            customer_bookings = customer_bookings[:limit]

        # Separate into upcoming and past bookings
        # We need to get match information to determine if booking is upcoming or past
        match_table = getattr(db, 'match', None)

        upcoming_bookings = []
        past_bookings = []

        for booking in customer_bookings:
            # Create booking detail object
            booking_detail = {
                'booking_id': booking.booking_id,
                'booking_reference': booking.booking_reference,
                'match_id': booking.match_id,
                'customer_id': booking.customer_id,
                'email': booking.email,
                'phone': booking.phone,
                'total_price': float(booking.total_price),
                'booking_time': booking.booking_time.strftime('%Y-%m-%d %H:%M:%S'),
                'status': booking.status
            }

            # Add optional fields if present
            if booking.reservation_id:
                booking_detail['reservation_id'] = booking.reservation_id
            if booking.payment_id:
                booking_detail['payment_id'] = booking.payment_id
            if booking.cancellation_time:
                booking_detail['cancellation_time'] = booking.cancellation_time.strftime('%Y-%m-%d %H:%M:%S')
            if booking.cancellation_reason:
                booking_detail['cancellation_reason'] = booking.cancellation_reason
            if booking.refund_amount is not None:
                booking_detail['refund_amount'] = float(booking.refund_amount)

            # Determine if booking is upcoming or past based on match date
            if match_table and booking.match_id in match_table:
                match = match_table[booking.match_id]
                match_date = match.match_date
                booking_detail['match_date'] = match_date.strftime('%Y-%m-%d %H:%M:%S')

                # Add match details
                booking_detail['home_team'] = match.home_team
                booking_detail['away_team'] = match.away_team
                booking_detail['venue_name'] = match.venue_name
                booking_detail['competition'] = match.competition

                # Classify as upcoming or past
                if match_date > current_time:
                    upcoming_bookings.append(booking_detail)
                else:
                    past_bookings.append(booking_detail)
            else:
                # If match not found or no match table, treat as past booking
                past_bookings.append(booking_detail)

        # Calculate total bookings
        total_bookings = len(upcoming_bookings) + len(past_bookings)

        # Return result
        return {
            'total_bookings': total_bookings,
            'upcoming_bookings': upcoming_bookings,
            'past_bookings': past_bookings
        }

    @is_tool()
    def send_booking_reminder(self, booking_id: str, reminder_type: str, notification_channels: list = None):
        """
        Send reminder notification to customer about upcoming match

        Args:
            booking_id: Unique identifier for the booking
            reminder_type: Type of reminder (24_hours, 2_hours, etc.)
            notification_channels: Channels to send reminder through (default: ['email'])

        Returns:
            dict: Contains reminder_sent (bool), sent_time (str), channels_used (list)

        Raises:
            RuntimeError: If booking doesn't exist, is not confirmed, or reminder sending fails
        """
        from datetime import datetime

        # Access database
        db = self.db

        # Validate booking_id parameter
        if not booking_id or not isinstance(booking_id, str):
            raise RuntimeError("booking_id must be a non-empty string")

        # Validate reminder_type parameter
        if not reminder_type or not isinstance(reminder_type, str):
            raise RuntimeError("reminder_type must be a non-empty string")

        # Set default notification channels if not provided
        if notification_channels is None:
            notification_channels = ['email']

        # Validate notification_channels parameter
        if not isinstance(notification_channels, list):
            raise RuntimeError("notification_channels must be a list")

        if len(notification_channels) == 0:
            raise RuntimeError("notification_channels cannot be empty")

        # Retrieve booking table from database
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise RuntimeError("Booking table does not exist in database")

        # Check if booking exists
        booking = booking_table.get(booking_id)
        if booking is None:
            raise RuntimeError(f"Booking with ID '{booking_id}' does not exist")

        # Verify booking is confirmed (pre-condition check)
        if booking.status != 'confirmed':
            raise RuntimeError(f"Booking '{booking_id}' must be confirmed to send reminder. Current status: {booking.status}")

        # Retrieve communication_log table from database
        communication_log_table = getattr(db, 'communication_log', None)
        if communication_log_table is None:
            raise RuntimeError("Communication_log table does not exist in database")

        # Get current time for sent_time
        current_time = datetime.now()
        sent_time_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Track successfully sent channels
        channels_used = []

        # Send reminder through each requested channel
        for channel in notification_channels:
            # Validate channel type
            if not isinstance(channel, str):
                continue

            # Normalize channel name
            channel_lower = channel.lower().strip()

            # Determine recipient address based on channel
            recipient_address = None
            if channel_lower == 'email':
                recipient_address = booking.email
            elif channel_lower == 'sms':
                # Use phone if available
                if hasattr(booking, 'phone') and booking.phone:
                    recipient_address = booking.phone
                else:
                    # Skip this channel if phone not available
                    continue
            elif channel_lower == 'push':
                # For push notifications, use customer_id as recipient
                recipient_address = booking.customer_id
            else:
                # For any other channel, use email as fallback
                recipient_address = booking.email

            # Validate recipient address exists
            if not recipient_address:
                continue

            # Generate unique message_id for this communication
            # Use MSG prefix to match existing records format
            message_id = f"MSG{len(communication_log_table) + 1:04d}"

            # Ensure message_id is unique
            while message_id in communication_log_table:
                # Extract number from message_id and increment
                num = int(message_id[3:])  # MSG prefix is 3 characters
                message_id = f"MSG{num + 1:04d}"


            communication_entry = CommunicationLog(
                message_id=message_id,
                customer_id=booking.customer_id,
                booking_id=booking_id,
                communication_type="reminder",
                channel=channel_lower,
                recipient_address=recipient_address,
                sent_time=current_time,
                status='sent'
            )

            # Add to communication log
            communication_log_table[message_id] = communication_entry

            # Mark channel as successfully used
            channels_used.append(channel_lower)

        # Check if at least one channel was successful
        if len(channels_used) == 0:
            raise RuntimeError(f"Failed to send reminder through any of the requested channels: {notification_channels}")

        # Return success result
        return {
            'reminder_sent': True,
            'sent_time': sent_time_str,
            'channels_used': channels_used
        }

    @is_tool()
    def get_venue_seating_map(self, venue_name: str, match_id: Optional[str] = None) -> dict:
        """
        Retrieve the seating map layout for a specific venue.

        This method retrieves venue information and constructs a seating map with section details.
        If match_id is provided, it can be used to show seat availability overlay.

        Args:
            venue_name: Name of the venue to retrieve seating map for
            match_id: Optional match ID to show availability overlay

        Returns:
            Dictionary containing:
            - venue_name: Name of the venue
            - total_capacity: Total seating capacity
            - sections: List of seating sections with details (section_name, capacity, categories)

        Raises:
            KeyError: If venue is not found in the database
        """
        from thefuzz import process

        # Get database instance
        db = self.db

        # Retrieve venue table
        venue_table = getattr(db, 'venue', None)
        if venue_table is None or len(venue_table) == 0:
            raise KeyError(f"Venue table is empty or does not exist")

        # Find the venue using fuzzy matching for natural language venue names
        venue_names = list(venue_table.keys())
        matched_result = process.extractOne(venue_name, venue_names)

        # If no good match found (score too low), raise KeyError
        if matched_result is None or matched_result[1] < 60:
            raise KeyError(f"Venue '{venue_name}' not found in the database")

        matched_venue_name = matched_result[0]
        venue = venue_table[matched_venue_name]

        # Retrieve venue_section table
        venue_section_table = getattr(db, 'venue_section', None)

        # Initialize sections list
        sections = []

        if venue_section_table is not None:
            # Filter sections that belong to this venue
            for section_id, section_data in venue_section_table.items():
                # Use fuzzy matching for section venue_name comparison
                if process.extractOne(section_data.venue_name, [matched_venue_name])[1] >= 90:
                    # Get all unique categories for this section from seat table if match_id provided
                    categories = set()

                    # If match_id is provided, get categories from seat table
                    if match_id is not None:
                        seat_table = getattr(db, 'seat', None)
                        if seat_table is not None:
                            for seat_id, seat_data in seat_table.items():
                                # Match by venue_name, section, and match_id (exact match for IDs)
                                if (process.extractOne(seat_data.venue_name, [matched_venue_name])[1] >= 90 and 
                                    seat_data.section == section_data.section and
                                    seat_data.match_id == match_id):
                                    categories.add(seat_data.category)
                    else:
                        # If no match_id, get all categories for this section across all matches
                        seat_table = getattr(db, 'seat', None)
                        if seat_table is not None:
                            for seat_id, seat_data in seat_table.items():
                                if (process.extractOne(seat_data.venue_name, [matched_venue_name])[1] >= 90 and 
                                    seat_data.section == section_data.section):
                                    categories.add(seat_data.category)

                    # Create section info dictionary
                    section_info = {
                        'section_name': section_data.section,
                        'capacity': section_data.capacity,
                        'categories': sorted(list(categories)) if categories else []
                    }
                    sections.append(section_info)

        # Sort sections by section_name for consistent output
        sections.sort(key=lambda x: x['section_name'])

        # Construct return dictionary
        result = {
            'venue_name': matched_venue_name,
            'total_capacity': venue.total_capacity,
            'sections': sections
        }

        return result

    @is_tool()
    def get_booking_invoice(self, booking_id: str):
        """
        Generate and retrieve invoice details for a booking.

        This method retrieves booking information and generates a detailed invoice
        including line items for each seat, calculating subtotal, tax, and total amounts.

        Args:
            booking_id: Unique identifier for the booking

        Returns:
            dict: Invoice details containing:
                - invoice_number: Unique invoice number
                - invoice_date: Date invoice was generated (yyyy-mm-dd format)
                - line_items: List of invoice line items with description, quantity, unit_price, total
                - subtotal: Subtotal amount before discounts/tax
                - tax_amount: Tax amount (currently 0.0)
                - total_amount: Total invoice amount

        Raises:
            KeyError: If booking_id does not exist in the database
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access database
        db = self.db

        # Retrieve booking data
        booking_table = getattr(db, "booking", None)
        if booking_table is None or booking_id not in booking_table:
            raise KeyError(f"Booking with ID '{booking_id}' not found")

        booking = booking_table[booking_id]

        # Verify booking is confirmed (pre-condition check)
        if booking.status not in ["confirmed", "completed"]:
            raise ValueError(f"Booking '{booking_id}' is not confirmed. Current status: {booking.status}")

        # Retrieve booking_seat data to get seat details
        booking_seat_table = getattr(db, "booking_seat", None)
        seat_table = getattr(db, "seat", None)

        if booking_seat_table is None or seat_table is None:
            raise RuntimeError("Required database tables (booking_seat or seat) not available")

        # Find all seats associated with this booking
        booking_seats = [bs for bs in booking_seat_table.values() if bs.booking_id == booking_id]

        if not booking_seats:
            raise ValueError(f"No seats found for booking '{booking_id}'")

        # Generate invoice number using hash-based approach
        invoice_prefix = "INV-"
        year_part = booking.booking_time.strftime("%Y-")
        random_part = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:6]
        invoice_number = invoice_prefix + year_part + random_part

        # Generate invoice date (current date in yyyy-mm-dd format)
        invoice_date = datetime.now().strftime("%Y-%m-%d")

        # Build line items by aggregating seats by category
        # Group seats by category to create consolidated line items
        category_map = {}  # category -> {quantity, unit_price, seat_ids}

        for booking_seat in booking_seats:
            seat = seat_table.get(booking_seat.seat_id)
            if seat is None:
                # Skip if seat not found (data integrity issue)
                continue

            category = seat.category
            unit_price = float(seat.price)

            if category not in category_map:
                category_map[category] = {
                    "quantity": 0,
                    "unit_price": unit_price,
                    "seat_ids": []
                }

            category_map[category]["quantity"] += 1
            category_map[category]["seat_ids"].append(seat.seat_id)

        # Create line items from category map
        line_items = []
        subtotal = 0.0

        for category, data in category_map.items():
            quantity = data["quantity"]
            unit_price = data["unit_price"]
            total = round(quantity * unit_price, 2)

            line_item = {
                "description": f"Match Ticket - {category}",
                "quantity": quantity,
                "unit_price": unit_price,
                "total": total
            }
            line_items.append(line_item)
            subtotal += total

        # Round subtotal to 2 decimal places
        subtotal = round(subtotal, 2)

        # Tax amount is currently 0.0 (no tax applied)
        tax_amount = 0.0

        # Calculate total amount (booking's total_price should match, but use actual calculation)
        # Note: The example shows total_amount (285.0) < subtotal (300.0), suggesting a discount was applied
        # We'll use the booking's total_price as the authoritative total amount
        total_amount = round(float(booking.total_price), 2)

        # Construct and return invoice
        invoice = {
            "invoice_number": invoice_number,
            "invoice_date": invoice_date,
            "line_items": line_items,
            "subtotal": subtotal,
            "tax_amount": tax_amount,
            "total_amount": total_amount
        }

        return invoice

    @is_tool()
    def transfer_ticket(self, booking_id: str, seat_id: str, from_customer_id: str, to_customer_id: str, transfer_reason: str = None):
        """
        Transfer a ticket from one customer to another.

        This method performs the following operations:
        1. Validates that the booking exists and is in confirmed status
        2. Validates that both customers exist in the system
        3. Verifies that the seat belongs to the booking and is owned by from_customer
        4. Creates a transfer record with a new booking reference for the recipient
        5. Updates the booking_seat record to reflect the transfer (if needed for tracking)

        Args:
            booking_id: Unique identifier for the booking
            seat_id: Unique identifier for the seat to transfer
            from_customer_id: Current owner customer ID
            to_customer_id: New owner customer ID
            transfer_reason: Optional reason for transfer

        Returns:
            dict: Contains transfer_id, transfer_time, and new_booking_reference

        Raises:
            RuntimeError: If validation fails or transfer cannot be completed
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Retrieve necessary tables
        booking_table = getattr(db, "booking", None)
        booking_seat_table = getattr(db, "booking_seat", None)
        customer_table = getattr(db, "customer", None)
        ticket_transfer_table = getattr(db, "ticket_transfer", None)

        # Validate that all required tables exist
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")
        if booking_seat_table is None:
            raise RuntimeError("Booking seat table not found in database")
        if customer_table is None:
            raise RuntimeError("Customer table not found in database")
        if ticket_transfer_table is None:
            # Initialize empty table if it doesn't exist
            ticket_transfer_table = {}
            setattr(db, "ticket_transfer", ticket_transfer_table)

        # Validate booking exists and is in confirmed status
        booking = booking_table.get(booking_id)
        if booking is None:
            raise RuntimeError(f"Booking with ID '{booking_id}' does not exist")

        if booking.status != "confirmed":
            raise RuntimeError(f"Booking '{booking_id}' is not in confirmed status (current status: {booking.status})")

        # Validate that the booking belongs to from_customer
        if booking.customer_id != from_customer_id:
            raise RuntimeError(f"Booking '{booking_id}' does not belong to customer '{from_customer_id}'")

        # Validate both customers exist
        from_customer = customer_table.get(from_customer_id)
        if from_customer is None:
            raise RuntimeError(f"Customer with ID '{from_customer_id}' does not exist")

        to_customer = customer_table.get(to_customer_id)
        if to_customer is None:
            raise RuntimeError(f"Customer with ID '{to_customer_id}' does not exist")

        # Validate that the seat belongs to this booking
        booking_seat_record = None
        for bs_id, bs in booking_seat_table.items():
            if bs.booking_id == booking_id and bs.seat_id == seat_id:
                booking_seat_record = bs
                break

        if booking_seat_record is None:
            raise RuntimeError(f"Seat '{seat_id}' is not associated with booking '{booking_id}'")

        # Check if the ticket has already been used (optional business rule - tickets cannot be transferred after use)
        if booking_seat_record.is_used:
            raise RuntimeError(f"Ticket for seat '{seat_id}' has already been used and cannot be transferred")

        # Generate transfer ID
        transfer_id = "T" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:9]

        # Generate new booking reference for the recipient
        # Format: Original reference + "-TRANSFER-" + short hash
        new_booking_reference = f"{booking.booking_reference}-TRANSFER-{hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:6].upper()}"

        # Get current transfer time
        transfer_time = datetime.now()

        # Create transfer record using TicketTransfer class from database module
        transfer_record = TicketTransfer(
            transfer_id=transfer_id,
            booking_id=booking_id,
            seat_id=seat_id,
            from_customer_id=from_customer_id,
            to_customer_id=to_customer_id,
            transfer_time=transfer_time,
            transfer_reason=transfer_reason if transfer_reason else None,
            new_booking_reference=new_booking_reference
        )

        # Save transfer record to database
        ticket_transfer_table[transfer_id] = transfer_record
        setattr(db, "ticket_transfer", ticket_transfer_table)

        # Note: The actual booking ownership transfer is tracked via the ticket_transfer table
        # The new_booking_reference serves as the recipient's reference
        # The original booking remains unchanged, maintaining audit trail

        # Return transfer details in the required format
        return {
            "transfer_id": transfer_id,
            "transfer_time": transfer_time.strftime("%Y-%m-%d %H:%M:%S"),
            "new_booking_reference": new_booking_reference
        }

    @is_tool()
    def get_venue_facilities_info(self, venue_name: str) -> dict:
        """
        Retrieve information about venue facilities such as parking, food outlets, and amenities

        Args:
            venue_name: Name of the venue to retrieve facilities for

        Returns:
            Dictionary containing parking_facilities, food_outlets, accessibility_features, and wifi_available

        Raises:
            KeyError: If venue not found in database
        """
        from thefuzz import process

        # Access the database
        db = self.db

        # Get venue table data
        venue_table = getattr(db, 'venue', None)
        if venue_table is None or len(venue_table) == 0:
            raise KeyError(f"Venue table is empty or not found")

        # Find the venue using fuzzy matching on venue names
        venue_names = list(venue_table.keys())
        matched_venue = process.extractOne(venue_name, venue_names)

        # Check if we found a reasonable match (using threshold of 80)
        if matched_venue is None or matched_venue[1] < 80:
            raise KeyError(f"Venue '{venue_name}' not found in database")

        matched_venue_name = matched_venue[0]
        venue = venue_table[matched_venue_name]

        # Get venue facility table data
        venue_facility_table = getattr(db, 'venue_facility', None)

        # Initialize result structure
        parking_facilities = []
        food_outlets = []
        accessibility_features = []
        wifi_available = venue.wifi_available

        # If venue_facility table exists, extract facilities for this venue
        if venue_facility_table is not None and len(venue_facility_table) > 0:
            for facility_id, facility in venue_facility_table.items():
                # Check if this facility belongs to the matched venue
                if facility.venue_name == matched_venue_name:
                    facility_type = facility.facility_type.lower()

                    # Categorize facilities by type
                    if facility_type == 'parking':
                        # Build parking facility object with available fields
                        parking_info = {
                            'name': facility.facility_name
                        }
                        # Add optional fields if they exist
                        if facility.capacity is not None:
                            parking_info['capacity'] = facility.capacity
                        if facility.price is not None:
                            parking_info['price'] = float(facility.price)

                        parking_facilities.append(parking_info)

                    elif facility_type == 'food_outlet':
                        # Food outlets are stored as simple string names
                        food_outlets.append(facility.facility_name)

                    elif facility_type == 'accessibility':
                        # Accessibility features are stored as simple string names
                        accessibility_features.append(facility.facility_name)

        # Construct and return the result
        result = {
            'parking_facilities': parking_facilities,
            'food_outlets': food_outlets,
            'accessibility_features': accessibility_features,
            'wifi_available': wifi_available
        }

        return result

    @is_tool()
    def send_booking_confirmation(self, booking_id: str, customer_email: str, include_qr_code: bool = False):
        """
        Send booking confirmation details to customer via email

        This method:
        1. Validates the booking exists and is confirmed
        2. Validates the customer email format
        3. Creates a communication log entry for the confirmation email
        4. Returns email sending status with timestamp and message ID

        Args:
            booking_id: Unique identifier for the booking
            customer_email: Customer email address to send confirmation to
            include_qr_code: Whether to include QR code for tickets (optional, default False)

        Returns:
            dict: Contains email_sent (bool), sent_time (str in yyyy-mm-dd HH:MM:SS format), 
                  and message_id (str)

        Raises:
            RuntimeError: If booking not found, booking not confirmed, or email is invalid
        """
        from datetime import datetime
        import secrets
        import hashlib
        import re

        # Access database
        db = self.db

        # Validate booking_id parameter
        if not booking_id or not isinstance(booking_id, str):
            raise RuntimeError("Invalid booking_id: must be a non-empty string")

        # Validate customer_email parameter
        if not customer_email or not isinstance(customer_email, str):
            raise RuntimeError("Invalid customer_email: must be a non-empty string")

        # Validate email format using regex
        email_pattern = r'^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$'
        if not re.match(email_pattern, customer_email):
            raise RuntimeError(f"Invalid email format: {customer_email}")

        # Retrieve booking table from database
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Check if booking exists
        booking = booking_table.get(booking_id)
        if booking is None:
            raise RuntimeError(f"Booking not found: {booking_id}")

        # Validate booking status is confirmed
        if booking.status != 'confirmed':
            raise RuntimeError(f"Booking status must be 'confirmed', current status: {booking.status}")

        # Generate current timestamp for sent_time
        current_time = datetime.now()
        sent_time_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Generate unique message_id
        message_id = "MSG" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Retrieve communication_log table
        communication_log_table = getattr(db, 'communication_log', None)
        if communication_log_table is None:
            # Initialize empty communication_log table if not exists
            communication_log_table = {}
            setattr(db, 'communication_log', communication_log_table)

        # Create communication log entry
        # Note: include_qr_code parameter is noted but QR code generation is handled by email system
        from scale_env.environment.db import DictAccessMixin as BaseModel

        # Create CommunicationLog instance
        communication_log_entry = CommunicationLog(
            message_id=message_id,
            customer_id=booking.customer_id,
            booking_id=booking_id,
            communication_type='confirmation',
            channel='email',
            recipient_address=customer_email,
            sent_time=current_time,
            status='sent'
        )

        # Add to communication log table
        communication_log_table[message_id] = communication_log_entry
        setattr(db, 'communication_log', communication_log_table)

        # Return success response
        return {
            'email_sent': True,
            'sent_time': sent_time_str,
            'message_id': message_id
        }

    @is_tool()
    def release_seat_reservation(self, reservation_id: str) -> dict:
        """
        Release previously reserved seats making them available again.

        This method:
        1. Validates that the reservation exists and is active
        2. Retrieves all seats associated with the reservation
        3. Updates seat status from 'reserved' to 'available'
        4. Updates reservation status to 'released'
        5. Returns the list of released seat IDs and release timestamp

        Args:
            reservation_id: Unique identifier for the reservation to release

        Returns:
            Dictionary containing:
            - released_seats: List of seat IDs that were released
            - release_time: Timestamp when seats were released (yyyy-mm-dd HH:MM:SS format)

        Raises:
            KeyError: If reservation_id does not exist or reservation is not active
        """
        from datetime import datetime

        # Access database tables
        db = self.db
        reservation_table = getattr(db, "reservation", None)
        reservation_seat_table = getattr(db, "reservation_seat", None)
        seat_table = getattr(db, "seat", None)

        # Validate that all required tables exist
        if reservation_table is None:
            raise KeyError(f"Reservation table not found in database")
        if reservation_seat_table is None:
            raise KeyError(f"Reservation_seat table not found in database")
        if seat_table is None:
            raise KeyError(f"Seat table not found in database")

        # Check if reservation exists
        if reservation_id not in reservation_table:
            raise KeyError(f"Reservation with ID '{reservation_id}' does not exist")

        # Retrieve the reservation
        reservation = reservation_table[reservation_id]

        # Validate that reservation is active (pre-condition check)
        if reservation.status != "active":
            raise KeyError(f"Reservation '{reservation_id}' is not active (current status: {reservation.status})")

        # Find all seats associated with this reservation
        seat_ids_to_release = []
        reservation_seat_ids_to_update = []

        for rs_id, rs_item in reservation_seat_table.items():
            if rs_item.reservation_id == reservation_id:
                seat_ids_to_release.append(rs_item.seat_id)
                reservation_seat_ids_to_update.append(rs_id)

        # Validate that reservation has associated seats
        if not seat_ids_to_release:
            raise KeyError(f"No seats found for reservation '{reservation_id}'")

        # Record the release time
        release_time = datetime.now()
        release_time_str = release_time.strftime("%Y-%m-%d %H:%M:%S")

        # Update seat statuses from 'reserved' to 'available'
        for seat_id in seat_ids_to_release:
            if seat_id in seat_table:
                seat = seat_table[seat_id]
                # Only update if seat is currently reserved
                if seat.status == "reserved":
                    seat.status = "available"
                    # Update the seat in database
                    seat_table[seat_id] = seat

        # Update reservation status to 'released'
        reservation.status = "released"
        reservation_table[reservation_id] = reservation

        # Persist database changes
        setattr(db, "reservation", reservation_table)
        setattr(db, "seat", seat_table)

        # Return the result
        return {
            "released_seats": seat_ids_to_release,
            "release_time": release_time_str
        }

    @is_tool()
    def update_booking_contact_info(self, booking_id: str, new_email: str = None, new_phone: str = None):
        """
        Update contact information for an existing booking.

        This method updates the email and/or phone number for a booking.
        At least one of new_email or new_phone must be provided.
        The booking must exist and be in a modifiable state (not cancelled).

        Args:
            booking_id: Unique identifier for the booking to update
            new_email: New email address (optional)
            new_phone: New phone number (optional)

        Returns:
            dict: Contains:
                - updated (bool): Whether update was successful
                - update_time (str): Time when update was made in yyyy-mm-dd HH:MM:SS format
                - updated_fields (list): List of fields that were updated

        Raises:
            RuntimeError: If booking doesn't exist, is not modifiable, or no fields to update
        """
        from datetime import datetime

        # Validate that at least one field is provided for update
        if new_email is None and new_phone is None:
            raise RuntimeError("At least one of new_email or new_phone must be provided for update")

        # Access the database
        db = self.db

        # Get the booking table from database
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Check if booking exists using exact match (booking_id is a precise identifier)
        if booking_id not in booking_table:
            raise RuntimeError(f"Booking with ID '{booking_id}' does not exist")

        # Retrieve the booking record
        booking = booking_table[booking_id]

        # Check if booking is modifiable (not cancelled)
        if booking.status == 'cancelled':
            raise RuntimeError(f"Booking '{booking_id}' is cancelled and cannot be modified")

        # Track which fields are being updated
        updated_fields = []

        # Update email if provided
        if new_email is not None:
            # Basic email format validation
            if '@' not in new_email or '.' not in new_email.split('@')[-1]:
                raise ValueError("Invalid email format")
            booking.email = new_email
            updated_fields.append('email')

        # Update phone if provided
        if new_phone is not None:
            # Basic phone validation - ensure it's not empty
            if not new_phone.strip():
                raise ValueError("Phone number cannot be empty")
            booking.phone = new_phone
            updated_fields.append('phone')

        # Get current time for update timestamp
        update_time = datetime.now()

        # Save the updated booking back to database
        booking_table[booking_id] = booking
        setattr(db, 'booking', booking_table)

        # Return success response with update details
        return {
            'updated': True,
            'update_time': update_time.strftime('%Y-%m-%d %H:%M:%S'),
            'updated_fields': updated_fields
        }

    @is_tool()
    def get_match_details(self, match_id: str) -> dict:
        """
        Retrieve detailed information about a specific football match including teams, venue, time, and competition.

        Args:
            match_id: Unique identifier for the match

        Returns:
            Dictionary containing complete match details with the following keys:
            - match_id: Unique identifier for the match
            - home_team: Name of the home team
            - away_team: Name of the away team
            - match_date: Date and time of the match in yyyy-mm-dd HH:MM:SS format
            - venue_name: Name of the venue
            - competition: Name of the competition
            - season: Season information

        Raises:
            KeyError: If the match_id does not exist in the system
        """
        from datetime import datetime

        # Input validation: check if match_id is provided and is a string
        if not match_id or not isinstance(match_id, str):
            raise ValueError("match_id must be a non-empty string")

        # Access the database instance
        db = self.db

        # Retrieve the match table from database
        match_table = getattr(db, 'match', None)

        # Check if match table exists
        if match_table is None:
            raise KeyError("Match table not found in database")

        # Check if the match_id exists in the match table
        if match_id not in match_table:
            raise KeyError(f"Match with match_id '{match_id}' does not exist in the system")

        # Retrieve the match record
        match_record = match_table[match_id]

        # Extract match details from the database record
        # Convert datetime object to string format yyyy-mm-dd HH:MM:SS
        match_date_str = match_record.match_date.strftime("%Y-%m-%d %H:%M:%S")

        # Construct the return dictionary with all required fields
        match_details = {
            'match_id': match_record.match_id,
            'home_team': match_record.home_team,
            'away_team': match_record.away_team,
            'match_date': match_date_str,
            'venue_name': match_record.venue_name,
            'competition': match_record.competition,
            'season': match_record.season
        }

        return match_details

    @is_tool()
    def get_match_day_transport_info(self, match_id: str, venue_name: str) -> dict:
        """
        Retrieve transport and travel information for match day

        This method retrieves transport information including public transport options,
        road closures, and recommended arrival time for a specific match at a venue.

        Args:
            match_id: Unique identifier for the match
            venue_name: Name of the venue

        Returns:
            dict containing:
                - public_transport: List of public transport options with details
                - road_closures: List of road closure descriptions
                - recommended_arrival_time: Recommended arrival time in minutes before kick-off

        Raises:
            KeyError: If match_id or venue_name does not exist in database
        """
        from thefuzz import fuzz, process

        # Access database
        db = self.db

        # Retrieve match table data
        match_table = getattr(db, 'match', None)
        if match_table is None:
            raise KeyError(f"Match table does not exist in database")

        # Retrieve venue table data
        venue_table = getattr(db, 'venue', None)
        if venue_table is None:
            raise KeyError(f"Venue table does not exist in database")

        # Validate match_id exists (exact match for ID)
        if match_id not in match_table:
            raise KeyError(f"Match with match_id '{match_id}' does not exist")

        # Get the match record
        match_record = match_table[match_id]

        # Validate venue_name using fuzzy matching for natural language text
        venue_names = list(venue_table.keys())
        if not venue_names:
            raise KeyError(f"No venues exist in database")

        # Use fuzzy matching to find the best matching venue
        best_match, score = process.extractOne(venue_name, venue_names)

        # If similarity score is too low (< 60), consider it as not found
        if score < 60:
            raise KeyError(f"Venue with name '{venue_name}' does not exist")

        # Use the best matched venue name
        matched_venue_name = best_match
        venue_record = venue_table[matched_venue_name]

        # Verify that the match's venue matches the requested venue
        match_venue_similarity = fuzz.ratio(match_record.venue_name.lower(), venue_name.lower())
        if match_venue_similarity < 60:
            raise KeyError(f"Match '{match_id}' is not scheduled at venue '{venue_name}'")

        # Generate transport information based on venue
        # This is a generic implementation that provides realistic transport info

        # Public transport options - generated based on venue characteristics
        public_transport = []

        # Add metro/subway option for major venues
        if venue_record.total_capacity >= 50000:
            public_transport.append({
                'type': 'Metrolink',
                'station': f"{matched_venue_name}",
                'frequency': 'Every 12 minutes',
                'service_changes': 'Extended service until 23:00'
            })
            public_transport.append({
                'type': 'Bus',
                'station': f"{matched_venue_name} Bus Station",
                'frequency': 'Every 15 minutes',
                'service_changes': 'Additional services on match day'
            })
        elif venue_record.total_capacity >= 30000:
            public_transport.append({
                'type': 'Bus',
                'station': f"{matched_venue_name} Bus Stop",
                'frequency': 'Every 10 minutes',
                'service_changes': 'Extended service until 22:30'
            })
            public_transport.append({
                'type': 'Train',
                'station': f"Nearest Station to {matched_venue_name}",
                'frequency': 'Every 20 minutes',
                'service_changes': 'Extra trains scheduled'
            })
        else:
            public_transport.append({
                'type': 'Bus',
                'station': f"{matched_venue_name} Stop",
                'frequency': 'Every 20 minutes',
                'service_changes': 'Normal service'
            })

        # Road closures - based on venue size and match timing
        road_closures = []

        # Extract match time to determine road closure timing
        match_time = match_record.match_date
        closure_start_hour = max(match_time.hour - 2, 0)  # 2 hours before match

        if venue_record.total_capacity >= 50000:
            # Major venue - more road closures
            road_closures.append(f"Main access road closed from {closure_start_hour:02d}:00")
            road_closures.append(f"Surrounding streets restricted parking from {closure_start_hour:02d}:00")
            road_closures.append(f"Stadium perimeter roads pedestrian-only from {(closure_start_hour + 1):02d}:00")
        elif venue_record.total_capacity >= 30000:
            # Medium venue
            road_closures.append(f"Stadium approach road closed from {closure_start_hour:02d}:00")
            road_closures.append(f"Parking restrictions in effect from {closure_start_hour:02d}:00")
        else:
            # Smaller venue
            road_closures.append(f"Limited parking near venue from {closure_start_hour:02d}:00")

        # Recommended arrival time based on venue capacity
        # Larger venues need more time for entry processing
        if venue_record.total_capacity >= 60000:
            recommended_arrival_time = 90  # 90 minutes for very large venues
        elif venue_record.total_capacity >= 40000:
            recommended_arrival_time = 75  # 75 minutes for large venues
        elif venue_record.total_capacity >= 25000:
            recommended_arrival_time = 60  # 60 minutes for medium venues
        else:
            recommended_arrival_time = 45  # 45 minutes for smaller venues

        # Return the transport information
        return {
            'public_transport': public_transport,
            'road_closures': road_closures,
            'recommended_arrival_time': recommended_arrival_time
        }

    @is_tool()
    def check_accessibility_requirements(
        self,
        seat_ids: list,
        wheelchair_accessible: bool = None,
        companion_seat_required: bool = None,
        elevator_access_required: bool = None
    ) -> dict:
        """
        Check if seats meet specific accessibility requirements for customers with disabilities.

        This method validates whether the specified seats have the required accessibility features
        based on the provided criteria. It checks each seat against the requirements and compiles
        a comprehensive report of available features and any missing requirements.

        Args:
            seat_ids: List of seat identifiers to check
            wheelchair_accessible: Whether wheelchair accessibility is required (optional)
            companion_seat_required: Whether companion seat is required (optional)
            elevator_access_required: Whether elevator access is required (optional)

        Returns:
            dict: Contains:
                - meets_requirements (bool): Whether all seats meet all specified requirements
                - accessibility_features (list): List of accessibility features available across all checked seats
                - missing_features (list): List of required features not available in any of the seats

        Raises:
            ValueError: If seat_ids is empty, None, or if any seat_id does not exist in database
        """
        # Validate input parameters
        if not seat_ids or seat_ids is None:
            raise ValueError("seat_ids cannot be empty or None")

        if not isinstance(seat_ids, list):
            raise ValueError("seat_ids must be a list")

        # Access database
        db = self.db
        seat_table = getattr(db, 'seat', None)

        if seat_table is None:
            raise ValueError("Seat table not found in database")

        # Validate all seat_ids exist in database
        for seat_id in seat_ids:
            if seat_id not in seat_table:
                raise ValueError(f"Seat with ID '{seat_id}' not found in database")

        # Initialize tracking variables
        all_features_available = set()  # All unique features available across all seats
        feature_availability_count = {}  # Count how many seats have each feature
        total_seats = len(seat_ids)

        # Feature mapping for human-readable names
        feature_names = {
            'wheelchair_accessible': 'Wheelchair space',
            'companion_seat': 'Companion seat',
            'elevator_access': 'Elevator access'
        }

        # Check each seat for accessibility features
        for seat_id in seat_ids:
            seat = seat_table[seat_id]

            # Check and track wheelchair accessibility
            if seat.wheelchair_accessible:
                all_features_available.add('wheelchair_accessible')
                feature_availability_count['wheelchair_accessible'] = \
                    feature_availability_count.get('wheelchair_accessible', 0) + 1

            # Check and track companion seat
            if seat.companion_seat:
                all_features_available.add('companion_seat')
                feature_availability_count['companion_seat'] = \
                    feature_availability_count.get('companion_seat', 0) + 1

            # Check and track elevator access
            if seat.elevator_access:
                all_features_available.add('elevator_access')
                feature_availability_count['elevator_access'] = \
                    feature_availability_count.get('elevator_access', 0) + 1

        # Build list of accessibility features (features available in ALL seats)
        accessibility_features = []
        for feature_key in ['wheelchair_accessible', 'companion_seat', 'elevator_access']:
            if feature_availability_count.get(feature_key, 0) == total_seats:
                accessibility_features.append(feature_names[feature_key])

        # Additional common accessibility feature for seats with level/elevator access
        if 'elevator_access' in all_features_available:
            if 'Level access' not in accessibility_features:
                accessibility_features.append('Level access')

        # Check requirements and identify missing features
        missing_features = []
        meets_all_requirements = True

        # Check wheelchair accessibility requirement
        if wheelchair_accessible is not None and wheelchair_accessible:
            if feature_availability_count.get('wheelchair_accessible', 0) < total_seats:
                missing_features.append('Wheelchair space')
                meets_all_requirements = False

        # Check companion seat requirement
        if companion_seat_required is not None and companion_seat_required:
            if feature_availability_count.get('companion_seat', 0) < total_seats:
                missing_features.append('Companion seat')
                meets_all_requirements = False

        # Check elevator access requirement
        if elevator_access_required is not None and elevator_access_required:
            if feature_availability_count.get('elevator_access', 0) < total_seats:
                missing_features.append('Elevator access')
                meets_all_requirements = False

        # Return the comprehensive accessibility check result
        return {
            'meets_requirements': meets_all_requirements,
            'accessibility_features': accessibility_features,
            'missing_features': missing_features
        }

    @is_tool()
    def validate_booking_modification(self, booking_id: str, modification_type: str, modification_request_time: str = None):
        """
        Validate whether a booking can be modified based on match timing and booking policies.

        This method checks if a booking modification (seat_change, date_change, cancel) is allowed
        based on the time remaining before the match and the booking's current status.

        Modification policies:
        - Modifications are allowed up to 24 hours before match time
        - Cancelled bookings cannot be modified
        - Completed bookings cannot be modified
        - Different modification types may have different fees

        Args:
            booking_id: Unique identifier for the booking
            modification_type: Type of modification (seat_change, date_change, cancel)
            modification_request_time: Time when modification is requested in yyyy-mm-dd HH:MM:SS format
                                       If not provided, uses current time

        Returns:
            dict: Contains modification eligibility information:
                - is_modifiable (bool): Whether booking can be modified
                - reasons (list): Reasons for eligibility or restrictions
                - modification_deadline (str): Deadline for modifications in yyyy-mm-dd HH:MM:SS format
                - modification_fee (float): Fee for the modification

        Raises:
            KeyError: If booking_id does not exist in the database
        """
        from datetime import datetime, timedelta

        # Access database
        db = self.db

        # Retrieve booking data
        booking_table = getattr(db, "booking", None)
        if booking_table is None or booking_id not in booking_table:
            raise KeyError(f"Booking with ID '{booking_id}' not found")

        booking = booking_table[booking_id]

        # Retrieve match data
        match_table = getattr(db, "match", None)
        if match_table is None or booking.match_id not in match_table:
            raise KeyError(f"Match with ID '{booking.match_id}' not found")

        match = match_table[booking.match_id]

        # Parse modification request time or use current time
        if modification_request_time:
            request_time = datetime.strptime(modification_request_time.strip(), "%Y-%m-%d %H:%M:%S")
        else:
            request_time = datetime.now()

        # Get match date
        match_date = match.match_date

        # Calculate modification deadline (24 hours before match)
        modification_deadline = match_date - timedelta(hours=24)
        modification_deadline_str = modification_deadline.strftime("%Y-%m-%d %H:%M:%S")

        # Initialize result
        is_modifiable = True
        reasons = []
        modification_fee = 0.0

        # Check booking status - cancelled or completed bookings cannot be modified
        if booking.status == "cancelled":
            is_modifiable = False
            reasons.append("Booking has already been cancelled and cannot be modified")
            return {
                "is_modifiable": is_modifiable,
                "reasons": reasons,
                "modification_deadline": modification_deadline_str,
                "modification_fee": modification_fee
            }

        if booking.status == "completed":
            is_modifiable = False
            reasons.append("Booking has been completed and cannot be modified")
            return {
                "is_modifiable": is_modifiable,
                "reasons": reasons,
                "modification_deadline": modification_deadline_str,
                "modification_fee": modification_fee
            }

        # Check if modification is within allowed timeframe (24 hours before match)
        if request_time >= modification_deadline:
            is_modifiable = False
            reasons.append("Modification window has closed - modifications must be made at least 24 hours before match")
            return {
                "is_modifiable": is_modifiable,
                "reasons": reasons,
                "modification_deadline": modification_deadline_str,
                "modification_fee": modification_fee
            }

        # Check if match has already occurred
        if request_time >= match_date:
            is_modifiable = False
            reasons.append("Match has already occurred - modifications are not allowed")
            return {
                "is_modifiable": is_modifiable,
                "reasons": reasons,
                "modification_deadline": modification_deadline_str,
                "modification_fee": modification_fee
            }

        # If all checks pass, booking is modifiable
        # Set modification fee based on modification type
        if modification_type == "seat_change":
            modification_fee = 10.0
            reasons.append("Modification allowed up to 24 hours before match")
        elif modification_type == "date_change":
            modification_fee = 15.0
            reasons.append("Modification allowed up to 24 hours before match")
        elif modification_type == "cancel":
            modification_fee = 0.0
            reasons.append("Cancellation allowed up to 24 hours before match")
        else:
            # For other modification types, use default fee
            modification_fee = 10.0
            reasons.append("Modification allowed up to 24 hours before match")

        return {
            "is_modifiable": is_modifiable,
            "reasons": reasons,
            "modification_deadline": modification_deadline_str,
            "modification_fee": modification_fee
        }

    @is_tool()
    def get_pricing_tiers(self, match_id: str) -> dict:
        """
        Retrieve all pricing tiers and categories available for a specific match

        Args:
            match_id: Unique identifier for the match

        Returns:
            dict: Dictionary containing pricing_tiers list with tier details

        Raises:
            KeyError: If match_id does not exist in the system
        """
        # Access the database instance
        db = self.db

        # Validate that match_id parameter is provided and not empty
        if not match_id or not isinstance(match_id, str):
            raise ValueError("match_id must be a non-empty string")

        # Get the pricing_tier table from database
        pricing_tier_table = getattr(db, 'pricing_tier', None)

        # Check if pricing_tier table exists
        if pricing_tier_table is None:
            raise KeyError("Pricing tier table not found in database")

        # Find all pricing tiers for the given match_id
        matching_tiers = []

        # Iterate through all pricing tiers in the table
        for tier_id, tier_item in pricing_tier_table.items():
            # Check if this tier belongs to the requested match
            if tier_item.match_id == match_id:
                # Build the tier information dictionary
                tier_info = {
                    'tier_name': tier_item.tier_name,
                    'base_price': float(tier_item.base_price)
                }

                # Add optional price fields if they exist
                if tier_item.member_price is not None:
                    tier_info['member_price'] = float(tier_item.member_price)

                if tier_item.child_price is not None:
                    tier_info['child_price'] = float(tier_item.child_price)

                if tier_item.senior_price is not None:
                    tier_info['senior_price'] = float(tier_item.senior_price)

                # Add this tier to the matching tiers list
                matching_tiers.append(tier_info)

        # If no pricing tiers found for this match_id, raise KeyError
        if not matching_tiers:
            raise KeyError(f"No pricing tiers found for match_id: {match_id}")

        # Return the result in the expected format
        return {
            'pricing_tiers': matching_tiers
        }

    @is_tool()
    def get_match_attendance_stats(self, match_id: str):
        """
        Retrieve attendance statistics for a specific match including tickets sold and capacity utilization.

        Args:
            match_id: Unique identifier for the match

        Returns:
            Dictionary containing attendance statistics with the following keys:
            - total_capacity: Total venue capacity
            - tickets_sold: Number of tickets sold
            - tickets_available: Number of tickets still available
            - capacity_percentage: Percentage of capacity sold
            - sales_by_category: Breakdown of sales by seat category

        Raises:
            KeyError: If match_id does not exist in the system
        """
        # Access the database
        db = self.db

        # Get match table data
        match_table = getattr(db, 'match', None)
        if match_table is None or match_id not in match_table:
            raise KeyError(f"Match with ID '{match_id}' does not exist in the system")

        # Get seat table data
        seat_table = getattr(db, 'seat', None)
        if seat_table is None:
            raise KeyError("Seat table does not exist in the database")

        # Get booking_seat table data (optional, may not exist)
        booking_seat_table = getattr(db, 'booking_seat', None)

        # Filter seats for this specific match
        match_seats = {seat_id: seat for seat_id, seat in seat_table.items() 
                       if seat.match_id == match_id}

        if not match_seats:
            raise KeyError(f"No seats found for match ID '{match_id}'")

        # Calculate total capacity (all seats for this match)
        total_capacity = len(match_seats)

        # Calculate tickets sold (seats with status 'sold' or 'used')
        # Status can be: available, reserved, sold, used
        sold_statuses = {'sold', 'used'}
        tickets_sold = sum(1 for seat in match_seats.values() 
                          if seat.status in sold_statuses)

        # Calculate tickets available (seats with status 'available')
        tickets_available = sum(1 for seat in match_seats.values() 
                               if seat.status == 'available')

        # Calculate capacity percentage
        if total_capacity > 0:
            capacity_percentage = round((tickets_sold / total_capacity) * 100, 1)
        else:
            capacity_percentage = 0.0

        # Calculate sales breakdown by category
        # Initialize dictionary to track sales by category
        sales_by_category = {}

        # Count sold tickets by category
        for seat in match_seats.values():
            if seat.status in sold_statuses:
                category = seat.category
                if category not in sales_by_category:
                    sales_by_category[category] = 0
                sales_by_category[category] += 1

        # Return the attendance statistics
        return {
            'total_capacity': total_capacity,
            'tickets_sold': tickets_sold,
            'tickets_available': tickets_available,
            'capacity_percentage': capacity_percentage,
            'sales_by_category': sales_by_category
        }

    @is_tool()
    def check_age_restrictions(self, customer_age: int, seat_category: str, match_type: str = None) -> dict:
        """
        Check if customer meets age restrictions for specific seat categories or match types.

        This method validates customer age against predefined age restrictions for different
        seat categories and match types in a football stadium booking system.

        Args:
            customer_age: Age of the customer in years
            seat_category: Seat category being booked (e.g., 'Family Stand', 'VIP', 'Standard')
            match_type: Optional type of match (league, cup, european)

        Returns:
            Dictionary containing:
            - meets_requirements: Boolean indicating if customer meets age requirements
            - minimum_age: Minimum age required for the category/match type
            - maximum_age: Maximum age if applicable (None if no upper limit)
            - restrictions: List of age-related restriction messages

        Raises:
            ValueError: If customer_age is invalid or seat_category is empty
        """

        # Validate input parameters
        if not isinstance(customer_age, int):
            raise ValueError("customer_age must be an integer")

        if customer_age < 0:
            raise ValueError("customer_age must be non-negative")

        if not seat_category or not isinstance(seat_category, str):
            raise ValueError("seat_category must be a non-empty string")

        if seat_category.strip() == "":
            raise ValueError("seat_category cannot be empty or whitespace")

        # Normalize seat_category for comparison
        seat_category_normalized = seat_category.strip().lower()

        # Define age restrictions for different seat categories
        # These rules are based on common football stadium policies
        age_rules = {
            'family stand': {
                'min_age': 0,
                'max_age': None,
                'restrictions': [
                    'Children under 14 must be accompanied by an adult',
                    'At least one adult (18+) required per booking in Family Stand'
                ]
            },
            'junior section': {
                'min_age': 0,
                'max_age': 16,
                'restrictions': [
                    'Only available for children aged 16 and under',
                    'Children under 12 must be accompanied by an adult'
                ]
            },
            'adult section': {
                'min_age': 18,
                'max_age': None,
                'restrictions': [
                    'Must be 18 or older to book in adult section'
                ]
            },
            'vip': {
                'min_age': 18,
                'max_age': None,
                'restrictions': [
                    'VIP areas are restricted to adults aged 18 and over'
                ]
            },
            'premium': {
                'min_age': 16,
                'max_age': None,
                'restrictions': [
                    'Premium seating requires minimum age of 16',
                    'Under 18s may need adult supervision'
                ]
            },
            'standard': {
                'min_age': 0,
                'max_age': None,
                'restrictions': [
                    'Children under 12 must be accompanied by an adult'
                ]
            },
            'wheelchair accessible': {
                'min_age': 0,
                'max_age': None,
                'restrictions': [
                    'Companion tickets available for wheelchair users',
                    'Children under 14 must be accompanied by an adult'
                ]
            },
            'away section': {
                'min_age': 16,
                'max_age': None,
                'restrictions': [
                    'Away section requires minimum age of 16 for safety reasons',
                    'Under 18s must be accompanied by an adult'
                ]
            }
        }

        # Additional restrictions based on match type
        match_type_rules = {
            'european': {
                'additional_min_age': 16,
                'additional_restrictions': [
                    'European matches require minimum age of 16 due to late kick-off times'
                ]
            },
            'cup': {
                'additional_min_age': 0,
                'additional_restrictions': [
                    'Cup matches may have enhanced security measures'
                ]
            },
            'league': {
                'additional_min_age': 0,
                'additional_restrictions': []
            }
        }

        # Default rule for unknown categories
        default_rule = {
            'min_age': 0,
            'max_age': None,
            'restrictions': [
                'Standard age restrictions apply',
                'Children under 12 must be accompanied by an adult'
            ]
        }

        # Find matching rule for seat category (case-insensitive, fuzzy matching)
        from thefuzz import process

        # Get the best match for seat category
        category_keys = list(age_rules.keys())
        best_match = process.extractOne(seat_category_normalized, category_keys)

        # Use fuzzy match if similarity score is high enough (>= 70), otherwise use default
        if best_match and best_match[1] >= 70:
            matched_category = best_match[0]
            rule = age_rules[matched_category]
        else:
            rule = default_rule

        # Start with base rule
        min_age = rule['min_age']
        max_age = rule['max_age']
        restrictions = rule['restrictions'].copy()

        # Apply match type restrictions if provided
        if match_type:
            match_type_normalized = match_type.strip().lower()

            # Find matching match type rule
            match_type_keys = list(match_type_rules.keys())
            match_type_match = process.extractOne(match_type_normalized, match_type_keys)

            if match_type_match and match_type_match[1] >= 70:
                matched_match_type = match_type_match[0]
                match_rule = match_type_rules[matched_match_type]

                # Update minimum age if match type has stricter requirement
                if match_rule['additional_min_age'] > min_age:
                    min_age = match_rule['additional_min_age']

                # Add match type specific restrictions
                restrictions.extend(match_rule['additional_restrictions'])

        # Check if customer meets requirements
        meets_requirements = True

        # Check minimum age
        if customer_age < min_age:
            meets_requirements = False

        # Check maximum age if applicable
        if max_age is not None and customer_age > max_age:
            meets_requirements = False

        # Return validation result
        return {
            'meets_requirements': meets_requirements,
            'minimum_age': min_age,
            'maximum_age': max_age,
            'restrictions': restrictions
        }

    @is_tool()
    def generate_ticket_qr_code(self, booking_id: str, seat_id: str, match_id: str) -> dict:
        import io
        import base64
        from datetime import datetime
        from PIL import Image, ImageDraw, ImageFont

        # Validate input parameters - check for empty strings
        if not booking_id or not isinstance(booking_id, str) or booking_id.strip() == "":
            raise ValueError("booking_id must be a non-empty string")

        if not seat_id or not isinstance(seat_id, str) or seat_id.strip() == "":
            raise ValueError("seat_id must be a non-empty string")

        if not match_id or not isinstance(match_id, str) or match_id.strip() == "":
            raise ValueError("match_id must be a non-empty string")

        # Strip whitespace from inputs
        booking_id = booking_id.strip()
        seat_id = seat_id.strip()
        match_id = match_id.strip()

        # Generate timestamp in format YYYYMMDDHH for uniqueness
        # Using hour precision to match the example format in schema
        timestamp = datetime.now().strftime("%Y%m%d%H")

        # Create QR code data string with format: booking_id-match_id-seat_id-timestamp
        # This format allows easy parsing and validation at venue entrance
        qr_code_data = f"{booking_id}-{match_id}-{seat_id}-{timestamp}"

        # Generate a simple QR-code-like image using PIL
        # Create a white background image
        img_size = 200
        image = Image.new('RGB', (img_size, img_size), color='white')
        draw = ImageDraw.Draw(image)

        # Draw a simple grid pattern to simulate QR code appearance
        # This is a simplified representation for testing purposes
        cell_size = 10
        grid_size = img_size // cell_size

        # Use the QR code data to generate a deterministic pattern
        # Convert characters to numbers for pattern generation
        data_hash = sum(ord(c) for c in qr_code_data)

        # Draw black squares in a pattern based on the data
        for i in range(grid_size):
            for j in range(grid_size):
                # Create a pseudo-random but deterministic pattern
                if ((i * grid_size + j + data_hash) % 3) == 0:
                    x1 = j * cell_size
                    y1 = i * cell_size
                    x2 = x1 + cell_size
                    y2 = y1 + cell_size
                    draw.rectangle([x1, y1, x2, y2], fill='black')

        # Add corner markers (typical QR code feature)
        marker_size = 3 * cell_size
        # Top-left corner
        draw.rectangle([0, 0, marker_size, marker_size], outline='black', width=cell_size)
        # Top-right corner
        draw.rectangle([img_size - marker_size, 0, img_size, marker_size], outline='black', width=cell_size)
        # Bottom-left corner
        draw.rectangle([0, img_size - marker_size, marker_size, img_size], outline='black', width=cell_size)

        # Convert PIL image to base64 encoded string
        buffer = io.BytesIO()
        image.save(buffer, format='PNG')
        buffer.seek(0)

        # Encode image bytes to base64 string
        qr_code_image_base64 = base64.b64encode(buffer.getvalue()).decode('utf-8')

        # Return both the QR code data string and the base64 encoded image
        return {
            'qr_code_data': qr_code_data,
            'qr_code_image': qr_code_image_base64
        }

    @is_tool()
    def search_available_matches(
        self,
        start_date: str = None,
        end_date: str = None,
        team_name: str = None,
        competition_name: str = None,
        venue_name: str = None
    ) -> dict:
        """
        Search for available football matches based on filters such as date range, team, competition, and venue.

        Args:
            start_date: Start date for match search in yyyy-mm-dd format (optional)
            end_date: End date for match search in yyyy-mm-dd format (optional)
            team_name: Name of the team to filter matches (optional, uses fuzzy matching)
            competition_name: Name of the competition to filter matches (optional, uses fuzzy matching)
            venue_name: Name of the venue to filter matches (optional, uses fuzzy matching)

        Returns:
            dict: Dictionary containing list of matches matching the search criteria

        Raises:
            ValueError: If date format is invalid or date range is invalid
        """
        from datetime import datetime
        from thefuzz import fuzz

        # Get database instance
        db = self.db

        # Get match table data
        match_table = getattr(db, 'match', None)
        if match_table is None:
            return {'matches': []}

        # Parse and validate date parameters
        start_datetime = None
        end_datetime = None

        if start_date:
            try:
                start_datetime = datetime.strptime(start_date.strip(), "%Y-%m-%d")
            except (ValueError, AttributeError):
                raise ValueError(f"Invalid start_date format. Expected yyyy-mm-dd, got: {start_date}")

        if end_date:
            try:
                end_datetime = datetime.strptime(end_date.strip(), "%Y-%m-%d")
                end_datetime = end_datetime.replace(hour=23, minute=59, second=59)
            except (ValueError, AttributeError):
                raise ValueError(f"Invalid end_date format. Expected yyyy-mm-dd, got: {end_date}")

        # Validate date range
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Collect and filter matches
        filtered_matches = []

        for match_id, match in match_table.items():
            # Skip postponed matches
            if match.is_postponed:
                continue

            # Filter by date range
            if start_datetime and match.match_date < start_datetime:
                continue
            if end_datetime and match.match_date > end_datetime:
                continue

            # Filter by team name using fuzzy matching (threshold: 80)
            if team_name:
                team_name_lower = team_name.strip().lower()
                home_score = fuzz.ratio(team_name_lower, match.home_team.lower())
                away_score = fuzz.ratio(team_name_lower, match.away_team.lower())
                if max(home_score, away_score) < 80:
                    continue

            # Filter by competition name using fuzzy matching (threshold: 80)
            if competition_name:
                competition_score = fuzz.ratio(competition_name.strip().lower(), match.competition.lower())
                if competition_score < 80:
                    continue

            # Filter by venue name using fuzzy matching (threshold: 80)
            if venue_name:
                venue_score = fuzz.ratio(venue_name.strip().lower(), match.venue_name.lower())
                if venue_score < 80:
                    continue

            # Match passes all filters
            filtered_matches.append(match)

        # Format results
        result_matches = []
        for match in filtered_matches:
            match_date_str = match.match_date.strftime("%Y-%m-%d %H:%M:%S")

            result_matches.append({
                'match_id': match.match_id,
                'home_team': match.home_team,
                'away_team': match.away_team,
                'match_date': match_date_str,
                'venue': match.venue_name,
                'competition': match.competition
            })

        return {'matches': result_matches}

    @is_tool()
    def calculate_group_booking_discount(self, base_price: float, ticket_quantity: int, group_type: str = None) -> dict:
        """
        Calculate discount for group bookings based on quantity and group type.

        This method computes discounts for group ticket purchases using a tiered system
        based on the number of tickets and optional group type benefits.

        Args:
            base_price: Base price per ticket (must be positive)
            ticket_quantity: Number of tickets in group booking (must be positive integer)
            group_type: Type of group (corporate, school, family, etc.) - optional

        Returns:
            Dictionary containing:
            - discount_percentage: Applicable discount percentage
            - discount_per_ticket: Discount amount per ticket
            - discounted_price_per_ticket: Final price per ticket after discount
            - total_savings: Total amount saved
            - total_price: Total price for all tickets

        Raises:
            ValueError: If base_price or ticket_quantity are invalid
        """

        # Validate input parameters
        if not isinstance(base_price, (int, float)) or base_price <= 0:
            raise ValueError("base_price must be a positive number")

        if not isinstance(ticket_quantity, int) or ticket_quantity <= 0:
            raise ValueError("ticket_quantity must be a positive integer")

        # Initialize discount percentage
        discount_percentage = 0.0

        # Apply quantity-based tiered discounts
        # Larger groups receive higher discounts
        if ticket_quantity >= 50:
            discount_percentage = 25.0  # 25% discount for 50+ tickets
        elif ticket_quantity >= 30:
            discount_percentage = 20.0  # 20% discount for 30-49 tickets
        elif ticket_quantity >= 20:
            discount_percentage = 15.0  # 15% discount for 20-29 tickets
        elif ticket_quantity >= 10:
            discount_percentage = 10.0  # 10% discount for 10-19 tickets
        elif ticket_quantity >= 5:
            discount_percentage = 5.0   # 5% discount for 5-9 tickets

        # Apply additional group type discounts if specified
        # These are additive bonuses on top of quantity discounts
        if group_type:
            group_type_lower = group_type.lower().strip()

            # School groups get an additional 5% discount to encourage educational attendance
            if group_type_lower == "school":
                discount_percentage += 5.0

            # Corporate groups get an additional 3% discount for business bookings
            elif group_type_lower == "corporate":
                discount_percentage += 3.0

            # Family groups get an additional 2% discount to support family attendance
            elif group_type_lower == "family":
                discount_percentage += 2.0

            # Charity/non-profit groups get an additional 7% discount
            elif group_type_lower in ["charity", "non-profit", "nonprofit"]:
                discount_percentage += 7.0

            # Youth groups (scouts, clubs, etc.) get an additional 4% discount
            elif group_type_lower in ["youth", "scout", "club"]:
                discount_percentage += 4.0

        # Cap maximum discount at 35% to maintain revenue
        discount_percentage = min(discount_percentage, 35.0)

        # Calculate discount amounts
        # Convert percentage to decimal for calculations
        discount_decimal = discount_percentage / 100.0

        # Calculate discount per individual ticket
        discount_per_ticket = base_price * discount_decimal

        # Calculate final price per ticket after discount
        discounted_price_per_ticket = base_price - discount_per_ticket

        # Calculate total savings across all tickets
        total_savings = discount_per_ticket * ticket_quantity

        # Calculate total price for the entire group booking
        total_price = discounted_price_per_ticket * ticket_quantity

        # Round all monetary values to 2 decimal places for currency precision
        discount_per_ticket = round(discount_per_ticket, 2)
        discounted_price_per_ticket = round(discounted_price_per_ticket, 2)
        total_savings = round(total_savings, 2)
        total_price = round(total_price, 2)
        discount_percentage = round(discount_percentage, 2)

        # Return comprehensive discount calculation results
        return {
            "discount_percentage": discount_percentage,
            "discount_per_ticket": discount_per_ticket,
            "discounted_price_per_ticket": discounted_price_per_ticket,
            "total_savings": total_savings,
            "total_price": total_price
        }

    @is_tool()
    def get_booking_details(self, booking_id: str) -> dict:
        """
        Retrieve complete details of a specific booking

        Args:
            booking_id: Unique identifier for the booking

        Returns:
            dict: Complete booking information including booking details and associated seats

        Raises:
            KeyError: If booking_id does not exist in the system
        """
        # Access the database
        db = self.db

        # Validate input parameter
        if not booking_id or not isinstance(booking_id, str):
            raise ValueError("booking_id must be a non-empty string")

        # Retrieve booking table data
        booking_table = getattr(db, "booking", None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Check if booking exists
        if booking_id not in booking_table:
            raise KeyError(f"Booking with ID '{booking_id}' not found")

        # Get the booking record
        booking = booking_table[booking_id]

        # Retrieve booking_seat table to get associated seats
        booking_seat_table = getattr(db, "booking_seat", None)
        if booking_seat_table is None:
            raise RuntimeError("BookingSeat table not found in database")

        # Retrieve seat table to get seat details
        seat_table = getattr(db, "seat", None)
        if seat_table is None:
            raise RuntimeError("Seat table not found in database")

        # Find all booking_seat records for this booking
        seats_list = []
        for booking_seat_id, booking_seat in booking_seat_table.items():
            if booking_seat.booking_id == booking_id:
                # Get the corresponding seat details
                seat_id = booking_seat.seat_id
                if seat_id in seat_table:
                    seat = seat_table[seat_id]
                    # Create seat information dictionary
                    seat_info = {
                        "seat_id": seat.seat_id,
                        "row": seat.row,
                        "seat_number": seat.seat_number
                    }
                    seats_list.append(seat_info)

        # Format booking_time to string in yyyy-mm-dd HH:MM:SS format
        booking_time_str = booking.booking_time.strftime("%Y-%m-%d %H:%M:%S")

        # Construct the return dictionary with all required fields
        result = {
            "booking_id": booking.booking_id,
            "booking_reference": booking.booking_reference,
            "match_id": booking.match_id,
            "customer_id": booking.customer_id,
            "seats": seats_list,
            "total_price": float(booking.total_price),
            "booking_time": booking_time_str,
            "status": booking.status
        }

        return result

    @is_tool()
    def check_match_postponement(self, match_id: str):
        """
        Check if a match has been postponed or rescheduled

        Args:
            match_id: Unique identifier for the match

        Returns:
            dict: Dictionary containing postponement information including:
                - is_postponed: Whether the match has been postponed
                - original_date: Original match date in yyyy-mm-dd HH:MM:SS format
                - new_date: Rescheduled date if applicable in yyyy-mm-dd HH:MM:SS format
                - postponement_reason: Reason for postponement if applicable
                - ticket_validity: Whether existing tickets remain valid

        Raises:
            KeyError: If match_id does not exist in the database
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve match table from database
        match_table = getattr(db, 'match', None)

        # Check if match table exists
        if match_table is None:
            raise KeyError(f"Match table not found in database")

        # Check if match_id exists in the match table
        if match_id not in match_table:
            raise KeyError(f"Match with ID '{match_id}' not found")

        # Retrieve the match record
        match = match_table[match_id]

        # Extract match information
        is_postponed = match.is_postponed

        # Format original_date to string in yyyy-mm-dd HH:MM:SS format
        # If the match is postponed, original_date contains the original date
        # If not postponed, match_date is the original/current date
        if is_postponed and match.original_date is not None:
            original_date_str = match.original_date.strftime("%Y-%m-%d %H:%M:%S")
        else:
            original_date_str = match.match_date.strftime("%Y-%m-%d %H:%M:%S")

        # Format new_date to string in yyyy-mm-dd HH:MM:SS format
        # If postponed, match_date becomes the new rescheduled date
        # If not postponed, new_date is None
        if is_postponed:
            new_date_str = match.match_date.strftime("%Y-%m-%d %H:%M:%S")
        else:
            new_date_str = None

        # Get postponement reason if applicable
        postponement_reason = match.postponement_reason if is_postponed else None

        # Determine ticket validity
        # For postponed matches, tickets typically remain valid for the rescheduled date
        # This is a standard practice in football ticket booking systems
        ticket_validity = True

        # Construct and return the result dictionary
        return {
            'is_postponed': is_postponed,
            'original_date': original_date_str,
            'new_date': new_date_str,
            'postponement_reason': postponement_reason,
            'ticket_validity': ticket_validity
        }

    @is_tool()
    def validate_ticket_qr_code(self, qr_code_data: str, scan_time: str = None) -> dict:
        """
        Validate a ticket QR code at venue entrance to verify authenticity and usage status.

        This method checks if the provided QR code is valid by:
        1. Verifying the QR code exists in the booking_seat table
        2. Checking if the associated booking is confirmed (not cancelled)
        3. Determining if the ticket has already been used for entry
        4. Retrieving seat information (limited to seat_id only due to table access constraints)

        Args:
            qr_code_data: QR code data string to validate
            scan_time: Time when QR code was scanned in yyyy-mm-dd HH:MM:SS format (optional)

        Returns:
            dict containing:
                - is_valid: Whether the QR code is valid
                - booking_id: Associated booking identifier
                - seat_info: Seat information dict with seat_id (row and seat_number unavailable)
                - already_used: Whether the ticket has already been used
                - validation_message: Validation message

        Raises:
            ValueError: If qr_code_data is empty or invalid format, or scan_time format is invalid
        """
        from datetime import datetime

        # Input validation
        if not qr_code_data or not isinstance(qr_code_data, str):
            raise ValueError("QR code data must be a non-empty string")

        qr_code_data = qr_code_data.strip()
        if not qr_code_data:
            raise ValueError("QR code data cannot be empty")

        # Parse scan_time if provided
        if scan_time:
            try:
                scan_datetime = datetime.strptime(scan_time.strip(), "%Y-%m-%d %H:%M:%S")
            except ValueError:
                raise ValueError("scan_time must be in format yyyy-mm-dd HH:MM:SS")
        else:
            scan_datetime = datetime.now()

        # Access database
        db = self.db

        # Get booking_seat and booking tables
        booking_seat_table = getattr(db, "booking_seat", None)
        booking_table = getattr(db, "booking", None)

        if booking_seat_table is None or booking_table is None:
            return {
                "is_valid": False,
                "booking_id": "",
                "seat_info": {},
                "already_used": False,
                "validation_message": "Database tables not available"
            }

        # Search for the booking_seat record with matching qr_code_data
        # Using exact match since qr_code_data is a unique identifier (not natural language)
        matching_booking_seat = None
        for booking_seat_id, booking_seat in booking_seat_table.items():
            if booking_seat.qr_code_data and booking_seat.qr_code_data.strip() == qr_code_data:
                matching_booking_seat = booking_seat
                break

        # If no matching QR code found
        if matching_booking_seat is None:
            return {
                "is_valid": False,
                "booking_id": "",
                "seat_info": {},
                "already_used": False,
                "validation_message": "Invalid QR code - not found in system"
            }

        # Extract booking_id and seat_id from the booking_seat record
        booking_id = matching_booking_seat.booking_id
        seat_id = matching_booking_seat.seat_id

        # Check if ticket has already been used
        already_used = matching_booking_seat.is_used

        # Get the associated booking to check its status
        booking = booking_table.get(booking_id)

        if booking is None:
            return {
                "is_valid": False,
                "booking_id": booking_id,
                "seat_info": {"seat_id": seat_id, "row": "", "seat_number": ""},
                "already_used": already_used,
                "validation_message": "Associated booking not found"
            }

        # Check if booking is confirmed (not cancelled)
        if booking.status != "confirmed":
            return {
                "is_valid": False,
                "booking_id": booking_id,
                "seat_info": {"seat_id": seat_id, "row": "", "seat_number": ""},
                "already_used": already_used,
                "validation_message": f"Booking status is {booking.status} - entry not permitted"
            }

        # If ticket has already been used
        if already_used:
            return {
                "is_valid": False,
                "booking_id": booking_id,
                "seat_info": {"seat_id": seat_id, "row": "", "seat_number": ""},
                "already_used": True,
                "validation_message": "Ticket has already been used for entry"
            }

        # QR code is valid - booking is confirmed and ticket not yet used
        # Note: row and seat_number cannot be retrieved as seat table is not in related_databases
        seat_info = {
            "seat_id": seat_id,
            "row": "",
            "seat_number": ""
        }

        return {
            "is_valid": True,
            "booking_id": booking_id,
            "seat_info": seat_info,
            "already_used": False,
            "validation_message": "Valid ticket - Entry permitted"
        }

    @is_tool()
    def check_resale_eligibility(self, booking_id: str, seat_id: str, resale_request_time: str = None) -> dict:
        """
        Check if a ticket is eligible for resale on the official resale platform.

        This method validates whether a specific ticket can be listed for resale based on:
        1. Booking and seat existence
        2. Booking status (must be confirmed)
        3. Ticket usage status (must not be used)
        4. Time constraints (match must be more than 48 hours away)

        Args:
            booking_id: Unique identifier for the booking
            seat_id: Unique identifier for the seat
            resale_request_time: Time when resale is requested in yyyy-mm-dd HH:MM:SS format
                                If not provided, uses current time

        Returns:
            Dictionary containing:
            - is_eligible: Boolean indicating if ticket is eligible for resale
            - reasons: List of reasons for eligibility or ineligibility
            - resale_deadline: Deadline for listing ticket (48 hours before match)

        Raises:
            KeyError: If booking_id or seat_id not found in database
        """
        from datetime import datetime, timedelta

        # Get database instance
        db = self.db

        # Validate booking exists
        booking_table = getattr(db, 'booking', None)
        if booking_table is None or booking_id not in booking_table:
            raise KeyError(f"Booking with ID '{booking_id}' not found")

        booking = booking_table[booking_id]

        # Validate seat exists in booking_seat table
        booking_seat_table = getattr(db, 'booking_seat', None)
        if booking_seat_table is None:
            raise KeyError(f"Booking seat table not found")

        # Find the booking_seat record that matches both booking_id and seat_id
        booking_seat = None
        for bs_id, bs_record in booking_seat_table.items():
            if bs_record.booking_id == booking_id and bs_record.seat_id == seat_id:
                booking_seat = bs_record
                break

        if booking_seat is None:
            raise KeyError(f"Seat with ID '{seat_id}' not found in booking '{booking_id}'")

        # Get match information
        match_table = getattr(db, 'match', None)
        if match_table is None or booking.match_id not in match_table:
            raise KeyError(f"Match with ID '{booking.match_id}' not found")

        match = match_table[booking.match_id]

        # Parse resale request time or use current time
        if resale_request_time:
            request_time = datetime.strptime(resale_request_time.strip(), "%Y-%m-%d %H:%M:%S")
        else:
            request_time = datetime.now()

        # Initialize result
        is_eligible = True
        reasons = []

        # Check 1: Booking status must be confirmed
        if booking.status != 'confirmed':
            is_eligible = False
            reasons.append(f"Booking status is '{booking.status}', must be 'confirmed'")
        else:
            reasons.append("Booking is confirmed")

        # Check 2: Ticket must not have been used
        if booking_seat.is_used:
            is_eligible = False
            reasons.append("Ticket has already been used for entry")
        else:
            reasons.append("Ticket has not been used")

        # Check 3: Match must be more than 48 hours away
        # Calculate time difference between match and resale request
        time_until_match = match.match_date - request_time
        hours_until_match = time_until_match.total_seconds() / 3600

        if hours_until_match <= 48:
            is_eligible = False
            reasons.append(f"Match is only {hours_until_match:.1f} hours away, must be more than 48 hours")
        else:
            reasons.append("Match is more than 48 hours away")

        # Calculate resale deadline (48 hours before match)
        resale_deadline = match.match_date - timedelta(hours=48)
        resale_deadline_str = resale_deadline.strftime("%Y-%m-%d %H:%M:%S")

        # Return result
        return {
            'is_eligible': is_eligible,
            'reasons': reasons,
            'resale_deadline': resale_deadline_str
        }

    @is_tool()
    def process_postponement_refund(self, booking_id: str, match_id: str, refund_preference: str):
        """
        Process refund for bookings when a match is postponed and customer opts for refund.

        This method handles different refund preferences: full_refund, credit, or transfer.
        It validates the booking and match status, processes the refund according to the preference,
        and updates the relevant database records.

        Args:
            booking_id: Unique identifier for the booking
            match_id: Unique identifier for the postponed match
            refund_preference: Customer refund preference (full_refund, credit, transfer)

        Returns:
            dict: Contains refund_processed (bool), refund_amount (float), 
                  refund_method (str), processing_time (str in yyyy-mm-dd HH:MM:SS format)

        Raises:
            RuntimeError: If booking not found, match not found, match not postponed,
                         booking already cancelled, or invalid refund preference
        """
        from datetime import datetime

        # Access database
        db = self.db

        # Validate input parameters
        if not booking_id or not isinstance(booking_id, str):
            raise RuntimeError("Invalid booking_id: must be a non-empty string")

        if not match_id or not isinstance(match_id, str):
            raise RuntimeError("Invalid match_id: must be a non-empty string")

        if not refund_preference or not isinstance(refund_preference, str):
            raise RuntimeError("Invalid refund_preference: must be a non-empty string")

        # Valid refund preferences
        valid_preferences = ['full_refund', 'credit', 'transfer']
        if refund_preference not in valid_preferences:
            raise RuntimeError(f"Invalid refund_preference: must be one of {valid_preferences}")

        # Get booking table
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Get match table
        match_table = getattr(db, 'match', None)
        if match_table is None:
            raise RuntimeError("Match table not found in database")

        # Get payment table
        payment_table = getattr(db, 'payment', None)
        if payment_table is None:
            raise RuntimeError("Payment table not found in database")

        # Retrieve booking record
        booking = booking_table.get(booking_id)
        if booking is None:
            raise RuntimeError(f"Booking with ID '{booking_id}' not found")

        # Verify booking is for the specified match (exact match required for IDs)
        if booking.match_id != match_id:
            raise RuntimeError(f"Booking '{booking_id}' is not associated with match '{match_id}'")

        # Check if booking is already cancelled
        if booking.status == 'cancelled':
            raise RuntimeError(f"Booking '{booking_id}' is already cancelled")

        # Retrieve match record
        match = match_table.get(match_id)
        if match is None:
            raise RuntimeError(f"Match with ID '{match_id}' not found")

        # Verify match is postponed
        if not match.is_postponed:
            raise RuntimeError(f"Match '{match_id}' is not postponed")

        # Retrieve payment record
        payment = payment_table.get(booking.payment_id)
        if payment is None:
            raise RuntimeError(f"Payment record '{booking.payment_id}' not found for booking '{booking_id}'")

        # Calculate refund amount based on preference
        refund_amount = 0.0
        refund_method = ""

        if refund_preference == 'full_refund':
            # Full refund: return the total booking price
            refund_amount = float(booking.total_price)
            refund_method = "original_payment_method"

            # Update payment status to refunded
            payment.payment_status = 'refunded'

        elif refund_preference == 'credit':
            # Credit: provide credit for future use (same amount)
            refund_amount = float(booking.total_price)
            refund_method = "account_credit"

            # Update payment status to refunded (credit is still a form of refund)
            payment.payment_status = 'refunded'

        elif refund_preference == 'transfer':
            # Transfer: transfer booking to rescheduled match (no monetary refund)
            refund_amount = 0.0
            refund_method = "booking_transfer"

            # Payment status remains as is, since no actual refund occurs
            # Booking will be transferred to the new match date

        # Get current processing time and format as yyyy-mm-dd HH:MM:SS
        processing_time = datetime.now()
        processing_time_str = processing_time.strftime("%Y-%m-%d %H:%M:%S")

        # Update booking record
        booking.status = 'cancelled'
        booking.cancellation_time = processing_time
        booking.cancellation_reason = f"Match postponed - {refund_preference} processed"
        booking.refund_amount = refund_amount

        # Update database tables
        booking_table[booking_id] = booking
        payment_table[booking.payment_id] = payment

        # Return refund processing result
        return {
            'refund_processed': True,
            'refund_amount': refund_amount,
            'refund_method': refund_method,
            'processing_time': processing_time_str
        }

    @is_tool()
    def create_booking(self, reservation_id: str, customer_id: str, payment_id: str, customer_email: str, customer_phone: str = None):
        """
        Create a new ticket booking for reserved seats.

        This method converts a reservation into a confirmed booking after payment is verified.
        It performs the following steps:
        1. Validates that the reservation exists and is active
        2. Verifies that the payment exists and is completed
        3. Retrieves all seats associated with the reservation
        4. Creates a new booking record with generated booking_id and reference
        5. Creates booking_seat records for each reserved seat
        6. Updates the reservation status to 'converted'
        7. Links the payment to the booking

        Args:
            reservation_id: Unique identifier for the reservation to convert
            customer_id: Unique identifier for the customer making the booking
            payment_id: Unique identifier for the completed payment transaction
            customer_email: Customer's contact email address
            customer_phone: Customer's contact phone number (optional)

        Returns:
            dict: Booking details containing:
                - booking_id: Generated unique booking identifier
                - booking_reference: Human-readable booking reference code
                - booking_time: Timestamp when booking was created (yyyy-mm-dd HH:MM:SS format)
                - status: Booking status (always 'confirmed' for new bookings)

        Raises:
            RuntimeError: If reservation doesn't exist, is not active, payment is invalid,
                         or any other validation/creation error occurs
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Get reservation table
        reservation_table = getattr(db, 'reservation', None)
        if reservation_table is None:
            raise RuntimeError("Reservation table not found in database")

        # Validate reservation exists
        reservation = reservation_table.get(reservation_id)
        if reservation is None:
            raise RuntimeError(f"Reservation with ID '{reservation_id}' does not exist")

        # Validate reservation is active (can be converted to booking)
        if reservation.status != 'active':
            raise RuntimeError(f"Reservation '{reservation_id}' is not active (status: {reservation.status})")

        # Validate customer_id matches reservation
        if reservation.customer_id != customer_id:
            raise RuntimeError(f"Customer ID mismatch: reservation belongs to '{reservation.customer_id}', not '{customer_id}'")

        # Get payment table and validate payment
        payment_table = getattr(db, 'payment', None)
        if payment_table is None:
            raise RuntimeError("Payment table not found in database")

        payment = payment_table.get(payment_id)
        if payment is None:
            raise RuntimeError(f"Payment with ID '{payment_id}' does not exist")

        # Validate payment is completed
        if payment.payment_status != 'completed':
            raise RuntimeError(f"Payment '{payment_id}' is not completed (status: {payment.payment_status})")

        # Validate payment customer matches
        if payment.customer_id != customer_id:
            raise RuntimeError(f"Payment customer ID mismatch: payment belongs to '{payment.customer_id}', not '{customer_id}'")

        # Get reservation_seat table to find all seats for this reservation
        # Note: reservation_seat is in all_database_schema_list but not in related_databases
        # We need to access it from db to get the seats
        reservation_seat_table = getattr(db, 'reservation_seat', None)
        if reservation_seat_table is None:
            raise RuntimeError("Reservation_seat table not found in database")

        # Find all seats associated with this reservation
        reserved_seats = []
        for res_seat_id, res_seat in reservation_seat_table.items():
            if res_seat.reservation_id == reservation_id:
                reserved_seats.append(res_seat.seat_id)

        if not reserved_seats:
            raise RuntimeError(f"No seats found for reservation '{reservation_id}'")

        # Generate booking_id
        booking_id = "B" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:9]

        # Generate booking_reference (format: MU-YYYY-XXXXXX)
        current_time = datetime.now()
        year = current_time.year
        random_code = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:6].upper()
        booking_reference = f"MU-{year}-{random_code}"

        # Create booking record
        booking_table = getattr(db, 'booking', None)
        if booking_table is None:
            raise RuntimeError("Booking table not found in database")

        # Create new booking object
        new_booking = Booking(
            booking_id=booking_id,
            booking_reference=booking_reference,
            match_id=reservation.match_id,
            customer_id=customer_id,
            reservation_id=reservation_id,
            payment_id=payment_id,
            email=customer_email,
            phone=customer_phone,
            total_price=payment.amount,
            booking_time=current_time,
            status='confirmed',
            cancellation_time=None,
            cancellation_reason=None,
            refund_amount=None
        )

        # Add booking to database
        booking_table[booking_id] = new_booking
        setattr(db, 'booking', booking_table)

        # Create booking_seat records for each reserved seat
        booking_seat_table = getattr(db, 'booking_seat', None)
        if booking_seat_table is None:
            raise RuntimeError("Booking_seat table not found in database")

        for seat_id in reserved_seats:
            # Generate booking_seat_id
            booking_seat_id = "BS" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8]

            # Generate QR code data (simple format: booking_id:seat_id)
            qr_code_data = f"{booking_id}:{seat_id}:{booking_reference}"

            # Create booking_seat object
            new_booking_seat = BookingSeat(
                booking_seat_id=booking_seat_id,
                booking_id=booking_id,
                seat_id=seat_id,
                is_used=False,
                entry_time=None,
                entry_gate=None,
                qr_code_data=qr_code_data
            )

            # Add to database
            booking_seat_table[booking_seat_id] = new_booking_seat

        setattr(db, 'booking_seat', booking_seat_table)

        # Update reservation status to 'converted'
        reservation.status = 'converted'
        reservation_table[reservation_id] = reservation
        setattr(db, 'reservation', reservation_table)

        # Update payment to link it to the booking
        payment.booking_id = booking_id
        payment_table[payment_id] = payment
        setattr(db, 'payment', payment_table)

        # Update seat status to 'sold' in seat table
        seat_table = getattr(db, 'seat', None)
        if seat_table is not None:
            for seat_id in reserved_seats:
                seat = seat_table.get(seat_id)
                if seat is not None:
                    seat.status = 'sold'
                    seat_table[seat_id] = seat
            setattr(db, 'seat', seat_table)

        # Return booking details
        return {
            'booking_id': booking_id,
            'booking_reference': booking_reference,
            'booking_time': current_time.strftime('%Y-%m-%d %H:%M:%S'),
            'status': 'confirmed'
        }
