Skip to content

Commit 81efd6e

Browse files
✨ Added new Error to TypedDict (#14225)
Fixes #4617 This allows the following code to trigger the error `typeddict-unknown-key` ```python A = T.TypedDict("A", {"x": int}) def f(x: A) -> None: ... f({"x": 1, "y": "foo"}) # err: typeddict-unknown-key f({"y": "foo"}) # err: typeddict-unknown-key & typeddict-item f({"x": 'err', "y": "foo"}) # err: typeddict-unknown-key & typeddict-item a: A = { 'x': 1 } # You can set extra attributes a['extra'] = 'extra' # err: typeddict-unknown-key # Reading them produces the normal item error err = a['does not exist'] # err: typeddict-item ``` The user can then safely ignore this specific error at their disgression. Co-authored-by: Ivan Levkivskyi <[email protected]>
1 parent e778a58 commit 81efd6e

7 files changed

+103
-34
lines changed

docs/source/error_code_list.rst

+50
Original file line numberDiff line numberDiff line change
@@ -430,6 +430,56 @@ Example:
430430
# Error: Incompatible types (expression has type "float",
431431
# TypedDict item "x" has type "int") [typeddict-item]
432432
p: Point = {'x': 1.2, 'y': 4}
433+
434+
Check TypedDict Keys [typeddict-unknown-key]
435+
--------------------------------------------
436+
437+
When constructing a ``TypedDict`` object, mypy checks whether the definition
438+
contains unknown keys. For convenience's sake, mypy will not generate an error
439+
when a ``TypedDict`` has extra keys if it's passed to a function as an argument.
440+
However, it will generate an error when these are created. Example:
441+
442+
.. code-block:: python
443+
444+
from typing_extensions import TypedDict
445+
446+
class Point(TypedDict):
447+
x: int
448+
y: int
449+
450+
class Point3D(Point):
451+
z: int
452+
453+
def add_x_coordinates(a: Point, b: Point) -> int:
454+
return a["x"] + b["x"]
455+
456+
a: Point = {"x": 1, "y": 4}
457+
b: Point3D = {"x": 2, "y": 5, "z": 6}
458+
459+
# OK
460+
add_x_coordinates(a, b)
461+
# Error: Extra key "z" for TypedDict "Point" [typeddict-unknown-key]
462+
add_x_coordinates(a, {"x": 1, "y": 4, "z": 5})
463+
464+
465+
Setting an unknown value on a ``TypedDict`` will also generate this error:
466+
467+
.. code-block:: python
468+
469+
a: Point = {"x": 1, "y": 2}
470+
# Error: Extra key "z" for TypedDict "Point" [typeddict-unknown-key]
471+
a["z"] = 3
472+
473+
474+
Whereas reading an unknown value will generate the more generic/serious
475+
``typeddict-item``:
476+
477+
.. code-block:: python
478+
479+
a: Point = {"x": 1, "y": 2}
480+
# Error: TypedDict "Point" has no key "z" [typeddict-item]
481+
_ = a["z"]
482+
433483
434484
Check that type of target is known [has-type]
435485
---------------------------------------------

mypy/checkexpr.py

+12-6
Original file line numberDiff line numberDiff line change
@@ -790,17 +790,21 @@ def check_typeddict_call_with_kwargs(
790790
context: Context,
791791
orig_callee: Type | None,
792792
) -> Type:
793-
if not (callee.required_keys <= set(kwargs.keys()) <= set(callee.items.keys())):
793+
actual_keys = kwargs.keys()
794+
if not (callee.required_keys <= actual_keys <= callee.items.keys()):
794795
expected_keys = [
795796
key
796797
for key in callee.items.keys()
797-
if key in callee.required_keys or key in kwargs.keys()
798+
if key in callee.required_keys or key in actual_keys
798799
]
799-
actual_keys = kwargs.keys()
800800
self.msg.unexpected_typeddict_keys(
801801
callee, expected_keys=expected_keys, actual_keys=list(actual_keys), context=context
802802
)
803-
return AnyType(TypeOfAny.from_error)
803+
if callee.required_keys > actual_keys:
804+
# found_set is a sub-set of the required_keys
805+
# This means we're missing some keys and as such, we can't
806+
# properly type the object
807+
return AnyType(TypeOfAny.from_error)
804808

805809
orig_callee = get_proper_type(orig_callee)
806810
if isinstance(orig_callee, CallableType):
@@ -3777,7 +3781,9 @@ def nonliteral_tuple_index_helper(self, left_type: TupleType, index: Expression)
37773781
return self.chk.named_generic_type("builtins.tuple", [union])
37783782
return union
37793783

3780-
def visit_typeddict_index_expr(self, td_type: TypedDictType, index: Expression) -> Type:
3784+
def visit_typeddict_index_expr(
3785+
self, td_type: TypedDictType, index: Expression, setitem: bool = False
3786+
) -> Type:
37813787
if isinstance(index, StrExpr):
37823788
key_names = [index.value]
37833789
else:
@@ -3806,7 +3812,7 @@ def visit_typeddict_index_expr(self, td_type: TypedDictType, index: Expression)
38063812
for key_name in key_names:
38073813
value_type = td_type.items.get(key_name)
38083814
if value_type is None:
3809-
self.msg.typeddict_key_not_found(td_type, key_name, index)
3815+
self.msg.typeddict_key_not_found(td_type, key_name, index, setitem)
38103816
return AnyType(TypeOfAny.from_error)
38113817
else:
38123818
value_types.append(value_type)

mypy/checkmember.py

+3-1
Original file line numberDiff line numberDiff line change
@@ -1073,7 +1073,9 @@ def analyze_typeddict_access(
10731073
if isinstance(mx.context, IndexExpr):
10741074
# Since we can get this during `a['key'] = ...`
10751075
# it is safe to assume that the context is `IndexExpr`.
1076-
item_type = mx.chk.expr_checker.visit_typeddict_index_expr(typ, mx.context.index)
1076+
item_type = mx.chk.expr_checker.visit_typeddict_index_expr(
1077+
typ, mx.context.index, setitem=True
1078+
)
10771079
else:
10781080
# It can also be `a.__setitem__(...)` direct call.
10791081
# In this case `item_type` can be `Any`,

mypy/errorcodes.py

+3
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,9 @@ def __str__(self) -> str:
6767
TYPEDDICT_ITEM: Final = ErrorCode(
6868
"typeddict-item", "Check items when constructing TypedDict", "General"
6969
)
70+
TYPPEDICT_UNKNOWN_KEY: Final = ErrorCode(
71+
"typeddict-unknown-key", "Check unknown keys when constructing TypedDict", "General"
72+
)
7073
HAS_TYPE: Final = ErrorCode(
7174
"has-type", "Check that type of reference can be determined", "General"
7275
)

mypy/messages.py

+25-23
Original file line numberDiff line numberDiff line change
@@ -1637,30 +1637,28 @@ def unexpected_typeddict_keys(
16371637
expected_set = set(expected_keys)
16381638
if not typ.is_anonymous():
16391639
# Generate simpler messages for some common special cases.
1640-
if actual_set < expected_set:
1641-
# Use list comprehension instead of set operations to preserve order.
1642-
missing = [key for key in expected_keys if key not in actual_set]
1640+
# Use list comprehension instead of set operations to preserve order.
1641+
missing = [key for key in expected_keys if key not in actual_set]
1642+
if missing:
16431643
self.fail(
16441644
"Missing {} for TypedDict {}".format(
16451645
format_key_list(missing, short=True), format_type(typ)
16461646
),
16471647
context,
16481648
code=codes.TYPEDDICT_ITEM,
16491649
)
1650+
extra = [key for key in actual_keys if key not in expected_set]
1651+
if extra:
1652+
self.fail(
1653+
"Extra {} for TypedDict {}".format(
1654+
format_key_list(extra, short=True), format_type(typ)
1655+
),
1656+
context,
1657+
code=codes.TYPPEDICT_UNKNOWN_KEY,
1658+
)
1659+
if missing or extra:
1660+
# No need to check for further errors
16501661
return
1651-
else:
1652-
extra = [key for key in actual_keys if key not in expected_set]
1653-
if extra:
1654-
# If there are both extra and missing keys, only report extra ones for
1655-
# simplicity.
1656-
self.fail(
1657-
"Extra {} for TypedDict {}".format(
1658-
format_key_list(extra, short=True), format_type(typ)
1659-
),
1660-
context,
1661-
code=codes.TYPEDDICT_ITEM,
1662-
)
1663-
return
16641662
found = format_key_list(actual_keys, short=True)
16651663
if not expected_keys:
16661664
self.fail(f"Unexpected TypedDict {found}", context)
@@ -1680,8 +1678,15 @@ def typeddict_key_must_be_string_literal(self, typ: TypedDictType, context: Cont
16801678
)
16811679

16821680
def typeddict_key_not_found(
1683-
self, typ: TypedDictType, item_name: str, context: Context
1681+
self, typ: TypedDictType, item_name: str, context: Context, setitem: bool = False
16841682
) -> None:
1683+
"""Handle error messages for TypedDicts that have unknown keys.
1684+
1685+
Note, that we differentiate in between reading a value and setting a
1686+
value.
1687+
Setting a value on a TypedDict is an 'unknown-key' error, whereas
1688+
reading it is the more serious/general 'item' error.
1689+
"""
16851690
if typ.is_anonymous():
16861691
self.fail(
16871692
'"{}" is not a valid TypedDict key; expected one of {}'.format(
@@ -1690,17 +1695,14 @@ def typeddict_key_not_found(
16901695
context,
16911696
)
16921697
else:
1698+
err_code = codes.TYPPEDICT_UNKNOWN_KEY if setitem else codes.TYPEDDICT_ITEM
16931699
self.fail(
1694-
f'TypedDict {format_type(typ)} has no key "{item_name}"',
1695-
context,
1696-
code=codes.TYPEDDICT_ITEM,
1700+
f'TypedDict {format_type(typ)} has no key "{item_name}"', context, code=err_code
16971701
)
16981702
matches = best_matches(item_name, typ.items.keys(), n=3)
16991703
if matches:
17001704
self.note(
1701-
"Did you mean {}?".format(pretty_seq(matches, "or")),
1702-
context,
1703-
code=codes.TYPEDDICT_ITEM,
1705+
"Did you mean {}?".format(pretty_seq(matches, "or")), context, code=err_code
17041706
)
17051707

17061708
def typeddict_context_ambiguous(self, types: list[TypedDictType], context: Context) -> None:

test-data/unit/check-errorcodes.test

+8-3
Original file line numberDiff line numberDiff line change
@@ -455,11 +455,15 @@ class E(TypedDict):
455455
y: int
456456

457457
a: D = {'x': ''} # E: Incompatible types (expression has type "str", TypedDict item "x" has type "int") [typeddict-item]
458-
b: D = {'y': ''} # E: Extra key "y" for TypedDict "D" [typeddict-item]
458+
b: D = {'y': ''} # E: Missing key "x" for TypedDict "D" [typeddict-item] \
459+
# E: Extra key "y" for TypedDict "D" [typeddict-unknown-key]
459460
c = D(x=0) if int() else E(x=0, y=0)
460461
c = {} # E: Expected TypedDict key "x" but found no keys [typeddict-item]
462+
d: D = {'x': '', 'y': 1} # E: Extra key "y" for TypedDict "D" [typeddict-unknown-key] \
463+
# E: Incompatible types (expression has type "str", TypedDict item "x" has type "int") [typeddict-item]
461464

462-
a['y'] = 1 # E: TypedDict "D" has no key "y" [typeddict-item]
465+
466+
a['y'] = 1 # E: TypedDict "D" has no key "y" [typeddict-unknown-key]
463467
a['x'] = 'x' # E: Value of "x" has incompatible type "str"; expected "int" [typeddict-item]
464468
a['y'] # E: TypedDict "D" has no key "y" [typeddict-item]
465469
[builtins fixtures/dict.pyi]
@@ -472,7 +476,8 @@ class A(TypedDict):
472476
two_commonparts: int
473477

474478
a: A = {'one_commonpart': 1, 'two_commonparts': 2}
475-
a['other_commonpart'] = 3 # type: ignore[typeddict-item]
479+
a['other_commonpart'] = 3 # type: ignore[typeddict-unknown-key]
480+
not_exist = a['not_exist'] # type: ignore[typeddict-item]
476481
[builtins fixtures/dict.pyi]
477482
[typing fixtures/typing-typeddict.pyi]
478483

test-data/unit/check-typeddict.test

+2-1
Original file line numberDiff line numberDiff line change
@@ -2030,7 +2030,8 @@ v = {union: 2} # E: Expected TypedDict key to be string literal
20302030
num2: Literal['num']
20312031
v = {num2: 2}
20322032
bad2: Literal['bad']
2033-
v = {bad2: 2} # E: Extra key "bad" for TypedDict "Value"
2033+
v = {bad2: 2} # E: Missing key "num" for TypedDict "Value" \
2034+
# E: Extra key "bad" for TypedDict "Value"
20342035

20352036
[builtins fixtures/dict.pyi]
20362037
[typing fixtures/typing-typeddict.pyi]

0 commit comments

Comments
 (0)