mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 07:31:38 +00:00 
			
		
		
		
	gh-128302: Fix bugs in xml.dom.xmlbuilder (GH-128284)
* Allow DOMParser.parse() to correctly handle DOMInputSource instances
  that only have a systemId attribute set.
* Fix DOMEntityResolver.resolveEntity(), which was broken by the
  Python 3.0 transition.
* Add Lib/test/test_xml_dom_xmlbuilder.py with few tests.
(cherry picked from commit 6ea04da270)
Co-authored-by: Stephen Morton <git@tungol.org>
		
	
			
		
			
				
	
	
		
			88 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			88 lines
		
	
	
	
		
			3.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import io
 | 
						|
import unittest
 | 
						|
from http import client
 | 
						|
from test.test_httplib import FakeSocket
 | 
						|
from unittest import mock
 | 
						|
from xml.dom import getDOMImplementation, minidom, xmlbuilder
 | 
						|
 | 
						|
SMALL_SAMPLE = b"""<?xml version="1.0"?>
 | 
						|
<html xmlns="http://www.w3.org/1999/xhtml" xmlns:xdc="http://www.xml.com/books">
 | 
						|
<!-- A comment -->
 | 
						|
<title>Introduction to XSL</title>
 | 
						|
<hr/>
 | 
						|
<p><xdc:author xdc:attrib="prefixed attribute" attrib="other attrib">A. Namespace</xdc:author></p>
 | 
						|
</html>"""
 | 
						|
 | 
						|
 | 
						|
class XMLBuilderTest(unittest.TestCase):
 | 
						|
    def test_entity_resolver(self):
 | 
						|
        body = (
 | 
						|
            b"HTTP/1.1 200 OK\r\nContent-Type: text/xml; charset=utf-8\r\n\r\n"
 | 
						|
            + SMALL_SAMPLE
 | 
						|
        )
 | 
						|
 | 
						|
        sock = FakeSocket(body)
 | 
						|
        response = client.HTTPResponse(sock)
 | 
						|
        response.begin()
 | 
						|
        attrs = {"open.return_value": response}
 | 
						|
        opener = mock.Mock(**attrs)
 | 
						|
 | 
						|
        resolver = xmlbuilder.DOMEntityResolver()
 | 
						|
 | 
						|
        with mock.patch("urllib.request.build_opener") as mock_build:
 | 
						|
            mock_build.return_value = opener
 | 
						|
            source = resolver.resolveEntity(None, "http://example.com/2000/svg")
 | 
						|
 | 
						|
        self.assertIsInstance(source, xmlbuilder.DOMInputSource)
 | 
						|
        self.assertIsNone(source.publicId)
 | 
						|
        self.assertEqual(source.systemId, "http://example.com/2000/svg")
 | 
						|
        self.assertEqual(source.baseURI, "http://example.com/2000/")
 | 
						|
        self.assertEqual(source.encoding, "utf-8")
 | 
						|
        self.assertIs(source.byteStream, response)
 | 
						|
 | 
						|
        self.assertIsNone(source.characterStream)
 | 
						|
        self.assertIsNone(source.stringData)
 | 
						|
 | 
						|
    def test_builder(self):
 | 
						|
        imp = getDOMImplementation()
 | 
						|
        self.assertIsInstance(imp, xmlbuilder.DOMImplementationLS)
 | 
						|
 | 
						|
        builder = imp.createDOMBuilder(imp.MODE_SYNCHRONOUS, None)
 | 
						|
        self.assertIsInstance(builder, xmlbuilder.DOMBuilder)
 | 
						|
 | 
						|
    def test_parse_uri(self):
 | 
						|
        body = (
 | 
						|
            b"HTTP/1.1 200 OK\r\nContent-Type: text/xml; charset=utf-8\r\n\r\n"
 | 
						|
            + SMALL_SAMPLE
 | 
						|
        )
 | 
						|
 | 
						|
        sock = FakeSocket(body)
 | 
						|
        response = client.HTTPResponse(sock)
 | 
						|
        response.begin()
 | 
						|
        attrs = {"open.return_value": response}
 | 
						|
        opener = mock.Mock(**attrs)
 | 
						|
 | 
						|
        with mock.patch("urllib.request.build_opener") as mock_build:
 | 
						|
            mock_build.return_value = opener
 | 
						|
 | 
						|
            imp = getDOMImplementation()
 | 
						|
            builder = imp.createDOMBuilder(imp.MODE_SYNCHRONOUS, None)
 | 
						|
            document = builder.parseURI("http://example.com/2000/svg")
 | 
						|
 | 
						|
        self.assertIsInstance(document, minidom.Document)
 | 
						|
        self.assertEqual(len(document.childNodes), 1)
 | 
						|
 | 
						|
    def test_parse_with_systemId(self):
 | 
						|
        response = io.BytesIO(SMALL_SAMPLE)
 | 
						|
 | 
						|
        with mock.patch("urllib.request.urlopen") as mock_open:
 | 
						|
            mock_open.return_value = response
 | 
						|
 | 
						|
            imp = getDOMImplementation()
 | 
						|
            source = imp.createDOMInputSource()
 | 
						|
            builder = imp.createDOMBuilder(imp.MODE_SYNCHRONOUS, None)
 | 
						|
            source.systemId = "http://example.com/2000/svg"
 | 
						|
            document = builder.parse(source)
 | 
						|
 | 
						|
        self.assertIsInstance(document, minidom.Document)
 | 
						|
        self.assertEqual(len(document.childNodes), 1)
 |