Skip to content
/ databox Public

A library for boxing values or exceptions for passing data through channels of core.async.

License

Notifications You must be signed in to change notification settings

tyano/databox

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

35 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

databox

A Clojure library for boxing values or exceptions for passing data through channels of core.async.

Build

The latest version on Clojars.

Clojars Project

Usage

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

Disable databox

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.

License

Copyright © 2019 YANO Tsutomu

Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.

About

A library for boxing values or exceptions for passing data through channels of core.async.

Resources

License

Stars

Watchers

Forks

Packages

No packages published