diff options
author | Yury Selivanov <yselivanov@sprymix.com> | 2015-05-28 01:59:27 (GMT) |
---|---|---|
committer | Yury Selivanov <yselivanov@sprymix.com> | 2015-05-28 01:59:27 (GMT) |
commit | 72f389fed0f583da9755979179316655fa42664a (patch) | |
tree | 6c8c52aee8007743a84a5b5d970697717894b984 /Lib | |
parent | fa0e7f67ab80b307fa0c3202ed4ed0e2874499ca (diff) | |
parent | 507cd3cf91333d133241c8126d21c5e33001024c (diff) | |
download | cpython-72f389fed0f583da9755979179316655fa42664a.zip cpython-72f389fed0f583da9755979179316655fa42664a.tar.gz cpython-72f389fed0f583da9755979179316655fa42664a.tar.bz2 |
Issue 24298: Fix signature() to properly unwrap wrappers around bound methods
Diffstat (limited to 'Lib')
-rw-r--r-- | Lib/inspect.py | 9 | ||||
-rw-r--r-- | Lib/test/test_inspect.py | 13 |
2 files changed, 22 insertions, 0 deletions
diff --git a/Lib/inspect.py b/Lib/inspect.py index 3debc24..57cb3dc 100644 --- a/Lib/inspect.py +++ b/Lib/inspect.py @@ -2130,6 +2130,15 @@ def _signature_from_callable(obj, *, # Was this function wrapped by a decorator? if follow_wrapper_chains: obj = unwrap(obj, stop=(lambda f: hasattr(f, "__signature__"))) + if isinstance(obj, types.MethodType): + # If the unwrapped object is a *method*, we might want to + # skip its first parameter (self). + # See test_signature_wrapped_bound_method for details. + return _signature_from_callable( + obj, + follow_wrapper_chains=follow_wrapper_chains, + skip_bound_arg=skip_bound_arg, + sigcls=sigcls) try: sig = obj.__signature__ diff --git a/Lib/test/test_inspect.py b/Lib/test/test_inspect.py index 9492cad..9f09b83 100644 --- a/Lib/test/test_inspect.py +++ b/Lib/test/test_inspect.py @@ -2086,6 +2086,19 @@ class TestSignatureObject(unittest.TestCase): with self.assertRaisesRegex(ValueError, 'invalid method signature'): self.signature(Test()) + def test_signature_wrapped_bound_method(self): + # Issue 24298 + class Test: + def m1(self, arg1, arg2=1) -> int: + pass + @functools.wraps(Test().m1) + def m1d(*args, **kwargs): + pass + self.assertEqual(self.signature(m1d), + ((('arg1', ..., ..., "positional_or_keyword"), + ('arg2', 1, ..., "positional_or_keyword")), + int)) + def test_signature_on_classmethod(self): class Test: @classmethod |