Python在不同的日志级别记录到多个处理程序?

时间:2014-08-07 15:59:37

标签: python logging config

民间,

我在python日志配置上摸不着头脑,我无法做对。

我们说我安装了以下软件包:

mypackage/
   data/mypackage.logging.conf
   module1.py
   module2.py
   scripts/main.py

由于脚本可以交互使用或从crontab运行,因此我有以下要求:

  1. 没有打印报表,所有内容都通过记录;

  2. 在DEBUG级别使用timedRotatingFileHandler始终进行日志记录;

  3. 在INFO级别使用mailinglogger.SummarisingLogger始终进行日志记录;

  4. 登录到控制台,默认情况下将级别设置为INFO或通过命令行选项覆盖。

  5. 问题是,我可以通过命令行更改日志级别,控制台日志级别也会相应更改,但其他处理程序也会更改,我不想要......: - /

    在日志记录配置文件中,我不确定我是否理解根记录器级别之间的优先级,其他记录器'级别和处理程序'等级设置。

    以下是一些示例代码。任何线索将不胜感激: - )

    # mypackage/data/mypackage.logging.conf
    [loggers]
    root,mypackage
    
    [handlers]
    keys=consoleHandler,timedRotatingFileHandler,summarisingHandler
    
    [formatters]
    keys=simpleFormatter,consoleFormatter,mypackageFormatter
    
    [logger_root]
    #level=INFO
    handlers=consoleHandler
    
    [logger_mypackage]
    #level=INFO
    handlers=timedRotatingFileHandler,summarisingHandler
    qualname=mypackage
    
    [handler_consoleHandler]
    class=StreamHandler
    #level=INFO
    formatter=consoleFormatter
    args=(sys.stdout,)
    
    [handler_timedRotatingFileHandler]
    class=logging.handlers.TimedRotatingFileHandler
    level=DEBUG
    formatter=mypackageFormatter
    args=('mypackage.log', 'M', 1, 5)
    
    [handler_summarisingHandler]
    class=mailinglogger.SummarisingLogger
    level=INFO
    formatter=mypackageFormatter
    args=('mypackage@someserver.somewhere.com', ('mypackage-alerts@somewhere.com',), 'relay.somewhere.com')
    

    #mypackage/scripts/main.py:
    import logging
    import logging.config
    import os
    import sys
    
    import mypackage.module1
    import mypackage.module2
    
    logging.config.fileConfig('data/mypackage.logging.conf')
    log = logging.getLogger(__name__)
    
    if __name__ == '__main__':
        loglevel = 'INFO'
        if len(sys.argv) > 1:
            loglevel = sys.argv[1].upper()
    
        logging.getLogger('').setLevel(getattr(logging, loglevel))
        # or logging.getLogger('mypackage').setLevel(getattr(logging, loglevel)) ?
    
        mypackage.module1.do_something()
        mypackage.module2.do_something_else()
    

    #mypackage/module1.py:
    import logging
    
    log = logging.getLogger(__name__)
    log.addHandler(NullHandler())
    
    def do_something():
        log.debug("some debug message from:" + __name__)
        log.info("some info message from:" + __name__)
        log.error("some error message from:" + __name__)
    

    #mypackage/module2.py:
    import logging
    
    log = logging.getLogger(__name__)
    log.addHandler(NullHandler())
    
    def do_something_else():
        log.debug("some debug message from:" + __name__)
        log.info("some info message from:" + __name__)
        log.error("some error message from:" + __name__)
    

    更新1

    与此同时,我发现this answer并成功修改了我的代码:

    #mypackage/scripts/main.py:
    import logging
    import logging.config
    import os
    import sys
    import mailinglogger
    
    import mypackage.module1
    import mypackage.module2
    
    def main():
        # get the console log level from the command-line
        loglevel = 'INFO'
        if len(sys.argv) > 1:
            loglevel = sys.argv[1].upper()
    
        # create formatters
        simple_formatter = logging.Formatter("%(name)s:%(levelname)s: %(message)s")
        detailed_formatter = logging.Formatter("%(asctime)s %(name)s[%(process)d]: %(levelname)s - %(message)s")
    
        # get a top-level "mypackage" logger,
        # set its log level to DEBUG,
        # BUT PREVENT IT from propagating messages to the root logger
        #
        log = logging.getLogger('mypackage')
        log.setLevel(logging.DEBUG)
        log.propagate = 0
    
        # create a console handler
        # and set its log level to the command-line option 
        # 
        console_handler = logging.StreamHandler(sys.stdout)
        console_handler.setLevel(getattr(logging, loglevel))
        console_handler.setFormatter(simple_formatter)
    
        # create a file handler
        # and set its log level to DEBUG
        #
        file_handler = logging.handlers.TimedRotatingFileHandler('mypackage.log', 'M', 1, 5)
        file_handler.setLevel(logging.DEBUG)
        file_handler.setFormatter(detailed_formatter)
    
        # create a mail handler
        # and set its log level to INFO
        #
        mail_handler = mailinglogger.SummarisingLogger(
            'mypackage@someserver.somewhere.com', ('mypackage-alerts@somewhere.com',), 'relay.somewhere.com')
        mail_handler.setLevel(logging.INFO)
        mail_handler.setFormatter(detailed_formatter)
    
        # add handlers to the "mypackage" logger
        #
        log.addHandler(console_handler)
        log.addHandler(file_handler)
        log.addHandler(mail_handler)
    
        # let the modules do their stuff 
        # and log to the "mypackage.module1" and "mypackage.module2" loggers
        #
        mypackage.module1.do_something()
        mypackage.module2.do_something_else()
    
    
    if __name__ == '__main__':
        main()
    

    现在,我将尝试在logging.config文件中翻译它......


    更新2

    这是我找到的最佳日志配置和代码组合。

    在mypackage.logging.conf文件中," mypackage"记录器是:

    • 仅使用文件和电子邮件处理程序进行设置;
    • 其传播设置为false;
    • 其级别设置为DEBUG;
    • ,而文件和电子邮件处理程序分别设置为INFO和DEBUG。

    #mypackage/data/mypackage.logging.conf
    [loggers]
    keys=root,mypackage
    
    [handlers]
    keys=consoleHandler,timedRotatingFileHandler,summarisingHandler
    
    [formatters]
    keys=simpleFormatter,consoleFormatter,mypackageFormatter
    
    [logger_root]
    #level=INFO
    handlers=consoleHandler
    
    [logger_mypackage]
    level=DEBUG
    handlers=timedRotatingFileHandler,summarisingHandler
    qualname=mypackage
    propagate=0
    
    [handler_consoleHandler]
    class=StreamHandler
    #level=INFO
    formatter=consoleFormatter
    args=(sys.stdout,)
    
    [handler_timedRotatingFileHandler]
    class=logging.handlers.TimedRotatingFileHandler
    level=DEBUG
    formatter=mypackageFormatter
    args=('mypackage.log', 'M', 1, 5)
    
    [handler_summarisingHandler]
    class=mailinglogger.SummarisingLogger
    level=INFO
    formatter=mypackageFormatter
    args=('mypackage@someserver.somewhere.com', ('mypackage-alerts@somewhere.com',), 'relay.somewhere.com')
    
    [formatter_consoleFormatter]
    format=%(levelname)s: %(message)s
    datefmt=
    
    [formatter_mypackageFormatter]
    format=%(asctime)s %(name)s[%(process)d]: %(levelname)s - %(message)s
    datefmt=
    

    在剧本中:

    1. 读取日志配置;

    2. (重新)创建了console_formatter;

    3. 使用命令行选项中的日志级别创建控制台处理程序,然后将其添加到" mypackage"记录器。


    4. import logging
      import logging.config
      import os
      import sys
      
      import mypackage.module1
      import mypackage.module2
      
      def setup_logging(loglevel):
          #
          # load logging config from file
          #
          logging.config.fileConfig('data/mypackage.logging.conf')
      
          # (re-)create formatter
          console_formatter = logging.Formatter("%(name)s:%(levelname)s: %(message)s")
      
          # create a console handler
          # and set its log level to the command-line option 
          # 
          console_handler = logging.StreamHandler(sys.stdout)
          console_handler.setFormatter(console_formatter)
          console_handler.setLevel(getattr(logging, loglevel))
      
          # add console handler to the pre-configured "mypackage" logger
          #
          logger = logging.getLogger('mypackage')
          logger.addHandler(console_handler)
      
      
      def main():
          # get the console log level from the command-line
          loglevel = 'INFO'
          if len(sys.argv) > 1:
              loglevel = sys.argv[1].upper()
      
          # load logging config and setup console handler
          #
          setup_logging(loglevel)
      
          # log from the script to the "mypackage.scripts.main" logger
          #
          log = logging.getLogger(__name__)
          log.debug("some debug message from:" + __name__)
          log.info("some info message from:" + __name__)
          log.error("some error message from:" + __name__)
      
          # let the modules do their stuff 
          # and log to the "mypackage.module1" and "mypackage.module2" loggers
          #
          mypackage.module1.do_something()
          mypackage.module2.do_something_else()
      
      if __name__== '__main__':
          main()
      

      如果处理程序是“可寻址的”,那么事情会变得更简单。从配置文件加载时的名称。

      然后,我们可以在配置文件中设置mypackage控制台处理程序,并在代码中更改其日志级别,如下所示:

      def setup_logging(loglevel):
          logging.config.fileConfig('data/mypackage.logging.conf')
      
          logger = logging.getLogger('mypackage')
          console_handler = logger.getHandler('consoleHandler')
          console_handler.setLevel(getattr(logging, loglevel))
      

      无需重新创建格式化程序......

      (上次更新:是的,我很清楚https://docs.python.org/3/library/logging.config.html#incremental-configuration,但在这种情况下,我仍然坚持使用Python 2.6 ...: - )

2 个答案:

答案 0 :(得分:1)

更新处理程序的方法:

import logging

from rootmodule.mymodule import mylogger

def update_handler_level(logger,  handler_type, level="INFO"):
    # if not root logger user logger.parent
    for handler in logger.handlers or logger.parent.handlers:
        if isinstance(handler, handler_type):
            print(handler.level)
            handler.setLevel(getattr(logging, level, "INFO"))
            print(handler.level)

mylogger.debug('test')
update_handler_level(mylogger, logging.StreamHandler)
mylogger.debug('test')

我的logging.cfg与你的非常相似,只是记录器名称设置在一个常量模块中(可以在不破坏日志记录配置的情况下重命名模块)

要从命令行更新,必须在opts值和logging.Handler子类名之间建立映射。

答案 1 :(得分:1)

使用dictConfig。这是在dictConfig中使用单独的句柄记录到多个文件的示例。这并不是您要找的东西,但是您可以修改此示例,并只需更改要使用的每个处理程序的级别即可。

import os, logging
from logging.config import dictConfig

FORMAT = "%(asctime)s {app} [%(thread)d] %(levelname)-5s %(name)s - %(message)s. [file=%(filename)s:%(lineno)d]"
DATE_FORMAT = None


def setup_logging(name, level="INFO", fmt=FORMAT):
    formatted = fmt.format(app=name)
    log_dir = r'C:/log_directory'
    if not os.path.exists(log_dir):
        os.makedirs(log_dir)

    logging_config = {
        "version": 1,
        'disable_existing_loggers': False,
        "formatters": {
            'standard': {
                'format': formatted
            }
        },
        "handlers": {
            'default': {
                'class': 'logging.StreamHandler',
                'formatter': 'standard',
                'level': level,
                'stream': 'ext://sys.stdout'
            },
            'file': {
                'class': 'logging.handlers.TimedRotatingFileHandler',
                'when': 'midnight',
                'utc': True,
                'backupCount': 5,
                'level': level,
                'filename': '{}/app_manager.log'.format(log_dir),
                'formatter': 'standard',
            },
            'file2': {
                'class': 'logging.handlers.TimedRotatingFileHandler',
                'when': 'midnight',
                'utc': True,
                'backupCount': 5,
                'level': level,
                'filename': '{}/unified_log.log'.format(log_dir),
                'formatter': 'standard',
            }
        },
        "loggers": {
            "": {
                'handlers': ['default', 'file'],
                'level': level
            },
            "second_log": {
                'handlers': ['default', 'file2'],
                'level': level
            }
        }
    }

    dictConfig(logging_config)

log.setup_logging(name="log-name", level=LEVEL
logger = logging.getLogger(__name__)
second_logger = logging.getLogger('second_log')
second_logger.propagate = False