Skip to content

Commit d8f60bb

Browse files
committed
Implement _StrPromise attribute hook.
This implements an attribute hook that provides type information for methods that are available on `builtins.str` for `_StrPromise` except the supported operators. This allows us to avoid copying stubs from the builtins for all supported methods on `str`. Signed-off-by: Zixuan James Li <[email protected]>
1 parent 3f39f02 commit d8f60bb

File tree

5 files changed

+79
-0
lines changed

5 files changed

+79
-0
lines changed

django-stubs/utils/functional.pyi

+2
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,8 @@ class _StrPromise(Promise, Sequence[str]):
4040
def __mod__(self, __x: Any) -> str: ...
4141
def __mul__(self, __n: SupportsIndex) -> str: ...
4242
def __rmul__(self, __n: SupportsIndex) -> str: ...
43+
# Mypy requires this for the attribute hook to take effect
44+
def __getattribute__(self, __name: str) -> Any: ...
4345

4446
_C = TypeVar("_C", bound=Callable)
4547

mypy_django_plugin/lib/fullnames.py

+2
Original file line numberDiff line numberDiff line change
@@ -41,3 +41,5 @@
4141
F_EXPRESSION_FULLNAME = "django.db.models.expressions.F"
4242

4343
ANY_ATTR_ALLOWED_CLASS_FULLNAME = "django_stubs_ext.AnyAttrAllowed"
44+
45+
STR_PROMISE_FULLNAME = "django.utils.functional._StrPromise"

mypy_django_plugin/main.py

+4
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
from mypy_django_plugin.django.context import DjangoContext
2323
from mypy_django_plugin.lib import fullnames, helpers
2424
from mypy_django_plugin.transformers import fields, forms, init_create, meta, querysets, request, settings
25+
from mypy_django_plugin.transformers.functional import resolve_str_promise_attribute
2526
from mypy_django_plugin.transformers.managers import (
2627
create_new_manager_class_from_from_queryset_method,
2728
resolve_manager_method,
@@ -285,6 +286,9 @@ def get_attribute_hook(self, fullname: str) -> Optional[Callable[[AttributeConte
285286
):
286287
return resolve_manager_method
287288

289+
if info and info.has_base(fullnames.STR_PROMISE_FULLNAME):
290+
return resolve_str_promise_attribute
291+
288292
return None
289293

290294
def get_type_analyze_hook(self, fullname: str) -> Optional[Callable[[AnalyzeTypeContext], MypyType]]:
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
from mypy.errorcodes import ATTR_DEFINED
2+
from mypy.nodes import CallExpr, MemberExpr
3+
from mypy.plugin import AttributeContext
4+
from mypy.types import AnyType, CallableType
5+
from mypy.types import Type as MypyType
6+
from mypy.types import TypeOfAny
7+
8+
from mypy_django_plugin.lib import helpers
9+
10+
11+
def resolve_str_promise_attribute(ctx: AttributeContext) -> MypyType:
12+
if isinstance(ctx.context, MemberExpr):
13+
method_name = ctx.context.name
14+
elif isinstance(ctx.context, CallExpr) and isinstance(ctx.context.callee, MemberExpr):
15+
method_name = ctx.context.callee.name
16+
else:
17+
ctx.api.fail(f'Cannot resolve the attribute of "{ctx.type}"', ctx.context, code=ATTR_DEFINED)
18+
return AnyType(TypeOfAny.from_error)
19+
20+
str_info = helpers.lookup_fully_qualified_typeinfo(helpers.get_typechecker_api(ctx), f"builtins.str")
21+
assert str_info is not None
22+
method = str_info.get(method_name)
23+
24+
if method is None or method.type is None:
25+
ctx.api.fail(f'"{ctx.type}" has no attribute "{method_name}"', ctx.context, code=ATTR_DEFINED)
26+
return AnyType(TypeOfAny.from_error)
27+
28+
if isinstance(method.type, CallableType):
29+
# The proxied str methods are only meant to be used as instance methods.
30+
# We need to drop the first `self` argument in them.
31+
assert method.type.arg_names[0] == "self"
32+
return method.type.copy_modified(
33+
arg_kinds=method.type.arg_kinds[1:],
34+
arg_names=method.type.arg_names[1:],
35+
arg_types=method.type.arg_types[1:],
36+
)
37+
else:
38+
# Not possible with `builtins.str`, but we have error handling for this anyway.
39+
ctx.api.fail(f'"{method_name}" on "{ctx.type}" is not a method', ctx.context)
40+
return AnyType(TypeOfAny.from_error)

tests/typecheck/utils/test_functional.yml

+31
Original file line numberDiff line numberDiff line change
@@ -16,3 +16,34 @@
1616
f = Foo()
1717
reveal_type(f.attr) # N: Revealed type is "builtins.list[builtins.str]"
1818
f.attr.name # E: "List[str]" has no attribute "name"
19+
20+
- case: str_promise_proxy
21+
main: |
22+
from typing import Union
23+
24+
from django.utils.functional import Promise, lazystr, _StrPromise
25+
26+
s = lazystr("asd")
27+
28+
reveal_type(s) # N: Revealed type is "django.utils.functional._StrPromise"
29+
30+
reveal_type(s.format("asd")) # N: Revealed type is "builtins.str"
31+
reveal_type(s.capitalize()) # N: Revealed type is "builtins.str"
32+
reveal_type(s.swapcase) # N: Revealed type is "def () -> builtins.str"
33+
reveal_type(s.__getnewargs__) # N: Revealed type is "def () -> Tuple[builtins.str]"
34+
s.nonsense # E: "django.utils.functional._StrPromise" has no attribute "nonsense"
35+
f: Union[_StrPromise, str]
36+
reveal_type(f.format("asd")) # N: Revealed type is "builtins.str"
37+
38+
reveal_type(s + "bar") # N: Revealed type is "builtins.str"
39+
reveal_type("foo" + s) # N: Revealed type is "Any"
40+
reveal_type(s % "asd") # N: Revealed type is "builtins.str"
41+
42+
def foo(content: str) -> None:
43+
...
44+
45+
def bar(content: Promise) -> None:
46+
...
47+
48+
foo(s) # E: Argument 1 to "foo" has incompatible type "_StrPromise"; expected "str"
49+
bar(s)

0 commit comments

Comments
 (0)