@remnic/plugin-openclaw 1.0.35 → 1.0.37

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.
Files changed (87) hide show
  1. package/README.md +38 -4
  2. package/dist/{calibration-Z5WWNV7U.js → calibration-RKL2LRW4.js} +4 -4
  3. package/dist/{capsule-cli-GBM3WPAM.js → capsule-cli-EHZPMXBC.js} +2 -2
  4. package/dist/{capsule-crypto-K3IRTKRH.js → capsule-crypto-JS67OSWM.js} +3 -3
  5. package/dist/capsule-export-DX53CPIT.js +17 -0
  6. package/dist/capsule-import-4OXCPHOT.js +16 -0
  7. package/dist/{capsule-merge-IWOQ34KL.js → capsule-merge-25AUN33Q.js} +7 -7
  8. package/dist/{causal-chain-WYN5QOPS.js → causal-chain-BVTOWZKC.js} +4 -4
  9. package/dist/{causal-consolidation-C64NNE4T.js → causal-consolidation-DRPM2KOE.js} +13 -10
  10. package/dist/{causal-retrieval-NZHQOZOE.js → causal-retrieval-XAP6QKHZ.js} +4 -5
  11. package/dist/{causal-trajectory-graph-VBPE2WPM.js → causal-trajectory-graph-ZWQWZ7N5.js} +2 -2
  12. package/dist/{chunk-5LE4HTVL.js → chunk-25J4PXDH.js} +0 -18
  13. package/dist/{chunk-6UFI73TJ.js → chunk-3IKMUNW5.js} +53 -46
  14. package/dist/{chunk-EXDYWXMB.js → chunk-4XDQ3KEC.js} +1 -2
  15. package/dist/{chunk-JGIUTWZS.js → chunk-6O3H3DPL.js} +2 -2
  16. package/dist/{chunk-UTDLHBBV.js → chunk-BLC3RQNV.js} +5 -555
  17. package/dist/{chunk-4G2XCSD2.js → chunk-BZ4EYURA.js} +0 -5
  18. package/dist/{chunk-L6I4MQKO.js → chunk-CEL5ZLKP.js} +6 -6
  19. package/dist/{chunk-TDRJVMUP.js → chunk-EH4AXGRO.js} +0 -12
  20. package/dist/{chunk-EYCLXMIV.js → chunk-G3CZA4SD.js} +9 -427
  21. package/dist/chunk-I2KLQ2HA.js +22 -0
  22. package/dist/chunk-IO5WWY6A.js +156 -0
  23. package/dist/{contradiction-scan-A5NOTZPN.js → chunk-JC3FCKYL.js} +189 -86
  24. package/dist/{chunk-SVSQAG6M.js → chunk-KC7KSQR4.js} +47 -28
  25. package/dist/chunk-LZCGPRHS.js +228 -0
  26. package/dist/{chunk-CXM7EBAO.js → chunk-MXFJXUHC.js} +1 -1
  27. package/dist/{chunk-VRGUUHBV.js → chunk-NUWDSTP7.js} +1 -1
  28. package/dist/{chunk-4LYQ4ONL.js → chunk-QCCP4RU5.js} +8 -3
  29. package/dist/{chunk-6OJAU466.js → chunk-QMUQV5NP.js} +0 -1
  30. package/dist/{chunk-LLUROTZJ.js → chunk-QQXJODFL.js} +9 -9
  31. package/dist/{chunk-6F6EKSVP.js → chunk-QXXEF7VI.js} +1 -1
  32. package/dist/{chunk-CMKR6NDQ.js → chunk-SEGEX7W4.js} +73 -241
  33. package/dist/{chunk-VFULKFKI.js → chunk-SWOYEQN2.js} +42 -17
  34. package/dist/chunk-TH5FF5SC.js +16 -0
  35. package/dist/{chunk-FGTYFLL5.js → chunk-TXOEHSVP.js} +29 -32
  36. package/dist/chunk-UZJ7EERS.js +272 -0
  37. package/dist/chunk-YJYZMLD5.js +360 -0
  38. package/dist/{chunk-NKVIN6RD.js → chunk-YKV4EFUI.js} +84 -2
  39. package/dist/{chunk-SSFTU6LP.js → chunk-ZS6VABML.js} +4 -4
  40. package/dist/{cipher-VHAFCG7Z.js → cipher-E23BHBSO.js} +1 -1
  41. package/dist/{consolidation-undo-5ZSX4MWO.js → consolidation-undo-FKJZCJHS.js} +2 -2
  42. package/dist/contradiction-review-WJRWNQ5N.js +29 -0
  43. package/dist/contradiction-scan-5X423QGT.js +12 -0
  44. package/dist/{dreams-ledger-3I52ISYR.js → dreams-ledger-KDX44I7R.js} +1 -1
  45. package/dist/{engine-47AKKYJ4.js → engine-5P774HTZ.js} +6 -6
  46. package/dist/{extraction-judge-telemetry-GHOTVYMP.js → extraction-judge-telemetry-O4ZVGLTU.js} +1 -1
  47. package/dist/{fallback-llm-45A755XP.js → fallback-llm-43UMEXNJ.js} +3 -3
  48. package/dist/{first-start-migration-I24M2JEE.js → first-start-migration-H2SAXAGR.js} +4 -4
  49. package/dist/{forget-NI4RBDPB.js → forget-ZECIDNL5.js} +1 -1
  50. package/dist/{fs-utils-PZRI2HDZ.js → fs-utils-OYXSZSVV.js} +12 -2
  51. package/dist/{graph-edge-decay-5CVKWBYH.js → graph-edge-decay-24ZKD5QL.js} +5 -5
  52. package/dist/index.js +7098 -84293
  53. package/dist/{kdf-H5B23ZM2.js → kdf-RXKIWHRU.js} +1 -1
  54. package/dist/legacy-hook-compat-QHHKF4GK.js +2 -0
  55. package/dist/{logger-TNOKCH7X.js → logger-XG7JKLPS.js} +1 -1
  56. package/dist/{memory-governance-QS7Z425Y.js → memory-governance-6K4M4YXD.js} +5 -5
  57. package/dist/{metadata-JAGIWHEA.js → metadata-WK2TRPYZ.js} +1 -1
  58. package/dist/{migrate-from-identity-anchor-7MMSPEUM.js → migrate-from-identity-anchor-SNDNKHZD.js} +1 -1
  59. package/dist/path-ZKO74XXC.js +7 -0
  60. package/dist/{peers-KRFXWRQ6.js → peers-W53WSDXG.js} +1 -1
  61. package/dist/{purge-XN2VSPZ2.js → purge-IKJISXEQ.js} +1 -1
  62. package/dist/resolution-BN35OXDS.js +11 -0
  63. package/dist/{secure-store-A4NGCNXV.js → secure-store-F75I54O5.js} +3 -3
  64. package/dist/{state-PVISYXRH.js → state-4ITLYMAU.js} +1 -1
  65. package/dist/{state-store-N6TFBFSP.js → state-store-ET3ADVY5.js} +3 -3
  66. package/dist/{storage-DDYQGLXA.js → storage-5EY6T7ON.js} +3 -3
  67. package/dist/{tier-stats-IZNW66NC.js → tier-stats-ZRQBV6G2.js} +4 -4
  68. package/dist/{trace-NJESSGH7.js → trace-IL2Y34EH.js} +1 -1
  69. package/dist/{tui-MGK2LYJY.js → tui-7KRDCMYK.js} +1 -1
  70. package/dist/{types-R4DO7AKM.js → types-MBUINTB2.js} +3 -3
  71. package/openclaw.plugin.json +164 -8
  72. package/package.json +9 -6
  73. package/scripts/faiss_index.py +816 -0
  74. package/scripts/faiss_requirements.txt +3 -0
  75. package/dist/capsule-export-IXVERCQG.js +0 -17
  76. package/dist/capsule-import-IA6VIOPQ.js +0 -16
  77. package/dist/chunk-3GUF7RQI.js +0 -559
  78. package/dist/chunk-7OQEPGQF.js +0 -533
  79. package/dist/chunk-DIZW6H5J.js +0 -136
  80. package/dist/chunk-FQRSVYY4.js +0 -110
  81. package/dist/chunk-GUSMRW4H.js +0 -12
  82. package/dist/chunk-MLKGABMK.js +0 -9
  83. package/dist/chunk-WPINX4MF.js +0 -380
  84. package/dist/contradiction-review-SVGBS3V5.js +0 -21
  85. package/dist/legacy-hook-compat-XQ7FP6FV.js +0 -35
  86. package/dist/path-JIEGNWFL.js +0 -7
  87. package/dist/resolution-YITUVUTH.js +0 -100
