bmad-plus 0.9.1 → 0.12.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/CHANGELOG.md +60 -0
- package/README.md +1 -1
- package/osint-agent-package/skills/bmad-osint-investigate/osint/SKILL.md +30 -0
- package/osint-agent-package/skills/bmad-osint-investigate/osint/assets/dossier-template.md +10 -0
- package/osint-agent-package/skills/bmad-osint-investigate/osint/assets/lawful-basis-record.md +48 -0
- package/osint-agent-package/skills/bmad-osint-investigate/osint/references/gdpr-osint.md +48 -0
- package/package.json +3 -1
- package/tools/build/README.md +78 -0
- package/tools/build/adapters.config.js +117 -0
- package/tools/build/generate-adapters.js +485 -0
- package/tools/build/generate.js +284 -0
- package/tools/build/generated-adapters/.codex/AGENTS.md +121 -0
- package/tools/build/generated-adapters/.cursor/rules/bmad-plus.mdc +126 -0
- package/tools/build/generated-adapters/.opencode/AGENTS.md +121 -0
- package/tools/build/generated-adapters/AGENTS.md +119 -0
- package/tools/build/generated-adapters/CLAUDE.md +122 -0
- package/tools/build/generated-adapters/CONVENTIONS.md +121 -0
- package/tools/build/generated-adapters/GEMINI.md +126 -0
- package/tools/build/generated-adapters/README.md +79 -0
- package/tools/cli/bmad-plus-cli.js +11 -0
- package/tools/cli/commands/autoconfig.js +18 -1
- package/tools/cli/commands/doctor.js +12 -0
- package/tools/cli/commands/install.js +66 -0
- package/tools/cli/commands/memory-journal-cmd.js +311 -0
- package/tools/cli/commands/scan.js +18 -1
- package/tools/cli/commands/uninstall.js +3 -1
- package/tools/cli/commands/update.js +19 -2
- package/tools/cli/lib/README-memory-journal.md +125 -0
- package/tools/cli/lib/memory-journal.js +0 -0
- package/tools/cli/lib/packs.js +209 -114
- package/tools/cli/lib/python-provision.js +508 -0
- package/tools/cli/lib/validate.js +8 -3
|
@@ -17,6 +17,21 @@ const { copyPackFiles } = require('../lib/pack-copy');
|
|
|
17
17
|
const { initMemory } = require('../lib/memory-init');
|
|
18
18
|
const { generateIDEConfig, IDE_CONFIGS } = require('../lib/ide-config');
|
|
19
19
|
const { validateUserName } = require('../lib/validate');
|
|
20
|
+
const { provisionPack } = require('../lib/python-provision');
|
|
21
|
+
|
|
22
|
+
// Packs whose runtime includes Python (registry.yaml: runtime includes "python").
|
|
23
|
+
// Provisioned only when the user opts in with --provision-python, so the default
|
|
24
|
+
// install (and the test suite) never spawns Python. Kills SEO "broken on arrival".
|
|
25
|
+
const PYTHON_PACKS = {
|
|
26
|
+
seo: {
|
|
27
|
+
requirements: ['oveanet-pack', 'seo-audit-360', 'requirements.txt'],
|
|
28
|
+
verifyModules: ['requests', 'bs4', 'defusedxml', 'lxml'],
|
|
29
|
+
},
|
|
30
|
+
memory: {
|
|
31
|
+
requirements: ['mcp-server', 'requirements.txt'],
|
|
32
|
+
verifyModules: ['chromadb'],
|
|
33
|
+
},
|
|
34
|
+
};
|
|
20
35
|
|
|
21
36
|
// Pack definitions are imported from the shared module: require('../lib/packs').PACKS
|
|
22
37
|
|
|
@@ -29,6 +44,7 @@ module.exports = {
|
|
|
29
44
|
['-y, --yes', 'Accept all defaults, skip prompts'],
|
|
30
45
|
['-l, --lang <code>', 'Language code: en, fr, es, de, pt-br, ru, zh, he, ja, it (overrides auto-detection)'],
|
|
31
46
|
['--tools <tools>', 'Comma-separated IDE IDs (default: auto-detect)'],
|
|
47
|
+
['--provision-python', 'Provision an isolated Python env for packs that need one (seo, memory)'],
|
|
32
48
|
],
|
|
33
49
|
action: async (options) => {
|
|
34
50
|
const projectDir = path.resolve(options.directory || process.cwd());
|
|
@@ -225,6 +241,36 @@ module.exports = {
|
|
|
225
241
|
}
|
|
226
242
|
}
|
|
227
243
|
|
|
244
|
+
// ── Python provisioning (opt-in) — for packs whose runtime needs Python ──
|
|
245
|
+
// Off by default so `install` never spawns Python unless asked. Closes the
|
|
246
|
+
// "SEO pack ships prompts that call scripts never installed" gap (PROD-02).
|
|
247
|
+
if (options.provisionPython) {
|
|
248
|
+
const pythonPacks = selectedPacks.filter((p) => PYTHON_PACKS[p]);
|
|
249
|
+
for (const packId of pythonPacks) {
|
|
250
|
+
const cfg = PYTHON_PACKS[packId];
|
|
251
|
+
const requirementsPath = path.join(projectRoot, ...cfg.requirements);
|
|
252
|
+
if (!fs.existsSync(requirementsPath)) {
|
|
253
|
+
clack.log.warn(`Python provisioning skipped for ${packId}: ${requirementsPath} not found.`);
|
|
254
|
+
continue;
|
|
255
|
+
}
|
|
256
|
+
const pySpinner = clack.spinner();
|
|
257
|
+
pySpinner.start(`Provisioning Python runtime for ${PACKS[packId].name}…`);
|
|
258
|
+
const res = provisionPack({
|
|
259
|
+
envDir: path.join(projectDir, '.bmad', 'venv', packId),
|
|
260
|
+
requirementsPath,
|
|
261
|
+
verifyModules: cfg.verifyModules,
|
|
262
|
+
});
|
|
263
|
+
pySpinner.stop(
|
|
264
|
+
res.ok
|
|
265
|
+
? `${PACKS[packId].name}: Python runtime ready (${res.tool}).`
|
|
266
|
+
: `${PACKS[packId].name}: Python not provisioned — ${res.messages[res.messages.length - 1] || 'see guidance'}`
|
|
267
|
+
);
|
|
268
|
+
if (!res.ok) {
|
|
269
|
+
clack.log.warn(`Run \`${PACKS[packId].name}\` Python tools after installing Python ≥3.11; the pack's prompts still work, only the local scripts need the runtime.`);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
|
|
228
274
|
// Copy module config
|
|
229
275
|
const moduleYaml = path.join(bmadSrc, 'module.yaml');
|
|
230
276
|
if (fs.existsSync(moduleYaml)) {
|
|
@@ -246,12 +292,32 @@ module.exports = {
|
|
|
246
292
|
ideSpinner.start(i.configuring_ides);
|
|
247
293
|
|
|
248
294
|
const configContent = generateIDEConfig(userName, commLang, selectedPacks);
|
|
295
|
+
// Marker present in every BMAD+-generated IDE config (see lib/ide-config.js).
|
|
296
|
+
const BMAD_MARKER = 'BMAD+ — AI Agent Configuration';
|
|
249
297
|
|
|
250
298
|
for (const ideId of detectedIDEs) {
|
|
251
299
|
const ide = IDE_CONFIGS[ideId];
|
|
252
300
|
if (!ide) continue;
|
|
253
301
|
|
|
254
302
|
const configPath = path.join(projectDir, ide.configFile);
|
|
303
|
+
|
|
304
|
+
// NODE-02: never clobber a hand-authored config without protecting it.
|
|
305
|
+
if (fs.existsSync(configPath)) {
|
|
306
|
+
const existing = fs.readFileSync(configPath, 'utf8');
|
|
307
|
+
if (!existing.includes(BMAD_MARKER)) {
|
|
308
|
+
if (!options.yes) {
|
|
309
|
+
// Interactive: keep the user's file untouched.
|
|
310
|
+
clack.log.warn(`⚠️ ${ide.configFile} already exists and was not created by BMAD+ — kept your file (skipped). Re-run with --yes to back it up and overwrite.`);
|
|
311
|
+
continue;
|
|
312
|
+
}
|
|
313
|
+
// Non-interactive: back up before overwriting, never destroy data.
|
|
314
|
+
let backupPath = `${configPath}.bak`;
|
|
315
|
+
if (fs.existsSync(backupPath)) backupPath = `${configPath}.${Date.now()}.bak`;
|
|
316
|
+
fs.copyFileSync(configPath, backupPath);
|
|
317
|
+
clack.log.warn(`⚠️ Backed up existing ${ide.configFile} → ${path.basename(backupPath)}`);
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
|
|
255
321
|
fs.writeFileSync(configPath, configContent, 'utf8');
|
|
256
322
|
}
|
|
257
323
|
|
|
@@ -0,0 +1,311 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* BMAD+ `mem` Command — portable Karpathy memory loop (Pillar 3)
|
|
3
|
+
*
|
|
4
|
+
* Exposes tools/cli/lib/memory-journal.js through the CLI so recall / write /
|
|
5
|
+
* reinforce behave IDENTICALLY no matter which agent CLI is driving
|
|
6
|
+
* (claude-code, gemini-cli, codex-cli, cursor, opencode, aider, antigravity):
|
|
7
|
+
*
|
|
8
|
+
* bmad-plus mem recall <query...> — ranked retrieval over journal + memory files
|
|
9
|
+
* bmad-plus mem write — append a structured event (--agent --cli --model --task --outcome)
|
|
10
|
+
* bmad-plus mem reinforce — apply a reward to a pattern score; any resulting
|
|
11
|
+
* promotion is persisted as PROPOSED, never applied
|
|
12
|
+
*
|
|
13
|
+
* NOTE: registered as `mem` (not `memory`) — `memory` is already taken by the
|
|
14
|
+
* brain scanner in tools/cli/commands/memory.js and must not be disturbed.
|
|
15
|
+
*
|
|
16
|
+
* Design rules (mirroring memory-journal.js):
|
|
17
|
+
* - The clock is read ONCE inside the action (new Date().toISOString()) and
|
|
18
|
+
* injected into every library call. Nothing reads the clock at import time.
|
|
19
|
+
* - The run* handlers take an injected `now` + `log`, so tests drive them
|
|
20
|
+
* deterministically against a tmp dir (tests/unit/memory-journal-cmd.test.js).
|
|
21
|
+
* - Output goes through plain log lines (+ --json for machine consumption) so
|
|
22
|
+
* ANY driving CLI can parse results — no interactive prompts, no TTY needs.
|
|
23
|
+
* - Governance guard: reinforce may PROPOSE a pattern promotion when the
|
|
24
|
+
* posterior mean crosses the threshold, but the record is always written
|
|
25
|
+
* with status PROPOSED (memory-journal.appendPromotion forces it anyway).
|
|
26
|
+
*
|
|
27
|
+
* Author: Laurent Rochetta
|
|
28
|
+
*/
|
|
29
|
+
|
|
30
|
+
'use strict';
|
|
31
|
+
|
|
32
|
+
const path = require('node:path');
|
|
33
|
+
const fs = require('node:fs');
|
|
34
|
+
const mj = require('../lib/memory-journal');
|
|
35
|
+
|
|
36
|
+
// ── Pattern score store ──────────────────────────────────────────────────────
|
|
37
|
+
// Lives next to the journal in the north-star scope (.bmad/memory/). Keyed by
|
|
38
|
+
// patternId (the `### heading` in patterns.md). This file is CLI-owned state;
|
|
39
|
+
// memory-journal.js stays a pure library and never touches it.
|
|
40
|
+
|
|
41
|
+
const SCORES_RELPATH = path.join('.bmad', 'memory', 'pattern-scores.json');
|
|
42
|
+
|
|
43
|
+
// Promotion proposal thresholds (candidate → validated). Tuned conservatively:
|
|
44
|
+
// the posterior mean is decayed-Bayesian (memory-journal.updatePatternScore),
|
|
45
|
+
// so 0.7 over >= 3 updates means a genuinely recent, repeated success signal.
|
|
46
|
+
const PROMOTION_MEAN_THRESHOLD = 0.7;
|
|
47
|
+
const PROMOTION_MIN_UPDATES = 3;
|
|
48
|
+
|
|
49
|
+
function scoresPath(baseDir) {
|
|
50
|
+
return path.join(baseDir, SCORES_RELPATH);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
function readScores(baseDir) {
|
|
54
|
+
const file = scoresPath(baseDir);
|
|
55
|
+
if (!fs.existsSync(file)) return {};
|
|
56
|
+
try {
|
|
57
|
+
const parsed = JSON.parse(fs.readFileSync(file, 'utf8'));
|
|
58
|
+
return parsed && typeof parsed === 'object' ? parsed : {};
|
|
59
|
+
} catch {
|
|
60
|
+
// Corrupt store never blocks the loop — reinforcement restarts from priors.
|
|
61
|
+
return {};
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function writeScores(baseDir, scores) {
|
|
66
|
+
const file = scoresPath(baseDir);
|
|
67
|
+
fs.mkdirSync(path.dirname(file), { recursive: true });
|
|
68
|
+
fs.writeFileSync(file, JSON.stringify(scores, null, 2) + '\n', 'utf8');
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// ── Flag parsing helpers ─────────────────────────────────────────────────────
|
|
72
|
+
|
|
73
|
+
/** Map CLI flags to the memory-journal signals shape { evalScore, acceptance, ci }. */
|
|
74
|
+
function parseSignals(opts) {
|
|
75
|
+
const signals = {};
|
|
76
|
+
if (opts.eval !== undefined && opts.eval !== null) {
|
|
77
|
+
signals.evalScore = Number(opts.eval);
|
|
78
|
+
}
|
|
79
|
+
if (opts.accept !== undefined && opts.accept !== null) {
|
|
80
|
+
const raw = String(opts.accept).toLowerCase();
|
|
81
|
+
if (raw === 'true' || raw === 'yes') signals.acceptance = true;
|
|
82
|
+
else if (raw === 'false' || raw === 'no') signals.acceptance = false;
|
|
83
|
+
else signals.acceptance = Number(opts.accept);
|
|
84
|
+
}
|
|
85
|
+
if (opts.ci !== undefined && opts.ci !== null) {
|
|
86
|
+
signals.ci = opts.ci === 'pass' || opts.ci === 'fail' ? opts.ci : Number(opts.ci);
|
|
87
|
+
}
|
|
88
|
+
return signals;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
function truncate(text, max = 120) {
|
|
92
|
+
const oneLine = String(text).replace(/\s+/g, ' ').trim();
|
|
93
|
+
return oneLine.length > max ? oneLine.slice(0, max - 1) + '…' : oneLine;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
// ── Action handlers (dependency-injected, unit-testable) ─────────────────────
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* `mem recall <query...>` — ranked retrieval over journal events + memory notes.
|
|
100
|
+
*
|
|
101
|
+
* @param {string} query
|
|
102
|
+
* @param {object} opts - { baseDir, scope, portfolioDir, limit, now, json, log }
|
|
103
|
+
* `now` is the INJECTED clock (enables recency decay).
|
|
104
|
+
* @returns {object[]} ranked entries (same shape as memory-journal.recall)
|
|
105
|
+
*/
|
|
106
|
+
function runRecall(query, opts = {}) {
|
|
107
|
+
const { baseDir = process.cwd(), scope = 'project', portfolioDir = null, limit = 8, now = null, json = false, log = console.log } = opts;
|
|
108
|
+
|
|
109
|
+
const results = mj.recall(query, { baseDir, scope, portfolioDir, limit, now });
|
|
110
|
+
|
|
111
|
+
if (json) {
|
|
112
|
+
log(JSON.stringify({ query, scope, count: results.length, results }, null, 2));
|
|
113
|
+
return results;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
if (results.length === 0) {
|
|
117
|
+
log(`mem recall: no matches for "${query}" (scope: ${scope})`);
|
|
118
|
+
return results;
|
|
119
|
+
}
|
|
120
|
+
log(`mem recall: ${results.length} match(es) for "${query}" (scope: ${scope})`);
|
|
121
|
+
results.forEach((r, i) => {
|
|
122
|
+
const origin = r.kind === 'event' ? `event ${r.source}#${r.ref}` : `note ${r.source} § ${r.ref}`;
|
|
123
|
+
log(` ${i + 1}. [${r.score.toFixed(3)}] ${origin}`);
|
|
124
|
+
log(` ${truncate(r.text)}`);
|
|
125
|
+
});
|
|
126
|
+
return results;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* `mem write` — append a structured event to the journal.
|
|
131
|
+
*
|
|
132
|
+
* @param {object} opts - { baseDir, agent, cli, model, task, outcome, signals, now, json, log }
|
|
133
|
+
* `now` is the INJECTED clock used as the event ts.
|
|
134
|
+
* @returns {object} the persisted event
|
|
135
|
+
*/
|
|
136
|
+
function runWrite(opts = {}) {
|
|
137
|
+
const { baseDir = process.cwd(), agent, cli = null, model = null, task = '', outcome = undefined, signals = {}, now, json = false, log = console.log } = opts;
|
|
138
|
+
|
|
139
|
+
const event = mj.appendEvent(baseDir, {
|
|
140
|
+
ts: now,
|
|
141
|
+
agent,
|
|
142
|
+
cli,
|
|
143
|
+
model,
|
|
144
|
+
task,
|
|
145
|
+
outcome,
|
|
146
|
+
signals,
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
if (json) {
|
|
150
|
+
log(JSON.stringify({ written: event }, null, 2));
|
|
151
|
+
} else {
|
|
152
|
+
log(`mem write: event ${event.id} appended (${event.ts})`);
|
|
153
|
+
log(` agent=${event.agent} cli=${event.cli || '-'} model=${event.model || '-'} outcome=${event.outcome || '-'}`);
|
|
154
|
+
if (event.task) log(` task: ${truncate(event.task)}`);
|
|
155
|
+
}
|
|
156
|
+
return event;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
/**
|
|
160
|
+
* `mem reinforce` — fold reward signals into a pattern's score.
|
|
161
|
+
*
|
|
162
|
+
* Updates the CLI-owned pattern-scores.json (Elo for ranking, decayed-Bayesian
|
|
163
|
+
* mean for promotion). When the mean crosses PROMOTION_MEAN_THRESHOLD with
|
|
164
|
+
* enough updates, a promotion is PROPOSED via memory-journal's governance
|
|
165
|
+
* guard — persisted as PROPOSED, applied only after human/Shield approval.
|
|
166
|
+
*
|
|
167
|
+
* @param {object} opts - { baseDir, patternId, signals, evidence, now, json, log }
|
|
168
|
+
* @returns {{patternId:string, reward:number, previous:object, next:object, promotion:object|null}}
|
|
169
|
+
*/
|
|
170
|
+
function runReinforce(opts = {}) {
|
|
171
|
+
const { baseDir = process.cwd(), patternId, signals = {}, evidence = [], now, json = false, log = console.log } = opts;
|
|
172
|
+
|
|
173
|
+
if (typeof patternId !== 'string' || patternId.trim().length === 0) {
|
|
174
|
+
throw new TypeError('mem reinforce: --pattern <id> is required (the pattern heading in patterns.md)');
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
const reward = mj.computeReward(signals);
|
|
178
|
+
const scores = readScores(baseDir);
|
|
179
|
+
const previous = scores[patternId] || null;
|
|
180
|
+
const next = mj.updatePatternScore(previous, reward, { ts: now });
|
|
181
|
+
scores[patternId] = next;
|
|
182
|
+
writeScores(baseDir, scores);
|
|
183
|
+
|
|
184
|
+
// Governance: promotion is only ever PROPOSED here. appendPromotion() forces
|
|
185
|
+
// status PROPOSED on disk regardless, and applying requires
|
|
186
|
+
// assertPromotionApplicable() to pass with a human/Shield approval.
|
|
187
|
+
let promotion = null;
|
|
188
|
+
if (next.mean >= PROMOTION_MEAN_THRESHOLD && next.updates >= PROMOTION_MIN_UPDATES) {
|
|
189
|
+
promotion = mj.appendPromotion(
|
|
190
|
+
baseDir,
|
|
191
|
+
mj.proposePromotion({
|
|
192
|
+
patternId,
|
|
193
|
+
ts: now,
|
|
194
|
+
reason: `posterior mean ${next.mean.toFixed(3)} >= ${PROMOTION_MEAN_THRESHOLD} over ${next.updates} update(s)`,
|
|
195
|
+
evidence,
|
|
196
|
+
score: next,
|
|
197
|
+
})
|
|
198
|
+
);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
const result = { patternId, reward, previous, next, promotion };
|
|
202
|
+
|
|
203
|
+
if (json) {
|
|
204
|
+
log(JSON.stringify(result, null, 2));
|
|
205
|
+
return result;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const prevElo = previous ? previous.elo : mj.INITIAL_PATTERN_SCORE.elo;
|
|
209
|
+
log(`mem reinforce: pattern "${patternId}"`);
|
|
210
|
+
log(` reward ${reward.toFixed(3)} (from ${JSON.stringify(signals)})`);
|
|
211
|
+
log(` elo ${prevElo.toFixed(1)} -> ${next.elo.toFixed(1)}`);
|
|
212
|
+
log(` mean ${next.mean.toFixed(3)} (alpha=${next.alpha.toFixed(2)}, beta=${next.beta.toFixed(2)}, updates=${next.updates})`);
|
|
213
|
+
if (promotion) {
|
|
214
|
+
log(` promotion PROPOSED (${promotion.id}): ${promotion.fromStatus} -> ${promotion.toStatus}`);
|
|
215
|
+
log(` awaiting human/Shield approval — never auto-applied (governance guard)`);
|
|
216
|
+
} else {
|
|
217
|
+
log(` promotion: none proposed (needs mean >= ${PROMOTION_MEAN_THRESHOLD} and >= ${PROMOTION_MIN_UPDATES} updates)`);
|
|
218
|
+
}
|
|
219
|
+
return result;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
// ── Commander wiring ─────────────────────────────────────────────────────────
|
|
223
|
+
|
|
224
|
+
module.exports = {
|
|
225
|
+
command: 'mem <action> [query...]',
|
|
226
|
+
description: 'Karpathy memory loop — recall | write | reinforce (portable across agent CLIs)',
|
|
227
|
+
options: [
|
|
228
|
+
['-d, --directory <path>', 'Project directory (default: current directory)'],
|
|
229
|
+
['--scope <scope>', 'Recall scope: project | portfolio', 'project'],
|
|
230
|
+
['--portfolio <path>', 'Portfolio brain directory (used with --scope portfolio)'],
|
|
231
|
+
['--limit <n>', 'Max recall results', '8'],
|
|
232
|
+
['--agent <name>', 'Agent that produced the event (write)'],
|
|
233
|
+
['--cli <name>', 'Driving CLI: claude-code, gemini-cli, codex-cli, cursor, opencode, aider, antigravity'],
|
|
234
|
+
['--model <id>', 'Model id used (model-agnostic: claude/gpt/gemini/local)'],
|
|
235
|
+
['--task <text>', 'What was attempted (write)'],
|
|
236
|
+
['--outcome <outcome>', 'success | failure | partial | abandoned (write)'],
|
|
237
|
+
['--pattern <id>', 'Pattern id to reinforce (its heading in patterns.md)'],
|
|
238
|
+
['--eval <score>', 'Eval suite score in [0,1] (write/reinforce signal)'],
|
|
239
|
+
['--accept <bool>', 'User acceptance: true | false | [0,1] (write/reinforce signal)'],
|
|
240
|
+
['--ci <result>', 'CI outcome: pass | fail | [0,1] (write/reinforce signal)'],
|
|
241
|
+
['--evidence <ids>', 'Comma-separated journal event ids backing a reinforcement'],
|
|
242
|
+
['--json', 'Machine-readable JSON output'],
|
|
243
|
+
],
|
|
244
|
+
subcommands: {
|
|
245
|
+
recall: 'Ranked retrieval over the journal + memory files',
|
|
246
|
+
write: 'Append a structured event to .bmad/memory/journal.ndjson',
|
|
247
|
+
reinforce: 'Apply a reward to a pattern score (promotions PROPOSED only)',
|
|
248
|
+
},
|
|
249
|
+
|
|
250
|
+
action: async (action, query, options = {}) => {
|
|
251
|
+
// Clock is read HERE, at call time, then injected everywhere below —
|
|
252
|
+
// memory-journal.js never reads it (see its determinism contract).
|
|
253
|
+
const now = new Date().toISOString();
|
|
254
|
+
const baseDir = path.resolve(options.directory || process.cwd());
|
|
255
|
+
const json = Boolean(options.json);
|
|
256
|
+
const queryText = Array.isArray(query) ? query.join(' ') : query || '';
|
|
257
|
+
|
|
258
|
+
try {
|
|
259
|
+
if (action === 'recall') {
|
|
260
|
+
runRecall(queryText, {
|
|
261
|
+
baseDir,
|
|
262
|
+
scope: options.scope || 'project',
|
|
263
|
+
portfolioDir: options.portfolio || null,
|
|
264
|
+
limit: Number(options.limit || 8),
|
|
265
|
+
now,
|
|
266
|
+
json,
|
|
267
|
+
});
|
|
268
|
+
} else if (action === 'write') {
|
|
269
|
+
runWrite({
|
|
270
|
+
baseDir,
|
|
271
|
+
agent: options.agent,
|
|
272
|
+
cli: options.cli || null,
|
|
273
|
+
model: options.model || null,
|
|
274
|
+
task: options.task || '',
|
|
275
|
+
outcome: options.outcome,
|
|
276
|
+
signals: parseSignals(options),
|
|
277
|
+
now,
|
|
278
|
+
json,
|
|
279
|
+
});
|
|
280
|
+
} else if (action === 'reinforce') {
|
|
281
|
+
runReinforce({
|
|
282
|
+
baseDir,
|
|
283
|
+
patternId: options.pattern,
|
|
284
|
+
signals: parseSignals(options),
|
|
285
|
+
evidence: options.evidence ? String(options.evidence).split(',').map(s => s.trim()).filter(Boolean) : [],
|
|
286
|
+
now,
|
|
287
|
+
json,
|
|
288
|
+
});
|
|
289
|
+
} else {
|
|
290
|
+
console.error(`mem: unknown action '${action}' — expected recall | write | reinforce`);
|
|
291
|
+
process.exitCode = 1;
|
|
292
|
+
}
|
|
293
|
+
} catch (err) {
|
|
294
|
+
console.error(`mem ${action}: ${err.message}`);
|
|
295
|
+
process.exitCode = 1;
|
|
296
|
+
}
|
|
297
|
+
},
|
|
298
|
+
|
|
299
|
+
// Exported for tests + future MCP wrapper (memory.recall / memory.write tools)
|
|
300
|
+
_internal: {
|
|
301
|
+
runRecall,
|
|
302
|
+
runWrite,
|
|
303
|
+
runReinforce,
|
|
304
|
+
parseSignals,
|
|
305
|
+
readScores,
|
|
306
|
+
writeScores,
|
|
307
|
+
SCORES_RELPATH,
|
|
308
|
+
PROMOTION_MEAN_THRESHOLD,
|
|
309
|
+
PROMOTION_MIN_UPDATES,
|
|
310
|
+
},
|
|
311
|
+
};
|
|
@@ -118,6 +118,22 @@ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0, activeDays = 30,
|
|
|
118
118
|
return projects;
|
|
119
119
|
}
|
|
120
120
|
|
|
121
|
+
/**
|
|
122
|
+
* Normalize a filesystem path for stable hashing.
|
|
123
|
+
* On Windows the same project can be referenced as `D:\proj` or `d:\proj`
|
|
124
|
+
* (case-insensitive drive) with mixed separators. Resolving + lowercasing the
|
|
125
|
+
* drive letter ensures the same project produces the same index key and is not
|
|
126
|
+
* double-indexed.
|
|
127
|
+
* @param {string} p - Raw filesystem path
|
|
128
|
+
* @returns {string} Normalized path suitable for hashing
|
|
129
|
+
*/
|
|
130
|
+
function normalizePathForHash(p) {
|
|
131
|
+
let resolved = path.resolve(p);
|
|
132
|
+
// Lowercase a leading Windows drive letter (e.g. "D:" → "d:").
|
|
133
|
+
resolved = resolved.replace(/^([A-Za-z]):/, (_m, drive) => drive.toLowerCase() + ':');
|
|
134
|
+
return resolved;
|
|
135
|
+
}
|
|
136
|
+
|
|
121
137
|
/**
|
|
122
138
|
* Index a single project in the global brain by writing its metadata YAML file.
|
|
123
139
|
* @param {object} project - Project metadata object
|
|
@@ -125,7 +141,7 @@ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0, activeDays = 30,
|
|
|
125
141
|
* @returns {void}
|
|
126
142
|
*/
|
|
127
143
|
function indexProject(project, globalBrainDir) {
|
|
128
|
-
const hash = crypto.createHash('sha256').update(project.path).digest('hex').slice(0, 8);
|
|
144
|
+
const hash = crypto.createHash('sha256').update(normalizePathForHash(project.path)).digest('hex').slice(0, 8);
|
|
129
145
|
const meta = {
|
|
130
146
|
path: project.path,
|
|
131
147
|
name: project.name,
|
|
@@ -341,6 +357,7 @@ module.exports = {
|
|
|
341
357
|
getProjectName,
|
|
342
358
|
hasBmadInstalled,
|
|
343
359
|
scanDirectory,
|
|
360
|
+
normalizePathForHash,
|
|
344
361
|
indexProject,
|
|
345
362
|
indexProjects,
|
|
346
363
|
},
|
|
@@ -86,7 +86,9 @@ module.exports = {
|
|
|
86
86
|
const p = path.join(projectDir, configFile);
|
|
87
87
|
if (fs.existsSync(p)) {
|
|
88
88
|
const content = fs.readFileSync(p, 'utf8');
|
|
89
|
-
|
|
89
|
+
// NODE-05: only remove a config BMAD+ actually generated (it carries the
|
|
90
|
+
// marker below). A hand-authored file that merely mentions "BMAD+" is kept.
|
|
91
|
+
if (content.includes('BMAD+ — AI Agent Configuration')) {
|
|
90
92
|
fs.unlinkSync(p);
|
|
91
93
|
removed++;
|
|
92
94
|
}
|
|
@@ -39,7 +39,21 @@ module.exports = {
|
|
|
39
39
|
return;
|
|
40
40
|
}
|
|
41
41
|
|
|
42
|
-
|
|
42
|
+
let manifest;
|
|
43
|
+
try {
|
|
44
|
+
manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
|
|
45
|
+
} catch (err) {
|
|
46
|
+
clack.log.error(`Install manifest is unreadable or corrupt: ${err.message}`);
|
|
47
|
+
clack.log.info('Re-run `npx bmad-plus install` to repair the installation.');
|
|
48
|
+
clack.outro(pc.red('Update aborted.'));
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
if (!manifest || typeof manifest !== 'object') {
|
|
52
|
+
clack.log.error('Install manifest is malformed (not an object).');
|
|
53
|
+
clack.log.info('Re-run `npx bmad-plus install` to repair the installation.');
|
|
54
|
+
clack.outro(pc.red('Update aborted.'));
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
43
57
|
const lang = options.lang || manifest.uiLanguage || 'en';
|
|
44
58
|
const i = t(lang);
|
|
45
59
|
|
|
@@ -51,7 +65,10 @@ module.exports = {
|
|
|
51
65
|
return;
|
|
52
66
|
}
|
|
53
67
|
|
|
54
|
-
|
|
68
|
+
// Guard against a missing/malformed `packs` field (NODE-04): coerce to an array.
|
|
69
|
+
const selectedPacks = Array.isArray(manifest.packs) && manifest.packs.length > 0
|
|
70
|
+
? manifest.packs
|
|
71
|
+
: ['core'];
|
|
55
72
|
clack.log.info(`${i.selected_packs}: ${selectedPacks.join(', ')}`);
|
|
56
73
|
|
|
57
74
|
const confirm = await clack.confirm({
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
# memory-journal.js — Karpathy Learning Layer core (Pillar 3)
|
|
2
|
+
|
|
3
|
+
Portable data structures + helpers for the BMAD+ **memory → reward → reinforcement** loop.
|
|
4
|
+
Prompt-level learning only: scores steer retrieval and pattern promotion — there is
|
|
5
|
+
**no base-model fine-tuning** (see `audit/2026-07-01/north-star/registry.yaml` →
|
|
6
|
+
`memory.reward_signal.applies_to`).
|
|
7
|
+
|
|
8
|
+
Builds **on top of** the existing `pack-memory` (Zecher, Karpathy guardrails G1–G4,
|
|
9
|
+
`decisions/lessons/patterns/context.md` templates). It never modifies those files or
|
|
10
|
+
`tools/cli/lib/memory-init.js` — it adds a structured, machine-readable layer beside them.
|
|
11
|
+
|
|
12
|
+
```
|
|
13
|
+
.bmad/memory/journal.ndjson ← structured event log (this module, north-star scope)
|
|
14
|
+
.bmad/memory/promotions.ndjson ← governance queue (always PROPOSED)
|
|
15
|
+
.agents/memory/*.md ← human memory (pack-memory, current layout) — READ ONLY here
|
|
16
|
+
.bmad/memory/*.md ← human memory (north-star layout) — READ ONLY here
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
## Hard rules baked into the module
|
|
20
|
+
|
|
21
|
+
| Rule | Enforcement |
|
|
22
|
+
|---|---|
|
|
23
|
+
| No hidden clock / randomness | `ts` is a **required, caller-injected** field; ids are content hashes (sha256). The module never calls `Date.now()` or `Math.random()` — at import or runtime. |
|
|
24
|
+
| Node stdlib only | `fs`, `path`, `crypto`. No network, no native deps → runs identically under every CLI. |
|
|
25
|
+
| Journal is append-only, corruption-tolerant | `readJournal`/`readPromotions` skip torn lines instead of throwing (concurrent CLIs may write). |
|
|
26
|
+
| Promotions are never auto-applied | `proposePromotion()` only emits `status: 'PROPOSED'`; `appendPromotion()` **forces** `PROPOSED` + clears approval fields on disk even for tampered records; `assertPromotionApplicable()` throws unless `status === 'APPROVED'` **and** `approvedBy` names a human/Shield reviewer. |
|
|
27
|
+
|
|
28
|
+
## API
|
|
29
|
+
|
|
30
|
+
### 1. Journal
|
|
31
|
+
|
|
32
|
+
```js
|
|
33
|
+
const mj = require('./memory-journal');
|
|
34
|
+
|
|
35
|
+
mj.appendEvent(projectDir, {
|
|
36
|
+
ts: new Date().toISOString(), // REQUIRED — injected by the caller
|
|
37
|
+
agent: 'forge', // REQUIRED
|
|
38
|
+
cli: 'claude-code', // claude-code | gemini-cli | antigravity | cursor | codex-cli | opencode | aider
|
|
39
|
+
model: 'claude', // model-agnostic by contract (claude/gpt/gemini/local)
|
|
40
|
+
task: 'refactor postgres pooling',
|
|
41
|
+
outcome: 'success', // success | failure | partial | abandoned
|
|
42
|
+
signals: { evalScore: 0.9, acceptance: true, ci: 'pass' },
|
|
43
|
+
artifactHashes: ['abc123'], // traceability to produced artifacts
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
mj.readJournal(projectDir); // → events[], oldest first, corrupt lines skipped
|
|
47
|
+
```
|
|
48
|
+
|
|
49
|
+
### 2. Recall (lexical first cut + vector-backend seam)
|
|
50
|
+
|
|
51
|
+
```js
|
|
52
|
+
mj.recall('postgres pooling', {
|
|
53
|
+
baseDir: projectDir,
|
|
54
|
+
scope: 'project', // or 'portfolio' + portfolioDir: 'D:/travail/DEV/_brain'
|
|
55
|
+
limit: 8,
|
|
56
|
+
now: new Date().toISOString(), // optional injected clock → recency decay on events
|
|
57
|
+
halfLifeDays: 30,
|
|
58
|
+
});
|
|
59
|
+
// → [{ score, kind: 'event'|'note', source, ref, text, event? }] ranked desc
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
Sources merged: `journal.ndjson` events + `### `-sectioned entries from
|
|
63
|
+
`decisions.md` / `lessons.md` / `patterns.md` in **both** `.bmad/memory/` (north-star)
|
|
64
|
+
and `.agents/memory/` (current pack-memory layout), plus `<portfolioDir>/memory/*.md`
|
|
65
|
+
when `scope: 'portfolio'`.
|
|
66
|
+
|
|
67
|
+
**ChromaDB seam** — pass `backend: { search(query, opts) }` and ranking is delegated
|
|
68
|
+
wholesale to it. The intended production backend is the existing RAG stack
|
|
69
|
+
(`mcp-server/rag.py`: ChromaDB + SentenceTransformers — `registry.yaml → memory.index`).
|
|
70
|
+
Backends must return the same entry shape as the lexical fallback, so callers never
|
|
71
|
+
know which engine served them. The lexical scorer is the zero-dependency fallback for
|
|
72
|
+
machines without Python provisioned.
|
|
73
|
+
|
|
74
|
+
### 3. Reward + pattern score
|
|
75
|
+
|
|
76
|
+
```js
|
|
77
|
+
const reward = mj.computeReward({ evalScore: 0.8, acceptance: true, ci: 'fail' });
|
|
78
|
+
// weights eval 0.5 / acceptance 0.3 / ci 0.2 (registry.yaml → memory.reward_signal.inputs)
|
|
79
|
+
// missing signals renormalize the remaining weights; result always in [0, 1]
|
|
80
|
+
|
|
81
|
+
let score = mj.INITIAL_PATTERN_SCORE; // { elo: 1200, alpha: 1, beta: 1, mean: 0.5, ... }
|
|
82
|
+
score = mj.updatePatternScore(score, reward, { ts: eventTs });
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
Two complementary estimators per pattern:
|
|
86
|
+
|
|
87
|
+
- **Elo** (`k=32`, baseline 1200): `elo' = elo + K·(reward − expected)` — fast-moving,
|
|
88
|
+
ordinal, used to **rank** patterns in recall. Fresh pattern + reward 1 → 1216.
|
|
89
|
+
- **Decayed Bayesian (Beta)**: evidence decays multiplicatively toward the uniform
|
|
90
|
+
prior (1,1) — per-update (`decay=0.98`) and time-based (`halfLifeDays=90`, only when
|
|
91
|
+
`ts` is injected) — so `mean = α/(α+β)` tracks the **recent** success rate, used for
|
|
92
|
+
**promotion thresholds** (`candidate → validated → deprecated` in `patterns.md`).
|
|
93
|
+
|
|
94
|
+
Pure function: never mutates input, never reads the clock.
|
|
95
|
+
|
|
96
|
+
### 4. Governance guard
|
|
97
|
+
|
|
98
|
+
```js
|
|
99
|
+
const p = mj.proposePromotion({ patternId: 'chromadb batch ingestion', ts, reason: 'mean 0.82 / 12 events', evidence: [eventIds] });
|
|
100
|
+
mj.appendPromotion(projectDir, p); // persisted as PROPOSED, always
|
|
101
|
+
// ... a human / Shield reviewer flips it to APPROVED with approvedBy elsewhere ...
|
|
102
|
+
mj.assertPromotionApplicable(approved); // the gate every apply path MUST call
|
|
103
|
+
```
|
|
104
|
+
|
|
105
|
+
Governed by **Shield** (`registry.yaml → memory.reward_signal.governed_by`): bounded
|
|
106
|
+
self-modification, versioned (append-only ndjson) and reversible (a promotion record
|
|
107
|
+
never rewrites history; a rollback is just another proposal).
|
|
108
|
+
|
|
109
|
+
## Testing
|
|
110
|
+
|
|
111
|
+
```
|
|
112
|
+
npx jest tests/unit/memory-journal.test.js
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
32 tests: append/recall round-trip on a tmp dir, corrupt-line tolerance, recency decay
|
|
116
|
+
with injected clock, portfolio scope, backend-seam delegation, exact Elo/Beta math,
|
|
117
|
+
purity, and the anti-tamper governance guard.
|
|
118
|
+
|
|
119
|
+
## Future wiring (done by the orchestrator, not this module)
|
|
120
|
+
|
|
121
|
+
- **MCP tools** `memory.write` / `memory.recall` in `mcp-server/` — thin wrappers over
|
|
122
|
+
this ndjson contract so every MCP-capable CLI shares one memory (Pillar 5).
|
|
123
|
+
- **CLI commands** `bmad-plus memory log|recall|promote` in `tools/cli/commands/memory.js`.
|
|
124
|
+
- **Zecher consolidation**: pack-memory's archivist reads `journal.ndjson` during
|
|
125
|
+
session consolidation and proposes pattern promotions via `proposePromotion()`.
|
|
Binary file
|