Created
February 24, 2025 01:43
-
-
Save federicobond/54bdc70e2e7c40cfcb8424b0de0c5534 to your computer and use it in GitHub Desktop.
Mypy Celery plugin to patch the return type of shared_task and Celery.task decorators with a Django-specific task class.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from collections.abc import Callable | |
from mypy.nodes import MypyFile | |
from mypy.plugin import FunctionContext, MethodContext, Plugin | |
from mypy.types import CallableType, Instance, Type, get_proper_type | |
DJANGO_TASK_FULLNAME = "celery.contrib.django.task.DjangoTask" | |
class CeleryPlugin(Plugin): | |
def get_function_hook( | |
self, fullname: str | |
) -> Callable[[FunctionContext], Type] | None: | |
if fullname == "celery.app.shared_task": | |
return self.celery_task_hook | |
return None | |
def get_method_hook(self, fullname: str) -> Callable[[MethodContext], Type] | None: | |
if fullname == "celery.app.base.Celery.task": | |
return self.celery_task_hook | |
return None | |
def celery_task_hook(self, context: FunctionContext | MethodContext) -> Type: | |
sym = self.lookup_fully_qualified(DJANGO_TASK_FULLNAME) | |
return_type = get_proper_type(context.default_return_type) | |
if isinstance(return_type, CallableType): | |
# decorator called with arguments | |
return return_type.copy_modified( | |
ret_type=Instance(sym.node, return_type.ret_type.args) | |
) | |
return Instance(sym.node, return_type.args) | |
def get_additional_deps(self, file: MypyFile) -> list[tuple[int, str, int]]: | |
if "celery" in file.fullname: | |
return [(10, DJANGO_TASK_FULLNAME.rsplit(".", 1)[0], -1)] | |
return [] | |
def plugin(version: str) -> type[CeleryPlugin]: | |
return CeleryPlugin |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment