memory-bank-skill 5.9.0 → 5.11.0

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 CHANGED
@@ -192,5 +192,5 @@ service=memory-bank Plugin initialized (unified) {"projectRoot":"..."}
192
192
 
193
193
  ## 版本
194
194
 
195
- - **版本**: 5.8.0
196
- - **主要更新**: 修复 skill 路径(`skill/` → `skills/`)+ 写入守卫 + Writer Agent 自动注册
195
+ - **版本**: 5.11.0
196
+ - **主要更新**: Writer 自动清理(目录文件数检查 + active.md 归档)+ 确认职责分离
package/dist/cli.js CHANGED
@@ -10,7 +10,7 @@ import { fileURLToPath } from "url";
10
10
  // package.json
11
11
  var package_default = {
12
12
  name: "memory-bank-skill",
13
- version: "5.9.0",
13
+ version: "5.11.0",
14
14
  description: "Memory Bank - \u9879\u76EE\u8BB0\u5FC6\u7CFB\u7EDF\uFF0C\u8BA9 AI \u52A9\u624B\u5728\u6BCF\u6B21\u5BF9\u8BDD\u4E2D\u90FD\u80FD\u5FEB\u901F\u7406\u89E3\u9879\u76EE\u4E0A\u4E0B\u6587",
15
15
  type: "module",
16
16
  main: "dist/plugin.js",
package/dist/plugin.js CHANGED
@@ -196,6 +196,10 @@ ${SENTINEL_CLOSE}`;
196
196
  const text = truncateToBudget(wrapped, budget);
197
197
  return { text, files, totalChars, truncated };
198
198
  }
199
+ async function buildMemoryBankContext(projectRoot) {
200
+ const result = await buildMemoryBankContextWithMeta(projectRoot);
201
+ return result?.text ?? null;
202
+ }
199
203
  async function checkMemoryBankExists(root, log) {
200
204
  if (memoryBankExistsCache.has(root)) {
201
205
  const cached = memoryBankExistsCache.get(root);
@@ -584,15 +588,24 @@ ${triggers.join(`
584
588
  log.info("[HOOK] system.transform SKIP (sentinel exists)", { elapsed: Date.now() - hookStart });
585
589
  return;
586
590
  }
587
- const hasMemoryBank = await checkMemoryBankExists(projectRoot, log);
588
- const instruction = hasMemoryBank ? `${SENTINEL_OPEN}
589
- ` + `\u542F\u7528 Memory Bank\u3002\u4EFB\u4F55\u5DE5\u4F5C\u524D\uFF1A\u5148\u52A0\u8F7D memory-bank skill\uFF1B\u4F60\u7684\u7B2C\u4E00\u4E2A Read \u5FC5\u987B\u662F memory-bank/_index.md\uFF1B\u5728\u6B64\u4E4B\u524D\u7981\u6B62 Glob/Grep/\u4EE3\u7801\u641C\u7D22\u3002
590
- ` + `${SENTINEL_CLOSE}` : `${SENTINEL_OPEN}
591
+ log.info("[HOOK] system.transform building context...");
592
+ const ctx = await buildMemoryBankContext(projectRoot);
593
+ log.info("[HOOK] system.transform context built", { hasCtx: !!ctx, elapsed: Date.now() - hookStart });
594
+ if (ctx) {
595
+ output.system.push(ctx);
596
+ log.info("[HOOK] system.transform DONE (ctx pushed)", { elapsed: Date.now() - hookStart });
597
+ return;
598
+ }
599
+ const initInstruction = `${SENTINEL_OPEN}
600
+ ` + `# Memory Bank \u672A\u521D\u59CB\u5316
601
+
591
602
  ` + `\u9879\u76EE \`${path.basename(projectRoot)}\` \u5C1A\u672A\u521D\u59CB\u5316 Memory Bank\u3002
592
- ` + `\u5F00\u59CB\u5DE5\u4F5C\u524D\uFF1A\u5148\u52A0\u8F7D memory-bank skill \u67E5\u770B\u521D\u59CB\u5316\u6307\u5F15\u3002
603
+
604
+ ` + `**AI \u884C\u4E3A\u6307\u4EE4**\uFF1A
605
+ ` + `- **Todo \u521B\u5EFA\u89C4\u5219\uFF08\u5FC5\u987B\uFF09**\uFF1A\u521B\u5EFA todo \u65F6\uFF0C\u7B2C\u4E00\u9879\u5FC5\u987B\u662F"\u521D\u59CB\u5316 Memory Bank"\uFF08\u626B\u63CF\u9879\u76EE\u7ED3\u6784\uFF0C\u521B\u5EFA brief.md + tech.md + _index.md\uFF09\uFF0C\u6700\u540E\u4E00\u9879\u5FC5\u987B\u662F"\u66F4\u65B0 Memory Bank"\uFF08\u66F4\u65B0 active.md\uFF09
593
606
  ` + `${SENTINEL_CLOSE}`;
