]> granicus.if.org Git - python/commitdiff
bpo-31411: Prevent raising a SystemError in case warnings.onceregistry is not a dicti...
authorOren Milman <orenmn@gmail.com>
Mon, 11 Sep 2017 06:28:39 +0000 (09:28 +0300)
committerSerhiy Storchaka <storchaka@gmail.com>
Mon, 11 Sep 2017 06:28:39 +0000 (09:28 +0300)
Lib/test/test_warnings/__init__.py
Misc/NEWS.d/next/Core and Builtins/2017-09-11-08-50-41.bpo-31411.HZz82I.rst [new file with mode: 0644]
Python/_warnings.c

index 69623c40f4677c7232d102185cfe388b39ae7571..75539cf1c87f8a56310c83fa36ed568686ae7855 100644 (file)
@@ -794,6 +794,17 @@ class _WarningsTests(BaseTest, unittest.TestCase):
         self.assertNotIn(b'Warning!', stderr)
         self.assertNotIn(b'Error', stderr)
 
+    @support.cpython_only
+    def test_issue31411(self):
+        # warn_explicit() shouldn't raise a SystemError in case
+        # warnings.onceregistry isn't a dictionary.
+        wmod = self.module
+        with original_warnings.catch_warnings(module=wmod):
+            wmod.filterwarnings('once')
+            with support.swap_attr(wmod, 'onceregistry', None):
+                with self.assertRaises(TypeError):
+                    wmod.warn_explicit('foo', Warning, 'bar', 1, registry=None)
+
 
 class WarningsDisplayTests(BaseTest):
 
diff --git a/Misc/NEWS.d/next/Core and Builtins/2017-09-11-08-50-41.bpo-31411.HZz82I.rst b/Misc/NEWS.d/next/Core and Builtins/2017-09-11-08-50-41.bpo-31411.HZz82I.rst
new file mode 100644 (file)
index 0000000..ad1b4b8
--- /dev/null
@@ -0,0 +1,2 @@
+Raise a TypeError instead of SystemError in case warnings.onceregistry is
+not a dictionary. Patch by Oren Milman.
index 5230318788c9ca115df897faffb73e07167fdd62..078bdcb0399321ecf43debbddee93ca6027d646d 100644 (file)
@@ -86,6 +86,12 @@ get_once_registry(void)
             return NULL;
         return _PyRuntime.warnings.once_registry;
     }
+    if (!PyDict_Check(registry)) {
+        PyErr_SetString(PyExc_TypeError,
+                        "warnings.onceregistry must be a dict");
+        Py_DECREF(registry);
+        return NULL;
+    }
     Py_DECREF(_PyRuntime.warnings.once_registry);
     _PyRuntime.warnings.once_registry = registry;
     return registry;
@@ -437,7 +443,7 @@ warn_explicit(PyObject *category, PyObject *message,
         Py_RETURN_NONE;
 
     if (registry && !PyDict_Check(registry) && (registry != Py_None)) {
-        PyErr_SetString(PyExc_TypeError, "'registry' must be a dict");
+        PyErr_SetString(PyExc_TypeError, "'registry' must be a dict or None");
         return NULL;
     }