Commit 4e81296b authored by Eric V. Smith's avatar Eric V. Smith Committed by GitHub

bpo-33536: Validate make_dataclass() field names. (GH-6906)

parent 5db5c066
...@@ -3,6 +3,7 @@ import sys ...@@ -3,6 +3,7 @@ import sys
import copy import copy
import types import types
import inspect import inspect
import keyword
__all__ = ['dataclass', __all__ = ['dataclass',
'field', 'field',
...@@ -1100,6 +1101,9 @@ def make_dataclass(cls_name, fields, *, bases=(), namespace=None, init=True, ...@@ -1100,6 +1101,9 @@ def make_dataclass(cls_name, fields, *, bases=(), namespace=None, init=True,
# Copy namespace since we're going to mutate it. # Copy namespace since we're going to mutate it.
namespace = namespace.copy() namespace = namespace.copy()
# While we're looking through the field names, validate that they
# are identifiers, are not keywords, and not duplicates.
seen = set()
anns = {} anns = {}
for item in fields: for item in fields:
if isinstance(item, str): if isinstance(item, str):
...@@ -1110,6 +1114,17 @@ def make_dataclass(cls_name, fields, *, bases=(), namespace=None, init=True, ...@@ -1110,6 +1114,17 @@ def make_dataclass(cls_name, fields, *, bases=(), namespace=None, init=True,
elif len(item) == 3: elif len(item) == 3:
name, tp, spec = item name, tp, spec = item
namespace[name] = spec namespace[name] = spec
else:
raise TypeError(f'Invalid field: {item!r}')
if not isinstance(name, str) or not name.isidentifier():
raise TypeError(f'Field names must be valid identifers: {name!r}')
if keyword.iskeyword(name):
raise TypeError(f'Field names must not be keywords: {name!r}')
if name in seen:
raise TypeError(f'Field name duplicated: {name!r}')
seen.add(name)
anns[name] = tp anns[name] = tp
namespace['__annotations__'] = anns namespace['__annotations__'] = anns
......
This diff is collapsed.
dataclasses.make_dataclass now checks for invalid field names and duplicate
fields. Also, added a check for invalid field specifications.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment