[Zope-Checkins] CVS: ZODB3/zLOG - LogHandlers.py:1.6 component.xml:1.6 datatypes.py:1.8 factory.py:1.3

Fred L. Drake, Jr. fred@zope.com
Thu, 23 Jan 2003 13:03:09 -0500


Update of /cvs-repository/ZODB3/zLOG
In directory cvs.zope.org:/tmp/cvs-serv21955/zLOG

Modified Files:
	LogHandlers.py component.xml datatypes.py factory.py 
Log Message:
Change the factory machinery to be a little less ad hoc.

=== ZODB3/zLOG/LogHandlers.py 1.5 => 1.6 ===
--- ZODB3/zLOG/LogHandlers.py:1.5	Thu Jan 23 11:27:13 2003
+++ ZODB3/zLOG/LogHandlers.py	Thu Jan 23 13:03:05 2003
@@ -16,7 +16,8 @@
 
 from logging import Handler, StreamHandler
 from logging.handlers import SysLogHandler
-from logging.handlers import HTTPHandler, SMTPHandler, NTEventLogHandler
+from logging.handlers import HTTPHandler, SMTPHandler
+from logging.handlers import NTEventLogHandler as Win32EventLogHandler
 
 class FileHandler(StreamHandler):
     """


=== ZODB3/zLOG/component.xml 1.5 => 1.6 ===
--- ZODB3/zLOG/component.xml:1.5	Thu Jan 23 11:57:25 2003
+++ ZODB3/zLOG/component.xml	Thu Jan 23 13:03:05 2003
@@ -12,14 +12,14 @@
     <key name="level" default="info" datatype=".logging_level"/>
   </sectiontype>
 
-  <sectiontype name="logfile" datatype=".file_handler"
+  <sectiontype name="logfile" datatype=".FileHandlerFactory"
                implements="loghandler" extends="base-log-handler">
     <key name="path" required="yes"/>
     <key name="format" default="------\n%(asctime)s %(message)s"
          datatype=".log_format"/>
   </sectiontype>
 
-  <sectiontype name="syslog" datatype=".syslog_handler"
+  <sectiontype name="syslog" datatype=".SyslogHandlerFactory"
                implements="loghandler" extends="base-log-handler">
     <key name="facility" default="user" datatype=".syslog_facility"/>
     <key name="address" datatype="socket-address" default="localhost:514"/>
@@ -27,14 +27,14 @@
          datatype=".log_format"/>
   </sectiontype>
 
-  <sectiontype name="win32-eventlog" datatype=".nteventlog_handler"
+  <sectiontype name="win32-eventlog" datatype=".Win32EventLogFactory"
                implements="loghandler" extends="base-log-handler">
     <key name="appname" default="Zope"/>
     <key name="format" default="%(message)s"
          datatype=".log_format"/>
   </sectiontype>
 
-  <sectiontype name="http-logger" datatype=".http_handler"
+  <sectiontype name="http-logger" datatype=".HTTPHandlerFactory"
                implements="loghandler" extends="base-log-handler">
     <key name="url" default="http://localhost/" datatype=".http_handler_url"/>
     <key name="method" default="GET" datatype=".get_or_post"/>
@@ -42,7 +42,7 @@
          datatype=".log_format"/>
   </sectiontype>
 
-  <sectiontype name="email-notifier" datatype=".smtp_handler"
+  <sectiontype name="email-notifier" datatype=".SMTPHandlerFactory"
                implements="loghandler" extends="base-log-handler">
     <key name="from" required="yes" attribute="fromaddr"/>
     <multikey name="to" required="yes" attribute="toaddrs"/>


=== ZODB3/zLOG/datatypes.py 1.7 => 1.8 ===
--- ZODB3/zLOG/datatypes.py:1.7	Thu Jan 23 11:57:25 2003
+++ ZODB3/zLOG/datatypes.py	Thu Jan 23 13:03:05 2003
@@ -1,6 +1,6 @@
 ##############################################################################
 #
-# Copyright (c) 2002, 2003 Zope Corporation and Contributors.
+# Copyright (c) 2002 Zope Corporation and Contributors.
 # All Rights Reserved.
 #
 # This software is subject to the provisions of the Zope Public License,
@@ -75,24 +75,32 @@
         value = value.replace(pattern, replacement)
     return value
 
-def file_handler(section):
-    path = section.path
 
-    def callback(inst,
-                 format=section.format,
-                 dateformat=section.dateformat,
-                 level=section.level):
-        import logging
-        inst.setFormatter(logging.Formatter(format, dateformat))
-        inst.setLevel(level)
+class HandlerFactory(Factory):
+    def __init__(self, section):
+        Factory.__init__(self)
+        self.section = section
 
-    # XXX should pick up sys.{stderr,stdout} when the factory is invoked
-    if path == "STDERR":
-        return Factory('zLOG.LogHandlers.StreamHandler', callback, sys.stderr)
-    elif path == "STDOUT":
-        return Factory('zLOG.LogHandlers.StreamHandler', callback, sys.stdout)
-    else:
-        return Factory('zLOG.LogHandlers.FileHandler', callback, path)
+    def create_logger(self):
+        raise NotImplementedError("subclasses must override create_logger()")
+
+    def create(self):
+        import logging
+        logger = self.create_logger()
+        logger.setFormatter(logging.Formatter(self.section.format,
+                                              self.section.dateformat))
+        logger.setLevel(self.section.level)
+        return logger
+
+class FileHandlerFactory(HandlerFactory):
+    def create_logger(self):
+        from zLOG.LogHandlers import StreamHandler, FileHandler
+        path = self.section.path
+        if path == "STDERR":
+            return StreamHandler(sys.stderr)
+        if path == "STDOUT":
+            return StreamHandler(sys.stdout)
+        return FileHandler(path)
 
 _syslog_facilities = {
     "auth": 1,
@@ -125,30 +133,16 @@
         raise ValueError("Syslog facility must be one of " + ", ".join(L))
     return value
 
-def syslog_handler(section):
-    def callback(inst,
-                 format=section.format,
-                 dateformat=section.dateformat,
-                 level=section.level):
-        import logging
-        inst.setFormatter(logging.Formatter(format, dateformat))
-        inst.setLevel(level)
-
-    return Factory('zLOG.LogHandlers.SysLogHandler', callback,
-                   section.address.address,
-                   section.facility)
-
-def nteventlog_handler(section):
-    def callback(inst,
-                 format=section.format,
-                 dateformat=section.dateformat,
-                 level=section.level):
-        import logging
-        inst.setFormatter(logging.Formatter(format, dateformat))
-        inst.setLevel(level)
-
-    return Factory('zLOG.LogHandlers.NTEventLogHandler', callback,
-                   section.appname)
+class SyslogHandlerFactory(HandlerFactory):
+    def create_logger(self):
+        from zLOG.LogHandlers import SysLogHandler
+        return SysLogHandler(self.section.address.address,
+                             self.section.facility)
+
+class Win32EventLogFactory(HandlerFactory):
+    def create_logger(self):
+        from zLOG.LogHandlers import Win32EventLogHandler
+        return Win32EventLogHandler(self.section.appname)
 
 def http_handler_url(value):
     import urlparse
@@ -178,45 +172,25 @@
                          + repr(value))
     return value
 
-def http_handler(section):
-    def callback(inst,
-                 format=section.format,
-                 dateformat=section.dateformat,
-                 level=section.level):
-        import logging
-        inst.setFormatter(logging.Formatter(format, dateformat))
-        inst.setLevel(level)
-
-    host, selector = section.url
-    return Factory('zLOG.LogHandlers.HTTPHandler',
-                   callback, host, selector, section.method)
-
-def smtp_handler(section):
-    def callback(inst,
-                 format=section.format,
-                 dateformat=section.dateformat,
-                 level=section.level):
-        import logging
-        inst.setFormatter(logging.Formatter(format, dateformat))
-        inst.setLevel(level)
-
-    host, port = section.smtp_server
-    if not port:
-        mailhost = host
-    else:
-        mailhost = host, port
-
-    return Factory('zLOG.LogHandlers.SMTPHandler',
-                   callback,
-                   mailhost,
-                   section.fromaddr,
-                   section.toaddrs,
-                   section.subject)
+class HTTPHandlerFactory(HandlerFactory):
+    def create_logger(self):
+        from zLOG.LogHandlers import HTTPHandler
+        host, selector = self.section.url
+        return HTTPHandler(host, selector, self.section.method)
+
+class SMTPHandlerFactory(HandlerFactory):
+    def create_logger(self):
+        from zLOG.LogHandlers import SMTPHandler
+        host, port = self.section.smtp_server
+        if not port:
+            mailhost = host
+        else:
+            mailhost = host, port
+        return SMTPHandler(mailhost, self.section.fromaddr,
+                           self.section.toaddrs, self.section.subject)
 
 
-_marker = []
-
-class EventLogFactory:
+class EventLogFactory(Factory):
     """
     A wrapper used to create loggers while delaying actual logger
     instance construction.  We need to do this because we may
