A Clojure library for boxing values or exceptions for passing data through channels of core.async.
The latest version on Clojars.
You can wrap any value with box/value
, and transform it's value with box/map
, box/filter
, box/mapcat
.
A boxed value is derefable by deref
or @
.
(require '[databox.core :as box] '[clojure.string :as string])
(let [boxed (box/value :a)]
(-> boxed
(box/map name)
(box/map string/upper-case)
(deref)))
;; "A"
All exceptions occurred in transformers like box/map
, box/filter
aren't thrown, but converted as failure box. And transformers ignore failure box. So we can handle boxed data in channel pipeline without any care of exceptions. Just throw them. Transformer functions never called on failure box and thrown exceptions in transformers automatically converted to failure box. The wrapped exception in failure box will be thrown when the box is dereferenced.
box/value
never be nested. If you call box/value
on already boxed data, box/value
returns the data unchanged.
(let [boxed (box/value :a)]
(= boxed (box/value boxed))) ; => true
If you wrap a throwable object with box/value
, the box become a box/failure
and will throw the contained throwable when it's dereference.
(let [boxed (box/value (ex-info "error" {}))]
@boxed) ;; <-- throw an exception
All transformers like box/map
will convert a normal value to boxed value automatically, so you can pass any value to the transformers in safe.
(let [data (box/map :a name)] ; :a is not a boxed data.
(assert (true? (box/box? data)))
(assert (= "a" @data)))
You can also use transformers as transducers.
(doseq [boxed (sequence (box/map name) [(box/value :a) (box/value :b) (box/value :c)])]
(println @boxed))
;; A
;; B
;; C
All transformers ignore failure-boxed data. You don't need to care that if sequences or channels contains error data. Just transform them with box-transformers.
with sequence:
(let [data-coll [(box/value :a) (box/failure (ex-info "error" {})) (box/value :c)]
converted (sequence (box/map name) data-coll)] ;; No exception because failure-box is ignored by box/map
(doseq [boxed converted]
(println @boxed))) ;; will thow an exception when the second item is unwrapped.
with channels:
(let [my-channel (chan)
ch (pipe my-channel
(chan 1 (box/map name)))] ;; You can connect a tranformer as a transducer to a channel.
(go
;; Here we send a (boxed) throwable to a channel and it will be transformed by `box/map`,
;; But `box/map` ignores all error data so exception never be thrown.
;; We can apply box transformers on channel pipelines in safe.
(onto-chan my-channel [(box/value :a) (box/value (ex-info "error" {})) (box/value :c)]))
(go-loop []
(when-let [data (<! ch)]
;; all exceptions will be thrown when it derefed.
;; so all errors in channel pipelines never be thrown IN pipeline,
;; but be thrown at the END of pipeline.
(println @data)
(recur))))
You can not only convert a data by box/map, but can do filter, mapcat and distinct them.
(let [mychannel (chan)
ch (-> mychannel
;; string/split returns a seq of string and the each string will be boxed by `box/mapcat`
(pipe (chan 1 (box/mapcat #(string/split % #","))))
;; all "a" are removed
(pipe (chan 1 (box/filter #(not= "a" %))))
;; all duplicated items will be removed
(pipe (chan 1 (box/distinct))))]
(go
(>! mychannel (box/value "a,b,a,c,c,b,d")))
(go-loop []
(when-let [item (<! ch)]
(println @item)
(recur))))
;; b
;; c
;; d
With maplet
, you can safely unwrap boxed values and process them. But currently only one value is supported.
maplet is a syntax sugar of box/map
.
(maplet [v boxed-value]
(name v))
If you don't certain it a value is boxed or normal value, you can deref it with box/unbox
in safe.
box/unbox
returns an unboxed (derefed) value or the value itself if the value is not boxed.
(box/unbox :a)
;; :a
You can check if a boxed value contains an exception or not by box/failure?
or box/success?
.
if a value contains a throwable, box/failure?
returns true.
(let [boxed (boxed/failure (ex-info "error" {}))]
(box/failure? boxed) ;=> true
(box/success? boxed) ;=> false
)
If you need the exception wrapped in a boxed data, you can use box/exception
.
(let [boxed (boxed/failure (ex-info "error" {}))
ex (box/exception boxed)]
(ex-data ex))
box/box?
return true if a data is boxed data.
(box/box? (box/value :a)) ; => true
(box/box? :a) ; => false
You cat apply a function to the value contained in a databox by using box/apply fn.
(box/apply (box/success :a) name)
;=> "a"
If you use the 'box/apply' onto a failure box, it just returns a nil, or the :failure value of option map.
(box/apply (box/failure (ex-info "Test Exception" {})) name)
;;=> nil
(box/apply (box/failure (ex-info "Test Exception" {})) name :failure :error)
;;=> :error
You can 'disable' or 'enable' databoxes. Disabled boxes never accept tranformers even if it is not failure-boxes.
You can disable a box by box/disable
.
(box/map (box/disable (box/success :a)) name)
;=> (box/success :a)
And enable the disabled box by box/enable
.
(def disabled (box/disable (box/success :a)))
(box/enable disabled) ;=> returns (box/success :a) that is not disabled.
Copyright © 2019 YANO Tsutomu
Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.