[Zope-Checkins] CVS: ZODB3/ZEO - runsvr.py:1.10

Guido van Rossum guido@python.org
Fri, 22 Nov 2002 13:00:25 -0500


Update of /cvs-repository/ZODB3/ZEO
In directory cvs.zope.org:/tmp/cvs-serv26949

Modified Files:
	runsvr.py 
Log Message:
Add ZConfig support.  The -s option goes away; if you want multiple
storages, it's much easier to specify those in a config file.

XXX This still needs a unit test suite.


=== ZODB3/ZEO/runsvr.py 1.9 => 1.10 ===
--- ZODB3/ZEO/runsvr.py:1.9	Thu Nov 21 16:34:33 2002
+++ ZODB3/ZEO/runsvr.py	Fri Nov 22 13:00:25 2002
@@ -14,17 +14,16 @@
 ##############################################################################
 """Start the ZEO storage server.
 
-Usage: %s [-a ADDRESS] [-f FILENAME] [-s STORAGE]
+Usage: %s [-a ADDRESS] [-C URL] [-f FILENAME] [-h]
 
 Options:
 -a/--address ADDRESS -- server address of the form PORT, HOST:PORT, or PATH
                         (a PATH must contain at least one "/")
+-C/--configuration URL -- configuration file or URL
 -f/--filename FILENAME -- filename for FileStorage
--s/--storage STORAGE -- storage specification of the form
-                        NAME=MODULE[:ATTRIBUTE]
-                        (multiple -s options are supported)
+-h/--help -- print this usage message and exit
 
--a is required; either -f must be used or -s must be used.
+Unless -C is specified, -a and -f are required.
 """
 
 # The code here is designed to be reused by other, similar servers.
@@ -41,6 +40,10 @@
 
 import zLOG
 
+import ZConfig
+import ZConfig.Common
+import ZConfig.Storage
+
 
 class Options:
 
@@ -53,6 +56,9 @@
     errors related to the command line.
     """
 
+    configuration = None
+    rootconf = None
+
     args = []
 
     def __init__(self, args=None, progname=None, doc=None):
@@ -90,8 +96,8 @@
         self.check_options()
 
     # Default set of options.  Subclasses should override.
-    _short_options = "h"
-    _long_options = ["--help"]
+    _short_options = "C:h"
+    _long_options = ["--configuration=", "--help"]
 
     def handle_option(self, opt, arg):
         """Handle one option.  Subclasses should override.
@@ -100,15 +106,25 @@
 
         When -h is detected, print the module docstring to stdout and exit(0).
         """
-        if opt == "-h" or opt == "--help":
+        if opt in ("-C", "--configuration"):
+            self.set_configuration(arg)
+        if opt in ("-h", "--help"):
             self.help()
 
+    def set_configuration(self, arg):
+        self.configuration = arg
+
     def check_options(self):
         """Check options.  Subclasses may override.
 
         This can be used to ensure certain options are set, etc.
         """
-        pass
+        self.load_configuration()
+
+    def load_configuration(self):
+        if self.rootconf or not self.configuration:
+            return
+        self.rootconf = ZConfig.load(self.configuration)
 
     def help(self):
         """Print a long help message (self.doc) to stdout and exit(0).
@@ -130,17 +146,20 @@
 
 class ZEOOptions(Options):
 
-    family = None
-    address = None
-    storages = None
-    filename = None
+    hostname = None                     # A subclass may set this
+    hostconf = None                     # <Host> section
+    zeoconf = None                      # <ZEO> section
+
+    family = None                       # set by -a; AF_UNIX or AF_INET
+    address = None                      # set by -a; string or (host, port)
+    storages = None                     # set by -f
 
-    _short_options = "a:f:hs:"
+    _short_options = "a:C:f:h"
     _long_options = [
         "--address=",
+        "--configuration=",
         "--filename=",
         "--help",
-        "--storage=",
         ]
 
     def handle_option(self, opt, arg):
@@ -162,32 +181,74 @@
                     self.usage("invalid port number: %r" % port)
                 self.address = (host, port)
         elif opt in ("-f", "--filename"):
