from datetime import datetime
import json
import requests
from loguru import logger
from dataclasses import dataclass
from datetime import timezone
import time
from requests.adapters import HTTPAdapter
from urllib3.util.retry import Retry

# Configure loguru logger
logger.add(
    "solana_transactions.log",
    rotation="500 MB",
    retention="10 days",
    level="INFO",
    format="{time} {level} {message}",
)

# Reliable public RPC endpoints
RPC_ENDPOINTS = [
    "https://api.mainnet-beta.solana.com",
    "https://solana.public-rpc.com",
    "https://rpc.ankr.com/solana",
]


@dataclass
class TransactionError:
    """Data class to represent transaction errors"""

    error_type: str
    message: str
    timestamp: str = datetime.now(timezone.utc).isoformat()


class SolanaAPIException(Exception):
    """Custom exception for Solana API related errors"""

    pass


def create_http_session() -> requests.Session:
    """
    Creates a requests session with retry logic and timeouts
    """
    session = requests.Session()

    # Configure retry strategy
    retry_strategy = Retry(
        total=3,
        backoff_factor=0.5,
        status_forcelist=[429, 500, 502, 503, 504],
    )

    adapter = HTTPAdapter(max_retries=retry_strategy)
    session.mount("http://", adapter)
    session.mount("https://", adapter)

    return session


def get_working_endpoint(session: requests.Session) -> str:
    """
    Tests endpoints and returns the first working one.

    Args:
        session: requests.Session object with retry logic

    Returns:
        str: Working RPC endpoint URL

    Raises:
        SolanaAPIException: If no working endpoint is found
    """
    for endpoint in RPC_ENDPOINTS:
        try:
            payload = {
                "jsonrpc": "2.0",
                "id": 1,
                "method": "getHealth",
            }
            response = session.post(endpoint, json=payload, timeout=5)
            if response.status_code == 200:
                logger.info(f"Using RPC endpoint: {endpoint}")
                return endpoint
        except Exception as e:
            logger.warning(
                f"Endpoint {endpoint} failed health check: {str(e)}"
            )
            continue

    raise SolanaAPIException("No working RPC endpoints found")


def fetch_wallet_transactions(wallet_address: str) -> str:
    """
    Fetches all transactions for a given Solana wallet address using public RPC endpoints.

    Args:
        wallet_address (str): The Solana wallet address to fetch transactions for
            Example: "CtBLg4AX6LQfKVtPPUWqJyQ5cRfHydUwuZZ87rmojA1P"

    Returns:
        str: JSON string containing the list of transactions and their details
            Format: {
                "success": bool,
                "transactions": List[Dict],
                "error": Optional[Dict]
            }
    """
    try:
        # Validate wallet address format (basic check)
        if (
            not isinstance(wallet_address, str)
            or len(wallet_address) != 44
        ):
            raise ValueError(
                f"Invalid Solana wallet address format: {wallet_address}"
            )

        logger.info(
            f"Fetching transactions for wallet: {wallet_address}"
        )

        # Create session with retry logic
        session = create_http_session()

        # Get working endpoint
        api_endpoint = get_working_endpoint(session)

        # Initialize variables for pagination
        all_transactions = []
        before_signature = None
        limit = 25  # Smaller batch size to be more conservative

        while True:
            try:
                # Prepare request payload
                payload = {
                    "jsonrpc": "2.0",
                    "id": "1",
                    "method": "getSignaturesForAddress",
                    "params": [
                        wallet_address,
                        {"limit": limit, "before": before_signature},
                    ],
                }

                # Make API request
                response = session.post(
                    api_endpoint, json=payload, timeout=10
                )

                data = response.json()

                if "error" in data:
                    error_code = data.get("error", {}).get("code")
                    if error_code == 429:  # Rate limit
                        time.sleep(1)  # Wait before trying again
                        continue

                    raise SolanaAPIException(
                        f"API error: {data['error']}"
                    )

                # Extract transactions from response
                transactions = data.get("result", [])

                if not transactions:
                    break

                # Add transactions to our list
                all_transactions.extend(transactions)

                # Update pagination cursor
                before_signature = transactions[-1]["signature"]

                logger.info(
                    f"Fetched {len(transactions)} transactions. Total: {len(all_transactions)}"
                )

                # Break if we received fewer transactions than the limit
                if len(transactions) < limit:
                    break

                # Add small delay between batches
                time.sleep(0.2)

            except Exception as e:
                logger.error(
                    f"Error during transaction fetch: {str(e)}"
                )
                # Try to get a new endpoint if the current one fails
                api_endpoint = get_working_endpoint(session)
                continue

        # Enrich transaction data with additional details
        enriched_transactions = []
        for tx in all_transactions:
            try:
                tx_payload = {
                    "jsonrpc": "2.0",
                    "id": "1",
                    "method": "getTransaction",
                    "params": [
                        tx["signature"],
                        {
                            "encoding": "json",
                            "maxSupportedTransactionVersion": 0,
                        },
                    ],
                }

                response = session.post(
                    api_endpoint, json=tx_payload, timeout=10
                )
                tx_data = response.json()

                if "result" in tx_data and tx_data["result"]:
                    enriched_transactions.append(
                        {
                            "signature": tx["signature"],
                            "slot": tx["slot"],
                            "timestamp": tx["blockTime"],
                            "status": (
                                "success"
                                if not tx.get("err")
                                else "error"
                            ),
                            "details": tx_data["result"],
                        }
                    )

                # Small delay between transaction fetches
                time.sleep(0.1)

                # print(tx)
                logger.info(f"Enriched transaction: {tx}")

            except Exception as e:
                logger.warning(
                    f"Failed to fetch details for transaction {tx['signature']}: {str(e)}"
                )
                continue

        logger.info(
            f"Successfully fetched and enriched {len(enriched_transactions)} transactions"
        )

        return json.dumps(
            {
                "success": True,
                "transactions": enriched_transactions,
                "error": None,
            }
        )

    except SolanaAPIException as e:
        error = TransactionError(
            error_type="API_ERROR", message=str(e)
        )
        logger.error(f"API error: {error.message}")
        return json.dumps(
            {
                "success": False,
                "transactions": [],
                "error": error.__dict__,
            }
        )

    except Exception as e:
        error = TransactionError(
            error_type="UNKNOWN_ERROR",
            message=f"An unexpected error occurred: {str(e)}",
        )
        logger.error(f"Unexpected error: {error.message}")
        return json.dumps(
            {
                "success": False,
                "transactions": [],
                "error": error.__dict__,
            }
        )


# Example usage
if __name__ == "__main__":
    wallet = "CtBLg4AX6LQfKVtPPUWqJyQ5cRfHydUwuZZ87rmojA1P"

    try:
        result = fetch_wallet_transactions(wallet)
        print(json.dumps(json.loads(result), indent=2))
    except Exception as e:
        logger.error(f"Failed to fetch transactions: {str(e)}")