[Zope-Checkins] CVS: Packages/ZConfig - SectionMatcher.py:1.1.2.1 SchemaInfo.py:1.1.2.3 SchemaParser.py:1.1.4.2

Fred L. Drake, Jr. fred@zope.com
Mon, 9 Dec 2002 12:02:26 -0500


Update of /cvs-repository/Packages/ZConfig
In directory cvs.zope.org:/tmp/cvs-serv8712

Modified Files:
      Tag: zconfig-schema-devel-branch
	SchemaInfo.py SchemaParser.py 
Added Files:
      Tag: zconfig-schema-devel-branch
	SectionMatcher.py 
Log Message:
Checkpoint: more progress.

=== Added File Packages/ZConfig/SectionMatcher.py ===
"""Utility that manages the binding of configuration data to a section."""

import ZConfig

class SectionMatcher:
    def __init__(self, info, name=None):
        if name is not None:
            self.name = info.nametype.convert(name)
            if not info.isAllowedName(self.name):
                raise ZConfig.ConfigurationError(
                    "%s is not an allowed name for %s sections"
                    % (`name`, `info.name`))
        elif info.allowUnnamed():
            self.name = None
        else:
            raise ZConfig.ConfigurationError(
                `info.name` + " sections may not be unnamed")
        self.info = info
        self._values = {}

    def addKeyValue(self, key, value):
        keyinfo = self.info.getinfo(key)
        if keyinfo.issection():
            if keyinfo.name:
                extra = " in %s sections" % `keyinfo.name`
            else:
                extra = ""
            raise ZConfig.ConfigurationError(
                "%s is not a valid key name%s" % (`key`, extra))
        L = self._values.get(key)
        if L is None:
            L = []
        if keyinfo.maxOccurs is None:
            L.append(value)
        elif len(L) = keyinfo.maxOccurs:
            raise ZConfig.ConfigurationError(
                "too many values for " + `name`)
        else:
            L.append(value)
        self._values[key] = L

    def finish(self):
        """Check the constraints of the section and convert to an application
        object."""
        values = {}
        for key in self.info.getchildnames():
            ci = self.info.getinfo(key)
            if self._values.has_key(key):
                L = self._values[key]
                if len(L) < ci.minOccurs:
                    raise ZConfig.ConfigurationError(
                        "not enough values for %s; %d found, %d required"
                        % (`key`, len(L), ci.minOccurs))
            else:
                L = ci.getdefault()

    def constuct(self):
        keys = self.info.getchildnames()
        values = [None] * len(keys)
        for i in range(len(keys)):
            name = keys[i]
            ci = self.info.getinfo(name)
            if self._values.has_key(name):
                L = self._values[key]
            elif ci.ismulti():
                L = ci.getdefault()
            else:
                L = [ci.getdefault()]
            if ci.ismulti():
                values[i] = [ci.datatype.convert(s) for s in L]
            else:
                assert len(L) == 1
                values[i] = ci.datatype.convert(L[0])
        v = SectionValue(keys, values)
        # XXX  Really should delay this until after all the
        # XXX  sibling SectionValue instances have been created and
        # XXX  we're ready to construct the parent.
        if self.info.handler:
            v = self.info.handler(v)
        return v


class SectionValue:
    def __init__(self, keys, values):
        self._keys = keys
        self._values = values

    def __len__(self):
        return len(self._keys)

    def __getitem__(self, index):
        if isinstance(index, slice):
            raise TypeError("SectionValue does not support slicing")
        return self._values[index]

    def __getattr__(self, name):
        try:
            i = self._keys.index(name)
        except ValueError:
            raise AttributeError, name
        return self._values[i]


=== Packages/ZConfig/SchemaInfo.py 1.1.2.2 => 1.1.2.3 ===
--- Packages/ZConfig/SchemaInfo.py:1.1.2.2	Fri Dec  6 21:37:46 2002
+++ Packages/ZConfig/SchemaInfo.py	Mon Dec  9 12:02:25 2002
@@ -5,10 +5,11 @@
     """Information about a single configuration key."""
 
     def __init__(self, name, datatype, minOccurs, maxOccurs, handler):
-        if maxOccurs < 1:
-            raise ValueError("maxOccurs must be at least 1")
-        if minOccurs < maxOccurs:
-            raise ValueError("minOccurs must be at least maxOccurs")
+        if maxOccurs is not None and maxOccurs < 1:
+            if maxOccurs < 1:
+                raise ValueError("maxOccurs must be at least 1")
+            if minOccurs is not None and minOccurs < maxOccurs:
+                raise ValueError("minOccurs must be at least maxOccurs")
         self.name = name
         self.datatype = datatype
         self.minOccurs = minOccurs
