blob: d0a99a7fb9f11a52f9cfab5b2d2f359eb94f7dfa [file] [log] [blame]
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001# test for xml.dom.minidom
Paul Prescod7993bcc2000-07-01 14:54:16 +00002
Guido van Rossume2ae77b2001-10-24 20:42:55 +00003import os
Fred Drake17647f52000-07-03 16:37:42 +00004import sys
Martin v. Löwisfd6aaa12003-01-25 22:02:52 +00005import pickle
Fred Drake17647f52000-07-03 16:37:42 +00006import traceback
Martin v. Löwisfd6aaa12003-01-25 22:02:52 +00007from StringIO import StringIO
Barry Warsaw04f357c2002-07-23 19:04:11 +00008from test.test_support import verbose
Fred Drake17647f52000-07-03 16:37:42 +00009
Thomas Wouters0e3f5912006-08-11 14:57:12 +000010import xml.dom
11import xml.dom.minidom
12import xml.parsers.expat
Fred Drakec441f7b2002-07-19 22:16:41 +000013
Thomas Wouters0e3f5912006-08-11 14:57:12 +000014from xml.dom.minidom import parse, Node, Document, parseString
15from xml.dom.minidom import getDOMImplementation
Martin v. Löwisfd6aaa12003-01-25 22:02:52 +000016
Fred Drakec441f7b2002-07-19 22:16:41 +000017
Fred Drake17647f52000-07-03 16:37:42 +000018if __name__ == "__main__":
19 base = sys.argv[0]
20else:
21 base = __file__
Guido van Rossume2ae77b2001-10-24 20:42:55 +000022tstfile = os.path.join(os.path.dirname(base), "test"+os.extsep+"xml")
Fred Drake17647f52000-07-03 16:37:42 +000023del base
Paul Prescod7993bcc2000-07-01 14:54:16 +000024
Jeremy Hylton3b0c6002000-10-12 17:31:36 +000025def confirm(test, testname = "Test"):
Fred Drakec441f7b2002-07-19 22:16:41 +000026 if not test:
Guido van Rossumbe19ed72007-02-09 05:37:30 +000027 print("Failed " + testname)
Paul Prescod10d27662000-09-18 19:07:26 +000028 raise Exception
29
Paul Prescod10d27662000-09-18 19:07:26 +000030def testParseFromFile():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +000031 dom = parse(StringIO(open(tstfile).read()))
Paul Prescod4c799192000-09-19 19:33:02 +000032 dom.unlink()
Martin v. Löwis89c528b2000-09-19 16:22:10 +000033 confirm(isinstance(dom,Document))
Paul Prescod10d27662000-09-18 19:07:26 +000034
Jeremy Hylton3b0c6002000-10-12 17:31:36 +000035def testGetElementsByTagName():
36 dom = parse(tstfile)
37 confirm(dom.getElementsByTagName("LI") == \
38 dom.documentElement.getElementsByTagName("LI"))
Paul Prescod7993bcc2000-07-01 14:54:16 +000039 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +000040
Jeremy Hylton3b0c6002000-10-12 17:31:36 +000041def testInsertBefore():
Fred Drakea1bde802000-11-21 22:02:43 +000042 dom = parseString("<doc><foo/></doc>")
43 root = dom.documentElement
44 elem = root.childNodes[0]
45 nelem = dom.createElement("element")
46 root.insertBefore(nelem, elem)
47 confirm(len(root.childNodes) == 2
Fred Drake946f7b12001-09-28 20:31:50 +000048 and root.childNodes.length == 2
Fred Drakea1bde802000-11-21 22:02:43 +000049 and root.childNodes[0] is nelem
Fred Drake946f7b12001-09-28 20:31:50 +000050 and root.childNodes.item(0) is nelem
Fred Drakea1bde802000-11-21 22:02:43 +000051 and root.childNodes[1] is elem
Fred Drake946f7b12001-09-28 20:31:50 +000052 and root.childNodes.item(1) is elem
Fred Drakea1bde802000-11-21 22:02:43 +000053 and root.firstChild is nelem
54 and root.lastChild is elem
55 and root.toxml() == "<doc><element/><foo/></doc>"
56 , "testInsertBefore -- node properly placed in tree")
57 nelem = dom.createElement("element")
58 root.insertBefore(nelem, None)
59 confirm(len(root.childNodes) == 3
Fred Drake946f7b12001-09-28 20:31:50 +000060 and root.childNodes.length == 3
Fred Drakea1bde802000-11-21 22:02:43 +000061 and root.childNodes[1] is elem
Fred Drake946f7b12001-09-28 20:31:50 +000062 and root.childNodes.item(1) is elem
Fred Drakea1bde802000-11-21 22:02:43 +000063 and root.childNodes[2] is nelem
Fred Drake946f7b12001-09-28 20:31:50 +000064 and root.childNodes.item(2) is nelem
Fred Drakea1bde802000-11-21 22:02:43 +000065 and root.lastChild is nelem
66 and nelem.previousSibling is elem
67 and root.toxml() == "<doc><element/><foo/><element/></doc>"
68 , "testInsertBefore -- node properly placed in tree")
69 nelem2 = dom.createElement("bar")
70 root.insertBefore(nelem2, nelem)
71 confirm(len(root.childNodes) == 4
Fred Drake946f7b12001-09-28 20:31:50 +000072 and root.childNodes.length == 4
Fred Drakea1bde802000-11-21 22:02:43 +000073 and root.childNodes[2] is nelem2
Fred Drake946f7b12001-09-28 20:31:50 +000074 and root.childNodes.item(2) is nelem2
Fred Drakea1bde802000-11-21 22:02:43 +000075 and root.childNodes[3] is nelem
Fred Drake946f7b12001-09-28 20:31:50 +000076 and root.childNodes.item(3) is nelem
Fred Drakea1bde802000-11-21 22:02:43 +000077 and nelem2.nextSibling is nelem
78 and nelem.previousSibling is nelem2
79 and root.toxml() == "<doc><element/><foo/><bar/><element/></doc>"
80 , "testInsertBefore -- node properly placed in tree")
Paul Prescod7993bcc2000-07-01 14:54:16 +000081 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +000082
Fred Drakee50959a2001-12-06 04:32:18 +000083def _create_fragment_test_nodes():
84 dom = parseString("<doc/>")
85 orig = dom.createTextNode("original")
86 c1 = dom.createTextNode("foo")
87 c2 = dom.createTextNode("bar")
88 c3 = dom.createTextNode("bat")
89 dom.documentElement.appendChild(orig)
90 frag = dom.createDocumentFragment()
91 frag.appendChild(c1)
92 frag.appendChild(c2)
93 frag.appendChild(c3)
94 return dom, orig, c1, c2, c3, frag
95
96def testInsertBeforeFragment():
97 dom, orig, c1, c2, c3, frag = _create_fragment_test_nodes()
98 dom.documentElement.insertBefore(frag, None)
99 confirm(tuple(dom.documentElement.childNodes) == (orig, c1, c2, c3),
100 "insertBefore(<fragment>, None)")
101 frag.unlink()
102 dom.unlink()
103 #
104 dom, orig, c1, c2, c3, frag = _create_fragment_test_nodes()
105 dom.documentElement.insertBefore(frag, orig)
106 confirm(tuple(dom.documentElement.childNodes) == (c1, c2, c3, orig),
107 "insertBefore(<fragment>, orig)")
108 frag.unlink()
109 dom.unlink()
110
Paul Prescod7993bcc2000-07-01 14:54:16 +0000111def testAppendChild():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000112 dom = parse(tstfile)
113 dom.documentElement.appendChild(dom.createComment(u"Hello"))
114 confirm(dom.documentElement.childNodes[-1].nodeName == "#comment")
115 confirm(dom.documentElement.childNodes[-1].data == "Hello")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000116 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000117
Fred Drakee50959a2001-12-06 04:32:18 +0000118def testAppendChildFragment():
119 dom, orig, c1, c2, c3, frag = _create_fragment_test_nodes()
120 dom.documentElement.appendChild(frag)
121 confirm(tuple(dom.documentElement.childNodes) == (orig, c1, c2, c3),
122 "appendChild(<fragment>)")
123 frag.unlink()
124 dom.unlink()
125
126def testReplaceChildFragment():
127 dom, orig, c1, c2, c3, frag = _create_fragment_test_nodes()
128 dom.documentElement.replaceChild(frag, orig)
129 orig.unlink()
130 confirm(tuple(dom.documentElement.childNodes) == (c1, c2, c3),
131 "replaceChild(<fragment>)")
132 frag.unlink()
133 dom.unlink()
134
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000135def testLegalChildren():
136 dom = Document()
137 elem = dom.createElement('element')
138 text = dom.createTextNode('text')
Fredrik Lundhf7850422001-01-17 21:51:36 +0000139
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000140 try: dom.appendChild(text)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000141 except xml.dom.HierarchyRequestErr: pass
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000142 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000143 print("dom.appendChild didn't raise HierarchyRequestErr")
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000144
145 dom.appendChild(elem)
146 try: dom.insertBefore(text, elem)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000147 except xml.dom.HierarchyRequestErr: pass
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000148 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000149 print("dom.appendChild didn't raise HierarchyRequestErr")
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000150
151 try: dom.replaceChild(text, elem)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000152 except xml.dom.HierarchyRequestErr: pass
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000153 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000154 print("dom.appendChild didn't raise HierarchyRequestErr")
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000155
Tim Peters0009c4e2001-02-21 07:29:48 +0000156 nodemap = elem.attributes
Andrew M. Kuchlingbc8f72c2001-02-21 01:30:26 +0000157 try: nodemap.setNamedItem(text)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000158 except xml.dom.HierarchyRequestErr: pass
Andrew M. Kuchlingbc8f72c2001-02-21 01:30:26 +0000159 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000160 print("NamedNodeMap.setNamedItem didn't raise HierarchyRequestErr")
Andrew M. Kuchlingbc8f72c2001-02-21 01:30:26 +0000161
162 try: nodemap.setNamedItemNS(text)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000163 except xml.dom.HierarchyRequestErr: pass
Andrew M. Kuchlingbc8f72c2001-02-21 01:30:26 +0000164 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000165 print("NamedNodeMap.setNamedItemNS didn't raise HierarchyRequestErr")
Andrew M. Kuchlingbc8f72c2001-02-21 01:30:26 +0000166
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000167 elem.appendChild(text)
Fredrik Lundhf7850422001-01-17 21:51:36 +0000168 dom.unlink()
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000169
Fred Drake2998a552001-12-06 18:27:48 +0000170def testNamedNodeMapSetItem():
171 dom = Document()
172 elem = dom.createElement('element')
173 attrs = elem.attributes
174 attrs["foo"] = "bar"
175 a = attrs.item(0)
176 confirm(a.ownerDocument is dom,
177 "NamedNodeMap.__setitem__() sets ownerDocument")
178 confirm(a.ownerElement is elem,
179 "NamedNodeMap.__setitem__() sets ownerElement")
180 confirm(a.value == "bar",
181 "NamedNodeMap.__setitem__() sets value")
182 confirm(a.nodeValue == "bar",
183 "NamedNodeMap.__setitem__() sets nodeValue")
184 elem.unlink()
185 dom.unlink()
186
Paul Prescod7993bcc2000-07-01 14:54:16 +0000187def testNonZero():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000188 dom = parse(tstfile)
189 confirm(dom)# should not be zero
190 dom.appendChild(dom.createComment("foo"))
191 confirm(not dom.childNodes[-1].childNodes)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000192 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000193
194def testUnlink():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000195 dom = parse(tstfile)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000196 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000197
198def testElement():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000199 dom = Document()
200 dom.appendChild(dom.createElement("abc"))
201 confirm(dom.documentElement)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000202 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000203
204def testAAA():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000205 dom = parseString("<abc/>")
206 el = dom.documentElement
207 el.setAttribute("spam", "jam2")
Fred Drakea1bde802000-11-21 22:02:43 +0000208 confirm(el.toxml() == '<abc spam="jam2"/>', "testAAA")
Fred Drake2998a552001-12-06 18:27:48 +0000209 a = el.getAttributeNode("spam")
210 confirm(a.ownerDocument is dom,
211 "setAttribute() sets ownerDocument")
212 confirm(a.ownerElement is dom.documentElement,
213 "setAttribute() sets ownerElement")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000214 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000215
216def testAAB():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000217 dom = parseString("<abc/>")
218 el = dom.documentElement
219 el.setAttribute("spam", "jam")
220 el.setAttribute("spam", "jam2")
Fred Drakea1bde802000-11-21 22:02:43 +0000221 confirm(el.toxml() == '<abc spam="jam2"/>', "testAAB")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000222 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000223
224def testAddAttr():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000225 dom = Document()
226 child = dom.appendChild(dom.createElement("abc"))
Paul Prescod7993bcc2000-07-01 14:54:16 +0000227
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000228 child.setAttribute("def", "ghi")
229 confirm(child.getAttribute("def") == "ghi")
230 confirm(child.attributes["def"].value == "ghi")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000231
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000232 child.setAttribute("jkl", "mno")
233 confirm(child.getAttribute("jkl") == "mno")
234 confirm(child.attributes["jkl"].value == "mno")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000235
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000236 confirm(len(child.attributes) == 2)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000237
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000238 child.setAttribute("def", "newval")
239 confirm(child.getAttribute("def") == "newval")
240 confirm(child.attributes["def"].value == "newval")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000241
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000242 confirm(len(child.attributes) == 2)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000243 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000244
245def testDeleteAttr():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000246 dom = Document()
247 child = dom.appendChild(dom.createElement("abc"))
Paul Prescod7993bcc2000-07-01 14:54:16 +0000248
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000249 confirm(len(child.attributes) == 0)
250 child.setAttribute("def", "ghi")
251 confirm(len(child.attributes) == 1)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000252 del child.attributes["def"]
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000253 confirm(len(child.attributes) == 0)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000254 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000255
256def testRemoveAttr():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000257 dom = Document()
258 child = dom.appendChild(dom.createElement("abc"))
Paul Prescod7993bcc2000-07-01 14:54:16 +0000259
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000260 child.setAttribute("def", "ghi")
261 confirm(len(child.attributes) == 1)
262 child.removeAttribute("def")
263 confirm(len(child.attributes) == 0)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000264
265 dom.unlink()
266
267def testRemoveAttrNS():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000268 dom = Document()
269 child = dom.appendChild(
270 dom.createElementNS("http://www.python.org", "python:abc"))
Fred Drake004d5e62000-10-23 17:22:08 +0000271 child.setAttributeNS("http://www.w3.org", "xmlns:python",
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000272 "http://www.python.org")
273 child.setAttributeNS("http://www.python.org", "python:abcattr", "foo")
274 confirm(len(child.attributes) == 2)
275 child.removeAttributeNS("http://www.python.org", "abcattr")
276 confirm(len(child.attributes) == 1)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000277
278 dom.unlink()
Fred Drake004d5e62000-10-23 17:22:08 +0000279
Paul Prescod7993bcc2000-07-01 14:54:16 +0000280def testRemoveAttributeNode():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000281 dom = Document()
282 child = dom.appendChild(dom.createElement("foo"))
283 child.setAttribute("spam", "jam")
284 confirm(len(child.attributes) == 1)
285 node = child.getAttributeNode("spam")
286 child.removeAttributeNode(node)
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000287 confirm(len(child.attributes) == 0
288 and child.getAttributeNode("spam") is None)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000289
290 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000291
292def testChangeAttr():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000293 dom = parseString("<abc/>")
294 el = dom.documentElement
295 el.setAttribute("spam", "jam")
296 confirm(len(el.attributes) == 1)
297 el.setAttribute("spam", "bam")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000298 # Set this attribute to be an ID and make sure that doesn't change
299 # when changing the value:
300 el.setIdAttribute("spam")
301 confirm(len(el.attributes) == 1
302 and el.attributes["spam"].value == "bam"
303 and el.attributes["spam"].nodeValue == "bam"
304 and el.getAttribute("spam") == "bam"
305 and el.getAttributeNode("spam").isId)
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000306 el.attributes["spam"] = "ham"
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000307 confirm(len(el.attributes) == 1
308 and el.attributes["spam"].value == "ham"
309 and el.attributes["spam"].nodeValue == "ham"
310 and el.getAttribute("spam") == "ham"
311 and el.attributes["spam"].isId)
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000312 el.setAttribute("spam2", "bam")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000313 confirm(len(el.attributes) == 2
314 and el.attributes["spam"].value == "ham"
315 and el.attributes["spam"].nodeValue == "ham"
316 and el.getAttribute("spam") == "ham"
317 and el.attributes["spam2"].value == "bam"
318 and el.attributes["spam2"].nodeValue == "bam"
319 and el.getAttribute("spam2") == "bam")
320 el.attributes["spam2"] = "bam2"
321 confirm(len(el.attributes) == 2
322 and el.attributes["spam"].value == "ham"
323 and el.attributes["spam"].nodeValue == "ham"
324 and el.getAttribute("spam") == "ham"
325 and el.attributes["spam2"].value == "bam2"
326 and el.attributes["spam2"].nodeValue == "bam2"
327 and el.getAttribute("spam2") == "bam2")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000328 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000329
330def testGetAttrList():
331 pass
332
333def testGetAttrValues(): pass
334
335def testGetAttrLength(): pass
336
337def testGetAttribute(): pass
338
339def testGetAttributeNS(): pass
340
341def testGetAttributeNode(): pass
342
Martin v. Löwis351c3d02001-06-03 14:27:02 +0000343def testGetElementsByTagNameNS():
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000344 d="""<foo xmlns:minidom='http://pyxml.sf.net/minidom'>
Martin v. Löwis351c3d02001-06-03 14:27:02 +0000345 <minidom:myelem/>
346 </foo>"""
347 dom = parseString(d)
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000348 elems = dom.getElementsByTagNameNS("http://pyxml.sf.net/minidom", "myelem")
349 confirm(len(elems) == 1
350 and elems[0].namespaceURI == "http://pyxml.sf.net/minidom"
351 and elems[0].localName == "myelem"
352 and elems[0].prefix == "minidom"
353 and elems[0].tagName == "minidom:myelem"
354 and elems[0].nodeName == "minidom:myelem")
Martin v. Löwis351c3d02001-06-03 14:27:02 +0000355 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000356
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000357def get_empty_nodelist_from_elements_by_tagName_ns_helper(doc, nsuri, lname):
358 nodelist = doc.getElementsByTagNameNS(nsuri, lname)
359 confirm(len(nodelist) == 0)
360
361def testGetEmptyNodeListFromElementsByTagNameNS():
362 doc = parseString('<doc/>')
363 get_empty_nodelist_from_elements_by_tagName_ns_helper(
364 doc, 'http://xml.python.org/namespaces/a', 'localname')
365 get_empty_nodelist_from_elements_by_tagName_ns_helper(
366 doc, '*', 'splat')
367 get_empty_nodelist_from_elements_by_tagName_ns_helper(
368 doc, 'http://xml.python.org/namespaces/a', '*')
369
370 doc = parseString('<doc xmlns="http://xml.python.org/splat"><e/></doc>')
371 get_empty_nodelist_from_elements_by_tagName_ns_helper(
372 doc, "http://xml.python.org/splat", "not-there")
373 get_empty_nodelist_from_elements_by_tagName_ns_helper(
374 doc, "*", "not-there")
375 get_empty_nodelist_from_elements_by_tagName_ns_helper(
376 doc, "http://somewhere.else.net/not-there", "e")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000377
378def testElementReprAndStr():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000379 dom = Document()
380 el = dom.appendChild(dom.createElement("abc"))
381 string1 = repr(el)
382 string2 = str(el)
383 confirm(string1 == string2)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000384 dom.unlink()
385
386# commented out until Fredrick's fix is checked in
387def _testElementReprAndStrUnicode():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000388 dom = Document()
389 el = dom.appendChild(dom.createElement(u"abc"))
390 string1 = repr(el)
391 string2 = str(el)
392 confirm(string1 == string2)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000393 dom.unlink()
394
395# commented out until Fredrick's fix is checked in
396def _testElementReprAndStrUnicodeNS():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000397 dom = Document()
398 el = dom.appendChild(
399 dom.createElementNS(u"http://www.slashdot.org", u"slash:abc"))
400 string1 = repr(el)
401 string2 = str(el)
402 confirm(string1 == string2)
403 confirm(string1.find("slash:abc") != -1)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000404 dom.unlink()
405
406def testAttributeRepr():
Jeremy Hylton3b0c6002000-10-12 17:31:36 +0000407 dom = Document()
408 el = dom.appendChild(dom.createElement(u"abc"))
409 node = el.setAttribute("abc", "def")
410 confirm(str(node) == repr(node))
Paul Prescod7993bcc2000-07-01 14:54:16 +0000411 dom.unlink()
412
413def testTextNodeRepr(): pass
414
Martin v. Löwis0a84a332000-10-06 22:42:55 +0000415def testWriteXML():
Georg Brandl905a01a2005-08-25 22:14:15 +0000416 str = '<?xml version="1.0" ?><a b="c"/>'
Martin v. Löwis0a84a332000-10-06 22:42:55 +0000417 dom = parseString(str)
418 domstr = dom.toxml()
419 dom.unlink()
420 confirm(str == domstr)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000421
Georg Brandl905a01a2005-08-25 22:14:15 +0000422def testAltNewline():
423 str = '<?xml version="1.0" ?>\n<a b="c"/>\n'
424 dom = parseString(str)
425 domstr = dom.toprettyxml(newl="\r\n")
426 dom.unlink()
427 confirm(domstr == str.replace("\n", "\r\n"))
428
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000429def testProcessingInstruction():
430 dom = parseString('<e><?mypi \t\n data \t\n ?></e>')
431 pi = dom.documentElement.firstChild
432 confirm(pi.target == "mypi"
433 and pi.data == "data \t\n "
434 and pi.nodeName == "mypi"
435 and pi.nodeType == Node.PROCESSING_INSTRUCTION_NODE
436 and pi.attributes is None
437 and not pi.hasChildNodes()
438 and len(pi.childNodes) == 0
439 and pi.firstChild is None
440 and pi.lastChild is None
441 and pi.localName is None
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000442 and pi.namespaceURI == xml.dom.EMPTY_NAMESPACE)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000443
444def testProcessingInstructionRepr(): pass
445
446def testTextRepr(): pass
447
448def testWriteText(): pass
449
450def testDocumentElement(): pass
451
Fred Drakea1bde802000-11-21 22:02:43 +0000452def testTooManyDocumentElements():
453 doc = parseString("<doc/>")
454 elem = doc.createElement("extra")
455 try:
456 doc.appendChild(elem)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000457 except xml.dom.HierarchyRequestErr:
Fred Drakec441f7b2002-07-19 22:16:41 +0000458 pass
Fred Drakea1bde802000-11-21 22:02:43 +0000459 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000460 print("Failed to catch expected exception when" \
461 " adding extra document element.")
Fred Drakea1bde802000-11-21 22:02:43 +0000462 elem.unlink()
463 doc.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000464
465def testCreateElementNS(): pass
466
Andrew M. Kuchlingad4a5582000-12-31 04:03:27 +0000467def testCreateAttributeNS(): pass
Paul Prescod7993bcc2000-07-01 14:54:16 +0000468
469def testParse(): pass
470
471def testParseString(): pass
472
473def testComment(): pass
474
475def testAttrListItem(): pass
476
477def testAttrListItems(): pass
478
479def testAttrListItemNS(): pass
480
481def testAttrListKeys(): pass
482
483def testAttrListKeysNS(): pass
484
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000485def testRemoveNamedItem():
486 doc = parseString("<doc a=''/>")
487 e = doc.documentElement
488 attrs = e.attributes
489 a1 = e.getAttributeNode("a")
490 a2 = attrs.removeNamedItem("a")
491 confirm(a1.isSameNode(a2))
492 try:
493 attrs.removeNamedItem("a")
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000494 except xml.dom.NotFoundErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000495 pass
496
497def testRemoveNamedItemNS():
498 doc = parseString("<doc xmlns:a='http://xml.python.org/' a:b=''/>")
499 e = doc.documentElement
500 attrs = e.attributes
501 a1 = e.getAttributeNodeNS("http://xml.python.org/", "b")
502 a2 = attrs.removeNamedItemNS("http://xml.python.org/", "b")
503 confirm(a1.isSameNode(a2))
504 try:
505 attrs.removeNamedItemNS("http://xml.python.org/", "b")
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000506 except xml.dom.NotFoundErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000507 pass
508
Paul Prescod7993bcc2000-07-01 14:54:16 +0000509def testAttrListValues(): pass
510
511def testAttrListLength(): pass
512
513def testAttrList__getitem__(): pass
514
515def testAttrList__setitem__(): pass
516
517def testSetAttrValueandNodeValue(): pass
518
519def testParseElement(): pass
520
521def testParseAttributes(): pass
522
523def testParseElementNamespaces(): pass
524
525def testParseAttributeNamespaces(): pass
526
527def testParseProcessingInstructions(): pass
528
529def testChildNodes(): pass
530
531def testFirstChild(): pass
532
533def testHasChildNodes(): pass
534
Fred Drakea1bde802000-11-21 22:02:43 +0000535def testCloneElementShallow():
536 dom, clone = _setupCloneElement(0)
537 confirm(len(clone.childNodes) == 0
Fred Drake946f7b12001-09-28 20:31:50 +0000538 and clone.childNodes.length == 0
Fred Drakea1bde802000-11-21 22:02:43 +0000539 and clone.parentNode is None
540 and clone.toxml() == '<doc attr="value"/>'
541 , "testCloneElementShallow")
542 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000543
Fred Drakea1bde802000-11-21 22:02:43 +0000544def testCloneElementDeep():
545 dom, clone = _setupCloneElement(1)
546 confirm(len(clone.childNodes) == 1
Fred Drake946f7b12001-09-28 20:31:50 +0000547 and clone.childNodes.length == 1
Fred Drakea1bde802000-11-21 22:02:43 +0000548 and clone.parentNode is None
549 and clone.toxml() == '<doc attr="value"><foo/></doc>'
550 , "testCloneElementDeep")
551 dom.unlink()
Paul Prescod7993bcc2000-07-01 14:54:16 +0000552
Fred Drakea1bde802000-11-21 22:02:43 +0000553def _setupCloneElement(deep):
554 dom = parseString("<doc attr='value'><foo/></doc>")
555 root = dom.documentElement
556 clone = root.cloneNode(deep)
557 _testCloneElementCopiesAttributes(
558 root, clone, "testCloneElement" + (deep and "Deep" or "Shallow"))
559 # mutilate the original so shared data is detected
560 root.tagName = root.nodeName = "MODIFIED"
561 root.setAttribute("attr", "NEW VALUE")
562 root.setAttribute("added", "VALUE")
563 return dom, clone
564
565def _testCloneElementCopiesAttributes(e1, e2, test):
566 attrs1 = e1.attributes
567 attrs2 = e2.attributes
568 keys1 = attrs1.keys()
569 keys2 = attrs2.keys()
570 keys1.sort()
571 keys2.sort()
572 confirm(keys1 == keys2, "clone of element has same attribute keys")
573 for i in range(len(keys1)):
574 a1 = attrs1.item(i)
575 a2 = attrs2.item(i)
576 confirm(a1 is not a2
577 and a1.value == a2.value
578 and a1.nodeValue == a2.nodeValue
579 and a1.namespaceURI == a2.namespaceURI
580 and a1.localName == a2.localName
581 , "clone of attribute node has proper attribute values")
582 confirm(a2.ownerElement is e2,
583 "clone of attribute node correctly owned")
Fredrik Lundhf7850422001-01-17 21:51:36 +0000584
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000585def testCloneDocumentShallow():
586 doc = parseString("<?xml version='1.0'?>\n"
587 "<!-- comment -->"
588 "<!DOCTYPE doc [\n"
589 "<!NOTATION notation SYSTEM 'http://xml.python.org/'>\n"
590 "]>\n"
591 "<doc attr='value'/>")
592 doc2 = doc.cloneNode(0)
593 confirm(doc2 is None,
594 "testCloneDocumentShallow:"
595 " shallow cloning of documents makes no sense!")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000596
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000597def testCloneDocumentDeep():
598 doc = parseString("<?xml version='1.0'?>\n"
599 "<!-- comment -->"
600 "<!DOCTYPE doc [\n"
601 "<!NOTATION notation SYSTEM 'http://xml.python.org/'>\n"
602 "]>\n"
603 "<doc attr='value'/>")
604 doc2 = doc.cloneNode(1)
605 confirm(not (doc.isSameNode(doc2) or doc2.isSameNode(doc)),
606 "testCloneDocumentDeep: document objects not distinct")
607 confirm(len(doc.childNodes) == len(doc2.childNodes),
608 "testCloneDocumentDeep: wrong number of Document children")
609 confirm(doc2.documentElement.nodeType == Node.ELEMENT_NODE,
610 "testCloneDocumentDeep: documentElement not an ELEMENT_NODE")
611 confirm(doc2.documentElement.ownerDocument.isSameNode(doc2),
612 "testCloneDocumentDeep: documentElement owner is not new document")
613 confirm(not doc.documentElement.isSameNode(doc2.documentElement),
614 "testCloneDocumentDeep: documentElement should not be shared")
615 if doc.doctype is not None:
616 # check the doctype iff the original DOM maintained it
617 confirm(doc2.doctype.nodeType == Node.DOCUMENT_TYPE_NODE,
618 "testCloneDocumentDeep: doctype not a DOCUMENT_TYPE_NODE")
619 confirm(doc2.doctype.ownerDocument.isSameNode(doc2))
620 confirm(not doc.doctype.isSameNode(doc2.doctype))
Paul Prescod7993bcc2000-07-01 14:54:16 +0000621
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000622def testCloneDocumentTypeDeepOk():
623 doctype = create_nonempty_doctype()
624 clone = doctype.cloneNode(1)
625 confirm(clone is not None
626 and clone.nodeName == doctype.nodeName
627 and clone.name == doctype.name
628 and clone.publicId == doctype.publicId
629 and clone.systemId == doctype.systemId
630 and len(clone.entities) == len(doctype.entities)
631 and clone.entities.item(len(clone.entities)) is None
632 and len(clone.notations) == len(doctype.notations)
633 and clone.notations.item(len(clone.notations)) is None
634 and len(clone.childNodes) == 0)
635 for i in range(len(doctype.entities)):
636 se = doctype.entities.item(i)
637 ce = clone.entities.item(i)
638 confirm((not se.isSameNode(ce))
639 and (not ce.isSameNode(se))
640 and ce.nodeName == se.nodeName
641 and ce.notationName == se.notationName
642 and ce.publicId == se.publicId
643 and ce.systemId == se.systemId
644 and ce.encoding == se.encoding
645 and ce.actualEncoding == se.actualEncoding
646 and ce.version == se.version)
647 for i in range(len(doctype.notations)):
648 sn = doctype.notations.item(i)
649 cn = clone.notations.item(i)
650 confirm((not sn.isSameNode(cn))
651 and (not cn.isSameNode(sn))
652 and cn.nodeName == sn.nodeName
653 and cn.publicId == sn.publicId
654 and cn.systemId == sn.systemId)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000655
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000656def testCloneDocumentTypeDeepNotOk():
657 doc = create_doc_with_doctype()
658 clone = doc.doctype.cloneNode(1)
659 confirm(clone is None, "testCloneDocumentTypeDeepNotOk")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000660
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000661def testCloneDocumentTypeShallowOk():
662 doctype = create_nonempty_doctype()
663 clone = doctype.cloneNode(0)
664 confirm(clone is not None
665 and clone.nodeName == doctype.nodeName
666 and clone.name == doctype.name
667 and clone.publicId == doctype.publicId
668 and clone.systemId == doctype.systemId
669 and len(clone.entities) == 0
670 and clone.entities.item(0) is None
671 and len(clone.notations) == 0
672 and clone.notations.item(0) is None
673 and len(clone.childNodes) == 0)
Paul Prescod7993bcc2000-07-01 14:54:16 +0000674
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000675def testCloneDocumentTypeShallowNotOk():
676 doc = create_doc_with_doctype()
677 clone = doc.doctype.cloneNode(0)
678 confirm(clone is None, "testCloneDocumentTypeShallowNotOk")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000679
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000680def check_import_document(deep, testName):
681 doc1 = parseString("<doc/>")
682 doc2 = parseString("<doc/>")
683 try:
684 doc1.importNode(doc2, deep)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000685 except xml.dom.NotSupportedErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000686 pass
687 else:
688 raise Exception(testName +
689 ": expected NotSupportedErr when importing a document")
690
691def testImportDocumentShallow():
692 check_import_document(0, "testImportDocumentShallow")
693
694def testImportDocumentDeep():
695 check_import_document(1, "testImportDocumentDeep")
696
697# The tests of DocumentType importing use these helpers to construct
698# the documents to work with, since not all DOM builders actually
699# create the DocumentType nodes.
700
701def create_doc_without_doctype(doctype=None):
702 return getDOMImplementation().createDocument(None, "doc", doctype)
703
704def create_nonempty_doctype():
705 doctype = getDOMImplementation().createDocumentType("doc", None, None)
706 doctype.entities._seq = []
707 doctype.notations._seq = []
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000708 notation = xml.dom.minidom.Notation("my-notation", None,
709 "http://xml.python.org/notations/my")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000710 doctype.notations._seq.append(notation)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000711 entity = xml.dom.minidom.Entity("my-entity", None,
712 "http://xml.python.org/entities/my",
713 "my-notation")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000714 entity.version = "1.0"
715 entity.encoding = "utf-8"
716 entity.actualEncoding = "us-ascii"
717 doctype.entities._seq.append(entity)
718 return doctype
719
720def create_doc_with_doctype():
721 doctype = create_nonempty_doctype()
722 doc = create_doc_without_doctype(doctype)
723 doctype.entities.item(0).ownerDocument = doc
724 doctype.notations.item(0).ownerDocument = doc
725 return doc
726
727def testImportDocumentTypeShallow():
728 src = create_doc_with_doctype()
729 target = create_doc_without_doctype()
730 try:
731 imported = target.importNode(src.doctype, 0)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000732 except xml.dom.NotSupportedErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000733 pass
734 else:
735 raise Exception(
736 "testImportDocumentTypeShallow: expected NotSupportedErr")
737
738def testImportDocumentTypeDeep():
739 src = create_doc_with_doctype()
740 target = create_doc_without_doctype()
741 try:
742 imported = target.importNode(src.doctype, 1)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000743 except xml.dom.NotSupportedErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000744 pass
745 else:
746 raise Exception(
747 "testImportDocumentTypeDeep: expected NotSupportedErr")
748
749# Testing attribute clones uses a helper, and should always be deep,
750# even if the argument to cloneNode is false.
751def check_clone_attribute(deep, testName):
752 doc = parseString("<doc attr='value'/>")
753 attr = doc.documentElement.getAttributeNode("attr")
754 assert attr is not None
755 clone = attr.cloneNode(deep)
756 confirm(not clone.isSameNode(attr))
757 confirm(not attr.isSameNode(clone))
758 confirm(clone.ownerElement is None,
759 testName + ": ownerElement should be None")
760 confirm(clone.ownerDocument.isSameNode(attr.ownerDocument),
761 testName + ": ownerDocument does not match")
762 confirm(clone.specified,
763 testName + ": cloned attribute must have specified == True")
764
765def testCloneAttributeShallow():
766 check_clone_attribute(0, "testCloneAttributeShallow")
767
768def testCloneAttributeDeep():
769 check_clone_attribute(1, "testCloneAttributeDeep")
770
771def check_clone_pi(deep, testName):
772 doc = parseString("<?target data?><doc/>")
773 pi = doc.firstChild
774 assert pi.nodeType == Node.PROCESSING_INSTRUCTION_NODE
775 clone = pi.cloneNode(deep)
776 confirm(clone.target == pi.target
777 and clone.data == pi.data)
778
779def testClonePIShallow():
780 check_clone_pi(0, "testClonePIShallow")
781
782def testClonePIDeep():
783 check_clone_pi(1, "testClonePIDeep")
Paul Prescod7993bcc2000-07-01 14:54:16 +0000784
Fred Drakea1bde802000-11-21 22:02:43 +0000785def testNormalize():
786 doc = parseString("<doc/>")
787 root = doc.documentElement
788 root.appendChild(doc.createTextNode("first"))
789 root.appendChild(doc.createTextNode("second"))
Fred Drake946f7b12001-09-28 20:31:50 +0000790 confirm(len(root.childNodes) == 2
791 and root.childNodes.length == 2, "testNormalize -- preparation")
Fred Drakea1bde802000-11-21 22:02:43 +0000792 doc.normalize()
793 confirm(len(root.childNodes) == 1
Fred Drake946f7b12001-09-28 20:31:50 +0000794 and root.childNodes.length == 1
Fred Drakea1bde802000-11-21 22:02:43 +0000795 and root.firstChild is root.lastChild
796 and root.firstChild.data == "firstsecond"
797 , "testNormalize -- result")
798 doc.unlink()
799
Fred Drake3277da02000-12-14 18:20:22 +0000800 doc = parseString("<doc/>")
801 root = doc.documentElement
802 root.appendChild(doc.createTextNode(""))
803 doc.normalize()
Fred Drake946f7b12001-09-28 20:31:50 +0000804 confirm(len(root.childNodes) == 0
805 and root.childNodes.length == 0,
Fred Drake3277da02000-12-14 18:20:22 +0000806 "testNormalize -- single empty node removed")
807 doc.unlink()
808
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000809def testSiblings():
810 doc = parseString("<doc><?pi?>text?<elm/></doc>")
811 root = doc.documentElement
812 (pi, text, elm) = root.childNodes
Paul Prescod7993bcc2000-07-01 14:54:16 +0000813
Fred Drake004d5e62000-10-23 17:22:08 +0000814 confirm(pi.nextSibling is text and
815 pi.previousSibling is None and
816 text.nextSibling is elm and
817 text.previousSibling is pi and
818 elm.nextSibling is None and
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000819 elm.previousSibling is text, "testSiblings")
820
821 doc.unlink()
822
823def testParents():
824 doc = parseString("<doc><elm1><elm2/><elm2><elm3/></elm2></elm1></doc>")
825 root = doc.documentElement
826 elm1 = root.childNodes[0]
827 (elm2a, elm2b) = elm1.childNodes
828 elm3 = elm2b.childNodes[0]
829
830 confirm(root.parentNode is doc and
831 elm1.parentNode is root and
832 elm2a.parentNode is elm1 and
833 elm2b.parentNode is elm1 and
834 elm3.parentNode is elm2b, "testParents")
835
836 doc.unlink()
837
Fred Drake946f7b12001-09-28 20:31:50 +0000838def testNodeListItem():
839 doc = parseString("<doc><e/><e/></doc>")
840 children = doc.childNodes
841 docelem = children[0]
842 confirm(children[0] is children.item(0)
843 and children.item(1) is None
844 and docelem.childNodes.item(0) is docelem.childNodes[0]
845 and docelem.childNodes.item(1) is docelem.childNodes[1]
846 and docelem.childNodes.item(0).childNodes.item(0) is None,
847 "test NodeList.item()")
848 doc.unlink()
849
Lars Gustäbel5bad5a42000-10-13 20:54:10 +0000850def testSAX2DOM():
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000851 from xml.dom import pulldom
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000852
Lars Gustäbel5bad5a42000-10-13 20:54:10 +0000853 sax2dom = pulldom.SAX2DOM()
854 sax2dom.startDocument()
855 sax2dom.startElement("doc", {})
856 sax2dom.characters("text")
857 sax2dom.startElement("subelm", {})
858 sax2dom.characters("text")
859 sax2dom.endElement("subelm")
Fred Drake004d5e62000-10-23 17:22:08 +0000860 sax2dom.characters("text")
Lars Gustäbel5bad5a42000-10-13 20:54:10 +0000861 sax2dom.endElement("doc")
862 sax2dom.endDocument()
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000863
Lars Gustäbel5bad5a42000-10-13 20:54:10 +0000864 doc = sax2dom.document
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000865 root = doc.documentElement
866 (text1, elm1, text2) = root.childNodes
867 text3 = elm1.childNodes[0]
868
869 confirm(text1.previousSibling is None and
870 text1.nextSibling is elm1 and
871 elm1.previousSibling is text1 and
872 elm1.nextSibling is text2 and
873 text2.previousSibling is elm1 and
874 text2.nextSibling is None and
875 text3.previousSibling is None and
Lars Gustäbel5bad5a42000-10-13 20:54:10 +0000876 text3.nextSibling is None, "testSAX2DOM - siblings")
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000877
878 confirm(root.parentNode is doc and
879 text1.parentNode is root and
880 elm1.parentNode is root and
881 text2.parentNode is root and
Lars Gustäbel5bad5a42000-10-13 20:54:10 +0000882 text3.parentNode is elm1, "testSAX2DOM - parents")
Fred Drake004d5e62000-10-23 17:22:08 +0000883
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +0000884 doc.unlink()
885
Martin v. Löwis7d650ca2002-06-30 15:05:00 +0000886def testEncodings():
887 doc = parseString('<foo>&#x20ac;</foo>')
Georg Brandl905a01a2005-08-25 22:14:15 +0000888 confirm(doc.toxml() == u'<?xml version="1.0" ?><foo>\u20ac</foo>'
889 and doc.toxml('utf-8') == '<?xml version="1.0" encoding="utf-8"?><foo>\xe2\x82\xac</foo>'
890 and doc.toxml('iso-8859-15') == '<?xml version="1.0" encoding="iso-8859-15"?><foo>\xa4</foo>',
Martin v. Löwis7d650ca2002-06-30 15:05:00 +0000891 "testEncodings - encoding EURO SIGN")
Neal Norwitz484d9a42005-09-30 04:46:49 +0000892
893 # Verify that character decoding errors throw exceptions instead of crashing
894 try:
895 doc = parseString('<fran\xe7ais>Comment \xe7a va ? Tr\xe8s bien ?</fran\xe7ais>')
896 except UnicodeDecodeError:
897 pass
898 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000899 print('parsing with bad encoding should raise a UnicodeDecodeError')
Neal Norwitz484d9a42005-09-30 04:46:49 +0000900
Martin v. Löwis7d650ca2002-06-30 15:05:00 +0000901 doc.unlink()
902
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000903class UserDataHandler:
904 called = 0
905 def handle(self, operation, key, data, src, dst):
906 dst.setUserData(key, data + 1, self)
907 src.setUserData(key, None, None)
908 self.called = 1
909
910def testUserData():
911 dom = Document()
912 n = dom.createElement('e')
913 confirm(n.getUserData("foo") is None)
914 n.setUserData("foo", None, None)
915 confirm(n.getUserData("foo") is None)
916 n.setUserData("foo", 12, 12)
917 n.setUserData("bar", 13, 13)
918 confirm(n.getUserData("foo") == 12)
919 confirm(n.getUserData("bar") == 13)
920 n.setUserData("foo", None, None)
921 confirm(n.getUserData("foo") is None)
922 confirm(n.getUserData("bar") == 13)
923
924 handler = UserDataHandler()
925 n.setUserData("bar", 12, handler)
926 c = n.cloneNode(1)
927 confirm(handler.called
928 and n.getUserData("bar") is None
929 and c.getUserData("bar") == 13)
930 n.unlink()
931 c.unlink()
932 dom.unlink()
933
934def testRenameAttribute():
935 doc = parseString("<doc a='v'/>")
936 elem = doc.documentElement
937 attrmap = elem.attributes
938 attr = elem.attributes['a']
939
940 # Simple renaming
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000941 attr = doc.renameNode(attr, xml.dom.EMPTY_NAMESPACE, "b")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000942 confirm(attr.name == "b"
943 and attr.nodeName == "b"
944 and attr.localName is None
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000945 and attr.namespaceURI == xml.dom.EMPTY_NAMESPACE
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000946 and attr.prefix is None
947 and attr.value == "v"
948 and elem.getAttributeNode("a") is None
949 and elem.getAttributeNode("b").isSameNode(attr)
950 and attrmap["b"].isSameNode(attr)
951 and attr.ownerDocument.isSameNode(doc)
952 and attr.ownerElement.isSameNode(elem))
953
954 # Rename to have a namespace, no prefix
955 attr = doc.renameNode(attr, "http://xml.python.org/ns", "c")
956 confirm(attr.name == "c"
957 and attr.nodeName == "c"
958 and attr.localName == "c"
959 and attr.namespaceURI == "http://xml.python.org/ns"
960 and attr.prefix is None
961 and attr.value == "v"
962 and elem.getAttributeNode("a") is None
963 and elem.getAttributeNode("b") is None
964 and elem.getAttributeNode("c").isSameNode(attr)
965 and elem.getAttributeNodeNS(
966 "http://xml.python.org/ns", "c").isSameNode(attr)
967 and attrmap["c"].isSameNode(attr)
968 and attrmap[("http://xml.python.org/ns", "c")].isSameNode(attr))
969
970 # Rename to have a namespace, with prefix
971 attr = doc.renameNode(attr, "http://xml.python.org/ns2", "p:d")
972 confirm(attr.name == "p:d"
973 and attr.nodeName == "p:d"
974 and attr.localName == "d"
975 and attr.namespaceURI == "http://xml.python.org/ns2"
976 and attr.prefix == "p"
977 and attr.value == "v"
978 and elem.getAttributeNode("a") is None
979 and elem.getAttributeNode("b") is None
980 and elem.getAttributeNode("c") is None
981 and elem.getAttributeNodeNS(
982 "http://xml.python.org/ns", "c") is None
983 and elem.getAttributeNode("p:d").isSameNode(attr)
984 and elem.getAttributeNodeNS(
985 "http://xml.python.org/ns2", "d").isSameNode(attr)
986 and attrmap["p:d"].isSameNode(attr)
987 and attrmap[("http://xml.python.org/ns2", "d")].isSameNode(attr))
988
989 # Rename back to a simple non-NS node
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000990 attr = doc.renameNode(attr, xml.dom.EMPTY_NAMESPACE, "e")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000991 confirm(attr.name == "e"
992 and attr.nodeName == "e"
993 and attr.localName is None
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000994 and attr.namespaceURI == xml.dom.EMPTY_NAMESPACE
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +0000995 and attr.prefix is None
996 and attr.value == "v"
997 and elem.getAttributeNode("a") is None
998 and elem.getAttributeNode("b") is None
999 and elem.getAttributeNode("c") is None
1000 and elem.getAttributeNode("p:d") is None
1001 and elem.getAttributeNodeNS(
1002 "http://xml.python.org/ns", "c") is None
1003 and elem.getAttributeNode("e").isSameNode(attr)
1004 and attrmap["e"].isSameNode(attr))
1005
1006 try:
1007 doc.renameNode(attr, "http://xml.python.org/ns", "xmlns")
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001008 except xml.dom.NamespaceErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001009 pass
1010 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001011 print("expected NamespaceErr")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001012
1013 checkRenameNodeSharedConstraints(doc, attr)
1014 doc.unlink()
1015
1016def testRenameElement():
1017 doc = parseString("<doc/>")
1018 elem = doc.documentElement
1019
1020 # Simple renaming
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001021 elem = doc.renameNode(elem, xml.dom.EMPTY_NAMESPACE, "a")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001022 confirm(elem.tagName == "a"
1023 and elem.nodeName == "a"
1024 and elem.localName is None
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001025 and elem.namespaceURI == xml.dom.EMPTY_NAMESPACE
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001026 and elem.prefix is None
1027 and elem.ownerDocument.isSameNode(doc))
1028
1029 # Rename to have a namespace, no prefix
1030 elem = doc.renameNode(elem, "http://xml.python.org/ns", "b")
1031 confirm(elem.tagName == "b"
1032 and elem.nodeName == "b"
1033 and elem.localName == "b"
1034 and elem.namespaceURI == "http://xml.python.org/ns"
1035 and elem.prefix is None
1036 and elem.ownerDocument.isSameNode(doc))
1037
1038 # Rename to have a namespace, with prefix
1039 elem = doc.renameNode(elem, "http://xml.python.org/ns2", "p:c")
1040 confirm(elem.tagName == "p:c"
1041 and elem.nodeName == "p:c"
1042 and elem.localName == "c"
1043 and elem.namespaceURI == "http://xml.python.org/ns2"
1044 and elem.prefix == "p"
1045 and elem.ownerDocument.isSameNode(doc))
1046
1047 # Rename back to a simple non-NS node
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001048 elem = doc.renameNode(elem, xml.dom.EMPTY_NAMESPACE, "d")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001049 confirm(elem.tagName == "d"
1050 and elem.nodeName == "d"
1051 and elem.localName is None
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001052 and elem.namespaceURI == xml.dom.EMPTY_NAMESPACE
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001053 and elem.prefix is None
1054 and elem.ownerDocument.isSameNode(doc))
1055
1056 checkRenameNodeSharedConstraints(doc, elem)
1057 doc.unlink()
1058
1059def checkRenameNodeSharedConstraints(doc, node):
1060 # Make sure illegal NS usage is detected:
1061 try:
1062 doc.renameNode(node, "http://xml.python.org/ns", "xmlns:foo")
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001063 except xml.dom.NamespaceErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001064 pass
1065 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001066 print("expected NamespaceErr")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001067
1068 doc2 = parseString("<doc/>")
1069 try:
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001070 doc2.renameNode(node, xml.dom.EMPTY_NAMESPACE, "foo")
1071 except xml.dom.WrongDocumentErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001072 pass
1073 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001074 print("expected WrongDocumentErr")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001075
1076def testRenameOther():
1077 # We have to create a comment node explicitly since not all DOM
1078 # builders used with minidom add comments to the DOM.
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001079 doc = xml.dom.minidom.getDOMImplementation().createDocument(
1080 xml.dom.EMPTY_NAMESPACE, "e", None)
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001081 node = doc.createComment("comment")
1082 try:
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001083 doc.renameNode(node, xml.dom.EMPTY_NAMESPACE, "foo")
1084 except xml.dom.NotSupportedErr:
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001085 pass
1086 else:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001087 print("expected NotSupportedErr when renaming comment node")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001088 doc.unlink()
1089
1090def checkWholeText(node, s):
1091 t = node.wholeText
1092 confirm(t == s, "looking for %s, found %s" % (repr(s), repr(t)))
1093
1094def testWholeText():
1095 doc = parseString("<doc>a</doc>")
1096 elem = doc.documentElement
1097 text = elem.childNodes[0]
1098 assert text.nodeType == Node.TEXT_NODE
1099
1100 checkWholeText(text, "a")
1101 elem.appendChild(doc.createTextNode("b"))
1102 checkWholeText(text, "ab")
1103 elem.insertBefore(doc.createCDATASection("c"), text)
1104 checkWholeText(text, "cab")
1105
1106 # make sure we don't cross other nodes
1107 splitter = doc.createComment("comment")
1108 elem.appendChild(splitter)
1109 text2 = doc.createTextNode("d")
1110 elem.appendChild(text2)
1111 checkWholeText(text, "cab")
1112 checkWholeText(text2, "d")
1113
1114 x = doc.createElement("x")
1115 elem.replaceChild(x, splitter)
1116 splitter = x
1117 checkWholeText(text, "cab")
1118 checkWholeText(text2, "d")
1119
1120 x = doc.createProcessingInstruction("y", "z")
1121 elem.replaceChild(x, splitter)
1122 splitter = x
1123 checkWholeText(text, "cab")
1124 checkWholeText(text2, "d")
1125
1126 elem.removeChild(splitter)
1127 checkWholeText(text, "cabd")
1128 checkWholeText(text2, "cabd")
1129
Andrew M. Kuchling841d25e2005-11-22 19:03:16 +00001130def testPatch1094164 ():
1131 doc = parseString("<doc><e/></doc>")
1132 elem = doc.documentElement
1133 e = elem.firstChild
1134 confirm(e.parentNode is elem, "Before replaceChild()")
1135 # Check that replacing a child with itself leaves the tree unchanged
1136 elem.replaceChild(e, e)
1137 confirm(e.parentNode is elem, "After replaceChild()")
Tim Peters536cf992005-12-25 23:18:31 +00001138
1139
1140
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001141def testReplaceWholeText():
1142 def setup():
1143 doc = parseString("<doc>a<e/>d</doc>")
1144 elem = doc.documentElement
1145 text1 = elem.firstChild
1146 text2 = elem.lastChild
1147 splitter = text1.nextSibling
1148 elem.insertBefore(doc.createTextNode("b"), splitter)
1149 elem.insertBefore(doc.createCDATASection("c"), text1)
1150 return doc, elem, text1, splitter, text2
1151
1152 doc, elem, text1, splitter, text2 = setup()
1153 text = text1.replaceWholeText("new content")
1154 checkWholeText(text, "new content")
1155 checkWholeText(text2, "d")
1156 confirm(len(elem.childNodes) == 3)
1157
1158 doc, elem, text1, splitter, text2 = setup()
1159 text = text2.replaceWholeText("new content")
1160 checkWholeText(text, "new content")
1161 checkWholeText(text1, "cab")
1162 confirm(len(elem.childNodes) == 5)
1163
1164 doc, elem, text1, splitter, text2 = setup()
1165 text = text1.replaceWholeText("")
1166 checkWholeText(text2, "d")
1167 confirm(text is None
1168 and len(elem.childNodes) == 2)
1169
1170def testSchemaType():
1171 doc = parseString(
1172 "<!DOCTYPE doc [\n"
1173 " <!ENTITY e1 SYSTEM 'http://xml.python.org/e1'>\n"
1174 " <!ENTITY e2 SYSTEM 'http://xml.python.org/e2'>\n"
1175 " <!ATTLIST doc id ID #IMPLIED \n"
1176 " ref IDREF #IMPLIED \n"
1177 " refs IDREFS #IMPLIED \n"
1178 " enum (a|b) #IMPLIED \n"
1179 " ent ENTITY #IMPLIED \n"
1180 " ents ENTITIES #IMPLIED \n"
1181 " nm NMTOKEN #IMPLIED \n"
1182 " nms NMTOKENS #IMPLIED \n"
1183 " text CDATA #IMPLIED \n"
1184 " >\n"
1185 "]><doc id='name' notid='name' text='splat!' enum='b'"
1186 " ref='name' refs='name name' ent='e1' ents='e1 e2'"
1187 " nm='123' nms='123 abc' />")
1188 elem = doc.documentElement
1189 # We don't want to rely on any specific loader at this point, so
1190 # just make sure we can get to all the names, and that the
1191 # DTD-based namespace is right. The names can vary by loader
1192 # since each supports a different level of DTD information.
1193 t = elem.schemaType
1194 confirm(t.name is None
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001195 and t.namespace == xml.dom.EMPTY_NAMESPACE)
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001196 names = "id notid text enum ref refs ent ents nm nms".split()
1197 for name in names:
1198 a = elem.getAttributeNode(name)
1199 t = a.schemaType
1200 confirm(hasattr(t, "name")
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001201 and t.namespace == xml.dom.EMPTY_NAMESPACE)
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001202
1203def testSetIdAttribute():
1204 doc = parseString("<doc a1='v' a2='w'/>")
1205 e = doc.documentElement
1206 a1 = e.getAttributeNode("a1")
1207 a2 = e.getAttributeNode("a2")
1208 confirm(doc.getElementById("v") is None
1209 and not a1.isId
1210 and not a2.isId)
1211 e.setIdAttribute("a1")
1212 confirm(e.isSameNode(doc.getElementById("v"))
1213 and a1.isId
1214 and not a2.isId)
1215 e.setIdAttribute("a2")
1216 confirm(e.isSameNode(doc.getElementById("v"))
1217 and e.isSameNode(doc.getElementById("w"))
1218 and a1.isId
1219 and a2.isId)
1220 # replace the a1 node; the new node should *not* be an ID
1221 a3 = doc.createAttribute("a1")
1222 a3.value = "v"
1223 e.setAttributeNode(a3)
1224 confirm(doc.getElementById("v") is None
1225 and e.isSameNode(doc.getElementById("w"))
1226 and not a1.isId
1227 and a2.isId
1228 and not a3.isId)
Georg Brandl7eb4b7d2005-07-22 21:49:32 +00001229 # renaming an attribute should not affect its ID-ness:
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001230 doc.renameNode(a2, xml.dom.EMPTY_NAMESPACE, "an")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001231 confirm(e.isSameNode(doc.getElementById("w"))
1232 and a2.isId)
1233
1234def testSetIdAttributeNS():
1235 NS1 = "http://xml.python.org/ns1"
1236 NS2 = "http://xml.python.org/ns2"
1237 doc = parseString("<doc"
1238 " xmlns:ns1='" + NS1 + "'"
1239 " xmlns:ns2='" + NS2 + "'"
1240 " ns1:a1='v' ns2:a2='w'/>")
1241 e = doc.documentElement
1242 a1 = e.getAttributeNodeNS(NS1, "a1")
1243 a2 = e.getAttributeNodeNS(NS2, "a2")
1244 confirm(doc.getElementById("v") is None
1245 and not a1.isId
1246 and not a2.isId)
1247 e.setIdAttributeNS(NS1, "a1")
1248 confirm(e.isSameNode(doc.getElementById("v"))
1249 and a1.isId
1250 and not a2.isId)
1251 e.setIdAttributeNS(NS2, "a2")
1252 confirm(e.isSameNode(doc.getElementById("v"))
1253 and e.isSameNode(doc.getElementById("w"))
1254 and a1.isId
1255 and a2.isId)
1256 # replace the a1 node; the new node should *not* be an ID
1257 a3 = doc.createAttributeNS(NS1, "a1")
1258 a3.value = "v"
1259 e.setAttributeNode(a3)
1260 confirm(e.isSameNode(doc.getElementById("w")))
1261 confirm(not a1.isId)
1262 confirm(a2.isId)
1263 confirm(not a3.isId)
1264 confirm(doc.getElementById("v") is None)
Georg Brandl7eb4b7d2005-07-22 21:49:32 +00001265 # renaming an attribute should not affect its ID-ness:
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001266 doc.renameNode(a2, xml.dom.EMPTY_NAMESPACE, "an")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001267 confirm(e.isSameNode(doc.getElementById("w"))
1268 and a2.isId)
1269
1270def testSetIdAttributeNode():
1271 NS1 = "http://xml.python.org/ns1"
1272 NS2 = "http://xml.python.org/ns2"
1273 doc = parseString("<doc"
1274 " xmlns:ns1='" + NS1 + "'"
1275 " xmlns:ns2='" + NS2 + "'"
1276 " ns1:a1='v' ns2:a2='w'/>")
1277 e = doc.documentElement
1278 a1 = e.getAttributeNodeNS(NS1, "a1")
1279 a2 = e.getAttributeNodeNS(NS2, "a2")
1280 confirm(doc.getElementById("v") is None
1281 and not a1.isId
1282 and not a2.isId)
1283 e.setIdAttributeNode(a1)
1284 confirm(e.isSameNode(doc.getElementById("v"))
1285 and a1.isId
1286 and not a2.isId)
1287 e.setIdAttributeNode(a2)
1288 confirm(e.isSameNode(doc.getElementById("v"))
1289 and e.isSameNode(doc.getElementById("w"))
1290 and a1.isId
1291 and a2.isId)
1292 # replace the a1 node; the new node should *not* be an ID
1293 a3 = doc.createAttributeNS(NS1, "a1")
1294 a3.value = "v"
1295 e.setAttributeNode(a3)
1296 confirm(e.isSameNode(doc.getElementById("w")))
1297 confirm(not a1.isId)
1298 confirm(a2.isId)
1299 confirm(not a3.isId)
1300 confirm(doc.getElementById("v") is None)
Georg Brandl7eb4b7d2005-07-22 21:49:32 +00001301 # renaming an attribute should not affect its ID-ness:
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001302 doc.renameNode(a2, xml.dom.EMPTY_NAMESPACE, "an")
Martin v. Löwisaa5af8d2003-01-25 21:39:09 +00001303 confirm(e.isSameNode(doc.getElementById("w"))
1304 and a2.isId)
1305
1306def testPickledDocument():
1307 doc = parseString("<?xml version='1.0' encoding='us-ascii'?>\n"
1308 "<!DOCTYPE doc PUBLIC 'http://xml.python.org/public'"
1309 " 'http://xml.python.org/system' [\n"
1310 " <!ELEMENT e EMPTY>\n"
1311 " <!ENTITY ent SYSTEM 'http://xml.python.org/entity'>\n"
1312 "]><doc attr='value'> text\n"
1313 "<?pi sample?> <!-- comment --> <e/> </doc>")
1314 s = pickle.dumps(doc)
1315 doc2 = pickle.loads(s)
1316 stack = [(doc, doc2)]
1317 while stack:
1318 n1, n2 = stack.pop()
1319 confirm(n1.nodeType == n2.nodeType
1320 and len(n1.childNodes) == len(n2.childNodes)
1321 and n1.nodeName == n2.nodeName
1322 and not n1.isSameNode(n2)
1323 and not n2.isSameNode(n1))
1324 if n1.nodeType == Node.DOCUMENT_TYPE_NODE:
1325 len(n1.entities)
1326 len(n2.entities)
1327 len(n1.notations)
1328 len(n2.notations)
1329 confirm(len(n1.entities) == len(n2.entities)
1330 and len(n1.notations) == len(n2.notations))
1331 for i in range(len(n1.notations)):
1332 no1 = n1.notations.item(i)
1333 no2 = n1.notations.item(i)
1334 confirm(no1.name == no2.name
1335 and no1.publicId == no2.publicId
1336 and no1.systemId == no2.systemId)
1337 statck.append((no1, no2))
1338 for i in range(len(n1.entities)):
1339 e1 = n1.entities.item(i)
1340 e2 = n2.entities.item(i)
1341 confirm(e1.notationName == e2.notationName
1342 and e1.publicId == e2.publicId
1343 and e1.systemId == e2.systemId)
1344 stack.append((e1, e2))
1345 if n1.nodeType != Node.DOCUMENT_NODE:
1346 confirm(n1.ownerDocument.isSameNode(doc)
1347 and n2.ownerDocument.isSameNode(doc2))
1348 for i in range(len(n1.childNodes)):
1349 stack.append((n1.childNodes[i], n2.childNodes[i]))
1350
1351
Lars Gustäbelf27f5ab2000-10-11 22:36:00 +00001352# --- MAIN PROGRAM
Fred Drake004d5e62000-10-23 17:22:08 +00001353
Jeremy Hylton3b0c6002000-10-12 17:31:36 +00001354names = globals().keys()
Paul Prescod7993bcc2000-07-01 14:54:16 +00001355names.sort()
Paul Prescod10d27662000-09-18 19:07:26 +00001356
Fred Drakeacfb3f62001-02-01 18:11:29 +00001357failed = []
Paul Prescod10d27662000-09-18 19:07:26 +00001358
Fred Drake6dd7d072002-09-12 17:03:02 +00001359try:
1360 Node.allnodes
1361except AttributeError:
Martin v. Löwisfd6aaa12003-01-25 22:02:52 +00001362 # We don't actually have the minidom from the standard library,
Fred Drake6dd7d072002-09-12 17:03:02 +00001363 # but are picking up the PyXML version from site-packages.
1364 def check_allnodes():
1365 pass
1366else:
1367 def check_allnodes():
1368 confirm(len(Node.allnodes) == 0,
1369 "assertion: len(Node.allnodes) == 0")
1370 if len(Node.allnodes):
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001371 print("Garbage left over:")
Fred Drake6dd7d072002-09-12 17:03:02 +00001372 if verbose:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001373 print(Node.allnodes.items()[0:10])
Fred Drake6dd7d072002-09-12 17:03:02 +00001374 else:
1375 # Don't print specific nodes if repeatable results
1376 # are needed
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001377 print(len(Node.allnodes))
Fred Drake6dd7d072002-09-12 17:03:02 +00001378 Node.allnodes = {}
1379
Paul Prescod7993bcc2000-07-01 14:54:16 +00001380for name in names:
Jeremy Hylton3b0c6002000-10-12 17:31:36 +00001381 if name.startswith("test"):
1382 func = globals()[name]
Paul Prescod7993bcc2000-07-01 14:54:16 +00001383 try:
1384 func()
Fred Drake6dd7d072002-09-12 17:03:02 +00001385 check_allnodes()
Fred Drakeacfb3f62001-02-01 18:11:29 +00001386 except:
1387 failed.append(name)
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001388 print("Test Failed: ", name)
Fred Drake1703cf62000-12-15 21:31:59 +00001389 sys.stdout.flush()
Jeremy Hylton3b0c6002000-10-12 17:31:36 +00001390 traceback.print_exception(*sys.exc_info())
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001391 print(repr(sys.exc_info()[1]))
Jeremy Hylton3b0c6002000-10-12 17:31:36 +00001392 Node.allnodes = {}
Paul Prescod10d27662000-09-18 19:07:26 +00001393
Fred Drakeacfb3f62001-02-01 18:11:29 +00001394if failed:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001395 print("\n\n\n**** Check for failures in these tests:")
Fred Drakeacfb3f62001-02-01 18:11:29 +00001396 for name in failed:
Guido van Rossumbe19ed72007-02-09 05:37:30 +00001397 print(" " + name)