2016-09-22 22:56:20 +00:00
|
|
|
(defproject datomish "0.1.1-SNAPSHOT"
|
2016-07-05 19:05:49 +00:00
|
|
|
:description "A persistent, embedded knowledge base inspired by Datomic and DataScript."
|
|
|
|
:url "https://github.com/mozilla/datomish"
|
|
|
|
:license {:name "Mozilla Public License Version 2.0"
|
2016-07-06 01:55:55 +00:00
|
|
|
:url "https://github.com/mozilla/datomish/blob/master/LICENSE"}
|
2016-09-08 00:03:03 +00:00
|
|
|
:dependencies [[org.clojure/clojurescript "1.9.229"]
|
2016-07-11 23:48:40 +00:00
|
|
|
[org.clojure/clojure "1.8.0"]
|
|
|
|
[org.clojure/core.async "0.2.385"]
|
2016-07-06 23:53:08 +00:00
|
|
|
[datascript "0.15.1"]
|
2016-08-05 20:06:42 +00:00
|
|
|
[honeysql "0.8.0"]
|
Completely rewrite main transaction logic to be faster.
This is almost complete; it passes the test suite save for retracting
fulltext datoms correctly.
There's a lot to say about this approach, but I don't have time to give
too many details. The broad outline is as follows. We collect datoms
to add and retract in a tx_lookup table. Depending on flags ("search
value" sv and "search value type tag" svalue_type_tag) we "complete" the
tx_lookup table by joining matching datoms. This allows us to find
datoms that are present (and should not be added as part of the
transaction, or should be retracted as part of the transaction, or
should be replaced as part of the transaction. We complete the
tx_lookup (in place!) in two separate INSERTs to avoid a quadratic
two-table walk (explain the queries to observe that both INSERTs walk
the lookup table once and then use the datoms indexes to complete the
matching values).
We could simplify the code by using multiple lookup tables, both for the
two cases of search parameters (eav vs. ea) and for the incomplete and
completed rows. Right now we differentiate the former with NULL checks,
and the latter by incrementing the added0 column. It performs well
enough, so I haven't tried to understand the performance of separating
these things.
After the tx_lookup table is completed, we build the transaction from
it; and update the datoms materialized view table as well. Observe the
careful handling of the "search value" sv parameters to handle replacing
:db.cardinality/one datoms.
Finally, we read the processed transaction back to produce to the API.
This is strictly to match the Datomic API; we might make allow to skip
this, since many consumers will not want to stream this over the wire.
Rough timings show the transactor processing a single >50k datom
transaction in about 3.5s, of which less than 0.5s is spent in the
expensive joins. Further, repeating the processing of the same
transaction is only about 3.5s again! That's the worst possible for the
joins, since every single inserted datom will already be present in the
database, making the most expensive join match every row.
2016-08-09 06:30:45 +00:00
|
|
|
[com.taoensso/tufte "1.0.2"]
|
2016-07-11 23:48:40 +00:00
|
|
|
[jamesmacaulay/cljs-promises "0.1.0"]]
|
2016-07-06 01:13:29 +00:00
|
|
|
|
2016-09-21 20:47:20 +00:00
|
|
|
;; The browser will never require from the .JAR anyway.
|
2016-09-22 17:38:45 +00:00
|
|
|
:source-paths [
|
|
|
|
"src/common"
|
|
|
|
;; Can't be enabled by default: layers on top of cljsbuild!
|
|
|
|
;; Instead, add the :node profile:
|
|
|
|
;; lein with-profile node install
|
|
|
|
;"src/node"
|
|
|
|
]
|
2016-09-21 20:47:20 +00:00
|
|
|
|
2016-09-08 00:00:47 +00:00
|
|
|
:cljsbuild {:builds
|
|
|
|
{
|
|
|
|
:release-node
|
|
|
|
{
|
2016-09-22 22:56:20 +00:00
|
|
|
:source-paths ["src/common" "src/node"]
|
2016-09-08 00:00:47 +00:00
|
|
|
:assert false
|
|
|
|
:compiler
|
|
|
|
{
|
2016-09-22 22:56:20 +00:00
|
|
|
;; :externs specified in deps.cljs.
|
2016-09-08 00:00:47 +00:00
|
|
|
:elide-asserts true
|
|
|
|
:hashbang false
|
|
|
|
:language-in :ecmascript5
|
|
|
|
:language-out :ecmascript5
|
|
|
|
:optimizations :advanced
|
2016-09-22 22:56:20 +00:00
|
|
|
:output-dir "target/release-node"
|
|
|
|
:output-to "target/release-node/datomish.bare.js"
|
2016-09-08 00:00:47 +00:00
|
|
|
:output-wrapper false
|
|
|
|
:parallel-build true
|
2016-09-15 22:11:58 +00:00
|
|
|
:pretty-print true
|
|
|
|
:pseudo-names true
|
|
|
|
:static-fns true
|
2016-09-08 00:00:47 +00:00
|
|
|
:target :nodejs
|
|
|
|
}
|
|
|
|
:notify-command ["release-node/wrap_bare.sh"]}
|
|
|
|
|
|
|
|
:release-browser
|
|
|
|
;; Release builds for use in Firefox must:
|
|
|
|
;; * Use :optimizations > :none, so that a single file is generated
|
|
|
|
;; without a need to import Closure's own libs.
|
|
|
|
;; * Be wrapped, so that a CommonJS module is produced.
|
|
|
|
;; * Have a preload script that defines what `println` does.
|
|
|
|
;;
|
|
|
|
;; There's no point in generating a source map -- it'll be wrong
|
|
|
|
;; due to wrapping.
|
|
|
|
{
|
2016-09-22 17:38:45 +00:00
|
|
|
:source-paths ["src/common" "src/browser"]
|
2016-09-08 00:00:47 +00:00
|
|
|
:assert false
|
|
|
|
:compiler
|
|
|
|
{
|
|
|
|
:elide-asserts true
|
2016-09-22 22:56:20 +00:00
|
|
|
:externs ["src/browser/externs/datomish.js"]
|
2016-09-08 00:00:47 +00:00
|
|
|
:language-in :ecmascript5
|
|
|
|
:language-out :ecmascript5
|
|
|
|
:optimizations :advanced
|
2016-09-22 22:56:20 +00:00
|
|
|
:output-dir "target/release-browser"
|
|
|
|
:output-to "target/release-browser/datomish.bare.js"
|
2016-09-08 00:00:47 +00:00
|
|
|
:output-wrapper false
|
|
|
|
:parallel-build true
|
|
|
|
:preloads [datomish.preload]
|
|
|
|
:pretty-print true
|
|
|
|
:pseudo-names true
|
|
|
|
:static-fns true
|
|
|
|
}
|
|
|
|
:notify-command ["release-browser/wrap_bare.sh"]}
|
|
|
|
|
|
|
|
:test
|
|
|
|
{
|
2016-09-22 22:56:20 +00:00
|
|
|
:source-paths ["src/common" "src/node" "test"]
|
2016-09-08 00:00:47 +00:00
|
|
|
:compiler
|
|
|
|
{
|
|
|
|
:language-in :ecmascript5
|
|
|
|
:language-out :ecmascript5
|
|
|
|
:main datomish.test
|
|
|
|
:optimizations :none
|
|
|
|
:output-dir "target/test"
|
|
|
|
:output-to "target/test/datomish.js"
|
|
|
|
:parallel-build true
|
|
|
|
:source-map true
|
|
|
|
:target :nodejs
|
|
|
|
}}
|
|
|
|
}}
|
2016-07-06 01:13:29 +00:00
|
|
|
|
2016-09-22 17:38:45 +00:00
|
|
|
:profiles {:node {:source-paths ["src/common" "src/node"]}
|
|
|
|
:dev {:dependencies [[cljsbuild "1.1.3"]
|
2016-07-12 05:02:53 +00:00
|
|
|
[tempfile "0.2.0"]
|
2016-07-08 19:44:57 +00:00
|
|
|
[com.cemerick/piggieback "0.2.1"]
|
|
|
|
[org.clojure/tools.nrepl "0.2.10"]
|
2016-07-12 05:02:53 +00:00
|
|
|
[org.clojure/java.jdbc "0.6.2-alpha1"]
|
|
|
|
[org.xerial/sqlite-jdbc "3.8.11.2"]]
|
2016-08-19 16:20:50 +00:00
|
|
|
:jvm-opts ["-Xss4m"]
|
2016-07-06 01:13:29 +00:00
|
|
|
:repl-options {:nrepl-middleware [cemerick.piggieback/wrap-cljs-repl]}
|
2016-07-08 19:44:57 +00:00
|
|
|
:plugins [[lein-cljsbuild "1.1.3"]
|
2016-08-26 00:23:25 +00:00
|
|
|
[lein-doo "0.1.6"]
|
|
|
|
[venantius/ultra "0.4.1"]
|
|
|
|
[com.jakemccrary/lein-test-refresh "0.16.0"]]
|
2016-07-06 01:55:55 +00:00
|
|
|
}}
|
|
|
|
|
|
|
|
:doo {:build "test"}
|
2016-07-06 01:13:29 +00:00
|
|
|
|
2016-09-22 22:56:20 +00:00
|
|
|
:clean-targets ^{:protect false} ["target"]
|
2016-07-06 01:13:29 +00:00
|
|
|
)
|