Source code for sunset.protocols

import weakref

from types import GenericAlias
from typing import (
    Any,
    Callable,
    Generic,
    Iterator,
    Optional,
    Protocol,
    TypeVar,
    Union,
    runtime_checkable,
)


Self = TypeVar("Self")
_T = TypeVar("_T")


# pylint: disable=unnecessary-ellipsis
# The ellipses in the protocol definitions below are in fact necessary: they let
# the type checker know it's fine we're not returning values.


[docs] @runtime_checkable class Serializable(Protocol): """ A protocol to be implemented by a class in order to enable storing instances of that class in a Key. The two methods to be implemented are :meth:`toStr()` and (classmethod) :meth:`fromStr()`. """
[docs] def toStr(self) -> str: """ Returns a string representation of this instance that can be used by :meth:`fromStr()` to reconstruct a copy of this instance. """ ...
[docs] @classmethod def fromStr(cls: type[Self], string: str) -> Optional[Self]: """ Takes a string that represents a serialized instance of this class, and returns a newly created instance that corresponds to that representation, or None if the string is not a valid serialized representation of an instance of this class. """ ...
[docs] class Serializer(Generic[_T], Protocol): """ A protocol that describes a way to serialize and deserialize an arbitrary type. SunsetSettings provides its own serializers for common types (int, float, bool, str, enum.Enum). In order to store an arbitrary type in a Key, users need to provide a serializer for that type when instantiating a Key. That serializer should be an implementation of this protocol. """
[docs] def toStr(self, value: _T) -> str: """ Returns a string representation of the given value, that can be used by :meth:`fromStr()` to reconstruct a copy of that value. """ ...
[docs] def fromStr(self, string: str) -> Optional[_T]: """ Takes a string that represents a serialized instance of a value, and returns a newly created instance that corresponds to that string, or None if the string is not a valid representation of a value for this serializer's type. """ ...
@runtime_checkable class Inheriter(Protocol): def setParent(self: Self, parent: Optional[Self]) -> None: ... def parent(self: Self) -> Optional[Self]: ... def children(self: Self) -> Iterator[Self]: ... @runtime_checkable class Dumpable(Protocol): def dumpFields(self) -> Iterator[tuple[str, Optional[str]]]: ... def restoreField(self, path: str, value: Optional[str]) -> bool: ... def isSet(self) -> bool: ... @runtime_checkable class UpdateNotifier(Protocol): def onUpdateCall(self, callback: Callable[[Any], Any]) -> None: ... @runtime_checkable class ItemTemplate(Protocol): def _typeHint(self) -> Union[type, GenericAlias]: ... def _newInstance(self: Self) -> Self: ... @runtime_checkable class Containable(Protocol): _PATH_SEPARATOR: str def _setContainer( self, label: str, container: Optional["Container"] ) -> None: ... def _container(self) -> Optional["Container"]: ... def fieldLabel(self) -> str: ... def fieldPath(self) -> str: ... def skipOnSave(self) -> bool: ... @runtime_checkable class Container(Containable, UpdateNotifier, Protocol): def _containsFieldWithLabel(self, label: str, field: "Containable") -> bool: ... def _triggerUpdateNotification( self, field: "Optional[UpdateNotifier]" ) -> None: ... class ContainableImpl: """ A ready-to-use implementation of the Containable protocol. """ _PATH_SEPARATOR: str = "." _field_label: str = "" _container_ref: Optional[weakref.ref[Container]] = None def _setContainer(self, label: str, container: Optional[Container]) -> None: """ Internal. """ if container is None: self._container_ref = None self._field_label = "" else: self._container_ref = weakref.ref(container) self._field_label = label def _container(self) -> Optional[Container]: """ Internal. """ if self._container_ref is None: return None container = self._container_ref() if container is None: return None # Make sure this Containable is in fact still held in its supposed # Container. Else update the situation. if not container._containsFieldWithLabel(self._field_label, self): self._setContainer("", None) return None return container def fieldLabel(self) -> str: """ Internal. Returns the label under which this entity was created in whatever entity contains it, if any. For a class-type entity, the label would be e.g. the attribute name. For a list-type entity, an index into the list. Returns: An opaque identifier. Don't rely on it not changing between SunsetSettings releases. Example: >>> from sunset import Settings, Key >>> class TestSettings(Settings): ... a_key: Key[str] = Key(default="") >>> settings = TestSettings() >>> print(settings.a_key.fieldLabel()) a_key """ if self._container() is None: self._field_label = "" return self._field_label def fieldPath(self) -> str: """ Internal. Returns an opaque string that uniquely identifies this element in the settings hierarchy. Returns: An opaque identifier. Don't rely on it not changing between SunsetSettings releases. Example: >>> from sunset import Bunch, Key, Settings >>> class TestSettings(Settings): ... class TestBunch(Bunch): ... a_key: Key[str] = Key(default="") ... a_bunch: TestBunch = TestBunch() >>> settings = TestSettings() >>> section = settings.newSection("A section") >>> print(section.a_bunch.a_key.fieldPath()) main/asection/a_bunch.a_key """ path = ( "" if (container := self._container()) is None else container.fieldPath() ) return path + self.fieldLabel() def skipOnSave(self) -> bool: """ Internal. Returns whether this entity should be disregarded when saving these settings. For entities with an attribute name, it's equivalent to checking if the attribute is private (its name starts with an underscore). For entities with a section name, it's equivalent to checking if the section name is empty. Can be overridden in subclasses. Returns: A bool used internally by the settings saving logic. """ return self.fieldLabel().startswith("_") assert isinstance(ContainableImpl, Containable) @runtime_checkable class Field( Containable, Dumpable, Inheriter, ItemTemplate, UpdateNotifier, Protocol, ): ...