"""
Database schemas for express_logistics_Courier_Service_Platform

This module contains all database schema classes for the express_logistics domain.
All classes are Pydantic BaseModel subclasses.

Generated by ClassDBAgent.
"""

import json
from datetime import datetime
from datetime import datetime, date
from pathlib import Path
from pydantic import Field
from pydantic import PrivateAttr, Field
from scale_env.data_model.thread_safe_base import ThreadSafeBase, with_instance_key
from scale_env.environment.db import DB, DictAccessMixin as BaseModel
from typing import Dict, List, Any, Optional
from typing import Dict, List, Any, Optional, Literal

@with_instance_key("order_id")
class ShipmentOrder(BaseModel, ThreadSafeBase["ShipmentOrder"]):
    order_id: str = Field(..., description="Unique identifier for the shipment order")
    tracking_number: str = Field(..., description="Tracking number for the shipment")
    status: Literal[
        "pending",
        "picked_up",
        "in_transit",
        "out_for_delivery",
        "delivered",
        "failed",
        "cancelled",
        "returned"
    ] = Field("pending", description="Current status of the shipment")
    sender_name: str = Field(..., description="Sender's full name")
    sender_phone: str = Field(..., description="Sender's phone number")
    sender_address: str = Field(..., description="Sender's complete address")
    recipient_name: str = Field(..., description="Recipient's full name")
    recipient_phone: str = Field(..., description="Recipient's phone number")
    recipient_address: str = Field(..., description="Recipient's complete address")
    weight: float = Field(..., description="Package weight in kilograms")
    length: float = Field(..., description="Package length in centimeters")
    width: float = Field(..., description="Package width in centimeters")
    height: float = Field(..., description="Package height in centimeters")
    service_type: Literal["standard", "express", "same_day", "next_day"] = Field(
        ..., description="Type of courier service"
    )
    declared_value: Optional[float] = Field(
        None, description="Declared value of package contents in currency units"
    )
    total_cost: Optional[float] = Field(
        None, description="Total shipping cost including all charges"
    )
    created_at: datetime = Field(..., description="Order creation timestamp")
    updated_at: Optional[datetime] = Field(
        None, description="Last update timestamp"
    )

@with_instance_key("courier_id")
class Courier(BaseModel, ThreadSafeBase["Courier"]):
    courier_id: str = Field(..., description="Unique identifier for the courier")
    courier_name: str = Field(..., description="Courier's full name")
    phone_number: str = Field(..., description="Courier's phone number")
    current_latitude: Optional[float] = Field(None, description="Current latitude coordinate")
    current_longitude: Optional[float] = Field(None, description="Current longitude coordinate")
    last_location_update: Optional[datetime] = Field(None, description="Last location update timestamp")
    status: Literal["available", "busy", "offline"] = Field("available", description="Current working status of the courier")
    created_at: datetime = Field(..., description="Courier registration timestamp")

@with_instance_key("assignment_id")
class CourierAssignment(BaseModel, ThreadSafeBase["CourierAssignment"]):      
    assignment_id: str = Field(default=..., description="Unique identifier for the assignment")
    order_id: str = Field(default=..., description="Reference to shipment order")
    courier_id: str = Field(default=..., description="Reference to courier")
    assignment_type: Literal["pickup", "delivery"] = Field(default=..., description="Type of assignment")
    scheduled_time: datetime = Field(default=..., description="Scheduled pickup or delivery time")
    assignment_status: Literal["assigned", "in_progress", "completed", "cancelled"] = Field(
        default="assigned", description="Status of the assignment"
    )
    created_at: datetime = Field(default=..., description="Assignment creation timestamp")

@with_instance_key("pickup_record_id")
class PickupRecord(BaseModel, ThreadSafeBase["PickupRecord"]):
    pickup_record_id: str = Field(..., description="Unique identifier for the pickup record")
    order_id: str = Field(..., description="Reference to shipment order")
    courier_id: str = Field(..., description="Reference to courier who performed pickup")
    pickup_time: datetime = Field(..., description="Actual pickup timestamp")
    pickup_signature: Optional[str] = Field(None, description="Digital signature or confirmation code from sender")
    created_at: datetime = Field(..., description="Record creation timestamp")

@with_instance_key("delivery_record_id")
class DeliveryRecord(BaseModel, ThreadSafeBase["DeliveryRecord"]):
    delivery_record_id: str = Field(..., description="Unique identifier for the delivery record")
    order_id: str = Field(..., description="Reference to shipment order")
    courier_id: str = Field(..., description="Reference to courier who performed delivery")
    actual_delivery_time: datetime = Field(..., description="Actual delivery timestamp")
    recipient_signature: str = Field(..., description="Digital signature or confirmation code from recipient")
    received_by: str = Field(..., description="Name of the person who received the package")
    photo_evidence: Optional[str] = Field(None, description="Photo evidence of delivery (URL or base64 data)")
    created_at: datetime = Field(..., description="Record creation timestamp")

@with_instance_key("attempt_record_id")
class DeliveryAttempt(BaseModel, ThreadSafeBase["DeliveryAttempt"]):
    attempt_record_id: str = Field(default=..., description="Unique identifier for the delivery attempt record")
    order_id: str = Field(default=..., description="Reference to shipment order")
    courier_id: str = Field(default=..., description="Reference to courier who attempted delivery")
    attempt_time: datetime = Field(default=..., description="Delivery attempt timestamp")
    failure_reason: Literal["recipient_not_home", "incorrect_address", "refused_delivery", "access_denied", "weather_conditions", "other"] = Field(
        default=..., description="Reason for delivery failure"
    )
    notes: Optional[str] = Field(default=None, description="Additional notes about the delivery attempt")
    attempt_count: int = Field(default=..., description="Sequential attempt number for this order")
    created_at: datetime = Field(default=..., description="Record creation timestamp")

@with_instance_key("event_id")
class TrackingEvent(BaseModel, ThreadSafeBase["TrackingEvent"]):
    event_id: str = Field(..., description="Unique identifier for the tracking event")
    tracking_number: str = Field(..., description="Tracking number for the shipment")
    event_type: Literal[
        "order_created",
        "picked_up",
        "in_transit",
        "arrived_at_facility",
        "out_for_delivery",
        "delivered",
        "exception",
        "returned"
    ] = Field(..., description="Type of tracking event")
    event_location: str = Field(..., description="Location where the event occurred")
    event_time: datetime = Field(..., description="Event timestamp")
    event_description: Optional[str] = Field(None, description="Detailed description of the event")
    created_at: datetime = Field(..., description="Event addition timestamp")

class ExpressLogisticsDB(DB):
    """Database containing all express_logistics_Courier_Service_Platform-related data"""
    shipment_order: Optional[Dict[str, ShipmentOrder]] = Field(
        default=None,
        description="Schema ShipmentOrder"
    )
    courier: Optional[Dict[str, Courier]] = Field(
        default=None,
        description="Schema Courier"
    )
    courier_assignment: Optional[Dict[str, CourierAssignment]] = Field(
        default=None,
        description="Schema CourierAssignment"
    )
    pickup_record: Optional[Dict[str, PickupRecord]] = Field(
        default=None,
        description="Schema PickupRecord"
    )
    delivery_record: Optional[Dict[str, DeliveryRecord]] = Field(
        default=None,
        description="Schema DeliveryRecord"
    )
    delivery_attempt: Optional[Dict[str, DeliveryAttempt]] = Field(
        default=None,
        description="Schema DeliveryAttempt"
    )
    tracking_event: Optional[Dict[str, TrackingEvent]] = Field(
        default=None,
        description="Schema TrackingEvent"
    )