Tools for livecoding performances
Project description
#doitlive - Tools for livecoding performances in Python.
#doitlive allows you to “refresh” a class by reloading the source in an almost-sane and almost-safe way. It also provides facilities for catching and handling common errors that come up with livecoding, such as undeclared variables.
It works under the philosophy that it’s better to have weird behavior than to crash.
Besides livecoding, it can also be used as a tool for debugging or experimenting – you can tweak algorithms as they run.
Projects
Installation
Install from pip:
pip install refreshable
Or install the latest (ha!) from git:
git clone https://github.com/zbanks/doitlive.git cd doitlive python setup.py install
refreshable.SafeRefreshableMixin
Provides a .refresh() method to reload a class
Adding the SafeRefreshMixin to a class allows you to “refresh” the class using the .refresh() method. This method reloads the python file the class came from and replaces all methods and class variables (not instance variables!) with the versions from the new file.
The refresh is “safe” because it tries very hard to keep the program running. On each refresh, a snapshot of the class is saved to form a history. If an error is encountered while performing the refresh, the state is reverted.
In general, you can wrap calls to methods of your refreshed class in a try block that catches all errors and call .revert() on failure.
Additionally, DEFAULTS and AUTO_NONE provide options for handling missing attributes (preventing AttributeErrors).
Usage
You can configure the behavior by setting the following class variables:
STATICS : List of variable names (strings) that are not refreshed.
DEFAULTS : Dictionary of variable names -> values. If an AttributeError is caught, and the attribute is in DEFAULTS, the attribute is populated from the dictionary. This can be useful if you need to initialize a new state variable.
AUTO_NONE: If True, catch AttributeErrors and set the attribute to None if the attribute is not in DEFAULTS.
Additionally, there are the .pre_refresh() and .post_refresh() hooks which can be overridden.
Once initialized, instances have the following methods:
.init_defaults(): Initialize attributes from the DEFAULTS dict.
.refresh() : Attempt to reload the class from disk.
.revert() : Revert the changes from the previous .refresh().
.purge() : Remove the state history. Each call to .refresh() takes a snapshot of the class. If you refresh often w/ a big class, this can get large.
Limitations
.refresh() assumes all methods are bound (take a self parameter). As a result, static/class methods (methods declared with @staticmethod, or @classmethod) will not be refreshed properly. These method names should be added to STATICS and they will not be refreshed.
This framework was designed around the singleton model with one instance of the given refreshed class. It hasn’t been extensively tested with multiple instances, and may cause weird behavior around class variables.
The __main__ module cannot be reloaded, so the class must exist in an imported module.
refreshable.SafeRefreshableLoop
Run a function in a loop while making the parent class refreshable.
The function .step() is called repeatedly while the loop is running. You can start the loop in one of two ways:
.start(): Run the loop in a thread.
.run() : (the target of the thread) Run the loop “inline”.
The loop can also be paused with .stop() and unpaused with .restart().
If you subclass, make sure you call threading.Thread.__init__
As with the SafeRefreshMixin, you can set the following class variables:
STATICS : List of variable names (strings) that are not refreshed.
DEFAULTS : Dictionary of variable names -> values. If an AttributeError is caught, and the attribute is in DEFAULTS, the attribute is populated from the dictionary. This can be useful if you need to initialize a new state variable.
AUTO_NONE: If True, catch AttributeErrors and set the attribute to None if the attribute is not in DEFAULTS.
And call the following methods:
.refresh(): Attempt to reload the class from disk.
.revert() : Revert the changes from the previous .refresh().
.purge() : Remove the state history. Each call to .refresh() takes a snapshot of the class. If you refresh often w/ a big class, this can get large.
Additionally, there are the .pre_refresh() and .post_refresh() hooks which can be overridden.
Testing
Test it out by running python run_test.py and modifying test.py.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
File details
Details for the file refreshable-1.0.2.tar.gz
.
File metadata
- Download URL: refreshable-1.0.2.tar.gz
- Upload date:
- Size: 5.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 5e1f7fa52fdd12fe0385b72254a24c9840b95afebcb335990637390f4da69b7f |
|
MD5 | cd252796ccac0afdd6f7456093c53c31 |
|
BLAKE2b-256 | ce303a4c4ff1de2817ddbac528cffb54d9003e4dae5da225ce6f2463d265b4b5 |