"""
Database schemas for second_hand_marketplace

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

Generated by ClassDBAgent.
"""

import json
from datetime import datetime
from datetime import datetime, date
from decimal import Decimal
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("user_id")
class User(BaseModel, ThreadSafeBase["User"]):      
    user_id: str = Field(..., description="Unique identifier for the user.")
    username: str = Field(..., description="Unique username for login and display.")
    email: str = Field(..., description="User's email address for communication and verification.")
    password_hash: str = Field(..., description="Hashed password for authentication.")
    phone_number: Optional[str] = Field(default=None, description="User's phone number, optional for verification.")
    bio: Optional[str] = Field(default=None, description="Short biography for the user's profile.")
    location: Optional[str] = Field(default=None, description="User's general location, e.g., city or region.")
    join_date: date = Field(default_factory=date.today, description="The date the user joined the marketplace.")
    verification_status: Literal["pending_verification", "verified", "verification_failed"] = Field(default="pending_verification", description="Status of the user's identity verification.")
    average_rating: Optional[Decimal] = Field(default=None, description="The user's average rating as a seller and buyer.")

@with_instance_key("category_id")
class Category(BaseModel, ThreadSafeBase["Category"]):      
    category_id: str = Field(default=..., description="Unique identifier for the category.")
    name: str = Field(default=..., description="Display name of the category.")
    parent_category_id: Optional[str] = Field(default=None, description="Reference to a parent category for hierarchical structure.")

@with_instance_key("listing_id")
class ItemListing(BaseModel, ThreadSafeBase["ItemListing"]):      
    listing_id: str = Field(default=..., description="Unique identifier for the item listing.")
    user_id: str = Field(default=..., description="The user who created the listing.")
    title: str = Field(default=..., description="The title of the item listing.")
    description: str = Field(default=..., description="Detailed description of the item.")
    price: float = Field(default=..., description="The asking price for the item.")
    category_id: str = Field(default=..., description="The category of the item.")
    condition: Literal["new", "like_new", "good", "fair", "poor"] = Field(default=..., description="The condition of the item.") # enum field
    photo_urls: Optional[List[str]] = Field(default=None, description="A JSON array of URLs for the item's photos.") # photo_urls is optional
    status: Literal["active", "inactive", "sold", "deleted"] = Field(default="active", description="The current status of the listing.") # enum field
    created_at: datetime = Field(default=..., description="Timestamp when the listing was created.")
    updated_at: datetime = Field(default=..., description="Timestamp when the listing was last updated.")

@with_instance_key("order_id")
class Order(BaseModel, ThreadSafeBase["Order"]):      
    order_id: str = Field(default=..., description="Unique identifier for the order.")
    listing_id: str = Field(default=..., description="The item listing being purchased.")
    user_id: str = Field(default=..., description="The user making the purchase.")
    seller_id: Optional[str] = Field(default=None, description="The user selling the item (derived from listing).")
    quantity: int = Field(default=1, description="Number of items purchased.")
    agreed_price: Decimal = Field(default=..., description="The final price agreed upon for the transaction.")
    total_amount: Decimal = Field(default=..., description="Total amount to be paid, including fees.")
    status: Literal["pending_payment", "paid", "shipped", "delivered", "completed", "cancelled", "refunded"] = Field(default="pending_payment", description="The current status of the order.") # enum field
    logistics_method: Optional[Literal["courier", "meetup"]] = Field(default=None, description="Method for completing the transaction.") # enum field, optional
    tracking_number: Optional[str] = Field(default=None, description="Tracking number for shipped items.") # optional
    created_at: datetime = Field(default_factory=datetime.now, description="Timestamp when the order was created.")
    updated_at: datetime = Field(default_factory=datetime.now, description="Timestamp when the order was last updated.")

@with_instance_key("offer_id")
class Offer(BaseModel, ThreadSafeBase["Offer"]):      
    offer_id: str = Field(default=..., description="Unique identifier for the offer.")
    listing_id: str = Field(default=..., description="The item listing the offer is for.")
    user_id: str = Field(default=..., description="The user making the offer.")
    price: float = Field(default=..., description="The price offered by the buyer.")
    content: Optional[str] = Field(default=None, description="Optional message accompanying the offer.") # content is optional
    status: Literal["pending", "accepted", "rejected", "expired"] = Field(default="pending", description="The status of the offer.") # enum field
    expires_at: datetime = Field(default=..., description="Timestamp when the offer expires.")
    created_at: datetime = Field(default=..., description="Timestamp when the offer was created.")

@with_instance_key("message_id")
class Message(BaseModel, ThreadSafeBase["Message"]):      
    message_id: str = Field(default=..., description="Unique identifier for the message.")
    user_id: str = Field(default=..., description="The user who sent the message.")
    user_id: str = Field(default=..., description="The user who received the message.")
    listing_id: Optional[str] = Field(default=None, description="The listing the message is about.") # listing_id is optional
    order_id: Optional[str] = Field(default=None, description="The order the message is about.") # order_id is optional
    content: str = Field(default=..., description="The text content of the message.")
    timestamp: datetime = Field(default=..., description="Timestamp when the message was sent.")

