@memtensor/memos-local-openclaw-plugin 0.1.3 → 0.1.5

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 (117) hide show
  1. package/.env.example +13 -5
  2. package/README.md +283 -91
  3. package/dist/capture/index.d.ts +5 -7
  4. package/dist/capture/index.d.ts.map +1 -1
  5. package/dist/capture/index.js +72 -43
  6. package/dist/capture/index.js.map +1 -1
  7. package/dist/ingest/dedup.d.ts +8 -0
  8. package/dist/ingest/dedup.d.ts.map +1 -1
  9. package/dist/ingest/dedup.js +21 -0
  10. package/dist/ingest/dedup.js.map +1 -1
  11. package/dist/ingest/providers/anthropic.d.ts +16 -0
  12. package/dist/ingest/providers/anthropic.d.ts.map +1 -1
  13. package/dist/ingest/providers/anthropic.js +214 -1
  14. package/dist/ingest/providers/anthropic.js.map +1 -1
  15. package/dist/ingest/providers/bedrock.d.ts +16 -5
  16. package/dist/ingest/providers/bedrock.d.ts.map +1 -1
  17. package/dist/ingest/providers/bedrock.js +210 -6
  18. package/dist/ingest/providers/bedrock.js.map +1 -1
  19. package/dist/ingest/providers/gemini.d.ts +16 -0
  20. package/dist/ingest/providers/gemini.d.ts.map +1 -1
  21. package/dist/ingest/providers/gemini.js +202 -1
  22. package/dist/ingest/providers/gemini.js.map +1 -1
  23. package/dist/ingest/providers/index.d.ts +31 -0
  24. package/dist/ingest/providers/index.d.ts.map +1 -1
  25. package/dist/ingest/providers/index.js +134 -4
  26. package/dist/ingest/providers/index.js.map +1 -1
  27. package/dist/ingest/providers/openai.d.ts +24 -0
  28. package/dist/ingest/providers/openai.d.ts.map +1 -1
  29. package/dist/ingest/providers/openai.js +255 -1
  30. package/dist/ingest/providers/openai.js.map +1 -1
  31. package/dist/ingest/task-processor.d.ts +65 -0
  32. package/dist/ingest/task-processor.d.ts.map +1 -0
  33. package/dist/ingest/task-processor.js +354 -0
  34. package/dist/ingest/task-processor.js.map +1 -0
  35. package/dist/ingest/worker.d.ts +3 -1
  36. package/dist/ingest/worker.d.ts.map +1 -1
  37. package/dist/ingest/worker.js +131 -23
  38. package/dist/ingest/worker.js.map +1 -1
  39. package/dist/recall/engine.d.ts +1 -0
  40. package/dist/recall/engine.d.ts.map +1 -1
  41. package/dist/recall/engine.js +22 -11
  42. package/dist/recall/engine.js.map +1 -1
  43. package/dist/recall/mmr.d.ts.map +1 -1
  44. package/dist/recall/mmr.js +3 -1
  45. package/dist/recall/mmr.js.map +1 -1
  46. package/dist/skill/bundled-memory-guide.d.ts +6 -0
  47. package/dist/skill/bundled-memory-guide.d.ts.map +1 -0
  48. package/dist/skill/bundled-memory-guide.js +95 -0
  49. package/dist/skill/bundled-memory-guide.js.map +1 -0
  50. package/dist/skill/evaluator.d.ts +31 -0
  51. package/dist/skill/evaluator.d.ts.map +1 -0
  52. package/dist/skill/evaluator.js +194 -0
  53. package/dist/skill/evaluator.js.map +1 -0
  54. package/dist/skill/evolver.d.ts +22 -0
  55. package/dist/skill/evolver.d.ts.map +1 -0
  56. package/dist/skill/evolver.js +193 -0
  57. package/dist/skill/evolver.js.map +1 -0
  58. package/dist/skill/generator.d.ts +25 -0
  59. package/dist/skill/generator.d.ts.map +1 -0
  60. package/dist/skill/generator.js +477 -0
  61. package/dist/skill/generator.js.map +1 -0
  62. package/dist/skill/installer.d.ts +16 -0
  63. package/dist/skill/installer.d.ts.map +1 -0
  64. package/dist/skill/installer.js +89 -0
  65. package/dist/skill/installer.js.map +1 -0
  66. package/dist/skill/upgrader.d.ts +19 -0
  67. package/dist/skill/upgrader.d.ts.map +1 -0
  68. package/dist/skill/upgrader.js +263 -0
  69. package/dist/skill/upgrader.js.map +1 -0
  70. package/dist/skill/validator.d.ts +29 -0
  71. package/dist/skill/validator.d.ts.map +1 -0
  72. package/dist/skill/validator.js +227 -0
  73. package/dist/skill/validator.js.map +1 -0
  74. package/dist/storage/sqlite.d.ts +141 -1
  75. package/dist/storage/sqlite.d.ts.map +1 -1
  76. package/dist/storage/sqlite.js +664 -7
  77. package/dist/storage/sqlite.js.map +1 -1
  78. package/dist/types.d.ts +93 -0
  79. package/dist/types.d.ts.map +1 -1
  80. package/dist/types.js +8 -0
  81. package/dist/types.js.map +1 -1
  82. package/dist/viewer/html.d.ts +1 -1
  83. package/dist/viewer/html.d.ts.map +1 -1
  84. package/dist/viewer/html.js +2391 -159
  85. package/dist/viewer/html.js.map +1 -1
  86. package/dist/viewer/server.d.ts +16 -0
  87. package/dist/viewer/server.d.ts.map +1 -1
  88. package/dist/viewer/server.js +346 -3
  89. package/dist/viewer/server.js.map +1 -1
  90. package/index.ts +572 -89
  91. package/openclaw.plugin.json +20 -45
  92. package/package.json +3 -4
  93. package/skill/memos-memory-guide/SKILL.md +86 -0
  94. package/src/capture/index.ts +85 -45
  95. package/src/ingest/dedup.ts +29 -0
  96. package/src/ingest/providers/anthropic.ts +258 -1
  97. package/src/ingest/providers/bedrock.ts +256 -6
  98. package/src/ingest/providers/gemini.ts +252 -1
  99. package/src/ingest/providers/index.ts +156 -8
  100. package/src/ingest/providers/openai.ts +304 -1
  101. package/src/ingest/task-processor.ts +396 -0
  102. package/src/ingest/worker.ts +145 -34
  103. package/src/recall/engine.ts +23 -12
  104. package/src/recall/mmr.ts +3 -1
  105. package/src/skill/bundled-memory-guide.ts +91 -0
  106. package/src/skill/evaluator.ts +220 -0
  107. package/src/skill/evolver.ts +169 -0
  108. package/src/skill/generator.ts +506 -0
  109. package/src/skill/installer.ts +59 -0
  110. package/src/skill/upgrader.ts +257 -0
  111. package/src/skill/validator.ts +227 -0
  112. package/src/storage/sqlite.ts +802 -7
  113. package/src/types.ts +96 -0
  114. package/src/viewer/html.ts +2391 -159
  115. package/src/viewer/server.ts +346 -3
  116. package/SKILL.md +0 -43
  117. package/www/index.html +0 -632
