home *** CD-ROM | disk | FTP | other *** search
/ Chip 2004 July / CMCD0704.ISO / Software / Shareware / Comunicatii / jyte / jyte.exe / minidom.py < prev    next >
Text File  |  2003-11-11  |  68KB  |  1,943 lines

  1. """\
  2. minidom.py -- a lightweight DOM implementation.
  3.  
  4. parse("foo.xml")
  5.  
  6. parseString("<foo><bar/></foo>")
  7.  
  8. Todo:
  9. =====
  10.  * convenience methods for getting elements and text.
  11.  * more testing
  12.  * bring some of the writer and linearizer code into conformance with this
  13.         interface
  14.  * SAX 2 namespaces
  15. """
  16.  
  17. import sys
  18. import xml.dom
  19.  
  20. from xml.dom import EMPTY_NAMESPACE, EMPTY_PREFIX, XMLNS_NAMESPACE, domreg
  21. from xml.dom.minicompat import *
  22. from xml.dom.xmlbuilder import DOMImplementationLS, DocumentLS
  23.  
  24. _TupleType = type(())
  25.  
  26. # This is used by the ID-cache invalidation checks; the list isn't
  27. # actually complete, since the nodes being checked will never be the
  28. # DOCUMENT_NODE or DOCUMENT_FRAGMENT_NODE.  (The node being checked is
  29. # the node being added or removed, not the node being modified.)
  30. #
  31. _nodeTypes_with_children = (xml.dom.Node.ELEMENT_NODE,
  32.                             xml.dom.Node.ENTITY_REFERENCE_NODE)
  33.  
  34.  
  35. class Node(xml.dom.Node, GetattrMagic):
  36.     namespaceURI = None # this is non-null only for elements and attributes
  37.     parentNode = None
  38.     ownerDocument = None
  39.     nextSibling = None
  40.     previousSibling = None
  41.  
  42.     prefix = EMPTY_PREFIX # non-null only for NS elements and attributes
  43.  
  44.     def __nonzero__(self):
  45.         return True
  46.  
  47.     def toxml(self, encoding = None):
  48.         return self.toprettyxml("", "", encoding)
  49.  
  50.     def toprettyxml(self, indent="\t", newl="\n", encoding = None):
  51.         # indent = the indentation string to prepend, per level
  52.         # newl = the newline string to append
  53.         writer = _get_StringIO()
  54.         if encoding is not None:
  55.             import codecs
  56.             # Can't use codecs.getwriter to preserve 2.0 compatibility
  57.             writer = codecs.lookup(encoding)[3](writer)
  58.         if self.nodeType == Node.DOCUMENT_NODE:
  59.             # Can pass encoding only to document, to put it into XML header
  60.             self.writexml(writer, "", indent, newl, encoding)
  61.         else:
  62.             self.writexml(writer, "", indent, newl)
  63.         return writer.getvalue()
  64.  
  65.     def hasChildNodes(self):
  66.         if self.childNodes:
  67.             return True
  68.         else:
  69.             return False
  70.  
  71.     def _get_childNodes(self):
  72.         return self.childNodes
  73.  
  74.     def _get_firstChild(self):
  75.         if self.childNodes:
  76.             return self.childNodes[0]
  77.  
  78.     def _get_lastChild(self):
  79.         if self.childNodes:
  80.             return self.childNodes[-1]
  81.  
  82.     def insertBefore(self, newChild, refChild):
  83.         if newChild.nodeType == self.DOCUMENT_FRAGMENT_NODE:
  84.             for c in tuple(newChild.childNodes):
  85.                 self.insertBefore(c, refChild)
  86.             ### The DOM does not clearly specify what to return in this case
  87.             return newChild
  88.         if newChild.nodeType not in self._child_node_types:
  89.             raise xml.dom.HierarchyRequestErr(
  90.                 "%s cannot be child of %s" % (repr(newChild), repr(self)))
  91.         if newChild.parentNode is not None:
  92.             newChild.parentNode.removeChild(newChild)
  93.         if refChild is None:
  94.             self.appendChild(newChild)
  95.         else:
  96.             try:
  97.                 index = self.childNodes.index(refChild)
  98.             except ValueError:
  99.                 raise xml.dom.NotFoundErr()
  100.             if newChild.nodeType in _nodeTypes_with_children:
  101.                 _clear_id_cache(self)
  102.             self.childNodes.insert(index, newChild)
  103.             newChild.nextSibling = refChild
  104.             refChild.previousSibling = newChild
  105.             if index:
  106.                 node = self.childNodes[index-1]
  107.                 node.nextSibling = newChild
  108.                 newChild.previousSibling = node
  109.             else:
  110.                 newChild.previousSibling = None
  111.             newChild.parentNode = self
  112.         return newChild
  113.  
  114.     def appendChild(self, node):
  115.         if node.nodeType == self.DOCUMENT_FRAGMENT_NODE:
  116.             for c in tuple(node.childNodes):
  117.                 self.appendChild(c)
  118.             ### The DOM does not clearly specify what to return in this case
  119.             return node
  120.         if node.nodeType not in self._child_node_types:
  121.             raise xml.dom.HierarchyRequestErr(
  122.                 "%s cannot be child of %s" % (repr(node), repr(self)))
  123.         elif node.nodeType in _nodeTypes_with_children:
  124.             _clear_id_cache(self)
  125.         if node.parentNode is not None:
  126.             node.parentNode.removeChild(node)
  127.         _append_child(self, node)
  128.         node.nextSibling = None
  129.         return node
  130.  
  131.     def replaceChild(self, newChild, oldChild):
  132.         if newChild.nodeType == self.DOCUMENT_FRAGMENT_NODE:
  133.             refChild = oldChild.nextSibling
  134.             self.removeChild(oldChild)
  135.             return self.insertBefore(newChild, refChild)
  136.         if newChild.nodeType not in self._child_node_types:
  137.             raise xml.dom.HierarchyRequestErr(
  138.                 "%s cannot be child of %s" % (repr(newChild), repr(self)))
  139.         if newChild.parentNode is not None:
  140.             newChild.parentNode.removeChild(newChild)
  141.         if newChild is oldChild:
  142.             return
  143.         try:
  144.             index = self.childNodes.index(oldChild)
  145.         except ValueError:
  146.             raise xml.dom.NotFoundErr()
  147.         self.childNodes[index] = newChild
  148.         newChild.parentNode = self
  149.         oldChild.parentNode = None
  150.         if (newChild.nodeType in _nodeTypes_with_children
  151.             or oldChild.nodeType in _nodeTypes_with_children):
  152.             _clear_id_cache(self)
  153.         newChild.nextSibling = oldChild.nextSibling
  154.         newChild.previousSibling = oldChild.previousSibling
  155.         oldChild.nextSibling = None
  156.         oldChild.previousSibling = None
  157.         if newChild.previousSibling:
  158.             newChild.previousSibling.nextSibling = newChild
  159.         if newChild.nextSibling:
  160.             newChild.nextSibling.previousSibling = newChild
  161.         return oldChild
  162.  
  163.     def removeChild(self, oldChild):
  164.         try:
  165.             self.childNodes.remove(oldChild)
  166.         except ValueError:
  167.             raise xml.dom.NotFoundErr()
  168.         if oldChild.nextSibling is not None:
  169.             oldChild.nextSibling.previousSibling = oldChild.previousSibling
  170.         if oldChild.previousSibling is not None:
  171.             oldChild.previousSibling.nextSibling = oldChild.nextSibling
  172.         oldChild.nextSibling = oldChild.previousSibling = None
  173.         if oldChild.nodeType in _nodeTypes_with_children:
  174.             _clear_id_cache(self)
  175.  
  176.         oldChild.parentNode = None
  177.         return oldChild
  178.  
  179.     def normalize(self):
  180.         L = []
  181.         for child in self.childNodes:
  182.             if child.nodeType == Node.TEXT_NODE:
  183.                 data = child.data
  184.                 if data and L and L[-1].nodeType == child.nodeType:
  185.                     # collapse text node
  186.                     node = L[-1]
  187.                     node.data = node.data + child.data
  188.                     node.nextSibling = child.nextSibling
  189.                     child.unlink()
  190.                 elif data:
  191.                     if L:
  192.                         L[-1].nextSibling = child
  193.                         child.previousSibling = L[-1]
  194.                     else:
  195.                         child.previousSibling = None
  196.                     L.append(child)
  197.                 else:
  198.                     # empty text node; discard
  199.                     child.unlink()
  200.             else:
  201.                 if L:
  202.                     L[-1].nextSibling = child
  203.                     child.previousSibling = L[-1]
  204.                 else:
  205.                     child.previousSibling = None
  206.                 L.append(child)
  207.                 if child.nodeType == Node.ELEMENT_NODE:
  208.                     child.normalize()
  209.         self.childNodes[:] = L
  210.  
  211.     def cloneNode(self, deep):
  212.         return _clone_node(self, deep, self.ownerDocument or self)
  213.  
  214.     def isSupported(self, feature, version):
  215.         return self.ownerDocument.implementation.hasFeature(feature, version)
  216.  
  217.     def _get_localName(self):
  218.         # Overridden in Element and Attr where localName can be Non-Null
  219.         return None
  220.  
  221.     # Node interfaces from Level 3 (WD 9 April 2002)
  222.  
  223.     def isSameNode(self, other):
  224.         return self is other
  225.  
  226.     def getInterface(self, feature):
  227.         if self.isSupported(feature, None):
  228.             return self
  229.         else:
  230.             return None
  231.  
  232.     # The "user data" functions use a dictionary that is only present
  233.     # if some user data has been set, so be careful not to assume it
  234.     # exists.
  235.  
  236.     def getUserData(self, key):
  237.         try:
  238.             return self._user_data[key][0]
  239.         except (AttributeError, KeyError):
  240.             return None
  241.  
  242.     def setUserData(self, key, data, handler):
  243.         old = None
  244.         try:
  245.             d = self._user_data
  246.         except AttributeError:
  247.             d = {}
  248.             self._user_data = d
  249.         if d.has_key(key):
  250.             old = d[key][0]
  251.         if data is None:
  252.             # ignore handlers passed for None
  253.             handler = None
  254.             if old is not None:
  255.                 del d[key]
  256.         else:
  257.             d[key] = (data, handler)
  258.         return old
  259.  
  260.     def _call_user_data_handler(self, operation, src, dst):
  261.         if hasattr(self, "_user_data"):
  262.             for key, (data, handler) in self._user_data.items():
  263.                 if handler is not None:
  264.                     handler.handle(operation, key, data, src, dst)
  265.  
  266.     # minidom-specific API:
  267.  
  268.     def unlink(self):
  269.         self.parentNode = self.ownerDocument = None
  270.         if self.childNodes:
  271.             for child in self.childNodes:
  272.                 child.unlink()
  273.             self.childNodes = NodeList()
  274.         self.previousSibling = None
  275.         self.nextSibling = None
  276.  
  277. defproperty(Node, "firstChild", doc="First child node, or None.")
  278. defproperty(Node, "lastChild",  doc="Last child node, or None.")
  279. defproperty(Node, "localName",  doc="Namespace-local name of this node.")
  280.  
  281.  
  282. def _append_child(self, node):
  283.     # fast path with less checks; usable by DOM builders if careful
  284.     childNodes = self.childNodes
  285.     if childNodes:
  286.         last = childNodes[-1]
  287.         node.__dict__["previousSibling"] = last
  288.         last.__dict__["nextSibling"] = node
  289.     childNodes.append(node)
  290.     node.__dict__["parentNode"] = self
  291.  
  292. def _in_document(node):
  293.     # return True iff node is part of a document tree
  294.     while node is not None:
  295.         if node.nodeType == Node.DOCUMENT_NODE:
  296.             return True
  297.         node = node.parentNode
  298.     return False
  299.  
  300. def _write_data(writer, data):
  301.     "Writes datachars to writer."
  302.     data = data.replace("&", "&").replace("<", "<")
  303.     data = data.replace("\"", """).replace(">", ">")
  304.     writer.write(data)
  305.  
  306. def _get_elements_by_tagName_helper(parent, name, rc):
  307.     for node in parent.childNodes:
  308.         if node.nodeType == Node.ELEMENT_NODE and \
  309.             (name == "*" or node.tagName == name):
  310.             rc.append(node)
  311.         _get_elements_by_tagName_helper(node, name, rc)
  312.     return rc
  313.  
  314. def _get_elements_by_tagName_ns_helper(parent, nsURI, localName, rc):
  315.     for node in parent.childNodes:
  316.         if node.nodeType == Node.ELEMENT_NODE:
  317.             if ((localName == "*" or node.localName == localName) and
  318.                 (nsURI == "*" or node.namespaceURI == nsURI)):
  319.                 rc.append(node)
  320.             _get_elements_by_tagName_ns_helper(node, nsURI, localName, rc)
  321.     return rc
  322.  
  323. class DocumentFragment(Node):
  324.     nodeType = Node.DOCUMENT_FRAGMENT_NODE
  325.     nodeName = "#document-fragment"
  326.     nodeValue = None
  327.     attributes = None
  328.     parentNode = None
  329.     _child_node_types = (Node.ELEMENT_NODE,
  330.                          Node.TEXT_NODE,
  331.                          Node.CDATA_SECTION_NODE,
  332.                          Node.ENTITY_REFERENCE_NODE,
  333.                          Node.PROCESSING_INSTRUCTION_NODE,
  334.                          Node.COMMENT_NODE,
  335.                          Node.NOTATION_NODE)
  336.  
  337.     def __init__(self):
  338.         self.childNodes = NodeList()
  339.  
  340.  
  341. class Attr(Node):
  342.     nodeType = Node.ATTRIBUTE_NODE
  343.     attributes = None
  344.     ownerElement = None
  345.     specified = False
  346.     _is_id = False
  347.  
  348.     _child_node_types = (Node.TEXT_NODE, Node.ENTITY_REFERENCE_NODE)
  349.  
  350.     def __init__(self, qName, namespaceURI=EMPTY_NAMESPACE, localName=None,
  351.                  prefix=None):
  352.         # skip setattr for performance
  353.         d = self.__dict__
  354.         d["nodeName"] = d["name"] = qName
  355.         d["namespaceURI"] = namespaceURI
  356.         d["prefix"] = prefix
  357.         d['childNodes'] = NodeList()
  358.  
  359.         # Add the single child node that represents the value of the attr
  360.         self.childNodes.append(Text())
  361.  
  362.         # nodeValue and value are set elsewhere
  363.  
  364.     def _get_localName(self):
  365.         return self.nodeName.split(":", 1)[-1]
  366.  
  367.     def _get_name(self):
  368.         return self.name
  369.  
  370.     def _get_specified(self):
  371.         return self.specified
  372.  
  373.     def __setattr__(self, name, value):
  374.         d = self.__dict__
  375.         if name in ("value", "nodeValue"):
  376.             d["value"] = d["nodeValue"] = value
  377.             d2 = self.childNodes[0].__dict__
  378.             d2["data"] = d2["nodeValue"] = value
  379.             if self.ownerElement is not None:
  380.                 _clear_id_cache(self.ownerElement)
  381.         elif name in ("name", "nodeName"):
  382.             d["name"] = d["nodeName"] = value
  383.             if self.ownerElement is not None:
  384.                 _clear_id_cache(self.ownerElement)
  385.         else:
  386.             d[name] = value
  387.  
  388.     def _set_prefix(self, prefix):
  389.         nsuri = self.namespaceURI
  390.         if prefix == "xmlns":
  391.             if nsuri and nsuri != XMLNS_NAMESPACE:
  392.                 raise xml.dom.NamespaceErr(
  393.                     "illegal use of 'xmlns' prefix for the wrong namespace")
  394.         d = self.__dict__
  395.         d['prefix'] = prefix
  396.         if prefix is None:
  397.             newName = self.localName
  398.         else:
  399.             newName = "%s:%s" % (prefix, self.localName)
  400.         if self.ownerElement:
  401.             _clear_id_cache(self.ownerElement)
  402.         d['nodeName'] = d['name'] = newName
  403.  
  404.     def _set_value(self, value):
  405.         d = self.__dict__
  406.         d['value'] = d['nodeValue'] = value
  407.         if self.ownerElement:
  408.             _clear_id_cache(self.ownerElement)
  409.         self.childNodes[0].data = value
  410.  
  411.     def unlink(self):
  412.         # This implementation does not call the base implementation
  413.         # since most of that is not needed, and the expense of the
  414.         # method call is not warranted.  We duplicate the removal of
  415.         # children, but that's all we needed from the base class.
  416.         elem = self.ownerElement
  417.         if elem is not None:
  418.             del elem._attrs[self.nodeName]
  419.             del elem._attrsNS[(self.namespaceURI, self.localName)]
  420.             if self._is_id:
  421.                 self._is_id = False
  422.                 elem._magic_id_nodes -= 1
  423.                 self.ownerDocument._magic_id_count -= 1
  424.         for child in self.childNodes:
  425.             child.unlink()
  426.         del self.childNodes[:]
  427.  
  428.     def _get_isId(self):
  429.         if self._is_id:
  430.             return True
  431.         doc = self.ownerDocument
  432.         elem = self.ownerElement
  433.         if doc is None or elem is None:
  434.             return False
  435.  
  436.         info = doc._get_elem_info(elem)
  437.         if info is None:
  438.             return False
  439.         if self.namespaceURI:
  440.             return info.isIdNS(self.namespaceURI, self.localName)
  441.         else:
  442.             return info.isId(self.nodeName)
  443.  
  444.     def _get_schemaType(self):
  445.         doc = self.ownerDocument
  446.         elem = self.ownerElement
  447.         if doc is None or elem is None:
  448.             return _no_type
  449.  
  450.         info = doc._get_elem_info(elem)
  451.         if info is None:
  452.             return _no_type
  453.         if self.namespaceURI:
  454.             return info.getAttributeTypeNS(self.namespaceURI, self.localName)
  455.         else:
  456.             return info.getAttributeType(self.nodeName)
  457.  
  458. defproperty(Attr, "isId",       doc="True if this attribute is an ID.")
  459. defproperty(Attr, "localName",  doc="Namespace-local name of this attribute.")
  460. defproperty(Attr, "schemaType", doc="Schema type for this attribute.")
  461.  
  462.  
  463. class NamedNodeMap(NewStyle, GetattrMagic):
  464.     """The attribute list is a transient interface to the underlying
  465.     dictionaries.  Mutations here will change the underlying element's
  466.     dictionary.
  467.  
  468.     Ordering is imposed artificially and does not reflect the order of
  469.     attributes as found in an input document.
  470.     """
  471.  
  472.     __slots__ = ('_attrs', '_attrsNS', '_ownerElement')
  473.  
  474.     def __init__(self, attrs, attrsNS, ownerElement):
  475.         self._attrs = attrs
  476.         self._attrsNS = attrsNS
  477.         self._ownerElement = ownerElement
  478.  
  479.     def _get_length(self):
  480.         return len(self._attrs)
  481.  
  482.     def item(self, index):
  483.         try:
  484.             return self[self._attrs.keys()[index]]
  485.         except IndexError:
  486.             return None
  487.  
  488.     def items(self):
  489.         L = []
  490.         for node in self._attrs.values():
  491.             L.append((node.nodeName, node.value))
  492.         return L
  493.  
  494.     def itemsNS(self):
  495.         L = []
  496.         for node in self._attrs.values():
  497.             L.append(((node.namespaceURI, node.localName), node.value))
  498.         return L
  499.  
  500.     def has_key(self, key):
  501.         if isinstance(key, StringTypes):
  502.             return self._attrs.has_key(key)
  503.         else:
  504.             return self._attrsNS.has_key(key)
  505.  
  506.     def keys(self):
  507.         return self._attrs.keys()
  508.  
  509.     def keysNS(self):
  510.         return self._attrsNS.keys()
  511.  
  512.     def values(self):
  513.         return self._attrs.values()
  514.  
  515.     def get(self, name, value=None):
  516.         return self._attrs.get(name, value)
  517.  
  518.     __len__ = _get_length
  519.  
  520.     def __cmp__(self, other):
  521.         if self._attrs is getattr(other, "_attrs", None):
  522.             return 0
  523.         else:
  524.             return cmp(id(self), id(other))
  525.  
  526.     def __getitem__(self, attname_or_tuple):
  527.         if isinstance(attname_or_tuple, _TupleType):
  528.             return self._attrsNS[attname_or_tuple]
  529.         else:
  530.             return self._attrs[attname_or_tuple]
  531.  
  532.     # same as set
  533.     def __setitem__(self, attname, value):
  534.         if isinstance(value, StringTypes):
  535.             try:
  536.                 node = self._attrs[attname]
  537.             except KeyError:
  538.                 node = Attr(attname)
  539.                 node.ownerDocument = self._ownerElement.ownerDocument
  540.                 self.setNamedItem(node)
  541.             node.value = value
  542.         else:
  543.             if not isinstance(value, Attr):
  544.                 raise TypeError, "value must be a string or Attr object"
  545.             node = value
  546.             self.setNamedItem(node)
  547.  
  548.     def getNamedItem(self, name):
  549.         try:
  550.             return self._attrs[name]
  551.         except KeyError:
  552.             return None
  553.  
  554.     def getNamedItemNS(self, namespaceURI, localName):
  555.         try:
  556.             return self._attrsNS[(namespaceURI, localName)]
  557.         except KeyError:
  558.             return None
  559.  
  560.     def removeNamedItem(self, name):
  561.         n = self.getNamedItem(name)
  562.         if n is not None:
  563.             _clear_id_cache(self._ownerElement)
  564.             del self._attrs[n.nodeName]
  565.             del self._attrsNS[(n.namespaceURI, n.localName)]
  566.             if n.__dict__.has_key('ownerElement'):
  567.                 n.__dict__['ownerElement'] = None
  568.             return n
  569.         else:
  570.             raise xml.dom.NotFoundErr()
  571.  
  572.     def removeNamedItemNS(self, namespaceURI, localName):
  573.         n = self.getNamedItemNS(namespaceURI, localName)
  574.         if n is not None:
  575.             _clear_id_cache(self._ownerElement)
  576.             del self._attrsNS[(n.namespaceURI, n.localName)]
  577.             del self._attrs[n.nodeName]
  578.             if n.__dict__.has_key('ownerElement'):
  579.                 n.__dict__['ownerElement'] = None
  580.             return n
  581.         else:
  582.             raise xml.dom.NotFoundErr()
  583.  
  584.     def setNamedItem(self, node):
  585.         if not isinstance(node, Attr):
  586.             raise xml.dom.HierarchyRequestErr(
  587.                 "%s cannot be child of %s" % (repr(node), repr(self)))
  588.         old = self._attrs.get(node.name)
  589.         if old:
  590.             old.unlink()
  591.         self._attrs[node.name] = node
  592.         self._attrsNS[(node.namespaceURI, node.localName)] = node
  593.         node.ownerElement = self._ownerElement
  594.         _clear_id_cache(node.ownerElement)
  595.         return old
  596.  
  597.     def setNamedItemNS(self, node):
  598.         return self.setNamedItem(node)
  599.  
  600.     def __delitem__(self, attname_or_tuple):
  601.         node = self[attname_or_tuple]
  602.         _clear_id_cache(node.ownerElement)
  603.         node.unlink()
  604.  
  605.     def __getstate__(self):
  606.         return self._attrs, self._attrsNS, self._ownerElement
  607.  
  608.     def __setstate__(self, state):
  609.         self._attrs, self._attrsNS, self._ownerElement = state
  610.  
  611. defproperty(NamedNodeMap, "length",
  612.             doc="Number of nodes in the NamedNodeMap.")
  613.  
  614. AttributeList = NamedNodeMap
  615.  
  616.  
  617. class TypeInfo(NewStyle):
  618.     __slots__ = 'namespace', 'name'
  619.  
  620.     def __init__(self, namespace, name):
  621.         self.namespace = namespace
  622.         self.name = name
  623.  
  624.     def __repr__(self):
  625.         if self.namespace:
  626.             return "<TypeInfo %s (from %s)>" % (`self.name`, `self.namespace`)
  627.         else:
  628.             return "<TypeInfo %s>" % `self.name`
  629.  
  630.     def _get_name(self):
  631.         return self.name
  632.  
  633.     def _get_namespace(self):
  634.         return self.namespace
  635.  
  636. _no_type = TypeInfo(None, None)
  637.  
  638. class Element(Node):
  639.     nodeType = Node.ELEMENT_NODE
  640.     nodeValue = None
  641.     schemaType = _no_type
  642.  
  643.     _magic_id_nodes = 0
  644.  
  645.     _child_node_types = (Node.ELEMENT_NODE,
  646.                          Node.PROCESSING_INSTRUCTION_NODE,
  647.                          Node.COMMENT_NODE,
  648.                          Node.TEXT_NODE,
  649.                          Node.CDATA_SECTION_NODE,
  650.                          Node.ENTITY_REFERENCE_NODE)
  651.  
  652.     def __init__(self, tagName, namespaceURI=EMPTY_NAMESPACE, prefix=None,
  653.                  localName=None):
  654.         self.tagName = self.nodeName = tagName
  655.         self.prefix = prefix
  656.         self.namespaceURI = namespaceURI
  657.         self.childNodes = NodeList()
  658.  
  659.         self._attrs = {}   # attributes are double-indexed:
  660.         self._attrsNS = {} #    tagName -> Attribute
  661.                            #    URI,localName -> Attribute
  662.                            # in the future: consider lazy generation
  663.                            # of attribute objects this is too tricky
  664.                            # for now because of headaches with
  665.                            # namespaces.
  666.  
  667.     def _get_localName(self):
  668.         return self.tagName.split(":", 1)[-1]
  669.  
  670.     def _get_tagName(self):
  671.         return self.tagName
  672.  
  673.     def unlink(self):
  674.         for attr in self._attrs.values():
  675.             attr.unlink()
  676.         self._attrs = None
  677.         self._attrsNS = None
  678.         Node.unlink(self)
  679.  
  680.     def getAttribute(self, attname):
  681.         try:
  682.             return self._attrs[attname].value
  683.         except KeyError:
  684.             return ""
  685.  
  686.     def getAttributeNS(self, namespaceURI, localName):
  687.         try:
  688.             return self._attrsNS[(namespaceURI, localName)].value
  689.         except KeyError:
  690.             return ""
  691.  
  692.     def setAttribute(self, attname, value):
  693.         attr = self.getAttributeNode(attname)
  694.         if attr is None:
  695.             attr = Attr(attname)
  696.             # for performance
  697.             d = attr.__dict__
  698.             d["value"] = d["nodeValue"] = value
  699.             d["ownerDocument"] = self.ownerDocument
  700.             self.setAttributeNode(attr)
  701.         elif value != attr.value:
  702.             d = attr.__dict__
  703.             d["value"] = d["nodeValue"] = value
  704.             if attr.isId:
  705.                 _clear_id_cache(self)
  706.  
  707.     def setAttributeNS(self, namespaceURI, qualifiedName, value):
  708.         prefix, localname = _nssplit(qualifiedName)
  709.         attr = self.getAttributeNodeNS(namespaceURI, localname)
  710.         if attr is None:
  711.             # for performance
  712.             attr = Attr(qualifiedName, namespaceURI, localname, prefix)
  713.             d = attr.__dict__
  714.             d["prefix"] = prefix
  715.             d["nodeName"] = qualifiedName
  716.             d["value"] = d["nodeValue"] = value
  717.             d["ownerDocument"] = self.ownerDocument
  718.             self.setAttributeNode(attr)
  719.         else:
  720.             d = attr.__dict__
  721.             if value != attr.value:
  722.                 d["value"] = d["nodeValue"] = value
  723.                 if attr.isId:
  724.                     _clear_id_cache(self)
  725.             if attr.prefix != prefix:
  726.                 d["prefix"] = prefix
  727.                 d["nodeName"] = qualifiedName
  728.  
  729.     def getAttributeNode(self, attrname):
  730.         return self._attrs.get(attrname)
  731.  
  732.     def getAttributeNodeNS(self, namespaceURI, localName):
  733.         return self._attrsNS.get((namespaceURI, localName))
  734.  
  735.     def setAttributeNode(self, attr):
  736.         if attr.ownerElement not in (None, self):
  737.             raise xml.dom.InuseAttributeErr("attribute node already owned")
  738.         old1 = self._attrs.get(attr.name, None)
  739.         if old1 is not None:
  740.             self.removeAttributeNode(old1)
  741.         old2 = self._attrsNS.get((attr.namespaceURI, attr.localName), None)
  742.         if old2 is not None and old2 is not old1:
  743.             self.removeAttributeNode(old2)
  744.         _set_attribute_node(self, attr)
  745.  
  746.         if old1 is not attr:
  747.             # It might have already been part of this node, in which case
  748.             # it doesn't represent a change, and should not be returned.
  749.             return old1
  750.         if old2 is not attr:
  751.             return old2
  752.  
  753.     setAttributeNodeNS = setAttributeNode
  754.  
  755.     def removeAttribute(self, name):
  756.         try:
  757.             attr = self._attrs[name]
  758.         except KeyError:
  759.             raise xml.dom.NotFoundErr()
  760.         self.removeAttributeNode(attr)
  761.  
  762.     def removeAttributeNS(self, namespaceURI, localName):
  763.         try:
  764.             attr = self._attrsNS[(namespaceURI, localName)]
  765.         except KeyError:
  766.             raise xml.dom.NotFoundErr()
  767.         self.removeAttributeNode(attr)
  768.  
  769.     def removeAttributeNode(self, node):
  770.         if node is None:
  771.             raise xml.dom.NotFoundErr()
  772.         try:
  773.             self._attrs[node.name]
  774.         except KeyError:
  775.             raise xml.dom.NotFoundErr()
  776.         _clear_id_cache(self)
  777.         node.unlink()
  778.         # Restore this since the node is still useful and otherwise
  779.         # unlinked
  780.         node.ownerDocument = self.ownerDocument
  781.  
  782.     removeAttributeNodeNS = removeAttributeNode
  783.  
  784.     def hasAttribute(self, name):
  785.         return self._attrs.has_key(name)
  786.  
  787.     def hasAttributeNS(self, namespaceURI, localName):
  788.         return self._attrsNS.has_key((namespaceURI, localName))
  789.  
  790.     def getElementsByTagName(self, name):
  791.         return _get_elements_by_tagName_helper(self, name, NodeList())
  792.  
  793.     def getElementsByTagNameNS(self, namespaceURI, localName):
  794.         return _get_elements_by_tagName_ns_helper(
  795.             self, namespaceURI, localName, NodeList())
  796.  
  797.     def __repr__(self):
  798.         # On some systems (RH10) id() can be a negative number. 
  799.         # work around this.
  800.         MAX = 2L*sys.maxint+1
  801.         return "<DOM Element: %s at %#x>" % (self.tagName, id(self)&MAX)
  802.  
  803.     def writexml(self, writer, indent="", addindent="", newl=""):
  804.         # indent = current indentation
  805.         # addindent = indentation to add to higher levels
  806.         # newl = newline string
  807.         writer.write(indent+"<" + self.tagName)
  808.  
  809.         attrs = self._get_attributes()
  810.         a_names = attrs.keys()
  811.         a_names.sort()
  812.  
  813.         for a_name in a_names:
  814.             writer.write(" %s=\"" % a_name)
  815.             _write_data(writer, attrs[a_name].value)
  816.             writer.write("\"")
  817.         if self.childNodes:
  818.             writer.write(">%s"%(newl))
  819.             for node in self.childNodes:
  820.                 node.writexml(writer,indent+addindent,addindent,newl)
  821.             writer.write("%s</%s>%s" % (indent,self.tagName,newl))
  822.         else:
  823.             writer.write("/>%s"%(newl))
  824.  
  825.     def _get_attributes(self):
  826.         return NamedNodeMap(self._attrs, self._attrsNS, self)
  827.  
  828.     def hasAttributes(self):
  829.         if self._attrs:
  830.             return True
  831.         else:
  832.             return False
  833.  
  834.     # DOM Level 3 attributes, based on the 22 Oct 2002 draft
  835.  
  836.     def setIdAttribute(self, name):
  837.         idAttr = self.getAttributeNode(name)
  838.         self.setIdAttributeNode(idAttr)
  839.  
  840.     def setIdAttributeNS(self, namespaceURI, localName):
  841.         idAttr = self.getAttributeNodeNS(namespaceURI, localName)
  842.         self.setIdAttributeNode(idAttr)
  843.  
  844.     def setIdAttributeNode(self, idAttr):
  845.         if idAttr is None or not self.isSameNode(idAttr.ownerElement):
  846.             raise xml.dom.NotFoundErr()
  847.         if _get_containing_entref(self) is not None:
  848.             raise xml.dom.NoModificationAllowedErr()
  849.         if not idAttr._is_id:
  850.             idAttr.__dict__['_is_id'] = True
  851.             self._magic_id_nodes += 1
  852.             self.ownerDocument._magic_id_count += 1
  853.             _clear_id_cache(self)
  854.  
  855. defproperty(Element, "attributes",
  856.             doc="NamedNodeMap of attributes on the element.")
  857. defproperty(Element, "localName",
  858.             doc="Namespace-local name of this element.")
  859.  
  860.  
  861. def _set_attribute_node(element, attr):
  862.     _clear_id_cache(element)
  863.     element._attrs[attr.name] = attr
  864.     element._attrsNS[(attr.namespaceURI, attr.localName)] = attr
  865.  
  866.     # This creates a circular reference, but Element.unlink()
  867.     # breaks the cycle since the references to the attribute
  868.     # dictionaries are tossed.
  869.     attr.__dict__['ownerElement'] = element
  870.  
  871.  
  872. class Childless:
  873.     """Mixin that makes childless-ness easy to implement and avoids
  874.     the complexity of the Node methods that deal with children.
  875.     """
  876.  
  877.     attributes = None
  878.     childNodes = EmptyNodeList()
  879.     firstChild = None
  880.     lastChild = None
  881.  
  882.     def _get_firstChild(self):
  883.         return None
  884.  
  885.     def _get_lastChild(self):
  886.         return None
  887.  
  888.     def appendChild(self, node):
  889.         raise xml.dom.HierarchyRequestErr(
  890.             self.nodeName + " nodes cannot have children")
  891.  
  892.     def hasChildNodes(self):
  893.         return False
  894.  
  895.     def insertBefore(self, newChild, refChild):
  896.         raise xml.dom.HierarchyRequestErr(
  897.             self.nodeName + " nodes do not have children")
  898.  
  899.     def removeChild(self, oldChild):
  900.         raise xml.dom.NotFoundErr(
  901.             self.nodeName + " nodes do not have children")
  902.  
  903.     def replaceChild(self, newChild, oldChild):
  904.         raise xml.dom.HierarchyRequestErr(
  905.             self.nodeName + " nodes do not have children")
  906.  
  907.  
  908. class ProcessingInstruction(Childless, Node):
  909.     nodeType = Node.PROCESSING_INSTRUCTION_NODE
  910.  
  911.     def __init__(self, target, data):
  912.         self.target = self.nodeName = target
  913.         self.data = self.nodeValue = data
  914.  
  915.     def _get_data(self):
  916.         return self.data
  917.     def _set_data(self, value):
  918.         d = self.__dict__
  919.         d['data'] = d['nodeValue'] = value
  920.  
  921.     def _get_target(self):
  922.         return self.target
  923.     def _set_target(self, value):
  924.         d = self.__dict__
  925.         d['target'] = d['nodeName'] = value
  926.  
  927.     def __setattr__(self, name, value):
  928.         if name == "data" or name == "nodeValue":
  929.             self.__dict__['data'] = self.__dict__['nodeValue'] = value
  930.         elif name == "target" or name == "nodeName":
  931.             self.__dict__['target'] = self.__dict__['nodeName'] = value
  932.         else:
  933.             self.__dict__[name] = value
  934.  
  935.     def writexml(self, writer, indent="", addindent="", newl=""):
  936.         writer.write("%s<?%s %s?>%s" % (indent,self.target, self.data, newl))
  937.  
  938.  
  939. class CharacterData(Childless, Node):
  940.     def _get_length(self):
  941.         return len(self.data)
  942.     __len__ = _get_length
  943.  
  944.     def _get_data(self):
  945.         return self.__dict__['data']
  946.     def _set_data(self, data):
  947.         d = self.__dict__
  948.         d['data'] = d['nodeValue'] = data
  949.  
  950.     _get_nodeValue = _get_data
  951.     _set_nodeValue = _set_data
  952.  
  953.     def __setattr__(self, name, value):
  954.         if name == "data" or name == "nodeValue":
  955.             self.__dict__['data'] = self.__dict__['nodeValue'] = value
  956.         else:
  957.             self.__dict__[name] = value
  958.  
  959.     def __repr__(self):
  960.         data = self.data
  961.         if len(data) > 10:
  962.             dotdotdot = "..."
  963.         else:
  964.             dotdotdot = ""
  965.         return "<DOM %s node \"%s%s\">" % (
  966.             self.__class__.__name__, data[0:10], dotdotdot)
  967.  
  968.     def substringData(self, offset, count):
  969.         if offset < 0:
  970.             raise xml.dom.IndexSizeErr("offset cannot be negative")
  971.         if offset >= len(self.data):
  972.             raise xml.dom.IndexSizeErr("offset cannot be beyond end of data")
  973.         if count < 0:
  974.             raise xml.dom.IndexSizeErr("count cannot be negative")
  975.         return self.data[offset:offset+count]
  976.  
  977.     def appendData(self, arg):
  978.         self.data = self.data + arg
  979.  
  980.     def insertData(self, offset, arg):
  981.         if offset < 0:
  982.             raise xml.dom.IndexSizeErr("offset cannot be negative")
  983.         if offset >= len(self.data):
  984.             raise xml.dom.IndexSizeErr("offset cannot be beyond end of data")
  985.         if arg:
  986.             self.data = "%s%s%s" % (
  987.                 self.data[:offset], arg, self.data[offset:])
  988.  
  989.     def deleteData(self, offset, count):
  990.         if offset < 0:
  991.             raise xml.dom.IndexSizeErr("offset cannot be negative")
  992.         if offset >= len(self.data):
  993.             raise xml.dom.IndexSizeErr("offset cannot be beyond end of data")
  994.         if count < 0:
  995.             raise xml.dom.IndexSizeErr("count cannot be negative")
  996.         if count:
  997.             self.data = self.data[:offset] + self.data[offset+count:]
  998.  
  999.     def replaceData(self, offset, count, arg):
  1000.         if offset < 0:
  1001.             raise xml.dom.IndexSizeErr("offset cannot be negative")
  1002.         if offset >= len(self.data):
  1003.             raise xml.dom.IndexSizeErr("offset cannot be beyond end of data")
  1004.         if count < 0:
  1005.             raise xml.dom.IndexSizeErr("count cannot be negative")
  1006.         if count:
  1007.             self.data = "%s%s%s" % (
  1008.                 self.data[:offset], arg, self.data[offset+count:])
  1009.  
  1010. defproperty(CharacterData, "length", doc="Length of the string data.")
  1011.  
  1012.  
  1013. class Text(CharacterData):
  1014.     # Make sure we don't add an instance __dict__ if we don't already
  1015.     # have one, at least when that's possible:
  1016.     # XXX this does not work, CharacterData is an old-style class
  1017.     # __slots__ = ()
  1018.  
  1019.     nodeType = Node.TEXT_NODE
  1020.     nodeName = "#text"
  1021.     attributes = None
  1022.  
  1023.     def splitText(self, offset):
  1024.         if offset < 0 or offset > len(self.data):
  1025.             raise xml.dom.IndexSizeErr("illegal offset value")
  1026.         newText = self.__class__()
  1027.         newText.data = self.data[offset:]
  1028.         newText.ownerDocument = self.ownerDocument
  1029.         next = self.nextSibling
  1030.         if self.parentNode and self in self.parentNode.childNodes:
  1031.             if next is None:
  1032.                 self.parentNode.appendChild(newText)
  1033.             else:
  1034.                 self.parentNode.insertBefore(newText, next)
  1035.         self.data = self.data[:offset]
  1036.         return newText
  1037.  
  1038.     def writexml(self, writer, indent="", addindent="", newl=""):
  1039.         _write_data(writer, "%s%s%s"%(indent, self.data, newl))
  1040.  
  1041.     # DOM Level 3 (WD 9 April 2002)
  1042.  
  1043.     def _get_wholeText(self):
  1044.         L = [self.data]
  1045.         n = self.previousSibling
  1046.         while n is not None:
  1047.             if n.nodeType in (Node.TEXT_NODE, Node.CDATA_SECTION_NODE):
  1048.                 L.insert(0, n.data)
  1049.                 n = n.previousSibling
  1050.             else:
  1051.                 break
  1052.         n = self.nextSibling
  1053.         while n is not None:
  1054.             if n.nodeType in (Node.TEXT_NODE, Node.CDATA_SECTION_NODE):
  1055.                 L.append(n.data)
  1056.                 n = n.nextSibling
  1057.             else:
  1058.                 break
  1059.         return ''.join(L)
  1060.  
  1061.     def replaceWholeText(self, content):
  1062.         # XXX This needs to be seriously changed if minidom ever
  1063.         # supports EntityReference nodes.
  1064.         parent = self.parentNode
  1065.         n = self.previousSibling
  1066.         while n is not None:
  1067.             if n.nodeType in (Node.TEXT_NODE, Node.CDATA_SECTION_NODE):
  1068.                 next = n.previousSibling
  1069.                 parent.removeChild(n)
  1070.                 n = next
  1071.             else:
  1072.                 break
  1073.         n = self.nextSibling
  1074.         if not content:
  1075.             parent.removeChild(self)
  1076.         while n is not None:
  1077.             if n.nodeType in (Node.TEXT_NODE, Node.CDATA_SECTION_NODE):
  1078.                 next = n.nextSibling
  1079.                 parent.removeChild(n)
  1080.                 n = next
  1081.             else:
  1082.                 break
  1083.         if content:
  1084.             d = self.__dict__
  1085.             d['data'] = content
  1086.             d['nodeValue'] = content
  1087.             return self
  1088.         else:
  1089.             return None
  1090.  
  1091.     def _get_isWhitespaceInElementContent(self):
  1092.         if self.data.strip():
  1093.             return False
  1094.         elem = _get_containing_element(self)
  1095.         if elem is None:
  1096.             return False
  1097.         info = self.ownerDocument._get_elem_info(elem)
  1098.         if info is None:
  1099.             return False
  1100.         else:
  1101.             return info.isElementContent()
  1102.  
  1103. defproperty(Text, "isWhitespaceInElementContent",
  1104.             doc="True iff this text node contains only whitespace"
  1105.                 " and is in element content.")
  1106. defproperty(Text, "wholeText",
  1107.             doc="The text of all logically-adjacent text nodes.")
  1108.  
  1109.  
  1110. def _get_containing_element(node):
  1111.     c = node.parentNode
  1112.     while c is not None:
  1113.         if c.nodeType == Node.ELEMENT_NODE:
  1114.             return c
  1115.         c = c.parentNode
  1116.     return None
  1117.  
  1118. def _get_containing_entref(node):
  1119.     c = node.parentNode
  1120.     while c is not None:
  1121.         if c.nodeType == Node.ENTITY_REFERENCE_NODE:
  1122.             return c
  1123.         c = c.parentNode
  1124.     return None
  1125.  
  1126.  
  1127. class Comment(Childless, CharacterData):
  1128.     nodeType = Node.COMMENT_NODE
  1129.     nodeName = "#comment"
  1130.  
  1131.     def __init__(self, data):
  1132.         self.data = self.nodeValue = data
  1133.  
  1134.     def writexml(self, writer, indent="", addindent="", newl=""):
  1135.         writer.write("%s<!--%s-->%s" % (indent, self.data, newl))
  1136.  
  1137.  
  1138. class CDATASection(Text):
  1139.     # Make sure we don't add an instance __dict__ if we don't already
  1140.     # have one, at least when that's possible:
  1141.     # XXX this does not work, Text is an old-style class
  1142.     # __slots__ = ()
  1143.  
  1144.     nodeType = Node.CDATA_SECTION_NODE
  1145.     nodeName = "#cdata-section"
  1146.  
  1147.     def writexml(self, writer, indent="", addindent="", newl=""):
  1148.         if self.data.find("]]>") >= 0:
  1149.             raise ValueError("']]>' not allowed in a CDATA section")
  1150.         writer.write("<![CDATA[%s]]>" % self.data)
  1151.  
  1152.  
  1153. class ReadOnlySequentialNamedNodeMap(NewStyle, GetattrMagic):
  1154.     __slots__ = '_seq',
  1155.  
  1156.     def __init__(self, seq=()):
  1157.         # seq should be a list or tuple
  1158.         self._seq = seq
  1159.  
  1160.     def __len__(self):
  1161.         return len(self._seq)
  1162.  
  1163.     def _get_length(self):
  1164.         return len(self._seq)
  1165.  
  1166.     def getNamedItem(self, name):
  1167.         for n in self._seq:
  1168.             if n.nodeName == name:
  1169.                 return n
  1170.  
  1171.     def getNamedItemNS(self, namespaceURI, localName):
  1172.         for n in self._seq:
  1173.             if n.namespaceURI == namespaceURI and n.localName == localName:
  1174.                 return n
  1175.  
  1176.     def __getitem__(self, name_or_tuple):
  1177.         if isinstance(name_or_tuple, _TupleType):
  1178.             node = self.getNamedItemNS(*name_or_tuple)
  1179.         else:
  1180.             node = self.getNamedItem(name_or_tuple)
  1181.         if node is None:
  1182.             raise KeyError, name_or_tuple
  1183.         return node
  1184.  
  1185.     def item(self, index):
  1186.         if index < 0:
  1187.             return None
  1188.         try:
  1189.             return self._seq[index]
  1190.         except IndexError:
  1191.             return None
  1192.  
  1193.     def removeNamedItem(self, name):
  1194.         raise xml.dom.NoModificationAllowedErr(
  1195.             "NamedNodeMap instance is read-only")
  1196.  
  1197.     def removeNamedItemNS(self, namespaceURI, localName):
  1198.         raise xml.dom.NoModificationAllowedErr(
  1199.             "NamedNodeMap instance is read-only")
  1200.  
  1201.     def setNamedItem(self, node):
  1202.         raise xml.dom.NoModificationAllowedErr(
  1203.             "NamedNodeMap instance is read-only")
  1204.  
  1205.     def setNamedItemNS(self, node):
  1206.         raise xml.dom.NoModificationAllowedErr(
  1207.             "NamedNodeMap instance is read-only")
  1208.  
  1209.     def __getstate__(self):
  1210.         return [self._seq]
  1211.  
  1212.     def __setstate__(self, state):
  1213.         self._seq = state[0]
  1214.  
  1215. defproperty(ReadOnlySequentialNamedNodeMap, "length",
  1216.             doc="Number of entries in the NamedNodeMap.")
  1217.  
  1218.  
  1219. class Identified:
  1220.     """Mix-in class that supports the publicId and systemId attributes."""
  1221.  
  1222.     # XXX this does not work, this is an old-style class
  1223.     # __slots__ = 'publicId', 'systemId'
  1224.  
  1225.     def _identified_mixin_init(self, publicId, systemId):
  1226.         self.publicId = publicId
  1227.         self.systemId = systemId
  1228.  
  1229.     def _get_publicId(self):
  1230.         return self.publicId
  1231.  
  1232.     def _get_systemId(self):
  1233.         return self.systemId
  1234.  
  1235. class DocumentType(Identified, Childless, Node):
  1236.     nodeType = Node.DOCUMENT_TYPE_NODE
  1237.     nodeValue = None
  1238.     name = None
  1239.     publicId = None
  1240.     systemId = None
  1241.     internalSubset = None
  1242.  
  1243.     def __init__(self, qualifiedName):
  1244.         self.entities = ReadOnlySequentialNamedNodeMap()
  1245.         self.notations = ReadOnlySequentialNamedNodeMap()
  1246.         if qualifiedName:
  1247.             prefix, localname = _nssplit(qualifiedName)
  1248.             self.name = localname
  1249.         self.nodeName = self.name
  1250.  
  1251.     def _get_internalSubset(self):
  1252.         return self.internalSubset
  1253.  
  1254.     def cloneNode(self, deep):
  1255.         if self.ownerDocument is None:
  1256.             # it's ok
  1257.             clone = DocumentType(None)
  1258.             clone.name = self.name
  1259.             clone.nodeName = self.name
  1260.             operation = xml.dom.UserDataHandler.NODE_CLONED
  1261.             if deep:
  1262.                 clone.entities._seq = []
  1263.                 clone.notations._seq = []
  1264.                 for n in self.notations._seq:
  1265.                     notation = Notation(n.nodeName, n.publicId, n.systemId)
  1266.                     clone.notations._seq.append(notation)
  1267.                     n._call_user_data_handler(operation, n, notation)
  1268.                 for e in self.entities._seq:
  1269.                     entity = Entity(e.nodeName, e.publicId, e.systemId,
  1270.                                     e.notationName)
  1271.                     entity.actualEncoding = e.actualEncoding
  1272.                     entity.encoding = e.encoding
  1273.                     entity.version = e.version
  1274.                     clone.entities._seq.append(entity)
  1275.                     e._call_user_data_handler(operation, n, entity)
  1276.             self._call_user_data_handler(operation, self, clone)
  1277.             return clone
  1278.         else:
  1279.             return None
  1280.  
  1281.     def writexml(self, writer, indent="", addindent="", newl=""):
  1282.         writer.write("<!DOCTYPE ")
  1283.         writer.write(self.name)
  1284.         if self.publicId:
  1285.             writer.write("\n  PUBLIC '%s'\n  '%s'"
  1286.                          % (self.publicId, self.systemId))
  1287.         elif self.systemId:
  1288.             writer.write("\n  SYSTEM '%s'" % self.systemId)
  1289.         if self.internalSubset is not None:
  1290.             writer.write(" [")
  1291.             writer.write(self.internalSubset)
  1292.             writer.write("]")
  1293.         writer.write(">\n")
  1294.  
  1295. class Entity(Identified, Node):
  1296.     attributes = None
  1297.     nodeType = Node.ENTITY_NODE
  1298.     nodeValue = None
  1299.  
  1300.     actualEncoding = None
  1301.     encoding = None
  1302.     version = None
  1303.  
  1304.     def __init__(self, name, publicId, systemId, notation):
  1305.         self.nodeName = name
  1306.         self.notationName = notation
  1307.         self.childNodes = NodeList()
  1308.         self._identified_mixin_init(publicId, systemId)
  1309.  
  1310.     def _get_actualEncoding(self):
  1311.         return self.actualEncoding
  1312.  
  1313.     def _get_encoding(self):
  1314.         return self.encoding
  1315.  
  1316.     def _get_version(self):
  1317.         return self.version
  1318.  
  1319.     def appendChild(self, newChild):
  1320.         raise xml.dom.HierarchyRequestErr(
  1321.             "cannot append children to an entity node")
  1322.  
  1323.     def insertBefore(self, newChild, refChild):
  1324.         raise xml.dom.HierarchyRequestErr(
  1325.             "cannot insert children below an entity node")
  1326.  
  1327.     def removeChild(self, oldChild):
  1328.         raise xml.dom.HierarchyRequestErr(
  1329.             "cannot remove children from an entity node")
  1330.  
  1331.     def replaceChild(self, newChild, oldChild):
  1332.         raise xml.dom.HierarchyRequestErr(
  1333.             "cannot replace children of an entity node")
  1334.  
  1335. class Notation(Identified, Childless, Node):
  1336.     nodeType = Node.NOTATION_NODE
  1337.     nodeValue = None
  1338.  
  1339.     def __init__(self, name, publicId, systemId):
  1340.         self.nodeName = name
  1341.         self._identified_mixin_init(publicId, systemId)
  1342.  
  1343.  
  1344. class DOMImplementation(DOMImplementationLS):
  1345.     _features = [("core", "1.0"),
  1346.                  ("core", "2.0"),
  1347.                  ("core", "3.0"),
  1348.                  ("core", None),
  1349.                  ("xml", "1.0"),
  1350.                  ("xml", "2.0"),
  1351.                  ("xml", "3.0"),
  1352.                  ("xml", None),
  1353.                  ("ls-load", "3.0"),
  1354.                  ("ls-load", None),
  1355.                  ]
  1356.  
  1357.     def hasFeature(self, feature, version):
  1358.         if version == "":
  1359.             version = None
  1360.         return (feature.lower(), version) in self._features
  1361.  
  1362.     def createDocument(self, namespaceURI, qualifiedName, doctype):
  1363.         if doctype and doctype.parentNode is not None:
  1364.             raise xml.dom.WrongDocumentErr(
  1365.                 "doctype object owned by another DOM tree")
  1366.         doc = self._create_document()
  1367.  
  1368.         add_root_element = not (namespaceURI is None
  1369.                                 and qualifiedName is None
  1370.                                 and doctype is None)
  1371.  
  1372.         if not qualifiedName and add_root_element:
  1373.             # The spec is unclear what to raise here; SyntaxErr
  1374.             # would be the other obvious candidate. Since Xerces raises
  1375.             # InvalidCharacterErr, and since SyntaxErr is not listed
  1376.             # for createDocument, that seems to be the better choice.
  1377.             # XXX: need to check for illegal characters here and in
  1378.             # createElement.
  1379.  
  1380.             # DOM Level III clears this up when talking about the return value
  1381.             # of this function.  If namespaceURI, qName and DocType are
  1382.             # Null the document is returned without a document element
  1383.             # Otherwise if doctype or namespaceURI are not None
  1384.             # Then we go back to the above problem
  1385.             raise xml.dom.InvalidCharacterErr("Element with no name")
  1386.  
  1387.         if add_root_element:
  1388.             prefix, localname = _nssplit(qualifiedName)
  1389.             if prefix == "xml" \
  1390.                and namespaceURI != "http://www.w3.org/XML/1998/namespace":
  1391.                 raise xml.dom.NamespaceErr("illegal use of 'xml' prefix")
  1392.             if prefix and not namespaceURI:
  1393.                 raise xml.dom.NamespaceErr(
  1394.                     "illegal use of prefix without namespaces")
  1395.             element = doc.createElementNS(namespaceURI, qualifiedName)
  1396.             if doctype:
  1397.                 doc.appendChild(doctype)
  1398.             doc.appendChild(element)
  1399.  
  1400.         if doctype:
  1401.             doctype.parentNode = doctype.ownerDocument = doc
  1402.  
  1403.         doc.doctype = doctype
  1404.         doc.implementation = self
  1405.         return doc
  1406.  
  1407.     def createDocumentType(self, qualifiedName, publicId, systemId):
  1408.         doctype = DocumentType(qualifiedName)
  1409.         doctype.publicId = publicId
  1410.         doctype.systemId = systemId
  1411.         return doctype
  1412.  
  1413.     # DOM Level 3 (WD 9 April 2002)
  1414.  
  1415.     def getInterface(self, feature):
  1416.         if self.hasFeature(feature, None):
  1417.             return self
  1418.         else:
  1419.             return None
  1420.  
  1421.     # internal
  1422.     def _create_document(self):
  1423.         return Document()
  1424.  
  1425. class ElementInfo(NewStyle):
  1426.     """Object that represents content-model information for an element.
  1427.  
  1428.     This implementation is not expected to be used in practice; DOM
  1429.     builders should provide implementations which do the right thing
  1430.     using information available to it.
  1431.  
  1432.     """
  1433.  
  1434.     __slots__ = 'tagName',
  1435.  
  1436.     def __init__(self, name):
  1437.         self.tagName = name
  1438.  
  1439.     def getAttributeType(self, aname):
  1440.         return _no_type
  1441.  
  1442.     def getAttributeTypeNS(self, namespaceURI, localName):
  1443.         return _no_type
  1444.  
  1445.     def isElementContent(self):
  1446.         return False
  1447.  
  1448.     def isEmpty(self):
  1449.         """Returns true iff this element is declared to have an EMPTY
  1450.         content model."""
  1451.         return False
  1452.  
  1453.     def isId(self, aname):
  1454.         """Returns true iff the named attribte is a DTD-style ID."""
  1455.         return False
  1456.  
  1457.     def isIdNS(self, namespaceURI, localName):
  1458.         """Returns true iff the identified attribute is a DTD-style ID."""
  1459.         return False
  1460.  
  1461.     def __getstate__(self):
  1462.         return self.tagName
  1463.  
  1464.     def __setstate__(self, state):
  1465.         self.tagName = state
  1466.  
  1467. def _clear_id_cache(node):
  1468.     if node.nodeType == Node.DOCUMENT_NODE:
  1469.         node._id_cache.clear()
  1470.         node._id_search_stack = None
  1471.     elif _in_document(node):
  1472.         node.ownerDocument._id_cache.clear()
  1473.         node.ownerDocument._id_search_stack= None
  1474.  
  1475. class Document(Node, DocumentLS):
  1476.     _child_node_types = (Node.ELEMENT_NODE, Node.PROCESSING_INSTRUCTION_NODE,
  1477.                          Node.COMMENT_NODE, Node.DOCUMENT_TYPE_NODE)
  1478.  
  1479.     nodeType = Node.DOCUMENT_NODE
  1480.     nodeName = "#document"
  1481.     nodeValue = None
  1482.     attributes = None
  1483.     doctype = None
  1484.     parentNode = None
  1485.     previousSibling = nextSibling = None
  1486.  
  1487.     implementation = DOMImplementation()
  1488.  
  1489.     # Document attributes from Level 3 (WD 9 April 2002)
  1490.  
  1491.     actualEncoding = None
  1492.     encoding = None
  1493.     standalone = None
  1494.     version = None
  1495.     strictErrorChecking = False
  1496.     errorHandler = None
  1497.     documentURI = None
  1498.  
  1499.     _magic_id_count = 0
  1500.  
  1501.     def __init__(self):
  1502.         self.childNodes = NodeList()
  1503.         # mapping of (namespaceURI, localName) -> ElementInfo
  1504.         #        and tagName -> ElementInfo
  1505.         self._elem_info = {}
  1506.         self._id_cache = {}
  1507.         self._id_search_stack = None
  1508.  
  1509.     def _get_elem_info(self, element):
  1510.         if element.namespaceURI:
  1511.             key = element.namespaceURI, element.localName
  1512.         else:
  1513.             key = element.tagName
  1514.         return self._elem_info.get(key)
  1515.  
  1516.     def _get_actualEncoding(self):
  1517.         return self.actualEncoding
  1518.  
  1519.     def _get_doctype(self):
  1520.         return self.doctype
  1521.  
  1522.     def _get_documentURI(self):
  1523.         return self.documentURI
  1524.  
  1525.     def _get_encoding(self):
  1526.         return self.encoding
  1527.  
  1528.     def _get_errorHandler(self):
  1529.         return self.errorHandler
  1530.  
  1531.     def _get_standalone(self):
  1532.         return self.standalone
  1533.  
  1534.     def _get_strictErrorChecking(self):
  1535.         return self.strictErrorChecking
  1536.  
  1537.     def _get_version(self):
  1538.         return self.version
  1539.  
  1540.     def appendChild(self, node):
  1541.         if node.nodeType not in self._child_node_types:
  1542.             raise xml.dom.HierarchyRequestErr(
  1543.                 "%s cannot be child of %s" % (repr(node), repr(self)))
  1544.         if node.parentNode is not None:
  1545.             # This needs to be done before the next test since this
  1546.             # may *be* the document element, in which case it should
  1547.             # end up re-ordered to the end.
  1548.             node.parentNode.removeChild(node)
  1549.  
  1550.         if node.nodeType == Node.ELEMENT_NODE \
  1551.            and self._get_documentElement():
  1552.             raise xml.dom.HierarchyRequestErr(
  1553.                 "two document elements disallowed")
  1554.         return Node.appendChild(self, node)
  1555.  
  1556.     def removeChild(self, oldChild):
  1557.         try:
  1558.             self.childNodes.remove(oldChild)
  1559.         except ValueError:
  1560.             raise xml.dom.NotFoundErr()
  1561.         oldChild.nextSibling = oldChild.previousSibling = None
  1562.         oldChild.parentNode = None
  1563.         if self.documentElement is oldChild:
  1564.             self.documentElement = None
  1565.  
  1566.         return oldChild
  1567.  
  1568.     def _get_documentElement(self):
  1569.         for node in self.childNodes:
  1570.             if node.nodeType == Node.ELEMENT_NODE:
  1571.                 return node
  1572.  
  1573.     def unlink(self):
  1574.         if self.doctype is not None:
  1575.             self.doctype.unlink()
  1576.             self.doctype = None
  1577.         Node.unlink(self)
  1578.  
  1579.     def cloneNode(self, deep):
  1580.         if not deep:
  1581.             return None
  1582.         clone = self.implementation.createDocument(None, None, None)
  1583.         clone.encoding = self.encoding
  1584.         clone.standalone = self.standalone
  1585.         clone.version = self.version
  1586.         for n in self.childNodes:
  1587.             childclone = _clone_node(n, deep, clone)
  1588.             assert childclone.ownerDocument.isSameNode(clone)
  1589.             clone.childNodes.append(childclone)
  1590.             if childclone.nodeType == Node.DOCUMENT_NODE:
  1591.                 assert clone.documentElement is None
  1592.             elif childclone.nodeType == Node.DOCUMENT_TYPE_NODE:
  1593.                 assert clone.doctype is None
  1594.                 clone.doctype = childclone
  1595.             childclone.parentNode = clone
  1596.         self._call_user_data_handler(xml.dom.UserDataHandler.NODE_CLONED,
  1597.                                      self, clone)
  1598.         return clone
  1599.  
  1600.     def createDocumentFragment(self):
  1601.         d = DocumentFragment()
  1602.         d.ownerDocument = self
  1603.         return d
  1604.  
  1605.     def createElement(self, tagName):
  1606.         e = Element(tagName)
  1607.         e.ownerDocument = self
  1608.         return e
  1609.  
  1610.     def createTextNode(self, data):
  1611.         if not isinstance(data, StringTypes):
  1612.             raise TypeError, "node contents must be a string"
  1613.         t = Text()
  1614.         t.data = data
  1615.         t.ownerDocument = self
  1616.         return t
  1617.  
  1618.     def createCDATASection(self, data):
  1619.         if not isinstance(data, StringTypes):
  1620.             raise TypeError, "node contents must be a string"
  1621.         c = CDATASection()
  1622.         c.data = data
  1623.         c.ownerDocument = self
  1624.         return c
  1625.  
  1626.     def createComment(self, data):
  1627.         c = Comment(data)
  1628.         c.ownerDocument = self
  1629.         return c
  1630.  
  1631.     def createProcessingInstruction(self, target, data):
  1632.         p = ProcessingInstruction(target, data)
  1633.         p.ownerDocument = self
  1634.         return p
  1635.  
  1636.     def createAttribute(self, qName):
  1637.         a = Attr(qName)
  1638.         a.ownerDocument = self
  1639.         a.value = ""
  1640.         return a
  1641.  
  1642.     def createElementNS(self, namespaceURI, qualifiedName):
  1643.         prefix, localName = _nssplit(qualifiedName)
  1644.         e = Element(qualifiedName, namespaceURI, prefix)
  1645.         e.ownerDocument = self
  1646.         return e
  1647.  
  1648.     def createAttributeNS(self, namespaceURI, qualifiedName):
  1649.         prefix, localName = _nssplit(qualifiedName)
  1650.         a = Attr(qualifiedName, namespaceURI, localName, prefix)
  1651.         a.ownerDocument = self
  1652.         a.value = ""
  1653.         return a
  1654.  
  1655.     # A couple of implementation-specific helpers to create node types
  1656.     # not supported by the W3C DOM specs:
  1657.  
  1658.     def _create_entity(self, name, publicId, systemId, notationName):
  1659.         e = Entity(name, publicId, systemId, notationName)
  1660.         e.ownerDocument = self
  1661.         return e
  1662.  
  1663.     def _create_notation(self, name, publicId, systemId):
  1664.         n = Notation(name, publicId, systemId)
  1665.         n.ownerDocument = self
  1666.         return n
  1667.  
  1668.     def getElementById(self, id):
  1669.         if self._id_cache.has_key(id):
  1670.             return self._id_cache[id]
  1671.         if not (self._elem_info or self._magic_id_count):
  1672.             return None
  1673.  
  1674.         stack = self._id_search_stack
  1675.         if stack is None:
  1676.             # we never searched before, or the cache has been cleared
  1677.             stack = [self.documentElement]
  1678.             self._id_search_stack = stack
  1679.         elif not stack:
  1680.             # Previous search was completed and cache is still valid;
  1681.             # no matching node.
  1682.             return None
  1683.  
  1684.         result = None
  1685.         while stack:
  1686.             node = stack.pop()
  1687.             # add child elements to stack for continued searching
  1688.             stack.extend([child for child in node.childNodes
  1689.                           if child.nodeType in _nodeTypes_with_children])
  1690.             # check this node
  1691.             info = self._get_elem_info(node)
  1692.             if info:
  1693.                 # We have to process all ID attributes before
  1694.                 # returning in order to get all the attributes set to
  1695.                 # be IDs using Element.setIdAttribute*().
  1696.                 for attr in node.attributes.values():
  1697.                     if attr.namespaceURI:
  1698.                         if info.isIdNS(attr.namespaceURI, attr.localName):
  1699.                             self._id_cache[attr.value] = node
  1700.                             if attr.value == id:
  1701.                                 result = node
  1702.                             elif not node._magic_id_nodes:
  1703.                                 break
  1704.                     elif info.isId(attr.name):
  1705.                         self._id_cache[attr.value] = node
  1706.                         if attr.value == id:
  1707.                             result = node
  1708.                         elif not node._magic_id_nodes:
  1709.                             break
  1710.                     elif attr._is_id:
  1711.                         self._id_cache[attr.value] = node
  1712.                         if attr.value == id:
  1713.                             result = node
  1714.                         elif node._magic_id_nodes == 1:
  1715.                             break
  1716.             elif node._magic_id_nodes:
  1717.                 for attr in node.attributes.values():
  1718.                     if attr._is_id:
  1719.                         self._id_cache[attr.value] = node
  1720.                         if attr.value == id:
  1721.                             result = node
  1722.             if result is not None:
  1723.                 break
  1724.         return result
  1725.  
  1726.     def getElementsByTagName(self, name):
  1727.         return _get_elements_by_tagName_helper(self, name, NodeList())
  1728.  
  1729.     def getElementsByTagNameNS(self, namespaceURI, localName):
  1730.         return _get_elements_by_tagName_ns_helper(
  1731.             self, namespaceURI, localName, NodeList())
  1732.  
  1733.     def isSupported(self, feature, version):
  1734.         return self.implementation.hasFeature(feature, version)
  1735.  
  1736.     def importNode(self, node, deep):
  1737.         if node.nodeType == Node.DOCUMENT_NODE:
  1738.             raise xml.dom.NotSupportedErr("cannot import document nodes")
  1739.         elif node.nodeType == Node.DOCUMENT_TYPE_NODE:
  1740.             raise xml.dom.NotSupportedErr("cannot import document type nodes")
  1741.         return _clone_node(node, deep, self)
  1742.  
  1743.     def writexml(self, writer, indent="", addindent="", newl="",
  1744.                  encoding = None):
  1745.         if encoding is None:
  1746.             writer.write('<?xml version="1.0" ?>\n')
  1747.         else:
  1748.             writer.write('<?xml version="1.0" encoding="%s"?>\n' % encoding)
  1749.         for node in self.childNodes:
  1750.             node.writexml(writer, indent, addindent, newl)
  1751.  
  1752.     # DOM Level 3 (WD 9 April 2002)
  1753.  
  1754.     def renameNode(self, n, namespaceURI, name):
  1755.         if n.ownerDocument is not self:
  1756.             raise xml.dom.WrongDocumentErr(
  1757.                 "cannot rename nodes from other documents;\n"
  1758.                 "expected %s,\nfound %s" % (self, n.ownerDocument))
  1759.         if n.nodeType not in (Node.ELEMENT_NODE, Node.ATTRIBUTE_NODE):
  1760.             raise xml.dom.NotSupportedErr(
  1761.                 "renameNode() only applies to element and attribute nodes")
  1762.         if namespaceURI != EMPTY_NAMESPACE:
  1763.             if ':' in name:
  1764.                 prefix, localName = name.split(':', 1)
  1765.                 if (  prefix == "xmlns"
  1766.                       and namespaceURI != xml.dom.XMLNS_NAMESPACE):
  1767.                     raise xml.dom.NamespaceErr(
  1768.                         "illegal use of 'xmlns' prefix")
  1769.             else:
  1770.                 if (  name == "xmlns"
  1771.                       and namespaceURI != xml.dom.XMLNS_NAMESPACE
  1772.                       and n.nodeType == Node.ATTRIBUTE_NODE):
  1773.                     raise xml.dom.NamespaceErr(
  1774.                         "illegal use of the 'xmlns' attribute")
  1775.                 prefix = None
  1776.                 localName = name
  1777.         else:
  1778.             prefix = None
  1779.             localName = None
  1780.         if n.nodeType == Node.ATTRIBUTE_NODE:
  1781.             element = n.ownerElement
  1782.             if element is not None:
  1783.                 is_id = n._is_id
  1784.                 element.removeAttributeNode(n)
  1785.         else:
  1786.             element = None
  1787.         # avoid __setattr__
  1788.         d = n.__dict__
  1789.         d['prefix'] = prefix
  1790.         d['localName'] = localName
  1791.         d['namespaceURI'] = namespaceURI
  1792.         d['nodeName'] = name
  1793.         if n.nodeType == Node.ELEMENT_NODE:
  1794.             d['tagName'] = name
  1795.         else:
  1796.             # attribute node
  1797.             d['name'] = name
  1798.             if element is not None:
  1799.                 element.setAttributeNode(n)
  1800.                 if is_id:
  1801.                     element.setIdAttributeNode(n)
  1802.         # It's not clear from a semantic perspective whether we should
  1803.         # call the user data handlers for the NODE_RENAMED event since
  1804.         # we're re-using the existing node.  The draft spec has been
  1805.         # interpreted as meaning "no, don't call the handler unless a
  1806.         # new node is created."
  1807.         return n
  1808.  
  1809. defproperty(Document, "documentElement",
  1810.             doc="Top-level element of this document.")
  1811.  
  1812.  
  1813. def _clone_node(node, deep, newOwnerDocument):
  1814.     """
  1815.     Clone a node and give it the new owner document.
  1816.     Called by Node.cloneNode and Document.importNode
  1817.     """
  1818.     if node.ownerDocument.isSameNode(newOwnerDocument):
  1819.         operation = xml.dom.UserDataHandler.NODE_CLONED
  1820.     else:
  1821.         operation = xml.dom.UserDataHandler.NODE_IMPORTED
  1822.     if node.nodeType == Node.ELEMENT_NODE:
  1823.         clone = newOwnerDocument.createElementNS(node.namespaceURI,
  1824.                                                  node.nodeName)
  1825.         for attr in node.attributes.values():
  1826.             clone.setAttributeNS(attr.namespaceURI, attr.nodeName, attr.value)
  1827.             a = clone.getAttributeNodeNS(attr.namespaceURI, attr.localName)
  1828.             a.specified = attr.specified
  1829.  
  1830.         if deep:
  1831.             for child in node.childNodes:
  1832.                 c = _clone_node(child, deep, newOwnerDocument)
  1833.                 clone.appendChild(c)
  1834.  
  1835.     elif node.nodeType == Node.DOCUMENT_FRAGMENT_NODE:
  1836.         clone = newOwnerDocument.createDocumentFragment()
  1837.         if deep:
  1838.             for child in node.childNodes:
  1839.                 c = _clone_node(child, deep, newOwnerDocument)
  1840.                 clone.appendChild(c)
  1841.  
  1842.     elif node.nodeType == Node.TEXT_NODE:
  1843.         clone = newOwnerDocument.createTextNode(node.data)
  1844.     elif node.nodeType == Node.CDATA_SECTION_NODE:
  1845.         clone = newOwnerDocument.createCDATASection(node.data)
  1846.     elif node.nodeType == Node.PROCESSING_INSTRUCTION_NODE:
  1847.         clone = newOwnerDocument.createProcessingInstruction(node.target,
  1848.                                                              node.data)
  1849.     elif node.nodeType == Node.COMMENT_NODE:
  1850.         clone = newOwnerDocument.createComment(node.data)
  1851.     elif node.nodeType == Node.ATTRIBUTE_NODE:
  1852.         clone = newOwnerDocument.createAttributeNS(node.namespaceURI,
  1853.                                                    node.nodeName)
  1854.         clone.specified = True
  1855.         clone.value = node.value
  1856.     elif node.nodeType == Node.DOCUMENT_TYPE_NODE:
  1857.         assert node.ownerDocument is not newOwnerDocument
  1858.         operation = xml.dom.UserDataHandler.NODE_IMPORTED
  1859.         clone = newOwnerDocument.implementation.createDocumentType(
  1860.             node.name, node.publicId, node.systemId)
  1861.         clone.ownerDocument = newOwnerDocument
  1862.         if deep:
  1863.             clone.entities._seq = []
  1864.             clone.notations._seq = []
  1865.             for n in node.notations._seq:
  1866.                 notation = Notation(n.nodeName, n.publicId, n.systemId)
  1867.                 notation.ownerDocument = newOwnerDocument
  1868.                 clone.notations._seq.append(notation)
  1869.                 if hasattr(n, '_call_user_data_handler'):
  1870.                     n._call_user_data_handler(operation, n, notation)
  1871.             for e in node.entities._seq:
  1872.                 entity = Entity(e.nodeName, e.publicId, e.systemId,
  1873.                                 e.notationName)
  1874.                 entity.actualEncoding = e.actualEncoding
  1875.                 entity.encoding = e.encoding
  1876.                 entity.version = e.version
  1877.                 entity.ownerDocument = newOwnerDocument
  1878.                 clone.entities._seq.append(entity)
  1879.                 if hasattr(e, '_call_user_data_handler'):
  1880.                     e._call_user_data_handler(operation, n, entity)
  1881.     else:
  1882.         # Note the cloning of Document and DocumentType nodes is
  1883.         # implemenetation specific.  minidom handles those cases
  1884.         # directly in the cloneNode() methods.
  1885.         raise xml.dom.NotSupportedErr("Cannot clone node %s" % repr(node))
  1886.  
  1887.     # Check for _call_user_data_handler() since this could conceivably
  1888.     # used with other DOM implementations (one of the FourThought
  1889.     # DOMs, perhaps?).
  1890.     if hasattr(node, '_call_user_data_handler'):
  1891.         node._call_user_data_handler(operation, node, clone)
  1892.     return clone
  1893.  
  1894.  
  1895. def _nssplit(qualifiedName):
  1896.     fields = qualifiedName.split(':', 1)
  1897.     if len(fields) == 2:
  1898.         return fields
  1899.     else:
  1900.         return (None, fields[0])
  1901.  
  1902.  
  1903. def _get_StringIO():
  1904.     # we can't use cStringIO since it doesn't support Unicode strings
  1905.     from StringIO import StringIO
  1906.     return StringIO()
  1907.  
  1908. def _do_pulldom_parse(func, args, kwargs):
  1909.     events = func(*args, **kwargs)
  1910.     toktype, rootNode = events.getEvent()
  1911.     events.expandNode(rootNode)
  1912.     events.clear()
  1913.     return rootNode
  1914.  
  1915. def parse(file, parser=None, bufsize=None):
  1916.     """Parse a file into a DOM by filename or file object."""
  1917.     if parser is None and not bufsize:
  1918.         from xml.dom import expatbuilder
  1919.         return expatbuilder.parse(file)
  1920.     else:
  1921.         from xml.dom import pulldom
  1922.         return _do_pulldom_parse(pulldom.parse, (file,),
  1923.             {'parser': parser, 'bufsize': bufsize})
  1924.  
  1925. def parseString(string, parser=None):
  1926.     """Parse a file into a DOM from a string."""
  1927.     if parser is None:
  1928.         from xml.dom import expatbuilder
  1929.         return expatbuilder.parseString(string)
  1930.     else:
  1931.         from xml.dom import pulldom
  1932.         return _do_pulldom_parse(pulldom.parseString, (string,),
  1933.                                  {'parser': parser})
  1934.  
  1935. def getDOMImplementation(features=None):
  1936.     if features:
  1937.         if isinstance(features, StringTypes):
  1938.             features = domreg._parse_feature_string(features)
  1939.         for f, v in features:
  1940.             if not Document.implementation.hasFeature(f, v):
  1941.                 return None
  1942.     return Document.implementation
  1943.