#! -*- coding: utf-8
import logging
import os
import os.path as path
import sys
import typing
from enum import IntEnum
from typing import TextIO

__all__ = ["LOGFORMAT", "LogLevel", "config_logger"]

# LOGFORMAT = "[%(asctime)s] %(process)d - %(name)s - <%(levelname)s>  [%(module)s.%(funcName)s#%(lineno)d] %(message)s"
LOGFORMAT = "[%(asctime)s] %(process)d %(name)s <%(levelname)s> [%(module)s.%(funcName)s#%(lineno)d] %(message)s"


class LogLevel(IntEnum):
    TRACE = 5
    DEBUG = 10
    INFO = 20
    WARNING = 30
    ERROR = 40
    CRITICAL = 50

    @classmethod
    def nameof(cls, name: str):
        name = name.upper()
        for e in cls:
            if e.name == name:
                return e
        raise ValueError(f"Unknown {cls.name} name: {name}")

    @classmethod
    def value_of(cls, value: int):
        for e in cls:
            if e.value == value:
                return e
        raise ValueError(f"Unknown {cls.name} name: {value}")

    def __str__(self):
        return self.name.upper()


# add Trace log
logging.addLevelName(LogLevel.TRACE.value, LogLevel.TRACE.name)
# set loglevel name.
setattr(logging, LogLevel.TRACE.name, LogLevel.TRACE.value)


def __trace__(msg, *args, **kwargs):
    """ Define trace log function """
    logging.log(LogLevel.TRACE.value, msg, *args, **kwargs)


logging.trace = __trace__


def config_logger(logformat: str = LOGFORMAT,
                  loglevel: LogLevel = LogLevel.INFO,
                  stream: TextIO = sys.stderr,
                  logfile: str = None, filemode="w", filters: typing.List[typing.Callable] = []):
    # log message formatter.
    formatter = logging.Formatter(fmt=logformat, style="%")

    handlers = []  # log handers
    if stream:
        # set stream handler
        handler = logging.StreamHandler(stream=stream)
        handler.setLevel(loglevel)
        handler.setFormatter(formatter)
        for filter in filters:
            handler.addFilter(filter)
        handlers.append(handler)

    if logfile:
        dirname = path.dirname(path.abspath(logfile))
        os.makedirs(dirname, exist_ok=True)
        handler = logging.FileHandler(logfile, mode=filemode)
        handler.setLevel(loglevel)
        handler.setFormatter(formatter)
        for filter in filters:
            handler.addFilter(filter)
        handlers.append(handler)

    # set root logger.
    # logging.basicConfig(level=logging.NOTSET, handlers=handlers)
    logging.basicConfig(level=int(loglevel), handlers=handlers)
