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,248 @@
1
+ (ns datahike.test.query-rules-test
2
+ (:require
3
+ #?(:cljs [cljs.test :as t :refer-macros [is deftest testing]]
4
+ :clj [clojure.test :as t :refer [is deftest testing]])
5
+ [datahike.api :as d]
6
+ [datahike.db :as db]))
7
+
8
+ #?(:cljs (def Throwable js/Error))
9
+
10
+ (deftest test-rules
11
+ (let [db [[5 :follow 3]
12
+ [1 :follow 2] [2 :follow 3] [3 :follow 4] [4 :follow 6]
13
+ [2 :follow 4]]]
14
+ (is (= (d/q '[:find ?e1 ?e2
15
+ :in $ %
16
+ :where (follow ?e1 ?e2)]
17
+ db
18
+ '[[(follow ?x ?y)
19
+ [?x :follow ?y]]])
20
+ #{[1 2] [2 3] [3 4] [2 4] [5 3] [4 6]}))
21
+
22
+ (testing "Joining regular clauses with rule"
23
+ (is (= (d/q '[:find ?y ?x
24
+ :in $ %
25
+ :where [_ _ ?x]
26
+ (rule ?x ?y)
27
+ [(even? ?x)]]
28
+ db
29
+ '[[(rule ?a ?b)
30
+ [?a :follow ?b]]])
31
+ #{[3 2] [6 4] [4 2]})))
32
+
33
+ (testing "Rule context is isolated from outer context"
34
+ (is (= (d/q '[:find ?x
35
+ :in $ %
36
+ :where [?e _ _]
37
+ (rule ?x)]
38
+ db
39
+ '[[(rule ?e)
40
+ [_ ?e _]]])
41
+ #{[:follow]})))
42
+
43
+ (testing "Rule with branches"
44
+ (is (= (d/q '[:find ?e2
45
+ :in $ ?e1 %
46
+ :where (follow ?e1 ?e2)]
47
+ db
48
+ 1
49
+ '[[(follow ?e2 ?e1)
50
+ [?e2 :follow ?e1]]
51
+ [(follow ?e2 ?e1)
52
+ [?e2 :follow ?t]
53
+ [?t :follow ?e1]]])
54
+ #{[2] [3] [4]})))
55
+
56
+ (testing "Recursive rules"
57
+ (is (= (d/q '[:find ?e2
58
+ :in $ ?e1 %
59
+ :where (follow ?e1 ?e2)]
60
+ db
61
+ 1
62
+ '[[(follow ?e1 ?e2)
63
+ [?e1 :follow ?e2]]
64
+ [(follow ?e1 ?e2)
65
+ [?e1 :follow ?t]
66
+ (follow ?t ?e2)]])
67
+ #{[2] [3] [4] [6]}))
68
+
69
+ (is (= (d/q '[:find ?e1 ?e2
70
+ :in $ %
71
+ :where (follow ?e1 ?e2)]
72
+ [[1 :follow 2] [2 :follow 3]]
73
+ '[[(follow ?e1 ?e2)
74
+ [?e1 :follow ?e2]]
75
+ [(follow ?e1 ?e2)
76
+ (follow ?e2 ?e1)]])
77
+ #{[1 2] [2 3] [2 1] [3 2]}))
78
+
79
+ (is (= (d/q '[:find ?e1 ?e2
80
+ :in $ %
81
+ :where (follow ?e1 ?e2)]
82
+ [[1 :follow 2] [2 :follow 3] [3 :follow 1]]
83
+ '[[(follow ?e1 ?e2)
84
+ [?e1 :follow ?e2]]
85
+ [(follow ?e1 ?e2)
86
+ (follow ?e2 ?e1)]])
87
+ #{[1 2] [2 3] [3 1] [2 1] [3 2] [1 3]})))
88
+
89
+ (testing "Mutually recursive rules"
90
+ (is (= (d/q '[:find ?e1 ?e2
91
+ :in $ %
92
+ :where (f1 ?e1 ?e2)]
93
+ [[0 :f1 1]
94
+ [1 :f2 2]
95
+ [2 :f1 3]
96
+ [3 :f2 4]
97
+ [4 :f1 5]
98
+ [5 :f2 6]]
99
+ '[[(f1 ?e1 ?e2)
100
+ [?e1 :f1 ?e2]]
101
+ [(f1 ?e1 ?e2)
102
+ [?t :f1 ?e2]
103
+ (f2 ?e1 ?t)]
104
+ [(f2 ?e1 ?e2)
105
+ [?e1 :f2 ?e2]]
106
+ [(f2 ?e1 ?e2)
107
+ [?t :f2 ?e2]
108
+ (f1 ?e1 ?t)]])
109
+ #{[0 1] [0 3] [0 5]
110
+ [1 3] [1 5]
111
+ [2 3] [2 5]
112
+ [3 5]
113
+ [4 5]})))
114
+
115
+ (testing "Passing ins to rule"
116
+ (is (= (d/q '[:find ?x ?y
117
+ :in $ % ?even
118
+ :where
119
+ (match ?even ?x ?y)]
120
+ db
121
+ '[[(match ?pred ?e ?e2)
122
+ [?e :follow ?e2]
123
+ [(?pred ?e)]
124
+ [(?pred ?e2)]]]
125
+ even?)
126
+ #{[4 6] [2 4]})))
127
+
128
+ (testing "Using built-ins inside rule"
129
+ (is (= (d/q '[:find ?x ?y
130
+ :in $ %
131
+ :where (match ?x ?y)]
132
+ db
133
+ '[[(match ?e ?e2)
134
+ [?e :follow ?e2]
135
+ [(even? ?e)]
136
+ [(even? ?e2)]]])
137
+ #{[4 6] [2 4]})))
138
+ (testing "Calling rule twice (#44)"
139
+ (d/q '[:find ?p
140
+ :in $ % ?fn
141
+ :where (rule ?p ?fn "a")
142
+ (rule ?p ?fn "b")]
143
+ [[1 :attr "a"]]
144
+ '[[(rule ?p ?fn ?x)
145
+ [?p :attr ?x]
146
+ [(?fn ?x)]]]
147
+ (constantly true))))
148
+
149
+ (testing "Specifying db to rule"
150
+ (is (= (d/q '[:find ?n
151
+ :in $sexes $ages %
152
+ :where ($sexes male ?n)
153
+ ($ages adult ?n)]
154
+ [["Ivan" :male] ["Darya" :female] ["Oleg" :male] ["Igor" :male]]
155
+ [["Ivan" 15] ["Oleg" 66] ["Darya" 32]]
156
+ '[[(male ?x)
157
+ [?x :male]]
158
+ [(adult ?y)
159
+ [?y ?a]
160
+ [(>= ?a 18)]]])
161
+ #{["Oleg"]}))))
162
+
163
+ ;; https://github.com/tonsky/datahike/issues/218
164
+
165
+ (deftest test-false-arguments
166
+ (let [db (d/db-with (db/empty-db)
167
+ [[:db/add 1 :attr true]
168
+ [:db/add 2 :attr false]])
169
+ rules '[[(is ?id ?val)
170
+ [?id :attr ?val]]]]
171
+ (is (= (d/q '[:find ?id :in $ %
172
+ :where (is ?id true)]
173
+ db rules)
174
+ #{[1]}))
175
+ (is (= (d/q '[:find ?id :in $ %
176
+ :where (is ?id false)] db rules)
177
+ #{[2]}))))
178
+
179
+ (deftest test-rule-arguments
180
+ (let [cfg {:store {:backend :memory
181
+ :id #uuid "a0000000-0000-0000-0000-00000000000a"}
182
+ :name "rule-test"
183
+ :keep-history? true
184
+ :schema-flexibility :write
185
+ :attribute-refs? true}
186
+ schema [{:db/ident :name
187
+ :db/cardinality :db.cardinality/one
188
+ :db/index true
189
+ :db/unique :db.unique/identity
190
+ :db/valueType :db.type/string}
191
+ {:db/ident :parents
192
+ :db/cardinality :db.cardinality/many
193
+ :db/valueType :db.type/ref}
194
+ {:db/ident :age
195
+ :db/cardinality :db.cardinality/one
196
+ :db/valueType :db.type/long}]
197
+ rules '[[(parent-info ?child ?name ?age)
198
+ [?child :parents ?p]
199
+ [(ground ["Alice" "Bob"]) [?name ...]]
200
+ [?p :name ?name]
201
+ [?p :age ?age]]]
202
+ _ (d/delete-database cfg)
203
+ _ (d/create-database cfg)
204
+ conn (d/connect cfg)]
205
+
206
+ (d/transact conn {:tx-data schema})
207
+ (d/transact conn {:tx-data [{:name "Alice"
208
+ :age 25}
209
+ {:name "Bob"
210
+ :age 30}]})
211
+ (d/transact conn {:tx-data [{:name "Charlie"
212
+ :age 5
213
+ :parents [[:name "Alice"]
214
+ [:name "Bob"]]}]})
215
+
216
+ (is (= #{[25]}
217
+ (d/q {:query '{:find [?age]
218
+ :in [$ ?n ?pn %]
219
+ :where
220
+ [[?child :name ?n]
221
+ (parent-info ?child ?pn ?age)]}
222
+ :args [@conn "Charlie" "Alice" rules]})))
223
+
224
+ (is (= #{[25]}
225
+ (d/q {:query '{:find [?age]
226
+ :in [$ ?n [?pn ...] %]
227
+ :where
228
+ [[?child :name ?n]
229
+ (parent-info ?child ?pn ?age)]}
230
+ :args [@conn "Charlie" ["Alice"] rules]})))
231
+
232
+ (is (= #{[25]}
233
+ (d/q {:query '{:find [?age]
234
+ :in [$ ?n %]
235
+ :where
236
+ [[?child :name ?n]
237
+ (parent-info ?child "Alice" ?age)]}
238
+ :args [@conn "Charlie" rules]})))
239
+
240
+ (is (thrown-with-msg? Throwable
241
+ #"Bad format for value in pattern, must be a scalar, nil or a vector of two elements."
242
+ (d/q {:query '{:find [?age]
243
+ :in [$ ?n %]
244
+ :where
245
+ [[?child :name ?n]
246
+ (parent-info ?child ["Alice"] ?age)]}
247
+ :args [@conn "Charlie" rules]})))
248
+ (d/release conn)))
@@ -0,0 +1,218 @@
1
+ (ns datahike.test.query-stats-test
2
+ (:require #?(:cljs [cljs.test :as t :refer-macros [is deftest use-fixtures]]
3
+ :clj [clojure.test :as t :refer [is deftest use-fixtures]])
4
+ [clojure.string :as str]
5
+ [clojure.walk :as cw]
6
+ [datahike.api :as d]
7
+ [datahike.query :refer [query-stats]]
8
+ [datahike.test.utils :refer [with-db]]))
9
+
10
+ (def config {:store {:backend :memory :id #uuid "00190000-0000-0000-0000-000000000019"}})
11
+
12
+ (def test-schema [{:db/ident :name
13
+ :db/valueType :db.type/string
14
+ :db/cardinality :db.cardinality/one}
15
+ {:db/ident :age
16
+ :db/valueType :db.type/long
17
+ :db/cardinality :db.cardinality/one}])
18
+
19
+ (def test-data [{:db/id 1 :name "Ivan" :age 10}
20
+ {:db/id 2 :name "Ivan" :age 20}
21
+ {:db/id 3 :name "Oleg" :age 30}
22
+ {:db/id 4 :name "Oleg" :age 40}
23
+ {:db/id 5 :name "Ivan" :age 50}
24
+ {:db/id 6 :name "Michal" :age 60}])
25
+
26
+ (use-fixtures :once (partial with-db config (into test-schema test-data)))
27
+
28
+ (defn unify-stats [stats]
29
+ (cw/postwalk
30
+ #(cond-> %
31
+ (and (map? %) (contains? % :t)) (assoc :t :measurement)
32
+ (and (symbol? %) (re-find #"__auto__" (name %))) (-> name (str/replace #"__auto__\d*" "_tmp") symbol))
33
+ stats))
34
+
35
+ (deftest test-not
36
+ (is (= {:consts {}
37
+ :query '{:find [[?a ...]]
38
+ :where [[?e :age ?a]
39
+ (not [?e :age 60])]}
40
+ :ret [10 20 40 30 50]
41
+ :rules {}
42
+ :stats [{:clause '[?e :age ?a]
43
+ :rels [{:bound #{'?a '?e} :rows 6}]
44
+ :t :measurement
45
+ :type :lookup}
46
+ {:branches [{:clause '[?e :age 60]
47
+ :rels [{:bound #{'?a '?e} :rows 1}]
48
+ :t :measurement
49
+ :type :lookup}]
50
+ :clause '(not [?e :age 60])
51
+ :rels [{:bound #{'?a '?e} :rows 5}]
52
+ :t :measurement
53
+ :type :not}]}
54
+ (unify-stats (query-stats '[:find [?a ...] :where
55
+ [?e :age ?a]
56
+ (not [?e :age 60])]
57
+ @(d/connect config))))))
58
+
59
+ (deftest test-not-join
60
+ (is (= {:consts {}
61
+ :query '{:find [?a]
62
+ :where [[?e :name]
63
+ [?e :age ?a]
64
+ (not-join [?e]
65
+ [?e :name "Oleg"]
66
+ [?e :age ?a])]}
67
+ :ret #{[10] [20] [50] [60]}
68
+ :rules {}
69
+ :stats [{:clause '[?e :name]
70
+ :rels [{:bound #{'?e} :rows 6}]
71
+ :t :measurement
72
+ :type :lookup}
73
+ {:clause '[?e :age ?a]
74
+ :rels [{:bound #{'?a '?e} :rows 6}]
75
+ :t :measurement
76
+ :type :lookup}
77
+ {:branches [{:clause '[?e :name "Oleg"]
78
+ :rels [{:bound #{'?e} :rows 2}]
79
+ :t :measurement
80
+ :type :lookup}
81
+ {:clause '[?e :age ?a]
82
+ :rels [{:bound #{'?a '?e} :rows 2}]
83
+ :t :measurement
84
+ :type :lookup}]
85
+ :clause '(not-join [?e]
86
+ [?e :name "Oleg"]
87
+ [?e :age ?a])
88
+ :rels [{:bound #{'?a '?e} :rows 4}]
89
+ :t :measurement
90
+ :type :not}]}
91
+ (unify-stats (query-stats '[:find ?a :where
92
+ [?e :name]
93
+ [?e :age ?a]
94
+ (not-join [?e]
95
+ [?e :name "Oleg"]
96
+ [?e :age ?a])]
97
+ @(d/connect config))))))
98
+
99
+ (deftest test-or
100
+ (is (= {:consts {}
101
+ :query '{:find [?a]
102
+ :where [[?e :age ?a]
103
+ (or [?e :name "Ivan"]
104
+ [?e :name "Oleg"])]}
105
+ :ret #{[10] [20] [30] [40] [50]}
106
+ :rules {}
107
+ :stats [{:clause '[?e :age ?a]
108
+ :rels [{:bound #{'?a '?e}
109
+ :rows 6}]
110
+ :t :measurement
111
+ :type :lookup}
112
+ {:branches [[{:clause '[?e :name "Ivan"]
113
+ :rels [{:bound #{'?a '?e} :rows 3}]
114
+ :t :measurement
115
+ :type :lookup}]
116
+ [{:clause '[?e :name "Oleg"]
117
+ :rels [{:bound #{'?a '?e} :rows 2}]
118
+ :t :measurement
119
+ :type :lookup}]]
120
+ :clause '(or [?e :name "Ivan"]
121
+ [?e :name "Oleg"])
122
+ :rels [{:bound #{'?a '?e} :rows 5}]
123
+ :t :measurement
124
+ :type :or}]}
125
+ (unify-stats (query-stats '[:find ?a :where
126
+ [?e :age ?a]
127
+ (or [?e :name "Ivan"]
128
+ [?e :name "Oleg"])]
129
+ @(d/connect config))))))
130
+
131
+ (deftest test-rules
132
+ (let [db [[1 :follow 2]
133
+ [2 :follow 3]
134
+ [2 :follow 4]
135
+ [3 :follow 4]
136
+ [4 :follow 6]
137
+ [5 :follow 3]]]
138
+ (is (= {:consts {},
139
+ :query '{:find [?y ?x], :in [$ %], :where [[_ _ ?x] (rule ?x ?y) [(even? ?x)]]},
140
+ :ret #{[3 2] [4 2] [6 4]},
141
+ :rules {'rule '[[(rule ?a ?b) [?a :follow ?b]]]},
142
+ :stats [{:clause '[_ _ ?x], :rels [{:bound #{'?x}, :rows 6}], :t :measurement, :type :lookup}
143
+ {:branches [{:branches [],
144
+ :clause '(rule ?x ?y),
145
+ :clauses (),
146
+ :rels [{:bound #{'?x}, :rows 6}],
147
+ :t :measurement,
148
+ :type :solve}
149
+ {:branches [{:clause '[?x :follow ?y],
150
+ :rels [{:bound #{'?x '?y}, :rows 6}],
151
+ :t :measurement,
152
+ :type :lookup}],
153
+ :clause '([?x :follow ?y]),
154
+ :clauses '([?x :follow ?y]),
155
+ :rels [{:bound #{'?x '?y}, :rows 6}],
156
+ :t :measurement,
157
+ :type :solve}],
158
+ :clause '(rule ?x ?y),
159
+ :rels [{:bound #{'?x '?y}, :rows 10}],
160
+ :t :measurement,
161
+ :type :rule}
162
+ {:clause '[(even? ?x)], :rels [{:bound #{'?x '?y}, :rows 6}], :t :measurement}]}
163
+ (unify-stats (query-stats '[:find ?y ?x
164
+ :in $ %
165
+ :where [_ _ ?x]
166
+ (rule ?x ?y)
167
+ [(even? ?x)]]
168
+ db
169
+ '[[(rule ?a ?b)
170
+ [?a :follow ?b]]]))))
171
+
172
+ (is (= {:consts '{?e1 1},
173
+ :query {:find '[?e2], :in '[$ ?e1 %], :where '[(follow ?e1 ?e2)]},
174
+ :ret #{[2] [3] [4]},
175
+ :rules {'follow '[[(follow ?e2 ?e1) [?e2 :follow ?e1]]
176
+ [(follow ?e2 ?e1) [?e2 :follow ?t] [?t :follow ?e1]]]},
177
+ :stats [{:branches [{:branches [],
178
+ :clause '(follow ?e1 ?e2),
179
+ :clauses (),
180
+ :rels [],
181
+ :t :measurement,
182
+ :type :solve}
183
+ {:branches [{:clause '[?e1 :follow ?e2],
184
+ :rels [{:bound #{'?e1 '?e2}, :rows 1}],
185
+ :t :measurement,
186
+ :type :lookup}],
187
+ :clause '([?e1 :follow ?e2]),
188
+ :clauses '([?e1 :follow ?e2]),
189
+ :rels [{:bound #{'?e1 '?e2}, :rows 1}],
190
+ :t :measurement,
191
+ :type :solve}
192
+ {:branches '[{:clause [?e1 :follow ?t_tmp],
193
+ :rels [{:bound #{?e1 ?t_tmp}, :rows 1}],
194
+ :t :measurement,
195
+ :type :lookup}
196
+ {:clause [?t_tmp :follow ?e2],
197
+ :rels [{:bound #{?e1 ?e2 ?t_tmp}, :rows 2}],
198
+ :t :measurement,
199
+ :type :lookup}],
200
+ :clause '([?e1 :follow ?t_tmp] [?t_tmp :follow ?e2]),
201
+ :clauses '([?e1 :follow ?t_tmp] [?t_tmp :follow ?e2]),
202
+ :rels [{:bound #{'?e1 '?e2 '?t_tmp}, :rows 2}],
203
+ :t :measurement,
204
+ :type :solve}],
205
+ :clause '(follow ?e1 ?e2),
206
+ :rels [{:bound #{'?e1 '?e2}, :rows 3}],
207
+ :t :measurement,
208
+ :type :rule}]}
209
+ (unify-stats (query-stats '[:find ?e2
210
+ :in $ ?e1 %
211
+ :where (follow ?e1 ?e2)]
212
+ db
213
+ 1
214
+ '[[(follow ?e2 ?e1)
215
+ [?e2 :follow ?e1]]
216
+ [(follow ?e2 ?e1)
217
+ [?e2 :follow ?t]
218
+ [?t :follow ?e1]]]))))))