bpo-41031: Match C and Python code formatting of unprintable exceptions and exceptions in the __main__ module. (GH-28139)

This commit is contained in:
Irit Katriel 2021-09-05 16:54:13 +01:00 committed by GitHub
parent b01fd533fe
commit 9e31b3952f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 55 additions and 14 deletions

View File

@ -1071,19 +1071,29 @@ class UnraisableHookTest(unittest.TestCase):
self.assertTrue(report.endswith("\n")) self.assertTrue(report.endswith("\n"))
def test_original_unraisablehook_exception_qualname(self): def test_original_unraisablehook_exception_qualname(self):
# See bpo-41031, bpo-45083.
# Check that the exception is printed with its qualified name
# rather than just classname, and the module names appears
# unless it is one of the hard-coded exclusions.
class A: class A:
class B: class B:
class X(Exception): class X(Exception):
pass pass
for moduleName in 'builtins', '__main__', 'some_module':
with self.subTest(moduleName=moduleName):
A.B.X.__module__ = moduleName
with test.support.captured_stderr() as stderr, \ with test.support.captured_stderr() as stderr, \
test.support.swap_attr(sys, 'unraisablehook', test.support.swap_attr(sys, 'unraisablehook',
sys.__unraisablehook__): sys.__unraisablehook__):
expected = self.write_unraisable_exc( expected = self.write_unraisable_exc(
A.B.X(), "msg", "obj"); A.B.X(), "msg", "obj");
report = stderr.getvalue() report = stderr.getvalue()
testName = 'test_original_unraisablehook_exception_qualname' self.assertIn(A.B.X.__qualname__, report)
self.assertIn(f"{testName}.<locals>.A.B.X", report) if moduleName in ['builtins', '__main__']:
self.assertNotIn(moduleName + '.', report)
else:
self.assertIn(moduleName + '.', report)
def test_original_unraisablehook_wrong_type(self): def test_original_unraisablehook_wrong_type(self):
exc = ValueError(42) exc = ValueError(42)

View File

@ -172,7 +172,7 @@ class TracebackCases(unittest.TestCase):
1/0 1/0
err = traceback.format_exception_only(X, X()) err = traceback.format_exception_only(X, X())
self.assertEqual(len(err), 1) self.assertEqual(len(err), 1)
str_value = '<unprintable %s object>' % X.__name__ str_value = '<exception str() failed>'
if X.__module__ in ('__main__', 'builtins'): if X.__module__ in ('__main__', 'builtins'):
str_name = X.__qualname__ str_name = X.__qualname__
else: else:
@ -1171,19 +1171,45 @@ class BaseExceptionReportingTests:
exp = "\n".join(expected) exp = "\n".join(expected)
self.assertEqual(exp, err) self.assertEqual(exp, err)
def test_format_exception_only_qualname(self): def test_exception_qualname(self):
class A: class A:
class B: class B:
class X(Exception): class X(Exception):
def __str__(self): def __str__(self):
return "I am X" return "I am X"
pass
err = self.get_report(A.B.X()) err = self.get_report(A.B.X())
str_value = 'I am X' str_value = 'I am X'
str_name = '.'.join([A.B.X.__module__, A.B.X.__qualname__]) str_name = '.'.join([A.B.X.__module__, A.B.X.__qualname__])
exp = "%s: %s\n" % (str_name, str_value) exp = "%s: %s\n" % (str_name, str_value)
self.assertEqual(exp, err) self.assertEqual(exp, err)
def test_exception_modulename(self):
class X(Exception):
def __str__(self):
return "I am X"
for modulename in '__main__', 'builtins', 'some_module':
X.__module__ = modulename
with self.subTest(modulename=modulename):
err = self.get_report(X())
str_value = 'I am X'
if modulename in ['builtins', '__main__']:
str_name = X.__qualname__
else:
str_name = '.'.join([X.__module__, X.__qualname__])
exp = "%s: %s\n" % (str_name, str_value)
self.assertEqual(exp, err)
def test_exception_bad__str__(self):
class X(Exception):
def __str__(self):
1/0
err = self.get_report(X())
str_value = '<exception str() failed>'
str_name = '.'.join([X.__module__, X.__qualname__])
self.assertEqual(err, f"{str_name}: {str_value}\n")
class PyExcReportingTests(BaseExceptionReportingTests, unittest.TestCase): class PyExcReportingTests(BaseExceptionReportingTests, unittest.TestCase):
# #

View File

@ -169,7 +169,7 @@ def _some_str(value):
try: try:
return str(value) return str(value)
except: except:
return '<unprintable %s object>' % type(value).__name__ return '<exception str() failed>'
# -- # --

View File

@ -0,0 +1 @@
Match C and Python code formatting of unprintable exceptions and exceptions in the :mod:`__main__` module.

View File

@ -25,6 +25,7 @@ extern char *strerror(int);
extern "C" { extern "C" {
#endif #endif
_Py_IDENTIFIER(__main__);
_Py_IDENTIFIER(__module__); _Py_IDENTIFIER(__module__);
_Py_IDENTIFIER(builtins); _Py_IDENTIFIER(builtins);
_Py_IDENTIFIER(stderr); _Py_IDENTIFIER(stderr);
@ -1297,7 +1298,8 @@ write_unraisable_exc_file(PyThreadState *tstate, PyObject *exc_type,
} }
} }
else { else {
if (!_PyUnicode_EqualToASCIIId(modulename, &PyId_builtins)) { if (!_PyUnicode_EqualToASCIIId(modulename, &PyId_builtins) &&
!_PyUnicode_EqualToASCIIId(modulename, &PyId___main__)) {
if (PyFile_WriteObject(modulename, file, Py_PRINT_RAW) < 0) { if (PyFile_WriteObject(modulename, file, Py_PRINT_RAW) < 0) {
Py_DECREF(modulename); Py_DECREF(modulename);
return -1; return -1;

View File

@ -35,6 +35,7 @@
#endif #endif
_Py_IDENTIFIER(__main__);
_Py_IDENTIFIER(builtins); _Py_IDENTIFIER(builtins);
_Py_IDENTIFIER(excepthook); _Py_IDENTIFIER(excepthook);
_Py_IDENTIFIER(flush); _Py_IDENTIFIER(flush);
@ -974,7 +975,8 @@ print_exception(PyObject *f, PyObject *value)
err = PyFile_WriteString("<unknown>", f); err = PyFile_WriteString("<unknown>", f);
} }
else { else {
if (!_PyUnicode_EqualToASCIIId(modulename, &PyId_builtins)) if (!_PyUnicode_EqualToASCIIId(modulename, &PyId_builtins) &&
!_PyUnicode_EqualToASCIIId(modulename, &PyId___main__))
{ {
err = PyFile_WriteObject(modulename, f, Py_PRINT_RAW); err = PyFile_WriteObject(modulename, f, Py_PRINT_RAW);
err += PyFile_WriteString(".", f); err += PyFile_WriteString(".", f);