Commit 43536e9e authored by Serhiy Storchaka's avatar Serhiy Storchaka

Issue #17089: Expat parser now correctly works with string input not only when

an internal XML encoding is UTF-8 or US-ASCII.  It now accepts bytes and
strings larger than 2 GiB.
parent 95b7110a
...@@ -52,6 +52,7 @@ data = b'''\ ...@@ -52,6 +52,7 @@ data = b'''\
<sub2><![CDATA[contents of CDATA section]]></sub2> <sub2><![CDATA[contents of CDATA section]]></sub2>
&external_entity; &external_entity;
&skipped_entity; &skipped_entity;
\xb5
</root> </root>
''' '''
...@@ -195,13 +196,13 @@ class ParseTest(unittest.TestCase): ...@@ -195,13 +196,13 @@ class ParseTest(unittest.TestCase):
"End element: 'sub2'", "End element: 'sub2'",
"External entity ref: (None, 'entity.file', None)", "External entity ref: (None, 'entity.file', None)",
('Skipped entity', ('skipped_entity', 0)), ('Skipped entity', ('skipped_entity', 0)),
"Character data: '\xb5'",
"End element: 'root'", "End element: 'root'",
] ]
for operation, expected_operation in zip(operations, expected_operations): for operation, expected_operation in zip(operations, expected_operations):
self.assertEqual(operation, expected_operation) self.assertEqual(operation, expected_operation)
def test_unicode(self): def test_parse_bytes(self):
# Try the parse again, this time producing Unicode output
out = self.Outputter() out = self.Outputter()
parser = expat.ParserCreate(namespace_separator='!') parser = expat.ParserCreate(namespace_separator='!')
self._hookup_callbacks(parser, out) self._hookup_callbacks(parser, out)
...@@ -213,6 +214,16 @@ class ParseTest(unittest.TestCase): ...@@ -213,6 +214,16 @@ class ParseTest(unittest.TestCase):
# Issue #6697. # Issue #6697.
self.assertRaises(AttributeError, getattr, parser, '\uD800') self.assertRaises(AttributeError, getattr, parser, '\uD800')
def test_parse_str(self):
out = self.Outputter()
parser = expat.ParserCreate(namespace_separator='!')
self._hookup_callbacks(parser, out)
parser.Parse(data.decode('iso-8859-1'), 1)
operations = out.out
self._verify_parse_output(operations)
def test_parse_file(self): def test_parse_file(self):
# Try parsing a file # Try parsing a file
out = self.Outputter() out = self.Outputter()
...@@ -269,7 +280,7 @@ class InterningTest(unittest.TestCase): ...@@ -269,7 +280,7 @@ class InterningTest(unittest.TestCase):
L.append(name) L.append(name)
p.StartElementHandler = collector p.StartElementHandler = collector
p.EndElementHandler = collector p.EndElementHandler = collector
p.Parse("<e> <e/> <e></e> </e>", 1) p.Parse(b"<e> <e/> <e></e> </e>", 1)
tag = L[0] tag = L[0]
self.assertEqual(len(L), 6) self.assertEqual(len(L), 6)
for entry in L: for entry in L:
...@@ -285,7 +296,7 @@ class InterningTest(unittest.TestCase): ...@@ -285,7 +296,7 @@ class InterningTest(unittest.TestCase):
def ExternalEntityRefHandler(self, context, base, sysId, pubId): def ExternalEntityRefHandler(self, context, base, sysId, pubId):
external_parser = self.parser.ExternalEntityParserCreate("") external_parser = self.parser.ExternalEntityParserCreate("")
self.parser_result = external_parser.Parse("", 1) self.parser_result = external_parser.Parse(b"", 1)
return 1 return 1
parser = expat.ParserCreate(namespace_separator='!') parser = expat.ParserCreate(namespace_separator='!')
...@@ -336,7 +347,7 @@ class BufferTextTest(unittest.TestCase): ...@@ -336,7 +347,7 @@ class BufferTextTest(unittest.TestCase):
def test_buffering_enabled(self): def test_buffering_enabled(self):
# Make sure buffering is turned on # Make sure buffering is turned on
self.assertTrue(self.parser.buffer_text) self.assertTrue(self.parser.buffer_text)
self.parser.Parse("<a>1<b/>2<c/>3</a>", 1) self.parser.Parse(b"<a>1<b/>2<c/>3</a>", 1)
self.assertEqual(self.stuff, ['123'], self.assertEqual(self.stuff, ['123'],
"buffered text not properly collapsed") "buffered text not properly collapsed")
...@@ -344,39 +355,39 @@ class BufferTextTest(unittest.TestCase): ...@@ -344,39 +355,39 @@ class BufferTextTest(unittest.TestCase):
# XXX This test exposes more detail of Expat's text chunking than we # XXX This test exposes more detail of Expat's text chunking than we
# XXX like, but it tests what we need to concisely. # XXX like, but it tests what we need to concisely.
self.setHandlers(["StartElementHandler"]) self.setHandlers(["StartElementHandler"])
self.parser.Parse("<a>1<b buffer-text='no'/>2\n3<c buffer-text='yes'/>4\n5</a>", 1) self.parser.Parse(b"<a>1<b buffer-text='no'/>2\n3<c buffer-text='yes'/>4\n5</a>", 1)
self.assertEqual(self.stuff, self.assertEqual(self.stuff,
["<a>", "1", "<b>", "2", "\n", "3", "<c>", "4\n5"], ["<a>", "1", "<b>", "2", "\n", "3", "<c>", "4\n5"],
"buffering control not reacting as expected") "buffering control not reacting as expected")
def test2(self): def test2(self):
self.parser.Parse("<a>1<b/>&lt;2&gt;<c/>&#32;\n&#x20;3</a>", 1) self.parser.Parse(b"<a>1<b/>&lt;2&gt;<c/>&#32;\n&#x20;3</a>", 1)
self.assertEqual(self.stuff, ["1<2> \n 3"], self.assertEqual(self.stuff, ["1<2> \n 3"],
"buffered text not properly collapsed") "buffered text not properly collapsed")
def test3(self): def test3(self):
self.setHandlers(["StartElementHandler"]) self.setHandlers(["StartElementHandler"])
self.parser.Parse("<a>1<b/>2<c/>3</a>", 1) self.parser.Parse(b"<a>1<b/>2<c/>3</a>", 1)
self.assertEqual(self.stuff, ["<a>", "1", "<b>", "2", "<c>", "3"], self.assertEqual(self.stuff, ["<a>", "1", "<b>", "2", "<c>", "3"],
"buffered text not properly split") "buffered text not properly split")
def test4(self): def test4(self):
self.setHandlers(["StartElementHandler", "EndElementHandler"]) self.setHandlers(["StartElementHandler", "EndElementHandler"])
self.parser.CharacterDataHandler = None self.parser.CharacterDataHandler = None
self.parser.Parse("<a>1<b/>2<c/>3</a>", 1) self.parser.Parse(b"<a>1<b/>2<c/>3</a>", 1)
self.assertEqual(self.stuff, self.assertEqual(self.stuff,
["<a>", "<b>", "</b>", "<c>", "</c>", "</a>"]) ["<a>", "<b>", "</b>", "<c>", "</c>", "</a>"])
def test5(self): def test5(self):
self.setHandlers(["StartElementHandler", "EndElementHandler"]) self.setHandlers(["StartElementHandler", "EndElementHandler"])
self.parser.Parse("<a>1<b></b>2<c/>3</a>", 1) self.parser.Parse(b"<a>1<b></b>2<c/>3</a>", 1)
self.assertEqual(self.stuff, self.assertEqual(self.stuff,
["<a>", "1", "<b>", "</b>", "2", "<c>", "</c>", "3", "</a>"]) ["<a>", "1", "<b>", "</b>", "2", "<c>", "</c>", "3", "</a>"])
def test6(self): def test6(self):
self.setHandlers(["CommentHandler", "EndElementHandler", self.setHandlers(["CommentHandler", "EndElementHandler",
"StartElementHandler"]) "StartElementHandler"])
self.parser.Parse("<a>1<b/>2<c></c>345</a> ", 1) self.parser.Parse(b"<a>1<b/>2<c></c>345</a> ", 1)
self.assertEqual(self.stuff, self.assertEqual(self.stuff,
["<a>", "1", "<b>", "</b>", "2", "<c>", "</c>", "345", "</a>"], ["<a>", "1", "<b>", "</b>", "2", "<c>", "</c>", "345", "</a>"],
"buffered text not properly split") "buffered text not properly split")
...@@ -384,7 +395,7 @@ class BufferTextTest(unittest.TestCase): ...@@ -384,7 +395,7 @@ class BufferTextTest(unittest.TestCase):
def test7(self): def test7(self):
self.setHandlers(["CommentHandler", "EndElementHandler", self.setHandlers(["CommentHandler", "EndElementHandler",
"StartElementHandler"]) "StartElementHandler"])
self.parser.Parse("<a>1<b/>2<c></c>3<!--abc-->4<!--def-->5</a> ", 1) self.parser.Parse(b"<a>1<b/>2<c></c>3<!--abc-->4<!--def-->5</a> ", 1)
self.assertEqual(self.stuff, self.assertEqual(self.stuff,
["<a>", "1", "<b>", "</b>", "2", "<c>", "</c>", "3", ["<a>", "1", "<b>", "</b>", "2", "<c>", "</c>", "3",
"<!--abc-->", "4", "<!--def-->", "5", "</a>"], "<!--abc-->", "4", "<!--def-->", "5", "</a>"],
...@@ -400,7 +411,7 @@ class HandlerExceptionTest(unittest.TestCase): ...@@ -400,7 +411,7 @@ class HandlerExceptionTest(unittest.TestCase):
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.StartElementHandler = self.StartElementHandler parser.StartElementHandler = self.StartElementHandler
try: try:
parser.Parse("<a><b><c/></b></a>", 1) parser.Parse(b"<a><b><c/></b></a>", 1)
self.fail() self.fail()
except RuntimeError as e: except RuntimeError as e:
self.assertEqual(e.args[0], 'a', self.assertEqual(e.args[0], 'a',
...@@ -436,7 +447,7 @@ class PositionTest(unittest.TestCase): ...@@ -436,7 +447,7 @@ class PositionTest(unittest.TestCase):
self.expected_list = [('s', 0, 1, 0), ('s', 5, 2, 1), ('s', 11, 3, 2), self.expected_list = [('s', 0, 1, 0), ('s', 5, 2, 1), ('s', 11, 3, 2),
('e', 15, 3, 6), ('e', 17, 4, 1), ('e', 22, 5, 0)] ('e', 15, 3, 6), ('e', 17, 4, 1), ('e', 22, 5, 0)]
xml = '<a>\n <b>\n <c/>\n </b>\n</a>' xml = b'<a>\n <b>\n <c/>\n </b>\n</a>'
self.parser.Parse(xml, 1) self.parser.Parse(xml, 1)
...@@ -457,7 +468,7 @@ class sf1296433Test(unittest.TestCase): ...@@ -457,7 +468,7 @@ class sf1296433Test(unittest.TestCase):
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.CharacterDataHandler = handler parser.CharacterDataHandler = handler
self.assertRaises(Exception, parser.Parse, xml) self.assertRaises(Exception, parser.Parse, xml.encode('iso8859'))
class ChardataBufferTest(unittest.TestCase): class ChardataBufferTest(unittest.TestCase):
""" """
...@@ -480,8 +491,8 @@ class ChardataBufferTest(unittest.TestCase): ...@@ -480,8 +491,8 @@ class ChardataBufferTest(unittest.TestCase):
self.assertRaises(ValueError, f, 0) self.assertRaises(ValueError, f, 0)
def test_unchanged_size(self): def test_unchanged_size(self):
xml1 = ("<?xml version='1.0' encoding='iso8859'?><s>%s" % ('a' * 512)) xml1 = b"<?xml version='1.0' encoding='iso8859'?><s>" + b'a' * 512
xml2 = 'a'*512 + '</s>' xml2 = b'a'*512 + b'</s>'
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.CharacterDataHandler = self.counting_handler parser.CharacterDataHandler = self.counting_handler
parser.buffer_size = 512 parser.buffer_size = 512
...@@ -503,9 +514,9 @@ class ChardataBufferTest(unittest.TestCase): ...@@ -503,9 +514,9 @@ class ChardataBufferTest(unittest.TestCase):
def test_disabling_buffer(self): def test_disabling_buffer(self):
xml1 = "<?xml version='1.0' encoding='iso8859'?><a>%s" % ('a' * 512) xml1 = b"<?xml version='1.0' encoding='iso8859'?><a>" + b'a' * 512
xml2 = ('b' * 1024) xml2 = b'b' * 1024
xml3 = "%s</a>" % ('c' * 1024) xml3 = b'c' * 1024 + b'</a>';
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.CharacterDataHandler = self.counting_handler parser.CharacterDataHandler = self.counting_handler
parser.buffer_text = 1 parser.buffer_text = 1
...@@ -532,16 +543,11 @@ class ChardataBufferTest(unittest.TestCase): ...@@ -532,16 +543,11 @@ class ChardataBufferTest(unittest.TestCase):
parser.Parse(xml3, 1) parser.Parse(xml3, 1)
self.assertEqual(self.n, 12) self.assertEqual(self.n, 12)
def make_document(self, bytes):
return ("<?xml version='1.0'?><tag>" + bytes * 'a' + '</tag>')
def counting_handler(self, text): def counting_handler(self, text):
self.n += 1 self.n += 1
def small_buffer_test(self, buffer_len): def small_buffer_test(self, buffer_len):
xml = "<?xml version='1.0' encoding='iso8859'?><s>%s</s>" % ('a' * buffer_len) xml = b"<?xml version='1.0' encoding='iso8859'?><s>" + b'a' * buffer_len + b'</s>'
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.CharacterDataHandler = self.counting_handler parser.CharacterDataHandler = self.counting_handler
parser.buffer_size = 1024 parser.buffer_size = 1024
...@@ -552,8 +558,8 @@ class ChardataBufferTest(unittest.TestCase): ...@@ -552,8 +558,8 @@ class ChardataBufferTest(unittest.TestCase):
return self.n return self.n
def test_change_size_1(self): def test_change_size_1(self):
xml1 = "<?xml version='1.0' encoding='iso8859'?><a><s>%s" % ('a' * 1024) xml1 = b"<?xml version='1.0' encoding='iso8859'?><a><s>" + b'a' * 1024
xml2 = "aaa</s><s>%s</s></a>" % ('a' * 1025) xml2 = b'aaa</s><s>' + b'a' * 1025 + b'</s></a>'
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.CharacterDataHandler = self.counting_handler parser.CharacterDataHandler = self.counting_handler
parser.buffer_text = 1 parser.buffer_text = 1
...@@ -568,8 +574,8 @@ class ChardataBufferTest(unittest.TestCase): ...@@ -568,8 +574,8 @@ class ChardataBufferTest(unittest.TestCase):
self.assertEqual(self.n, 2) self.assertEqual(self.n, 2)
def test_change_size_2(self): def test_change_size_2(self):
xml1 = "<?xml version='1.0' encoding='iso8859'?><a>a<s>%s" % ('a' * 1023) xml1 = b"<?xml version='1.0' encoding='iso8859'?><a>a<s>" + b'a' * 1023
xml2 = "aaa</s><s>%s</s></a>" % ('a' * 1025) xml2 = b'aaa</s><s>' + b'a' * 1025 + b'</s></a>'
parser = expat.ParserCreate() parser = expat.ParserCreate()
parser.CharacterDataHandler = self.counting_handler parser.CharacterDataHandler = self.counting_handler
parser.buffer_text = 1 parser.buffer_text = 1
...@@ -585,7 +591,7 @@ class ChardataBufferTest(unittest.TestCase): ...@@ -585,7 +591,7 @@ class ChardataBufferTest(unittest.TestCase):
class MalformedInputTest(unittest.TestCase): class MalformedInputTest(unittest.TestCase):
def test1(self): def test1(self):
xml = "\0\r\n" xml = b"\0\r\n"
parser = expat.ParserCreate() parser = expat.ParserCreate()
try: try:
parser.Parse(xml, True) parser.Parse(xml, True)
...@@ -594,7 +600,8 @@ class MalformedInputTest(unittest.TestCase): ...@@ -594,7 +600,8 @@ class MalformedInputTest(unittest.TestCase):
self.assertEqual(str(e), 'unclosed token: line 2, column 0') self.assertEqual(str(e), 'unclosed token: line 2, column 0')
def test2(self): def test2(self):
xml = "<?xml version\xc2\x85='1.0'?>\r\n" # \xc2\x85 is UTF-8 encoded U+0085 (NEXT LINE)
xml = b"<?xml version\xc2\x85='1.0'?>\r\n"
parser = expat.ParserCreate() parser = expat.ParserCreate()
try: try:
parser.Parse(xml, True) parser.Parse(xml, True)
...@@ -609,7 +616,7 @@ class ErrorMessageTest(unittest.TestCase): ...@@ -609,7 +616,7 @@ class ErrorMessageTest(unittest.TestCase):
errors.messages[errors.codes[errors.XML_ERROR_SYNTAX]]) errors.messages[errors.codes[errors.XML_ERROR_SYNTAX]])
def test_expaterror(self): def test_expaterror(self):
xml = '<' xml = b'<'
parser = expat.ParserCreate() parser = expat.ParserCreate()
try: try:
parser.Parse(xml, True) parser.Parse(xml, True)
...@@ -638,7 +645,7 @@ class ForeignDTDTests(unittest.TestCase): ...@@ -638,7 +645,7 @@ class ForeignDTDTests(unittest.TestCase):
parser.UseForeignDTD(True) parser.UseForeignDTD(True)
parser.SetParamEntityParsing(expat.XML_PARAM_ENTITY_PARSING_ALWAYS) parser.SetParamEntityParsing(expat.XML_PARAM_ENTITY_PARSING_ALWAYS)
parser.ExternalEntityRefHandler = resolve_entity parser.ExternalEntityRefHandler = resolve_entity
parser.Parse("<?xml version='1.0'?><element/>") parser.Parse(b"<?xml version='1.0'?><element/>")
self.assertEqual(handler_call_args, [(None, None)]) self.assertEqual(handler_call_args, [(None, None)])
# test UseForeignDTD() is equal to UseForeignDTD(True) # test UseForeignDTD() is equal to UseForeignDTD(True)
...@@ -648,7 +655,7 @@ class ForeignDTDTests(unittest.TestCase): ...@@ -648,7 +655,7 @@ class ForeignDTDTests(unittest.TestCase):
parser.UseForeignDTD() parser.UseForeignDTD()
parser.SetParamEntityParsing(expat.XML_PARAM_ENTITY_PARSING_ALWAYS) parser.SetParamEntityParsing(expat.XML_PARAM_ENTITY_PARSING_ALWAYS)
parser.ExternalEntityRefHandler = resolve_entity parser.ExternalEntityRefHandler = resolve_entity
parser.Parse("<?xml version='1.0'?><element/>") parser.Parse(b"<?xml version='1.0'?><element/>")
self.assertEqual(handler_call_args, [(None, None)]) self.assertEqual(handler_call_args, [(None, None)])
def test_ignore_use_foreign_dtd(self): def test_ignore_use_foreign_dtd(self):
...@@ -667,7 +674,7 @@ class ForeignDTDTests(unittest.TestCase): ...@@ -667,7 +674,7 @@ class ForeignDTDTests(unittest.TestCase):
parser.SetParamEntityParsing(expat.XML_PARAM_ENTITY_PARSING_ALWAYS) parser.SetParamEntityParsing(expat.XML_PARAM_ENTITY_PARSING_ALWAYS)
parser.ExternalEntityRefHandler = resolve_entity parser.ExternalEntityRefHandler = resolve_entity
parser.Parse( parser.Parse(
"<?xml version='1.0'?><!DOCTYPE foo PUBLIC 'bar' 'baz'><element/>") b"<?xml version='1.0'?><!DOCTYPE foo PUBLIC 'bar' 'baz'><element/>")
self.assertEqual(handler_call_args, [("bar", "baz")]) self.assertEqual(handler_call_args, [("bar", "baz")])
......
...@@ -212,6 +212,10 @@ Core and Builtins ...@@ -212,6 +212,10 @@ Core and Builtins
Library Library
------- -------
- Issue #17089: Expat parser now correctly works with string input not only when
an internal XML encoding is UTF-8 or US-ASCII. It now accepts bytes and
strings larger than 2 GiB.
- Issue #16903: Popen.communicate() on Unix now accepts strings when - Issue #16903: Popen.communicate() on Unix now accepts strings when
universal_newlines is true as on Windows. universal_newlines is true as on Windows.
......
...@@ -777,17 +777,52 @@ PyDoc_STRVAR(xmlparse_Parse__doc__, ...@@ -777,17 +777,52 @@ PyDoc_STRVAR(xmlparse_Parse__doc__,
"Parse(data[, isfinal])\n\ "Parse(data[, isfinal])\n\
Parse XML data. `isfinal' should be true at end of input."); Parse XML data. `isfinal' should be true at end of input.");
#define MAX_CHUNK_SIZE (1 << 20)
static PyObject * static PyObject *
xmlparse_Parse(xmlparseobject *self, PyObject *args) xmlparse_Parse(xmlparseobject *self, PyObject *args)
{ {
char *s; PyObject *data;
int slen;
int isFinal = 0; int isFinal = 0;
const char *s;
Py_ssize_t slen;
Py_buffer view;
int rc;
if (!PyArg_ParseTuple(args, "s#|i:Parse", &s, &slen, &isFinal)) if (!PyArg_ParseTuple(args, "O|i:Parse", &data, &isFinal))
return NULL; return NULL;
return get_parse_result(self, XML_Parse(self->itself, s, slen, isFinal)); if (PyUnicode_Check(data)) {
PyObject *bytes;
bytes = PyUnicode_AsUTF8String(data);
if (bytes == NULL)
return NULL;
view.buf = NULL;
s = PyBytes_AS_STRING(bytes);
slen = PyBytes_GET_SIZE(bytes);
/* Explicitly set UTF-8 encoding. Return code ignored. */
(void)XML_SetEncoding(self->itself, "utf-8");
}
else {
if (PyObject_GetBuffer(data, &view, PyBUF_SIMPLE) < 0)
return NULL;
s = view.buf;
slen = view.len;
}
while (slen > MAX_CHUNK_SIZE) {
rc = XML_Parse(self->itself, s, MAX_CHUNK_SIZE, 0);
if (!rc)
goto done;
s += MAX_CHUNK_SIZE;
slen -= MAX_CHUNK_SIZE;
}
rc = XML_Parse(self->itself, s, slen, isFinal);
done:
if (view.buf != NULL)
PyBuffer_Release(&view);
return get_parse_result(self, rc);
} }
/* File reading copied from cPickle */ /* File reading copied from cPickle */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment