pushwork 2.0.0-preview → 2.0.0-preview.3
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/branches.d.ts +1 -0
- package/dist/branches.d.ts.map +1 -1
- package/dist/cli/commands.d.ts +71 -0
- package/dist/cli/commands.d.ts.map +1 -0
- package/dist/cli/commands.js +794 -0
- package/dist/cli/commands.js.map +1 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +19 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli.js +67 -112
- package/dist/cli.js.map +1 -1
- package/dist/commands.d.ts +58 -0
- package/dist/commands.d.ts.map +1 -0
- package/dist/commands.js +975 -0
- package/dist/commands.js.map +1 -0
- package/dist/config/index.d.ts +71 -0
- package/dist/config/index.d.ts.map +1 -0
- package/dist/config/index.js +314 -0
- package/dist/config/index.js.map +1 -0
- package/dist/config.d.ts +1 -2
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +1 -2
- package/dist/config.js.map +1 -1
- package/dist/core/change-detection.d.ts +80 -0
- package/dist/core/change-detection.d.ts.map +1 -0
- package/dist/core/change-detection.js +560 -0
- package/dist/core/change-detection.js.map +1 -0
- package/dist/core/config.d.ts +81 -0
- package/dist/core/config.d.ts.map +1 -0
- package/dist/core/config.js +304 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/index.d.ts +6 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +22 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/move-detection.d.ts +34 -0
- package/dist/core/move-detection.d.ts.map +1 -0
- package/dist/core/move-detection.js +128 -0
- package/dist/core/move-detection.js.map +1 -0
- package/dist/core/snapshot.d.ts +105 -0
- package/dist/core/snapshot.d.ts.map +1 -0
- package/dist/core/snapshot.js +254 -0
- package/dist/core/snapshot.js.map +1 -0
- package/dist/core/sync-engine.d.ts +177 -0
- package/dist/core/sync-engine.d.ts.map +1 -0
- package/dist/core/sync-engine.js +1471 -0
- package/dist/core/sync-engine.js.map +1 -0
- package/dist/index.d.ts +2 -4
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4 -14
- package/dist/index.js.map +1 -1
- package/dist/pushwork.d.ts +28 -61
- package/dist/pushwork.d.ts.map +1 -1
- package/dist/pushwork.js +127 -445
- package/dist/pushwork.js.map +1 -1
- package/dist/shapes/types.d.ts +1 -0
- package/dist/shapes/types.d.ts.map +1 -1
- package/dist/shapes/types.js.map +1 -1
- package/dist/shapes/vfs.d.ts.map +1 -1
- package/dist/shapes/vfs.js +6 -2
- package/dist/shapes/vfs.js.map +1 -1
- package/dist/snarf.d.ts +21 -0
- package/dist/snarf.d.ts.map +1 -0
- package/dist/snarf.js +117 -0
- package/dist/snarf.js.map +1 -0
- package/dist/stash.d.ts +0 -2
- package/dist/stash.d.ts.map +1 -1
- package/dist/stash.js +0 -1
- package/dist/stash.js.map +1 -1
- package/dist/types/config.d.ts +102 -0
- package/dist/types/config.d.ts.map +1 -0
- package/dist/types/config.js +10 -0
- package/dist/types/config.js.map +1 -0
- package/dist/types/documents.d.ts +88 -0
- package/dist/types/documents.d.ts.map +1 -0
- package/dist/types/documents.js +23 -0
- package/dist/types/documents.js.map +1 -0
- package/dist/types/index.d.ts +4 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +20 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/snapshot.d.ts +64 -0
- package/dist/types/snapshot.d.ts.map +1 -0
- package/dist/types/snapshot.js +3 -0
- package/dist/types/snapshot.js.map +1 -0
- package/dist/utils/content-similarity.d.ts +53 -0
- package/dist/utils/content-similarity.d.ts.map +1 -0
- package/dist/utils/content-similarity.js +155 -0
- package/dist/utils/content-similarity.js.map +1 -0
- package/dist/utils/content.d.ts +10 -0
- package/dist/utils/content.d.ts.map +1 -0
- package/dist/utils/content.js +35 -0
- package/dist/utils/content.js.map +1 -0
- package/dist/utils/directory.d.ts +24 -0
- package/dist/utils/directory.d.ts.map +1 -0
- package/dist/utils/directory.js +56 -0
- package/dist/utils/directory.js.map +1 -0
- package/dist/utils/fs.d.ts +74 -0
- package/dist/utils/fs.d.ts.map +1 -0
- package/dist/utils/fs.js +298 -0
- package/dist/utils/fs.js.map +1 -0
- package/dist/utils/index.d.ts +5 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +21 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/mime-types.d.ts +13 -0
- package/dist/utils/mime-types.d.ts.map +1 -0
- package/dist/utils/mime-types.js +247 -0
- package/dist/utils/mime-types.js.map +1 -0
- package/dist/utils/network-sync.d.ts +30 -0
- package/dist/utils/network-sync.d.ts.map +1 -0
- package/dist/utils/network-sync.js +391 -0
- package/dist/utils/network-sync.js.map +1 -0
- package/dist/utils/node-polyfills.d.ts +9 -0
- package/dist/utils/node-polyfills.d.ts.map +1 -0
- package/dist/utils/node-polyfills.js +9 -0
- package/dist/utils/node-polyfills.js.map +1 -0
- package/dist/utils/output.d.ts +129 -0
- package/dist/utils/output.d.ts.map +1 -0
- package/dist/utils/output.js +375 -0
- package/dist/utils/output.js.map +1 -0
- package/dist/utils/repo-factory.d.ts +15 -0
- package/dist/utils/repo-factory.d.ts.map +1 -0
- package/dist/utils/repo-factory.js +156 -0
- package/dist/utils/repo-factory.js.map +1 -0
- package/dist/utils/string-similarity.d.ts +14 -0
- package/dist/utils/string-similarity.d.ts.map +1 -0
- package/dist/utils/string-similarity.js +43 -0
- package/dist/utils/string-similarity.js.map +1 -0
- package/dist/utils/text-diff.d.ts +37 -0
- package/dist/utils/text-diff.d.ts.map +1 -0
- package/dist/utils/text-diff.js +131 -0
- package/dist/utils/text-diff.js.map +1 -0
- package/dist/utils/trace.d.ts +19 -0
- package/dist/utils/trace.d.ts.map +1 -0
- package/dist/utils/trace.js +68 -0
- package/dist/utils/trace.js.map +1 -0
- package/dist/version.d.ts +11 -0
- package/dist/version.d.ts.map +1 -0
- package/dist/version.js +93 -0
- package/dist/version.js.map +1 -0
- package/package.json +5 -1
- package/.prettierrc +0 -9
- package/flake.lock +0 -128
- package/flake.nix +0 -66
- package/pnpm-workspace.yaml +0 -5
- package/src/branches.ts +0 -93
- package/src/cli.ts +0 -292
- package/src/config.ts +0 -64
- package/src/fs-tree.ts +0 -70
- package/src/ignore.ts +0 -33
- package/src/index.ts +0 -38
- package/src/log.ts +0 -8
- package/src/pushwork.ts +0 -1055
- package/src/repo.ts +0 -76
- package/src/shapes/custom.ts +0 -29
- package/src/shapes/file.ts +0 -115
- package/src/shapes/index.ts +0 -19
- package/src/shapes/patchwork-folder.ts +0 -156
- package/src/shapes/types.ts +0 -79
- package/src/shapes/vfs.ts +0 -93
- package/src/stash.ts +0 -106
- package/test/integration/branches.test.ts +0 -389
- package/test/integration/pushwork.test.ts +0 -547
- package/test/setup.ts +0 -29
- package/test/unit/doc-shape.test.ts +0 -612
- package/tsconfig.json +0 -22
- package/vitest.config.ts +0 -14
|
@@ -0,0 +1,391 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.waitForBidirectionalSync = waitForBidirectionalSync;
|
|
37
|
+
exports.waitForSync = waitForSync;
|
|
38
|
+
const A = __importStar(require("@automerge/automerge"));
|
|
39
|
+
const output_1 = require("./output");
|
|
40
|
+
const directory_1 = require("./directory");
|
|
41
|
+
const isDebug = !!process.env.DEBUG;
|
|
42
|
+
function debug(...args) {
|
|
43
|
+
if (isDebug)
|
|
44
|
+
console.error("[pushwork:sync]", ...args);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Wait for bidirectional sync to stabilize.
|
|
48
|
+
* This function waits until document heads stop changing, indicating that
|
|
49
|
+
* both outgoing and incoming sync has completed.
|
|
50
|
+
*
|
|
51
|
+
* @param repo - The Automerge repository
|
|
52
|
+
* @param rootDirectoryUrl - The root directory URL to start traversal from
|
|
53
|
+
* @param options - Configuration options
|
|
54
|
+
*/
|
|
55
|
+
async function waitForBidirectionalSync(repo, rootDirectoryUrl, options = {}) {
|
|
56
|
+
const { timeoutMs = 10000, pollIntervalMs = 100, stableChecksRequired = 3, handles, } = options;
|
|
57
|
+
if (!rootDirectoryUrl) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
const startTime = Date.now();
|
|
61
|
+
let lastSeenHeads = new Map();
|
|
62
|
+
let stableCount = 0;
|
|
63
|
+
let pollCount = 0;
|
|
64
|
+
let dynamicTimeoutMs = timeoutMs;
|
|
65
|
+
debug(`waitForBidirectionalSync: starting (timeout=${timeoutMs}ms, stableChecks=${stableChecksRequired}${handles ? `, tracking ${handles.length} handles` : ', full tree scan'})`);
|
|
66
|
+
while (Date.now() - startTime < dynamicTimeoutMs) {
|
|
67
|
+
pollCount++;
|
|
68
|
+
// Get current heads: use provided handles if available, otherwise full tree scan
|
|
69
|
+
const currentHeads = handles
|
|
70
|
+
? getHandleHeads(handles)
|
|
71
|
+
: await getAllDocumentHeads(repo, rootDirectoryUrl);
|
|
72
|
+
// After first scan: scale timeout to tree size and reset the clock.
|
|
73
|
+
// The first scan is just establishing a baseline — its duration
|
|
74
|
+
// shouldn't count against the stability-wait timeout.
|
|
75
|
+
if (pollCount === 1) {
|
|
76
|
+
const scanDuration = Date.now() - startTime;
|
|
77
|
+
dynamicTimeoutMs = Math.max(timeoutMs, 5000 + currentHeads.size * 50) + scanDuration;
|
|
78
|
+
debug(`waitForBidirectionalSync: first scan took ${scanDuration}ms, timeout now ${dynamicTimeoutMs}ms for ${currentHeads.size} docs`);
|
|
79
|
+
}
|
|
80
|
+
// Check if heads are stable (no changes since last check)
|
|
81
|
+
const isStable = headsMapEqual(lastSeenHeads, currentHeads);
|
|
82
|
+
if (isStable) {
|
|
83
|
+
stableCount++;
|
|
84
|
+
debug(`waitForBidirectionalSync: stable check ${stableCount}/${stableChecksRequired} (${currentHeads.size} docs, poll #${pollCount})`);
|
|
85
|
+
if (stableCount >= stableChecksRequired) {
|
|
86
|
+
const elapsed = Date.now() - startTime;
|
|
87
|
+
debug(`waitForBidirectionalSync: converged in ${elapsed}ms after ${pollCount} polls (${currentHeads.size} docs)`);
|
|
88
|
+
output_1.out.taskLine(`Bidirectional sync converged (${currentHeads.size} docs, ${elapsed}ms)`);
|
|
89
|
+
return; // Converged!
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
else {
|
|
93
|
+
// Find which docs changed
|
|
94
|
+
if (lastSeenHeads.size > 0) {
|
|
95
|
+
const changedDocs = [];
|
|
96
|
+
for (const [url, heads] of currentHeads) {
|
|
97
|
+
if (lastSeenHeads.get(url) !== heads) {
|
|
98
|
+
changedDocs.push(url);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
const newDocs = currentHeads.size - lastSeenHeads.size;
|
|
102
|
+
if (newDocs > 0) {
|
|
103
|
+
debug(`waitForBidirectionalSync: ${newDocs} new docs discovered, ${changedDocs.length} docs changed heads (poll #${pollCount})`);
|
|
104
|
+
}
|
|
105
|
+
else if (changedDocs.length > 0) {
|
|
106
|
+
debug(`waitForBidirectionalSync: ${changedDocs.length} docs changed heads: ${changedDocs.slice(0, 5).join(", ")}${changedDocs.length > 5 ? ` ...and ${changedDocs.length - 5} more` : ""} (poll #${pollCount})`);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
debug(`waitForBidirectionalSync: initial scan found ${currentHeads.size} docs (poll #${pollCount})`);
|
|
111
|
+
}
|
|
112
|
+
if (stableCount > 0) {
|
|
113
|
+
debug(`waitForBidirectionalSync: heads changed after ${stableCount} stable checks, resetting`);
|
|
114
|
+
}
|
|
115
|
+
stableCount = 0;
|
|
116
|
+
lastSeenHeads = currentHeads;
|
|
117
|
+
}
|
|
118
|
+
await new Promise((r) => setTimeout(r, pollIntervalMs));
|
|
119
|
+
}
|
|
120
|
+
// Timeout - but don't throw, just log a warning
|
|
121
|
+
// The sync may still work, we just couldn't confirm stability
|
|
122
|
+
const elapsed = Date.now() - startTime;
|
|
123
|
+
debug(`waitForBidirectionalSync: timed out after ${elapsed}ms (${pollCount} polls, ${lastSeenHeads.size} docs tracked, reached ${stableCount}/${stableChecksRequired} stable checks)`);
|
|
124
|
+
output_1.out.taskLine(`Bidirectional sync timed out after ${(elapsed / 1000).toFixed(1)}s - document heads were still changing after ${pollCount} checks across ${lastSeenHeads.size} docs (reached ${stableCount}/${stableChecksRequired} stability checks). This may mean another peer is actively editing, or the sync server is slow to relay changes. The sync will continue but some remote changes may not be reflected yet.`, true);
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Get heads from a pre-collected set of handles (cheap, synchronous reads).
|
|
128
|
+
* Used for post-push stabilization where we already know which documents changed.
|
|
129
|
+
*/
|
|
130
|
+
function getHandleHeads(handles) {
|
|
131
|
+
const heads = new Map();
|
|
132
|
+
for (const handle of handles) {
|
|
133
|
+
heads.set((0, directory_1.getPlainUrl)(handle.url), JSON.stringify(handle.heads()));
|
|
134
|
+
}
|
|
135
|
+
return heads;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Get all document heads in the directory hierarchy.
|
|
139
|
+
* Returns a map of document URL -> serialized heads.
|
|
140
|
+
* Uses plain URLs (without heads) to ensure we see current document state.
|
|
141
|
+
*/
|
|
142
|
+
async function getAllDocumentHeads(repo, rootDirectoryUrl) {
|
|
143
|
+
const heads = new Map();
|
|
144
|
+
// Pass URL as-is; collectHeadsRecursive will strip heads
|
|
145
|
+
await collectHeadsRecursive(repo, rootDirectoryUrl, heads);
|
|
146
|
+
return heads;
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Recursively collect document heads from the directory hierarchy.
|
|
150
|
+
* Uses getPlainUrl to strip heads and always see the CURRENT state of documents.
|
|
151
|
+
*/
|
|
152
|
+
async function collectHeadsRecursive(repo, directoryUrl, heads) {
|
|
153
|
+
try {
|
|
154
|
+
const plainUrl = (0, directory_1.getPlainUrl)(directoryUrl);
|
|
155
|
+
const handle = await repo.find(plainUrl);
|
|
156
|
+
const doc = await handle.doc();
|
|
157
|
+
// Record this directory's heads (use plain URL as key for consistency)
|
|
158
|
+
heads.set(plainUrl, JSON.stringify(handle.heads()));
|
|
159
|
+
if (!doc || !doc.docs) {
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
162
|
+
// Process all entries in the directory concurrently
|
|
163
|
+
await Promise.all(doc.docs.map(async (entry) => {
|
|
164
|
+
if (entry.type === "folder") {
|
|
165
|
+
// Recurse into subdirectory (entry.url may have stale heads)
|
|
166
|
+
await collectHeadsRecursive(repo, entry.url, heads);
|
|
167
|
+
}
|
|
168
|
+
else if (entry.type === "file") {
|
|
169
|
+
// Get file document heads (strip heads from entry.url)
|
|
170
|
+
try {
|
|
171
|
+
const fileUrl = (0, directory_1.getPlainUrl)(entry.url);
|
|
172
|
+
const fileHandle = await repo.find(fileUrl);
|
|
173
|
+
heads.set(fileUrl, JSON.stringify(fileHandle.heads()));
|
|
174
|
+
}
|
|
175
|
+
catch {
|
|
176
|
+
// File document may not exist yet
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
}));
|
|
180
|
+
}
|
|
181
|
+
catch {
|
|
182
|
+
// Directory may not exist yet
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Compare two heads maps for equality.
|
|
187
|
+
*/
|
|
188
|
+
function headsMapEqual(a, b) {
|
|
189
|
+
if (a.size !== b.size) {
|
|
190
|
+
return false;
|
|
191
|
+
}
|
|
192
|
+
for (const [key, value] of a) {
|
|
193
|
+
if (b.get(key) !== value) {
|
|
194
|
+
return false;
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
return true;
|
|
198
|
+
}
|
|
199
|
+
/** Maximum documents to sync concurrently to avoid flooding the server */
|
|
200
|
+
const SYNC_BATCH_SIZE = 10;
|
|
201
|
+
/**
|
|
202
|
+
* Wait for a single document handle to sync to the server.
|
|
203
|
+
* Resolves with the handle on success, rejects with the handle on timeout.
|
|
204
|
+
*/
|
|
205
|
+
function waitForHandleSync(handle, syncServerStorageId, timeoutMs, startTime) {
|
|
206
|
+
return new Promise((resolve, reject) => {
|
|
207
|
+
let pollInterval;
|
|
208
|
+
const cleanup = () => {
|
|
209
|
+
clearTimeout(timeout);
|
|
210
|
+
clearInterval(pollInterval);
|
|
211
|
+
handle.off("remote-heads", onRemoteHeads);
|
|
212
|
+
};
|
|
213
|
+
const onConverged = () => {
|
|
214
|
+
debug(`waitForSync: ${handle.url}... converged in ${Date.now() - startTime}ms`);
|
|
215
|
+
cleanup();
|
|
216
|
+
resolve(handle);
|
|
217
|
+
};
|
|
218
|
+
const timeout = setTimeout(() => {
|
|
219
|
+
debug(`waitForSync: ${handle.url}... timed out after ${timeoutMs}ms`);
|
|
220
|
+
cleanup();
|
|
221
|
+
reject(handle);
|
|
222
|
+
}, timeoutMs);
|
|
223
|
+
const isConverged = () => {
|
|
224
|
+
const localHeads = handle.heads();
|
|
225
|
+
const info = handle.getSyncInfo(syncServerStorageId);
|
|
226
|
+
return A.equals(localHeads, info?.lastHeads);
|
|
227
|
+
};
|
|
228
|
+
const onRemoteHeads = ({ storageId, }) => {
|
|
229
|
+
if (storageId === syncServerStorageId && isConverged()) {
|
|
230
|
+
onConverged();
|
|
231
|
+
}
|
|
232
|
+
};
|
|
233
|
+
// Initial check
|
|
234
|
+
if (isConverged()) {
|
|
235
|
+
cleanup();
|
|
236
|
+
resolve(handle);
|
|
237
|
+
return;
|
|
238
|
+
}
|
|
239
|
+
// Start polling and event listening
|
|
240
|
+
pollInterval = setInterval(() => {
|
|
241
|
+
if (isConverged()) {
|
|
242
|
+
onConverged();
|
|
243
|
+
}
|
|
244
|
+
}, 100);
|
|
245
|
+
handle.on("remote-heads", onRemoteHeads);
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Wait for documents to sync to the remote server.
|
|
250
|
+
* Processes handles in batches to avoid flooding the server.
|
|
251
|
+
* Returns a result with any failed handles instead of throwing,
|
|
252
|
+
* so callers can attempt recovery (e.g. recreating documents).
|
|
253
|
+
*/
|
|
254
|
+
async function waitForSync(handlesToWaitOn, syncServerStorageId, timeoutMs = 60000) {
|
|
255
|
+
const startTime = Date.now();
|
|
256
|
+
if (handlesToWaitOn.length === 0) {
|
|
257
|
+
debug("waitForSync: no documents to sync");
|
|
258
|
+
return { failed: [] };
|
|
259
|
+
}
|
|
260
|
+
// When no StorageId is available (Subduction mode), use head-stability
|
|
261
|
+
// polling. The SubductionSource handles sync internally — we just wait
|
|
262
|
+
// for each handle's heads to stop changing.
|
|
263
|
+
if (!syncServerStorageId) {
|
|
264
|
+
debug(`waitForSync: no storage ID, using head-stability polling for ${handlesToWaitOn.length} documents`);
|
|
265
|
+
output_1.out.taskLine(`Waiting for ${handlesToWaitOn.length} documents to sync`);
|
|
266
|
+
return waitForSyncViaHeadStability(handlesToWaitOn, timeoutMs, startTime);
|
|
267
|
+
}
|
|
268
|
+
debug(`waitForSync: waiting for ${handlesToWaitOn.length} documents (timeout=${timeoutMs}ms, batchSize=${SYNC_BATCH_SIZE})`);
|
|
269
|
+
// Separate already-synced from needs-sync
|
|
270
|
+
const needsSync = [];
|
|
271
|
+
let alreadySynced = 0;
|
|
272
|
+
for (const handle of handlesToWaitOn) {
|
|
273
|
+
const heads = handle.heads();
|
|
274
|
+
const syncInfo = handle.getSyncInfo(syncServerStorageId);
|
|
275
|
+
const remoteHeads = syncInfo?.lastHeads;
|
|
276
|
+
if (A.equals(heads, remoteHeads)) {
|
|
277
|
+
alreadySynced++;
|
|
278
|
+
debug(`waitForSync: ${handle.url}... already synced`);
|
|
279
|
+
}
|
|
280
|
+
else {
|
|
281
|
+
debug(`waitForSync: ${handle.url}... needs sync (remoteHeads=${remoteHeads ? 'present' : 'missing'})`);
|
|
282
|
+
needsSync.push(handle);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
if (needsSync.length > 0) {
|
|
286
|
+
debug(`waitForSync: ${alreadySynced} already synced, ${needsSync.length} need sync`);
|
|
287
|
+
output_1.out.taskLine(`Uploading: ${alreadySynced}/${handlesToWaitOn.length} already synced, waiting for ${needsSync.length} more`);
|
|
288
|
+
}
|
|
289
|
+
else {
|
|
290
|
+
debug(`waitForSync: all ${handlesToWaitOn.length} already synced`);
|
|
291
|
+
return { failed: [] };
|
|
292
|
+
}
|
|
293
|
+
// Process in batches to avoid flooding the server
|
|
294
|
+
const failed = [];
|
|
295
|
+
let synced = alreadySynced;
|
|
296
|
+
for (let i = 0; i < needsSync.length; i += SYNC_BATCH_SIZE) {
|
|
297
|
+
const batch = needsSync.slice(i, i + SYNC_BATCH_SIZE);
|
|
298
|
+
const batchNum = Math.floor(i / SYNC_BATCH_SIZE) + 1;
|
|
299
|
+
const totalBatches = Math.ceil(needsSync.length / SYNC_BATCH_SIZE);
|
|
300
|
+
if (totalBatches > 1) {
|
|
301
|
+
debug(`waitForSync: batch ${batchNum}/${totalBatches} (${batch.length} docs)`);
|
|
302
|
+
output_1.out.update(`Uploading batch ${batchNum}/${totalBatches} (${synced}/${handlesToWaitOn.length} done)`);
|
|
303
|
+
}
|
|
304
|
+
const results = await Promise.allSettled(batch.map(handle => waitForHandleSync(handle, syncServerStorageId, timeoutMs, startTime)));
|
|
305
|
+
for (const result of results) {
|
|
306
|
+
if (result.status === "rejected") {
|
|
307
|
+
failed.push(result.reason);
|
|
308
|
+
}
|
|
309
|
+
else {
|
|
310
|
+
synced++;
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
const elapsed = Date.now() - startTime;
|
|
315
|
+
if (failed.length > 0) {
|
|
316
|
+
debug(`waitForSync: ${failed.length} documents failed after ${elapsed}ms`);
|
|
317
|
+
output_1.out.taskLine(`Upload: ${synced} synced, ${failed.length} failed after ${(elapsed / 1000).toFixed(1)}s`, true);
|
|
318
|
+
}
|
|
319
|
+
else {
|
|
320
|
+
debug(`waitForSync: all ${handlesToWaitOn.length} documents synced in ${elapsed}ms (${alreadySynced} were already synced)`);
|
|
321
|
+
output_1.out.taskLine(`All ${handlesToWaitOn.length} documents uploaded to server (${(elapsed / 1000).toFixed(1)}s)`);
|
|
322
|
+
}
|
|
323
|
+
return { failed };
|
|
324
|
+
}
|
|
325
|
+
/**
|
|
326
|
+
* Wait for sync by polling head stability (Subduction mode).
|
|
327
|
+
* Each handle's heads are polled until they remain unchanged for
|
|
328
|
+
* several consecutive checks, indicating the SubductionSource has
|
|
329
|
+
* finished syncing.
|
|
330
|
+
*/
|
|
331
|
+
async function waitForSyncViaHeadStability(handles, timeoutMs, startTime) {
|
|
332
|
+
const failed = [];
|
|
333
|
+
let synced = 0;
|
|
334
|
+
// Process in batches
|
|
335
|
+
for (let i = 0; i < handles.length; i += SYNC_BATCH_SIZE) {
|
|
336
|
+
const batch = handles.slice(i, i + SYNC_BATCH_SIZE);
|
|
337
|
+
const results = await Promise.allSettled(batch.map(handle => waitForHandleHeadStability(handle, timeoutMs, startTime)));
|
|
338
|
+
for (const result of results) {
|
|
339
|
+
if (result.status === "rejected") {
|
|
340
|
+
failed.push(result.reason);
|
|
341
|
+
}
|
|
342
|
+
else {
|
|
343
|
+
synced++;
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
const elapsed = Date.now() - startTime;
|
|
348
|
+
if (failed.length > 0) {
|
|
349
|
+
debug(`waitForSync(heads): ${failed.length} documents failed after ${elapsed}ms`);
|
|
350
|
+
output_1.out.taskLine(`Sync: ${synced} synced, ${failed.length} timed out after ${(elapsed / 1000).toFixed(1)}s`, true);
|
|
351
|
+
}
|
|
352
|
+
else {
|
|
353
|
+
debug(`waitForSync(heads): all ${handles.length} documents synced in ${elapsed}ms`);
|
|
354
|
+
output_1.out.taskLine(`All ${handles.length} documents synced (${(elapsed / 1000).toFixed(1)}s)`);
|
|
355
|
+
}
|
|
356
|
+
return { failed };
|
|
357
|
+
}
|
|
358
|
+
/**
|
|
359
|
+
* Wait for a single handle's heads to stabilize.
|
|
360
|
+
* Polls heads at 100ms intervals; resolves after 3 consecutive stable
|
|
361
|
+
* checks, rejects on timeout.
|
|
362
|
+
*/
|
|
363
|
+
function waitForHandleHeadStability(handle, timeoutMs, startTime) {
|
|
364
|
+
return new Promise((resolve, reject) => {
|
|
365
|
+
let lastHeads = JSON.stringify(handle.heads());
|
|
366
|
+
let stableCount = 0;
|
|
367
|
+
const stableRequired = 3;
|
|
368
|
+
const pollInterval = setInterval(() => {
|
|
369
|
+
const currentHeads = JSON.stringify(handle.heads());
|
|
370
|
+
if (currentHeads === lastHeads) {
|
|
371
|
+
stableCount++;
|
|
372
|
+
if (stableCount >= stableRequired) {
|
|
373
|
+
clearInterval(pollInterval);
|
|
374
|
+
clearTimeout(timeout);
|
|
375
|
+
debug(`waitForSync(heads): ${handle.url}... converged in ${Date.now() - startTime}ms`);
|
|
376
|
+
resolve(handle);
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
else {
|
|
380
|
+
stableCount = 0;
|
|
381
|
+
lastHeads = currentHeads;
|
|
382
|
+
}
|
|
383
|
+
}, 100);
|
|
384
|
+
const timeout = setTimeout(() => {
|
|
385
|
+
clearInterval(pollInterval);
|
|
386
|
+
debug(`waitForSync(heads): ${handle.url}... timed out after ${timeoutMs}ms`);
|
|
387
|
+
reject(handle);
|
|
388
|
+
}, timeoutMs);
|
|
389
|
+
});
|
|
390
|
+
}
|
|
391
|
+
//# sourceMappingURL=network-sync.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"network-sync.js","sourceRoot":"","sources":["../../src/utils/network-sync.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAyBA,4DA0FC;AA6KD,kCAqFC;AA/WD,wDAA0C;AAC1C,qCAA+B;AAE/B,2CAA0C;AAE1C,MAAM,OAAO,GAAG,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC;AACpC,SAAS,KAAK,CAAC,GAAG,IAAW;IAC3B,IAAI,OAAO;QAAE,OAAO,CAAC,KAAK,CAAC,iBAAiB,EAAE,GAAG,IAAI,CAAC,CAAC;AACzD,CAAC;AAED;;;;;;;;GAQG;AACI,KAAK,UAAU,wBAAwB,CAC5C,IAAU,EACV,gBAA0C,EAC1C,UAKI,EAAE;IAEN,MAAM,EACJ,SAAS,GAAG,KAAK,EACjB,cAAc,GAAG,GAAG,EACpB,oBAAoB,GAAG,CAAC,EACxB,OAAO,GACR,GAAG,OAAO,CAAC;IAEZ,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACtB,OAAO;IACT,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAC7B,IAAI,aAAa,GAAG,IAAI,GAAG,EAAkB,CAAC;IAC9C,IAAI,WAAW,GAAG,CAAC,CAAC;IACpB,IAAI,SAAS,GAAG,CAAC,CAAC;IAClB,IAAI,gBAAgB,GAAG,SAAS,CAAC;IAEjC,KAAK,CAAC,+CAA+C,SAAS,oBAAoB,oBAAoB,GAAG,OAAO,CAAC,CAAC,CAAC,cAAc,OAAO,CAAC,MAAM,UAAU,CAAC,CAAC,CAAC,kBAAkB,GAAG,CAAC,CAAC;IAEnL,OAAO,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,GAAG,gBAAgB,EAAE,CAAC;QACjD,SAAS,EAAE,CAAC;QACZ,iFAAiF;QACjF,MAAM,YAAY,GAAG,OAAO;YAC1B,CAAC,CAAC,cAAc,CAAC,OAAO,CAAC;YACzB,CAAC,CAAC,MAAM,mBAAmB,CAAC,IAAI,EAAE,gBAAgB,CAAC,CAAC;QAEtD,oEAAoE;QACpE,gEAAgE;QAChE,sDAAsD;QACtD,IAAI,SAAS,KAAK,CAAC,EAAE,CAAC;YACpB,MAAM,YAAY,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YAC5C,gBAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,GAAG,YAAY,CAAC,IAAI,GAAG,EAAE,CAAC,GAAG,YAAY,CAAC;YACrF,KAAK,CAAC,6CAA6C,YAAY,mBAAmB,gBAAgB,UAAU,YAAY,CAAC,IAAI,OAAO,CAAC,CAAC;QACxI,CAAC;QAED,0DAA0D;QAC1D,MAAM,QAAQ,GAAG,aAAa,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC;QAE5D,IAAI,QAAQ,EAAE,CAAC;YACb,WAAW,EAAE,CAAC;YACd,KAAK,CAAC,0CAA0C,WAAW,IAAI,oBAAoB,KAAK,YAAY,CAAC,IAAI,gBAAgB,SAAS,GAAG,CAAC,CAAC;YACvI,IAAI,WAAW,IAAI,oBAAoB,EAAE,CAAC;gBACxC,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;gBACvC,KAAK,CAAC,0CAA0C,OAAO,YAAY,SAAS,WAAW,YAAY,CAAC,IAAI,QAAQ,CAAC,CAAC;gBAClH,YAAG,CAAC,QAAQ,CAAC,iCAAiC,YAAY,CAAC,IAAI,UAAU,OAAO,KAAK,CAAC,CAAC;gBACvF,OAAO,CAAC,aAAa;YACvB,CAAC;QACH,CAAC;aAAM,CAAC;YACN,0BAA0B;YAC1B,IAAI,aAAa,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;gBAC3B,MAAM,WAAW,GAAa,EAAE,CAAC;gBACjC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,YAAY,EAAE,CAAC;oBACxC,IAAI,aAAa,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,KAAK,EAAE,CAAC;wBACrC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBACxB,CAAC;gBACH,CAAC;gBACD,MAAM,OAAO,GAAG,YAAY,CAAC,IAAI,GAAG,aAAa,CAAC,IAAI,CAAC;gBACvD,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;oBAChB,KAAK,CAAC,6BAA6B,OAAO,yBAAyB,WAAW,CAAC,MAAM,8BAA8B,SAAS,GAAG,CAAC,CAAC;gBACnI,CAAC;qBAAM,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAClC,KAAK,CAAC,6BAA6B,WAAW,CAAC,MAAM,wBAAwB,WAAW,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,WAAW,CAAC,MAAM,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,WAAW,SAAS,GAAG,CAAC,CAAC;gBACnN,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,KAAK,CAAC,gDAAgD,YAAY,CAAC,IAAI,gBAAgB,SAAS,GAAG,CAAC,CAAC;YACvG,CAAC;YACD,IAAI,WAAW,GAAG,CAAC,EAAE,CAAC;gBACpB,KAAK,CAAC,iDAAiD,WAAW,2BAA2B,CAAC,CAAC;YACjG,CAAC;YACD,WAAW,GAAG,CAAC,CAAC;YAChB,aAAa,GAAG,YAAY,CAAC;QAC/B,CAAC;QAED,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,cAAc,CAAC,CAAC,CAAC;IAC1D,CAAC;IAED,gDAAgD;IAChD,8DAA8D;IAC9D,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;IACvC,KAAK,CAAC,6CAA6C,OAAO,OAAO,SAAS,WAAW,aAAa,CAAC,IAAI,0BAA0B,WAAW,IAAI,oBAAoB,iBAAiB,CAAC,CAAC;IACvL,YAAG,CAAC,QAAQ,CAAC,sCAAsC,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,gDAAgD,SAAS,kBAAkB,aAAa,CAAC,IAAI,kBAAkB,WAAW,IAAI,oBAAoB,2LAA2L,EAAE,IAAI,CAAC,CAAC;AACra,CAAC;AAED;;;GAGG;AACH,SAAS,cAAc,CACrB,OAA6B;IAE7B,MAAM,KAAK,GAAG,IAAI,GAAG,EAAkB,CAAC;IACxC,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;QAC7B,KAAK,CAAC,GAAG,CAAC,IAAA,uBAAW,EAAC,MAAM,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;IACrE,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;;;GAIG;AACH,KAAK,UAAU,mBAAmB,CAChC,IAAU,EACV,gBAA8B;IAE9B,MAAM,KAAK,GAAG,IAAI,GAAG,EAAkB,CAAC;IACxC,yDAAyD;IACzD,MAAM,qBAAqB,CAAC,IAAI,EAAE,gBAAgB,EAAE,KAAK,CAAC,CAAC;IAC3D,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;;GAGG;AACH,KAAK,UAAU,qBAAqB,CAClC,IAAU,EACV,YAA0B,EAC1B,KAA0B;IAE1B,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,IAAA,uBAAW,EAAC,YAAY,CAAC,CAAC;QAC3C,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAoB,QAAQ,CAAC,CAAC;QAC5D,MAAM,GAAG,GAAG,MAAM,MAAM,CAAC,GAAG,EAAE,CAAC;QAE/B,uEAAuE;QACvE,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QAEpD,IAAI,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;YACtB,OAAO;QACT,CAAC;QAED,oDAAoD;QACpD,MAAM,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,KAAwD,EAAE,EAAE;YAChG,IAAI,KAAK,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;gBAC5B,6DAA6D;gBAC7D,MAAM,qBAAqB,CAAC,IAAI,EAAE,KAAK,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;YACtD,CAAC;iBAAM,IAAI,KAAK,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;gBACjC,uDAAuD;gBACvD,IAAI,CAAC;oBACH,MAAM,OAAO,GAAG,IAAA,uBAAW,EAAC,KAAK,CAAC,GAAG,CAAC,CAAC;oBACvC,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;oBAC5C,KAAK,CAAC,GAAG,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;gBACzD,CAAC;gBAAC,MAAM,CAAC;oBACP,kCAAkC;gBACpC,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC,CAAC;IACN,CAAC;IAAC,MAAM,CAAC;QACP,8BAA8B;IAChC,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,aAAa,CACpB,CAAsB,EACtB,CAAsB;IAEtB,IAAI,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;QACtB,OAAO,KAAK,CAAC;IACf,CAAC;IACD,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;QAC7B,IAAI,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,KAAK,EAAE,CAAC;YACzB,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AASD,0EAA0E;AAC1E,MAAM,eAAe,GAAG,EAAE,CAAC;AAE3B;;;GAGG;AACH,SAAS,iBAAiB,CACxB,MAA0B,EAC1B,mBAA8B,EAC9B,SAAiB,EACjB,SAAiB;IAEjB,OAAO,IAAI,OAAO,CAAqB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACzD,IAAI,YAA4B,CAAC;QAEjC,MAAM,OAAO,GAAG,GAAG,EAAE;YACnB,YAAY,CAAC,OAAO,CAAC,CAAC;YACtB,aAAa,CAAC,YAAY,CAAC,CAAC;YAC5B,MAAM,CAAC,GAAG,CAAC,cAAc,EAAE,aAAa,CAAC,CAAC;QAC5C,CAAC,CAAC;QAEF,MAAM,WAAW,GAAG,GAAG,EAAE;YACvB,KAAK,CAAC,gBAAgB,MAAM,CAAC,GAAG,oBAAoB,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,IAAI,CAAC,CAAC;YAChF,OAAO,EAAE,CAAC;YACV,OAAO,CAAC,MAAM,CAAC,CAAC;QAClB,CAAC,CAAC;QAEF,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;YAC9B,KAAK,CAAC,gBAAgB,MAAM,CAAC,GAAG,uBAAuB,SAAS,IAAI,CAAC,CAAC;YACtE,OAAO,EAAE,CAAC;YACV,MAAM,CAAC,MAAM,CAAC,CAAC;QACjB,CAAC,EAAE,SAAS,CAAC,CAAC;QAEd,MAAM,WAAW,GAAG,GAAG,EAAE;YACvB,MAAM,UAAU,GAAG,MAAM,CAAC,KAAK,EAAE,CAAC;YAClC,MAAM,IAAI,GAAG,MAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;YACrD,OAAO,CAAC,CAAC,MAAM,CAAC,UAAU,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC;QAEF,MAAM,aAAa,GAAG,CAAC,EACrB,SAAS,GAIV,EAAE,EAAE;YACH,IAAI,SAAS,KAAK,mBAAmB,IAAI,WAAW,EAAE,EAAE,CAAC;gBACvD,WAAW,EAAE,CAAC;YAChB,CAAC;QACH,CAAC,CAAC;QAEF,gBAAgB;QAChB,IAAI,WAAW,EAAE,EAAE,CAAC;YAClB,OAAO,EAAE,CAAC;YACV,OAAO,CAAC,MAAM,CAAC,CAAC;YAChB,OAAO;QACT,CAAC;QAED,oCAAoC;QACpC,YAAY,GAAG,WAAW,CAAC,GAAG,EAAE;YAC9B,IAAI,WAAW,EAAE,EAAE,CAAC;gBAClB,WAAW,EAAE,CAAC;YAChB,CAAC;QACH,CAAC,EAAE,GAAG,CAAC,CAAC;QAER,MAAM,CAAC,EAAE,CAAC,cAAc,EAAE,aAAa,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;GAKG;AACI,KAAK,UAAU,WAAW,CAC/B,eAAqC,EACrC,mBAA+B,EAC/B,YAAoB,KAAK;IAEzB,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAE7B,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjC,KAAK,CAAC,mCAAmC,CAAC,CAAC;QAC3C,OAAO,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;IACxB,CAAC;IAED,uEAAuE;IACvE,uEAAuE;IACvE,4CAA4C;IAC5C,IAAI,CAAC,mBAAmB,EAAE,CAAC;QACzB,KAAK,CAAC,gEAAgE,eAAe,CAAC,MAAM,YAAY,CAAC,CAAC;QAC1G,YAAG,CAAC,QAAQ,CAAC,eAAe,eAAe,CAAC,MAAM,oBAAoB,CAAC,CAAC;QACxE,OAAO,2BAA2B,CAAC,eAAe,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC;IAC5E,CAAC;IAED,KAAK,CAAC,4BAA4B,eAAe,CAAC,MAAM,uBAAuB,SAAS,iBAAiB,eAAe,GAAG,CAAC,CAAC;IAE7H,0CAA0C;IAC1C,MAAM,SAAS,GAAyB,EAAE,CAAC;IAC3C,IAAI,aAAa,GAAG,CAAC,CAAC;IAEtB,KAAK,MAAM,MAAM,IAAI,eAAe,EAAE,CAAC;QACrC,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,EAAE,CAAC;QAC7B,MAAM,QAAQ,GAAG,MAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,CAAC;QACzD,MAAM,WAAW,GAAG,QAAQ,EAAE,SAAS,CAAC;QACxC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,WAAW,CAAC,EAAE,CAAC;YACjC,aAAa,EAAE,CAAC;YAChB,KAAK,CAAC,gBAAgB,MAAM,CAAC,GAAG,oBAAoB,CAAC,CAAC;QACxD,CAAC;aAAM,CAAC;YACN,KAAK,CAAC,gBAAgB,MAAM,CAAC,GAAG,+BAA+B,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC;YACvG,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACzB,CAAC;IACH,CAAC;IAED,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACzB,KAAK,CAAC,gBAAgB,aAAa,oBAAoB,SAAS,CAAC,MAAM,YAAY,CAAC,CAAC;QACrF,YAAG,CAAC,QAAQ,CAAC,cAAc,aAAa,IAAI,eAAe,CAAC,MAAM,gCAAgC,SAAS,CAAC,MAAM,OAAO,CAAC,CAAC;IAC7H,CAAC;SAAM,CAAC;QACN,KAAK,CAAC,oBAAoB,eAAe,CAAC,MAAM,iBAAiB,CAAC,CAAC;QACnE,OAAO,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;IACxB,CAAC;IAED,kDAAkD;IAClD,MAAM,MAAM,GAAyB,EAAE,CAAC;IACxC,IAAI,MAAM,GAAG,aAAa,CAAC;IAE3B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,SAAS,CAAC,MAAM,EAAE,CAAC,IAAI,eAAe,EAAE,CAAC;QAC3D,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,eAAe,CAAC,CAAC;QACtD,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,eAAe,CAAC,GAAG,CAAC,CAAC;QACrD,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,GAAG,eAAe,CAAC,CAAC;QAEnE,IAAI,YAAY,GAAG,CAAC,EAAE,CAAC;YACrB,KAAK,CAAC,sBAAsB,QAAQ,IAAI,YAAY,KAAK,KAAK,CAAC,MAAM,QAAQ,CAAC,CAAC;YAC/E,YAAG,CAAC,MAAM,CAAC,mBAAmB,QAAQ,IAAI,YAAY,KAAK,MAAM,IAAI,eAAe,CAAC,MAAM,QAAQ,CAAC,CAAC;QACvG,CAAC;QAED,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CACtC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,iBAAiB,CAAC,MAAM,EAAE,mBAAmB,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC,CAC1F,CAAC;QAEF,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,MAA4B,CAAC,CAAC;YACnD,CAAC;iBAAM,CAAC;gBACN,MAAM,EAAE,CAAC;YACX,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;IACvC,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,KAAK,CAAC,gBAAgB,MAAM,CAAC,MAAM,2BAA2B,OAAO,IAAI,CAAC,CAAC;QAC3E,YAAG,CAAC,QAAQ,CAAC,WAAW,MAAM,YAAY,MAAM,CAAC,MAAM,iBAAiB,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IAChH,CAAC;SAAM,CAAC;QACN,KAAK,CAAC,oBAAoB,eAAe,CAAC,MAAM,wBAAwB,OAAO,OAAO,aAAa,uBAAuB,CAAC,CAAC;QAC5H,YAAG,CAAC,QAAQ,CAAC,OAAO,eAAe,CAAC,MAAM,kCAAkC,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;IAC/G,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,CAAC;AACpB,CAAC;AAED;;;;;GAKG;AACH,KAAK,UAAU,2BAA2B,CACxC,OAA6B,EAC7B,SAAiB,EACjB,SAAiB;IAEjB,MAAM,MAAM,GAAyB,EAAE,CAAC;IACxC,IAAI,MAAM,GAAG,CAAC,CAAC;IAEf,qBAAqB;IACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,IAAI,eAAe,EAAE,CAAC;QACzD,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,eAAe,CAAC,CAAC;QAEpD,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CACtC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,0BAA0B,CAAC,MAAM,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC,CAC9E,CAAC;QAEF,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,MAA4B,CAAC,CAAC;YACnD,CAAC;iBAAM,CAAC;gBACN,MAAM,EAAE,CAAC;YACX,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;IACvC,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,KAAK,CAAC,uBAAuB,MAAM,CAAC,MAAM,2BAA2B,OAAO,IAAI,CAAC,CAAC;QAClF,YAAG,CAAC,QAAQ,CAAC,SAAS,MAAM,YAAY,MAAM,CAAC,MAAM,oBAAoB,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IACjH,CAAC;SAAM,CAAC;QACN,KAAK,CAAC,2BAA2B,OAAO,CAAC,MAAM,wBAAwB,OAAO,IAAI,CAAC,CAAC;QACpF,YAAG,CAAC,QAAQ,CAAC,OAAO,OAAO,CAAC,MAAM,sBAAsB,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;IAC3F,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,CAAC;AACpB,CAAC;AAED;;;;GAIG;AACH,SAAS,0BAA0B,CACjC,MAA0B,EAC1B,SAAiB,EACjB,SAAiB;IAEjB,OAAO,IAAI,OAAO,CAAqB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACzD,IAAI,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;QAC/C,IAAI,WAAW,GAAG,CAAC,CAAC;QACpB,MAAM,cAAc,GAAG,CAAC,CAAC;QAEzB,MAAM,YAAY,GAAG,WAAW,CAAC,GAAG,EAAE;YACpC,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;YACpD,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;gBAC/B,WAAW,EAAE,CAAC;gBACd,IAAI,WAAW,IAAI,cAAc,EAAE,CAAC;oBAClC,aAAa,CAAC,YAAY,CAAC,CAAC;oBAC5B,YAAY,CAAC,OAAO,CAAC,CAAC;oBACtB,KAAK,CAAC,uBAAuB,MAAM,CAAC,GAAG,oBAAoB,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,IAAI,CAAC,CAAC;oBACvF,OAAO,CAAC,MAAM,CAAC,CAAC;gBAClB,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,WAAW,GAAG,CAAC,CAAC;gBAChB,SAAS,GAAG,YAAY,CAAC;YAC3B,CAAC;QACH,CAAC,EAAE,GAAG,CAAC,CAAC;QAER,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;YAC9B,aAAa,CAAC,YAAY,CAAC,CAAC;YAC5B,KAAK,CAAC,uBAAuB,MAAM,CAAC,GAAG,uBAAuB,SAAS,IAAI,CAAC,CAAC;YAC7E,MAAM,CAAC,MAAM,CAAC,CAAC;QACjB,CAAC,EAAE,SAAS,CAAC,CAAC;IAChB,CAAC,CAAC,CAAC;AACL,CAAC"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Polyfills for browser APIs required by @automerge/automerge-subduction.
|
|
3
|
+
* Must be imported before any subduction code.
|
|
4
|
+
*
|
|
5
|
+
* The Subduction WASM module uses IndexedDB for key persistence
|
|
6
|
+
* (via WebCryptoSigner). In Node.js we provide a fake-indexeddb polyfill.
|
|
7
|
+
*/
|
|
8
|
+
import "fake-indexeddb/auto";
|
|
9
|
+
//# sourceMappingURL=node-polyfills.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"node-polyfills.d.ts","sourceRoot":"","sources":["../../src/utils/node-polyfills.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AACH,OAAO,qBAAqB,CAAC"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Polyfills for browser APIs required by @automerge/automerge-subduction.
|
|
3
|
+
* Must be imported before any subduction code.
|
|
4
|
+
*
|
|
5
|
+
* The Subduction WASM module uses IndexedDB for key persistence
|
|
6
|
+
* (via WebCryptoSigner). In Node.js we provide a fake-indexeddb polyfill.
|
|
7
|
+
*/
|
|
8
|
+
import "fake-indexeddb/auto";
|
|
9
|
+
//# sourceMappingURL=node-polyfills.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"node-polyfills.js","sourceRoot":"","sources":["../../src/utils/node-polyfills.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AACH,OAAO,qBAAqB,CAAC"}
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Clean terminal output manager (Singleton)
|
|
3
|
+
* - Progress stays on one line (spinner updates in place)
|
|
4
|
+
* - No emojis
|
|
5
|
+
* - Background colors for section headers
|
|
6
|
+
* - Minimal output
|
|
7
|
+
* - Supports scrolling task lines (max-lines)
|
|
8
|
+
*/
|
|
9
|
+
export declare class Output {
|
|
10
|
+
#private;
|
|
11
|
+
private static instance;
|
|
12
|
+
private spinner;
|
|
13
|
+
private taskStartTime;
|
|
14
|
+
private taskOriginalMessage;
|
|
15
|
+
private taskCurrentMessage;
|
|
16
|
+
private taskLines;
|
|
17
|
+
private taskMaxLines;
|
|
18
|
+
private constructor();
|
|
19
|
+
/**
|
|
20
|
+
* Get the singleton instance
|
|
21
|
+
*/
|
|
22
|
+
static getInstance(): Output;
|
|
23
|
+
/**
|
|
24
|
+
* Reset the singleton (useful for testing)
|
|
25
|
+
*/
|
|
26
|
+
static reset(): void;
|
|
27
|
+
/**
|
|
28
|
+
* Start a task with spinner - updates in place
|
|
29
|
+
* Completes any previous task before starting the new one
|
|
30
|
+
* @param message - The task message
|
|
31
|
+
* @param maxLines - Maximum number of task lines to show (0 = unlimited, lines scroll)
|
|
32
|
+
*/
|
|
33
|
+
task(message: string, maxLines?: number): void;
|
|
34
|
+
/**
|
|
35
|
+
* Update spinner text (stays on same line)
|
|
36
|
+
*/
|
|
37
|
+
update(message: string): void;
|
|
38
|
+
/**
|
|
39
|
+
* Add a line to the active task (appears below spinner, scrolls if max-lines set)
|
|
40
|
+
* Lines are dimmed and temporary - they disappear when task completes unless kept
|
|
41
|
+
* If no task is active, displays as a regular log message
|
|
42
|
+
*/
|
|
43
|
+
taskLine(message: string, keepOnComplete?: boolean): void;
|
|
44
|
+
/**
|
|
45
|
+
* Clear all task lines (useful when you want to reset the scrolling window)
|
|
46
|
+
*/
|
|
47
|
+
clearTaskLines(): void;
|
|
48
|
+
/**
|
|
49
|
+
* Complete task with optional duration display
|
|
50
|
+
* Defaults to showing the original task message with duration
|
|
51
|
+
* Task lines marked with keepOnComplete will be preserved, others are cleared
|
|
52
|
+
*/
|
|
53
|
+
done(message?: string, showTime?: boolean): void;
|
|
54
|
+
/**
|
|
55
|
+
* Show an object as a table of key-value pairs
|
|
56
|
+
* Filters out undefined values and applies optional transforms
|
|
57
|
+
* Automatically calculates key padding from max key length
|
|
58
|
+
*/
|
|
59
|
+
obj(obj: Record<string, any>, keyTransform?: (key: string) => string, valueTransform?: (value: any, key: string) => string): void;
|
|
60
|
+
/**
|
|
61
|
+
* Display array as bulleted list
|
|
62
|
+
* Each item shown with dim bullet and white text
|
|
63
|
+
*/
|
|
64
|
+
arr(items: any[]): void;
|
|
65
|
+
/**
|
|
66
|
+
* Show plain message with optional color
|
|
67
|
+
*/
|
|
68
|
+
log(message: string, color?: "red" | "green" | "yellow" | "blue" | "cyan" | "magenta" | "gray" | "dim"): void;
|
|
69
|
+
/**
|
|
70
|
+
* Show success message (green text)
|
|
71
|
+
*/
|
|
72
|
+
success(message: string): void;
|
|
73
|
+
/**
|
|
74
|
+
* Show success block (green background label + optional message)
|
|
75
|
+
*/
|
|
76
|
+
successBlock(label: string, message?: string): void;
|
|
77
|
+
/**
|
|
78
|
+
* Show success message (green text)
|
|
79
|
+
*/
|
|
80
|
+
spicy(message: string): void;
|
|
81
|
+
/**
|
|
82
|
+
* Show success block (green background label + optional message)
|
|
83
|
+
*/
|
|
84
|
+
spicyBlock(label: string, message?: string): void;
|
|
85
|
+
/**
|
|
86
|
+
* Show message with rainbow gradient
|
|
87
|
+
*/
|
|
88
|
+
rainbow(message: string): void;
|
|
89
|
+
/**
|
|
90
|
+
* Show info message (dim text)
|
|
91
|
+
*/
|
|
92
|
+
info(message: string): void;
|
|
93
|
+
/**
|
|
94
|
+
* Show info block (grey background label + optional message)
|
|
95
|
+
*/
|
|
96
|
+
infoBlock(label: string, message?: string): void;
|
|
97
|
+
/**
|
|
98
|
+
* Show error message (red text) - fails spinner if running
|
|
99
|
+
*/
|
|
100
|
+
error(message: string | Error | unknown): void;
|
|
101
|
+
/**
|
|
102
|
+
* Show error block (red background label + optional message) - fails spinner if running
|
|
103
|
+
*/
|
|
104
|
+
errorBlock(label: string, message?: string): void;
|
|
105
|
+
/**
|
|
106
|
+
* Show warning message (yellow text)
|
|
107
|
+
*/
|
|
108
|
+
warn(message: string): void;
|
|
109
|
+
/**
|
|
110
|
+
* Show warning block (yellow background label + optional message)
|
|
111
|
+
*/
|
|
112
|
+
warnBlock(label: string, message?: string): void;
|
|
113
|
+
/**
|
|
114
|
+
* Show detailed error information and exit the program
|
|
115
|
+
* Use this when an unexpected/unrecoverable error occurs
|
|
116
|
+
* Shows error message and stack trace, then exits
|
|
117
|
+
*/
|
|
118
|
+
crash(error: unknown, exitCode?: number): never;
|
|
119
|
+
/**
|
|
120
|
+
* Exit with code
|
|
121
|
+
*/
|
|
122
|
+
exit(code?: number): never;
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Global singleton output instance
|
|
126
|
+
* Import and use this anywhere in your code
|
|
127
|
+
*/
|
|
128
|
+
export declare const out: Output;
|
|
129
|
+
//# sourceMappingURL=output.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"output.d.ts","sourceRoot":"","sources":["../../src/utils/output.ts"],"names":[],"mappings":"AAGA;;;;;;;GAOG;AACH,qBAAa,MAAM;;IACjB,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAuB;IAC9C,OAAO,CAAC,OAAO,CAAoB;IACnC,OAAO,CAAC,aAAa,CAAuB;IAC5C,OAAO,CAAC,mBAAmB,CAAuB;IAClD,OAAO,CAAC,kBAAkB,CAAuB;IACjD,OAAO,CAAC,SAAS,CAAgB;IACjC,OAAO,CAAC,YAAY,CAAa;IAEjC,OAAO;IAEP;;OAEG;IACH,MAAM,CAAC,WAAW,IAAI,MAAM;IAO5B;;OAEG;IACH,MAAM,CAAC,KAAK,IAAI,IAAI;IAQpB;;;;;OAKG;IACH,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,GAAE,MAAU,GAAG,IAAI;IAcjD;;OAEG;IACH,MAAM,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IAO7B;;;;OAIG;IACH,QAAQ,CAAC,OAAO,EAAE,MAAM,EAAE,cAAc,GAAE,OAAe,GAAG,IAAI;IAkBhE;;OAEG;IACH,cAAc,IAAI,IAAI;IAiCtB;;;;OAIG;IACH,IAAI,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,QAAQ,GAAE,OAAc,GAAG,IAAI;IAuCtD;;;;OAIG;IACH,GAAG,CACD,GAAG,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EACxB,YAAY,CAAC,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,MAAM,EACtC,cAAc,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,MAAM,KAAK,MAAM,GACnD,IAAI;IA0BP;;;OAGG;IACH,GAAG,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,IAAI;IASvB;;OAEG;IACH,GAAG,CACD,OAAO,EAAE,MAAM,EACf,KAAK,CAAC,EACF,KAAK,GACL,OAAO,GACP,QAAQ,GACR,MAAM,GACN,MAAM,GACN,SAAS,GACT,MAAM,GACN,KAAK,GACR,IAAI;IAWP;;OAEG;IACH,OAAO,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IAK9B;;OAEG;IACH,YAAY,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,GAAE,MAAW,GAAG,IAAI;IAOvD;;OAEG;IACH,KAAK,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IAK5B;;OAEG;IACH,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,GAAE,MAAW,GAAG,IAAI;IAOrD;;OAEG;IACH,OAAO,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IA8B9B;;OAEG;IACH,IAAI,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IAK3B;;OAEG;IACH,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,GAAE,MAAW,GAAG,IAAI;IAOpD;;OAEG;IACH,KAAK,CAAC,OAAO,EAAE,MAAM,GAAG,KAAK,GAAG,OAAO,GAAG,IAAI;IAmB9C;;OAEG;IACH,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,GAAE,MAAW,GAAG,IAAI;IAarD;;OAEG;IACH,IAAI,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI;IAK3B;;OAEG;IACH,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,GAAE,MAAW,GAAG,IAAI;IAOpD;;;;OAIG;IACH,KAAK,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,GAAE,MAAU,GAAG,KAAK;IAuBlD;;OAEG;IACH,IAAI,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,KAAK;CAoB3B;AAED;;;GAGG;AACH,eAAO,MAAM,GAAG,QAAuB,CAAC"}
|