Mar 3, 2024 • Arnout Roemers
The redelay library - the unassuming state lifecycle tool - had several unreleased improvements piling up. Some of these are breaking, so a new major version has finally been released. Upgrading should be easy though. This short post goes through the improvements.
The redelay library is for starting and stopping the stateful parts of your Clojure application, in the correct order.
There are more libraries like this - like integrant, component, or mount.
Redelay brings this down its core in a flexible way, by having State
objects that can live anywhere.
Think of them like Clojure’s delay
, but stoppable.
To give a small example, this is what it looks like:
Just plain objects, no framework to follow. All of the above has not changed with the new major release. The devil is in the smaller details.
The State
object now supports Clojure’s peek
function.
This can be used to inspect the value of a State
, but it won’t start it if is not realized already.
This makes the State
object more complete and useable with Clojure’s fantastic core toolbelt.
The defstate
macro - which is similar to (def .. (state ..))
- is now automatically tagged as :dynamic
.
This allows for easy rebinding the state in tests.
For example:
So you can remove the ^:dynamic
tags from your code if you were using the defstate
like this already.
Or, perhaps it inspires you for structuring your tests like this.
The library has an extension point, called the watchpoint
.
You can add callbacks to this, using Clojure’s add-watch
.
This is still the case in version 2.0, however the arguments to the callback are now different.
As you might know, a watch callback receives four arguments: a key, the reference, its old-state, and its new-state.
In version 1.0 a stopped State
would be put in the third (old-state) argument, and a started State
would be put in the fourth (new-state) argument.
In version 2.0 the arguments to the callback are different, and the callbacks happen more often.
The third argument is now one of :starting
, :started
, :stopping
and :stopped
, and the fourth argument is the State
object it concerns.
For example:
This allows for more thorough logging for example, and it could even be used to deduce a dependency graph while starting a State
.
So, if you are using this extension point currently, you will have to update it a bit when upgrading to this new version.
The State
object previously supported Clojure’s with-meta
.
Since this is a pure function - and was implemented as such - this resulted in a new State
object.
This could lead to undesirable and suprising effects.
In version 2.0 support for with-meta
has been replaced with Clojure’s alter-meta!
and reset-meta!
.
These functions update the meta-data of State
objects in place.
No more suprises.
That’s it for the major changes to the library. The smaller additions, changes and fixes you can find in the changelog. The updated documentation can be found at cljdoc.
As always, have fun!