# Copyright (c) 2024 westlake-repl
# SPDX-License-Identifier: MIT

import logging
import os
import sys
import colorlog
import re
import torch
from REDRec.utils.utils import get_local_time, ensure_dir
from colorama import init

log_colors_config = {
    'DEBUG': 'cyan',
    'WARNING': 'yellow',
    'ERROR': 'red',
    'CRITICAL': 'red',
}


class RemoveColorFilter(logging.Filter):

    def filter(self, record):
        if record:
            ansi_escape = re.compile(r'\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])')
            record.msg = ansi_escape.sub('', str(record.msg))
        return True


def set_color(log, color, highlight=True):
    color_set = ['black', 'red', 'green', 'yellow', 'blue', 'pink', 'cyan', 'white']
    try:
        index = color_set.index(color)
    except:
        index = len(color_set) - 1
    prev_log = '\033['
    if highlight:
        prev_log += '1;3'
    else:
        prev_log += '0;3'
    prev_log += str(index) + 'm'
    return prev_log + log + '\033[0m'


def init_logger(config):
    """
    A logger that can show a message on standard output and write it into the
    file named `filename` simultaneously.
    All the message that you want to log MUST be str.

    Args:
        config (Config): An instance object of Config, used to record parameter information.

    Example:
        >>> logger = logging.getLogger(config)
        >>> logger.debug(train_state)
        >>> logger.info(train_result)
    """
    
    init(autoreset=True)

    LOGROOT = os.path.join(config.saver.log_dir, config.saver.saved_model_name, 'logger')
    rank = torch.distributed.get_rank()
    if rank == 0:
        ensure_dir(LOGROOT)
    
    
    torch.distributed.barrier()
    logfilename = '{}.log'.format(get_local_time())

    logfilepath = os.path.join(LOGROOT, logfilename)
    if config.get('log_path', None):
        logfilepath = os.path.join(LOGROOT, config['log_path'])

    filefmt = "%(asctime)-15s %(levelname)s  %(message)s"
    filedatefmt = "%a %d %b %Y %H:%M:%S"
    fileformatter = logging.Formatter(filefmt, filedatefmt)

    sfmt = "%(log_color)s%(asctime)-15s %(levelname)s  %(message)s"
    sdatefmt = "%d %b %H:%M"
    sformatter = colorlog.ColoredFormatter(sfmt, sdatefmt, log_colors=log_colors_config)
    state = config.get('state', None)
    if not state or state.lower() == 'info':
        level = logging.INFO
    elif state.lower() == 'debug':
        level = logging.DEBUG
    elif state.lower() == 'error':
        level = logging.ERROR
    elif state.lower() == 'warning':
        level = logging.WARNING
    elif state.lower() == 'critical':
        level = logging.CRITICAL
    else:
        level = logging.INFO

    fh = logging.FileHandler(logfilepath)
    fh.setLevel(level)
    fh.setFormatter(fileformatter)
    remove_color_filter = RemoveColorFilter()
    fh.addFilter(remove_color_filter)

    sh = logging.StreamHandler(sys.stdout)
    sh.setLevel(level)
    sh.setFormatter(sformatter)

    logging.basicConfig(level=level if rank in [-1, 0] else logging.WARN, handlers=[sh, fh])
