]> git.ipfire.org Git - thirdparty/Python/cpython.git/commitdiff
[3.7] bpo-43660: Fix crash when displaying exceptions with custom values for sys...
authorPablo Galindo <Pablogsal@gmail.com>
Mon, 29 Mar 2021 23:24:17 +0000 (00:24 +0100)
committerGitHub <noreply@github.com>
Mon, 29 Mar 2021 23:24:17 +0000 (00:24 +0100)
(cherry picked from commit 09b90a037d18f5d4acdf1b14082e57bda78e85d3)

Co-authored-by: Pablo Galindo <Pablogsal@gmail.com>
Lib/test/test_sys.py
Misc/NEWS.d/next/Core and Builtins/2021-03-29-19-50-34.bpo-43660.scTgag.rst [new file with mode: 0644]
Python/pythonrun.c

index 84927a393f17593620d67bc7eff5219409dfb0f2..0478f20cd33b4d444e0af3f9523765fc0085a4f6 100644 (file)
@@ -1292,6 +1292,21 @@ class SizeofTest(unittest.TestCase):
         self.assertIsNone(cur.firstiter)
         self.assertIsNone(cur.finalizer)
 
+    def test_changing_sys_stderr_and_removing_reference(self):
+        # If the default displayhook doesn't take a strong reference
+        # to sys.stderr the following code can crash. See bpo-43660
+        # for more details.
+        code = textwrap.dedent('''
+            import sys
+            class MyStderr:
+                def write(self, s):
+                    sys.stderr = None
+            sys.stderr = MyStderr()
+            1/0
+        ''')
+        rc, out, err = assert_python_failure('-c', code)
+        self.assertEqual(out, b"")
+        self.assertEqual(err, b"")
 
 def test_main():
     test.support.run_unittest(SysModuleTest, SizeofTest)
diff --git a/Misc/NEWS.d/next/Core and Builtins/2021-03-29-19-50-34.bpo-43660.scTgag.rst b/Misc/NEWS.d/next/Core and Builtins/2021-03-29-19-50-34.bpo-43660.scTgag.rst
new file mode 100644 (file)
index 0000000..9841950
--- /dev/null
@@ -0,0 +1,3 @@
+Fix crash that happens when replacing ``sys.stderr`` with a callable that
+can remove the object while an exception is being printed. Patch by Pablo
+Galindo.
index 4c974cef39c78bc88a0952426ca3dd0682006e48..b7b77d1f9ee3febc0f656d29f5337c874cb373f4 100644 (file)
@@ -932,7 +932,9 @@ PyErr_Display(PyObject *exception, PyObject *value, PyObject *tb)
         seen = PySet_New(NULL);
         if (seen == NULL)
             PyErr_Clear();
+        Py_INCREF(f);
         print_exception_recursive(f, value, seen);
+        Py_DECREF(f);
         Py_XDECREF(seen);
     }
 }