from __future__ import annotations

import logging
import os
from pathlib import Path, PurePath

from rich.logging import RichHandler

_SET_UP_LOGGERS = set()
_ADDITIONAL_HANDLERS = []

logging.TRACE = 5  # type: ignore
logging.addLevelName(logging.TRACE, "TRACE")  # type: ignore


def _interpret_level_from_env(level: str | None, *, default=logging.DEBUG) -> int:
    if not level:
        return default
    if level.isnumeric():
        return int(level)
    return getattr(logging, level.upper())


# Change default log level to DEBUG
_STREAM_LEVEL = _interpret_level_from_env(os.environ.get("SWE_AGENT_LOG_STREAM_LEVEL"), default=logging.DEBUG)
_FILE_LEVEL = _interpret_level_from_env(os.environ.get("SWE_AGENT_LOG_FILE_LEVEL"), default=logging.DEBUG)


def get_logger(name: str) -> logging.Logger:
    """Get logger. Use this instead of `logging.getLogger` to ensure
    that the logger is set up with the correct handlers.
    """
    logger = logging.getLogger(name)
    if name in _SET_UP_LOGGERS:
        # Already set up
        return logger
        
    # Rich handler setup
    handler = RichHandler(
        show_time=True,  # Always show time
        show_path=True,  # Show log location
        markup=True,  # Enable Rich markup
    )
    handler.setLevel(_STREAM_LEVEL)
    
    # Formatter setup
    formatter = logging.Formatter(
        '%(asctime)s [%(levelname)s] %(name)s: %(message)s',
        datefmt='%Y-%m-%d %H:%M:%S'
    )
    handler.setFormatter(formatter)
    
    logger.setLevel(min(_STREAM_LEVEL, _FILE_LEVEL))
    logger.addHandler(handler)
    logger.propagate = False
    _SET_UP_LOGGERS.add(name)
    
    # Apply additional handlers
    for handler in _ADDITIONAL_HANDLERS:
        logger.addHandler(handler)
        
    return logger


def _get_next_log_file(base_path: Path) -> Path:
    """Get the next available log file path by appending a number if the file already exists."""
    if not base_path.exists():
        return base_path
    
    directory = base_path.parent
    stem = base_path.stem
    suffix = base_path.suffix
    counter = 1
    
    while True:
        new_path = directory / f"{stem}_{counter}{suffix}"
        if not new_path.exists():
            return new_path
        counter += 1


def add_file_handler(path: PurePath | str) -> None:
    """Adds a file handler to all loggers that we have set up
    and all future loggers that will be set up with `get_logger`.
    If the log file already exists, creates a new file with an incremented number.
    """
    path_obj = Path(path)
    actual_path = _get_next_log_file(path_obj)
    
    handler = logging.FileHandler(actual_path)
    formatter = logging.Formatter("%(asctime)s %(levelname)s %(message)s")
    handler.setFormatter(formatter)
    handler.setLevel(_FILE_LEVEL)
    for name in _SET_UP_LOGGERS:
        logger = logging.getLogger(name)
        logger.addHandler(handler)
    _ADDITIONAL_HANDLERS.append(handler)


default_logger = get_logger("swe-agent")
