boundlessdb 0.4.0 → 0.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +85 -0
- package/README.md +199 -17
- package/dist/browser.d.ts +2 -2
- package/dist/browser.d.ts.map +1 -1
- package/dist/browser.js +1 -1
- package/dist/browser.js.map +1 -1
- package/dist/event-store.browser.d.ts.map +1 -1
- package/dist/event-store.browser.js +16 -34
- package/dist/event-store.browser.js.map +1 -1
- package/dist/event-store.d.ts.map +1 -1
- package/dist/event-store.js +14 -29
- package/dist/event-store.js.map +1 -1
- package/dist/index.d.ts +2 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/query-builder.d.ts +35 -3
- package/dist/query-builder.d.ts.map +1 -1
- package/dist/query-builder.js +63 -4
- package/dist/query-builder.js.map +1 -1
- package/dist/storage/memory.d.ts +17 -0
- package/dist/storage/memory.d.ts.map +1 -1
- package/dist/storage/memory.js +52 -6
- package/dist/storage/memory.js.map +1 -1
- package/dist/storage/postgres.d.ts +18 -0
- package/dist/storage/postgres.d.ts.map +1 -1
- package/dist/storage/postgres.js +150 -138
- package/dist/storage/postgres.js.map +1 -1
- package/dist/storage/sqlite.d.ts +21 -0
- package/dist/storage/sqlite.d.ts.map +1 -1
- package/dist/storage/sqlite.js +139 -13
- package/dist/storage/sqlite.js.map +1 -1
- package/dist/storage/sqljs.d.ts +13 -0
- package/dist/storage/sqljs.d.ts.map +1 -1
- package/dist/storage/sqljs.js +139 -45
- package/dist/storage/sqljs.js.map +1 -1
- package/dist/types.d.ts +40 -5
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js +25 -1
- package/dist/types.js.map +1 -1
- package/package.json +1 -1
package/dist/event-store.js
CHANGED
|
@@ -5,7 +5,7 @@ import { randomUUID, createHash } from 'node:crypto';
|
|
|
5
5
|
import { KeyExtractor } from './config/extractor.js';
|
|
6
6
|
import { validateConfig } from './config/validator.js';
|
|
7
7
|
import { SqliteStorage } from './storage/sqlite.js';
|
|
8
|
-
import { QueryResult,
|
|
8
|
+
import { QueryResult, normalizeCondition, hasKeys, } from './types.js';
|
|
9
9
|
import { QueryBuilder } from './query-builder.js';
|
|
10
10
|
/**
|
|
11
11
|
* Recursively sort object keys for deterministic JSON
|
|
@@ -61,32 +61,12 @@ export class EventStore {
|
|
|
61
61
|
const storedHash = this.storage.getConfigHash();
|
|
62
62
|
if (storedHash === null) {
|
|
63
63
|
// First run — just store the hash
|
|
64
|
-
console.log('[EventStore] First run, storing config hash:', currentHash.substring(0, 16) + '...');
|
|
65
64
|
this.storage.setConfigHash(currentHash);
|
|
66
65
|
}
|
|
67
66
|
else if (storedHash !== currentHash) {
|
|
68
|
-
// Config changed — reindex
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
console.log(`[EventStore] New hash: ${currentHash.substring(0, 16)}...`);
|
|
72
|
-
const startTime = Date.now();
|
|
73
|
-
let eventCount = 0;
|
|
74
|
-
let keyCount = 0;
|
|
75
|
-
this.storage.reindex((event) => {
|
|
76
|
-
eventCount++;
|
|
77
|
-
// Convert StoredEvent to Event format for KeyExtractor
|
|
78
|
-
const keys = this.keyExtractor.extract({
|
|
79
|
-
type: event.type,
|
|
80
|
-
data: event.data,
|
|
81
|
-
metadata: event.metadata
|
|
82
|
-
});
|
|
83
|
-
keyCount += keys.length;
|
|
84
|
-
return keys;
|
|
85
|
-
});
|
|
86
|
-
// Update stored hash
|
|
87
|
-
this.storage.setConfigHash(currentHash);
|
|
88
|
-
const duration = Date.now() - startTime;
|
|
89
|
-
console.log(`[EventStore] ✅ Reindex complete: ${eventCount} events, ${keyCount} keys (${duration}ms)`);
|
|
67
|
+
// Config changed — throw error, require explicit reindex via script
|
|
68
|
+
throw new Error(`Config hash mismatch (stored: ${storedHash}, current: ${currentHash}). ` +
|
|
69
|
+
`Run the reindex script before starting the application.`);
|
|
90
70
|
}
|
|
91
71
|
}
|
|
92
72
|
/**
|
|
@@ -210,22 +190,27 @@ export class EventStore {
|
|
|
210
190
|
*/
|
|
211
191
|
buildConditionsFromEvents(events, originalCondition) {
|
|
212
192
|
// Start with original conditions if provided
|
|
193
|
+
// Use a Map with a stable key to deduplicate
|
|
213
194
|
const conditions = new Map();
|
|
214
195
|
if (originalCondition !== null) {
|
|
215
196
|
for (const cond of originalCondition.failIfEventsMatch) {
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
197
|
+
const normalized = normalizeCondition(cond);
|
|
198
|
+
if (hasKeys(normalized)) {
|
|
199
|
+
// Build a dedup key from all keys
|
|
200
|
+
const keysStr = normalized.keys.map(k => `${k.name}:${k.value}`).sort().join('|');
|
|
201
|
+
const dedupKey = `${normalized.type}:${keysStr}`;
|
|
202
|
+
conditions.set(dedupKey, normalized);
|
|
219
203
|
}
|
|
220
204
|
}
|
|
221
205
|
}
|
|
222
206
|
// Add conditions from the newly appended events
|
|
207
|
+
// Each event generates one condition per extracted key (single-key conditions)
|
|
223
208
|
for (const event of events) {
|
|
224
209
|
const extractedKeys = this.keyExtractor.extract(event);
|
|
225
210
|
for (const extracted of extractedKeys) {
|
|
226
211
|
const cond = { type: event.type, key: extracted.name, value: extracted.value };
|
|
227
|
-
const
|
|
228
|
-
conditions.set(
|
|
212
|
+
const dedupKey = `${cond.type}:${extracted.name}:${extracted.value}`;
|
|
213
|
+
conditions.set(dedupKey, cond);
|
|
229
214
|
}
|
|
230
215
|
}
|
|
231
216
|
return Array.from(conditions.values());
|
package/dist/event-store.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"event-store.js","sourceRoot":"","sources":["../src/event-store.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAEvD,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AACpD,OAAO,EACL,WAAW,
|
|
1
|
+
{"version":3,"file":"event-store.js","sourceRoot":"","sources":["../src/event-store.ts"],"names":[],"mappings":"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"}
|
package/dist/index.d.ts
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* DCB Event Store - Public API
|
|
3
3
|
*/
|
|
4
|
-
export type { Event, EventWithMetadata, StoredEvent, Query, QueryCondition, UnconstrainedCondition, ConstrainedCondition, ConsistencyConfig, ConsistencyKeyDef, EventTypeConfig, ExtractedKey, AppendResult, ConflictResult, AppendCondition, EventStoreOptions, } from './types.js';
|
|
5
|
-
export { QueryResult, isConflict, isConstrainedCondition } from './types.js';
|
|
4
|
+
export type { Event, EventWithMetadata, StoredEvent, Query, QueryCondition, UnconstrainedCondition, ConstrainedCondition, MultiKeyConstrainedCondition, ConsistencyConfig, ConsistencyKeyDef, EventTypeConfig, ExtractedKey, AppendResult, ConflictResult, AppendCondition, EventStoreOptions, } from './types.js';
|
|
5
|
+
export { QueryResult, isConflict, isConstrainedCondition, isMultiKeyCondition, normalizeCondition, hasKeys } from './types.js';
|
|
6
6
|
export { EventStore, createEventStore, type EventStoreConfig } from './event-store.js';
|
|
7
7
|
export type { EventStorage, EventToStore } from './storage/interface.js';
|
|
8
8
|
export { InMemoryStorage } from './storage/memory.js';
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,YAAY,EAEV,KAAK,EACL,iBAAiB,EACjB,WAAW,EAEX,KAAK,EACL,cAAc,EACd,sBAAsB,EACtB,oBAAoB,
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,YAAY,EAEV,KAAK,EACL,iBAAiB,EACjB,WAAW,EAEX,KAAK,EACL,cAAc,EACd,sBAAsB,EACtB,oBAAoB,EACpB,4BAA4B,EAE5B,iBAAiB,EACjB,iBAAiB,EACjB,eAAe,EACf,YAAY,EAEZ,YAAY,EACZ,cAAc,EACd,eAAe,EACf,iBAAiB,GAClB,MAAM,YAAY,CAAC;AAEpB,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,kBAAkB,EAAE,OAAO,EAAE,MAAM,YAAY,CAAC;AAG/H,OAAO,EAAE,UAAU,EAAE,gBAAgB,EAAE,KAAK,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAGvF,YAAY,EAAE,YAAY,EAAE,YAAY,EAAE,MAAM,wBAAwB,CAAC;AACzE,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AACpD,OAAO,EAAE,YAAY,EAAE,KAAK,mBAAmB,EAAE,MAAM,oBAAoB,CAAC;AAC5E,OAAO,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAGxD,OAAO,EAAE,YAAY,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,qBAAqB,EAAE,MAAM,uBAAuB,CAAC;AAG9E,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC"}
|
package/dist/index.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* DCB Event Store - Public API
|
|
3
3
|
*/
|
|
4
|
-
export { QueryResult, isConflict, isConstrainedCondition } from './types.js';
|
|
4
|
+
export { QueryResult, isConflict, isConstrainedCondition, isMultiKeyCondition, normalizeCondition, hasKeys } from './types.js';
|
|
5
5
|
// Event Store
|
|
6
6
|
export { EventStore, createEventStore } from './event-store.js';
|
|
7
7
|
export { InMemoryStorage } from './storage/memory.js';
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;GAEG;
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AA0BH,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,kBAAkB,EAAE,OAAO,EAAE,MAAM,YAAY,CAAC;AAE/H,cAAc;AACd,OAAO,EAAE,UAAU,EAAE,gBAAgB,EAAyB,MAAM,kBAAkB,CAAC;AAIvF,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AACpD,OAAO,EAAE,YAAY,EAA4B,MAAM,oBAAoB,CAAC;AAC5E,OAAO,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAExD,SAAS;AACT,OAAO,EAAE,YAAY,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,qBAAqB,EAAE,MAAM,uBAAuB,CAAC;AAE9E,gBAAgB;AAChB,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC"}
|
package/dist/query-builder.d.ts
CHANGED
|
@@ -14,11 +14,24 @@ export interface QueryExecutor<E extends Event> {
|
|
|
14
14
|
*
|
|
15
15
|
* @example
|
|
16
16
|
* ```typescript
|
|
17
|
+
* // Single-key query (backward compatible):
|
|
17
18
|
* const result = await store.query<CourseEvent>()
|
|
18
|
-
* .matchType('CourseCreated')
|
|
19
19
|
* .matchTypeAndKey('StudentSubscribed', 'course', 'cs101')
|
|
20
|
-
* .
|
|
21
|
-
*
|
|
20
|
+
* .read();
|
|
21
|
+
*
|
|
22
|
+
* // Multi-key AND query:
|
|
23
|
+
* const result = await store.query<CourseEvent>()
|
|
24
|
+
* .matchType('StudentSubscribed')
|
|
25
|
+
* .withKey('course', 'cs101')
|
|
26
|
+
* .withKey('student', 'alice')
|
|
27
|
+
* .read();
|
|
28
|
+
*
|
|
29
|
+
* // Mixed (AND + OR):
|
|
30
|
+
* const result = await store.query<CourseEvent>()
|
|
31
|
+
* .matchType('StudentSubscribed')
|
|
32
|
+
* .withKey('course', 'cs101')
|
|
33
|
+
* .withKey('student', 'alice') // AND on condition 0
|
|
34
|
+
* .matchTypeAndKey('CourseCancelled', 'course', 'cs101') // OR (condition 1)
|
|
22
35
|
* .read();
|
|
23
36
|
* ```
|
|
24
37
|
*/
|
|
@@ -30,22 +43,41 @@ export declare class QueryBuilder<E extends Event> {
|
|
|
30
43
|
constructor(executor: QueryExecutor<E>);
|
|
31
44
|
/**
|
|
32
45
|
* Add an unconstrained condition (match all events of type).
|
|
46
|
+
* Use `.withKey()` after to add key constraints (AND).
|
|
33
47
|
*
|
|
34
48
|
* @example
|
|
35
49
|
* ```typescript
|
|
36
50
|
* .matchType('CourseCreated') // matches ALL CourseCreated events
|
|
51
|
+
* .matchType('StudentSubscribed').withKey('course', 'cs101') // type + key
|
|
37
52
|
* ```
|
|
38
53
|
*/
|
|
39
54
|
matchType(type: string): this;
|
|
40
55
|
/**
|
|
41
56
|
* Add a constrained condition (match events of type where key equals value).
|
|
57
|
+
* Shorthand for `.matchType(type).withKey(key, value)`.
|
|
58
|
+
* Use `.withKey()` after to add more key constraints (AND).
|
|
42
59
|
*
|
|
43
60
|
* @example
|
|
44
61
|
* ```typescript
|
|
45
62
|
* .matchTypeAndKey('StudentSubscribed', 'course', 'cs101')
|
|
63
|
+
* .matchTypeAndKey('StudentSubscribed', 'course', 'cs101').withKey('student', 'alice')
|
|
46
64
|
* ```
|
|
47
65
|
*/
|
|
48
66
|
matchTypeAndKey(type: string, key: string, value: string): this;
|
|
67
|
+
/**
|
|
68
|
+
* Add a key constraint to the last condition (AND).
|
|
69
|
+
* Must be called after `.matchType()` or `.matchTypeAndKey()`.
|
|
70
|
+
*
|
|
71
|
+
* @throws Error if no preceding condition exists
|
|
72
|
+
*
|
|
73
|
+
* @example
|
|
74
|
+
* ```typescript
|
|
75
|
+
* .matchType('StudentSubscribed')
|
|
76
|
+
* .withKey('course', 'cs101')
|
|
77
|
+
* .withKey('student', 'alice') // AND: both keys must match
|
|
78
|
+
* ```
|
|
79
|
+
*/
|
|
80
|
+
withKey(key: string, value: string): this;
|
|
49
81
|
/**
|
|
50
82
|
* Start reading from a specific position.
|
|
51
83
|
*
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"query-builder.d.ts","sourceRoot":"","sources":["../src/query-builder.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,KAAK,EAAE,cAAc,EAAE,WAAW,
|
|
1
|
+
{"version":3,"file":"query-builder.d.ts","sourceRoot":"","sources":["../src/query-builder.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,KAAK,EAAE,cAAc,EAAE,WAAW,EAAgC,MAAM,YAAY,CAAC;AAEnG,MAAM,WAAW,aAAa,CAAC,CAAC,SAAS,KAAK;IAC5C,IAAI,CAAC,KAAK,EAAE;QACV,UAAU,EAAE,cAAc,EAAE,CAAC;QAC7B,YAAY,CAAC,EAAE,MAAM,CAAC;QACtB,KAAK,CAAC,EAAE,MAAM,CAAC;KAChB,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;CAC7B;AAED;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AACH,qBAAa,YAAY,CAAC,CAAC,SAAS,KAAK;IAK3B,OAAO,CAAC,QAAQ,CAAC,QAAQ;IAJrC,OAAO,CAAC,UAAU,CAAwB;IAC1C,OAAO,CAAC,aAAa,CAAC,CAAS;IAC/B,OAAO,CAAC,MAAM,CAAC,CAAS;gBAEK,QAAQ,EAAE,aAAa,CAAC,CAAC,CAAC;IAEvD;;;;;;;;;OASG;IACH,SAAS,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAK7B;;;;;;;;;;OAUG;IACH,eAAe,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI;IAK/D;;;;;;;;;;;;OAYG;IACH,OAAO,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI;IA8BzC;;;;;;;OAOG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAKpC;;;;;;;OAOG;IACH,KAAK,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAK1B;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;CAOtC"}
|
package/dist/query-builder.js
CHANGED
|
@@ -6,11 +6,24 @@
|
|
|
6
6
|
*
|
|
7
7
|
* @example
|
|
8
8
|
* ```typescript
|
|
9
|
+
* // Single-key query (backward compatible):
|
|
9
10
|
* const result = await store.query<CourseEvent>()
|
|
10
|
-
* .matchType('CourseCreated')
|
|
11
11
|
* .matchTypeAndKey('StudentSubscribed', 'course', 'cs101')
|
|
12
|
-
* .
|
|
13
|
-
*
|
|
12
|
+
* .read();
|
|
13
|
+
*
|
|
14
|
+
* // Multi-key AND query:
|
|
15
|
+
* const result = await store.query<CourseEvent>()
|
|
16
|
+
* .matchType('StudentSubscribed')
|
|
17
|
+
* .withKey('course', 'cs101')
|
|
18
|
+
* .withKey('student', 'alice')
|
|
19
|
+
* .read();
|
|
20
|
+
*
|
|
21
|
+
* // Mixed (AND + OR):
|
|
22
|
+
* const result = await store.query<CourseEvent>()
|
|
23
|
+
* .matchType('StudentSubscribed')
|
|
24
|
+
* .withKey('course', 'cs101')
|
|
25
|
+
* .withKey('student', 'alice') // AND on condition 0
|
|
26
|
+
* .matchTypeAndKey('CourseCancelled', 'course', 'cs101') // OR (condition 1)
|
|
14
27
|
* .read();
|
|
15
28
|
* ```
|
|
16
29
|
*/
|
|
@@ -24,10 +37,12 @@ export class QueryBuilder {
|
|
|
24
37
|
}
|
|
25
38
|
/**
|
|
26
39
|
* Add an unconstrained condition (match all events of type).
|
|
40
|
+
* Use `.withKey()` after to add key constraints (AND).
|
|
27
41
|
*
|
|
28
42
|
* @example
|
|
29
43
|
* ```typescript
|
|
30
44
|
* .matchType('CourseCreated') // matches ALL CourseCreated events
|
|
45
|
+
* .matchType('StudentSubscribed').withKey('course', 'cs101') // type + key
|
|
31
46
|
* ```
|
|
32
47
|
*/
|
|
33
48
|
matchType(type) {
|
|
@@ -36,14 +51,58 @@ export class QueryBuilder {
|
|
|
36
51
|
}
|
|
37
52
|
/**
|
|
38
53
|
* Add a constrained condition (match events of type where key equals value).
|
|
54
|
+
* Shorthand for `.matchType(type).withKey(key, value)`.
|
|
55
|
+
* Use `.withKey()` after to add more key constraints (AND).
|
|
39
56
|
*
|
|
40
57
|
* @example
|
|
41
58
|
* ```typescript
|
|
42
59
|
* .matchTypeAndKey('StudentSubscribed', 'course', 'cs101')
|
|
60
|
+
* .matchTypeAndKey('StudentSubscribed', 'course', 'cs101').withKey('student', 'alice')
|
|
43
61
|
* ```
|
|
44
62
|
*/
|
|
45
63
|
matchTypeAndKey(type, key, value) {
|
|
46
|
-
this.conditions.push({ type, key, value });
|
|
64
|
+
this.conditions.push({ type, keys: [{ name: key, value }] });
|
|
65
|
+
return this;
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Add a key constraint to the last condition (AND).
|
|
69
|
+
* Must be called after `.matchType()` or `.matchTypeAndKey()`.
|
|
70
|
+
*
|
|
71
|
+
* @throws Error if no preceding condition exists
|
|
72
|
+
*
|
|
73
|
+
* @example
|
|
74
|
+
* ```typescript
|
|
75
|
+
* .matchType('StudentSubscribed')
|
|
76
|
+
* .withKey('course', 'cs101')
|
|
77
|
+
* .withKey('student', 'alice') // AND: both keys must match
|
|
78
|
+
* ```
|
|
79
|
+
*/
|
|
80
|
+
withKey(key, value) {
|
|
81
|
+
if (this.conditions.length === 0) {
|
|
82
|
+
throw new Error('.withKey() requires a preceding .matchType() or .matchTypeAndKey()');
|
|
83
|
+
}
|
|
84
|
+
const lastIdx = this.conditions.length - 1;
|
|
85
|
+
const last = this.conditions[lastIdx];
|
|
86
|
+
// Convert the last condition to multi-key format if needed
|
|
87
|
+
if ('keys' in last && Array.isArray(last.keys)) {
|
|
88
|
+
// Already multi-key format — add to it
|
|
89
|
+
last.keys.push({ name: key, value });
|
|
90
|
+
}
|
|
91
|
+
else if ('key' in last && 'value' in last) {
|
|
92
|
+
// Legacy single-key format — convert to multi-key
|
|
93
|
+
const legacy = last;
|
|
94
|
+
this.conditions[lastIdx] = {
|
|
95
|
+
type: legacy.type,
|
|
96
|
+
keys: [{ name: legacy.key, value: legacy.value }, { name: key, value }],
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
// Unconstrained (type only) — convert to multi-key
|
|
101
|
+
this.conditions[lastIdx] = {
|
|
102
|
+
type: last.type,
|
|
103
|
+
keys: [{ name: key, value }],
|
|
104
|
+
};
|
|
105
|
+
}
|
|
47
106
|
return this;
|
|
48
107
|
}
|
|
49
108
|
/**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"query-builder.js","sourceRoot":"","sources":["../src/query-builder.ts"],"names":[],"mappings":"AAAA;;GAEG;AAYH
|
|
1
|
+
{"version":3,"file":"query-builder.js","sourceRoot":"","sources":["../src/query-builder.ts"],"names":[],"mappings":"AAAA;;GAEG;AAYH;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AACH,MAAM,OAAO,YAAY;IAKM;IAJrB,UAAU,GAAqB,EAAE,CAAC;IAClC,aAAa,CAAU;IACvB,MAAM,CAAU;IAExB,YAA6B,QAA0B;QAA1B,aAAQ,GAAR,QAAQ,CAAkB;IAAG,CAAC;IAE3D;;;;;;;;;OASG;IACH,SAAS,CAAC,IAAY;QACpB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QAC/B,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;;;;OAUG;IACH,eAAe,CAAC,IAAY,EAAE,GAAW,EAAE,KAAa;QACtD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,EAAkC,CAAC,CAAC;QAC7F,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;;;;;;OAYG;IACH,OAAO,CAAC,GAAW,EAAE,KAAa;QAChC,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACjC,MAAM,IAAI,KAAK,CAAC,oEAAoE,CAAC,CAAC;QACxF,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC;QAC3C,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAEtC,2DAA2D;QAC3D,IAAI,MAAM,IAAI,IAAI,IAAI,KAAK,CAAC,OAAO,CAAE,IAAqC,CAAC,IAAI,CAAC,EAAE,CAAC;YACjF,uCAAuC;YACtC,IAAqC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;QACzE,CAAC;aAAM,IAAI,KAAK,IAAI,IAAI,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;YAC5C,kDAAkD;YAClD,MAAM,MAAM,GAAG,IAAoD,CAAC;YACpE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,GAAG;gBACzB,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,CAAC,GAAG,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,EAAE,EAAE,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC;aACxC,CAAC;QACpC,CAAC;aAAM,CAAC;YACN,mDAAmD;YACnD,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,GAAG;gBACzB,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC;aACG,CAAC;QACpC,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;OAOG;IACH,YAAY,CAAC,QAAgB;QAC3B,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC;QAC9B,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,KAAa;QACjB,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QACpB,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,IAAI;QACR,OAAO,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;YACxB,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,YAAY,EAAE,IAAI,CAAC,aAAa;YAChC,KAAK,EAAE,IAAI,CAAC,MAAM;SACnB,CAAC,CAAC;IACL,CAAC;CACF"}
|
package/dist/storage/memory.d.ts
CHANGED
|
@@ -22,5 +22,22 @@ export declare class InMemoryStorage implements EventStorage {
|
|
|
22
22
|
* Clear all events (for testing)
|
|
23
23
|
*/
|
|
24
24
|
clear(): void;
|
|
25
|
+
/**
|
|
26
|
+
* Reindex all events with new keys (in-memory: just re-extract all keys in one go)
|
|
27
|
+
* @deprecated Use reindexBatch() instead
|
|
28
|
+
*/
|
|
29
|
+
reindex(extractKeys: (event: StoredEvent) => ExtractedKey[]): void;
|
|
30
|
+
/**
|
|
31
|
+
* Batch-based reindex for in-memory storage.
|
|
32
|
+
* Since everything is in memory, no batching needed — just re-extracts all keys.
|
|
33
|
+
*/
|
|
34
|
+
reindexBatch(extractKeys: (event: StoredEvent) => ExtractedKey[], options?: {
|
|
35
|
+
batchSize?: number;
|
|
36
|
+
onProgress?: (done: number, total: number) => void;
|
|
37
|
+
}): {
|
|
38
|
+
events: number;
|
|
39
|
+
keys: number;
|
|
40
|
+
durationMs: number;
|
|
41
|
+
};
|
|
25
42
|
}
|
|
26
43
|
//# sourceMappingURL=memory.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"memory.d.ts","sourceRoot":"","sources":["../../src/storage/memory.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,
|
|
1
|
+
{"version":3,"file":"memory.d.ts","sourceRoot":"","sources":["../../src/storage/memory.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAuD,KAAK,YAAY,EAAE,KAAK,cAAc,EAAE,KAAK,WAAW,EAAqC,MAAM,aAAa,CAAC;AAC/K,OAAO,KAAK,EAAE,YAAY,EAAE,YAAY,EAAE,sBAAsB,EAAE,yBAAyB,EAAE,MAAM,gBAAgB,CAAC;AAMpH;;;GAGG;AACH,qBAAa,eAAgB,YAAW,YAAY;IAClD,OAAO,CAAC,MAAM,CAA6B;IAC3C,OAAO,CAAC,YAAY,CAAc;IAE5B,mBAAmB,CACvB,aAAa,EAAE,YAAY,EAAE,EAC7B,IAAI,EAAE,YAAY,EAAE,EAAE,EACtB,SAAS,EAAE,sBAAsB,GAAG,IAAI,GACvC,OAAO,CAAC,yBAAyB,CAAC;IA8C/B,KAAK,CACT,UAAU,EAAE,cAAc,EAAE,EAC5B,YAAY,CAAC,EAAE,MAAM,EACrB,KAAK,CAAC,EAAE,MAAM,GACb,OAAO,CAAC,WAAW,EAAE,CAAC;IAkDnB,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC;IAOpC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAI5B;;OAEG;IACH,YAAY,IAAI,WAAW,EAAE;IAI7B;;OAEG;IACH,KAAK,IAAI,IAAI;IAKb;;;OAGG;IACH,OAAO,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,YAAY,EAAE,GAAG,IAAI;IAclE;;;OAGG;IACH,YAAY,CACV,WAAW,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,YAAY,EAAE,EACnD,OAAO,CAAC,EAAE;QACR,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,UAAU,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;KACpD,GACA;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,MAAM,CAAA;KAAE;CA0BxD"}
|
package/dist/storage/memory.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* In-Memory Storage implementation (for testing)
|
|
3
3
|
*/
|
|
4
|
-
import {
|
|
4
|
+
import { normalizeCondition, hasKeys } from '../types.js';
|
|
5
5
|
/**
|
|
6
6
|
* In-memory event storage for testing purposes
|
|
7
7
|
* NOT suitable for production use
|
|
@@ -54,20 +54,22 @@ export class InMemoryStorage {
|
|
|
54
54
|
const limited = limit !== undefined ? matching.slice(0, limit) : matching;
|
|
55
55
|
return limited.map(({ keys: _keys, ...event }) => event);
|
|
56
56
|
}
|
|
57
|
+
// Normalize conditions to internal format
|
|
58
|
+
const normalized = conditions.map(normalizeCondition);
|
|
57
59
|
// Filter by conditions
|
|
58
60
|
matching = matching.filter(event => {
|
|
59
|
-
// Must match at least one condition
|
|
60
|
-
return
|
|
61
|
+
// Must match at least one condition (OR across conditions)
|
|
62
|
+
return normalized.some(cond => {
|
|
61
63
|
// Type must match
|
|
62
64
|
if (event.type !== cond.type) {
|
|
63
65
|
return false;
|
|
64
66
|
}
|
|
65
67
|
// If unconstrained, type match is enough
|
|
66
|
-
if (!
|
|
68
|
+
if (!hasKeys(cond)) {
|
|
67
69
|
return true;
|
|
68
70
|
}
|
|
69
|
-
// Constrained:
|
|
70
|
-
return event.keys.some(
|
|
71
|
+
// Constrained: ALL keys must match (AND within a condition)
|
|
72
|
+
return cond.keys.every(requiredKey => event.keys.some(eventKey => eventKey.name === requiredKey.name && eventKey.value === requiredKey.value));
|
|
71
73
|
});
|
|
72
74
|
});
|
|
73
75
|
// Sort by position
|
|
@@ -99,5 +101,49 @@ export class InMemoryStorage {
|
|
|
99
101
|
this.events = [];
|
|
100
102
|
this.nextPosition = 1n;
|
|
101
103
|
}
|
|
104
|
+
/**
|
|
105
|
+
* Reindex all events with new keys (in-memory: just re-extract all keys in one go)
|
|
106
|
+
* @deprecated Use reindexBatch() instead
|
|
107
|
+
*/
|
|
108
|
+
reindex(extractKeys) {
|
|
109
|
+
for (const internal of this.events) {
|
|
110
|
+
const event = {
|
|
111
|
+
id: internal.id,
|
|
112
|
+
type: internal.type,
|
|
113
|
+
data: internal.data,
|
|
114
|
+
metadata: internal.metadata,
|
|
115
|
+
timestamp: internal.timestamp,
|
|
116
|
+
position: internal.position,
|
|
117
|
+
};
|
|
118
|
+
internal.keys = extractKeys(event);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Batch-based reindex for in-memory storage.
|
|
123
|
+
* Since everything is in memory, no batching needed — just re-extracts all keys.
|
|
124
|
+
*/
|
|
125
|
+
reindexBatch(extractKeys, options) {
|
|
126
|
+
const onProgress = options?.onProgress;
|
|
127
|
+
const startTime = Date.now();
|
|
128
|
+
let totalKeys = 0;
|
|
129
|
+
for (let i = 0; i < this.events.length; i++) {
|
|
130
|
+
const internal = this.events[i];
|
|
131
|
+
const event = {
|
|
132
|
+
id: internal.id,
|
|
133
|
+
type: internal.type,
|
|
134
|
+
data: internal.data,
|
|
135
|
+
metadata: internal.metadata,
|
|
136
|
+
timestamp: internal.timestamp,
|
|
137
|
+
position: internal.position,
|
|
138
|
+
};
|
|
139
|
+
const keys = extractKeys(event);
|
|
140
|
+
internal.keys = keys;
|
|
141
|
+
totalKeys += keys.length;
|
|
142
|
+
if (onProgress) {
|
|
143
|
+
onProgress(i + 1, this.events.length);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
return { events: this.events.length, keys: totalKeys, durationMs: Date.now() - startTime };
|
|
147
|
+
}
|
|
102
148
|
}
|
|
103
149
|
//# sourceMappingURL=memory.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"memory.js","sourceRoot":"","sources":["../../src/storage/memory.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"memory.js","sourceRoot":"","sources":["../../src/storage/memory.ts"],"names":[],"mappings":"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"}
|
|
@@ -39,6 +39,11 @@ export declare class PostgresStorage implements EventStorage {
|
|
|
39
39
|
init(): Promise<void>;
|
|
40
40
|
private ensureInitialized;
|
|
41
41
|
appendWithCondition(eventsToStore: EventToStore[], keys: ExtractedKey[][], condition: StorageAppendCondition | null): Promise<AppendWithConditionResult>;
|
|
42
|
+
/**
|
|
43
|
+
* Build PostgreSQL query from normalized conditions.
|
|
44
|
+
* Shared between queryWithClient and query methods.
|
|
45
|
+
*/
|
|
46
|
+
private buildPostgresQuery;
|
|
42
47
|
/**
|
|
43
48
|
* Query with specific client (for use within transactions)
|
|
44
49
|
*/
|
|
@@ -72,8 +77,21 @@ export declare class PostgresStorage implements EventStorage {
|
|
|
72
77
|
setConfigHash(hash: string): Promise<void>;
|
|
73
78
|
/**
|
|
74
79
|
* Reindex all events with new keys
|
|
80
|
+
* @deprecated Use reindexBatch() for production-safe batch-based reindexing
|
|
75
81
|
*/
|
|
76
82
|
reindex(extractKeys: (event: StoredEvent) => ExtractedKey[]): Promise<void>;
|
|
83
|
+
/**
|
|
84
|
+
* Batch-based reindex: processes events in cursor-based batches.
|
|
85
|
+
* Crash-safe via reindex_position metadata. Resumes from last completed batch.
|
|
86
|
+
*/
|
|
87
|
+
reindexBatch(extractKeys: (event: StoredEvent) => ExtractedKey[], options?: {
|
|
88
|
+
batchSize?: number;
|
|
89
|
+
onProgress?: (done: number, total: number) => void;
|
|
90
|
+
}): Promise<{
|
|
91
|
+
events: number;
|
|
92
|
+
keys: number;
|
|
93
|
+
durationMs: number;
|
|
94
|
+
}>;
|
|
77
95
|
private rowToEvent;
|
|
78
96
|
}
|
|
79
97
|
//# sourceMappingURL=postgres.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"postgres.d.ts","sourceRoot":"","sources":["../../src/storage/postgres.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAoB,KAAK,UAAU,EAAE,MAAM,IAAI,CAAC;AACvD,OAAO,
|
|
1
|
+
{"version":3,"file":"postgres.d.ts","sourceRoot":"","sources":["../../src/storage/postgres.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAoB,KAAK,UAAU,EAAE,MAAM,IAAI,CAAC;AACvD,OAAO,EAA4E,KAAK,YAAY,EAAE,KAAK,cAAc,EAAE,KAAK,WAAW,EAAkE,MAAM,aAAa,CAAC;AACjO,OAAO,KAAK,EAAE,YAAY,EAAE,YAAY,EAAE,sBAAsB,EAAE,yBAAyB,EAAE,MAAM,gBAAgB,CAAC;AA2CpH;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AACH,qBAAa,eAAgB,YAAW,YAAY;IAClD,OAAO,CAAC,IAAI,CAAO;IACnB,OAAO,CAAC,WAAW,CAAS;gBAEhB,wBAAwB,EAAE,MAAM,GAAG,UAAU;IAQzD;;;OAGG;IACG,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAY3B,OAAO,CAAC,iBAAiB;IAMnB,mBAAmB,CACvB,aAAa,EAAE,YAAY,EAAE,EAC7B,IAAI,EAAE,YAAY,EAAE,EAAE,EACtB,SAAS,EAAE,sBAAsB,GAAG,IAAI,GACvC,OAAO,CAAC,yBAAyB,CAAC;IAwGrC;;;OAGG;IACH,OAAO,CAAC,kBAAkB;IAmK1B;;OAEG;YACW,eAAe;IAWvB,KAAK,CACT,UAAU,EAAE,cAAc,EAAE,EAC5B,YAAY,CAAC,EAAE,MAAM,EACrB,KAAK,CAAC,EAAE,MAAM,GACb,OAAO,CAAC,WAAW,EAAE,CAAC;IAQnB,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC;IASpC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAM5B;;OAEG;IACG,YAAY,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;IAY5C;;OAEG;IACG,UAAU,IAAI,OAAO,CAAC,KAAK,CAAC;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAgB7F;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB5B;;OAEG;IACG,aAAa,IAAI,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IAS7C;;OAEG;IACG,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAUhD;;;OAGG;IACG,OAAO,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,YAAY,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAgCjF;;;OAGG;IACG,YAAY,CAChB,WAAW,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,YAAY,EAAE,EACnD,OAAO,CAAC,EAAE;QACR,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,UAAU,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;KACpD,GACA,OAAO,CAAC;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,MAAM,CAAA;KAAE,CAAC;IA+GhE,OAAO,CAAC,UAAU;CAUnB"}
|