123 lines
4.5 KiB
Python
123 lines
4.5 KiB
Python
"""Utility to create classes from which frozen or mutable dataclasses can be derived.
|
|
|
|
This module enabled a non-breaking transition from mutable to frozen dataclasses
|
|
derived from EntityDescription and sub classes thereof.
|
|
"""
|
|
from __future__ import annotations
|
|
|
|
import dataclasses
|
|
import sys
|
|
from typing import Any
|
|
|
|
from typing_extensions import dataclass_transform
|
|
|
|
|
|
def _class_fields(cls: type, kw_only: bool) -> list[tuple[str, Any, Any]]:
|
|
"""Return a list of dataclass fields.
|
|
|
|
Extracted from dataclasses._process_class.
|
|
"""
|
|
# pylint: disable=protected-access
|
|
cls_annotations = cls.__dict__.get("__annotations__", {})
|
|
|
|
cls_fields: list[dataclasses.Field[Any]] = []
|
|
|
|
_dataclasses = sys.modules[dataclasses.__name__]
|
|
for name, _type in cls_annotations.items():
|
|
# See if this is a marker to change the value of kw_only.
|
|
if dataclasses._is_kw_only(type, _dataclasses) or ( # type: ignore[attr-defined]
|
|
isinstance(_type, str)
|
|
and dataclasses._is_type( # type: ignore[attr-defined]
|
|
_type,
|
|
cls,
|
|
_dataclasses,
|
|
dataclasses.KW_ONLY,
|
|
dataclasses._is_kw_only, # type: ignore[attr-defined]
|
|
)
|
|
):
|
|
kw_only = True
|
|
else:
|
|
# Otherwise it's a field of some type.
|
|
cls_fields.append(dataclasses._get_field(cls, name, _type, kw_only)) # type: ignore[attr-defined]
|
|
|
|
return [(field.name, field.type, field) for field in cls_fields]
|
|
|
|
|
|
@dataclass_transform(
|
|
field_specifiers=(dataclasses.field, dataclasses.Field),
|
|
frozen_default=True, # Set to allow setting frozen in child classes
|
|
kw_only_default=True, # Set to allow setting kw_only in child classes
|
|
)
|
|
class FrozenOrThawed(type):
|
|
"""Metaclass which which makes classes which behave like a dataclass.
|
|
|
|
This allows child classes to be either mutable or frozen dataclasses.
|
|
"""
|
|
|
|
def _make_dataclass(cls, name: str, bases: tuple[type, ...], kw_only: bool) -> None:
|
|
class_fields = _class_fields(cls, kw_only)
|
|
dataclass_bases = []
|
|
for base in bases:
|
|
dataclass_bases.append(getattr(base, "_dataclass", base))
|
|
cls._dataclass = dataclasses.make_dataclass(
|
|
name, class_fields, bases=tuple(dataclass_bases), frozen=True
|
|
)
|
|
|
|
def __new__(
|
|
mcs, # noqa: N804 ruff bug, ruff does not understand this is a metaclass
|
|
name: str,
|
|
bases: tuple[type, ...],
|
|
namespace: dict[Any, Any],
|
|
frozen_or_thawed: bool = False,
|
|
**kwargs: Any,
|
|
) -> Any:
|
|
"""Pop frozen_or_thawed and store it in the namespace."""
|
|
namespace["_FrozenOrThawed__frozen_or_thawed"] = frozen_or_thawed
|
|
return super().__new__(mcs, name, bases, namespace)
|
|
|
|
def __init__(
|
|
cls,
|
|
name: str,
|
|
bases: tuple[type, ...],
|
|
namespace: dict[Any, Any],
|
|
**kwargs: Any,
|
|
) -> None:
|
|
"""Optionally create a dataclass and store it in cls._dataclass.
|
|
|
|
A dataclass will be created if frozen_or_thawed is set, if not we assume the
|
|
class will be a real dataclass, i.e. it's decorated with @dataclass.
|
|
"""
|
|
if not namespace["_FrozenOrThawed__frozen_or_thawed"]:
|
|
# This class is a real dataclass, optionally inject the parent's annotations
|
|
if all(dataclasses.is_dataclass(base) for base in bases):
|
|
# All direct parents are dataclasses, rely on dataclass inheritance
|
|
return
|
|
# Parent is not a dataclass, inject all parents' annotations
|
|
annotations: dict = {}
|
|
for parent in cls.__mro__[::-1]:
|
|
if parent is object:
|
|
continue
|
|
annotations |= parent.__annotations__
|
|
cls.__annotations__ = annotations
|
|
return
|
|
|
|
# First try without setting the kw_only flag, and if that fails, try setting it
|
|
try:
|
|
cls._make_dataclass(name, bases, False)
|
|
except TypeError:
|
|
cls._make_dataclass(name, bases, True)
|
|
|
|
def __new__(*args: Any, **kwargs: Any) -> object:
|
|
"""Create a new instance.
|
|
|
|
The function has no named arguments to avoid name collisions with dataclass
|
|
field names.
|
|
"""
|
|
cls, *_args = args
|
|
if dataclasses.is_dataclass(cls):
|
|
return object.__new__(cls)
|
|
return cls._dataclass(*_args, **kwargs)
|
|
|
|
cls.__init__ = cls._dataclass.__init__ # type: ignore[misc]
|
|
cls.__new__ = __new__ # type: ignore[method-assign]
|