|
|
@@ -1,586 +0,0 @@
|
|
|
-#!/usr/bin/env python
|
|
|
-"Makes working with XML feel like you are working with JSON"
|
|
|
-
|
|
|
-try:
|
|
|
- from defusedexpat import pyexpat as expat
|
|
|
-except ImportError:
|
|
|
- from xml.parsers import expat
|
|
|
-
|
|
|
-from xml.sax.saxutils import XMLGenerator
|
|
|
-from xml.sax.xmlreader import AttributesImpl
|
|
|
-
|
|
|
-try: # pragma no cover
|
|
|
- from cStringIO import StringIO
|
|
|
-except ImportError: # pragma no cover
|
|
|
- try:
|
|
|
- from StringIO import StringIO
|
|
|
- except ImportError:
|
|
|
- from io import StringIO
|
|
|
-
|
|
|
-from inspect import isgenerator
|
|
|
-
|
|
|
-
|
|
|
-class ObjectDict(dict):
|
|
|
- def __getattr__(self, name):
|
|
|
- if name in self:
|
|
|
- return self[name]
|
|
|
- else:
|
|
|
- raise AttributeError("No such attribute: " + name)
|
|
|
-
|
|
|
-
|
|
|
-try: # pragma no cover
|
|
|
- _basestring = basestring
|
|
|
-except NameError: # pragma no cover
|
|
|
- _basestring = str
|
|
|
-try: # pragma no cover
|
|
|
- _unicode = unicode
|
|
|
-except NameError: # pragma no cover
|
|
|
- _unicode = str
|
|
|
-
|
|
|
-__author__ = "Martin Blech"
|
|
|
-__version__ = "0.12.0"
|
|
|
-__license__ = "MIT"
|
|
|
-
|
|
|
-
|
|
|
-class ParsingInterrupted(Exception):
|
|
|
- pass
|
|
|
-
|
|
|
-
|
|
|
-class _DictSAXHandler(object):
|
|
|
- def __init__(
|
|
|
- self,
|
|
|
- item_depth=0,
|
|
|
- item_callback=lambda *args: True,
|
|
|
- xml_attribs=True,
|
|
|
- attr_prefix="@",
|
|
|
- cdata_key="#text",
|
|
|
- force_cdata=False,
|
|
|
- cdata_separator="",
|
|
|
- postprocessor=None,
|
|
|
- dict_constructor=ObjectDict,
|
|
|
- strip_whitespace=True,
|
|
|
- namespace_separator=":",
|
|
|
- namespaces=None,
|
|
|
- force_list=None,
|
|
|
- comment_key="#comment",
|
|
|
- ):
|
|
|
- self.path = []
|
|
|
- self.stack = []
|
|
|
- self.data = []
|
|
|
- self.item = None
|
|
|
- self.item_depth = item_depth
|
|
|
- self.xml_attribs = xml_attribs
|
|
|
- self.item_callback = item_callback
|
|
|
- self.attr_prefix = attr_prefix
|
|
|
- self.cdata_key = cdata_key
|
|
|
- self.force_cdata = force_cdata
|
|
|
- self.cdata_separator = cdata_separator
|
|
|
- self.postprocessor = postprocessor
|
|
|
- self.dict_constructor = dict_constructor
|
|
|
- self.strip_whitespace = strip_whitespace
|
|
|
- self.namespace_separator = namespace_separator
|
|
|
- self.namespaces = namespaces
|
|
|
- self.namespace_declarations = ObjectDict()
|
|
|
- self.force_list = force_list
|
|
|
- self.comment_key = comment_key
|
|
|
-
|
|
|
- def _build_name(self, full_name):
|
|
|
- if self.namespaces is None:
|
|
|
- return full_name
|
|
|
- i = full_name.rfind(self.namespace_separator)
|
|
|
- if i == -1:
|
|
|
- return full_name
|
|
|
- namespace, name = full_name[:i], full_name[i + 1 :]
|
|
|
- try:
|
|
|
- short_namespace = self.namespaces[namespace]
|
|
|
- except KeyError:
|
|
|
- short_namespace = namespace
|
|
|
- if not short_namespace:
|
|
|
- return name
|
|
|
- else:
|
|
|
- return self.namespace_separator.join((short_namespace, name))
|
|
|
-
|
|
|
- def _attrs_to_dict(self, attrs):
|
|
|
- if isinstance(attrs, dict):
|
|
|
- return attrs
|
|
|
- return self.dict_constructor(zip(attrs[0::2], attrs[1::2]))
|
|
|
-
|
|
|
- def startNamespaceDecl(self, prefix, uri):
|
|
|
- self.namespace_declarations[prefix or ""] = uri
|
|
|
-
|
|
|
- def startElement(self, full_name, attrs):
|
|
|
- name = self._build_name(full_name)
|
|
|
- attrs = self._attrs_to_dict(attrs)
|
|
|
- if attrs and self.namespace_declarations:
|
|
|
- attrs["xmlns"] = self.namespace_declarations
|
|
|
- self.namespace_declarations = ObjectDict()
|
|
|
- self.path.append((name, attrs or None))
|
|
|
- if len(self.path) > self.item_depth:
|
|
|
- self.stack.append((self.item, self.data))
|
|
|
- if self.xml_attribs:
|
|
|
- attr_entries = []
|
|
|
- for key, value in attrs.items():
|
|
|
- key = self.attr_prefix + self._build_name(key)
|
|
|
- if self.postprocessor:
|
|
|
- entry = self.postprocessor(self.path, key, value)
|
|
|
- else:
|
|
|
- entry = (key, value)
|
|
|
- if entry:
|
|
|
- attr_entries.append(entry)
|
|
|
- attrs = self.dict_constructor(attr_entries)
|
|
|
- else:
|
|
|
- attrs = None
|
|
|
- self.item = attrs or None
|
|
|
- self.data = []
|
|
|
-
|
|
|
- def endElement(self, full_name):
|
|
|
- name = self._build_name(full_name)
|
|
|
- if len(self.path) == self.item_depth:
|
|
|
- item = self.item
|
|
|
- if item is None:
|
|
|
- item = None if not self.data else self.cdata_separator.join(self.data)
|
|
|
-
|
|
|
- should_continue = self.item_callback(self.path, item)
|
|
|
- if not should_continue:
|
|
|
- raise ParsingInterrupted()
|
|
|
- if len(self.stack):
|
|
|
- data = None if not self.data else self.cdata_separator.join(self.data)
|
|
|
- item = self.item
|
|
|
- self.item, self.data = self.stack.pop()
|
|
|
- if self.strip_whitespace and data:
|
|
|
- data = data.strip() or None
|
|
|
- if data and self.force_cdata and item is None:
|
|
|
- item = self.dict_constructor()
|
|
|
- if item is not None:
|
|
|
- if data:
|
|
|
- self.push_data(item, self.cdata_key, data)
|
|
|
- self.item = self.push_data(self.item, name, item)
|
|
|
- else:
|
|
|
- self.item = self.push_data(self.item, name, data)
|
|
|
- else:
|
|
|
- self.item = None
|
|
|
- self.data = []
|
|
|
- self.path.pop()
|
|
|
-
|
|
|
- def characters(self, data):
|
|
|
- if not self.data:
|
|
|
- self.data = [data]
|
|
|
- else:
|
|
|
- self.data.append(data)
|
|
|
-
|
|
|
- def comments(self, data):
|
|
|
- if self.strip_whitespace:
|
|
|
- data = data.strip()
|
|
|
- self.item = self.push_data(self.item, self.comment_key, data)
|
|
|
-
|
|
|
- def push_data(self, item, key, data):
|
|
|
- if self.postprocessor is not None:
|
|
|
- result = self.postprocessor(self.path, key, data)
|
|
|
- if result is None:
|
|
|
- return item
|
|
|
- key, data = result
|
|
|
- if item is None:
|
|
|
- item = self.dict_constructor()
|
|
|
- try:
|
|
|
- value = item[key]
|
|
|
- if isinstance(value, list):
|
|
|
- value.append(data)
|
|
|
- else:
|
|
|
- item[key] = [value, data]
|
|
|
- except KeyError:
|
|
|
- if self._should_force_list(key, data):
|
|
|
- item[key] = [data]
|
|
|
- else:
|
|
|
- item[key] = data
|
|
|
- return item
|
|
|
-
|
|
|
- def _should_force_list(self, key, value):
|
|
|
- if not self.force_list:
|
|
|
- return False
|
|
|
- if isinstance(self.force_list, bool):
|
|
|
- return self.force_list
|
|
|
- try:
|
|
|
- return key in self.force_list
|
|
|
- except TypeError:
|
|
|
- return self.force_list(self.path[:-1], key, value)
|
|
|
-
|
|
|
-
|
|
|
-def parse(
|
|
|
- xml_input,
|
|
|
- encoding=None,
|
|
|
- expat=expat,
|
|
|
- process_namespaces=False,
|
|
|
- namespace_separator=":",
|
|
|
- disable_entities=True,
|
|
|
- process_comments=False,
|
|
|
- **kwargs
|
|
|
-):
|
|
|
- """Parse the given XML input and convert it into a dictionary.
|
|
|
-
|
|
|
- `xml_input` can either be a `string`, a file-like object, or a generator of strings.
|
|
|
-
|
|
|
- If `xml_attribs` is `True`, element attributes are put in the dictionary
|
|
|
- among regular child elements, using `@` as a prefix to avoid collisions. If
|
|
|
- set to `False`, they are just ignored.
|
|
|
-
|
|
|
- Simple example::
|
|
|
-
|
|
|
- >>> import xmltodict
|
|
|
- >>> doc = xmltodict.parse(\"\"\"
|
|
|
- ... <a prop="x">
|
|
|
- ... <b>1</b>
|
|
|
- ... <b>2</b>
|
|
|
- ... </a>
|
|
|
- ... \"\"\")
|
|
|
- >>> doc['a']['@prop']
|
|
|
- u'x'
|
|
|
- >>> doc['a']['b']
|
|
|
- [u'1', u'2']
|
|
|
-
|
|
|
- If `item_depth` is `0`, the function returns a dictionary for the root
|
|
|
- element (default behavior). Otherwise, it calls `item_callback` every time
|
|
|
- an item at the specified depth is found and returns `None` in the end
|
|
|
- (streaming mode).
|
|
|
-
|
|
|
- The callback function receives two parameters: the `path` from the document
|
|
|
- root to the item (name-attribs pairs), and the `item` (dict). If the
|
|
|
- callback's return value is false-ish, parsing will be stopped with the
|
|
|
- :class:`ParsingInterrupted` exception.
|
|
|
-
|
|
|
- Streaming example::
|
|
|
-
|
|
|
- >>> def handle(path, item):
|
|
|
- ... print('path:%s item:%s' % (path, item))
|
|
|
- ... return True
|
|
|
- ...
|
|
|
- >>> xmltodict.parse(\"\"\"
|
|
|
- ... <a prop="x">
|
|
|
- ... <b>1</b>
|
|
|
- ... <b>2</b>
|
|
|
- ... </a>\"\"\", item_depth=2, item_callback=handle)
|
|
|
- path:[(u'a', {u'prop': u'x'}), (u'b', None)] item:1
|
|
|
- path:[(u'a', {u'prop': u'x'}), (u'b', None)] item:2
|
|
|
-
|
|
|
- The optional argument `postprocessor` is a function that takes `path`,
|
|
|
- `key` and `value` as positional arguments and returns a new `(key, value)`
|
|
|
- pair where both `key` and `value` may have changed. Usage example::
|
|
|
-
|
|
|
- >>> def postprocessor(path, key, value):
|
|
|
- ... try:
|
|
|
- ... return key + ':int', int(value)
|
|
|
- ... except (ValueError, TypeError):
|
|
|
- ... return key, value
|
|
|
- >>> xmltodict.parse('<a><b>1</b><b>2</b><b>x</b></a>',
|
|
|
- ... postprocessor=postprocessor)
|
|
|
- ObjectDict([(u'a', ObjectDict([(u'b:int', [1, 2]), (u'b', u'x')]))])
|
|
|
-
|
|
|
- You can pass an alternate version of `expat` (such as `defusedexpat`) by
|
|
|
- using the `expat` parameter. E.g:
|
|
|
-
|
|
|
- >>> import defusedexpat
|
|
|
- >>> xmltodict.parse('<a>hello</a>', expat=defusedexpat.pyexpat)
|
|
|
- ObjectDict([(u'a', u'hello')])
|
|
|
-
|
|
|
- You can use the force_list argument to force lists to be created even
|
|
|
- when there is only a single child of a given level of hierarchy. The
|
|
|
- force_list argument is a tuple of keys. If the key for a given level
|
|
|
- of hierarchy is in the force_list argument, that level of hierarchy
|
|
|
- will have a list as a child (even if there is only one sub-element).
|
|
|
- The index_keys operation takes precedence over this. This is applied
|
|
|
- after any user-supplied postprocessor has already run.
|
|
|
-
|
|
|
- For example, given this input:
|
|
|
- <servers>
|
|
|
- <server>
|
|
|
- <name>host1</name>
|
|
|
- <os>Linux</os>
|
|
|
- <interfaces>
|
|
|
- <interface>
|
|
|
- <name>em0</name>
|
|
|
- <ip_address>10.0.0.1</ip_address>
|
|
|
- </interface>
|
|
|
- </interfaces>
|
|
|
- </server>
|
|
|
- </servers>
|
|
|
-
|
|
|
- If called with force_list=('interface',), it will produce
|
|
|
- this dictionary:
|
|
|
- {'servers':
|
|
|
- {'server':
|
|
|
- {'name': 'host1',
|
|
|
- 'os': 'Linux'},
|
|
|
- 'interfaces':
|
|
|
- {'interface':
|
|
|
- [ {'name': 'em0', 'ip_address': '10.0.0.1' } ] } } }
|
|
|
-
|
|
|
- `force_list` can also be a callable that receives `path`, `key` and
|
|
|
- `value`. This is helpful in cases where the logic that decides whether
|
|
|
- a list should be forced is more complex.
|
|
|
-
|
|
|
-
|
|
|
- If `process_comment` is `True` then comment will be added with comment_key
|
|
|
- (default=`'#comment'`) to then tag which contains comment
|
|
|
-
|
|
|
- For example, given this input:
|
|
|
- <a>
|
|
|
- <b>
|
|
|
- <!-- b comment -->
|
|
|
- <c>
|
|
|
- <!-- c comment -->
|
|
|
- 1
|
|
|
- </c>
|
|
|
- <d>2</d>
|
|
|
- </b>
|
|
|
- </a>
|
|
|
-
|
|
|
- If called with process_comment=True, it will produce
|
|
|
- this dictionary:
|
|
|
- 'a': {
|
|
|
- 'b': {
|
|
|
- '#comment': 'b comment',
|
|
|
- 'c': {
|
|
|
-
|
|
|
- '#comment': 'c comment',
|
|
|
- '#text': '1',
|
|
|
- },
|
|
|
- 'd': '2',
|
|
|
- },
|
|
|
- }
|
|
|
- """
|
|
|
- handler = _DictSAXHandler(namespace_separator=namespace_separator, **kwargs)
|
|
|
- if isinstance(xml_input, _unicode):
|
|
|
- if not encoding:
|
|
|
- encoding = "utf-8"
|
|
|
- xml_input = xml_input.encode(encoding)
|
|
|
- if not process_namespaces:
|
|
|
- namespace_separator = None
|
|
|
- parser = expat.ParserCreate(encoding, namespace_separator)
|
|
|
- try:
|
|
|
- parser.ordered_attributes = True
|
|
|
- except AttributeError:
|
|
|
- # Jython's expat does not support ordered_attributes
|
|
|
- pass
|
|
|
- parser.StartNamespaceDeclHandler = handler.startNamespaceDecl
|
|
|
- parser.StartElementHandler = handler.startElement
|
|
|
- parser.EndElementHandler = handler.endElement
|
|
|
- parser.CharacterDataHandler = handler.characters
|
|
|
- if process_comments:
|
|
|
- parser.CommentHandler = handler.comments
|
|
|
- parser.buffer_text = True
|
|
|
- if disable_entities:
|
|
|
- try:
|
|
|
- # Attempt to disable DTD in Jython's expat parser (Xerces-J).
|
|
|
- feature = "http://apache.org/xml/features/disallow-doctype-decl"
|
|
|
- parser._reader.setFeature(feature, True)
|
|
|
- except AttributeError:
|
|
|
- # For CPython / expat parser.
|
|
|
- # Anything not handled ends up here and entities aren't expanded.
|
|
|
- parser.DefaultHandler = lambda x: None
|
|
|
- # Expects an integer return; zero means failure -> expat.ExpatError.
|
|
|
- parser.ExternalEntityRefHandler = lambda *x: 1
|
|
|
- if hasattr(xml_input, "read"):
|
|
|
- parser.ParseFile(xml_input)
|
|
|
- elif isgenerator(xml_input):
|
|
|
- for chunk in xml_input:
|
|
|
- parser.Parse(chunk, False)
|
|
|
- parser.Parse(b"", True)
|
|
|
- else:
|
|
|
- parser.Parse(xml_input, True)
|
|
|
- return handler.item
|
|
|
-
|
|
|
-
|
|
|
-def _process_namespace(name, namespaces, ns_sep=":", attr_prefix="@"):
|
|
|
- if not namespaces:
|
|
|
- return name
|
|
|
- try:
|
|
|
- ns, name = name.rsplit(ns_sep, 1)
|
|
|
- except ValueError:
|
|
|
- pass
|
|
|
- else:
|
|
|
- ns_res = namespaces.get(ns.strip(attr_prefix))
|
|
|
- name = (
|
|
|
- "{}{}{}{}".format(
|
|
|
- attr_prefix if ns.startswith(attr_prefix) else "", ns_res, ns_sep, name
|
|
|
- )
|
|
|
- if ns_res
|
|
|
- else name
|
|
|
- )
|
|
|
- return name
|
|
|
-
|
|
|
-
|
|
|
-def _emit(
|
|
|
- key,
|
|
|
- value,
|
|
|
- content_handler,
|
|
|
- attr_prefix="@",
|
|
|
- cdata_key="#text",
|
|
|
- depth=0,
|
|
|
- preprocessor=None,
|
|
|
- pretty=False,
|
|
|
- newl="\n",
|
|
|
- indent="\t",
|
|
|
- namespace_separator=":",
|
|
|
- namespaces=None,
|
|
|
- full_document=True,
|
|
|
- expand_iter=None,
|
|
|
-):
|
|
|
- key = _process_namespace(key, namespaces, namespace_separator, attr_prefix)
|
|
|
- if preprocessor is not None:
|
|
|
- result = preprocessor(key, value)
|
|
|
- if result is None:
|
|
|
- return
|
|
|
- key, value = result
|
|
|
- if (
|
|
|
- not hasattr(value, "__iter__")
|
|
|
- or isinstance(value, _basestring)
|
|
|
- or isinstance(value, dict)
|
|
|
- ):
|
|
|
- value = [value]
|
|
|
- for index, v in enumerate(value):
|
|
|
- if full_document and depth == 0 and index > 0:
|
|
|
- raise ValueError("document with multiple roots")
|
|
|
- if v is None:
|
|
|
- v = ObjectDict()
|
|
|
- elif isinstance(v, bool):
|
|
|
- if v:
|
|
|
- v = _unicode("true")
|
|
|
- else:
|
|
|
- v = _unicode("false")
|
|
|
- elif not isinstance(v, dict):
|
|
|
- if (
|
|
|
- expand_iter
|
|
|
- and hasattr(v, "__iter__")
|
|
|
- and not isinstance(v, _basestring)
|
|
|
- ):
|
|
|
- v = ObjectDict(((expand_iter, v),))
|
|
|
- else:
|
|
|
- v = _unicode(v)
|
|
|
- if isinstance(v, _basestring):
|
|
|
- v = ObjectDict(((cdata_key, v),))
|
|
|
- cdata = None
|
|
|
- attrs = ObjectDict()
|
|
|
- children = []
|
|
|
- for ik, iv in v.items():
|
|
|
- if ik == cdata_key:
|
|
|
- cdata = iv
|
|
|
- continue
|
|
|
- if ik.startswith(attr_prefix):
|
|
|
- ik = _process_namespace(
|
|
|
- ik, namespaces, namespace_separator, attr_prefix
|
|
|
- )
|
|
|
- if ik == "@xmlns" and isinstance(iv, dict):
|
|
|
- for k, v in iv.items():
|
|
|
- attr = "xmlns{}".format(":{}".format(k) if k else "")
|
|
|
- attrs[attr] = _unicode(v)
|
|
|
- continue
|
|
|
- if not isinstance(iv, _unicode):
|
|
|
- iv = _unicode(iv)
|
|
|
- attrs[ik[len(attr_prefix) :]] = iv
|
|
|
- continue
|
|
|
- children.append((ik, iv))
|
|
|
- if pretty:
|
|
|
- content_handler.ignorableWhitespace(depth * indent)
|
|
|
- content_handler.startElement(key, AttributesImpl(attrs))
|
|
|
- if pretty and children:
|
|
|
- content_handler.ignorableWhitespace(newl)
|
|
|
- for child_key, child_value in children:
|
|
|
- _emit(
|
|
|
- child_key,
|
|
|
- child_value,
|
|
|
- content_handler,
|
|
|
- attr_prefix,
|
|
|
- cdata_key,
|
|
|
- depth + 1,
|
|
|
- preprocessor,
|
|
|
- pretty,
|
|
|
- newl,
|
|
|
- indent,
|
|
|
- namespaces=namespaces,
|
|
|
- namespace_separator=namespace_separator,
|
|
|
- expand_iter=expand_iter,
|
|
|
- )
|
|
|
- if cdata is not None:
|
|
|
- content_handler.characters(cdata)
|
|
|
- if pretty and children:
|
|
|
- content_handler.ignorableWhitespace(depth * indent)
|
|
|
- content_handler.endElement(key)
|
|
|
- if pretty and depth:
|
|
|
- content_handler.ignorableWhitespace(newl)
|
|
|
-
|
|
|
-
|
|
|
-def unparse(
|
|
|
- input_dict,
|
|
|
- output=None,
|
|
|
- encoding="utf-8",
|
|
|
- full_document=True,
|
|
|
- short_empty_elements=False,
|
|
|
- **kwargs
|
|
|
-):
|
|
|
- """Emit an XML document for the given `input_dict` (reverse of `parse`).
|
|
|
-
|
|
|
- The resulting XML document is returned as a string, but if `output` (a
|
|
|
- file-like object) is specified, it is written there instead.
|
|
|
-
|
|
|
- Dictionary keys prefixed with `attr_prefix` (default=`'@'`) are interpreted
|
|
|
- as XML node attributes, whereas keys equal to `cdata_key`
|
|
|
- (default=`'#text'`) are treated as character data.
|
|
|
-
|
|
|
- The `pretty` parameter (default=`False`) enables pretty-printing. In this
|
|
|
- mode, lines are terminated with `'\n'` and indented with `'\t'`, but this
|
|
|
- can be customized with the `newl` and `indent` parameters.
|
|
|
-
|
|
|
- """
|
|
|
- if full_document and len(input_dict) != 1:
|
|
|
- raise ValueError("Document must have exactly one root.")
|
|
|
- must_return = False
|
|
|
- if output is None:
|
|
|
- output = StringIO()
|
|
|
- must_return = True
|
|
|
- if short_empty_elements:
|
|
|
- content_handler = XMLGenerator(output, encoding, True)
|
|
|
- else:
|
|
|
- content_handler = XMLGenerator(output, encoding)
|
|
|
- if full_document:
|
|
|
- content_handler.startDocument()
|
|
|
- for key, value in input_dict.items():
|
|
|
- _emit(key, value, content_handler, full_document=full_document, **kwargs)
|
|
|
- if full_document:
|
|
|
- content_handler.endDocument()
|
|
|
- if must_return:
|
|
|
- value = output.getvalue()
|
|
|
- try: # pragma no cover
|
|
|
- value = value.decode(encoding)
|
|
|
- except AttributeError: # pragma no cover
|
|
|
- pass
|
|
|
- return value
|
|
|
-
|
|
|
-
|
|
|
-if __name__ == "__main__": # pragma: no cover
|
|
|
- import sys
|
|
|
- import marshal
|
|
|
-
|
|
|
- try:
|
|
|
- stdin = sys.stdin.buffer
|
|
|
- stdout = sys.stdout.buffer
|
|
|
- except AttributeError:
|
|
|
- stdin = sys.stdin
|
|
|
- stdout = sys.stdout
|
|
|
-
|
|
|
- (item_depth,) = sys.argv[1:]
|
|
|
- item_depth = int(item_depth)
|
|
|
-
|
|
|
- def handle_item(path, item):
|
|
|
- marshal.dump((path, item), stdout)
|
|
|
- return True
|
|
|
-
|
|
|
- try:
|
|
|
- root = parse(
|
|
|
- stdin,
|
|
|
- item_depth=item_depth,
|
|
|
- item_callback=handle_item,
|
|
|
- dict_constructor=dict,
|
|
|
- )
|
|
|
- if item_depth == 0:
|
|
|
- handle_item([], root)
|
|
|
- except KeyboardInterrupt:
|
|
|
- pass
|