@@ -0,0 +1,354 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TaskProcessor = void 0;
4
+ const uuid_1 = require("uuid");
5
+ const types_1 = require("../types");
6
+ const providers_1 = require("./providers");
7
+ const TRIVIAL_PATTERNS = [
8
+ /^(test|testing|hello|hi|hey|ok|okay|yes|no|yeah|nope|sure|thanks|thank you|thx|ping|pong|哈哈|好的|嗯|是的|不是|谢谢|你好|测试)\s*[.!?。!?]*$/,
9
+ /^(aaa+|bbb+|xxx+|zzz+|123+|asdf+|qwer+|haha+|lol+|hmm+)\s*$/,
10
+ /^[\s\p{P}\p{S}]*$/u,
11
+ ];
12
+ const SKIP_REASONS = {
13
+ noChunks: "该任务没有对话内容,已自动跳过。",
14
+ };
15
+ /**
16
+ * Asynchronous task-level processor.
17
+ *
18
+ * After each ingestion batch, checks whether the current conversation
19
+ * constitutes a "new task" compared to the previous one. If so:
20
+ * 1. Finalizes the previous task (generates a detailed summary).
21
+ * 2. Creates a new active task for incoming chunks.
22
+ *
23
+ * Task boundary detection:
24
+ * - Session change → always new task
25
+ * - Time gap > 2h → always new task
26
+ * - LLM judges whether new user message starts a different topic
27
+ */
28
+ class TaskProcessor {
29
+ store;
30
+ ctx;
31
+ summarizer;
32
+ processing = false;
33
+ onTaskCompletedCallback;
34
+ constructor(store, ctx) {
35
+ this.store = store;
36
+ this.ctx = ctx;
37
+ this.summarizer = new providers_1.Summarizer(ctx.config.summarizer, ctx.log);
38
+ }
39
+ onTaskCompleted(cb) {
40
+ this.onTaskCompletedCallback = cb;
41
+ }
42
+ /**
43
+ * Called after new chunks are ingested.
44
+ * Determines if a new task boundary was crossed and handles transition.
45
+ */
46
+ async onChunksIngested(sessionKey, latestTimestamp) {
47
+ this.ctx.log.debug(`TaskProcessor.onChunksIngested called session=${sessionKey} ts=${latestTimestamp} processing=${this.processing}`);
48
+ if (this.processing) {
49
+ this.ctx.log.debug("TaskProcessor.onChunksIngested skipped — already processing");
50
+ return;
51
+ }
52
+ this.processing = true;
53
+ try {
54
+ await this.detectAndProcess(sessionKey, latestTimestamp);
55
+ }
56
+ catch (err) {
57
+ this.ctx.log.error(`TaskProcessor error: ${err}`);
58
+ }
59
+ finally {
60
+ this.processing = false;
61
+ }
62
+ }
63
+ async detectAndProcess(sessionKey, latestTimestamp) {
64
+ this.ctx.log.debug(`TaskProcessor.detectAndProcess session=${sessionKey}`);
65
+ // Finalize any active tasks from OTHER sessions (session change = task boundary)
66
+ const allActive = this.store.getAllActiveTasks();
67
+ for (const t of allActive) {
68
+ if (t.sessionKey !== sessionKey) {
69
+ this.ctx.log.info(`Session changed: finalizing task=${t.id} from session=${t.sessionKey}`);
70
+ await this.finalizeTask(t);
71
+ }
72
+ }
73
+ const activeTask = this.store.getActiveTask(sessionKey);
74
+ this.ctx.log.debug(`TaskProcessor.detectAndProcess activeTask=${activeTask?.id ?? "none"}`);
75
+ if (!activeTask) {
76
+ await this.createNewTask(sessionKey, latestTimestamp);
77
+ return;
78
+ }
79
+ const isNewTask = await this.isTaskBoundary(activeTask, sessionKey, latestTimestamp);
80
+ if (isNewTask) {
81
+ await this.finalizeTask(activeTask);
82
+ await this.createNewTask(sessionKey, latestTimestamp);
83
+ }
84
+ else {
85
+ this.assignUnassignedChunks(sessionKey, activeTask.id);
86
+ this.store.updateTask(activeTask.id, { endedAt: undefined });
87
+ }
88
+ }
89
+ async isTaskBoundary(activeTask, sessionKey, latestTimestamp) {
90
+ if (activeTask.sessionKey !== sessionKey)
91
+ return true;
92
+ const chunks = this.store.getChunksByTask(activeTask.id);
93
+ if (chunks.length === 0)
94
+ return false;
95
+ const lastChunkTs = Math.max(...chunks.map((c) => c.createdAt));
96
+ const gap = latestTimestamp - lastChunkTs;
97
+ // Hard timeout: always split after 2h regardless of topic
98
+ if (gap > types_1.DEFAULTS.taskIdleTimeoutMs) {
99
+ this.ctx.log.info(`Task boundary: time gap ${Math.round(gap / 60000)}min > ${Math.round(types_1.DEFAULTS.taskIdleTimeoutMs / 60000)}min`);
100
+ return true;
101
+ }
102
+ // LLM topic judgment: build context from existing task and compare with new message
103
+ const newUserChunks = this.store.getUnassignedChunks(sessionKey).filter((c) => c.role === "user");
104
+ if (newUserChunks.length === 0)
105
+ return false;
106
+ const existingUserChunks = chunks.filter((c) => c.role === "user");
107
+ if (existingUserChunks.length === 0)
108
+ return false;
109
+ const currentContext = this.buildContextSummary(chunks);
110
+ const newMessage = newUserChunks.map((c) => c.content).join("\n");
111
+ const isNew = await this.summarizer.judgeNewTopic(currentContext, newMessage);
112
+ if (isNew === null) {
113
+ this.ctx.log.debug("Topic judge unavailable (no LLM configured), keeping current task");
114
+ return false;
115
+ }
116
+ if (isNew) {
117
+ this.ctx.log.info(`Task boundary: LLM judged new topic. New message: "${newMessage.slice(0, 80)}..."`);
118
+ }
119
+ else {
120
+ this.ctx.log.debug(`LLM judged SAME topic, continuing task=${activeTask.id}`);
121
+ }
122
+ return isNew;
123
+ }
124
+ /**
125
+ * Build a concise context string from existing task chunks for the LLM topic judge.
126
+ * Takes recent user/assistant summaries to keep token usage low.
127
+ */
128
+ buildContextSummary(chunks) {
129
+ const relevant = chunks
130
+ .filter((c) => c.role === "user" || c.role === "assistant")
131
+ .slice(-6);
132
+ return relevant
133
+ .map((c) => `[${c.role === "user" ? "User" : "Assistant"}]: ${c.summary || c.content.slice(0, 150)}`)
134
+ .join("\n");
135
+ }
136
+ async createNewTask(sessionKey, timestamp) {
137
+ const taskId = (0, uuid_1.v4)();
138
+ const task = {
139
+ id: taskId,
140
+ sessionKey,
141
+ title: "",
142
+ summary: "",
143
+ status: "active",
144
+ startedAt: timestamp,
145
+ endedAt: null,
146
+ updatedAt: timestamp,
147
+ };
148
+ this.store.insertTask(task);
149
+ this.assignUnassignedChunks(sessionKey, taskId);
150
+ this.ctx.log.info(`Created new task=${taskId} session=${sessionKey}`);
151
+ }
152
+ assignUnassignedChunks(sessionKey, taskId) {
153
+ const unassigned = this.store.getUnassignedChunks(sessionKey);
154
+ for (const chunk of unassigned) {
155
+ this.store.setChunkTaskId(chunk.id, taskId);
156
+ }
157
+ if (unassigned.length > 0) {
158
+ this.ctx.log.debug(`Assigned ${unassigned.length} chunks to task=${taskId}`);
159
+ }
160
+ }
161
+ async finalizeTask(task) {
162
+ const chunks = this.store.getChunksByTask(task.id);
163
+ const fallbackTitle = chunks.length > 0 ? this.extractTitle(chunks) : "";
164
+ if (chunks.length === 0) {
165
+ this.ctx.log.info(`Task ${task.id} skipped: no chunks`);
166
+ this.store.updateTask(task.id, { title: fallbackTitle, summary: SKIP_REASONS.noChunks, status: "skipped", endedAt: Date.now() });
167
+ return;
168
+ }
169
+ const skipReason = this.shouldSkipSummary(chunks);
170
+ if (skipReason) {
171
+ this.ctx.log.info(`Task ${task.id} skipped: ${skipReason} (chunks=${chunks.length}, title="${fallbackTitle}")`);
172
+ const reason = this.humanReadableSkipReason(skipReason, chunks);
173
+ this.store.updateTask(task.id, { title: fallbackTitle, summary: reason, status: "skipped", endedAt: Date.now() });
174
+ return;
175
+ }
176
+ const conversationText = this.buildConversationText(chunks);
177
+ let summary;
178
+ try {
179
+ summary = await this.summarizer.summarizeTask(conversationText);
180
+ }
181
+ catch (err) {
182
+ this.ctx.log.warn(`Task summary generation failed for task=${task.id}: ${err}`);
183
+ summary = this.fallbackSummary(chunks);
184
+ }
185
+ const { title: llmTitle, body } = this.parseTitleFromSummary(summary);
186
+ const title = llmTitle || fallbackTitle;
187
+ this.store.updateTask(task.id, {
188
+ title,
189
+ summary: body,
190
+ status: "completed",
191
+ endedAt: Date.now(),
192
+ });
193
+ this.ctx.log.info(`Finalized task=${task.id} title="${title}" chunks=${chunks.length} summaryLen=${body.length}`);
194
+ if (this.onTaskCompletedCallback) {
195
+ const finalized = this.store.getTask(task.id);
196
+ if (finalized) {
197
+ try {
198
+ this.onTaskCompletedCallback(finalized);
199
+ }
200
+ catch (err) {
201
+ this.ctx.log.warn(`TaskProcessor onTaskCompleted callback error: ${err}`);
202
+ }
203
+ }
204
+ }
205
+ }
206
+ /**
207
+ * Determine if a task is too trivial to warrant an LLM summary call.
208
+ * Returns a skip reason string, or null if summary should proceed.
209
+ *
210
+ * Skip conditions (any one triggers skip):
211
+ * 1. Total chunks < 4 — too few messages to form a meaningful task
212
+ * 2. Real conversation turns < 2 — no back-and-forth dialogue
213
+ * 3. No user messages — purely system/tool generated, no user intent
214
+ * 4. Total content < 200 chars — not enough substance
215
+ * 5. User content is trivial/test data — "hello", "test", "ok" etc.
216
+ * 6. All messages are tool results — automated output, no conversation
217
+ * 7. High content repetition — user repeated the same thing (debug loops)
218
+ */
219
+ shouldSkipSummary(chunks) {
220
+ const userChunks = chunks.filter((c) => c.role === "user");
221
+ const assistantChunks = chunks.filter((c) => c.role === "assistant");
222
+ const toolChunks = chunks.filter((c) => c.role === "tool");
223
+ // 1. Too few chunks
224
+ if (chunks.length < 4) {
225
+ return `too few chunks (${chunks.length} < 4 minimum)`;
226
+ }
227
+ // 2. Not enough real conversation turns (need at least 2 user-assistant exchanges)
228
+ const turns = Math.min(userChunks.length, assistantChunks.length);
229
+ if (turns < 2) {
230
+ return `too few conversation turns (${turns} < 2 minimum)`;
231
+ }
232
+ // 3. No user messages at all — purely automated
233
+ if (userChunks.length === 0) {
234
+ return "no user messages — task appears to be automated/system-generated";
235
+ }
236
+ // 4. Total content too short
237
+ // CJK characters carry more info per char, so use a lower threshold
238
+ const totalContentLen = chunks.reduce((sum, c) => sum + c.content.length, 0);
239
+ const hasCJK = /[\u4e00-\u9fff\u3040-\u30ff\uac00-\ud7af]/.test(userChunks[0]?.content ?? "");
240
+ const minContentLen = hasCJK ? 80 : 200;
241
+ if (totalContentLen < minContentLen) {
242
+ return `content too short (${totalContentLen} chars < ${minContentLen} minimum)`;
243
+ }
244
+ // 5. User content is trivial/test data
245
+ const userContent = userChunks.map((c) => c.content).join("\n");
246
+ if (this.looksLikeTrivialContent(userContent)) {
247
+ return "user content appears to be test/trivial data";
248
+ }
249
+ // 6. Assistant content is also trivial (both sides are low-value)
250
+ const assistantContent = assistantChunks.map((c) => c.content).join("\n");
251
+ if (this.looksLikeTrivialContent(userContent + "\n" + assistantContent)) {
252
+ return "conversation content (both user and assistant) appears trivial";
253
+ }
254
+ // 7. Almost all messages are tool results with minimal user interaction
255
+ if (toolChunks.length > 0 && toolChunks.length >= chunks.length * 0.7 && userChunks.length <= 1) {
256
+ return `dominated by tool results (${toolChunks.length}/${chunks.length} chunks) with minimal user input`;
257
+ }
258
+ // 8. High repetition — user keeps saying the same thing
259
+ if (userChunks.length >= 3) {
260
+ const uniqueUserMsgs = new Set(userChunks.map((c) => c.content.trim().toLowerCase()));
261
+ const uniqueRatio = uniqueUserMsgs.size / userChunks.length;
262
+ if (uniqueRatio < 0.4) {
263
+ return `high content repetition (${uniqueUserMsgs.size} unique out of ${userChunks.length} user messages)`;
264
+ }
265
+ }
266
+ return null;
267
+ }
268
+ looksLikeTrivialContent(text) {
269
+ const lines = text.toLowerCase().split(/\n/).map((l) => l.trim()).filter(Boolean);
270
+ if (lines.length === 0)
271
+ return true;
272
+ const trivialCount = lines.filter((line) => {
273
+ if (line.length < 5)
274
+ return true;
275
+ if (TRIVIAL_PATTERNS.some((p) => p.test(line)))
276
+ return true;
277
+ return false;
278
+ }).length;
279
+ return trivialCount / lines.length > 0.7;
280
+ }
281
+ buildConversationText(chunks) {
282
+ const lines = [];
283
+ for (const c of chunks) {
284
+ const roleLabel = c.role === "user" ? "User" : c.role === "assistant" ? "Assistant" : c.role;
285
+ lines.push(`[${roleLabel}]: ${c.content}`);
286
+ }
287
+ return lines.join("\n\n");
288
+ }
289
+ /**
290
+ * Extract the LLM-generated title from the summary output.
291
+ * The LLM is prompted to output "📌 Title\n<title text>" as the first section.
292
+ * Returns the title and the remaining body (with the title section stripped).
293
+ */
294
+ parseTitleFromSummary(summary) {
295
+ const titleMatch = summary.match(/📌\s*(?:Title|标题)\s*\n(.+)/);
296
+ if (titleMatch) {
297
+ const title = titleMatch[1].trim().slice(0, 80);
298
+ const body = summary.replace(/📌\s*(?:Title|标题)\s*\n.+\n?/, "").trim();
299
+ return { title, body };
300
+ }
301
+ return { title: "", body: summary };
302
+ }
303
+ extractTitle(chunks) {
304
+ const firstUser = chunks.find((c) => c.role === "user");
305
+ if (!firstUser)
306
+ return "Untitled Task";
307
+ const text = firstUser.content.trim();
308
+ if (text.length <= 60)
309
+ return text;
310
+ return text.slice(0, 57) + "...";
311
+ }
312
+ humanReadableSkipReason(reason, chunks) {
313
+ const userCount = chunks.filter((c) => c.role === "user").length;
314
+ const assistantCount = chunks.filter((c) => c.role === "assistant").length;
315
+ if (reason.includes("too few chunks")) {
316
+ return `对话内容过少(${chunks.length} 条消息),不足以生成有效摘要。至少需要 4 条消息。`;
317
+ }
318
+ if (reason.includes("too few conversation turns")) {
319
+ return `对话轮次不足(${Math.min(userCount, assistantCount)} 轮),需要至少 2 轮完整的问答交互才能生成摘要。`;
320
+ }
321
+ if (reason.includes("no user messages")) {
322
+ return "该任务没有用户消息,仅包含系统或工具自动生成的内容。";
323
+ }
324
+ if (reason.includes("content too short")) {
325
+ return "对话内容过短,信息量不足以生成有意义的摘要。";
326
+ }
327
+ if (reason.includes("trivial")) {
328
+ return "对话内容为简单问候或测试数据(如 hello、test、ok),无需生成摘要。";
329
+ }
330
+ if (reason.includes("tool results")) {
331
+ return "该任务主要由工具执行结果组成,缺少足够的用户交互内容。";
332
+ }
333
+ if (reason.includes("repetition")) {
334
+ return "对话中存在大量重复内容,无法提取有效信息生成摘要。";
335
+ }
336
+ return `对话未达到生成摘要的条件:${reason}`;
337
+ }
338
+ fallbackSummary(chunks) {
339
+ const title = this.extractTitle(chunks);
340
+ const summaries = chunks
341
+ .filter((c) => c.summary)
342
+ .map((c) => `- ${c.summary}`);
343
+ const lines = [
344
+ `🎯 Goal`,
345
+ title,
346
+ ``,
347
+ `📋 Key Steps`,
348
+ ...summaries.slice(0, 20),
349
+ ];
350
+ return lines.join("\n");
351
+ }
352
+ }
353
+ exports.TaskProcessor = TaskProcessor;
354
+ //# sourceMappingURL=task-processor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task-processor.js","sourceRoot":"","sources":["../../src/ingest/task-processor.ts"],"names":[],"mappings":"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"}
@@ -1,21 +1,23 @@
1
1
  import type { ConversationMessage, PluginContext } from "../types";
