Многострочные записи журнала в syslog
Итак, я настроил приложение Python для входа в syslog с Python SysLogHandler, и все работает нормально. Кроме многострочной обработки. Не то, чтобы мне нужно было издавать многострочные записи журнала так плохо (я делаю немного), но мне нужно иметь возможность читать исключения на Python. Я использую Ubuntu с rsyslog 4.2.0. Это то, что я получаю:
Mar 28 20:11:59 telemachos root: ERROR 'EXCEPTION'#012Traceback (most recent call last):#012 File "./test.py", line 22, in <module>#012 foo()#012 File "./test.py", line 13, in foo#012 bar()#012 File "./test.py", line 16, in bar#012 bla()#012 File "./test.py", line 19, in bla#012 raise Exception("EXCEPTION!")#012Exception: EXCEPTION!
Тестовый код в случае необходимости:
import logging
from logging.handlers import SysLogHandler
logger = logging.getLogger()
logger.setLevel(logging.INFO)
syslog = SysLogHandler(address='/dev/log', facility='local0')
formatter = logging.Formatter('%(name)s: %(levelname)s %(message)r')
syslog.setFormatter(formatter)
logger.addHandler(syslog)
def foo():
bar()
def bar():
bla()
def bla():
raise Exception("EXCEPTION!")
try:
foo()
except:
logger.exception("EXCEPTION")
Ответы
Ответ 1
ОК, наконец выяснилось...
rsyslog по умолчанию экранирует все странные символы (ASCII < 32), и это включает в себя символы новой строки (а также вкладки и другие). Просто добавьте это в свою конфигурацию rsyslog, чтобы отключить это:
$EscapeControlCharactersOnReceive off
Ответ 2
В качестве альтернативы, если вы хотите, чтобы ваш syslog был неактивен в одной строке для синтаксического анализа, вы можете просто заменить символы при просмотре журнала.
tail -f /var/log/syslog | sed 's/#012/\n\t/g'
Ответ 3
Другой вариант заключается в подклассе SysLogHandler и переопределении emit()
- вы могли бы затем вызвать суперкласс emit()
для каждой строки в отправляемом тексте. Что-то вроде:
from logging import LogRecord
from logging.handlers import SysLogHandler
class MultilineSysLogHandler(SysLogHandler):
def emit(self, record):
if '\n' in record.msg:
record_args = [record.args] if isinstance(record.args, dict) else record.args
for single_line in record.msg.split('\n'):
single_line_record = LogRecord(
name=record.name,
level=record.levelno,
pathname=record.pathname,
msg=single_line,
args=record_args,
exc_info=record.exc_info,
func=record.funcName
)
super(MultilineSysLogHandler, self).emit(single_line_record)
else:
super(MultilineSysLogHandler, self).emit(record)