@with_instance_key("review_id")
class Review(BaseModel, ThreadSafeBase["Review"]):      
    review_id: str = Field(default=..., description="Unique identifier for the review.")
    order_id: str = Field(default=..., description="The completed order this review is for.")
    user_id: str = Field(default=..., description="The user writing the review.")
    user_id: str = Field(default=..., description="The user being reviewed (the other party in the transaction).")
    rating: int = Field(default=..., description="The rating given, typically 1-5.")
    content: Optional[str] = Field(default=None, description="The textual content of the review.") # content is optional
    timestamp: datetime = Field(default=..., description="Timestamp when the review was submitted.")

@with_instance_key("dispute_id")
class Dispute(BaseModel, ThreadSafeBase["Dispute"]):      
    dispute_id: str = Field(default=..., description="Unique identifier for the dispute case.")
    order_id: str = Field(default=..., description="The order under dispute.")
    user_id: str = Field(default=..., description="The user who initiated the dispute.")
    reason: Literal["item_not_as_described", "item_damaged", "item_not_received", "wrong_item_shipped"] = Field(default=..., description="The reason for initiating the dispute.") # enum field
    description: str = Field(default=..., description="Detailed description of the issue.")
    status: Literal["awaiting_response", "under_review", "resolved"] = Field(default="awaiting_response", description="The current status of the dispute.") # enum field
    resolution: Optional[Literal["refund_buyer", "return_item_for_refund", "release_funds_to_seller"]] = Field(default=None, description="The final resolution of the dispute.") # enum field, optional
    created_at: datetime = Field(default=..., description="Timestamp when the dispute was opened.")
    resolved_at: Optional[datetime] = Field(default=None, description="Timestamp when the dispute was resolved.") # optional

@with_instance_key("evidence_id")
class DisputeEvidence(BaseModel, ThreadSafeBase["DisputeEvidence"]):      
    evidence_id: str = Field(default=..., description="Unique identifier for the evidence.")
    dispute_id: str = Field(default=..., description="The dispute case this evidence belongs to.")
    user_id: str = Field(default=..., description="The user who submitted the evidence.")
    evidence_type: Literal["photo", "video", "document", "chat_log"] = Field(default=..., description="The type of evidence being submitted.") # enum field
    evidence_url: str = Field(default=..., description="URL to the uploaded evidence file.")
    description: Optional[str] = Field(default=None, description="Description of the evidence.") # description is optional
    submitted_at: datetime = Field(default=..., description="Timestamp when the evidence was submitted.")

@with_instance_key("comment_id")
class DisputeComment(BaseModel, ThreadSafeBase["DisputeComment"]):      
    comment_id: str = Field(default=..., description="Unique identifier for the comment.")
    dispute_id: str = Field(default=..., description="The dispute case this comment belongs to.")
    user_id: str = Field(default=..., description="The user or moderator writing the comment.")
    content: str = Field(default=..., description="The text content of the comment.")
    timestamp: datetime = Field(default=..., description="Timestamp when the comment was added.")

@with_instance_key("notification_id")
class Notification(BaseModel, ThreadSafeBase["Notification"]):      
    notification_id: str = Field(default=..., description="Unique identifier for the notification.")
    user_id: str = Field(default=..., description="The recipient user.")
    notification_type: Literal["new_message", "offer_received", "order_update", "payment_received", "item_sold"] = Field(default=..., description="The type of notification.") # enum field
    title: str = Field(default=..., description="The title of the notification.")
    content: str = Field(default=..., description="The main content of the notification.")
    related_entity_id: Optional[str] = Field(default=None, description="ID of the related entity (e.g., listing_id, order_id).") # related_entity_id is optional
    is_read: bool = Field(default=False, description="Whether the notification has been read by the user.")
    timestamp: datetime = Field(default=..., description="Timestamp when the notification was created.")

@with_instance_key("user_id")
class UserBlock(BaseModel, ThreadSafeBase["UserBlock"]):      
    user_id: str = Field(..., description="The user who initiated the block.")
    blocked_user_id: str = Field(..., description="The user who was blocked.")
    blocked_at: datetime = Field(default_factory=datetime.now, description="Timestamp when the block was created.")

class SecondHandMarketplaceDB(DB):
    """Database containing all second_hand_marketplace-related data"""
    user: Optional[Dict[str, User]] = Field(
        default=None,
        description="Schema User"
    )
    category: Optional[Dict[str, Category]] = Field(
        default=None,
        description="Schema Category"
    )
    item_listing: Optional[Dict[str, ItemListing]] = Field(
        default=None,
        description="Schema ItemListing"
    )
    order: Optional[Dict[str, Order]] = Field(
        default=None,
        description="Schema Order"
    )
    offer: Optional[Dict[str, Offer]] = Field(
        default=None,
        description="Schema Offer"
    )
    message: Optional[Dict[str, Message]] = Field(
        default=None,
        description="Schema Message"
    )
    review: Optional[Dict[str, Review]] = Field(
        default=None,
        description="Schema Review"
    )
    dispute: Optional[Dict[str, Dispute]] = Field(
        default=None,
        description="Schema Dispute"
    )
    dispute_evidence: Optional[Dict[str, DisputeEvidence]] = Field(
        default=None,
        description="Schema DisputeEvidence"
    )
    dispute_comment: Optional[Dict[str, DisputeComment]] = Field(
        default=None,
        description="Schema DisputeComment"
    )
    notification: Optional[Dict[str, Notification]] = Field(
        default=None,
        description="Schema Notification"
    )
    user_block: Optional[Dict[str, UserBlock]] = Field(
        default=None,
        description="Schema UserBlock"
    )