Скопировать подпись вызова в декоратор

Если я сделаю следующее

def mydecorator(f):
    def wrapper(*args, **kwargs):
        f(*args, **kwargs)
    wrapper.__doc__ = f.__doc__
    wrapper.__name__ = f.__name__
    return wrapper

@mydecorator
def myfunction(a,b,c):
    '''My docstring'''
    pass

И затем введите help myfunction, я получаю:

Help on function myfunction in module __main__:

myfunction(*args, **kwargs)
    My docstring

Итак, имя и docstring правильно скопированы. Есть ли способ также скопировать фактическую подпись вызова, в этом случае (a, b, c)?

Ответ 1

Вот пример, используя Michele Simionato модуль декоратора, чтобы исправить подпись:

import decorator

@decorator.decorator
def mydecorator(f,*args, **kwargs):
    return f(*args, **kwargs)

@mydecorator
def myfunction(a,b,c):
    '''My docstring'''
    pass

help(myfunction)
# Help on function myfunction in module __main__:

# myfunction(a, b, c)
#     My docstring

Ответ 3

Эта функциональность предоставляется стандартной библиотекой Python проверять модуль, в частности, inspect.getargspec.

>>> import inspect
>>> def f(a, b, c=0, *args, **kwargs): return
... 
>>> inspect.getargspec(f)
ArgSpec(args=['a', 'b', 'c'], varargs='args', keywords='kwargs', defaults=(0,))