openclaw-cortex-memory 0.1.0-Alpha.2 → 0.1.0-Alpha.21
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 +163 -203
- package/SKILL.md +71 -268
- package/dist/index.d.ts +88 -15
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +859 -1189
- package/dist/index.js.map +1 -1
- package/dist/openclaw.plugin.json +362 -14
- package/dist/src/dedup/three_stage_deduplicator.d.ts +25 -0
- package/dist/src/dedup/three_stage_deduplicator.d.ts.map +1 -0
- package/dist/src/dedup/three_stage_deduplicator.js +224 -0
- package/dist/src/dedup/three_stage_deduplicator.js.map +1 -0
- package/dist/src/engine/memory_engine.d.ts +2 -1
- package/dist/src/engine/memory_engine.d.ts.map +1 -1
- package/dist/src/engine/ts_engine.d.ts +126 -0
- package/dist/src/engine/ts_engine.d.ts.map +1 -1
- package/dist/src/engine/ts_engine.js +1172 -44
- package/dist/src/engine/ts_engine.js.map +1 -1
- package/dist/src/engine/types.d.ts +12 -0
- package/dist/src/engine/types.d.ts.map +1 -1
- package/dist/src/graph/ontology.d.ts +103 -0
- package/dist/src/graph/ontology.d.ts.map +1 -0
- package/dist/src/graph/ontology.js +564 -0
- package/dist/src/graph/ontology.js.map +1 -0
- package/dist/src/quality/llm_output_validator.d.ts +48 -0
- package/dist/src/quality/llm_output_validator.d.ts.map +1 -0
- package/dist/src/quality/llm_output_validator.js +404 -0
- package/dist/src/quality/llm_output_validator.js.map +1 -0
- package/dist/src/reflect/reflector.d.ts +7 -0
- package/dist/src/reflect/reflector.d.ts.map +1 -1
- package/dist/src/reflect/reflector.js +358 -8
- package/dist/src/reflect/reflector.js.map +1 -1
- package/dist/src/rules/rule_store.d.ts.map +1 -1
- package/dist/src/rules/rule_store.js +75 -16
- package/dist/src/rules/rule_store.js.map +1 -1
- package/dist/src/session/session_end.d.ts +33 -0
- package/dist/src/session/session_end.d.ts.map +1 -1
- package/dist/src/session/session_end.js +67 -64
- package/dist/src/session/session_end.js.map +1 -1
- package/dist/src/store/archive_store.d.ts +128 -0
- package/dist/src/store/archive_store.d.ts.map +1 -0
- package/dist/src/store/archive_store.js +481 -0
- package/dist/src/store/archive_store.js.map +1 -0
- package/dist/src/store/embedding_utils.d.ts +32 -0
- package/dist/src/store/embedding_utils.d.ts.map +1 -0
- package/dist/src/store/embedding_utils.js +173 -0
- package/dist/src/store/embedding_utils.js.map +1 -0
- package/dist/src/store/graph_memory_store.d.ts +44 -0
- package/dist/src/store/graph_memory_store.d.ts.map +1 -0
- package/dist/src/store/graph_memory_store.js +168 -0
- package/dist/src/store/graph_memory_store.js.map +1 -0
- package/dist/src/store/read_store.d.ts +86 -0
- package/dist/src/store/read_store.d.ts.map +1 -1
- package/dist/src/store/read_store.js +1681 -25
- package/dist/src/store/read_store.js.map +1 -1
- package/dist/src/store/vector_store.d.ts +44 -0
- package/dist/src/store/vector_store.d.ts.map +1 -0
- package/dist/src/store/vector_store.js +201 -0
- package/dist/src/store/vector_store.js.map +1 -0
- package/dist/src/store/write_store.d.ts +52 -0
- package/dist/src/store/write_store.d.ts.map +1 -1
- package/dist/src/store/write_store.js +245 -3
- package/dist/src/store/write_store.js.map +1 -1
- package/dist/src/sync/session_sync.d.ts +100 -2
- package/dist/src/sync/session_sync.d.ts.map +1 -1
- package/dist/src/sync/session_sync.js +673 -22
- package/dist/src/sync/session_sync.js.map +1 -1
- package/dist/src/utils/runtime_env.d.ts +4 -0
- package/dist/src/utils/runtime_env.d.ts.map +1 -0
- package/dist/src/utils/runtime_env.js +51 -0
- package/dist/src/utils/runtime_env.js.map +1 -0
- package/openclaw.plugin.json +362 -14
- package/package.json +23 -6
- package/scripts/cli.js +19 -14
- package/scripts/uninstall.js +22 -5
- package/index.ts +0 -2092
- package/scripts/install.js +0 -27
|
@@ -36,6 +36,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
36
36
|
exports.createReflector = createReflector;
|
|
37
37
|
const fs = __importStar(require("fs"));
|
|
38
38
|
const path = __importStar(require("path"));
|
|
39
|
+
const crypto = __importStar(require("crypto"));
|
|
40
|
+
const MAX_PROCESSED_EVENT_IDS = 50000;
|
|
39
41
|
function readJsonl(filePath) {
|
|
40
42
|
if (!fs.existsSync(filePath)) {
|
|
41
43
|
return [];
|
|
@@ -60,51 +62,399 @@ function textOf(record) {
|
|
|
60
62
|
}
|
|
61
63
|
return "";
|
|
62
64
|
}
|
|
65
|
+
function normalizeBaseUrl(value) {
|
|
66
|
+
if (!value)
|
|
67
|
+
return "";
|
|
68
|
+
return value.endsWith("/") ? value.slice(0, -1) : value;
|
|
69
|
+
}
|
|
70
|
+
function nowIso() {
|
|
71
|
+
return new Date().toISOString();
|
|
72
|
+
}
|
|
73
|
+
function normalizeRuleText(value) {
|
|
74
|
+
return value.replace(/\s+/g, " ").trim();
|
|
75
|
+
}
|
|
76
|
+
function bumpReason(metrics, reason) {
|
|
77
|
+
const key = reason.trim() || "unknown";
|
|
78
|
+
metrics.per_reason[key] = (metrics.per_reason[key] || 0) + 1;
|
|
79
|
+
}
|
|
80
|
+
function readReflectState(filePath) {
|
|
81
|
+
try {
|
|
82
|
+
if (!fs.existsSync(filePath)) {
|
|
83
|
+
return {
|
|
84
|
+
version: "1",
|
|
85
|
+
processed_event_ids: [],
|
|
86
|
+
total_runs: 0,
|
|
87
|
+
updated_at: nowIso(),
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
const content = fs.readFileSync(filePath, "utf-8").trim();
|
|
91
|
+
if (!content) {
|
|
92
|
+
return {
|
|
93
|
+
version: "1",
|
|
94
|
+
processed_event_ids: [],
|
|
95
|
+
total_runs: 0,
|
|
96
|
+
updated_at: nowIso(),
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
const parsed = JSON.parse(content);
|
|
100
|
+
return {
|
|
101
|
+
version: "1",
|
|
102
|
+
processed_event_ids: Array.isArray(parsed.processed_event_ids)
|
|
103
|
+
? parsed.processed_event_ids.filter(item => typeof item === "string" && item.trim()).map(item => item.trim())
|
|
104
|
+
: [],
|
|
105
|
+
total_runs: typeof parsed.total_runs === "number" && Number.isFinite(parsed.total_runs)
|
|
106
|
+
? Math.max(0, Math.floor(parsed.total_runs))
|
|
107
|
+
: 0,
|
|
108
|
+
updated_at: typeof parsed.updated_at === "string" && parsed.updated_at.trim() ? parsed.updated_at : nowIso(),
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
catch {
|
|
112
|
+
return {
|
|
113
|
+
version: "1",
|
|
114
|
+
processed_event_ids: [],
|
|
115
|
+
total_runs: 0,
|
|
116
|
+
updated_at: nowIso(),
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
function writeReflectState(filePath, state) {
|
|
121
|
+
const dir = path.dirname(filePath);
|
|
122
|
+
if (!fs.existsSync(dir)) {
|
|
123
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
124
|
+
}
|
|
125
|
+
const deduped = [...new Set(state.processed_event_ids.filter(Boolean))];
|
|
126
|
+
const normalized = {
|
|
127
|
+
version: "1",
|
|
128
|
+
processed_event_ids: deduped.slice(-MAX_PROCESSED_EVENT_IDS),
|
|
129
|
+
total_runs: Math.max(0, Math.floor(state.total_runs || 0)),
|
|
130
|
+
updated_at: nowIso(),
|
|
131
|
+
};
|
|
132
|
+
fs.writeFileSync(filePath, JSON.stringify(normalized, null, 2), "utf-8");
|
|
133
|
+
}
|
|
134
|
+
function appendMetrics(filePath, metrics) {
|
|
135
|
+
const dir = path.dirname(filePath);
|
|
136
|
+
if (!fs.existsSync(dir)) {
|
|
137
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
138
|
+
}
|
|
139
|
+
fs.appendFileSync(filePath, `${JSON.stringify(metrics)}\n`, "utf-8");
|
|
140
|
+
}
|
|
141
|
+
function recordEventId(record) {
|
|
142
|
+
const candidates = [record.id, record.canonical_id, record.source_event_id];
|
|
143
|
+
for (const candidate of candidates) {
|
|
144
|
+
if (typeof candidate === "string" && candidate.trim()) {
|
|
145
|
+
return candidate.trim();
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
const payload = JSON.stringify({
|
|
149
|
+
session_id: typeof record.session_id === "string" ? record.session_id : "",
|
|
150
|
+
event_type: typeof record.event_type === "string" ? record.event_type : "",
|
|
151
|
+
summary: typeof record.summary === "string" ? record.summary : "",
|
|
152
|
+
content: typeof record.content === "string" ? record.content : "",
|
|
153
|
+
outcome: typeof record.outcome === "string" ? record.outcome : "",
|
|
154
|
+
timestamp: typeof record.timestamp === "string" ? record.timestamp : "",
|
|
155
|
+
});
|
|
156
|
+
return `derived:${crypto.createHash("sha1").update(payload).digest("hex")}`;
|
|
157
|
+
}
|
|
158
|
+
function buildFallbackRule(summary, outcome) {
|
|
159
|
+
const trimmedSummary = summary.replace(/\s+/g, " ").trim().slice(0, 180);
|
|
160
|
+
const trimmedOutcome = outcome.replace(/\s+/g, " ").trim().slice(0, 120) || "expected";
|
|
161
|
+
return `For similar tasks, prioritize reproducing "${trimmedSummary}" and verify the final outcome is "${trimmedOutcome}".`;
|
|
162
|
+
}
|
|
163
|
+
function validateRuleText(raw) {
|
|
164
|
+
const normalized = normalizeRuleText(raw);
|
|
165
|
+
if (!normalized) {
|
|
166
|
+
return { ok: false, normalized, reason: "quality_empty" };
|
|
167
|
+
}
|
|
168
|
+
if (normalized.length < 24) {
|
|
169
|
+
return { ok: false, normalized, reason: "quality_too_short" };
|
|
170
|
+
}
|
|
171
|
+
if (normalized.length > 420) {
|
|
172
|
+
return { ok: false, normalized, reason: "quality_too_long" };
|
|
173
|
+
}
|
|
174
|
+
if (/^(#|[-*]\s+|\d+\.\s+)/.test(normalized)) {
|
|
175
|
+
return { ok: false, normalized, reason: "quality_markdown_style" };
|
|
176
|
+
}
|
|
177
|
+
if (/(as an ai|i cannot|i can't|sorry)/i.test(normalized)) {
|
|
178
|
+
return { ok: false, normalized, reason: "quality_non_actionable" };
|
|
179
|
+
}
|
|
180
|
+
if (/(api[_-]?key|authorization\s*:\s*bearer|password|secret|token|sk-[a-z0-9]{16,})/i.test(normalized)) {
|
|
181
|
+
return { ok: false, normalized, reason: "quality_sensitive_content" };
|
|
182
|
+
}
|
|
183
|
+
return { ok: true, normalized };
|
|
184
|
+
}
|
|
185
|
+
function normalizeCandidateKey(text) {
|
|
186
|
+
return text.toLowerCase().replace(/\s+/g, " ").trim();
|
|
187
|
+
}
|
|
188
|
+
function looksLikeNoise(text) {
|
|
189
|
+
const normalized = text.trim();
|
|
190
|
+
if (!normalized)
|
|
191
|
+
return true;
|
|
192
|
+
if (normalized.length < 20)
|
|
193
|
+
return true;
|
|
194
|
+
if (normalized.length > 420)
|
|
195
|
+
return true;
|
|
196
|
+
if (/^\s*(hi|hello|thanks|thank you|ok|okay|got it|好的|谢谢|收到|嗯)\s*[.!?]*\s*$/i.test(normalized)) {
|
|
197
|
+
return true;
|
|
198
|
+
}
|
|
199
|
+
if (/[??]\s*$/.test(normalized)) {
|
|
200
|
+
return true;
|
|
201
|
+
}
|
|
202
|
+
if (/(can you|could you|please|帮我|能不能|可以吗)/i.test(normalized)) {
|
|
203
|
+
return true;
|
|
204
|
+
}
|
|
205
|
+
return false;
|
|
206
|
+
}
|
|
207
|
+
function signalScore(text) {
|
|
208
|
+
const normalized = text.toLowerCase();
|
|
209
|
+
let score = 0;
|
|
210
|
+
if (/(must|should|ensure|avoid|prefer|always|never|fallback|verify|sanitize|validate|dedup|idempotent|retry|timeout)/i.test(normalized)) {
|
|
211
|
+
score += 2;
|
|
212
|
+
}
|
|
213
|
+
if (/(fix|resolved|resolved|success|stable|pass|passed|deploy|release|migration|rollback|incident|postmortem)/i.test(normalized)) {
|
|
214
|
+
score += 2;
|
|
215
|
+
}
|
|
216
|
+
if (/(确保|避免|优先|必须|建议|回退|重试|校验|去重|幂等|已修复|成功|发布|稳定)/.test(text)) {
|
|
217
|
+
score += 2;
|
|
218
|
+
}
|
|
219
|
+
if (/^[A-Z][^.]{10,}\.$/.test(text.trim())) {
|
|
220
|
+
score += 1;
|
|
221
|
+
}
|
|
222
|
+
return score;
|
|
223
|
+
}
|
|
224
|
+
function promotionWeight(record, content) {
|
|
225
|
+
let score = 1;
|
|
226
|
+
const role = typeof record.role === "string" ? record.role.trim().toLowerCase() : "";
|
|
227
|
+
const outcome = typeof record.outcome === "string" ? record.outcome.trim().toLowerCase() : "";
|
|
228
|
+
const eventType = typeof record.event_type === "string" ? record.event_type.trim().toLowerCase() : "";
|
|
229
|
+
const confidence = typeof record.confidence === "number"
|
|
230
|
+
? Math.max(0, Math.min(1, record.confidence))
|
|
231
|
+
: undefined;
|
|
232
|
+
if (role === "assistant" || role === "system")
|
|
233
|
+
score += 1;
|
|
234
|
+
if (["success", "resolved", "done", "completed", "ok"].includes(outcome))
|
|
235
|
+
score += 1;
|
|
236
|
+
if (["fix", "decision", "insight", "retrospective", "requirement", "constraint"].includes(eventType))
|
|
237
|
+
score += 2;
|
|
238
|
+
if (typeof confidence === "number" && confidence >= 0.7)
|
|
239
|
+
score += 1;
|
|
240
|
+
if (content.length >= 30 && content.length <= 260)
|
|
241
|
+
score += 1;
|
|
242
|
+
score += signalScore(content);
|
|
243
|
+
return score;
|
|
244
|
+
}
|
|
245
|
+
async function requestRuleFromLlm(args) {
|
|
246
|
+
const endpoint = args.baseUrl.endsWith("/chat/completions")
|
|
247
|
+
? args.baseUrl
|
|
248
|
+
: `${args.baseUrl}/chat/completions`;
|
|
249
|
+
const prompt = `事件摘要: ${args.summary}\n结果: ${args.outcome}\n请生成一条可复用的工程规则,要求简洁、可执行、单句输出。`;
|
|
250
|
+
const body = {
|
|
251
|
+
model: args.model,
|
|
252
|
+
temperature: 0.2,
|
|
253
|
+
messages: [
|
|
254
|
+
{ role: "system", content: "你是工程规则提炼器。输出只包含规则正文,不要编号,不要解释。" },
|
|
255
|
+
{ role: "user", content: prompt },
|
|
256
|
+
],
|
|
257
|
+
};
|
|
258
|
+
let lastError = null;
|
|
259
|
+
for (let attempt = 0; attempt < 3; attempt += 1) {
|
|
260
|
+
const controller = new AbortController();
|
|
261
|
+
const timeoutId = setTimeout(() => controller.abort(), 15000);
|
|
262
|
+
try {
|
|
263
|
+
const response = await fetch(endpoint, {
|
|
264
|
+
method: "POST",
|
|
265
|
+
headers: {
|
|
266
|
+
"content-type": "application/json",
|
|
267
|
+
authorization: `Bearer ${args.apiKey}`,
|
|
268
|
+
},
|
|
269
|
+
body: JSON.stringify(body),
|
|
270
|
+
signal: controller.signal,
|
|
271
|
+
});
|
|
272
|
+
clearTimeout(timeoutId);
|
|
273
|
+
if (!response.ok) {
|
|
274
|
+
lastError = new Error(`llm_http_${response.status}`);
|
|
275
|
+
continue;
|
|
276
|
+
}
|
|
277
|
+
const json = await response.json();
|
|
278
|
+
const content = json?.choices?.[0]?.message?.content?.trim() || "";
|
|
279
|
+
if (content) {
|
|
280
|
+
return content.slice(0, 500);
|
|
281
|
+
}
|
|
282
|
+
lastError = new Error("llm_empty");
|
|
283
|
+
}
|
|
284
|
+
catch (error) {
|
|
285
|
+
clearTimeout(timeoutId);
|
|
286
|
+
lastError = error;
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
if (lastError) {
|
|
290
|
+
throw lastError;
|
|
291
|
+
}
|
|
292
|
+
return null;
|
|
293
|
+
}
|
|
63
294
|
function createReflector(options) {
|
|
64
295
|
const memoryRoot = options.dbPath ? path.resolve(options.dbPath) : path.join(options.projectRoot, "data", "memory");
|
|
65
296
|
const activeSessionsPath = path.join(memoryRoot, "sessions", "active", "sessions.jsonl");
|
|
66
297
|
const archiveSessionsPath = path.join(memoryRoot, "sessions", "archive", "sessions.jsonl");
|
|
298
|
+
const reflectStatePath = path.join(memoryRoot, ".reflect_state.json");
|
|
299
|
+
const reflectMetricsPath = path.join(memoryRoot, ".reflect_metrics.jsonl");
|
|
67
300
|
async function reflectMemory() {
|
|
301
|
+
const runId = `${Date.now()}_${crypto.randomBytes(4).toString("hex")}`;
|
|
302
|
+
const startedAt = nowIso();
|
|
303
|
+
const metrics = {
|
|
304
|
+
run_id: runId,
|
|
305
|
+
started_at: startedAt,
|
|
306
|
+
finished_at: startedAt,
|
|
307
|
+
scanned: 0,
|
|
308
|
+
attempted: 0,
|
|
309
|
+
reflected: 0,
|
|
310
|
+
skipped_no_summary: 0,
|
|
311
|
+
skipped_processed: 0,
|
|
312
|
+
skipped_quality_gate: 0,
|
|
313
|
+
rule_store_duplicate: 0,
|
|
314
|
+
llm_generated: 0,
|
|
315
|
+
llm_failed: 0,
|
|
316
|
+
fallback_used: 0,
|
|
317
|
+
per_reason: {},
|
|
318
|
+
};
|
|
319
|
+
const state = readReflectState(reflectStatePath);
|
|
320
|
+
const processedSet = new Set(state.processed_event_ids);
|
|
68
321
|
const archiveRecords = readJsonl(archiveSessionsPath).slice(-100);
|
|
322
|
+
metrics.scanned = archiveRecords.length;
|
|
69
323
|
let reflected = 0;
|
|
70
324
|
for (const record of archiveRecords) {
|
|
71
325
|
const summary = textOf(record);
|
|
72
326
|
if (!summary) {
|
|
327
|
+
metrics.skipped_no_summary += 1;
|
|
328
|
+
bumpReason(metrics, "skip_no_summary");
|
|
73
329
|
continue;
|
|
74
330
|
}
|
|
331
|
+
const eventId = recordEventId(record);
|
|
332
|
+
if (processedSet.has(eventId)) {
|
|
333
|
+
metrics.skipped_processed += 1;
|
|
334
|
+
bumpReason(metrics, "skip_processed");
|
|
335
|
+
continue;
|
|
336
|
+
}
|
|
337
|
+
metrics.attempted += 1;
|
|
75
338
|
const outcome = typeof record.outcome === "string" ? record.outcome : "unknown";
|
|
76
|
-
|
|
339
|
+
let ruleText = buildFallbackRule(summary, outcome);
|
|
340
|
+
let usedFallback = true;
|
|
341
|
+
const llmModel = options.llm?.model || "";
|
|
342
|
+
const llmApiKey = options.llm?.apiKey || "";
|
|
343
|
+
const llmBaseUrl = normalizeBaseUrl(options.llm?.baseURL || options.llm?.baseUrl);
|
|
344
|
+
if (llmModel && llmApiKey && llmBaseUrl) {
|
|
345
|
+
try {
|
|
346
|
+
const generated = await requestRuleFromLlm({
|
|
347
|
+
summary,
|
|
348
|
+
outcome,
|
|
349
|
+
model: llmModel,
|
|
350
|
+
apiKey: llmApiKey,
|
|
351
|
+
baseUrl: llmBaseUrl,
|
|
352
|
+
});
|
|
353
|
+
if (generated) {
|
|
354
|
+
ruleText = generated;
|
|
355
|
+
usedFallback = false;
|
|
356
|
+
metrics.llm_generated += 1;
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
catch (error) {
|
|
360
|
+
metrics.llm_failed += 1;
|
|
361
|
+
bumpReason(metrics, "llm_failed");
|
|
362
|
+
options.logger.warn(`LLM reflection failed, fallback to template rule: ${error}`);
|
|
363
|
+
}
|
|
364
|
+
}
|
|
365
|
+
if (usedFallback) {
|
|
366
|
+
metrics.fallback_used += 1;
|
|
367
|
+
}
|
|
368
|
+
const quality = validateRuleText(ruleText);
|
|
369
|
+
if (!quality.ok) {
|
|
370
|
+
if (!usedFallback) {
|
|
371
|
+
const fallbackQuality = validateRuleText(buildFallbackRule(summary, outcome));
|
|
372
|
+
if (fallbackQuality.ok) {
|
|
373
|
+
ruleText = fallbackQuality.normalized;
|
|
374
|
+
}
|
|
375
|
+
else {
|
|
376
|
+
metrics.skipped_quality_gate += 1;
|
|
377
|
+
bumpReason(metrics, fallbackQuality.reason || "skip_quality_gate");
|
|
378
|
+
continue;
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
else {
|
|
382
|
+
metrics.skipped_quality_gate += 1;
|
|
383
|
+
bumpReason(metrics, quality.reason || "skip_quality_gate");
|
|
384
|
+
continue;
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
else {
|
|
388
|
+
ruleText = quality.normalized;
|
|
389
|
+
}
|
|
77
390
|
const added = options.ruleStore.addRule({
|
|
78
391
|
sectionTitle: "Reflected Rule",
|
|
79
392
|
content: ruleText,
|
|
80
393
|
});
|
|
394
|
+
processedSet.add(eventId);
|
|
81
395
|
if (added.added) {
|
|
82
396
|
reflected += 1;
|
|
397
|
+
metrics.reflected += 1;
|
|
398
|
+
}
|
|
399
|
+
else if (added.reason === "duplicate_rule") {
|
|
400
|
+
metrics.rule_store_duplicate += 1;
|
|
401
|
+
bumpReason(metrics, "duplicate_rule");
|
|
402
|
+
}
|
|
403
|
+
else if (added.reason) {
|
|
404
|
+
bumpReason(metrics, added.reason);
|
|
83
405
|
}
|
|
84
406
|
}
|
|
85
|
-
|
|
86
|
-
|
|
407
|
+
state.processed_event_ids = [...processedSet];
|
|
408
|
+
state.total_runs += 1;
|
|
409
|
+
writeReflectState(reflectStatePath, state);
|
|
410
|
+
metrics.finished_at = nowIso();
|
|
411
|
+
appendMetrics(reflectMetricsPath, metrics);
|
|
412
|
+
options.logger.info(`TS reflector run=${metrics.run_id} scanned=${metrics.scanned} attempted=${metrics.attempted} reflected=${metrics.reflected} duplicate=${metrics.rule_store_duplicate} quality_skip=${metrics.skipped_quality_gate} llm_generated=${metrics.llm_generated} llm_failed=${metrics.llm_failed} fallback=${metrics.fallback_used}`);
|
|
413
|
+
return {
|
|
414
|
+
status: "ok",
|
|
415
|
+
message: `Reflection completed: reflected=${reflected}, attempted=${metrics.attempted}, skipped_processed=${metrics.skipped_processed}, skipped_quality=${metrics.skipped_quality_gate}`,
|
|
416
|
+
reflected_count: reflected,
|
|
417
|
+
};
|
|
87
418
|
}
|
|
88
419
|
async function promoteMemory() {
|
|
89
420
|
const activeRecords = readJsonl(activeSessionsPath).slice(-500);
|
|
90
421
|
const counter = new Map();
|
|
422
|
+
let skippedNoise = 0;
|
|
423
|
+
let skippedQuality = 0;
|
|
91
424
|
for (const record of activeRecords) {
|
|
92
425
|
const content = typeof record.content === "string" ? record.content.trim() : "";
|
|
93
426
|
if (!content) {
|
|
94
427
|
continue;
|
|
95
428
|
}
|
|
96
|
-
|
|
429
|
+
if (looksLikeNoise(content)) {
|
|
430
|
+
skippedNoise += 1;
|
|
431
|
+
continue;
|
|
432
|
+
}
|
|
433
|
+
const quality = validateRuleText(content);
|
|
434
|
+
if (!quality.ok) {
|
|
435
|
+
skippedQuality += 1;
|
|
436
|
+
continue;
|
|
437
|
+
}
|
|
438
|
+
const normalized = quality.normalized;
|
|
439
|
+
const key = normalizeCandidateKey(normalized);
|
|
440
|
+
const weight = promotionWeight(record, normalized);
|
|
441
|
+
const existing = counter.get(key);
|
|
97
442
|
if (existing) {
|
|
98
443
|
existing.count += 1;
|
|
444
|
+
existing.score += weight;
|
|
99
445
|
}
|
|
100
446
|
else {
|
|
101
|
-
counter.set(
|
|
447
|
+
counter.set(key, { content: normalized, count: 1, score: weight });
|
|
102
448
|
}
|
|
103
449
|
}
|
|
104
450
|
let promoted = 0;
|
|
105
|
-
const
|
|
451
|
+
const minOccurrences = 3;
|
|
452
|
+
const minScore = 8;
|
|
106
453
|
for (const item of counter.values()) {
|
|
107
|
-
if (item.count <
|
|
454
|
+
if (item.count < minOccurrences) {
|
|
455
|
+
continue;
|
|
456
|
+
}
|
|
457
|
+
if (item.score < minScore) {
|
|
108
458
|
continue;
|
|
109
459
|
}
|
|
110
460
|
const added = options.ruleStore.addRule({
|
|
@@ -115,7 +465,7 @@ function createReflector(options) {
|
|
|
115
465
|
promoted += 1;
|
|
116
466
|
}
|
|
117
467
|
}
|
|
118
|
-
options.logger.info(`TS reflector promoted ${promoted} rules`);
|
|
468
|
+
options.logger.info(`TS reflector promoted ${promoted} rules from ${activeRecords.length} active records (candidates=${counter.size}, skipped_noise=${skippedNoise}, skipped_quality=${skippedQuality})`);
|
|
119
469
|
return { status: "ok", promoted_count: promoted };
|
|
120
470
|
}
|
|
121
471
|
options.logger.debug(`TS reflector initialized with memory root ${memoryRoot}`);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"reflector.js","sourceRoot":"","sources":["../../../src/reflect/reflector.ts"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"reflector.js","sourceRoot":"","sources":["../../../src/reflect/reflector.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"rule_store.d.ts","sourceRoot":"","sources":["../../../src/rules/rule_store.ts"],"names":[],"mappings":"AAIA,UAAU,UAAU;IAClB,KAAK,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,CAAC;IACrD,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,CAAC;IACpD,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,CAAC;CACrD;AAMD,UAAU,gBAAgB;IACxB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,UAAU,CAAC;CACpB;
|
|
1
|
+
{"version":3,"file":"rule_store.d.ts","sourceRoot":"","sources":["../../../src/rules/rule_store.ts"],"names":[],"mappings":"AAIA,UAAU,UAAU;IAClB,KAAK,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,CAAC;IACrD,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,CAAC;IACpD,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,CAAC;CACrD;AAMD,UAAU,gBAAgB;IACxB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,UAAU,CAAC;CACpB;AAmDD,wBAAgB,eAAe,CAAC,OAAO,EAAE,gBAAgB,GAAG;IAC1D,OAAO,CAAC,IAAI,EAAE;QAAE,YAAY,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,GAAG;QAAE,KAAK,EAAE,OAAO,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC;CAC9G,CA2EA"}
|
|
@@ -37,6 +37,13 @@ exports.createRuleStore = createRuleStore;
|
|
|
37
37
|
const crypto = __importStar(require("crypto"));
|
|
38
38
|
const fs = __importStar(require("fs"));
|
|
39
39
|
const path = __importStar(require("path"));
|
|
40
|
+
const LOCK_WAIT_MS = 5000;
|
|
41
|
+
const LOCK_RETRY_MS = 25;
|
|
42
|
+
function sleepMs(ms) {
|
|
43
|
+
const sab = new SharedArrayBuffer(4);
|
|
44
|
+
const arr = new Int32Array(sab);
|
|
45
|
+
Atomics.wait(arr, 0, 0, Math.max(0, ms));
|
|
46
|
+
}
|
|
40
47
|
function readState(filePath) {
|
|
41
48
|
try {
|
|
42
49
|
if (!fs.existsSync(filePath)) {
|
|
@@ -61,7 +68,12 @@ function writeState(filePath, state) {
|
|
|
61
68
|
if (!fs.existsSync(dir)) {
|
|
62
69
|
fs.mkdirSync(dir, { recursive: true });
|
|
63
70
|
}
|
|
64
|
-
|
|
71
|
+
const normalized = {
|
|
72
|
+
hashes: [...new Set((state.hashes || []).filter(item => typeof item === "string" && item.trim()))],
|
|
73
|
+
};
|
|
74
|
+
const tempPath = `${filePath}.tmp-${process.pid}-${Date.now()}`;
|
|
75
|
+
fs.writeFileSync(tempPath, JSON.stringify(normalized, null, 2), "utf-8");
|
|
76
|
+
fs.renameSync(tempPath, filePath);
|
|
65
77
|
}
|
|
66
78
|
function normalizeRule(content) {
|
|
67
79
|
return content.replace(/\s+/g, " ").trim();
|
|
@@ -73,28 +85,75 @@ function createRuleStore(options) {
|
|
|
73
85
|
const memoryRoot = options.dbPath ? path.resolve(options.dbPath) : path.join(options.projectRoot, "data", "memory");
|
|
74
86
|
const rulesPath = path.join(memoryRoot, "CORTEX_RULES.md");
|
|
75
87
|
const statePath = path.join(memoryRoot, ".rule_store_state.json");
|
|
88
|
+
const lockPath = path.join(memoryRoot, ".rule_store.lock");
|
|
89
|
+
function withLock(run) {
|
|
90
|
+
const lockDir = path.dirname(lockPath);
|
|
91
|
+
if (!fs.existsSync(lockDir)) {
|
|
92
|
+
fs.mkdirSync(lockDir, { recursive: true });
|
|
93
|
+
}
|
|
94
|
+
const started = Date.now();
|
|
95
|
+
let lockFd = null;
|
|
96
|
+
while (Date.now() - started < LOCK_WAIT_MS) {
|
|
97
|
+
try {
|
|
98
|
+
lockFd = fs.openSync(lockPath, "wx");
|
|
99
|
+
fs.writeFileSync(lockFd, `${process.pid}:${Date.now()}`, "utf-8");
|
|
100
|
+
break;
|
|
101
|
+
}
|
|
102
|
+
catch (error) {
|
|
103
|
+
const err = error;
|
|
104
|
+
if (err?.code !== "EEXIST") {
|
|
105
|
+
options.logger.warn(`TS rule_store lock error: ${error}`);
|
|
106
|
+
return { ok: false, reason: "lock_error" };
|
|
107
|
+
}
|
|
108
|
+
sleepMs(LOCK_RETRY_MS);
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
if (lockFd === null) {
|
|
112
|
+
return { ok: false, reason: "lock_timeout" };
|
|
113
|
+
}
|
|
114
|
+
try {
|
|
115
|
+
return { ok: true, value: run() };
|
|
116
|
+
}
|
|
117
|
+
finally {
|
|
118
|
+
try {
|
|
119
|
+
fs.closeSync(lockFd);
|
|
120
|
+
}
|
|
121
|
+
catch { }
|
|
122
|
+
try {
|
|
123
|
+
fs.unlinkSync(lockPath);
|
|
124
|
+
}
|
|
125
|
+
catch { }
|
|
126
|
+
}
|
|
127
|
+
}
|
|
76
128
|
function addRule(args) {
|
|
77
129
|
const normalized = normalizeRule(args.content);
|
|
78
130
|
if (!normalized) {
|
|
79
131
|
return { added: false, reason: "empty_rule" };
|
|
80
132
|
}
|
|
81
133
|
const contentHash = hashRule(normalized);
|
|
82
|
-
const
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
fs.
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
fs.
|
|
134
|
+
const lockResult = withLock(() => {
|
|
135
|
+
const state = readState(statePath);
|
|
136
|
+
if (state.hashes.includes(contentHash)) {
|
|
137
|
+
return { added: false, reason: "duplicate_rule", hash: contentHash };
|
|
138
|
+
}
|
|
139
|
+
const ruleDir = path.dirname(rulesPath);
|
|
140
|
+
if (!fs.existsSync(ruleDir)) {
|
|
141
|
+
fs.mkdirSync(ruleDir, { recursive: true });
|
|
142
|
+
}
|
|
143
|
+
if (!fs.existsSync(rulesPath)) {
|
|
144
|
+
fs.writeFileSync(rulesPath, "# CORTEX_RULES.md\n", "utf-8");
|
|
145
|
+
}
|
|
146
|
+
const sectionTitle = (args.sectionTitle || "Rule").replace(/[\r\n]+/g, " ").trim() || "Rule";
|
|
147
|
+
fs.appendFileSync(rulesPath, `\n## ${sectionTitle}\n${normalized}\n`, "utf-8");
|
|
148
|
+
state.hashes.push(contentHash);
|
|
149
|
+
writeState(statePath, state);
|
|
150
|
+
options.logger.info(`TS rule_store appended ${sectionTitle}`);
|
|
151
|
+
return { added: true, hash: contentHash };
|
|
152
|
+
});
|
|
153
|
+
if (!lockResult.ok) {
|
|
154
|
+
return { added: false, reason: lockResult.reason };
|
|
92
155
|
}
|
|
93
|
-
|
|
94
|
-
state.hashes.push(contentHash);
|
|
95
|
-
writeState(statePath, state);
|
|
96
|
-
options.logger.info(`TS rule_store appended ${args.sectionTitle}`);
|
|
97
|
-
return { added: true, hash: contentHash };
|
|
156
|
+
return lockResult.value;
|
|
98
157
|
}
|
|
99
158
|
options.logger.debug(`TS rule_store initialized at ${rulesPath}`);
|
|
100
159
|
return { addRule };
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"rule_store.js","sourceRoot":"","sources":["../../../src/rules/rule_store.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
1
|
+
{"version":3,"file":"rule_store.js","sourceRoot":"","sources":["../../../src/rules/rule_store.ts"],"names":[],"mappings":"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"}
|