@@ -226,24 +200,22 @@
     logger object.
     """
     def __init__(self, section):
+        Factory.__init__(self)
         self.level = section.level
         self.handler_factories = section.handlers
-        self.resolved = _marker
 
-    def __call__(self):
-        if self.resolved is _marker:
-            # set the logger up
-            import logging
-            logger = logging.getLogger("event")
-            logger.handlers = []
-            logger.propagate = 0
-            logger.setLevel(self.level)
-            if self.handler_factories:
-                for handler_factory in self.handler_factories:
-                    handler = handler_factory()
-                    logger.addHandler(handler)
-            else:
-                from zLOG.LogHandlers import NullHandler
-                logger.addHandler(NullHandler())
-            self.resolved = logger
-        return self.resolved
+    def create(self):
+        # set the logger up
+        import logging
+        logger = logging.getLogger("event")
+        logger.handlers = []
+        logger.propagate = 0
+        logger.setLevel(self.level)
+        if self.handler_factories:
+            for handler_factory in self.handler_factories:
+                handler = handler_factory()
+                logger.addHandler(handler)
+        else:
+            from zLOG.LogHandlers import NullHandler
+            logger.addHandler(NullHandler())
+        return logger


=== ZODB3/zLOG/factory.py 1.2 => 1.3 ===
--- ZODB3/zLOG/factory.py:1.2	Mon Jan 20 17:40:07 2003
+++ ZODB3/zLOG/factory.py	Thu Jan 23 13:03:05 2003
@@ -14,45 +14,23 @@
 
 _marker = []
 
-def importer(name):
-    components = name.split('.')
-    start = components[0]
-    g = globals()
-    package = __import__(start, g, g)
-    modulenames = [start]
-    for component in components[1:]:
-        modulenames.append(component)
-        try:
-            package = getattr(package, component)
-        except AttributeError:
-            name = '.'.join(modulenames)
-            package = __import__(name, g, g, component)
-    return package
-
 class Factory:
+    """Generic wrapper for instance construction.
+
+    Calling the factory causes the instance to be created if it hasn't
+    already been created, and returns the object.  Calling the factory
+    multiple times returns the same object.
+
+    The instance is created using the factory's create() method, which
+    must be overriden by subclasses.
     """
-    A generic wrapper for instance construction and function calling used
-    to delay construction/call until necessary.  The class path is the dotted
-    name to the class or function, args are the positional args, kw are the
-    keyword args. If it is specified, 'callback' is a function which will be
-    called back after constructing an instance or calling a function.  It must
-    take the instance (or the result of the function) as a single argument.
-    """
-    def __init__(self, class_path, callback, *args, **kw):
-        self.class_path = class_path
-        self.callback = callback
-        self.args = args
-        self.kw = kw
+    def __init__(self):
         self.instance = _marker
 
-    def __repr__(self):
-        return ('<Factory instance for class "%s" with positional args "%s" '
-                'and keword args "%s"' % (self.class_path, self.args, self.kw))
-
     def __call__(self):
         if self.instance is _marker:
-            constructor = importer(self.class_path)
-            self.instance = constructor(*self.args, **self.kw)
-            if self.callback is not None:
-                self.callback(self.instance)
+            self.instance = self.create()
         return self.instance
+
+    def create(self):
+        raise NotImplementedError("subclasses need to override create()")