@livestore/common 0.3.0-dev.5 → 0.3.0-dev.50

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 (491) hide show
  1. package/dist/.tsbuildinfo +1 -1
  2. package/dist/__tests__/fixture.d.ts +83 -221
  3. package/dist/__tests__/fixture.d.ts.map +1 -1
  4. package/dist/__tests__/fixture.js +33 -11
  5. package/dist/__tests__/fixture.js.map +1 -1
  6. package/dist/adapter-types.d.ts +120 -64
  7. package/dist/adapter-types.d.ts.map +1 -1
  8. package/dist/adapter-types.js +39 -8
  9. package/dist/adapter-types.js.map +1 -1
  10. package/dist/bounded-collections.d.ts.map +1 -1
  11. package/dist/debug-info.d.ts +1 -1
  12. package/dist/debug-info.d.ts.map +1 -1
  13. package/dist/debug-info.js +1 -0
  14. package/dist/debug-info.js.map +1 -1
  15. package/dist/devtools/devtools-messages-client-session.d.ts +390 -0
  16. package/dist/devtools/devtools-messages-client-session.d.ts.map +1 -0
  17. package/dist/devtools/devtools-messages-client-session.js +97 -0
  18. package/dist/devtools/devtools-messages-client-session.js.map +1 -0
  19. package/dist/devtools/devtools-messages-common.d.ts +68 -0
  20. package/dist/devtools/devtools-messages-common.d.ts.map +1 -0
  21. package/dist/devtools/devtools-messages-common.js +60 -0
  22. package/dist/devtools/devtools-messages-common.js.map +1 -0
  23. package/dist/devtools/devtools-messages-leader.d.ts +394 -0
  24. package/dist/devtools/devtools-messages-leader.d.ts.map +1 -0
  25. package/dist/devtools/devtools-messages-leader.js +147 -0
  26. package/dist/devtools/devtools-messages-leader.js.map +1 -0
  27. package/dist/devtools/devtools-messages.d.ts +3 -580
  28. package/dist/devtools/devtools-messages.d.ts.map +1 -1
  29. package/dist/devtools/devtools-messages.js +3 -174
  30. package/dist/devtools/devtools-messages.js.map +1 -1
  31. package/dist/devtools/devtools-sessioninfo.d.ts +32 -0
  32. package/dist/devtools/devtools-sessioninfo.d.ts.map +1 -0
  33. package/dist/devtools/devtools-sessioninfo.js +36 -0
  34. package/dist/devtools/devtools-sessioninfo.js.map +1 -0
  35. package/dist/devtools/mod.d.ts +55 -0
  36. package/dist/devtools/mod.d.ts.map +1 -0
  37. package/dist/devtools/mod.js +33 -0
  38. package/dist/devtools/mod.js.map +1 -0
  39. package/dist/index.d.ts +7 -13
  40. package/dist/index.d.ts.map +1 -1
  41. package/dist/index.js +7 -9
  42. package/dist/index.js.map +1 -1
  43. package/dist/leader-thread/LeaderSyncProcessor.d.ts +62 -0
  44. package/dist/leader-thread/LeaderSyncProcessor.d.ts.map +1 -0
  45. package/dist/leader-thread/LeaderSyncProcessor.js +595 -0
  46. package/dist/leader-thread/LeaderSyncProcessor.js.map +1 -0
  47. package/dist/leader-thread/connection.d.ts +34 -6
  48. package/dist/leader-thread/connection.d.ts.map +1 -1
  49. package/dist/leader-thread/connection.js +22 -7
  50. package/dist/leader-thread/connection.js.map +1 -1
  51. package/dist/leader-thread/eventlog.d.ts +27 -0
  52. package/dist/leader-thread/eventlog.d.ts.map +1 -0
  53. package/dist/leader-thread/eventlog.js +119 -0
  54. package/dist/leader-thread/eventlog.js.map +1 -0
  55. package/dist/leader-thread/leader-worker-devtools.d.ts +1 -1
  56. package/dist/leader-thread/leader-worker-devtools.d.ts.map +1 -1
  57. package/dist/leader-thread/leader-worker-devtools.js +155 -80
  58. package/dist/leader-thread/leader-worker-devtools.js.map +1 -1
  59. package/dist/leader-thread/make-leader-thread-layer.d.ts +23 -11
  60. package/dist/leader-thread/make-leader-thread-layer.d.ts.map +1 -1
  61. package/dist/leader-thread/make-leader-thread-layer.js +72 -47
  62. package/dist/leader-thread/make-leader-thread-layer.js.map +1 -1
  63. package/dist/leader-thread/materialize-event.d.ts +16 -0
  64. package/dist/leader-thread/materialize-event.d.ts.map +1 -0
  65. package/dist/leader-thread/materialize-event.js +109 -0
  66. package/dist/leader-thread/materialize-event.js.map +1 -0
  67. package/dist/leader-thread/mod.d.ts +1 -1
  68. package/dist/leader-thread/mod.d.ts.map +1 -1
  69. package/dist/leader-thread/mod.js +1 -1
  70. package/dist/leader-thread/mod.js.map +1 -1
  71. package/dist/leader-thread/recreate-db.d.ts +4 -2
  72. package/dist/leader-thread/recreate-db.d.ts.map +1 -1
  73. package/dist/leader-thread/recreate-db.js +33 -31
  74. package/dist/leader-thread/recreate-db.js.map +1 -1
  75. package/dist/leader-thread/shutdown-channel.d.ts +2 -5
  76. package/dist/leader-thread/shutdown-channel.d.ts.map +1 -1
  77. package/dist/leader-thread/shutdown-channel.js +2 -4
  78. package/dist/leader-thread/shutdown-channel.js.map +1 -1
  79. package/dist/leader-thread/types.d.ts +87 -40
  80. package/dist/leader-thread/types.d.ts.map +1 -1
  81. package/dist/leader-thread/types.js +1 -3
  82. package/dist/leader-thread/types.js.map +1 -1
  83. package/dist/make-client-session.d.ts +23 -0
  84. package/dist/make-client-session.d.ts.map +1 -0
  85. package/dist/make-client-session.js +57 -0
  86. package/dist/make-client-session.js.map +1 -0
  87. package/dist/materializer-helper.d.ts +23 -0
  88. package/dist/materializer-helper.d.ts.map +1 -0
  89. package/dist/materializer-helper.js +86 -0
  90. package/dist/materializer-helper.js.map +1 -0
  91. package/dist/otel.d.ts +2 -0
  92. package/dist/otel.d.ts.map +1 -1
  93. package/dist/otel.js +5 -0
  94. package/dist/otel.js.map +1 -1
  95. package/dist/rematerialize-from-eventlog.d.ts +14 -0
  96. package/dist/rematerialize-from-eventlog.d.ts.map +1 -0
  97. package/dist/rematerialize-from-eventlog.js +64 -0
  98. package/dist/rematerialize-from-eventlog.js.map +1 -0
  99. package/dist/schema/EventDef.d.ts +146 -0
  100. package/dist/schema/EventDef.d.ts.map +1 -0
  101. package/dist/schema/EventDef.js +58 -0
  102. package/dist/schema/EventDef.js.map +1 -0
  103. package/dist/schema/EventId.d.ts +43 -25
  104. package/dist/schema/EventId.d.ts.map +1 -1
  105. package/dist/schema/EventId.js +56 -18
  106. package/dist/schema/EventId.js.map +1 -1
  107. package/dist/schema/EventId.test.d.ts +2 -0
  108. package/dist/schema/EventId.test.d.ts.map +1 -0
  109. package/dist/schema/EventId.test.js +11 -0
  110. package/dist/schema/EventId.test.js.map +1 -0
  111. package/dist/schema/EventNumber.d.ts +57 -0
  112. package/dist/schema/EventNumber.d.ts.map +1 -0
  113. package/dist/schema/EventNumber.js +82 -0
  114. package/dist/schema/EventNumber.js.map +1 -0
  115. package/dist/schema/EventNumber.test.d.ts +2 -0
  116. package/dist/schema/EventNumber.test.d.ts.map +1 -0
  117. package/dist/schema/EventNumber.test.js +11 -0
  118. package/dist/schema/EventNumber.test.js.map +1 -0
  119. package/dist/schema/EventSequenceNumber.d.ts +57 -0
  120. package/dist/schema/EventSequenceNumber.d.ts.map +1 -0
  121. package/dist/schema/EventSequenceNumber.js +82 -0
  122. package/dist/schema/EventSequenceNumber.js.map +1 -0
  123. package/dist/schema/EventSequenceNumber.test.d.ts +2 -0
  124. package/dist/schema/EventSequenceNumber.test.d.ts.map +1 -0
  125. package/dist/schema/EventSequenceNumber.test.js +11 -0
  126. package/dist/schema/EventSequenceNumber.test.js.map +1 -0
  127. package/dist/schema/LiveStoreEvent.d.ts +257 -0
  128. package/dist/schema/LiveStoreEvent.d.ts.map +1 -0
  129. package/dist/schema/LiveStoreEvent.js +117 -0
  130. package/dist/schema/LiveStoreEvent.js.map +1 -0
  131. package/dist/schema/events.d.ts +2 -0
  132. package/dist/schema/events.d.ts.map +1 -0
  133. package/dist/schema/events.js +2 -0
  134. package/dist/schema/events.js.map +1 -0
  135. package/dist/schema/mod.d.ts +8 -6
  136. package/dist/schema/mod.d.ts.map +1 -1
  137. package/dist/schema/mod.js +8 -6
  138. package/dist/schema/mod.js.map +1 -1
  139. package/dist/schema/schema.d.ts +50 -32
  140. package/dist/schema/schema.d.ts.map +1 -1
  141. package/dist/schema/schema.js +36 -43
  142. package/dist/schema/schema.js.map +1 -1
  143. package/dist/schema/state/mod.d.ts +3 -0
  144. package/dist/schema/state/mod.d.ts.map +1 -0
  145. package/dist/schema/state/mod.js +3 -0
  146. package/dist/schema/state/mod.js.map +1 -0
  147. package/dist/schema/state/sqlite/client-document-def.d.ts +223 -0
  148. package/dist/schema/state/sqlite/client-document-def.d.ts.map +1 -0
  149. package/dist/schema/state/sqlite/client-document-def.js +170 -0
  150. package/dist/schema/state/sqlite/client-document-def.js.map +1 -0
  151. package/dist/schema/state/sqlite/client-document-def.test.d.ts +2 -0
  152. package/dist/schema/state/sqlite/client-document-def.test.d.ts.map +1 -0
  153. package/dist/schema/state/sqlite/client-document-def.test.js +201 -0
  154. package/dist/schema/state/sqlite/client-document-def.test.js.map +1 -0
  155. package/dist/schema/state/sqlite/db-schema/ast/sqlite.d.ts +69 -0
  156. package/dist/schema/state/sqlite/db-schema/ast/sqlite.d.ts.map +1 -0
  157. package/dist/schema/state/sqlite/db-schema/ast/sqlite.js +71 -0
  158. package/dist/schema/state/sqlite/db-schema/ast/sqlite.js.map +1 -0
  159. package/dist/schema/state/sqlite/db-schema/ast/validate.d.ts +3 -0
  160. package/dist/schema/state/sqlite/db-schema/ast/validate.d.ts.map +1 -0
  161. package/dist/schema/state/sqlite/db-schema/ast/validate.js +12 -0
  162. package/dist/schema/state/sqlite/db-schema/ast/validate.js.map +1 -0
  163. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.d.ts +90 -0
  164. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.d.ts.map +1 -0
  165. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.js +87 -0
  166. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.js.map +1 -0
  167. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.test.d.ts +2 -0
  168. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.test.d.ts.map +1 -0
  169. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.test.js +29 -0
  170. package/dist/schema/state/sqlite/db-schema/dsl/field-defs.test.js.map +1 -0
  171. package/dist/schema/state/sqlite/db-schema/dsl/mod.d.ts +90 -0
  172. package/dist/schema/state/sqlite/db-schema/dsl/mod.d.ts.map +1 -0
  173. package/dist/schema/state/sqlite/db-schema/dsl/mod.js +41 -0
  174. package/dist/schema/state/sqlite/db-schema/dsl/mod.js.map +1 -0
  175. package/dist/schema/state/sqlite/db-schema/hash.d.ts +2 -0
  176. package/dist/schema/state/sqlite/db-schema/hash.d.ts.map +1 -0
  177. package/dist/schema/state/sqlite/db-schema/hash.js +14 -0
  178. package/dist/schema/state/sqlite/db-schema/hash.js.map +1 -0
  179. package/dist/schema/state/sqlite/db-schema/mod.d.ts +3 -0
  180. package/dist/schema/state/sqlite/db-schema/mod.d.ts.map +1 -0
  181. package/dist/schema/state/sqlite/db-schema/mod.js +3 -0
  182. package/dist/schema/state/sqlite/db-schema/mod.js.map +1 -0
  183. package/dist/schema/state/sqlite/mod.d.ts +17 -0
  184. package/dist/schema/state/sqlite/mod.d.ts.map +1 -0
  185. package/dist/schema/state/sqlite/mod.js +41 -0
  186. package/dist/schema/state/sqlite/mod.js.map +1 -0
  187. package/dist/schema/state/sqlite/query-builder/api.d.ts +294 -0
  188. package/dist/schema/state/sqlite/query-builder/api.d.ts.map +1 -0
  189. package/dist/schema/state/sqlite/query-builder/api.js +6 -0
  190. package/dist/schema/state/sqlite/query-builder/api.js.map +1 -0
  191. package/dist/schema/state/sqlite/query-builder/astToSql.d.ts +7 -0
  192. package/dist/schema/state/sqlite/query-builder/astToSql.d.ts.map +1 -0
  193. package/dist/schema/state/sqlite/query-builder/astToSql.js +190 -0
  194. package/dist/schema/state/sqlite/query-builder/astToSql.js.map +1 -0
  195. package/dist/schema/state/sqlite/query-builder/impl.d.ts +7 -0
  196. package/dist/schema/state/sqlite/query-builder/impl.d.ts.map +1 -0
  197. package/dist/schema/state/sqlite/query-builder/impl.js +286 -0
  198. package/dist/schema/state/sqlite/query-builder/impl.js.map +1 -0
  199. package/dist/schema/state/sqlite/query-builder/impl.test.d.ts +87 -0
  200. package/dist/schema/state/sqlite/query-builder/impl.test.d.ts.map +1 -0
  201. package/dist/schema/state/sqlite/query-builder/impl.test.js +563 -0
  202. package/dist/schema/state/sqlite/query-builder/impl.test.js.map +1 -0
  203. package/dist/{query-builder → schema/state/sqlite/query-builder}/mod.d.ts +7 -0
  204. package/dist/schema/state/sqlite/query-builder/mod.d.ts.map +1 -0
  205. package/dist/{query-builder → schema/state/sqlite/query-builder}/mod.js +7 -0
  206. package/dist/schema/state/sqlite/query-builder/mod.js.map +1 -0
  207. package/dist/schema/state/sqlite/schema-helpers.d.ts.map +1 -0
  208. package/dist/schema/{schema-helpers.js → state/sqlite/schema-helpers.js} +1 -1
  209. package/dist/schema/state/sqlite/schema-helpers.js.map +1 -0
  210. package/dist/schema/state/sqlite/system-tables.d.ts +574 -0
  211. package/dist/schema/state/sqlite/system-tables.d.ts.map +1 -0
  212. package/dist/schema/state/sqlite/system-tables.js +88 -0
  213. package/dist/schema/state/sqlite/system-tables.js.map +1 -0
  214. package/dist/schema/state/sqlite/table-def.d.ts +84 -0
  215. package/dist/schema/state/sqlite/table-def.d.ts.map +1 -0
  216. package/dist/schema/state/sqlite/table-def.js +36 -0
  217. package/dist/schema/state/sqlite/table-def.js.map +1 -0
  218. package/dist/schema-management/common.d.ts +7 -7
  219. package/dist/schema-management/common.d.ts.map +1 -1
  220. package/dist/schema-management/common.js.map +1 -1
  221. package/dist/schema-management/migrations.d.ts +6 -6
  222. package/dist/schema-management/migrations.d.ts.map +1 -1
  223. package/dist/schema-management/migrations.js +27 -18
  224. package/dist/schema-management/migrations.js.map +1 -1
  225. package/dist/schema-management/validate-schema.d.ts +8 -0
  226. package/dist/schema-management/validate-schema.d.ts.map +1 -0
  227. package/dist/schema-management/validate-schema.js +39 -0
  228. package/dist/schema-management/validate-schema.js.map +1 -0
  229. package/dist/sql-queries/misc.d.ts.map +1 -1
  230. package/dist/sql-queries/sql-queries.d.ts +1 -1
  231. package/dist/sql-queries/sql-queries.d.ts.map +1 -1
  232. package/dist/sql-queries/sql-queries.js.map +1 -1
  233. package/dist/sql-queries/sql-query-builder.d.ts +1 -1
  234. package/dist/sql-queries/sql-query-builder.d.ts.map +1 -1
  235. package/dist/sql-queries/sql-query-builder.js.map +1 -1
  236. package/dist/sql-queries/types.d.ts +2 -1
  237. package/dist/sql-queries/types.d.ts.map +1 -1
  238. package/dist/sql-queries/types.js.map +1 -1
  239. package/dist/sync/ClientSessionSyncProcessor.d.ts +66 -0
  240. package/dist/sync/ClientSessionSyncProcessor.d.ts.map +1 -0
  241. package/dist/sync/ClientSessionSyncProcessor.js +209 -0
  242. package/dist/sync/ClientSessionSyncProcessor.js.map +1 -0
  243. package/dist/sync/index.d.ts +1 -1
  244. package/dist/sync/index.d.ts.map +1 -1
  245. package/dist/sync/index.js +1 -1
  246. package/dist/sync/index.js.map +1 -1
  247. package/dist/sync/next/compact-events.d.ts.map +1 -1
  248. package/dist/sync/next/compact-events.js +38 -35
  249. package/dist/sync/next/compact-events.js.map +1 -1
  250. package/dist/sync/next/facts.d.ts +21 -21
  251. package/dist/sync/next/facts.d.ts.map +1 -1
  252. package/dist/sync/next/facts.js +11 -11
  253. package/dist/sync/next/facts.js.map +1 -1
  254. package/dist/sync/next/history-dag-common.d.ts +9 -7
  255. package/dist/sync/next/history-dag-common.d.ts.map +1 -1
  256. package/dist/sync/next/history-dag-common.js +10 -5
  257. package/dist/sync/next/history-dag-common.js.map +1 -1
  258. package/dist/sync/next/history-dag.d.ts +0 -2
  259. package/dist/sync/next/history-dag.d.ts.map +1 -1
  260. package/dist/sync/next/history-dag.js +16 -14
  261. package/dist/sync/next/history-dag.js.map +1 -1
  262. package/dist/sync/next/rebase-events.d.ts +10 -8
  263. package/dist/sync/next/rebase-events.d.ts.map +1 -1
  264. package/dist/sync/next/rebase-events.js +18 -10
  265. package/dist/sync/next/rebase-events.js.map +1 -1
  266. package/dist/sync/next/test/compact-events.calculator.test.js +39 -34
  267. package/dist/sync/next/test/compact-events.calculator.test.js.map +1 -1
  268. package/dist/sync/next/test/compact-events.test.js +77 -77
  269. package/dist/sync/next/test/compact-events.test.js.map +1 -1
  270. package/dist/sync/next/test/{mutation-fixtures.d.ts → event-fixtures.d.ts} +38 -28
  271. package/dist/sync/next/test/event-fixtures.d.ts.map +1 -0
  272. package/dist/sync/next/test/{mutation-fixtures.js → event-fixtures.js} +81 -38
  273. package/dist/sync/next/test/event-fixtures.js.map +1 -0
  274. package/dist/sync/next/test/mod.d.ts +1 -1
  275. package/dist/sync/next/test/mod.d.ts.map +1 -1
  276. package/dist/sync/next/test/mod.js +1 -1
  277. package/dist/sync/next/test/mod.js.map +1 -1
  278. package/dist/sync/sync.d.ts +60 -25
  279. package/dist/sync/sync.d.ts.map +1 -1
  280. package/dist/sync/sync.js +10 -6
  281. package/dist/sync/sync.js.map +1 -1
  282. package/dist/sync/syncstate.d.ts +213 -82
  283. package/dist/sync/syncstate.d.ts.map +1 -1
  284. package/dist/sync/syncstate.js +337 -139
  285. package/dist/sync/syncstate.js.map +1 -1
  286. package/dist/sync/syncstate.test.js +310 -286
  287. package/dist/sync/syncstate.test.js.map +1 -1
  288. package/dist/sync/validate-push-payload.d.ts +2 -2
  289. package/dist/sync/validate-push-payload.d.ts.map +1 -1
  290. package/dist/sync/validate-push-payload.js +4 -4
  291. package/dist/sync/validate-push-payload.js.map +1 -1
  292. package/dist/util.d.ts +2 -2
  293. package/dist/util.d.ts.map +1 -1
  294. package/dist/version.d.ts +2 -2
  295. package/dist/version.d.ts.map +1 -1
  296. package/dist/version.js +2 -2
  297. package/dist/version.js.map +1 -1
  298. package/package.json +13 -6
  299. package/src/__tests__/fixture.ts +36 -15
  300. package/src/adapter-types.ts +107 -68
  301. package/src/debug-info.ts +1 -0
  302. package/src/devtools/devtools-messages-client-session.ts +142 -0
  303. package/src/devtools/devtools-messages-common.ts +115 -0
  304. package/src/devtools/devtools-messages-leader.ts +191 -0
  305. package/src/devtools/devtools-messages.ts +3 -246
  306. package/src/devtools/devtools-sessioninfo.ts +101 -0
  307. package/src/devtools/mod.ts +59 -0
  308. package/src/index.ts +7 -15
  309. package/src/leader-thread/LeaderSyncProcessor.ts +940 -0
  310. package/src/leader-thread/connection.ts +54 -9
  311. package/src/leader-thread/eventlog.ts +199 -0
  312. package/src/leader-thread/leader-worker-devtools.ts +227 -104
  313. package/src/leader-thread/make-leader-thread-layer.ts +128 -78
  314. package/src/leader-thread/materialize-event.ts +173 -0
  315. package/src/leader-thread/mod.ts +1 -1
  316. package/src/leader-thread/recreate-db.ts +38 -39
  317. package/src/leader-thread/shutdown-channel.ts +2 -4
  318. package/src/leader-thread/types.ts +96 -50
  319. package/src/make-client-session.ts +136 -0
  320. package/src/materializer-helper.ts +138 -0
  321. package/src/otel.ts +8 -0
  322. package/src/rematerialize-from-eventlog.ts +117 -0
  323. package/src/schema/EventDef.ts +227 -0
  324. package/src/schema/EventSequenceNumber.test.ts +12 -0
  325. package/src/schema/EventSequenceNumber.ts +121 -0
  326. package/src/schema/LiveStoreEvent.ts +240 -0
  327. package/src/schema/events.ts +1 -0
  328. package/src/schema/mod.ts +8 -6
  329. package/src/schema/schema.ts +88 -84
  330. package/src/schema/state/mod.ts +2 -0
  331. package/src/schema/state/sqlite/client-document-def.test.ts +238 -0
  332. package/src/schema/state/sqlite/client-document-def.ts +444 -0
  333. package/src/schema/state/sqlite/db-schema/ast/sqlite.ts +142 -0
  334. package/src/schema/state/sqlite/db-schema/ast/validate.ts +13 -0
  335. package/src/schema/state/sqlite/db-schema/dsl/__snapshots__/field-defs.test.ts.snap +206 -0
  336. package/src/schema/state/sqlite/db-schema/dsl/field-defs.test.ts +35 -0
  337. package/src/schema/state/sqlite/db-schema/dsl/field-defs.ts +242 -0
  338. package/src/schema/state/sqlite/db-schema/dsl/mod.ts +222 -0
  339. package/src/schema/state/sqlite/db-schema/hash.ts +14 -0
  340. package/src/schema/state/sqlite/db-schema/mod.ts +2 -0
  341. package/src/schema/state/sqlite/mod.ts +73 -0
  342. package/src/schema/state/sqlite/query-builder/api.ts +440 -0
  343. package/src/schema/state/sqlite/query-builder/astToSql.ts +232 -0
  344. package/src/schema/state/sqlite/query-builder/impl.test.ts +617 -0
  345. package/src/schema/state/sqlite/query-builder/impl.ts +351 -0
  346. package/src/{query-builder → schema/state/sqlite/query-builder}/mod.ts +7 -0
  347. package/src/schema/{schema-helpers.ts → state/sqlite/schema-helpers.ts} +1 -1
  348. package/src/schema/state/sqlite/system-tables.ts +117 -0
  349. package/src/schema/state/sqlite/table-def.ts +197 -0
  350. package/src/schema-management/common.ts +7 -7
  351. package/src/schema-management/migrations.ts +37 -31
  352. package/src/schema-management/validate-schema.ts +61 -0
  353. package/src/sql-queries/sql-queries.ts +1 -1
  354. package/src/sql-queries/sql-query-builder.ts +1 -2
  355. package/src/sql-queries/types.ts +3 -1
  356. package/src/sync/ClientSessionSyncProcessor.ts +332 -0
  357. package/src/sync/index.ts +1 -1
  358. package/src/sync/next/compact-events.ts +38 -35
  359. package/src/sync/next/facts.ts +43 -41
  360. package/src/sync/next/history-dag-common.ts +17 -10
  361. package/src/sync/next/history-dag.ts +16 -17
  362. package/src/sync/next/rebase-events.ts +29 -17
  363. package/src/sync/next/test/compact-events.calculator.test.ts +46 -46
  364. package/src/sync/next/test/compact-events.test.ts +79 -79
  365. package/src/sync/next/test/event-fixtures.ts +226 -0
  366. package/src/sync/next/test/mod.ts +1 -1
  367. package/src/sync/sync.ts +60 -24
  368. package/src/sync/syncstate.test.ts +347 -320
  369. package/src/sync/syncstate.ts +422 -230
  370. package/src/sync/validate-push-payload.ts +6 -6
  371. package/src/version.ts +2 -2
  372. package/dist/derived-mutations.d.ts +0 -109
  373. package/dist/derived-mutations.d.ts.map +0 -1
  374. package/dist/derived-mutations.js +0 -54
  375. package/dist/derived-mutations.js.map +0 -1
  376. package/dist/derived-mutations.test.d.ts +0 -2
  377. package/dist/derived-mutations.test.d.ts.map +0 -1
  378. package/dist/derived-mutations.test.js +0 -93
  379. package/dist/derived-mutations.test.js.map +0 -1
  380. package/dist/devtools/devtools-bridge.d.ts +0 -13
  381. package/dist/devtools/devtools-bridge.d.ts.map +0 -1
  382. package/dist/devtools/devtools-bridge.js +0 -2
  383. package/dist/devtools/devtools-bridge.js.map +0 -1
  384. package/dist/devtools/devtools-window-message.d.ts +0 -29
  385. package/dist/devtools/devtools-window-message.d.ts.map +0 -1
  386. package/dist/devtools/devtools-window-message.js +0 -33
  387. package/dist/devtools/devtools-window-message.js.map +0 -1
  388. package/dist/devtools/index.d.ts +0 -42
  389. package/dist/devtools/index.d.ts.map +0 -1
  390. package/dist/devtools/index.js +0 -48
  391. package/dist/devtools/index.js.map +0 -1
  392. package/dist/init-singleton-tables.d.ts +0 -4
  393. package/dist/init-singleton-tables.d.ts.map +0 -1
  394. package/dist/init-singleton-tables.js +0 -16
  395. package/dist/init-singleton-tables.js.map +0 -1
  396. package/dist/leader-thread/apply-mutation.d.ts +0 -8
  397. package/dist/leader-thread/apply-mutation.d.ts.map +0 -1
  398. package/dist/leader-thread/apply-mutation.js +0 -95
  399. package/dist/leader-thread/apply-mutation.js.map +0 -1
  400. package/dist/leader-thread/leader-sync-processor.d.ts +0 -47
  401. package/dist/leader-thread/leader-sync-processor.d.ts.map +0 -1
  402. package/dist/leader-thread/leader-sync-processor.js +0 -425
  403. package/dist/leader-thread/leader-sync-processor.js.map +0 -1
  404. package/dist/leader-thread/mutationlog.d.ts +0 -10
  405. package/dist/leader-thread/mutationlog.d.ts.map +0 -1
  406. package/dist/leader-thread/mutationlog.js +0 -28
  407. package/dist/leader-thread/mutationlog.js.map +0 -1
  408. package/dist/leader-thread/pull-queue-set.d.ts +0 -7
  409. package/dist/leader-thread/pull-queue-set.d.ts.map +0 -1
  410. package/dist/leader-thread/pull-queue-set.js +0 -39
  411. package/dist/leader-thread/pull-queue-set.js.map +0 -1
  412. package/dist/mutation.d.ts +0 -13
  413. package/dist/mutation.d.ts.map +0 -1
  414. package/dist/mutation.js +0 -57
  415. package/dist/mutation.js.map +0 -1
  416. package/dist/query-builder/api.d.ts +0 -190
  417. package/dist/query-builder/api.d.ts.map +0 -1
  418. package/dist/query-builder/api.js +0 -8
  419. package/dist/query-builder/api.js.map +0 -1
  420. package/dist/query-builder/impl.d.ts +0 -12
  421. package/dist/query-builder/impl.d.ts.map +0 -1
  422. package/dist/query-builder/impl.js +0 -244
  423. package/dist/query-builder/impl.js.map +0 -1
  424. package/dist/query-builder/impl.test.d.ts +0 -2
  425. package/dist/query-builder/impl.test.d.ts.map +0 -1
  426. package/dist/query-builder/impl.test.js +0 -212
  427. package/dist/query-builder/impl.test.js.map +0 -1
  428. package/dist/query-builder/mod.d.ts.map +0 -1
  429. package/dist/query-builder/mod.js.map +0 -1
  430. package/dist/query-info.d.ts +0 -38
  431. package/dist/query-info.d.ts.map +0 -1
  432. package/dist/query-info.js +0 -7
  433. package/dist/query-info.js.map +0 -1
  434. package/dist/rehydrate-from-mutationlog.d.ts +0 -14
  435. package/dist/rehydrate-from-mutationlog.d.ts.map +0 -1
  436. package/dist/rehydrate-from-mutationlog.js +0 -72
  437. package/dist/rehydrate-from-mutationlog.js.map +0 -1
  438. package/dist/schema/MutationEvent.d.ts +0 -166
  439. package/dist/schema/MutationEvent.d.ts.map +0 -1
  440. package/dist/schema/MutationEvent.js +0 -72
  441. package/dist/schema/MutationEvent.js.map +0 -1
  442. package/dist/schema/mutations.d.ts +0 -107
  443. package/dist/schema/mutations.d.ts.map +0 -1
  444. package/dist/schema/mutations.js +0 -42
  445. package/dist/schema/mutations.js.map +0 -1
  446. package/dist/schema/schema-helpers.d.ts.map +0 -1
  447. package/dist/schema/schema-helpers.js.map +0 -1
  448. package/dist/schema/system-tables.d.ts +0 -399
  449. package/dist/schema/system-tables.d.ts.map +0 -1
  450. package/dist/schema/system-tables.js +0 -58
  451. package/dist/schema/system-tables.js.map +0 -1
  452. package/dist/schema/table-def.d.ts +0 -156
  453. package/dist/schema/table-def.d.ts.map +0 -1
  454. package/dist/schema/table-def.js +0 -79
  455. package/dist/schema/table-def.js.map +0 -1
  456. package/dist/schema-management/validate-mutation-defs.d.ts +0 -8
  457. package/dist/schema-management/validate-mutation-defs.d.ts.map +0 -1
  458. package/dist/schema-management/validate-mutation-defs.js +0 -39
  459. package/dist/schema-management/validate-mutation-defs.js.map +0 -1
  460. package/dist/sync/client-session-sync-processor.d.ts +0 -45
  461. package/dist/sync/client-session-sync-processor.d.ts.map +0 -1
  462. package/dist/sync/client-session-sync-processor.js +0 -131
  463. package/dist/sync/client-session-sync-processor.js.map +0 -1
  464. package/dist/sync/next/test/mutation-fixtures.d.ts.map +0 -1
  465. package/dist/sync/next/test/mutation-fixtures.js.map +0 -1
  466. package/src/derived-mutations.test.ts +0 -101
  467. package/src/derived-mutations.ts +0 -166
  468. package/src/devtools/devtools-bridge.ts +0 -14
  469. package/src/devtools/devtools-window-message.ts +0 -27
  470. package/src/devtools/index.ts +0 -48
  471. package/src/init-singleton-tables.ts +0 -24
  472. package/src/leader-thread/apply-mutation.ts +0 -143
  473. package/src/leader-thread/leader-sync-processor.ts +0 -670
  474. package/src/leader-thread/mutationlog.ts +0 -46
  475. package/src/leader-thread/pull-queue-set.ts +0 -58
  476. package/src/mutation.ts +0 -81
  477. package/src/query-builder/api.ts +0 -289
  478. package/src/query-builder/impl.test.ts +0 -239
  479. package/src/query-builder/impl.ts +0 -285
  480. package/src/query-info.ts +0 -78
  481. package/src/rehydrate-from-mutationlog.ts +0 -127
  482. package/src/schema/EventId.ts +0 -60
  483. package/src/schema/MutationEvent.ts +0 -180
  484. package/src/schema/mutations.ts +0 -192
  485. package/src/schema/system-tables.ts +0 -104
  486. package/src/schema/table-def.ts +0 -343
  487. package/src/schema-management/validate-mutation-defs.ts +0 -63
  488. package/src/sync/client-session-sync-processor.ts +0 -207
  489. package/src/sync/next/test/mutation-fixtures.ts +0 -224
  490. package/tsconfig.json +0 -11
  491. /package/dist/schema/{schema-helpers.d.ts → state/sqlite/schema-helpers.d.ts} +0 -0