2
2
  import type { SqliteStore } from "../storage/sqlite";
3
3
  import type { Embedder } from "../embedding";
4
+ import { TaskProcessor } from "./task-processor";
4
5
  export declare class IngestWorker {
5
6
  private store;
6
7
  private embedder;
7
8
  private ctx;
8
9
  private summarizer;
10
+ private taskProcessor;
9
11
  private queue;
10
12
  private processing;
11
13
  private flushResolvers;
12
14
  constructor(store: SqliteStore, embedder: Embedder, ctx: PluginContext);
15
+ getTaskProcessor(): TaskProcessor;
13
16
  enqueue(messages: ConversationMessage[]): void;
14
17
  /** Wait until all queued messages have been processed. */
15
18
  flush(): Promise<void>;
16
19
  private processQueue;
17
20
  private ingestMessage;
18
- private ingestToolResult;
19
21
  private storeChunk;
20
22
  }
21
23
  //# sourceMappingURL=worker.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"worker.d.ts","sourceRoot":"","sources":["../../src/ingest/worker.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,mBAAmB,EAAS,aAAa,EAAE,MAAM,UAAU,CAAC;AAC1E,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AACrD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAC;AAK7C,qBAAa,YAAY;IAOrB,OAAO,CAAC,KAAK;IACb,OAAO,CAAC,QAAQ;IAChB,OAAO,CAAC,GAAG;IARb,OAAO,CAAC,UAAU,CAAa;IAC/B,OAAO,CAAC,KAAK,CAA6B;IAC1C,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,cAAc,CAAyB;gBAGrC,KAAK,EAAE,WAAW,EAClB,QAAQ,EAAE,QAAQ,EAClB,GAAG,EAAE,aAAa;IAK5B,OAAO,CAAC,QAAQ,EAAE,mBAAmB,EAAE,GAAG,IAAI;IAU9C,0DAA0D;IACpD,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;YAOd,YAAY;YAiBZ,aAAa;YAeb,gBAAgB;YAKhB,UAAU;CAoDzB"}