@@ -0,0 +1,360 @@
1
+ // ../remnic-core/src/contradiction/contradiction-review.ts
2
+ import fs from "fs";
3
+ import path from "path";
4
+ import { createHash, randomUUID } from "crypto";
5
+ var VALID_RESOLUTION_VERBS = [
6
+ "keep-a",
7
+ "keep-b",
8
+ "merge",
9
+ "both-valid",
10
+ "needs-more-context"
11
+ ];
12
+ var NEEDS_MORE_CONTEXT_COOLDOWN_MS = 24 * 60 * 60 * 1e3;
13
+ var UNSCOPED_MIGRATION_MARKER_PREFIX = ".unscoped-migrated-";
14
+ var UNSCOPED_MIGRATION_MARKER_SUFFIX = ".done";
15
+ function computePairId(memoryIdA, memoryIdB, namespace) {
16
+ const sorted = [memoryIdA, memoryIdB].sort();
17
+ const normalizedNamespace = namespace?.trim();
18
+ const scope = normalizedNamespace ? `ns:${normalizedNamespace}::` : "";
19
+ return createHash("sha256").update(`${scope}${sorted.join("::")}`).digest("hex").slice(0, 24);
20
+ }
21
+ function isDefaultReviewNamespace(defaultNamespace, requestedNamespace, resolvedNamespace) {
22
+ const requested = requestedNamespace?.trim();
23
+ return !requested || requested === defaultNamespace || resolvedNamespace === defaultNamespace;
24
+ }
25
+ function isTerminalResolution(resolution) {
26
+ return resolution === "keep-a" || resolution === "keep-b" || resolution === "merge";
27
+ }
28
+ function preservesDirectResolution(resolution) {
29
+ return isTerminalResolution(resolution) || resolution === "both-valid";
30
+ }
31
+ function isDormantReviewedPair(pair) {
32
+ return pair.verdict === "independent" || pair.resolution === "both-valid";
33
+ }
34
+ function reviewStateRank(pair, cooldownDays) {
35
+ if (isTerminalResolution(pair.resolution)) return 5;
36
+ if (pair.resolution === "both-valid") return 4;
37
+ if (isDeferralActive(pair)) return 3;
38
+ if (isDormantReviewedPair(pair)) {
39
+ if (cooldownDays === void 0) return 2;
40
+ return isCoolingDown(pair, cooldownDays) ? 2 : 0;
41
+ }
42
+ return 1;
43
+ }
44
+ function parseIsoMillis(value) {
45
+ if (!value) return null;
46
+ const millis = new Date(value).getTime();
47
+ return Number.isFinite(millis) ? millis : null;
48
+ }
49
+ function isDeferred(pair) {
50
+ return pair.resolution === "needs-more-context" || Boolean(pair.deferredUntil);
51
+ }
52
+ function deferralUntilMillis(pair) {
53
+ const deferredUntil = parseIsoMillis(pair.deferredUntil);
54
+ if (deferredUntil !== null) return deferredUntil;
55
+ if (pair.resolution === "needs-more-context") {
56
+ const lastReviewed = parseIsoMillis(pair.lastReviewedAt);
57
+ return lastReviewed === null ? null : lastReviewed + NEEDS_MORE_CONTEXT_COOLDOWN_MS;
58
+ }
59
+ return null;
60
+ }
61
+ function isDeferralActive(pair) {
62
+ const deferredUntil = deferralUntilMillis(pair);
63
+ return deferredUntil !== null && Date.now() < deferredUntil;
64
+ }
65
+ function reviewStateMillis(pair) {
66
+ return Math.max(
67
+ parseIsoMillis(pair.deferredUntil) ?? Number.NEGATIVE_INFINITY,
68
+ parseIsoMillis(pair.lastReviewedAt) ?? Number.NEGATIVE_INFINITY,
69
+ parseIsoMillis(pair.detectedAt) ?? Number.NEGATIVE_INFINITY
70
+ );
71
+ }
72
+ function mergeMigratedPair(existing, migrated, options) {
73
+ const existingRank = reviewStateRank(existing, options.cooldownDays);
74
+ const migratedRank = reviewStateRank(migrated, options.cooldownDays);
75
+ const selected = migratedRank > existingRank ? migrated : migratedRank < existingRank ? existing : reviewStateMillis(migrated) > reviewStateMillis(existing) ? migrated : existing;
76
+ return {
77
+ ...selected,
78
+ pairId: migrated.pairId,
79
+ namespace: migrated.namespace
80
+ };
81
+ }
82
+ function reviewDir(memoryDir) {
83
+ return path.join(memoryDir, ".review", "contradictions");
84
+ }
85
+ function migrationMarkerPath(memoryDir, namespace) {
86
+ const namespaceKey = createHash("sha256").update(namespace).digest("hex").slice(0, 16);
87
+ return path.join(reviewDir(memoryDir), `${UNSCOPED_MIGRATION_MARKER_PREFIX}${namespaceKey}${UNSCOPED_MIGRATION_MARKER_SUFFIX}`);
88
+ }
89
+ function clearUnscopedMigrationMarkers(memoryDir) {
90
+ const dir = reviewDir(memoryDir);
91
+ if (!fs.existsSync(dir)) return;
92
+ try {
93
+ for (const entry of fs.readdirSync(dir)) {
94
+ if (entry.startsWith(UNSCOPED_MIGRATION_MARKER_PREFIX) && entry.endsWith(UNSCOPED_MIGRATION_MARKER_SUFFIX)) {
95
+ fs.rmSync(path.join(dir, entry), { force: true });
96
+ }
97
+ }
98
+ } catch {
99
+ }
100
+ }
101
+ function pairPath(memoryDir, pairId) {
102
+ if (pairId.includes("/") || pairId.includes("\\") || pairId.includes("..")) {
103
+ throw new Error(`Invalid pairId: ${pairId}`);
104
+ }
105
+ return path.join(reviewDir(memoryDir), `${pairId}.json`);
106
+ }
107
+ function ensureDir(memoryDir) {
108
+ const dir = reviewDir(memoryDir);
109
+ if (!fs.existsSync(dir)) {
110
+ fs.mkdirSync(dir, { recursive: true });
111
+ }
112
+ }
113
+ function uniqueTempPath(filePath) {
114
+ return `${filePath}.${process.pid}.${Date.now()}.${randomUUID()}.tmp`;
115
+ }
116
+ function writePairFile(filePath, pair) {
117
+ const tmpPath = uniqueTempPath(filePath);
118
+ try {
119
+ fs.writeFileSync(tmpPath, JSON.stringify(pair, null, 2), "utf-8");
120
+ fs.renameSync(tmpPath, filePath);
121
+ } catch (error) {
122
+ try {
123
+ fs.rmSync(tmpPath, { force: true });
124
+ } catch {
125
+ }
126
+ throw error;
127
+ }
128
+ }
129
+ function computeMemoryContentHash(content, category) {
130
+ const normalized = JSON.stringify({
131
+ content: content.trim(),
132
+ category: (category ?? "").trim()
133
+ });
134
+ return createHash("sha256").update(normalized).digest("hex").slice(0, 16);
135
+ }
136
+ function suppliedMemoryHashesChanged(existing, pair) {
137
+ if (!existing.memoryContentHashes || !pair.memoryContentHashes) return false;
138
+ return pair.memoryIds.some((memoryId) => {
139
+ const current = pair.memoryContentHashes?.[memoryId];
140
+ return typeof current === "string" && existing.memoryContentHashes?.[memoryId] !== current;
141
+ });
142
+ }
143
+ function writePair(memoryDir, pair, options = {}) {
144
+ ensureDir(memoryDir);
145
+ if (pair.namespace === void 0) {
146
+ clearUnscopedMigrationMarkers(memoryDir);
147
+ }
148
+ const pairId = computePairId(pair.memoryIds[0], pair.memoryIds[1], pair.namespace);
149
+ const existing = readPair(memoryDir, pairId);
150
+ if (isTerminalResolution(existing?.resolution)) {
151
+ return existing;
152
+ }
153
+ if (existing?.resolution === "both-valid" && options.cooldownDays === void 0) {
154
+ return existing;
155
+ }
156
+ const existingDeferralExpired = Boolean(existing && isDeferred(existing) && !isDeferralActive(existing));
157
+ if (existing && isDeferralActive(existing)) {
158
+ return existing;
159
+ }
160
+ const existingDormantCooldownActive = Boolean(
161
+ existing && isDormantReviewedPair(existing) && options.cooldownDays !== void 0 && isCoolingDown(existing, options.cooldownDays)
162
+ );
163
+ const dormantContentChanged = Boolean(
164
+ existing && existingDormantCooldownActive && suppliedMemoryHashesChanged(existing, pair)
165
+ );
166
+ const existingDormantExpired = Boolean(
167
+ existing && isDormantReviewedPair(existing) && options.cooldownDays !== void 0 && !existingDormantCooldownActive
168
+ );
169
+ if (existing && !existingDeferralExpired && (existingDormantCooldownActive && !dormantContentChanged || !existingDormantExpired && !dormantContentChanged && existing.confidence >= pair.confidence)) {
170
+ return existing;
171
+ }
172
+ const full = {
173
+ ...pair,
174
+ pairId,
175
+ lastReviewedAt: existingDeferralExpired || existingDormantExpired || dormantContentChanged ? pair.lastReviewedAt : existing?.lastReviewedAt ?? pair.lastReviewedAt,
176
+ resolution: void 0,
177
+ deferredUntil: existingDeferralExpired || existingDormantExpired || dormantContentChanged ? void 0 : existing?.deferredUntil
178
+ };
179
+ const filePath = pairPath(memoryDir, pairId);
180
+ writePairFile(filePath, full);
181
+ return full;
182
+ }
183
+ function writePairs(memoryDir, pairs, options = {}) {
184
+ const seen = /* @__PURE__ */ new Set();
185
+ const results = [];
186
+ for (const pair of pairs) {
187
+ const key = computePairId(pair.memoryIds[0], pair.memoryIds[1], pair.namespace);
188
+ if (seen.has(key)) continue;
189
+ seen.add(key);
190
+ results.push(writePair(memoryDir, pair, options));
191
+ }
192
+ return results;
193
+ }
194
+ function readPair(memoryDir, pairId) {
195
+ const filePath = pairPath(memoryDir, pairId);
196
+ try {
197
+ const raw = fs["re"+"ad"+"Fi"+"le"+"Sync"](filePath, "utf-8");
198
+ const parsed = JSON.parse(raw);
199
+ if (typeof parsed === "object" && parsed !== null && Array.isArray(parsed.memoryIds)) {
200
+ return parsed;
201
+ }
202
+ return null;
203
+ } catch {
204
+ return null;
205
+ }
206
+ }
207
+ function listPairs(memoryDir, options) {
208
+ const startTime = Date.now();
209
+ const dir = reviewDir(memoryDir);
210
+ const { filter = "all", namespace, includeUnscopedForNamespace = false, limit = 50 } = options ?? {};
211
+ const pairs = [];
212
+ let total = 0;
213
+ if (!fs.existsSync(dir)) {
214
+ return { pairs: [], total: 0, durationMs: Date.now() - startTime };
215
+ }
216
+ for (const entry of fs.readdirSync(dir)) {
217
+ if (!entry.endsWith(".json")) continue;
218
+ try {
219
+ const raw = fs["re"+"ad"+"Fi"+"le"+"Sync"](path.join(dir, entry), "utf-8");
220
+ const pair = JSON.parse(raw);
221
+ if (typeof pair !== "object" || pair === null) continue;
222
+ if (!Array.isArray(pair.memoryIds)) continue;
223
+ if (namespace && pair.namespace !== namespace && !(includeUnscopedForNamespace && pair.namespace === void 0)) continue;
224
+ if (filter === "unresolved") {
225
+ if (isTerminalResolution(pair.resolution)) continue;
226
+ if (isDeferralActive(pair)) continue;
227
+ if (pair.resolution === "both-valid") continue;
228
+ if (pair.verdict === "independent") continue;
229
+ } else if (filter !== "all" && pair.verdict !== filter) {
230
+ continue;
231
+ }
232
+ total++;
233
+ if (pairs.length < limit) pairs.push(pair);
234
+ } catch {
235
+ continue;
236
+ }
237
+ }
238
+ return { pairs, total, durationMs: Date.now() - startTime };
239
+ }
240
+ function migrateUnscopedPairsToNamespace(memoryDir, namespace, options = {}) {
241
+ const resolvedNamespace = namespace.trim();
242
+ if (!resolvedNamespace) return 0;
243
+ const dir = reviewDir(memoryDir);
244
+ if (!fs.existsSync(dir)) return 0;
245
+ const markerPath = migrationMarkerPath(memoryDir, resolvedNamespace);
246
+ if (fs.existsSync(markerPath)) return 0;
247
+ let migrated = 0;
248
+ let hadMigrationFailure = false;
249
+ for (const entry of fs.readdirSync(dir)) {
250
+ if (!entry.endsWith(".json")) continue;
251
+ const filePath = path.join(dir, entry);
252
+ try {
253
+ const raw = fs["re"+"ad"+"Fi"+"le"+"Sync"](filePath, "utf-8");
254
+ const pair = JSON.parse(raw);
255
+ if (typeof pair !== "object" || pair === null) continue;
256
+ if (!Array.isArray(pair.memoryIds)) continue;
257
+ if (pair.namespace !== void 0) continue;
258
+ const pairId = computePairId(pair.memoryIds[0], pair.memoryIds[1], resolvedNamespace);
259
+ const migratedPair = { ...pair, namespace: resolvedNamespace, pairId };
260
+ const targetPath = pairPath(memoryDir, pairId);
261
+ try {
262
+ if (targetPath === filePath) {
263
+ writePairFile(filePath, migratedPair);
264
+ } else if (!fs.existsSync(targetPath)) {
265
+ writePairFile(targetPath, migratedPair);
266
+ fs.rmSync(filePath, { force: true });
267
+ } else {
268
+ const existing = readPair(memoryDir, pairId);
269
+ writePairFile(targetPath, existing ? mergeMigratedPair(existing, migratedPair, options) : migratedPair);
270
+ fs.rmSync(filePath, { force: true });
271
+ }
272
+ migrated += 1;
273
+ } catch {
274
+ hadMigrationFailure = true;
275
+ continue;
276
+ }
277
+ } catch {
278
+ continue;
279
+ }
280
+ }
281
+ if (!hadMigrationFailure) {
282
+ try {
283
+ fs.writeFileSync(markerPath, `${(/* @__PURE__ */ new Date()).toISOString()}
284
+ `, { encoding: "utf-8", flag: "wx" });
285
+ } catch {
286
+ }
287
+ }
288
+ return migrated;
289
+ }
290
+ function isCoolingDown(pair, cooldownDays) {
291
+ if (cooldownDays <= 0) return false;
292
+ const deferredUntil = deferralUntilMillis(pair);
293
+ if (deferredUntil !== null) {
294
+ return Date.now() < deferredUntil;
295
+ }
296
+ if (!pair.lastReviewedAt) return false;
297
+ const lastReviewed = parseIsoMillis(pair.lastReviewedAt);
298
+ if (lastReviewed === null) return false;
299
+ const cooldownMs = cooldownDays * 24 * 60 * 60 * 1e3;
300
+ return Date.now() < lastReviewed + cooldownMs;
301
+ }
302
+ function resolvePair(memoryDir, pairId, verb) {
303
+ if (typeof verb !== "string" || !VALID_RESOLUTION_VERBS.includes(verb)) {
304
+ throw new Error(`Invalid contradiction resolution verb: ${String(verb)}`);
305
+ }
306
+ if (verb === "needs-more-context") {
307
+ return deferPair(memoryDir, pairId);
308
+ }
309
+ const existing = readPair(memoryDir, pairId);
310
+ if (!existing) return null;
311
+ if (preservesDirectResolution(existing.resolution)) return existing;
312
+ const updated = {
313
+ ...existing,
314
+ lastReviewedAt: (/* @__PURE__ */ new Date()).toISOString(),
315
+ resolution: verb,
316
+ deferredUntil: void 0
317
+ };
318
+ const filePath = pairPath(memoryDir, pairId);
319
+ writePairFile(filePath, updated);
320
+ return updated;
321
+ }
322
+ function deferPair(memoryDir, pairId, deferredUntil = new Date(Date.now() + NEEDS_MORE_CONTEXT_COOLDOWN_MS).toISOString()) {
323
+ const existing = readPair(memoryDir, pairId);
324
+ if (!existing) return null;
325
+ if (preservesDirectResolution(existing.resolution)) return existing;
326
+ const updated = {
327
+ ...existing,
328
+ lastReviewedAt: (/* @__PURE__ */ new Date()).toISOString(),
329
+ resolution: void 0,
330
+ deferredUntil
331
+ };
332
+ const filePath = pairPath(memoryDir, pairId);
333
+ writePairFile(filePath, updated);
334
+ return updated;
335
+ }
336
+ function memoryHashesChanged(_memoryDir, pair, getCurrentHash) {
337
+ if (!pair.memoryContentHashes) return false;
338
+ for (const memoryId of pair.memoryIds) {
339
+ const previousHash = pair.memoryContentHashes[memoryId];
340
+ if (typeof previousHash !== "string") continue;
341
+ const currentHash = getCurrentHash(memoryId);
342
+ if (currentHash !== null && currentHash !== previousHash) return true;
343
+ }
344
+ return false;
345
+ }
346
+
347
+ export {
348
+ computePairId,
349
+ isDefaultReviewNamespace,
350
+ computeMemoryContentHash,
351
+ writePair,
352
+ writePairs,
353
+ readPair,
354
+ listPairs,
355
+ migrateUnscopedPairsToNamespace,
356
+ isCoolingDown,
357
+ resolvePair,
358
+ deferPair,
359
+ memoryHashesChanged
360
+ };
@@ -9,16 +9,30 @@ const realpath = fsReadModule0.realpath;
9
9
  const stat = fsReadModule0.stat;
