@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.
- package/README.md +38 -4
- package/dist/{calibration-Z5WWNV7U.js → calibration-RKL2LRW4.js} +4 -4
- package/dist/{capsule-cli-GBM3WPAM.js → capsule-cli-EHZPMXBC.js} +2 -2
- package/dist/{capsule-crypto-K3IRTKRH.js → capsule-crypto-JS67OSWM.js} +3 -3
- package/dist/capsule-export-DX53CPIT.js +17 -0
- package/dist/capsule-import-4OXCPHOT.js +16 -0
- package/dist/{capsule-merge-IWOQ34KL.js → capsule-merge-25AUN33Q.js} +7 -7
- package/dist/{causal-chain-WYN5QOPS.js → causal-chain-BVTOWZKC.js} +4 -4
- package/dist/{causal-consolidation-C64NNE4T.js → causal-consolidation-DRPM2KOE.js} +13 -10
- package/dist/{causal-retrieval-NZHQOZOE.js → causal-retrieval-XAP6QKHZ.js} +4 -5
- package/dist/{causal-trajectory-graph-VBPE2WPM.js → causal-trajectory-graph-ZWQWZ7N5.js} +2 -2
- package/dist/{chunk-5LE4HTVL.js → chunk-25J4PXDH.js} +0 -18
- package/dist/{chunk-6UFI73TJ.js → chunk-3IKMUNW5.js} +53 -46
- package/dist/{chunk-EXDYWXMB.js → chunk-4XDQ3KEC.js} +1 -2
- package/dist/{chunk-JGIUTWZS.js → chunk-6O3H3DPL.js} +2 -2
- package/dist/{chunk-UTDLHBBV.js → chunk-BLC3RQNV.js} +5 -555
- package/dist/{chunk-4G2XCSD2.js → chunk-BZ4EYURA.js} +0 -5
- package/dist/{chunk-L6I4MQKO.js → chunk-CEL5ZLKP.js} +6 -6
- package/dist/{chunk-TDRJVMUP.js → chunk-EH4AXGRO.js} +0 -12
- package/dist/{chunk-EYCLXMIV.js → chunk-G3CZA4SD.js} +9 -427
- package/dist/chunk-I2KLQ2HA.js +22 -0
- package/dist/chunk-IO5WWY6A.js +156 -0
- package/dist/{contradiction-scan-A5NOTZPN.js → chunk-JC3FCKYL.js} +189 -86
- package/dist/{chunk-SVSQAG6M.js → chunk-KC7KSQR4.js} +47 -28
- package/dist/chunk-LZCGPRHS.js +228 -0
- package/dist/{chunk-CXM7EBAO.js → chunk-MXFJXUHC.js} +1 -1
- package/dist/{chunk-VRGUUHBV.js → chunk-NUWDSTP7.js} +1 -1
- package/dist/{chunk-4LYQ4ONL.js → chunk-QCCP4RU5.js} +8 -3
- package/dist/{chunk-6OJAU466.js → chunk-QMUQV5NP.js} +0 -1
- package/dist/{chunk-LLUROTZJ.js → chunk-QQXJODFL.js} +9 -9
- package/dist/{chunk-6F6EKSVP.js → chunk-QXXEF7VI.js} +1 -1
- package/dist/{chunk-CMKR6NDQ.js → chunk-SEGEX7W4.js} +73 -241
- package/dist/{chunk-VFULKFKI.js → chunk-SWOYEQN2.js} +42 -17
- package/dist/chunk-TH5FF5SC.js +16 -0
- package/dist/{chunk-FGTYFLL5.js → chunk-TXOEHSVP.js} +29 -32
- package/dist/chunk-UZJ7EERS.js +272 -0
- package/dist/chunk-YJYZMLD5.js +360 -0
- package/dist/{chunk-NKVIN6RD.js → chunk-YKV4EFUI.js} +84 -2
- package/dist/{chunk-SSFTU6LP.js → chunk-ZS6VABML.js} +4 -4
- package/dist/{cipher-VHAFCG7Z.js → cipher-E23BHBSO.js} +1 -1
- package/dist/{consolidation-undo-5ZSX4MWO.js → consolidation-undo-FKJZCJHS.js} +2 -2
- package/dist/contradiction-review-WJRWNQ5N.js +29 -0
- package/dist/contradiction-scan-5X423QGT.js +12 -0
- package/dist/{dreams-ledger-3I52ISYR.js → dreams-ledger-KDX44I7R.js} +1 -1
- package/dist/{engine-47AKKYJ4.js → engine-5P774HTZ.js} +6 -6
- package/dist/{extraction-judge-telemetry-GHOTVYMP.js → extraction-judge-telemetry-O4ZVGLTU.js} +1 -1
- package/dist/{fallback-llm-45A755XP.js → fallback-llm-43UMEXNJ.js} +3 -3
- package/dist/{first-start-migration-I24M2JEE.js → first-start-migration-H2SAXAGR.js} +4 -4
- package/dist/{forget-NI4RBDPB.js → forget-ZECIDNL5.js} +1 -1
- package/dist/{fs-utils-PZRI2HDZ.js → fs-utils-OYXSZSVV.js} +12 -2
- package/dist/{graph-edge-decay-5CVKWBYH.js → graph-edge-decay-24ZKD5QL.js} +5 -5
- package/dist/index.js +7098 -84293
- package/dist/{kdf-H5B23ZM2.js → kdf-RXKIWHRU.js} +1 -1
- package/dist/legacy-hook-compat-QHHKF4GK.js +2 -0
- package/dist/{logger-TNOKCH7X.js → logger-XG7JKLPS.js} +1 -1
- package/dist/{memory-governance-QS7Z425Y.js → memory-governance-6K4M4YXD.js} +5 -5
- package/dist/{metadata-JAGIWHEA.js → metadata-WK2TRPYZ.js} +1 -1
- package/dist/{migrate-from-identity-anchor-7MMSPEUM.js → migrate-from-identity-anchor-SNDNKHZD.js} +1 -1
- package/dist/path-ZKO74XXC.js +7 -0
- package/dist/{peers-KRFXWRQ6.js → peers-W53WSDXG.js} +1 -1
- package/dist/{purge-XN2VSPZ2.js → purge-IKJISXEQ.js} +1 -1
- package/dist/resolution-BN35OXDS.js +11 -0
- package/dist/{secure-store-A4NGCNXV.js → secure-store-F75I54O5.js} +3 -3
- package/dist/{state-PVISYXRH.js → state-4ITLYMAU.js} +1 -1
- package/dist/{state-store-N6TFBFSP.js → state-store-ET3ADVY5.js} +3 -3
- package/dist/{storage-DDYQGLXA.js → storage-5EY6T7ON.js} +3 -3
- package/dist/{tier-stats-IZNW66NC.js → tier-stats-ZRQBV6G2.js} +4 -4
- package/dist/{trace-NJESSGH7.js → trace-IL2Y34EH.js} +1 -1
- package/dist/{tui-MGK2LYJY.js → tui-7KRDCMYK.js} +1 -1
- package/dist/{types-R4DO7AKM.js → types-MBUINTB2.js} +3 -3
- package/openclaw.plugin.json +164 -8
- package/package.json +9 -6
- package/scripts/faiss_index.py +816 -0
- package/scripts/faiss_requirements.txt +3 -0
- package/dist/capsule-export-IXVERCQG.js +0 -17
- package/dist/capsule-import-IA6VIOPQ.js +0 -16
- package/dist/chunk-3GUF7RQI.js +0 -559
- package/dist/chunk-7OQEPGQF.js +0 -533
- package/dist/chunk-DIZW6H5J.js +0 -136
- package/dist/chunk-FQRSVYY4.js +0 -110
- package/dist/chunk-GUSMRW4H.js +0 -12
- package/dist/chunk-MLKGABMK.js +0 -9
- package/dist/chunk-WPINX4MF.js +0 -380
- package/dist/contradiction-review-SVGBS3V5.js +0 -21
- package/dist/legacy-hook-compat-XQ7FP6FV.js +0 -35
- package/dist/path-JIEGNWFL.js +0 -7
- 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
|
|
18
|
-
const buf = Buffer.from(content
|
|
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-
|
|
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-
|
|
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-
|
|
137
|
+
const { generateSalt } = await import("./cipher-E23BHBSO.js");
|
|
138
138
|
const salt = generateSalt();
|
|
139
|
-
const { DEFAULT_ARGON2ID_PARAMS } = await import("./kdf-
|
|
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,
|
|
@@ -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,15 +1,15 @@
|
|
|
1
1
|
import {
|
|
2
2
|
CompoundingEngine,
|
|
3
3
|
defaultTierMigrationCycleBudget
|
|
4
|
-
} from "./chunk-
|
|
5
|
-
import "./chunk-
|
|
6
|
-
import "./chunk-
|
|
7
|
-
import "./chunk-
|
|
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-
|
|
12
|
-
import "./chunk-
|
|
11
|
+
import "./chunk-EH4AXGRO.js";
|
|
12
|
+
import "./chunk-I2KLQ2HA.js";
|
|
13
13
|
export {
|
|
14
14
|
CompoundingEngine,
|
|
15
15
|
defaultTierMigrationCycleBudget
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
import {
|
|
2
2
|
FallbackLlmClient
|
|
3
|
-
} from "./chunk-
|
|
3
|
+
} from "./chunk-SWOYEQN2.js";
|
|
4
4
|
import "./chunk-3A5ELHTT.js";
|
|
5
5
|
import "./chunk-UFU5GGGA.js";
|
|
6
|
-
import "./chunk-
|
|
7
|
-
import "./chunk-
|
|
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-
|
|
9
|
-
import "./chunk-
|
|
8
|
+
} from "./chunk-LZCGPRHS.js";
|
|
9
|
+
import "./chunk-BZ4EYURA.js";
|
|
10
10
|
import "./chunk-3G7FAF6S.js";
|
|
11
|
-
import "./chunk-
|
|
12
|
-
import "./chunk-
|
|
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";
|
|
@@ -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-
|
|
15
|
-
import "./chunk-
|
|
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
|
};
|