@rotorsoft/act-sqlite 0.3.4 → 0.5.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/README.md +4 -0
- package/package.json +9 -1
- package/dist/.tsbuildinfo +0 -1
- package/dist/@types/SqliteStore.d.ts +0 -72
- package/dist/@types/SqliteStore.d.ts.map +0 -1
- package/dist/@types/index.d.ts +0 -7
- package/dist/@types/index.d.ts.map +0 -1
- package/dist/index.cjs +0 -476
- package/dist/index.cjs.map +0 -1
- package/dist/index.js +0 -438
- package/dist/index.js.map +0 -1
package/dist/index.cjs
DELETED
|
@@ -1,476 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __create = Object.create;
|
|
3
|
-
var __defProp = Object.defineProperty;
|
|
4
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
-
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
-
var __export = (target, all) => {
|
|
9
|
-
for (var name in all)
|
|
10
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
-
};
|
|
12
|
-
var __copyProps = (to, from, except, desc) => {
|
|
13
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
-
for (let key of __getOwnPropNames(from))
|
|
15
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
-
}
|
|
18
|
-
return to;
|
|
19
|
-
};
|
|
20
|
-
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
-
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
-
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
-
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
-
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
-
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
-
mod
|
|
27
|
-
));
|
|
28
|
-
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
29
|
-
|
|
30
|
-
// src/index.ts
|
|
31
|
-
var index_exports = {};
|
|
32
|
-
__export(index_exports, {
|
|
33
|
-
SqliteStore: () => SqliteStore,
|
|
34
|
-
streamPatternToLike: () => streamPatternToLike
|
|
35
|
-
});
|
|
36
|
-
module.exports = __toCommonJS(index_exports);
|
|
37
|
-
|
|
38
|
-
// src/SqliteStore.ts
|
|
39
|
-
var import_client = require("@libsql/client");
|
|
40
|
-
var DEFAULT_CONFIG = {
|
|
41
|
-
url: "file::memory:"
|
|
42
|
-
};
|
|
43
|
-
function streamPatternToLike(input) {
|
|
44
|
-
let s = input;
|
|
45
|
-
const start = s.startsWith("^");
|
|
46
|
-
const end = s.endsWith("$");
|
|
47
|
-
if (start) s = s.slice(1);
|
|
48
|
-
if (end) s = s.slice(0, -1);
|
|
49
|
-
s = s.replace(/\.\*/g, "%").replace(/\./g, "_");
|
|
50
|
-
const out = (start ? "" : "%") + s + (end ? "" : "%");
|
|
51
|
-
return out.replace(/%+/g, "%");
|
|
52
|
-
}
|
|
53
|
-
var SqliteStore = class {
|
|
54
|
-
client;
|
|
55
|
-
constructor(config = {}) {
|
|
56
|
-
const cfg = { ...DEFAULT_CONFIG, ...config };
|
|
57
|
-
this.client = (0, import_client.createClient)({
|
|
58
|
-
url: cfg.url,
|
|
59
|
-
authToken: cfg.authToken
|
|
60
|
-
});
|
|
61
|
-
}
|
|
62
|
-
async seed() {
|
|
63
|
-
await this.client.execute("PRAGMA journal_mode=WAL");
|
|
64
|
-
await this.client.execute(`
|
|
65
|
-
CREATE TABLE IF NOT EXISTS events (
|
|
66
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
67
|
-
stream TEXT NOT NULL,
|
|
68
|
-
version INTEGER NOT NULL,
|
|
69
|
-
name TEXT NOT NULL,
|
|
70
|
-
data TEXT NOT NULL,
|
|
71
|
-
meta TEXT NOT NULL,
|
|
72
|
-
created TEXT NOT NULL,
|
|
73
|
-
UNIQUE(stream, version)
|
|
74
|
-
)
|
|
75
|
-
`);
|
|
76
|
-
await this.client.execute(
|
|
77
|
-
"CREATE INDEX IF NOT EXISTS idx_events_stream ON events(stream)"
|
|
78
|
-
);
|
|
79
|
-
await this.client.execute(
|
|
80
|
-
"CREATE INDEX IF NOT EXISTS idx_events_name ON events(name)"
|
|
81
|
-
);
|
|
82
|
-
await this.client.execute(`
|
|
83
|
-
CREATE TABLE IF NOT EXISTS streams (
|
|
84
|
-
stream TEXT PRIMARY KEY,
|
|
85
|
-
source TEXT,
|
|
86
|
-
at INTEGER NOT NULL DEFAULT -1,
|
|
87
|
-
retry INTEGER NOT NULL DEFAULT 0,
|
|
88
|
-
blocked INTEGER NOT NULL DEFAULT 0,
|
|
89
|
-
error TEXT NOT NULL DEFAULT '',
|
|
90
|
-
leased_by TEXT,
|
|
91
|
-
leased_until TEXT
|
|
92
|
-
)
|
|
93
|
-
`);
|
|
94
|
-
}
|
|
95
|
-
async drop() {
|
|
96
|
-
await this.client.execute("DROP TABLE IF EXISTS events");
|
|
97
|
-
await this.client.execute("DROP TABLE IF EXISTS streams");
|
|
98
|
-
}
|
|
99
|
-
async dispose() {
|
|
100
|
-
await Promise.resolve();
|
|
101
|
-
this.client.close();
|
|
102
|
-
}
|
|
103
|
-
// --- commit: transaction + optimistic concurrency ---
|
|
104
|
-
async commit(stream, msgs, meta, expectedVersion) {
|
|
105
|
-
const tx = await this.client.transaction("write");
|
|
106
|
-
try {
|
|
107
|
-
const versionRow = await tx.execute({
|
|
108
|
-
sql: "SELECT COALESCE(MAX(version), -1) as v FROM events WHERE stream = ?",
|
|
109
|
-
args: [stream]
|
|
110
|
-
});
|
|
111
|
-
const currentVersion = Number(versionRow.rows[0].v);
|
|
112
|
-
if (typeof expectedVersion === "number" && currentVersion !== expectedVersion) {
|
|
113
|
-
const { ConcurrencyError } = await import("@rotorsoft/act");
|
|
114
|
-
throw new ConcurrencyError(
|
|
115
|
-
stream,
|
|
116
|
-
currentVersion,
|
|
117
|
-
msgs,
|
|
118
|
-
expectedVersion
|
|
119
|
-
);
|
|
120
|
-
}
|
|
121
|
-
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
122
|
-
const committed = [];
|
|
123
|
-
let version = currentVersion + 1;
|
|
124
|
-
for (const { name, data } of msgs) {
|
|
125
|
-
const result = await tx.execute({
|
|
126
|
-
sql: "INSERT INTO events (stream, version, name, data, meta, created) VALUES (?, ?, ?, ?, ?, ?)",
|
|
127
|
-
args: [
|
|
128
|
-
stream,
|
|
129
|
-
version,
|
|
130
|
-
name,
|
|
131
|
-
JSON.stringify(data),
|
|
132
|
-
JSON.stringify(meta),
|
|
133
|
-
now
|
|
134
|
-
]
|
|
135
|
-
});
|
|
136
|
-
committed.push({
|
|
137
|
-
id: Number(result.lastInsertRowid),
|
|
138
|
-
stream,
|
|
139
|
-
version,
|
|
140
|
-
created: new Date(now),
|
|
141
|
-
name,
|
|
142
|
-
data,
|
|
143
|
-
meta
|
|
144
|
-
});
|
|
145
|
-
version++;
|
|
146
|
-
}
|
|
147
|
-
await tx.commit();
|
|
148
|
-
return committed;
|
|
149
|
-
} catch (e) {
|
|
150
|
-
await tx.rollback();
|
|
151
|
-
throw e;
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
// --- query: read-only, no transaction needed ---
|
|
155
|
-
async query(callback, query) {
|
|
156
|
-
let sql = "SELECT * FROM events WHERE 1=1";
|
|
157
|
-
const args = [];
|
|
158
|
-
if (query?.stream) {
|
|
159
|
-
if (query.stream_exact) {
|
|
160
|
-
sql += " AND stream = ?";
|
|
161
|
-
args.push(query.stream);
|
|
162
|
-
} else {
|
|
163
|
-
sql += " AND stream LIKE ?";
|
|
164
|
-
args.push(streamPatternToLike(query.stream));
|
|
165
|
-
}
|
|
166
|
-
}
|
|
167
|
-
if (query?.names) {
|
|
168
|
-
sql += ` AND name IN (${query.names.map(() => "?").join(",")})`;
|
|
169
|
-
args.push(...query.names);
|
|
170
|
-
}
|
|
171
|
-
if (query?.correlation) {
|
|
172
|
-
sql += " AND json_extract(meta, '$.correlation') = ?";
|
|
173
|
-
args.push(query.correlation);
|
|
174
|
-
}
|
|
175
|
-
if (query?.after !== void 0) {
|
|
176
|
-
sql += " AND id > ?";
|
|
177
|
-
args.push(query.after);
|
|
178
|
-
}
|
|
179
|
-
if (query?.before !== void 0) {
|
|
180
|
-
sql += " AND id < ?";
|
|
181
|
-
args.push(query.before);
|
|
182
|
-
}
|
|
183
|
-
if (query?.created_after) {
|
|
184
|
-
sql += " AND created > ?";
|
|
185
|
-
args.push(query.created_after.toISOString());
|
|
186
|
-
}
|
|
187
|
-
if (query?.created_before) {
|
|
188
|
-
sql += " AND created < ?";
|
|
189
|
-
args.push(query.created_before.toISOString());
|
|
190
|
-
}
|
|
191
|
-
if (!query?.with_snaps) {
|
|
192
|
-
sql += " AND name != '__snapshot__'";
|
|
193
|
-
}
|
|
194
|
-
sql += query?.backward ? " ORDER BY id DESC" : " ORDER BY id ASC";
|
|
195
|
-
if (query?.limit) {
|
|
196
|
-
sql += " LIMIT ?";
|
|
197
|
-
args.push(query.limit);
|
|
198
|
-
}
|
|
199
|
-
const result = await this.client.execute({ sql, args });
|
|
200
|
-
let count = 0;
|
|
201
|
-
for (const row of result.rows) {
|
|
202
|
-
callback({
|
|
203
|
-
id: Number(row.id),
|
|
204
|
-
stream: row.stream,
|
|
205
|
-
version: Number(row.version),
|
|
206
|
-
created: new Date(row.created),
|
|
207
|
-
name: row.name,
|
|
208
|
-
data: JSON.parse(row.data),
|
|
209
|
-
meta: JSON.parse(row.meta)
|
|
210
|
-
});
|
|
211
|
-
count++;
|
|
212
|
-
}
|
|
213
|
-
return count;
|
|
214
|
-
}
|
|
215
|
-
// --- subscribe: idempotent INSERT OR IGNORE (= PG ON CONFLICT DO NOTHING) ---
|
|
216
|
-
async subscribe(streams) {
|
|
217
|
-
const tx = await this.client.transaction("write");
|
|
218
|
-
try {
|
|
219
|
-
let subscribed = 0;
|
|
220
|
-
for (const { stream, source } of streams) {
|
|
221
|
-
const result = await tx.execute({
|
|
222
|
-
sql: "INSERT OR IGNORE INTO streams (stream, source) VALUES (?, ?)",
|
|
223
|
-
args: [stream, source ?? null]
|
|
224
|
-
});
|
|
225
|
-
if (result.rowsAffected > 0) subscribed++;
|
|
226
|
-
}
|
|
227
|
-
const wm = await tx.execute(
|
|
228
|
-
"SELECT COALESCE(MAX(at), -1) as w FROM streams"
|
|
229
|
-
);
|
|
230
|
-
await tx.commit();
|
|
231
|
-
return { subscribed, watermark: Number(wm.rows[0].w) };
|
|
232
|
-
} catch (e) {
|
|
233
|
-
await tx.rollback();
|
|
234
|
-
throw e;
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
// --- claim: write transaction (SQLite serializes writes = equivalent
|
|
238
|
-
// to PG FOR UPDATE SKIP LOCKED for single-server) ---
|
|
239
|
-
async claim(lagging, leading, by, millis) {
|
|
240
|
-
const tx = await this.client.transaction("write");
|
|
241
|
-
try {
|
|
242
|
-
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
243
|
-
const result = await tx.execute({
|
|
244
|
-
sql: `SELECT stream, source, at FROM streams
|
|
245
|
-
WHERE blocked = 0 AND (leased_until IS NULL OR leased_until <= ?)
|
|
246
|
-
ORDER BY at ASC`,
|
|
247
|
-
args: [now]
|
|
248
|
-
});
|
|
249
|
-
const candidates = [];
|
|
250
|
-
for (const row of result.rows) {
|
|
251
|
-
const stream = row.stream;
|
|
252
|
-
const source = row.source;
|
|
253
|
-
const at = Number(row.at);
|
|
254
|
-
let hasEvents;
|
|
255
|
-
if (source) {
|
|
256
|
-
const check = await tx.execute({
|
|
257
|
-
sql: `SELECT 1 FROM events WHERE id > ? AND name != '__snapshot__' AND stream LIKE ? LIMIT 1`,
|
|
258
|
-
args: [at, streamPatternToLike(source)]
|
|
259
|
-
});
|
|
260
|
-
hasEvents = check.rows.length > 0;
|
|
261
|
-
} else {
|
|
262
|
-
const check = await tx.execute({
|
|
263
|
-
sql: `SELECT 1 FROM events WHERE id > ? AND name != '__snapshot__' LIMIT 1`,
|
|
264
|
-
args: [at]
|
|
265
|
-
});
|
|
266
|
-
hasEvents = check.rows.length > 0;
|
|
267
|
-
}
|
|
268
|
-
if (hasEvents) {
|
|
269
|
-
candidates.push({ stream, source: source ?? void 0, at });
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
const lag = candidates.slice(0, lagging);
|
|
273
|
-
const lead = [...candidates].sort((a, b) => b.at - a.at).slice(0, leading);
|
|
274
|
-
const seen = /* @__PURE__ */ new Set();
|
|
275
|
-
const combined = [...lag, ...lead].filter((p) => {
|
|
276
|
-
if (seen.has(p.stream)) return false;
|
|
277
|
-
seen.add(p.stream);
|
|
278
|
-
return true;
|
|
279
|
-
});
|
|
280
|
-
const leases = [];
|
|
281
|
-
const until = new Date(Date.now() + millis).toISOString();
|
|
282
|
-
for (const row of combined) {
|
|
283
|
-
await tx.execute({
|
|
284
|
-
sql: "UPDATE streams SET leased_by = ?, leased_until = ?, retry = retry + 1 WHERE stream = ?",
|
|
285
|
-
args: [by, until, row.stream]
|
|
286
|
-
});
|
|
287
|
-
leases.push({
|
|
288
|
-
stream: row.stream,
|
|
289
|
-
source: row.source,
|
|
290
|
-
at: row.at,
|
|
291
|
-
by,
|
|
292
|
-
retry: 0,
|
|
293
|
-
lagging: row.at < 0
|
|
294
|
-
});
|
|
295
|
-
}
|
|
296
|
-
await tx.commit();
|
|
297
|
-
return leases;
|
|
298
|
-
} catch (e) {
|
|
299
|
-
await tx.rollback();
|
|
300
|
-
throw e;
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
// --- ack: transaction + ownership check (= PG WHERE leased_by) ---
|
|
304
|
-
async ack(leases) {
|
|
305
|
-
const tx = await this.client.transaction("write");
|
|
306
|
-
try {
|
|
307
|
-
const result = [];
|
|
308
|
-
for (const l of leases) {
|
|
309
|
-
const r = await tx.execute({
|
|
310
|
-
sql: `UPDATE streams SET at = ?, leased_by = NULL, leased_until = NULL, retry = -1
|
|
311
|
-
WHERE stream = ? AND leased_by = ?`,
|
|
312
|
-
args: [l.at, l.stream, l.by]
|
|
313
|
-
});
|
|
314
|
-
if (r.rowsAffected > 0) result.push(l);
|
|
315
|
-
}
|
|
316
|
-
await tx.commit();
|
|
317
|
-
return result;
|
|
318
|
-
} catch (e) {
|
|
319
|
-
await tx.rollback();
|
|
320
|
-
throw e;
|
|
321
|
-
}
|
|
322
|
-
}
|
|
323
|
-
// --- block: transaction + ownership + idempotent (= PG) ---
|
|
324
|
-
async block(leases) {
|
|
325
|
-
const tx = await this.client.transaction("write");
|
|
326
|
-
try {
|
|
327
|
-
const result = [];
|
|
328
|
-
for (const l of leases) {
|
|
329
|
-
const r = await tx.execute({
|
|
330
|
-
sql: `UPDATE streams SET blocked = 1, error = ?
|
|
331
|
-
WHERE stream = ? AND leased_by = ? AND blocked = 0`,
|
|
332
|
-
args: [l.error, l.stream, l.by]
|
|
333
|
-
});
|
|
334
|
-
if (r.rowsAffected > 0) result.push(l);
|
|
335
|
-
}
|
|
336
|
-
await tx.commit();
|
|
337
|
-
return result;
|
|
338
|
-
} catch (e) {
|
|
339
|
-
await tx.rollback();
|
|
340
|
-
throw e;
|
|
341
|
-
}
|
|
342
|
-
}
|
|
343
|
-
// --- reset: transactional ---
|
|
344
|
-
async reset(streams) {
|
|
345
|
-
const tx = await this.client.transaction("write");
|
|
346
|
-
try {
|
|
347
|
-
let count = 0;
|
|
348
|
-
for (const stream of streams) {
|
|
349
|
-
const r = await tx.execute({
|
|
350
|
-
sql: `UPDATE streams SET at = -1, retry = 0, blocked = 0, error = '',
|
|
351
|
-
leased_by = NULL, leased_until = NULL WHERE stream = ?`,
|
|
352
|
-
args: [stream]
|
|
353
|
-
});
|
|
354
|
-
count += r.rowsAffected;
|
|
355
|
-
}
|
|
356
|
-
await tx.commit();
|
|
357
|
-
return count;
|
|
358
|
-
} catch (e) {
|
|
359
|
-
await tx.rollback();
|
|
360
|
-
throw e;
|
|
361
|
-
}
|
|
362
|
-
}
|
|
363
|
-
// --- query_streams: read-only introspection with filters ---
|
|
364
|
-
async query_streams(callback, query) {
|
|
365
|
-
const limit = query?.limit ?? 100;
|
|
366
|
-
let sql = "SELECT stream, source, at, retry, blocked, error, leased_by, leased_until FROM streams WHERE 1=1";
|
|
367
|
-
const args = [];
|
|
368
|
-
if (query?.stream !== void 0) {
|
|
369
|
-
if (query.stream_exact) {
|
|
370
|
-
sql += " AND stream = ?";
|
|
371
|
-
args.push(query.stream);
|
|
372
|
-
} else {
|
|
373
|
-
sql += " AND stream LIKE ?";
|
|
374
|
-
args.push(streamPatternToLike(query.stream));
|
|
375
|
-
}
|
|
376
|
-
}
|
|
377
|
-
if (query?.source !== void 0) {
|
|
378
|
-
sql += " AND source IS NOT NULL";
|
|
379
|
-
if (query.source_exact) {
|
|
380
|
-
sql += " AND source = ?";
|
|
381
|
-
args.push(query.source);
|
|
382
|
-
} else {
|
|
383
|
-
sql += " AND source LIKE ?";
|
|
384
|
-
args.push(streamPatternToLike(query.source));
|
|
385
|
-
}
|
|
386
|
-
}
|
|
387
|
-
if (query?.blocked !== void 0) {
|
|
388
|
-
sql += " AND blocked = ?";
|
|
389
|
-
args.push(query.blocked ? 1 : 0);
|
|
390
|
-
}
|
|
391
|
-
if (query?.after !== void 0) {
|
|
392
|
-
sql += " AND stream > ?";
|
|
393
|
-
args.push(query.after);
|
|
394
|
-
}
|
|
395
|
-
sql += " ORDER BY stream LIMIT ?";
|
|
396
|
-
args.push(limit);
|
|
397
|
-
const [streamsResult, maxResult] = await Promise.all([
|
|
398
|
-
this.client.execute({ sql, args }),
|
|
399
|
-
this.client.execute("SELECT COALESCE(MAX(id), -1) AS m FROM events")
|
|
400
|
-
]);
|
|
401
|
-
let count = 0;
|
|
402
|
-
for (const row of streamsResult.rows) {
|
|
403
|
-
const leased_until = row.leased_until;
|
|
404
|
-
callback({
|
|
405
|
-
stream: row.stream,
|
|
406
|
-
source: row.source ?? void 0,
|
|
407
|
-
at: Number(row.at),
|
|
408
|
-
retry: Number(row.retry),
|
|
409
|
-
blocked: Number(row.blocked) === 1,
|
|
410
|
-
error: row.error,
|
|
411
|
-
leased_by: row.leased_by ?? void 0,
|
|
412
|
-
leased_until: leased_until ? new Date(leased_until) : void 0
|
|
413
|
-
});
|
|
414
|
-
count++;
|
|
415
|
-
}
|
|
416
|
-
return { maxEventId: Number(maxResult.rows[0].m), count };
|
|
417
|
-
}
|
|
418
|
-
// --- truncate: transactional delete + seed ---
|
|
419
|
-
async truncate(targets) {
|
|
420
|
-
const result = /* @__PURE__ */ new Map();
|
|
421
|
-
const tx = await this.client.transaction("write");
|
|
422
|
-
try {
|
|
423
|
-
for (const { stream, snapshot, meta } of targets) {
|
|
424
|
-
const countRow = await tx.execute({
|
|
425
|
-
sql: "SELECT COUNT(*) as c FROM events WHERE stream = ?",
|
|
426
|
-
args: [stream]
|
|
427
|
-
});
|
|
428
|
-
const deleted = Number(countRow.rows[0].c);
|
|
429
|
-
await tx.execute({
|
|
430
|
-
sql: "DELETE FROM events WHERE stream = ?",
|
|
431
|
-
args: [stream]
|
|
432
|
-
});
|
|
433
|
-
await tx.execute({
|
|
434
|
-
sql: "DELETE FROM streams WHERE stream = ?",
|
|
435
|
-
args: [stream]
|
|
436
|
-
});
|
|
437
|
-
const eventName = snapshot !== void 0 ? "__snapshot__" : "__tombstone__";
|
|
438
|
-
const eventMeta = meta ?? { correlation: "", causation: {} };
|
|
439
|
-
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
440
|
-
const ins = await tx.execute({
|
|
441
|
-
sql: "INSERT INTO events (stream, version, name, data, meta, created) VALUES (?, 0, ?, ?, ?, ?)",
|
|
442
|
-
args: [
|
|
443
|
-
stream,
|
|
444
|
-
eventName,
|
|
445
|
-
JSON.stringify(snapshot ?? {}),
|
|
446
|
-
JSON.stringify(eventMeta),
|
|
447
|
-
now
|
|
448
|
-
]
|
|
449
|
-
});
|
|
450
|
-
result.set(stream, {
|
|
451
|
-
deleted,
|
|
452
|
-
committed: {
|
|
453
|
-
id: Number(ins.lastInsertRowid),
|
|
454
|
-
stream,
|
|
455
|
-
version: 0,
|
|
456
|
-
created: new Date(now),
|
|
457
|
-
name: eventName,
|
|
458
|
-
data: snapshot ?? {},
|
|
459
|
-
meta: eventMeta
|
|
460
|
-
}
|
|
461
|
-
});
|
|
462
|
-
}
|
|
463
|
-
await tx.commit();
|
|
464
|
-
} catch (e) {
|
|
465
|
-
await tx.rollback();
|
|
466
|
-
throw e;
|
|
467
|
-
}
|
|
468
|
-
return result;
|
|
469
|
-
}
|
|
470
|
-
};
|
|
471
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
472
|
-
0 && (module.exports = {
|
|
473
|
-
SqliteStore,
|
|
474
|
-
streamPatternToLike
|
|
475
|
-
});
|
|
476
|
-
//# sourceMappingURL=index.cjs.map
|
package/dist/index.cjs.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/SqliteStore.ts"],"sourcesContent":["/**\n * @packageDocumentation\n * @module act-sqlite\n * Main entry point for the Act-SQLite adapter. Re-exports all core APIs\n */\nexport * from \"./SqliteStore.js\";\n","import { type Client, createClient } from \"@libsql/client\";\nimport type {\n BlockedLease,\n Committed,\n EventMeta,\n Lease,\n Message,\n Query,\n QueryStreams,\n QueryStreamsResult,\n Schemas,\n Store,\n StreamPosition,\n} from \"@rotorsoft/act\";\n\n/**\n * SQLite store configuration\n */\nexport interface SqliteConfig {\n /** Path to the SQLite database file (default: \":memory:\") */\n url: string;\n /** Auth token for libSQL server connections (optional) */\n authToken?: string;\n}\n\nconst DEFAULT_CONFIG: SqliteConfig = {\n url: \"file::memory:\",\n};\n\n/** Translate a stream filter (regex-shaped or plain substring) into a\n * SQL LIKE pattern. Honors `^` / `$` anchors and converts `.*` → `%`,\n * `.` → `_`. Unanchored input gets `%` wildcards on both sides.\n *\n * Examples:\n * - `^abc$` → `abc` (exact)\n * - `^abc.*` → `abc%` (starts-with)\n * - `.*abc$` → `%abc` (ends-with)\n * - `abc` → `%abc%` (contains)\n * - `a.c` → `%a_c%` (single-char wildcard, contains)\n *\n * @internal exported for testing\n */\nexport function streamPatternToLike(input: string): string {\n let s = input;\n const start = s.startsWith(\"^\");\n const end = s.endsWith(\"$\");\n if (start) s = s.slice(1);\n if (end) s = s.slice(0, -1);\n s = s.replace(/\\.\\*/g, \"%\").replace(/\\./g, \"_\");\n const out = (start ? \"\" : \"%\") + s + (end ? \"\" : \"%\");\n // Collapse adjacent `%` — e.g. `^a.*` would otherwise yield `a%%`.\n // Same matching semantics, cleaner output.\n return out.replace(/%+/g, \"%\");\n}\n\n/**\n * SQLite event store adapter for [@rotorsoft/act](https://www.npmjs.com/package/@rotorsoft/act).\n *\n * Provides persistent event storage using SQLite via `@libsql/client`.\n * All write operations use transactions for ACID guarantees.\n * Since SQLite serializes writes at the database level, the concurrency\n * model is equivalent to PostgreSQL's `FOR UPDATE SKIP LOCKED` for\n * single-server deployments.\n *\n * @example\n * ```typescript\n * import { store } from \"@rotorsoft/act\";\n * import { SqliteStore } from \"@rotorsoft/act-sqlite\";\n *\n * store(new SqliteStore({ url: \"file:myapp.db\" }));\n * await store().seed();\n * ```\n */\nexport class SqliteStore implements Store {\n private client: Client;\n\n constructor(config: Partial<SqliteConfig> = {}) {\n const cfg = { ...DEFAULT_CONFIG, ...config };\n this.client = createClient({\n url: cfg.url,\n authToken: cfg.authToken,\n });\n }\n\n async seed() {\n await this.client.execute(\"PRAGMA journal_mode=WAL\");\n await this.client.execute(`\n CREATE TABLE IF NOT EXISTS events (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n stream TEXT NOT NULL,\n version INTEGER NOT NULL,\n name TEXT NOT NULL,\n data TEXT NOT NULL,\n meta TEXT NOT NULL,\n created TEXT NOT NULL,\n UNIQUE(stream, version)\n )\n `);\n await this.client.execute(\n \"CREATE INDEX IF NOT EXISTS idx_events_stream ON events(stream)\"\n );\n await this.client.execute(\n \"CREATE INDEX IF NOT EXISTS idx_events_name ON events(name)\"\n );\n await this.client.execute(`\n CREATE TABLE IF NOT EXISTS streams (\n stream TEXT PRIMARY KEY,\n source TEXT,\n at INTEGER NOT NULL DEFAULT -1,\n retry INTEGER NOT NULL DEFAULT 0,\n blocked INTEGER NOT NULL DEFAULT 0,\n error TEXT NOT NULL DEFAULT '',\n leased_by TEXT,\n leased_until TEXT\n )\n `);\n }\n\n async drop() {\n await this.client.execute(\"DROP TABLE IF EXISTS events\");\n await this.client.execute(\"DROP TABLE IF EXISTS streams\");\n }\n\n async dispose() {\n await Promise.resolve();\n this.client.close();\n }\n\n // --- commit: transaction + optimistic concurrency ---\n async commit<E extends Schemas>(\n stream: string,\n msgs: Message<E, keyof E>[],\n meta: EventMeta,\n expectedVersion?: number\n ): Promise<Committed<E, keyof E>[]> {\n const tx = await this.client.transaction(\"write\");\n try {\n const versionRow = await tx.execute({\n sql: \"SELECT COALESCE(MAX(version), -1) as v FROM events WHERE stream = ?\",\n args: [stream],\n });\n const currentVersion = Number(versionRow.rows[0].v);\n\n if (\n typeof expectedVersion === \"number\" &&\n currentVersion !== expectedVersion\n ) {\n const { ConcurrencyError } = await import(\"@rotorsoft/act\");\n throw new ConcurrencyError(\n stream,\n currentVersion,\n msgs as Message<Schemas, keyof Schemas>[],\n expectedVersion\n );\n }\n\n const now = new Date().toISOString();\n const committed: Committed<E, keyof E>[] = [];\n let version = currentVersion + 1;\n\n for (const { name, data } of msgs) {\n const result = await tx.execute({\n sql: \"INSERT INTO events (stream, version, name, data, meta, created) VALUES (?, ?, ?, ?, ?, ?)\",\n args: [\n stream,\n version,\n name as string,\n JSON.stringify(data),\n JSON.stringify(meta),\n now,\n ],\n });\n committed.push({\n id: Number(result.lastInsertRowid),\n stream,\n version,\n created: new Date(now),\n name,\n data,\n meta,\n });\n version++;\n }\n\n await tx.commit();\n return committed;\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n }\n\n // --- query: read-only, no transaction needed ---\n async query<E extends Schemas>(\n callback: (event: Committed<E, keyof E>) => void,\n query?: Query\n ): Promise<number> {\n let sql = \"SELECT * FROM events WHERE 1=1\";\n const args: unknown[] = [];\n\n if (query?.stream) {\n if (query.stream_exact) {\n sql += \" AND stream = ?\";\n args.push(query.stream);\n } else {\n sql += \" AND stream LIKE ?\";\n args.push(streamPatternToLike(query.stream));\n }\n }\n if (query?.names) {\n sql += ` AND name IN (${query.names.map(() => \"?\").join(\",\")})`;\n args.push(...query.names);\n }\n if ((query as any)?.correlation) {\n sql += \" AND json_extract(meta, '$.correlation') = ?\";\n args.push((query as any).correlation);\n }\n if (query?.after !== undefined) {\n sql += \" AND id > ?\";\n args.push(query.after);\n }\n if (query?.before !== undefined) {\n sql += \" AND id < ?\";\n args.push(query.before);\n }\n if (query?.created_after) {\n sql += \" AND created > ?\";\n args.push(query.created_after.toISOString());\n }\n if (query?.created_before) {\n sql += \" AND created < ?\";\n args.push(query.created_before.toISOString());\n }\n if (!query?.with_snaps) {\n sql += \" AND name != '__snapshot__'\";\n }\n\n sql += query?.backward ? \" ORDER BY id DESC\" : \" ORDER BY id ASC\";\n\n if (query?.limit) {\n sql += \" LIMIT ?\";\n args.push(query.limit);\n }\n\n const result = await this.client.execute({ sql, args: args as any[] });\n let count = 0;\n\n for (const row of result.rows) {\n callback({\n id: Number(row.id),\n stream: row.stream as string,\n version: Number(row.version),\n created: new Date(row.created as string),\n name: row.name as string,\n data: JSON.parse(row.data as string),\n meta: JSON.parse(row.meta as string),\n });\n count++;\n }\n\n return count;\n }\n\n // --- subscribe: idempotent INSERT OR IGNORE (= PG ON CONFLICT DO NOTHING) ---\n async subscribe(streams: Array<{ stream: string; source?: string }>) {\n const tx = await this.client.transaction(\"write\");\n try {\n let subscribed = 0;\n for (const { stream, source } of streams) {\n const result = await tx.execute({\n sql: \"INSERT OR IGNORE INTO streams (stream, source) VALUES (?, ?)\",\n args: [stream, source ?? null],\n });\n if (result.rowsAffected > 0) subscribed++;\n }\n const wm = await tx.execute(\n \"SELECT COALESCE(MAX(at), -1) as w FROM streams\"\n );\n await tx.commit();\n return { subscribed, watermark: Number(wm.rows[0].w) };\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n }\n\n // --- claim: write transaction (SQLite serializes writes = equivalent\n // to PG FOR UPDATE SKIP LOCKED for single-server) ---\n async claim(lagging: number, leading: number, by: string, millis: number) {\n const tx = await this.client.transaction(\"write\");\n try {\n const now = new Date().toISOString();\n\n const result = await tx.execute({\n sql: `SELECT stream, source, at FROM streams\n WHERE blocked = 0 AND (leased_until IS NULL OR leased_until <= ?)\n ORDER BY at ASC`,\n args: [now],\n });\n\n const candidates: {\n stream: string;\n source: string | undefined;\n at: number;\n }[] = [];\n for (const row of result.rows) {\n const stream = row.stream as string;\n const source = row.source as string | null;\n const at = Number(row.at);\n\n let hasEvents: boolean;\n if (source) {\n const check = await tx.execute({\n sql: `SELECT 1 FROM events WHERE id > ? AND name != '__snapshot__' AND stream LIKE ? LIMIT 1`,\n args: [at, streamPatternToLike(source)],\n });\n hasEvents = check.rows.length > 0;\n } else {\n const check = await tx.execute({\n sql: `SELECT 1 FROM events WHERE id > ? AND name != '__snapshot__' LIMIT 1`,\n args: [at],\n });\n hasEvents = check.rows.length > 0;\n }\n\n if (hasEvents) {\n candidates.push({ stream, source: source ?? undefined, at });\n }\n }\n\n // Dual frontier: lagging (oldest first) + leading (newest first)\n const lag = candidates.slice(0, lagging);\n const lead = [...candidates]\n .sort((a, b) => b.at - a.at)\n .slice(0, leading);\n const seen = new Set<string>();\n const combined = [...lag, ...lead].filter((p) => {\n if (seen.has(p.stream)) return false;\n seen.add(p.stream);\n return true;\n });\n\n const leases: Lease[] = [];\n const until = new Date(Date.now() + millis).toISOString();\n for (const row of combined) {\n await tx.execute({\n sql: \"UPDATE streams SET leased_by = ?, leased_until = ?, retry = retry + 1 WHERE stream = ?\",\n args: [by, until, row.stream],\n });\n leases.push({\n stream: row.stream,\n source: row.source,\n at: row.at,\n by,\n retry: 0,\n lagging: row.at < 0,\n });\n }\n\n await tx.commit();\n return leases;\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n }\n\n // --- ack: transaction + ownership check (= PG WHERE leased_by) ---\n async ack(leases: Lease[]) {\n const tx = await this.client.transaction(\"write\");\n try {\n const result: Lease[] = [];\n for (const l of leases) {\n const r = await tx.execute({\n sql: `UPDATE streams SET at = ?, leased_by = NULL, leased_until = NULL, retry = -1\n WHERE stream = ? AND leased_by = ?`,\n args: [l.at, l.stream, l.by],\n });\n if (r.rowsAffected > 0) result.push(l);\n }\n await tx.commit();\n return result;\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n }\n\n // --- block: transaction + ownership + idempotent (= PG) ---\n async block(leases: BlockedLease[]) {\n const tx = await this.client.transaction(\"write\");\n try {\n const result: BlockedLease[] = [];\n for (const l of leases) {\n const r = await tx.execute({\n sql: `UPDATE streams SET blocked = 1, error = ?\n WHERE stream = ? AND leased_by = ? AND blocked = 0`,\n args: [l.error, l.stream, l.by],\n });\n if (r.rowsAffected > 0) result.push(l);\n }\n await tx.commit();\n return result;\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n }\n\n // --- reset: transactional ---\n async reset(streams: string[]) {\n const tx = await this.client.transaction(\"write\");\n try {\n let count = 0;\n for (const stream of streams) {\n const r = await tx.execute({\n sql: `UPDATE streams SET at = -1, retry = 0, blocked = 0, error = '',\n leased_by = NULL, leased_until = NULL WHERE stream = ?`,\n args: [stream],\n });\n count += r.rowsAffected;\n }\n await tx.commit();\n return count;\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n }\n\n // --- query_streams: read-only introspection with filters ---\n async query_streams(\n callback: (position: StreamPosition) => void,\n query?: QueryStreams\n ): Promise<QueryStreamsResult> {\n const limit = query?.limit ?? 100;\n let sql =\n \"SELECT stream, source, at, retry, blocked, error, leased_by, leased_until FROM streams WHERE 1=1\";\n const args: unknown[] = [];\n\n if (query?.stream !== undefined) {\n if (query.stream_exact) {\n sql += \" AND stream = ?\";\n args.push(query.stream);\n } else {\n sql += \" AND stream LIKE ?\";\n args.push(streamPatternToLike(query.stream));\n }\n }\n if (query?.source !== undefined) {\n sql += \" AND source IS NOT NULL\";\n if (query.source_exact) {\n sql += \" AND source = ?\";\n args.push(query.source);\n } else {\n sql += \" AND source LIKE ?\";\n args.push(streamPatternToLike(query.source));\n }\n }\n if (query?.blocked !== undefined) {\n sql += \" AND blocked = ?\";\n args.push(query.blocked ? 1 : 0);\n }\n if (query?.after !== undefined) {\n sql += \" AND stream > ?\";\n args.push(query.after);\n }\n sql += \" ORDER BY stream LIMIT ?\";\n args.push(limit);\n\n const [streamsResult, maxResult] = await Promise.all([\n this.client.execute({ sql, args: args as any[] }),\n this.client.execute(\"SELECT COALESCE(MAX(id), -1) AS m FROM events\"),\n ]);\n\n let count = 0;\n for (const row of streamsResult.rows) {\n const leased_until = row.leased_until as string | null;\n callback({\n stream: row.stream as string,\n source: (row.source as string | null) ?? undefined,\n at: Number(row.at),\n retry: Number(row.retry),\n blocked: Number(row.blocked) === 1,\n error: row.error as string,\n leased_by: (row.leased_by as string | null) ?? undefined,\n leased_until: leased_until ? new Date(leased_until) : undefined,\n });\n count++;\n }\n\n return { maxEventId: Number(maxResult.rows[0].m), count };\n }\n\n // --- truncate: transactional delete + seed ---\n async truncate(\n targets: Array<{\n stream: string;\n snapshot?: Record<string, unknown>;\n meta?: EventMeta;\n }>\n ) {\n const result = new Map<\n string,\n { deleted: number; committed: Committed<Schemas, keyof Schemas> }\n >();\n\n const tx = await this.client.transaction(\"write\");\n try {\n for (const { stream, snapshot, meta } of targets) {\n const countRow = await tx.execute({\n sql: \"SELECT COUNT(*) as c FROM events WHERE stream = ?\",\n args: [stream],\n });\n const deleted = Number(countRow.rows[0].c);\n await tx.execute({\n sql: \"DELETE FROM events WHERE stream = ?\",\n args: [stream],\n });\n await tx.execute({\n sql: \"DELETE FROM streams WHERE stream = ?\",\n args: [stream],\n });\n\n const eventName =\n snapshot !== undefined ? \"__snapshot__\" : \"__tombstone__\";\n const eventMeta = meta ?? { correlation: \"\", causation: {} };\n const now = new Date().toISOString();\n const ins = await tx.execute({\n sql: \"INSERT INTO events (stream, version, name, data, meta, created) VALUES (?, 0, ?, ?, ?, ?)\",\n args: [\n stream,\n eventName,\n JSON.stringify(snapshot ?? {}),\n JSON.stringify(eventMeta),\n now,\n ],\n });\n\n result.set(stream, {\n deleted,\n committed: {\n id: Number(ins.lastInsertRowid),\n stream,\n version: 0,\n created: new Date(now),\n name: eventName,\n data: snapshot ?? {},\n meta: eventMeta,\n },\n });\n }\n await tx.commit();\n } catch (e) {\n await tx.rollback();\n throw e;\n }\n\n return result;\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;;ACAA,oBAA0C;AAyB1C,IAAM,iBAA+B;AAAA,EACnC,KAAK;AACP;AAeO,SAAS,oBAAoB,OAAuB;AACzD,MAAI,IAAI;AACR,QAAM,QAAQ,EAAE,WAAW,GAAG;AAC9B,QAAM,MAAM,EAAE,SAAS,GAAG;AAC1B,MAAI,MAAO,KAAI,EAAE,MAAM,CAAC;AACxB,MAAI,IAAK,KAAI,EAAE,MAAM,GAAG,EAAE;AAC1B,MAAI,EAAE,QAAQ,SAAS,GAAG,EAAE,QAAQ,OAAO,GAAG;AAC9C,QAAM,OAAO,QAAQ,KAAK,OAAO,KAAK,MAAM,KAAK;AAGjD,SAAO,IAAI,QAAQ,OAAO,GAAG;AAC/B;AAoBO,IAAM,cAAN,MAAmC;AAAA,EAChC;AAAA,EAER,YAAY,SAAgC,CAAC,GAAG;AAC9C,UAAM,MAAM,EAAE,GAAG,gBAAgB,GAAG,OAAO;AAC3C,SAAK,aAAS,4BAAa;AAAA,MACzB,KAAK,IAAI;AAAA,MACT,WAAW,IAAI;AAAA,IACjB,CAAC;AAAA,EACH;AAAA,EAEA,MAAM,OAAO;AACX,UAAM,KAAK,OAAO,QAAQ,yBAAyB;AACnD,UAAM,KAAK,OAAO,QAAQ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,KAWzB;AACD,UAAM,KAAK,OAAO;AAAA,MAChB;AAAA,IACF;AACA,UAAM,KAAK,OAAO;AAAA,MAChB;AAAA,IACF;AACA,UAAM,KAAK,OAAO,QAAQ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,KAWzB;AAAA,EACH;AAAA,EAEA,MAAM,OAAO;AACX,UAAM,KAAK,OAAO,QAAQ,6BAA6B;AACvD,UAAM,KAAK,OAAO,QAAQ,8BAA8B;AAAA,EAC1D;AAAA,EAEA,MAAM,UAAU;AACd,UAAM,QAAQ,QAAQ;AACtB,SAAK,OAAO,MAAM;AAAA,EACpB;AAAA;AAAA,EAGA,MAAM,OACJ,QACA,MACA,MACA,iBACkC;AAClC,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,YAAM,aAAa,MAAM,GAAG,QAAQ;AAAA,QAClC,KAAK;AAAA,QACL,MAAM,CAAC,MAAM;AAAA,MACf,CAAC;AACD,YAAM,iBAAiB,OAAO,WAAW,KAAK,CAAC,EAAE,CAAC;AAElD,UACE,OAAO,oBAAoB,YAC3B,mBAAmB,iBACnB;AACA,cAAM,EAAE,iBAAiB,IAAI,MAAM,OAAO,gBAAgB;AAC1D,cAAM,IAAI;AAAA,UACR;AAAA,UACA;AAAA,UACA;AAAA,UACA;AAAA,QACF;AAAA,MACF;AAEA,YAAM,OAAM,oBAAI,KAAK,GAAE,YAAY;AACnC,YAAM,YAAqC,CAAC;AAC5C,UAAI,UAAU,iBAAiB;AAE/B,iBAAW,EAAE,MAAM,KAAK,KAAK,MAAM;AACjC,cAAM,SAAS,MAAM,GAAG,QAAQ;AAAA,UAC9B,KAAK;AAAA,UACL,MAAM;AAAA,YACJ;AAAA,YACA;AAAA,YACA;AAAA,YACA,KAAK,UAAU,IAAI;AAAA,YACnB,KAAK,UAAU,IAAI;AAAA,YACnB;AAAA,UACF;AAAA,QACF,CAAC;AACD,kBAAU,KAAK;AAAA,UACb,IAAI,OAAO,OAAO,eAAe;AAAA,UACjC;AAAA,UACA;AAAA,UACA,SAAS,IAAI,KAAK,GAAG;AAAA,UACrB;AAAA,UACA;AAAA,UACA;AAAA,QACF,CAAC;AACD;AAAA,MACF;AAEA,YAAM,GAAG,OAAO;AAChB,aAAO;AAAA,IACT,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,MACJ,UACA,OACiB;AACjB,QAAI,MAAM;AACV,UAAM,OAAkB,CAAC;AAEzB,QAAI,OAAO,QAAQ;AACjB,UAAI,MAAM,cAAc;AACtB,eAAO;AACP,aAAK,KAAK,MAAM,MAAM;AAAA,MACxB,OAAO;AACL,eAAO;AACP,aAAK,KAAK,oBAAoB,MAAM,MAAM,CAAC;AAAA,MAC7C;AAAA,IACF;AACA,QAAI,OAAO,OAAO;AAChB,aAAO,iBAAiB,MAAM,MAAM,IAAI,MAAM,GAAG,EAAE,KAAK,GAAG,CAAC;AAC5D,WAAK,KAAK,GAAG,MAAM,KAAK;AAAA,IAC1B;AACA,QAAK,OAAe,aAAa;AAC/B,aAAO;AACP,WAAK,KAAM,MAAc,WAAW;AAAA,IACtC;AACA,QAAI,OAAO,UAAU,QAAW;AAC9B,aAAO;AACP,WAAK,KAAK,MAAM,KAAK;AAAA,IACvB;AACA,QAAI,OAAO,WAAW,QAAW;AAC/B,aAAO;AACP,WAAK,KAAK,MAAM,MAAM;AAAA,IACxB;AACA,QAAI,OAAO,eAAe;AACxB,aAAO;AACP,WAAK,KAAK,MAAM,cAAc,YAAY,CAAC;AAAA,IAC7C;AACA,QAAI,OAAO,gBAAgB;AACzB,aAAO;AACP,WAAK,KAAK,MAAM,eAAe,YAAY,CAAC;AAAA,IAC9C;AACA,QAAI,CAAC,OAAO,YAAY;AACtB,aAAO;AAAA,IACT;AAEA,WAAO,OAAO,WAAW,sBAAsB;AAE/C,QAAI,OAAO,OAAO;AAChB,aAAO;AACP,WAAK,KAAK,MAAM,KAAK;AAAA,IACvB;AAEA,UAAM,SAAS,MAAM,KAAK,OAAO,QAAQ,EAAE,KAAK,KAAoB,CAAC;AACrE,QAAI,QAAQ;AAEZ,eAAW,OAAO,OAAO,MAAM;AAC7B,eAAS;AAAA,QACP,IAAI,OAAO,IAAI,EAAE;AAAA,QACjB,QAAQ,IAAI;AAAA,QACZ,SAAS,OAAO,IAAI,OAAO;AAAA,QAC3B,SAAS,IAAI,KAAK,IAAI,OAAiB;AAAA,QACvC,MAAM,IAAI;AAAA,QACV,MAAM,KAAK,MAAM,IAAI,IAAc;AAAA,QACnC,MAAM,KAAK,MAAM,IAAI,IAAc;AAAA,MACrC,CAAC;AACD;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,MAAM,UAAU,SAAqD;AACnE,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,UAAI,aAAa;AACjB,iBAAW,EAAE,QAAQ,OAAO,KAAK,SAAS;AACxC,cAAM,SAAS,MAAM,GAAG,QAAQ;AAAA,UAC9B,KAAK;AAAA,UACL,MAAM,CAAC,QAAQ,UAAU,IAAI;AAAA,QAC/B,CAAC;AACD,YAAI,OAAO,eAAe,EAAG;AAAA,MAC/B;AACA,YAAM,KAAK,MAAM,GAAG;AAAA,QAClB;AAAA,MACF;AACA,YAAM,GAAG,OAAO;AAChB,aAAO,EAAE,YAAY,WAAW,OAAO,GAAG,KAAK,CAAC,EAAE,CAAC,EAAE;AAAA,IACvD,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA;AAAA,EAIA,MAAM,MAAM,SAAiB,SAAiB,IAAY,QAAgB;AACxE,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,YAAM,OAAM,oBAAI,KAAK,GAAE,YAAY;AAEnC,YAAM,SAAS,MAAM,GAAG,QAAQ;AAAA,QAC9B,KAAK;AAAA;AAAA;AAAA,QAGL,MAAM,CAAC,GAAG;AAAA,MACZ,CAAC;AAED,YAAM,aAIA,CAAC;AACP,iBAAW,OAAO,OAAO,MAAM;AAC7B,cAAM,SAAS,IAAI;AACnB,cAAM,SAAS,IAAI;AACnB,cAAM,KAAK,OAAO,IAAI,EAAE;AAExB,YAAI;AACJ,YAAI,QAAQ;AACV,gBAAM,QAAQ,MAAM,GAAG,QAAQ;AAAA,YAC7B,KAAK;AAAA,YACL,MAAM,CAAC,IAAI,oBAAoB,MAAM,CAAC;AAAA,UACxC,CAAC;AACD,sBAAY,MAAM,KAAK,SAAS;AAAA,QAClC,OAAO;AACL,gBAAM,QAAQ,MAAM,GAAG,QAAQ;AAAA,YAC7B,KAAK;AAAA,YACL,MAAM,CAAC,EAAE;AAAA,UACX,CAAC;AACD,sBAAY,MAAM,KAAK,SAAS;AAAA,QAClC;AAEA,YAAI,WAAW;AACb,qBAAW,KAAK,EAAE,QAAQ,QAAQ,UAAU,QAAW,GAAG,CAAC;AAAA,QAC7D;AAAA,MACF;AAGA,YAAM,MAAM,WAAW,MAAM,GAAG,OAAO;AACvC,YAAM,OAAO,CAAC,GAAG,UAAU,EACxB,KAAK,CAAC,GAAG,MAAM,EAAE,KAAK,EAAE,EAAE,EAC1B,MAAM,GAAG,OAAO;AACnB,YAAM,OAAO,oBAAI,IAAY;AAC7B,YAAM,WAAW,CAAC,GAAG,KAAK,GAAG,IAAI,EAAE,OAAO,CAAC,MAAM;AAC/C,YAAI,KAAK,IAAI,EAAE,MAAM,EAAG,QAAO;AAC/B,aAAK,IAAI,EAAE,MAAM;AACjB,eAAO;AAAA,MACT,CAAC;AAED,YAAM,SAAkB,CAAC;AACzB,YAAM,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAI,MAAM,EAAE,YAAY;AACxD,iBAAW,OAAO,UAAU;AAC1B,cAAM,GAAG,QAAQ;AAAA,UACf,KAAK;AAAA,UACL,MAAM,CAAC,IAAI,OAAO,IAAI,MAAM;AAAA,QAC9B,CAAC;AACD,eAAO,KAAK;AAAA,UACV,QAAQ,IAAI;AAAA,UACZ,QAAQ,IAAI;AAAA,UACZ,IAAI,IAAI;AAAA,UACR;AAAA,UACA,OAAO;AAAA,UACP,SAAS,IAAI,KAAK;AAAA,QACpB,CAAC;AAAA,MACH;AAEA,YAAM,GAAG,OAAO;AAChB,aAAO;AAAA,IACT,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,IAAI,QAAiB;AACzB,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,YAAM,SAAkB,CAAC;AACzB,iBAAW,KAAK,QAAQ;AACtB,cAAM,IAAI,MAAM,GAAG,QAAQ;AAAA,UACzB,KAAK;AAAA;AAAA,UAEL,MAAM,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE;AAAA,QAC7B,CAAC;AACD,YAAI,EAAE,eAAe,EAAG,QAAO,KAAK,CAAC;AAAA,MACvC;AACA,YAAM,GAAG,OAAO;AAChB,aAAO;AAAA,IACT,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,MAAM,QAAwB;AAClC,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,YAAM,SAAyB,CAAC;AAChC,iBAAW,KAAK,QAAQ;AACtB,cAAM,IAAI,MAAM,GAAG,QAAQ;AAAA,UACzB,KAAK;AAAA;AAAA,UAEL,MAAM,CAAC,EAAE,OAAO,EAAE,QAAQ,EAAE,EAAE;AAAA,QAChC,CAAC;AACD,YAAI,EAAE,eAAe,EAAG,QAAO,KAAK,CAAC;AAAA,MACvC;AACA,YAAM,GAAG,OAAO;AAChB,aAAO;AAAA,IACT,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,MAAM,SAAmB;AAC7B,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,UAAI,QAAQ;AACZ,iBAAW,UAAU,SAAS;AAC5B,cAAM,IAAI,MAAM,GAAG,QAAQ;AAAA,UACzB,KAAK;AAAA;AAAA,UAEL,MAAM,CAAC,MAAM;AAAA,QACf,CAAC;AACD,iBAAS,EAAE;AAAA,MACb;AACA,YAAM,GAAG,OAAO;AAChB,aAAO;AAAA,IACT,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,cACJ,UACA,OAC6B;AAC7B,UAAM,QAAQ,OAAO,SAAS;AAC9B,QAAI,MACF;AACF,UAAM,OAAkB,CAAC;AAEzB,QAAI,OAAO,WAAW,QAAW;AAC/B,UAAI,MAAM,cAAc;AACtB,eAAO;AACP,aAAK,KAAK,MAAM,MAAM;AAAA,MACxB,OAAO;AACL,eAAO;AACP,aAAK,KAAK,oBAAoB,MAAM,MAAM,CAAC;AAAA,MAC7C;AAAA,IACF;AACA,QAAI,OAAO,WAAW,QAAW;AAC/B,aAAO;AACP,UAAI,MAAM,cAAc;AACtB,eAAO;AACP,aAAK,KAAK,MAAM,MAAM;AAAA,MACxB,OAAO;AACL,eAAO;AACP,aAAK,KAAK,oBAAoB,MAAM,MAAM,CAAC;AAAA,MAC7C;AAAA,IACF;AACA,QAAI,OAAO,YAAY,QAAW;AAChC,aAAO;AACP,WAAK,KAAK,MAAM,UAAU,IAAI,CAAC;AAAA,IACjC;AACA,QAAI,OAAO,UAAU,QAAW;AAC9B,aAAO;AACP,WAAK,KAAK,MAAM,KAAK;AAAA,IACvB;AACA,WAAO;AACP,SAAK,KAAK,KAAK;AAEf,UAAM,CAAC,eAAe,SAAS,IAAI,MAAM,QAAQ,IAAI;AAAA,MACnD,KAAK,OAAO,QAAQ,EAAE,KAAK,KAAoB,CAAC;AAAA,MAChD,KAAK,OAAO,QAAQ,+CAA+C;AAAA,IACrE,CAAC;AAED,QAAI,QAAQ;AACZ,eAAW,OAAO,cAAc,MAAM;AACpC,YAAM,eAAe,IAAI;AACzB,eAAS;AAAA,QACP,QAAQ,IAAI;AAAA,QACZ,QAAS,IAAI,UAA4B;AAAA,QACzC,IAAI,OAAO,IAAI,EAAE;AAAA,QACjB,OAAO,OAAO,IAAI,KAAK;AAAA,QACvB,SAAS,OAAO,IAAI,OAAO,MAAM;AAAA,QACjC,OAAO,IAAI;AAAA,QACX,WAAY,IAAI,aAA+B;AAAA,QAC/C,cAAc,eAAe,IAAI,KAAK,YAAY,IAAI;AAAA,MACxD,CAAC;AACD;AAAA,IACF;AAEA,WAAO,EAAE,YAAY,OAAO,UAAU,KAAK,CAAC,EAAE,CAAC,GAAG,MAAM;AAAA,EAC1D;AAAA;AAAA,EAGA,MAAM,SACJ,SAKA;AACA,UAAM,SAAS,oBAAI,IAGjB;AAEF,UAAM,KAAK,MAAM,KAAK,OAAO,YAAY,OAAO;AAChD,QAAI;AACF,iBAAW,EAAE,QAAQ,UAAU,KAAK,KAAK,SAAS;AAChD,cAAM,WAAW,MAAM,GAAG,QAAQ;AAAA,UAChC,KAAK;AAAA,UACL,MAAM,CAAC,MAAM;AAAA,QACf,CAAC;AACD,cAAM,UAAU,OAAO,SAAS,KAAK,CAAC,EAAE,CAAC;AACzC,cAAM,GAAG,QAAQ;AAAA,UACf,KAAK;AAAA,UACL,MAAM,CAAC,MAAM;AAAA,QACf,CAAC;AACD,cAAM,GAAG,QAAQ;AAAA,UACf,KAAK;AAAA,UACL,MAAM,CAAC,MAAM;AAAA,QACf,CAAC;AAED,cAAM,YACJ,aAAa,SAAY,iBAAiB;AAC5C,cAAM,YAAY,QAAQ,EAAE,aAAa,IAAI,WAAW,CAAC,EAAE;AAC3D,cAAM,OAAM,oBAAI,KAAK,GAAE,YAAY;AACnC,cAAM,MAAM,MAAM,GAAG,QAAQ;AAAA,UAC3B,KAAK;AAAA,UACL,MAAM;AAAA,YACJ;AAAA,YACA;AAAA,YACA,KAAK,UAAU,YAAY,CAAC,CAAC;AAAA,YAC7B,KAAK,UAAU,SAAS;AAAA,YACxB;AAAA,UACF;AAAA,QACF,CAAC;AAED,eAAO,IAAI,QAAQ;AAAA,UACjB;AAAA,UACA,WAAW;AAAA,YACT,IAAI,OAAO,IAAI,eAAe;AAAA,YAC9B;AAAA,YACA,SAAS;AAAA,YACT,SAAS,IAAI,KAAK,GAAG;AAAA,YACrB,MAAM;AAAA,YACN,MAAM,YAAY,CAAC;AAAA,YACnB,MAAM;AAAA,UACR;AAAA,QACF,CAAC;AAAA,MACH;AACA,YAAM,GAAG,OAAO;AAAA,IAClB,SAAS,GAAG;AACV,YAAM,GAAG,SAAS;AAClB,YAAM;AAAA,IACR;AAEA,WAAO;AAAA,EACT;AACF;","names":[]}
|