bpo-32999: Fix ABC.__subclasscheck__ crash (GH-6002)
This commit is contained in:
parent
bc3f2289b9
commit
fc7df0e664
|
@ -392,6 +392,24 @@ def test_factory(abc_ABCMeta, abc_get_cache_token):
|
||||||
self.assertIsInstance(42, A)
|
self.assertIsInstance(42, A)
|
||||||
self.assertIsInstance(42, (A,))
|
self.assertIsInstance(42, (A,))
|
||||||
|
|
||||||
|
def test_issubclass_bad_arguments(self):
|
||||||
|
class A(metaclass=abc_ABCMeta):
|
||||||
|
pass
|
||||||
|
|
||||||
|
with self.assertRaises(TypeError):
|
||||||
|
issubclass({}, A) # unhashable
|
||||||
|
|
||||||
|
with self.assertRaises(TypeError):
|
||||||
|
issubclass(42, A) # No __mro__
|
||||||
|
|
||||||
|
# Python version supports any iterable as __mro__.
|
||||||
|
# But it's implementation detail and don't emulate it in C version.
|
||||||
|
class C:
|
||||||
|
__mro__ = 42 # __mro__ is not tuple
|
||||||
|
|
||||||
|
with self.assertRaises(TypeError):
|
||||||
|
issubclass(C(), A)
|
||||||
|
|
||||||
def test_all_new_methods_are_called(self):
|
def test_all_new_methods_are_called(self):
|
||||||
class A(metaclass=abc_ABCMeta):
|
class A(metaclass=abc_ABCMeta):
|
||||||
pass
|
pass
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
Fix C implemetation of ``ABC.__subclasscheck__(cls, subclass)`` crashed when
|
||||||
|
``subclass`` is not a type object.
|
|
@ -16,6 +16,7 @@ _Py_IDENTIFIER(__abstractmethods__);
|
||||||
_Py_IDENTIFIER(__class__);
|
_Py_IDENTIFIER(__class__);
|
||||||
_Py_IDENTIFIER(__dict__);
|
_Py_IDENTIFIER(__dict__);
|
||||||
_Py_IDENTIFIER(__bases__);
|
_Py_IDENTIFIER(__bases__);
|
||||||
|
_Py_IDENTIFIER(__mro__);
|
||||||
_Py_IDENTIFIER(_abc_impl);
|
_Py_IDENTIFIER(_abc_impl);
|
||||||
_Py_IDENTIFIER(__subclasscheck__);
|
_Py_IDENTIFIER(__subclasscheck__);
|
||||||
_Py_IDENTIFIER(__subclasshook__);
|
_Py_IDENTIFIER(__subclasshook__);
|
||||||
|
@ -568,7 +569,7 @@ _abc__abc_subclasscheck_impl(PyObject *module, PyObject *self,
|
||||||
PyObject *subclass)
|
PyObject *subclass)
|
||||||
/*[clinic end generated code: output=b56c9e4a530e3894 input=1d947243409d10b8]*/
|
/*[clinic end generated code: output=b56c9e4a530e3894 input=1d947243409d10b8]*/
|
||||||
{
|
{
|
||||||
PyObject *ok, *mro, *subclasses = NULL, *result = NULL;
|
PyObject *ok, *mro = NULL, *subclasses = NULL, *result = NULL;
|
||||||
Py_ssize_t pos;
|
Py_ssize_t pos;
|
||||||
int incache;
|
int incache;
|
||||||
_abc_data *impl = _get_impl(self);
|
_abc_data *impl = _get_impl(self);
|
||||||
|
@ -637,14 +638,24 @@ _abc__abc_subclasscheck_impl(PyObject *module, PyObject *self,
|
||||||
}
|
}
|
||||||
Py_DECREF(ok);
|
Py_DECREF(ok);
|
||||||
|
|
||||||
/* 4. Check if it's a direct subclass. */
|
/* 4. Check if it's a direct subclass.
|
||||||
mro = ((PyTypeObject *)subclass)->tp_mro;
|
*
|
||||||
assert(PyTuple_Check(mro));
|
* if cls in getattr(subclass, '__mro__', ()):
|
||||||
for (pos = 0; pos < PyTuple_GET_SIZE(mro); pos++) {
|
* cls._abc_cache.add(subclass)
|
||||||
PyObject *mro_item = PyTuple_GET_ITEM(mro, pos);
|
* return True
|
||||||
if (mro_item == NULL) {
|
*/
|
||||||
|
if (_PyObject_LookupAttrId(subclass, &PyId___mro__, &mro) < 0) {
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
if (mro != NULL) {
|
||||||
|
if (!PyTuple_Check(mro)) {
|
||||||
|
// Python version supports non-tuple iterable. Keep it as
|
||||||
|
// implementation detail.
|
||||||
|
PyErr_SetString(PyExc_TypeError, "__mro__ is not a tuple");
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
for (pos = 0; pos < PyTuple_GET_SIZE(mro); pos++) {
|
||||||
|
PyObject *mro_item = PyTuple_GET_ITEM(mro, pos);
|
||||||
if ((PyObject *)self == mro_item) {
|
if ((PyObject *)self == mro_item) {
|
||||||
if (_add_to_weak_set(&impl->_abc_cache, subclass) < 0) {
|
if (_add_to_weak_set(&impl->_abc_cache, subclass) < 0) {
|
||||||
goto end;
|
goto end;
|
||||||
|
@ -653,6 +664,7 @@ _abc__abc_subclasscheck_impl(PyObject *module, PyObject *self,
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* 5. Check if it's a subclass of a registered class (recursive). */
|
/* 5. Check if it's a subclass of a registered class (recursive). */
|
||||||
if (subclasscheck_check_registry(impl, subclass, &result)) {
|
if (subclasscheck_check_registry(impl, subclass, &result)) {
|
||||||
|
@ -690,7 +702,8 @@ _abc__abc_subclasscheck_impl(PyObject *module, PyObject *self,
|
||||||
result = Py_False;
|
result = Py_False;
|
||||||
|
|
||||||
end:
|
end:
|
||||||
Py_XDECREF(impl);
|
Py_DECREF(impl);
|
||||||
|
Py_XDECREF(mro);
|
||||||
Py_XDECREF(subclasses);
|
Py_XDECREF(subclasses);
|
||||||
Py_XINCREF(result);
|
Py_XINCREF(result);
|
||||||
return result;
|
return result;
|
||||||
|
|
Loading…
Reference in New Issue