mirror of
https://github.com/python/cpython.git
synced 2025-07-08 03:45:36 +00:00
gh-102493: fix normalization in PyErr_SetObject (#102502)
Co-authored-by: Jelle Zijlstra <jelle.zijlstra@gmail.com>
This commit is contained in:
parent
54060ae91d
commit
a33ca2ad1f
4 changed files with 56 additions and 4 deletions
|
@ -140,6 +140,34 @@ class Test_ErrSetAndRestore(unittest.TestCase):
|
|||
self.assertEqual(1, v.args[0])
|
||||
self.assertIs(tb, v.__traceback__.tb_next)
|
||||
|
||||
def test_set_object(self):
|
||||
|
||||
# new exception as obj is not an exception
|
||||
with self.assertRaises(ValueError) as e:
|
||||
_testcapi.exc_set_object(ValueError, 42)
|
||||
self.assertEqual(e.exception.args, (42,))
|
||||
|
||||
# wraps the exception because unrelated types
|
||||
with self.assertRaises(ValueError) as e:
|
||||
_testcapi.exc_set_object(ValueError, TypeError(1,2,3))
|
||||
wrapped = e.exception.args[0]
|
||||
self.assertIsInstance(wrapped, TypeError)
|
||||
self.assertEqual(wrapped.args, (1, 2, 3))
|
||||
|
||||
# is superclass, so does not wrap
|
||||
with self.assertRaises(PermissionError) as e:
|
||||
_testcapi.exc_set_object(OSError, PermissionError(24))
|
||||
self.assertEqual(e.exception.args, (24,))
|
||||
|
||||
class Meta(type):
|
||||
def __subclasscheck__(cls, sub):
|
||||
1/0
|
||||
|
||||
class Broken(Exception, metaclass=Meta):
|
||||
pass
|
||||
|
||||
with self.assertRaises(ZeroDivisionError) as e:
|
||||
_testcapi.exc_set_object(Broken, Broken())
|
||||
|
||||
if __name__ == "__main__":
|
||||
unittest.main()
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
Fix regression in semantics of normalisation in ``PyErr_SetObject``.
|
|
@ -78,6 +78,20 @@ make_exception_with_doc(PyObject *self, PyObject *args, PyObject *kwargs)
|
|||
return PyErr_NewExceptionWithDoc(name, doc, base, dict);
|
||||
}
|
||||
|
||||
static PyObject *
|
||||
exc_set_object(PyObject *self, PyObject *args)
|
||||
{
|
||||
PyObject *exc;
|
||||
PyObject *obj;
|
||||
|
||||
if (!PyArg_ParseTuple(args, "OO:exc_set_object", &exc, &obj)) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
PyErr_SetObject(exc, obj);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static PyObject *
|
||||
raise_exception(PyObject *self, PyObject *args)
|
||||
{
|
||||
|
@ -247,6 +261,7 @@ static PyMethodDef test_methods[] = {
|
|||
PyDoc_STR("fatal_error(message, release_gil=False): call Py_FatalError(message)")},
|
||||
{"make_exception_with_doc", _PyCFunction_CAST(make_exception_with_doc),
|
||||
METH_VARARGS | METH_KEYWORDS},
|
||||
{"exc_set_object", exc_set_object, METH_VARARGS},
|
||||
{"raise_exception", raise_exception, METH_VARARGS},
|
||||
{"raise_memoryerror", raise_memoryerror, METH_NOARGS},
|
||||
{"set_exc_info", test_set_exc_info, METH_VARARGS},
|
||||
|
|
|
@ -149,9 +149,16 @@ _PyErr_SetObject(PyThreadState *tstate, PyObject *exception, PyObject *value)
|
|||
exception);
|
||||
return;
|
||||
}
|
||||
Py_XINCREF(value);
|
||||
/* Normalize the exception */
|
||||
if (value == NULL || (PyObject *)Py_TYPE(value) != exception) {
|
||||
int is_subclass = 0;
|
||||
if (value != NULL && PyExceptionInstance_Check(value)) {
|
||||
is_subclass = PyObject_IsSubclass((PyObject *)Py_TYPE(value), exception);
|
||||
if (is_subclass < 0) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
Py_XINCREF(value);
|
||||
if (!is_subclass) {
|
||||
/* We must normalize the value right now */
|
||||
PyObject *fixed_value;
|
||||
|
||||
|
@ -206,9 +213,10 @@ _PyErr_SetObject(PyThreadState *tstate, PyObject *exception, PyObject *value)
|
|||
Py_DECREF(exc_value);
|
||||
}
|
||||
}
|
||||
if (value != NULL && PyExceptionInstance_Check(value))
|
||||
assert(value != NULL);
|
||||
if (PyExceptionInstance_Check(value))
|
||||
tb = PyException_GetTraceback(value);
|
||||
_PyErr_Restore(tstate, Py_XNewRef(exception), value, tb);
|
||||
_PyErr_Restore(tstate, Py_NewRef(Py_TYPE(value)), value, tb);
|
||||
}
|
||||
|
||||
void
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue