-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.html
115 lines (105 loc) · 17.4 KB
/
index.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
<!DOCTYPE html>
<html>
<head>
<title>domino-clj</title>
<meta charset="utf-8">
<!-- KLIPSE ASSETS -->
<link rel="stylesheet" type="text/css" href="codemirror.css">
<script>
window.klipse_settings = {
selector: '.lang-eval-clojure', // css selector for the html elements you want to klipsify
};
</script>
<!-- END KLIPSE ASSETS -->
<link rel="stylesheet" type="text/css" href="style.css">
<link rel="stylesheet" type="text/css" href="highlight.pack.css">
<link rel="apple-touch-icon" sizes="180x180" href="logo/apple-touch-icon.png">
<link rel="icon" type="image/png" sizes="32x32" href="logo/favicon-32x32.png">
<link rel="icon" type="image/png" sizes="16x16" href="logo/favicon-16x16.png">
<link rel="manifest" href="logo/site.webmanifest">
<link rel="mask-icon" href="logo/safari-pinned-tab.svg" color="#74a995">
<meta name="msapplication-TileColor" content="#74a995">
<meta name="theme-color" content="#ffffff">
</head>
<body>
<p><h1><img src="logo/logo.png" title="" style="margin-bottom: -10px; margin-right: 10px;" width="85px"><a href="https://github.com/domino-clj/domino">Domino</a></h1></p><p><a href='https://circleci.com/gh/domino-clj/domino'><img src="https://img.shields.io/circleci/build/gh/domino-clj/domino?label=CircleCI&logo=circleci&style=flat-square" alt="CircleCI" /></a> <a href='https://clojars.org/domino/core'><img src="https://img.shields.io/clojars/v/domino/core?&style=flat-square" alt="Clojars Project" /></a> <a href='https://clojurians.slack.com/messages/domino-clj'><img src="https://img.shields.io/badge/slack-%40clojurians%2Fdomino–clj-blue?logo=slack&style=flat-square" alt="Slack" /></a> <a href='https://clojars.org/domino/core'><img src="https://img.shields.io/clojars/dt/domino/core?color=blue&style=flat-square" alt="Clojars Downloads" /></a> <a href='https://github.com/domino-clj/domino/stargazers'><img src="https://img.shields.io/github/stars/domino-clj/domino?logo=github&style=flat-square" alt="GitHub Stars" /></a></p><p><h3 class="hidden">See <a href="https://domino-clj.github.io">here</a> for interactive documentation.</h3></p><p>Domino is a data flow engine that helps you organize the interactions between your data model and events. Domino allows you to declare your business logic using a directed acyclic graph of events and effects. Whenever an external change is transacted to the data model, the graph determines the chain of events that will be executed, and side effects triggered as a result of the computation.</p><p>Without a way to formalize the interactions between different parts of the application, relationships in code become implicit. This results in code that's difficult to maintain because of the mental overhead involved in tracking these relationships. Domino makes the interactions between pieces of business logic explicit and centralized.</p><p>Domino explicitly separates logic that makes changes to the data model from side effectful functions. Business logic functions in Domino explicitly declare how they interact with the data model by declaring their inputs and outputs. Domino then uses these declarations to build a graphs of related events. This approach handles cascading business logic out of the box, and provides a data specification for relationships in code. Once the changes are transacted, the effectful functions are called against the new state.</p><h2>Version 0.4.0</h2><p>Version 0.4.0 is in pre-alpha and is not on the master branch. We do not recommend using it at this point (or even looking at it really). Please continue to use version 0.3.3 for the time being.</p><h2>Concepts</h2><p>Domino consists of three main concepts:</p><h3>1. Model</h3><p>The model represents the paths within an EDN data structure. These paths will typically represent fields within a document. Each path entry is a tuple where the first value is the path segment, and the second value is the metadata associated with it. If the path is to be used for effects and/or events, the metadata must contain the <code>:id</code> key.</p><p>For example, <code>[:amount {:id :amount}]</code> is the path entry to the <code>:amount</code> key within the data model and can be referenced in your events and effects as <code>:amount</code> (defined by the <code>:id</code>). You can nest paths within each other, such as the following model definition:</p><pre><code class="clojure">[[:patient [:first-name {:id :fname}]]]
</code></pre><h3>2. Events</h3><p>The events define the business logic associated with the changes of the model. Whenever a value is transacted, associated events are computed. Events are defined by three keys; an <code>:inputs</code> vector, an <code>:outputs</code> vector, and a <code>:handler</code> function.</p><p>The handler accepts three arguments: a context containing the current state of the engine, a list of the input values, and a list of the output values. The function should produce a vector of outputs matching the declared <code>:outputs</code> key. For example:</p><pre><code class="clojure">{:inputs [:amount]
:outputs [:total]
:handler (fn [ctx {:keys [amount]} {:keys [total]}]
{:total (+ total amount)})}
</code></pre><p>Domino also provides a <code>domino.core/event</code> helper for declaring events, so the above event can also be written as follows:</p><pre><code class="clojure">(domino.core/event [ctx {:keys [amount]} {:keys [total]}]
{:total (+ total amount)})
</code></pre><p>The macro requires that the <code>:keys</code> destructuring syntax is used for input and outputs, and expands the the event map with the <code>:inputs</code> and <code>:outputs</code> keys being inferred from the ones specified using the <code>:keys</code> in the event declaration.</p><p>It's also possible to declare async events by providing the <code>:async?</code> key, e.g:</p><pre><code class="clojure">{:async? true
:inputs [:amount]
:outputs [:total]
:handler (fn [ctx {:keys [amount]} {:keys [total]} callback]
(callback {:total (+ total amount)}))}
</code></pre><p>Async event handler takes an additional argument that specifies the callback function that should be called with the result.</p><h3>3. Effects</h3><p>Effects are used for effectful operations, such as IO, that happen at the edges of the computation. The effects do not cascade. An effect can contain the following keys:</p><ul><li><code>:id</code> - optional unique identifier for the event</li><li><code>:inputs</code> - optional set of inputs that trigger the event to run when changed</li><li><code>:outputs</code> - optional set of outpus that the event will produce when running the handler</li><li><code>:handler</code> - a function that handles the business logic for the effect</li></ul><h4>Incoming Effects</h4><p>Effects that declare <code>:outputs</code> are used to generate the initial input to the engine. For example, an effect that injects a timestamp can look as follows:</p><pre><code class="clojure">{:id :timestamp
:outputs [:ts]
:handler (fn [_ {:keys [ts]}]
{:ts (.getTime (java.util.Date.))})}
</code></pre><p>The effect has an <code>:id</code> key specifying the unique identifier that is used be trigger the event by calling the <code>domino.core/trigger-effects</code> function. This function accepts a collection of event ids, e.g: <code>(trigger-effects ctx [:timestamp])</code>.</p><p>The handler accepts two arguments: a context containing the current state of the engine, and a list of output values.</p><h4>Outgoing Effects</h4><p>Effects that declare <code>:inputs</code> will be run after events have been transacted and the new context is produced. These effects are defined as a map of <code>:inputs</code> and a <code>:handler</code> function.</p><p>The handler accepts two arguments: a context containing the current state of the engine, and a list of input values.</p><p> For example:</p><pre><code class="clojure">{:inputs [:total]
:handler (fn [ctx {:keys [total]}]
(when (> total 1337)
(println "Woah. That's a lot.")))}
</code></pre><h2>Usage</h2><p><strong>1. Require <code>domino.core</code></strong></p><p><pre><code class="language-clojure lang-eval-clojure" data-external-libs="https://raw.githubusercontent.com/domino-clj/domino/master/src"> (require '[domino.core :as domino]) </code></pre></p><p><strong>2. Declare your schema</strong></p><p>Let's take a look at a simple engine that accumulates a total. Whenever an amount is set, this value is added to the current value of the total. If the total exceeds <code>1337</code> at any point, it prints out a statement that says <code>"Woah. That's a lot."</code></p><pre><code class="clojure lang-eval-clojure">(def schema
{:model [[:amount {:id :amount}]
[:total {:id :total}]]
:events [{:id :update-total
:inputs [:amount]
:outputs [:total]
:handler (fn [ctx {:keys [amount]} {:keys [total]}]
{:total (+ total amount)})}]
:effects [{:inputs [:total]
:handler (fn [ctx {:keys [total]}]
(when (> total 1337)
(js/alert "Woah. That's a lot.")))}]})
</code></pre><p>This schema declaration is a map containing three keys:</p><ul><li>The <code>:model</code> key declares the shape of the data model used by Domino.</li><li>The <code>:events</code> key contains pure functions that represent events that are triggered when their inputs change. The events produce updated values that are persisted in the state.</li><li>The <code>:effects</code> key contains the functions that produce side effects based on the updated state.</li></ul><p>Using a unified model referenced by the event functions allows us to easily tell how a particular piece of business logic is triggered.</p><p>The event engine generates a direct acyclic graph (DAG) based on the <code>:input</code> keys declared by each event that's used to compute the new state in a transaction. This approach removes any ambiguity regarding when and how business logic is executed.</p><p>Domino explicitly separates the code that modifies the state of the data from the code that causes side effects. This encourages keeping business logic pure and keeping the effects at the edges of the application.</p><p><strong>3. Initialize the engine</strong></p><p>The <code>schema</code> that we declared above provides a specification for the internal data model and the code that operates on it. Once we've created a schema, we will need to initialize the data flow engine. This is done by calling the <code>domino/initialize</code> function. This function can be called by providing a schema along with an optional initial state map. In our example, we will give it the <code>schema</code> that we defined above, and an initial value for the state with the <code>:total</code> set to <code>0</code>.</p><pre><code class="clojure lang-eval-clojure">(def ctx (atom (domino/initialize schema {:total 0})))
</code></pre><p>Calling the <code>initialize</code> function creates a context <code>ctx</code> that's used as the initial state for the engine. The context will contain the model, events, effects, event graph, and db (state). In our example we use an atom in order to easily update the state of the engine.</p><p><strong>4. Transact your external data changes</strong></p><p>We can update the state of the data by calling <code>domino/transact</code> that accepts the current <code>ctx</code> along with an inputs vector, returning the updated <code>ctx</code>. The input vector is a collection of path-value pairs. For example, to set the value of <code>:amount</code> to <code>10</code>, you would pass in the following input vector <code>[[[:amount] 10]]</code>.</p><pre><code class="clojure lang-eval-clojure">(swap! ctx domino/transact [[[:amount] 10]])
</code></pre><p>The updated <code>ctx</code> contains <code>:domino.core/change-history</code> key which is a simple vector of all the changes as they were applied to the data in execution order of the events that were triggered.</p><pre><code class="clojure lang-eval-clojure">(:domino.core/change-history @ctx)
</code></pre><p>We can see the new context contains the updated total amount and the change history shows the order in which the changes were applied.</p><p>The <code>:domino.core/db</code> key in the context will contain the updated state reflecting the changes applied by running the events.</p><pre><code class="clojure lang-eval-clojure">(:domino.core/db @ctx)
</code></pre><p>Finally, let's update the <code>:amount</code> to a value that triggers an effect.</p><pre><code class="clojure lang-eval-clojure">(require '[reagent.core :as reagent])
(defn button []
[:button
{:on-click #(swap! ctx domino/transact [[[:amount] 2000]])}
"trigger effect"])
(reagent/render-component [button] js/klipse-container)
</code></pre><h3>Interceptors</h3><p>Domino provides the ability to add interceptors pre and post event execution. Interceptors are defined in the schema's model. If there are multiple interceptors applicable, they are composed together.</p><p>In the metadata map for a model key, you can add a <code>:pre</code> and <code>:post</code> key to define these interceptors. Returning a <code>nil</code> value from an interceptor will short circuit execution. For example, we could check if the context is authorized before running the events as follows:</p><pre><code class="clojure lang-eval-clojure">(let [ctx (domino/initialize
{:model [[:foo {:id :foo
:pre [(fn [handler]
(fn [ctx inputs outputs]
;; only run the handler if ctx contains
;; :authorized key
(when (:authorized ctx)
(handler ctx inputs outputs))))]
:post [(fn [handler]
(fn [result]
(handler (update result :foo #(or % -1)))))]}]]
:events [{:inputs [:foo]
:outputs [:foo]
:handler (fn [ctx {:keys [foo]} outputs]
{:foo (inc foo)})}]})]
(map :domino.core/db
[(domino/transact ctx [[[:foo] 0]])
(domino/transact (assoc ctx :authorized true) [[[:foo] 0]])]))
</code></pre><h3>Triggering Effects</h3><p>Effects can act as inputs to the data flow engine. For example, this might happen when a button is clicked and you want a value to increment. This can be accomplished with a call to <code>trigger-effects</code>.</p><p><code>trigger-effects</code> takes a list of effects that you would like trigger and calls <code>transact</code> with the current state of the data from all the inputs of the effects. For example:</p><pre><code class="clojure lang-eval-clojure">(let [ctx
(domino.core/initialize
{:model [[:total {:id :total}]]
:effects [{:id :increment-total
:outputs [:total]
:handler (fn [_ current-state]
(update current-state :total inc))}]}
{:total 0})]
(:domino.core/db (domino.core/trigger-effects ctx [:increment-total])))
</code></pre><p>This wraps up everything you need to know to start using Domino. You can see a more detailed example using Domino with re-frame <a href='https://domino-clj.github.io/demo'>here</a>.</p><h2>Possible Use Cases</h2><ul><li>UI state management</li><li>FSM</li><li>Reactive systems / spreadsheet-like models</li></ul><h2>Example App</h2><ul><li>demo applications can be found <a href='https://github.com/domino-clj/examples'>here</a></li></ul><h2>Inspirations</h2><ul><li><a href='https://github.com/Day8/re-frame'>re-frame</a></li><li><a href='https://github.com/hoplon/javelin'>javelin</a></li><li><a href='https://github.com/metosin/reitit'>reitit</a></li></ul><h2>License</h2><p>Copyright © 2019</p><p>Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.</p>
<script src="highlight.pack.js"></script>
<script>
document.addEventListener('DOMContentLoaded', (event) => {
document.querySelectorAll('pre code.clojure').forEach((block) => {
hljs.highlightBlock(block);
});
});
</script>
<script src="https://storage.googleapis.com/app.klipse.tech/plugin/js/klipse_plugin.js"></script>
</body>
</html>