Source code for sunset.bunch

import dataclasses
import inspect
import weakref

from typing import (
    Any,
    Callable,
    Iterator,
    MutableSet,
    Optional,
    cast,
)

try:
    from typing import Self
except ImportError:
    # TODO: Remove once we deprecate support for Python 3.10.
    from typing_extensions import Self


from sunset.notifier import Notifier
from sunset.protocols import (
    BaseField,
    Field,
    ItemTemplate,
    UpdateNotifier,
)
from sunset.sets import WeakNonHashableSet


[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: Optional[weakref.ref["Bunch"]] _children_set: MutableSet["Bunch"] _fields: dict[str, Field] _update_notifier: Notifier[[UpdateNotifier]] _loaded_notifier: Notifier[[]] 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" dataclass_class: Optional[type[Self]] = None if (dataclass_class := getattr(cls, _dataclass_attr, None)) is None: # We haven't yet constructed a dataclass from this class. Construct # one here. cls_parents = cls.__bases__ dataclasses_fields: list[Any] = [] potential_fields = list(vars(cls).items()) for name, attr in potential_fields: if inspect.isclass(attr): if attr.__name__ == name: # This is probably a class definition that just happens # to be located inside the containing Bunch definition. # This is fine. continue raise TypeError( f"Field '{name}' in the definition of '{cls.__name__}'" " is uninstantiated" ) if isinstance(attr, ItemTemplate): # Safety check: make sure the user isn't accidentally # overriding an existing attribute. for cls_parent in cls_parents: if getattr(cls_parent, name, None) is not None: raise TypeError( 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" ) # Create a proper field from the attribute. field = dataclasses.field(default_factory=attr._newInstance) dataclasses_fields.append((name, attr._typeHint(), field)) # Note that we delete the attribute now that the field is # created. This helps avoid a hard-to-debug problem if the # user subclasses a Bunch with a custom __init__() that # doesn't call super().__init__(). That would seem to work, # but any updates made to attributes of that bunch would # really be applied to the attribute of the Bunch *class*, # not the instance, which would cause 'weird action at a # distance' bugs. Deleting the original class attribute # prevents this entirely. delattr(cls, name) # Create a dataclass based on this class. Note that we will be # providing our own __init__() override below. dataclass_class = cast( type[Self], dataclasses.make_dataclass( cls.__qualname__, dataclasses_fields, init=False, bases=(cls,) + cls_parents, ), ) # And store it on the class itself for later. setattr(cls, _dataclass_attr, dataclass_class) # And finally, return an instance of the dataclass. Phew. return super().__new__(dataclass_class) # type: ignore def __init__(self) -> None: super().__init__() # Set up the Bunch fields. # First, look up internal dataclass attribute names. fields_attr: str = getattr(dataclasses, "_FIELDS") field_type: Any = getattr(dataclasses, "_FIELD") 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: continue if field.default_factory is dataclasses.MISSING: continue setattr(self, field.name, field.default_factory()) self.__post_init__() def __post_init__(self) -> None: self._parent_ref = None self._children_set = WeakNonHashableSet[Bunch]() self._fields = {} self._update_notifier = Notifier() self._loaded_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) self._loaded_notifier.add(field._loaded_notifier.trigger)
[docs] def setParent(self, parent: Optional[Self]) -> 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: if type(self) is not type(parent): return old_parent = self.parent() if old_parent is not None: old_parent._children_set.discard(self) if parent is None: self._parent_ref = None else: self._parent_ref = weakref.ref(parent) parent._children_set.add(self) 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: # 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) field.setParent(parent_field)
[docs] def parent(self) -> Optional[Self]: """ 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(Optional[weakref.ref[Self]], self._parent_ref) return parent() if parent is not None else None
[docs] def children(self) -> Iterator[Self]: """ Returns an iterator over the Bunch instances that have this Bunch as their parent. Returns: An iterator over Bunch instances of the same type as this one. """ # Note that we iterate on a copy so that this will not break if a # different thread updates the contents during the iteration. for child in list(self._children_set): yield cast(Self, child)
[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)
[docs] def onLoadedCall(self, callback: Callable[[], Any]) -> None: """ Adds a callback to be called whenever settings were just loaded. This Bunch itself may or may not have been modified during the load. Args: callback: A callable that takes no argument. Note: This method does not increase the reference count of the given callback. """ self._loaded_notifier.add(callback)
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()) def dumpFields(self) -> Iterator[tuple[str, Optional[str]]]: """ Internal. """ sep = self._PATH_SEPARATOR if not self.skipOnSave(): for label, field in sorted(self._fields.items()): yield from ( (label + sep + path if path else label, item) for path, item in field.dumpFields() ) def restoreField(self, path: str, value: Optional[str]) -> bool: """ Internal. """ with self._update_notifier.inhibit(): if self._PATH_SEPARATOR in path: field_label, path = path.split(self._PATH_SEPARATOR, 1) else: field_label, path = path, "" if (field := self._fields.get(field_label)) is not None: return field.restoreField(path, value) return False 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__()