[Zope-Checkins] CVS: Packages/ZConfig - info.py:1.1.2.4 matcher.py:1.1.2.5 schema.py:1.1.2.3
Fred L. Drake, Jr.
fred@zope.com
Wed, 11 Dec 2002 01:10:37 -0500
Update of /cvs-repository/Packages/ZConfig
In directory cvs.zope.org:/tmp/cvs-serv2496
Modified Files:
Tag: zconfig-schema-devel-branch
info.py matcher.py schema.py
Log Message:
Basic nested sections work! Not very clean, but working.
Needs refactoring/cleanup, lots of unit tests.
=== Packages/ZConfig/info.py 1.1.2.3 => 1.1.2.4 ===
--- Packages/ZConfig/info.py:1.1.2.3 Tue Dec 10 18:25:01 2002
+++ Packages/ZConfig/info.py Wed Dec 11 01:10:36 2002
@@ -106,51 +106,75 @@
assert keytype is not None
self.typename = typename
self.keytype = keytype
- self.names = names # '*', '+', or [name1, ...]
+ self.names = names # '*', '+', or [name1, ...]
self.nametype = nametype
- self._children = [] # [(name, info), ...]
- self._attrmap = {} # {attribute: index, ...}
+ self._children = [] # [info, ...]
+ self._attrmap = {} # {attribute: index, ...}
+ self._keymap = {} # {key: index, ...}
+ self._secttypemap = {} # {section type: info, ...}
+ if datatype is None:
+ datatype = _null_conversion
KeyInfo.__init__(self, None, datatype, minOccurs, maxOccurs,
handler, attribute)
- def _add_child(self, name, thing):
- for n, info in self._children:
- if name and n == name:
- raise ValueError("child name %s already used" % name)
- if info.attribute and info.attribute == thing.attribute:
- raise ValueError("child attribute name %s already used"
- % thing.attribute)
- self._children.append((name, thing))
+ def __len__(self):
+ return len(self._children)
+
+ def __getitem__(self, index):
+ return self._children[index]
+
+ def _add_child(self, key, info):
+ # check naming constraints
+ assert key or info.attribute
+ if key and self._keymap.has_key(key):
+ raise ValueError("child name %s already used" % key)
+ if info.attribute and self._attrmap.has_key(info.attribute):
+ raise ValueError("child attribute name %s already used"
+ % info.attribute)
+ # a-ok, add the item to the appropriate maps
+ if info.attribute:
+ self._attrmap[info.attribute] = len(self._children)
+ if key:
+ self._keymap[key] = len(self._children)
+ self._children.append((key, info))
def addkey(self, keyinfo):
self._add_child(keyinfo.name, keyinfo)
def addsection(self, name, sectinfo):
assert name not in ("*", "+")
+ # Check that this isn't a different definition of the section type,
+ # since once _add_child() returns we must succeed, or undo a whole
+ # bunch of stuff. This is the easier alternative. ;-)
+ oldinfo = self._secttypemap.get(sectinfo.typename)
+ if oldinfo not in (None, sectinfo):
+ raise ZConfig.ConfigurationError(
+ "duplicate definition of section type " + `name`)
self._add_child(name, sectinfo)
+ self._secttypemap[sectinfo.typename] = sectinfo
- def getinfo(self, name):
- if not name:
+ def getinfo(self, key):
+ if not key:
raise ZConfig.ConfigurationError(
"cannot match a key without a name")
- for n, info in self._children:
- if n == name:
- return info
- raise ZConfig.ConfigurationError("no key matching " + `name`)
+ index = self._keymap.get(key)
+ if index is None:
+ raise ZConfig.ConfigurationError("no key matching " + `key`)
+ else:
+ return self._children[index][1]
def getchildnames(self):
- return [n for (n,info) in self._children]
+ return [key for (key, info) in self._children]
def getsectionindex(self, type, name):
index = -1
- for n, info in self._children:
+ for key, info in self._children:
index += 1
- if n:
- if n == name:
+ if key:
+ if key == name:
if not info.issection():
raise ZConfig.ConfigurationError(
- "section name %s already in use for key"
- % str(n))
+ "section name %s already in use for key" % key)
if not info.typename == type:
raise ZConfig.ConfigurationError(
"name %s must be used for a %s section"
@@ -194,3 +218,10 @@
def isAllowedName(self, name):
return False
+
+
+class NullConversion:
+ def convert(self, value):
+ return value
+
+_null_conversion = NullConversion()
=== Packages/ZConfig/matcher.py 1.1.2.4 => 1.1.2.5 ===
--- Packages/ZConfig/matcher.py:1.1.2.4 Tue Dec 10 18:25:01 2002
+++ Packages/ZConfig/matcher.py Wed Dec 11 01:10:36 2002
@@ -19,68 +19,85 @@
raise ZConfig.ConfigurationError(
`info.name` + " sections may not be unnamed")
self.info = info
- self._values = {}
+ self._values = [None] * len(info)
def addSection(self, type, name, sectvalue):
i = self.info.getsectionindex(type, name)
- raise NotImpementedError("still working on this...")
+ ci = self.info.getsectioninfo(type, name)
+ v = self._values[i]
+ if v is None and ci.ismulti():
+ v = [sectvalue]
+ self._values[i] = v
+ elif ci.ismulti():
+ v.append(sectvalue)
+ else:
+ self._values[i] = sectvalue
def addValue(self, key, value):
- keyinfo = self.info.getinfo(key)
- if keyinfo.issection():
- if keyinfo.name:
+ length = len(self.info)
+ for i in range(length):
+ k, ci = self.info[i]
+ if k == key:
+ break
+ else:
+ raise ZConfig.ConfigurationError(
+ `key` + " is not a known key name")
+ if ci.issection():
+ if ci.name:
extra = " in %s sections" % `self.info.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`)
+
+ v = self._values[i]
+ if v is None:
+ if ci.ismulti():
+ v = []
+ self._values[i] = v
+ elif not ci.ismulti():
+ raise ZConfig.ConfigurationError(
+ `key` + " does not support multiple values")
+ elif len(v) == ci.maxOccurs:
+ raise ZConfig.ConfigurationError(
+ "too many values for " + `name`)
+
+ if ci.ismulti():
+ v.append(value)
else:
- L.append(value)
- self._values[key] = L
+ self._values[i] = value
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()
- return self.constuct()
-
- 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)
+ length = len(self.info)
+ values = [None] * length
+ keys = [None] * length
+ for i in range(length):
+ key, ci = self.info[i]
keys[i] = ci.attribute
- if self._values.has_key(name):
- L = self._values[name]
- elif ci.ismulti():
- L = ci.getdefault()
- else:
- L = [ci.getdefault()]
+ if ci.ismulti() and len(self._values[i]) < ci.minOccurs:
+ raise ZConfig.ConfigurationError(
+ "not enough values for %s; %d found, %d required"
+ % (`key`, len(L), ci.minOccurs))
+ v = self._values[i]
+ if v is None:
+ if ci.ismulti():
+ v = ci.getdefault()[:]
+ else:
+ v = ci.getdefault()
+ self._values[i] = v
+ return self.constuct(keys, values)
+
+ def constuct(self, keys, values):
+ length = len(self.info)
+ for i in range(length):
+ name, ci = self.info[i]
+ dt = ci.datatype
if ci.ismulti():
- values[i] = [ci.datatype.convert(s) for s in L]
- elif ci.issection() and ci.datatype is None:
- values[i] = L[0]
+ values[i] = [dt.convert(s) for s in self._values[i]]
else:
- assert len(L) == 1
- values[i] = ci.datatype.convert(L[0])
+ values[i] = dt.convert(self._values[i])
v = SectionValue(keys, values)
# XXX Really should delay this until after all the
# XXX sibling SectionValue instances have been created and
=== Packages/ZConfig/schema.py 1.1.2.2 => 1.1.2.3 ===
--- Packages/ZConfig/schema.py:1.1.2.2 Tue Dec 10 18:25:01 2002
+++ Packages/ZConfig/schema.py Wed Dec 11 01:10:36 2002
@@ -40,15 +40,15 @@
def startElement(self, name, attrs):
attrs = dict(attrs)
if name == "schema":
- if self._schema:
+ if self._schema is not None:
self.doSchemaError("schema element improperly nested")
self.start_schema(attrs)
elif name in self._handled_tags:
- if not self._schema:
+ if self._schema is None:
self.doSchemaError(name + " element outside of schema")
getattr(self, "start_" + name)(attrs)
elif name in self._cdata_tags:
- if not self._schema:
+ if self._schema is None:
self.doSchemaError(name + " element outside of schema")
if self._cdata is not None:
self.doSchemaError(name + " element improperly nested")