@livestore/common 0.4.0-dev.8 → 0.4.0-dev.9
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/devtools/devtools-messages-client-session.d.ts +21 -21
- package/dist/devtools/devtools-messages-common.d.ts +6 -6
- package/dist/devtools/devtools-messages-leader.d.ts +24 -24
- package/dist/testing/event-factory.d.ts +68 -0
- package/dist/testing/event-factory.d.ts.map +1 -0
- package/dist/testing/event-factory.js +80 -0
- package/dist/testing/event-factory.js.map +1 -0
- package/dist/testing/mod.d.ts +2 -0
- package/dist/testing/mod.d.ts.map +1 -0
- package/dist/testing/mod.js +2 -0
- package/dist/testing/mod.js.map +1 -0
- package/dist/version.d.ts +1 -1
- package/dist/version.js +1 -1
- package/package.json +6 -5
- package/src/testing/event-factory.ts +133 -0
- package/src/testing/mod.ts +1 -0
- package/src/version.ts +1 -1
@@ -0,0 +1,133 @@
|
|
1
|
+
/**
|
2
|
+
* Helpers for synthesizing LiveStore events in tests while keeping track of
|
3
|
+
* sequence numbers, parent pointers, and authoring client identity. Inspired
|
4
|
+
* by the effect-based schema utilities, the factory exposes a namespaced API
|
5
|
+
* where each event definition maps to a helper with `next`, `advanceTo`, and
|
6
|
+
* `setParent` functions that share a single sequence stream.
|
7
|
+
*
|
8
|
+
* @example
|
9
|
+
* ```ts
|
10
|
+
* import { EventFactory } from '@livestore/common/testing'
|
11
|
+
* import { events } from './schema'
|
12
|
+
*
|
13
|
+
* const eventFactory = EventFactory.makeFactory(events)({
|
14
|
+
* client: EventFactory.clientIdentity('test-client'),
|
15
|
+
* startSeq: 1,
|
16
|
+
* initialParent: 'root',
|
17
|
+
* })
|
18
|
+
*
|
19
|
+
* const bootstrap = eventFactory.todoCreated.next({
|
20
|
+
* id: 'todo-1',
|
21
|
+
* text: 'write tests',
|
22
|
+
* completed: false,
|
23
|
+
* })
|
24
|
+
*
|
25
|
+
* eventFactory.todoCreated.advanceTo(42)
|
26
|
+
* const branched = eventFactory.todoUpdated.next({
|
27
|
+
* id: 'todo-1',
|
28
|
+
* text: 'ship feature',
|
29
|
+
* completed: true,
|
30
|
+
* })
|
31
|
+
* ```
|
32
|
+
*/
|
33
|
+
|
34
|
+
import { Schema } from '@livestore/utils/effect'
|
35
|
+
|
36
|
+
import type { EventDef } from '../schema/EventDef.ts'
|
37
|
+
import * as EventSequenceNumber from '../schema/EventSequenceNumber.ts'
|
38
|
+
import * as LiveStoreEvent from '../schema/LiveStoreEvent.ts'
|
39
|
+
|
40
|
+
export interface ClientIdentity {
|
41
|
+
clientId: string
|
42
|
+
sessionId: string
|
43
|
+
}
|
44
|
+
|
45
|
+
export const clientIdentity = (label: string, session = `${label}-session`): ClientIdentity => ({
|
46
|
+
clientId: label,
|
47
|
+
sessionId: session,
|
48
|
+
})
|
49
|
+
|
50
|
+
export type SequenceValue = 'root' | number
|
51
|
+
|
52
|
+
type EventFactoriesArgs<TDefs extends Record<string, EventDef.Any>> = {
|
53
|
+
[K in keyof TDefs]: Parameters<TDefs[K]>[0]
|
54
|
+
}
|
55
|
+
|
56
|
+
type EventFactories<TDefs extends Record<string, EventDef.Any>, TResult> = {
|
57
|
+
[K in keyof TDefs]: {
|
58
|
+
next: (args: EventFactoriesArgs<TDefs>[K]) => TResult
|
59
|
+
advanceTo: (seq: number, parent?: SequenceValue) => void
|
60
|
+
setParent: (parent: SequenceValue) => void
|
61
|
+
current: () => { seq: number; parent: SequenceValue }
|
62
|
+
}
|
63
|
+
}
|
64
|
+
|
65
|
+
export interface EventFactoriesConfig {
|
66
|
+
client: ClientIdentity
|
67
|
+
/**
|
68
|
+
* @default 1
|
69
|
+
*/
|
70
|
+
startSeq?: number
|
71
|
+
/**
|
72
|
+
* @default 0 (root)
|
73
|
+
*/
|
74
|
+
initialParent?: SequenceValue
|
75
|
+
}
|
76
|
+
|
77
|
+
export const makeFactory =
|
78
|
+
<TDefs extends Record<string, EventDef.Any>>(eventDefs: TDefs) =>
|
79
|
+
({
|
80
|
+
client,
|
81
|
+
startSeq = 1,
|
82
|
+
initialParent = 'root',
|
83
|
+
}: EventFactoriesConfig): EventFactories<TDefs, LiveStoreEvent.AnyEncodedGlobal> => {
|
84
|
+
let nextSeq = startSeq
|
85
|
+
let parentRef: SequenceValue = initialParent
|
86
|
+
|
87
|
+
const advanceTo = (seq: number, parent: SequenceValue = 'root') => {
|
88
|
+
nextSeq = seq
|
89
|
+
parentRef = parent
|
90
|
+
}
|
91
|
+
|
92
|
+
const setParent = (parent: SequenceValue) => {
|
93
|
+
parentRef = parent
|
94
|
+
}
|
95
|
+
|
96
|
+
const current = () => ({ seq: nextSeq, parent: parentRef })
|
97
|
+
|
98
|
+
const factories: Partial<EventFactories<TDefs, LiveStoreEvent.AnyEncodedGlobal>> = {}
|
99
|
+
|
100
|
+
for (const [name, eventDef] of Object.entries(eventDefs) as [keyof TDefs, TDefs[keyof TDefs]][]) {
|
101
|
+
const next = (args: EventFactoriesArgs<TDefs>[typeof name]) => {
|
102
|
+
const decoded = eventDef(args)
|
103
|
+
const encodedArgs = Schema.encodeSync(eventDef.schema)(decoded.args)
|
104
|
+
const encoded = eventDef.encoded(encodedArgs)
|
105
|
+
|
106
|
+
const event = LiveStoreEvent.AnyEncodedGlobal.make({
|
107
|
+
name: encoded.name,
|
108
|
+
args: encoded.args,
|
109
|
+
seqNum: EventSequenceNumber.globalEventSequenceNumber(nextSeq),
|
110
|
+
parentSeqNum:
|
111
|
+
parentRef === 'root'
|
112
|
+
? EventSequenceNumber.ROOT.global
|
113
|
+
: EventSequenceNumber.globalEventSequenceNumber(parentRef),
|
114
|
+
clientId: client.clientId,
|
115
|
+
sessionId: client.sessionId,
|
116
|
+
})
|
117
|
+
|
118
|
+
parentRef = nextSeq
|
119
|
+
nextSeq = nextSeq + 1
|
120
|
+
|
121
|
+
return event
|
122
|
+
}
|
123
|
+
|
124
|
+
factories[name] = {
|
125
|
+
next,
|
126
|
+
advanceTo,
|
127
|
+
setParent,
|
128
|
+
current,
|
129
|
+
} as EventFactories<TDefs, LiveStoreEvent.AnyEncodedGlobal>[typeof name]
|
130
|
+
}
|
131
|
+
|
132
|
+
return factories as EventFactories<TDefs, LiveStoreEvent.AnyEncodedGlobal>
|
133
|
+
}
|
@@ -0,0 +1 @@
|
|
1
|
+
export * as EventFactory from './event-factory.ts'
|
package/src/version.ts
CHANGED
@@ -2,7 +2,7 @@
|
|
2
2
|
// import packageJson from '../package.json' with { type: 'json' }
|
3
3
|
// export const liveStoreVersion = packageJson.version
|
4
4
|
|
5
|
-
export const liveStoreVersion = '0.4.0-dev.
|
5
|
+
export const liveStoreVersion = '0.4.0-dev.9' as const
|
6
6
|
|
7
7
|
/**
|
8
8
|
* This version number is incremented whenever the internal storage format changes in a breaking way.
|