import datetime import functools import inspect import logging import os import shutil import subprocess from termcolor import colored def add_fillers(text, filler="=", fill_side="both"): terminal_width = shutil.get_terminal_size().columns text = text.strip() text_width = len(text) if text_width >= terminal_width: return text if fill_side[0].lower() == "b": leading_fill_str = filler * ((terminal_width - text_width) // 2 - 1) + " " trailing_fill_str = " " + filler * ( terminal_width - text_width - len(leading_fill_str) - 1 ) elif fill_side[0].lower() == "l": leading_fill_str = filler * (terminal_width - text_width - 1) + " " trailing_fill_str = "" elif fill_side[0].lower() == "r": leading_fill_str = "" trailing_fill_str = " " + filler * (terminal_width - text_width - 1) else: raise ValueError("Invalid fill_side") filled_str = f"{leading_fill_str}{text}{trailing_fill_str}" return filled_str class Logger: LOG_METHODS = { "err": ("error", "red"), "warn": ("warning", "light_red"), "note": ("info", "light_magenta"), "mesg": ("info", "light_cyan"), "file": ("info", "light_blue"), "line": ("info", "white"), "success": ("info", "light_green"), "fail": ("info", "light_red"), "back": ("debug", "light_cyan"), } INDENT_METHODS = [ "indent", "set_indent", "reset_indent", "store_indent", "restore_indent", "log_indent", ] LEVEL_METHODS = [ "set_level", "store_level", "restore_level", "quiet", "enter_quiet", "exit_quiet", ] LEVEL_NAMES = { "critical": logging.CRITICAL, "error": logging.ERROR, "warning": logging.WARNING, "info": logging.INFO, "debug": logging.DEBUG, } def __init__(self, name=None, prefix=False): if not name: frame = inspect.stack()[1] module = inspect.getmodule(frame[0]) name = module.__name__ self.logger = logging.getLogger(name) self.logger.setLevel(logging.INFO) if prefix: formatter_prefix = "[%(asctime)s] - [%(name)s] - [%(levelname)s]\n" else: formatter_prefix = "" self.formatter = logging.Formatter(formatter_prefix + "%(message)s") self.handler = logging.StreamHandler() self.handler.setLevel(logging.INFO) self.handler.setFormatter(self.formatter) self.logger.addHandler(self.handler) self.log_indent = 0 self.log_indents = [] self.log_level = "info" self.log_levels = [] self.bind_functions() def indent(self, indent=2): self.log_indent += indent def set_indent(self, indent=2): self.log_indent = indent def reset_indent(self): self.log_indent = 0 def store_indent(self): self.log_indents.append(self.log_indent) def restore_indent(self): self.log_indent = self.log_indents.pop(-1) def set_level(self, level): self.log_level = level self.logger.setLevel(self.LEVEL_NAMES[level]) def store_level(self): self.log_levels.append(self.log_level) def restore_level(self): self.log_level = self.log_levels.pop(-1) self.set_level(self.log_level) def quiet(self): self.set_level("critical") def enter_quiet(self, quiet=False): if quiet: self.store_level() self.quiet() def exit_quiet(self, quiet=False): if quiet: self.restore_level() def log(self, method, msg, indent=0, fill=False, fill_side="both", *args, **kwargs): if type(msg) == str: msg_str = msg else: msg_str = repr(msg) quotes = ["'", '"'] if msg_str[0] in quotes and msg_str[-1] in quotes: msg_str = msg_str[1:-1] msg_lines = msg_str.splitlines() self.store_indent() self.indent(indent) indent_str = " " * self.log_indent self.restore_indent() level, color = self.LOG_METHODS[method] indented_msg = "\n".join([f"{indent_str}{line}" for line in msg_lines]) if fill: indented_msg = add_fillers(indented_msg, fill_side=fill_side) getattr(self.logger, level)(colored(indented_msg, color), *args, **kwargs) def bind_functions(self): for method in self.LOG_METHODS: setattr(self.logger, method, functools.partial(self.log, method)) for method in self.INDENT_METHODS: setattr(self.logger, method, getattr(self, method)) for method in self.LEVEL_METHODS: setattr(self.logger, method, getattr(self, method)) logger = Logger().logger def shell_cmd(cmd, getoutput=False, showcmd=True, env=None): if showcmd: logger.info(colored(f"\n$ [{os.getcwd()}]", "light_blue")) logger.info(colored(f" $ {cmd}\n", "light_cyan")) if getoutput: output = subprocess.getoutput(cmd, env=env) return output else: subprocess.run(cmd, shell=True, env=env) class Runtimer: def __enter__(self): self.t1, _ = self.start_time() return self def __exit__(self, exc_type, exc_value, traceback): self.t2, _ = self.end_time() self.elapsed_time(self.t2 - self.t1) def start_time(self): t1 = datetime.datetime.now() self.logger_time("start", t1) return t1, self.time2str(t1) def end_time(self): t2 = datetime.datetime.now() self.logger_time("end", t2) return t2, self.time2str(t2) def elapsed_time(self, dt=None): if dt is None: dt = self.t2 - self.t1 self.logger_time("elapsed", dt) return dt, self.time2str(dt) def logger_time(self, time_type, t): time_types = { "start": "Start", "end": "End", "elapsed": "Elapsed", } time_str = add_fillers( colored( f"{time_types[time_type]} time: [ {self.time2str(t)} ]", "light_magenta", ), fill_side="both", ) logger.line(time_str) # Convert time to string def time2str(self, t): datetime_str_format = "%Y-%m-%d %H:%M:%S" if isinstance(t, datetime.datetime): return t.strftime(datetime_str_format) elif isinstance(t, datetime.timedelta): hours = t.seconds // 3600 hour_str = f"{hours} hr" if hours > 0 else "" minutes = (t.seconds // 60) % 60 minute_str = f"{minutes:>2} min" if minutes > 0 else "" seconds = t.seconds % 60 second_str = f"{seconds:>2} s" time_str = " ".join([hour_str, minute_str, second_str]).strip() return time_str else: return str(t)