mirror of
https://github.com/SpaceVim/SpaceVim.git
synced 2025-01-24 06:10:05 +08:00
167 lines
5.6 KiB
Python
167 lines
5.6 KiB
Python
# ============================================================================
|
|
# FILE: logger.py
|
|
# AUTHOR: Tommy Allen <tommy at esdf.io>
|
|
# License: MIT license
|
|
# ============================================================================
|
|
import sys
|
|
import time
|
|
import logging
|
|
import typing
|
|
|
|
from deoplete.util import Nvim
|
|
|
|
from functools import wraps
|
|
from collections import defaultdict
|
|
|
|
log_format = '%(asctime)s %(levelname)-8s [%(process)d] (%(name)s) %(message)s'
|
|
log_message_cooldown = 0.5
|
|
|
|
root = logging.getLogger('deoplete')
|
|
root.propagate = False
|
|
init = False
|
|
|
|
FUNC = typing.Callable[..., typing.Any]
|
|
|
|
|
|
def getLogger(name: str) -> logging.Logger:
|
|
"""Get a logger that is a child of the 'root' logger.
|
|
"""
|
|
return root.getChild(name)
|
|
|
|
|
|
def setup(vim: Nvim, level: str, output_file: str = '') -> None:
|
|
"""Setup logging for Deoplete
|
|
"""
|
|
global init
|
|
if init:
|
|
return
|
|
init = True
|
|
|
|
if output_file:
|
|
formatter = logging.Formatter(log_format)
|
|
handler = logging.FileHandler(filename=output_file)
|
|
handler.setFormatter(formatter)
|
|
handler.addFilter(DeopleteLogFilter(vim))
|
|
root.addHandler(handler)
|
|
|
|
level = str(level).upper()
|
|
if level not in ('DEBUG', 'INFO', 'WARN', 'WARNING', 'ERROR',
|
|
'CRITICAL', 'FATAL'):
|
|
level = 'DEBUG'
|
|
root.setLevel(getattr(logging, level))
|
|
|
|
try:
|
|
import pkg_resources
|
|
|
|
pynvim_version = pkg_resources.get_distribution('pynvim').version
|
|
except Exception:
|
|
pynvim_version = 'unknown'
|
|
|
|
log = getLogger('logging')
|
|
log.info('--- Deoplete Log Start ---')
|
|
log.info('%s, Python %s, pynvim %s',
|
|
vim.call('deoplete#util#neovim_version'),
|
|
'.'.join(map(str, sys.version_info[:3])),
|
|
pynvim_version)
|
|
|
|
if 'deoplete#_logging_notified' not in vim.vars:
|
|
vim.vars['deoplete#_logging_notified'] = 1
|
|
vim.call('deoplete#util#print_debug', 'Logging to %s' % (
|
|
output_file))
|
|
|
|
|
|
def logmethod(func: FUNC) -> typing.Callable[[FUNC], FUNC]:
|
|
"""Decorator for setting up the logger in LoggingMixin subclasses.
|
|
|
|
This does not guarantee that log messages will be generated. If
|
|
`LoggingMixin.is_debug_enabled` is True, it will be propagated up to the
|
|
root 'deoplete' logger.
|
|
"""
|
|
@wraps(func)
|
|
def wrapper(self, # type: ignore
|
|
*args: typing.Any,
|
|
**kwargs: typing.Any) -> typing.Any:
|
|
if not init or not self.is_debug_enabled:
|
|
return
|
|
if self._logger is None:
|
|
self._logger = getLogger(getattr(self, 'name', 'unknown'))
|
|
return func(self, *args, **kwargs)
|
|
return wrapper
|
|
|
|
|
|
class LoggingMixin(object):
|
|
"""Class that adds logging functions to a subclass.
|
|
"""
|
|
is_debug_enabled = False
|
|
_logger = None # type: logging.Logger
|
|
|
|
@logmethod
|
|
def debug(self, msg: str,
|
|
*args: typing.Any, **kwargs: typing.Any) -> None:
|
|
self._logger.debug(msg, *args, **kwargs)
|
|
|
|
@logmethod
|
|
def info(self, msg: str,
|
|
*args: typing.Any, **kwargs: typing.Any) -> None:
|
|
self._logger.info(msg, *args, **kwargs)
|
|
|
|
@logmethod
|
|
def warning(self, msg: str,
|
|
*args: typing.Any, **kwargs: typing.Any) -> None:
|
|
self._logger.warning(msg, *args, **kwargs)
|
|
warn = warning
|
|
|
|
@logmethod
|
|
def error(self, msg: str,
|
|
*args: typing.Any, **kwargs: typing.Any) -> None:
|
|
self._logger.error(msg, *args, **kwargs)
|
|
|
|
@logmethod
|
|
def exception(self, msg: str,
|
|
*args: typing.Any, **kwargs: typing.Any) -> None:
|
|
# This will not produce a log message if there is no exception to log.
|
|
self._logger.exception(msg, *args, **kwargs)
|
|
|
|
@logmethod
|
|
def critical(self, msg: str,
|
|
*args: typing.Any, **kwargs: typing.Any) -> None:
|
|
self._logger.critical(msg, *args, **kwargs)
|
|
fatal = critical
|
|
|
|
|
|
class DeopleteLogFilter(logging.Filter):
|
|
def __init__(self, vim: Nvim, name: str = ''):
|
|
self.vim = vim
|
|
self.counter: typing.Dict[str, int] = defaultdict(int)
|
|
self.last_message_time: float = 0
|
|
self.last_message: typing.Tuple[typing.Any, ...] = ()
|
|
|
|
def filter(self, record: logging.LogRecord) -> bool:
|
|
t = time.time()
|
|
elapsed = t - self.last_message_time
|
|
self.last_message_time = t
|
|
|
|
message = (record.levelno, record.name, record.msg, record.args)
|
|
if message == self.last_message and elapsed < log_message_cooldown:
|
|
# Ignore if the same message comes in too fast.
|
|
return False
|
|
self.last_message = message
|
|
|
|
if record.levelno >= logging.ERROR:
|
|
self.counter[record.name] += 1
|
|
if self.counter[record.name] <= 2:
|
|
# Only permit 2 errors in succession from a logging source to
|
|
# display errors inside of Neovim. After this, it is no longer
|
|
# permitted to emit any more errors and should be addressed.
|
|
self.vim.call('deoplete#util#print_error', record.getMessage(),
|
|
record.name)
|
|
if record.exc_info and record.stack_info:
|
|
# Add a penalty for messages that generate exceptions to avoid
|
|
# making the log harder to read with doubled stack traces.
|
|
self.counter[record.name] += 1
|
|
elif self.counter[record.name] < 2:
|
|
# If below the threshold for silencing a logging source, reset its
|
|
# counter.
|
|
self.counter[record.name] = 0
|
|
return True
|