[Zope-Checkins] CVS: Zope/lib/python/zLOG - BaseLogger.py:1.2 EventLogger.py:1.2 LogHandlers.py:1.2 __init__.py:1.13 MinimalLogger.py:NONE
Chris McDonough
chrism@zope.com
Sun, 24 Nov 2002 00:24:43 -0500
Update of /cvs-repository/Zope/lib/python/zLOG
In directory cvs.zope.org:/tmp/cvs-serv18617
Modified Files:
__init__.py
Added Files:
BaseLogger.py EventLogger.py LogHandlers.py
Removed Files:
MinimalLogger.py
Log Message:
Use PEP282-based logger instead of homegrown.
=== Zope/lib/python/zLOG/BaseLogger.py 1.1 => 1.2 ===
--- /dev/null Sun Nov 24 00:24:43 2002
+++ Zope/lib/python/zLOG/BaseLogger.py Sun Nov 24 00:24:43 2002
@@ -0,0 +1,24 @@
+##############################################################################
+#
+# Copyright (c) 2001, 2002 Zope Corporation and Contributors.
+# All Rights Reserved.
+#
+# This software is subject to the provisions of the Zope Public License,
+# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
+# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
+# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
+# FOR A PARTICULAR PURPOSE
+#
+##############################################################################
+"""
+An abstract logger meant to provide features to the access logger,
+the event logger, and the debug logger.
+"""
+
+class BaseLogger:
+ def reopen(self):
+ for handler in self.logger.handlers:
+ if hasattr(handler, 'reopen') and callable(handler.reopen):
+ handler.reopen()
+
=== Zope/lib/python/zLOG/EventLogger.py 1.1 => 1.2 ===
--- /dev/null Sun Nov 24 00:24:43 2002
+++ Zope/lib/python/zLOG/EventLogger.py Sun Nov 24 00:24:43 2002
@@ -0,0 +1,191 @@
+##############################################################################
+#
+# Copyright (c) 2001, 2002 Zope Corporation and Contributors.
+# All Rights Reserved.
+#
+# This software is subject to the provisions of the Zope Public License,
+# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
+# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
+# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
+# FOR A PARTICULAR PURPOSE
+#
+##############################################################################
+"""
+A logging module which handles event messages.
+
+This uses Vinay Sajip's PEP 282 logging module.
+"""
+
+__version__='$Revision$'[11:-2]
+
+import os, sys, time
+try:
+ import textwrap
+except ImportError:
+ textwrap = None
+import logging
+from BaseLogger import BaseLogger
+from LogHandlers import FileHandler, NullHandler, SysLogHandler
+from logging import StreamHandler, Formatter
+
+class EventLogger(BaseLogger):
+ logger = logging.getLogger('event')
+ logger.addHandler(NullHandler())
+ log_format = '%(sev)s %(subsys)s %(summary)s%(detail)s'
+
+ def log(self, subsystem, severity, summary, detail, error):
+ if error:
+ kw = {'exc_info':1}
+ else:
+ kw = {}
+
+ if detail:
+ detail = '\n' + detail
+ else:
+ detail = ''
+
+ msg = self.log_format % {
+ 'sev' : severity_string(severity),
+ 'subsys' : subsystem,
+ 'summary': summary,
+ 'detail' : detail,
+ }
+
+ if textwrap and len(msg) > 80:
+ msg = '\n'.join(textwrap.wrap(
+ msg, width=79, subsequent_indent=" "*20,
+ break_long_words=0))
+
+ severity = zlog_to_pep282_severity(severity)
+ self.logger.log(severity, msg, **kw)
+
+EventLogger = EventLogger()
+
+log_write = EventLogger.log
+
+def severity_string(severity, mapping={
+ -300: 'TRACE',
+ -200: 'DEBUG',
+ -100: 'BLATHER',
+ 0: 'INFO',
+ 100: 'PROBLEM',
+ 200: 'ERROR',
+ 300: 'PANIC',
+ }):
+ """Convert a severity code to a string."""
+ s = mapping.get(int(severity), '')
+ return "%s(%s)" % (s, severity)
+
+def zlog_to_pep282_severity(zlog_severity):
+ """
+ We map zLOG severities to PEP282 severities here.
+ This is how they are mapped:
+
+ zLOG severity PEP282 severity
+ ------------- ---------------
+ PANIC (300) critical (50)
+ ERROR (200), PROBLEM (100) error (40)
+ INFO (0) warn (30)
+ BLATHER (-100) info (20)
+ DEBUG (-200), TRACE (-300) debug (10)
+ """
+ sev = zlog_severity
+ if sev >= 300:
+ return logging.CRITICAL
+ if sev >= 100:
+ return logging.ERROR
+ if sev >= 0:
+ return logging.WARN
+ if sev >= -100:
+ return logging.INFO
+ else:
+ return logging.DEBUG
+
+def log_time():
+ """Return a simple time string without spaces suitable for logging."""
+ return ("%4.4d-%2.2d-%2.2dT%2.2d:%2.2d:%2.2d"
+ % time.localtime()[:6])
+
+def get_env_severity_info():
+ # EVENT_LOG_SEVERITY is the preferred envvar, but we accept
+ # STUPID_LOG_SEVERITY also
+ eget = os.environ.get
+ severity = eget('EVENT_LOG_SEVERITY') or eget('STUPID_LOG_SEVERITY')
+ if severity:
+ severity = int(severity)
+ else:
+ severity = 0 # INFO
+ return severity
+
+def get_env_syslog_info():
+ eget = os.environ.get
+ addr = None
+ port = None
+ path = eget('ZSYSLOG')
+ facility = eget('ZSYSLOG_FACILITY', 'user')
+ server = eget('ZSYSLOG_SERVER')
+ if server:
+ addr, port = server.split(':')
+ port = int(port)
+ if addr:
+ return (facility, (addr, port))
+ else:
+ return (facility, path)
+
+def get_env_file_info():
+ eget = os.environ.get
+ # EVENT_LOG_FILE is the preferred envvar, but we accept
+ # STUPID_LOG_FILE also
+ path = eget('EVENT_LOG_FILE')
+ if path is None:
+ path = eget('STUPID_LOG_FILE')
+ if path is None:
+ dest = None
+ else:
+ dest = path
+ return dest
+
+formatters = {
+ 'file': Formatter(fmt='------\n%(asctime)s %(message)s',
+ datefmt='%Y-%m-%dT%H:%M:%S'),
+ 'syslog': Formatter(fmt='%(message)s'),
+ }
+
+def initialize_from_environment():
+ """ Reinitialize the event logger from the environment """
+ # clear the current handlers from the event logger
+ EventLogger.logger.handlers = []
+
+ handlers = []
+
+ # set up syslog handler if necessary
+ facility, syslogdest = get_env_syslog_info()
+ if syslogdest:
+ handler = SysLogHandler(syslogdest, facility)
+ handler.setFormatter(formatters['syslog'])
+ handlers.append(handler)
+
+ # set up file handler if necessary
+ filedest = get_env_file_info()
+ if filedest:
+ handler = FileHandler(filedest)
+ handler.setFormatter(formatters['file'])
+ handlers.append(handler)
+ elif filedest == '':
+ # if dest is an empty string, log to standard error
+ handler = StreamHandler()
+ handler.setFormatter(formatters['file'])
+ handlers.append(handler)
+ else:
+ # log to nowhere, but install a 'null' handler in order to
+ # prevent error messages from emanating due to a missing handler
+ handlers.append(NullHandler())
+
+ severity = get_env_severity_info()
+ severity = zlog_to_pep282_severity(severity)
+ EventLogger.logger.setLevel(severity)
+
+ for handler in handlers:
+ EventLogger.logger.addHandler(handler)
+
=== Zope/lib/python/zLOG/LogHandlers.py 1.1 => 1.2 ===
--- /dev/null Sun Nov 24 00:24:43 2002
+++ Zope/lib/python/zLOG/LogHandlers.py Sun Nov 24 00:24:43 2002
@@ -0,0 +1,78 @@
+##############################################################################
+#
+# Copyright (c) 2001 Zope Corporation and Contributors. All Rights Reserved.
+#
+# This software is subject to the provisions of the Zope Public License,
+# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
+# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
+# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
+# FOR A PARTICULAR PURPOSE
+#
+##############################################################################
+""" Handlers which can plug into a PEP 282 logger """
+
+from logging import Handler, StreamHandler
+from logging.handlers import SysLogHandler
+
+class FileHandler(StreamHandler):
+ """
+ A file handler which allows for reopening of logs in favor of the
+ 'rollover' features of the standard PEP282 FileHandler.
+ """
+ def __init__(self, filename, mode="a+"):
+ StreamHandler.__init__(self, open(filename, mode))
+ self.baseFilename = filename
+ self.mode = mode
+
+ def close(self):
+ self.stream.close()
+
+ def reopen(self):
+ self.close()
+ self.stream = open(self.baseFilename, self.mode)
+
+class NullHandler(Handler):
+ """
+ A null handler. Does nothing.
+ """
+ def emit(self, record):
+ pass
+
+ def handle(self, record):
+ pass
+
+class StartupHandler(Handler):
+ """
+ A handler which outputs messages to a stream but also buffers them until
+ they can be flushed to a target handler. Useful at startup before we can
+ know that we can safely write to a config-specified handler.
+ """
+ def __init__(self, stream=None):
+ Handler.__init__(self)
+ if not stream:
+ stream = sys.stderr
+ self.stream = stream
+ self.buffer = []
+
+ def emit(self, record):
+ try:
+ self.buffer.append(record)
+ msg = self.format(record)
+ self.stream.write("%s\n" % msg)
+ self.flush()
+ except:
+ self.handleError()
+
+ def flush(self):
+ self.stream.flush()
+
+ def flushBufferTo(self, target):
+ for record in self.buffer:
+ target.handle(record)
+ self.buffer = []
+
+
+
+
+
=== Zope/lib/python/zLOG/__init__.py 1.12 => 1.13 ===
--- Zope/lib/python/zLOG/__init__.py:1.12 Fri Nov 22 13:17:39 2002
+++ Zope/lib/python/zLOG/__init__.py Sun Nov 24 00:24:43 2002
@@ -88,8 +88,8 @@
"""
__version__='$Revision$'[11:-2]
-from MinimalLogger import log_write, log_time, severity_string, \
- _set_log_dest, initialize
+from EventLogger import log_write, log_time, severity_string, \
+ initialize_from_environment as initialize
from traceback import format_exception
# Standard severities
=== Removed File Zope/lib/python/zLOG/MinimalLogger.py ===