import json
import logging
import random

import numpy as np
import torch
from hydra.core.hydra_config import HydraConfig
from lightning.pytorch import seed_everything
from omegaconf import OmegaConf


def init_seed(seed=42):
    # Python random
    random.seed(seed)
    # Numpy
    np.random.seed(seed)
    # Pytorch
    torch.manual_seed(seed)
    torch.cuda.manual_seed(seed)
    seed_everything(seed)


def setup_logger(name: str = __name__, level: int = logging.INFO) -> logging.Logger:
    """
    Configure a logger with the given name and level.

    Args:
        name: Name of the logger
        level: Logging level

    Returns:
        The configured logger
    """
    # Set the root logger level first to ensure all loggers inherit this level by default
    root_logger = logging.getLogger()
    root_logger.setLevel(level)

    # Then configure the specific logger
    logger = logging.getLogger(name)
    logger.setLevel(level)
    logger.handlers.clear()
    # When using hydra, the logger is already set up by hydra itself.
    # out_handler = logging.StreamHandler(sys.stdout)
    # out_handler.setLevel(logging.DEBUG)
    #
    # err_handler = logging.StreamHandler(sys.stderr)
    # err_handler.setLevel(logging.ERROR)
    #
    # logger.addHandler(out_handler)
    # logger.addHandler(err_handler)
    return logger


def set_all_loggers_level(level: int) -> None:
    """
    Set the log level for all existing loggers in the project.

    Args:
        level: Logging level to set
    """
    # Set root logger level
    logging.getLogger().setLevel(level)

    # Set all existing loggers to the specified level
    for logger_name in logging.root.manager.loggerDict:
        if logger_name.startswith("research.wsl_ece"):
            logger = logging.getLogger(logger_name)
            logger.setLevel(level)


def num_job():
    return HydraConfig.get().job.num if not OmegaConf.is_missing(HydraConfig.get().job, "num") else 1


def setup_device_config(device="default"):
    """Set up the device config.

    This function avoids using multiple GPUs because of the limitation that trainer with DDPSampler doesn't correctly
    aggregate metrics. See https://github.com/Lightning-AI/pytorch-lightning/issues/12862
    """
    if device == "default":
        return (
            [num_job() % torch.cuda.device_count()]
            if torch.cuda.device_count()
            else [0]
            if torch.cuda.is_available()
            else "auto"
        )
    elif device in ["dp", "auto"]:
        return "auto"
    else:
        return json.loads(str(device))
