import dataclasses
import inspect
import sys
import weakref
from collections.abc import Callable, Iterable, MutableSet
from typing import TYPE_CHECKING, Any, cast
if sys.version_info < (3, 11): # pragma: no cover
from typing_extensions import Self
else:
from typing import Self
if TYPE_CHECKING: # pragma: no cover
from types import GenericAlias
from sunset.lock import SettingsLock
from sunset.notifier import Notifier
from sunset.protocols import BaseField, Field, UpdateNotifier
from sunset.sets import WeakNonHashableSet
from sunset.stringutils import collate_by_prefix, split_on
[docs]
class Bunch(BaseField):
"""
A collection of related Keys.
Under the hood, a Bunch is a dataclass, and can be used in the same manner,
i.e. by defining attributes directly on the class itself.
Example:
>>> from sunset import Bunch, Key
>>> class Appearance(Bunch):
... class Font(Bunch):
... name: Key[str] = Key(default="Arial")
... size: Key[int] = Key(default=14)
... main_font: Font = Font()
... secondary_font: Font = Font()
>>> appearance = Appearance()
>>> appearance.main_font.name.get()
'Arial'
>>> appearance.secondary_font.name.get()
'Arial'
>>> appearance.main_font.name.set("Times New Roman")
True
>>> appearance.secondary_font.name.set("Calibri")
True
>>> appearance.main_font.name.get()
'Times New Roman'
>>> appearance.secondary_font.name.get()
'Calibri'
"""
_parent_ref: weakref.ref["Bunch"] | None
_children_set: MutableSet["Bunch"]
_fields: dict[str, Field]
_update_notifier: Notifier[[UpdateNotifier]]
def __new__(cls) -> Self:
# Build and return a dataclass constructed from this class. Keep a
# reference to that dataclass as a private class attribute, so that we
# only construct it once. This allows type identity checks (as in
# "type(a) is type(b)") to work.
dataclass_attr = "__DATACLASS_CLASS"
orig_class_attr = "__ORIG_CLASS"
dataclass_class: type[Self] | None = None
if dataclasses.is_dataclass(cls):
# If this class is already a dataclass, then no need to construct a new one.
# Just use this one directly.
dataclass_class = cls
else:
# Else use the dataclass recorded on this class, if there is one. Note the
# use of vars(), in order to look up the dataclass for this specific
# class, and not one of its parents.
dataclass_class = vars(cls).get(dataclass_attr, None)
if dataclass_class is None:
# We haven't yet constructed a dataclass from this class. Construct
# one here.
cls_parents = cls.__bases__
dataclass_fields: list[tuple[str, type | GenericAlias, Field]] = []
potential_fields = [(name, getattr(cls, name, None)) for name in dir(cls)]
for name, attr in potential_fields:
if inspect.isclass(attr) and issubclass(attr, Bunch):
if name in (attr.__name__, dataclass_attr):
# This is probably a class definition that just happens
# to be located inside the containing Bunch definition.
# This is fine.
continue
msg = (
f"Field '{name}' in the definition of '{cls.__name__}' is"
f" uninstantiated. Did you mean '{attr.__name__}()'?"
)
raise TypeError(msg)
if not isinstance(attr, Field):
# Not actually a field, then.
continue
# Safety check: make sure the user isn't accidentally overriding an
# existing attribute. We do however allow overriding an attribute
# with an attribute of the same type, which allows the user to
# override a Key with a Key of the same type but a different
# default value, for instance.
for cls_parent in cls_parents:
if (
parent_attr := getattr(cls_parent, name, None)
) is not None and type(parent_attr) is not type(attr):
msg = (
f"Field '{name}' in the definition of"
f" '{cls.__name__}' overrides attribute of the"
" same name declared in parent class"
f" '{cls_parent.__name__}'; consider renaming"
f" this field to '{name}_' for instance"
)
raise TypeError(msg)
# Create a proper field from the attribute.
field = dataclasses.field(default_factory=attr._newInstance) # noqa: SLF001
dataclass_fields.append((name, attr._typeHint(), field)) # noqa: SLF001
# Create a dataclass based on this class. Note that we will be
# providing our own __init__() override below.
kwargs: dict[str, Any] = (
{} if sys.version_info < (3, 12) else {"module": cls.__module__}
)
dataclass_class = cast(
type[Self],
dataclasses.make_dataclass(
cls.__qualname__,
dataclass_fields,
init=False,
bases=(cls, *cls_parents),
**kwargs,
),
)
# And store it on the class itself so it can be reused if this class is
# instantiated again.
setattr(cls, dataclass_attr, dataclass_class)
# Also keep a reference to the original class, so that's not lost.
setattr(dataclass_class, orig_class_attr, cls)
# Create an instance of the dataclass.
new_cls: Self = super().__new__(dataclass_class)
# Set up the fields that were identified above as instance attributes.
new_cls.__setup_fields__()
return new_cls
def __setup_fields__(self) -> None:
# Set up the Bunch fields as instance attributes.
# First, look up internal dataclass attribute names.
fields_attr: str = getattr(dataclasses, "_FIELDS") # noqa: B009
field_type: Any = getattr(dataclasses, "_FIELD") # noqa: B009
fields: dict[str, dataclasses.Field[Any]] = getattr(self, fields_attr, {})
# Then look up the fields stored in the dataclass.
for field in fields.values():
if (
getattr(field, "_field_type", None) is not field_type
): # pragma: no cover
continue
if field.default_factory is dataclasses.MISSING: # pragma: no cover
continue
setattr(self, field.name, field.default_factory())
def __init__(self) -> None:
super().__init__()
self._parent_ref = None
self._children_set = WeakNonHashableSet[Bunch]()
self._fields = {}
self._update_notifier = Notifier()
for label, field in vars(self).items():
if isinstance(field, Field):
self._fields[label] = field
field.meta().update(label=label, container=self)
field._update_notifier.add(self._update_notifier.trigger) # noqa: SLF001
self.__post_init__()
def __post_init__(self) -> None:
"""
DEPRECATED. Will be removed in v1.0.
Use __init__() instead.
"""
[docs]
@SettingsLock.with_write_lock
def setParent(self, parent: Self | None) -> None:
"""
Makes the given Bunch the parent of this one. If None, remove this
Bunch's parent, if any.
All the Key, List and Bunch fields defined on this Bunch instance
will be recursively reparented to the corresponding Key / List / Bunch
field on the given parent.
This method is for internal purposes and you will typically not need to
call it directly.
Args:
parent: Either a Bunch that will become this Bunch's parent, or
None. The parent Bunch must have the same type as this
Bunch.
Note:
A Bunch and its parent, if any, do not increase each other's
reference count.
"""
# Runtime check to affirm the type check of the method.
if parent is not None and type(self) is not type(parent): # pragma: no cover
return
old_parent = self.parent()
if old_parent is not None:
old_parent._children_set.discard(self) # noqa: SLF001
if parent is None:
self._parent_ref = None
else:
self._parent_ref = weakref.ref(parent)
parent._children_set.add(self) # noqa: SLF001
for label, field in self._fields.items():
if parent is None:
field.setParent(None)
continue
parent_field = parent._fields.get(label)
if parent_field is None: # pragma: no cover
# This is a safety check, but it shouldn't happen. By
# construction self should be of the same type as parent, so
# they should have the same attributes.
continue
assert type(field) is type(parent_field) # noqa: S101
field.setParent(parent_field)
[docs]
@SettingsLock.with_read_lock
def parent(self) -> Self | None:
"""
Returns the parent of this Bunch, if any.
Returns:
A Bunch instance of the same type as this one, or None.
"""
# Make the type of self._parent_ref more specific for the purpose of
# type checking.
parent = cast(weakref.ref[Self] | None, self._parent_ref)
return parent() if parent is not None else None
[docs]
@SettingsLock.with_read_lock
def children(self) -> Iterable[Self]:
"""
Returns an iterable with the Bunch instances that have this Bunch as their
parent.
Returns:
An iterable of Bunch instances of the same type as this one.
"""
return [cast(Self, child) for child in self._children_set]
[docs]
def onUpdateCall(self, callback: Callable[[Any], Any]) -> None:
"""
Adds a callback to be called whenever this Bunch is updated. A Bunch
is considered updated when any of its fields is updated.
The callback will be called with as its argument whichever field was
just updated.
Args:
callback: A callable that will be called with one argument of type
:class:`~sunset.List`, :class:`~sunset.Bunch` or
:class:`~sunset.Key`.
Note:
This method does not increase the reference count of the given
callback.
"""
self._update_notifier.add(callback)
@SettingsLock.with_read_lock
def isSet(self) -> bool:
"""
Indicates whether this Bunch holds any field that is set.
Returns:
True if any field set on this Bunch is set, else False.
"""
return any(field.isSet() for field in self._fields.values())
@SettingsLock.with_write_lock
def clear(self) -> None:
for field in self._fields.values():
field.clear()
@SettingsLock.with_read_lock
def dumpFields(self) -> Iterable[tuple[str, str | None]]:
"""
Internal.
"""
ret: list[tuple[str, str | None]] = []
sep = self._PATH_SEPARATOR
if not self.skipOnSave():
for label, field in sorted(self._fields.items()):
ret.extend(
(label + sep + path if path else label, item)
for path, item in field.dumpFields()
)
return ret
@SettingsLock.with_write_lock
def restoreFields(self, fields: Iterable[tuple[str, str | None]]) -> bool:
"""
Internal.
"""
any_change = False
by_field = collate_by_prefix(fields, split_on(self._PATH_SEPARATOR))
with self._update_notifier.inhibit():
for field_name, field in self._fields.items():
if field_name not in by_field:
# If the field is not in the fields to restore, clear it.
any_change = any_change or field.isSet()
field.clear()
else:
any_change = field.restoreFields(by_field[field_name]) or any_change
return any_change
def _typeHint(self) -> type:
return type(self)
def _newInstance(self) -> Self:
"""
Internal. Returns a new instance of this Bunch with the same fields.
Returns:
A Bunch instance.
"""
return self.__class__()