-            self.filename = arg
-        elif opt in ("-s", "--storage"):
-            if self.storages is None:
-                self.storages = {}
-            if not "=" in arg:
-                self.usage("use -s/--storage storagename=module[:attribute]")
-            name, rest = arg.split("=", 1)
-            if ":" in rest:
-                module, attr = rest.split(":", 1)
-            else:
-                module = rest
-                attr = name
-            self.storages[name] = module, attr
+            from ZODB.FileStorage import FileStorage
+            self.storages = {"1": (FileStorage, {"file_name": arg})}
         else:
             # Pass it to the base class, for --help/-h
             Options.handle_option(self, opt, arg)
 
     def check_options(self):
-        if self.storages and self.filename:
-            self.usage("can't use both -s/--storage and -f/--filename")
-        if not self.storages and not self.filename:
-            self.usage("need one of -s/--storage or -f/--filename")
+        Options.check_options(self) # Calls load_configuration()
+        if not self.storages:
+            self.usage("no storages specified; use -f or -C")
         if self.family is None:
-            self.usage("need -a/--address [host:]port or unix path")
+            self.usage("no server address specified; use -a or -C")
         if self.args:
-            self.usage("no positional arguments supported")
+            self.usage("positional arguments are not supported")
+
+    def load_configuration(self):
+        Options.load_configuration(self) # Sets self.rootconf
+        if not self.rootconf:
+            return
+        try:
+            self.hostconf = self.rootconf.getSection("Host")
+        except ZConfig.Common.ConfigurationConflictingSectionError:
+            if not self.hostname:
+                self.hostname = socket.getfqdn()
+            self.hostconf = self.rootconf.getSection("Host", self.hostname)
+        if self.hostconf is None:
+            # If no <Host> section exists, fall back to the root
+            self.hostconf = self.rootconf
+        self.zeoconf = self.hostconf.getSection("ZEO")
+        if self.zeoconf is None:
+            # If no <ZEO> section exists, fall back to the host (or root)
+            self.zeoconf = self.hostconf
+
+        # Now extract options from various configuration sections
+        self.load_zeoconf()
+        self.load_storages()
+
+    def load_zeoconf(self):
+        # Get some option defaults from the configuration
+        if self.family:
+            # -a option overrides
+            return
+        port = self.zeoconf.getint("server-port")
+        path = self.zeoconf.get("path")
+        if port and path:
+            self.usage(
+                "Configuration contains conflicting ZEO information:\n"
+                "Exactly one of 'path' and 'server-port' may be given.")
+        if port:
+            host = self.hostconf.get("hostname", "")
+            self.family = socket.AF_INET
+            self.address = (host, port)
+        elif path:
+            self.family = socket.AF_UNIX
+            self.address = path
+
+    def load_storages(self):
+        # Get the storage specifications
+        if self.storages:
+            # -f option overrides
+            return
+        storagesections = self.zeoconf.getChildSections("Storage")
+        self.storages = {}
+        for section in storagesections:
+            name = section.name
+            if not name:
+                name = str(1 + len(self.storages))
+            self.storages[name] = ZConfig.Storage.getStorageInfo(section)
 
 
 class ZEOServer:
@@ -234,37 +295,11 @@
                 pass
 
     def open_storages(self):
-        if self.options.storages:
-            self.load_storages(self.options.storages)
-        else:
-            from ZODB.FileStorage import FileStorage
-            info("opening storage '1': %r" % self.options.filename)
-            storage = FileStorage(self.options.filename)
-            self.storages = {"1": storage}
-
-    def load_storages(self, storages):
         self.storages = {}
-        for name, (module, attr) in storages.items():
-            info("opening storage %r (%r:%r)" % (name, module, attr))
-            self.storages[name] = self.get_storage(module, attr)
-
-    _storage_cache = {}
-
-    def get_storage(self, module, attr):
-        # XXX This may fail with ImportError or AttributeError
-        path = sys.path
-        dir, module = os.path.split(module)
-        if module.lower().endswith('.py'):
-            module = module[:-3]
-        im = self._storage_cache.get((dir, module))
-        if im is None:
-            if dir:
-                path = [dir] + path
-            import imp
-            im = imp.find_module(module, path)
-            im = imp.load_module(module, *im)
-            self._storage_cache[(dir, module)] = im
-        return getattr(im, attr)
+        for name, (cls, args) in self.options.storages.items():
+            info("open storage %r: %s.%s(**%r)" %
+                 (name, cls.__module__, cls.__name__, args))
+            self.storages[name] = cls(**args)
 
     def setup_signals(self):
         """Set up signal handlers.