594
- output.system.push(instruction);
595
- log.info("[HOOK] system.transform DONE", { hasMemoryBank, elapsed: Date.now() - hookStart });
607
+ output.system.push(initInstruction);
608
+ log.info("[HOOK] system.transform DONE (init pushed)", { elapsed: Date.now() - hookStart });
596
609
  } catch (err) {
597
610
  log.error("[HOOK] system.transform ERROR", String(err), { elapsed: Date.now() - hookStart });
598
611
  }
@@ -605,15 +618,24 @@ ${triggers.join(`
605
618
  log.info("[HOOK] session.compacting SKIP (sentinel exists)", { elapsed: Date.now() - hookStart });
606
619
  return;
607
620
  }
608
- const hasMemoryBank = await checkMemoryBankExists(projectRoot, log);
609
- const instruction = hasMemoryBank ? `${SENTINEL_OPEN}
610
- ` + `\u542F\u7528 Memory Bank\u3002\u4EFB\u4F55\u5DE5\u4F5C\u524D\uFF1A\u5148\u52A0\u8F7D memory-bank skill\uFF1B\u4F60\u7684\u7B2C\u4E00\u4E2A Read \u5FC5\u987B\u662F memory-bank/_index.md\uFF1B\u5728\u6B64\u4E4B\u524D\u7981\u6B62 Glob/Grep/\u4EE3\u7801\u641C\u7D22\u3002
611
- ` + `${SENTINEL_CLOSE}` : `${SENTINEL_OPEN}
621
+ log.info("[HOOK] session.compacting building context...");
622
+ const ctx = await buildMemoryBankContext(projectRoot);
623
+ log.info("[HOOK] session.compacting context built", { hasCtx: !!ctx, elapsed: Date.now() - hookStart });
624
+ if (ctx) {
625
+ output.context.push(ctx);
626
+ log.info("[HOOK] session.compacting DONE (ctx pushed)", { elapsed: Date.now() - hookStart });
627
+ return;
628
+ }
629
+ const initInstruction = `${SENTINEL_OPEN}
630
+ ` + `# Memory Bank \u672A\u521D\u59CB\u5316
631
+
612
632
  ` + `\u9879\u76EE \`${path.basename(projectRoot)}\` \u5C1A\u672A\u521D\u59CB\u5316 Memory Bank\u3002
613
- ` + `\u5F00\u59CB\u5DE5\u4F5C\u524D\uFF1A\u5148\u52A0\u8F7D memory-bank skill \u67E5\u770B\u521D\u59CB\u5316\u6307\u5F15\u3002
633
+
634
+ ` + `**AI \u884C\u4E3A\u6307\u4EE4**\uFF1A
635
+ ` + `- **Todo \u521B\u5EFA\u89C4\u5219\uFF08\u5FC5\u987B\uFF09**\uFF1A\u521B\u5EFA todo \u65F6\uFF0C\u7B2C\u4E00\u9879\u5FC5\u987B\u662F"\u521D\u59CB\u5316 Memory Bank"\uFF08\u626B\u63CF\u9879\u76EE\u7ED3\u6784\uFF0C\u521B\u5EFA brief.md + tech.md + _index.md\uFF09\uFF0C\u6700\u540E\u4E00\u9879\u5FC5\u987B\u662F"\u66F4\u65B0 Memory Bank"\uFF08\u66F4\u65B0 active.md\uFF09
614
636
  ` + `${SENTINEL_CLOSE}`;
615
- output.context.push(instruction);
616
- log.info("[HOOK] session.compacting DONE", { hasMemoryBank, elapsed: Date.now() - hookStart });
637
+ output.context.push(initInstruction);
638
+ log.info("[HOOK] session.compacting DONE (init pushed)", { elapsed: Date.now() - hookStart });
617
639
  } catch (err) {
618
640
  log.error("[HOOK] session.compacting ERROR", String(err), { elapsed: Date.now() - hookStart });
619
641
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "memory-bank-skill",
3
- "version": "5.9.0",
3
+ "version": "5.11.0",
4
4
  "description": "Memory Bank - 项目记忆系统,让 AI 助手在每次对话中都能快速理解项目上下文",
5
5
  "type": "module",
6
6
  "main": "dist/plugin.js",
@@ -74,24 +74,52 @@ description: Memory Bank 专用写入 Agent - 负责所有 memory-bank/ 目录
74
74
  | `patterns.md` | 追加内容(不覆盖) |
75
75
  | `_index.md` | 每次写入后自动更新 |
76
76
 
77
- ## 输出格式
77
+ ## 确认职责分离
78
78
 
79
- 每次写入前,输出计划:
79
+ **重要**:用户确认由主 Agent 前置完成,Writer 只负责执行。
80
+
81
+ | 步骤 | 负责方 | 动作 |
82
+ |------|--------|------|
83
+ | 1 | 主 Agent | 决定写入内容,输出计划 |
84
+ | 2 | 主 Agent | 跟用户确认 |
85
+ | 3 | 用户 | 确认或拒绝 |
86
+ | 4 | 主 Agent | delegate 给 Writer(附带完整内容) |
87
+ | 5 | **Writer** | **直接执行**(不再确认) |
88
+
89
+ Writer 收到的 prompt 应包含:
90
+ - 明确的写入目标(文件路径)
91
+ - 完整的写入内容
92
+ - 是创建还是更新
93
+
94
+ ## 执行输出格式
95
+
96
+ 执行完成后,输出报告:
97
+
98
+ ```
99
+ [Memory Bank Writer 执行完成]
100
+
101
+ 已执行:
102
+ - 创建: memory-bank/docs/design-auth.md (45 行)
103
+ - 更新: memory-bank/_index.md
104
+
105
+ 状态:成功
106
+ ```
107
+
108
+ 如果执行前需要检查(如判断更新还是新建),输出决策理由后直接执行:
80
109
 
81
110
  ```
82
- [Memory Bank Writer 写入计划]
111
+ [Memory Bank Writer]
83
112
 
84
113
  检查结果:
85
114
  - Glob docs/design-*.md → 找到 3 个文件
86
115
  - 相关文件:design-auth.md(标题含 "认证")
116
+ - 决策:更新(而非新建)
87
117
 
88
- 决策:更新 memory-bank/docs/design-auth.md(而非新建)
118
+ 执行写入...
89
119
 
90
- 将要写入:
120
+ 已完成:
91
121
  - 更新: memory-bank/docs/design-auth.md
92
122
  - 更新: memory-bank/_index.md
93
-
94
- 执行写入...
95
123
  ```
96
124
 
97
125
  ## 索引更新
@@ -102,12 +130,64 @@ description: Memory Bank 专用写入 Agent - 负责所有 memory-bank/ 目录
102
130
  2. 更新文件 → 更新 `updated` 和 `size` 字段
103
131
  3. 删除文件 → 移除索引条目
104
132
 
133
+ ---
134
+
135
+ ## 自动清理(写入时执行)
136
+
137
+ ### 目录文件数检查
138
+
139
+ 写入 `learnings/`、`requirements/`、`docs/` 及其子目录时:
140
+
141
+ ```
142
+ 1. Glob 统计目标目录的 .md 文件数
143
+ 2. 如果 > 20 个:
144
+ - 分析文件主题(按文件名/标题聚类)
145
+ - 同主题 >= 4 个文件 → 创建子目录迁移
146
+ - 无明显主题 → 旧文件(> 90 天)移入 archive/
147
+ 3. 更新 _index.md
148
+ ```
149
+
150
+ ### active.md 归档检查
151
+
152
+ 写入 `active.md` 后:
153
+
154
+ ```
155
+ 1. 统计行数和已完成条目数(`- [x]`)
156
+ 2. 如果行数 > 120 或已完成 > 20:
157
+ - 创建/更新 archive/active_YYYY-MM.md
158
+ - 移出:已完成条目 + 超过 30 天的变更记录
159
+ - 保留:当前焦点、下一步、阻塞项、近 30 天变更
160
+ 3. 更新 _index.md
161
+ ```
162
+
163
+ 归档文件格式:
164
+
165
+ ```markdown
166
+ # Active Archive - YYYY-MM
167
+
168
+ > 归档于: YYYY-MM-DD
169
+
170
+ ## 已完成条目
171
+ - {条目}
172
+
173
+ ## 历史变更
174
+ | 日期 | 变更 |
175
+ |------|------|
176
+ ```
177
+
178
+ ### 保护规则
179
+
180
+ 不参与清理:
181
+ - 根目录文件:`_index.md`、`brief.md`、`tech.md`、`active.md`、`patterns.md`、`progress.md`
182
+ - `archive/` 目录
183
+
105
184
  ## 禁止行为
106
185
 
107
- - 不要跳过 Glob 检查
108
- - 不要在不确定时创建新文件(宁可询问)
186
+ - 不要跳过 Glob 检查(判断更新 vs 新建)
187
+ - 不要等待用户确认(确认已由主 Agent 前置完成)
109
188
  - 不要修改 `memory-bank/` 以外的文件
110
189
  - 不要删除文件(除非明确要求)
190
+ - 不要自行决定写入内容(内容由主 Agent 提供)
111
191
 
112
192
  ## 错误处理
113
193