A dynamic approach to application components
objection is about objects that acquire resources globally such as connections, connection pools, threads, thread pools, servers, processes etc.
It provides a means of registering such objects in a global registry giving you oversight of what is 'running' in your program.
(defn jdbc-conn
"Returns a registered jdbc db-spec. Stop with obj/stop!"
[url]
(obj/register (db/pool url) {:name "JDBC Conn" :stopfn db/shutdown}))
(jdbc-conn some-url)
(obj/status)
;; =>
;; 1 objects registered.
;; -------
;; objects:
;; -------
;; 81e73f11-dc5f-4576-b706-420fa53856d7 - JDBC Conn
(obj/stop! "81e")
- Provides functions to register 'objects' in your program.
- Objects can be registered at any time and on any thread.
- Registered objects can be stopped, made dependent on one another, have data associated with them, be named etc.
- Objects that are stopped shutdown their dependent objects first.
- Objects can be anything really, no wrappers or protocol impls required.
Certain objects in most programs have global implications that are not managed by the garbage collector. They often have life cycles, acquire resources or have side-effects just by being 'active' (e.g a thread). They are often dependent on one another, and those dependencies are often implicit and the management of clean shutdowns is often hand-rolled.
In concurrent scenarios all this becomes even more difficult, particularly in programs where such objects are being spawned/shutdown at runtime on multiple threads.
The repl complicates things further, how often have you created a core.async process only to lose the reference to a channel?
Objection provides a set of tools to manage the complexity inherent in objects of this nature, helping you write robust programs to have a better time at the repl when your program is changing as you edit it.
Component/Integrant/Mount already solves this problem, why not use that or make a PR or something
I am a big fan of the excellent integrant and component libraries and have been using them for almost as long as I have been using clojure.
However objection takes a different approach that is rather more dynamic than component, integrant or mount, after feeling that they bank too hard on a static application whose topology does not change at runtime.
One thing that is particularily nice about both integrant and component is the declarative wiring of components - I would suggest using either library alongside objection if that has value for your application.
[riverford/objection "0.1.3"]
Objection manages regular old objects that have been registered with objection.
You can register an object with register
, it returns the object passed as-is.
(require
'[objection.core :as obj]
'[ring.adapter.jetty :as jetty]
'[ring.util.response :as resp])
(defn start-server
[handler port]
(-> (jetty/run-jetty handler {:port port :join? false})
(obj/register
{ ;; all optional
:name (str "Jetty Server on port " port)
:alias [:jetty-server port]
:data {:handler handler :port port}
;; optional, but wise!
:stopfn (fn [server] (.stop server))})))
(start-server (fn [_] (resp/response "Hello World")) 8080)
Each registered object is assigned an id, you can get all the currently registered object ids from the id-seq
function. Alternatively use the function (status)
to print some useful data.
(obj/status)
;; 1 objects registered.
;; -------
;; objects:
;; -------
;; 81e73f11-dc5f-4576-b706-420fa53856d7 - Jetty Server on port 8080
Each registered object can be queried. Each function that takes a registered object will work on an id (or prefix), alias, as well as the object itself.
describe
will return data about the object.
(obj/describe "81e7")
;; =>
{:registered? true,
:id "81e73f11-dc5f-4576-b706-420fa53856d7",
:name "Jetty Server on port 8080",
:data {:handler #object[user$eval1843$fn__1844 0x45a21de2 "user$eval1843$fn__1844@45a21de2"] :port 8080},
:aliases #{[:jetty-server 8080]},
:deps #{},
:dependents #{}}
object
will return the object instance itself.
(obj/object "81e7")
id
will return the id of the object or alias if it is registered.
(obj/id [:jetty-server 8080])
(obj/id (obj/object "81e7"))
(obj/id "81e7")
;; all return the string
;; =>
"81e73f11-dc5f-4576-b706-420fa53856d7"
Registered objects can be stopped using the stop! function. Again an alias/id etc can be used interchangeably with the object.
stop!
will call the :stopfn
if one was supplied on registry, if not it will look for an implementation of the protocol obj/IAutoStoppable
or java.lang.AutoCloseable
.
(obj/stop! "81e73f11-dc5f-4576-b706-420fa53856d7")
You can use stop-all!
to stop each and every object currently registered.
(obj/stop-all!)
Registered objects can be dependent on one another, manage dependencies through
the :deps
opt on registry, or using the depend
/undepend
functions.
When constructing an object that is dependent on other objects, it is better to
use the construct
macro as it protects against dependencies
being stopped on other threads while the construction logic is run.
construct
takes the same options as register
, but takes them before the body containing
the construction code.
(defn arbitrary-object
[server]
(obj/construct
{:deps [server]
:stopfn (fn [_] (println "stopping object"))}
(Object.)))
;; restart the server and construct the object.
(let [server (start-server (fn [_] (resp/response "Hello World")) 8080)]
(arbitrary-object server))
;;b2af4b34-d37a-4f6a-892e-36db94aa95ac
(obj/status)
;; 2 objects registered.
;; -------
;; objects:
;; -------
;; b2af4b34-d37a-4f6a-892e-36db94aa95ac - Jetty Server on port 8080
;; f0094e78-e886-4ff0-9e9d-8dd632ea66df - java.lang.Object
;; now if we stop the server, objection will first stop the dependent object.
(obj/stop! [:jetty-server 8080])
;; stopping object
;; => nil
Sometimes global singletons are not so bad if they are used carefully.
For example a good candidate for a singleton is a threadpool that is local to a namespace and used to optimize functions whose api in no way needs to reflect the implementation detail of the thread pool.
e.g the go
macro in core.async
Define a singleton with defsingleton
, defsingleton does not evaluate its body, so they are safe to define in any order.
Redefinition of the singleton will stop any existing instance for the singleton (and any dependent objects).
(obj/defsingleton :my-threadpool
;; the register is optional as singletons will always be registered
;; but you can use it if you want to supply a name or deps etc
(obj/register
(java.util.concurrent.Executors/newFixedThreadPool 4)
{:name "My Threadpool"
:stopfn (fn [tp] (println "Closing threadpool") (.shutdown tp))}))
Grab a singleton with singleton
, at this point the singleton definition will be evaluated
and a registered object will be returned. Repeatedly calling singleton with the same key will return the same object.
(obj/singleton :my-threadpool)
Singletons are always registered and aliased with the key of the singleton. So you can call any of the normal objection functions with the singleton key e.g
(obj/describe :my-threadpool)
;; =>
{:registered? true,
:singleton-key :my-threadpool,
:singleton-ns user,
:id "adb8b07b-959d-4327-8442-722d813e17e0",
:aliases #{:my-threadpool},
:deps #{},
:dependents #{}}
You can associate arbitrary data with an object on registry via a :data
key.
Later you can retrieve it with obj/data
, or alter it via alter-data!
.
(obj/register foo {:data {:fred 42}})
(obj/data foo) ;; => {:fred 42}
Pull requests wecome!
- Clojurescript support
- tools.namespace reloaded support
- More documentation, examples, usage guidelines
- More introspection
- Better query based on user data
- Middleware (on-stop/on-start etc)
- Visualization and tools for humans
Copyright © 2017 Riverford Organic Farmers
Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.