mirror of
https://github.com/graphql-python/graphene.git
synced 2025-07-18 12:02:19 +03:00
Use black in dataclasses
This commit is contained in:
parent
015bfa3452
commit
c17103005c
|
@ -9,21 +9,21 @@ import types
|
||||||
import inspect
|
import inspect
|
||||||
import keyword
|
import keyword
|
||||||
|
|
||||||
__all__ = ['dataclass',
|
__all__ = [
|
||||||
'field',
|
"dataclass",
|
||||||
'Field',
|
"field",
|
||||||
'FrozenInstanceError',
|
"Field",
|
||||||
'InitVar',
|
"FrozenInstanceError",
|
||||||
'MISSING',
|
"InitVar",
|
||||||
|
"MISSING",
|
||||||
# Helper functions.
|
# Helper functions.
|
||||||
'fields',
|
"fields",
|
||||||
'asdict',
|
"asdict",
|
||||||
'astuple',
|
"astuple",
|
||||||
'make_dataclass',
|
"make_dataclass",
|
||||||
'replace',
|
"replace",
|
||||||
'is_dataclass',
|
"is_dataclass",
|
||||||
]
|
]
|
||||||
|
|
||||||
# Conditions for adding methods. The boxes indicate what action the
|
# Conditions for adding methods. The boxes indicate what action the
|
||||||
# dataclass decorator takes. For all of these tables, when I talk
|
# dataclass decorator takes. For all of these tables, when I talk
|
||||||
|
@ -152,20 +152,26 @@ __all__ = ['dataclass',
|
||||||
|
|
||||||
|
|
||||||
# Raised when an attempt is made to modify a frozen class.
|
# Raised when an attempt is made to modify a frozen class.
|
||||||
class FrozenInstanceError(AttributeError): pass
|
class FrozenInstanceError(AttributeError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
# A sentinel object for default values to signal that a default
|
# A sentinel object for default values to signal that a default
|
||||||
# factory will be used. This is given a nice repr() which will appear
|
# factory will be used. This is given a nice repr() which will appear
|
||||||
# in the function signature of dataclasses' constructors.
|
# in the function signature of dataclasses' constructors.
|
||||||
class _HAS_DEFAULT_FACTORY_CLASS:
|
class _HAS_DEFAULT_FACTORY_CLASS:
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return '<factory>'
|
return "<factory>"
|
||||||
|
|
||||||
|
|
||||||
_HAS_DEFAULT_FACTORY = _HAS_DEFAULT_FACTORY_CLASS()
|
_HAS_DEFAULT_FACTORY = _HAS_DEFAULT_FACTORY_CLASS()
|
||||||
|
|
||||||
# A sentinel object to detect if a parameter is supplied or not. Use
|
# A sentinel object to detect if a parameter is supplied or not. Use
|
||||||
# a class to give it a better repr.
|
# a class to give it a better repr.
|
||||||
class _MISSING_TYPE:
|
class _MISSING_TYPE:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
MISSING = _MISSING_TYPE()
|
MISSING = _MISSING_TYPE()
|
||||||
|
|
||||||
# Since most per-field metadata will be unused, create an empty
|
# Since most per-field metadata will be unused, create an empty
|
||||||
|
@ -176,33 +182,38 @@ _EMPTY_METADATA = types.MappingProxyType({})
|
||||||
class _FIELD_BASE:
|
class _FIELD_BASE:
|
||||||
def __init__(self, name):
|
def __init__(self, name):
|
||||||
self.name = name
|
self.name = name
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return self.name
|
return self.name
|
||||||
_FIELD = _FIELD_BASE('_FIELD')
|
|
||||||
_FIELD_CLASSVAR = _FIELD_BASE('_FIELD_CLASSVAR')
|
|
||||||
_FIELD_INITVAR = _FIELD_BASE('_FIELD_INITVAR')
|
_FIELD = _FIELD_BASE("_FIELD")
|
||||||
|
_FIELD_CLASSVAR = _FIELD_BASE("_FIELD_CLASSVAR")
|
||||||
|
_FIELD_INITVAR = _FIELD_BASE("_FIELD_INITVAR")
|
||||||
|
|
||||||
# The name of an attribute on the class where we store the Field
|
# The name of an attribute on the class where we store the Field
|
||||||
# objects. Also used to check if a class is a Data Class.
|
# objects. Also used to check if a class is a Data Class.
|
||||||
_FIELDS = '__dataclass_fields__'
|
_FIELDS = "__dataclass_fields__"
|
||||||
|
|
||||||
# The name of an attribute on the class that stores the parameters to
|
# The name of an attribute on the class that stores the parameters to
|
||||||
# @dataclass.
|
# @dataclass.
|
||||||
_PARAMS = '__dataclass_params__'
|
_PARAMS = "__dataclass_params__"
|
||||||
|
|
||||||
# The name of the function, that if it exists, is called at the end of
|
# The name of the function, that if it exists, is called at the end of
|
||||||
# __init__.
|
# __init__.
|
||||||
_POST_INIT_NAME = '__post_init__'
|
_POST_INIT_NAME = "__post_init__"
|
||||||
|
|
||||||
# String regex that string annotations for ClassVar or InitVar must match.
|
# String regex that string annotations for ClassVar or InitVar must match.
|
||||||
# Allows "identifier.identifier[" or "identifier[".
|
# Allows "identifier.identifier[" or "identifier[".
|
||||||
# https://bugs.python.org/issue33453 for details.
|
# https://bugs.python.org/issue33453 for details.
|
||||||
_MODULE_IDENTIFIER_RE = re.compile(r'^(?:\s*(\w+)\s*\.)?\s*(\w+)')
|
_MODULE_IDENTIFIER_RE = re.compile(r"^(?:\s*(\w+)\s*\.)?\s*(\w+)")
|
||||||
|
|
||||||
|
|
||||||
class _InitVarMeta(type):
|
class _InitVarMeta(type):
|
||||||
def __getitem__(self, params):
|
def __getitem__(self, params):
|
||||||
return self
|
return self
|
||||||
|
|
||||||
|
|
||||||
class InitVar(metaclass=_InitVarMeta):
|
class InitVar(metaclass=_InitVarMeta):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -218,20 +229,20 @@ class InitVar(metaclass=_InitVarMeta):
|
||||||
# When cls._FIELDS is filled in with a list of Field objects, the name
|
# When cls._FIELDS is filled in with a list of Field objects, the name
|
||||||
# and type fields will have been populated.
|
# and type fields will have been populated.
|
||||||
class Field:
|
class Field:
|
||||||
__slots__ = ('name',
|
__slots__ = (
|
||||||
'type',
|
"name",
|
||||||
'default',
|
"type",
|
||||||
'default_factory',
|
"default",
|
||||||
'repr',
|
"default_factory",
|
||||||
'hash',
|
"repr",
|
||||||
'init',
|
"hash",
|
||||||
'compare',
|
"init",
|
||||||
'metadata',
|
"compare",
|
||||||
'_field_type', # Private: not to be used by user code.
|
"metadata",
|
||||||
|
"_field_type", # Private: not to be used by user code.
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, default, default_factory, init, repr, hash, compare,
|
def __init__(self, default, default_factory, init, repr, hash, compare, metadata):
|
||||||
metadata):
|
|
||||||
self.name = None
|
self.name = None
|
||||||
self.type = None
|
self.type = None
|
||||||
self.default = default
|
self.default = default
|
||||||
|
@ -240,24 +251,28 @@ class Field:
|
||||||
self.repr = repr
|
self.repr = repr
|
||||||
self.hash = hash
|
self.hash = hash
|
||||||
self.compare = compare
|
self.compare = compare
|
||||||
self.metadata = (_EMPTY_METADATA
|
self.metadata = (
|
||||||
if metadata is None or len(metadata) == 0 else
|
_EMPTY_METADATA
|
||||||
types.MappingProxyType(metadata))
|
if metadata is None or len(metadata) == 0
|
||||||
|
else types.MappingProxyType(metadata)
|
||||||
|
)
|
||||||
self._field_type = None
|
self._field_type = None
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return ('Field('
|
return (
|
||||||
f'name={self.name!r},'
|
"Field("
|
||||||
f'type={self.type!r},'
|
f"name={self.name!r},"
|
||||||
f'default={self.default!r},'
|
f"type={self.type!r},"
|
||||||
f'default_factory={self.default_factory!r},'
|
f"default={self.default!r},"
|
||||||
f'init={self.init!r},'
|
f"default_factory={self.default_factory!r},"
|
||||||
f'repr={self.repr!r},'
|
f"init={self.init!r},"
|
||||||
f'hash={self.hash!r},'
|
f"repr={self.repr!r},"
|
||||||
f'compare={self.compare!r},'
|
f"hash={self.hash!r},"
|
||||||
f'metadata={self.metadata!r},'
|
f"compare={self.compare!r},"
|
||||||
f'_field_type={self._field_type}'
|
f"metadata={self.metadata!r},"
|
||||||
')')
|
f"_field_type={self._field_type}"
|
||||||
|
")"
|
||||||
|
)
|
||||||
|
|
||||||
# This is used to support the PEP 487 __set_name__ protocol in the
|
# This is used to support the PEP 487 __set_name__ protocol in the
|
||||||
# case where we're using a field that contains a descriptor as a
|
# case where we're using a field that contains a descriptor as a
|
||||||
|
@ -268,7 +283,7 @@ class Field:
|
||||||
# with the default value, so the end result is a descriptor that
|
# with the default value, so the end result is a descriptor that
|
||||||
# had __set_name__ called on it at the right time.
|
# had __set_name__ called on it at the right time.
|
||||||
def __set_name__(self, owner, name):
|
def __set_name__(self, owner, name):
|
||||||
func = getattr(type(self.default), '__set_name__', None)
|
func = getattr(type(self.default), "__set_name__", None)
|
||||||
if func:
|
if func:
|
||||||
# There is a __set_name__ method on the descriptor, call
|
# There is a __set_name__ method on the descriptor, call
|
||||||
# it.
|
# it.
|
||||||
|
@ -276,12 +291,13 @@ class Field:
|
||||||
|
|
||||||
|
|
||||||
class _DataclassParams:
|
class _DataclassParams:
|
||||||
__slots__ = ('init',
|
__slots__ = (
|
||||||
'repr',
|
"init",
|
||||||
'eq',
|
"repr",
|
||||||
'order',
|
"eq",
|
||||||
'unsafe_hash',
|
"order",
|
||||||
'frozen',
|
"unsafe_hash",
|
||||||
|
"frozen",
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, init, repr, eq, order, unsafe_hash, frozen):
|
def __init__(self, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
|
@ -293,21 +309,31 @@ class _DataclassParams:
|
||||||
self.frozen = frozen
|
self.frozen = frozen
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return ('_DataclassParams('
|
return (
|
||||||
f'init={self.init!r},'
|
"_DataclassParams("
|
||||||
f'repr={self.repr!r},'
|
f"init={self.init!r},"
|
||||||
f'eq={self.eq!r},'
|
f"repr={self.repr!r},"
|
||||||
f'order={self.order!r},'
|
f"eq={self.eq!r},"
|
||||||
f'unsafe_hash={self.unsafe_hash!r},'
|
f"order={self.order!r},"
|
||||||
f'frozen={self.frozen!r}'
|
f"unsafe_hash={self.unsafe_hash!r},"
|
||||||
')')
|
f"frozen={self.frozen!r}"
|
||||||
|
")"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
# This function is used instead of exposing Field creation directly,
|
# This function is used instead of exposing Field creation directly,
|
||||||
# so that a type checker can be told (via overloads) that this is a
|
# so that a type checker can be told (via overloads) that this is a
|
||||||
# function whose type depends on its parameters.
|
# function whose type depends on its parameters.
|
||||||
def field(*, default=MISSING, default_factory=MISSING, init=True, repr=True,
|
def field(
|
||||||
hash=None, compare=True, metadata=None):
|
*,
|
||||||
|
default=MISSING,
|
||||||
|
default_factory=MISSING,
|
||||||
|
init=True,
|
||||||
|
repr=True,
|
||||||
|
hash=None,
|
||||||
|
compare=True,
|
||||||
|
metadata=None,
|
||||||
|
):
|
||||||
"""Return an object to identify dataclass fields.
|
"""Return an object to identify dataclass fields.
|
||||||
|
|
||||||
default is the default value of the field. default_factory is a
|
default is the default value of the field. default_factory is a
|
||||||
|
@ -323,9 +349,8 @@ def field(*, default=MISSING, default_factory=MISSING, init=True, repr=True,
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if default is not MISSING and default_factory is not MISSING:
|
if default is not MISSING and default_factory is not MISSING:
|
||||||
raise ValueError('cannot specify both default and default_factory')
|
raise ValueError("cannot specify both default and default_factory")
|
||||||
return Field(default, default_factory, init, repr, hash, compare,
|
return Field(default, default_factory, init, repr, hash, compare, metadata)
|
||||||
metadata)
|
|
||||||
|
|
||||||
|
|
||||||
def _tuple_str(obj_name, fields):
|
def _tuple_str(obj_name, fields):
|
||||||
|
@ -335,27 +360,26 @@ def _tuple_str(obj_name, fields):
|
||||||
|
|
||||||
# Special case for the 0-tuple.
|
# Special case for the 0-tuple.
|
||||||
if not fields:
|
if not fields:
|
||||||
return '()'
|
return "()"
|
||||||
# Note the trailing comma, needed if this turns out to be a 1-tuple.
|
# Note the trailing comma, needed if this turns out to be a 1-tuple.
|
||||||
return f'({",".join([f"{obj_name}.{f.name}" for f in fields])},)'
|
return f'({",".join([f"{obj_name}.{f.name}" for f in fields])},)'
|
||||||
|
|
||||||
|
|
||||||
def _create_fn(name, args, body, *, globals=None, locals=None,
|
def _create_fn(name, args, body, *, globals=None, locals=None, return_type=MISSING):
|
||||||
return_type=MISSING):
|
|
||||||
# Note that we mutate locals when exec() is called. Caller
|
# Note that we mutate locals when exec() is called. Caller
|
||||||
# beware! The only callers are internal to this module, so no
|
# beware! The only callers are internal to this module, so no
|
||||||
# worries about external callers.
|
# worries about external callers.
|
||||||
if locals is None:
|
if locals is None:
|
||||||
locals = {}
|
locals = {}
|
||||||
return_annotation = ''
|
return_annotation = ""
|
||||||
if return_type is not MISSING:
|
if return_type is not MISSING:
|
||||||
locals['_return_type'] = return_type
|
locals["_return_type"] = return_type
|
||||||
return_annotation = '->_return_type'
|
return_annotation = "->_return_type"
|
||||||
args = ','.join(args)
|
args = ",".join(args)
|
||||||
body = '\n'.join(f' {b}' for b in body)
|
body = "\n".join(f" {b}" for b in body)
|
||||||
|
|
||||||
# Compute the text of the entire function.
|
# Compute the text of the entire function.
|
||||||
txt = f'def {name}({args}){return_annotation}:\n{body}'
|
txt = f"def {name}({args}){return_annotation}:\n{body}"
|
||||||
|
|
||||||
exec(txt, globals, locals)
|
exec(txt, globals, locals)
|
||||||
return locals[name]
|
return locals[name]
|
||||||
|
@ -369,23 +393,25 @@ def _field_assign(frozen, name, value, self_name):
|
||||||
# self_name is what "self" is called in this function: don't
|
# self_name is what "self" is called in this function: don't
|
||||||
# hard-code "self", since that might be a field name.
|
# hard-code "self", since that might be a field name.
|
||||||
if frozen:
|
if frozen:
|
||||||
return f'object.__setattr__({self_name},{name!r},{value})'
|
return f"object.__setattr__({self_name},{name!r},{value})"
|
||||||
return f'{self_name}.{name}={value}'
|
return f"{self_name}.{name}={value}"
|
||||||
|
|
||||||
|
|
||||||
def _field_init(f, frozen, globals, self_name):
|
def _field_init(f, frozen, globals, self_name):
|
||||||
# Return the text of the line in the body of __init__ that will
|
# Return the text of the line in the body of __init__ that will
|
||||||
# initialize this field.
|
# initialize this field.
|
||||||
|
|
||||||
default_name = f'_dflt_{f.name}'
|
default_name = f"_dflt_{f.name}"
|
||||||
if f.default_factory is not MISSING:
|
if f.default_factory is not MISSING:
|
||||||
if f.init:
|
if f.init:
|
||||||
# This field has a default factory. If a parameter is
|
# This field has a default factory. If a parameter is
|
||||||
# given, use it. If not, call the factory.
|
# given, use it. If not, call the factory.
|
||||||
globals[default_name] = f.default_factory
|
globals[default_name] = f.default_factory
|
||||||
value = (f'{default_name}() '
|
value = (
|
||||||
f'if {f.name} is _HAS_DEFAULT_FACTORY '
|
f"{default_name}() "
|
||||||
f'else {f.name}')
|
f"if {f.name} is _HAS_DEFAULT_FACTORY "
|
||||||
|
f"else {f.name}"
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
# This is a field that's not in the __init__ params, but
|
# This is a field that's not in the __init__ params, but
|
||||||
# has a default factory function. It needs to be
|
# has a default factory function. It needs to be
|
||||||
|
@ -402,7 +428,7 @@ def _field_init(f, frozen, globals, self_name):
|
||||||
# (which, after all, is why we have a factory function!).
|
# (which, after all, is why we have a factory function!).
|
||||||
|
|
||||||
globals[default_name] = f.default_factory
|
globals[default_name] = f.default_factory
|
||||||
value = f'{default_name}()'
|
value = f"{default_name}()"
|
||||||
else:
|
else:
|
||||||
# No default factory.
|
# No default factory.
|
||||||
if f.init:
|
if f.init:
|
||||||
|
@ -435,15 +461,15 @@ def _init_param(f):
|
||||||
if f.default is MISSING and f.default_factory is MISSING:
|
if f.default is MISSING and f.default_factory is MISSING:
|
||||||
# There's no default, and no default_factory, just output the
|
# There's no default, and no default_factory, just output the
|
||||||
# variable name and type.
|
# variable name and type.
|
||||||
default = ''
|
default = ""
|
||||||
elif f.default is not MISSING:
|
elif f.default is not MISSING:
|
||||||
# There's a default, this will be the name that's used to look
|
# There's a default, this will be the name that's used to look
|
||||||
# it up.
|
# it up.
|
||||||
default = f'=_dflt_{f.name}'
|
default = f"=_dflt_{f.name}"
|
||||||
elif f.default_factory is not MISSING:
|
elif f.default_factory is not MISSING:
|
||||||
# There's a factory function. Set a marker.
|
# There's a factory function. Set a marker.
|
||||||
default = '=_HAS_DEFAULT_FACTORY'
|
default = "=_HAS_DEFAULT_FACTORY"
|
||||||
return f'{f.name}:_type_{f.name}{default}'
|
return f"{f.name}:_type_{f.name}{default}"
|
||||||
|
|
||||||
|
|
||||||
def _init_fn(fields, frozen, has_post_init, self_name):
|
def _init_fn(fields, frozen, has_post_init, self_name):
|
||||||
|
@ -461,11 +487,11 @@ def _init_fn(fields, frozen, has_post_init, self_name):
|
||||||
if not (f.default is MISSING and f.default_factory is MISSING):
|
if not (f.default is MISSING and f.default_factory is MISSING):
|
||||||
seen_default = True
|
seen_default = True
|
||||||
elif seen_default:
|
elif seen_default:
|
||||||
raise TypeError(f'non-default argument {f.name!r} '
|
raise TypeError(
|
||||||
'follows default argument')
|
f"non-default argument {f.name!r} " "follows default argument"
|
||||||
|
)
|
||||||
|
|
||||||
globals = {'MISSING': MISSING,
|
globals = {"MISSING": MISSING, "_HAS_DEFAULT_FACTORY": _HAS_DEFAULT_FACTORY}
|
||||||
'_HAS_DEFAULT_FACTORY': _HAS_DEFAULT_FACTORY}
|
|
||||||
|
|
||||||
body_lines = []
|
body_lines = []
|
||||||
for f in fields:
|
for f in fields:
|
||||||
|
@ -477,54 +503,66 @@ def _init_fn(fields, frozen, has_post_init, self_name):
|
||||||
|
|
||||||
# Does this class have a post-init function?
|
# Does this class have a post-init function?
|
||||||
if has_post_init:
|
if has_post_init:
|
||||||
params_str = ','.join(f.name for f in fields
|
params_str = ",".join(f.name for f in fields if f._field_type is _FIELD_INITVAR)
|
||||||
if f._field_type is _FIELD_INITVAR)
|
body_lines.append(f"{self_name}.{_POST_INIT_NAME}({params_str})")
|
||||||
body_lines.append(f'{self_name}.{_POST_INIT_NAME}({params_str})')
|
|
||||||
|
|
||||||
# If no body lines, use 'pass'.
|
# If no body lines, use 'pass'.
|
||||||
if not body_lines:
|
if not body_lines:
|
||||||
body_lines = ['pass']
|
body_lines = ["pass"]
|
||||||
|
|
||||||
locals = {f'_type_{f.name}': f.type for f in fields}
|
locals = {f"_type_{f.name}": f.type for f in fields}
|
||||||
return _create_fn('__init__',
|
return _create_fn(
|
||||||
|
"__init__",
|
||||||
[self_name] + [_init_param(f) for f in fields if f.init],
|
[self_name] + [_init_param(f) for f in fields if f.init],
|
||||||
body_lines,
|
body_lines,
|
||||||
locals=locals,
|
locals=locals,
|
||||||
globals=globals,
|
globals=globals,
|
||||||
return_type=None)
|
return_type=None,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def _repr_fn(fields):
|
def _repr_fn(fields):
|
||||||
return _create_fn('__repr__',
|
return _create_fn(
|
||||||
('self',),
|
"__repr__",
|
||||||
['return self.__class__.__qualname__ + f"(' +
|
("self",),
|
||||||
', '.join([f"{f.name}={{self.{f.name}!r}}"
|
[
|
||||||
for f in fields]) +
|
'return self.__class__.__qualname__ + f"('
|
||||||
')"'])
|
+ ", ".join([f"{f.name}={{self.{f.name}!r}}" for f in fields])
|
||||||
|
+ ')"'
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def _frozen_get_del_attr(cls, fields):
|
def _frozen_get_del_attr(cls, fields):
|
||||||
# XXX: globals is modified on the first call to _create_fn, then
|
# XXX: globals is modified on the first call to _create_fn, then
|
||||||
# the modified version is used in the second call. Is this okay?
|
# the modified version is used in the second call. Is this okay?
|
||||||
globals = {'cls': cls,
|
globals = {"cls": cls, "FrozenInstanceError": FrozenInstanceError}
|
||||||
'FrozenInstanceError': FrozenInstanceError}
|
|
||||||
if fields:
|
if fields:
|
||||||
fields_str = '(' + ','.join(repr(f.name) for f in fields) + ',)'
|
fields_str = "(" + ",".join(repr(f.name) for f in fields) + ",)"
|
||||||
else:
|
else:
|
||||||
# Special case for the zero-length tuple.
|
# Special case for the zero-length tuple.
|
||||||
fields_str = '()'
|
fields_str = "()"
|
||||||
return (_create_fn('__setattr__',
|
return (
|
||||||
('self', 'name', 'value'),
|
_create_fn(
|
||||||
(f'if type(self) is cls or name in {fields_str}:',
|
"__setattr__",
|
||||||
|
("self", "name", "value"),
|
||||||
|
(
|
||||||
|
f"if type(self) is cls or name in {fields_str}:",
|
||||||
' raise FrozenInstanceError(f"cannot assign to field {name!r}")',
|
' raise FrozenInstanceError(f"cannot assign to field {name!r}")',
|
||||||
f'super(cls, self).__setattr__(name, value)'),
|
f"super(cls, self).__setattr__(name, value)",
|
||||||
globals=globals),
|
),
|
||||||
_create_fn('__delattr__',
|
globals=globals,
|
||||||
('self', 'name'),
|
),
|
||||||
(f'if type(self) is cls or name in {fields_str}:',
|
_create_fn(
|
||||||
|
"__delattr__",
|
||||||
|
("self", "name"),
|
||||||
|
(
|
||||||
|
f"if type(self) is cls or name in {fields_str}:",
|
||||||
' raise FrozenInstanceError(f"cannot delete field {name!r}")',
|
' raise FrozenInstanceError(f"cannot delete field {name!r}")',
|
||||||
f'super(cls, self).__delattr__(name)'),
|
f"super(cls, self).__delattr__(name)",
|
||||||
globals=globals),
|
),
|
||||||
|
globals=globals,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -534,18 +572,20 @@ def _cmp_fn(name, op, self_tuple, other_tuple):
|
||||||
# '(self.x,self.y)' and other_tuple is the string
|
# '(self.x,self.y)' and other_tuple is the string
|
||||||
# '(other.x,other.y)'.
|
# '(other.x,other.y)'.
|
||||||
|
|
||||||
return _create_fn(name,
|
return _create_fn(
|
||||||
('self', 'other'),
|
name,
|
||||||
[ 'if other.__class__ is self.__class__:',
|
("self", "other"),
|
||||||
f' return {self_tuple}{op}{other_tuple}',
|
[
|
||||||
'return NotImplemented'])
|
"if other.__class__ is self.__class__:",
|
||||||
|
f" return {self_tuple}{op}{other_tuple}",
|
||||||
|
"return NotImplemented",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def _hash_fn(fields):
|
def _hash_fn(fields):
|
||||||
self_tuple = _tuple_str('self', fields)
|
self_tuple = _tuple_str("self", fields)
|
||||||
return _create_fn('__hash__',
|
return _create_fn("__hash__", ("self",), [f"return hash({self_tuple})"])
|
||||||
('self',),
|
|
||||||
[f'return hash({self_tuple})'])
|
|
||||||
|
|
||||||
|
|
||||||
def _is_classvar(a_type, typing):
|
def _is_classvar(a_type, typing):
|
||||||
|
@ -658,12 +698,12 @@ def _get_field(cls, a_name, a_type):
|
||||||
# annotation to be a ClassVar. So, only look for ClassVar if
|
# annotation to be a ClassVar. So, only look for ClassVar if
|
||||||
# typing has been imported by any module (not necessarily cls's
|
# typing has been imported by any module (not necessarily cls's
|
||||||
# module).
|
# module).
|
||||||
typing = sys.modules.get('typing')
|
typing = sys.modules.get("typing")
|
||||||
if typing:
|
if typing:
|
||||||
if (_is_classvar(a_type, typing)
|
if _is_classvar(a_type, typing) or (
|
||||||
or (isinstance(f.type, str)
|
isinstance(f.type, str)
|
||||||
and _is_type(f.type, cls, typing, typing.ClassVar,
|
and _is_type(f.type, cls, typing, typing.ClassVar, _is_classvar)
|
||||||
_is_classvar))):
|
):
|
||||||
f._field_type = _FIELD_CLASSVAR
|
f._field_type = _FIELD_CLASSVAR
|
||||||
|
|
||||||
# If the type is InitVar, or if it's a matching string annotation,
|
# If the type is InitVar, or if it's a matching string annotation,
|
||||||
|
@ -672,10 +712,10 @@ def _get_field(cls, a_name, a_type):
|
||||||
# The module we're checking against is the module we're
|
# The module we're checking against is the module we're
|
||||||
# currently in (dataclasses.py).
|
# currently in (dataclasses.py).
|
||||||
dataclasses = sys.modules[__name__]
|
dataclasses = sys.modules[__name__]
|
||||||
if (_is_initvar(a_type, dataclasses)
|
if _is_initvar(a_type, dataclasses) or (
|
||||||
or (isinstance(f.type, str)
|
isinstance(f.type, str)
|
||||||
and _is_type(f.type, cls, dataclasses, dataclasses.InitVar,
|
and _is_type(f.type, cls, dataclasses, dataclasses.InitVar, _is_initvar)
|
||||||
_is_initvar))):
|
):
|
||||||
f._field_type = _FIELD_INITVAR
|
f._field_type = _FIELD_INITVAR
|
||||||
|
|
||||||
# Validations for individual fields. This is delayed until now,
|
# Validations for individual fields. This is delayed until now,
|
||||||
|
@ -685,8 +725,7 @@ def _get_field(cls, a_name, a_type):
|
||||||
# Special restrictions for ClassVar and InitVar.
|
# Special restrictions for ClassVar and InitVar.
|
||||||
if f._field_type in (_FIELD_CLASSVAR, _FIELD_INITVAR):
|
if f._field_type in (_FIELD_CLASSVAR, _FIELD_INITVAR):
|
||||||
if f.default_factory is not MISSING:
|
if f.default_factory is not MISSING:
|
||||||
raise TypeError(f'field {f.name} cannot have a '
|
raise TypeError(f"field {f.name} cannot have a " "default factory")
|
||||||
'default factory')
|
|
||||||
# Should I check for other field settings? default_factory
|
# Should I check for other field settings? default_factory
|
||||||
# seems the most serious to check for. Maybe add others. For
|
# seems the most serious to check for. Maybe add others. For
|
||||||
# example, how about init=False (or really,
|
# example, how about init=False (or really,
|
||||||
|
@ -695,8 +734,10 @@ def _get_field(cls, a_name, a_type):
|
||||||
|
|
||||||
# For real fields, disallow mutable defaults for known types.
|
# For real fields, disallow mutable defaults for known types.
|
||||||
if f._field_type is _FIELD and isinstance(f.default, (list, dict, set)):
|
if f._field_type is _FIELD and isinstance(f.default, (list, dict, set)):
|
||||||
raise ValueError(f'mutable default {type(f.default)} for field '
|
raise ValueError(
|
||||||
f'{f.name} is not allowed: use default_factory')
|
f"mutable default {type(f.default)} for field "
|
||||||
|
f"{f.name} is not allowed: use default_factory"
|
||||||
|
)
|
||||||
|
|
||||||
return f
|
return f
|
||||||
|
|
||||||
|
@ -715,17 +756,20 @@ def _set_new_attribute(cls, name, value):
|
||||||
# take. The common case is to do nothing, so instead of providing a
|
# take. The common case is to do nothing, so instead of providing a
|
||||||
# function that is a no-op, use None to signify that.
|
# function that is a no-op, use None to signify that.
|
||||||
|
|
||||||
|
|
||||||
def _hash_set_none(cls, fields):
|
def _hash_set_none(cls, fields):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
def _hash_add(cls, fields):
|
def _hash_add(cls, fields):
|
||||||
flds = [f for f in fields if (f.compare if f.hash is None else f.hash)]
|
flds = [f for f in fields if (f.compare if f.hash is None else f.hash)]
|
||||||
return _hash_fn(flds)
|
return _hash_fn(flds)
|
||||||
|
|
||||||
|
|
||||||
def _hash_exception(cls, fields):
|
def _hash_exception(cls, fields):
|
||||||
# Raise an exception.
|
# Raise an exception.
|
||||||
raise TypeError(f'Cannot overwrite attribute __hash__ '
|
raise TypeError(f"Cannot overwrite attribute __hash__ " f"in class {cls.__name__}")
|
||||||
f'in class {cls.__name__}')
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# +-------------------------------------- unsafe_hash?
|
# +-------------------------------------- unsafe_hash?
|
||||||
|
@ -736,23 +780,24 @@ def _hash_exception(cls, fields):
|
||||||
# | | | | +------- action
|
# | | | | +------- action
|
||||||
# | | | | |
|
# | | | | |
|
||||||
# v v v v v
|
# v v v v v
|
||||||
_hash_action = {(False, False, False, False): None,
|
_hash_action = {
|
||||||
(False, False, False, True ): None,
|
(False, False, False, False): None,
|
||||||
|
(False, False, False, True): None,
|
||||||
(False, False, True, False): None,
|
(False, False, True, False): None,
|
||||||
(False, False, True, True ): None,
|
(False, False, True, True): None,
|
||||||
(False, True, False, False): _hash_set_none,
|
(False, True, False, False): _hash_set_none,
|
||||||
(False, True, False, True ): None,
|
(False, True, False, True): None,
|
||||||
(False, True, True, False): _hash_add,
|
(False, True, True, False): _hash_add,
|
||||||
(False, True, True, True ): None,
|
(False, True, True, True): None,
|
||||||
(True, False, False, False): _hash_add,
|
(True, False, False, False): _hash_add,
|
||||||
(True, False, False, True ): _hash_exception,
|
(True, False, False, True): _hash_exception,
|
||||||
(True, False, True, False): _hash_add,
|
(True, False, True, False): _hash_add,
|
||||||
(True, False, True, True ): _hash_exception,
|
(True, False, True, True): _hash_exception,
|
||||||
(True, True, False, False): _hash_add,
|
(True, True, False, False): _hash_add,
|
||||||
(True, True, False, True ): _hash_exception,
|
(True, True, False, True): _hash_exception,
|
||||||
(True, True, True, False): _hash_add,
|
(True, True, True, False): _hash_add,
|
||||||
(True, True, True, True ): _hash_exception,
|
(True, True, True, True): _hash_exception,
|
||||||
}
|
}
|
||||||
# See https://bugs.python.org/issue32929#msg312829 for an if-statement
|
# See https://bugs.python.org/issue32929#msg312829 for an if-statement
|
||||||
# version of this table.
|
# version of this table.
|
||||||
|
|
||||||
|
@ -764,8 +809,7 @@ def _process_class(cls, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
# is defined by the base class, which is found first.
|
# is defined by the base class, which is found first.
|
||||||
fields = {}
|
fields = {}
|
||||||
|
|
||||||
setattr(cls, _PARAMS, _DataclassParams(init, repr, eq, order,
|
setattr(cls, _PARAMS, _DataclassParams(init, repr, eq, order, unsafe_hash, frozen))
|
||||||
unsafe_hash, frozen))
|
|
||||||
|
|
||||||
# Find our base classes in reverse MRO order, and exclude
|
# Find our base classes in reverse MRO order, and exclude
|
||||||
# ourselves. In reversed order so that more derived classes
|
# ourselves. In reversed order so that more derived classes
|
||||||
|
@ -796,13 +840,12 @@ def _process_class(cls, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
# actual default value. Pseudo-fields ClassVars and InitVars are
|
# actual default value. Pseudo-fields ClassVars and InitVars are
|
||||||
# included, despite the fact that they're not real fields. That's
|
# included, despite the fact that they're not real fields. That's
|
||||||
# dealt with later.
|
# dealt with later.
|
||||||
cls_annotations = cls.__dict__.get('__annotations__', {})
|
cls_annotations = cls.__dict__.get("__annotations__", {})
|
||||||
|
|
||||||
# Now find fields in our class. While doing so, validate some
|
# Now find fields in our class. While doing so, validate some
|
||||||
# things, and set the default values (as class attributes) where
|
# things, and set the default values (as class attributes) where
|
||||||
# we can.
|
# we can.
|
||||||
cls_fields = [_get_field(cls, name, type)
|
cls_fields = [_get_field(cls, name, type) for name, type in cls_annotations.items()]
|
||||||
for name, type in cls_annotations.items()]
|
|
||||||
for f in cls_fields:
|
for f in cls_fields:
|
||||||
fields[f.name] = f
|
fields[f.name] = f
|
||||||
|
|
||||||
|
@ -825,19 +868,17 @@ def _process_class(cls, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
# Do we have any Field members that don't also have annotations?
|
# Do we have any Field members that don't also have annotations?
|
||||||
for name, value in cls.__dict__.items():
|
for name, value in cls.__dict__.items():
|
||||||
if isinstance(value, Field) and not name in cls_annotations:
|
if isinstance(value, Field) and not name in cls_annotations:
|
||||||
raise TypeError(f'{name!r} is a field but has no type annotation')
|
raise TypeError(f"{name!r} is a field but has no type annotation")
|
||||||
|
|
||||||
# Check rules that apply if we are derived from any dataclasses.
|
# Check rules that apply if we are derived from any dataclasses.
|
||||||
if has_dataclass_bases:
|
if has_dataclass_bases:
|
||||||
# Raise an exception if any of our bases are frozen, but we're not.
|
# Raise an exception if any of our bases are frozen, but we're not.
|
||||||
if any_frozen_base and not frozen:
|
if any_frozen_base and not frozen:
|
||||||
raise TypeError('cannot inherit non-frozen dataclass from a '
|
raise TypeError("cannot inherit non-frozen dataclass from a " "frozen one")
|
||||||
'frozen one')
|
|
||||||
|
|
||||||
# Raise an exception if we're frozen, but none of our bases are.
|
# Raise an exception if we're frozen, but none of our bases are.
|
||||||
if not any_frozen_base and frozen:
|
if not any_frozen_base and frozen:
|
||||||
raise TypeError('cannot inherit frozen dataclass from a '
|
raise TypeError("cannot inherit frozen dataclass from a " "non-frozen one")
|
||||||
'non-frozen one')
|
|
||||||
|
|
||||||
# Remember all of the fields on our class (including bases). This
|
# Remember all of the fields on our class (including bases). This
|
||||||
# also marks this class as being a dataclass.
|
# also marks this class as being a dataclass.
|
||||||
|
@ -848,32 +889,35 @@ def _process_class(cls, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
# set __hash__ to None. This is a heuristic, as it's possible
|
# set __hash__ to None. This is a heuristic, as it's possible
|
||||||
# that such a __hash__ == None was not auto-generated, but it
|
# that such a __hash__ == None was not auto-generated, but it
|
||||||
# close enough.
|
# close enough.
|
||||||
class_hash = cls.__dict__.get('__hash__', MISSING)
|
class_hash = cls.__dict__.get("__hash__", MISSING)
|
||||||
has_explicit_hash = not (class_hash is MISSING or
|
has_explicit_hash = not (
|
||||||
(class_hash is None and '__eq__' in cls.__dict__))
|
class_hash is MISSING or (class_hash is None and "__eq__" in cls.__dict__)
|
||||||
|
)
|
||||||
|
|
||||||
# If we're generating ordering methods, we must be generating the
|
# If we're generating ordering methods, we must be generating the
|
||||||
# eq methods.
|
# eq methods.
|
||||||
if order and not eq:
|
if order and not eq:
|
||||||
raise ValueError('eq must be true if order is true')
|
raise ValueError("eq must be true if order is true")
|
||||||
|
|
||||||
if init:
|
if init:
|
||||||
# Does this class have a post-init function?
|
# Does this class have a post-init function?
|
||||||
has_post_init = hasattr(cls, _POST_INIT_NAME)
|
has_post_init = hasattr(cls, _POST_INIT_NAME)
|
||||||
|
|
||||||
# Include InitVars and regular fields (so, not ClassVars).
|
# Include InitVars and regular fields (so, not ClassVars).
|
||||||
flds = [f for f in fields.values()
|
flds = [f for f in fields.values() if f._field_type in (_FIELD, _FIELD_INITVAR)]
|
||||||
if f._field_type in (_FIELD, _FIELD_INITVAR)]
|
_set_new_attribute(
|
||||||
_set_new_attribute(cls, '__init__',
|
cls,
|
||||||
_init_fn(flds,
|
"__init__",
|
||||||
|
_init_fn(
|
||||||
|
flds,
|
||||||
frozen,
|
frozen,
|
||||||
has_post_init,
|
has_post_init,
|
||||||
# The name to use for the "self"
|
# The name to use for the "self"
|
||||||
# param in __init__. Use "self"
|
# param in __init__. Use "self"
|
||||||
# if possible.
|
# if possible.
|
||||||
'__dataclass_self__' if 'self' in fields
|
"__dataclass_self__" if "self" in fields else "self",
|
||||||
else 'self',
|
),
|
||||||
))
|
)
|
||||||
|
|
||||||
# Get the fields as a list, and include only real fields. This is
|
# Get the fields as a list, and include only real fields. This is
|
||||||
# used in all of the following methods.
|
# used in all of the following methods.
|
||||||
|
@ -881,54 +925,58 @@ def _process_class(cls, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
|
|
||||||
if repr:
|
if repr:
|
||||||
flds = [f for f in field_list if f.repr]
|
flds = [f for f in field_list if f.repr]
|
||||||
_set_new_attribute(cls, '__repr__', _repr_fn(flds))
|
_set_new_attribute(cls, "__repr__", _repr_fn(flds))
|
||||||
|
|
||||||
if eq:
|
if eq:
|
||||||
# Create _eq__ method. There's no need for a __ne__ method,
|
# Create _eq__ method. There's no need for a __ne__ method,
|
||||||
# since python will call __eq__ and negate it.
|
# since python will call __eq__ and negate it.
|
||||||
flds = [f for f in field_list if f.compare]
|
flds = [f for f in field_list if f.compare]
|
||||||
self_tuple = _tuple_str('self', flds)
|
self_tuple = _tuple_str("self", flds)
|
||||||
other_tuple = _tuple_str('other', flds)
|
other_tuple = _tuple_str("other", flds)
|
||||||
_set_new_attribute(cls, '__eq__',
|
_set_new_attribute(
|
||||||
_cmp_fn('__eq__', '==',
|
cls, "__eq__", _cmp_fn("__eq__", "==", self_tuple, other_tuple)
|
||||||
self_tuple, other_tuple))
|
)
|
||||||
|
|
||||||
if order:
|
if order:
|
||||||
# Create and set the ordering methods.
|
# Create and set the ordering methods.
|
||||||
flds = [f for f in field_list if f.compare]
|
flds = [f for f in field_list if f.compare]
|
||||||
self_tuple = _tuple_str('self', flds)
|
self_tuple = _tuple_str("self", flds)
|
||||||
other_tuple = _tuple_str('other', flds)
|
other_tuple = _tuple_str("other", flds)
|
||||||
for name, op in [('__lt__', '<'),
|
for name, op in [
|
||||||
('__le__', '<='),
|
("__lt__", "<"),
|
||||||
('__gt__', '>'),
|
("__le__", "<="),
|
||||||
('__ge__', '>='),
|
("__gt__", ">"),
|
||||||
|
("__ge__", ">="),
|
||||||
]:
|
]:
|
||||||
if _set_new_attribute(cls, name,
|
if _set_new_attribute(
|
||||||
_cmp_fn(name, op, self_tuple, other_tuple)):
|
cls, name, _cmp_fn(name, op, self_tuple, other_tuple)
|
||||||
raise TypeError(f'Cannot overwrite attribute {name} '
|
):
|
||||||
f'in class {cls.__name__}. Consider using '
|
raise TypeError(
|
||||||
'functools.total_ordering')
|
f"Cannot overwrite attribute {name} "
|
||||||
|
f"in class {cls.__name__}. Consider using "
|
||||||
|
"functools.total_ordering"
|
||||||
|
)
|
||||||
|
|
||||||
if frozen:
|
if frozen:
|
||||||
for fn in _frozen_get_del_attr(cls, field_list):
|
for fn in _frozen_get_del_attr(cls, field_list):
|
||||||
if _set_new_attribute(cls, fn.__name__, fn):
|
if _set_new_attribute(cls, fn.__name__, fn):
|
||||||
raise TypeError(f'Cannot overwrite attribute {fn.__name__} '
|
raise TypeError(
|
||||||
f'in class {cls.__name__}')
|
f"Cannot overwrite attribute {fn.__name__} "
|
||||||
|
f"in class {cls.__name__}"
|
||||||
|
)
|
||||||
|
|
||||||
# Decide if/how we're going to create a hash function.
|
# Decide if/how we're going to create a hash function.
|
||||||
hash_action = _hash_action[bool(unsafe_hash),
|
hash_action = _hash_action[
|
||||||
bool(eq),
|
bool(unsafe_hash), bool(eq), bool(frozen), has_explicit_hash
|
||||||
bool(frozen),
|
]
|
||||||
has_explicit_hash]
|
|
||||||
if hash_action:
|
if hash_action:
|
||||||
# No need to call _set_new_attribute here, since by the time
|
# No need to call _set_new_attribute here, since by the time
|
||||||
# we're here the overwriting is unconditional.
|
# we're here the overwriting is unconditional.
|
||||||
cls.__hash__ = hash_action(cls, field_list)
|
cls.__hash__ = hash_action(cls, field_list)
|
||||||
|
|
||||||
if not getattr(cls, '__doc__'):
|
if not getattr(cls, "__doc__"):
|
||||||
# Create a class doc-string.
|
# Create a class doc-string.
|
||||||
cls.__doc__ = (cls.__name__ +
|
cls.__doc__ = cls.__name__ + str(inspect.signature(cls)).replace(" -> None", "")
|
||||||
str(inspect.signature(cls)).replace(' -> None', ''))
|
|
||||||
|
|
||||||
return cls
|
return cls
|
||||||
|
|
||||||
|
@ -936,8 +984,16 @@ def _process_class(cls, init, repr, eq, order, unsafe_hash, frozen):
|
||||||
# _cls should never be specified by keyword, so start it with an
|
# _cls should never be specified by keyword, so start it with an
|
||||||
# underscore. The presence of _cls is used to detect if this
|
# underscore. The presence of _cls is used to detect if this
|
||||||
# decorator is being called with parameters or not.
|
# decorator is being called with parameters or not.
|
||||||
def dataclass(_cls=None, *, init=True, repr=True, eq=True, order=False,
|
def dataclass(
|
||||||
unsafe_hash=False, frozen=False):
|
_cls=None,
|
||||||
|
*,
|
||||||
|
init=True,
|
||||||
|
repr=True,
|
||||||
|
eq=True,
|
||||||
|
order=False,
|
||||||
|
unsafe_hash=False,
|
||||||
|
frozen=False,
|
||||||
|
):
|
||||||
"""Returns the same class as was passed in, with dunder methods
|
"""Returns the same class as was passed in, with dunder methods
|
||||||
added based on the fields defined in the class.
|
added based on the fields defined in the class.
|
||||||
|
|
||||||
|
@ -973,7 +1029,7 @@ def fields(class_or_instance):
|
||||||
try:
|
try:
|
||||||
fields = getattr(class_or_instance, _FIELDS)
|
fields = getattr(class_or_instance, _FIELDS)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
raise TypeError('must be called with a dataclass type or instance')
|
raise TypeError("must be called with a dataclass type or instance")
|
||||||
|
|
||||||
# Exclude pseudo-fields. Note that fields is sorted by insertion
|
# Exclude pseudo-fields. Note that fields is sorted by insertion
|
||||||
# order, so the order of the tuple is as the fields were defined.
|
# order, so the order of the tuple is as the fields were defined.
|
||||||
|
@ -1025,8 +1081,10 @@ def _asdict_inner(obj, dict_factory):
|
||||||
elif isinstance(obj, (list, tuple)):
|
elif isinstance(obj, (list, tuple)):
|
||||||
return type(obj)(_asdict_inner(v, dict_factory) for v in obj)
|
return type(obj)(_asdict_inner(v, dict_factory) for v in obj)
|
||||||
elif isinstance(obj, dict):
|
elif isinstance(obj, dict):
|
||||||
return type(obj)((_asdict_inner(k, dict_factory), _asdict_inner(v, dict_factory))
|
return type(obj)(
|
||||||
for k, v in obj.items())
|
(_asdict_inner(k, dict_factory), _asdict_inner(v, dict_factory))
|
||||||
|
for k, v in obj.items()
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
return copy.deepcopy(obj)
|
return copy.deepcopy(obj)
|
||||||
|
|
||||||
|
@ -1065,15 +1123,27 @@ def _astuple_inner(obj, tuple_factory):
|
||||||
elif isinstance(obj, (list, tuple)):
|
elif isinstance(obj, (list, tuple)):
|
||||||
return type(obj)(_astuple_inner(v, tuple_factory) for v in obj)
|
return type(obj)(_astuple_inner(v, tuple_factory) for v in obj)
|
||||||
elif isinstance(obj, dict):
|
elif isinstance(obj, dict):
|
||||||
return type(obj)((_astuple_inner(k, tuple_factory), _astuple_inner(v, tuple_factory))
|
return type(obj)(
|
||||||
for k, v in obj.items())
|
(_astuple_inner(k, tuple_factory), _astuple_inner(v, tuple_factory))
|
||||||
|
for k, v in obj.items()
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
return copy.deepcopy(obj)
|
return copy.deepcopy(obj)
|
||||||
|
|
||||||
|
|
||||||
def make_dataclass(cls_name, fields, *, bases=(), namespace=None, init=True,
|
def make_dataclass(
|
||||||
repr=True, eq=True, order=False, unsafe_hash=False,
|
cls_name,
|
||||||
frozen=False):
|
fields,
|
||||||
|
*,
|
||||||
|
bases=(),
|
||||||
|
namespace=None,
|
||||||
|
init=True,
|
||||||
|
repr=True,
|
||||||
|
eq=True,
|
||||||
|
order=False,
|
||||||
|
unsafe_hash=False,
|
||||||
|
frozen=False,
|
||||||
|
):
|
||||||
"""Return a new dynamically created dataclass.
|
"""Return a new dynamically created dataclass.
|
||||||
|
|
||||||
The dataclass name will be 'cls_name'. 'fields' is an iterable
|
The dataclass name will be 'cls_name'. 'fields' is an iterable
|
||||||
|
@ -1110,31 +1180,38 @@ def make_dataclass(cls_name, fields, *, bases=(), namespace=None, init=True,
|
||||||
for item in fields:
|
for item in fields:
|
||||||
if isinstance(item, str):
|
if isinstance(item, str):
|
||||||
name = item
|
name = item
|
||||||
tp = 'typing.Any'
|
tp = "typing.Any"
|
||||||
elif len(item) == 2:
|
elif len(item) == 2:
|
||||||
name, tp, = item
|
name, tp, = item
|
||||||
elif len(item) == 3:
|
elif len(item) == 3:
|
||||||
name, tp, spec = item
|
name, tp, spec = item
|
||||||
namespace[name] = spec
|
namespace[name] = spec
|
||||||
else:
|
else:
|
||||||
raise TypeError(f'Invalid field: {item!r}')
|
raise TypeError(f"Invalid field: {item!r}")
|
||||||
|
|
||||||
if not isinstance(name, str) or not name.isidentifier():
|
if not isinstance(name, str) or not name.isidentifier():
|
||||||
raise TypeError(f'Field names must be valid identifers: {name!r}')
|
raise TypeError(f"Field names must be valid identifers: {name!r}")
|
||||||
if keyword.iskeyword(name):
|
if keyword.iskeyword(name):
|
||||||
raise TypeError(f'Field names must not be keywords: {name!r}')
|
raise TypeError(f"Field names must not be keywords: {name!r}")
|
||||||
if name in seen:
|
if name in seen:
|
||||||
raise TypeError(f'Field name duplicated: {name!r}')
|
raise TypeError(f"Field name duplicated: {name!r}")
|
||||||
|
|
||||||
seen.add(name)
|
seen.add(name)
|
||||||
anns[name] = tp
|
anns[name] = tp
|
||||||
|
|
||||||
namespace['__annotations__'] = anns
|
namespace["__annotations__"] = anns
|
||||||
# We use `types.new_class()` instead of simply `type()` to allow dynamic creation
|
# We use `types.new_class()` instead of simply `type()` to allow dynamic creation
|
||||||
# of generic dataclassses.
|
# of generic dataclassses.
|
||||||
cls = types.new_class(cls_name, bases, {}, lambda ns: ns.update(namespace))
|
cls = types.new_class(cls_name, bases, {}, lambda ns: ns.update(namespace))
|
||||||
return dataclass(cls, init=init, repr=repr, eq=eq, order=order,
|
return dataclass(
|
||||||
unsafe_hash=unsafe_hash, frozen=frozen)
|
cls,
|
||||||
|
init=init,
|
||||||
|
repr=repr,
|
||||||
|
eq=eq,
|
||||||
|
order=order,
|
||||||
|
unsafe_hash=unsafe_hash,
|
||||||
|
frozen=frozen,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def replace(obj, **changes):
|
def replace(obj, **changes):
|
||||||
|
@ -1165,9 +1242,11 @@ def replace(obj, **changes):
|
||||||
if not f.init:
|
if not f.init:
|
||||||
# Error if this field is specified in changes.
|
# Error if this field is specified in changes.
|
||||||
if f.name in changes:
|
if f.name in changes:
|
||||||
raise ValueError(f'field {f.name} is declared with '
|
raise ValueError(
|
||||||
'init=False, it cannot be specified with '
|
f"field {f.name} is declared with "
|
||||||
'replace()')
|
"init=False, it cannot be specified with "
|
||||||
|
"replace()"
|
||||||
|
)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if f.name not in changes:
|
if f.name not in changes:
|
||||||
|
|
Loading…
Reference in New Issue
Block a user