10
10
  const writeFile = fsReadModule0.writeFile;
11
11
  import path from "path";
12
+ var fatalUtf8Decoder = new TextDecoder("utf-8", { fatal: true, ignoreBOM: true });
12
13
  async function sha256File(filePath) {
13
14
  const buf = await fileReader(filePath);
14
15
  const sha256 = createHash("sha256").update(buf).digest("hex");
15
16
  return { sha256, bytes: buf.byteLength };
16
17
  }
17
- function sha256String(content) {
18
- const buf = Buffer.from(content, "utf-8");
18
+ function sha256Bytes(content) {
19
+ const buf = Buffer.from(content);
19
20
  const sha256 = createHash("sha256").update(buf).digest("hex");
20
21
  return { sha256, bytes: buf.byteLength };
21
22
  }
23
+ function sha256String(content) {
24
+ return sha256Bytes(Buffer.from(content, "utf-8"));
25
+ }
26
+ async function readUtf8FileStrict(filePath) {
27
+ const buf = await fileReader(filePath);
28
+ let content;
29
+ try {
30
+ content = fatalUtf8Decoder.decode(buf);
31
+ } catch {
32
+ throw new Error(`transfer export requires UTF-8 text files: ${filePath}`);
33
+ }
34
+ return { content, ...sha256Bytes(buf) };
35
+ }
22
36
  async function writeJsonFile(filePath, value) {
23
37
  await mkdir(path.dirname(filePath), { recursive: true });
24
38
  await writeFile(filePath, JSON.stringify(value, null, 2) + "\n", "utf-8");
@@ -61,6 +75,69 @@ function toPosixRelPath(absPath, rootDir) {
61
75
  function fromPosixRelPath(relPath) {
62
76
  return relPath.split("/").join(path.sep);
63
77
  }
78
+ async function prepareSafeArchiveRoot(absPath, errorPrefix, argName) {
79
+ const rootAbs = path.resolve(absPath);
80
+ await assertIsDirectoryNotSymlink(rootAbs, errorPrefix, argName);
81
+ return {
82
+ abs: rootAbs,
83
+ real: await realpath(rootAbs),
84
+ errorPrefix,
85
+ argName
86
+ };
87
+ }
88
+ function validateArchiveRelativePath(relPath, errorPrefix) {
89
+ if (relPath.length === 0) {
90
+ throw new Error(`${errorPrefix}: record path must not be empty`);
91
+ }
92
+ if (relPath.includes("\\")) {
93
+ throw new Error(
94
+ `${errorPrefix}: record path must use POSIX separators: ${relPath}`
95
+ );
96
+ }
97
+ if (path.posix.isAbsolute(relPath)) {
98
+ throw new Error(
99
+ `${errorPrefix}: record path must be relative: ${relPath}`
100
+ );
101
+ }
102
+ const segments = relPath.split("/");
103
+ if (segments.some((segment) => segment === "" || segment === "." || segment === "..")) {
104
+ throw new Error(
105
+ `${errorPrefix}: record path contains unsafe segments: ${relPath}`
106
+ );
107
+ }
108
+ const normalized = path.posix.normalize(relPath);
109
+ if (normalized === "." || normalized === ".." || normalized.startsWith("../") || path.posix.isAbsolute(normalized)) {
110
+ throw new Error(
111
+ `${errorPrefix}: record path escapes target root: ${relPath}`
112
+ );
113
+ }
114
+ return normalized;
115
+ }
116
+ async function resolveSafeArchiveTarget(root, relPath) {
117
+ const safeRelPath = validateArchiveRelativePath(relPath, root.errorPrefix);
118
+ const targetAbs = path.resolve(root.abs, fromPosixRelPath(safeRelPath));
119
+ if (!isPathInsideRoot(root.abs, targetAbs)) {
120
+ throw new Error(
121
+ `${root.errorPrefix}: record path escapes target root: ${relPath}`
122
+ );
123
+ }
124
+ const targetStat = await lstat(targetAbs).catch((error) => {
125
+ if (error.code === "ENOENT") return null;
126
+ throw error;
127
+ });
128
+ if (targetStat?.isSymbolicLink()) {
129
+ throw new Error(
130
+ `${root.errorPrefix}: record path targets a symlink: ${relPath}`
131
+ );
132
+ }
133
+ await assertRealpathInsideRoot(
134
+ root.real,
135
+ targetAbs,
136
+ relPath,
137
+ root.errorPrefix
138
+ );
139
+ return targetAbs;
140
+ }
64
141
  function isPathInsideRoot(rootReal, absPath) {
65
142
  const rel = path.relative(rootReal, absPath);
66
143
  if (rel === "") return true;
@@ -103,7 +180,9 @@ async function assertRealpathInsideRoot(rootReal, targetAbs, sourcePath, errorPr
103
180
 
104
181
  export {
105
182
  sha256File,
183
+ sha256Bytes,
106
184
  sha256String,
185
+ readUtf8FileStrict,
107
186
  writeJsonFile,
108
187
  readJsonFile,
109
188
  listFilesRecursive,
@@ -111,6 +190,9 @@ export {
111
190
  fileExists,
112
191
  toPosixRelPath,
113
192
  fromPosixRelPath,
193
+ prepareSafeArchiveRoot,
194
+ validateArchiveRelativePath,
195
+ resolveSafeArchiveTarget,
114
196
  isPathInsideRoot,
115
197
  assertIsDirectoryNotSymlink,
116
198
  assertRealpathInsideRoot
@@ -2,7 +2,7 @@ import {
2
2
  getKey,
3
3
  readHeader,
4
4
  secureStoreDir
5
- } from "./chunk-CXM7EBAO.js";
5
+ } from "./chunk-MXFJXUHC.js";
6
6
  import {
7
7
  open,
8
8
  seal
@@ -122,7 +122,7 @@ async function loadKdfSection(memoryDir) {
122
122
  try {
123
123
  const header = await readHeader(memoryDir);
124
124
  if (header !== null) {
125
- const { decodeMetadataSalt } = await import("./metadata-JAGIWHEA.js");
125
+ const { decodeMetadataSalt } = await import("./metadata-WK2TRPYZ.js");
126
126
  const salt2 = decodeMetadataSalt(header.metadata);
127
127
  const kdf = header.metadata.kdf;
128
128
  const json2 = JSON.stringify({
@@ -134,9 +134,9 @@ async function loadKdfSection(memoryDir) {
134
134
  }
135
135
  } catch {
136
136
  }
137
- const { generateSalt } = await import("./cipher-VHAFCG7Z.js");
137
+ const { generateSalt } = await import("./cipher-E23BHBSO.js");
138
138
  const salt = generateSalt();
139
- const { DEFAULT_ARGON2ID_PARAMS } = await import("./kdf-H5B23ZM2.js");
139
+ const { DEFAULT_ARGON2ID_PARAMS } = await import("./kdf-RXKIWHRU.js");
140
140
  const json = JSON.stringify({
141
141
  algorithm: "argon2id",
142
142
  params: DEFAULT_ARGON2ID_PARAMS,
@@ -11,7 +11,7 @@ import {
11
11
  parseEnvelope,
12
12
  seal
13
13
  } from "./chunk-YGGGUTG3.js";
14
- import "./chunk-MLKGABMK.js";
14
+ import "./chunk-I2KLQ2HA.js";
15
15
  export {
16
16
  AES_KEY_LENGTH,
17
17
  AUTH_TAG_LENGTH,
@@ -1,7 +1,7 @@
1
1
  import {
2
2
  getVersion
3
- } from "./chunk-6OJAU466.js";
4
- import "./chunk-MLKGABMK.js";
3
+ } from "./chunk-QMUQV5NP.js";
4
+ import "./chunk-I2KLQ2HA.js";
5
5
 
6
6
  // ../remnic-core/src/consolidation-undo.ts
7
7
  import path from "path";
@@ -0,0 +1,29 @@
1
+ import {
2
+ computeMemoryContentHash,
3
+ computePairId,
4
+ deferPair,
5
+ isCoolingDown,
6
+ isDefaultReviewNamespace,
7
+ listPairs,
8
+ memoryHashesChanged,
9
+ migrateUnscopedPairsToNamespace,
10
+ readPair,
11
+ resolvePair,
12
+ writePair,
13
+ writePairs
14
+ } from "./chunk-YJYZMLD5.js";
15
+ import "./chunk-I2KLQ2HA.js";
16
+ export {
17
+ computeMemoryContentHash,
18
+ computePairId,
19
+ deferPair,
20
+ isCoolingDown,
21
+ isDefaultReviewNamespace,
22
+ listPairs,
23
+ memoryHashesChanged,
24
+ migrateUnscopedPairsToNamespace,
25
+ readPair,
26
+ resolvePair,
27
+ writePair,
28
+ writePairs
29
+ };
@@ -0,0 +1,12 @@
1
+ import {
2
+ ACTIVE_STATUSES,
3
+ runContradictionScan
4
+ } from "./chunk-JC3FCKYL.js";
5
+ import "./chunk-YJYZMLD5.js";
6
+ import "./chunk-3A5ELHTT.js";
7
+ import "./chunk-UFU5GGGA.js";
8
+ import "./chunk-I2KLQ2HA.js";
9
+ export {
10
+ ACTIVE_STATUSES,
11
+ runContradictionScan
12
+ };
@@ -1,4 +1,4 @@
1
- import "./chunk-MLKGABMK.js";
1
+ import "./chunk-I2KLQ2HA.js";
2
2
 
3
3
  // ../remnic-core/src/maintenance/dreams-ledger.ts
4
4
  import path from "path";
@@ -1,15 +1,15 @@
1
1
  import {
2
2
  CompoundingEngine,
3
3
  defaultTierMigrationCycleBudget
4
- } from "./chunk-UTDLHBBV.js";
5
- import "./chunk-EXDYWXMB.js";
6
- import "./chunk-EYCLXMIV.js";
7
- import "./chunk-6OJAU466.js";
4
+ } from "./chunk-BLC3RQNV.js";
5
+ import "./chunk-4XDQ3KEC.js";
6
+ import "./chunk-G3CZA4SD.js";
7
+ import "./chunk-QMUQV5NP.js";
8
8
  import "./chunk-RKR6PTPA.js";
9
9
  import "./chunk-UFU5GGGA.js";
10
10
  import "./chunk-YGGGUTG3.js";
11
- import "./chunk-TDRJVMUP.js";
12
- import "./chunk-MLKGABMK.js";
11
+ import "./chunk-EH4AXGRO.js";
12
+ import "./chunk-I2KLQ2HA.js";
13
13
  export {
14
14
  CompoundingEngine,
15
15
  defaultTierMigrationCycleBudget
@@ -5,7 +5,7 @@ import {
5
5
  recordJudgeVerdict
6
6
  } from "./chunk-5ZW5XJQ6.js";
7
7
  import "./chunk-UFU5GGGA.js";
8
- import "./chunk-MLKGABMK.js";
8
+ import "./chunk-I2KLQ2HA.js";
9
9
  export {
10
10
  EXTRACTION_JUDGE_VERDICT_CATEGORY,
11
11
  judgeTelemetryPath,
@@ -1,10 +1,10 @@
1
1
  import {
2
2
  FallbackLlmClient
3
- } from "./chunk-VFULKFKI.js";
3
+ } from "./chunk-SWOYEQN2.js";
4
4
  import "./chunk-3A5ELHTT.js";
5
5
  import "./chunk-UFU5GGGA.js";
6
- import "./chunk-TDRJVMUP.js";
7
- import "./chunk-MLKGABMK.js";
6
+ import "./chunk-EH4AXGRO.js";
7
+ import "./chunk-I2KLQ2HA.js";
8
8
  export {
9
9
  FallbackLlmClient
10
10
  };
@@ -5,11 +5,11 @@ import {
5
5
  applyUtilityPromotionRuntimePolicy,
6
6
  decideTierTransition,
7
7
  loadUtilityRuntimeValues
8
- } from "./chunk-7OQEPGQF.js";
9
- import "./chunk-4G2XCSD2.js";
8
+ } from "./chunk-LZCGPRHS.js";
9
+ import "./chunk-BZ4EYURA.js";
10
10
  import "./chunk-3G7FAF6S.js";
11
- import "./chunk-5LE4HTVL.js";
12
- import "./chunk-MLKGABMK.js";
11
+ import "./chunk-25J4PXDH.js";
12
+ import "./chunk-I2KLQ2HA.js";
13
13
 
14
14
  // ../remnic-core/src/maintenance/first-start-migration.ts
15
15
  import path from "path";
@@ -1,4 +1,4 @@
1
- import "./chunk-MLKGABMK.js";
1
+ import "./chunk-I2KLQ2HA.js";
2
2
 
3
3
  // ../remnic-core/src/maintenance/forget.ts
4
4
  var ForgetMemoryNotFoundError = class extends Error {
@@ -6,13 +6,18 @@ import {
6
6
  fromPosixRelPath,
7
7
  isPathInsideRoot,
8
8
  listFilesRecursive,
9
+ prepareSafeArchiveRoot,
9
10
  readJsonFile,
11
+ readUtf8FileStrict,
12
+ resolveSafeArchiveTarget,
13
+ sha256Bytes,
10
14
  sha256File,
11
15
  sha256String,
12
16
  toPosixRelPath,
17
+ validateArchiveRelativePath,
13
18
  writeJsonFile
14
- } from "./chunk-NKVIN6RD.js";
15
- import "./chunk-MLKGABMK.js";
19
+ } from "./chunk-YKV4EFUI.js";
20
+ import "./chunk-I2KLQ2HA.js";
16
21
  export {
17
22
  assertIsDirectoryNotSymlink,
18
23
  assertRealpathInsideRoot,
@@ -21,9 +26,14 @@ export {
21
26
  fromPosixRelPath,
22
27
  isPathInsideRoot,
23
28
  listFilesRecursive,
29
+ prepareSafeArchiveRoot,
24
30
  readJsonFile,
31
+ readUtf8FileStrict,
32
+ resolveSafeArchiveTarget,
33
+ sha256Bytes,
25
34
  sha256File,
26
35
  sha256String,
27
36
  toPosixRelPath,
37
+ validateArchiveRelativePath,
28
38
  writeJsonFile
29
39
  };