Skip to content

Commit

Permalink
Add TypeVar resolution for as_manager and from_queryset querysets (ty…
Browse files Browse the repository at this point in the history
  • Loading branch information
moranabadie committed Aug 31, 2023
1 parent a8e42cb commit 719f7ed
Show file tree
Hide file tree
Showing 3 changed files with 81 additions and 1 deletion.
21 changes: 20 additions & 1 deletion mypy_django_plugin/transformers/managers.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
from mypy.plugin import AttributeContext, ClassDefContext, DynamicClassDefContext
from mypy.semanal import SemanticAnalyzer
from mypy.semanal_shared import has_placeholder
from mypy.types import AnyType, CallableType, Instance, ProperType, TypeOfAny
from mypy.types import AnyType, CallableType, Instance, ProperType, TypeOfAny, TypeVarType
from mypy.types import Type as MypyType
from mypy.typevars import fill_typevars

Expand Down Expand Up @@ -105,6 +105,9 @@ def get_funcdef_type(definition: Union[FuncBase, Decorator, None]) -> Optional[P
ret_type = Instance(queryset_info, [manager_instance.args[0], manager_instance.args[0]])
variables = []

if isinstance(ret_type, TypeVarType):
ret_type = _find_type_var(queryset_info, ret_type) or ret_type

# Drop any 'self' argument as our manager is already initialized
return method_type.copy_modified(
arg_types=method_type.arg_types[1:],
Expand All @@ -115,6 +118,22 @@ def get_funcdef_type(definition: Union[FuncBase, Decorator, None]) -> Optional[P
)


def _find_type_var(queryset_info: TypeInfo, ret_type: TypeVarType) -> Optional[MypyType]:
"""
Attempts to find the concrete type corresponding to a TypeVarType in a queryset's base types.
Example:
Suppose queryset_info is based on a generic class `QuerySet[T]` and ret_type corresponds
to `T`, then this function will return the concrete type that `T` was instantiated with.
"""
for base in queryset_info.bases:
for i, type_var in enumerate(base.type.type_vars):
type_var_path = f"{base.type.module_name}.{type_var}"
if type_var_path == ret_type.fullname and i < len(base.args):
return base.args[i]
return None


def get_method_type_from_reverse_manager(
api: TypeChecker, method_name: str, manager_type_info: TypeInfo
) -> Optional[ProperType]:
Expand Down
26 changes: 26 additions & 0 deletions tests/typecheck/managers/querysets/test_as_manager.yml
Original file line number Diff line number Diff line change
Expand Up @@ -146,6 +146,32 @@
class MyModel(models.Model):
objects = ManagerFromModelQuerySet
- case: handles_subclasses_of_queryset
main: |
from myapp.models import MyModel
reveal_type(MyModel.objects.example()) # N: Revealed type is "myapp.models.MyModel"
installed_apps:
- myapp
files:
- path: myapp/__init__.py
- path: myapp/models.py
content: |
from typing import TypeVar
from django.db import models
_CTE = TypeVar("_CTE", bound=models.Model)
class _MyModelQuerySet(models.QuerySet[_CTE]):
def example(self) -> _CTE: ...
class MyModelQuerySet(_MyModelQuerySet["MyModel"]):
...
class MyModel(models.Model):
objects = MyModelQuerySet.as_manager()
- case: reuses_generated_type_when_called_identically_for_multiple_managers
main: |
from myapp.models import MyModel
Expand Down
35 changes: 35 additions & 0 deletions tests/typecheck/managers/querysets/test_from_queryset.yml
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,41 @@
NewManager = BaseManager.from_queryset(ModelQuerySet)
class MyModel(models.Model):
objects = NewManager()
- case: handles_subclasses_of_queryset
main: |
from myapp.models import MyModel
reveal_type(MyModel.objects.example()) # N: Revealed type is "myapp.models.MyModel"
installed_apps:
- myapp
files:
- path: myapp/__init__.py
- path: myapp/querysets.py
content: |
from typing import TypeVar, TYPE_CHECKING
from django.db import models
from django.db.models.manager import BaseManager
if TYPE_CHECKING:
from .models import MyModel
_CTE = TypeVar("_CTE", bound=models.Model)
class _MyModelQuerySet(models.QuerySet[_CTE]):
def example(self) -> _CTE: ...
class MyModelQuerySet(_MyModelQuerySet["MyModel"]):
...
- path: myapp/models.py
content: |
from django.db import models
from django.db.models.manager import BaseManager
from .querysets import MyModelQuerySet
NewManager = BaseManager.from_queryset(MyModelQuerySet)
class MyModel(models.Model):
objects = NewManager()
- case: from_queryset_generated_manager_imported_from_other_module
main: |
Expand Down

0 comments on commit 719f7ed

Please sign in to comment.