@remnic/plugin-openclaw 1.0.6 → 1.0.8
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 +36 -0
- package/dist/{calibration-3JHF25QT.js → calibration-BAC7KNKR.js} +2 -1
- package/dist/{causal-consolidation-EBLROS42.js → causal-consolidation-33R5JTPX.js} +7 -5
- package/dist/chunk-3A5ELHTT.js +61 -0
- package/dist/chunk-5ZW5XJQ6.js +125 -0
- package/dist/chunk-6OJAU466.js +148 -0
- package/dist/{chunk-QHMR3D7U.js → chunk-BQLPVRIU.js} +109 -2
- package/dist/chunk-DIZW6H5J.js +136 -0
- package/dist/{chunk-KPMXWORS.js → chunk-JJSNPSCD.js} +608 -354
- package/dist/{chunk-3SA5F4WT.js → chunk-NXLHSCLU.js} +125 -69
- package/dist/{chunk-GUKYM4XZ.js → chunk-PFH73PN6.js} +3 -3
- package/dist/consolidation-undo-5ZSX4MWO.js +426 -0
- package/dist/contradiction-review-SVGBS3V5.js +21 -0
- package/dist/contradiction-scan-U3QKHWQN.js +412 -0
- package/dist/{engine-BU6GNUJ5.js → engine-M5G6ZJU7.js} +3 -2
- package/dist/extraction-judge-telemetry-GHOTVYMP.js +14 -0
- package/dist/{fallback-llm-HJRCHKSA.js → fallback-llm-QEAPMDW7.js} +2 -1
- package/dist/index.js +6467 -1285
- package/dist/resolution-YITUVUTH.js +100 -0
- package/dist/{storage-BA6OBLMK.js → storage-DM4ZGOCN.js} +2 -1
- package/openclaw.plugin.json +280 -9
- package/package.json +2 -2
|
@@ -0,0 +1,426 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getVersion
|
|
3
|
+
} from "./chunk-6OJAU466.js";
|
|
4
|
+
import "./chunk-MLKGABMK.js";
|
|
5
|
+
|
|
6
|
+
// ../remnic-core/src/consolidation-undo.ts
|
|
7
|
+
import path from "path";
|
|
8
|
+
import { mkdir, writeFile, access, realpath, lstat } from "fs/promises";
|
|
9
|
+
import { constants as fsConstants } from "fs";
|
|
10
|
+
var DERIVED_FROM_ENTRY_RE = /^(.+):(\d+)$/;
|
|
11
|
+
function parseEntry(entry) {
|
|
12
|
+
if (typeof entry !== "string") return null;
|
|
13
|
+
const match = entry.match(DERIVED_FROM_ENTRY_RE);
|
|
14
|
+
if (!match) return null;
|
|
15
|
+
return { pagePath: match[1], versionId: match[2] };
|
|
16
|
+
}
|
|
17
|
+
function isInsideDirectory(candidate, root) {
|
|
18
|
+
const normRoot = path.resolve(root);
|
|
19
|
+
const normCandidate = path.resolve(candidate);
|
|
20
|
+
const rel = path.relative(normRoot, normCandidate);
|
|
21
|
+
if (rel.length === 0) return true;
|
|
22
|
+
if (rel.startsWith("..")) return false;
|
|
23
|
+
if (path.isAbsolute(rel)) return false;
|
|
24
|
+
return true;
|
|
25
|
+
}
|
|
26
|
+
async function isInsideDirectoryRealpath(candidate, root) {
|
|
27
|
+
if (!isInsideDirectory(candidate, root)) return false;
|
|
28
|
+
const rawSegments = candidate.replace(/\\/g, "/").split("/");
|
|
29
|
+
if (rawSegments.some((s) => s === "..")) return false;
|
|
30
|
+
let resolvedRoot;
|
|
31
|
+
try {
|
|
32
|
+
resolvedRoot = await realpath(path.resolve(root));
|
|
33
|
+
} catch {
|
|
34
|
+
return false;
|
|
35
|
+
}
|
|
36
|
+
const normCandidate = path.resolve(candidate);
|
|
37
|
+
const normRoot = path.resolve(root);
|
|
38
|
+
const relFromRoot = path.relative(normRoot, normCandidate);
|
|
39
|
+
const segments = relFromRoot.length > 0 ? relFromRoot.split(path.sep) : [];
|
|
40
|
+
for (let i = 0; i <= segments.length; i++) {
|
|
41
|
+
const probe = i === 0 ? normRoot : path.join(normRoot, ...segments.slice(0, i));
|
|
42
|
+
try {
|
|
43
|
+
const st = await lstat(probe);
|
|
44
|
+
if (st.isSymbolicLink() && probe !== normRoot) {
|
|
45
|
+
let target;
|
|
46
|
+
try {
|
|
47
|
+
target = await realpath(probe);
|
|
48
|
+
} catch {
|
|
49
|
+
return false;
|
|
50
|
+
}
|
|
51
|
+
const rel = path.relative(resolvedRoot, target);
|
|
52
|
+
if (rel.length === 0) continue;
|
|
53
|
+
if (rel.startsWith("..") || path.isAbsolute(rel)) return false;
|
|
54
|
+
}
|
|
55
|
+
} catch {
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
const parts = normCandidate.split(path.sep);
|
|
59
|
+
for (let i = parts.length; i > 0; i--) {
|
|
60
|
+
const probe = parts.slice(0, i).join(path.sep) || path.sep;
|
|
61
|
+
try {
|
|
62
|
+
const resolved = await realpath(probe);
|
|
63
|
+
const trailing = parts.slice(i).join(path.sep);
|
|
64
|
+
const final = trailing.length > 0 ? path.join(resolved, trailing) : resolved;
|
|
65
|
+
const rel = path.relative(resolvedRoot, final);
|
|
66
|
+
if (rel.length === 0) return true;
|
|
67
|
+
if (rel.startsWith("..")) return false;
|
|
68
|
+
if (path.isAbsolute(rel)) return false;
|
|
69
|
+
return true;
|
|
70
|
+
} catch {
|
|
71
|
+
continue;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
return false;
|
|
75
|
+
}
|
|
76
|
+
var NON_ACTIVE_PREFIXES = ["archive/", "state/"];
|
|
77
|
+
function normalizeRelativePath(p) {
|
|
78
|
+
const parts = p.replace(/\\/g, "/").split("/");
|
|
79
|
+
const resolved = [];
|
|
80
|
+
for (const seg of parts) {
|
|
81
|
+
if (seg === "" || seg === ".") continue;
|
|
82
|
+
if (seg === "..") {
|
|
83
|
+
if (resolved.length > 0) resolved.pop();
|
|
84
|
+
} else {
|
|
85
|
+
resolved.push(seg);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
return resolved.join("/");
|
|
89
|
+
}
|
|
90
|
+
function isActiveMemoryRelativePath(pagePath, sidecarDir) {
|
|
91
|
+
const normalized = normalizeRelativePath(pagePath);
|
|
92
|
+
const prefixes = [...NON_ACTIVE_PREFIXES];
|
|
93
|
+
if (sidecarDir) {
|
|
94
|
+
const normSidecar = normalizeRelativePath(sidecarDir);
|
|
95
|
+
prefixes.push(normSidecar + "/");
|
|
96
|
+
}
|
|
97
|
+
for (const prefix of prefixes) {
|
|
98
|
+
if (normalized === prefix.slice(0, -1) || normalized.startsWith(prefix)) {
|
|
99
|
+
return false;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return true;
|
|
103
|
+
}
|
|
104
|
+
async function isRegularFile(p) {
|
|
105
|
+
try {
|
|
106
|
+
const st = await lstat(p);
|
|
107
|
+
return st.isFile();
|
|
108
|
+
} catch {
|
|
109
|
+
return false;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
async function fileExists(p) {
|
|
113
|
+
try {
|
|
114
|
+
await access(p, fsConstants.F_OK);
|
|
115
|
+
return true;
|
|
116
|
+
} catch {
|
|
117
|
+
return false;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
async function runConsolidationUndo(options) {
|
|
121
|
+
const { storage, memoryDir, targetPath, versioning } = options;
|
|
122
|
+
const dryRun = options.dryRun === true;
|
|
123
|
+
const result = {
|
|
124
|
+
targetPath,
|
|
125
|
+
targetArchived: false,
|
|
126
|
+
restores: [],
|
|
127
|
+
dryRun
|
|
128
|
+
};
|
|
129
|
+
if (!await isInsideDirectoryRealpath(targetPath, memoryDir)) {
|
|
130
|
+
result.error = `target path ${targetPath} is outside memory directory ${memoryDir}`;
|
|
131
|
+
return result;
|
|
132
|
+
}
|
|
133
|
+
const targetRel = path.relative(memoryDir, targetPath);
|
|
134
|
+
if (!isActiveMemoryRelativePath(targetRel, versioning.sidecarDir)) {
|
|
135
|
+
result.error = `target path "${targetRel}" is inside a non-active directory \u2014 refusing to operate`;
|
|
136
|
+
return result;
|
|
137
|
+
}
|
|
138
|
+
const target = await storage.readMemoryByPath(targetPath);
|
|
139
|
+
if (!target) {
|
|
140
|
+
result.error = `could not load target memory at ${targetPath}`;
|
|
141
|
+
return result;
|
|
142
|
+
}
|
|
143
|
+
const derivedFrom = target.frontmatter.derived_from;
|
|
144
|
+
if (!Array.isArray(derivedFrom) || derivedFrom.length === 0) {
|
|
145
|
+
result.error = "target memory has no derived_from entries \u2014 nothing to undo";
|
|
146
|
+
return result;
|
|
147
|
+
}
|
|
148
|
+
const plans = [];
|
|
149
|
+
for (const rawEntry of derivedFrom) {
|
|
150
|
+
const entry = typeof rawEntry === "string" ? rawEntry : String(rawEntry);
|
|
151
|
+
const parsed = parseEntry(rawEntry);
|
|
152
|
+
if (!parsed) {
|
|
153
|
+
plans.push({
|
|
154
|
+
kind: "skip",
|
|
155
|
+
restore: {
|
|
156
|
+
entry,
|
|
157
|
+
sourcePath: "",
|
|
158
|
+
outcome: "skipped_malformed_entry",
|
|
159
|
+
detail: `expected "<path>:<version>" shape`
|
|
160
|
+
}
|
|
161
|
+
});
|
|
162
|
+
continue;
|
|
163
|
+
}
|
|
164
|
+
if (path.isAbsolute(parsed.pagePath)) {
|
|
165
|
+
plans.push({
|
|
166
|
+
kind: "skip",
|
|
167
|
+
restore: {
|
|
168
|
+
entry,
|
|
169
|
+
sourcePath: parsed.pagePath,
|
|
170
|
+
outcome: "skipped_malformed_entry",
|
|
171
|
+
detail: `derived_from path must be relative, got absolute: "${parsed.pagePath}"`
|
|
172
|
+
}
|
|
173
|
+
});
|
|
174
|
+
continue;
|
|
175
|
+
}
|
|
176
|
+
const sourcePath = path.join(memoryDir, parsed.pagePath);
|
|
177
|
+
if (!await isInsideDirectoryRealpath(sourcePath, memoryDir)) {
|
|
178
|
+
plans.push({
|
|
179
|
+
kind: "skip",
|
|
180
|
+
restore: {
|
|
181
|
+
entry,
|
|
182
|
+
sourcePath,
|
|
183
|
+
outcome: "skipped_outside_memory_dir",
|
|
184
|
+
detail: `resolved path escapes memory directory ${memoryDir}`
|
|
185
|
+
}
|
|
186
|
+
});
|
|
187
|
+
continue;
|
|
188
|
+
}
|
|
189
|
+
let resolvedRelative = parsed.pagePath;
|
|
190
|
+
try {
|
|
191
|
+
const realBase = await realpath(memoryDir);
|
|
192
|
+
try {
|
|
193
|
+
const realSource = await realpath(sourcePath);
|
|
194
|
+
const rel = path.relative(realBase, realSource);
|
|
195
|
+
if (!rel.startsWith("..") && !path.isAbsolute(rel)) {
|
|
196
|
+
resolvedRelative = rel.replace(/\\/g, "/");
|
|
197
|
+
}
|
|
198
|
+
} catch {
|
|
199
|
+
const parentDir = path.dirname(sourcePath);
|
|
200
|
+
try {
|
|
201
|
+
const realParent = await realpath(parentDir);
|
|
202
|
+
const parentRel = path.relative(realBase, realParent);
|
|
203
|
+
if (!parentRel.startsWith("..") && !path.isAbsolute(parentRel)) {
|
|
204
|
+
const leafName = path.basename(sourcePath);
|
|
205
|
+
resolvedRelative = path.join(parentRel, leafName).replace(/\\/g, "/");
|
|
206
|
+
}
|
|
207
|
+
} catch {
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
} catch {
|
|
211
|
+
}
|
|
212
|
+
if (!isActiveMemoryRelativePath(parsed.pagePath, versioning.sidecarDir) || !isActiveMemoryRelativePath(resolvedRelative, versioning.sidecarDir)) {
|
|
213
|
+
plans.push({
|
|
214
|
+
kind: "skip",
|
|
215
|
+
restore: {
|
|
216
|
+
entry,
|
|
217
|
+
sourcePath,
|
|
218
|
+
outcome: "skipped_non_active_path",
|
|
219
|
+
detail: `source path "${parsed.pagePath}" is inside a non-active directory (archive/state/versions)`
|
|
220
|
+
}
|
|
221
|
+
});
|
|
222
|
+
continue;
|
|
223
|
+
}
|
|
224
|
+
if (path.resolve(sourcePath) === path.resolve(targetPath)) {
|
|
225
|
+
plans.push({
|
|
226
|
+
kind: "skip",
|
|
227
|
+
restore: {
|
|
228
|
+
entry,
|
|
229
|
+
sourcePath,
|
|
230
|
+
outcome: "skipped_self_referential",
|
|
231
|
+
detail: `derived_from entry "${entry}" resolves to the same file as the target \u2014 refusing to count as recovered`
|
|
232
|
+
}
|
|
233
|
+
});
|
|
234
|
+
continue;
|
|
235
|
+
}
|
|
236
|
+
if (await isRegularFile(sourcePath)) {
|
|
237
|
+
plans.push({ kind: "recovered_existing", entry, sourcePath });
|
|
238
|
+
continue;
|
|
239
|
+
}
|
|
240
|
+
if (await fileExists(sourcePath)) {
|
|
241
|
+
plans.push({
|
|
242
|
+
kind: "skip",
|
|
243
|
+
restore: {
|
|
244
|
+
entry,
|
|
245
|
+
sourcePath,
|
|
246
|
+
outcome: "skipped_non_regular_file",
|
|
247
|
+
detail: "source path is occupied by a non-regular-file; refusing to proceed"
|
|
248
|
+
}
|
|
249
|
+
});
|
|
250
|
+
continue;
|
|
251
|
+
}
|
|
252
|
+
let snapshotContent;
|
|
253
|
+
try {
|
|
254
|
+
snapshotContent = await getVersion(
|
|
255
|
+
sourcePath,
|
|
256
|
+
parsed.versionId,
|
|
257
|
+
versioning,
|
|
258
|
+
memoryDir
|
|
259
|
+
);
|
|
260
|
+
} catch {
|
|
261
|
+
plans.push({
|
|
262
|
+
kind: "skip",
|
|
263
|
+
restore: {
|
|
264
|
+
entry,
|
|
265
|
+
sourcePath,
|
|
266
|
+
outcome: "skipped_snapshot_missing",
|
|
267
|
+
detail: `no snapshot for version ${parsed.versionId}`
|
|
268
|
+
}
|
|
269
|
+
});
|
|
270
|
+
continue;
|
|
271
|
+
}
|
|
272
|
+
plans.push({ kind: "write", entry, sourcePath, content: snapshotContent });
|
|
273
|
+
}
|
|
274
|
+
const skipped = plans.filter((p) => p.kind === "skip");
|
|
275
|
+
if (skipped.length > 0) {
|
|
276
|
+
for (const p of plans) {
|
|
277
|
+
if (p.kind === "skip") {
|
|
278
|
+
result.restores.push(p.restore);
|
|
279
|
+
} else if (p.kind === "write") {
|
|
280
|
+
result.restores.push({
|
|
281
|
+
entry: p.entry,
|
|
282
|
+
sourcePath: p.sourcePath,
|
|
283
|
+
outcome: dryRun ? "skipped_dry_run" : "skipped_blocked_by_other_failures",
|
|
284
|
+
detail: dryRun ? "would restore from snapshot (blocked by other failures)" : "snapshot available but undo aborted due to other failures"
|
|
285
|
+
});
|
|
286
|
+
} else {
|
|
287
|
+
result.restores.push({
|
|
288
|
+
entry: p.entry,
|
|
289
|
+
sourcePath: p.sourcePath,
|
|
290
|
+
outcome: "skipped_file_exists",
|
|
291
|
+
detail: "source file already exists; no restore needed"
|
|
292
|
+
});
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
const recovered = result.restores.filter(
|
|
296
|
+
(r) => r.outcome === "restored" || r.outcome === "skipped_file_exists"
|
|
297
|
+
).length;
|
|
298
|
+
if (recovered === 0) {
|
|
299
|
+
result.error = "no sources could be recovered (all snapshots missing or paths unsafe); target not archived to preserve data";
|
|
300
|
+
} else {
|
|
301
|
+
result.error = `${skipped.length} of ${plans.length} sources could not be recovered; target not archived (undo is all-or-nothing)`;
|
|
302
|
+
}
|
|
303
|
+
return result;
|
|
304
|
+
}
|
|
305
|
+
const seenSourcePaths = /* @__PURE__ */ new Set();
|
|
306
|
+
const dedupedPlans = [];
|
|
307
|
+
for (const p of plans) {
|
|
308
|
+
if (p.kind === "write" || p.kind === "recovered_existing") {
|
|
309
|
+
if (seenSourcePaths.has(p.sourcePath)) {
|
|
310
|
+
dedupedPlans.push({
|
|
311
|
+
kind: "skip",
|
|
312
|
+
restore: {
|
|
313
|
+
entry: p.kind === "write" ? p.entry : p.entry,
|
|
314
|
+
sourcePath: p.sourcePath,
|
|
315
|
+
outcome: "skipped_file_exists",
|
|
316
|
+
detail: "duplicate derived_from entry \u2014 source already processed"
|
|
317
|
+
}
|
|
318
|
+
});
|
|
319
|
+
continue;
|
|
320
|
+
}
|
|
321
|
+
seenSourcePaths.add(p.sourcePath);
|
|
322
|
+
}
|
|
323
|
+
dedupedPlans.push(p);
|
|
324
|
+
}
|
|
325
|
+
if (dryRun) {
|
|
326
|
+
for (const p of dedupedPlans) {
|
|
327
|
+
if (p.kind === "write") {
|
|
328
|
+
result.restores.push({
|
|
329
|
+
entry: p.entry,
|
|
330
|
+
sourcePath: p.sourcePath,
|
|
331
|
+
outcome: "skipped_dry_run",
|
|
332
|
+
detail: "would restore from snapshot"
|
|
333
|
+
});
|
|
334
|
+
} else if (p.kind === "recovered_existing") {
|
|
335
|
+
result.restores.push({
|
|
336
|
+
entry: p.entry,
|
|
337
|
+
sourcePath: p.sourcePath,
|
|
338
|
+
outcome: "skipped_file_exists",
|
|
339
|
+
detail: "source file already exists; no restore needed"
|
|
340
|
+
});
|
|
341
|
+
} else if (p.kind === "skip" && p.restore) {
|
|
342
|
+
result.restores.push(p.restore);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
return result;
|
|
346
|
+
}
|
|
347
|
+
let writeFailed = false;
|
|
348
|
+
for (const p of dedupedPlans) {
|
|
349
|
+
if (p.kind === "skip") {
|
|
350
|
+
if (p.restore) result.restores.push(p.restore);
|
|
351
|
+
continue;
|
|
352
|
+
}
|
|
353
|
+
if (p.kind === "recovered_existing") {
|
|
354
|
+
result.restores.push({
|
|
355
|
+
entry: p.entry,
|
|
356
|
+
sourcePath: p.sourcePath,
|
|
357
|
+
outcome: "skipped_file_exists",
|
|
358
|
+
detail: "source file already exists; no restore needed"
|
|
359
|
+
});
|
|
360
|
+
continue;
|
|
361
|
+
}
|
|
362
|
+
if (p.kind === "write") {
|
|
363
|
+
if (writeFailed) {
|
|
364
|
+
result.restores.push({
|
|
365
|
+
entry: p.entry,
|
|
366
|
+
sourcePath: p.sourcePath,
|
|
367
|
+
outcome: "skipped_blocked_by_other_failures",
|
|
368
|
+
detail: "a prior source write failed; skipping remaining writes to honor all-or-nothing contract"
|
|
369
|
+
});
|
|
370
|
+
continue;
|
|
371
|
+
}
|
|
372
|
+
try {
|
|
373
|
+
await mkdir(path.dirname(p.sourcePath), { recursive: true });
|
|
374
|
+
await writeFile(p.sourcePath, p.content, { encoding: "utf-8", flag: "wx" });
|
|
375
|
+
result.restores.push({
|
|
376
|
+
entry: p.entry,
|
|
377
|
+
sourcePath: p.sourcePath,
|
|
378
|
+
outcome: "restored"
|
|
379
|
+
});
|
|
380
|
+
} catch (err) {
|
|
381
|
+
writeFailed = true;
|
|
382
|
+
result.restores.push({
|
|
383
|
+
entry: p.entry,
|
|
384
|
+
sourcePath: p.sourcePath,
|
|
385
|
+
outcome: "skipped_write_failed",
|
|
386
|
+
detail: `write failed: ${err instanceof Error ? err.message : String(err)}`
|
|
387
|
+
});
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
}
|
|
391
|
+
if (writeFailed) {
|
|
392
|
+
result.error = "one or more source writes failed mid-restore; target not archived to preserve data";
|
|
393
|
+
return result;
|
|
394
|
+
}
|
|
395
|
+
const archivedAt = await storage.archiveMemory(target, {
|
|
396
|
+
actor: "consolidate-undo",
|
|
397
|
+
reasonCode: "consolidation-undo"
|
|
398
|
+
});
|
|
399
|
+
result.targetArchived = archivedAt !== null;
|
|
400
|
+
if (!result.targetArchived) {
|
|
401
|
+
result.error = "sources restored successfully but archiving the consolidated target failed; inspect storage for manual cleanup";
|
|
402
|
+
}
|
|
403
|
+
return result;
|
|
404
|
+
}
|
|
405
|
+
function formatConsolidationUndoResult(result) {
|
|
406
|
+
const lines = [];
|
|
407
|
+
lines.push(`consolidate undo ${result.dryRun ? "(dry run) " : ""}\u2192 ${result.targetPath}`);
|
|
408
|
+
for (const r of result.restores) {
|
|
409
|
+
lines.push(` - ${r.entry} \u2192 ${r.outcome}${r.detail ? ` (${r.detail})` : ""}`);
|
|
410
|
+
}
|
|
411
|
+
if (result.error) {
|
|
412
|
+
lines.push(` ERROR: ${result.error}`);
|
|
413
|
+
return lines.join("\n");
|
|
414
|
+
}
|
|
415
|
+
lines.push(
|
|
416
|
+
result.dryRun ? " (dry run \u2014 no files were modified, target not archived)" : ` target archived: ${result.targetArchived ? "yes" : "no"}`
|
|
417
|
+
);
|
|
418
|
+
return lines.join("\n");
|
|
419
|
+
}
|
|
420
|
+
export {
|
|
421
|
+
formatConsolidationUndoResult,
|
|
422
|
+
isActiveMemoryRelativePath,
|
|
423
|
+
isInsideDirectory,
|
|
424
|
+
isInsideDirectoryRealpath,
|
|
425
|
+
runConsolidationUndo
|
|
426
|
+
};
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import {
|
|
2
|
+
computePairId,
|
|
3
|
+
isCoolingDown,
|
|
4
|
+
listPairs,
|
|
5
|
+
memoryHashesChanged,
|
|
6
|
+
readPair,
|
|
7
|
+
resolvePair,
|
|
8
|
+
writePair,
|
|
9
|
+
writePairs
|
|
10
|
+
} from "./chunk-DIZW6H5J.js";
|
|
11
|
+
import "./chunk-MLKGABMK.js";
|
|
12
|
+
export {
|
|
13
|
+
computePairId,
|
|
14
|
+
isCoolingDown,
|
|
15
|
+
listPairs,
|
|
16
|
+
memoryHashesChanged,
|
|
17
|
+
readPair,
|
|
18
|
+
resolvePair,
|
|
19
|
+
writePair,
|
|
20
|
+
writePairs
|
|
21
|
+
};
|