From d1772d410235592b482e3b08b1863f6624d9fe6b Mon Sep 17 00:00:00 2001 From: Daniel Baumann Date: Sun, 19 Feb 2023 15:52:21 +0100 Subject: Adding upstream version 2.0.3. Signed-off-by: Daniel Baumann --- deluge/log.py | 348 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 348 insertions(+) create mode 100644 deluge/log.py (limited to 'deluge/log.py') diff --git a/deluge/log.py b/deluge/log.py new file mode 100644 index 0000000..75e8308 --- /dev/null +++ b/deluge/log.py @@ -0,0 +1,348 @@ +# -*- coding: utf-8 -*- +# +# Copyright (C) 2007 Andrew Resch +# Copyright (C) 2010 Pedro Algarvio +# +# This file is part of Deluge and is licensed under GNU General Public License 3.0, or later, with +# the additional special exception to link portions of this program with the OpenSSL library. +# See LICENSE for more details. +# + +"""Logging functions""" +from __future__ import unicode_literals + +import inspect +import logging +import logging.handlers +import os +import sys + +from twisted.internet import defer +from twisted.python.log import PythonLoggingObserver + +from deluge import common + +__all__ = ('setup_logger', 'set_logger_level', 'get_plugin_logger', 'LOG') + +LoggingLoggerClass = logging.getLoggerClass() + +if 'dev' in common.get_version(): + DEFAULT_LOGGING_FORMAT = '%%(asctime)s.%%(msecs)03.0f [%%(levelname)-8s][%%(name)-%ds:%%(lineno)-4d] %%(message)s' +else: + DEFAULT_LOGGING_FORMAT = ( + '%%(asctime)s [%%(levelname)-8s][%%(name)-%ds:%%(lineno)-4d] %%(message)s' + ) +MAX_LOGGER_NAME_LENGTH = 10 + + +class Logging(LoggingLoggerClass): + def __init__(self, logger_name): + super(Logging, self).__init__(logger_name) + + # This makes module name padding increase to the biggest module name + # so that logs keep readability. + global MAX_LOGGER_NAME_LENGTH + if len(logger_name) > MAX_LOGGER_NAME_LENGTH: + MAX_LOGGER_NAME_LENGTH = len(logger_name) + for handler in logging.getLogger().handlers: + handler.setFormatter( + logging.Formatter( + DEFAULT_LOGGING_FORMAT % MAX_LOGGER_NAME_LENGTH, + datefmt='%H:%M:%S', + ) + ) + + @defer.inlineCallbacks + def garbage(self, msg, *args, **kwargs): + yield LoggingLoggerClass.log(self, 1, msg, *args, **kwargs) + + @defer.inlineCallbacks + def trace(self, msg, *args, **kwargs): + yield LoggingLoggerClass.log(self, 5, msg, *args, **kwargs) + + @defer.inlineCallbacks + def debug(self, msg, *args, **kwargs): + yield LoggingLoggerClass.debug(self, msg, *args, **kwargs) + + @defer.inlineCallbacks + def info(self, msg, *args, **kwargs): + yield LoggingLoggerClass.info(self, msg, *args, **kwargs) + + @defer.inlineCallbacks + def warning(self, msg, *args, **kwargs): + yield LoggingLoggerClass.warning(self, msg, *args, **kwargs) + + warn = warning + + @defer.inlineCallbacks + def error(self, msg, *args, **kwargs): + yield LoggingLoggerClass.error(self, msg, *args, **kwargs) + + @defer.inlineCallbacks + def critical(self, msg, *args, **kwargs): + yield LoggingLoggerClass.critical(self, msg, *args, **kwargs) + + @defer.inlineCallbacks + def exception(self, msg, *args, **kwargs): + yield LoggingLoggerClass.exception(self, msg, *args, **kwargs) + + def findCaller(self, stack_info=False): # NOQA: N802 + f = logging.currentframe().f_back + rv = '(unknown file)', 0, '(unknown function)' + while hasattr(f, 'f_code'): + co = f.f_code + filename = os.path.normcase(co.co_filename) + if filename in ( + __file__.replace('.pyc', '.py'), + defer.__file__.replace('.pyc', '.py'), + ): + f = f.f_back + continue + if common.PY2: + rv = (filename, f.f_lineno, co.co_name) + else: + rv = (filename, f.f_lineno, co.co_name, None) + break + return rv + + +levels = { + 'info': logging.INFO, + 'warn': logging.WARNING, + 'warning': logging.WARNING, + 'error': logging.ERROR, + 'none': logging.CRITICAL, + 'debug': logging.DEBUG, + 'trace': 5, + 'garbage': 1, +} + + +def setup_logger( + level='error', + filename=None, + filemode='w', + logrotate=None, + output_stream=sys.stdout, + twisted_observer=True, +): + """ + Sets up the basic logger and if `:param:filename` is set, then it will log + to that file instead of stdout. + + Args: + level (str): The log level to use (Default: 'error') + filename (str, optional): The log filename. Default is None meaning log + to terminal + filemode (str): The filemode to use when opening the log file + logrotate (int, optional): The size of the logfile in bytes when enabling + log rotation (Default is None meaning disabled) + output_stream (file descriptor): File descriptor to log to if not logging to file + twisted_observer (bool): Whether to setup the custom twisted logging observer. + """ + if logging.getLoggerClass() is not Logging: + logging.setLoggerClass(Logging) + logging.addLevelName(levels['trace'], 'TRACE') + logging.addLevelName(levels['garbage'], 'GARBAGE') + + level = levels.get(level, logging.ERROR) + + root_logger = logging.getLogger() + + if filename and logrotate: + handler = logging.handlers.RotatingFileHandler( + filename, maxBytes=logrotate, backupCount=5, encoding='utf-8' + ) + elif filename and filemode == 'w': + handler_cls = logging.FileHandler + if not common.windows_check(): + handler_cls = getattr( + logging.handlers, 'WatchedFileHandler', logging.FileHandler + ) + handler = handler_cls(filename, mode=filemode, encoding='utf-8') + else: + handler = logging.StreamHandler(stream=output_stream) + + handler.setLevel(level) + + formatter = logging.Formatter( + DEFAULT_LOGGING_FORMAT % MAX_LOGGER_NAME_LENGTH, datefmt='%H:%M:%S' + ) + + handler.setFormatter(formatter) + + # Check for existing handler to prevent duplicate logging. + if root_logger.handlers: + for handle in root_logger.handlers: + if not isinstance(handle, type(handler)): + root_logger.addHandler(handler) + else: + root_logger.addHandler(handler) + root_logger.setLevel(level) + + if twisted_observer: + twisted_logging = TwistedLoggingObserver() + twisted_logging.start() + + +class TwistedLoggingObserver(PythonLoggingObserver): + """ + Custom logging class to fix missing exception tracebacks in log output with new + twisted.logger module in twisted version >= 15.2. + + Related twisted bug ticket: https://twistedmatrix.com/trac/ticket/7927 + + """ + + def __init__(self): + PythonLoggingObserver.__init__(self, loggerName='twisted') + + def emit(self, event_dict): + log = logging.getLogger(__name__) + if 'log_failure' in event_dict: + fmt = '%(log_namespace)s \n%(log_failure)s' + getattr(LoggingLoggerClass, event_dict['log_level'].name)( + log, fmt % (event_dict) + ) + else: + PythonLoggingObserver.emit(self, event_dict) + + +def tweak_logging_levels(): + """This function allows tweaking the logging levels for all or some loggers. + This is mostly usefull for developing purposes hence the contents of the + file are NOT like regular deluge config file's. + + To use is, create a file named "logging.conf" on your Deluge's config dir + with contents like for example: + deluge:warn + deluge.core:debug + deluge.plugin:error + + What the above mean is the logger "deluge" will be set to the WARN level, + the "deluge.core" logger will be set to the DEBUG level and the + "deluge.plugin" will be set to the ERROR level. + + Remember, one rule per line and this WILL override the setting passed from + the command line. + """ + from deluge import configmanager + + logging_config_file = os.path.join(configmanager.get_config_dir(), 'logging.conf') + if not os.path.isfile(logging_config_file): + return + log = logging.getLogger(__name__) + log.warning( + 'logging.conf found! tweaking logging levels from %s', logging_config_file + ) + with open(logging_config_file, 'r') as _file: + for line in _file: + if line.strip().startswith('#'): + continue + name, level = line.strip().split(':') + if level not in levels: + continue + + log.warning('Setting logger "%s" to logging level "%s"', name, level) + set_logger_level(level, name) + + +def set_logger_level(level, logger_name=None): + """ + Sets the logger level. + + :param level: str, a string representing the desired level + :param logger_name: str, a string representing desired logger name for which + the level should change. The default is "None" will tweak + the root logger level. + + """ + logging.getLogger(logger_name).setLevel(levels.get(level, 'error')) + + +def get_plugin_logger(logger_name): + import warnings + + stack = inspect.stack() + stack.pop(0) # The logging call from this module + module_stack = stack.pop(0) # The module that called the log function + caller_module = inspect.getmodule(module_stack[0]) + # In some weird cases caller_module might be None, try to continue + caller_module_name = getattr(caller_module, '__name__', '') + warnings.warn_explicit( + DEPRECATION_WARNING, + DeprecationWarning, + module_stack[1], + module_stack[2], + caller_module_name, + ) + + if 'deluge.plugins.' in logger_name: + return logging.getLogger(logger_name) + return logging.getLogger('deluge.plugin.%s' % logger_name) + + +DEPRECATION_WARNING = """You seem to be using old style logging on your code, ie: + from deluge.log import LOG as log + +or: + from deluge.log import get_plugin_logger + +This has been deprecated in favour of an enhanced logging system and both "LOG" +and "get_plugin_logger" will be removed on the next major version release of Deluge, +meaning, code will break, specially plugins. +If you're seeing this message and you're not the developer of the plugin which +triggered this warning, please report to it's author. +If you're the developer, please stop using the above code and instead use: + + import logging + log = logging.getLogger(__name__) + + +The above will result in, regarding the "Label" plugin for example a log message similar to: + 15:33:54 [deluge.plugins.label.core:78 ][INFO ] *** Start Label plugin *** + +Triggering code: +""" + + +class _BackwardsCompatibleLOG(object): + def __getattribute__(self, name): + import warnings + + logger_name = 'deluge' + stack = inspect.stack() + stack.pop(0) # The logging call from this module + module_stack = stack.pop(0) # The module that called the log function + caller_module = inspect.getmodule(module_stack[0]) + # In some weird cases caller_module might be None, try to continue + caller_module_name = getattr(caller_module, '__name__', '') + warnings.warn_explicit( + DEPRECATION_WARNING, + DeprecationWarning, + module_stack[1], + module_stack[2], + caller_module_name, + ) + if caller_module: + for member in stack: + module = inspect.getmodule(member[0]) + if not module: + continue + if module.__name__ in ( + 'deluge.plugins.pluginbase', + 'deluge.plugins.init', + ): + logger_name += '.plugin.%s' % caller_module_name + # Monkey Patch The Plugin Module + caller_module.log = logging.getLogger(logger_name) + break + else: + logging.getLogger(logger_name).warning( + "Unable to monkey-patch the calling module's `log` attribute! " + 'You should really update and rebuild your plugins...' + ) + return getattr(logging.getLogger(logger_name), name) + + +LOG = _BackwardsCompatibleLOG() -- cgit v1.2.3