Подтвердить что ты не робот

Как перенаправить stdout и stderr в logger в Python

У меня есть журнал, который имеет RotatingFileHandler. Я хочу перенаправить все Stdout и Stderr в регистратор. Как это сделать?

4b9b3361

Ответ 1

Не хватает комментариев для комментариев, но я хотел добавить версию этого, которая сработала для меня, если другие находятся в аналогичной ситуации.

class LoggerWriter:
    def __init__(self, level):
        # self.level is really like using log.debug(message)
        # at least in my case
        self.level = level

    def write(self, message):
        # if statement reduces the amount of newlines that are
        # printed to the logger
        if message != '\n':
            self.level(message)

    def flush(self):
        # create a flush method so things can be flushed when
        # the system wants to. Not sure if simply 'printing'
        # sys.stderr is the correct way to do it, but it seemed
        # to work properly for me.
        self.level(sys.stderr)

и это будет выглядеть примерно так:

log = logging.getLogger('foobar')
sys.stdout = LoggerWriter(log.debug)
sys.stderr = LoggerWriter(log.warning)

Ответ 2

Если это система все-Python (т.е. нет библиотек C, которые пишут fds напрямую, как спрашивал Игнасио Васкес-Абрамс), тогда вы можете использовать подход, предложенный здесь:

class LoggerWriter:
    def __init__(self, logger, level):
        self.logger = logger
        self.level = level

    def write(self, message):
        if message != '\n':
            self.logger.log(self.level, message)

а затем установите для экземпляров sys.stdout и sys.stderr в LoggerWriter.

Ответ 3

Все предыдущие ответы, похоже, имеют проблемы с добавлением дополнительных строк, где они не нужны. Лучшее для меня решение - от http://www.electricmonk.nl/log/2011/08/14/redirect-stdout-and-stderr-to-a-logger-in-python/, где он демонстрирует, как отправить stdout и stderr в регистратор:

import logging
import sys

class StreamToLogger(object):
   """
   Fake file-like stream object that redirects writes to a logger instance.
   """
   def __init__(self, logger, log_level=logging.INFO):
      self.logger = logger
      self.log_level = log_level
      self.linebuf = ''

   def write(self, buf):
      for line in buf.rstrip().splitlines():
         self.logger.log(self.log_level, line.rstrip())

logging.basicConfig(
   level=logging.DEBUG,
   format='%(asctime)s:%(levelname)s:%(name)s:%(message)s',
   filename="out.log",
   filemode='a'
)

stdout_logger = logging.getLogger('STDOUT')
sl = StreamToLogger(stdout_logger, logging.INFO)
sys.stdout = sl

stderr_logger = logging.getLogger('STDERR')
sl = StreamToLogger(stderr_logger, logging.ERROR)
sys.stderr = sl

print "Test to standard out"
raise Exception('Test to standard error')

Результат выглядит следующим образом:

2011-08-14 14:46:20,573:INFO:STDOUT:Test to standard out
2011-08-14 14:46:20,573:ERROR:STDERR:Traceback (most recent call last):
2011-08-14 14:46:20,574:ERROR:STDERR:  File "redirect.py", line 33, in 
2011-08-14 14:46:20,574:ERROR:STDERR:raise Exception('Test to standard error')
2011-08-14 14:46:20,574:ERROR:STDERR:Exception
2011-08-14 14:46:20,574:ERROR:STDERR::
2011-08-14 14:46:20,574:ERROR:STDERR:Test to standard error

Обратите внимание, что self.linebuf = '' - это то, где обрабатывается флеш, а не реализация функции флеша.

Ответ 4

Вы можете использовать диспетчер контекстов redirect_stdout:

import logging
from contextlib import redirect_stdout

logging.basicConfig(stream=sys.stdout, level=logging.DEBUG)
logging.write = lambda msg: logging.info(msg) if msg != '\n' else None

with redirect_stdout(logging):
    print('Test')

или как это

import logging
from contextlib import redirect_stdout


logger = logging.getLogger('Meow')
logger.setLevel(logging.INFO)
formatter = logging.Formatter(
    fmt='[{name}] {asctime} {levelname}: {message}',
    datefmt='%m/%d/%Y %H:%M:%S',
    style='{'
)
ch = logging.StreamHandler()
ch.setLevel(logging.INFO)
ch.setFormatter(formatter)
logger.addHandler(ch)

logger.write = lambda msg: logger.info(msg) if msg != '\n' else None

with redirect_stdout(logger):
    print('Test')

Ответ 5

С добавлением флеша к Vinay Sajip ответ:

class LoggerWriter:
def __init__(self, logger, level): 
    self.logger = logger
    self.level = level 

def write(self, message):
    if message != '\n':
        self.logger.log(self.level, message)

def flush(self): 
    pass