import logging
from collections import deque
from copy import deepcopy
from decimal import Decimal
from typing import Callable, Deque, List

from trade import Trade

# Set up a dedicated logger for FIFOQueue
logger = logging.getLogger(__name__)


class FIFOQueue:
    """
    Crypto trading FIFO queue.

    Will track trades.
    """

    def __init__(self) -> None:
        self.__queue: Deque[Trade] = deque()
        self._cached_total: Decimal = Decimal(0)
        self._cache_valid: bool = True
        logger.info("FIFOQueue initialized with empty queue.")

    def __len__(self) -> int:
        return len(self.__queue)

    def get_copy(self) -> List[Trade]:
        """
        Helper for unit tests, to check internal state.
        """
        return list(deepcopy(self.__queue))

    def add(self, amount: Decimal, total_cost: Decimal, date: str) -> None:
        """
        Add a trade to the queue.
        """
        trade = Trade(amount, total_cost, date)
        self.__queue.append(trade)
        self._cache_valid = False
        logger.info(f"Added trade: {trade}.")

    def add_trade(self, trade: Trade) -> None:
        """
        Add a trade to the queue.
        """
        self.__queue.append(trade)
        self._cache_valid = False
        logger.info(f"Added trade: {trade}.")

    def remove_coins(self, amount: float | Decimal) -> List[Trade]:
        """
        Remove a specified amount of coins from the queue, returning the
        trades used to buy. This can be used to calculate profit/loss.
        """
        if amount <= 0:
            logger.error("Attempted to remove non-positive amount.")
            raise ValueError("The amount to remove must be positive.")

        amount = Decimal(amount)

        if amount > self.get_remaining_amount():
            logger.error(f"Insufficient assets to process sale of {amount}.")
            raise ValueError(
                f"Insufficient assets in queue to process sale of {amount}."
            )

        logger.debug(f"Removing {amount:.2f} coins from the queue.")
        logger.info("Cache invalidated before removing coins.")
        self._cache_valid = False

        remaining: Decimal = amount
        entries: List[Trade] = []

        while remaining > 0:
            trade = self.__queue[0]
            logger.debug(f"Processing trade: {trade}")

            if trade.amount > remaining:
                ppc = trade.price_per_coin
                trade.remove_coins(remaining)
                entries.append(Trade(remaining, remaining * ppc, trade.date))
                logger.info(f"Partial removal from trade: {remaining:.2f} coins.")
                break
            else:
                remaining -= trade.amount
                entries.append(trade)
                self.__queue.popleft()
                logger.info(
                    f"Removed full trade: {trade}. Remaining coins to remove: {remaining}"
                )

        return entries

    def get_remaining_amount(self) -> Decimal:
        """
        Calculate the total remaining amount in the queue.
        """
        if not self._cache_valid:
            logger.debug("Cache invalid, recalculating remaining amount.")
            self._cached_total = sum(
                (trade.amount for trade in self.__queue), Decimal(0)
            )
            self._cache_valid = True
            logger.info(f"Cache recalculated: {self._cached_total:.2f}")

        logger.debug(f"Returning cached remaining amount: {self._cached_total:.2f}")
        return self._cached_total

    def remove(self, predicate: Callable[[Trade], bool]) -> Trade:
        """
        Remove a trade from the queue based on a given predicate.

        Args:
            predicate (Callable[[Trade], bool]): A function that returns True for the trade to remove.

        Returns:
            Trade: The removed trade.

        Raises:
            ValueError: If no trade matches the predicate or multiple trades are found.
        """
        # Use filter to find matching trades
        matching_trades = list(filter(predicate, self.__queue))

        if len(matching_trades) == 0:
            logger.error("No matching trade found for removal.")
            raise ValueError("No trade matches the given predicate.")
        elif len(matching_trades) > 1:
            logger.error("Multiple matching trades found for removal.")
            raise ValueError(
                "Multiple trades match the given predicate. Please refine your criteria."
            )

        # Locate the exact match in the original queue
        trade_to_remove = matching_trades[0]
        self.__queue.remove(trade_to_remove)
        self._cache_valid = False
        logger.info(f"Removed trade: {trade_to_remove}.")
        return trade_to_remove