Skip to content

Commit c665ad2

Browse files
committed
Warn about redundant casts
A cast is considered redundant if the target type of the cast is the same as the inferred type of the expression. A cast to a supertype like `cast(object, 1)` is not considered redundant because such a cast could be needed to work around deficiencies in type inference.
1 parent c7cfb69 commit c665ad2

File tree

7 files changed

+54
-2
lines changed

7 files changed

+54
-2
lines changed

mypy/build.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,8 @@
6868
CHECK_UNTYPED_DEFS = 'check-untyped-defs'
6969
# Also check typeshed for missing annotations
7070
WARN_INCOMPLETE_STUB = 'warn-incomplete-stub'
71+
# Warn about casting an expression to its inferred type
72+
WARN_REDUNDANT_CASTS = 'warn-redundant-casts'
7173

7274
PYTHON_EXTENSIONS = ['.pyi', '.py']
7375

@@ -386,7 +388,8 @@ def __init__(self, data_dir: str,
386388
DISALLOW_UNTYPED_CALLS in self.flags,
387389
DISALLOW_UNTYPED_DEFS in self.flags,
388390
check_untyped_defs,
389-
WARN_INCOMPLETE_STUB in self.flags)
391+
WARN_INCOMPLETE_STUB in self.flags,
392+
WARN_REDUNDANT_CASTS in self.flags)
390393
self.missing_modules = set() # type: Set[str]
391394

392395
def all_imported_modules_in_file(self,

mypy/checker.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -380,12 +380,14 @@ class TypeChecker(NodeVisitor[Type]):
380380
# Should we check untyped function defs?
381381
check_untyped_defs = False
382382
warn_incomplete_stub = False
383+
warn_redundant_casts = False
383384
is_typeshed_stub = False
384385

385386
def __init__(self, errors: Errors, modules: Dict[str, MypyFile],
386387
pyversion: Tuple[int, int] = defaults.PYTHON3_VERSION,
387388
disallow_untyped_calls=False, disallow_untyped_defs=False,
388-
check_untyped_defs=False, warn_incomplete_stub=False) -> None:
389+
check_untyped_defs=False, warn_incomplete_stub=False,
390+
warn_redundant_casts=False) -> None:
389391
"""Construct a type checker.
390392
391393
Use errors to report type check errors.
@@ -411,6 +413,7 @@ def __init__(self, errors: Errors, modules: Dict[str, MypyFile],
411413
self.disallow_untyped_defs = disallow_untyped_defs
412414
self.check_untyped_defs = check_untyped_defs
413415
self.warn_incomplete_stub = warn_incomplete_stub
416+
self.warn_redundant_casts = warn_redundant_casts
414417

415418
def visit_file(self, file_node: MypyFile, path: str) -> None:
416419
"""Type check a mypy file with the given path."""

mypy/checkexpr.py

+2
Original file line numberDiff line numberDiff line change
@@ -1229,6 +1229,8 @@ def visit_cast_expr(self, expr: CastExpr) -> Type:
12291229
"""Type check a cast expression."""
12301230
source_type = self.accept(expr.expr, context=AnyType())
12311231
target_type = expr.type
1232+
if self.chk.warn_redundant_casts and is_same_type(source_type, target_type):
1233+
self.msg.redundant_cast(target_type, expr)
12321234
if not self.is_valid_cast(source_type, target_type):
12331235
self.msg.invalid_cast(target_type, source_type, expr)
12341236
return target_type

mypy/main.py

+4
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,8 @@ def parse_version(v):
153153
parser.add_argument('--warn-incomplete-stub', action='store_true',
154154
help="warn if missing type annotation in typeshed, only relevant with"
155155
" --check-untyped-defs enabled")
156+
parser.add_argument('--warn-redundant-casts', action='store_true',
157+
help="warn about casting an expression to its inferred type")
156158
parser.add_argument('--fast-parser', action='store_true',
157159
help="enable experimental fast parser")
158160
parser.add_argument('-i', '--incremental', action='store_true',
@@ -251,6 +253,8 @@ def parse_version(v):
251253

252254
if args.warn_incomplete_stub:
253255
options.build_flags.append(build.WARN_INCOMPLETE_STUB)
256+
if args.warn_redundant_casts:
257+
options.build_flags.append(build.WARN_REDUNDANT_CASTS)
254258

255259
# experimental
256260
if args.fast_parser:

mypy/messages.py

+3
Original file line numberDiff line numberDiff line change
@@ -841,6 +841,9 @@ def reveal_type(self, typ: Type, context: Context) -> None:
841841
def unsupported_type_type(self, item: Type, context: Context) -> None:
842842
self.fail('Unsupported type Type[{}]'.format(self.format(item)), context)
843843

844+
def redundant_cast(self, typ: Type, context: Context) -> None:
845+
self.note('Redundant cast to {}'.format(self.format(typ)), context)
846+
844847

845848
def capitalize(s: str) -> str:
846849
"""Capitalize the first character of a string."""

mypy/test/testcheck.py

+1
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,7 @@
6060
'check-bound.test',
6161
'check-optional.test',
6262
'check-fastparse.test',
63+
'check-warnings.test',
6364
]
6465

6566

test-data/unit/check-warnings.test

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
-- Test cases for warning generation.
2+
3+
-- Redundant casts
4+
-- ---------------
5+
6+
[case testRedundantCast]
7+
# flags: warn-redundant-casts
8+
from typing import cast
9+
a = 1
10+
b = cast(str, a)
11+
c = cast(int, a)
12+
[out]
13+
main:5: note: Redundant cast to "int"
14+
15+
[case testRedundantCastWithIsinstance]
16+
# flags: warn-redundant-casts
17+
from typing import cast, Union
18+
x = 1 # type: Union[int, str]
19+
if isinstance(x, str):
20+
cast(str, x)
21+
[builtins fixtures/isinstance.py]
22+
[out]
23+
main:5: note: Redundant cast to "str"
24+
25+
[case testCastToSuperclassNotRedundant]
26+
# flags: warn-redundant-casts
27+
from typing import cast, TypeVar, List
28+
T = TypeVar('T')
29+
def add(xs: List[T], ys: List[T]) -> List[T]: pass
30+
class A: pass
31+
class B(A): pass
32+
a = A()
33+
b = B()
34+
# Without the cast, the following line would fail to type check.
35+
c = add([cast(A, b)], [a])
36+
[builtins fixtures/list.py]

0 commit comments

Comments
 (0)