2016-07-27 21:29:16 +00:00
|
|
|
;; This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
;; License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
;; file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
(ns datomish.db-test
|
|
|
|
#?(:cljs
|
|
|
|
(:require-macros
|
|
|
|
[datomish.pair-chan :refer [go-pair <?]]
|
|
|
|
[datomish.node-tempfile-macros :refer [with-tempfile]]
|
|
|
|
[cljs.core.async.macros :as a :refer [go]]))
|
|
|
|
(:require
|
2016-08-05 00:40:38 +00:00
|
|
|
[datomish.api :as d]
|
2016-07-27 21:29:16 +00:00
|
|
|
[datomish.util :as util #?(:cljs :refer-macros :clj :refer) [raise cond-let]]
|
|
|
|
[datomish.sqlite :as s]
|
2016-08-03 15:47:55 +00:00
|
|
|
[datomish.sqlite-schema]
|
2016-07-27 21:29:16 +00:00
|
|
|
[datomish.datom]
|
2016-08-05 00:40:38 +00:00
|
|
|
[datomish.db :as db]
|
2016-07-27 21:29:16 +00:00
|
|
|
#?@(:clj [[datomish.pair-chan :refer [go-pair <?]]
|
|
|
|
[tempfile.core :refer [tempfile with-tempfile]]
|
|
|
|
[datomish.test-macros :refer [deftest-async]]
|
|
|
|
[clojure.test :as t :refer [is are deftest testing]]
|
|
|
|
[clojure.core.async :refer [go <! >!]]])
|
|
|
|
#?@(:cljs [[datomish.pair-chan]
|
|
|
|
[datomish.test-macros :refer-macros [deftest-async]]
|
|
|
|
[datomish.node-tempfile :refer [tempfile]]
|
|
|
|
[cljs.test :as t :refer-macros [is are deftest testing async]]
|
|
|
|
[cljs.core.async :as a :refer [<! >!]]]))
|
|
|
|
#?(:clj
|
|
|
|
(:import [clojure.lang ExceptionInfo]))
|
|
|
|
#?(:clj
|
|
|
|
(:import [datascript.db DB])))
|
|
|
|
|
2016-07-28 21:47:43 +00:00
|
|
|
#?(:cljs
|
|
|
|
(def Throwable js/Error))
|
|
|
|
|
2016-08-05 00:40:38 +00:00
|
|
|
(defn- tempids [tx]
|
|
|
|
(into {} (map (juxt (comp :idx first) second) (:tempids tx))))
|
|
|
|
|
2016-08-03 15:47:55 +00:00
|
|
|
(defn- <datoms-after [db tx]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [entids (zipmap (vals (db/idents db)) (keys (db/idents db)))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(go-pair
|
|
|
|
(->>
|
2016-08-03 22:41:40 +00:00
|
|
|
(s/all-rows (:sqlite-connection db) ["SELECT e, a, v, tx FROM datoms WHERE tx > ?" tx])
|
2016-07-27 21:29:16 +00:00
|
|
|
(<?)
|
2016-08-03 15:47:55 +00:00
|
|
|
(mapv #(vector (:e %) (get entids (:a %) (str "fail" (:a %))) (:v %)))
|
2016-07-27 21:29:16 +00:00
|
|
|
(filter #(not (= :db/txInstant (second %))))
|
|
|
|
(set)))))
|
|
|
|
|
2016-08-03 15:47:55 +00:00
|
|
|
(defn- <datoms [db]
|
|
|
|
(<datoms-after db 0))
|
|
|
|
|
2016-07-28 22:30:46 +00:00
|
|
|
(defn- <shallow-entity [db eid]
|
2016-08-04 21:01:11 +00:00
|
|
|
;; TODO: make this actually be <entity. Handle :db.cardinality/many and :db/isComponent.
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [entids (zipmap (vals (db/idents db)) (keys (db/idents db)))]
|
2016-07-28 22:30:46 +00:00
|
|
|
(go-pair
|
|
|
|
(->>
|
|
|
|
(s/all-rows (:sqlite-connection db) ["SELECT a, v FROM datoms WHERE e = ?" eid])
|
|
|
|
(<?)
|
|
|
|
(mapv #(vector (entids (:a %)) (:v %)))
|
|
|
|
(reduce conj {})))))
|
|
|
|
|
2016-08-03 15:47:55 +00:00
|
|
|
(defn- <transactions-after [db tx]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [entids (zipmap (vals (db/idents db)) (keys (db/idents db)))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(go-pair
|
|
|
|
(->>
|
2016-08-03 22:41:40 +00:00
|
|
|
(s/all-rows (:sqlite-connection db) ["SELECT e, a, v, tx, added FROM transactions WHERE tx > ? ORDER BY tx ASC, e, a, v, added" tx])
|
2016-07-27 21:29:16 +00:00
|
|
|
(<?)
|
|
|
|
(mapv #(vector (:e %) (entids (:a %)) (:v %) (:tx %) (:added %)))))))
|
|
|
|
|
2016-08-03 15:47:55 +00:00
|
|
|
(defn- <transactions [db]
|
|
|
|
(<transactions-after db 0))
|
|
|
|
|
2016-08-03 22:42:04 +00:00
|
|
|
(defn- <fulltext-values [db]
|
|
|
|
(go-pair
|
|
|
|
(->>
|
|
|
|
(s/all-rows (:sqlite-connection db) ["SELECT rowid, text FROM fulltext_values"])
|
|
|
|
(<?)
|
|
|
|
(mapv #(vector (:rowid %) (:text %))))))
|
|
|
|
|
2016-08-03 15:47:55 +00:00
|
|
|
;; TODO: use reverse refs!
|
2016-07-28 00:07:05 +00:00
|
|
|
(def test-schema
|
2016-08-05 00:40:38 +00:00
|
|
|
[{:db/id (d/id-literal :test -1)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :x
|
|
|
|
:db/unique :db.unique/identity
|
|
|
|
:db/valueType :db.type/integer}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -1)}
|
|
|
|
{:db/id (d/id-literal :test -2)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :name
|
|
|
|
:db/unique :db.unique/identity
|
|
|
|
:db/valueType :db.type/string}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -2)}
|
|
|
|
{:db/id (d/id-literal :test -3)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :y
|
|
|
|
:db/cardinality :db.cardinality/many
|
|
|
|
:db/valueType :db.type/integer}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -3)}
|
|
|
|
{:db/id (d/id-literal :test -5)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :aka
|
|
|
|
:db/cardinality :db.cardinality/many
|
|
|
|
:db/valueType :db.type/string}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -5)}
|
|
|
|
{:db/id (d/id-literal :test -6)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :age
|
|
|
|
:db/valueType :db.type/integer}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -6)}
|
|
|
|
{:db/id (d/id-literal :test -7)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :email
|
|
|
|
:db/unique :db.unique/identity
|
|
|
|
:db/valueType :db.type/string}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -7)}
|
|
|
|
{:db/id (d/id-literal :test -8)
|
2016-08-03 15:47:55 +00:00
|
|
|
:db/ident :spouse
|
|
|
|
:db/unique :db.unique/value
|
|
|
|
:db/valueType :db.type/string}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -8)}
|
|
|
|
{:db/id (d/id-literal :test -9)
|
2016-08-04 21:01:11 +00:00
|
|
|
:db/ident :friends
|
|
|
|
:db/cardinality :db.cardinality/many
|
|
|
|
:db/valueType :db.type/ref}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :test -9)}
|
2016-08-03 15:47:55 +00:00
|
|
|
])
|
2016-07-28 00:07:05 +00:00
|
|
|
|
2016-07-27 21:29:16 +00:00
|
|
|
(deftest-async test-add-one
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{tx0 :tx} (<? (d/<transact! conn test-schema))]
|
|
|
|
(let [{:keys [tx txInstant]} (<? (d/<transact! conn [[:db/add 0 :name "valuex"]]))]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:41:40 +00:00
|
|
|
#{[0 :name "valuex"]}))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<transactions-after (d/db conn) tx0))
|
2016-08-03 22:41:40 +00:00
|
|
|
[[0 :name "valuex" tx 1] ;; TODO: true, not 1.
|
2016-08-04 03:24:02 +00:00
|
|
|
[tx :db/txInstant txInstant tx 1]]))))
|
2016-07-27 21:29:16 +00:00
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-27 21:29:16 +00:00
|
|
|
|
|
|
|
(deftest-async test-add-two
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{tx0 :tx} (<? (d/<transact! conn test-schema))
|
|
|
|
{tx1 :tx txInstant1 :txInstant} (<? (d/<transact! conn [[:db/add 1 :name "Ivan"]]))
|
|
|
|
{tx2 :tx txInstant2 :txInstant} (<? (d/<transact! conn [[:db/add 1 :name "Petr"]]))
|
|
|
|
{tx3 :tx txInstant3 :txInstant} (<? (d/<transact! conn [[:db/add 1 :aka "Tupen"]]))
|
|
|
|
{tx4 :tx txInstant4 :txInstant} (<? (d/<transact! conn [[:db/add 1 :aka "Devil"]]))]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-07-28 00:07:05 +00:00
|
|
|
#{[1 :name "Petr"]
|
|
|
|
[1 :aka "Tupen"]
|
|
|
|
[1 :aka "Devil"]}))
|
2016-07-27 21:29:16 +00:00
|
|
|
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<transactions-after (d/db conn) tx0))
|
2016-07-28 00:07:05 +00:00
|
|
|
[[1 :name "Ivan" tx1 1] ;; TODO: true, not 1.
|
2016-08-04 03:24:02 +00:00
|
|
|
[tx1 :db/txInstant txInstant1 tx1 1]
|
2016-07-28 00:07:05 +00:00
|
|
|
[1 :name "Ivan" tx2 0]
|
|
|
|
[1 :name "Petr" tx2 1]
|
2016-08-04 03:24:02 +00:00
|
|
|
[tx2 :db/txInstant txInstant2 tx2 1]
|
2016-07-28 00:07:05 +00:00
|
|
|
[1 :aka "Tupen" tx3 1]
|
2016-08-04 03:24:02 +00:00
|
|
|
[tx3 :db/txInstant txInstant3 tx3 1]
|
2016-07-28 00:07:05 +00:00
|
|
|
[1 :aka "Devil" tx4 1]
|
2016-08-04 03:24:02 +00:00
|
|
|
[tx4 :db/txInstant txInstant4 tx4 1]])))
|
2016-08-03 22:41:40 +00:00
|
|
|
|
2016-07-27 21:29:16 +00:00
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-27 21:29:16 +00:00
|
|
|
|
|
|
|
(deftest-async test-retract
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{tx0 :tx} (<? (d/<transact! conn test-schema))
|
|
|
|
{tx1 :tx txInstant1 :txInstant} (<? (d/<transact! conn [[:db/add 0 :x 123]]))
|
|
|
|
{tx2 :tx txInstant2 :txInstant} (<? (d/<transact! conn [[:db/retract 0 :x 123]]))]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-07-27 21:29:16 +00:00
|
|
|
#{}))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<transactions-after (d/db conn) tx0))
|
2016-08-04 03:24:02 +00:00
|
|
|
[[0 :x 123 tx1 1]
|
|
|
|
[tx1 :db/txInstant txInstant1 tx1 1]
|
|
|
|
[0 :x 123 tx2 0]
|
|
|
|
[tx2 :db/txInstant txInstant2 tx2 1]])))
|
2016-07-27 21:29:16 +00:00
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-27 21:29:16 +00:00
|
|
|
|
|
|
|
(deftest-async test-id-literal-1
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx0 (:tx (<? (d/<transact! conn test-schema)))
|
|
|
|
report (<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user -1) :x 0]
|
|
|
|
[:db/add (d/id-literal :db.part/user -1) :y 1]
|
|
|
|
[:db/add (d/id-literal :db.part/user -2) :y 2]
|
|
|
|
[:db/add (d/id-literal :db.part/user -2) :y 3]]))]
|
2016-07-27 21:29:16 +00:00
|
|
|
(is (= (keys (:tempids report)) ;; TODO: include values.
|
2016-08-05 00:40:38 +00:00
|
|
|
[(d/id-literal :db.part/user -1)
|
|
|
|
(d/id-literal :db.part/user -2)]))
|
2016-07-27 21:29:16 +00:00
|
|
|
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [eid1 (get-in report [:tempids (d/id-literal :db.part/user -1)])
|
|
|
|
eid2 (get-in report [:tempids (d/id-literal :db.part/user -2)])]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-07-27 21:29:16 +00:00
|
|
|
#{[eid1 :x 0]
|
|
|
|
[eid1 :y 1]
|
|
|
|
[eid2 :y 2]
|
|
|
|
[eid2 :y 3]}))))
|
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-27 21:30:01 +00:00
|
|
|
|
2016-07-28 22:30:46 +00:00
|
|
|
(deftest-async test-unique
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-28 22:30:46 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx0 (:tx (<? (d/<transact! conn test-schema)))]
|
2016-08-03 22:41:40 +00:00
|
|
|
(testing "Multiple :db/unique values in tx-data violate unique constraint, no tempid"
|
|
|
|
(is (thrown-with-msg?
|
|
|
|
ExceptionInfo #"unique constraint"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [[:db/add 1 :x 0]
|
|
|
|
[:db/add 2 :x 0]])))))
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
(testing "Multiple :db/unique values in tx-data violate unique constraint, tempid"
|
|
|
|
(is (thrown-with-msg?
|
|
|
|
ExceptionInfo #"unique constraint"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user -1) :spouse "Dana"]
|
|
|
|
[:db/add (d/id-literal :db.part/user -2) :spouse "Dana"]]))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
2016-07-28 04:15:57 +00:00
|
|
|
(deftest-async test-valueType-keyword
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-28 04:15:57 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx0 (:tx (<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1)
|
|
|
|
:db/ident :test/kw
|
|
|
|
:db/unique :db.unique/identity
|
|
|
|
:db/valueType :db.type/keyword}
|
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :db.part/user -1)}])))]
|
|
|
|
|
|
|
|
(let [report (<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user -1) :test/kw :test/kw1]]))
|
|
|
|
eid (get-in report [:tempids (d/id-literal :db.part/user -1)])]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:41:40 +00:00
|
|
|
#{[eid :test/kw ":test/kw1"]})) ;; Value is raw.
|
|
|
|
|
|
|
|
(testing "Adding the same value compares existing values correctly."
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [[:db/add eid :test/kw :test/kw1]]))
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:41:40 +00:00
|
|
|
#{[eid :test/kw ":test/kw1"]}))) ;; Value is raw.
|
|
|
|
|
|
|
|
(testing "Upserting retracts existing value correctly."
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [[:db/add eid :test/kw :test/kw2]]))
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:41:40 +00:00
|
|
|
#{[eid :test/kw ":test/kw2"]}))) ;; Value is raw.
|
|
|
|
|
|
|
|
(testing "Retracting compares values correctly."
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [[:db/retract eid :test/kw :test/kw2]]))
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:41:40 +00:00
|
|
|
#{})))))
|
2016-07-28 04:15:57 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
|
|
|
(deftest-async test-vector-upsert
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-28 22:30:46 +00:00
|
|
|
(try
|
2016-08-03 15:47:55 +00:00
|
|
|
;; Not having DB-as-value really hurts us here. This test only works because all upserts
|
|
|
|
;; succeed on top of each other, so we never need to reset the underlying store.
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn test-schema))
|
|
|
|
(let [tx0 (:tx (<? (d/<transact! conn [{:db/id 101 :name "Ivan" :email "@1"}
|
|
|
|
{:db/id 102 :name "Petr" :email "@2"}])))]
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
(testing "upsert with tempid"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [report (<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user -1) :name "Ivan"]
|
|
|
|
[:db/add (d/id-literal :db.part/user -1) :age 12]]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 101))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Ivan" :age 12 :email "@1"}))
|
|
|
|
(is (= (tempids report)
|
|
|
|
{-1 101}))))
|
|
|
|
|
|
|
|
(testing "upsert with tempid, order does not matter"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [report (<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user -1) :age 13]
|
|
|
|
[:db/add (d/id-literal :db.part/user -1) :name "Petr"]]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 102))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Petr" :age 13 :email "@2"}))
|
|
|
|
(is (= (tempids report)
|
|
|
|
{-1 102}))))
|
|
|
|
|
|
|
|
(testing "Conflicting upserts fail"
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (thrown-with-msg? Throwable #"Conflicting upsert"
|
|
|
|
(<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user -1) :name "Ivan"]
|
|
|
|
[:db/add (d/id-literal :db.part/user -1) :age 35]
|
|
|
|
[:db/add (d/id-literal :db.part/user -1) :name "Petr"]
|
|
|
|
[:db/add (d/id-literal :db.part/user -1) :age 36]]))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
|
|
|
(deftest-async test-map-upsert
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-28 22:30:46 +00:00
|
|
|
(try
|
2016-08-03 15:47:55 +00:00
|
|
|
;; Not having DB-as-value really hurts us here. This test only works because all upserts
|
|
|
|
;; succeed on top of each other, so we never need to reset the underlying store.
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn test-schema))
|
|
|
|
(let [tx0 (:tx (<? (d/<transact! conn [{:db/id 101 :name "Ivan" :email "@1"}
|
|
|
|
{:db/id 102 :name "Petr" :email "@2"}])))]
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
(testing "upsert with tempid"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx (<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :age 35}]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 101))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Ivan" :email "@1" :age 35}))
|
|
|
|
(is (= (tempids tx)
|
|
|
|
{-1 101}))))
|
|
|
|
|
|
|
|
(testing "upsert by 2 attrs with tempid"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx (<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :email "@1" :age 35}]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 101))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Ivan" :email "@1" :age 35}))
|
|
|
|
(is (= (tempids tx)
|
|
|
|
{-1 101}))))
|
|
|
|
|
|
|
|
(testing "upsert with existing id"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx (<? (d/<transact! conn [{:db/id 101 :name "Ivan" :age 36}]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 101))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Ivan" :email "@1" :age 36}))
|
|
|
|
(is (= (tempids tx)
|
|
|
|
{}))))
|
|
|
|
|
|
|
|
(testing "upsert by 2 attrs with existing id"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [tx (<? (d/<transact! conn [{:db/id 101 :name "Ivan" :email "@1" :age 37}]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 101))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Ivan" :email "@1" :age 37}))
|
|
|
|
(is (= (tempids tx)
|
|
|
|
{}))))
|
|
|
|
|
|
|
|
(testing "upsert to two entities, resolve to same tempid, fails due to overlapping writes"
|
|
|
|
(is (thrown-with-msg? Throwable #"cardinality constraint"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :age 35}
|
|
|
|
{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :age 36}])))))
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
(testing "upsert to two entities, two tempids, fails due to overlapping writes"
|
|
|
|
(is (thrown-with-msg? Throwable #"cardinality constraint"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :age 35}
|
|
|
|
{:db/id (d/id-literal :db.part/user -2) :name "Ivan" :age 36}]))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
|
|
|
(deftest-async test-map-upsert-conflicts
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-28 22:30:46 +00:00
|
|
|
(try
|
2016-08-03 15:47:55 +00:00
|
|
|
;; Not having DB-as-value really hurts us here. This test only works because all upserts
|
|
|
|
;; fail until the final one, so we never need to reset the underlying store.
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn test-schema))
|
|
|
|
(let [tx0 (:tx (<? (d/<transact! conn [{:db/id 101 :name "Ivan" :email "@1"}
|
|
|
|
{:db/id 102 :name "Petr" :email "@2"}])))]
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
;; TODO: improve error message to refer to upsert inputs.
|
|
|
|
(testing "upsert conficts with existing id"
|
|
|
|
(is (thrown-with-msg? Throwable #"unique constraint"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 102 :name "Ivan" :age 36}])))))
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
;; TODO: improve error message to refer to upsert inputs.
|
|
|
|
(testing "upsert conficts with non-existing id"
|
|
|
|
(is (thrown-with-msg? Throwable #"unique constraint"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 103 :name "Ivan" :age 36}])))))
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
;; TODO: improve error message to refer to upsert inputs.
|
|
|
|
(testing "upsert by 2 conflicting fields"
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (thrown-with-msg? Throwable #"Conflicting upsert"
|
|
|
|
(<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :email "@2" :age 35}])))))
|
2016-08-03 22:41:40 +00:00
|
|
|
|
|
|
|
(testing "upsert by non-existing value resolves as update"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [report (<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user -1) :name "Ivan" :email "@3" :age 35}]))]
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 101))
|
2016-08-03 22:41:40 +00:00
|
|
|
{:name "Ivan" :email "@3" :age 35}))
|
|
|
|
(is (= (tempids report)
|
|
|
|
{-1 101})))))
|
2016-07-28 22:30:46 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
(deftest-async test-add-ident
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-29 23:10:07 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [report (<? (d/<transact! conn [[:db/add (d/id-literal :db.part/db -1) :db/ident :test/ident]]))
|
2016-07-29 23:10:07 +00:00
|
|
|
db-after (:db-after report)
|
2016-08-04 03:24:02 +00:00
|
|
|
tx (:tx db-after)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (:test/ident (db/idents db-after)) (get-in report [:tempids (d/id-literal :db.part/db -1)]))))
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
;; TODO: This should fail, but doesn't, due to stringification of :test/ident.
|
|
|
|
;; (is (thrown-with-msg?
|
|
|
|
;; ExceptionInfo #"Retracting a :db/ident is not yet supported, got "
|
2016-08-05 00:40:38 +00:00
|
|
|
;; (<? (d/<transact! conn [[:db/retract 44 :db/ident :test/ident]]))))
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
;; ;; Renaming looks like retraction and then assertion.
|
|
|
|
;; (is (thrown-with-msg?
|
|
|
|
;; ExceptionInfo #"Retracting a :db/ident is not yet supported, got"
|
2016-08-05 00:40:38 +00:00
|
|
|
;; (<? (d/<transact! conn [[:db/add 44 :db/ident :other-name]]))))
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
;; (is (thrown-with-msg?
|
|
|
|
;; ExceptionInfo #"Re-asserting a :db/ident is not yet supported, got"
|
2016-08-05 00:40:38 +00:00
|
|
|
;; (<? (d/<transact! conn [[:db/add 55 :db/ident :test/ident]]))))
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
(deftest-async test-add-schema
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))]
|
2016-07-29 23:10:07 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [es [[:db/add :db.part/db :db.install/attribute (d/id-literal :db.part/db -1)]
|
|
|
|
{:db/id (d/id-literal :db.part/db -1)
|
2016-07-29 23:10:07 +00:00
|
|
|
:db/ident :test/attr
|
|
|
|
:db/valueType :db.type/string
|
|
|
|
:db/cardinality :db.cardinality/one}]
|
2016-08-05 00:40:38 +00:00
|
|
|
report (<? (d/<transact! conn es))
|
2016-07-29 23:10:07 +00:00
|
|
|
db-after (:db-after report)
|
2016-08-04 03:24:02 +00:00
|
|
|
tx (:tx db-after)]
|
2016-07-29 23:10:07 +00:00
|
|
|
|
|
|
|
(testing "New ident is allocated"
|
|
|
|
(is (some? (get-in db-after [:idents :test/attr]))))
|
|
|
|
|
|
|
|
(testing "Schema is modified"
|
|
|
|
(is (= (get-in db-after [:symbolic-schema :test/attr])
|
2016-08-03 15:47:55 +00:00
|
|
|
{:db/valueType :db.type/string,
|
2016-07-29 23:10:07 +00:00
|
|
|
:db/cardinality :db.cardinality/one})))
|
|
|
|
|
|
|
|
(testing "Schema is used in subsequent transaction"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 100 :test/attr "value 1"}]))
|
|
|
|
(<? (d/<transact! conn [{:db/id 100 :test/attr "value 2"}]))
|
|
|
|
(is (= (<? (<shallow-entity (d/db conn) 100))
|
2016-07-29 23:10:07 +00:00
|
|
|
{:test/attr "value 2"}))))
|
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-08-03 22:42:04 +00:00
|
|
|
|
|
|
|
(deftest-async test-fulltext
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))
|
|
|
|
schema [{:db/id (d/id-literal :db.part/db -1)
|
2016-08-03 22:42:04 +00:00
|
|
|
:db/ident :test/fulltext
|
|
|
|
:db/valueType :db.type/string
|
|
|
|
:db/fulltext true
|
|
|
|
:db/unique :db.unique/identity}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :db.part/db -1)}
|
|
|
|
{:db/id (d/id-literal :db.part/db -2)
|
2016-08-03 22:42:04 +00:00
|
|
|
:db/ident :test/other
|
|
|
|
:db/valueType :db.type/string
|
|
|
|
:db/fulltext true
|
|
|
|
:db/cardinality :db.cardinality/one}
|
2016-08-05 00:40:38 +00:00
|
|
|
{:db/id :db.part/db :db.install/attribute (d/id-literal :db.part/db -2)}
|
2016-08-03 22:42:04 +00:00
|
|
|
]
|
2016-08-05 00:40:38 +00:00
|
|
|
tx0 (:tx (<? (d/<transact! conn schema)))]
|
2016-08-03 22:42:04 +00:00
|
|
|
(try
|
|
|
|
(testing "Can add fulltext indexed datoms"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [r (<? (d/<transact! conn [[:db/add 101 :test/fulltext "test this"]]))]
|
|
|
|
(is (= (<? (<fulltext-values (d/db conn)))
|
2016-08-03 22:42:04 +00:00
|
|
|
[[1 "test this"]]))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:42:04 +00:00
|
|
|
#{[101 :test/fulltext 1]})) ;; Values are raw; 1 is the rowid into fulltext_values.
|
|
|
|
))
|
|
|
|
|
|
|
|
(testing "Can replace fulltext indexed datoms"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [r (<? (d/<transact! conn [[:db/add 101 :test/fulltext "alternate thing"]]))]
|
|
|
|
(is (= (<? (<fulltext-values (d/db conn)))
|
2016-08-03 22:42:04 +00:00
|
|
|
[[1 "test this"]
|
|
|
|
[2 "alternate thing"]]))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:42:04 +00:00
|
|
|
#{[101 :test/fulltext 2]})) ;; Values are raw; 2 is the rowid into fulltext_values.
|
|
|
|
))
|
|
|
|
|
|
|
|
(testing "Can upsert keyed by fulltext indexed datoms"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [r (<? (d/<transact! conn [{:db/id (d/id-literal :db.part/user) :test/fulltext "alternate thing" :test/other "other"}]))]
|
|
|
|
(is (= (<? (<fulltext-values (d/db conn)))
|
2016-08-03 22:42:04 +00:00
|
|
|
[[1 "test this"]
|
|
|
|
[2 "alternate thing"]
|
|
|
|
[3 "other"]]))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:42:04 +00:00
|
|
|
#{[101 :test/fulltext 2] ;; Values are raw; 2, 3 are the rowids into fulltext_values.
|
|
|
|
[101 :test/other 3]}))
|
|
|
|
))
|
|
|
|
|
|
|
|
(testing "Can re-use fulltext indexed datoms"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [r (<? (d/<transact! conn [[:db/add 102 :test/other "test this"]]))]
|
|
|
|
(is (= (<? (<fulltext-values (d/db conn)))
|
2016-08-03 22:42:04 +00:00
|
|
|
[[1 "test this"]
|
|
|
|
[2 "alternate thing"]
|
|
|
|
[3 "other"]]))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:42:04 +00:00
|
|
|
#{[101 :test/fulltext 2]
|
|
|
|
[101 :test/other 3]
|
|
|
|
[102 :test/other 1]})) ;; Values are raw; 1, 2, 3 are the rowids into fulltext_values.
|
|
|
|
))
|
|
|
|
|
|
|
|
(testing "Can retract fulltext indexed datoms"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [r (<? (d/<transact! conn [[:db/retract 101 :test/fulltext "alternate thing"]]))]
|
|
|
|
(is (= (<? (<fulltext-values (d/db conn)))
|
2016-08-03 22:42:04 +00:00
|
|
|
[[1 "test this"]
|
|
|
|
[2 "alternate thing"]
|
|
|
|
[3 "other"]]))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-03 22:42:04 +00:00
|
|
|
#{[101 :test/other 3]
|
|
|
|
[102 :test/other 1]})) ;; Values are raw; 1, 3 are the rowids into fulltext_values.
|
|
|
|
))
|
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-08-04 03:24:02 +00:00
|
|
|
|
|
|
|
(deftest-async test-txInstant
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))
|
|
|
|
{tx0 :tx} (<? (d/<transact! conn test-schema))]
|
2016-08-04 03:24:02 +00:00
|
|
|
(try
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{txa :tx txInstantA :txInstant} (<? (d/<transact! conn []))]
|
2016-08-04 03:24:02 +00:00
|
|
|
(testing ":db/txInstant is set by default"
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<transactions-after (d/db conn) tx0))
|
2016-08-04 03:24:02 +00:00
|
|
|
[[txa :db/txInstant txInstantA txa 1]])))
|
|
|
|
|
|
|
|
;; TODO: range check txInstant values against DB clock.
|
|
|
|
(testing ":db/txInstant can be set explicitly"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{txb :tx txInstantB :txInstant} (<? (d/<transact! conn [[:db/add :db/tx :db/txInstant (+ txInstantA 1)]]))]
|
2016-08-04 03:24:02 +00:00
|
|
|
(is (= txInstantB (+ txInstantA 1)))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<transactions-after (d/db conn) txa))
|
2016-08-04 03:24:02 +00:00
|
|
|
[[txb :db/txInstant txInstantB txb 1]]))
|
|
|
|
|
|
|
|
(testing ":db/txInstant can be set explicitly, with additional datoms"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{txc :tx txInstantC :txInstant} (<? (d/<transact! conn [[:db/add :db/tx :db/txInstant (+ txInstantB 2)]
|
|
|
|
[:db/add :db/tx :x 123]]))]
|
2016-08-04 03:24:02 +00:00
|
|
|
(is (= txInstantC (+ txInstantB 2)))
|
2016-08-05 00:40:38 +00:00
|
|
|
(is (= (<? (<transactions-after (d/db conn) txb))
|
2016-08-04 03:24:02 +00:00
|
|
|
[[txc :db/txInstant txInstantC txc 1]
|
|
|
|
[txc :x 123 txc 1]]))
|
|
|
|
|
|
|
|
(testing "additional datoms can be added, without :db/txInstant explicitly"
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{txd :tx txInstantD :txInstant} (<? (d/<transact! conn [[:db/add :db/tx :x 456]]))]
|
|
|
|
(is (= (<? (<transactions-after (d/db conn) txc))
|
2016-08-04 03:24:02 +00:00
|
|
|
[[txd :db/txInstant txInstantD txd 1]
|
|
|
|
[txd :x 456 txd 1]])))))))))
|
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-08-04 17:04:15 +00:00
|
|
|
|
|
|
|
(deftest-async test-no-tx
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))
|
|
|
|
{tx0 :tx} (<? (d/<transact! conn test-schema))]
|
2016-08-04 17:04:15 +00:00
|
|
|
(try
|
|
|
|
(testing "Cannot specificy an explicit tx"
|
|
|
|
(is (thrown-with-msg?
|
|
|
|
ExceptionInfo #"Bad entity: too long"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [[:db/add (d/id-literal :db.part/user) :x 0 10101]])))))
|
2016-08-04 17:04:15 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-08-04 21:01:11 +00:00
|
|
|
|
|
|
|
(deftest-async test-explode-sequences
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))
|
|
|
|
{tx0 :tx} (<? (d/<transact! conn test-schema))]
|
2016-08-04 21:01:11 +00:00
|
|
|
(try
|
|
|
|
(testing ":db.cardinality/many sequences are accepted"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 101 :aka ["first" "second"]}]))
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-04 21:01:11 +00:00
|
|
|
#{[101 :aka "first"]
|
|
|
|
[101 :aka "second"]})))
|
|
|
|
|
|
|
|
(testing ":db.cardinality/many sequences are recursively applied, allowing unexpected sequence nesting"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 102 :aka [[["first"]] ["second"]]}]))
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-04 21:01:11 +00:00
|
|
|
#{[101 :aka "first"]
|
|
|
|
[101 :aka "second"]
|
|
|
|
[102 :aka "first"]
|
|
|
|
[102 :aka "second"]})))
|
|
|
|
|
|
|
|
(testing ":db.cardinality/one sequences fail"
|
|
|
|
(is (thrown-with-msg?
|
|
|
|
ExceptionInfo #"Sequential values"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 101 :email ["@1" "@2"]}])))))
|
2016-08-04 21:01:11 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-08-04 21:01:11 +00:00
|
|
|
|
|
|
|
(deftest-async test-explode-maps
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))
|
|
|
|
{tx0 :tx} (<? (d/<transact! conn test-schema))]
|
2016-08-04 21:01:11 +00:00
|
|
|
(try
|
|
|
|
(testing "nested maps are accepted"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 101 :friends {:name "Petr"}}]))
|
2016-08-04 21:01:11 +00:00
|
|
|
;; TODO: this works only because we have a single friend.
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{petr :friends} (<? (<shallow-entity (d/db conn) 101))]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-04 21:01:11 +00:00
|
|
|
#{[101 :friends petr]
|
|
|
|
[petr :name "Petr"]}))))
|
|
|
|
|
|
|
|
(testing "recursively nested maps are accepted"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 102 :friends {:name "Ivan" :friends {:name "Petr"}}}]))
|
2016-08-04 21:01:11 +00:00
|
|
|
;; This would be much easier with `entity` and lookup refs.
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [{ivan :friends} (<? (<shallow-entity (d/db conn) 102))
|
|
|
|
{petr :friends} (<? (<shallow-entity (d/db conn) ivan))]
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-04 21:01:11 +00:00
|
|
|
#{[101 :friends petr]
|
|
|
|
[petr :name "Petr"]
|
|
|
|
[102 :friends ivan]
|
|
|
|
[ivan :name "Ivan"]
|
|
|
|
[ivan :friends petr]}))))
|
|
|
|
|
|
|
|
(testing "nested maps without :db.type/ref fail"
|
|
|
|
(is (thrown-with-msg?
|
|
|
|
ExceptionInfo #"\{:db/valueType :db.type/ref\}"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 101 :aka {:name "Petr"}}])))))
|
2016-08-04 21:01:11 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|
2016-08-04 21:01:11 +00:00
|
|
|
|
|
|
|
(deftest-async test-explode-reverse-refs
|
|
|
|
(with-tempfile [t (tempfile)]
|
2016-08-05 00:40:38 +00:00
|
|
|
(let [conn (<? (d/<connect t))
|
|
|
|
{tx0 :tx} (<? (d/<transact! conn test-schema))]
|
2016-08-04 21:01:11 +00:00
|
|
|
(try
|
|
|
|
(testing "reverse refs are accepted"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 101 :name "Igor"}]))
|
|
|
|
(<? (d/<transact! conn [{:db/id 102 :name "Oleg" :_friends 101}]))
|
|
|
|
(is (= (<? (<datoms-after (d/db conn) tx0))
|
2016-08-04 21:01:11 +00:00
|
|
|
#{[101 :name "Igor"]
|
|
|
|
[102 :name "Oleg"]
|
|
|
|
[101 :friends 102]})))
|
|
|
|
|
|
|
|
(testing "reverse refs without :db.type/ref fail"
|
|
|
|
(is (thrown-with-msg?
|
|
|
|
ExceptionInfo #"\{:db/valueType :db.type/ref\}"
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<transact! conn [{:db/id 101 :_aka 102}])))))
|
2016-08-04 21:01:11 +00:00
|
|
|
|
|
|
|
(finally
|
2016-08-05 00:40:38 +00:00
|
|
|
(<? (d/<close conn)))))))
|