@@ -1,11 +1,39 @@
1
1
  import { Effect } from '@livestore/utils/effect';
2
- import type { SynchronousDatabase } from '../adapter-types.js';
2
+ import type { SqliteDb } from '../adapter-types.js';
3
3
  import { SqliteError } from '../adapter-types.js';
4
4
  import type { BindValues } from '../sql-queries/index.js';
5
5
  import type { PreparedBindValues } from '../util.js';
6
- export declare const configureConnection: (syncDb: SynchronousDatabase, { fkEnabled }: {
7
- fkEnabled: boolean;
8
- }) => Effect.Effect<void, SqliteError, never>;
9
- export declare const execSql: (syncDb: SynchronousDatabase, sql: string, bind: BindValues) => Effect.Effect<void, SqliteError, never>;
10
- export declare const execSqlPrepared: (syncDb: SynchronousDatabase, sql: string, bindValues: PreparedBindValues) => Effect.Effect<void, SqliteError, never>;
6
+ type ConnectionOptions = {
7
+ /**
8
+ * The database connection locking mode.
9
+ *
10
+ * @remarks
11
+ *
12
+ * This **option is ignored** when used on an **in-memory database** as they can only operate in exclusive locking mode.
13
+ * In-memory databases can’t share state between connections (unless using a
14
+ * {@link https://www.sqlite.org/sharedcache.html#shared_cache_and_in_memory_databases|shared cache}),
15
+ * making concurrent access impossible. This is functionally equivalent to exclusive locking.
16
+ *
17
+ * @defaultValue
18
+ * The default is `"NORMAL"` unless it was unless overridden at compile-time using `SQLITE_DEFAULT_LOCKING_MODE`.
19
+ *
20
+ * @see {@link https://www.sqlite.org/pragma.html#pragma_locking_mode|`locking_mode` pragma}
21
+ */
22
+ lockingMode?: 'NORMAL' | 'EXCLUSIVE';
23
+ /**
24
+ * Whether to enforce foreign key constraints.
25
+ *
26
+ * @privateRemarks
27
+ *
28
+ * We require a value for this option to minimize future problems, as the default value might change in future
29
+ * versions of SQLite.
30
+ *
31
+ * @see {@link https://www.sqlite.org/pragma.html#pragma_foreign_keys|`foreign_keys` pragma}
32
+ */
33
+ foreignKeys: boolean;
34
+ };
35
+ export declare const configureConnection: (sqliteDb: SqliteDb, { foreignKeys, lockingMode }: ConnectionOptions) => Effect.Effect<void, SqliteError, never>;
36
+ export declare const execSql: (sqliteDb: SqliteDb, sql: string, bind: BindValues) => Effect.Effect<void, SqliteError, never>;
37
+ export declare const execSqlPrepared: (sqliteDb: SqliteDb, sql: string, bindValues: PreparedBindValues) => Effect.Effect<void, SqliteError, never>;
38
+ export {};
11
39
  //# sourceMappingURL=connection.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"connection.d.ts","sourceRoot":"","sources":["../../src/leader-thread/connection.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAEhD,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,qBAAqB,CAAA;AAC9D,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAA;AACjD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAA;AACzD,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAQpD,eAAO,MAAM,mBAAmB,WAAY,mBAAmB,iBAAiB;IAAE,SAAS,EAAE,OAAO,CAAA;CAAE,4CASnG,CAAA;AAEH,eAAO,MAAM,OAAO,WAAY,mBAAmB,OAAO,MAAM,QAAQ,UAAU,4CAajF,CAAA;AAYD,eAAO,MAAM,eAAe,WAAY,mBAAmB,OAAO,MAAM,cAAc,kBAAkB,4CAgBvG,CAAA"}
