123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116 |
- """
- Internal logging utility. Adapted from
- https://github.com/theroyallab/tabbyAPI/blob/4cc0b59bdc94e6342b6d1d7acadbadc63c740ed9/common/logger.py
- """
- import logging
- from loguru import logger
- from rich.console import Console
- from rich.markup import escape
- from rich.progress import (
- Progress,
- TextColumn,
- BarColumn,
- TimeRemainingColumn,
- TaskProgressColumn,
- MofNCompleteColumn,
- )
- RICH_CONSOLE = Console()
- def unwrap(wrapped, default=None):
- """Unwrap function for Optionals."""
- if wrapped is None:
- return default
- return wrapped
- def get_loading_progress_bar():
- """Gets a pre-made progress bar for loading tasks."""
- return Progress(
- TextColumn("[progress.description]{task.description}"),
- BarColumn(),
- TaskProgressColumn(),
- MofNCompleteColumn(),
- TimeRemainingColumn(),
- console=RICH_CONSOLE,
- )
- def _log_formatter(record: dict):
- """Log message formatter."""
- color_map = {
- "TRACE": "dim blue",
- "DEBUG": "cyan",
- "INFO": "green",
- "SUCCESS": "bold green",
- "WARNING": "yellow",
- "ERROR": "red",
- "CRITICAL": "bold white on red",
- }
- level = record.get("level")
- level_color = color_map.get(level.name, "cyan")
- colored_level = f"[{level_color}]{level.name}[/{level_color}]:"
- separator = " " * (9 - len(level.name))
- message = unwrap(record.get("message"), "")
- # Replace once loguru allows for turning off str.format
- message = message.replace("{{", "{{").replace("}}", "}}")
- # Manually escape < and > characters
- message = message.replace("<", "\\<").replace(">", "\\>")
- message = escape(message)
- lines = message.splitlines()
- fmt = ""
- if len(lines) > 1:
- fmt = "\n".join(
- [f"{colored_level}{separator}{line}" for line in lines])
- else:
- fmt = f"{colored_level}{separator}{message}"
- return fmt
- # Uvicorn log handler
- # Uvicorn log portions inspired from https://github.com/encode/uvicorn/discussions/2027#discussioncomment-6432362
- class UvicornLoggingHandler(logging.Handler):
- def emit(self, record: logging.LogRecord) -> None:
- logger.opt(exception=record.exc_info).log(record.levelname,
- self.format(record).rstrip())
- # Uvicorn config for logging. Passed into run when creating all loggers in server
- UVICORN_LOG_CONFIG = {
- "version": 1,
- "disable_existing_loggers": False,
- "handlers": {
- "uvicorn": {
- "class":
- f"{UvicornLoggingHandler.__module__}.{UvicornLoggingHandler.__qualname__}", # noqa
- },
- },
- "root": {
- "handlers": ["uvicorn"],
- "propagate": False,
- "level": "INFO"
- },
- }
- def setup_logger():
- """Bootstrap the logger."""
- logger.remove()
- logger.add(
- RICH_CONSOLE.print,
- level="INFO",
- format=_log_formatter,
- colorize=True,
- )
|