@@ -20,6 +21,11 @@
     def finish(self):
         if self._finished:
             raise ValueError("cannot finish KeyInfo more than once")
+        if self.minOccurs is None:
+            if self._default is not None:
+                self.minOccurs = 1
+            else:
+                self.minOccurs = 0
         self._finished = True
 
     def adddefault(self, value):
@@ -41,8 +47,12 @@
             raise ValueError("cannot get default value of key before KeyInfo"
                              " has been completely initialized")
         if self._default is None and self.maxOccurs > 1:
-            self._default = []
-        return self._default
+            return []
+        else:
+            return self._default
+
+    def hasdefault(self):
+        return self._default is not None
 
     def ismulti(self):
         return self.maxOccurs > 1
@@ -68,12 +78,12 @@
 
     def _add_child(self, thing):
         for c in self._children:
-            if name == c.name:
-                raise ValueError("child name %s already used" % name)
+            if thing.name == c.name:
+                raise ValueError("child name %s already used" % thing.name)
         self._children.append(thing)
 
     def addkey(self, keyinfo):
-        self._add_child(keyinfo.name, keyinfo)
+        self._add_child(keyinfo)
 
     def getinfo(self, name):
         for c in self._children:


=== Packages/ZConfig/SchemaParser.py 1.1.4.1 => 1.1.4.2 ===
--- Packages/ZConfig/SchemaParser.py:1.1.4.1	Fri Dec  6 18:54:03 2002
+++ Packages/ZConfig/SchemaParser.py	Mon Dec  9 12:02:25 2002
@@ -23,6 +23,22 @@
         self._cdata = None
         self._locator = None
         self._prefixes = []
+        self._schema = None
+        self._stack = []
+        self._sections = []
+
+    def parseFile(self, file):
+        if isinstance(file, str) or isinstance(file, unicode):
+            file = open(file, 'rU')
+        return self.parseStream(file)
+
+    def parseString(self, s):
+        xml.sax.parseString(s, self)
+        return self._schema
+
+    def parseStream(self, stream):
+        xml.sax.parse(stream, self)
+        return self._schema
 
     # SAX 2 ContentHandler methods
 
@@ -70,8 +86,6 @@
     def endDocument(self):
         if not self._schema:
             self.doSchemaError("no schema found")
-        self.checkClasses()
-        self.context.setSchema(self._schema)
 
     # schema loading logic
 
@@ -83,7 +97,7 @@
 
     def push_prefix(self, attrs):
         name = attrs.get("prefix")
-        self._prefixes.append(self.get_classname(name) or "")
+        self._prefixes.append(self.get_classname(name or ""))
 
     def start_schema(self, attrs):
         self.start_section(attrs)
@@ -119,18 +133,21 @@
         else:
             names = None
         maxOccurs, minOccurs, handler = self.get_common_info(attrs)
-        
+        section = SectionInfo(None, None, minOccurs, maxOccurs, handler,
+                              keytype, names)
+        self._stack.append(section)
+        self._sections.append(section)
 
     def end_section(self):
         del self._prefixes[-1]
-        self._stack[-1].finish()
+        del self._sections[-1]
+        self._stack.pop().finish()
 
     def start_sectiongroup(self, attrs):
         self.push_prefix(attrs)
 
     def end_sectiongroup(self):
         del self._prefixes[-1]
-        self._stack.pop().finish()
 
     def start_key(self, attrs):
         name = attrs.get("name")
@@ -149,12 +166,22 @@
 
     def get_common_info(self, attrs):
         maxOccurs = attrs.get("maxOccurs")
-        if maxOccurs:
-            maxOccurs = int(maxOccurs)
+        if maxOccurs is not None:
+            if maxOccurs == "unbounded":
+                maxOccurs = None
+            else:
+                maxOccurs = int(maxOccurs)
+        else:
+            maxOccurs = 1
         minOccurs = attrs.get("minOccurs")
         if minOccurs:
             minOccurs = int(minOccurs)
+        else:
+            minOccurs = None
         handler = attrs.get("handler")
         if handler:
             handler = self.get_classname(handler)
         return maxOccurs, minOccurs, handler
+
+    def doSchemaError(self, message):
+        raise ZConfig.ConfigurationError(message)