Skip to content

Commit df64210

Browse files
[3.12] gh-109230: test_pyexpat no longer depends on the current directory (GH-109233) (#109241)
gh-109230: test_pyexpat no longer depends on the current directory (GH-109233) Fix test_pyexpat.test_exception(): it can now be run from a directory different than Python source code directory. Before, the test failed in this case. Skip the test if Modules/pyexpat.c source is not available. Skip also the test on Python implementations other than CPython. (cherry picked from commit e55aab9) Co-authored-by: Victor Stinner <vstinner@python.org>
1 parent b361381 commit df64210

File tree

2 files changed

+53
-24
lines changed

2 files changed

+53
-24
lines changed

Lib/test/test_pyexpat.py

+48-24
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,15 @@
11
# XXX TypeErrors on calling handlers, or on bad return values from a
22
# handler, are obscure and unhelpful.
33

4-
from io import BytesIO
54
import os
65
import platform
76
import sys
87
import sysconfig
98
import unittest
109
import traceback
10+
from io import BytesIO
11+
from test import support
12+
from test.support import os_helper
1113

1214
from xml.parsers import expat
1315
from xml.parsers.expat import errors
@@ -441,37 +443,59 @@ def test7(self):
441443
# Test handling of exception from callback:
442444
class HandlerExceptionTest(unittest.TestCase):
443445
def StartElementHandler(self, name, attrs):
444-
raise RuntimeError(name)
446+
raise RuntimeError(f'StartElementHandler: <{name}>')
445447

446448
def check_traceback_entry(self, entry, filename, funcname):
447-
self.assertEqual(os.path.basename(entry[0]), filename)
448-
self.assertEqual(entry[2], funcname)
449+
self.assertEqual(os.path.basename(entry.filename), filename)
450+
self.assertEqual(entry.name, funcname)
449451

452+
@support.cpython_only
450453
def test_exception(self):
454+
# gh-66652: test _PyTraceback_Add() used by pyexpat.c to inject frames
455+
456+
# Change the current directory to the Python source code directory
457+
# if it is available.
458+
src_dir = sysconfig.get_config_var('abs_builddir')
459+
if src_dir:
460+
have_source = os.path.isdir(src_dir)
461+
else:
462+
have_source = False
463+
if have_source:
464+
with os_helper.change_cwd(src_dir):
465+
self._test_exception(have_source)
466+
else:
467+
self._test_exception(have_source)
468+
469+
def _test_exception(self, have_source):
470+
# Use path relative to the current directory which should be the Python
471+
# source code directory (if it is available).
472+
PYEXPAT_C = os.path.join('Modules', 'pyexpat.c')
473+
451474
parser = expat.ParserCreate()
452475
parser.StartElementHandler = self.StartElementHandler
453476
try:
454477
parser.Parse(b"<a><b><c/></b></a>", True)
455-
self.fail()
456-
except RuntimeError as e:
457-
self.assertEqual(e.args[0], 'a',
458-
"Expected RuntimeError for element 'a', but" + \
459-
" found %r" % e.args[0])
460-
# Check that the traceback contains the relevant line in pyexpat.c
461-
entries = traceback.extract_tb(e.__traceback__)
462-
self.assertEqual(len(entries), 3)
463-
self.check_traceback_entry(entries[0],
464-
"test_pyexpat.py", "test_exception")
465-
self.check_traceback_entry(entries[1],
466-
"pyexpat.c", "StartElement")
467-
self.check_traceback_entry(entries[2],
468-
"test_pyexpat.py", "StartElementHandler")
469-
if (sysconfig.is_python_build()
470-
and not (sys.platform == 'win32' and platform.machine() == 'ARM')
471-
and not is_emscripten
472-
and not is_wasi
473-
):
474-
self.assertIn('call_with_frame("StartElement"', entries[1][3])
478+
479+
self.fail("the parser did not raise RuntimeError")
480+
except RuntimeError as exc:
481+
self.assertEqual(exc.args[0], 'StartElementHandler: <a>', exc)
482+
entries = traceback.extract_tb(exc.__traceback__)
483+
484+
self.assertEqual(len(entries), 3, entries)
485+
self.check_traceback_entry(entries[0],
486+
"test_pyexpat.py", "_test_exception")
487+
self.check_traceback_entry(entries[1],
488+
os.path.basename(PYEXPAT_C),
489+
"StartElement")
490+
self.check_traceback_entry(entries[2],
491+
"test_pyexpat.py", "StartElementHandler")
492+
493+
# Check that the traceback contains the relevant line in
494+
# Modules/pyexpat.c. Skip the test if Modules/pyexpat.c is not
495+
# available.
496+
if have_source and os.path.exists(PYEXPAT_C):
497+
self.assertIn('call_with_frame("StartElement"',
498+
entries[1].line)
475499

476500

477501
# Test Current* members:
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
Fix ``test_pyexpat.test_exception()``: it can now be run from a directory
2+
different than Python source code directory. Before, the test failed in this
3+
case. Skip the test if Modules/pyexpat.c source is not available. Skip also
4+
the test on Python implementations other than CPython. Patch by Victor
5+
Stinner.

0 commit comments

Comments
 (0)