@livestore/common 0.4.0-dev.18 → 0.4.0-dev.19
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.
- package/dist/.tsbuildinfo +1 -1
- package/dist/ClientSessionLeaderThreadProxy.d.ts +10 -10
- package/dist/ClientSessionLeaderThreadProxy.d.ts.map +1 -1
- package/dist/adapter-types.d.ts +5 -5
- package/dist/adapter-types.d.ts.map +1 -1
- package/dist/devtools/devtools-messages-client-session.d.ts +28 -23
- package/dist/devtools/devtools-messages-client-session.d.ts.map +1 -1
- package/dist/devtools/devtools-messages-client-session.js +2 -2
- package/dist/devtools/devtools-messages-client-session.js.map +1 -1
- package/dist/devtools/devtools-messages-common.d.ts +6 -6
- package/dist/devtools/devtools-messages-leader.d.ts +33 -28
- package/dist/devtools/devtools-messages-leader.d.ts.map +1 -1
- package/dist/devtools/devtools-messages-leader.js +8 -8
- package/dist/devtools/devtools-messages-leader.js.map +1 -1
- package/dist/errors.d.ts +6 -6
- package/dist/errors.d.ts.map +1 -1
- package/dist/errors.js +7 -7
- package/dist/errors.js.map +1 -1
- package/dist/leader-thread/LeaderSyncProcessor.d.ts +2 -2
- package/dist/leader-thread/LeaderSyncProcessor.d.ts.map +1 -1
- package/dist/leader-thread/LeaderSyncProcessor.js +28 -28
- package/dist/leader-thread/LeaderSyncProcessor.js.map +1 -1
- package/dist/leader-thread/eventlog.d.ts +13 -13
- package/dist/leader-thread/eventlog.d.ts.map +1 -1
- package/dist/leader-thread/eventlog.js +12 -11
- package/dist/leader-thread/eventlog.js.map +1 -1
- package/dist/leader-thread/leader-worker-devtools.d.ts +2 -2
- package/dist/leader-thread/leader-worker-devtools.d.ts.map +1 -1
- package/dist/leader-thread/leader-worker-devtools.js +3 -3
- package/dist/leader-thread/leader-worker-devtools.js.map +1 -1
- package/dist/leader-thread/make-leader-thread-layer.d.ts +2 -2
- package/dist/leader-thread/make-leader-thread-layer.d.ts.map +1 -1
- package/dist/leader-thread/make-leader-thread-layer.js +10 -8
- package/dist/leader-thread/make-leader-thread-layer.js.map +1 -1
- package/dist/leader-thread/materialize-event.d.ts +1 -1
- package/dist/leader-thread/materialize-event.d.ts.map +1 -1
- package/dist/leader-thread/materialize-event.js +2 -2
- package/dist/leader-thread/materialize-event.js.map +1 -1
- package/dist/leader-thread/recreate-db.d.ts +2 -2
- package/dist/leader-thread/recreate-db.d.ts.map +1 -1
- package/dist/leader-thread/recreate-db.js +5 -5
- package/dist/leader-thread/recreate-db.js.map +1 -1
- package/dist/leader-thread/shutdown-channel.d.ts +2 -2
- package/dist/leader-thread/shutdown-channel.d.ts.map +1 -1
- package/dist/leader-thread/shutdown-channel.js +2 -2
- package/dist/leader-thread/shutdown-channel.js.map +1 -1
- package/dist/leader-thread/types.d.ts +15 -15
- package/dist/leader-thread/types.d.ts.map +1 -1
- package/dist/make-client-session.d.ts +2 -2
- package/dist/make-client-session.d.ts.map +1 -1
- package/dist/materializer-helper.d.ts +5 -5
- package/dist/materializer-helper.d.ts.map +1 -1
- package/dist/materializer-helper.js.map +1 -1
- package/dist/rematerialize-from-eventlog.d.ts +2 -2
- package/dist/rematerialize-from-eventlog.d.ts.map +1 -1
- package/dist/rematerialize-from-eventlog.js +6 -6
- package/dist/rematerialize-from-eventlog.js.map +1 -1
- package/dist/schema/EventDef/define.d.ts +147 -0
- package/dist/schema/EventDef/define.d.ts.map +1 -0
- package/dist/schema/EventDef/define.js +139 -0
- package/dist/schema/EventDef/define.js.map +1 -0
- package/dist/schema/EventDef/event-def.d.ts +106 -0
- package/dist/schema/EventDef/event-def.d.ts.map +1 -0
- package/dist/schema/EventDef/event-def.js +2 -0
- package/dist/schema/EventDef/event-def.js.map +1 -0
- package/dist/schema/EventDef/facts.d.ts +118 -0
- package/dist/schema/EventDef/facts.d.ts.map +1 -0
- package/dist/schema/EventDef/facts.js +53 -0
- package/dist/schema/EventDef/facts.js.map +1 -0
- package/dist/schema/EventDef/materializer.d.ts +155 -0
- package/dist/schema/EventDef/materializer.d.ts.map +1 -0
- package/dist/schema/EventDef/materializer.js +83 -0
- package/dist/schema/EventDef/materializer.js.map +1 -0
- package/dist/schema/EventDef/mod.d.ts +5 -0
- package/dist/schema/EventDef/mod.d.ts.map +1 -0
- package/dist/schema/EventDef/mod.js +5 -0
- package/dist/schema/EventDef/mod.js.map +1 -0
- package/dist/schema/EventSequenceNumber/client.d.ts +136 -0
- package/dist/schema/EventSequenceNumber/client.d.ts.map +1 -0
- package/dist/schema/{EventSequenceNumber.js → EventSequenceNumber/client.js} +86 -39
- package/dist/schema/EventSequenceNumber/client.js.map +1 -0
- package/dist/schema/EventSequenceNumber/global.d.ts +15 -0
- package/dist/schema/EventSequenceNumber/global.d.ts.map +1 -0
- package/dist/schema/EventSequenceNumber/global.js +14 -0
- package/dist/schema/EventSequenceNumber/global.js.map +1 -0
- package/dist/schema/EventSequenceNumber/mod.d.ts +37 -0
- package/dist/schema/EventSequenceNumber/mod.d.ts.map +1 -0
- package/dist/schema/EventSequenceNumber/mod.js +37 -0
- package/dist/schema/EventSequenceNumber/mod.js.map +1 -0
- package/dist/schema/EventSequenceNumber.test.js +41 -41
- package/dist/schema/EventSequenceNumber.test.js.map +1 -1
- package/dist/schema/{LiveStoreEvent.d.ts → LiveStoreEvent/client.d.ts} +84 -101
- package/dist/schema/LiveStoreEvent/client.d.ts.map +1 -0
- package/dist/schema/{LiveStoreEvent.js → LiveStoreEvent/client.js} +69 -52
- package/dist/schema/LiveStoreEvent/client.js.map +1 -0
- package/dist/schema/LiveStoreEvent/for-event-def.d.ts +52 -0
- package/dist/schema/LiveStoreEvent/for-event-def.d.ts.map +1 -0
- package/dist/schema/LiveStoreEvent/for-event-def.js +2 -0
- package/dist/schema/LiveStoreEvent/for-event-def.js.map +1 -0
- package/dist/schema/LiveStoreEvent/global.d.ts +36 -0
- package/dist/schema/LiveStoreEvent/global.d.ts.map +1 -0
- package/dist/schema/LiveStoreEvent/global.js +31 -0
- package/dist/schema/LiveStoreEvent/global.js.map +1 -0
- package/dist/schema/LiveStoreEvent/input.d.ts +46 -0
- package/dist/schema/LiveStoreEvent/input.d.ts.map +1 -0
- package/dist/schema/LiveStoreEvent/input.js +26 -0
- package/dist/schema/LiveStoreEvent/input.js.map +1 -0
- package/dist/schema/LiveStoreEvent/mod.d.ts +5 -0
- package/dist/schema/LiveStoreEvent/mod.d.ts.map +1 -0
- package/dist/schema/LiveStoreEvent/mod.js +5 -0
- package/dist/schema/LiveStoreEvent/mod.js.map +1 -0
- package/dist/schema/events.d.ts +1 -1
- package/dist/schema/events.d.ts.map +1 -1
- package/dist/schema/events.js +1 -1
- package/dist/schema/events.js.map +1 -1
- package/dist/schema/mod.d.ts +3 -3
- package/dist/schema/mod.d.ts.map +1 -1
- package/dist/schema/mod.js +3 -3
- package/dist/schema/mod.js.map +1 -1
- package/dist/schema/schema.d.ts +1 -1
- package/dist/schema/schema.d.ts.map +1 -1
- package/dist/schema/state/sqlite/client-document-def.d.ts +1 -1
- package/dist/schema/state/sqlite/client-document-def.d.ts.map +1 -1
- package/dist/schema/state/sqlite/client-document-def.js +2 -2
- package/dist/schema/state/sqlite/client-document-def.js.map +1 -1
- package/dist/schema/state/sqlite/client-document-def.test.js.map +1 -1
- package/dist/schema/state/sqlite/column-def.js +13 -0
- package/dist/schema/state/sqlite/column-def.js.map +1 -1
- package/dist/schema/state/sqlite/column-def.test.js +2 -1
- package/dist/schema/state/sqlite/column-def.test.js.map +1 -1
- package/dist/schema/state/sqlite/mod.d.ts +2 -2
- package/dist/schema/state/sqlite/mod.d.ts.map +1 -1
- package/dist/schema/state/sqlite/mod.js +1 -1
- package/dist/schema/state/sqlite/mod.js.map +1 -1
- package/dist/schema/state/sqlite/query-builder/api.d.ts +12 -8
- package/dist/schema/state/sqlite/query-builder/api.d.ts.map +1 -1
- package/dist/schema/state/sqlite/query-builder/astToSql.d.ts.map +1 -1
- package/dist/schema/state/sqlite/query-builder/astToSql.js +18 -11
- package/dist/schema/state/sqlite/query-builder/astToSql.js.map +1 -1
- package/dist/schema/state/sqlite/query-builder/impl.test.js +119 -90
- package/dist/schema/state/sqlite/query-builder/impl.test.js.map +1 -1
- package/dist/schema/state/sqlite/system-tables/eventlog-tables.js +5 -5
- package/dist/schema/state/sqlite/system-tables/eventlog-tables.js.map +1 -1
- package/dist/schema/state/sqlite/system-tables/state-tables.js +3 -3
- package/dist/schema/state/sqlite/system-tables/state-tables.js.map +1 -1
- package/dist/schema/unknown-events.d.ts +3 -3
- package/dist/schema/unknown-events.d.ts.map +1 -1
- package/dist/schema-management/migrations.d.ts +2 -2
- package/dist/schema-management/migrations.d.ts.map +1 -1
- package/dist/schema-management/migrations.js.map +1 -1
- package/dist/schema-management/validate-schema.d.ts +3 -3
- package/dist/schema-management/validate-schema.d.ts.map +1 -1
- package/dist/schema-management/validate-schema.js +2 -2
- package/dist/schema-management/validate-schema.js.map +1 -1
- package/dist/sqlite-types.d.ts +3 -3
- package/dist/sqlite-types.d.ts.map +1 -1
- package/dist/sync/ClientSessionSyncProcessor.d.ts +5 -5
- package/dist/sync/ClientSessionSyncProcessor.d.ts.map +1 -1
- package/dist/sync/ClientSessionSyncProcessor.js +12 -12
- package/dist/sync/ClientSessionSyncProcessor.js.map +1 -1
- package/dist/sync/errors.d.ts +9 -4
- package/dist/sync/errors.d.ts.map +1 -1
- package/dist/sync/errors.js +6 -6
- package/dist/sync/errors.js.map +1 -1
- package/dist/sync/mock-sync-backend.d.ts +6 -6
- package/dist/sync/mock-sync-backend.d.ts.map +1 -1
- package/dist/sync/mock-sync-backend.js +4 -4
- package/dist/sync/mock-sync-backend.js.map +1 -1
- package/dist/sync/next/compact-events.js +2 -2
- package/dist/sync/next/compact-events.js.map +1 -1
- package/dist/sync/next/facts.d.ts +5 -5
- package/dist/sync/next/facts.d.ts.map +1 -1
- package/dist/sync/next/facts.js.map +1 -1
- package/dist/sync/next/history-dag-common.d.ts +5 -5
- package/dist/sync/next/history-dag-common.d.ts.map +1 -1
- package/dist/sync/next/history-dag-common.js +5 -5
- package/dist/sync/next/history-dag-common.js.map +1 -1
- package/dist/sync/next/history-dag.d.ts.map +1 -1
- package/dist/sync/next/history-dag.js +8 -8
- package/dist/sync/next/history-dag.js.map +1 -1
- package/dist/sync/next/rebase-events.d.ts +5 -5
- package/dist/sync/next/rebase-events.d.ts.map +1 -1
- package/dist/sync/next/rebase-events.js +5 -5
- package/dist/sync/next/rebase-events.js.map +1 -1
- package/dist/sync/next/test/event-fixtures.d.ts +2 -2
- package/dist/sync/next/test/event-fixtures.d.ts.map +1 -1
- package/dist/sync/next/test/event-fixtures.js +9 -9
- package/dist/sync/next/test/event-fixtures.js.map +1 -1
- package/dist/sync/sync-backend-kv.d.ts +3 -3
- package/dist/sync/sync-backend-kv.d.ts.map +1 -1
- package/dist/sync/sync-backend-kv.js +3 -3
- package/dist/sync/sync-backend-kv.js.map +1 -1
- package/dist/sync/sync-backend.d.ts +9 -9
- package/dist/sync/sync-backend.d.ts.map +1 -1
- package/dist/sync/syncstate.d.ts +55 -42
- package/dist/sync/syncstate.d.ts.map +1 -1
- package/dist/sync/syncstate.js +42 -42
- package/dist/sync/syncstate.js.map +1 -1
- package/dist/sync/syncstate.test.js +40 -40
- package/dist/sync/syncstate.test.js.map +1 -1
- package/dist/sync/validate-push-payload.d.ts +1 -1
- package/dist/sync/validate-push-payload.d.ts.map +1 -1
- package/dist/sync/validate-push-payload.js +2 -2
- package/dist/sync/validate-push-payload.js.map +1 -1
- package/dist/testing/event-factory.d.ts +3 -3
- package/dist/testing/event-factory.d.ts.map +1 -1
- package/dist/testing/event-factory.js +5 -7
- package/dist/testing/event-factory.js.map +1 -1
- package/dist/version.d.ts +1 -1
- package/dist/version.js +1 -1
- package/package.json +4 -4
- package/src/ClientSessionLeaderThreadProxy.ts +10 -10
- package/src/adapter-types.ts +5 -5
- package/src/devtools/devtools-messages-client-session.ts +2 -2
- package/src/devtools/devtools-messages-leader.ts +8 -8
- package/src/errors.ts +11 -13
- package/src/leader-thread/LeaderSyncProcessor.ts +54 -56
- package/src/leader-thread/eventlog.ts +21 -26
- package/src/leader-thread/leader-worker-devtools.ts +3 -3
- package/src/leader-thread/make-leader-thread-layer.ts +12 -10
- package/src/leader-thread/materialize-event.ts +3 -3
- package/src/leader-thread/recreate-db.ts +6 -6
- package/src/leader-thread/shutdown-channel.ts +2 -2
- package/src/leader-thread/types.ts +15 -15
- package/src/make-client-session.ts +2 -2
- package/src/materializer-helper.ts +5 -5
- package/src/rematerialize-from-eventlog.ts +6 -6
- package/src/schema/EventDef/define.ts +201 -0
- package/src/schema/EventDef/event-def.ts +120 -0
- package/src/schema/EventDef/facts.ts +135 -0
- package/src/schema/EventDef/materializer.ts +172 -0
- package/src/schema/EventDef/mod.ts +4 -0
- package/src/schema/EventSequenceNumber/client.ts +257 -0
- package/src/schema/EventSequenceNumber/global.ts +19 -0
- package/src/schema/EventSequenceNumber/mod.ts +37 -0
- package/src/schema/EventSequenceNumber.test.ts +68 -50
- package/src/schema/LiveStoreEvent/client.ts +221 -0
- package/src/schema/LiveStoreEvent/for-event-def.ts +60 -0
- package/src/schema/LiveStoreEvent/global.ts +45 -0
- package/src/schema/LiveStoreEvent/input.ts +63 -0
- package/src/schema/LiveStoreEvent/mod.ts +4 -0
- package/src/schema/events.ts +1 -1
- package/src/schema/mod.ts +3 -3
- package/src/schema/schema.ts +1 -1
- package/src/schema/state/sqlite/client-document-def.test.ts +2 -2
- package/src/schema/state/sqlite/client-document-def.ts +3 -3
- package/src/schema/state/sqlite/column-def.test.ts +2 -1
- package/src/schema/state/sqlite/column-def.ts +17 -0
- package/src/schema/state/sqlite/mod.ts +2 -2
- package/src/schema/state/sqlite/query-builder/api.ts +12 -8
- package/src/schema/state/sqlite/query-builder/astToSql.ts +20 -11
- package/src/schema/state/sqlite/query-builder/impl.test.ts +122 -90
- package/src/schema/state/sqlite/system-tables/eventlog-tables.ts +5 -5
- package/src/schema/state/sqlite/system-tables/state-tables.ts +3 -3
- package/src/schema/unknown-events.ts +3 -3
- package/src/schema-management/migrations.ts +2 -2
- package/src/schema-management/validate-schema.ts +3 -3
- package/src/sqlite-types.ts +3 -3
- package/src/sync/ClientSessionSyncProcessor.ts +17 -17
- package/src/sync/errors.ts +6 -6
- package/src/sync/mock-sync-backend.ts +16 -16
- package/src/sync/next/compact-events.ts +2 -2
- package/src/sync/next/facts.ts +6 -6
- package/src/sync/next/history-dag-common.ts +8 -8
- package/src/sync/next/history-dag.ts +14 -10
- package/src/sync/next/rebase-events.ts +11 -11
- package/src/sync/next/test/event-fixtures.ts +11 -11
- package/src/sync/sync-backend-kv.ts +3 -3
- package/src/sync/sync-backend.ts +9 -9
- package/src/sync/syncstate.test.ts +46 -46
- package/src/sync/syncstate.ts +59 -55
- package/src/sync/validate-push-payload.ts +4 -4
- package/src/testing/event-factory.ts +10 -12
- package/src/version.ts +1 -1
- package/dist/schema/EventDef.d.ts +0 -126
- package/dist/schema/EventDef.d.ts.map +0 -1
- package/dist/schema/EventDef.js +0 -46
- package/dist/schema/EventDef.js.map +0 -1
- package/dist/schema/EventSequenceNumber.d.ts +0 -89
- package/dist/schema/EventSequenceNumber.d.ts.map +0 -1
- package/dist/schema/EventSequenceNumber.js.map +0 -1
- package/dist/schema/LiveStoreEvent.d.ts.map +0 -1
- package/dist/schema/LiveStoreEvent.js.map +0 -1
- package/src/schema/EventDef.ts +0 -222
- package/src/schema/EventSequenceNumber.ts +0 -208
- package/src/schema/LiveStoreEvent.ts +0 -286
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Facts System for Event Constraints (Experimental)
|
|
3
|
+
*
|
|
4
|
+
* The facts system enables defining constraints and dependencies between events.
|
|
5
|
+
* Facts are key-value pairs that events can read, set, or require, allowing
|
|
6
|
+
* LiveStore to understand event relationships for:
|
|
7
|
+
* - History constraints (ordering requirements)
|
|
8
|
+
* - Event compaction (which events can be safely merged)
|
|
9
|
+
* - Conflict detection during sync
|
|
10
|
+
*
|
|
11
|
+
* @experimental This system is not fully implemented yet.
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```ts
|
|
15
|
+
* const facts = defineFacts({
|
|
16
|
+
* todoExists: (id: string) => [`todo:${id}`, true],
|
|
17
|
+
* userOwns: (userId: string, todoId: string) => [`owns:${userId}:${todoId}`, true],
|
|
18
|
+
* })
|
|
19
|
+
*
|
|
20
|
+
* const todoCreated = Events.synced({
|
|
21
|
+
* name: 'v1.TodoCreated',
|
|
22
|
+
* schema: Schema.Struct({ id: Schema.String, userId: Schema.String }),
|
|
23
|
+
* facts: ({ id, userId }) => ({
|
|
24
|
+
* modify: { set: [facts.todoExists(id), facts.userOwns(userId, id)] },
|
|
25
|
+
* require: [],
|
|
26
|
+
* }),
|
|
27
|
+
* })
|
|
28
|
+
* ```
|
|
29
|
+
* @module
|
|
30
|
+
*/
|
|
31
|
+
|
|
32
|
+
/** String key identifying a fact (e.g., `"todo:abc123"` or `"user:owner:xyz"`). */
|
|
33
|
+
export type EventDefKey = string
|
|
34
|
+
|
|
35
|
+
/** String identifier for a fact type. */
|
|
36
|
+
export type EventDefFact = string
|
|
37
|
+
|
|
38
|
+
/** Immutable map of fact keys to their current values. */
|
|
39
|
+
export type EventDefFacts = ReadonlyMap<string, any>
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Groups of facts that an event interacts with.
|
|
43
|
+
* Used internally to track how events modify and depend on facts.
|
|
44
|
+
*/
|
|
45
|
+
export type EventDefFactsGroup = {
|
|
46
|
+
/** Facts this event sets to a new value. */
|
|
47
|
+
modifySet: EventDefFacts
|
|
48
|
+
|
|
49
|
+
/** Facts this event removes/unsets. */
|
|
50
|
+
modifyUnset: EventDefFacts
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Facts this event requires to exist with specific values.
|
|
54
|
+
* Events on independent dependency branches are commutative,
|
|
55
|
+
* which can facilitate more prioritized syncing.
|
|
56
|
+
*/
|
|
57
|
+
depRequire: EventDefFacts
|
|
58
|
+
|
|
59
|
+
/** Facts this event reads (but doesn't require). */
|
|
60
|
+
depRead: EventDefFacts
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
/** Mutable snapshot of facts state at a point in time. */
|
|
64
|
+
export type EventDefFactsSnapshot = Map<string, any>
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Input format for specifying a fact.
|
|
68
|
+
* Either a simple key string (value defaults to `true`) or a `[key, value]` tuple.
|
|
69
|
+
*
|
|
70
|
+
* @example
|
|
71
|
+
* ```ts
|
|
72
|
+
* // Simple key (value = true)
|
|
73
|
+
* const fact1: EventDefFactInput = 'todo:abc123'
|
|
74
|
+
*
|
|
75
|
+
* // Key-value tuple
|
|
76
|
+
* const fact2: EventDefFactInput = ['todo:abc123', { status: 'active' }]
|
|
77
|
+
* ```
|
|
78
|
+
*/
|
|
79
|
+
export type EventDefFactInput = string | readonly [string, any]
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Callback function that defines how an event interacts with the facts system.
|
|
83
|
+
* Called during event processing to determine fact constraints.
|
|
84
|
+
*
|
|
85
|
+
* @example
|
|
86
|
+
* ```ts
|
|
87
|
+
* const factsCallback: FactsCallback<{ id: string }> = (args, currentFacts) => ({
|
|
88
|
+
* modify: {
|
|
89
|
+
* set: [`item:${args.id}`], // Create/update this fact
|
|
90
|
+
* unset: [], // No facts to remove
|
|
91
|
+
* },
|
|
92
|
+
* require: currentFacts.has('initialized') ? [] : ['initialized'],
|
|
93
|
+
* })
|
|
94
|
+
* ```
|
|
95
|
+
*/
|
|
96
|
+
export type FactsCallback<TTo> = (
|
|
97
|
+
args: TTo,
|
|
98
|
+
currentFacts: EventDefFacts,
|
|
99
|
+
) => {
|
|
100
|
+
modify: {
|
|
101
|
+
/** Facts to set (create or update). */
|
|
102
|
+
set: Iterable<EventDefFactInput>
|
|
103
|
+
/** Facts to unset (remove). */
|
|
104
|
+
unset: Iterable<EventDefFactInput>
|
|
105
|
+
}
|
|
106
|
+
/** Facts that must exist with specific values for this event to be valid. */
|
|
107
|
+
require: Iterable<EventDefFactInput>
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* Helper to define a typed record of fact constructors.
|
|
112
|
+
* Returns the input unchanged but provides type inference.
|
|
113
|
+
*
|
|
114
|
+
* @example
|
|
115
|
+
* ```ts
|
|
116
|
+
* const facts = defineFacts({
|
|
117
|
+
* // Simple fact (value = true)
|
|
118
|
+
* initialized: 'system:initialized',
|
|
119
|
+
*
|
|
120
|
+
* // Parameterized fact constructor
|
|
121
|
+
* todoExists: (id: string) => [`todo:${id}`, true] as const,
|
|
122
|
+
*
|
|
123
|
+
* // Fact with complex value
|
|
124
|
+
* todoStatus: (id: string, status: string) => [`todo:${id}:status`, status] as const,
|
|
125
|
+
* })
|
|
126
|
+
*
|
|
127
|
+
* // Usage
|
|
128
|
+
* facts.todoExists('abc') // => ['todo:abc', true]
|
|
129
|
+
* ```
|
|
130
|
+
*/
|
|
131
|
+
export const defineFacts = <
|
|
132
|
+
TRecord extends Record<string, EventDefFactInput | ((...args: any[]) => EventDefFactInput)>,
|
|
133
|
+
>(
|
|
134
|
+
record: TRecord,
|
|
135
|
+
): TRecord => record
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Materializer System
|
|
3
|
+
*
|
|
4
|
+
* Materializers transform events into SQL mutations (INSERT, UPDATE, DELETE).
|
|
5
|
+
* Every non-derived event must have a corresponding materializer that defines
|
|
6
|
+
* how it affects the SQLite database state.
|
|
7
|
+
*
|
|
8
|
+
* Materializers are pure functions that receive the event arguments and return
|
|
9
|
+
* SQL operations. They can also query current state when needed for complex
|
|
10
|
+
* transformations.
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```ts
|
|
14
|
+
* import { State } from '@livestore/livestore'
|
|
15
|
+
*
|
|
16
|
+
* const materializers = State.SQLite.materializers(events, {
|
|
17
|
+
* 'v1.TodoCreated': ({ id, text }) =>
|
|
18
|
+
* tables.todos.insert({ id, text, completed: false }),
|
|
19
|
+
*
|
|
20
|
+
* 'v1.TodoCompleted': ({ id }) =>
|
|
21
|
+
* tables.todos.update({ completed: true }).where({ id }),
|
|
22
|
+
*
|
|
23
|
+
* 'v1.TodoDeleted': ({ id, deletedAt }) =>
|
|
24
|
+
* tables.todos.update({ deletedAt }).where({ id }),
|
|
25
|
+
* })
|
|
26
|
+
* ```
|
|
27
|
+
* @module
|
|
28
|
+
*/
|
|
29
|
+
|
|
30
|
+
import type { SingleOrReadonlyArray } from '@livestore/utils'
|
|
31
|
+
|
|
32
|
+
import type { BindValues } from '../../sql-queries/sql-queries.ts'
|
|
33
|
+
import type { ParamsObject } from '../../util.ts'
|
|
34
|
+
import type * as LiveStoreEvent from '../LiveStoreEvent/mod.ts'
|
|
35
|
+
import type { QueryBuilder } from '../state/sqlite/query-builder/mod.ts'
|
|
36
|
+
import type { EventDef } from './event-def.ts'
|
|
37
|
+
import type { EventDefFacts } from './facts.ts'
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Result type for materializer functions.
|
|
41
|
+
*
|
|
42
|
+
* Can be one of:
|
|
43
|
+
* - A QueryBuilder operation (recommended for type safety)
|
|
44
|
+
* - A raw SQL string
|
|
45
|
+
* - An object with SQL, bind values, and optional write table tracking
|
|
46
|
+
*/
|
|
47
|
+
export type MaterializerResult =
|
|
48
|
+
| {
|
|
49
|
+
sql: string
|
|
50
|
+
bindValues: BindValues
|
|
51
|
+
writeTables?: ReadonlySet<string>
|
|
52
|
+
}
|
|
53
|
+
| QueryBuilder.Any
|
|
54
|
+
| string
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Function signature for querying current state within a materializer.
|
|
58
|
+
*
|
|
59
|
+
* Allows materializers to read existing data when computing mutations.
|
|
60
|
+
* Can be called with either raw SQL or a type-safe QueryBuilder.
|
|
61
|
+
*
|
|
62
|
+
* @example
|
|
63
|
+
* ```ts
|
|
64
|
+
* 'v1.TodoUpdated': ({ id, text }, { query }) => {
|
|
65
|
+
* const existing = query(tables.todos.select().where({ id }).first())
|
|
66
|
+
* if (!existing) return [] // No-op if todo doesn't exist
|
|
67
|
+
* return tables.todos.update({ text }).where({ id })
|
|
68
|
+
* }
|
|
69
|
+
* ```
|
|
70
|
+
*/
|
|
71
|
+
export type MaterializerContextQuery = {
|
|
72
|
+
/** Query with raw SQL and bind values. */
|
|
73
|
+
(args: { query: string; bindValues: ParamsObject }): ReadonlyArray<unknown>
|
|
74
|
+
/** Query with a type-safe QueryBuilder. */
|
|
75
|
+
<TResult>(qb: QueryBuilder<TResult, any, any>): TResult
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Function type for transforming an event into database mutations.
|
|
80
|
+
*
|
|
81
|
+
* Materializers are the bridge between events and SQLite state. They receive
|
|
82
|
+
* the decoded event arguments and return SQL operations to execute.
|
|
83
|
+
*
|
|
84
|
+
* @example
|
|
85
|
+
* ```ts
|
|
86
|
+
* const todoCreatedMaterializer: Materializer<typeof todoCreated> =
|
|
87
|
+
* ({ id, text }) => tables.todos.insert({ id, text, completed: false })
|
|
88
|
+
* ```
|
|
89
|
+
*/
|
|
90
|
+
export type Materializer<TEventDef extends EventDef.AnyWithoutFn = EventDef.AnyWithoutFn> = (
|
|
91
|
+
/** Decoded event arguments. */
|
|
92
|
+
event: TEventDef['schema']['Type'],
|
|
93
|
+
context: {
|
|
94
|
+
/** Current facts state (experimental). */
|
|
95
|
+
currentFacts: EventDefFacts
|
|
96
|
+
/** The event definition being materialized. */
|
|
97
|
+
eventDef: TEventDef
|
|
98
|
+
/** Function to query current database state. */
|
|
99
|
+
query: MaterializerContextQuery
|
|
100
|
+
/** Full event metadata including clientId, sessionId, sequence numbers. */
|
|
101
|
+
event: LiveStoreEvent.Client.Decoded
|
|
102
|
+
},
|
|
103
|
+
) => SingleOrReadonlyArray<MaterializerResult>
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Type-safe wrapper for defining a single materializer.
|
|
107
|
+
*
|
|
108
|
+
* Useful when defining materializers separately from the `materializers()` builder.
|
|
109
|
+
* The first argument provides type inference for the second.
|
|
110
|
+
*
|
|
111
|
+
* @example
|
|
112
|
+
* ```ts
|
|
113
|
+
* const todoCreatedHandler = defineMaterializer(
|
|
114
|
+
* events.todoCreated,
|
|
115
|
+
* ({ id, text }) => tables.todos.insert({ id, text, completed: false })
|
|
116
|
+
* )
|
|
117
|
+
* ```
|
|
118
|
+
*/
|
|
119
|
+
export const defineMaterializer = <TEventDef extends EventDef.AnyWithoutFn>(
|
|
120
|
+
_eventDef: TEventDef,
|
|
121
|
+
materializer: Materializer<TEventDef>,
|
|
122
|
+
): Materializer<TEventDef> => {
|
|
123
|
+
return materializer
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Builder function for creating a type-safe materializer map.
|
|
128
|
+
*
|
|
129
|
+
* This is the primary way to define materializers in LiveStore. It ensures:
|
|
130
|
+
* - Every non-derived event has a corresponding materializer
|
|
131
|
+
* - Materializer argument types match their event schemas
|
|
132
|
+
* - Derived events are excluded from the required handlers
|
|
133
|
+
*
|
|
134
|
+
* @example
|
|
135
|
+
* ```ts
|
|
136
|
+
* import { State } from '@livestore/livestore'
|
|
137
|
+
*
|
|
138
|
+
* const handlers = State.SQLite.materializers(events, {
|
|
139
|
+
* // Handler for each event - argument types are inferred
|
|
140
|
+
* 'v1.TodoCreated': ({ id, text, completed }) =>
|
|
141
|
+
* tables.todos.insert({ id, text, completed }),
|
|
142
|
+
*
|
|
143
|
+
* 'v1.TodoUpdated': ({ id, text }) =>
|
|
144
|
+
* tables.todos.update({ text }).where({ id }),
|
|
145
|
+
*
|
|
146
|
+
* // Can return multiple operations
|
|
147
|
+
* 'v1.UserCreatedWithDefaults': ({ userId, name }) => [
|
|
148
|
+
* tables.users.insert({ id: userId, name }),
|
|
149
|
+
* tables.settings.insert({ userId, theme: 'light' }),
|
|
150
|
+
* ],
|
|
151
|
+
*
|
|
152
|
+
* // Can query current state
|
|
153
|
+
* 'v1.TodoToggled': ({ id }, { query }) => {
|
|
154
|
+
* const todo = query(tables.todos.select().where({ id }).first())
|
|
155
|
+
* return tables.todos.update({ completed: !todo?.completed }).where({ id })
|
|
156
|
+
* },
|
|
157
|
+
* })
|
|
158
|
+
* ```
|
|
159
|
+
*/
|
|
160
|
+
export const materializers = <TInputRecord extends Record<string, EventDef.AnyWithoutFn>>(
|
|
161
|
+
_eventDefRecord: TInputRecord,
|
|
162
|
+
handlers: {
|
|
163
|
+
[TEventName in TInputRecord[keyof TInputRecord]['name'] as Extract<
|
|
164
|
+
TInputRecord[keyof TInputRecord],
|
|
165
|
+
{ name: TEventName }
|
|
166
|
+
>['options']['derived'] extends true
|
|
167
|
+
? never
|
|
168
|
+
: TEventName]: Materializer<Extract<TInputRecord[keyof TInputRecord], { name: TEventName }>>
|
|
169
|
+
},
|
|
170
|
+
) => {
|
|
171
|
+
return handlers
|
|
172
|
+
}
|
|
@@ -0,0 +1,257 @@
|
|
|
1
|
+
import { Brand, Schema as S } from '@livestore/utils/effect'
|
|
2
|
+
|
|
3
|
+
import { type Type as Global, Schema as GlobalSchema, make as makeGlobal } from './global.ts'
|
|
4
|
+
|
|
5
|
+
/** Branded integer type for client-local sequence numbers. */
|
|
6
|
+
export type Type = Brand.Branded<number, 'ClientEventSequenceNumber'>
|
|
7
|
+
|
|
8
|
+
const ClientBrand = Brand.nominal<Type>()
|
|
9
|
+
|
|
10
|
+
/** Effect Schema for encoding/decoding client sequence numbers. */
|
|
11
|
+
export const Schema = S.fromBrand(ClientBrand)(S.Int)
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Creates a branded client sequence number from a plain number.
|
|
15
|
+
*
|
|
16
|
+
* @example
|
|
17
|
+
* ```ts
|
|
18
|
+
* const clientSeq = EventSequenceNumber.Client.make(1)
|
|
19
|
+
* ```
|
|
20
|
+
*/
|
|
21
|
+
export const make = ClientBrand
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Default client sequence number (0). Used for confirmed/synced events.
|
|
25
|
+
*
|
|
26
|
+
* @example
|
|
27
|
+
* ```ts
|
|
28
|
+
* const defaultSeq = EventSequenceNumber.Client.DEFAULT // 0
|
|
29
|
+
* ```
|
|
30
|
+
*/
|
|
31
|
+
export const DEFAULT = 0 as any as Type
|
|
32
|
+
|
|
33
|
+
/** Default rebase generation (0). Increments each time the client rebases unconfirmed events. */
|
|
34
|
+
export const REBASE_GENERATION_DEFAULT = 0
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Composite event sequence number consisting of global + client + rebaseGeneration.
|
|
38
|
+
* Used for client-side event tracking with support for unconfirmed local events.
|
|
39
|
+
*
|
|
40
|
+
* For event notation documentation, see: contributor-docs/events-notation.md
|
|
41
|
+
*/
|
|
42
|
+
export type Composite = {
|
|
43
|
+
global: Global
|
|
44
|
+
client: Type
|
|
45
|
+
/**
|
|
46
|
+
* Generation integer that is incremented whenever the client rebased.
|
|
47
|
+
* Remains constant for all subsequent events until another rebase occurs.
|
|
48
|
+
*/
|
|
49
|
+
rebaseGeneration: number
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/** Input type for creating a Composite sequence number. Allows omitting rebaseGeneration (defaults to 0). */
|
|
53
|
+
export type CompositeInput =
|
|
54
|
+
| Composite
|
|
55
|
+
| (Omit<typeof CompositeSchema.Encoded, 'rebaseGeneration'> & { rebaseGeneration?: number })
|
|
56
|
+
|
|
57
|
+
/** A pair of sequence numbers representing an event and its parent. */
|
|
58
|
+
export type CompositePair = { seqNum: Composite; parentSeqNum: Composite }
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Compare two composite sequence numbers.
|
|
62
|
+
* Comparison hierarchy: global > client > rebaseGeneration
|
|
63
|
+
*/
|
|
64
|
+
export const compare = (a: Composite, b: Composite) => {
|
|
65
|
+
if (a.global !== b.global) {
|
|
66
|
+
return a.global - b.global
|
|
67
|
+
}
|
|
68
|
+
if (a.client !== b.client) {
|
|
69
|
+
return a.client - b.client
|
|
70
|
+
}
|
|
71
|
+
return a.rebaseGeneration - b.rebaseGeneration
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* Convert a composite sequence number to a string representation.
|
|
76
|
+
*
|
|
77
|
+
* For notation documentation, see: contributor-docs/events-notation.md
|
|
78
|
+
*/
|
|
79
|
+
export const toString = (seqNum: Composite) => {
|
|
80
|
+
const rebaseGenerationStr = seqNum.rebaseGeneration > 0 ? `r${seqNum.rebaseGeneration}` : ''
|
|
81
|
+
return seqNum.client === 0
|
|
82
|
+
? `e${seqNum.global}${rebaseGenerationStr}`
|
|
83
|
+
: `e${seqNum.global}.${seqNum.client}${rebaseGenerationStr}`
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Convert a string representation of a sequence number to a Composite.
|
|
88
|
+
* Parses strings in the format: e{global}[.{client}][r{rebaseGeneration}]
|
|
89
|
+
* Examples: "e0", "e0r1", "e0.1", "e0.1r1"
|
|
90
|
+
*
|
|
91
|
+
* For full notation documentation, see: contributor-docs/events-notation.md
|
|
92
|
+
*/
|
|
93
|
+
export const fromString = (str: string): Composite => {
|
|
94
|
+
if (!str.startsWith('e')) {
|
|
95
|
+
throw new Error('Invalid event sequence number string: must start with "e"')
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// Remove the 'e' prefix
|
|
99
|
+
const remaining = str.slice(1)
|
|
100
|
+
|
|
101
|
+
// Parse rebase generation if present
|
|
102
|
+
let rebaseGeneration = REBASE_GENERATION_DEFAULT
|
|
103
|
+
let withoutRebase = remaining
|
|
104
|
+
const rebaseMatch = remaining.match(/r(\d+)$/)
|
|
105
|
+
if (rebaseMatch !== null) {
|
|
106
|
+
rebaseGeneration = Number.parseInt(rebaseMatch[1]!, 10)
|
|
107
|
+
withoutRebase = remaining.slice(0, -rebaseMatch[0].length)
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// Parse global and client parts
|
|
111
|
+
const parts = withoutRebase.split('.')
|
|
112
|
+
|
|
113
|
+
// Validate that parts contain only digits (and possibly empty for client)
|
|
114
|
+
if (parts[0] === '' || !/^\d+$/.test(parts[0]!)) {
|
|
115
|
+
throw new Error('Invalid event sequence number string: invalid number format')
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
if (parts.length > 1 && parts[1] !== undefined && (parts[1] === '' || !/^\d+$/.test(parts[1]))) {
|
|
119
|
+
throw new Error('Invalid event sequence number string: invalid number format')
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
const global = Number.parseInt(parts[0]!, 10)
|
|
123
|
+
const client = parts.length > 1 && parts[1] !== undefined ? Number.parseInt(parts[1], 10) : 0
|
|
124
|
+
|
|
125
|
+
if (Number.isNaN(global) || Number.isNaN(client) || Number.isNaN(rebaseGeneration)) {
|
|
126
|
+
throw new TypeError('Invalid event sequence number string: invalid number format')
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
return {
|
|
130
|
+
global: global as any as Global,
|
|
131
|
+
client: client as any as Type,
|
|
132
|
+
rebaseGeneration,
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/** Creates a Composite sequence number from a global sequence number (client=0, rebaseGeneration=0). */
|
|
137
|
+
export const fromGlobal = (seqNum: Global): Composite => ({
|
|
138
|
+
global: seqNum,
|
|
139
|
+
client: DEFAULT,
|
|
140
|
+
rebaseGeneration: REBASE_GENERATION_DEFAULT,
|
|
141
|
+
})
|
|
142
|
+
|
|
143
|
+
/** Returns true if two Composite sequence numbers are structurally equal. */
|
|
144
|
+
export const isEqual = (a: Composite, b: Composite) =>
|
|
145
|
+
a.global === b.global && a.client === b.client && a.rebaseGeneration === b.rebaseGeneration
|
|
146
|
+
|
|
147
|
+
/** Returns true if `a` is strictly greater than `b` (compares global, then client). */
|
|
148
|
+
export const isGreaterThan = (a: Composite, b: Composite) => {
|
|
149
|
+
return a.global > b.global || (a.global === b.global && a.client > b.client)
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
/** Returns true if `a` is greater than or equal to `b` (compares global, then client). */
|
|
153
|
+
export const isGreaterThanOrEqual = (a: Composite, b: Composite) => {
|
|
154
|
+
return a.global > b.global || (a.global === b.global && a.client >= b.client)
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
/** Returns the larger of two Composite sequence numbers. */
|
|
158
|
+
export const max = (a: Composite, b: Composite) => {
|
|
159
|
+
return a.global > b.global || (a.global === b.global && a.client > b.client) ? a : b
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
/** Returns the difference between two Composite sequence numbers (a - b) for global and client components. */
|
|
163
|
+
export const diff = (a: Composite, b: Composite) => {
|
|
164
|
+
return {
|
|
165
|
+
global: a.global - b.global,
|
|
166
|
+
client: a.client - b.client,
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Schema for the composite event sequence number.
|
|
172
|
+
* NOTE: Client mutation events with a non-0 client id won't be synced to the sync backend.
|
|
173
|
+
*/
|
|
174
|
+
const CompositeSchema = S.Struct({
|
|
175
|
+
global: GlobalSchema,
|
|
176
|
+
/** Only increments for client-local events */
|
|
177
|
+
client: Schema,
|
|
178
|
+
// Client only
|
|
179
|
+
rebaseGeneration: S.Int,
|
|
180
|
+
}).annotations({
|
|
181
|
+
title: 'EventSequenceNumber.Composite',
|
|
182
|
+
pretty: () => (seqNum) => toString(seqNum),
|
|
183
|
+
})
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* Creates a validated Composite sequence number from input.
|
|
187
|
+
* If rebaseGeneration is omitted, defaults to REBASE_GENERATION_DEFAULT (0).
|
|
188
|
+
*/
|
|
189
|
+
const makeComposite = (seqNum: CompositeInput): Composite => {
|
|
190
|
+
return S.is(CompositeSchema)(seqNum)
|
|
191
|
+
? seqNum
|
|
192
|
+
: S.decodeSync(CompositeSchema)({
|
|
193
|
+
...seqNum,
|
|
194
|
+
rebaseGeneration: seqNum.rebaseGeneration ?? REBASE_GENERATION_DEFAULT,
|
|
195
|
+
})
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
/**
|
|
199
|
+
* Effect Schema for the composite event sequence number (global + client + rebaseGeneration).
|
|
200
|
+
* Also includes a `make` helper for creating validated Composite values.
|
|
201
|
+
*
|
|
202
|
+
* @example
|
|
203
|
+
* ```ts
|
|
204
|
+
* const seqNum: EventSequenceNumber.Client.Composite = {
|
|
205
|
+
* global: EventSequenceNumber.Global.make(5),
|
|
206
|
+
* client: EventSequenceNumber.Client.DEFAULT,
|
|
207
|
+
* rebaseGeneration: 0
|
|
208
|
+
* }
|
|
209
|
+
*
|
|
210
|
+
* const validated = EventSequenceNumber.Client.Composite.make({ global: 5, client: 0, rebaseGeneration: 0 })
|
|
211
|
+
* ```
|
|
212
|
+
*/
|
|
213
|
+
export const Composite = Object.assign(CompositeSchema, { make: makeComposite })
|
|
214
|
+
|
|
215
|
+
/** The root sequence number (global=0, client=0, rebaseGeneration=0). Parent of the first event. */
|
|
216
|
+
export const ROOT = {
|
|
217
|
+
global: makeGlobal(0),
|
|
218
|
+
client: DEFAULT,
|
|
219
|
+
rebaseGeneration: REBASE_GENERATION_DEFAULT,
|
|
220
|
+
} satisfies Composite
|
|
221
|
+
|
|
222
|
+
/**
|
|
223
|
+
* Computes the next sequence number and its parent based on the current position.
|
|
224
|
+
*
|
|
225
|
+
* For client-local events (isClient=true): increments the client component, keeps global.
|
|
226
|
+
* For global events (isClient=false): increments global, resets client to 0.
|
|
227
|
+
*/
|
|
228
|
+
export const nextPair = ({
|
|
229
|
+
seqNum,
|
|
230
|
+
isClient,
|
|
231
|
+
rebaseGeneration,
|
|
232
|
+
}: {
|
|
233
|
+
seqNum: Composite
|
|
234
|
+
isClient: boolean
|
|
235
|
+
rebaseGeneration?: number
|
|
236
|
+
}): CompositePair => {
|
|
237
|
+
if (isClient) {
|
|
238
|
+
return {
|
|
239
|
+
seqNum: {
|
|
240
|
+
global: seqNum.global,
|
|
241
|
+
client: (seqNum.client + 1) as any as Type,
|
|
242
|
+
rebaseGeneration: rebaseGeneration ?? seqNum.rebaseGeneration,
|
|
243
|
+
},
|
|
244
|
+
parentSeqNum: seqNum,
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
return {
|
|
249
|
+
seqNum: {
|
|
250
|
+
global: (seqNum.global + 1) as any as Global,
|
|
251
|
+
client: DEFAULT,
|
|
252
|
+
rebaseGeneration: rebaseGeneration ?? seqNum.rebaseGeneration,
|
|
253
|
+
},
|
|
254
|
+
// NOTE we always point to `client: 0` for non-client-local events
|
|
255
|
+
parentSeqNum: { global: seqNum.global, client: DEFAULT, rebaseGeneration: seqNum.rebaseGeneration },
|
|
256
|
+
}
|
|
257
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { Brand, Schema as S } from '@livestore/utils/effect'
|
|
2
|
+
|
|
3
|
+
/** Branded integer type for global sequence numbers assigned by the sync backend. */
|
|
4
|
+
export type Type = Brand.Branded<number, 'GlobalEventSequenceNumber'>
|
|
5
|
+
|
|
6
|
+
const GlobalBrand = Brand.nominal<Type>()
|
|
7
|
+
|
|
8
|
+
/** Effect Schema for encoding/decoding global sequence numbers. */
|
|
9
|
+
export const Schema = S.fromBrand(GlobalBrand)(S.Int)
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Creates a branded global sequence number from a plain number.
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```ts
|
|
16
|
+
* const seqNum = EventSequenceNumber.Global.make(5)
|
|
17
|
+
* ```
|
|
18
|
+
*/
|
|
19
|
+
export const make = GlobalBrand
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Client sequence numbers track events created locally before sync confirmation.
|
|
3
|
+
* Also contains the Composite type combining global and client sequence numbers with a rebase
|
|
4
|
+
* generation to fully identify an event's position in the eventlog.
|
|
5
|
+
*
|
|
6
|
+
* For event notation documentation, see: contributor-docs/events-notation.md
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```ts
|
|
10
|
+
* import { EventSequenceNumber } from '@livestore/common'
|
|
11
|
+
*
|
|
12
|
+
* // Client sequence number
|
|
13
|
+
* const clientSeq = EventSequenceNumber.Client.make(1)
|
|
14
|
+
*
|
|
15
|
+
* // Composite sequence number
|
|
16
|
+
* const composite: EventSequenceNumber.Client.Composite = {
|
|
17
|
+
* global: EventSequenceNumber.Global.make(5),
|
|
18
|
+
* client: EventSequenceNumber.Client.DEFAULT,
|
|
19
|
+
* rebaseGeneration: 0,
|
|
20
|
+
* }
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
export * as Client from './client.ts'
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Global sequence numbers are assigned by the sync backend and represent
|
|
27
|
+
* the canonical ordering of events across all clients. They are monotonically
|
|
28
|
+
* increasing integers that establish the authoritative event timeline.
|
|
29
|
+
*
|
|
30
|
+
* @example
|
|
31
|
+
* ```ts
|
|
32
|
+
* import { EventSequenceNumber } from '@livestore/common'
|
|
33
|
+
*
|
|
34
|
+
* const globalSeq = EventSequenceNumber.Global.make(5)
|
|
35
|
+
* ```
|
|
36
|
+
*/
|
|
37
|
+
export * as Global from './global.ts'
|