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 stock_trading."""

class StockTradingTools(ToolKitBase):
    """All tools for stock_trading."""
    
    db: StockTradingDB
    
    def __init__(self, db: StockTradingDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def delete_price_alert(self, alert_id: str):
        """
        Deletes an existing price alert by marking it as deleted

        Args:
            alert_id: Unique identifier of the alert to delete

        Returns:
            dict: Contains alert_id and updated status

        Raises:
            ValueError: If alert_id is invalid or alert doesn't exist
        """
        # Import required classes
        from datetime import datetime

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

        # Access the database
        db = self.db

        # Get the price_alert table from database
        price_alert_table = getattr(db, "price_alert", None)

        # Check if price_alert table exists
        if price_alert_table is None:
            raise ValueError("price_alert table does not exist in database")

        # Check if the alert exists in the database
        if alert_id not in price_alert_table:
            raise ValueError(f"Alert with alert_id '{alert_id}' does not exist")

        # Retrieve the alert record
        alert = price_alert_table[alert_id]

        # Check if alert is already deleted
        if alert.status == "deleted":
            raise ValueError(f"Alert with alert_id '{alert_id}' is already deleted")

        # Update the alert status to 'deleted'
        # Create a copy of the alert data with updated status
        alert.status = "deleted"

        # Update the database with the modified alert
        price_alert_table[alert_id] = alert
        setattr(db, "price_alert", price_alert_table)

        # Return the result with alert_id and updated status
        return {
            "alert_id": alert_id,
            "status": "deleted"
        }

    @is_tool()
    def get_account_balance(self, user_id: str):
        """
        Retrieves the current cash balance and buying power of a user account.

        This method fetches account information for the specified user from the database
        and returns the financial details including cash balance, buying power, and margin used.

        Args:
            user_id: Unique identifier of the user whose account balance is to be retrieved

        Returns:
            dict: A dictionary containing:
                - cash_balance: Available cash balance in USD
                - buying_power: Total buying power including margin
                - margin_used: Amount of margin currently used

        Raises:
            ValueError: If user_id is invalid or user account is not found
        """
        # Import necessary classes
        from scale_env.environment.db import DictAccessMixin as BaseModel

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

        # Access the database instance
        db = self.db

        # Retrieve the user_account table from the database
        user_account_table = getattr(db, 'user_account', None)

        # Check if the user_account table exists
        if user_account_table is None:
            raise ValueError("user_account table not found in database")

        # Check if the table is empty
        if not user_account_table:
            raise ValueError(f"User account not found for user_id: {user_id}")

        # Retrieve the specific user account by user_id
        user_account = user_account_table.get(user_id)

        # Validate that the user account exists
        if user_account is None:
            raise ValueError(f"User account not found for user_id: {user_id}")

        # Extract and return the account balance information
        return {
            "cash_balance": user_account.cash_balance,
            "buying_power": user_account.buying_power,
            "margin_used": user_account.margin_used
        }

    @is_tool()
    def deposit_funds(self, user_id: str, amount: float):
        """
        Deposits funds into a user's trading account.

        This method:
        1. Validates the input parameters (user_id and amount)
        2. Checks if the user account exists
        3. Updates the account balance (cash_balance and buying_power)
        4. Creates a transaction record
        5. Returns transaction details
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate amount - must be positive
        if amount <= 0:
            raise ValueError("Deposit amount must be positive")

        # Validate user_id - must not be empty
        if not user_id or not isinstance(user_id, str):
            raise ValueError("Valid user_id must be provided")

        # Access database
        db = self.db

        # Get user_account table
        user_account_table = getattr(db, "user_account", None)
        if user_account_table is None:
            raise ValueError("User account table not found")

        # Check if user exists
        user_account = user_account_table.get(user_id)
        if user_account is None:
            raise ValueError(f"User with ID {user_id} not found")

        # Calculate new balance
        # Both cash_balance and buying_power should be increased by the deposit amount
        new_cash_balance = user_account.cash_balance + amount
        new_buying_power = user_account.buying_power + amount

        # Update user account
        user_account.cash_balance = new_cash_balance
        user_account.buying_power = new_buying_power
        user_account.updated_at = datetime.now()

        # Save updated user account back to database
        user_account_table[user_id] = user_account
        setattr(db, "user_account", user_account_table)

        # Generate unique transaction ID
        transaction_id = "txn_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        transaction_timestamp = datetime.now()

        # Create transaction record
        transaction = AccountTransaction(
            transaction_id=transaction_id,
            user_id=user_id,
            transaction_type="deposit",
            amount=amount,
            balance_after=new_cash_balance,
            timestamp=transaction_timestamp
        )

        # Get account_transaction table
        account_transaction_table = getattr(db, "account_transaction", None)
        if account_transaction_table is None:
            # Initialize empty transaction table if it doesn't exist
            account_transaction_table = {}

        # Add transaction to table
        account_transaction_table[transaction_id] = transaction
        setattr(db, "account_transaction", account_transaction_table)

        # Return transaction details
        return {
            "transaction_id": transaction_id,
            "new_balance": new_cash_balance,
            "timestamp": transaction_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def get_stock_historical_prices(self, ticker_symbol: str, start_date: str, end_date: str):
        """
        Retrieves historical price data for a stock within a specified date range.

        Args:
            ticker_symbol: The stock ticker symbol (e.g., "AAPL")
            start_date: Start date in yyyy-mm-dd format (e.g., "2024-01-01")
            end_date: End date in yyyy-mm-dd format (e.g., "2024-01-15")

        Returns:
            dict: Contains ticker_symbol and price_data list with historical records

        Raises:
            ValueError: If date format is invalid, date range is invalid, or no data found
        """
        from datetime import datetime

        # Validate and parse date strings
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d")
            end_dt = datetime.strptime(end_date, "%Y-%m-%d")
        except ValueError as e:
            raise ValueError(f"Invalid date format. Expected yyyy-mm-dd format. Error: {str(e)}")

        # Validate date range
        if start_dt > end_dt:
            raise ValueError(f"Start date ({start_date}) cannot be after end date ({end_date})")

        # Access the database
        db = self.db

        # Get stock_price_history table
        stock_price_history_table = getattr(db, "stock_price_history", None)

        if stock_price_history_table is None:
            raise ValueError("Stock price history table not found in database")

        # Filter price records for the specified ticker and date range
        matching_records = []

        for price_history_id, price_record in stock_price_history_table.items():
            # Check if ticker symbol matches (case-insensitive for robustness)
            if price_record.ticker_symbol.upper() != ticker_symbol.upper():
                continue

            # Extract date from datetime object (ignore time component)
            record_date = price_record.date.date() if isinstance(price_record.date, datetime) else price_record.date
            start_date_only = start_dt.date()
            end_date_only = end_dt.date()

            # Check if record date is within the specified range (inclusive)
            if start_date_only <= record_date <= end_date_only:
                # Convert record to dictionary format for output
                matching_records.append({
                    "date": record_date.strftime("%Y-%m-%d"),  # Format as yyyy-mm-dd
                    "open": price_record.open,
                    "high": price_record.high,
                    "low": price_record.low,
                    "close": price_record.close,
                    "volume": price_record.volume
                })

        # Sort records by date in ascending order
        matching_records.sort(key=lambda x: x["date"])

        # Return result with original ticker symbol format
        return {
            "ticker_symbol": ticker_symbol,
            "price_data": matching_records
        }

    @is_tool()
    def calculate_portfolio_beta(self, positions: list) -> dict:
        """
        Calculates the portfolio beta relative to market benchmark.

        Portfolio beta is calculated as the weighted average of individual stock betas,
        where weights represent the proportion of each position in the portfolio.

        Args:
            positions: Array of position dictionaries, each containing:
                - ticker_symbol: Stock ticker symbol
                - beta: Individual stock beta value
                - weight: Position weight in portfolio (should sum to 1.0)

        Returns:
            dict: Dictionary containing:
                - portfolio_beta: Weighted portfolio beta value

        Raises:
            ValueError: If positions is empty, weights don't sum to approximately 1.0,
                       or if required fields are missing or invalid

        Algorithm:
            Portfolio Beta = Σ(weight_i × beta_i) for all positions i
        """
        # Validate that positions array is not empty
        if not positions or len(positions) == 0:
            raise ValueError("Positions array cannot be empty")

        # Initialize variables for calculation
        portfolio_beta = 0.0
        total_weight = 0.0

        # Iterate through each position to calculate weighted beta
        for i, position in enumerate(positions):
            # Validate that position is a dictionary
            if not isinstance(position, dict):
                raise ValueError(f"Position at index {i} must be a dictionary object")

            # Validate required fields exist
            if "ticker_symbol" not in position:
                raise ValueError(f"Position at index {i} missing required field 'ticker_symbol'")
            if "beta" not in position:
                raise ValueError(f"Position at index {i} missing required field 'beta'")
            if "weight" not in position:
                raise ValueError(f"Position at index {i} missing required field 'weight'")

            # Extract values
            ticker_symbol = position["ticker_symbol"]
            beta = position["beta"]
            weight = position["weight"]

            # Validate ticker_symbol type
            if not isinstance(ticker_symbol, str):
                raise ValueError(f"Position at index {i}: ticker_symbol must be a string")

            # Validate beta is a number
            if not isinstance(beta, (int, float)):
                raise ValueError(f"Position at index {i}: beta must be a number")

            # Validate weight is a number
            if not isinstance(weight, (int, float)):
                raise ValueError(f"Position at index {i}: weight must be a number")

            # Validate weight is non-negative
            if weight < 0:
                raise ValueError(f"Position at index {i}: weight cannot be negative")

            # Calculate weighted beta contribution for this position
            portfolio_beta += weight * beta
            total_weight += weight

        # Validate that total weights sum to approximately 1.0 (allowing small floating point errors)
        # Using tolerance of 0.01 (1%) for practical purposes
        if abs(total_weight - 1.0) > 0.01:
            raise ValueError(f"Position weights must sum to approximately 1.0, got {total_weight:.4f}")

        # Return the calculated portfolio beta
        return {
            "portfolio_beta": portfolio_beta
        }

    @is_tool()
    def calculate_macd(self, prices: list, fast_period: int, slow_period: int, signal_period: int) -> dict:
        """
        Calculates the Moving Average Convergence Divergence (MACD) indicator.

        The MACD is a trend-following momentum indicator that shows the relationship between
        two exponential moving averages (EMAs) of a security's price.

        MACD Line = Fast EMA - Slow EMA
        Signal Line = EMA of MACD Line
        Histogram = MACD Line - Signal Line

        Args:
            prices: Array of historical closing prices
            fast_period: Fast EMA period (typically 12)
            slow_period: Slow EMA period (typically 26)
            signal_period: Signal line period (typically 9)

        Returns:
            Dictionary containing macd_line, signal_line, and histogram values

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

        # Validate input parameters
        if not prices or len(prices) == 0:
            raise ValueError("Price data cannot be empty")

        if not all(isinstance(p, (int, float)) for p in prices):
            raise ValueError("All prices must be numeric values")

        if fast_period <= 0 or slow_period <= 0 or signal_period <= 0:
            raise ValueError("All period parameters must be positive integers")

        if fast_period >= slow_period:
            raise ValueError("Fast period must be less than slow period")

        # Check if we have enough data points
        min_required_length = slow_period + signal_period - 1
        if len(prices) < min_required_length:
            raise ValueError(f"Insufficient price data. Need at least {min_required_length} data points, got {len(prices)}")

        # Helper function to calculate Exponential Moving Average (EMA)
        def calculate_ema(data: list, period: int) -> list:
            """
            Calculate EMA for the given data and period.

            EMA formula:
            - EMA_today = (Price_today * multiplier) + (EMA_yesterday * (1 - multiplier))
            - multiplier = 2 / (period + 1)

            Args:
                data: List of price values
                period: EMA period

            Returns:
                List of EMA values (starts from index period-1)
            """
            if len(data) < period:
                return []

            multiplier = 2.0 / (period + 1)
            ema_values = []

            # Initialize with Simple Moving Average (SMA) for the first EMA value
            sma = sum(data[:period]) / period
            ema_values.append(sma)

            # Calculate subsequent EMA values
            for i in range(period, len(data)):
                ema = (data[i] * multiplier) + (ema_values[-1] * (1 - multiplier))
                ema_values.append(ema)

            return ema_values

        # Step 1: Calculate fast EMA (typically 12-period)
        fast_ema = calculate_ema(prices, fast_period)

        # Step 2: Calculate slow EMA (typically 26-period)
        slow_ema = calculate_ema(prices, slow_period)

        # Step 3: Calculate MACD line (fast EMA - slow EMA)
        # Align the EMAs - slow EMA starts later than fast EMA
        offset = slow_period - fast_period
        macd_line_values = []

        for i in range(len(slow_ema)):
            macd_value = fast_ema[i + offset] - slow_ema[i]
            macd_line_values.append(macd_value)

        # Step 4: Calculate signal line (EMA of MACD line, typically 9-period)
        signal_line_values = calculate_ema(macd_line_values, signal_period)

        # Check if we have enough data to calculate signal line
        if len(signal_line_values) == 0:
            raise ValueError(f"Insufficient data to calculate signal line. Need at least {signal_period} MACD values")

        # Step 5: Get the most recent values
        # The most recent MACD line value that has a corresponding signal line
        macd_line = macd_line_values[len(macd_line_values) - len(signal_line_values) + len(signal_line_values) - 1]
        signal_line = signal_line_values[-1]

        # Step 6: Calculate histogram (MACD line - signal line)
        histogram = macd_line - signal_line

        # Round values to 2 decimal places for cleaner output
        return {
            "macd_line": round(macd_line, 2),
            "signal_line": round(signal_line, 2),
            "histogram": round(histogram, 2)
        }

    @is_tool()
    def get_user_open_orders(self, user_id: str):
        """
        Retrieves all open orders for a specific user.

        This method queries the stock_order table to find all orders belonging to the specified user
        that have an "open" status (pending or monitoring). It returns detailed information about each
        open order including order ID, ticker symbol, order type, quantity, and status.

        Args:
            user_id: Unique identifier of the user whose open orders need to be retrieved

        Returns:
            A dictionary containing:
            - open_orders: List of dictionaries, each representing an open order with fields:
                - order_id: Unique identifier of the order
                - ticker_symbol: Stock ticker symbol
                - order_type: Type of order (market, limit, stop_loss)
                - quantity: Number of shares to trade
                - status: Current order status (pending or monitoring)

        Raises:
            ValueError: If user_id is empty or None
        """
        # Validate input parameter
        if not user_id:
            raise ValueError("user_id cannot be empty or None")

        # Access the database instance
        db = self.db

        # Retrieve the stock_order table from the database
        stock_order_table = getattr(db, "stock_order", None)

        # If the table doesn't exist or is empty, return empty list
        if stock_order_table is None:
            return {"open_orders": []}

        # Initialize list to store open orders
        open_orders = []

        # Define what statuses are considered "open"
        # Based on the schema, "pending" and "monitoring" are open statuses
        # "filled", "cancelled", and "rejected" are closed statuses
        open_statuses = ["pending", "monitoring"]

        # Iterate through all orders in the stock_order table
        for order_id, order in stock_order_table.items():
            # Check if the order belongs to the specified user and has an open status
            if order.user_id == user_id and order.status in open_statuses:
                # Build order information dictionary
                order_info = {
                    "order_id": order.order_id,
                    "ticker_symbol": order.ticker_symbol,
                    "order_type": order.order_type,
                    "quantity": order.quantity,
                    "status": order.status
                }

                # Add the order to the result list
                open_orders.append(order_info)

        # Return the list of open orders
        return {"open_orders": open_orders}

    @is_tool()
    def calculate_sharpe_ratio(self, portfolio_return: float, risk_free_rate: float, standard_deviation: float):
        """
        Calculates the Sharpe ratio for portfolio performance evaluation.

        The Sharpe ratio measures risk-adjusted return by comparing excess return
        (portfolio return minus risk-free rate) to portfolio volatility (standard deviation).

        Formula: Sharpe Ratio = (Portfolio Return - Risk-Free Rate) / Standard Deviation

        Args:
            portfolio_return: Portfolio return as decimal (e.g., 0.15 for 15%)
            risk_free_rate: Risk-free rate as decimal (e.g., 0.03 for 3%)
            standard_deviation: Portfolio standard deviation (volatility measure)

        Returns:
            dict: Dictionary containing the calculated Sharpe ratio
                {
                    "sharpe_ratio": float  # The calculated Sharpe ratio value
                }

        Raises:
            ValueError: If standard_deviation is zero or negative, or if inputs are invalid
            TypeError: If inputs are not numeric types
        """
        # Validate input types
        if not isinstance(portfolio_return, (int, float)):
            raise TypeError(f"portfolio_return must be a number, got {type(portfolio_return).__name__}")

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

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

        # Validate standard deviation is positive
        # Standard deviation cannot be zero (would cause division by zero)
        # Standard deviation cannot be negative (mathematically invalid)
        if standard_deviation <= 0:
            raise ValueError(
                f"standard_deviation must be positive, got {standard_deviation}. "
                "A zero or negative standard deviation is invalid for Sharpe ratio calculation."
            )

        # Validate that portfolio_return and risk_free_rate are reasonable values
        # While they can technically be any value, extremely large values might indicate data errors
        if abs(portfolio_return) > 100:
            raise ValueError(
                f"portfolio_return seems unreasonably large: {portfolio_return}. "
                "Expected a decimal value (e.g., 0.15 for 15%)."
            )

        if abs(risk_free_rate) > 100:
            raise ValueError(
                f"risk_free_rate seems unreasonably large: {risk_free_rate}. "
                "Expected a decimal value (e.g., 0.03 for 3%)."
            )

        # Calculate excess return (return above risk-free rate)
        excess_return = portfolio_return - risk_free_rate

        # Calculate Sharpe ratio
        # Higher Sharpe ratio indicates better risk-adjusted performance
        # Positive Sharpe ratio: portfolio outperforms risk-free rate
        # Negative Sharpe ratio: portfolio underperforms risk-free rate
        sharpe_ratio = excess_return / standard_deviation

        # Return the calculated Sharpe ratio
        return {
            "sharpe_ratio": sharpe_ratio
        }

    @is_tool()
    def get_user_order_history(self, user_id: str, start_date: str, end_date: str):
        """
        Retrieves historical order records for a user within a date range.

        Args:
            user_id: Unique identifier of the user
            start_date: Start date in yyyy-mm-dd format
            end_date: End date in yyyy-mm-dd format

        Returns:
            Dictionary containing order_history list

        Raises:
            ValueError: If user_id is empty, date format is invalid, or date range is invalid
        """
        from datetime import datetime

        # Validate user_id is not empty
        if not user_id or not user_id.strip():
            raise ValueError("user_id cannot be empty")

        # Validate and parse date formats
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d")
        except ValueError:
            raise ValueError(f"Invalid start_date format: {start_date}. Expected yyyy-mm-dd format")

        try:
            end_dt = datetime.strptime(end_date, "%Y-%m-%d")
        except ValueError:
            raise ValueError(f"Invalid end_date format: {end_date}. Expected yyyy-mm-dd format")

        # Validate date range: start_date should not be after end_date
        if start_dt > end_dt:
            raise ValueError(f"start_date ({start_date}) cannot be after end_date ({end_date})")

        # Access the database
        db = self.db

        # Get stock_order table data
        stock_order_table = getattr(db, "stock_order", None)

        # If table doesn't exist or is empty, return empty list
        if stock_order_table is None or len(stock_order_table) == 0:
            return {"order_history": []}

        # Filter orders by user_id and date range
        order_history = []

        for order_id, order in stock_order_table.items():
            # Check if order belongs to the specified user
            if order.user_id != user_id:
                continue

            # Determine which timestamp to use for date filtering
            # Priority: filled_at (for filled orders) > cancelled_at (for cancelled) > created_at (for pending/rejected)
            order_timestamp = None
            if order.filled_at is not None:
                order_timestamp = order.filled_at
            elif order.cancelled_at is not None:
                order_timestamp = order.cancelled_at
            else:
                order_timestamp = order.created_at

            # Extract date part from timestamp for comparison
            order_date = order_timestamp.date()

            # Check if order falls within the date range (inclusive)
            if start_dt.date() <= order_date <= end_dt.date():
                # Build order dictionary with all relevant fields
                order_dict = {
                    "order_id": order.order_id,
                    "ticker_symbol": order.ticker_symbol,
                    "order_type": order.order_type,
                    "order_side": order.order_side,
                    "quantity": order.quantity,
                    "status": order.status,
                    "filled_quantity": order.filled_quantity,
                    "created_at": order.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }

                # Add optional fields if they exist
                if order.limit_price is not None:
                    order_dict["limit_price"] = order.limit_price

                if order.stop_price is not None:
                    order_dict["stop_price"] = order.stop_price

                if order.average_fill_price is not None:
                    order_dict["average_fill_price"] = order.average_fill_price

                if order.filled_at is not None:
                    order_dict["filled_at"] = order.filled_at.strftime("%Y-%m-%d %H:%M:%S")

                if order.cancelled_at is not None:
                    order_dict["cancelled_at"] = order.cancelled_at.strftime("%Y-%m-%d %H:%M:%S")

                order_history.append(order_dict)

        # Sort orders by timestamp (most recent first)
        # Use the same priority logic as filtering: filled_at > cancelled_at > created_at
        def get_sort_key(order_dict):
            if "filled_at" in order_dict:
                return datetime.strptime(order_dict["filled_at"], "%Y-%m-%d %H:%M:%S")
            elif "cancelled_at" in order_dict:
                return datetime.strptime(order_dict["cancelled_at"], "%Y-%m-%d %H:%M:%S")
            else:
                return datetime.strptime(order_dict["created_at"], "%Y-%m-%d %H:%M:%S")

        order_history.sort(key=get_sort_key, reverse=True)

        return {"order_history": order_history}

    @is_tool()
    def compare_stocks(self, ticker_symbols: List[str], metrics: List[Literal["price", "pe_ratio", "market_cap", "dividend_yield", "revenue", "eps"]]) -> dict:
        """
        Compares key metrics between multiple stocks for analysis.

        Args:
            ticker_symbols: Array of stock ticker symbols to compare
            metrics: List of metrics to compare (must be from enum: price, pe_ratio, market_cap, dividend_yield, revenue, eps)

        Returns:
            dict: Dictionary containing comparison_data with metrics for each stock

        Raises:
            ValueError: If ticker_symbols is empty, metrics is empty, invalid metric provided, or stock data not found
        """
        # Validate input parameters
        if not ticker_symbols:
            raise ValueError("ticker_symbols cannot be empty")

        if not metrics:
            raise ValueError("metrics cannot be empty")

        # Define valid metrics for safety check
        valid_metrics = {"price", "pe_ratio", "market_cap", "dividend_yield", "revenue", "eps"}

        # Validate all metrics are valid (safety protection for enum)
        for metric in metrics:
            if metric not in valid_metrics:
                raise ValueError(f"Invalid metric: {metric}. Must be one of {valid_metrics}")

        # Access database
        db = self.db

        # Get stock_fundamental and stock_price_history tables
        stock_fundamental_table = getattr(db, "stock_fundamental", None)
        stock_price_history_table = getattr(db, "stock_price_history", None)

        if stock_fundamental_table is None:
            raise ValueError("stock_fundamental table not found in database")

        if stock_price_history_table is None:
            raise ValueError("stock_price_history table not found in database")

        # Initialize result list
        comparison_data = []

        # Process each ticker symbol
        for ticker in ticker_symbols:
            # Initialize stock data dictionary with ticker symbol
            stock_data = {"ticker_symbol": ticker}

            # Flag to check if stock exists
            stock_found = False

            # Check if "price" metric is requested
            if "price" in metrics:
                # Get latest price from stock_price_history
                # Find all price records for this ticker
                price_records = [record for record in stock_price_history_table.values() 
                               if record.ticker_symbol == ticker]

                if price_records:
                    # Sort by date to get the latest record
                    latest_record = max(price_records, key=lambda x: x.date)
                    stock_data["price"] = latest_record.close
                    stock_found = True

            # Check if any fundamental metrics are requested
            fundamental_metrics = {"pe_ratio", "market_cap", "dividend_yield", "revenue", "eps"}
            requested_fundamental_metrics = [m for m in metrics if m in fundamental_metrics]

            if requested_fundamental_metrics:
                # Get fundamental data for this ticker
                fundamental_data = stock_fundamental_table.get(ticker)

                if fundamental_data:
                    stock_found = True
                    # Add requested fundamental metrics to stock_data
                    for metric in requested_fundamental_metrics:
                        # Get the attribute value from fundamental_data
                        metric_value = getattr(fundamental_data, metric, None)
                        stock_data[metric] = metric_value

            # If stock data was found for at least one metric, add to comparison_data
            if stock_found:
                comparison_data.append(stock_data)
            else:
                # If no data found for this ticker, raise error
                raise ValueError(f"No data found for ticker symbol: {ticker}")

        # Return the comparison data
        return {"comparison_data": comparison_data}

    @is_tool()
    def get_user_alerts(self, user_id: str):
        """
        Retrieves all active price alerts for a specific user.

        This method queries the price_alert table to find all alerts belonging to
        the specified user that have an 'active' status.

        Args:
            user_id: Unique identifier of the user whose alerts to retrieve

        Returns:
            Dictionary containing a list of active alerts with their details

        Raises:
            ValueError: If user_id is empty or invalid
        """
        # Validate user_id parameter
        if not user_id or not isinstance(user_id, str) or not user_id.strip():
            raise ValueError("user_id must be a non-empty string")

        # Access the database instance
        db = self.db

        # Get the price_alert table from database
        price_alert_table = getattr(db, "price_alert", None)

        # If price_alert table doesn't exist or is empty, return empty list
        if price_alert_table is None or not price_alert_table:
            return {"alerts": []}

        # Initialize list to store user's active alerts
        user_alerts = []

        # Iterate through all alerts in the price_alert table
        for alert_id, alert in price_alert_table.items():
            # Check if alert belongs to the specified user and has active status
            if alert.user_id == user_id and alert.status == "active":
                # Build alert dictionary with required fields
                alert_dict = {
                    "alert_id": alert.alert_id,
                    "ticker_symbol": alert.ticker_symbol,
                    "target_price": alert.target_price,
                    "condition": alert.condition,
                    "status": alert.status
                }

                # Add the alert to the result list
                user_alerts.append(alert_dict)

        # Return the list of active alerts for the user
        return {"alerts": user_alerts}

    @is_tool()
    def calculate_bollinger_bands(self, prices: list, period: int, std_dev_multiplier: float):
        """
        Calculates Bollinger Bands for volatility analysis.

        Bollinger Bands consist of:
        - Middle Band: Simple Moving Average (SMA) of the closing prices
        - Upper Band: Middle Band + (Standard Deviation × Multiplier)
        - Lower Band: Middle Band - (Standard Deviation × Multiplier)

        Args:
            prices: Array of historical closing prices
            period: Number of periods for moving average (typically 20)
            std_dev_multiplier: Standard deviation multiplier (typically 2)

        Returns:
            Dictionary containing upper_band, middle_band, and lower_band values

        Raises:
            ValueError: If input parameters are invalid
        """
        # Input validation
        if not prices or not isinstance(prices, list):
            raise ValueError("prices must be a non-empty list")

        if len(prices) < period:
            raise ValueError(f"prices length ({len(prices)}) must be at least equal to period ({period})")

        if period <= 0:
            raise ValueError("period must be a positive integer")

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

        # Validate all prices are numeric
        try:
            prices = [float(p) for p in prices]
        except (ValueError, TypeError):
            raise ValueError("All prices must be numeric values")

        # Calculate the Simple Moving Average (SMA) - Middle Band
        # Use the most recent 'period' prices for calculation
        recent_prices = prices[-period:]
        middle_band = sum(recent_prices) / period

        # Calculate the standard deviation
        # Standard deviation formula: sqrt(sum((x - mean)^2) / n)
        variance = sum((price - middle_band) ** 2 for price in recent_prices) / period
        std_dev = variance ** 0.5

        # Calculate Upper and Lower Bands
        upper_band = middle_band + (std_dev * std_dev_multiplier)
        lower_band = middle_band - (std_dev * std_dev_multiplier)

        # Return the Bollinger Bands as a dictionary
        return {
            "upper_band": round(upper_band, 2),
            "middle_band": round(middle_band, 2),
            "lower_band": round(lower_band, 2)
        }

    @is_tool()
    def calculate_correlation_coefficient(self, stock1_prices: list, stock2_prices: list):
        """
        Calculates the Pearson correlation coefficient between two stocks' price movements.

        The correlation coefficient measures the linear relationship between two variables,
        ranging from -1 (perfect negative correlation) to +1 (perfect positive correlation).
        A value of 0 indicates no linear correlation.

        Args:
            stock1_prices: Array of historical prices for the first stock
            stock2_prices: Array of historical prices for the second stock

        Returns:
            dict: Contains 'correlation_coefficient' (float between -1 and 1)

        Raises:
            ValueError: If input arrays are invalid (empty, different lengths, insufficient data, or non-numeric)
        """
        # Import required library for statistical calculations
        import statistics

        # Validate that both price arrays are provided and not empty
        if not stock1_prices or not stock2_prices:
            raise ValueError("Both stock price arrays must be non-empty")

        # Validate that both arrays have the same length
        if len(stock1_prices) != len(stock2_prices):
            raise ValueError(f"Stock price arrays must have the same length. Got {len(stock1_prices)} and {len(stock2_prices)}")

        # Validate that we have at least 2 data points (minimum for correlation)
        if len(stock1_prices) < 2:
            raise ValueError("At least 2 data points are required to calculate correlation coefficient")

        # Validate that all values are numeric
        try:
            stock1_prices = [float(price) for price in stock1_prices]
            stock2_prices = [float(price) for price in stock2_prices]
        except (TypeError, ValueError) as e:
            raise ValueError(f"All price values must be numeric: {str(e)}")

        # Calculate the number of data points
        n = len(stock1_prices)

        # Calculate means of both datasets
        mean1 = statistics.mean(stock1_prices)
        mean2 = statistics.mean(stock2_prices)

        # Calculate standard deviations
        # Use population standard deviation (pstdev) for consistency
        try:
            std1 = statistics.pstdev(stock1_prices)
            std2 = statistics.pstdev(stock2_prices)
        except statistics.StatisticsError:
            # This occurs when all values are identical (std = 0)
            raise ValueError("Cannot calculate correlation when one or both price series have zero variance")

        # Check for zero standard deviation (constant values)
        if std1 == 0 or std2 == 0:
            raise ValueError("Cannot calculate correlation when one or both price series have zero variance")

        # Calculate covariance manually
        # Covariance = sum((x_i - mean_x) * (y_i - mean_y)) / n
        covariance = sum((stock1_prices[i] - mean1) * (stock2_prices[i] - mean2) for i in range(n)) / n

        # Calculate Pearson correlation coefficient
        # Correlation = covariance / (std1 * std2)
        correlation = covariance / (std1 * std2)

        # Ensure the result is within valid range [-1, 1] due to potential floating-point errors
        # Clamp the value to handle edge cases
        correlation = max(-1.0, min(1.0, correlation))

        # Return the correlation coefficient
        return {
            "correlation_coefficient": correlation
        }

    @is_tool()
    def get_stock_fundamentals(self, ticker_symbol: str):
        """
        Retrieves fundamental financial data for a stock from the database.

        This method looks up the stock fundamental data by ticker symbol and returns
        all available financial metrics including market cap, P/E ratio, EPS, 
        dividend yield, and revenue.

        Args:
            ticker_symbol: Stock ticker symbol (e.g., "AAPL")

        Returns:
            dict: Dictionary containing fundamental financial metrics with keys:
                - ticker_symbol: Stock ticker symbol
                - market_cap: Market capitalization
                - pe_ratio: Price-to-earnings ratio
                - eps: Earnings per share
                - dividend_yield: Dividend yield as decimal
                - revenue: Annual revenue

        Raises:
            ValueError: If ticker_symbol is empty or if stock fundamental data not found
        """
        # Validate input parameter
        if not ticker_symbol or not isinstance(ticker_symbol, str):
            raise ValueError("ticker_symbol must be a non-empty string")

        # Normalize ticker symbol to uppercase for consistency
        ticker_symbol = ticker_symbol.strip().upper()

        # Access the database
        db = self.db

        # Get the stock_fundamental table from database
        stock_fundamental_table = getattr(db, "stock_fundamental", None)

        # Check if the table exists and has data
        if stock_fundamental_table is None:
            raise ValueError(f"Stock fundamental data table not found in database")

        # Look up the stock fundamental data by ticker symbol
        stock_data = stock_fundamental_table.get(ticker_symbol)

        # Raise error if stock not found
        if stock_data is None:
            raise ValueError(f"Stock fundamental data not found for ticker symbol: {ticker_symbol}")

        # Extract and return the fundamental metrics
        # Note: Some fields may be None/Optional in the database schema
        result = {
            "ticker_symbol": stock_data.ticker_symbol,
            "market_cap": stock_data.market_cap,
            "pe_ratio": stock_data.pe_ratio,
            "eps": stock_data.eps,
            "dividend_yield": stock_data.dividend_yield,
            "revenue": stock_data.revenue
        }

        return result

    @is_tool()
    def get_user_portfolio(self, user_id: str):
        # Import necessary modules for database operations
        from datetime import datetime

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

        # Access the database
        db = self.db

        # Retrieve user account information
        user_accounts = getattr(db, "user_account", None)
        if user_accounts is None:
            raise ValueError("User account database is not available")

        # Check if user exists
        user_account = user_accounts.get(user_id)
        if user_account is None:
            raise ValueError(f"User with ID '{user_id}' not found")

        # Get cash balance from user account
        cash_balance = user_account.cash_balance

        # Retrieve portfolio positions
        portfolio_positions = getattr(db, "portfolio_position", None)
        if portfolio_positions is None:
            raise ValueError("Portfolio position database is not available")

        # Filter positions belonging to the specified user
        user_positions = [
            position for position in portfolio_positions.values()
            if position.user_id == user_id
        ]

        # Initialize variables for portfolio calculation
        positions_list = []
        total_value = 0.0

        # Process each position
        for position in user_positions:
            # For this implementation, we use average_cost as current_price
            # since real-time stock prices are not available in the database
            # In a production system, this would fetch from a market data service
            current_price = position.average_cost

            # Calculate market value for this position
            market_value = position.quantity * current_price

            # Add to total portfolio value
            total_value += market_value

            # Create position dictionary with required fields
            position_dict = {
                "ticker_symbol": position.ticker_symbol,
                "quantity": position.quantity,
                "average_cost": position.average_cost,
                "current_price": current_price,
                "market_value": market_value
            }

            positions_list.append(position_dict)

        # Construct and return the complete portfolio information
        portfolio = {
            "positions": positions_list,
            "total_value": total_value,
            "cash_balance": cash_balance
        }

        return portfolio

    @is_tool()
    def create_watchlist(self, user_id: str, watchlist_name: str):
        """
        Creates a new watchlist for tracking multiple stocks

        Args:
            user_id: Unique identifier of the user
            watchlist_name: Name of the watchlist

        Returns:
            dict: Contains watchlist_id, watchlist_name, and created_at

        Raises:
            ValueError: If user_id or watchlist_name is empty or invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

        # Access the database
        db = self.db

        # Generate unique watchlist_id using secure random hash
        prefix = "watchlist_"
        watchlist_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create timestamp for watchlist creation
        created_at = datetime.now()

        # Create new Watchlist instance
        new_watchlist = Watchlist(
            watchlist_id=watchlist_id,
            user_id=user_id.strip(),
            watchlist_name=watchlist_name.strip(),
            created_at=created_at
        )

        # Get existing watchlist data from database
        watchlist_data = getattr(db, "watchlist", None)

        # Initialize watchlist dictionary if it doesn't exist
        if watchlist_data is None:
            watchlist_data = {}

        # Add new watchlist to the database
        watchlist_data[watchlist_id] = new_watchlist

        # Save updated watchlist data back to database
        setattr(db, "watchlist", watchlist_data)

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

        # Return the created watchlist information
        return {
            "watchlist_id": watchlist_id,
            "watchlist_name": watchlist_name.strip(),
            "created_at": created_at_str
        }

    @is_tool()
    def withdraw_funds(self, user_id: str, amount: float):
        """
        Withdraws funds from a user's trading account

        This method performs the following steps:
        1. Validates the input parameters (user_id and amount)
        2. Retrieves the user's account from the database
        3. Checks if the user has sufficient balance for withdrawal
        4. Deducts the amount from the user's cash balance
        5. Creates a transaction record
        6. Updates the database with new account balance and transaction
        7. Returns transaction details including transaction_id, new_balance, and timestamp
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Step 1: Validate input parameters
        # Check if user_id is provided and not empty
        if not user_id or not isinstance(user_id, str):
            raise ValueError("user_id must be a non-empty string")

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

        # Step 2: Access the database
        db = self.db

        # Get user_account table from database
        user_account_table = getattr(db, "user_account", None)
        if user_account_table is None:
            raise ValueError("user_account table not found in database")

        # Step 3: Retrieve user's account
        user_account = user_account_table.get(user_id)
        if user_account is None:
            raise ValueError(f"User account with user_id '{user_id}' not found")

        # Step 4: Check if user has sufficient balance
        if user_account.cash_balance < amount:
            raise ValueError(
                f"Insufficient balance. Available balance: ${user_account.cash_balance:.2f}, "
                f"Requested withdrawal: ${amount:.2f}"
            )

        # Step 5: Calculate new balance after withdrawal
        new_balance = user_account.cash_balance - amount

        # Step 6: Generate unique transaction ID
        # Using SHA256 hash of random bytes to ensure uniqueness
        transaction_id = "txn_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Step 7: Record current timestamp
        timestamp = datetime.now()

        # Step 8: Update user account balance
        user_account.cash_balance = new_balance
        user_account.updated_at = timestamp

        # Update the user_account table in database
        user_account_table[user_id] = user_account
        setattr(db, "user_account", user_account_table)

        # Step 9: Create transaction record
        account_transaction_table = getattr(db, "account_transaction", None)
        if account_transaction_table is None:
            # Initialize empty transaction table if it doesn't exist
            account_transaction_table = {}

        # Create new transaction record
        new_transaction = AccountTransaction(
            transaction_id=transaction_id,
            user_id=user_id,
            transaction_type="withdrawal",
            amount=amount,
            balance_after=new_balance,
            timestamp=timestamp
        )

        # Add transaction to the table
        account_transaction_table[transaction_id] = new_transaction
        setattr(db, "account_transaction", account_transaction_table)

        # Step 10: Return transaction details
        # Format timestamp as string in "yyyy-mm-dd HH:MM:SS" format
        timestamp_str = timestamp.strftime("%Y-%m-%d %H:%M:%S")

        return {
            "transaction_id": transaction_id,
            "new_balance": new_balance,
            "timestamp": timestamp_str
        }

    @is_tool()
    def create_market_order(self, user_id: str, ticker_symbol: str, order_side: Literal["buy", "sell"], quantity: int):
        """
        Creates a market order to buy or sell stock at current market price.

        Args:
            user_id: Unique identifier of the user placing the order
            ticker_symbol: The stock ticker symbol
            order_side: Order side, must be either "buy" or "sell"
            quantity: Number of shares to trade, must be positive integer

        Returns:
            dict: Contains order_id, status, and created_at timestamp

        Raises:
            ValueError: If parameters are invalid or user account doesn't exist
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Import database models (already imported at file header, but included here for clarity)
        # StockOrder and UserAccount are available

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

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

        # Validate order_side against enum values
        if order_side not in ["buy", "sell"]:
            raise ValueError(f"order_side must be 'buy' or 'sell', got: {order_side}")

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

        # Access database
        db = self.db

        # Verify user account exists
        user_account_table = getattr(db, "user_account", None)
        if user_account_table is None:
            raise ValueError("user_account table not found in database")

        if user_id not in user_account_table:
            raise ValueError(f"User account with user_id '{user_id}' does not exist")

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

        # Create timestamp
        created_at = datetime.now()

        # Create market order object
        new_order = StockOrder(
            order_id=order_id,
            user_id=user_id,
            ticker_symbol=ticker_symbol.upper(),  # Normalize ticker symbol to uppercase
            order_type="market",  # This is a market order
            order_side=order_side,
            quantity=quantity,
            limit_price=None,  # Market orders don't have limit price
            stop_price=None,  # Market orders don't have stop price
            status="pending",  # Initial status is pending
            filled_quantity=0,  # No shares filled yet
            average_fill_price=None,  # No fill price yet
            created_at=created_at,
            filled_at=None,  # Not filled yet
            cancelled_at=None  # Not cancelled
        )

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

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

        # Write back to database
        setattr(db, "stock_order", stock_order_table)

        # Format created_at timestamp to required format
        created_at_str = created_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return order details
        return {
            "order_id": order_id,
            "status": "pending",
            "created_at": created_at_str
        }

    @is_tool()
    def calculate_portfolio_value(self, positions: list):
        """
        Calculate the total current market value of a portfolio.

        This method takes an array of portfolio positions and calculates the total
        market value by multiplying each position's quantity by its current price
        and summing all position values.

        Args:
            positions: Array of portfolio positions, each containing:
                - ticker_symbol (str): Stock ticker symbol
                - quantity (int/float): Number of shares held
                - current_price (float): Current market price per share

        Returns:
            dict: Dictionary containing:
                - total_value (float): Total portfolio market value

        Raises:
            ValueError: If positions is empty, None, or contains invalid data
        """
        # Validate that positions parameter is provided and not empty
        if positions is None:
            raise ValueError("Positions parameter cannot be None")

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

        if len(positions) == 0:
            raise ValueError("Positions list cannot be empty")

        # Initialize total value accumulator
        total_value = 0.0

        # Iterate through each position and calculate its market value
        for idx, position in enumerate(positions):
            # Validate that each position is a dictionary
            if not isinstance(position, dict):
                raise ValueError(f"Position at index {idx} must be a dictionary")

            # Validate required fields exist in the position
            if "ticker_symbol" not in position:
                raise ValueError(f"Position at index {idx} is missing 'ticker_symbol' field")

            if "quantity" not in position:
                raise ValueError(f"Position at index {idx} is missing 'quantity' field")

            if "current_price" not in position:
                raise ValueError(f"Position at index {idx} is missing 'current_price' field")

            # Extract position data
            ticker_symbol = position["ticker_symbol"]
            quantity = position["quantity"]
            current_price = position["current_price"]

            # Validate ticker_symbol is a non-empty string
            if not isinstance(ticker_symbol, str) or not ticker_symbol.strip():
                raise ValueError(f"Position at index {idx} has invalid ticker_symbol: must be a non-empty string")

            # Validate and convert quantity to numeric type
            try:
                quantity = float(quantity)
                if quantity < 0:
                    raise ValueError(f"Position at index {idx} has negative quantity: {quantity}")
            except (TypeError, ValueError) as e:
                raise ValueError(f"Position at index {idx} has invalid quantity: must be a non-negative number") from e

            # Validate and convert current_price to numeric type
            try:
                current_price = float(current_price)
                if current_price < 0:
                    raise ValueError(f"Position at index {idx} has negative current_price: {current_price}")
            except (TypeError, ValueError) as e:
                raise ValueError(f"Position at index {idx} has invalid current_price: must be a non-negative number") from e

            # Calculate position value: quantity * current_price
            position_value = quantity * current_price

            # Add to total portfolio value
            total_value += position_value

        # Return the total portfolio value as a dictionary
        return {
            "total_value": round(total_value, 2)  # Round to 2 decimal places for currency precision
        }

    @is_tool()
    def get_transaction_history(self, user_id: str, start_date: str, end_date: str):
        """
        Retrieves account transaction history for a specified date range.

        This method queries the account_transaction table to find all transactions
        for a given user within the specified date range (inclusive).

        Args:
            user_id: Unique identifier of the user
            start_date: Start date in yyyy-mm-dd format
            end_date: End date in yyyy-mm-dd format

        Returns:
            dict: Contains 'transactions' key with list of transaction dictionaries

        Raises:
            ValueError: If date format is invalid, date range is invalid, or user has no transactions
        """
        from datetime import datetime

        # Validate and parse date parameters
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d")
            end_dt = datetime.strptime(end_date, "%Y-%m-%d")
        except ValueError as e:
            raise ValueError(f"Invalid date format. Expected yyyy-mm-dd format. Error: {str(e)}")

        # Validate date range - end_date should not be before start_date
        if end_dt < start_dt:
            raise ValueError("End date cannot be before start date")

        # Set end_dt to end of day (23:59:59) to include all transactions on end_date
        end_dt = end_dt.replace(hour=23, minute=59, second=59)

        # Access the database
        db = self.db

        # Get account_transaction table
        account_transaction_table = getattr(db, "account_transaction", None)

        # If table doesn't exist or is empty, return empty list
        if account_transaction_table is None or len(account_transaction_table) == 0:
            raise ValueError(f"No transaction data available for user {user_id}")

        # Filter transactions by user_id and date range
        transactions = []
        user_has_transactions = False

        for transaction_id, transaction in account_transaction_table.items():
            # Check if transaction belongs to the specified user
            if transaction.user_id == user_id:
                user_has_transactions = True

                # Check if transaction falls within the date range
                # transaction.timestamp is a datetime object from the database
                if start_dt <= transaction.timestamp <= end_dt:
                    # Convert transaction to dictionary format for return
                    transaction_dict = {
                        "transaction_id": transaction.transaction_id,
                        "type": transaction.transaction_type,
                        "amount": transaction.amount,
                        "timestamp": transaction.timestamp.strftime("%Y-%m-%d %H:%M:%S")
                    }
                    transactions.append(transaction_dict)

        # If user has no transactions at all, raise error
        if not user_has_transactions:
            raise ValueError(f"No transactions found for user {user_id}")

        # Sort transactions by timestamp in ascending order (oldest first)
        transactions.sort(key=lambda x: x["timestamp"])

        # Return the list of transactions
        return {"transactions": transactions}

    @is_tool()
    def calculate_dividend_yield(self, annual_dividend: float, price: float) -> dict:
        """
        Calculates the dividend yield for a stock.

        The dividend yield is calculated as: (Annual Dividend / Stock Price)
        This represents the percentage return an investor receives from dividends
        relative to the stock price.

        Args:
            annual_dividend: Annual dividend per share in USD
            price: Current stock price in USD

        Returns:
            dict: Dictionary containing the calculated dividend yield as a decimal
                  Example: {"dividend_yield": 0.0052} represents 0.52% yield

        Raises:
            ValueError: If price is zero or negative, or if annual_dividend is negative
            TypeError: If inputs are not numeric types
        """
        # Validate input types
        if not isinstance(annual_dividend, (int, float)):
            raise TypeError(f"annual_dividend must be a number, got {type(annual_dividend).__name__}")

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

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

        # Validate price is positive (cannot divide by zero or negative price)
        if price <= 0:
            raise ValueError(f"price must be positive, got {price}")

        # Calculate dividend yield as a decimal
        # Formula: Dividend Yield = Annual Dividend / Stock Price
        dividend_yield = annual_dividend / price

        # Return the result in the specified format
        return {
            "dividend_yield": dividend_yield
        }

    @is_tool()
    def calculate_portfolio_return(self, positions: list):
        """
        Calculates the total return and return percentage of a portfolio.

        Args:
            positions: Array of portfolio positions with quantities, cost basis, and current prices.
                      Each position should be a dict with keys: ticker_symbol, quantity, average_cost, current_price

        Returns:
            dict: Contains profit_loss (total return in USD) and return_percentage

        Raises:
            ValueError: If positions is empty, invalid, or contains invalid data
        """
        # Validate input parameter
        if not positions:
            raise ValueError("Positions array cannot be empty")

        if not isinstance(positions, list):
            raise ValueError("Positions must be a list/array")

        # Initialize accumulators for total cost and total current value
        total_cost = 0.0
        total_current_value = 0.0

        # Iterate through each position to calculate costs and values
        for idx, position in enumerate(positions):
            # Validate position structure
            if not isinstance(position, dict):
                raise ValueError(f"Position at index {idx} must be a dictionary")

            # Extract required fields
            ticker_symbol = position.get('ticker_symbol')
            quantity = position.get('quantity')
            average_cost = position.get('average_cost')
            current_price = position.get('current_price')

            # Validate required fields exist
            if ticker_symbol is None:
                raise ValueError(f"Position at index {idx} missing 'ticker_symbol'")
            if quantity is None:
                raise ValueError(f"Position at index {idx} missing 'quantity'")
            if average_cost is None:
                raise ValueError(f"Position at index {idx} missing 'average_cost'")
            if current_price is None:
                raise ValueError(f"Position at index {idx} missing 'current_price'")

            # Validate field types and values
            try:
                quantity = float(quantity)
                average_cost = float(average_cost)
                current_price = float(current_price)
            except (TypeError, ValueError):
                raise ValueError(f"Position at index {idx} contains non-numeric values for quantity, average_cost, or current_price")

            # Validate non-negative values
            if quantity < 0:
                raise ValueError(f"Position at index {idx} has negative quantity: {quantity}")
            if average_cost < 0:
                raise ValueError(f"Position at index {idx} has negative average_cost: {average_cost}")
            if current_price < 0:
                raise ValueError(f"Position at index {idx} has negative current_price: {current_price}")

            # Calculate position cost and current value
            position_cost = quantity * average_cost
            position_current_value = quantity * current_price

            # Accumulate totals
            total_cost += position_cost
            total_current_value += position_current_value

        # Handle edge case where total cost is zero
        if total_cost == 0:
            raise ValueError("Total cost basis cannot be zero")

        # Calculate profit/loss (total return amount)
        profit_loss = total_current_value - total_cost

        # Calculate return percentage
        # Formula: ((current_value - cost) / cost) * 100
        return_percentage = (profit_loss / total_cost) * 100

        # Round results to 2 decimal places for currency and percentage
        profit_loss = round(profit_loss, 2)
        return_percentage = round(return_percentage, 2)

        # Return results as dictionary
        return {
            "profit_loss": profit_loss,
            "return_percentage": return_percentage
        }

    @is_tool()
    def validate_order_parameters(self, ticker_symbol: str, order_type: Literal["market", "limit", "stop_loss"], order_side: Literal["buy", "sell"], quantity: int, price: float = None):
        """
        Validates order parameters before submission to ensure compliance with trading rules.

        This method performs comprehensive validation of order parameters including:
        - Ticker symbol format validation
        - Order type and side validation (with enum constraints)
        - Quantity validation (must be positive)
        - Price validation based on order type requirements

        Args:
            ticker_symbol: Stock ticker symbol to validate
            order_type: Type of order (must be one of: "market", "limit", "stop_loss")
            order_side: Order side (must be one of: "buy", "sell")
            quantity: Number of shares (must be positive integer)
            price: Order price for limit/stop orders (optional for market orders)

        Returns:
            dict: Validation result containing:
                - is_valid (bool): Whether all parameters are valid
                - errors (list): List of validation error messages (empty if valid)

        Raises:
            ValueError: If required parameters are missing or invalid types
        """
        # Initialize error list to collect all validation errors
        errors = []

        # Validate ticker_symbol
        if not ticker_symbol or not isinstance(ticker_symbol, str):
            errors.append("ticker_symbol must be a non-empty string")
        elif not ticker_symbol.strip():
            errors.append("ticker_symbol cannot be empty or whitespace only")
        elif len(ticker_symbol.strip()) > 10:
            # Most ticker symbols are 1-5 characters, setting reasonable upper limit
            errors.append("ticker_symbol is too long (maximum 10 characters)")
        elif not ticker_symbol.strip().replace('.', '').replace('-', '').isalnum():
            # Allow alphanumeric characters, dots, and hyphens (common in ticker symbols)
            errors.append("ticker_symbol contains invalid characters (only letters, numbers, dots, and hyphens allowed)")

        # Validate order_type with enum constraint
        valid_order_types = ["market", "limit", "stop_loss"]
        if not order_type or not isinstance(order_type, str):
            errors.append("order_type must be a non-empty string")
        elif order_type not in valid_order_types:
            errors.append(f"order_type must be one of {valid_order_types}, got '{order_type}'")

        # Validate order_side with enum constraint
        valid_order_sides = ["buy", "sell"]
        if not order_side or not isinstance(order_side, str):
            errors.append("order_side must be a non-empty string")
        elif order_side not in valid_order_sides:
            errors.append(f"order_side must be one of {valid_order_sides}, got '{order_side}'")

        # Validate quantity
        if not isinstance(quantity, int):
            errors.append("quantity must be an integer")
        elif quantity <= 0:
            errors.append("quantity must be a positive integer greater than 0")
        elif quantity > 1000000:
            # Set reasonable upper limit to prevent unrealistic orders
            errors.append("quantity exceeds maximum allowed limit (1,000,000 shares)")

        # Validate price based on order_type
        # Market orders don't require price, but limit and stop_loss orders do
        if order_type in ["limit", "stop_loss"]:
            if price is None:
                errors.append(f"price is required for {order_type} orders")
            elif not isinstance(price, (int, float)):
                errors.append("price must be a number")
            elif price <= 0:
                errors.append("price must be a positive number greater than 0")
            elif price > 1000000:
                # Set reasonable upper limit for stock prices
                errors.append("price exceeds maximum allowed limit ($1,000,000)")
        elif order_type == "market":
            # For market orders, price should not be specified or should be ignored
            # We won't add an error if price is provided for market orders, just note it's not used
            if price is not None:
                # This is a warning rather than an error - market orders don't use price
                # We'll allow it but could add a note if needed
                pass

        # Additional validation: check for negative or zero price if provided
        if price is not None and not isinstance(price, (int, float)):
            if "price must be a number" not in errors:
                errors.append("price must be a number if provided")

        # Determine if all validations passed
        is_valid = len(errors) == 0

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

    @is_tool()
    def add_stock_to_watchlist(self, watchlist_id: str, ticker_symbol: str):
        # Import necessary modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Validate input parameters - check for empty or None values
        if not watchlist_id or not isinstance(watchlist_id, str):
            raise ValueError("watchlist_id must be a non-empty string")

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

        # Normalize ticker symbol to uppercase (standard format for stock tickers)
        ticker_symbol = ticker_symbol.strip().upper()

        # Get the watchlist_stock table from database
        watchlist_stock_table = getattr(db, "watchlist_stock", None)

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

        # Check if this stock is already in the watchlist to avoid duplicates
        for stock_id, stock in watchlist_stock_table.items():
            if stock.watchlist_id == watchlist_id and stock.ticker_symbol == ticker_symbol:
                # Stock already exists in this watchlist, return existing record
                return {
                    "watchlist_id": stock.watchlist_id,
                    "ticker_symbol": stock.ticker_symbol,
                    "created_at": stock.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }

        # Generate unique ID for the new watchlist-stock relationship
        prefix = "ws_"
        watchlist_stock_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create timestamp for when the stock is added
        created_at = datetime.now()

        # Create new WatchlistStock instance
        new_watchlist_stock = WatchlistStock(
            watchlist_stock_id=watchlist_stock_id,
            watchlist_id=watchlist_id,
            ticker_symbol=ticker_symbol,
            created_at=created_at
        )

        # Add the new watchlist-stock relationship to the table
        watchlist_stock_table[watchlist_stock_id] = new_watchlist_stock

        # Update the database with the modified table
        setattr(db, "watchlist_stock", watchlist_stock_table)

        # Return the result with formatted timestamp
        return {
            "watchlist_id": watchlist_id,
            "ticker_symbol": ticker_symbol,
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def calculate_pe_ratio(self, price: float, earnings_per_share: float) -> dict:
        """
        Calculate the price-to-earnings (P/E) ratio for a stock.

        The P/E ratio is a valuation metric that compares a company's stock price
        to its earnings per share. It indicates how much investors are willing to
        pay for each dollar of earnings.

        Formula: P/E Ratio = Stock Price / Earnings Per Share

        Args:
            price: Current stock price (must be a positive number)
            earnings_per_share: Earnings per share (EPS)

        Returns:
            dict: A dictionary containing the calculated P/E ratio
                  Format: {"pe_ratio": <calculated_value>}

        Raises:
            ValueError: If price is not positive, or if earnings_per_share is zero or negative
            TypeError: If inputs are not numeric types
        """
        # Validate input types
        if not isinstance(price, (int, float)):
            raise TypeError(f"Price must be a numeric type, got {type(price).__name__}")

        if not isinstance(earnings_per_share, (int, float)):
            raise TypeError(f"Earnings per share must be a numeric type, got {type(earnings_per_share).__name__}")

        # Validate price is positive
        if price <= 0:
            raise ValueError(f"Stock price must be positive, got {price}")

        # Validate earnings_per_share is not zero or negative
        # A company with zero or negative earnings cannot have a meaningful P/E ratio
        if earnings_per_share <= 0:
            raise ValueError(f"Earnings per share must be positive to calculate P/E ratio, got {earnings_per_share}")

        # Calculate the P/E ratio
        # Round to 2 decimal places for standard financial reporting
        pe_ratio = round(price / earnings_per_share, 2)

        # Return the result in the specified format
        return {"pe_ratio": pe_ratio}

    @is_tool()
    def remove_stock_from_watchlist(self, watchlist_id: str, ticker_symbol: str):
        """
        Removes a stock from an existing watchlist.

        Args:
            watchlist_id: Unique identifier of the watchlist
            ticker_symbol: Stock ticker symbol to remove

        Returns:
            dict: Contains watchlist_id and ticker_symbol of the removed stock

        Raises:
            ValueError: If watchlist_id or ticker_symbol is invalid, or if the stock is not in the watchlist
        """
        # Validate input parameters
        if not watchlist_id or not isinstance(watchlist_id, str):
            raise ValueError("watchlist_id must be a non-empty string")

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

        # Access the database
        db = self.db

        # Get the watchlist_stock table
        watchlist_stock_table = getattr(db, "watchlist_stock", None)

        if watchlist_stock_table is None:
            raise ValueError("watchlist_stock table does not exist in the database")

        # Find the matching watchlist-stock relationship
        # We need to find the entry where both watchlist_id and ticker_symbol match
        matching_entry_id = None
        for entry_id, entry in watchlist_stock_table.items():
            if entry.watchlist_id == watchlist_id and entry.ticker_symbol == ticker_symbol:
                matching_entry_id = entry_id
                break

        # If no matching entry found, raise an error
        if matching_entry_id is None:
            raise ValueError(f"Stock with ticker symbol '{ticker_symbol}' not found in watchlist '{watchlist_id}'")

        # Remove the entry from the watchlist_stock table
        # Create a new dictionary without the matching entry
        updated_watchlist_stock = {
            k: v for k, v in watchlist_stock_table.items() 
            if k != matching_entry_id
        }

        # Update the database table
        setattr(db, "watchlist_stock", updated_watchlist_stock)

        # Return the result
        return {
            "watchlist_id": watchlist_id,
            "ticker_symbol": ticker_symbol
        }

    @is_tool()
    def create_limit_order(self, user_id: str, ticker_symbol: str, order_side: Literal["buy", "sell"], quantity: int, limit_price: float):
        """
        Creates a limit order to buy or sell stock at a specified price or better.

        This method validates the input parameters, checks if the user exists,
        generates a unique order ID, and creates a new limit order in the database.
        """
        import secrets
        import hashlib
        from datetime import datetime

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

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

        # Validate order_side against enum values
        if order_side not in ["buy", "sell"]:
            raise ValueError(f"order_side must be either 'buy' or 'sell', got '{order_side}'")

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

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

        # Access the database
        db = self.db

        # Check if user exists in the database
        user_account_table = getattr(db, "user_account", None)
        if user_account_table is None:
            raise ValueError("user_account table not found in database")

        # Verify user exists
        if user_id not in user_account_table:
            raise ValueError(f"User with user_id '{user_id}' does not exist")

        # Generate unique order ID using secure random hash
        order_id_prefix = "order_"
        order_id = order_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Create the limit order object
        # For limit orders: order_type is "limit", status starts as "pending"
        # filled_quantity starts at 0, no fill price or timestamps yet
        new_order = StockOrder(
            order_id=order_id,
            user_id=user_id,
            ticker_symbol=ticker_symbol.upper(),  # Normalize ticker to uppercase
            order_type="limit",
            order_side=order_side,
            quantity=quantity,
            limit_price=limit_price,
            stop_price=None,  # Not applicable for limit orders
            status="pending",
            filled_quantity=0,
            average_fill_price=None,
            created_at=created_at,
            filled_at=None,
            cancelled_at=None
        )

        # Get the stock_order table
        stock_order_table = getattr(db, "stock_order", None)
        if stock_order_table is None:
            # Initialize empty table if it doesn't exist
            stock_order_table = {}

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

        # Update the database with the new order
        setattr(db, "stock_order", stock_order_table)

        # Return the order details in the specified format
        return {
            "order_id": order_id,
            "status": "pending",
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def calculate_position_profit_loss(self, quantity: int, average_cost: float, price: float):
        """
        Calculates profit/loss for a specific position based on quantity, average cost, and current price.

        This method computes both the absolute profit/loss amount and the percentage gain/loss
        for a stock position. It performs input validation to ensure all parameters are valid
        numeric values.

        Args:
            quantity: Number of shares held (must be positive integer)
            average_cost: Average cost per share (must be positive number)
            price: Current market price per share (must be positive number)

        Returns:
            dict: Contains profit_loss (float) and profit_loss_percentage (float)

        Raises:
            ValueError: If any parameter is invalid (negative, zero, or non-numeric)
        """
        # Validate quantity parameter
        if not isinstance(quantity, int):
            raise ValueError(f"quantity must be an integer, got {type(quantity).__name__}")
        if quantity <= 0:
            raise ValueError(f"quantity must be positive, got {quantity}")

        # Validate average_cost parameter
        if not isinstance(average_cost, (int, float)):
            raise ValueError(f"average_cost must be a number, got {type(average_cost).__name__}")
        if average_cost <= 0:
            raise ValueError(f"average_cost must be positive, got {average_cost}")

        # Validate price parameter
        if not isinstance(price, (int, float)):
            raise ValueError(f"price must be a number, got {type(price).__name__}")
        if price <= 0:
            raise ValueError(f"price must be positive, got {price}")

        # Calculate total cost basis (initial investment)
        total_cost = quantity * average_cost

        # Calculate current market value of the position
        current_value = quantity * price

        # Calculate absolute profit/loss
        # Positive value indicates profit, negative indicates loss
        profit_loss = current_value - total_cost

        # Calculate profit/loss percentage
        # Formula: ((current_value - total_cost) / total_cost) * 100
        # This represents the percentage gain or loss relative to the initial investment
        profit_loss_percentage = (profit_loss / total_cost) * 100

        # Round results to 2 decimal places for currency precision
        profit_loss = round(profit_loss, 2)
        profit_loss_percentage = round(profit_loss_percentage, 2)

        # Return results as a dictionary matching the schema
        return {
            "profit_loss": profit_loss,
            "profit_loss_percentage": profit_loss_percentage
        }

    @is_tool()
    def calculate_moving_average(self, prices: list, period: int):
        """
        Calculates the moving average of stock prices over a specified period.

        This method computes a simple moving average (SMA) by taking the mean of 
        the most recent 'period' number of prices from the provided price array.

        Args:
            prices: Array of historical closing prices (floats or ints)
            period: Number of periods for moving average calculation

        Returns:
            dict: Dictionary containing the calculated moving average value

        Raises:
            ValueError: If prices list is empty, period is invalid, or insufficient data
        """
        # Validate that prices list is not empty
        if not prices:
            raise ValueError("Prices list cannot be empty")

        # Validate that all prices are numeric (int or float)
        if not all(isinstance(price, (int, float)) for price in prices):
            raise ValueError("All prices must be numeric values (int or float)")

        # Validate that period is a positive integer
        if not isinstance(period, int) or period <= 0:
            raise ValueError("Period must be a positive integer")

        # Check if we have enough data points to calculate the moving average
        if len(prices) < period:
            raise ValueError(
                f"Insufficient data: need at least {period} prices to calculate "
                f"{period}-period moving average, but only {len(prices)} prices provided"
            )

        # Extract the most recent 'period' number of prices
        # For example, if prices = [100, 101, 102, 103, 104] and period = 3,
        # we take the last 3 prices: [102, 103, 104]
        recent_prices = prices[-period:]

        # Calculate the simple moving average (SMA)
        # SMA = sum of recent prices / number of periods
        moving_average = sum(recent_prices) / period

        # Round to 2 decimal places for cleaner output (typical for financial data)
        moving_average = round(moving_average, 2)

        # Return the result in the specified format
        return {
            "moving_average": moving_average
        }

    @is_tool()
    def create_stop_loss_order(self, user_id: str, ticker_symbol: str, quantity: int, stop_price: float):
        """
        Creates a stop-loss order to automatically sell when price falls to specified level.

        This method:
        1. Validates the input parameters
        2. Verifies the user account exists
        3. Generates a unique order ID
        4. Creates a new stop-loss order with monitoring status
        5. Stores the order in the database
        6. Returns the order details with creation timestamp
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

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

        # Access the database
        db = self.db

        # Verify user account exists
        user_account_table = getattr(db, "user_account", None)
        if user_account_table is None:
            raise ValueError("user_account table not found in database")

        if user_id not in user_account_table:
            raise ValueError(f"User account with user_id '{user_id}' does not exist")

        # Generate unique order ID
        order_id_prefix = "order_"
        order_id = order_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        created_at = datetime.now()

        # Create the stop-loss order
        # Stop-loss orders are sell orders that trigger when price falls to stop_price
        new_order = StockOrder(
            order_id=order_id,
            user_id=user_id,
            ticker_symbol=ticker_symbol.upper(),  # Normalize ticker symbol to uppercase
            order_type="stop_loss",
            order_side="sell",  # Stop-loss orders are typically sell orders
            quantity=quantity,
            limit_price=None,  # Stop-loss orders don't have limit prices
            stop_price=stop_price,
            status="monitoring",  # Initial status is monitoring the price
            filled_quantity=0,  # No shares filled yet
            average_fill_price=None,  # No execution price yet
            created_at=created_at,
            filled_at=None,  # Not filled yet
            cancelled_at=None  # Not cancelled
        )

        # Store the order in the database
        stock_order_table = getattr(db, "stock_order", None)
        if stock_order_table is None:
            # Initialize the table if it doesn't exist
            stock_order_table = {}

        # Add the new order to the table
        stock_order_table[order_id] = new_order
        setattr(db, "stock_order", stock_order_table)

        # Return the order details in the required format
        return {
            "order_id": order_id,
            "status": "monitoring",
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def check_buying_power(self, user_id: str, order_value: float):
        """
        Checks if user has sufficient buying power for a proposed trade.

        This method validates whether a user has enough buying power to execute
        a trade of the specified value. It retrieves the user's account information
        from the database and compares their available buying power against the
        requested order value.

        Args:
            user_id: Unique identifier of the user
            order_value: Total value of the proposed order in USD

        Returns:
            dict: Contains three fields:
                - has_sufficient_power (bool): Whether user has sufficient buying power
                - buying_power (float): User's available buying power
                - shortfall (float): Amount short if insufficient (0 if sufficient)

        Raises:
            ValueError: If user_id is empty/None, order_value is negative,
                       or user account is not found
        """
        # Input validation
        if not user_id or not isinstance(user_id, str) or user_id.strip() == "":
            raise ValueError("user_id must be a non-empty string")

        if not isinstance(order_value, (int, float)):
            raise ValueError("order_value must be a number")

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

        # Access the database
        db = self.db

        # Get user_account table from database
        user_account_table = getattr(db, "user_account", None)

        if user_account_table is None:
            raise ValueError("user_account table not found in database")

        # Retrieve user account information
        user_account = user_account_table.get(user_id)

        if user_account is None:
            raise ValueError(f"User account not found for user_id: {user_id}")

        # Get user's buying power
        buying_power = user_account.buying_power

        # Calculate if user has sufficient buying power
        has_sufficient_power = buying_power >= order_value

        # Calculate shortfall (0 if sufficient, positive value if insufficient)
        shortfall = max(0.0, order_value - buying_power)

        # Return the result
        return {
            "has_sufficient_power": has_sufficient_power,
            "buying_power": buying_power,
            "shortfall": shortfall
        }

    @is_tool()
    def get_user_watchlists(self, user_id: str):
        """
        Retrieves all watchlists for a specific user

        Args:
            user_id: Unique identifier of the user

        Returns:
            Dictionary containing list of user watchlists with their details

        Raises:
            ValueError: If user_id is invalid or empty
        """
        # Validate user_id parameter
        if not user_id or not isinstance(user_id, str):
            raise ValueError("Invalid user_id: must be a non-empty string")

        # Strip whitespace from user_id
        user_id = user_id.strip()
        if not user_id:
            raise ValueError("Invalid user_id: cannot be empty or whitespace only")

        # Access the database instance
        db = self.db

        # Get the watchlist table from database
        watchlist_table = getattr(db, "watchlist", None)

        # If watchlist table doesn't exist or is empty, return empty list
        if watchlist_table is None or not watchlist_table:
            return {"watchlists": []}

        # Initialize list to store user's watchlists
        user_watchlists = []

        # Iterate through all watchlists in the table
        for watchlist_id, watchlist_item in watchlist_table.items():
            # Check if this watchlist belongs to the specified user
            if watchlist_item.user_id == user_id:
                # Get stock count for this watchlist
                # Since the schema doesn't include a stocks field, we'll need to check
                # if there's a related table. For now, we'll set stock_count to 0
                # as we don't have stock information in the current schema
                stock_count = 0

                # Create watchlist info dictionary
                watchlist_info = {
                    "watchlist_id": watchlist_item.watchlist_id,
                    "watchlist_name": watchlist_item.watchlist_name,
                    "stock_count": stock_count
                }

                # Add to user's watchlists
                user_watchlists.append(watchlist_info)

        # Return the list of watchlists
        return {"watchlists": user_watchlists}

    @is_tool()
    def get_real_time_stock_price(self, ticker_symbol: str):
        """
        Retrieves the current real-time price of a specific stock by its ticker symbol.

        This method queries the stock_price_history table to find the most recent price
        record for the given ticker symbol and returns the current price with timestamp.

        Args:
            ticker_symbol: The stock ticker symbol (e.g., AAPL, GOOGL)

        Returns:
            dict: Contains ticker_symbol, current_price, and timestamp

        Raises:
            ValueError: If ticker_symbol is invalid or no price data found
        """
        from datetime import datetime

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

        # Normalize ticker symbol to uppercase for consistency
        ticker_symbol = ticker_symbol.strip().upper()

        # Access the database
        db = self.db

        # Get stock_price_history table
        stock_price_history = getattr(db, "stock_price_history", None)

        # Check if table exists and has data
        if stock_price_history is None or len(stock_price_history) == 0:
            raise ValueError(f"No price data available in the database")

        # Find all price records for the given ticker symbol
        matching_records = []
        for price_history_id, record in stock_price_history.items():
            if record.ticker_symbol.upper() == ticker_symbol:
                matching_records.append(record)

        # Check if any records found for this ticker
        if not matching_records:
            raise ValueError(f"No price data found for ticker symbol: {ticker_symbol}")

        # Find the most recent record by comparing datetime objects
        # Sort by date in descending order to get the latest record
        latest_record = max(matching_records, key=lambda x: x.date)

        # Extract the current price (using close price as the current price)
        # In real-time scenarios, the most recent close price represents the current price
        current_price = latest_record.close

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

        # Construct and return the result
        return {
            "ticker_symbol": ticker_symbol,
            "current_price": current_price,
            "timestamp": timestamp
        }

    @is_tool()
    def get_market_indices(self):
        """
        Retrieves current values and changes for major market indices.

        This method returns simulated real-time data for major market indices including
        their current values, absolute changes, and percentage changes. The data represents
        major US and international market indices.

        Returns:
            dict: A dictionary containing:
                - indices (list): List of market index dictionaries, each containing:
                    - index_name (str): Name of the market index
                    - value (float): Current index value
                    - change (float): Absolute change from previous close
                    - change_percentage (float): Percentage change from previous close

        Raises:
            RuntimeError: If there is an error retrieving market index data
        """
        try:
            # Import required modules for generating realistic market data
            import random

            # Define major market indices with their typical value ranges
            # These represent the most commonly tracked market indices
            major_indices = [
                {
                    "name": "S&P 500",
                    "base_value": 4750.0,
                    "volatility": 50.0  # Typical daily range
                },
                {
                    "name": "Dow Jones Industrial Average",
                    "base_value": 37500.0,
                    "volatility": 300.0
                },
                {
                    "name": "NASDAQ Composite",
                    "base_value": 14800.0,
                    "volatility": 150.0
                },
                {
                    "name": "Russell 2000",
                    "base_value": 2050.0,
                    "volatility": 30.0
                },
                {
                    "name": "FTSE 100",
                    "base_value": 7650.0,
                    "volatility": 80.0
                },
                {
                    "name": "DAX",
                    "base_value": 16500.0,
                    "volatility": 200.0
                },
                {
                    "name": "Nikkei 225",
                    "base_value": 33000.0,
                    "volatility": 400.0
                },
                {
                    "name": "Shanghai Composite",
                    "base_value": 3050.0,
                    "volatility": 40.0
                }
            ]

            # Generate current market data for each index
            indices_data = []

            for index_info in major_indices:
                # Generate a realistic change value within the volatility range
                # Using a normal distribution to simulate realistic market movements
                change = random.uniform(-index_info["volatility"], index_info["volatility"])

                # Calculate current value based on base value and change
                current_value = index_info["base_value"] + change

                # Calculate percentage change
                # Formula: (change / previous_close) * 100
                previous_close = index_info["base_value"]
                change_percentage = (change / previous_close) * 100

                # Round values to appropriate decimal places for readability
                # Index values typically shown to 2 decimal places
                # Percentage changes shown to 2 decimal places
                index_data = {
                    "index_name": index_info["name"],
                    "value": round(current_value, 2),
                    "change": round(change, 2),
                    "change_percentage": round(change_percentage, 2)
                }

                indices_data.append(index_data)

            # Return the compiled market indices data
            return {
                "indices": indices_data
            }

        except Exception as e:
            # Catch any unexpected errors and raise as RuntimeError
            # This ensures consistent error handling as specified in the schema
            raise RuntimeError(f"Failed to retrieve market indices data: {str(e)}")

    @is_tool()
    def calculate_position_risk(self, quantity: int, price: float, volatility: float, confidence_level: float):
        """
        Calculates risk metrics for a specific position including value at risk (VaR).

        This method computes:
        1. Position value: Total market value of the position
        2. Value at Risk (VaR): Maximum expected loss at a given confidence level
        3. Risk percentage: VaR as a percentage of position value

        The VaR is calculated using the parametric (variance-covariance) method,
        which assumes normal distribution of returns.

        Args:
            quantity: Number of shares held (must be positive integer)
            price: Current market price per share (must be positive)
            volatility: Historical volatility as standard deviation (must be positive)
            confidence_level: Confidence level for VaR calculation (e.g., 0.95 for 95%)
                             Must be between 0 and 1

        Returns:
            dict: Dictionary containing:
                - value_at_risk: Value at Risk (VaR) amount in USD
                - position_value: Total position value in USD
                - risk_percentage: Risk as percentage of position value

        Raises:
            ValueError: If any input parameter is invalid (negative values, 
                       confidence_level outside [0,1] range, etc.)
        """
        import math

        # Validate input parameters
        if quantity <= 0:
            raise ValueError(f"Quantity must be positive, got {quantity}")

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

        if volatility < 0:
            raise ValueError(f"Volatility must be non-negative, got {volatility}")

        if not (0 < confidence_level < 1):
            raise ValueError(f"Confidence level must be between 0 and 1, got {confidence_level}")

        # Calculate position value
        # Position value = quantity * current price
        position_value = float(quantity * price)

        # Calculate z-score for the given confidence level
        # Using inverse normal distribution (quantile function)
        # For common confidence levels:
        # 90% -> z ≈ 1.28, 95% -> z ≈ 1.645, 99% -> z ≈ 2.33
        # We use the error function (erf) to approximate the inverse normal distribution
        z_score = math.sqrt(2) * self._inverse_erf(2 * confidence_level - 1)

        # Calculate Value at Risk (VaR)
        # VaR = Position Value * z-score * volatility
        # This represents the maximum expected loss at the given confidence level
        # assuming normal distribution of returns
        value_at_risk = position_value * z_score * volatility

        # Calculate risk as percentage of position value
        # This helps understand what proportion of the position is at risk
        risk_percentage = (value_at_risk / position_value) * 100 if position_value > 0 else 0.0

        # Return risk metrics
        return {
            "value_at_risk": round(value_at_risk, 2),
            "position_value": round(position_value, 2),
            "risk_percentage": round(risk_percentage, 2)
        }

    def _inverse_erf(self, x):
        """
        Helper method to calculate inverse error function (erf^-1).

        This is used to compute z-scores from confidence levels.
        Uses a polynomial approximation for computational efficiency.

        Args:
            x: Input value between -1 and 1

        Returns:
            float: Inverse error function value

        Raises:
            ValueError: If x is outside [-1, 1] range
        """
        import math

        if not (-1 <= x <= 1):
            raise ValueError(f"Input to inverse_erf must be in [-1, 1], got {x}")

        # Handle boundary cases
        if x == 1:
            return float('inf')
        if x == -1:
            return float('-inf')
        if x == 0:
            return 0.0

        # Use polynomial approximation for inverse error function
        # This is based on the Abramowitz and Stegun approximation
        a = 0.147

        # Calculate intermediate values
        ln_term = math.log(1 - x * x)
        first_term = (2 / (math.pi * a)) + (ln_term / 2)
        second_term = ln_term / a

        # Calculate result
        result = math.sqrt(-first_term + math.sqrt(first_term * first_term - second_term))

        # Apply sign based on input
        return result if x >= 0 else -result

    @is_tool()
    def get_sector_performance(self, time_period: Literal["1D", "1W", "1M", "3M", "1Y", "YTD"]):
        """
        Retrieves performance metrics for different market sectors based on the specified time period.

        This method calculates sector performance by:
        1. Validating the time_period parameter against allowed values
        2. Calculating the start date based on the time period
        3. Aggregating stock price data by sector
        4. Computing return and volatility metrics for each sector

        Args:
            time_period: Time period for performance calculation, must be one of ["1D", "1W", "1M", "3M", "1Y", "YTD"]

        Returns:
            dict: Contains 'sector_performance' key with list of sector performance metrics
                  Each sector entry includes: sector name, return (decimal), and volatility (decimal)

        Raises:
            ValueError: If time_period is not in the allowed enum values
        """
        from datetime import datetime, timedelta
        from collections import defaultdict
        import math

        # Validate time_period parameter
        allowed_periods = ["1D", "1W", "1M", "3M", "1Y", "YTD"]
        if time_period not in allowed_periods:
            raise ValueError(f"Invalid time_period: {time_period}. Must be one of {allowed_periods}")

        # Get current date for calculations
        current_date = datetime.now()

        # Calculate start date based on time_period
        if time_period == "1D":
            start_date = current_date - timedelta(days=1)
        elif time_period == "1W":
            start_date = current_date - timedelta(weeks=1)
        elif time_period == "1M":
            start_date = current_date - timedelta(days=30)
        elif time_period == "3M":
            start_date = current_date - timedelta(days=90)
        elif time_period == "1Y":
            start_date = current_date - timedelta(days=365)
        elif time_period == "YTD":
            # Year-to-date: from January 1st of current year
            start_date = datetime(current_date.year, 1, 1)

        # Access database
        db = self.db

        # Get stock fundamentals to map tickers to sectors
        stock_fundamentals = getattr(db, "stock_fundamental", None)
        if stock_fundamentals is None:
            stock_fundamentals = {}

        # Get stock price history
        stock_price_history = getattr(db, "stock_price_history", None)
        if stock_price_history is None:
            stock_price_history = {}

        # Group price data by sector
        # Structure: {sector: {ticker: [(date, close_price), ...]}}
        sector_data = defaultdict(lambda: defaultdict(list))

        # Iterate through price history and organize by sector
        for price_id, price_record in stock_price_history.items():
            ticker = price_record.ticker_symbol
            price_date = price_record.date
            close_price = price_record.close

            # Filter by date range
            if price_date < start_date:
                continue

            # Get sector for this ticker
            fundamental = stock_fundamentals.get(ticker)
            if fundamental and fundamental.sector:
                sector = fundamental.sector
                sector_data[sector][ticker].append((price_date, close_price))

        # Calculate performance metrics for each sector
        sector_performance = []

        for sector, tickers_data in sector_data.items():
            # Calculate sector-level returns and volatility
            sector_returns = []

            for ticker, price_list in tickers_data.items():
                if len(price_list) < 2:
                    # Need at least 2 data points to calculate return
                    continue

                # Sort by date
                price_list.sort(key=lambda x: x[0])

                # Calculate return for this ticker: (final_price - initial_price) / initial_price
                initial_price = price_list[0][1]
                final_price = price_list[-1][1]

                if initial_price > 0:
                    ticker_return = (final_price - initial_price) / initial_price
                    sector_returns.append(ticker_return)

            if not sector_returns:
                # No valid data for this sector
                continue

            # Calculate average return for the sector
            avg_return = sum(sector_returns) / len(sector_returns)

            # Calculate volatility (standard deviation of returns)
            if len(sector_returns) > 1:
                mean_return = avg_return
                variance = sum((r - mean_return) ** 2 for r in sector_returns) / len(sector_returns)
                volatility = math.sqrt(variance)
            else:
                # If only one ticker, set volatility to 0
                volatility = 0.0

            # Add sector performance to results
            sector_performance.append({
                "sector": sector,
                "return": round(avg_return, 6),  # Round to 6 decimal places for precision
                "volatility": round(volatility, 6)
            })

        # Sort by sector name for consistent output
        sector_performance.sort(key=lambda x: x["sector"])

        return {
            "sector_performance": sector_performance
        }

    @is_tool()
    def cancel_order(self, order_id: str):
        """
        Cancels an existing pending order by updating its status to 'cancelled'
        and recording the cancellation timestamp.

        Args:
            order_id: Unique identifier of the order to cancel

        Returns:
            dict: Contains order_id, status ('cancelled'), and cancelled_at timestamp

        Raises:
            ValueError: If order doesn't exist or is not in a cancellable status
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the stock_order table
        stock_order_table = getattr(db, 'stock_order', None)
        if stock_order_table is None:
            raise ValueError("Stock order table not found in database")

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

        # Get the order object
        order = stock_order_table[order_id]

        # Check if the order is in a cancellable status
        # Orders can only be cancelled if they are in 'pending' or 'monitoring' status
        # Orders that are 'filled', 'cancelled', or 'rejected' cannot be cancelled
        cancellable_statuses = ['pending', 'monitoring']
        if order.status not in cancellable_statuses:
            raise ValueError(
                f"Order with ID '{order_id}' cannot be cancelled. "
                f"Current status is '{order.status}', but only orders with status "
                f"{cancellable_statuses} can be cancelled"
            )

        # Record the cancellation timestamp
        cancellation_time = datetime.now()

        # Update the order status and cancellation timestamp
        order.status = 'cancelled'
        order.cancelled_at = cancellation_time

        # Update the order in the database
        stock_order_table[order_id] = order
        setattr(db, 'stock_order', stock_order_table)

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

        # Return the result
        return {
            "order_id": order_id,
            "status": "cancelled",
            "cancelled_at": cancelled_at_str
        }

    @is_tool()
    def calculate_portfolio_diversification(self, positions: list):
        """
        Calculates portfolio diversification metrics including sector and position weights.

        Args:
            positions: Array of portfolio positions with sector and market value.
                      Each position should contain: ticker_symbol, sector, market_value

        Returns:
            Dictionary containing:
            - sector_weights: Percentage allocation by sector
            - position_weights: Percentage allocation by position
            - concentration_risk: Risk level based on concentration (low, medium, high)

        Raises:
            ValueError: If positions is empty or invalid data is provided
        """
        # Validate input
        if not positions or not isinstance(positions, list):
            raise ValueError("Positions must be a non-empty list")

        if len(positions) == 0:
            raise ValueError("Positions list cannot be empty")

        # Validate each position has required fields
        for i, pos in enumerate(positions):
            if not isinstance(pos, dict):
                raise ValueError(f"Position at index {i} must be a dictionary")
            if "ticker_symbol" not in pos:
                raise ValueError(f"Position at index {i} missing required field: ticker_symbol")
            if "sector" not in pos:
                raise ValueError(f"Position at index {i} missing required field: sector")
            if "market_value" not in pos:
                raise ValueError(f"Position at index {i} missing required field: market_value")
            if not isinstance(pos["market_value"], (int, float)) or pos["market_value"] < 0:
                raise ValueError(f"Position at index {i} has invalid market_value: must be a non-negative number")

        # Calculate total portfolio value
        total_value = sum(pos["market_value"] for pos in positions)

        if total_value == 0:
            raise ValueError("Total portfolio value cannot be zero")

        # Calculate sector weights
        sector_totals = {}
        for pos in positions:
            sector = pos["sector"]
            market_value = pos["market_value"]
            sector_totals[sector] = sector_totals.get(sector, 0) + market_value

        # Convert sector totals to percentages
        sector_weights = {
            sector: round((value / total_value) * 100, 1)
            for sector, value in sector_totals.items()
        }

        # Calculate position weights
        position_weights = []
        for pos in positions:
            weight = round((pos["market_value"] / total_value) * 100, 1)
            position_weights.append({
                "ticker_symbol": pos["ticker_symbol"],
                "weight": weight
            })

        # Sort position weights by weight descending for better readability
        position_weights.sort(key=lambda x: x["weight"], reverse=True)

        # Calculate concentration risk
        # Risk assessment based on:
        # 1. Single position concentration (if any position > 40% = high risk)
        # 2. Top 3 positions concentration (if > 70% = high risk, > 50% = medium risk)
        # 3. Sector concentration (if any sector > 60% = high risk, > 40% = medium risk)
        # 4. Number of positions (< 5 positions = higher risk)

        max_position_weight = max(pw["weight"] for pw in position_weights)
        max_sector_weight = max(sector_weights.values())
        num_positions = len(positions)

        # Calculate top 3 positions concentration
        top_3_concentration = sum(pw["weight"] for pw in position_weights[:min(3, len(position_weights))])

        # Determine risk level
        risk_factors = 0

        # Check single position concentration
        if max_position_weight > 40:
            risk_factors += 3  # High risk factor
        elif max_position_weight > 25:
            risk_factors += 2  # Medium risk factor
        elif max_position_weight > 15:
            risk_factors += 1  # Low risk factor

        # Check top 3 positions concentration
        if top_3_concentration > 70:
            risk_factors += 2
        elif top_3_concentration > 50:
            risk_factors += 1

        # Check sector concentration
        if max_sector_weight > 60:
            risk_factors += 2
        elif max_sector_weight > 40:
            risk_factors += 1

        # Check number of positions (diversification level)
        if num_positions < 5:
            risk_factors += 2
        elif num_positions < 10:
            risk_factors += 1

        # Determine final risk level based on accumulated risk factors
        if risk_factors >= 5:
            concentration_risk = "high"
        elif risk_factors >= 2:
            concentration_risk = "medium"
        else:
            concentration_risk = "low"

        return {
            "sector_weights": sector_weights,
            "position_weights": position_weights,
            "concentration_risk": concentration_risk
        }

    @is_tool()
    def calculate_rsi(self, prices: list, period: int):
        """
        Calculates the Relative Strength Index (RSI) indicator for stock price momentum analysis.

        The RSI is a momentum oscillator that measures the speed and magnitude of price changes.
        It ranges from 0 to 100, with values above 70 typically indicating overbought conditions
        and values below 30 indicating oversold conditions.

        Algorithm:
        1. Calculate price changes between consecutive periods
        2. Separate gains (positive changes) and losses (negative changes)
        3. Calculate average gain and average loss over the specified period
        4. Calculate Relative Strength (RS) = Average Gain / Average Loss
        5. Calculate RSI = 100 - (100 / (1 + RS))

        Args:
            prices: Array of historical closing prices
            period: Number of periods for RSI calculation (typically 14)

        Returns:
            dict: Contains 'rsi_value' - RSI indicator value between 0 and 100

        Raises:
            ValueError: If prices list is too short, period is invalid, or prices contain invalid values
        """

        # Validate input parameters
        if not prices or len(prices) < 2:
            raise ValueError("Prices list must contain at least 2 values")

        if period <= 0:
            raise ValueError("Period must be a positive integer")

        if len(prices) < period + 1:
            raise ValueError(f"Prices list must contain at least {period + 1} values for RSI calculation with period {period}")

        # Validate all prices are valid numbers
        try:
            prices = [float(p) for p in prices]
        except (TypeError, ValueError):
            raise ValueError("All prices must be valid numeric values")

        if any(p < 0 for p in prices):
            raise ValueError("Prices cannot be negative")

        # Calculate price changes (deltas) between consecutive periods
        deltas = []
        for i in range(1, len(prices)):
            delta = prices[i] - prices[i - 1]
            deltas.append(delta)

        # Separate gains and losses
        # Gains are positive changes, losses are absolute values of negative changes
        gains = [delta if delta > 0 else 0 for delta in deltas]
        losses = [-delta if delta < 0 else 0 for delta in deltas]

        # Calculate initial average gain and average loss using simple moving average
        # for the first 'period' values
        if len(gains) < period:
            raise ValueError(f"Not enough price data to calculate RSI with period {period}")

        # Initial averages: simple average of first 'period' gains/losses
        avg_gain = sum(gains[:period]) / period
        avg_loss = sum(losses[:period]) / period

        # For subsequent periods, use the Wilder smoothing method:
        # New Average = (Previous Average * (period - 1) + Current Value) / period
        for i in range(period, len(gains)):
            avg_gain = (avg_gain * (period - 1) + gains[i]) / period
            avg_loss = (avg_loss * (period - 1) + losses[i]) / period

        # Calculate RSI
        # Handle edge case where average loss is zero (all gains, no losses)
        if avg_loss == 0:
            if avg_gain == 0:
                # No price movement at all
                rsi_value = 50.0
            else:
                # Only gains, no losses - RSI approaches 100
                rsi_value = 100.0
        else:
            # Standard RSI calculation
            # RS (Relative Strength) = Average Gain / Average Loss
            rs = avg_gain / avg_loss
            # RSI = 100 - (100 / (1 + RS))
            rsi_value = 100.0 - (100.0 / (1.0 + rs))

        # Ensure RSI is within valid range [0, 100]
        rsi_value = max(0.0, min(100.0, rsi_value))

        # Round to one decimal place for readability
        rsi_value = round(rsi_value, 1)

        return {
            "rsi_value": rsi_value
        }

    @is_tool()
    def get_order_status(self, order_id: str):
        """
        Retrieves the current status and details of a specific order.

        Args:
            order_id: Unique identifier of the order to retrieve

        Returns:
            Dictionary containing complete order information including:
            - order_id: Unique identifier of the order
            - ticker_symbol: Stock ticker symbol
            - order_type: Type of order (market, limit, stop_loss)
            - order_side: Order side (buy, sell)
            - quantity: Number of shares
            - status: Current order status
            - filled_quantity: Number of shares filled
            - average_fill_price: Average execution price

        Raises:
            ValueError: If order_id is invalid or order does not exist
        """
        # Validate input parameter
        if not order_id or not isinstance(order_id, str):
            raise ValueError("Invalid order_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Get the stock_order table from database
        stock_order_table = getattr(db, "stock_order", None)

        # Check if stock_order table exists
        if stock_order_table is None:
            raise ValueError("Stock order table not found in database")

        # Retrieve the order by order_id
        order = stock_order_table.get(order_id)

        # Check if order exists
        if order is None:
            raise ValueError(f"Order with order_id '{order_id}' does not exist")

        # Construct the return dictionary with all required fields
        result = {
            "order_id": order.order_id,
            "ticker_symbol": order.ticker_symbol,
            "order_type": order.order_type,
            "order_side": order.order_side,
            "quantity": order.quantity,
            "status": order.status,
            "filled_quantity": order.filled_quantity,
            "average_fill_price": order.average_fill_price
        }

        return result

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

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

        # Retrieve watchlist table from database
        watchlist_table = getattr(db, "watchlist", None)
        if watchlist_table is None:
            raise ValueError("Watchlist table not found in database")

        # Check if watchlist exists
        if watchlist_id not in watchlist_table:
            raise ValueError(f"Watchlist with ID '{watchlist_id}' not found")

        # Get watchlist information
        watchlist = watchlist_table[watchlist_id]
        watchlist_name = watchlist.watchlist_name

        # Retrieve watchlist_stock table from database
        watchlist_stock_table = getattr(db, "watchlist_stock", None)
        if watchlist_stock_table is None:
            raise ValueError("WatchlistStock table not found in database")

        # Find all stocks associated with this watchlist
        stocks_list = []
        for watchlist_stock_id, watchlist_stock in watchlist_stock_table.items():
            # Check if this stock belongs to the requested watchlist
            if watchlist_stock.watchlist_id == watchlist_id:
                # Note: The tool schema indicates current prices should be returned,
                # but there is no stock price table in related_databases.
                # We can only return ticker symbols from watchlist_stock table.
                # Price data would need to come from an external source or another tool.
                stock_info = {
                    "ticker_symbol": watchlist_stock.ticker_symbol,
                    "current_price": 0.0,
                    "change": 0.0,
                    "change_percentage": 0.0
                }
                stocks_list.append(stock_info)

        # Construct and return the result
        result = {
            "watchlist_id": watchlist_id,
            "watchlist_name": watchlist_name,
            "stocks": stocks_list
        }

        return result

    @is_tool()
    def get_stock_news(self, ticker_symbol: str, limit: int = 10):
        """
        Retrieves recent news articles related to a specific stock.

        This method fetches news articles from external news APIs or sources
        based on the provided ticker symbol. It does not interact with the database
        as this is a read operation for external market data.

        Args:
            ticker_symbol: Stock ticker symbol (e.g., "AAPL", "GOOGL")
            limit: Maximum number of news articles to return (default: 10)

        Returns:
            dict: Dictionary containing list of news articles with format:
                  {
                      "news_articles": [
                          {
                              "title": str,
                              "source": str,
                              "published_at": str (format: "yyyy-mm-dd HH:MM:SS"),
                              "url": str
                          },
                          ...
                      ]
                  }

        Raises:
            ValueError: If ticker_symbol is empty/invalid or limit is non-positive
        """
        import requests
        from datetime import datetime, timedelta

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

        # Normalize ticker symbol to uppercase for consistency
        ticker_symbol = ticker_symbol.strip().upper()

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

        # Initialize result structure
        news_articles = []

        try:
            # Attempt to fetch news from a financial news API
            # Note: In production, you would use a real API like Alpha Vantage, NewsAPI, or Yahoo Finance
            # Here we demonstrate the structure with a mock implementation

            # Option 1: Using Alpha Vantage NEWS_SENTIMENT endpoint (example)
            # API_KEY would typically be stored in environment variables
            # url = f"https://www.alphavantage.co/query?function=NEWS_SENTIMENT&tickers={ticker_symbol}&apikey={API_KEY}&limit={limit}"

            # Option 2: Using NewsAPI (example)
            # url = f"https://newsapi.org/v2/everything?q={ticker_symbol}&sortBy=publishedAt&apiKey={API_KEY}&pageSize={limit}"

            # For demonstration, we'll simulate a successful API response structure
            # In production, replace this with actual API call:
            # response = requests.get(url, timeout=10)
            # response.raise_for_status()
            # data = response.json()

            # Mock implementation: Generate sample news data
            # This simulates what would be returned from a real news API
            current_time = datetime.now()

            # Sample news sources
            sources = ["Reuters", "Bloomberg", "CNBC", "Financial Times", "Wall Street Journal", 
                       "MarketWatch", "Yahoo Finance", "Seeking Alpha"]

            # Generate mock news articles based on limit
            for i in range(min(limit, 10)):  # Cap at 10 for mock data
                # Create timestamp for news article (recent dates)
                published_time = current_time - timedelta(hours=i*2, minutes=i*15)

                article = {
                    "title": f"{ticker_symbol} Stock Analysis and Market Update #{i+1}",
                    "source": sources[i % len(sources)],
                    "published_at": published_time.strftime("%Y-%m-%d %H:%M:%S"),
                    "url": f"https://example.com/news/{ticker_symbol.lower()}/article-{i+1}"
                }
                news_articles.append(article)

            # In production with real API, you would parse the response:
            # Example for NewsAPI structure:
            # if "articles" in data:
            #     for article in data["articles"][:limit]:
            #         parsed_date = datetime.strptime(article["publishedAt"], "%Y-%m-%dT%H:%M:%SZ")
            #         news_articles.append({
            #             "title": article.get("title", ""),
            #             "source": article.get("source", {}).get("name", "Unknown"),
            #             "published_at": parsed_date.strftime("%Y-%m-%d %H:%M:%S"),
            #             "url": article.get("url", "")
            #         })

        except requests.exceptions.RequestException as e:
            # Handle network-related errors
            raise ValueError(f"Failed to fetch news for ticker {ticker_symbol}: {str(e)}")
        except Exception as e:
            # Handle any other unexpected errors
            raise ValueError(f"Error retrieving news articles: {str(e)}")

        # Return the structured result
        return {
            "news_articles": news_articles
        }

    @is_tool()
    def calculate_tax_liability(self, short_term_gains: float, long_term_gains: float, short_term_tax_rate: float, long_term_tax_rate: float):
        """
        Calculates estimated tax liability on realized capital gains.

        This method computes the tax liability separately for short-term and long-term capital gains,
        then returns the total tax liability along with the breakdown.

        Args:
            short_term_gains: Short-term capital gains (held < 1 year) in USD
            long_term_gains: Long-term capital gains (held >= 1 year) in USD
            short_term_tax_rate: Tax rate for short-term gains as decimal (e.g., 0.24 for 24%)
            long_term_tax_rate: Tax rate for long-term gains as decimal (e.g., 0.15 for 15%)

        Returns:
            dict: A dictionary containing:
                - total_tax_liability: Total estimated tax liability
                - short_term_tax: Tax on short-term gains
                - long_term_tax: Tax on long-term gains

        Raises:
            ValueError: If any of the gains or tax rates are negative, or if tax rates exceed 1.0
            TypeError: If any of the parameters are not numeric
        """

        # Validate input types
        try:
            short_term_gains = float(short_term_gains)
            long_term_gains = float(long_term_gains)
            short_term_tax_rate = float(short_term_tax_rate)
            long_term_tax_rate = float(long_term_tax_rate)
        except (TypeError, ValueError) as e:
            raise TypeError(f"All parameters must be numeric values: {str(e)}")

        # Validate that gains are non-negative
        if short_term_gains < 0:
            raise ValueError(f"Short-term gains cannot be negative: {short_term_gains}")

        if long_term_gains < 0:
            raise ValueError(f"Long-term gains cannot be negative: {long_term_gains}")

        # Validate that tax rates are within valid range [0, 1]
        if short_term_tax_rate < 0 or short_term_tax_rate > 1:
            raise ValueError(f"Short-term tax rate must be between 0 and 1: {short_term_tax_rate}")

        if long_term_tax_rate < 0 or long_term_tax_rate > 1:
            raise ValueError(f"Long-term tax rate must be between 0 and 1: {long_term_tax_rate}")

        # Calculate tax on short-term capital gains
        # Short-term gains are typically taxed at ordinary income tax rates
        short_term_tax = short_term_gains * short_term_tax_rate

        # Calculate tax on long-term capital gains
        # Long-term gains usually benefit from preferential tax rates
        long_term_tax = long_term_gains * long_term_tax_rate

        # Calculate total tax liability
        total_tax_liability = short_term_tax + long_term_tax

        # Round all values to 2 decimal places for currency precision
        short_term_tax = round(short_term_tax, 2)
        long_term_tax = round(long_term_tax, 2)
        total_tax_liability = round(total_tax_liability, 2)

        # Return the tax breakdown
        return {
            "total_tax_liability": total_tax_liability,
            "short_term_tax": short_term_tax,
            "long_term_tax": long_term_tax
        }

    @is_tool()
    def create_price_alert(self, user_id: str, ticker_symbol: str, target_price: float, condition: Literal["above", "below"]):
        """
        Creates a price alert that triggers when stock reaches specified price level.

        This method:
        1. Validates all required input parameters
        2. Generates a unique alert_id
        3. Creates a new PriceAlert instance with status 'active'
        4. Stores the alert in the database
        5. Returns alert details including alert_id, status, and creation timestamp

        Args:
            user_id: Unique identifier of the user creating the alert
            ticker_symbol: Stock ticker symbol to monitor
            target_price: Price level that triggers the alert
            condition: Alert condition, must be either "above" or "below"

        Returns:
            dict: Contains alert_id, status, and created_at timestamp

        Raises:
            ValueError: If any required parameter is invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

        # Validate target_price
        if not isinstance(target_price, (int, float)):
            raise ValueError("target_price must be a number")
        if target_price <= 0:
            raise ValueError("target_price must be greater than 0")

        # Validate condition (enum parameter with safety protection)
        # The Literal type annotation already enforces this at type-check time,
        # but we add runtime validation for safety
        valid_conditions = ["above", "below"]
        if condition not in valid_conditions:
            raise ValueError(f"condition must be one of {valid_conditions}, got '{condition}'")

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

        # Get current timestamp
        created_at = datetime.now()

        # Access the database
        db = self.db

        # Get the price_alert table from database
        price_alert_table = getattr(db, "price_alert", None)
        if price_alert_table is None:
            # Initialize empty table if it doesn't exist
            price_alert_table = {}

        # Import PriceAlert class (already imported at file header)
        # Create new PriceAlert instance
        new_alert = PriceAlert(
            alert_id=alert_id,
            user_id=user_id,
            ticker_symbol=ticker_symbol.upper(),  # Normalize ticker to uppercase
            target_price=float(target_price),
            condition=condition,
            status="active",  # New alerts start with 'active' status
            created_at=created_at,
            triggered_at=None  # Not triggered yet
        )

        # Add the new alert to the table
        price_alert_table[alert_id] = new_alert

        # Save the updated table back to database
        setattr(db, "price_alert", price_alert_table)

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

        # Return alert details
        return {
            "alert_id": alert_id,
            "status": "active",
            "created_at": created_at_str
        }
