-
-
Notifications
You must be signed in to change notification settings - Fork 2.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
When using Class as a Decorator, on decorated Instance Methods or Class Methods, MyPy reports 'Missing positional argument "self" [or "cls"] in call to "__call__" of "Decorator"'. Type Checks Okay for Static Methods and Normal Functions. #13222
Comments
This is (unfortunately) how One (admittedly ugly) workaround is to define two different versions of the decorator — one for methods and one for normal functions. The version for methods needs to use a def with_auth_method(wrapped: Callable[Concatenate[Any, P], T]) -> Auth[P, T]: ... |
@erictraut, thanks for the info. Is there a way to do what I'm trying to do without using ParamSpec? Maybe with Protocols? |
You could use a TypeVar bound to a callable. It's not ideal because it involves a C = TypeVar("C", bound=Callable[..., Any])
def with_auth(wrapped: C) -> C:
@Auth
def decorated(*args: Any, **kwargs: Any) -> C:
return wrapped(*args, **kwargs)
return cast(C, decorated) |
Hmm. With that, the decorator removes the ability to type check decorated methods. :-\ I changed it up a bit (so I'm not doing that decorating the decorator bit), but it's got the same functionality as your example. But when I run this through mypy in the mypy playground with import functools
from typing import (
cast,
Any,
Callable,
Generic,
NewType,
TypeVar,
)
UserId = NewType('UserId', str)
F = TypeVar('F', bound=Callable[..., Any])
class Auth:
def login(self) -> dict[str, str]:
return {'accessToken': 'wow', 'tokenType': 'Bearer'}
class WithAuth(Generic[F]):
def __init__(self, wrapped: F) -> None:
functools.update_wrapper(self, wrapped)
self.wrapped = wrapped
self.auth = Auth()
def __call__(self, *args: Any, **kwargs: Any) -> Any:
self.token = self.auth.login()
return cast(F, self.wrapped(*args, **kwargs))
class Client:
@WithAuth
def one(self, user_id: UserId) -> str:
return f'Making call with token {self.one.token}'
@classmethod
@WithAuth
def two(cls, user_id: UserId) -> str:
return f'Making call with token {cls.two.token}'
@staticmethod
@WithAuth
def three(user_id: UserId) -> str:
return f'Making call with token {Client.three.token}'
@WithAuth
def four(user_id: UserId) -> str:
return f'Making call with token {four.token}'
@WithAuth
def five(user_id: UserId) -> str:
return f'Making call with token {five.token}'
# Succeeds as expected.
UserId('asdf')
# Fails Successfully: Argument 1 to "UserId" has incompatible type "int"; expected "str" [arg-type]
UserId(42)
client = Client()
# Every call here checks successfully with mypy, but only those which cast the str to a UserId should pass
client.one(user_id=UserId('asdf'))
client.one(user_id='asdf')
client.one(user_id=42)
Client.two(user_id=UserId('asdf'))
Client.two(user_id='asdf')
Client.two(user_id=42)
Client.three(user_id=UserId('asdf'))
Client.three(user_id='asdf')
Client.three(user_id=42)
four(user_id=UserId('asdf'))
four(user_id='asdf')
four(user_id=42)
five(user_id=UserId('asdf'))
five(user_id='asdf')
five(user_id=42) |
So the example here is a bit complex, but I think I've run into the same issue, with a simpler example: from typing import Protocol, ParamSpec
P = ParamSpec('P')
class Thing(Protocol[P]):
def __call__(self, *args: P.args, **kwargs: P.kwargs) -> str:
...
def decorate(f: Thing[P]) -> Thing[P]:
return f
class Foo:
@decorate
def bar(self, x: int) -> str:
return str(x)
reveal_type(Foo().bar)
Foo().bar(5) Here,
@erictraut Above you mention that "This is (unfortunately) how When I make |
@bjoernpollex-sc, If you use def decorate(f: Callable[P, str]) -> Callable[P, str]:
return f |
@erictraut Thanks for the explanation. Do you know if there is any work in progress to adress this? Specifically, what is the intended way to type-hint decorators that add attributes to the decorated functions? |
I'm encountering this exact same issue, and also getting an "Incompatible types in assignment" error when trying to instantiate a class that should conform to my protocol. Here's a simple reproduction of the issue: from typing import Callable, Protocol, TypeVar
from typing_extensions import ParamSpec
PS = ParamSpec("PS")
RT = TypeVar("RT", covariant=True)
class MyCallable(Protocol[PS, RT]):
def __call__(self, *args: PS.args, **kwargs: PS.kwargs) -> RT:
...
def decorate(func: Callable[PS, RT], /) -> MyCallable[PS, RT]:
return func
class Animal(Protocol):
@decorate
def speak(self) -> None:
...
class Dog:
def speak(self) -> None:
print("Woof!")
dog: Animal = Dog()
dog.speak() $ mypy app.py
app.py:28: error: Incompatible types in assignment (expression has type "Dog", variable has type "Animal") [assignment]
app:28: note: Following member(s) of "Dog" have conflicts:
app:28: note: speak: expected "MyCallable[[Animal], None]", got "Callable[[], None]"
app:30: error: Missing positional argument "self" in call to "__call__" of "MyCallable" [call-arg]
Found 2 errors in 1 file (checked 1 source file) I appear to have a very similar use-case to @bjoernpollex-sc, as I would like the |
I think mypy is working correctly here according to PEP 612. I don't see any actionable bugs in any of the above code samples. I think this issue can be closed. |
Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
…achedMethod" Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
…achedMethod" Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
…achedMethod" Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... cc voznesenskym penguinwu EikanWang jgong5 Guobing-Chen XiaobingSuper zhuhaozhe blzheng wenzhe-nrv jiayisunx peterbell10 ipiszy yf225 chenyang78 kadeng muchulee8 aakhundov ColinPeppler [ghstack-poisoned]
Previously, I was unsure how to properly type the parameters of a decorated method. Then I found python/mypy#13222 (comment) which explains how to use `Concatenate` to hackily achieve it. Not entirely sure why we can't write a user-defined version of `Callable` that works seamlessly for both functions and methods... Pull Request resolved: #114161 Approved by: https://github.com/Skylion007
Bug Report
I'm trying to use a class as a decorator to inject auth data into function calls, but I'm having a tough time getting Mypy to work correctly. Everything works correctly for a non-class method or a staticmethod, but the type checking fails for an instance method or a classmethod. The failure is Missing positional argument "self" in call to "call" of "Auth" [call-arg] for instance methods and Missing positional argument "cls" in call to "call" of "Auth" [call-arg]. These failures happen even when the UserId is passed like it should be.
I guess the implicit self and cls values aren't being recognized by mypy as being passed to the Auth.call method when the functions are called in their normal way. Note in the examples When I explicitly add a positional parameter as the first parameter to the instance method and to the classmethod which is the instance or class respectively, mypy stops complaining about the missing positional parameter and instead checks the type of the passed parameter. But obviously forcing users to call python methods in some weird way just to satisfy mypy isn't going to fly.
To Reproduce
Expected Behavior
I expected calls to all decorated methods to type check the parameters that are used in the normal calls to the methods.
Actual Behavior
I found that calls like the following to instance methods:
Your Environment
mypy.ini
(and other config files): Used mypy playground with no flags and then strict and local with no flags and then strict.The text was updated successfully, but these errors were encountered: