datahike-browser-tests 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (324) hide show
  1. package/.circleci/config.yml +405 -0
  2. package/.circleci/scripts/gen_ci.clj +194 -0
  3. package/.cirrus.yml +60 -0
  4. package/.clj-kondo/babashka/sci/config.edn +1 -0
  5. package/.clj-kondo/babashka/sci/sci/core.clj +9 -0
  6. package/.clj-kondo/config.edn +95 -0
  7. package/.dir-locals.el +2 -0
  8. package/.github/FUNDING.yml +3 -0
  9. package/.github/ISSUE_TEMPLATE/1-bug-report.yml +68 -0
  10. package/.github/ISSUE_TEMPLATE/2-feature-request.yml +28 -0
  11. package/.github/ISSUE_TEMPLATE/config.yml +6 -0
  12. package/.github/pull_request_template.md +24 -0
  13. package/.github/workflows/native-image.yml +84 -0
  14. package/LICENSE +203 -0
  15. package/README.md +273 -0
  16. package/bb/deps.edn +9 -0
  17. package/bb/resources/github-fingerprints +3 -0
  18. package/bb/resources/native-image-tests/run-bb-pod-tests.clj +162 -0
  19. package/bb/resources/native-image-tests/run-libdatahike-tests +12 -0
  20. package/bb/resources/native-image-tests/run-native-image-tests +74 -0
  21. package/bb/resources/native-image-tests/run-python-tests +22 -0
  22. package/bb/resources/native-image-tests/testconfig.attr-refs.edn +6 -0
  23. package/bb/resources/native-image-tests/testconfig.edn +5 -0
  24. package/bb/resources/template/.settings/org.eclipse.jdt.apt.core.prefs +2 -0
  25. package/bb/resources/template/.settings/org.eclipse.jdt.core.prefs +9 -0
  26. package/bb/resources/template/.settings/org.eclipse.m2e.core.prefs +4 -0
  27. package/bb/resources/template/pom.xml +22 -0
  28. package/bb/src/tools/build.clj +132 -0
  29. package/bb/src/tools/clj_kondo.clj +32 -0
  30. package/bb/src/tools/deploy.clj +26 -0
  31. package/bb/src/tools/examples.clj +19 -0
  32. package/bb/src/tools/npm.clj +100 -0
  33. package/bb/src/tools/python.clj +14 -0
  34. package/bb/src/tools/release.clj +94 -0
  35. package/bb/src/tools/test.clj +148 -0
  36. package/bb/src/tools/version.clj +47 -0
  37. package/bb.edn +269 -0
  38. package/benchmark/src/benchmark/cli.clj +195 -0
  39. package/benchmark/src/benchmark/compare.clj +157 -0
  40. package/benchmark/src/benchmark/config.clj +316 -0
  41. package/benchmark/src/benchmark/measure.clj +187 -0
  42. package/benchmark/src/benchmark/store.clj +190 -0
  43. package/benchmark/test/benchmark/measure_test.clj +156 -0
  44. package/build.clj +30 -0
  45. package/config.edn +49 -0
  46. package/deps.edn +138 -0
  47. package/dev/sandbox.clj +82 -0
  48. package/dev/sandbox.cljs +127 -0
  49. package/dev/sandbox_benchmarks.clj +27 -0
  50. package/dev/sandbox_client.clj +87 -0
  51. package/dev/sandbox_transact_bench.clj +109 -0
  52. package/dev/user.clj +79 -0
  53. package/doc/README.md +96 -0
  54. package/doc/adl/README.md +6 -0
  55. package/doc/adl/adr-000-adr.org +28 -0
  56. package/doc/adl/adr-001-attribute-references.org +15 -0
  57. package/doc/adl/adr-002-build-tooling.org +54 -0
  58. package/doc/adl/adr-003-db-meta-data.md +52 -0
  59. package/doc/adl/adr-004-github-flow.md +40 -0
  60. package/doc/adl/adr-XYZ-template.md +30 -0
  61. package/doc/adl/index.org +3 -0
  62. package/doc/assets/datahike-logo.svg +3 -0
  63. package/doc/assets/datahiking-invoice.org +85 -0
  64. package/doc/assets/hhtree2.png +0 -0
  65. package/doc/assets/network_topology.svg +624 -0
  66. package/doc/assets/perf.png +0 -0
  67. package/doc/assets/schema_mindmap.mm +132 -0
  68. package/doc/assets/schema_mindmap.svg +970 -0
  69. package/doc/assets/temporal_index.mm +74 -0
  70. package/doc/backend-development.md +78 -0
  71. package/doc/bb-pod.md +89 -0
  72. package/doc/benchmarking.md +360 -0
  73. package/doc/bindings/edn-conversion.md +383 -0
  74. package/doc/cli.md +162 -0
  75. package/doc/cljdoc.edn +27 -0
  76. package/doc/cljs-support.md +133 -0
  77. package/doc/config.md +406 -0
  78. package/doc/contributing.md +114 -0
  79. package/doc/datalog-vs-sql.md +210 -0
  80. package/doc/datomic_differences.md +109 -0
  81. package/doc/development/pull-api-ns.md +186 -0
  82. package/doc/development/pull-frame-state-diagram.jpg +0 -0
  83. package/doc/distributed.md +566 -0
  84. package/doc/entity_spec.md +92 -0
  85. package/doc/gc.md +273 -0
  86. package/doc/java-api.md +808 -0
  87. package/doc/javascript-api.md +421 -0
  88. package/doc/libdatahike.md +86 -0
  89. package/doc/logging_and_error_handling.md +43 -0
  90. package/doc/norms.md +66 -0
  91. package/doc/schema-migration.md +85 -0
  92. package/doc/schema.md +287 -0
  93. package/doc/storage-backends.md +363 -0
  94. package/doc/store-id-refactoring.md +596 -0
  95. package/doc/time_variance.md +325 -0
  96. package/doc/unstructured.md +167 -0
  97. package/doc/versioning.md +261 -0
  98. package/examples/basic/README.md +19 -0
  99. package/examples/basic/deps.edn +6 -0
  100. package/examples/basic/docker-compose.yml +13 -0
  101. package/examples/basic/src/examples/core.clj +60 -0
  102. package/examples/basic/src/examples/schema.clj +155 -0
  103. package/examples/basic/src/examples/store.clj +60 -0
  104. package/examples/basic/src/examples/time_travel.clj +185 -0
  105. package/examples/java/.settings/org.eclipse.core.resources.prefs +3 -0
  106. package/examples/java/.settings/org.eclipse.jdt.apt.core.prefs +2 -0
  107. package/examples/java/.settings/org.eclipse.jdt.core.prefs +9 -0
  108. package/examples/java/.settings/org.eclipse.m2e.core.prefs +4 -0
  109. package/examples/java/README.md +162 -0
  110. package/examples/java/pom.xml +62 -0
  111. package/examples/java/src/main/java/examples/QuickStart.java +115 -0
  112. package/examples/java/src/main/java/examples/SchemaExample.java +148 -0
  113. package/examples/java/src/main/java/examples/TimeTravelExample.java +121 -0
  114. package/flake.lock +27 -0
  115. package/flake.nix +27 -0
  116. package/http-server/datahike/http/middleware.clj +75 -0
  117. package/http-server/datahike/http/server.clj +269 -0
  118. package/java/src/datahike/java/Database.java +274 -0
  119. package/java/src/datahike/java/Datahike.java +281 -0
  120. package/java/src/datahike/java/DatahikeGeneratedTest.java +349 -0
  121. package/java/src/datahike/java/DatahikeTest.java +370 -0
  122. package/java/src/datahike/java/EDN.java +170 -0
  123. package/java/src/datahike/java/IEntity.java +11 -0
  124. package/java/src/datahike/java/Keywords.java +161 -0
  125. package/java/src/datahike/java/SchemaFlexibility.java +52 -0
  126. package/java/src/datahike/java/Util.java +219 -0
  127. package/karma.conf.js +19 -0
  128. package/libdatahike/compile-cpp +7 -0
  129. package/libdatahike/src/datahike/impl/LibDatahikeBase.java +203 -0
  130. package/libdatahike/src/datahike/impl/libdatahike.clj +59 -0
  131. package/libdatahike/src/test_cpp.cpp +61 -0
  132. package/npm-package/PUBLISHING.md +140 -0
  133. package/npm-package/README.md +226 -0
  134. package/npm-package/package.template.json +34 -0
  135. package/npm-package/test-isomorphic.ts +281 -0
  136. package/npm-package/test.js +557 -0
  137. package/npm-package/typescript-test.ts +70 -0
  138. package/package.json +16 -0
  139. package/pydatahike/README.md +569 -0
  140. package/pydatahike/pyproject.toml +91 -0
  141. package/pydatahike/setup.py +42 -0
  142. package/pydatahike/src/datahike/__init__.py +134 -0
  143. package/pydatahike/src/datahike/_native.py +250 -0
  144. package/pydatahike/src/datahike/_version.py +2 -0
  145. package/pydatahike/src/datahike/database.py +722 -0
  146. package/pydatahike/src/datahike/edn.py +311 -0
  147. package/pydatahike/src/datahike/py.typed +0 -0
  148. package/pydatahike/tests/conftest.py +17 -0
  149. package/pydatahike/tests/test_basic.py +170 -0
  150. package/pydatahike/tests/test_database.py +51 -0
  151. package/pydatahike/tests/test_edn_conversion.py +299 -0
  152. package/pydatahike/tests/test_query.py +99 -0
  153. package/pydatahike/tests/test_schema.py +55 -0
  154. package/resources/clj-kondo.exports/io.replikativ/datahike/config.edn +5 -0
  155. package/resources/example_server.edn +4 -0
  156. package/shadow-cljs.edn +56 -0
  157. package/src/data_readers.clj +7 -0
  158. package/src/datahike/api/impl.cljc +176 -0
  159. package/src/datahike/api/specification.cljc +633 -0
  160. package/src/datahike/api/types.cljc +261 -0
  161. package/src/datahike/api.cljc +41 -0
  162. package/src/datahike/array.cljc +99 -0
  163. package/src/datahike/cli.clj +166 -0
  164. package/src/datahike/cljs.cljs +6 -0
  165. package/src/datahike/codegen/cli.clj +406 -0
  166. package/src/datahike/codegen/clj_kondo.clj +291 -0
  167. package/src/datahike/codegen/java.clj +403 -0
  168. package/src/datahike/codegen/naming.cljc +33 -0
  169. package/src/datahike/codegen/native.clj +559 -0
  170. package/src/datahike/codegen/pod.clj +488 -0
  171. package/src/datahike/codegen/python.clj +838 -0
  172. package/src/datahike/codegen/report.clj +55 -0
  173. package/src/datahike/codegen/typescript.clj +262 -0
  174. package/src/datahike/codegen/validation.clj +145 -0
  175. package/src/datahike/config.cljc +294 -0
  176. package/src/datahike/connections.cljc +16 -0
  177. package/src/datahike/connector.cljc +265 -0
  178. package/src/datahike/constants.cljc +142 -0
  179. package/src/datahike/core.cljc +297 -0
  180. package/src/datahike/datom.cljc +459 -0
  181. package/src/datahike/db/interface.cljc +119 -0
  182. package/src/datahike/db/search.cljc +305 -0
  183. package/src/datahike/db/transaction.cljc +937 -0
  184. package/src/datahike/db/utils.cljc +338 -0
  185. package/src/datahike/db.cljc +956 -0
  186. package/src/datahike/experimental/unstructured.cljc +126 -0
  187. package/src/datahike/experimental/versioning.cljc +172 -0
  188. package/src/datahike/externs.js +31 -0
  189. package/src/datahike/gc.cljc +69 -0
  190. package/src/datahike/http/client.clj +188 -0
  191. package/src/datahike/http/writer.clj +79 -0
  192. package/src/datahike/impl/entity.cljc +218 -0
  193. package/src/datahike/index/interface.cljc +93 -0
  194. package/src/datahike/index/persistent_set.cljc +469 -0
  195. package/src/datahike/index/utils.cljc +44 -0
  196. package/src/datahike/index.cljc +32 -0
  197. package/src/datahike/js/api.cljs +172 -0
  198. package/src/datahike/js/api_macros.clj +22 -0
  199. package/src/datahike/js.cljs +163 -0
  200. package/src/datahike/json.cljc +209 -0
  201. package/src/datahike/lru.cljc +146 -0
  202. package/src/datahike/migrate.clj +39 -0
  203. package/src/datahike/norm/norm.clj +245 -0
  204. package/src/datahike/online_gc.cljc +252 -0
  205. package/src/datahike/pod.clj +155 -0
  206. package/src/datahike/pull_api.cljc +325 -0
  207. package/src/datahike/query.cljc +1945 -0
  208. package/src/datahike/query_stats.cljc +88 -0
  209. package/src/datahike/readers.cljc +62 -0
  210. package/src/datahike/remote.cljc +218 -0
  211. package/src/datahike/schema.cljc +228 -0
  212. package/src/datahike/schema_cache.cljc +42 -0
  213. package/src/datahike/spec.cljc +101 -0
  214. package/src/datahike/store.cljc +80 -0
  215. package/src/datahike/tools.cljc +308 -0
  216. package/src/datahike/transit.cljc +80 -0
  217. package/src/datahike/writer.cljc +239 -0
  218. package/src/datahike/writing.cljc +362 -0
  219. package/src/deps.cljs +1 -0
  220. package/src-hitchhiker-tree/datahike/index/hitchhiker_tree/insert.cljc +76 -0
  221. package/src-hitchhiker-tree/datahike/index/hitchhiker_tree/upsert.cljc +128 -0
  222. package/src-hitchhiker-tree/datahike/index/hitchhiker_tree.cljc +213 -0
  223. package/test/datahike/backward_compatibility_test/src/backward_test.clj +37 -0
  224. package/test/datahike/integration_test/config_record_file_test.clj +14 -0
  225. package/test/datahike/integration_test/config_record_test.clj +14 -0
  226. package/test/datahike/integration_test/depr_config_uri_test.clj +15 -0
  227. package/test/datahike/integration_test/return_map_test.clj +62 -0
  228. package/test/datahike/integration_test.cljc +67 -0
  229. package/test/datahike/norm/norm_test.clj +124 -0
  230. package/test/datahike/norm/resources/naming-and-sorting-test/001-a1-example.edn +5 -0
  231. package/test/datahike/norm/resources/naming-and-sorting-test/002-a2-example.edn +5 -0
  232. package/test/datahike/norm/resources/naming-and-sorting-test/003-tx-fn-test.edn +1 -0
  233. package/test/datahike/norm/resources/naming-and-sorting-test/004-tx-data-and-tx-fn-test.edn +5 -0
  234. package/test/datahike/norm/resources/naming-and-sorting-test/01-transact-basic-characters.edn +2 -0
  235. package/test/datahike/norm/resources/naming-and-sorting-test/02 add occupation.edn +5 -0
  236. package/test/datahike/norm/resources/naming-and-sorting-test/checksums.edn +12 -0
  237. package/test/datahike/norm/resources/simple-test/001-a1-example.edn +5 -0
  238. package/test/datahike/norm/resources/simple-test/002-a2-example.edn +5 -0
  239. package/test/datahike/norm/resources/simple-test/checksums.edn +4 -0
  240. package/test/datahike/norm/resources/tx-data-and-tx-fn-test/first/001-a1-example.edn +5 -0
  241. package/test/datahike/norm/resources/tx-data-and-tx-fn-test/first/002-a2-example.edn +5 -0
  242. package/test/datahike/norm/resources/tx-data-and-tx-fn-test/first/003-tx-fn-test.edn +1 -0
  243. package/test/datahike/norm/resources/tx-data-and-tx-fn-test/first/checksums.edn +6 -0
  244. package/test/datahike/norm/resources/tx-data-and-tx-fn-test/second/004-tx-data-and-tx-fn-test.edn +5 -0
  245. package/test/datahike/norm/resources/tx-data-and-tx-fn-test/second/checksums.edn +2 -0
  246. package/test/datahike/norm/resources/tx-fn-test/first/001-a1-example.edn +5 -0
  247. package/test/datahike/norm/resources/tx-fn-test/first/002-a2-example.edn +5 -0
  248. package/test/datahike/norm/resources/tx-fn-test/first/checksums.edn +4 -0
  249. package/test/datahike/norm/resources/tx-fn-test/second/003-tx-fn-test.edn +1 -0
  250. package/test/datahike/norm/resources/tx-fn-test/second/checksums.edn +2 -0
  251. package/test/datahike/test/api_test.cljc +895 -0
  252. package/test/datahike/test/array_test.cljc +40 -0
  253. package/test/datahike/test/attribute_refs/datoms_test.cljc +140 -0
  254. package/test/datahike/test/attribute_refs/db_test.cljc +42 -0
  255. package/test/datahike/test/attribute_refs/differences_test.cljc +515 -0
  256. package/test/datahike/test/attribute_refs/entity_test.cljc +89 -0
  257. package/test/datahike/test/attribute_refs/pull_api_test.cljc +320 -0
  258. package/test/datahike/test/attribute_refs/query_find_specs_test.cljc +59 -0
  259. package/test/datahike/test/attribute_refs/query_fns_test.cljc +130 -0
  260. package/test/datahike/test/attribute_refs/query_interop_test.cljc +47 -0
  261. package/test/datahike/test/attribute_refs/query_not_test.cljc +193 -0
  262. package/test/datahike/test/attribute_refs/query_or_test.cljc +137 -0
  263. package/test/datahike/test/attribute_refs/query_pull_test.cljc +156 -0
  264. package/test/datahike/test/attribute_refs/query_rules_test.cljc +176 -0
  265. package/test/datahike/test/attribute_refs/query_test.cljc +241 -0
  266. package/test/datahike/test/attribute_refs/temporal_search.cljc +22 -0
  267. package/test/datahike/test/attribute_refs/transact_test.cljc +220 -0
  268. package/test/datahike/test/attribute_refs/utils.cljc +128 -0
  269. package/test/datahike/test/cache_test.cljc +38 -0
  270. package/test/datahike/test/components_test.cljc +92 -0
  271. package/test/datahike/test/config_test.cljc +158 -0
  272. package/test/datahike/test/core_test.cljc +105 -0
  273. package/test/datahike/test/datom_test.cljc +44 -0
  274. package/test/datahike/test/db_test.cljc +54 -0
  275. package/test/datahike/test/entity_spec_test.cljc +159 -0
  276. package/test/datahike/test/entity_test.cljc +103 -0
  277. package/test/datahike/test/explode_test.cljc +143 -0
  278. package/test/datahike/test/filter_test.cljc +75 -0
  279. package/test/datahike/test/gc_test.cljc +159 -0
  280. package/test/datahike/test/http/server_test.clj +192 -0
  281. package/test/datahike/test/http/writer_test.clj +86 -0
  282. package/test/datahike/test/ident_test.cljc +32 -0
  283. package/test/datahike/test/index_test.cljc +345 -0
  284. package/test/datahike/test/insert.cljc +125 -0
  285. package/test/datahike/test/java_bindings_test.clj +6 -0
  286. package/test/datahike/test/listen_test.cljc +41 -0
  287. package/test/datahike/test/lookup_refs_test.cljc +266 -0
  288. package/test/datahike/test/lru_test.cljc +27 -0
  289. package/test/datahike/test/migrate_test.clj +297 -0
  290. package/test/datahike/test/model/core.cljc +376 -0
  291. package/test/datahike/test/model/invariant.cljc +142 -0
  292. package/test/datahike/test/model/rng.cljc +82 -0
  293. package/test/datahike/test/model_test.clj +217 -0
  294. package/test/datahike/test/nodejs_test.cljs +262 -0
  295. package/test/datahike/test/online_gc_test.cljc +475 -0
  296. package/test/datahike/test/pod_test.clj +369 -0
  297. package/test/datahike/test/pull_api_test.cljc +474 -0
  298. package/test/datahike/test/purge_test.cljc +144 -0
  299. package/test/datahike/test/query_aggregates_test.cljc +101 -0
  300. package/test/datahike/test/query_find_specs_test.cljc +52 -0
  301. package/test/datahike/test/query_fns_test.cljc +523 -0
  302. package/test/datahike/test/query_interop_test.cljc +47 -0
  303. package/test/datahike/test/query_not_test.cljc +189 -0
  304. package/test/datahike/test/query_or_test.cljc +158 -0
  305. package/test/datahike/test/query_pull_test.cljc +147 -0
  306. package/test/datahike/test/query_rules_test.cljc +248 -0
  307. package/test/datahike/test/query_stats_test.cljc +218 -0
  308. package/test/datahike/test/query_test.cljc +984 -0
  309. package/test/datahike/test/schema_test.cljc +424 -0
  310. package/test/datahike/test/specification_test.cljc +30 -0
  311. package/test/datahike/test/store_test.cljc +78 -0
  312. package/test/datahike/test/stress_test.cljc +57 -0
  313. package/test/datahike/test/time_variance_test.cljc +518 -0
  314. package/test/datahike/test/tools_test.clj +134 -0
  315. package/test/datahike/test/transact_test.cljc +518 -0
  316. package/test/datahike/test/tuples_test.cljc +564 -0
  317. package/test/datahike/test/unstructured_test.cljc +291 -0
  318. package/test/datahike/test/upsert_impl_test.cljc +205 -0
  319. package/test/datahike/test/upsert_test.cljc +363 -0
  320. package/test/datahike/test/utils.cljc +110 -0
  321. package/test/datahike/test/validation_test.cljc +48 -0
  322. package/test/datahike/test/versioning_test.cljc +56 -0
  323. package/test/datahike/test.cljc +66 -0
  324. package/tests.edn +24 -0
@@ -0,0 +1,47 @@
1
+ (ns datahike.test.query-interop-test
2
+ (:require
3
+ #?(:cljs [cljs.test :as t :refer-macros [is are deftest]]
4
+ :clj [clojure.test :as t :refer [is are deftest]])
5
+ [datahike.api :as d]
6
+ [datahike.db :as db]))
7
+
8
+ #?(:cljs (def Exception js/Error))
9
+
10
+ (def test-db
11
+ (d/db-with
12
+ (db/empty-db)
13
+ [[:db/add 1 :name "Vlad"]
14
+ [:db/add 2 :name "Ivan"]
15
+ [:db/add 3 :name "Sergey"]]))
16
+
17
+ (deftest test-filter
18
+ (are [q expected] (= (d/q q test-db) expected)
19
+ '[:find ?v
20
+ :where [_ :name ?v]
21
+ [(.startsWith ?v "Ser")]]
22
+ #{["Sergey"]}
23
+
24
+ '[:find ?v
25
+ :where [_ :name ?v]
26
+ [(.contains ?v "a")]]
27
+ #{["Vlad"] ["Ivan"]}
28
+
29
+ '[:find ?v
30
+ :where [_ :name ?v]
31
+ [(.matches ?v ".+rg.+")]]
32
+ #{["Sergey"]}))
33
+
34
+ (deftest test-bind
35
+ (are [q expected] (= (d/q q test-db) expected)
36
+ '[:find ?V
37
+ :where
38
+ [?e :name ?v]
39
+ [(.toLowerCase ?v) ?V]]
40
+ #{["vlad"] ["ivan"] ["sergey"]}))
41
+
42
+ (deftest test-method-not-found
43
+ (is (thrown? Exception (d/q '[:find ?v
44
+ :where
45
+ [?e :name ?v]
46
+ [(.thisMethodDoesNotExist ?v 1)]]
47
+ test-db))))
@@ -0,0 +1,189 @@
1
+ (ns datahike.test.query-not-test
2
+ (:require
3
+ #?(:cljs [cljs.test :as t :refer-macros [are deftest]]
4
+ :clj [clojure.test :as t :refer [are deftest]])
5
+ [datahike.api :as d]
6
+ #?(:cljs [datahike.cljs :refer [Throwable]])
7
+ [datahike.db :as db]
8
+ [datahike.test.core-test]))
9
+
10
+ (def test-db
11
+ (delay
12
+ (d/db-with (db/empty-db)
13
+ [{:db/id 1 :name "Ivan" :age 10}
14
+ {:db/id 2 :name "Ivan" :age 20}
15
+ {:db/id 3 :name "Oleg" :age 10}
16
+ {:db/id 4 :name "Oleg" :age 20}
17
+ {:db/id 5 :name "Ivan" :age 10}
18
+ {:db/id 6 :name "Ivan" :age 20}])))
19
+
20
+ (deftest test-not
21
+ (are [q res] (= (set (d/q (into '[:find [?e ...] :where] (quote q)) @test-db))
22
+ res)
23
+ [[?e :name]
24
+ (not [?e :name "Ivan"])]
25
+ #{3 4}
26
+
27
+ [[?e :name]
28
+ (not
29
+ [?e :name "Ivan"]
30
+ [?e :age 10])]
31
+ #{2 3 4 6}
32
+
33
+ [[?e :name]
34
+ (not [?e :name "Ivan"])
35
+ (not [?e :age 10])]
36
+ #{4}
37
+
38
+ ;; full exclude
39
+ [[?e :name]
40
+ (not [?e :age])]
41
+ #{}
42
+
43
+ ;; not-intersecting rels
44
+ [[?e :name "Ivan"]
45
+ (not [?e :name "Oleg"])]
46
+ #{1 2 5 6}
47
+
48
+ ;; exclude empty set
49
+ [[?e :name]
50
+ (not [?e :name "Ivan"]
51
+ [?e :name "Oleg"])]
52
+ #{1 2 3 4 5 6}
53
+
54
+ ;; nested excludes
55
+ [[?e :name]
56
+ (not [?e :name "Ivan"]
57
+ (not [?e :age 10]))]
58
+ #{1 3 4 5}
59
+
60
+ ;; extra binding in not
61
+ [[?e :name ?a]
62
+ (not [?e :age ?f]
63
+ [?e :age 10])]
64
+ #{2 4 6}))
65
+
66
+ (deftest test-not-join
67
+ (are [q res] (= (d/q (into '[:find ?e ?a :where] (quote q)) @test-db)
68
+ res)
69
+ [[?e :name]
70
+ [?e :age ?a]
71
+ (not-join [?e]
72
+ [?e :name "Oleg"]
73
+ [?e :age ?a])]
74
+ #{[1 10] [2 20] [5 10] [6 20]}
75
+
76
+ [[?e :age ?a]
77
+ [?e :age 10]
78
+ (not-join [?e]
79
+ [?e :name "Oleg"]
80
+ [?e :age ?a]
81
+ [?e :age 10])]
82
+ #{[1 10] [5 10]}))
83
+
84
+ (deftest test-default-source
85
+ (let [db1 (d/db-with (db/empty-db)
86
+ [[:db/add 1 :name "Ivan"]
87
+ [:db/add 2 :name "Oleg"]])
88
+ db2 (d/db-with (db/empty-db)
89
+ [[:db/add 1 :age 10]
90
+ [:db/add 2 :age 20]])]
91
+ (are [q res] (= (set (d/q (into '[:find [?e ...]
92
+ :in $ $2
93
+ :where]
94
+ (quote q))
95
+ db1 db2))
96
+ res)
97
+ ;; NOT inherits default source
98
+ [[?e :name]
99
+ (not [?e :name "Ivan"])]
100
+ #{2}
101
+
102
+ ;; NOT can reference any source
103
+ [[?e :name]
104
+ (not [$2 ?e :age 10])]
105
+ #{2}
106
+
107
+ ;; NOT can change default source
108
+ [[?e :name]
109
+ ($2 not [?e :age 10])]
110
+ #{2}
111
+
112
+ ;; even with another default source, it can reference any other source explicitly
113
+ [[?e :name]
114
+ ($2 not [$ ?e :name "Ivan"])]
115
+ #{2}
116
+
117
+ ;; nested NOT keeps the default source
118
+ [[?e :name]
119
+ ($2 not (not [?e :age 10]))]
120
+ #{1}
121
+
122
+ ;; can override nested NOT source
123
+ [[?e :name]
124
+ ($2 not ($ not [?e :name "Ivan"]))]
125
+ #{1})))
126
+
127
+ (deftest test-impl-edge-cases
128
+ (are [q res] (= (d/q (quote q) @test-db)
129
+ res)
130
+ ;; const \ empty
131
+ [:find ?e
132
+ :where [?e :name "Oleg"]
133
+ [?e :age 10]
134
+ (not [?e :age 20])]
135
+ #{[3]}
136
+
137
+ ;; const \ const
138
+ [:find ?e
139
+ :where [?e :name "Oleg"]
140
+ [?e :age 10]
141
+ (not [?e :age 10])]
142
+ #{}
143
+
144
+ ;; rel \ const
145
+ [:find ?e
146
+ :where [?e :name "Oleg"]
147
+ (not [?e :age 10])]
148
+ #{[4]}
149
+
150
+ ;; 2 rels \ 2 rels
151
+ [:find ?e ?e2
152
+ :where [?e :name "Ivan"]
153
+ [?e2 :name "Ivan"]
154
+ (not [?e :age 10]
155
+ [?e2 :age 20])]
156
+ #{[2 1] [6 5] [1 1] [2 2] [5 5] [6 6] [2 5] [1 5] [2 6] [6 1] [5 1] [6 2]}
157
+
158
+ ;; 2 rels \ rel + const
159
+ [:find ?e ?e2
160
+ :where [?e :name "Ivan"]
161
+ [?e2 :name "Oleg"]
162
+ (not [?e :age 10]
163
+ [?e2 :age 20])]
164
+ #{[2 3] [1 3] [2 4] [6 3] [5 3] [6 4]}
165
+
166
+ ;; 2 rels \ 2 consts
167
+ [:find ?e ?e2
168
+ :where [?e :name "Oleg"]
169
+ [?e2 :name "Oleg"]
170
+ (not [?e :age 10]
171
+ [?e2 :age 20])]
172
+ #{[4 3] [3 3] [4 4]}))
173
+
174
+ (deftest test-insufficient-bindings
175
+ (are [q msg] (thrown-with-msg? Throwable msg
176
+ (d/q (into '[:find ?e :where] (quote q)) @test-db))
177
+ [(not [?e :name "Ivan"])
178
+ [?e :name]]
179
+ #"Insufficient bindings: none of #\{\?e\} is bound"
180
+
181
+ [[?e :name]
182
+ (not-join [?e]
183
+ (not [1 :age ?a])
184
+ [?e :age ?a])]
185
+ #"Insufficient bindings: none of #\{\?a\} is bound"
186
+
187
+ [[?e :name]
188
+ (not [?a :name "Ivan"])]
189
+ #"Insufficient bindings: none of #\{\?a\} is bound"))
@@ -0,0 +1,158 @@
1
+ (ns datahike.test.query-or-test
2
+ (:require
3
+ #?(:cljs [cljs.test :as t :refer-macros [is are deftest]]
4
+ :clj [clojure.test :as t :refer [is are deftest]])
5
+ [datahike.api :as d]
6
+ #?(:cljs [datahike.cljs :refer [Throwable]])
7
+ [datahike.db :as db]
8
+ [datahike.test.core-test]))
9
+
10
+ (def test-db
11
+ (delay
12
+ (d/db-with (db/empty-db)
13
+ [{:db/id 1 :name "Ivan" :age 10}
14
+ {:db/id 2 :name "Ivan" :age 20}
15
+ {:db/id 3 :name "Oleg" :age 10}
16
+ {:db/id 4 :name "Oleg" :age 20}
17
+ {:db/id 5 :name "Ivan" :age 10}
18
+ {:db/id 6 :name "Ivan" :age 20}])))
19
+
20
+ (deftest test-or
21
+ (are [q res] (= (d/q (into '[:find ?e :where] (quote q)) @test-db)
22
+ (into #{} (map vector) res))
23
+
24
+ ;; intersecting results
25
+ [(or [?e :name "Oleg"]
26
+ [?e :age 10])]
27
+ #{1 3 4 5}
28
+
29
+ ;; one branch empty
30
+ [(or [?e :name "Oleg"]
31
+ [?e :age 30])]
32
+ #{3 4}
33
+
34
+ ;; both empty
35
+ [(or [?e :name "Petr"]
36
+ [?e :age 30])]
37
+ #{}
38
+
39
+ ;; join with 1 var
40
+ [[?e :name "Ivan"]
41
+ (or [?e :name "Oleg"]
42
+ [?e :age 10])]
43
+ #{1 5}
44
+
45
+ ;; join with 2 vars
46
+ [[?e :age ?a]
47
+ (or (and [?e :name "Ivan"]
48
+ [1 :age ?a])
49
+ (and [?e :name "Oleg"]
50
+ [2 :age ?a]))]
51
+ #{1 5 4}
52
+
53
+ ;; OR introduces vars
54
+ [(or (and [?e :name "Ivan"]
55
+ [1 :age ?a])
56
+ (and [?e :name "Oleg"]
57
+ [2 :age ?a]))
58
+ [?e :age ?a]]
59
+ #{1 5 4}
60
+
61
+ ;; OR introduces vars in different order
62
+ [(or (and [?e :name "Ivan"]
63
+ [1 :age ?a])
64
+ (and [2 :age ?a]
65
+ [?e :name "Oleg"]))
66
+ [?e :age ?a]]
67
+ #{1 5 4}))
68
+
69
+ (deftest test-or-join
70
+ (are [q res] (= (d/q (into '[:find ?e :where] (quote q)) @test-db)
71
+ (into #{} (map vector) res))
72
+ [(or-join [?e]
73
+ [?e :name ?n]
74
+ (and [?e :age ?a]
75
+ [?e :name ?n]))]
76
+ #{1 2 3 4 5 6}
77
+
78
+ [[?e :name ?a]
79
+ [?e2 :name ?a]
80
+ (or-join [?e]
81
+ (and [?e :age ?a]
82
+ [?e2 :age ?a]))]
83
+ #{1 2 3 4 5 6})
84
+
85
+ (is (= #{[:a1 :b1 :c1]
86
+ [:a2 :b2 :c2]}
87
+ (d/q '[:find ?a ?b ?c
88
+ :in $xs $ys
89
+ :where [$xs ?a ?b ?c] ;; check join by ?a, ignoring ?b, dropping ?c ?d
90
+ (or-join [?a]
91
+ [$ys ?a ?b ?d])]
92
+ [[:a1 :b1 :c1]
93
+ [:a2 :b2 :c2]
94
+ [:a3 :b3 :c3]]
95
+ [[:a1 :b1 :d1] ;; same ?a, same ?b
96
+ [:a2 :b2* :d2] ;; same ?a, different ?b. Should still be joined
97
+ [:a4 :b4 :c4]]))) ;; different ?a, should be dropped
98
+
99
+ (is (= #{[:a1 :c1] [:a2 :c2]}
100
+ (d/q '[:find ?a ?c
101
+ :in $xs $ys
102
+ :where (or-join [?a ?c]
103
+ [$xs ?a ?b ?c] ; rel with hole (?b gets dropped, leaving {?a 0 ?c 2} and 3-element tuples)
104
+ [$ys ?a ?c])]
105
+ [[:a1 :b1 :c1]]
106
+ [[:a2 :c2]]))))
107
+
108
+ (deftest test-default-source
109
+ (let [db1 (d/db-with (db/empty-db)
110
+ [[:db/add 1 :name "Ivan"]
111
+ [:db/add 2 :name "Oleg"]])
112
+ db2 (d/db-with (db/empty-db)
113
+ [[:db/add 1 :age 10]
114
+ [:db/add 2 :age 20]])]
115
+ (are [q res] (= (d/q (into '[:find ?e :in $ $2 :where] (quote q)) db1 db2)
116
+ (into #{} (map vector) res))
117
+ ;; OR inherits default source
118
+ [[?e :name]
119
+ (or [?e :name "Ivan"])]
120
+ #{1}
121
+
122
+ ;; OR can reference any source
123
+ [[?e :name]
124
+ (or [$2 ?e :age 10])]
125
+ #{1}
126
+
127
+ ;; OR can change default source
128
+ [[?e :name]
129
+ ($2 or [?e :age 10])]
130
+ #{1}
131
+
132
+ ;; even with another default source, it can reference any other source explicitly
133
+ [[?e :name]
134
+ ($2 or [$ ?e :name "Ivan"])]
135
+ #{1}
136
+
137
+ ;; nested OR keeps the default source
138
+ [[?e :name]
139
+ ($2 or (or [?e :age 10]))]
140
+ #{1}
141
+
142
+ ;; can override nested OR source
143
+ [[?e :name]
144
+ ($2 or ($ or [?e :name "Ivan"]))]
145
+ #{1})))
146
+
147
+ (deftest test-errors
148
+ (is (thrown-with-msg? Throwable #"Join variable not declared inside clauses"
149
+ (d/q '[:find ?e
150
+ :where (or [?e :name _]
151
+ [?e :age ?a])]
152
+ @test-db)))
153
+
154
+ (is (thrown-with-msg? Throwable #"Insufficient bindings: #\{\?e\} not bound"
155
+ (d/q '[:find ?e
156
+ :where (or-join [[?e]]
157
+ [?e :name "Ivan"])]
158
+ @test-db))))
@@ -0,0 +1,147 @@
1
+ (ns datahike.test.query-pull-test
2
+ (:require
3
+ #?(:cljs [cljs.test :as t :refer-macros [is are deftest testing]]
4
+ :clj [clojure.test :as t :refer [is are deftest testing]])
5
+ [datahike.api :as d]
6
+ [datahike.db :as db]))
7
+
8
+ (def test-db (d/db-with (db/empty-db)
9
+ [{:db/id 1 :name "Petr" :age 44}
10
+ {:db/id 2 :name "Ivan" :age 25}
11
+ {:db/id 3 :name "Oleg" :age 11}]))
12
+
13
+ (deftest test-basics
14
+ (are [find res] (= (set (d/q {:find find
15
+ :where '[[?e :age ?a]
16
+ [(>= ?a 18)]]}
17
+ test-db))
18
+ res)
19
+ '[(pull ?e [:name])]
20
+ #{[{:name "Ivan"}] [{:name "Petr"}]}
21
+
22
+ '[(pull ?e [*])]
23
+ #{[{:db/id 2 :age 25 :name "Ivan"}] [{:db/id 1 :age 44 :name "Petr"}]}
24
+
25
+ '[?e (pull ?e [:name])]
26
+ #{[2 {:name "Ivan"}] [1 {:name "Petr"}]}
27
+
28
+ '[?e ?a (pull ?e [:name])]
29
+ #{[2 25 {:name "Ivan"}] [1 44 {:name "Petr"}]}
30
+
31
+ '[?e ?a (pull ?e #{:name})]
32
+ #{[2 25 {:name "Ivan"}] [1 44 {:name "Petr"}]}
33
+
34
+ '[?e (pull ?e [:name]) ?a]
35
+ #{[2 {:name "Ivan"} 25] [1 {:name "Petr"} 44]}))
36
+
37
+ (deftest test-var-pattern
38
+ (are [find pattern res] (= (set (d/q {:find find
39
+ :in '[$ ?pattern]
40
+ :where '[[?e :age ?a]
41
+ [(>= ?a 18)]]}
42
+ test-db pattern))
43
+ res)
44
+ '[(pull ?e ?pattern)] [:name]
45
+ #{[{:name "Ivan"}] [{:name "Petr"}]}
46
+
47
+ '[(pull ?e ?pattern)] #{:name}
48
+ #{[{:name "Ivan"}] [{:name "Petr"}]}
49
+
50
+ '[?e ?a ?pattern (pull ?e ?pattern)] [:name]
51
+ #{[2 25 [:name] {:name "Ivan"}] [1 44 [:name] {:name "Petr"}]}))
52
+
53
+ ;; not supported
54
+ #_(deftest test-multi-pattern
55
+ (is (= (set (d/q '[:find ?e ?p (pull ?e ?p)
56
+ :in $ [?p ...]
57
+ :where [?e :age ?a]
58
+ [>= ?a 18]]
59
+ test-db [[:name] [:age]]))
60
+ #{[2 [:name] {:name "Ivan"}]
61
+ [2 [:age] {:age 25}]
62
+ [1 [:name] {:name "Petr"}]
63
+ [1 [:age] {:age 44}]})))
64
+
65
+ (deftest test-multiple-sources
66
+ (let [db1 (d/db-with (db/empty-db) [{:db/id 1 :name "Ivan" :age 25}])
67
+ db2 (d/db-with (db/empty-db) [{:db/id 1 :name "Petr" :age 25}])]
68
+ (is (= (set (d/q '[:find ?e (pull $1 ?e [:name])
69
+ :in $1 $2
70
+ :where [$1 ?e :age 25]]
71
+ db1 db2))
72
+ #{[1 {:name "Ivan"}]}))
73
+
74
+ (is (= (set (d/q '[:find ?e (pull $2 ?e [:name])
75
+ :in $1 $2
76
+ :where [$2 ?e :age 25]]
77
+ db1 db2))
78
+ #{[1 {:name "Petr"}]}))
79
+
80
+ (testing "$ is default source"
81
+ (is (= (set (d/q '[:find ?e (pull ?e [:name])
82
+ :in $1 $
83
+ :where [$ ?e :age 25]]
84
+ db1 db2))
85
+ #{[1 {:name "Petr"}]})))))
86
+
87
+ (deftest test-find-spec
88
+ (is (= (d/q '[:find (pull ?e [:name]) .
89
+ :where [?e :age 25]]
90
+ test-db)
91
+ {:name "Ivan"}))
92
+ (is (= (d/q '[:find (pull ?e #{:name}) .
93
+ :where [?e :age 25]]
94
+ test-db)
95
+ {:name "Ivan"}))
96
+
97
+ (is (= (set (d/q '[:find [(pull ?e [:name]) ...]
98
+ :where [?e :age ?a]]
99
+ test-db))
100
+ #{{:name "Ivan"} {:name "Petr"} {:name "Oleg"}}))
101
+
102
+ (is (= (d/q '[:find [?e (pull ?e [:name])]
103
+ :where [?e :age 25]]
104
+ test-db)
105
+ [2 {:name "Ivan"}])))
106
+
107
+ (deftest test-find-spec-input
108
+ (is (= (d/q '[:find (pull ?e ?p) .
109
+ :in $ ?p
110
+ :where [(ground 2) ?e]]
111
+ test-db [:name])
112
+ {:name "Ivan"}))
113
+ (is (= (d/q '[:find (pull ?e ?p) .
114
+ :in $ ?p
115
+ :where [(ground 2) ?e]]
116
+ test-db #{:name})
117
+ {:name "Ivan"}))
118
+ (is (= (d/q '[:find (pull ?e p) .
119
+ :in $ p
120
+ :where [(ground 2) ?e]]
121
+ test-db [:name])
122
+ {:name "Ivan"})))
123
+
124
+ (deftest test-aggregates
125
+ (let [db (d/db-with (db/empty-db {:value {:db/cardinality :db.cardinality/many}})
126
+ [{:db/id 1 :name "Petr" :value [10 20 30 40]}
127
+ {:db/id 2 :name "Ivan" :value [14 16]}
128
+ {:db/id 3 :name "Oleg" :value 1}])]
129
+ (is (= (set (d/q '[:find ?e (pull ?e [:name]) (min ?v) (max ?v)
130
+ :where [?e :value ?v]]
131
+ db))
132
+ #{[1 {:name "Petr"} 10 40]
133
+ [2 {:name "Ivan"} 14 16]
134
+ [3 {:name "Oleg"} 1 1]}))))
135
+
136
+ (deftest test-lookup-refs
137
+ (let [db (d/db-with (db/empty-db {:name {:db/unique :db.unique/identity}})
138
+ [{:db/id 1 :name "Petr" :age 44}
139
+ {:db/id 2 :name "Ivan" :age 25}
140
+ {:db/id 3 :name "Oleg" :age 11}])]
141
+ (is (= (set (d/q '[:find ?ref ?a (pull ?ref [:db/id :name])
142
+ :in $ [?ref ...]
143
+ :where [?ref :age ?a]
144
+ [(>= ?a 18)]]
145
+ db [[:name "Ivan"] [:name "Oleg"] [:name "Petr"]]))
146
+ #{[[:name "Petr"] 44 {:db/id 1 :name "Petr"}]
147
+ [[:name "Ivan"] 25 {:db/id 2 :name "Ivan"}]}))))