|
| 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) |
0 commit comments