"""
Database schemas for stock_trading

This module contains all database schema classes for the stock_trading 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 pydantic import PrivateAttr, Field, field_validator
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 UserAccount(BaseModel, ThreadSafeBase["UserAccount"]):
    user_id: str = Field(..., description="Unique identifier for the user")
    cash_balance: float = Field(0.00, description="Available cash balance in USD")
    buying_power: float = Field(0.00, description="Total buying power including margin")
    margin_used: float = Field(0.00, description="Amount of margin currently used")
    created_at: datetime = Field(..., description="Account creation timestamp")
    updated_at: datetime = Field(..., description="Last update timestamp")

    @field_validator("created_at", "updated_at", mode="before")
    @classmethod
    def strip_and_parse_datetime(cls, v):
        if isinstance(v, str):
            v = v.strip()
        return v

@with_instance_key("order_id")
class StockOrder(BaseModel, ThreadSafeBase["StockOrder"]):
    order_id: str = Field(..., description="Unique identifier for the order")
    user_id: str = Field(..., description="User who placed the order")
    ticker_symbol: str = Field(..., description="Stock ticker symbol")
    order_type: Literal["market", "limit", "stop_loss"] = Field(..., description="Type of order")
    order_side: Literal["buy", "sell"] = Field(..., description="Order side: buy or sell")
    quantity: int = Field(..., description="Number of shares to trade")
    limit_price: Optional[float] = Field(None, description="Limit price for limit orders")
    stop_price: Optional[float] = Field(None, description="Stop price for stop-loss orders")
    status: Literal["pending", "monitoring", "filled", "cancelled", "rejected"] = Field(..., description="Current order status")
    filled_quantity: int = Field(0, description="Number of shares filled")
    average_fill_price: Optional[float] = Field(None, description="Average execution price")
    created_at: datetime = Field(..., description="Order creation timestamp")
    filled_at: Optional[datetime] = Field(None, description="Order fill timestamp")
    cancelled_at: Optional[datetime] = Field(None, description="Order cancellation timestamp")

@with_instance_key("position_id")
class PortfolioPosition(BaseModel, ThreadSafeBase["PortfolioPosition"]):
    position_id: str = Field(default=..., description="Unique identifier for the position")
    user_id: str = Field(default=..., description="User who owns the position")
    ticker_symbol: str = Field(default=..., description="Stock ticker symbol")
    quantity: int = Field(default=..., description="Number of shares held")
    average_cost: float = Field(default=..., description="Average cost per share")
    sector: Optional[str] = Field(default=None, description="Market sector of the stock")
    created_at: datetime = Field(default=..., description="Position creation timestamp")
    updated_at: datetime = Field(default=..., description="Last update timestamp")

@with_instance_key("transaction_id")
class AccountTransaction(BaseModel, ThreadSafeBase["AccountTransaction"]):
    transaction_id: str = Field(..., description="Unique identifier for the transaction")
    user_id: str = Field(..., description="User associated with the transaction")
    transaction_type: Literal["deposit", "withdrawal"] = Field(..., description="Type of transaction")
    amount: float = Field(..., description="Transaction amount in USD")
    balance_after: float = Field(..., description="Account balance after transaction")
    timestamp: datetime = Field(default_factory=datetime.now, description="Transaction timestamp")

@with_instance_key("alert_id")
class PriceAlert(BaseModel, ThreadSafeBase["PriceAlert"]):
    alert_id: str = Field(default=..., description="Unique identifier for the alert")
    user_id: str = Field(default=..., description="User who created the alert")
    ticker_symbol: str = Field(default=..., description="Stock ticker symbol")
    target_price: float = Field(default=..., description="Price level that triggers the alert")
    condition: Literal["above", "below"] = Field(default=..., description="Alert condition")
    status: Literal["active", "triggered", "deleted"] = Field(default=..., description="Alert status")
    created_at: datetime = Field(default=..., description="Alert creation timestamp")
    triggered_at: Optional[datetime] = Field(default=None, description="Alert trigger timestamp")  # triggered_at is optional

@with_instance_key("watchlist_id")
class Watchlist(BaseModel, ThreadSafeBase["Watchlist"]):
    watchlist_id: str = Field(..., description="Unique identifier for the watchlist")
    user_id: str = Field(..., description="User who owns the watchlist")
    watchlist_name: str = Field(..., description="Name of the watchlist")
    created_at: datetime = Field(default_factory=datetime.now, description="Watchlist creation timestamp")

@with_instance_key("watchlist_stock_id")
class WatchlistStock(BaseModel, ThreadSafeBase["WatchlistStock"]):
    watchlist_stock_id: str = Field(..., description="Unique identifier for the watchlist-stock relationship")
    watchlist_id: str = Field(..., description="Watchlist containing the stock")
    ticker_symbol: str = Field(..., description="Stock ticker symbol")
    created_at: datetime = Field(default_factory=datetime.now, description="Timestamp when stock was added to watchlist")

@with_instance_key("price_history_id")
class StockPriceHistory(BaseModel, ThreadSafeBase["StockPriceHistory"]):
    price_history_id: str = Field(default=..., description="Unique identifier for the price record")
    ticker_symbol: str = Field(default=..., description="Stock ticker symbol")
    date: datetime = Field(default=..., description="Date and time of the price record")
    open: float = Field(default=..., description="Opening price")
    high: float = Field(default=..., description="Highest price")
    low: float = Field(default=..., description="Lowest price")
    close: float = Field(default=..., description="Closing price")
    volume: int = Field(default=..., description="Trading volume")

@with_instance_key("ticker_symbol")
class StockFundamental(BaseModel, ThreadSafeBase["StockFundamental"]):
    ticker_symbol: str = Field(..., description="Stock ticker symbol")
    market_cap: Optional[int] = Field(default=None, description="Market capitalization")
    pe_ratio: Optional[float] = Field(default=None, description="Price-to-earnings ratio")
    eps: Optional[float] = Field(default=None, description="Earnings per share")
    dividend_yield: Optional[float] = Field(default=None, description="Dividend yield as decimal")
    revenue: Optional[int] = Field(default=None, description="Annual revenue")
    sector: Optional[str] = Field(default=None, description="Market sector")
    updated_at: datetime = Field(default=..., description="Last update timestamp")

class StockTradingDB(DB):
    """Database containing all stock_trading-related data"""
    user_account: Optional[Dict[str, UserAccount]] = Field(
        default=None,
        description="Schema UserAccount"
    )
    stock_order: Optional[Dict[str, StockOrder]] = Field(
        default=None,
        description="Schema StockOrder"
    )
    portfolio_position: Optional[Dict[str, PortfolioPosition]] = Field(
        default=None,
        description="Schema PortfolioPosition"
    )
    account_transaction: Optional[Dict[str, AccountTransaction]] = Field(
        default=None,
        description="Schema AccountTransaction"
    )
    price_alert: Optional[Dict[str, PriceAlert]] = Field(
        default=None,
        description="Schema PriceAlert"
    )
    watchlist: Optional[Dict[str, Watchlist]] = Field(
        default=None,
        description="Schema Watchlist"
    )
    watchlist_stock: Optional[Dict[str, WatchlistStock]] = Field(
        default=None,
        description="Schema WatchlistStock"
    )
    stock_price_history: Optional[Dict[str, StockPriceHistory]] = Field(
        default=None,
        description="Schema StockPriceHistory"
    )
    stock_fundamental: Optional[Dict[str, StockFundamental]] = Field(
        default=None,
        description="Schema StockFundamental"
    )