Люди,
Я почесываю голову на конфигурацию журнала python, которую я не могу получить.
Скажем, у меня установлен следующий пакет:
mypackage/
data/mypackage.logging.conf
module1.py
module2.py
scripts/main.py
Поскольку script может использоваться в интерактивном режиме или запускаться из crontab, у меня есть следующие требования:
-
нет операторов печати, все проходит через журнал,
-
с использованием
timedRotatingFileHandler
, всегда на уровне DEBUG; -
с использованием
mailinglogger.SummarisingLogger
, всегда на уровне INFO; -
log to console, с уровнем, установленным по умолчанию INFO или переопределенным с помощью опции командной строки.
Проблема в том, что я могу изменить уровень журнала через командную строку, и уровень журнала консоли изменился соответствующим образом, но другие обработчики также изменены, что я не хочу...: -/
В файле конфигурации протоколирования я не уверен, что я понимаю приоритет между уровнем корневого регистратора, уровнем уровня других регистраторов и уровнями обработчиков.
Вот пример кода. Любые подсказки будут оценены: -)
# 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=('[email protected]', ('[email protected]',), '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
Тем временем я обнаружил этот ответ и успешно изменил свой код следующим образом:
#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(
'[email protected]', ('[email protected]',), '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=('[email protected]', ('[email protected]',), '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=
В script:
-
прочитана конфигурация регистрации;
-
создается консольный_формат (повторно);
-
обработчик консоли создается с уровнем журнала из командной строки, а затем добавляется в журнал "mypackage".
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...: -)