1
+ {"version":3,"file":"connection.d.ts","sourceRoot":"","sources":["../../src/leader-thread/connection.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAEhD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAA;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAA;AACjD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAA;AACzD,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAQpD,KAAK,iBAAiB,GAAG;IACvB;;;;;;;;;;;;;;OAcG;IACH,WAAW,CAAC,EAAE,QAAQ,GAAG,WAAW,CAAA;IAEpC;;;;;;;;;OASG;IACH,WAAW,EAAE,OAAO,CAAA;CACrB,CAAA;AAED,eAAO,MAAM,mBAAmB,GAAI,UAAU,QAAQ,EAAE,8BAA8B,iBAAiB,4CAuBpG,CAAA;AAEH,eAAO,MAAM,OAAO,GAAI,UAAU,QAAQ,EAAE,KAAK,MAAM,EAAE,MAAM,UAAU,4CAaxE,CAAA;AAYD,eAAO,MAAM,eAAe,GAAI,UAAU,QAAQ,EAAE,KAAK,MAAM,EAAE,YAAY,kBAAkB,4CAgB9F,CAAA"}
@@ -2,15 +2,30 @@
2
2
  import { Effect } from '@livestore/utils/effect';
3
3
  import { SqliteError } from '../adapter-types.js';
4
4
  import { prepareBindValues, sql } from '../util.js';
5
- export const configureConnection = (syncDb, { fkEnabled }) => execSql(syncDb, sql `
5
+ export const configureConnection = (sqliteDb, { foreignKeys, lockingMode }) => execSql(sqliteDb,
6
+ // We use the WAL journal mode is significantly faster in most scenarios than the traditional rollback journal mode.
7
+ // It specifically significantly improves write performance. However, when using the WAL journal mode, transactions
8
+ // that involve changes against multiple ATTACHed databases are atomic for each database but are not atomic
9
+ // across all databases as a set. Additionally, it is not possible to change the page size after entering WAL mode,
10
+ // whether on an empty database or by using VACUUM or the backup API. To change the page size, we must switch to the
11
+ // rollback journal mode.
12
+ //
13
+ // When connected to an in-memory database, the WAL journal mode option is ignored because an in-memory database can
14
+ // only be in either the MEMORY or OFF options. By default, an in-memory database is in the MEMORY option, which
15
+ // means that it stores the rollback journal in volatile RAM. This saves disk I/O but at the expense of safety and
16
+ // integrity. If the thread using SQLite crashes in the middle of a transaction, then the database file will very
17
+ // likely go corrupt.
18
+ sql `
19
+ -- disable WAL until we have it working properly
20
+ -- PRAGMA journal_mode=WAL;
6
21
  PRAGMA page_size=8192;
7
- PRAGMA journal_mode=MEMORY;
8
- ${fkEnabled ? sql `PRAGMA foreign_keys='ON';` : sql `PRAGMA foreign_keys='OFF';`}
22
+ PRAGMA foreign_keys=${foreignKeys ? 'ON' : 'OFF'};
23
+ ${lockingMode === undefined ? '' : sql `PRAGMA locking_mode=${lockingMode};`}
9
24
  `, {});
10
- export const execSql = (syncDb, sql, bind) => {
25
+ export const execSql = (sqliteDb, sql, bind) => {
11
26
  const bindValues = prepareBindValues(bind, sql);
12
27
  return Effect.try({
13
- try: () => syncDb.execute(sql, bindValues),
28
+ try: () => sqliteDb.execute(sql, bindValues),
14
29
  catch: (cause) => new SqliteError({ cause, query: { bindValues, sql }, code: cause.code }),
15
30
  }).pipe(Effect.asVoid,
16
31
  // Effect.logDuration(`@livestore/common:execSql:${sql}`),
@@ -27,9 +42,9 @@ export const execSql = (syncDb, sql, bind) => {
27
42
  // })
28
43
  // }
29
44
  // TODO actually use prepared statements
30
- export const execSqlPrepared = (syncDb, sql, bindValues) => {
45
+ export const execSqlPrepared = (sqliteDb, sql, bindValues) => {
31
46
  return Effect.try({
32
- try: () => syncDb.execute(sql, bindValues),
47
+ try: () => sqliteDb.execute(sql, bindValues),
33
48
  catch: (cause) => new SqliteError({ cause, query: { bindValues, sql }, code: cause.code }),
34
49
  }).pipe(Effect.asVoid,
35
50
  // Effect.logDuration(`@livestore/common:execSqlPrepared:${sql}`),
@@ -1 +1 @@
1
- {"version":3,"file":"connection.js","sourceRoot":"","sources":["../../src/leader-thread/connection.ts"],"names":[],"mappings":"AAAA,yDAAyD;AACzD,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAGhD,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAA;AAGjD,OAAO,EAAE,iBAAiB,EAAE,GAAG,EAAE,MAAM,YAAY,CAAA;AAOnD,MAAM,CAAC,MAAM,mBAAmB,GAAG,CAAC,MAA2B,EAAE,EAAE,SAAS,EAA0B,EAAE,EAAE,CACxG,OAAO,CACL,MAAM,EACN,GAAG,CAAA;;;MAGD,SAAS,CAAC,CAAC,CAAC,GAAG,CAAA,2BAA2B,CAAC,CAAC,CAAC,GAAG,CAAA,4BAA4B;GAC/E,EACC,EAAE,CACH,CAAA;AAEH,MAAM,CAAC,MAAM,OAAO,GAAG,CAAC,MAA2B,EAAE,GAAW,EAAE,IAAgB,EAAE,EAAE;IACpF,MAAM,UAAU,GAAG,iBAAiB,CAAC,IAAI,EAAE,GAAG,CAAC,CAAA;IAC/C,OAAO,MAAM,CAAC,GAAG,CAAC;QAChB,GAAG,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,GAAG,EAAE,UAAU,CAAC;QAC1C,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE,CACf,IAAI,WAAW,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,UAAU,EAAE,GAAG,EAAE,EAAE,IAAI,EAAG,KAA8B,CAAC,IAAI,EAAE,CAAC;KACrG,CAAC,CAAC,IAAI,CACL,MAAM,CAAC,MAAM;IACb,0DAA0D;IAC1D,MAAM,CAAC,QAAQ,CAAC,2BAA2B,EAAE;QAC3C,UAAU,EAAE,EAAE,YAAY,EAAE,GAAG,EAAE,GAAG,EAAE,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE;KAC/E,CAAC,CACH,CAAA;AACH,CAAC,CAAA;AAED,gFAAgF;AAChF,yDAAyD;AACzD,wBAAwB;AACxB,6CAA6C;AAC7C,wBAAwB;AACxB,sHAAsH;AACtH,OAAO;AACP,IAAI;AAEJ,wCAAwC;AACxC,MAAM,CAAC,MAAM,eAAe,GAAG,CAAC,MAA2B,EAAE,GAAW,EAAE,UAA8B,EAAE,EAAE;IAC1G,OAAO,MAAM,CAAC,GAAG,CAAC;QAChB,GAAG,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,GAAG,EAAE,UAAU,CAAC;QAC1C,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE,CACf,IAAI,WAAW,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,UAAU,EAAE,GAAG,EAAE,EAAE,IAAI,EAAG,KAA8B,CAAC,IAAI,EAAE,CAAC;KACrG,CAAC,CAAC,IAAI,CACL,MAAM,CAAC,MAAM;IACb,kEAAkE;IAClE,MAAM,CAAC,QAAQ,CAAC,mCAAmC,EAAE;QACnD,UAAU,EAAE;YACV,YAAY,EAAE,GAAG;YACjB,GAAG;YACH,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC;SACvC;KACF,CAAC,CACH,CAAA;AACH,CAAC,CAAA"}
1
+ {"version":3,"file":"connection.js","sourceRoot":"","sources":["../../src/leader-thread/connection.ts"],"names":[],"mappings":"AAAA,yDAAyD;AACzD,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAGhD,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAA;AAGjD,OAAO,EAAE,iBAAiB,EAAE,GAAG,EAAE,MAAM,YAAY,CAAA;AAsCnD,MAAM,CAAC,MAAM,mBAAmB,GAAG,CAAC,QAAkB,EAAE,EAAE,WAAW,EAAE,WAAW,EAAqB,EAAE,EAAE,CACzG,OAAO,CACL,QAAQ;AACR,oHAAoH;AACpH,mHAAmH;AACnH,2GAA2G;AAC3G,mHAAmH;AACnH,oHAAoH;AACpH,yBAAyB;AACzB,EAAE;AACF,oHAAoH;AACpH,gHAAgH;AAChH,kHAAkH;AAClH,iHAAiH;AACjH,qBAAqB;AACrB,GAAG,CAAA;;;;0BAImB,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK;MAC9C,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAA,uBAAuB,WAAW,GAAG;GAC5E,EACC,EAAE,CACH,CAAA;AAEH,MAAM,CAAC,MAAM,OAAO,GAAG,CAAC,QAAkB,EAAE,GAAW,EAAE,IAAgB,EAAE,EAAE;IAC3E,MAAM,UAAU,GAAG,iBAAiB,CAAC,IAAI,EAAE,GAAG,CAAC,CAAA;IAC/C,OAAO,MAAM,CAAC,GAAG,CAAC;QAChB,GAAG,EAAE,GAAG,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,UAAU,CAAC;QAC5C,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE,CACf,IAAI,WAAW,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,UAAU,EAAE,GAAG,EAAE,EAAE,IAAI,EAAG,KAA8B,CAAC,IAAI,EAAE,CAAC;KACrG,CAAC,CAAC,IAAI,CACL,MAAM,CAAC,MAAM;IACb,0DAA0D;IAC1D,MAAM,CAAC,QAAQ,CAAC,2BAA2B,EAAE;QAC3C,UAAU,EAAE,EAAE,YAAY,EAAE,GAAG,EAAE,GAAG,EAAE,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE;KAC/E,CAAC,CACH,CAAA;AACH,CAAC,CAAA;AAED,gFAAgF;AAChF,yDAAyD;AACzD,wBAAwB;AACxB,6CAA6C;AAC7C,wBAAwB;AACxB,sHAAsH;AACtH,OAAO;AACP,IAAI;AAEJ,wCAAwC;AACxC,MAAM,CAAC,MAAM,eAAe,GAAG,CAAC,QAAkB,EAAE,GAAW,EAAE,UAA8B,EAAE,EAAE;IACjG,OAAO,MAAM,CAAC,GAAG,CAAC;QAChB,GAAG,EAAE,GAAG,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,UAAU,CAAC;QAC5C,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE,CACf,IAAI,WAAW,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,UAAU,EAAE,GAAG,EAAE,EAAE,IAAI,EAAG,KAA8B,CAAC,IAAI,EAAE,CAAC;KACrG,CAAC,CAAC,IAAI,CACL,MAAM,CAAC,MAAM;IACb,kEAAkE;IAClE,MAAM,CAAC,QAAQ,CAAC,mCAAmC,EAAE;QACnD,UAAU,EAAE;YACV,YAAY,EAAE,GAAG;YACjB,GAAG;YACH,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC;SACvC;KACF,CAAC,CACH,CAAA;AACH,CAAC,CAAA"}
@@ -0,0 +1,27 @@
1
+ import { Effect, Option, Schema } from '@livestore/utils/effect';
2
+ import type { SqliteDb } from '../adapter-types.js';
3
+ import * as EventSequenceNumber from '../schema/EventSequenceNumber.js';
4
+ import * as LiveStoreEvent from '../schema/LiveStoreEvent.js';
5
+ import { LeaderThreadCtx } from './types.js';
6
+ export declare const initEventlogDb: (dbEventlog: SqliteDb) => Effect.Effect<void, import("../adapter-types.js").SqliteError, never>;
7
+ /** Exclusive of the "since event" */
8
+ export declare const getEventsSince: (since: EventSequenceNumber.EventSequenceNumber) => Effect.Effect<ReadonlyArray<LiveStoreEvent.EncodedWithMeta>, never, LeaderThreadCtx>;
9
+ export declare const getClientHeadFromDb: (dbEventlog: SqliteDb) => EventSequenceNumber.EventSequenceNumber;
10
+ export declare const getBackendHeadFromDb: (dbEventlog: SqliteDb) => EventSequenceNumber.GlobalEventSequenceNumber;
11
+ export declare const updateBackendHead: (dbEventlog: SqliteDb, head: EventSequenceNumber.EventSequenceNumber) => void;
12
+ export declare const insertIntoEventlog: (eventEncoded: LiveStoreEvent.EncodedWithMeta, dbEventlog: SqliteDb, eventDefSchemaHash: number, clientId: string, sessionId: string) => Effect.Effect<void, import("../adapter-types.js").SqliteError, never>;
13
+ export declare const updateSyncMetadata: (items: ReadonlyArray<LiveStoreEvent.EncodedWithMeta>) => Effect.Effect<void, import("../adapter-types.js").SqliteError, LeaderThreadCtx>;
14
+ export declare const getSyncBackendCursorInfo: (remoteHead: EventSequenceNumber.GlobalEventSequenceNumber) => Effect.Effect<Option.None<{
15
+ cursor: {
16
+ global: EventSequenceNumber.GlobalEventSequenceNumber;
17
+ client: EventSequenceNumber.ClientEventSequenceNumber;
18
+ };
19
+ metadata: Option.Option<Schema.JsonValue>;
20
+ }> | Option.Some<{
21
+ cursor: {
22
+ global: EventSequenceNumber.GlobalEventSequenceNumber;
23
+ client: EventSequenceNumber.ClientEventSequenceNumber;
24
+ };
25
+ metadata: Option.Option<Schema.JsonValue>;
26
+ }>, never, LeaderThreadCtx>;
27
+ //# sourceMappingURL=eventlog.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eventlog.d.ts","sourceRoot":"","sources":["../../src/leader-thread/eventlog.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAEhE,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAA;AACnD,OAAO,KAAK,mBAAmB,MAAM,kCAAkC,CAAA;AACvE,OAAO,KAAK,cAAc,MAAM,6BAA6B,CAAA;AAc7D,OAAO,EAAE,eAAe,EAAE,MAAM,YAAY,CAAA;AAE5C,eAAO,MAAM,cAAc,GAAI,YAAY,QAAQ,0EAmB/C,CAAA;AAEJ,qCAAqC;AACrC,eAAO,MAAM,cAAc,GACzB,OAAO,mBAAmB,CAAC,mBAAmB,KAC7C,MAAM,CAAC,MAAM,CAAC,aAAa,CAAC,cAAc,CAAC,eAAe,CAAC,EAAE,KAAK,EAAE,eAAe,CA8ClF,CAAA;AAEJ,eAAO,MAAM,mBAAmB,GAAI,YAAY,QAAQ,KAAG,mBAAmB,CAAC,mBAS9E,CAAA;AAED,eAAO,MAAM,oBAAoB,GAAI,YAAY,QAAQ,KAAG,mBAAmB,CAAC,yBAGjC,CAAA;AAG/C,eAAO,MAAM,iBAAiB,GAAI,YAAY,QAAQ,EAAE,MAAM,mBAAmB,CAAC,mBAAmB,SACrB,CAAA;AAEhF,eAAO,MAAM,kBAAkB,GAC7B,cAAc,cAAc,CAAC,eAAe,EAC5C,YAAY,QAAQ,EACpB,oBAAoB,MAAM,EAC1B,UAAU,MAAM,EAChB,WAAW,MAAM,0EAsCf,CAAA;AAEJ,eAAO,MAAM,kBAAkB,GAAI,OAAO,aAAa,CAAC,cAAc,CAAC,eAAe,CAAC,oFAkBnF,CAAA;AAEJ,eAAO,MAAM,wBAAwB,GAAI,YAAY,mBAAmB,CAAC,yBAAyB;;;;;;;;;;;;2BAoBe,CAAA"}
@@ -0,0 +1,119 @@
1
+ import { LS_DEV, shouldNeverHappen } from '@livestore/utils';
2
+ import { Effect, Option, Schema } from '@livestore/utils/effect';
3
+ import * as EventSequenceNumber from '../schema/EventSequenceNumber.js';
4
+ import * as LiveStoreEvent from '../schema/LiveStoreEvent.js';
5
+ import { EVENTLOG_META_TABLE, eventlogMetaTable, eventlogSystemTables, sessionChangesetMetaTable, SYNC_STATUS_TABLE, } from '../schema/state/sqlite/system-tables.js';
6
+ import { migrateTable } from '../schema-management/migrations.js';
7
+ import { insertRow, updateRows } from '../sql-queries/sql-queries.js';
8
+ import { prepareBindValues, sql } from '../util.js';
9
+ import { execSql } from './connection.js';
10
+ import { LeaderThreadCtx } from './types.js';
11
+ export const initEventlogDb = (dbEventlog) => Effect.gen(function* () {
12
+ for (const tableDef of eventlogSystemTables) {
13
+ yield* migrateTable({
14
+ db: dbEventlog,
15
+ behaviour: 'create-if-not-exists',
16
+ tableAst: tableDef.sqliteDef.ast,
17
+ skipMetaTable: true,
18
+ });
19
+ }
20
+ // Create sync status row if it doesn't exist
21
+ yield* execSql(dbEventlog, sql `INSERT INTO ${SYNC_STATUS_TABLE} (head)
22
+ SELECT ${EventSequenceNumber.ROOT.global}
23
+ WHERE NOT EXISTS (SELECT 1 FROM ${SYNC_STATUS_TABLE})`, {});
24
+ });
25
+ /** Exclusive of the "since event" */
26
+ export const getEventsSince = (since) => Effect.gen(function* () {
27
+ const { dbEventlog, dbState } = yield* LeaderThreadCtx;
28
+ const query = eventlogMetaTable.where('seqNumGlobal', '>=', since.global).asSql();
29
+ const pendingEventsRaw = dbEventlog.select(query.query, prepareBindValues(query.bindValues, query.query));
30
+ const pendingEvents = Schema.decodeUnknownSync(eventlogMetaTable.rowSchema.pipe(Schema.Array))(pendingEventsRaw);
31
+ const sessionChangesetRows = sessionChangesetMetaTable.where('seqNumGlobal', '>=', since.global).asSql();
32
+ const sessionChangesetRowsRaw = dbState.select(sessionChangesetRows.query, prepareBindValues(sessionChangesetRows.bindValues, sessionChangesetRows.query));
33
+ const sessionChangesetRowsDecoded = Schema.decodeUnknownSync(sessionChangesetMetaTable.rowSchema.pipe(Schema.Array))(sessionChangesetRowsRaw);
34
+ return pendingEvents
35
+ .map((eventlogEvent) => {
36
+ const sessionChangeset = sessionChangesetRowsDecoded.find((readModelEvent) => readModelEvent.seqNumGlobal === eventlogEvent.seqNumGlobal &&
37
+ readModelEvent.seqNumClient === eventlogEvent.seqNumClient);
38
+ return LiveStoreEvent.EncodedWithMeta.make({
39
+ name: eventlogEvent.name,
40
+ args: eventlogEvent.argsJson,
41
+ seqNum: { global: eventlogEvent.seqNumGlobal, client: eventlogEvent.seqNumClient },
42
+ parentSeqNum: { global: eventlogEvent.parentSeqNumGlobal, client: eventlogEvent.parentSeqNumClient },
43
+ clientId: eventlogEvent.clientId,
44
+ sessionId: eventlogEvent.sessionId,
45
+ meta: {
46
+ sessionChangeset: sessionChangeset && sessionChangeset.changeset !== null
47
+ ? {
48
+ _tag: 'sessionChangeset',
49
+ data: sessionChangeset.changeset,
50
+ debug: sessionChangeset.debug,
51
+ }
52
+ : { _tag: 'unset' },
53
+ syncMetadata: eventlogEvent.syncMetadataJson,
54
+ },
55
+ });
56
+ })
57
+ .filter((_) => EventSequenceNumber.compare(_.seqNum, since) > 0)
58
+ .sort((a, b) => EventSequenceNumber.compare(a.seqNum, b.seqNum));
59
+ });
60
+ export const getClientHeadFromDb = (dbEventlog) => {
61
+ const res = dbEventlog.select(sql `select seqNumGlobal, seqNumClient from ${EVENTLOG_META_TABLE} order by seqNumGlobal DESC, seqNumClient DESC limit 1`)[0];
62
+ return res ? { global: res.seqNumGlobal, client: res.seqNumClient } : EventSequenceNumber.ROOT;
63
+ };
64
+ export const getBackendHeadFromDb = (dbEventlog) => dbEventlog.select(sql `select head from ${SYNC_STATUS_TABLE}`)[0]?.head ?? EventSequenceNumber.ROOT.global;
65
+ // TODO use prepared statements
66
+ export const updateBackendHead = (dbEventlog, head) => dbEventlog.execute(sql `UPDATE ${SYNC_STATUS_TABLE} SET head = ${head.global}`);
67
+ export const insertIntoEventlog = (eventEncoded, dbEventlog, eventDefSchemaHash, clientId, sessionId) => Effect.gen(function* () {
68
+ // Check history consistency during LS_DEV
69
+ if (LS_DEV && eventEncoded.parentSeqNum.global !== EventSequenceNumber.ROOT.global) {
70
+ const parentEventExists = dbEventlog.select(`SELECT COUNT(*) as count FROM ${EVENTLOG_META_TABLE} WHERE seqNumGlobal = ? AND seqNumClient = ?`, [eventEncoded.parentSeqNum.global, eventEncoded.parentSeqNum.client])[0].count === 1;
71
+ if (parentEventExists === false) {
72
+ shouldNeverHappen(`Parent mutation ${eventEncoded.parentSeqNum.global},${eventEncoded.parentSeqNum.client} does not exist`);
73
+ }
74
+ }
75
+ // TODO use prepared statements
76
+ yield* execSql(dbEventlog, ...insertRow({
77
+ tableName: EVENTLOG_META_TABLE,
78
+ columns: eventlogMetaTable.sqliteDef.columns,
79
+ values: {
80
+ seqNumGlobal: eventEncoded.seqNum.global,
81
+ seqNumClient: eventEncoded.seqNum.client,
82
+ parentSeqNumGlobal: eventEncoded.parentSeqNum.global,
83
+ parentSeqNumClient: eventEncoded.parentSeqNum.client,
84
+ name: eventEncoded.name,
85
+ argsJson: eventEncoded.args ?? {},
86
+ clientId,
87
+ sessionId,
88
+ schemaHash: eventDefSchemaHash,
89
+ syncMetadataJson: eventEncoded.meta.syncMetadata,
90
+ },
91
+ }));
92
+ });
93
+ export const updateSyncMetadata = (items) => Effect.gen(function* () {
94
+ const { dbEventlog } = yield* LeaderThreadCtx;
95
+ // TODO try to do this in a single query
96
+ for (let i = 0; i < items.length; i++) {
97
+ const event = items[i];
98
+ yield* execSql(dbEventlog, ...updateRows({
99
+ tableName: EVENTLOG_META_TABLE,
100
+ columns: eventlogMetaTable.sqliteDef.columns,
101
+ where: { seqNumGlobal: event.seqNum.global, seqNumClient: event.seqNum.client },
102
+ updateValues: { syncMetadataJson: event.meta.syncMetadata },
103
+ }));
104
+ }
105
+ });
106
+ export const getSyncBackendCursorInfo = (remoteHead) => Effect.gen(function* () {
107
+ const { dbEventlog } = yield* LeaderThreadCtx;
108
+ if (remoteHead === EventSequenceNumber.ROOT.global)
109
+ return Option.none();
110
+ const EventlogQuerySchema = Schema.Struct({
111
+ syncMetadataJson: Schema.parseJson(Schema.Option(Schema.JsonValue)),
112
+ }).pipe(Schema.pluck('syncMetadataJson'), Schema.Array, Schema.head);
113
+ const syncMetadataOption = yield* Effect.sync(() => dbEventlog.select(sql `SELECT syncMetadataJson FROM ${EVENTLOG_META_TABLE} WHERE seqNumGlobal = ${remoteHead} ORDER BY seqNumClient ASC LIMIT 1`)).pipe(Effect.andThen(Schema.decode(EventlogQuerySchema)), Effect.map(Option.flatten), Effect.orDie);
114
+ return Option.some({
115
+ cursor: { global: remoteHead, client: EventSequenceNumber.clientDefault },
116
+ metadata: syncMetadataOption,
117
+ });
118
+ }).pipe(Effect.withSpan('@livestore/common:eventlog:getSyncBackendCursorInfo', { attributes: { remoteHead } }));
119
+ //# sourceMappingURL=eventlog.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eventlog.js","sourceRoot":"","sources":["../../src/leader-thread/eventlog.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAA;AAC5D,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAGhE,OAAO,KAAK,mBAAmB,MAAM,kCAAkC,CAAA;AACvE,OAAO,KAAK,cAAc,MAAM,6BAA6B,CAAA;AAC7D,OAAO,EACL,mBAAmB,EACnB,iBAAiB,EACjB,oBAAoB,EACpB,yBAAyB,EACzB,iBAAiB,GAClB,MAAM,yCAAyC,CAAA;AAChD,OAAO,EAAE,YAAY,EAAE,MAAM,oCAAoC,CAAA;AACjE,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,+BAA+B,CAAA;AAErE,OAAO,EAAE,iBAAiB,EAAE,GAAG,EAAE,MAAM,YAAY,CAAA;AACnD,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB,CAAA;AAEzC,OAAO,EAAE,eAAe,EAAE,MAAM,YAAY,CAAA;AAE5C,MAAM,CAAC,MAAM,cAAc,GAAG,CAAC,UAAoB,EAAE,EAAE,CACrD,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAClB,KAAK,MAAM,QAAQ,IAAI,oBAAoB,EAAE,CAAC;QAC5C,KAAK,CAAC,CAAC,YAAY,CAAC;YAClB,EAAE,EAAE,UAAU;YACd,SAAS,EAAE,sBAAsB;YACjC,QAAQ,EAAE,QAAQ,CAAC,SAAS,CAAC,GAAG;YAChC,aAAa,EAAE,IAAI;SACpB,CAAC,CAAA;IACJ,CAAC;IAED,6CAA6C;IAC7C,KAAK,CAAC,CAAC,OAAO,CACZ,UAAU,EACV,GAAG,CAAA,eAAe,iBAAiB;mBACtB,mBAAmB,CAAC,IAAI,CAAC,MAAM;4CACN,iBAAiB,GAAG,EAC1D,EAAE,CACH,CAAA;AACH,CAAC,CAAC,CAAA;AAEJ,qCAAqC;AACrC,MAAM,CAAC,MAAM,cAAc,GAAG,CAC5B,KAA8C,EACwC,EAAE,CACxF,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAClB,MAAM,EAAE,UAAU,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,CAAC,eAAe,CAAA;IAEtD,MAAM,KAAK,GAAG,iBAAiB,CAAC,KAAK,CAAC,cAAc,EAAE,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,CAAA;IACjF,MAAM,gBAAgB,GAAG,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,EAAE,iBAAiB,CAAC,KAAK,CAAC,UAAU,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAA;IACzG,MAAM,aAAa,GAAG,MAAM,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAA;IAEhH,MAAM,oBAAoB,GAAG,yBAAyB,CAAC,KAAK,CAAC,cAAc,EAAE,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,CAAA;IACxG,MAAM,uBAAuB,GAAG,OAAO,CAAC,MAAM,CAC5C,oBAAoB,CAAC,KAAK,EAC1B,iBAAiB,CAAC,oBAAoB,CAAC,UAAU,EAAE,oBAAoB,CAAC,KAAK,CAAC,CAC/E,CAAA;IACD,MAAM,2BAA2B,GAAG,MAAM,CAAC,iBAAiB,CAC1D,yBAAyB,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CACvD,CAAC,uBAAuB,CAAC,CAAA;IAE1B,OAAO,aAAa;SACjB,GAAG,CAAC,CAAC,aAAa,EAAE,EAAE;QACrB,MAAM,gBAAgB,GAAG,2BAA2B,CAAC,IAAI,CACvD,CAAC,cAAc,EAAE,EAAE,CACjB,cAAc,CAAC,YAAY,KAAK,aAAa,CAAC,YAAY;YAC1D,cAAc,CAAC,YAAY,KAAK,aAAa,CAAC,YAAY,CAC7D,CAAA;QACD,OAAO,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC;YACzC,IAAI,EAAE,aAAa,CAAC,IAAI;YACxB,IAAI,EAAE,aAAa,CAAC,QAAQ;YAC5B,MAAM,EAAE,EAAE,MAAM,EAAE,aAAa,CAAC,YAAY,EAAE,MAAM,EAAE,aAAa,CAAC,YAAY,EAAE;YAClF,YAAY,EAAE,EAAE,MAAM,EAAE,aAAa,CAAC,kBAAkB,EAAE,MAAM,EAAE,aAAa,CAAC,kBAAkB,EAAE;YACpG,QAAQ,EAAE,aAAa,CAAC,QAAQ;YAChC,SAAS,EAAE,aAAa,CAAC,SAAS;YAClC,IAAI,EAAE;gBACJ,gBAAgB,EACd,gBAAgB,IAAI,gBAAgB,CAAC,SAAS,KAAK,IAAI;oBACrD,CAAC,CAAC;wBACE,IAAI,EAAE,kBAA2B;wBACjC,IAAI,EAAE,gBAAgB,CAAC,SAAS;wBAChC,KAAK,EAAE,gBAAgB,CAAC,KAAK;qBAC9B;oBACH,CAAC,CAAC,EAAE,IAAI,EAAE,OAAgB,EAAE;gBAChC,YAAY,EAAE,aAAa,CAAC,gBAAgB;aAC7C;SACF,CAAC,CAAA;IACJ,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC;SAC/D,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAA;AACpE,CAAC,CAAC,CAAA;AAEJ,MAAM,CAAC,MAAM,mBAAmB,GAAG,CAAC,UAAoB,EAA2C,EAAE;IACnG,MAAM,GAAG,GAAG,UAAU,CAAC,MAAM,CAI3B,GAAG,CAAA,0CAA0C,mBAAmB,wDAAwD,CACzH,CAAC,CAAC,CAAC,CAAA;IAEJ,OAAO,GAAG,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,YAAY,EAAE,MAAM,EAAE,GAAG,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC,mBAAmB,CAAC,IAAI,CAAA;AAChG,CAAC,CAAA;AAED,MAAM,CAAC,MAAM,oBAAoB,GAAG,CAAC,UAAoB,EAAiD,EAAE,CAC1G,UAAU,CAAC,MAAM,CACf,GAAG,CAAA,oBAAoB,iBAAiB,EAAE,CAC3C,CAAC,CAAC,CAAC,EAAE,IAAI,IAAI,mBAAmB,CAAC,IAAI,CAAC,MAAM,CAAA;AAE/C,+BAA+B;AAC/B,MAAM,CAAC,MAAM,iBAAiB,GAAG,CAAC,UAAoB,EAAE,IAA6C,EAAE,EAAE,CACvG,UAAU,CAAC,OAAO,CAAC,GAAG,CAAA,UAAU,iBAAiB,eAAe,IAAI,CAAC,MAAM,EAAE,CAAC,CAAA;AAEhF,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAChC,YAA4C,EAC5C,UAAoB,EACpB,kBAA0B,EAC1B,QAAgB,EAChB,SAAiB,EACjB,EAAE,CACF,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAClB,0CAA0C;IAC1C,IAAI,MAAM,IAAI,YAAY,CAAC,YAAY,CAAC,MAAM,KAAK,mBAAmB,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;QACnF,MAAM,iBAAiB,GACrB,UAAU,CAAC,MAAM,CACf,iCAAiC,mBAAmB,8CAA8C,EAClG,CAAC,YAAY,CAAC,YAAY,CAAC,MAAM,EAAE,YAAY,CAAC,YAAY,CAAC,MAAM,CAA8B,CAClG,CAAC,CAAC,CAAE,CAAC,KAAK,KAAK,CAAC,CAAA;QAEnB,IAAI,iBAAiB,KAAK,KAAK,EAAE,CAAC;YAChC,iBAAiB,CACf,mBAAmB,YAAY,CAAC,YAAY,CAAC,MAAM,IAAI,YAAY,CAAC,YAAY,CAAC,MAAM,iBAAiB,CACzG,CAAA;QACH,CAAC;IACH,CAAC;IAED,+BAA+B;IAC/B,KAAK,CAAC,CAAC,OAAO,CACZ,UAAU,EACV,GAAG,SAAS,CAAC;QACX,SAAS,EAAE,mBAAmB;QAC9B,OAAO,EAAE,iBAAiB,CAAC,SAAS,CAAC,OAAO;QAC5C,MAAM,EAAE;YACN,YAAY,EAAE,YAAY,CAAC,MAAM,CAAC,MAAM;YACxC,YAAY,EAAE,YAAY,CAAC,MAAM,CAAC,MAAM;YACxC,kBAAkB,EAAE,YAAY,CAAC,YAAY,CAAC,MAAM;YACpD,kBAAkB,EAAE,YAAY,CAAC,YAAY,CAAC,MAAM;YACpD,IAAI,EAAE,YAAY,CAAC,IAAI;YACvB,QAAQ,EAAE,YAAY,CAAC,IAAI,IAAI,EAAE;YACjC,QAAQ;YACR,SAAS;YACT,UAAU,EAAE,kBAAkB;YAC9B,gBAAgB,EAAE,YAAY,CAAC,IAAI,CAAC,YAAY;SACjD;KACF,CAAC,CACH,CAAA;AACH,CAAC,CAAC,CAAA;AAEJ,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAAC,KAAoD,EAAE,EAAE,CACzF,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAClB,MAAM,EAAE,UAAU,EAAE,GAAG,KAAK,CAAC,CAAC,eAAe,CAAA;IAE7C,wCAAwC;IACxC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACtC,MAAM,KAAK,GAAG,KAAK,CAAC,CAAC,CAAE,CAAA;QAEvB,KAAK,CAAC,CAAC,OAAO,CACZ,UAAU,EACV,GAAG,UAAU,CAAC;YACZ,SAAS,EAAE,mBAAmB;YAC9B,OAAO,EAAE,iBAAiB,CAAC,SAAS,CAAC,OAAO;YAC5C,KAAK,EAAE,EAAE,YAAY,EAAE,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE,YAAY,EAAE,KAAK,CAAC,MAAM,CAAC,MAAM,EAAE;YAC/E,YAAY,EAAE,EAAE,gBAAgB,EAAE,KAAK,CAAC,IAAI,CAAC,YAAY,EAAE;SAC5D,CAAC,CACH,CAAA;IACH,CAAC;AACH,CAAC,CAAC,CAAA;AAEJ,MAAM,CAAC,MAAM,wBAAwB,GAAG,CAAC,UAAyD,EAAE,EAAE,CACpG,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC;IAClB,MAAM,EAAE,UAAU,EAAE,GAAG,KAAK,CAAC,CAAC,eAAe,CAAA;IAE7C,IAAI,UAAU,KAAK,mBAAmB,CAAC,IAAI,CAAC,MAAM;QAAE,OAAO,MAAM,CAAC,IAAI,EAAE,CAAA;IAExE,MAAM,mBAAmB,GAAG,MAAM,CAAC,MAAM,CAAC;QACxC,gBAAgB,EAAE,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;KACpE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,kBAAkB,CAAC,EAAE,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,CAAA;IAEpE,MAAM,kBAAkB,GAAG,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,CACjD,UAAU,CAAC,MAAM,CACf,GAAG,CAAA,gCAAgC,mBAAmB,yBAAyB,UAAU,oCAAoC,CAC9H,CACF,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC,EAAE,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,MAAM,CAAC,KAAK,CAAC,CAAA;IAEpG,OAAO,MAAM,CAAC,IAAI,CAAC;QACjB,MAAM,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,mBAAmB,CAAC,aAAa,EAAE;QACzE,QAAQ,EAAE,kBAAkB;KAC7B,CAA2B,CAAA;AAC9B,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,qDAAqD,EAAE,EAAE,UAAU,EAAE,EAAE,UAAU,EAAE,EAAE,CAAC,CAAC,CAAA"}
@@ -2,5 +2,5 @@ import { Effect } from '@livestore/utils/effect';
2
2
  import { UnexpectedError } from '../index.js';
3
3
  import type { DevtoolsOptions } from './types.js';
4
4
  import { LeaderThreadCtx } from './types.js';
5
- export declare const bootDevtools: (options: DevtoolsOptions) => Effect.Effect<void, UnexpectedError, import("effect/Scope").Scope | import("@effect/platform/HttpClient").HttpClient | LeaderThreadCtx>;
5
+ export declare const bootDevtools: (options: DevtoolsOptions) => Effect.Effect<void, UnexpectedError, import("effect/Scope").Scope | LeaderThreadCtx | import("@effect/platform/HttpClient").HttpClient>;
6
6
  //# sourceMappingURL=leader-worker-devtools.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"leader-worker-devtools.d.ts","sourceRoot":"","sources":["../../src/leader-thread/leader-worker-devtools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAA6C,MAAM,yBAAyB,CAAA;AAE3F,OAAO,EAAwD,eAAe,EAAE,MAAM,aAAa,CAAA;AAEnG,OAAO,KAAK,EAAE,eAAe,EAAuB,MAAM,YAAY,CAAA;AACtE,OAAO,EAAE,eAAe,EAAE,MAAM,YAAY,CAAA;AAK5C,eAAO,MAAM,YAAY,YAAa,eAAe,4IAmDsB,CAAA"}
1
+ {"version":3,"file":"leader-worker-devtools.d.ts","sourceRoot":"","sources":["../../src/leader-thread/leader-worker-devtools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAA6C,MAAM,yBAAyB,CAAA;AAG3F,OAAO,EAAwD,eAAe,EAAE,MAAM,aAAa,CAAA;AAEnG,OAAO,KAAK,EAAE,eAAe,EAAuB,MAAM,YAAY,CAAA;AACtE,OAAO,EAAE,eAAe,EAAE,MAAM,YAAY,CAAA;AAK5C,eAAO,MAAM,YAAY,GAAI,SAAS,eAAe,4IAwDsB,CAAA"}
@@ -1,102 +1,132 @@
1
1
  import { Effect, FiberMap, Option, Stream, SubscriptionRef } from '@livestore/utils/effect';
2
+ import { nanoid } from '@livestore/utils/nanoid';
2
3
  import { Devtools, IntentionalShutdownCause, liveStoreVersion, UnexpectedError } from '../index.js';
3
- import { MUTATION_LOG_META_TABLE, SCHEMA_META_TABLE, SCHEMA_MUTATIONS_META_TABLE } from '../schema/mod.js';
4
+ import { SystemTables } from '../schema/mod.js';
4
5
  import { LeaderThreadCtx } from './types.js';
5
6
  // TODO bind scope to the webchannel lifetime
6
7
  export const bootDevtools = (options) => Effect.gen(function* () {
7
8
  if (options.enabled === false) {
8
9
  return;
9
10
  }
10
- const { connectedClientSessionPullQueues, syncProcessor, extraIncomingMessagesQueue } = yield* LeaderThreadCtx;
11
+ const { syncProcessor, extraIncomingMessagesQueue, clientId, storeId } = yield* LeaderThreadCtx;
11
12
  yield* listenToDevtools({
12
13
  incomingMessages: Stream.fromQueue(extraIncomingMessagesQueue),
13
14
  sendMessage: () => Effect.void,
14
15
  }).pipe(Effect.tapCauseLogPretty, Effect.forkScoped);
15
- const { persistenceInfo, devtoolsWebChannel } = yield* options.makeContext;
16
- const sendMessage = (message) => devtoolsWebChannel
17
- .send(message)
18
- .pipe(Effect.withSpan('@livestore/common:leader-thread:devtools:sendToDevtools'), Effect.interruptible, Effect.ignoreLogged);
19
- const { localHead } = yield* syncProcessor.syncState;
20
- // TODO close queue when devtools disconnects
21
- const pullQueue = yield* connectedClientSessionPullQueues.makeQueue(localHead);
22
- yield* Stream.fromQueue(pullQueue).pipe(Stream.tap((msg) => Effect.gen(function* () {
23
- if (msg.payload._tag === 'upstream-advance') {
24
- for (const mutationEventEncoded of msg.payload.newEvents) {
25
- // TODO refactor with push semantics
26
- yield* sendMessage(Devtools.MutationBroadcast.make({ mutationEventEncoded, liveStoreVersion }));
27
- }
28
- }
29
- else {
30
- yield* Effect.logWarning('TODO implement rebases in devtools');
31
- }
32
- })), Stream.runDrain, Effect.forkScoped);
33
- yield* listenToDevtools({
34
- incomingMessages: devtoolsWebChannel.listen.pipe(Stream.flatten(), Stream.orDie),
35
- sendMessage,
36
- persistenceInfo,
37
- }).pipe(Effect.tapCauseLogPretty, Effect.forkScoped);
16
+ const { node, persistenceInfo, mode } = yield* options.boot;
17
+ yield* node.listenForChannel.pipe(Stream.filter((res) => Devtools.isChannelName.devtoolsClientLeader(res.channelName, { storeId, clientId }) && res.mode === mode), Stream.tap(({ channelName, source }) => Effect.gen(function* () {
18
+ const channel = yield* node.makeChannel({
19
+ target: source,
20
+ channelName,
21
+ schema: { listen: Devtools.Leader.MessageToApp, send: Devtools.Leader.MessageFromApp },
22
+ mode,
23
+ });
24
+ const sendMessage = (message) => channel
25
+ .send(message)
26
+ .pipe(Effect.withSpan('@livestore/common:leader-thread:devtools:sendToDevtools'), Effect.interruptible, Effect.ignoreLogged);
27
+ const syncState = yield* syncProcessor.syncState;
28
+ const mergeCounter = syncProcessor.getMergeCounter();
29
+ yield* syncProcessor.pull({ cursor: { mergeCounter, eventNum: syncState.localHead } }).pipe(Stream.tap(({ payload }) => sendMessage(Devtools.Leader.SyncPull.make({ payload, liveStoreVersion }))), Stream.runDrain, Effect.forkScoped);
30
+ yield* listenToDevtools({
31
+ incomingMessages: channel.listen.pipe(Stream.flatten(), Stream.orDie),
32
+ sendMessage,
33
+ persistenceInfo,
34
+ });
35
+ }).pipe(Effect.tapCauseLogPretty, Effect.forkScoped)), Stream.runDrain);
38
36
  }).pipe(Effect.withSpan('@livestore/common:leader-thread:devtools:boot'));
39
37
  const listenToDevtools = ({ incomingMessages, sendMessage, persistenceInfo, }) => Effect.gen(function* () {
40
- const { syncBackend, makeSyncDb, db, dbLog, shutdownStateSubRef, shutdownChannel, syncProcessor } = yield* LeaderThreadCtx;
38
+ const { syncBackend, makeSqliteDb, dbState, dbEventlog, shutdownStateSubRef, shutdownChannel, syncProcessor, clientId, devtools, } = yield* LeaderThreadCtx;
41
39
  const subscriptionFiberMap = yield* FiberMap.make();
40
+ const handledRequestIds = new Set();
42
41
  yield* incomingMessages.pipe(Stream.tap((decodedEvent) => Effect.gen(function* () {
43
- // yield* Effect.logDebug('[@livestore/common:leader-thread:devtools] incomingMessage', decodedEvent)
44
- if (decodedEvent._tag === 'LSD.Disconnect') {
42
+ const { requestId } = decodedEvent;
43
+ const reqPayload = { requestId, liveStoreVersion, clientId };
44
+ // yield* Effect.logDebug(
45
+ // `[@livestore/common:leader-thread:devtools] incomingMessage: ${decodedEvent._tag} (${requestId})`,
46
+ // decodedEvent,
47
+ // )
48
+ if (decodedEvent._tag === 'LSD.Leader.Disconnect') {
45
49
  return;
46
50
  }
47
- const { requestId } = decodedEvent;
48
- const reqPayload = { requestId, liveStoreVersion };
51
+ // TODO we should try to move the duplicate message handling on the webmesh layer
52
+ // So far I could only observe this problem with webmesh proxy channels (e.g. for Expo)
53
+ // Proof: https://share.cleanshot.com/V9G87B0B
54
+ // Also see `store/devtools.ts` for same problem
55
+ if (handledRequestIds.has(requestId)) {
56
+ // yield* Effect.logWarning(`Duplicate message`, decodedEvent)
57
+ return;
58
+ }
59
+ handledRequestIds.add(requestId);
49
60
  switch (decodedEvent._tag) {
50
- case 'LSD.Ping': {
51
- yield* sendMessage(Devtools.Pong.make({ ...reqPayload }));
61
+ case 'LSD.Leader.Ping': {
62
+ yield* sendMessage(Devtools.Leader.Pong.make({ ...reqPayload }));
52
63
  return;
53
64
  }
54
65
  case 'LSD.Leader.SnapshotReq': {
55
- const snapshot = db.export();
56
- yield* sendMessage(Devtools.SnapshotRes.make({ snapshot, ...reqPayload }));
66
+ const snapshot = dbState.export();
67
+ yield* sendMessage(Devtools.Leader.SnapshotRes.make({ snapshot, ...reqPayload }));
57
68
  return;
58
69
  }
59
- case 'LSD.Leader.LoadDatabaseFileReq': {
70
+ case 'LSD.Leader.LoadDatabaseFile.Request': {
60
71
  const { data } = decodedEvent;
61
72
  let tableNames;
62
73
  try {
63
- const tmpSyncDb = yield* makeSyncDb({ _tag: 'in-memory' });
64
- tmpSyncDb.import(data);
65
- const tableNameResults = tmpSyncDb.select(`select name from sqlite_master where type = 'table'`);
74
+ const tmpDb = yield* makeSqliteDb({ _tag: 'in-memory' });
75
+ tmpDb.import(data);
76
+ const tableNameResults = tmpDb.select(`select name from sqlite_master where type = 'table'`);
66
77
  tableNames = new Set(tableNameResults.map((_) => _.name));
67
- tmpSyncDb.close();
78
+ tmpDb.close();
68
79
  }
69
- catch (e) {
70
- yield* Effect.logError(`Error importing database file`, e);
71
- yield* sendMessage(Devtools.LoadDatabaseFileRes.make({ ...reqPayload, status: 'unsupported-file' }));
80
+ catch (cause) {
81
+ yield* Effect.logError(`Error importing database file`, cause);
82
+ yield* sendMessage(Devtools.Leader.LoadDatabaseFile.Error.make({
83
+ ...reqPayload,
84
+ cause: { _tag: 'unexpected-error', cause },
85
+ }));
72
86
  return;
73
87
  }
74
- if (tableNames.has(MUTATION_LOG_META_TABLE)) {
75
- yield* SubscriptionRef.set(shutdownStateSubRef, 'shutting-down');
76
- dbLog.import(data);
77
- db.destroy();
78
- }
79
- else if (tableNames.has(SCHEMA_META_TABLE) && tableNames.has(SCHEMA_MUTATIONS_META_TABLE)) {
80
- yield* SubscriptionRef.set(shutdownStateSubRef, 'shutting-down');
81
- db.import(data);
82
- dbLog.destroy();
88
+ try {
89
+ if (tableNames.has(SystemTables.EVENTLOG_META_TABLE)) {
90
+ // Is eventlog db
91
+ yield* SubscriptionRef.set(shutdownStateSubRef, 'shutting-down');
92
+ dbEventlog.import(data);
93
+ dbState.destroy();
94
+ }
95
+ else if (tableNames.has(SystemTables.SCHEMA_META_TABLE) &&
96
+ tableNames.has(SystemTables.SCHEMA_EVENT_DEFS_META_TABLE)) {
97
+ // Is state db
98
+ yield* SubscriptionRef.set(shutdownStateSubRef, 'shutting-down');
99
+ dbState.import(data);
100
+ dbEventlog.destroy();
101
+ }
102
+ else {
103
+ yield* sendMessage(Devtools.Leader.LoadDatabaseFile.Error.make({
104
+ ...reqPayload,
105
+ cause: { _tag: 'unsupported-database' },
106
+ }));
107
+ return;
108
+ }
109
+ yield* sendMessage(Devtools.Leader.LoadDatabaseFile.Success.make({ ...reqPayload }));
110
+ yield* shutdownChannel.send(IntentionalShutdownCause.make({ reason: 'devtools-import' })) ?? Effect.void;
111
+ return;
83
112
  }
84
- else {
85
- yield* sendMessage(Devtools.LoadDatabaseFileRes.make({ ...reqPayload, status: 'unsupported-database' }));
113
+ catch (cause) {
114
+ yield* Effect.logError(`Error importing database file`, cause);
115
+ yield* sendMessage(Devtools.Leader.LoadDatabaseFile.Error.make({
116
+ ...reqPayload,
117
+ cause: { _tag: 'unexpected-error', cause },
118
+ }));
86
119
  return;
87
120
  }
88
- yield* sendMessage(Devtools.LoadDatabaseFileRes.make({ ...reqPayload, status: 'ok' }));
89
- yield* shutdownChannel.send(IntentionalShutdownCause.make({ reason: 'devtools-import' })) ?? Effect.void;
90
- return;
91
121
  }
92
- case 'LSD.Leader.ResetAllDataReq': {
122
+ case 'LSD.Leader.ResetAllData.Request': {
93
123
  const { mode } = decodedEvent;
94
124
  yield* SubscriptionRef.set(shutdownStateSubRef, 'shutting-down');
95
- db.destroy();
125
+ dbState.destroy();
96
126
  if (mode === 'all-data') {
97
- dbLog.destroy();
127
+ dbEventlog.destroy();
98
128
  }
99
- yield* sendMessage(Devtools.ResetAllDataRes.make({ ...reqPayload }));
129
+ yield* sendMessage(Devtools.Leader.ResetAllData.Success.make({ ...reqPayload }));
100
130
  yield* shutdownChannel.send(IntentionalShutdownCause.make({ reason: 'devtools-reset' })) ?? Effect.void;
101
131
  return;
102
132
  }
@@ -106,30 +136,40 @@ const listenToDevtools = ({ incomingMessages, sendMessage, persistenceInfo, }) =
106
136
  return;
107
137
  }
108
138
  const dbSizeQuery = `SELECT page_count * page_size as size FROM pragma_page_count(), pragma_page_size();`;
109
- const dbFileSize = db.select(dbSizeQuery, undefined)[0].size;
110
- const mutationLogFileSize = dbLog.select(dbSizeQuery, undefined)[0].size;
111
- yield* sendMessage(Devtools.DatabaseFileInfoRes.make({
112
- db: { fileSize: dbFileSize, persistenceInfo: persistenceInfo.db },
113
- mutationLog: { fileSize: mutationLogFileSize, persistenceInfo: persistenceInfo.mutationLog },
139
+ const dbFileSize = dbState.select(dbSizeQuery, undefined)[0].size;
140
+ const eventlogFileSize = dbEventlog.select(dbSizeQuery, undefined)[0].size;
141
+ yield* sendMessage(Devtools.Leader.DatabaseFileInfoRes.make({
142
+ state: { fileSize: dbFileSize, persistenceInfo: persistenceInfo.state },
143
+ eventlog: { fileSize: eventlogFileSize, persistenceInfo: persistenceInfo.eventlog },
114
144
  ...reqPayload,
115
145
  }));
116
146
  return;
117
147
  }
118
- case 'LSD.Leader.MutationLogReq': {
119
- const mutationLog = dbLog.export();
120
- yield* sendMessage(Devtools.MutationLogRes.make({ mutationLog, ...reqPayload }));
148
+ case 'LSD.Leader.EventlogReq': {
149
+ const eventlog = dbEventlog.export();
150
+ yield* sendMessage(Devtools.Leader.EventlogRes.make({ eventlog, ...reqPayload }));
121
151
  return;
122
152
  }
123
- case 'LSD.Leader.RunMutationReq': {
124
- yield* syncProcessor.pushPartial(decodedEvent.mutationEventEncoded);
125
- yield* sendMessage(Devtools.RunMutationRes.make({ ...reqPayload }));
153
+ case 'LSD.Leader.CommitEventReq': {
154
+ yield* syncProcessor.pushPartial({
155
+ event: decodedEvent.eventEncoded,
156
+ clientId: `devtools-${clientId}`,
157
+ sessionId: `devtools-${clientId}`,
158
+ });
159
+ yield* sendMessage(Devtools.Leader.CommitEventRes.make({ ...reqPayload }));
126
160
  return;
127
161
  }
128
162
  case 'LSD.Leader.SyncHistorySubscribe': {
129
- const { requestId } = decodedEvent;
163
+ const { subscriptionId } = decodedEvent;
130
164
  if (syncBackend !== undefined) {
131
165
  // TODO consider piggybacking on the existing leader-thread sync-pulling
132
- yield* syncBackend.pull(Option.none()).pipe(Stream.map((_) => _.batch), Stream.flattenIterables, Stream.tap(({ mutationEventEncoded, metadata }) => sendMessage(Devtools.SyncHistoryRes.make({ mutationEventEncoded, metadata, ...reqPayload }))), Stream.runDrain, Effect.acquireRelease(() => Effect.log('syncHistorySubscribe done')), Effect.interruptible, Effect.tapCauseLogPretty, FiberMap.run(subscriptionFiberMap, requestId));
166
+ yield* syncBackend.pull(Option.none()).pipe(Stream.map((_) => _.batch), Stream.flattenIterables, Stream.tap(({ eventEncoded, metadata }) => sendMessage(Devtools.Leader.SyncHistoryRes.make({
167
+ eventEncoded,
168
+ metadata,
169
+ subscriptionId,
170
+ ...reqPayload,
171
+ requestId: nanoid(10),
172
+ }))), Stream.runDrain, Effect.interruptible, Effect.tapCauseLogPretty, FiberMap.run(subscriptionFiberMap, subscriptionId));
133
173
  }
134
174
  return;
135
175
  }
@@ -140,24 +180,26 @@ const listenToDevtools = ({ incomingMessages, sendMessage, persistenceInfo, }) =
140
180
  return;
141
181
  }
142
182
  case 'LSD.Leader.SyncingInfoReq': {
143
- const syncingInfo = Devtools.SyncingInfo.make({
183
+ const syncingInfo = Devtools.Leader.SyncingInfo.make({
144
184
  enabled: syncBackend !== undefined,
145
- metadata: {},
185
+ metadata: syncBackend?.metadata ?? {},
146
186
  });
147
- yield* sendMessage(Devtools.SyncingInfoRes.make({ syncingInfo, ...reqPayload }));
187
+ yield* sendMessage(Devtools.Leader.SyncingInfoRes.make({ syncingInfo, ...reqPayload }));
148
188
  return;
149
189
  }
150
190
  case 'LSD.Leader.NetworkStatusSubscribe': {
151
191
  if (syncBackend !== undefined) {
152
- const { requestId } = decodedEvent;
192
+ const { subscriptionId } = decodedEvent;
153
193
  // TODO investigate and fix bug. seems that when sending messages right after
154
194
  // the devtools have connected get sometimes lost
155
195
  // This is probably the same "flaky databrowser loading" bug as we're seeing in the playwright tests
156
196
  yield* Effect.sleep(1000);
157
- yield* syncBackend.isConnected.changes.pipe(Stream.tap((isConnected) => sendMessage(Devtools.NetworkStatusRes.make({
158
- networkStatus: { isConnected, timestampMs: Date.now() },
197
+ yield* Stream.zipLatest(syncBackend.isConnected.changes, devtools.enabled ? devtools.syncBackendLatchState.changes : Stream.make({ latchClosed: false })).pipe(Stream.tap(([isConnected, { latchClosed }]) => sendMessage(Devtools.Leader.NetworkStatusRes.make({
198
+ networkStatus: { isConnected, timestampMs: Date.now(), latchClosed },
199
+ subscriptionId,
159
200
  ...reqPayload,
160
- }))), Stream.runDrain, Effect.interruptible, Effect.tapCauseLogPretty, FiberMap.run(subscriptionFiberMap, requestId));
201
+ requestId: nanoid(10),
202
+ }))), Stream.runDrain, Effect.interruptible, Effect.tapCauseLogPretty, FiberMap.run(subscriptionFiberMap, subscriptionId));
161
203
  }
162
204
  return;
163
205
  }
@@ -166,6 +208,39 @@ const listenToDevtools = ({ incomingMessages, sendMessage, persistenceInfo, }) =
166
208
  yield* FiberMap.remove(subscriptionFiberMap, requestId);
167
209
  return;
168
210
  }
211
+ case 'LSD.Leader.SyncHeadSubscribe': {
212
+ const { subscriptionId } = decodedEvent;
213
+ yield* syncProcessor.syncState.changes.pipe(Stream.tap((syncState) => sendMessage(Devtools.Leader.SyncHeadRes.make({
214
+ local: syncState.localHead,
215
+ upstream: syncState.upstreamHead,
216
+ subscriptionId,
217
+ ...reqPayload,
218
+ requestId: nanoid(10),
219
+ }))), Stream.runDrain, Effect.interruptible, Effect.tapCauseLogPretty, FiberMap.run(subscriptionFiberMap, subscriptionId));
220
+ return;
221
+ }
222
+ case 'LSD.Leader.SyncHeadUnsubscribe': {
223
+ const { subscriptionId } = decodedEvent;
224
+ yield* FiberMap.remove(subscriptionFiberMap, subscriptionId);
225
+ return;
226
+ }
227
+ case 'LSD.Leader.SetSyncLatch.Request': {
228
+ const { closeLatch } = decodedEvent;
229
+ if (devtools.enabled === false)
230
+ return;
231
+ if (closeLatch === true) {
232
+ yield* devtools.syncBackendLatch.close;
233
+ }
234
+ else {
235
+ yield* devtools.syncBackendLatch.open;
236
+ }
237
+ yield* SubscriptionRef.set(devtools.syncBackendLatchState, { latchClosed: closeLatch });
238
+ yield* sendMessage(Devtools.Leader.SetSyncLatch.Success.make({ ...reqPayload }));
239
+ return;
240
+ }
241
+ default: {
242
+ yield* Effect.logWarning(`TODO implement devtools message`, decodedEvent);
243
+ }
169
244
  }
170
245
  }).pipe(Effect.withSpan(`@livestore/common:leader-thread:onDevtoolsMessage:${decodedEvent._tag}`))), UnexpectedError.mapToUnexpectedErrorStream, Stream.runDrain);
171
246
  });