Skip to content

Commit e5d5522

Browse files
authored
gh-106917: fix super classmethod calls to non-classmethods (#106977)
1 parent b383703 commit e5d5522

File tree

5 files changed

+51
-5
lines changed

5 files changed

+51
-5
lines changed

Lib/test/test_super.py

+44-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,9 @@
55
from test import shadowed_super
66

77

8+
ADAPTIVE_WARMUP_DELAY = 2
9+
10+
811
class A:
912
def f(self):
1013
return 'A'
@@ -419,8 +422,47 @@ def test(name):
419422
super(MyType, type(mytype)).__setattr__(mytype, "bar", 1)
420423
self.assertEqual(mytype.bar, 1)
421424

422-
test("foo1")
423-
test("foo2")
425+
for _ in range(ADAPTIVE_WARMUP_DELAY):
426+
test("foo1")
427+
428+
def test_reassigned_new(self):
429+
class A:
430+
def __new__(cls):
431+
pass
432+
433+
def __init_subclass__(cls):
434+
if "__new__" not in cls.__dict__:
435+
cls.__new__ = cls.__new__
436+
437+
class B(A):
438+
pass
439+
440+
class C(B):
441+
def __new__(cls):
442+
return super().__new__(cls)
443+
444+
for _ in range(ADAPTIVE_WARMUP_DELAY):
445+
C()
446+
447+
def test_mixed_staticmethod_hierarchy(self):
448+
# This test is just a desugared version of `test_reassigned_new`
449+
class A:
450+
@staticmethod
451+
def some(cls, *args, **kwargs):
452+
self.assertFalse(args)
453+
self.assertFalse(kwargs)
454+
455+
class B(A):
456+
def some(cls, *args, **kwargs):
457+
return super().some(cls, *args, **kwargs)
458+
459+
class C(B):
460+
@staticmethod
461+
def some(cls):
462+
return super().some(cls)
463+
464+
for _ in range(ADAPTIVE_WARMUP_DELAY):
465+
C.some(C)
424466

425467

426468
if __name__ == "__main__":
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
Fix classmethod-style :func:`super` method calls (i.e., where the second
2+
argument to :func:`super`, or the implied second argument drawn from
3+
``self/cls`` in the case of zero-arg super, is a type) when the target of
4+
the call is not a classmethod.

Python/bytecodes.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -1721,7 +1721,7 @@ dummy_func(
17211721
PyTypeObject *cls = (PyTypeObject *)class;
17221722
int method_found = 0;
17231723
res2 = _PySuper_Lookup(cls, self, name,
1724-
cls->tp_getattro == PyObject_GenericGetAttr ? &method_found : NULL);
1724+
Py_TYPE(self)->tp_getattro == PyObject_GenericGetAttr ? &method_found : NULL);
17251725
Py_DECREF(global_super);
17261726
Py_DECREF(class);
17271727
if (res2 == NULL) {

Python/executor_cases.c.h

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Python/generated_cases.c.h

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)