Skip to main content

Observable managed attributes (aka, observable properties)

Project description

Observable managed attributes (also known as "observable properties")

In summary:

  • Declare observable properties using the @observable decorator instead of @property.
  • Subscribe/unsubscribe any number of callback functions to observable properties.
  • If the value of an observable property changes, all subscribed callbacks are executed.

How to use

Declare observable properties in your class

Use the @observable decorator as you do with @property:

from observable_properties import observable
class Test():
    def __init__(self):
        self._value = 0
    @observable
    def value(self):
        return self._value
    @value.setter
    def value(self, value):
        self._value = value

Needless to say, observable properties are also regular properties and they work just the same:

t = Test()
t.value = 1000
print(f"Current value of t: {t.value}")

Subscribe to property changes

In the context of this library, an observer is a callback function to be executed when the value of a property is overwritten in a given object. For example:

def observer(instance,property_name,new_value):
    print(f"{instance.__class__.__name__}.{property_name} = {new_value}")

To start observing a property, call subscribe() passing the instance and property name to observe along with the callback function:

from observable_properties import subscribe

subscribe(observer,t,"value")

Which means "subscribe observer to t.value". Now, the observer is executed after the value of the observed property changes at the observed object:

t.value = 2000

prints:

Test.value = 2000

Additionally:

  • Observers are not allowed to change the value of the observed property. ObservablePropertyError is raised in such a case.
  • The same observer can subscribe to many objects and properties.
  • Any number of observers can subscribe to the same object and property. All of them will run. The execution order depends on subscription order.

Unsubscribe

To stop observing a property, call unsubscribe() with the same parameters that were given to subscribe():

from observable_properties import unsubscribe

unsubscribe(observer,t,"value")

Syntactic sugar

To speed things up, derive your class from the Observable class (note the capital letter). The declaration itself does not differ too much from the previous example:

from observable_properties import Observable
class ObservableTest(Observable):
    def __init__(self):
        self._value = 0
    @observable
    def value(self):
        return self._value
    @value.setter
    def value(self, value):
        self._value = value

Observable classes offer subscribe() and unsubscribe() methods with a slightly different syntax to their function counterparts:

ot = ObservableTest()
ot.subscribe("value",observer)
ot.value = 900
ot.unsubscribe("value", observer)
ot.value = 500

prints:

ObservableTest.value = 900

Observable changes triggered from non-setter methods

A public setter is not mandatory for observable properties of an Observable instance. This is a common use case for computed properties.

Any method can notify changes in observable properties in two ways:

  • Compute the new value inside the context returned by self._observable(). The new value is automatically notified to all subscribers on exit. As an example, see Product.set() below.
  • Explicitly call self._observable_notify() after the computation is finished. As an example, see Product.set_alternate() below.
class Product(Observable):
    def __init__(self):
        self._a = 0
        self._b = 0
    @observable
    def value(self):
        return self._a * self._b
    def set(self, a, b):
        with self._observable("value"):
            self._a = a
            self._b = b
    def set_alternate(self, a, b):
        changed = (a!=self._a) or (b!=self._b)
        self._a = a
        self._b = b
        if (changed):
            self._observable_notify("value")
mult = Product()
mult.subscribe("value", observer)
mult.set(3, 5)
mult.set_alternate(4,3)

prints:

Product.value = 15
Product.value = 12

Other notes

  • subscribe(), unsubscribe(), _observable() and _observable_notify() raise ObservablePropertyError on non-observable or non-existing properties.
  • Subscribing twice to the same object and property has no effect.
  • Unsubscribing to non-subscribed properties has no effect, but unsubscribe() returns False.
  • Objects hold strong references to observers. Deleting an observer does not prevent it from running. Unsubscribe first.
  • Coroutines are accepted as observers. They are executed by the means of asyncio.run()

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

observable-properties-4.0.0.tar.gz (16.9 kB view details)

Uploaded Source

Built Distribution

observable_properties-4.0.0-py3-none-any.whl (15.0 kB view details)

Uploaded Python 3

File details

Details for the file observable-properties-4.0.0.tar.gz.

File metadata

  • Download URL: observable-properties-4.0.0.tar.gz
  • Upload date:
  • Size: 16.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.11.7

File hashes

Hashes for observable-properties-4.0.0.tar.gz
Algorithm Hash digest
SHA256 cf3f318ae07df428e6f496ec204f6257d5b28ed3097cd4463b6d2b91038d192d
MD5 863f2e577778b36e4798b79ecb31d2af
BLAKE2b-256 8bfaea9ddde11de55ffd41947da61fa27eb9c53b858c92cdf5b6c0d78a7bbb3f

See more details on using hashes here.

File details

Details for the file observable_properties-4.0.0-py3-none-any.whl.

File metadata

File hashes

Hashes for observable_properties-4.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 51664235f5bf9692ddafb4ef85cbef1f41de541f69105d6cb253f3a81468ab44
MD5 b59ce47f4a8dbfac92b1a89db4588d4d
BLAKE2b-256 4ceca7d2fc07ab94d2c57ea31de13d71e3368c6b3005298f8ff2675c19629608

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page