Viewing file: test_xml_etree_c.py (8.48 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
# xml.etree test for cElementTree import io import struct from test import support from test.support.import_helper import import_fresh_module import types import unittest
cET = import_fresh_module('xml.etree.ElementTree', fresh=['_elementtree']) cET_alias = import_fresh_module('xml.etree.cElementTree', fresh=['_elementtree', 'xml.etree'], deprecated=True)
@unittest.skipUnless(cET, 'requires _elementtree') class MiscTests(unittest.TestCase): # Issue #8651. @support.bigmemtest(size=support._2G + 100, memuse=1, dry_run=False) def test_length_overflow(self, size): data = b'x' * size parser = cET.XMLParser() try: self.assertRaises(OverflowError, parser.feed, data) finally: data = None
def test_del_attribute(self): element = cET.Element('tag')
element.tag = 'TAG' with self.assertRaises(AttributeError): del element.tag self.assertEqual(element.tag, 'TAG')
with self.assertRaises(AttributeError): del element.text self.assertIsNone(element.text) element.text = 'TEXT' with self.assertRaises(AttributeError): del element.text self.assertEqual(element.text, 'TEXT')
with self.assertRaises(AttributeError): del element.tail self.assertIsNone(element.tail) element.tail = 'TAIL' with self.assertRaises(AttributeError): del element.tail self.assertEqual(element.tail, 'TAIL')
with self.assertRaises(AttributeError): del element.attrib self.assertEqual(element.attrib, {}) element.attrib = {'A': 'B', 'C': 'D'} with self.assertRaises(AttributeError): del element.attrib self.assertEqual(element.attrib, {'A': 'B', 'C': 'D'})
def test_trashcan(self): # If this test fails, it will most likely die via segfault. e = root = cET.Element('root') for i in range(200000): e = cET.SubElement(e, 'x') del e del root support.gc_collect()
def test_parser_ref_cycle(self): # bpo-31499: xmlparser_dealloc() crashed with a segmentation fault when # xmlparser_gc_clear() was called previously by the garbage collector, # when the parser was part of a reference cycle.
def parser_ref_cycle(): parser = cET.XMLParser() # Create a reference cycle using an exception to keep the frame # alive, so the parser will be destroyed by the garbage collector try: raise ValueError except ValueError as exc: err = exc
# Create a parser part of reference cycle parser_ref_cycle() # Trigger an explicit garbage collection to break the reference cycle # and so destroy the parser support.gc_collect()
def test_bpo_31728(self): # A crash or an assertion failure shouldn't happen, in case garbage # collection triggers a call to clear() or a reading of text or tail, # while a setter or clear() or __setstate__() is already running. elem = cET.Element('elem') class X: def __del__(self): elem.text elem.tail elem.clear()
elem.text = X() elem.clear() # shouldn't crash
elem.tail = X() elem.clear() # shouldn't crash
elem.text = X() elem.text = X() # shouldn't crash elem.clear()
elem.tail = X() elem.tail = X() # shouldn't crash elem.clear()
elem.text = X() elem.__setstate__({'tag': 42}) # shouldn't cause an assertion failure elem.clear()
elem.tail = X() elem.__setstate__({'tag': 42}) # shouldn't cause an assertion failure
@support.cpython_only def test_uninitialized_parser(self): # The interpreter shouldn't crash in case of calling methods or # accessing attributes of uninitialized XMLParser objects. parser = cET.XMLParser.__new__(cET.XMLParser) self.assertRaises(ValueError, parser.close) self.assertRaises(ValueError, parser.feed, 'foo') class MockFile: def read(*args): return '' self.assertRaises(ValueError, parser._parse_whole, MockFile()) self.assertRaises(ValueError, parser._setevents, None) self.assertIsNone(parser.entity) self.assertIsNone(parser.target)
def test_setstate_leaks(self): # Test reference leaks elem = cET.Element.__new__(cET.Element) for i in range(100): elem.__setstate__({'tag': 'foo', 'attrib': {'bar': 42}, '_children': [cET.Element('child')], 'text': 'text goes here', 'tail': 'opposite of head'})
self.assertEqual(elem.tag, 'foo') self.assertEqual(elem.text, 'text goes here') self.assertEqual(elem.tail, 'opposite of head') self.assertEqual(list(elem.attrib.items()), [('bar', 42)]) self.assertEqual(len(elem), 1) self.assertEqual(elem[0].tag, 'child')
def test_iterparse_leaks(self): # Test reference leaks in TreeBuilder (issue #35502). # The test is written to be executed in the hunting reference leaks # mode. XML = '<a></a></b>' parser = cET.iterparse(io.StringIO(XML)) next(parser) del parser support.gc_collect()
def test_xmlpullparser_leaks(self): # Test reference leaks in TreeBuilder (issue #35502). # The test is written to be executed in the hunting reference leaks # mode. XML = '<a></a></b>' parser = cET.XMLPullParser() parser.feed(XML) del parser support.gc_collect()
def test_dict_disappearing_during_get_item(self): # test fix for seg fault reported in issue 27946 class X: def __hash__(self): e.attrib = {} # this frees e->extra->attrib [{i: i} for i in range(1000)] # exhaust the dict keys cache return 13
e = cET.Element("elem", {1: 2}) r = e.get(X()) self.assertIsNone(r)
@unittest.skipUnless(cET, 'requires _elementtree') class TestAliasWorking(unittest.TestCase): # Test that the cET alias module is alive def test_alias_working(self): e = cET_alias.Element('foo') self.assertEqual(e.tag, 'foo')
@unittest.skipUnless(cET, 'requires _elementtree') @support.cpython_only class TestAcceleratorImported(unittest.TestCase): # Test that the C accelerator was imported, as expected def test_correct_import_cET(self): # SubElement is a function so it retains _elementtree as its module. self.assertEqual(cET.SubElement.__module__, '_elementtree')
def test_correct_import_cET_alias(self): self.assertEqual(cET_alias.SubElement.__module__, '_elementtree')
def test_parser_comes_from_C(self): # The type of methods defined in Python code is types.FunctionType, # while the type of methods defined inside _elementtree is # <class 'wrapper_descriptor'> self.assertNotIsInstance(cET.Element.__init__, types.FunctionType)
@unittest.skipUnless(cET, 'requires _elementtree') @support.cpython_only class SizeofTest(unittest.TestCase): def setUp(self): self.elementsize = support.calcobjsize('5P') # extra self.extra = struct.calcsize('PnnP4P')
check_sizeof = support.check_sizeof
def test_element(self): e = cET.Element('a') self.check_sizeof(e, self.elementsize)
def test_element_with_attrib(self): e = cET.Element('a', href='about:') self.check_sizeof(e, self.elementsize + self.extra)
def test_element_with_children(self): e = cET.Element('a') for i in range(5): cET.SubElement(e, 'span') # should have space for 8 children now self.check_sizeof(e, self.elementsize + self.extra + struct.calcsize('8P'))
def test_main(): from test import test_xml_etree
# Run the tests specific to the C implementation support.run_unittest( MiscTests, TestAliasWorking, TestAcceleratorImported, SizeofTest, )
# Run the same test suite as the Python module test_xml_etree.test_main(module=cET)
if __name__ == '__main__': test_main()
|