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

"""Tools for express_logistics."""

class ExpressLogisticsTools(ToolKitBase):
    """All tools for express_logistics."""
    
    db: ExpressLogisticsDB
    
    def __init__(self, db: ExpressLogisticsDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def calculate_package_insurance_premium(
        self,
        declared_value: float,
        service_type: Literal["standard", "express", "same_day", "next_day"],
        insurance_rate: float
    ) -> dict:
        """
        Calculate insurance premium based on declared package value and service type.

        This method computes the insurance premium by applying the insurance rate to the
        declared value of the package. The premium amount is calculated as a percentage
        of the declared value, and the coverage amount equals the declared value.

        Args:
            declared_value: Declared value of package contents in currency units (must be non-negative)
            service_type: Type of courier service (must be one of: standard, express, same_day, next_day)
            insurance_rate: Insurance rate as percentage of declared value (must be non-negative)

        Returns:
            dict: A dictionary containing:
                - premium_amount: Calculated insurance premium in currency units
                - coverage_amount: Maximum coverage amount (equals declared_value)

        Raises:
            ValueError: If declared_value is negative, insurance_rate is negative,
                       or if service_type is not in the allowed enum values
        """
        # Validate declared_value is non-negative
        if declared_value < 0:
            raise ValueError(
                f"Declared value must be non-negative, got: {declared_value}"
            )

        # Validate insurance_rate is non-negative
        if insurance_rate < 0:
            raise ValueError(
                f"Insurance rate must be non-negative, got: {insurance_rate}"
            )

        # Validate service_type is within allowed enum values
        # Note: The Literal type annotation already provides type checking at runtime
        # for Python 3.8+, but we add explicit validation for robustness
        allowed_service_types = ["standard", "express", "same_day", "next_day"]
        if service_type not in allowed_service_types:
            raise ValueError(
                f"Invalid service_type '{service_type}'. "
                f"Must be one of: {', '.join(allowed_service_types)}"
            )

        # Calculate the insurance premium
        # Premium is calculated as: declared_value * (insurance_rate / 100)
        # The insurance_rate is expected to be a percentage value
        premium_amount = declared_value * (insurance_rate / 100.0)

        # The coverage amount equals the declared value
        # This represents the maximum amount that can be claimed in case of loss or damage
        coverage_amount = declared_value

        # Return the calculated results as a dictionary
        return {
            "premium_amount": premium_amount,
            "coverage_amount": coverage_amount
        }

    @is_tool()
    def calculate_multi_package_discount(
        self,
        package_count: int,
        individual_costs: list,
        discount_tier: Literal["tier1", "tier2", "tier3", "tier4"]
    ) -> dict:
        """
        Calculate discount for shipping multiple packages in a single order.

        This method computes the total cost before and after applying a discount based on
        the discount tier. The discount percentage is determined by the tier level.

        Args:
            package_count: Number of packages in the order
            individual_costs: List of individual package shipping costs
            discount_tier: Discount tier ("tier1", "tier2", "tier3", or "tier4")

        Returns:
            Dictionary containing:
            - total_before_discount: Total cost before discount
            - discount_amount: Total discount amount
            - total_after_discount: Total cost after discount
            - discount_percentage: Applied discount percentage

        Raises:
            ValueError: If inputs are invalid or inconsistent
        """
        # Validate package_count
        if not isinstance(package_count, int):
            raise ValueError("package_count must be an integer")
        if package_count <= 0:
            raise ValueError("package_count must be greater than 0")

        # Validate individual_costs
        if not isinstance(individual_costs, list):
            raise ValueError("individual_costs must be a list")
        if len(individual_costs) == 0:
            raise ValueError("individual_costs cannot be empty")

        # Validate that all costs are numeric and non-negative
        for i, cost in enumerate(individual_costs):
            if not isinstance(cost, (int, float)):
                raise ValueError(f"Cost at index {i} must be a number")
            if cost < 0:
                raise ValueError(f"Cost at index {i} cannot be negative")

        # Validate consistency between package_count and individual_costs length
        if len(individual_costs) != package_count:
            raise ValueError(
                f"Length of individual_costs ({len(individual_costs)}) must match package_count ({package_count})"
            )

        # Validate discount_tier (already constrained by Literal type, but add safety check)
        valid_tiers = ["tier1", "tier2", "tier3", "tier4"]
        if discount_tier not in valid_tiers:
            raise ValueError(f"discount_tier must be one of {valid_tiers}")

        # Define discount percentages for each tier
        # tier1: 5% discount (2-3 packages)
        # tier2: 10% discount (4-5 packages)
        # tier3: 15% discount (6-9 packages)
        # tier4: 20% discount (10+ packages)
        tier_discount_map = {
            "tier1": 5.0,
            "tier2": 10.0,
            "tier3": 15.0,
            "tier4": 20.0
        }

        # Get the discount percentage for the specified tier
        discount_percentage = tier_discount_map[discount_tier]

        # Calculate total cost before discount
        total_before_discount = sum(individual_costs)

        # Calculate discount amount
        discount_amount = total_before_discount * (discount_percentage / 100.0)

        # Calculate total cost after discount
        total_after_discount = total_before_discount - discount_amount

        # Round all monetary values to 2 decimal places for precision
        total_before_discount = round(total_before_discount, 2)
        discount_amount = round(discount_amount, 2)
        total_after_discount = round(total_after_discount, 2)

        # Return the result dictionary
        return {
            "total_before_discount": total_before_discount,
            "discount_amount": discount_amount,
            "total_after_discount": total_after_discount,
            "discount_percentage": discount_percentage
        }

    @is_tool()
    def calculate_volumetric_weight(self, length: float, width: float, height: float, divisor: float):
        """
        Calculate the volumetric weight of a package based on its dimensions.

        Volumetric weight is calculated using the formula:
        (length × width × height) / divisor

        This is commonly used in shipping to determine the billable weight when
        the package is large but light, as carriers charge based on the greater
        of actual weight or volumetric weight.

        Args:
            length: Package length in centimeters (must be positive)
            width: Package width in centimeters (must be positive)
            height: Package height in centimeters (must be positive)
            divisor: Volumetric divisor used for calculation (must be positive)
                    Default is typically 5000 for international shipping

        Returns:
            dict: Contains the calculated volumetric weight in kilograms
                  Format: {"volumetric_weight": float}

        Raises:
            ValueError: If any dimension or divisor is not positive
            TypeError: If any parameter is not a numeric type
        """
        # Validate input types - ensure all parameters are numeric
        if not isinstance(length, (int, float)):
            raise TypeError(f"Length must be a number, got {type(length).__name__}")
        if not isinstance(width, (int, float)):
            raise TypeError(f"Width must be a number, got {type(width).__name__}")
        if not isinstance(height, (int, float)):
            raise TypeError(f"Height must be a number, got {type(height).__name__}")
        if not isinstance(divisor, (int, float)):
            raise TypeError(f"Divisor must be a number, got {type(divisor).__name__}")

        # Validate that all dimensions and divisor are positive values
        if length <= 0:
            raise ValueError(f"Length must be positive, got {length}")
        if width <= 0:
            raise ValueError(f"Width must be positive, got {width}")
        if height <= 0:
            raise ValueError(f"Height must be positive, got {height}")
        if divisor <= 0:
            raise ValueError(f"Divisor must be positive, got {divisor}")

        # Calculate volumetric weight using the standard formula
        # Volume (cubic cm) divided by the volumetric divisor gives weight in kg
        volumetric_weight = (length * width * height) / divisor

        # Return the result as a dictionary with the calculated volumetric weight
        # Round to 2 decimal places for practical shipping weight representation
        return {
            "volumetric_weight": round(volumetric_weight, 2)
        }

    @is_tool()
    def estimate_sorting_center_processing_time(
        self,
        current_package_count: int,
        center_capacity: int,
        package_priority: Literal["low", "medium", "high", "urgent"]
    ) -> dict:
        """
        Estimate processing time at sorting center based on package volume and center capacity.

        This method calculates the estimated processing time and queue position for a package
        based on the current workload at the sorting center and the package's priority level.

        Args:
            current_package_count: Current number of packages at the sorting center
            center_capacity: Maximum processing capacity per hour
            package_priority: Priority level of the package (low/medium/high/urgent)

        Returns:
            dict: Contains estimated_processing_hours and queue_position

        Raises:
            ValueError: If input parameters are invalid
        """
        # Validate input parameters
        if current_package_count < 0:
            raise ValueError("current_package_count must be non-negative")

        if center_capacity <= 0:
            raise ValueError("center_capacity must be positive")

        # Validate package_priority is within allowed enum values
        allowed_priorities = ["low", "medium", "high", "urgent"]
        if package_priority not in allowed_priorities:
            raise ValueError(f"package_priority must be one of {allowed_priorities}, got '{package_priority}'")

        # Define priority weights for queue positioning
        # Higher priority packages get processed earlier (lower weight = earlier processing)
        priority_weights = {
            "urgent": 0.1,    # Urgent packages processed first
            "high": 0.3,      # High priority packages processed early
            "medium": 0.6,    # Medium priority packages processed normally
            "low": 1.0        # Low priority packages processed last
        }

        # Define priority multipliers for processing time estimation
        # Higher priority packages may get faster processing
        priority_time_multipliers = {
            "urgent": 0.5,    # Urgent packages get expedited processing
            "high": 0.7,      # High priority packages get faster processing
            "medium": 1.0,    # Medium priority packages get normal processing
            "low": 1.2        # Low priority packages may experience slight delays
        }

        # Get the weight and time multiplier for the given priority
        priority_weight = priority_weights[package_priority]
        time_multiplier = priority_time_multipliers[package_priority]

        # Calculate base processing time in hours
        # This represents how long it would take to process all current packages at full capacity
        base_processing_hours = current_package_count / center_capacity

        # Apply priority multiplier to get estimated processing time
        # Higher priority packages are estimated to be processed faster
        estimated_processing_hours = base_processing_hours * time_multiplier

        # Calculate queue position based on priority
        # The queue position represents where this package would be in the processing queue
        # considering its priority relative to other packages

        # Estimate how many packages are ahead based on priority weight
        # Lower priority weight means fewer packages ahead (earlier in queue)
        packages_ahead = int(current_package_count * priority_weight)

        # Queue position is the number of packages ahead plus 1 (for this package)
        queue_position = packages_ahead + 1

        # Round estimated processing hours to 2 decimal places for cleaner output
        estimated_processing_hours = round(estimated_processing_hours, 2)

        # Return the estimation results
        return {
            "estimated_processing_hours": estimated_processing_hours,
            "queue_position": queue_position
        }

    @is_tool()
    def calculate_delivery_priority_score(
        self,
        service_type: Literal["standard", "express", "same_day", "next_day"],
        order_time: str,
        deadline: str,
        declared_value: float,
        customer_tier: Literal["regular", "silver", "gold", "platinum"]
    ) -> dict:
        """
        Calculate priority score for delivery scheduling based on multiple factors.

        The priority score is calculated based on:
        1. Service type urgency (40% weight)
        2. Time urgency based on remaining time until deadline (30% weight)
        3. Package value (20% weight)
        4. Customer tier level (10% weight)

        Args:
            service_type: Type of courier service (standard/express/same_day/next_day)
            order_time: Order creation timestamp in yyyy-mm-dd HH:MM:SS format
            deadline: Delivery deadline in yyyy-mm-dd HH:MM:SS format
            declared_value: Declared value of package contents
            customer_tier: Customer tier level (regular/silver/gold/platinum)

        Returns:
            dict: Contains priority_score (0-100) and priority_level (low/medium/high/urgent)

        Raises:
            ValueError: If time format is invalid, deadline is before order_time, or declared_value is negative
        """
        from datetime import datetime

        # Validate and parse time parameters
        try:
            order_dt = datetime.strptime(order_time, "%Y-%m-%d %H:%M:%S")
            deadline_dt = datetime.strptime(deadline, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid time format. Expected 'yyyy-mm-dd HH:MM:SS'. Error: {str(e)}")

        # Validate deadline is after order time
        if deadline_dt <= order_dt:
            raise ValueError("Deadline must be after order time")

        # Validate declared value is non-negative
        if declared_value < 0:
            raise ValueError("Declared value cannot be negative")

        # Initialize score components
        score = 0.0

        # 1. Service Type Score (40% weight) - Higher urgency = higher score
        service_type_scores = {
            "same_day": 100.0,      # Most urgent
            "express": 80.0,        # High urgency
            "next_day": 60.0,       # Medium urgency
            "standard": 40.0        # Normal urgency
        }
        service_score = service_type_scores.get(service_type, 40.0)
        score += service_score * 0.4

        # 2. Time Urgency Score (30% weight) - Less remaining time = higher score
        # Calculate time remaining in hours
        time_remaining = (deadline_dt - order_dt).total_seconds() / 3600.0

        # Calculate urgency based on remaining time
        # Less than 3 hours: very urgent (100)
        # 3-6 hours: urgent (80)
        # 6-12 hours: medium (60)
        # 12-24 hours: normal (40)
        # More than 24 hours: low (20)
        if time_remaining < 3:
            time_score = 100.0
        elif time_remaining < 6:
            time_score = 80.0
        elif time_remaining < 12:
            time_score = 60.0
        elif time_remaining < 24:
            time_score = 40.0
        else:
            # Gradually decrease score for longer time windows
            # Max 72 hours (3 days) for minimum score
            time_score = max(20.0, 40.0 - (time_remaining - 24) / 48.0 * 20.0)

        score += time_score * 0.3

        # 3. Package Value Score (20% weight) - Higher value = higher priority
        # Use logarithmic scale to handle wide range of values
        # Values: 0-100: 20, 100-500: 40, 500-1000: 60, 1000-5000: 80, >5000: 100
        if declared_value <= 100:
            value_score = 20.0 + (declared_value / 100.0) * 20.0
        elif declared_value <= 500:
            value_score = 40.0 + ((declared_value - 100) / 400.0) * 20.0
        elif declared_value <= 1000:
            value_score = 60.0 + ((declared_value - 500) / 500.0) * 20.0
        elif declared_value <= 5000:
            value_score = 80.0 + ((declared_value - 1000) / 4000.0) * 15.0
        else:
            value_score = min(100.0, 95.0 + ((declared_value - 5000) / 10000.0) * 5.0)

        score += value_score * 0.2

        # 4. Customer Tier Score (10% weight) - Higher tier = higher priority
        customer_tier_scores = {
            "platinum": 100.0,
            "gold": 75.0,
            "silver": 50.0,
            "regular": 25.0
        }
        tier_score = customer_tier_scores.get(customer_tier, 25.0)
        score += tier_score * 0.1

        # Ensure score is within valid range [0, 100]
        priority_score = round(max(0.0, min(100.0, score)), 1)

        # Determine priority level based on final score
        if priority_score >= 80:
            priority_level = "urgent"
        elif priority_score >= 60:
            priority_level = "high"
        elif priority_score >= 40:
            priority_level = "medium"
        else:
            priority_level = "low"

        return {
            "priority_score": priority_score,
            "priority_level": priority_level
        }

    @is_tool()
    def cancel_shipment_order(self, order_id: str, cancellation_reason: str):
        """
        Cancel a shipment order before pickup

        This method cancels an existing shipment order if it's in a cancellable status.
        Only orders with status 'pending' or 'picked_up' can be cancelled.
        A refund amount is calculated based on the order's total cost and status.

        Args:
            order_id: Unique identifier for the shipment order
            cancellation_reason: Reason for cancelling the shipment

        Returns:
            dict: Contains cancellation status, refund amount, and cancellation timestamp

        Raises:
            ValueError: If order_id is empty, order doesn't exist, or order is not cancellable
        """
        from datetime import datetime

        # Validate input parameters
        if not order_id or not order_id.strip():
            raise ValueError("Order ID cannot be empty")

        if not cancellation_reason or not cancellation_reason.strip():
            raise ValueError("Cancellation reason cannot be empty")

        # Access the database
        db = self.db

        # Get the shipment_order table
        shipment_order_table = getattr(db, "shipment_order", None)

        # Check if the table exists
        if shipment_order_table is None:
            raise ValueError(f"Shipment order table not found in database")

        # Check if the order exists
        if order_id not in shipment_order_table:
            raise ValueError(f"Order with ID '{order_id}' does not exist")

        # Retrieve the order
        order = shipment_order_table[order_id]

        # Define cancellable statuses
        # Orders can only be cancelled if they are pending or just picked up
        cancellable_statuses = ["pending", "picked_up"]

        # Check if the order status allows cancellation
        if order.status not in cancellable_statuses:
            raise ValueError(
                f"Order with ID '{order_id}' cannot be cancelled. "
                f"Current status is '{order.status}'. "
                f"Only orders with status {cancellable_statuses} can be cancelled."
            )

        # Calculate refund amount based on order status
        # If order is still pending, full refund is given
        # If order is already picked up, a partial refund (90%) is given
        refund_amount = 0.0
        if order.total_cost is not None:
            if order.status == "pending":
                # Full refund for pending orders
                refund_amount = order.total_cost
            elif order.status == "picked_up":
                # 90% refund for picked up orders (10% processing fee)
                refund_amount = order.total_cost * 0.9

        # Round refund amount to 2 decimal places for currency
        refund_amount = round(refund_amount, 2)

        # Get current timestamp for cancellation
        cancelled_at = datetime.now()

        # Update the order status to cancelled
        order.status = "cancelled"
        order.updated_at = cancelled_at

        # Update the order in the database
        shipment_order_table[order_id] = order
        setattr(db, "shipment_order", shipment_order_table)

        # Format the cancellation timestamp as string
        cancelled_at_str = cancelled_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the cancellation result
        return {
            "cancelled": True,
            "refund_amount": refund_amount,
            "cancelled_at": cancelled_at_str
        }

    @is_tool()
    def update_shipment_status(self, order_id: str, new_status: Literal["pending", "picked_up", "in_transit", "out_for_delivery", "delivered", "failed", "cancelled", "returned"], status_note: str = None):
        """
        Update the status of a shipment order

        Args:
            order_id: Unique identifier for the shipment order
            new_status: New status to be set for the shipment (must be one of the valid status values)
            status_note: Optional additional note or comment about the status change

        Returns:
            dict: Contains 'updated' (bool) and 'updated_at' (str in yyyy-mm-dd HH:MM:SS format)

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

        # Access the database
        db = self.db

        # Get the shipment_order table from database
        shipment_order_table = getattr(db, "shipment_order", None)

        # Check if the table exists
        if shipment_order_table is None:
            raise KeyError(f"Shipment order table not found in database")

        # Check if the order exists in the table
        if order_id not in shipment_order_table:
            raise KeyError(f"Order with ID '{order_id}' not found")

        # Retrieve the order object
        order = shipment_order_table[order_id]

        # Update the status field
        order.status = new_status

        # Update the timestamp
        current_time = datetime.now()
        order.updated_at = current_time

        # Note: status_note is provided for context but not stored in the schema
        # If needed for logging/auditing, it would require a separate tracking mechanism
        # For now, we acknowledge it but don't persist it as the schema doesn't include a notes field

        # Write the updated order back to the database
        shipment_order_table[order_id] = order
        setattr(db, "shipment_order", shipment_order_table)

        # Format the timestamp as required (yyyy-mm-dd HH:MM:SS)
        updated_at_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response
        return {
            "updated": True,
            "updated_at": updated_at_str
        }

    @is_tool()
    def check_delivery_time_window(
        self,
        current_time: str,
        destination_latitude: float,
        destination_longitude: float,
        window_start: str,
        window_end: str,
        average_speed_kmh: float
    ):
        """
        Check if a delivery can be completed within the specified time window.

        This method calculates whether there is enough time to reach the destination
        before the delivery window closes, considering the current time, destination
        coordinates, and average travel speed.

        Note: This is a generic validation tool that does not require specific courier
        or order information. It performs a theoretical feasibility check based on
        provided parameters.
        """
        from datetime import datetime
        from math import radians, sin, cos, sqrt, atan2

        # Parse time strings to datetime objects
        try:
            current_dt = datetime.strptime(current_time, "%Y-%m-%d %H:%M:%S")
            window_start_dt = datetime.strptime(window_start, "%Y-%m-%d %H:%M:%S")
            window_end_dt = datetime.strptime(window_end, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid time format. Expected 'yyyy-mm-dd HH:MM:SS': {str(e)}")

        # Validate time window logic
        if window_start_dt >= window_end_dt:
            raise ValueError("Time window start must be before window end")

        if current_dt > window_end_dt:
            raise ValueError("Current time is already past the delivery window end")

        # Validate coordinates
        if not (-90 <= destination_latitude <= 90):
            raise ValueError("Destination latitude must be between -90 and 90")

        if not (-180 <= destination_longitude <= 180):
            raise ValueError("Destination longitude must be between -180 and 180")

        # Validate average speed
        if average_speed_kmh <= 0:
            raise ValueError("Average speed must be greater than 0")

        # Calculate distance using Haversine formula
        # Assumes starting point at origin (0, 0) for generic validation
        # In production, this would use actual courier location
        def haversine_distance(lat1, lon1, lat2, lon2):
            """
            Calculate the great circle distance between two points 
            on the earth (specified in decimal degrees).
            Returns distance in kilometers.
            """
            # Convert decimal degrees to radians
            lat1_rad, lon1_rad = radians(lat1), radians(lon1)
            lat2_rad, lon2_rad = radians(lat2), radians(lon2)

            # Haversine formula
            dlat = lat2_rad - lat1_rad
            dlon = lon2_rad - lon1_rad
            a = sin(dlat / 2)**2 + cos(lat1_rad) * cos(lat2_rad) * sin(dlon / 2)**2
            c = 2 * atan2(sqrt(a), sqrt(1 - a))

            # Radius of earth in kilometers
            earth_radius_km = 6371.0
            distance = earth_radius_km * c

            return distance

        # For generic validation, assume starting from origin (0, 0)
        # This provides a theoretical maximum distance calculation
        current_latitude = 0.0
        current_longitude = 0.0

        # Calculate distance to destination in kilometers
        distance_km = haversine_distance(
            current_latitude,
            current_longitude,
            destination_latitude,
            destination_longitude
        )

        # Calculate estimated travel time in hours
        estimated_travel_hours = distance_km / average_speed_kmh

        # Convert to minutes for easier comparison
        estimated_travel_minutes = estimated_travel_hours * 60

        # Calculate available time window in minutes
        # Time available = window_end - current_time
        available_time_seconds = (window_end_dt - current_dt).total_seconds()
        available_time_minutes = available_time_seconds / 60

        # Check feasibility: can we reach destination before window closes?
        is_feasible = estimated_travel_minutes <= available_time_minutes

        # Calculate buffer time (positive if feasible, negative if not)
        # Buffer time = available time - estimated travel time
        buffer_time_minutes = available_time_minutes - estimated_travel_minutes

        return {
            "is_feasible": is_feasible,
            "buffer_time_minutes": round(buffer_time_minutes, 2)
        }

    @is_tool()
    def record_delivery_attempt(
        self,
        order_id: str,
        courier_id: str,
        attempt_time: str,
        failure_reason: Literal["recipient_not_home", "incorrect_address", "refused_delivery", "access_denied", "weather_conditions", "other"],
        notes: str = None
    ):
        """
        Record a failed delivery attempt with reason.

        This method creates a new delivery attempt record in the system and updates
        the attempt count for the given order.

        Args:
            order_id: Unique identifier for the shipment order
            courier_id: Unique identifier for the courier
            attempt_time: Delivery attempt timestamp in yyyy-mm-dd HH:MM:SS format
            failure_reason: Reason for delivery failure (must be one of the enum values)
            notes: Additional notes about the delivery attempt (optional)

        Returns:
            dict: Contains attempt_record_id, attempt_count, and recorded_at timestamp

        Raises:
            KeyError: If order_id does not exist in the database
            ValueError: If attempt_time format is invalid or failure_reason is not in enum
        """
        from datetime import datetime

        # Access database
        db = self.db

        # Validate that order exists in database
        shipment_order_table = getattr(db, "shipment_order", None)
        if shipment_order_table is None or order_id not in shipment_order_table:
            raise KeyError(f"Order ID '{order_id}' does not exist in the system")

        # Validate failure_reason against enum values
        valid_reasons = ["recipient_not_home", "incorrect_address", "refused_delivery", 
                         "access_denied", "weather_conditions", "other"]
        if failure_reason not in valid_reasons:
            raise ValueError(f"Invalid failure_reason '{failure_reason}'. Must be one of: {valid_reasons}")

        # Parse and validate attempt_time format
        try:
            attempt_datetime = datetime.strptime(attempt_time, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid attempt_time format. Expected 'yyyy-mm-dd HH:MM:SS', got '{attempt_time}'") from e

        # Get current timestamp for record creation
        current_time = datetime.now()

        # Generate unique attempt_record_id using timestamp-based approach
        timestamp_str = current_time.strftime("%Y%m%d%H%M%S%f")
        attempt_record_id = f"ATM{timestamp_str[:14]}"

        # Get existing delivery attempts for this order to calculate attempt count
        delivery_attempt_table = getattr(db, "delivery_attempt", None)
        if delivery_attempt_table is None:
            delivery_attempt_table = {}

        # Count existing attempts for this order
        attempt_count = sum(1 for attempt in delivery_attempt_table.values() 
                           if attempt.order_id == order_id) + 1

        # Create new DeliveryAttempt record
        new_attempt = DeliveryAttempt(
            attempt_record_id=attempt_record_id,
            order_id=order_id,
            courier_id=courier_id,
            attempt_time=attempt_datetime,
            failure_reason=failure_reason,
            notes=notes,
            attempt_count=attempt_count,
            created_at=current_time
        )

        # Add the new attempt to the database
        delivery_attempt_table[attempt_record_id] = new_attempt
        setattr(db, "delivery_attempt", delivery_attempt_table)

        # Format recorded_at timestamp for return
        recorded_at_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "attempt_record_id": attempt_record_id,
            "attempt_count": attempt_count,
            "recorded_at": recorded_at_str
        }

    @is_tool()
    def get_tracking_history(self, tracking_number: str):
        # Get database instance
        db = self.db

        # Retrieve tracking_event table from database
        tracking_event_table = getattr(db, "tracking_event", None)
        if tracking_event_table is None:
            raise KeyError("tracking_event table not found in database")

        # Retrieve shipment_order table from database
        shipment_order_table = getattr(db, "shipment_order", None)
        if shipment_order_table is None:
            raise KeyError("shipment_order table not found in database")

        # Find the shipment order with the given tracking number
        shipment = None
        for order_id, order in shipment_order_table.items():
            if order.tracking_number == tracking_number:
                shipment = order
                break

        # If no shipment found with this tracking number, raise KeyError
        if shipment is None:
            raise KeyError(f"No shipment found with tracking number: {tracking_number}")

        # Collect all tracking events for this tracking number
        tracking_events = []
        for event_id, event in tracking_event_table.items():
            if event.tracking_number == tracking_number:
                # Convert datetime to string format "yyyy-mm-dd HH:MM:SS"
                timestamp_str = event.event_time.strftime("%Y-%m-%d %H:%M:%S")

                # Create event dictionary with required fields
                event_dict = {
                    "status": event.event_type,
                    "location": event.event_location,
                    "timestamp": timestamp_str
                }

                # Add optional description if available
                if event.event_description:
                    event_dict["description"] = event.event_description

                tracking_events.append(event_dict)

        # Sort tracking events by timestamp in chronological order (earliest first)
        # Parse timestamp strings back to datetime for proper sorting
        tracking_events.sort(key=lambda x: datetime.strptime(x["timestamp"], "%Y-%m-%d %H:%M:%S"))

        # Get current status from shipment order
        current_status = shipment.status

        # Return complete tracking history
        return {
            "tracking_events": tracking_events,
            "status": current_status
        }

    @is_tool()
    def calculate_fuel_surcharge(
        self,
        distance_km: float,
        fuel_price_per_liter: float,
        base_fuel_price: float,
        vehicle_efficiency: float
    ) -> dict:
        """
        Calculate fuel surcharge based on current fuel prices and distance.

        The fuel surcharge is calculated based on the difference between current fuel price
        and base fuel price, considering the distance traveled and vehicle fuel efficiency.

        Args:
            distance_km: Transportation distance in kilometers
            fuel_price_per_liter: Current fuel price per liter in currency units
            base_fuel_price: Base fuel price for surcharge calculation
            vehicle_efficiency: Vehicle fuel efficiency in kilometers per liter

        Returns:
            dict: Contains surcharge_amount and surcharge_percentage

        Raises:
            ValueError: If any input parameter is invalid (negative or zero where not allowed)
        """
        # Validate input parameters
        if distance_km <= 0:
            raise ValueError("Distance must be greater than zero")

        if fuel_price_per_liter <= 0:
            raise ValueError("Current fuel price must be greater than zero")

        if base_fuel_price <= 0:
            raise ValueError("Base fuel price must be greater than zero")

        if vehicle_efficiency <= 0:
            raise ValueError("Vehicle efficiency must be greater than zero")

        # Calculate fuel consumption for the distance
        # Fuel consumption (liters) = distance (km) / efficiency (km/liter)
        fuel_consumed_liters = distance_km / vehicle_efficiency

        # Calculate the price difference per liter
        price_difference_per_liter = fuel_price_per_liter - base_fuel_price

        # Calculate the fuel surcharge amount
        # Surcharge = fuel consumed * price difference per liter
        surcharge_amount = fuel_consumed_liters * price_difference_per_liter

        # Calculate base fuel cost (what it would cost at base price)
        base_fuel_cost = fuel_consumed_liters * base_fuel_price

        # Calculate surcharge as percentage of base cost
        # Percentage = (surcharge / base cost) * 100
        if base_fuel_cost > 0:
            surcharge_percentage = (surcharge_amount / base_fuel_cost) * 100
        else:
            surcharge_percentage = 0.0

        # Round results to 2 decimal places for currency precision
        surcharge_amount = round(surcharge_amount, 2)
        surcharge_percentage = round(surcharge_percentage, 2)

        return {
            "surcharge_amount": surcharge_amount,
            "surcharge_percentage": surcharge_percentage
        }

    @is_tool()
    def check_delivery_time_compliance(
        self,
        promised_delivery_time: str,
        actual_delivery_time: str,
        grace_period_minutes: int
    ):
        """
        Check if actual delivery time complies with service level agreement.

        This method compares the actual delivery time against the promised delivery time,
        taking into account a grace period. It returns compliance status and time difference.

        Args:
            promised_delivery_time: Promised delivery timestamp in yyyy-mm-dd HH:MM:SS format
            actual_delivery_time: Actual delivery timestamp in yyyy-mm-dd HH:MM:SS format
            grace_period_minutes: Grace period allowed beyond promised time in minutes

        Returns:
            dict: Contains is_compliant (bool), time_difference_minutes (float), 
                  and compliance_status (str)

        Raises:
            ValueError: If time format is invalid or grace_period_minutes is negative
        """
        from datetime import datetime

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

        # Parse time strings to datetime objects
        try:
            promised_dt = datetime.strptime(promised_delivery_time, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid promised_delivery_time format. Expected yyyy-mm-dd HH:MM:SS, got: {promised_delivery_time}") from e

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

        # Calculate time difference in minutes
        # Negative means early delivery, positive means late delivery
        time_delta = actual_dt - promised_dt
        time_difference_minutes = time_delta.total_seconds() / 60.0

        # Determine compliance status based on time difference
        if time_difference_minutes < 0:
            # Delivered early
            compliance_status = "early"
            is_compliant = True
        elif time_difference_minutes == 0:
            # Delivered exactly on time
            compliance_status = "on_time"
            is_compliant = True
        elif 0 < time_difference_minutes <= grace_period_minutes:
            # Delivered late but within grace period
            compliance_status = "late_within_grace"
            is_compliant = True
        else:
            # Delivered late beyond grace period
            compliance_status = "late_beyond_grace"
            is_compliant = False

        # Return compliance check result
        return {
            "is_compliant": is_compliant,
            "time_difference_minutes": time_difference_minutes,
            "compliance_status": compliance_status
        }

    @is_tool()
    def calculate_shipping_cost(
        self,
        weight: float,
        length: float,
        width: float,
        height: float,
        origin_latitude: float,
        origin_longitude: float,
        destination_latitude: float,
        destination_longitude: float,
        service_type: Literal["standard", "express", "same_day", "next_day"]
    ) -> dict:
        """
        Calculate the shipping cost based on package weight, dimensions, distance, and service type.

        The calculation considers:
        1. Base cost from weight and volumetric weight
        2. Distance-based pricing using Haversine formula
        3. Service type multiplier
        4. Fuel surcharge (percentage of base cost)
        """
        from typing import Literal
        import math

        # Validate input parameters
        if weight <= 0:
            raise ValueError("Weight must be greater than 0")
        if length <= 0 or width <= 0 or height <= 0:
            raise ValueError("All dimensions (length, width, height) must be greater than 0")
        if not (-90 <= origin_latitude <= 90) or not (-90 <= destination_latitude <= 90):
            raise ValueError("Latitude must be between -90 and 90 degrees")
        if not (-180 <= origin_longitude <= 180) or not (-180 <= destination_longitude <= 180):
            raise ValueError("Longitude must be between -180 and 180 degrees")

        # Calculate volumetric weight (dimensional weight)
        # Standard formula: (length × width × height) / volumetric_divisor
        # Using 5000 as volumetric divisor (common in logistics)
        volumetric_weight = (length * width * height) / 5000.0

        # Use the greater of actual weight or volumetric weight for pricing
        chargeable_weight = max(weight, volumetric_weight)

        # Calculate distance between origin and destination using Haversine formula
        # This gives the great-circle distance between two points on Earth
        def haversine_distance(lat1: float, lon1: float, lat2: float, lon2: float) -> float:
            """
            Calculate the great circle distance between two points on Earth
            Returns distance in kilometers
            """
            # Earth's radius in kilometers
            R = 6371.0

            # Convert latitude and longitude from degrees to radians
            lat1_rad = math.radians(lat1)
            lon1_rad = math.radians(lon1)
            lat2_rad = math.radians(lat2)
            lon2_rad = math.radians(lon2)

            # Differences
            dlat = lat2_rad - lat1_rad
            dlon = lon2_rad - lon1_rad

            # Haversine formula
            a = math.sin(dlat / 2)**2 + math.cos(lat1_rad) * math.cos(lat2_rad) * math.sin(dlon / 2)**2
            c = 2 * math.atan2(math.sqrt(a), math.sqrt(1 - a))
            distance = R * c

            return distance

        distance_km = haversine_distance(
            origin_latitude, origin_longitude,
            destination_latitude, destination_longitude
        )

        # Base rate per kg (can be adjusted based on business requirements)
        base_rate_per_kg = 5.0

        # Distance rate per km (can be adjusted based on business requirements)
        distance_rate_per_km = 0.5

        # Service type multipliers
        service_multipliers = {
            "standard": 1.0,      # Base rate
            "express": 1.5,       # 50% premium
            "next_day": 1.8,      # 80% premium
            "same_day": 2.5       # 150% premium
        }

        # Get service multiplier (with safety check)
        service_multiplier = service_multipliers.get(service_type, 1.0)

        # Calculate base cost components
        weight_cost = chargeable_weight * base_rate_per_kg
        distance_cost = distance_km * distance_rate_per_km

        # Calculate base cost before service multiplier
        base_cost_before_service = weight_cost + distance_cost

        # Apply service type multiplier
        base_cost = base_cost_before_service * service_multiplier

        # Calculate fuel surcharge (typically 8-12% of base cost in logistics)
        fuel_surcharge_rate = 0.10  # 10% fuel surcharge
        fuel_surcharge = base_cost * fuel_surcharge_rate

        # Calculate total cost
        total_cost = base_cost + fuel_surcharge

        # Round to 2 decimal places for currency
        base_cost = round(base_cost, 2)
        fuel_surcharge = round(fuel_surcharge, 2)
        total_cost = round(total_cost, 2)

        return {
            "base_cost": base_cost,
            "fuel_surcharge": fuel_surcharge,
            "total_cost": total_cost
        }

    @is_tool()
    def estimate_package_transit_stages(
        self,
        origin_latitude: float,
        origin_longitude: float,
        destination_latitude: float,
        destination_longitude: float,
        service_type: Literal["standard", "express", "same_day", "next_day"]
    ) -> dict:
        """
        Estimate the number and duration of transit stages for a shipment based on
        origin, destination coordinates, and service type.

        The estimation is based on:
        1. Distance calculation using Haversine formula
        2. Service type characteristics (speed, processing time)
        3. Standard logistics stages (pickup, sorting, transit, delivery)
        """
        import math

        # Validate input parameters
        if not (-90 <= origin_latitude <= 90):
            raise ValueError(f"Invalid origin_latitude: {origin_latitude}. Must be between -90 and 90.")
        if not (-180 <= origin_longitude <= 180):
            raise ValueError(f"Invalid origin_longitude: {origin_longitude}. Must be between -180 and 180.")
        if not (-90 <= destination_latitude <= 90):
            raise ValueError(f"Invalid destination_latitude: {destination_latitude}. Must be between -90 and 90.")
        if not (-180 <= destination_longitude <= 180):
            raise ValueError(f"Invalid destination_longitude: {destination_longitude}. Must be between -180 and 180.")

        # Safety check for service_type enum (though Literal already constrains it)
        valid_service_types = ["standard", "express", "same_day", "next_day"]
        if service_type not in valid_service_types:
            raise ValueError(f"Invalid service_type: {service_type}. Must be one of {valid_service_types}.")

        # Calculate distance using Haversine formula (in kilometers)
        def haversine_distance(lat1: float, lon1: float, lat2: float, lon2: float) -> float:
            """
            Calculate the great circle distance between two points on Earth
            using the Haversine formula.

            Returns distance in kilometers.
            """
            # Earth's radius in kilometers
            R = 6371.0

            # Convert degrees to radians
            lat1_rad = math.radians(lat1)
            lon1_rad = math.radians(lon1)
            lat2_rad = math.radians(lat2)
            lon2_rad = math.radians(lon2)

            # Differences
            dlat = lat2_rad - lat1_rad
            dlon = lon2_rad - lon1_rad

            # Haversine formula
            a = math.sin(dlat / 2)**2 + math.cos(lat1_rad) * math.cos(lat2_rad) * math.sin(dlon / 2)**2
            c = 2 * math.atan2(math.sqrt(a), math.sqrt(1 - a))
            distance = R * c

            return distance

        distance_km = haversine_distance(
            origin_latitude, origin_longitude,
            destination_latitude, destination_longitude
        )

        # Define service type characteristics
        # Each service type has different speed (km/h) and processing efficiency
        service_configs = {
            "same_day": {
                "avg_speed": 80,  # Fast transit speed
                "pickup_hours": 1,  # Quick pickup
                "sorting_hours": 1,  # Minimal sorting
                "delivery_hours": 1,  # Quick delivery
                "max_distance": 200  # Same day typically within city/region
            },
            "next_day": {
                "avg_speed": 70,
                "pickup_hours": 2,
                "sorting_hours": 2,
                "delivery_hours": 2,
                "max_distance": 1000
            },
            "express": {
                "avg_speed": 60,
                "pickup_hours": 3,
                "sorting_hours": 3,
                "delivery_hours": 2,
                "max_distance": float('inf')
            },
            "standard": {
                "avg_speed": 50,
                "pickup_hours": 4,
                "sorting_hours": 4,
                "delivery_hours": 3,
                "max_distance": float('inf')
            }
        }

        config = service_configs[service_type]

        # Initialize stages list
        stages = []

        # Stage 1: Pickup
        pickup_duration = config["pickup_hours"]
        stages.append({
            "stage": "pickup",
            "duration_hours": pickup_duration
        })

        # Stage 2: Initial sorting at origin facility
        origin_sorting_duration = config["sorting_hours"]
        stages.append({
            "stage": "origin_sorting",
            "duration_hours": origin_sorting_duration
        })

        # Stage 3: Transit stages
        # Calculate number of transit segments based on distance
        # Longer distances require more intermediate stops/transfers
        transit_duration = distance_km / config["avg_speed"]

        # Determine if intermediate sorting/transfer is needed
        # For distances > 500km, add intermediate facility processing
        if distance_km > 500:
            # Split transit into multiple segments with intermediate sorting
            num_intermediate_facilities = int(distance_km / 500)
            segment_duration = transit_duration / (num_intermediate_facilities + 1)

            for i in range(num_intermediate_facilities + 1):
                stages.append({
                    "stage": f"transit_segment_{i+1}",
                    "duration_hours": round(segment_duration, 2)
                })

                # Add intermediate sorting except after the last segment
                if i < num_intermediate_facilities:
                    stages.append({
                        "stage": f"intermediate_sorting_{i+1}",
                        "duration_hours": config["sorting_hours"] * 0.5  # Shorter than origin sorting
                    })
        else:
            # Single transit segment for shorter distances
            stages.append({
                "stage": "transit",
                "duration_hours": round(transit_duration, 2)
            })

        # Stage 4: Destination sorting
        destination_sorting_duration = config["sorting_hours"] * 0.8  # Slightly faster than origin
        stages.append({
            "stage": "destination_sorting",
            "duration_hours": round(destination_sorting_duration, 2)
        })

        # Stage 5: Out for delivery
        stages.append({
            "stage": "out_for_delivery",
            "duration_hours": config["delivery_hours"]
        })

        # Stage 6: Final delivery
        stages.append({
            "stage": "delivery",
            "duration_hours": 0.5  # Final handoff time
        })

        # Calculate total transit time
        total_transit_time = sum(stage["duration_hours"] for stage in stages)

        # Round total time to 2 decimal places
        total_transit_time = round(total_transit_time, 2)

        # Prepare return value
        result = {
            "stage_count": len(stages),
            "stages": stages,
            "total_transit_time_hours": total_transit_time
        }

        return result

    @is_tool()
    def validate_phone_number(self, phone_number: str, country_code: str):
        """
        Validate if a phone number meets the required format for the specified country.

        Args:
            phone_number: Phone number to validate
            country_code: ISO country code (e.g., "CN", "US", "GB")

        Returns:
            dict: Contains is_valid (bool) and formatted_number (str)

        Raises:
            ValueError: If phone_number or country_code is invalid
        """
        import re

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

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

        # Convert country code to uppercase for consistency
        country_code = country_code.upper()

        # Remove all non-digit characters from phone number for validation
        clean_number = re.sub(r'\D', '', phone_number)

        # Define validation rules and formatting patterns for different countries
        # Format: {country_code: (regex_pattern, country_calling_code, format_function)}
        country_rules = {
            # China: 11 digits, starts with 1
            "CN": {
                "pattern": r'^1[3-9]\d{9}$',
                "country_code": "+86",
                "format": lambda n: f"+86 {n[:3]} {n[3:7]} {n[7:]}"
            },
            # United States: 10 digits
            "US": {
                "pattern": r'^[2-9]\d{2}[2-9]\d{6}$',
                "country_code": "+1",
                "format": lambda n: f"+1 ({n[:3]}) {n[3:6]}-{n[6:]}"
            },
            # United Kingdom: 10-11 digits
            "GB": {
                "pattern": r'^[1-9]\d{9,10}$',
                "country_code": "+44",
                "format": lambda n: f"+44 {n[:4]} {n[4:7]} {n[7:]}" if len(n) == 11 else f"+44 {n[:3]} {n[3:6]} {n[6:]}"
            },
            # Japan: 10-11 digits
            "JP": {
                "pattern": r'^0[1-9]\d{8,9}$',
                "country_code": "+81",
                "format": lambda n: f"+81 {n[1:3]} {n[3:7]} {n[7:]}" if len(n) == 11 else f"+81 {n[1:2]} {n[2:6]} {n[6:]}"
            },
            # Germany: 10-11 digits
            "DE": {
                "pattern": r'^0[1-9]\d{8,10}$',
                "country_code": "+49",
                "format": lambda n: f"+49 {n[1:4]} {n[4:7]} {n[7:]}"
            },
            # France: 10 digits, starts with 0
            "FR": {
                "pattern": r'^0[1-9]\d{8}$',
                "country_code": "+33",
                "format": lambda n: f"+33 {n[1]} {n[2:4]} {n[4:6]} {n[6:8]} {n[8:]}"
            },
            # India: 10 digits
            "IN": {
                "pattern": r'^[6-9]\d{9}$',
                "country_code": "+91",
                "format": lambda n: f"+91 {n[:5]} {n[5:]}"
            },
            # Australia: 9 digits (mobile starts with 4)
            "AU": {
                "pattern": r'^[2-478]\d{8}$',
                "country_code": "+61",
                "format": lambda n: f"+61 {n[:3]} {n[3:6]} {n[6:]}"
            },
            # Canada: 10 digits (same as US)
            "CA": {
                "pattern": r'^[2-9]\d{2}[2-9]\d{6}$',
                "country_code": "+1",
                "format": lambda n: f"+1 ({n[:3]}) {n[3:6]}-{n[6:]}"
            },
            # South Korea: 10-11 digits
            "KR": {
                "pattern": r'^0[1-9]\d{8,9}$',
                "country_code": "+82",
                "format": lambda n: f"+82 {n[1:3]} {n[3:7]} {n[7:]}" if len(n) == 11 else f"+82 {n[1:2]} {n[2:6]} {n[6:]}"
            }
        }

        # Check if country code is supported
        if country_code not in country_rules:
            raise ValueError(f"Country code '{country_code}' is not supported. Supported countries: {', '.join(country_rules.keys())}")

        # Get validation rules for the specified country
        rules = country_rules[country_code]
        pattern = rules["pattern"]
        calling_code = rules["country_code"]
        format_func = rules["format"]

        # Validate the phone number against the country-specific pattern
        is_valid = bool(re.match(pattern, clean_number))

        # Format the phone number if valid, otherwise return the original
        if is_valid:
            formatted_number = format_func(clean_number)
        else:
            # Return a basic format with country code even if invalid
            formatted_number = f"{calling_code} {clean_number}"

        return {
            "is_valid": is_valid,
            "formatted_number": formatted_number
        }

    @is_tool()
    def get_courier_current_location(self, courier_id: str):
        # Import necessary modules for database access
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the courier table from the database
        courier_table = getattr(db, "courier", None)

        # Check if the courier table exists in the database
        if courier_table is None:
            raise KeyError(f"Courier table not found in database")

        # Check if the courier_id exists in the courier table
        if courier_id not in courier_table:
            raise KeyError(f"Courier with ID '{courier_id}' not found")

        # Retrieve the courier record
        courier = courier_table[courier_id]

        # Extract current location coordinates
        latitude = courier.current_latitude
        longitude = courier.current_longitude
        last_updated_datetime = courier.last_location_update

        # Check if location data is available
        # If coordinates are None, it means the courier's location has not been updated yet
        if latitude is None or longitude is None:
            raise KeyError(f"Location data not available for courier '{courier_id}'")

        # Format the last_updated timestamp to "yyyy-mm-dd HH:MM:SS" format
        # Convert datetime object to string in the required format
        if last_updated_datetime is not None:
            last_updated = last_updated_datetime.strftime("%Y-%m-%d %H:%M:%S")
        else:
            # If last_location_update is None, raise an error as location data is incomplete
            raise KeyError(f"Last location update timestamp not available for courier '{courier_id}'")

        # Return the courier's current location information
        return {
            "latitude": latitude,
            "longitude": longitude,
            "last_updated": last_updated
        }

    @is_tool()
    def add_tracking_event(
        self,
        tracking_number: str,
        event_type: Literal["order_created", "picked_up", "in_transit", "arrived_at_facility", "out_for_delivery", "delivered", "exception", "returned"],
        event_location: str,
        event_time: str,
        event_description: str = None
    ):
        """
        Add a new tracking event to a shipment's history.

        This method creates a new tracking event record in the database with the provided details.
        The event is associated with a shipment via its tracking number and includes information
        about the event type, location, time, and optional description.

        Args:
            tracking_number: Tracking number for the shipment
            event_type: Type of tracking event (must be one of the predefined enum values)
            event_location: Location where the event occurred
            event_time: Event timestamp in "yyyy-mm-dd HH:MM:SS" format
            event_description: Optional detailed description of the event

        Returns:
            dict: Contains event_id and added_at timestamp

        Raises:
            KeyError: If required parameters are missing or invalid
            ValueError: If event_time format is invalid or event_type is not in allowed values
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not tracking_number or not isinstance(tracking_number, str):
            raise KeyError("tracking_number is required and must be a non-empty string")

        if not event_type or not isinstance(event_type, str):
            raise KeyError("event_type is required and must be a non-empty string")

        # Validate event_type against allowed enum values
        allowed_event_types = [
            "order_created", "picked_up", "in_transit", "arrived_at_facility",
            "out_for_delivery", "delivered", "exception", "returned"
        ]
        if event_type not in allowed_event_types:
            raise ValueError(f"event_type must be one of {allowed_event_types}, got: {event_type}")

        if not event_location or not isinstance(event_location, str):
            raise KeyError("event_location is required and must be a non-empty string")

        if not event_time or not isinstance(event_time, str):
            raise KeyError("event_time is required and must be a non-empty string")

        # Parse and validate event_time format
        try:
            # Support both full datetime format "yyyy-mm-dd HH:MM:SS" and date-only format "yyyy-mm-dd"
            if len(event_time.strip()) == 10:  # Date only format
                parsed_event_time = datetime.strptime(event_time.strip(), "%Y-%m-%d")
            else:  # Full datetime format
                parsed_event_time = datetime.strptime(event_time.strip(), "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"event_time must be in 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd' format, got: {event_time}. Error: {str(e)}")

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

        # Get current timestamp for created_at
        created_at = datetime.now()

        # Access database
        db = self.db

        # Get tracking_event table, initialize if not exists
        tracking_event_table = getattr(db, "tracking_event", None)
        if tracking_event_table is None:
            tracking_event_table = {}
            setattr(db, "tracking_event", tracking_event_table)

        # Create new tracking event instance
        new_event = TrackingEvent(
            event_id=event_id,
            tracking_number=tracking_number,
            event_type=event_type,
            event_location=event_location,
            event_time=parsed_event_time,
            event_description=event_description,
            created_at=created_at
        )

        # Add event to database
        tracking_event_table[event_id] = new_event
        setattr(db, "tracking_event", tracking_event_table)

        # Format return timestamps
        added_at_str = created_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return event details
        return {
            "event_id": event_id,
            "added_at": added_at_str
        }

    @is_tool()
    def assign_courier_to_pickup(self, order_id: str, courier_id: str, scheduled_pickup_time: str):
        """
        Assign a courier to pick up a shipment order.

        This method creates a new courier assignment record for pickup task,
        validates the existence of order and courier, and updates the order status.

        Args:
            order_id: Unique identifier for the shipment order
            courier_id: Unique identifier for the courier
            scheduled_pickup_time: Scheduled pickup time in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains assignment_id and assigned_at timestamp

        Raises:
            KeyError: If order_id or courier_id does not exist in database
            ValueError: If scheduled_pickup_time format is invalid or order status is not valid for pickup
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Validate and parse scheduled_pickup_time format
        try:
            scheduled_time_dt = datetime.strptime(scheduled_pickup_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid scheduled_pickup_time format: {scheduled_pickup_time}. Expected format: yyyy-mm-dd HH:MM:SS")

        # Get shipment_order table
        shipment_order_table = getattr(db, "shipment_order", None)
        if shipment_order_table is None:
            raise KeyError(f"shipment_order table not found in database")

        # Validate order exists
        if order_id not in shipment_order_table:
            raise KeyError(f"Order with order_id '{order_id}' does not exist")

        order = shipment_order_table[order_id]

        # Validate order status - only pending orders can be assigned for pickup
        if order.status not in ["pending"]:
            raise ValueError(f"Order {order_id} cannot be assigned for pickup. Current status: {order.status}")

        # Get courier table
        courier_table = getattr(db, "courier", None)
        if courier_table is None:
            raise KeyError(f"courier table not found in database")

        # Validate courier exists
        if courier_id not in courier_table:
            raise KeyError(f"Courier with courier_id '{courier_id}' does not exist")

        courier = courier_table[courier_id]

        # Check courier availability (optional validation - courier should be available or busy, not offline)
        if courier.status == "offline":
            raise ValueError(f"Courier {courier_id} is offline and cannot be assigned")

        # Generate unique assignment_id
        prefix = "ASG"
        assignment_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for assignment creation
        assigned_at_dt = datetime.now()

        # Get courier_assignment table
        courier_assignment_table = getattr(db, "courier_assignment", None)
        if courier_assignment_table is None:
            # Initialize empty table if not exists
            courier_assignment_table = {}

        # Create new CourierAssignment instance
        new_assignment = CourierAssignment(
            assignment_id=assignment_id,
            order_id=order_id,
            courier_id=courier_id,
            assignment_type="pickup",
            scheduled_time=scheduled_time_dt,
            assignment_status="assigned",
            created_at=assigned_at_dt
        )

        # Add assignment to table
        courier_assignment_table[assignment_id] = new_assignment

        # Update courier_assignment table in database
        setattr(db, "courier_assignment", courier_assignment_table)

        # Update order status to indicate pickup has been scheduled
        # Note: We don't change status to "picked_up" yet, that happens when courier actually picks up
        # But we can update the updated_at timestamp
        order.updated_at = assigned_at_dt
        shipment_order_table[order_id] = order
        setattr(db, "shipment_order", shipment_order_table)

        # Update courier status to busy if currently available
        if courier.status == "available":
            courier.status = "busy"
            courier_table[courier_id] = courier
            setattr(db, "courier", courier_table)

        # Format return timestamps as strings
        assigned_at_str = assigned_at_dt.strftime("%Y-%m-%d %H:%M:%S")

        # Return assignment details
        return {
            "assignment_id": assignment_id,
            "assigned_at": assigned_at_str
        }

    @is_tool()
    def calculate_distance_between_coordinates(
        self,
        origin_latitude: float,
        origin_longitude: float,
        destination_latitude: float,
        destination_longitude: float
    ) -> dict:
        """
        Calculate the geographical distance between two coordinate points using Haversine formula.

        The Haversine formula calculates the great-circle distance between two points on a sphere
        given their longitudes and latitudes. This is useful for calculating distances on Earth.

        Args:
            origin_latitude: Origin location latitude in decimal degrees (-90 to 90)
            origin_longitude: Origin location longitude in decimal degrees (-180 to 180)
            destination_latitude: Destination location latitude in decimal degrees (-90 to 90)
            destination_longitude: Destination location longitude in decimal degrees (-180 to 180)

        Returns:
            dict: Dictionary containing the distance in kilometers
                - distance_km: Distance between the two coordinates in kilometers

        Raises:
            ValueError: If any coordinate values are out of valid range
        """
        import math

        # Validate latitude values (must be between -90 and 90)
        if not (-90 <= origin_latitude <= 90):
            raise ValueError(f"Origin latitude must be between -90 and 90 degrees, got {origin_latitude}")
        if not (-90 <= destination_latitude <= 90):
            raise ValueError(f"Destination latitude must be between -90 and 90 degrees, got {destination_latitude}")

        # Validate longitude values (must be between -180 and 180)
        if not (-180 <= origin_longitude <= 180):
            raise ValueError(f"Origin longitude must be between -180 and 180 degrees, got {origin_longitude}")
        if not (-180 <= destination_longitude <= 180):
            raise ValueError(f"Destination longitude must be between -180 and 180 degrees, got {destination_longitude}")

        # Earth's radius in kilometers
        EARTH_RADIUS_KM = 6371.0

        # Convert degrees to radians
        lat1_rad = math.radians(origin_latitude)
        lon1_rad = math.radians(origin_longitude)
        lat2_rad = math.radians(destination_latitude)
        lon2_rad = math.radians(destination_longitude)

        # Calculate differences
        dlat = lat2_rad - lat1_rad
        dlon = lon2_rad - lon1_rad

        # Apply Haversine formula
        # a = sin²(Δlat/2) + cos(lat1) * cos(lat2) * sin²(Δlon/2)
        a = (math.sin(dlat / 2) ** 2 + 
             math.cos(lat1_rad) * math.cos(lat2_rad) * math.sin(dlon / 2) ** 2)

        # c = 2 * atan2(√a, √(1−a))
        c = 2 * math.atan2(math.sqrt(a), math.sqrt(1 - a))

        # Calculate distance: d = R * c
        distance_km = EARTH_RADIUS_KM * c

        # Round to one decimal place for practical use
        distance_km = round(distance_km, 1)

        return {
            "distance_km": distance_km
        }

    @is_tool()
    def calculate_estimated_arrival_time(
        self,
        current_latitude: float,
        current_longitude: float,
        destination_latitude: float,
        destination_longitude: float,
        average_speed_kmh: float
    ):
        """
        Calculate estimated arrival time based on current location and destination.

        This method computes the great circle distance between two geographical points
        using the Haversine formula, then calculates the estimated arrival time based
        on the provided average speed.

        Args:
            current_latitude: Current latitude coordinate in decimal degrees
            current_longitude: Current longitude coordinate in decimal degrees
            destination_latitude: Destination latitude coordinate in decimal degrees
            destination_longitude: Destination longitude coordinate in decimal degrees
            average_speed_kmh: Average travel speed in kilometers per hour

        Returns:
            dict: Contains estimated_arrival_time (str in "yyyy-mm-dd HH:MM:SS" format)
                  and remaining_distance_km (float)

        Raises:
            ValueError: If coordinates are out of valid range or speed is non-positive
        """
        from math import radians, sin, cos, sqrt, atan2
        from datetime import datetime, timedelta

        # Validate input parameters
        if not (-90 <= current_latitude <= 90):
            raise ValueError(f"Invalid current_latitude: {current_latitude}. Must be between -90 and 90.")

        if not (-180 <= current_longitude <= 180):
            raise ValueError(f"Invalid current_longitude: {current_longitude}. Must be between -180 and 180.")

        if not (-90 <= destination_latitude <= 90):
            raise ValueError(f"Invalid destination_latitude: {destination_latitude}. Must be between -90 and 90.")

        if not (-180 <= destination_longitude <= 180):
            raise ValueError(f"Invalid destination_longitude: {destination_longitude}. Must be between -180 and 180.")

        if average_speed_kmh <= 0:
            raise ValueError(f"Invalid average_speed_kmh: {average_speed_kmh}. Must be greater than 0.")

        # Earth's radius in kilometers
        EARTH_RADIUS_KM = 6371.0

        # Convert latitude and longitude from degrees to radians
        lat1_rad = radians(current_latitude)
        lon1_rad = radians(current_longitude)
        lat2_rad = radians(destination_latitude)
        lon2_rad = radians(destination_longitude)

        # Calculate differences
        dlat = lat2_rad - lat1_rad
        dlon = lon2_rad - lon1_rad

        # Haversine formula to calculate great circle distance
        # a = sin²(Δlat/2) + cos(lat1) * cos(lat2) * sin²(Δlon/2)
        # c = 2 * atan2(√a, √(1−a))
        # distance = R * c
        a = sin(dlat / 2)**2 + cos(lat1_rad) * cos(lat2_rad) * sin(dlon / 2)**2
        c = 2 * atan2(sqrt(a), sqrt(1 - a))
        distance_km = EARTH_RADIUS_KM * c

        # Calculate travel time in hours
        travel_time_hours = distance_km / average_speed_kmh

        # Get current time and add travel time to calculate estimated arrival time
        current_time = datetime.now()
        estimated_arrival = current_time + timedelta(hours=travel_time_hours)

        # Format estimated arrival time as "yyyy-mm-dd HH:MM:SS"
        estimated_arrival_str = estimated_arrival.strftime("%Y-%m-%d %H:%M:%S")

        # Round distance to one decimal place for cleaner output
        remaining_distance_km = round(distance_km, 1)

        return {
            "estimated_arrival_time": estimated_arrival_str,
            "remaining_distance_km": remaining_distance_km
        }

    @is_tool()
    def calculate_weight_discrepancy(self, declared_weight: float, actual_weight: float, tolerance_percentage: float):
        """
        Calculate the discrepancy between declared weight and actual weight.

        This method computes the absolute and percentage difference between the declared
        and actual weights, and determines if the discrepancy exceeds the acceptable tolerance.

        Args:
            declared_weight: Declared package weight in kilograms (must be positive)
            actual_weight: Actual measured weight in kilograms (must be positive)
            tolerance_percentage: Acceptable weight tolerance as percentage (must be non-negative)

        Returns:
            dict: Contains discrepancy_kg, discrepancy_percentage, and exceeds_tolerance

        Raises:
            ValueError: If any weight value is non-positive or tolerance is negative
        """
        # Validate input parameters
        if declared_weight <= 0:
            raise ValueError("Declared weight must be a positive number")

        if actual_weight <= 0:
            raise ValueError("Actual weight must be a positive number")

        if tolerance_percentage < 0:
            raise ValueError("Tolerance percentage must be non-negative")

        # Calculate absolute weight discrepancy in kilograms
        # This represents the raw difference between actual and declared weights
        discrepancy_kg = abs(actual_weight - declared_weight)

        # Calculate percentage discrepancy relative to declared weight
        # Formula: (|actual - declared| / declared) * 100
        # This normalizes the discrepancy to make it comparable across different weight ranges
        discrepancy_percentage = (discrepancy_kg / declared_weight) * 100

        # Determine if the discrepancy exceeds the acceptable tolerance
        # Returns True if the percentage discrepancy is greater than the tolerance threshold
        exceeds_tolerance = discrepancy_percentage > tolerance_percentage

        # Return the calculated results as a dictionary
        return {
            "discrepancy_kg": round(discrepancy_kg, 2),  # Round to 2 decimal places for precision
            "discrepancy_percentage": round(discrepancy_percentage, 2),  # Round to 2 decimal places
            "exceeds_tolerance": exceeds_tolerance
        }

    @is_tool()
    def calculate_return_shipping_cost(
        self,
        original_shipping_cost: float,
        return_reason: Literal["customer_request", "damaged_goods", "wrong_item", "quality_issue", "other"],
        who_pays: Literal["sender", "recipient", "shared"],
        return_discount_percentage: float
    ) -> dict:
        """
        Calculate the cost for return shipping based on original shipment details.

        This method computes the return shipping cost by applying a discount to the original
        shipping cost, then determines the cost responsibility based on who pays for the return.

        Args:
            original_shipping_cost: Original shipping cost in currency units
            return_reason: Reason for return (must be one of the enum values)
            who_pays: Who is responsible for return shipping cost (must be one of the enum values)
            return_discount_percentage: Discount percentage for return shipping (0-100)

        Returns:
            dict: Contains return_shipping_cost, customer_responsibility, and sender_responsibility

        Raises:
            ValueError: If input parameters are invalid
        """
        # Validate original_shipping_cost
        if not isinstance(original_shipping_cost, (int, float)):
            raise ValueError("original_shipping_cost must be a number")
        if original_shipping_cost < 0:
            raise ValueError("original_shipping_cost must be non-negative")

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

        # Validate return_reason enum (safety protection for enum parameter)
        valid_return_reasons = ["customer_request", "damaged_goods", "wrong_item", "quality_issue", "other"]
        if return_reason not in valid_return_reasons:
            raise ValueError(f"return_reason must be one of {valid_return_reasons}, got '{return_reason}'")

        # Validate who_pays enum (safety protection for enum parameter)
        valid_who_pays = ["sender", "recipient", "shared"]
        if who_pays not in valid_who_pays:
            raise ValueError(f"who_pays must be one of {valid_who_pays}, got '{who_pays}'")

        # Calculate return shipping cost with discount applied
        # Formula: return_cost = original_cost * (1 - discount_percentage / 100)
        discount_multiplier = 1 - (return_discount_percentage / 100)
        return_shipping_cost = original_shipping_cost * discount_multiplier

        # Round to 2 decimal places for currency precision
        return_shipping_cost = round(return_shipping_cost, 2)

        # Determine cost responsibility based on who_pays
        customer_responsibility = 0.0
        sender_responsibility = 0.0

        if who_pays == "sender":
            # Sender pays the full return shipping cost
            sender_responsibility = return_shipping_cost
            customer_responsibility = 0.0
        elif who_pays == "recipient":
            # Recipient (customer) pays the full return shipping cost
            customer_responsibility = return_shipping_cost
            sender_responsibility = 0.0
        elif who_pays == "shared":
            # Cost is split equally between sender and recipient
            shared_cost = round(return_shipping_cost / 2, 2)
            customer_responsibility = shared_cost
            sender_responsibility = shared_cost

        # Return the calculated costs
        return {
            "return_shipping_cost": return_shipping_cost,
            "customer_responsibility": customer_responsibility,
            "sender_responsibility": sender_responsibility
        }

    @is_tool()
    def check_service_availability(
        self,
        service_type: Literal["standard", "express", "same_day", "next_day"],
        origin_latitude: float,
        origin_longitude: float,
        destination_latitude: float,
        destination_longitude: float
    ) -> dict:
        """
        Check if a specific courier service is available for given origin and destination.

        This method validates service availability based on:
        1. Service type validity (must be one of the allowed enum values)
        2. Geographic coordinate validity (latitude: -90 to 90, longitude: -180 to 180)
        3. Distance calculation between origin and destination
        4. Service-specific coverage rules

        Args:
            service_type: Type of courier service (standard/express/same_day/next_day)
            origin_latitude: Origin location latitude coordinate
            origin_longitude: Origin location longitude coordinate
            destination_latitude: Destination location latitude coordinate
            destination_longitude: Destination location longitude coordinate

        Returns:
            dict: Contains 'is_available' (bool) and 'unavailable_reason' (str or None)

        Raises:
            ValueError: If coordinates are invalid or service_type is not in allowed values
        """
        from math import radians, sin, cos, sqrt, atan2

        # Validate service_type against enum values (safety protection for enum parameter)
        allowed_service_types = ["standard", "express", "same_day", "next_day"]
        if service_type not in allowed_service_types:
            raise ValueError(
                f"Invalid service_type '{service_type}'. Must be one of: {allowed_service_types}"
            )

        # Validate latitude values (must be between -90 and 90)
        if not (-90 <= origin_latitude <= 90):
            raise ValueError(
                f"Invalid origin_latitude {origin_latitude}. Must be between -90 and 90"
            )
        if not (-90 <= destination_latitude <= 90):
            raise ValueError(
                f"Invalid destination_latitude {destination_latitude}. Must be between -90 and 90"
            )

        # Validate longitude values (must be between -180 and 180)
        if not (-180 <= origin_longitude <= 180):
            raise ValueError(
                f"Invalid origin_longitude {origin_longitude}. Must be between -180 and 180"
            )
        if not (-180 <= destination_longitude <= 180):
            raise ValueError(
                f"Invalid destination_longitude {destination_longitude}. Must be between -180 and 180"
            )

        # Calculate distance between origin and destination using Haversine formula
        # This formula calculates the great-circle distance between two points on Earth
        # given their latitude and longitude coordinates

        # Earth's radius in kilometers
        earth_radius_km = 6371.0

        # Convert latitude and longitude from degrees to radians
        lat1_rad = radians(origin_latitude)
        lon1_rad = radians(origin_longitude)
        lat2_rad = radians(destination_latitude)
        lon2_rad = radians(destination_longitude)

        # Calculate differences
        dlat = lat2_rad - lat1_rad
        dlon = lon2_rad - lon1_rad

        # Apply Haversine formula
        a = sin(dlat / 2)**2 + cos(lat1_rad) * cos(lat2_rad) * sin(dlon / 2)**2
        c = 2 * atan2(sqrt(a), sqrt(1 - a))
        distance_km = earth_radius_km * c

        # Define service coverage rules based on service type
        # These rules determine maximum distance and other constraints for each service
        service_rules = {
            "standard": {
                "max_distance_km": 5000,  # Standard service covers long distances
                "description": "Standard delivery service"
            },
            "express": {
                "max_distance_km": 2000,  # Express service has moderate range
                "description": "Express delivery service"
            },
            "same_day": {
                "max_distance_km": 100,   # Same-day delivery limited to local area
                "description": "Same-day delivery service"
            },
            "next_day": {
                "max_distance_km": 500,   # Next-day delivery covers regional area
                "description": "Next-day delivery service"
            }
        }

        # Get the rules for the requested service type
        service_rule = service_rules[service_type]
        max_distance = service_rule["max_distance_km"]

        # Check if the distance exceeds the service coverage
        if distance_km > max_distance:
            return {
                "is_available": False,
                "unavailable_reason": (
                    f"destination outside service area "
                    f"(distance: {distance_km:.2f}km exceeds {service_type} service "
                    f"maximum of {max_distance}km)"
                )
            }

        # Additional validation: Check if origin and destination are the same location
        # This might indicate an error in the request
        if distance_km < 0.1:  # Less than 100 meters
            return {
                "is_available": False,
                "unavailable_reason": (
                    "origin and destination are too close or identical "
                    "(distance less than 100 meters)"
                )
            }

        # Service is available if all checks pass
        return {
            "is_available": True,
            "unavailable_reason": None
        }

    @is_tool()
    def generate_barcode_data(self, tracking_number: str, destination_code: str, service_code: str):
        """
        Generate barcode data string for package labeling.

        This method creates a standardized barcode data string by combining the tracking number,
        destination sorting center code, and service type code. The barcode uses CODE128 format
        which is widely supported in logistics operations.

        Args:
            tracking_number: Tracking number for the shipment (e.g., "TRK20240115001")
            destination_code: Destination sorting center code (e.g., "SH-PD-01")
            service_code: Service type code (e.g., "EXP" for express)

        Returns:
            dict: A dictionary containing:
                - barcode: The encoded barcode data string combining all input parameters
                - barcode_format: The barcode format type (CODE128)

        Raises:
            ValueError: If any required parameter is empty or contains only whitespace
        """
        # Validate that all required parameters are provided and non-empty
        if not tracking_number or not tracking_number.strip():
            raise ValueError("tracking_number cannot be empty or whitespace")

        if not destination_code or not destination_code.strip():
            raise ValueError("destination_code cannot be empty or whitespace")

        if not service_code or not service_code.strip():
            raise ValueError("service_code cannot be empty or whitespace")

        # Strip whitespace from all inputs to ensure clean data
        tracking_number = tracking_number.strip()
        destination_code = destination_code.strip()
        service_code = service_code.strip()

        # Generate the barcode data string by concatenating the components with hyphens
        # Format: TRACKING_NUMBER-DESTINATION_CODE-SERVICE_CODE
        # This format is standardized for easy parsing by barcode scanners and logistics systems
        barcode_data = f"{tracking_number}-{destination_code}-{service_code}"

        # CODE128 is chosen as the barcode format because:
        # - It supports alphanumeric characters and special symbols
        # - It has high data density (can encode more information in less space)
        # - It's widely supported in logistics and shipping industries
        # - It has built-in error checking capabilities
        barcode_format = "CODE128"

        # Return the generated barcode data and format type
        return {
            "barcode": barcode_data,
            "barcode_format": barcode_format
        }

    @is_tool()
    def generate_delivery_proof(
        self,
        tracking_number: str,
        actual_delivery_time: str,
        recipient_signature: str,
        received_by: str,
        courier_id: str,
        photo_evidence: str = None
    ):
        """
        Generate delivery proof document with all verification details.

        This method creates a comprehensive delivery proof document that includes:
        - Shipment tracking information
        - Delivery timestamp
        - Recipient signature and name
        - Courier information
        - Optional photo evidence

        Args:
            tracking_number: Tracking number for the shipment
            actual_delivery_time: Actual delivery timestamp in yyyy-mm-dd HH:MM:SS format
            recipient_signature: Digital signature from recipient
            received_by: Name of person who received the package
            courier_id: Unique identifier for the courier
            photo_evidence: Optional photo evidence of delivery

        Returns:
            dict: Contains proof_id and proof_document with complete delivery details

        Raises:
            ValueError: If required parameters are invalid or missing
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters are not empty
        if not tracking_number or not tracking_number.strip():
            raise ValueError("tracking_number cannot be empty")
        if not actual_delivery_time or not actual_delivery_time.strip():
            raise ValueError("actual_delivery_time cannot be empty")
        if not recipient_signature or not recipient_signature.strip():
            raise ValueError("recipient_signature cannot be empty")
        if not received_by or not received_by.strip():
            raise ValueError("received_by cannot be empty")
        if not courier_id or not courier_id.strip():
            raise ValueError("courier_id cannot be empty")

        # Validate and parse delivery time format
        try:
            # Try to parse the datetime string to validate format
            delivery_datetime = datetime.strptime(actual_delivery_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            try:
                # Also support date-only format
                delivery_datetime = datetime.strptime(actual_delivery_time, "%Y-%m-%d")
                # Normalize to full datetime format
                actual_delivery_time = delivery_datetime.strftime("%Y-%m-%d %H:%M:%S")
            except ValueError:
                raise ValueError(
                    "actual_delivery_time must be in format 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd'"
                )

        # Generate unique proof ID using secure random hash
        proof_id = "PROOF" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Build the proof document structure
        proof_document = {
            "tracking": tracking_number,
            "delivered_at": actual_delivery_time,
            "recipient_signature": recipient_signature,
            "received_by": received_by,
            "courier_id": courier_id
        }

        # Add optional photo evidence if provided
        if photo_evidence and photo_evidence.strip():
            proof_document["photo_evidence"] = photo_evidence

        # Add metadata for document generation
        proof_document["proof_id"] = proof_id
        proof_document["generated_at"] = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Add verification status
        proof_document["verification_status"] = "verified"
        proof_document["document_type"] = "delivery_proof"

        # Return the complete proof data
        return {
            "proof_id": proof_id,
            "proof_document": proof_document
        }

    @is_tool()
    def create_shipment_order(
        self,
        sender_name: str,
        sender_phone: str,
        sender_address: str,
        recipient_name: str,
        recipient_phone: str,
        recipient_address: str,
        weight: float,
        length: float,
        width: float,
        height: float,
        service_type: Literal["standard", "express", "same_day", "next_day"],
        declared_value: float = None
    ):
        """
        Create a new shipment order with sender, recipient, and package details.

        This method validates all required parameters, generates unique identifiers,
        and stores the shipment order in the database.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required string parameters are not empty
        if not sender_name or not sender_name.strip():
            raise ValueError("Sender name cannot be empty")
        if not sender_phone or not sender_phone.strip():
            raise ValueError("Sender phone cannot be empty")
        if not sender_address or not sender_address.strip():
            raise ValueError("Sender address cannot be empty")
        if not recipient_name or not recipient_name.strip():
            raise ValueError("Recipient name cannot be empty")
        if not recipient_phone or not recipient_phone.strip():
            raise ValueError("Recipient phone cannot be empty")
        if not recipient_address or not recipient_address.strip():
            raise ValueError("Recipient address cannot be empty")

        # Validate numeric parameters (weight and dimensions must be positive)
        if weight <= 0:
            raise ValueError("Package weight must be greater than 0")
        if length <= 0:
            raise ValueError("Package length must be greater than 0")
        if width <= 0:
            raise ValueError("Package width must be greater than 0")
        if height <= 0:
            raise ValueError("Package height must be greater than 0")

        # Validate declared_value if provided
        if declared_value is not None and declared_value < 0:
            raise ValueError("Declared value cannot be negative")

        # Validate service_type against enum values (safety protection)
        valid_service_types = ["standard", "express", "same_day", "next_day"]
        if service_type not in valid_service_types:
            raise ValueError(f"Invalid service_type. Must be one of: {', '.join(valid_service_types)}")

        # Generate unique order_id with prefix "ORD"
        order_id = "ORD" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate unique tracking_number with prefix "TRK"
        tracking_number = "TRK" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for created_at
        created_at = datetime.now()

        # Access the database
        db = self.db

        # Initialize shipment_order table if it doesn't exist
        if getattr(db, "shipment_order", None) is None:
            setattr(db, "shipment_order", {})

        # Create new ShipmentOrder instance
        new_order = ShipmentOrder(
            order_id=order_id,
            tracking_number=tracking_number,
            status="pending",  # Default initial status
            sender_name=sender_name,
            sender_phone=sender_phone,
            sender_address=sender_address,
            recipient_name=recipient_name,
            recipient_phone=recipient_phone,
            recipient_address=recipient_address,
            weight=weight,
            length=length,
            width=width,
            height=height,
            service_type=service_type,
            declared_value=declared_value,
            created_at=created_at
        )

        # Get current shipment_order table data
        shipment_order_table = getattr(db, "shipment_order")

        # Add new order to the table
        shipment_order_table[order_id] = new_order

        # Update the database
        setattr(db, "shipment_order", shipment_order_table)

        # Return order details with formatted timestamp
        return {
            "order_id": order_id,
            "tracking_number": tracking_number,
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def update_courier_location(self, courier_id: str, latitude: float, longitude: float):
        """
        Update the GPS location of a courier

        Args:
            courier_id: Unique identifier for the courier
            latitude: Current latitude coordinate
            longitude: Current longitude coordinate

        Returns:
            dict: Contains 'updated' (bool) and 'updated_at' (str in yyyy-mm-dd HH:MM:SS format)

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

        # Access the database
        db = self.db

        # Get the courier table from database
        courier_table = getattr(db, "courier", None)

        # Check if courier table exists
        if courier_table is None:
            raise KeyError(f"Courier table does not exist in database")

        # Check if courier exists in the table
        if courier_id not in courier_table:
            raise KeyError(f"Courier with ID '{courier_id}' does not exist")

        # Get the courier record
        courier = courier_table[courier_id]

        # Update the courier's location coordinates
        courier.current_latitude = latitude
        courier.current_longitude = longitude

        # Update the last location update timestamp
        current_time = datetime.now()
        courier.last_location_update = current_time

        # Save the updated courier back to the database
        courier_table[courier_id] = courier
        setattr(db, "courier", courier_table)

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

        # Return the result
        return {
            "updated": True,
            "updated_at": updated_at_str
        }

    @is_tool()
    def get_courier_active_deliveries(self, courier_id: str):
        """
        Retrieve all active delivery assignments for a specific courier

        Args:
            courier_id: Unique identifier for the courier

        Returns:
            dict: Contains active_deliveries (list of order IDs) and total_count

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

        # Get courier table data
        courier_table = getattr(db, "courier", None)
        if courier_table is None:
            raise KeyError(f"Courier table not found in database")

        # Verify that the courier exists in the database
        if courier_id not in courier_table:
            raise KeyError(f"Courier with ID '{courier_id}' does not exist")

        # Get courier_assignment table data
        courier_assignment_table = getattr(db, "courier_assignment", None)
        if courier_assignment_table is None:
            # If no assignment table exists, return empty results
            return {
                "active_deliveries": [],
                "total_count": 0
            }

        # Initialize list to store active delivery order IDs
        active_delivery_order_ids = []

        # Iterate through all courier assignments to find active deliveries for this courier
        for assignment_id, assignment in courier_assignment_table.items():
            # Check if assignment belongs to the specified courier
            if assignment.courier_id != courier_id:
                continue

            # Check if assignment type is delivery (not pickup)
            if assignment.assignment_type != "delivery":
                continue

            # Check if assignment status is active (assigned or in_progress)
            # Active statuses exclude completed and cancelled
            if assignment.assignment_status in ["assigned", "in_progress"]:
                # Add the order_id to the active deliveries list
                active_delivery_order_ids.append(assignment.order_id)

        # Return the results with list of active delivery order IDs and total count
        return {
            "active_deliveries": active_delivery_order_ids,
            "total_count": len(active_delivery_order_ids)
        }

    @is_tool()
    def validate_address_format(self, address: str, country_code: str):
        """
        Validate if an address string meets the required format standards for a given country.

        This method checks various aspects of address format including:
        - Presence of required components (district, city, building/room info)
        - Proper structure and formatting
        - Country-specific validation rules

        Args:
            address: Complete address string to validate
            country_code: ISO country code for address validation rules (e.g., "CN", "US", "UK")

        Returns:
            dict: Dictionary containing:
                - is_valid (bool): Whether the address format is valid
                - validation_errors (list): List of validation error codes if any

        Raises:
            ValueError: If address is empty/None or country_code is invalid
        """
        # Input validation
        if not address or not isinstance(address, str) or not address.strip():
            raise ValueError("Address must be a non-empty string")

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

        # Normalize inputs
        address = address.strip()
        country_code = country_code.strip().upper()

        # Initialize validation result
        validation_errors = []

        # Define country-specific validation rules
        # Each country has different address format requirements
        country_rules = {
            "CN": {
                # Chinese address format: Province/City, District, Street/Building, Room
                "required_components": ["district", "building_or_room"],
                "min_length": 10,
                "common_keywords": ["省", "市", "区", "县", "路", "街", "号", "楼", "室", "栋", "单元"],
                "postal_code_pattern": r"\d{6}",  # 6-digit postal code
            },
            "US": {
                # US address format: Street Address, City, State ZIP
                "required_components": ["street", "city", "state", "zip"],
                "min_length": 15,
                "common_keywords": ["street", "st", "avenue", "ave", "road", "rd", "drive", "dr", "lane", "ln", "blvd"],
                "postal_code_pattern": r"\d{5}(-\d{4})?",  # 5-digit or ZIP+4
            },
            "UK": {
                # UK address format: Street Address, City, Postcode
                "required_components": ["street", "city", "postcode"],
                "min_length": 10,
                "common_keywords": ["street", "road", "avenue", "lane", "close", "way"],
                "postal_code_pattern": r"[A-Z]{1,2}\d{1,2}[A-Z]?\s?\d[A-Z]{2}",  # UK postcode format
            },
        }

        # Check if country code is supported
        if country_code not in country_rules:
            # For unsupported countries, perform basic validation only
            if len(address) < 10:
                validation_errors.append("address_too_short")

            # Check for at least some structural elements (numbers, commas, spaces)
            import re
            if not re.search(r'\d', address):
                validation_errors.append("missing_building_number")

            if ',' not in address and len(address.split()) < 3:
                validation_errors.append("insufficient_address_components")

            return {
                "is_valid": len(validation_errors) == 0,
                "validation_errors": validation_errors
            }

        # Get rules for the specified country
        rules = country_rules[country_code]

        # Check minimum length
        if len(address) < rules["min_length"]:
            validation_errors.append("address_too_short")

        # Country-specific validation
        if country_code == "CN":
            # Chinese address validation
            import re

            # Check for district/county component
            has_district = any(keyword in address for keyword in ["区", "县"])
            if not has_district:
                validation_errors.append("missing_district")

            # Check for building/room component
            has_building = any(keyword in address for keyword in ["号", "楼", "室", "栋", "单元"])
            if not has_building:
                validation_errors.append("missing_building_or_room_info")

            # Check for numbers (building/room numbers)
            if not re.search(r'\d', address):
                validation_errors.append("missing_building_number")

            # Optional: Check for postal code if present
            postal_match = re.search(rules["postal_code_pattern"], address)
            if address.replace(" ", "").isdigit() and len(address.replace(" ", "")) == 6:
                # If address contains only 6 digits, it might be just a postal code
                validation_errors.append("incomplete_address_only_postal_code")

        elif country_code == "US":
            # US address validation
            import re

            # Check for street component
            has_street = any(keyword.lower() in address.lower() for keyword in rules["common_keywords"])
            if not has_street and not re.search(r'\d+\s+\w+', address):
                validation_errors.append("missing_street_address")

            # Check for city (look for comma separation typical in US addresses)
            comma_count = address.count(',')
            if comma_count < 1:
                validation_errors.append("missing_city_or_state")

            # Check for ZIP code
            postal_match = re.search(rules["postal_code_pattern"], address)
            if not postal_match:
                validation_errors.append("missing_or_invalid_zip_code")

            # Check for state abbreviation (2 uppercase letters)
            if not re.search(r'\b[A-Z]{2}\b', address):
                validation_errors.append("missing_state_code")

        elif country_code == "UK":
            # UK address validation
            import re

            # Check for street component
            has_street = any(keyword.lower() in address.lower() for keyword in rules["common_keywords"])
            if not has_street:
                validation_errors.append("missing_street_name")

            # Check for postcode
            postal_match = re.search(rules["postal_code_pattern"], address, re.IGNORECASE)
            if not postal_match:
                validation_errors.append("missing_or_invalid_postcode")

            # Check for building number
            if not re.search(r'^\d+\s+|,\s*\d+\s+', address):
                validation_errors.append("missing_building_number")

        # Additional general validations
        # Check for suspicious patterns
        if address.count(',') == 0 and address.count(' ') < 2:
            validation_errors.append("insufficient_address_structure")

        # Check for excessive special characters
        import re
        special_char_count = len(re.findall(r'[^\w\s,.\-/]', address))
        if special_char_count > len(address) * 0.2:  # More than 20% special characters
            validation_errors.append("excessive_special_characters")

        # Return validation result
        return {
            "is_valid": len(validation_errors) == 0,
            "validation_errors": validation_errors
        }

    @is_tool()
    def generate_performance_metrics(
        self,
        total_deliveries: int,
        successful_deliveries: int,
        on_time_deliveries: int,
        total_distance_km: float,
        total_time_hours: float,
        customer_ratings: list
    ) -> dict:
        """
        Generate performance metrics for courier or service evaluation.

        This method calculates various performance indicators based on delivery statistics,
        including success rate, on-time rate, average speed, customer satisfaction, and
        an overall efficiency score.

        Args:
            total_deliveries: Total number of deliveries attempted
            successful_deliveries: Number of successful first-attempt deliveries
            on_time_deliveries: Number of on-time deliveries
            total_distance_km: Total distance covered in kilometers
            total_time_hours: Total time spent on deliveries in hours
            customer_ratings: List of customer rating scores (1-5)

        Returns:
            dict: Performance metrics including success_rate, on_time_rate, 
                  average_speed_kmh, average_customer_rating, and efficiency_score

        Raises:
            ValueError: If input parameters are invalid (negative values, empty lists, etc.)
        """
        # Validate input parameters
        if total_deliveries <= 0:
            raise ValueError("total_deliveries must be greater than 0")

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

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

        if successful_deliveries > total_deliveries:
            raise ValueError("successful_deliveries cannot exceed total_deliveries")

        if on_time_deliveries > total_deliveries:
            raise ValueError("on_time_deliveries cannot exceed total_deliveries")

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

        if total_time_hours <= 0:
            raise ValueError("total_time_hours must be greater than 0")

        if not customer_ratings or len(customer_ratings) == 0:
            raise ValueError("customer_ratings cannot be empty")

        # Validate customer ratings are within valid range (1-5)
        for rating in customer_ratings:
            if not isinstance(rating, (int, float)):
                raise ValueError("All customer ratings must be numeric")
            if rating < 1 or rating > 5:
                raise ValueError("Customer ratings must be between 1 and 5")

        # Calculate success rate (percentage of successful first-attempt deliveries)
        success_rate = round((successful_deliveries / total_deliveries) * 100, 2)

        # Calculate on-time delivery rate (percentage of deliveries completed on time)
        on_time_rate = round((on_time_deliveries / total_deliveries) * 100, 2)

        # Calculate average delivery speed in kilometers per hour
        average_speed_kmh = round(total_distance_km / total_time_hours, 2)

        # Calculate average customer rating score
        average_customer_rating = round(sum(customer_ratings) / len(customer_ratings), 2)

        # Calculate overall efficiency score (0-100)
        # The efficiency score is a weighted combination of key performance indicators:
        # - Success rate contributes 30% (normalized to 0-30 range)
        # - On-time rate contributes 30% (normalized to 0-30 range)
        # - Customer rating contributes 25% (normalized from 1-5 scale to 0-25 range)
        # - Speed efficiency contributes 15% (normalized based on reasonable speed expectations)

        # Normalize success rate component (0-30 points)
        success_component = (success_rate / 100) * 30

        # Normalize on-time rate component (0-30 points)
        on_time_component = (on_time_rate / 100) * 30

        # Normalize customer rating component (0-25 points)
        # Convert from 1-5 scale to 0-1 scale: (rating - 1) / 4
        rating_component = ((average_customer_rating - 1) / 4) * 25

        # Normalize speed component (0-15 points)
        # Assume optimal delivery speed is around 15-20 km/h for urban courier services
        # Speed above 20 km/h gets full points, below 5 km/h gets minimal points
        optimal_speed = 20.0
        min_speed = 5.0
        if average_speed_kmh >= optimal_speed:
            speed_component = 15.0
        elif average_speed_kmh <= min_speed:
            speed_component = 0.0
        else:
            # Linear interpolation between min_speed and optimal_speed
            speed_component = ((average_speed_kmh - min_speed) / (optimal_speed - min_speed)) * 15

        # Calculate final efficiency score
        efficiency_score = round(
            success_component + on_time_component + rating_component + speed_component,
            2
        )

        # Ensure efficiency score is within 0-100 range
        efficiency_score = max(0.0, min(100.0, efficiency_score))

        # Return performance metrics dictionary
        return {
            "success_rate": success_rate,
            "on_time_rate": on_time_rate,
            "average_speed_kmh": average_speed_kmh,
            "average_customer_rating": average_customer_rating,
            "efficiency_score": efficiency_score
        }

    @is_tool()
    def geocode_address(self, address: str):
        """
        Convert a text address to geographical coordinates using a geocoding algorithm.

        This method takes a complete address string and returns its geographical coordinates
        (latitude and longitude) along with an accuracy level. The implementation uses a
        deterministic hash-based approach to simulate geocoding without external API dependencies.

        Args:
            address: Complete address string to geocode

        Returns:
            dict: A dictionary containing:
                - latitude (float): Latitude coordinate of the address
                - longitude (float): Longitude coordinate of the address
                - accuracy (str): Geocoding accuracy level ("high", "medium", or "low")

        Raises:
            ValueError: If address is empty or invalid
        """
        import hashlib

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

        # Strip whitespace and check if address is meaningful
        address_cleaned = address.strip()
        if not address_cleaned:
            raise ValueError("Address cannot be empty or contain only whitespace")

        # Use hash-based deterministic geocoding simulation
        # This ensures the same address always returns the same coordinates
        address_hash = hashlib.sha256(address_cleaned.encode('utf-8')).hexdigest()

        # Extract numeric values from hash to generate coordinates
        # Use different parts of hash for latitude and longitude
        lat_hash = int(address_hash[:16], 16)
        lon_hash = int(address_hash[16:32], 16)

        # Generate latitude between -90 and 90
        # Normalize hash value to latitude range
        latitude = -90 + (lat_hash % 180000000) / 1000000.0

        # Generate longitude between -180 and 180
        # Normalize hash value to longitude range
        longitude = -180 + (lon_hash % 360000000) / 1000000.0

        # Determine accuracy based on address characteristics
        # More detailed addresses get higher accuracy ratings
        address_lower = address_cleaned.lower()

        # High accuracy: contains specific identifiers like room/building/unit numbers
        if any(keyword in address_lower for keyword in ['room', 'building', 'unit', 'floor', 'suite', 'apt', 'apartment']):
            accuracy = "high"
        # Medium accuracy: contains street-level information
        elif any(keyword in address_lower for keyword in ['street', 'road', 'avenue', 'boulevard', 'lane', 'drive']):
            accuracy = "medium"
        # Low accuracy: only general location information
        else:
            accuracy = "low"

        # Return geocoded result
        return {
            "latitude": round(latitude, 4),  # Round to 4 decimal places (~11 meters precision)
            "longitude": round(longitude, 4),
            "accuracy": accuracy
        }

    @is_tool()
    def validate_package_dimensions(
        self,
        length: float,
        width: float,
        height: float,
        weight: float,
        service_type: Literal["standard", "express", "same_day", "next_day"]
    ) -> dict:
        """
        Validate if package dimensions meet courier service requirements and restrictions.

        This method checks package dimensions and weight against predefined limits for different
        service types. It returns validation status and detailed violation reasons if any.

        Args:
            length: Package length in centimeters
            width: Package width in centimeters
            height: Package height in centimeters
            weight: Package weight in kilograms
            service_type: Type of courier service (must be one of: standard, express, same_day, next_day)

        Returns:
            dict: Validation result containing:
                - is_valid (bool): Whether package dimensions are valid
                - violation_reasons (list): List of violation reasons if validation fails

        Raises:
            ValueError: If input parameters are invalid (negative dimensions, invalid service type)
        """

        # Input validation - check for negative or zero values
        if length <= 0 or width <= 0 or height <= 0:
            raise ValueError("Package dimensions (length, width, height) must be positive numbers")

        if weight <= 0:
            raise ValueError("Package weight must be a positive number")

        # Validate service_type against enum values (additional safety check)
        valid_service_types = ["standard", "express", "same_day", "next_day"]
        if service_type not in valid_service_types:
            raise ValueError(f"Invalid service_type. Must be one of: {', '.join(valid_service_types)}")

        # Define service-specific dimension and weight restrictions
        # These restrictions are based on typical courier service requirements
        service_restrictions = {
            "standard": {
                "max_weight": 30.0,  # kg
                "max_length": 120.0,  # cm
                "max_width": 80.0,  # cm
                "max_height": 80.0,  # cm
                "max_girth": 300.0,  # cm (length + 2*(width + height))
                "max_volume": 500000.0  # cubic cm
            },
            "express": {
                "max_weight": 25.0,  # kg
                "max_length": 100.0,  # cm
                "max_width": 70.0,  # cm
                "max_height": 70.0,  # cm
                "max_girth": 250.0,  # cm
                "max_volume": 400000.0  # cubic cm
            },
            "same_day": {
                "max_weight": 15.0,  # kg
                "max_length": 80.0,  # cm
                "max_width": 60.0,  # cm
                "max_height": 60.0,  # cm
                "max_girth": 200.0,  # cm
                "max_volume": 250000.0  # cubic cm
            },
            "next_day": {
                "max_weight": 20.0,  # kg
                "max_length": 90.0,  # cm
                "max_width": 65.0,  # cm
                "max_height": 65.0,  # cm
                "max_girth": 230.0,  # cm
                "max_volume": 350000.0  # cubic cm
            }
        }

        # Get restrictions for the specified service type
        restrictions = service_restrictions[service_type]

        # Calculate package metrics
        volume = length * width * height  # cubic cm
        girth = length + 2 * (width + height)  # cm

        # Initialize validation result
        violation_reasons = []

        # Check weight restriction
        if weight > restrictions["max_weight"]:
            violation_reasons.append(
                f"exceeds_weight_limit: {weight}kg exceeds maximum {restrictions['max_weight']}kg for {service_type} service"
            )

        # Check individual dimension restrictions
        if length > restrictions["max_length"]:
            violation_reasons.append(
                f"exceeds_length_limit: {length}cm exceeds maximum {restrictions['max_length']}cm for {service_type} service"
            )

        if width > restrictions["max_width"]:
            violation_reasons.append(
                f"exceeds_width_limit: {width}cm exceeds maximum {restrictions['max_width']}cm for {service_type} service"
            )

        if height > restrictions["max_height"]:
            violation_reasons.append(
                f"exceeds_height_limit: {height}cm exceeds maximum {restrictions['max_height']}cm for {service_type} service"
            )

        # Check girth restriction (length + 2*(width + height))
        if girth > restrictions["max_girth"]:
            violation_reasons.append(
                f"exceeds_girth_limit: {girth:.2f}cm exceeds maximum {restrictions['max_girth']}cm for {service_type} service"
            )

        # Check volume restriction
        if volume > restrictions["max_volume"]:
            violation_reasons.append(
                f"exceeds_volume_limit: {volume:.2f}cm³ exceeds maximum {restrictions['max_volume']}cm³ for {service_type} service"
            )

        # Determine if package is valid (no violations found)
        is_valid = len(violation_reasons) == 0

        # Return validation result
        return {
            "is_valid": is_valid,
            "violation_reasons": violation_reasons
        }

    @is_tool()
    def calculate_carbon_footprint(self, distance_km: float, transport_mode: Literal["road", "air", "rail", "sea"], weight: float):
        """
        Calculate the estimated carbon footprint for a shipment based on distance, transport mode, and weight.

        This method uses industry-standard carbon emission factors for different transport modes.
        The calculation formula is: carbon_emissions = distance_km * weight * emission_factor

        Args:
            distance_km: Transportation distance in kilometers (must be positive)
            transport_mode: Mode of transportation (must be one of: "road", "air", "rail", "sea")
            weight: Package weight in kilograms (must be positive)

        Returns:
            dict: A dictionary containing:
                - carbon_emissions_kg: Estimated carbon emissions in kilograms of CO2

        Raises:
            ValueError: If distance_km or weight is not positive, or if transport_mode is invalid
        """

        # Input validation: ensure distance_km is positive
        if distance_km <= 0:
            raise ValueError("distance_km must be a positive number")

        # Input validation: ensure weight is positive
        if weight <= 0:
            raise ValueError("weight must be a positive number")

        # Input validation: ensure transport_mode is one of the allowed values
        # Note: Literal type annotation already restricts input, but we add safety check
        allowed_modes = ["road", "air", "rail", "sea"]
        if transport_mode not in allowed_modes:
            raise ValueError(f"transport_mode must be one of {allowed_modes}, got '{transport_mode}'")

        # Carbon emission factors (kg CO2 per ton-kilometer) for different transport modes
        # These are industry-standard average values based on environmental research
        # Source: Various environmental agencies and logistics industry standards
        emission_factors = {
            "road": 0.062,      # Road transport (trucks): ~62g CO2 per ton-km
            "air": 0.602,       # Air freight: ~602g CO2 per ton-km (highest emissions)
            "rail": 0.022,      # Rail transport: ~22g CO2 per ton-km (most efficient for land)
            "sea": 0.008        # Sea freight: ~8g CO2 per ton-km (most efficient overall)
        }

        # Get the emission factor for the selected transport mode
        emission_factor = emission_factors[transport_mode]

        # Calculate carbon emissions
        # Formula: emissions (kg CO2) = distance (km) × weight (kg) × emission_factor (kg CO2 per ton-km)
        # Note: emission_factor is per ton-km, so we divide weight by 1000 to convert kg to tons
        carbon_emissions_kg = distance_km * (weight / 1000.0) * emission_factor

        # Round to 2 decimal places for practical reporting
        carbon_emissions_kg = round(carbon_emissions_kg, 2)

        # Return the calculated carbon footprint
        return {
            "carbon_emissions_kg": carbon_emissions_kg
        }

    @is_tool()
    def calculate_package_handling_fee(
        self,
        is_fragile: bool,
        requires_signature: bool,
        is_oversized: bool,
        requires_refrigeration: bool,
        base_handling_fee: float
    ) -> dict:
        """
        Calculate additional handling fee for packages with special requirements.

        This method computes the total handling fee by applying multipliers and additional charges
        based on various special handling requirements. Each requirement adds a specific cost
        to the base handling fee.

        Args:
            is_fragile: Whether package contains fragile items (adds 100% of base fee)
            requires_signature: Whether signature is required for delivery (adds 50% of base fee)
            is_oversized: Whether package exceeds standard dimensions (adds 150% of base fee)
            requires_refrigeration: Whether package requires temperature control (adds 200% of base fee)
            base_handling_fee: Base handling fee in currency units

        Returns:
            dict: Dictionary containing:
                - total_handling_fee (float): Total calculated handling fee
                - fee_breakdown (dict): Breakdown of handling fees by category

        Raises:
            ValueError: If base_handling_fee is negative or any boolean parameter is not a valid boolean type
        """
        # Validate base_handling_fee
        if not isinstance(base_handling_fee, (int, float)):
            raise ValueError("base_handling_fee must be a numeric value")

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

        # Validate boolean parameters
        if not isinstance(is_fragile, bool):
            raise ValueError("is_fragile must be a boolean value")
        if not isinstance(requires_signature, bool):
            raise ValueError("requires_signature must be a boolean value")
        if not isinstance(is_oversized, bool):
            raise ValueError("is_oversized must be a boolean value")
        if not isinstance(requires_refrigeration, bool):
            raise ValueError("requires_refrigeration must be a boolean value")

        # Initialize fee breakdown dictionary
        fee_breakdown = {}

        # Start with base handling fee
        total_fee = base_handling_fee

        # Add fragile item handling fee (100% of base fee)
        if is_fragile:
            fragile_fee = base_handling_fee * 1.0
            fee_breakdown["fragile"] = fragile_fee
            total_fee += fragile_fee

        # Add signature requirement fee (50% of base fee)
        if requires_signature:
            signature_fee = base_handling_fee * 0.5
            fee_breakdown["signature"] = signature_fee
            total_fee += signature_fee

        # Add oversized package handling fee (150% of base fee)
        if is_oversized:
            oversized_fee = base_handling_fee * 1.5
            fee_breakdown["oversized"] = oversized_fee
            total_fee += oversized_fee

        # Add refrigeration requirement fee (200% of base fee)
        if requires_refrigeration:
            refrigeration_fee = base_handling_fee * 2.0
            fee_breakdown["refrigeration"] = refrigeration_fee
            total_fee += refrigeration_fee

        # Round total fee to 2 decimal places for currency precision
        total_fee = round(total_fee, 2)

        # Round each fee in breakdown to 2 decimal places
        for key in fee_breakdown:
            fee_breakdown[key] = round(fee_breakdown[key], 2)

        # Return result dictionary
        return {
            "total_handling_fee": total_fee,
            "fee_breakdown": fee_breakdown
        }

    @is_tool()
    def calculate_delivery_density(
        self,
        center_latitude: float,
        center_longitude: float,
        radius_km: float,
        delivery_count: int
    ) -> dict:
        """
        Calculate delivery density in a specific geographical area.

        This method computes the delivery density per square kilometer based on the
        area defined by a center point and radius, along with the number of deliveries.
        It also classifies the density into predefined categories.

        Args:
            center_latitude: Center point latitude of the area
            center_longitude: Center point longitude of the area
            radius_km: Radius of the area in kilometers
            delivery_count: Number of deliveries in the area

        Returns:
            dict: Contains density_per_sq_km and density_classification

        Raises:
            ValueError: If input parameters are invalid
        """
        import math

        # Validate input parameters
        if not isinstance(center_latitude, (int, float)):
            raise ValueError("center_latitude must be a number")
        if not isinstance(center_longitude, (int, float)):
            raise ValueError("center_longitude must be a number")
        if not isinstance(radius_km, (int, float)) or radius_km <= 0:
            raise ValueError("radius_km must be a positive number")
        if not isinstance(delivery_count, int) or delivery_count < 0:
            raise ValueError("delivery_count must be a non-negative integer")

        # Validate latitude range (-90 to 90)
        if center_latitude < -90 or center_latitude > 90:
            raise ValueError("center_latitude must be between -90 and 90 degrees")

        # Validate longitude range (-180 to 180)
        if center_longitude < -180 or center_longitude > 180:
            raise ValueError("center_longitude must be between -180 and 180 degrees")

        # Calculate the area of the circular region
        # Area = π * r²
        area_sq_km = math.pi * (radius_km ** 2)

        # Calculate delivery density per square kilometer
        # Density = number of deliveries / area
        if area_sq_km == 0:
            raise ValueError("Calculated area is zero, cannot compute density")

        density_per_sq_km = delivery_count / area_sq_km

        # Round to 2 decimal places for cleaner output
        density_per_sq_km = round(density_per_sq_km, 2)

        # Classify density based on predefined thresholds
        # These thresholds can be adjusted based on business requirements
        # sparse: < 2 deliveries per sq km
        # moderate: 2-10 deliveries per sq km
        # dense: 10-50 deliveries per sq km
        # very_dense: >= 50 deliveries per sq km
        if density_per_sq_km < 2:
            density_classification = "sparse"
        elif density_per_sq_km < 10:
            density_classification = "moderate"
        elif density_per_sq_km < 50:
            density_classification = "dense"
        else:
            density_classification = "very_dense"

        # Return the result as a dictionary
        return {
            "density_per_sq_km": density_per_sq_km,
            "density_classification": density_classification
        }

    @is_tool()
    def reverse_geocode_coordinates(self, latitude: float, longitude: float):
        """
        Convert geographical coordinates to a text address.

        This method simulates reverse geocoding by using a predefined mapping of coordinate ranges
        to address components. In a real implementation, this would call an external geocoding API.

        Args:
            latitude: Latitude coordinate (valid range: -90 to 90)
            longitude: Longitude coordinate (valid range: -180 to 180)

        Returns:
            dict: Dictionary containing:
                - address: Formatted address string
                - city: City name
                - district: District or area name

        Raises:
            ValueError: If coordinates are invalid or out of valid range
        """
        from typing import Dict, Tuple

        # Validate latitude range
        if not isinstance(latitude, (int, float)):
            raise ValueError(f"Latitude must be a number, got {type(latitude).__name__}")
        if not isinstance(longitude, (int, float)):
            raise ValueError(f"Longitude must be a number, got {type(longitude).__name__}")

        if latitude < -90 or latitude > 90:
            raise ValueError(f"Latitude must be between -90 and 90, got {latitude}")

        # Validate longitude range
        if longitude < -180 or longitude > 180:
            raise ValueError(f"Longitude must be between -180 and 180, got {longitude}")

        # Define coordinate-to-location mapping for common regions
        # This simulates a geocoding database with approximate coordinate ranges
        location_mapping = [
            # Beijing regions
            ((39.8, 40.0), (116.2, 116.6), "Beijing", "Chaoyang District"),
            ((39.8, 40.0), (116.1, 116.3), "Beijing", "Haidian District"),
            ((39.8, 40.0), (116.3, 116.5), "Beijing", "Dongcheng District"),
            ((39.7, 39.9), (116.2, 116.5), "Beijing", "Fengtai District"),

            # Shanghai regions
            ((31.1, 31.3), (121.3, 121.6), "Shanghai", "Pudong New Area"),
            ((31.2, 31.4), (121.4, 121.5), "Shanghai", "Huangpu District"),
            ((31.1, 31.3), (121.2, 121.4), "Shanghai", "Xuhui District"),

            # Guangzhou regions
            ((23.0, 23.2), (113.1, 113.4), "Guangzhou", "Tianhe District"),
            ((23.0, 23.2), (113.2, 113.3), "Guangzhou", "Yuexiu District"),

            # Shenzhen regions
            ((22.5, 22.6), (113.9, 114.1), "Shenzhen", "Futian District"),
            ((22.5, 22.6), (114.0, 114.2), "Shenzhen", "Luohu District"),

            # Chengdu regions
            ((30.5, 30.7), (104.0, 104.2), "Chengdu", "Jinjiang District"),
            ((30.6, 30.8), (104.0, 104.1), "Chengdu", "Wuhou District"),

            # Hangzhou regions
            ((30.2, 30.3), (120.1, 120.2), "Hangzhou", "Xihu District"),
            ((30.2, 30.4), (120.1, 120.3), "Hangzhou", "Gongshu District"),

            # Wuhan regions
            ((30.5, 30.6), (114.2, 114.4), "Wuhan", "Wuchang District"),
            ((30.5, 30.7), (114.2, 114.3), "Wuhan", "Hankou District"),

            # Xi'an regions
            ((34.2, 34.3), (108.9, 109.0), "Xi'an", "Beilin District"),
            ((34.2, 34.4), (108.8, 109.0), "Xi'an", "Yanta District"),
        ]

        # Search for matching location based on coordinates
        matched_city = None
        matched_district = None

        for lat_range, lon_range, city, district in location_mapping:
            if (lat_range[0] <= latitude <= lat_range[1] and 
                lon_range[0] <= longitude <= lon_range[1]):
                matched_city = city
                matched_district = district
                break

        # If no exact match found, provide generic location based on coordinate ranges
        if matched_city is None:
            # Determine general region based on latitude/longitude
            if 35 <= latitude <= 45 and 110 <= longitude <= 125:
                # Northern China region
                matched_city = "Northern China"
                matched_district = "Unknown District"
            elif 22 <= latitude <= 35 and 110 <= longitude <= 125:
                # Southern China region
                matched_city = "Southern China"
                matched_district = "Unknown District"
            elif 30 <= latitude <= 50 and 100 <= longitude <= 110:
                # Western China region
                matched_city = "Western China"
                matched_district = "Unknown District"
            else:
                # Generic location for coordinates outside common Chinese regions
                matched_city = "Unknown City"
                matched_district = "Unknown District"

        # Format the complete address string
        address = f"{matched_district}, {matched_city}"

        # Return the geocoded result
        return {
            "address": address,
            "city": matched_city,
            "district": matched_district
        }

    @is_tool()
    def calculate_customs_duty(
        self,
        declared_value: float,
        item_category: str,
        origin_country: str,
        destination_country: str
    ) -> dict:
        """
        Calculate estimated customs duty for international shipments.

        This method computes the customs duty based on declared value, item category,
        and the origin-destination country pair. Different item categories and country
        pairs have different tax rates.

        Args:
            declared_value: Declared value of package contents in currency units
            item_category: Category of items for customs classification
            origin_country: Origin country code (e.g., "CN", "US", "UK")
            destination_country: Destination country code (e.g., "US", "CN", "UK")

        Returns:
            dict: Contains duty_amount (estimated customs duty) and tax_rate (applied rate)

        Raises:
            ValueError: If declared_value is negative or zero, or if required parameters are invalid
        """
        # Validate declared_value
        if declared_value <= 0:
            raise ValueError("Declared value must be greater than zero")

        # Validate required string parameters
        if not item_category or not isinstance(item_category, str):
            raise ValueError("Item category must be a non-empty string")
        if not origin_country or not isinstance(origin_country, str):
            raise ValueError("Origin country must be a non-empty string")
        if not destination_country or not isinstance(destination_country, str):
            raise ValueError("Destination country must be a non-empty string")

        # Normalize inputs to uppercase for consistent matching
        item_category_normalized = item_category.strip().lower()
        origin_normalized = origin_country.strip().upper()
        destination_normalized = destination_country.strip().upper()

        # Define base tax rates by item category (as percentage)
        # These rates represent typical customs duty rates for different categories
        category_base_rates = {
            "electronics": 10.0,
            "clothing": 15.0,
            "textiles": 15.0,
            "apparel": 15.0,
            "food": 5.0,
            "beverages": 8.0,
            "cosmetics": 12.0,
            "beauty": 12.0,
            "books": 0.0,
            "documents": 0.0,
            "toys": 8.0,
            "furniture": 6.0,
            "jewelry": 20.0,
            "watches": 18.0,
            "automotive": 7.0,
            "machinery": 5.0,
            "medical": 3.0,
            "pharmaceuticals": 3.0,
            "sports": 9.0,
            "musical_instruments": 4.0,
            "art": 0.0,
            "antiques": 0.0
        }

        # Get base rate for the item category, default to 10% if category not found
        base_rate = category_base_rates.get(item_category_normalized, 10.0)

        # Define country-specific adjustment factors
        # These factors adjust the base rate based on trade agreements and policies
        # Positive values increase duty, negative values decrease it
        country_pair_adjustments = {
            # US as destination
            ("CN", "US"): 5.0,  # Additional tariffs on Chinese goods
            ("MX", "US"): -5.0,  # USMCA trade agreement benefits
            ("CA", "US"): -5.0,  # USMCA trade agreement benefits
            ("EU", "US"): 0.0,
            ("UK", "US"): 0.0,
            ("JP", "US"): 2.0,

            # China as destination
            ("US", "CN"): 3.0,
            ("EU", "CN"): 2.0,
            ("UK", "CN"): 2.0,
            ("JP", "CN"): 1.0,

            # EU as destination
            ("CN", "EU"): 3.0,
            ("US", "EU"): 0.0,
            ("UK", "EU"): -3.0,  # Post-Brexit adjustments

            # UK as destination
            ("CN", "UK"): 4.0,
            ("US", "UK"): 0.0,
            ("EU", "UK"): 2.0,  # Post-Brexit adjustments

            # Japan as destination
            ("CN", "JP"): 2.0,
            ("US", "JP"): 0.0,
            ("EU", "JP"): -2.0,  # Trade agreement benefits
        }

        # Check if origin is within EU (simplified list of major EU countries)
        eu_countries = {"DE", "FR", "IT", "ES", "NL", "BE", "PL", "SE", "AT", "DK", "FI", "IE", "PT", "GR", "CZ", "RO", "HU"}

        # Normalize origin/destination to EU if they're EU member states
        origin_for_lookup = "EU" if origin_normalized in eu_countries else origin_normalized
        destination_for_lookup = "EU" if destination_normalized in eu_countries else destination_normalized

        # Get country pair adjustment, default to 0 if pair not found
        country_adjustment = country_pair_adjustments.get(
            (origin_for_lookup, destination_for_lookup), 0.0
        )

        # Calculate final tax rate
        final_tax_rate = base_rate + country_adjustment

        # Ensure tax rate is non-negative
        final_tax_rate = max(0.0, final_tax_rate)

        # Apply duty-free threshold (common in many countries)
        # Shipments below $50 are often duty-free
        duty_free_threshold = 50.0
        if declared_value < duty_free_threshold:
            final_tax_rate = 0.0

        # Calculate duty amount
        duty_amount = round(declared_value * (final_tax_rate / 100.0), 2)

        # Round tax rate to 2 decimal places for consistency
        final_tax_rate = round(final_tax_rate, 2)

        return {
            "duty_amount": duty_amount,
            "tax_rate": final_tax_rate
        }

    @is_tool()
    def get_shipment_order_details(self, order_id: str):
        """
        Retrieve complete details of a shipment order by order ID.

        This method fetches all information about a specific shipment order from the database
        and returns it in a formatted dictionary with all relevant details.

        Args:
            order_id: Unique identifier for the shipment order to retrieve

        Returns:
            Dictionary containing complete shipment order details including:
            - order_id, tracking_number, status
            - sender information (name, phone, address)
            - recipient information (name, phone, address)
            - package details (weight, service_type)
            - cost and timestamp information

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

        # Retrieve the shipment_order table from database
        shipment_order_table = getattr(db, "shipment_order", None)

        # Check if the table exists
        if shipment_order_table is None:
            raise KeyError(f"Shipment order table not found in database")

        # Check if the order_id exists in the table
        if order_id not in shipment_order_table:
            raise KeyError(f"Order ID '{order_id}' not found in shipment orders")

        # Retrieve the order object
        order = shipment_order_table[order_id]

        # Format the created_at timestamp to required format "yyyy-mm-dd HH:MM:SS"
        created_at_str = order.created_at.strftime("%Y-%m-%d %H:%M:%S")

        # Construct the return dictionary with all required fields
        # Map database field 'weight' to return field 'package_weight'
        order_details = {
            "order_id": order.order_id,
            "tracking_number": order.tracking_number,
            "status": order.status,
            "sender_name": order.sender_name,
            "sender_phone": order.sender_phone,
            "sender_address": order.sender_address,
            "recipient_name": order.recipient_name,
            "recipient_phone": order.recipient_phone,
            "recipient_address": order.recipient_address,
            "package_weight": order.weight,  # Note: database field is 'weight', return field is 'package_weight'
            "service_type": order.service_type,
            "total_cost": order.total_cost,
            "created_at": created_at_str
        }

        return order_details

    @is_tool()
    def record_pickup_completion(self, order_id: str, courier_id: str, pickup_time: str, pickup_signature: Optional[str] = None):
        """
        Record the completion of a package pickup by a courier.

        This method creates a pickup record and updates the shipment order status to 'picked_up'.

        Args:
            order_id: Unique identifier for the shipment order
            courier_id: Unique identifier for the courier
            pickup_time: Actual pickup timestamp in yyyy-mm-dd HH:MM:SS format
            pickup_signature: Optional digital signature or confirmation code from sender

        Returns:
            dict: Contains pickup_record_id and recorded_at timestamp

        Raises:
            KeyError: If order_id does not exist in the database
            ValueError: If pickup_time format is invalid or order status is not valid for pickup
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate and parse pickup_time format
        try:
            pickup_datetime = datetime.strptime(pickup_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid pickup_time format: {pickup_time}. Expected format: yyyy-mm-dd HH:MM:SS")

        # Get shipment_order table
        shipment_order_table = getattr(db, "shipment_order", None)
        if shipment_order_table is None:
            raise KeyError(f"shipment_order table does not exist in database")

        # Check if order exists
        if order_id not in shipment_order_table:
            raise KeyError(f"Order with order_id '{order_id}' does not exist")

        # Get the shipment order
        order = shipment_order_table[order_id]

        # Validate order status - only pending orders can be picked up
        # Orders already picked up or in other states should not be picked up again
        if order.status != "pending":
            raise ValueError(f"Order {order_id} cannot be picked up. Current status: {order.status}. Only 'pending' orders can be picked up.")

        # Generate unique pickup_record_id
        prefix = "PKP"
        pickup_record_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for record creation
        recorded_at = datetime.now()

        # Create pickup record
        pickup_record = PickupRecord(
            pickup_record_id=pickup_record_id,
            order_id=order_id,
            courier_id=courier_id,
            pickup_time=pickup_datetime,
            pickup_signature=pickup_signature,
            created_at=recorded_at
        )

        # Get pickup_record table
        pickup_record_table = getattr(db, "pickup_record", None)
        if pickup_record_table is None:
            # Initialize pickup_record table if it doesn't exist
            pickup_record_table = {}

        # Add pickup record to database
        pickup_record_table[pickup_record_id] = pickup_record
        setattr(db, "pickup_record", pickup_record_table)

        # Update shipment order status to 'picked_up'
        order.status = "picked_up"
        order.updated_at = recorded_at

        # Update shipment order in database
        shipment_order_table[order_id] = order
        setattr(db, "shipment_order", shipment_order_table)

        # Return pickup record information
        return {
            "pickup_record_id": pickup_record_id,
            "recorded_at": recorded_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def record_delivery_completion(
        self,
        order_id: str,
        courier_id: str,
        actual_delivery_time: str,
        recipient_signature: str,
        received_by: str
    ):
        """
        Record the completion of a package delivery to recipient.

        This method creates a delivery record and updates the shipment order status to 'delivered'.
        It validates the existence of the order, checks the order status, and creates a new
        delivery record with all provided information.

        Args:
            order_id: Unique identifier for the shipment order
            courier_id: Unique identifier for the courier who performed the delivery
            actual_delivery_time: Actual delivery timestamp in "yyyy-mm-dd HH:MM:SS" format
            recipient_signature: Digital signature or confirmation code from recipient
            received_by: Name of the person who received the package

        Returns:
            dict: Contains delivery_record_id and recorded_at timestamp

        Raises:
            KeyError: If the order_id does not exist in the database
            ValueError: If actual_delivery_time format is invalid or order status prevents delivery
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Retrieve shipment_order table
        shipment_order_table = getattr(db, "shipment_order", None)
        if shipment_order_table is None:
            raise KeyError(f"shipment_order table not found in database")

        # Check if the order exists
        if order_id not in shipment_order_table:
            raise KeyError(f"Order with order_id '{order_id}' does not exist")

        # Retrieve the order
        order = shipment_order_table[order_id]

        # Validate order status - delivery can only be recorded for orders that are out for delivery
        # or in transit (flexible to allow recording delivery from various valid states)
        valid_statuses = ["out_for_delivery", "in_transit", "picked_up"]
        if order.status not in valid_statuses:
            raise ValueError(
                f"Cannot record delivery for order with status '{order.status}'. "
                f"Order must be in one of these statuses: {valid_statuses}"
            )

        # Parse and validate actual_delivery_time format
        try:
            delivery_datetime = datetime.strptime(actual_delivery_time, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(
                f"Invalid actual_delivery_time format: '{actual_delivery_time}'. "
                f"Expected format: 'yyyy-mm-dd HH:MM:SS' (e.g., '2024-01-16 10:30:00')"
            ) from e

        # Validate that delivery time is not in the future
        current_time = datetime.now()
        if delivery_datetime > current_time:
            raise ValueError(
                f"Delivery time cannot be in the future. "
                f"Provided: {actual_delivery_time}, Current time: {current_time.strftime('%Y-%m-%d %H:%M:%S')}"
            )

        # Validate that delivery time is after order creation time
        if delivery_datetime < order.created_at:
            raise ValueError(
                f"Delivery time cannot be before order creation time. "
                f"Order created at: {order.created_at.strftime('%Y-%m-%d %H:%M:%S')}, "
                f"Delivery time: {actual_delivery_time}"
            )

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

        # Get current timestamp for recorded_at
        recorded_at_datetime = datetime.now()

        # Create new delivery record
        from datetime import datetime
        new_delivery_record = DeliveryRecord(
            delivery_record_id=delivery_record_id,
            order_id=order_id,
            courier_id=courier_id,
            actual_delivery_time=delivery_datetime,
            recipient_signature=recipient_signature,
            received_by=received_by,
            photo_evidence=None,  # Optional field, not provided in this tool
            created_at=recorded_at_datetime
        )

        # Retrieve or initialize delivery_record table
        delivery_record_table = getattr(db, "delivery_record", None)
        if delivery_record_table is None:
            delivery_record_table = {}

        # Add the new delivery record to the table
        delivery_record_table[delivery_record_id] = new_delivery_record
        setattr(db, "delivery_record", delivery_record_table)

        # Update the shipment order status to 'delivered'
        order.status = "delivered"
        order.updated_at = recorded_at_datetime

        # Save the updated order back to the database
        shipment_order_table[order_id] = order
        setattr(db, "shipment_order", shipment_order_table)

        # Return the delivery record information
        return {
            "delivery_record_id": delivery_record_id,
            "recorded_at": recorded_at_datetime.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def generate_shipping_label(
        self,
        tracking_number: str,
        sender_name: str,
        sender_address: str,
        recipient_name: str,
        recipient_address: str,
        service_type: Literal["standard", "express", "same_day", "next_day"],
        weight: float
    ) -> dict:
        """
        Generate shipping label data with all required information.

        This method creates a structured label data object containing all shipping information
        and generates a barcode for the label. The barcode encodes key information including
        tracking number, locations, and service type.

        Args:
            tracking_number: Unique tracking identifier for the shipment
            sender_name: Full name of the package sender
            sender_address: Complete address of the sender
            recipient_name: Full name of the package recipient
            recipient_address: Complete address of the recipient
            service_type: Type of courier service (standard/express/same_day/next_day)
            weight: Package weight in kilograms

        Returns:
            Dictionary containing:
                - label_data: Complete structured label information
                - barcode: Generated barcode string for the label

        Raises:
            ValueError: If any required parameter is invalid or missing
        """
        # Validate required parameters are not empty
        if not tracking_number or not isinstance(tracking_number, str):
            raise ValueError("tracking_number must be a non-empty string")

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

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

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

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

        # Validate service_type is one of the allowed enum values
        allowed_service_types = ["standard", "express", "same_day", "next_day"]
        if service_type not in allowed_service_types:
            raise ValueError(f"service_type must be one of {allowed_service_types}, got '{service_type}'")

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

        # Extract city/region information from addresses for label routing
        # Parse sender location (extract key location identifier)
        sender_location_parts = sender_address.split(',')
        sender_city = sender_location_parts[-1].strip() if sender_location_parts else "Unknown"

        # Parse recipient location (extract key location identifier)
        recipient_location_parts = recipient_address.split(',')
        recipient_city = recipient_location_parts[-1].strip() if recipient_location_parts else "Unknown"

        # Generate service type abbreviation for barcode
        service_abbreviations = {
            "standard": "STD",
            "express": "EXP",
            "same_day": "SD",
            "next_day": "ND"
        }
        service_abbr = service_abbreviations.get(service_type, "STD")

        # Create structured label data with all shipping information
        label_data = {
            "tracking": tracking_number,
            "from": sender_city,
            "to": recipient_city,
            "sender": {
                "name": sender_name,
                "address": sender_address
            },
            "recipient": {
                "name": recipient_name,
                "address": recipient_address
            },
            "service_type": service_type,
            "weight_kg": weight,
            "routing_info": {
                "origin": sender_city,
                "destination": recipient_city,
                "service": service_type
            }
        }

        # Generate barcode string encoding key shipment information
        # Format: TRACKING-ORIGIN_ABBR-DEST_ABBR-WEIGHT_CODE-SERVICE_ABBR
        # Extract first 2 letters of city names for compact representation
        origin_code = ''.join(filter(str.isalpha, sender_city))[:2].upper()
        dest_code = ''.join(filter(str.isalpha, recipient_city))[:2].upper()

        # Create weight code (format: 2 digits representing integer part)
        weight_code = f"{int(weight):02d}"

        # Construct barcode with all encoded information
        barcode = f"{tracking_number}-{origin_code}-{dest_code}-{weight_code}-{service_abbr}"

        # Return complete label data with barcode
        return {
            "label_data": label_data,
            "barcode": barcode
        }

    @is_tool()
    def validate_customs_declaration(
        self,
        declared_value: float,
        item_descriptions: list,
        hs_codes: list,
        origin_country: str,
        destination_country: str
    ) -> dict:
        """
        Validate completeness and accuracy of customs declaration information.

        This method performs comprehensive validation of customs declaration data including:
        - Declared value validation (positive number)
        - Item descriptions completeness and format
        - HS codes format and completeness
        - Country codes validation
        - Consistency checks between items and HS codes

        Args:
            declared_value: Declared value of goods (must be positive)
            item_descriptions: List of item descriptions (must not be empty)
            hs_codes: List of Harmonized System codes (must match item count)
            origin_country: Country of origin (ISO country code)
            destination_country: Destination country (ISO country code)

        Returns:
            dict: Validation result containing:
                - is_valid: Boolean indicating if declaration is valid
                - validation_errors: List of error messages if validation fails

        Raises:
            ValueError: If input parameters are of incorrect type or format
        """
        validation_errors = []

        # Validate declared_value type and value
        if not isinstance(declared_value, (int, float)):
            raise ValueError("declared_value must be a number")

        if declared_value <= 0:
            validation_errors.append("declared_value_must_be_positive")

        # Validate item_descriptions type and content
        if not isinstance(item_descriptions, list):
            raise ValueError("item_descriptions must be a list")

        if len(item_descriptions) == 0:
            validation_errors.append("item_descriptions_cannot_be_empty")
        else:
            # Check each item description
            for idx, desc in enumerate(item_descriptions):
                if not isinstance(desc, str):
                    validation_errors.append(f"item_description_at_index_{idx}_must_be_string")
                elif not desc or desc.strip() == "":
                    validation_errors.append(f"item_description_at_index_{idx}_is_empty")

        # Validate hs_codes type and content
        if not isinstance(hs_codes, list):
            raise ValueError("hs_codes must be a list")

        if len(hs_codes) == 0:
            validation_errors.append("hs_codes_cannot_be_empty")
        else:
            # Check each HS code format
            for idx, code in enumerate(hs_codes):
                if not isinstance(code, str):
                    validation_errors.append(f"hs_code_at_index_{idx}_must_be_string")
                elif not code or code.strip() == "":
                    validation_errors.append(f"hs_code_at_index_{idx}_is_empty")
                else:
                    # HS codes typically follow format: XXXX.XX.XX (digits with dots)
                    # Remove dots and check if remaining characters are digits
                    code_cleaned = code.replace(".", "")
                    if not code_cleaned.isdigit():
                        validation_errors.append(f"hs_code_at_index_{idx}_invalid_format")
                    # HS codes are typically 6-10 digits
                    elif len(code_cleaned) < 4 or len(code_cleaned) > 10:
                        validation_errors.append(f"hs_code_at_index_{idx}_invalid_length")

        # Validate consistency between item_descriptions and hs_codes
        if isinstance(item_descriptions, list) and isinstance(hs_codes, list):
            if len(item_descriptions) != len(hs_codes):
                validation_errors.append("item_descriptions_and_hs_codes_count_mismatch")

        # Validate origin_country
        if not isinstance(origin_country, str):
            raise ValueError("origin_country must be a string")

        if not origin_country or origin_country.strip() == "":
            validation_errors.append("origin_country_is_empty")
        else:
            # Country codes are typically 2-3 characters (ISO 3166-1 alpha-2 or alpha-3)
            origin_country_clean = origin_country.strip().upper()
            if len(origin_country_clean) < 2 or len(origin_country_clean) > 3:
                validation_errors.append("origin_country_invalid_format")
            elif not origin_country_clean.isalpha():
                validation_errors.append("origin_country_must_be_alphabetic")

        # Validate destination_country
        if not isinstance(destination_country, str):
            raise ValueError("destination_country must be a string")

        if not destination_country or destination_country.strip() == "":
            validation_errors.append("destination_country_is_empty")
        else:
            # Country codes are typically 2-3 characters (ISO 3166-1 alpha-2 or alpha-3)
            destination_country_clean = destination_country.strip().upper()
            if len(destination_country_clean) < 2 or len(destination_country_clean) > 3:
                validation_errors.append("destination_country_invalid_format")
            elif not destination_country_clean.isalpha():
                validation_errors.append("destination_country_must_be_alphabetic")

        # Check if origin and destination are the same (potential warning)
        if (isinstance(origin_country, str) and isinstance(destination_country, str) and
            origin_country.strip().upper() == destination_country.strip().upper()):
            validation_errors.append("origin_and_destination_countries_are_same")

        # Determine overall validity
        is_valid = len(validation_errors) == 0

        return {
            "is_valid": is_valid,
            "validation_errors": validation_errors
        }

    @is_tool()
    def calculate_service_zone(self, latitude: float, longitude: float):
        """
        Determine the service zone based on geographical coordinates.

        This method calculates the service zone classification by analyzing the distance
        from a reference point (typically city center) and determining the zone type
        (urban, suburban, rural, or remote) based on distance thresholds.

        Args:
            latitude: Latitude coordinate of the location
            longitude: Longitude coordinate of the location

        Returns:
            dict: A dictionary containing:
                - zone_id: Service zone identifier (e.g., "ZONE_BJ_01")
                - zone_type: Type of service zone (urban/suburban/rural/remote)
                - service_level: Available service level in this zone

        Raises:
            ValueError: If latitude or longitude values are invalid
        """
        import math

        # Validate input coordinates
        if not isinstance(latitude, (int, float)) or not isinstance(longitude, (int, float)):
            raise ValueError("Latitude and longitude must be numeric values")

        if not (-90 <= latitude <= 90):
            raise ValueError("Latitude must be between -90 and 90 degrees")

        if not (-180 <= longitude <= 180):
            raise ValueError("Longitude must be between -180 and 180 degrees")

        # Define reference points for major cities (can be extended)
        # Using Beijing as the primary reference point based on the example
        city_centers = {
            "BJ": {"lat": 39.9042, "lon": 116.4074, "name": "Beijing"},
            "SH": {"lat": 31.2304, "lon": 121.4737, "name": "Shanghai"},
            "GZ": {"lat": 23.1291, "lon": 113.2644, "name": "Guangzhou"},
            "SZ": {"lat": 22.5431, "lon": 114.0579, "name": "Shenzhen"}
        }

        # Calculate distance from the nearest city center using Haversine formula
        def haversine_distance(lat1: float, lon1: float, lat2: float, lon2: float) -> float:
            """
            Calculate the great circle distance between two points on Earth.
            Returns distance in kilometers.
            """
            # Convert decimal degrees to radians
            lat1_rad = math.radians(lat1)
            lon1_rad = math.radians(lon1)
            lat2_rad = math.radians(lat2)
            lon2_rad = math.radians(lon2)

            # Haversine formula
            dlat = lat2_rad - lat1_rad
            dlon = lon2_rad - lon1_rad
            a = math.sin(dlat/2)**2 + math.cos(lat1_rad) * math.cos(lat2_rad) * math.sin(dlon/2)**2
            c = 2 * math.asin(math.sqrt(a))

            # Earth's radius in kilometers
            earth_radius = 6371

            return earth_radius * c

        # Find the nearest city center and calculate distance
        min_distance = float('inf')
        nearest_city_code = None

        for city_code, city_info in city_centers.items():
            distance = haversine_distance(
                latitude, longitude,
                city_info["lat"], city_info["lon"]
            )
            if distance < min_distance:
                min_distance = distance
                nearest_city_code = city_code

        # Determine zone type based on distance thresholds (in kilometers)
        # Urban: 0-20km from city center
        # Suburban: 20-50km from city center
        # Rural: 50-150km from city center
        # Remote: >150km from city center
        if min_distance <= 20:
            zone_type = "urban"
            service_level = "full_service"
            zone_suffix = "01"
        elif min_distance <= 50:
            zone_type = "suburban"
            service_level = "full_service"
            zone_suffix = "02"
        elif min_distance <= 150:
            zone_type = "rural"
            service_level = "standard_service"
            zone_suffix = "03"
        else:
            zone_type = "remote"
            service_level = "limited_service"
            zone_suffix = "04"

        # Generate zone_id based on nearest city and zone type
        zone_id = f"ZONE_{nearest_city_code}_{zone_suffix}"

        # Return the service zone information
        return {
            "zone_id": zone_id,
            "zone_type": zone_type,
            "service_level": service_level
        }