1
+ {"version":3,"file":"worker.d.ts","sourceRoot":"","sources":["../../src/ingest/worker.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,mBAAmB,EAAS,aAAa,EAAE,MAAM,UAAU,CAAC;AAC1E,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AACrD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAC;AAG7C,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AAEjD,qBAAa,YAAY;IAQrB,OAAO,CAAC,KAAK;IACb,OAAO,CAAC,QAAQ;IAChB,OAAO,CAAC,GAAG;IATb,OAAO,CAAC,UAAU,CAAa;IAC/B,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,KAAK,CAA6B;IAC1C,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,cAAc,CAAyB;gBAGrC,KAAK,EAAE,WAAW,EAClB,QAAQ,EAAE,QAAQ,EAClB,GAAG,EAAE,aAAa;IAM5B,gBAAgB,IAAI,aAAa;IAEjC,OAAO,CAAC,QAAQ,EAAE,mBAAmB,EAAE,GAAG,IAAI;IAU9C,0DAA0D;IACpD,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;YAOd,YAAY;YAqEZ,aAAa;YAYb,UAAU;CAkHzB"}
@@ -3,13 +3,14 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.IngestWorker = void 0;
4
4
  const uuid_1 = require("uuid");
5
5
  const providers_1 = require("./providers");
6
- const chunker_1 = require("./chunker");
7
6
  const dedup_1 = require("./dedup");
7
+ const task_processor_1 = require("./task-processor");
8
8
  class IngestWorker {
9
9
  store;
10
10
  embedder;
11
11
  ctx;
12
12
  summarizer;
13
+ taskProcessor;
13
14
  queue = [];
14
15
  processing = false;
15
16
  flushResolvers = [];
@@ -18,7 +19,9 @@ class IngestWorker {
18
19
  this.embedder = embedder;
19
20
  this.ctx = ctx;
20
21
  this.summarizer = new providers_1.Summarizer(ctx.config.summarizer, ctx.log);
22
+ this.taskProcessor = new task_processor_1.TaskProcessor(store, ctx);
21
23
  }
24
+ getTaskProcessor() { return this.taskProcessor; }
22
25
  enqueue(messages) {
23
26
  this.queue.push(...messages);
24
27
  if (!this.processing) {
@@ -38,35 +41,79 @@ class IngestWorker {
38
41
  }
39
42
  async processQueue() {
40
43
  this.processing = true;
44
+ const t0 = performance.now();
45
+ let lastSessionKey;
46
+ let lastTimestamp = 0;
47
+ let stored = 0;
48
+ let skipped = 0;
49
+ let merged = 0;
50
+ let duplicated = 0;
51
+ let errors = 0;
52
+ const resultLines = [];
53
+ const inputLines = [];
54
+ const totalMessages = this.queue.length;
41
55
  while (this.queue.length > 0) {
42
56
  const msg = this.queue.shift();
57
+ inputLines.push(`[${msg.role}] ${msg.content}`);
43
58
  try {
44
- await this.ingestMessage(msg);
59
+ const result = await this.ingestMessage(msg);
60
+ lastSessionKey = msg.sessionKey;
61
+ lastTimestamp = Math.max(lastTimestamp, msg.timestamp);
62
+ if (result === "skipped") {
63
+ skipped++;
64
+ resultLines.push(`[${msg.role}] ⏭ exact-dup → ${msg.content}`);
65
+ }
66
+ else if (result.action === "stored") {
67
+ stored++;
68
+ resultLines.push(`[${msg.role}] ✅ stored → ${result.summary ?? msg.content}`);
69
+ }
70
+ else if (result.action === "duplicate") {
71
+ duplicated++;
72
+ resultLines.push(`[${msg.role}] 🔁 dedup(${result.reason ?? "similar"}) → ${msg.content}`);
73
+ }
74
+ else if (result.action === "merged") {
75
+ merged++;
76
+ resultLines.push(`[${msg.role}] 🔀 merged → ${msg.content}`);
77
+ }
45
78
  }
46
79
  catch (err) {
80
+ errors++;
81
+ resultLines.push(`[${msg.role}] ❌ error → ${msg.content}`);
47
82
  this.ctx.log.error(`Failed to ingest message turn=${msg.turnId}: ${err}`);
48
83
  }
49
84
  }
85
+ const dur = performance.now() - t0;
86
+ if (stored + merged > 0 || skipped > 0 || duplicated > 0) {
87
+ this.store.recordToolCall("memory_add", dur, errors === 0);
88
+ try {
89
+ const inputInfo = {
90
+ session: lastSessionKey,
91
+ messages: totalMessages,
92
+ details: inputLines,
93
+ };
94
+ const stats = [`stored=${stored}`, skipped > 0 ? `skipped=${skipped}` : null, duplicated > 0 ? `dedup=${duplicated}` : null, merged > 0 ? `merged=${merged}` : null, errors > 0 ? `errors=${errors}` : null].filter(Boolean).join(", ");
95
+ this.store.recordApiLog("memory_add", inputInfo, `${stats}\n${resultLines.join("\n")}`, dur, errors === 0);
96
+ }
97
+ catch (_) { /* best-effort */ }
98
+ }
99
+ if (lastSessionKey) {
100
+ this.ctx.log.debug(`Calling TaskProcessor.onChunksIngested session=${lastSessionKey} ts=${lastTimestamp}`);
101
+ this.taskProcessor
102
+ .onChunksIngested(lastSessionKey, lastTimestamp)
103
+ .catch((err) => this.ctx.log.error(`TaskProcessor post-ingest error: ${err}`));
104
+ }
50
105
  this.processing = false;
51
106
  for (const resolve of this.flushResolvers)
52
107
  resolve();
53
108
  this.flushResolvers = [];
54
109
  }
55
110
  async ingestMessage(msg) {
56
- if (msg.role === "tool") {
57
- await this.ingestToolResult(msg);
58
- return;
111
+ if (this.store.chunkExistsByContent(msg.sessionKey, msg.role, msg.content)) {
112
+ this.ctx.log.debug(`Exact-dup (same session+role+hash), skipping: session=${msg.sessionKey} role=${msg.role} len=${msg.content.length}`);
113
+ return "skipped";
59
114
  }
60
- const rawChunks = (0, chunker_1.chunkText)(msg.content);
61
- this.ctx.log.debug(`Chunked turn=${msg.turnId} into ${rawChunks.length} chunks`);
62
- for (let seq = 0; seq < rawChunks.length; seq++) {
63
- const raw = rawChunks[seq];
64
- await this.storeChunk(msg, raw.content, raw.kind, seq);
65
- }
66
- }
67
- async ingestToolResult(msg) {
68
- this.ctx.log.debug(`Ingesting tool result turn=${msg.turnId} tool=${msg.toolName ?? "unknown"} len=${msg.content.length}`);
69
- await this.storeChunk(msg, msg.content, "tool_result", 0);
115
+ const kind = msg.role === "tool" ? "tool_result" : "paragraph";
116
+ return await this.storeChunk(msg, msg.content, kind, 0);
70
117
  }
71
118
  async storeChunk(msg, content, kind, seq) {
72
119
  const chunkId = (0, uuid_1.v4)();
@@ -78,13 +125,59 @@ class IngestWorker {
78
125
  catch (err) {
79
126
  this.ctx.log.warn(`Embedding failed for chunk=${chunkId}, storing without vector: ${err}`);
80
127
  }
128
+ let dedupStatus = "active";
129
+ let dedupTarget = null;
130
+ let dedupReason = null;
131
+ // Smart dedup: find Top-5 similar chunks, then ask LLM to judge
81
132
  if (embedding) {
82
- const dupId = (0, dedup_1.findDuplicate)(this.store, embedding, this.ctx.config.dedup?.similarityThreshold ?? 0.93, this.ctx.log);
83
- if (dupId) {
84
- this.store.updateSummary(dupId, summary);
85
- this.store.upsertEmbedding(dupId, embedding);
86
- this.ctx.log.debug(`Dedup-merged into existing chunk=${dupId}`);
87
- return;
133
+ const similarThreshold = this.ctx.config.dedup?.similarityThreshold ?? 0.75;
134
+ const topSimilar = (0, dedup_1.findTopSimilar)(this.store, embedding, similarThreshold, 5, this.ctx.log);
135
+ if (topSimilar.length > 0) {
136
+ const candidates = topSimilar.map((s, i) => {
137
+ const chunk = this.store.getChunk(s.chunkId);
138
+ return {
139
+ index: i + 1,
140
+ summary: chunk?.summary ?? "",
141
+ chunkId: s.chunkId,
142
+ };
143
+ }).filter(c => c.summary);
144
+ if (candidates.length > 0) {
145
+ const dedupResult = await this.summarizer.judgeDedup(summary, candidates);
146
+ if (dedupResult && dedupResult.action === "DUPLICATE" && dedupResult.targetIndex) {
147
+ const targetChunkId = candidates[dedupResult.targetIndex - 1]?.chunkId;
148
+ if (targetChunkId) {
149
+ this.store.recordMergeHit(targetChunkId, "DUPLICATE", dedupResult.reason);
150
+ dedupStatus = "duplicate";
151
+ dedupTarget = targetChunkId;
152
+ dedupReason = dedupResult.reason;
153
+ this.ctx.log.debug(`Smart dedup: DUPLICATE → target=${targetChunkId}, storing with status=duplicate, reason: ${dedupResult.reason}`);
154
+ }
155
+ }
156
+ if (dedupStatus === "active" && dedupResult && dedupResult.action === "UPDATE" && dedupResult.targetIndex && dedupResult.mergedSummary) {
157
+ const targetChunkId = candidates[dedupResult.targetIndex - 1]?.chunkId;
158
+ if (targetChunkId) {
159
+ const oldChunk = this.store.getChunk(targetChunkId);
160
+ const oldSummary = oldChunk?.summary ?? "";
161
+ this.store.recordMergeHit(targetChunkId, "UPDATE", dedupResult.reason, oldSummary, dedupResult.mergedSummary);
162
+ this.store.updateChunkSummaryAndContent(targetChunkId, dedupResult.mergedSummary, content);
163
+ try {
164
+ const [newEmb] = await this.embedder.embed([dedupResult.mergedSummary]);
165
+ if (newEmb)
166
+ this.store.upsertEmbedding(targetChunkId, newEmb);
167
+ }
168
+ catch (err) {
169
+ this.ctx.log.warn(`Re-embed after UPDATE failed: ${err}`);
170
+ }
171
+ dedupStatus = "merged";
172
+ dedupTarget = targetChunkId;
173
+ dedupReason = dedupResult.reason;
174
+ this.ctx.log.debug(`Smart dedup: UPDATE → merged into chunk=${targetChunkId}, storing with status=merged, reason: ${dedupResult.reason}`);
175
+ }
176
+ }
177
+ if (dedupStatus === "active") {
178
+ this.ctx.log.debug(`Smart dedup: NEW — creating active chunk (reason: ${dedupResult?.reason ?? "no_result"})`);
179
+ }
180
+ }
88
181
  }
89
182
  }
90
183
  const chunk = {
@@ -97,14 +190,29 @@ class IngestWorker {
97
190
  kind,
98
191
  summary,
99
192
  embedding: null,
193
+ taskId: null,
194
+ skillId: null,
195
+ dedupStatus,
196
+ dedupTarget,
197
+ dedupReason,
198
+ mergeCount: 0,
199
+ lastHitAt: null,
200
+ mergeHistory: "[]",
100
201
  createdAt: msg.timestamp,
101
202
  updatedAt: msg.timestamp,
102
203
  };
103
204
  this.store.insertChunk(chunk);
104
- if (embedding) {
205
+ if (embedding && dedupStatus === "active") {
105
206
  this.store.upsertEmbedding(chunkId, embedding);
106
207
  }
107
- this.ctx.log.debug(`Stored chunk=${chunkId} kind=${kind} role=${msg.role} len=${content.length} hasVec=${!!embedding}`);
208
+ this.ctx.log.debug(`Stored chunk=${chunkId} kind=${kind} role=${msg.role} dedup=${dedupStatus} len=${content.length} hasVec=${!!embedding && dedupStatus === "active"}`);
209
+ if (dedupStatus === "duplicate") {
210
+ return { action: "duplicate", summary, targetChunkId: dedupTarget ?? undefined, reason: dedupReason ?? undefined };
211
+ }
212
+ if (dedupStatus === "merged") {
213
+ return { action: "merged", summary, targetChunkId: dedupTarget ?? undefined, reason: dedupReason ?? undefined };
214
+ }
215
+ return { action: "stored", chunkId, summary };
108
216
  }
109
217
  }
110
218
  exports.IngestWorker = IngestWorker;