@entelligentsia/forgecli 0.11.2 → 0.15.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 +324 -0
- package/README.md +2 -1
- package/dist/CHANGELOG-forge-plugin.md +210 -0
- package/dist/bin/forge.js +20 -1
- package/dist/bin/forge.js.map +1 -1
- package/dist/extensions/forgecli/ask-user-tool.js +32 -20
- package/dist/extensions/forgecli/ask-user-tool.js.map +1 -1
- package/dist/extensions/forgecli/config-layer.d.ts +15 -0
- package/dist/extensions/forgecli/config-layer.js +4 -1
- package/dist/extensions/forgecli/config-layer.js.map +1 -1
- package/dist/extensions/forgecli/config-writer.js +4 -1
- package/dist/extensions/forgecli/config-writer.js.map +1 -1
- package/dist/extensions/forgecli/enhance.js +1 -1
- package/dist/extensions/forgecli/enhance.js.map +1 -1
- package/dist/extensions/forgecli/fix-bug.js +31 -1
- package/dist/extensions/forgecli/fix-bug.js.map +1 -1
- package/dist/extensions/forgecli/forge-cli-schema.json +19 -0
- package/dist/extensions/forgecli/forge-tools.js +80 -0
- package/dist/extensions/forgecli/forge-tools.js.map +1 -1
- package/dist/extensions/forgecli/forge-update-command.js +24 -18
- package/dist/extensions/forgecli/forge-update-command.js.map +1 -1
- package/dist/extensions/forgecli/friction-emit.d.ts +97 -0
- package/dist/extensions/forgecli/friction-emit.js +246 -0
- package/dist/extensions/forgecli/friction-emit.js.map +1 -0
- package/dist/extensions/forgecli/health-check.d.ts +10 -0
- package/dist/extensions/forgecli/health-check.js +160 -8
- package/dist/extensions/forgecli/health-check.js.map +1 -1
- package/dist/extensions/forgecli/hook-dispatcher.js +24 -2
- package/dist/extensions/forgecli/hook-dispatcher.js.map +1 -1
- package/dist/extensions/forgecli/hooks/write-guard.js +5 -1
- package/dist/extensions/forgecli/hooks/write-guard.js.map +1 -1
- package/dist/extensions/forgecli/index.js +29 -5
- package/dist/extensions/forgecli/index.js.map +1 -1
- package/dist/extensions/forgecli/lib/store-error-remediation.d.ts +65 -0
- package/dist/extensions/forgecli/lib/store-error-remediation.js +298 -0
- package/dist/extensions/forgecli/lib/store-error-remediation.js.map +1 -0
- package/dist/extensions/forgecli/regenerate.d.ts +22 -0
- package/dist/extensions/forgecli/regenerate.js +133 -3
- package/dist/extensions/forgecli/regenerate.js.map +1 -1
- package/dist/extensions/forgecli/run-sprint.js +16 -1
- package/dist/extensions/forgecli/run-sprint.js.map +1 -1
- package/dist/extensions/forgecli/run-task.js +30 -8
- package/dist/extensions/forgecli/run-task.js.map +1 -1
- package/dist/extensions/forgecli/skill-curation-flag.d.ts +21 -0
- package/dist/extensions/forgecli/skill-curation-flag.js +71 -0
- package/dist/extensions/forgecli/skill-curation-flag.js.map +1 -0
- package/dist/extensions/forgecli/skill-curator-subagent.d.ts +101 -0
- package/dist/extensions/forgecli/skill-curator-subagent.js +342 -0
- package/dist/extensions/forgecli/skill-curator-subagent.js.map +1 -0
- package/dist/extensions/forgecli/skill-retriever.d.ts +84 -0
- package/dist/extensions/forgecli/skill-retriever.js +246 -0
- package/dist/extensions/forgecli/skill-retriever.js.map +1 -0
- package/dist/extensions/forgecli/skill-usage-tracker.d.ts +91 -0
- package/dist/extensions/forgecli/skill-usage-tracker.js +224 -0
- package/dist/extensions/forgecli/skill-usage-tracker.js.map +1 -0
- package/dist/extensions/forgecli/store-resolver.d.ts +18 -0
- package/dist/extensions/forgecli/store-resolver.js +44 -4
- package/dist/extensions/forgecli/store-resolver.js.map +1 -1
- package/dist/extensions/forgecli/store-validator.d.ts +3 -0
- package/dist/extensions/forgecli/store-validator.js +4 -2
- package/dist/extensions/forgecli/store-validator.js.map +1 -1
- package/dist/forge-payload/.base-pack/personas/supervisor.md +9 -0
- package/dist/forge-payload/.base-pack/workflows/enhance.md +344 -18
- package/dist/forge-payload/.claude-plugin/plugin.json +1 -1
- package/dist/forge-payload/.schemas/event.schema.json +20 -2
- package/dist/forge-payload/.schemas/migrations.json +112 -0
- package/dist/forge-payload/.schemas/proposal.schema.json +40 -0
- package/dist/forge-payload/agents/store-query-validator.md +103 -0
- package/dist/forge-payload/agents/tomoshibi.md +185 -0
- package/dist/forge-payload/commands/regenerate.md +109 -20
- package/dist/forge-payload/hooks/check-update.js +378 -0
- package/dist/forge-payload/hooks/forge-permissions.js +158 -0
- package/dist/forge-payload/hooks/triage-error.js +71 -0
- package/dist/forge-payload/hooks/validate-write.js +236 -0
- package/dist/forge-payload/integrity.json +32 -0
- package/dist/forge-payload/meta/workflows/meta-enhance.md +344 -18
- package/dist/forge-payload/schemas/structure-manifest.json +511 -0
- package/dist/forge-payload/tools/build-persona-pack.cjs +120 -11
- package/dist/forge-payload/tools/compression-gate.cjs +192 -0
- package/dist/forge-payload/tools/delete-candidate-detector.cjs +114 -0
- package/dist/forge-payload/tools/judge-proposal.cjs +177 -0
- package/dist/forge-payload/tools/manage-versions.cjs +132 -4
- package/dist/forge-payload/tools/queue-drain.cjs +152 -0
- package/dist/forge-payload/tools/replay-scoring.cjs +117 -0
- package/node_modules/@mariozechner/clipboard/package.json +2 -1
- package/node_modules/@mariozechner/clipboard-linux-x64-musl/README.md +3 -0
- package/node_modules/@mariozechner/clipboard-linux-x64-musl/clipboard.linux-x64-musl.node +0 -0
- package/node_modules/@mariozechner/clipboard-linux-x64-musl/package.json +25 -0
- package/package.json +4 -2
|
@@ -50,6 +50,40 @@ Phases 2 and 3 write proposal artifacts to `.forge/enhancement-proposals/`. This
|
|
|
50
50
|
distinct from `.forge/enhancements/` (FR-007, S14 scope). This workflow uses `mkdir -p` before
|
|
51
51
|
writing the first proposal artifact to avoid assuming the directory exists. No conflict with S14.
|
|
52
52
|
|
|
53
|
+
### Sub-directory: `.forge/enhancement-proposals/queue/`
|
|
54
|
+
|
|
55
|
+
FORGE-S24-T07 introduces a project-local **enhancement queue** at
|
|
56
|
+
`.forge/enhancement-proposals/queue/<sprintId>/<taskId>-<ts>.json` — one file per
|
|
57
|
+
per-task curator run (T10). The queue is **append-only**: each curator run writes
|
|
58
|
+
a fresh file (the ISO compact `<ts>` suffix differentiates writes; nothing is
|
|
59
|
+
overwritten). Phase 2 drains the queue at sprint close, dedupes by
|
|
60
|
+
`{op, target_path, sha256(diff_body)}`, and feeds the merged batch into the
|
|
61
|
+
existing recurrence → delete-candidate → compression-gate → judge pipeline.
|
|
62
|
+
The result: **one batched review prompt per sprint, not one per task** (paper
|
|
63
|
+
§3.2.1 grouped reward). The drain is read-only — Phase 2 never deletes queue
|
|
64
|
+
files; operators triage them during retrospective if needed.
|
|
65
|
+
|
|
66
|
+
**Per-task curator (T10) write contract.** A curator MUST write via
|
|
67
|
+
`forge/tools/queue-drain.cjs` to preserve the append-only invariant:
|
|
68
|
+
|
|
69
|
+
```sh
|
|
70
|
+
node -e "
|
|
71
|
+
const { appendToQueue } = require('./forge/tools/queue-drain.cjs');
|
|
72
|
+
appendToQueue({
|
|
73
|
+
queueRoot: '.forge/enhancement-proposals/queue',
|
|
74
|
+
sprintId: process.env.FORGE_SPRINT_ID,
|
|
75
|
+
taskId: process.env.FORGE_TASK_ID,
|
|
76
|
+
ts: new Date().toISOString().replace(/[-:]|\\.\\d{3}/g, ''),
|
|
77
|
+
proposals: PROPOSALS_ARRAY,
|
|
78
|
+
});
|
|
79
|
+
"
|
|
80
|
+
```
|
|
81
|
+
|
|
82
|
+
`appendToQueue` throws if the exact file path already exists; curators MUST
|
|
83
|
+
choose a fresh `ts` per run rather than overwriting. The drain is empty-safe:
|
|
84
|
+
if no curator ever wrote (queue dir missing) or no files exist in the sprint
|
|
85
|
+
sub-dir, Phase 2 reports "no proposals" and exits cleanly (AC5).
|
|
86
|
+
|
|
53
87
|
## Confidence gating (Phase 1)
|
|
54
88
|
|
|
55
89
|
A key substitution is **high-confidence** when there is exactly one unambiguous signal source
|
|
@@ -64,9 +98,9 @@ Receive the phase flag from the command invocation:
|
|
|
64
98
|
|
|
65
99
|
| Flag | Mode |
|
|
66
100
|
|------|------|
|
|
67
|
-
| `--phase 1` or `--auto` | Auto-apply: placeholder fills only |
|
|
68
|
-
| `--phase 2` | Propose-diffs: sprint artifact + friction scan |
|
|
69
|
-
| `--phase 3` | Drift detection: full codebase vs structural-element comparison |
|
|
101
|
+
| `--phase 1` or `--auto` | Auto-apply: placeholder fills only — **use after** `/forge:init` completes to fill `{{KEY}}` placeholders from project signals |
|
|
102
|
+
| `--phase 2` | Propose-diffs: sprint artifact + friction scan — **use after** a sprint completes to turn friction events into persona/skill enrichments |
|
|
103
|
+
| `--phase 3` | Drift detection: full codebase vs structural-element comparison — **use on-demand** or after `/forge:calibrate` to detect stale references |
|
|
70
104
|
|
|
71
105
|
Default to `--phase 3` if no phase flag is given.
|
|
72
106
|
|
|
@@ -191,28 +225,320 @@ Invoked by T09 post-sprint hook or manually via `/forge:enhance --phase 2`.
|
|
|
191
225
|
"
|
|
192
226
|
```
|
|
193
227
|
|
|
194
|
-
|
|
228
|
+
1a. **Drain enhancement queue** (FORGE-S24-T07) — read per-task curator
|
|
229
|
+
proposals from `.forge/enhancement-proposals/queue/<sprintId>/`, dedupe by
|
|
230
|
+
`{op, target_path, sha256(diff_body)}`, and produce a `queuedProposals`
|
|
231
|
+
array that joins the synthesised proposals from step 5. This is what makes
|
|
232
|
+
the review **batched** rather than per-task (paper §3.2.1):
|
|
233
|
+
|
|
234
|
+
```sh
|
|
235
|
+
node -e "
|
|
236
|
+
const { drainQueue } = require('./forge/tools/queue-drain.cjs');
|
|
237
|
+
const drained = drainQueue({
|
|
238
|
+
queueRoot: '.forge/enhancement-proposals/queue',
|
|
239
|
+
sprintId: process.env.FORGE_SPRINT_ID,
|
|
240
|
+
});
|
|
241
|
+
process.stdout.write(JSON.stringify(drained));
|
|
242
|
+
"
|
|
243
|
+
```
|
|
244
|
+
|
|
245
|
+
Contract (per `forge/tools/queue-drain.cjs`):
|
|
246
|
+
- Returns `{ proposals: [...], files: [...], errors: [...] }`. `proposals`
|
|
247
|
+
is the deduped union of every per-task curator file in the sprint
|
|
248
|
+
sub-dir. `files` is the lexicographic-sorted list of source paths (used
|
|
249
|
+
by step 6 to log provenance). `errors` carries any malformed JSON files
|
|
250
|
+
skipped during read — log them, do not abort.
|
|
251
|
+
- Empty / missing queue → empty result. The drain never throws on absent
|
|
252
|
+
queue dir (first-run or no curators registered yet, AC5).
|
|
253
|
+
- The drain is read-only. Operators are responsible for queue triage
|
|
254
|
+
after sprint close.
|
|
255
|
+
|
|
256
|
+
2. **Zero-input guard**: If both the friction event list AND `queuedProposals`
|
|
257
|
+
are empty, print:
|
|
258
|
+
```
|
|
259
|
+
No friction events or queued proposals for the active sprint — nothing to enhance.
|
|
260
|
+
```
|
|
261
|
+
and exit Phase 2 immediately (skip steps 3–9; emit the enhancement event with `"notes": "{\"phase\":2,\"frictionCount\":0,\"queuedCount\":0}"`). Do not create `.forge/enhancement-proposals/` when there are no proposals.
|
|
262
|
+
|
|
263
|
+
3. **Deduplicate** friction events by composite key `workflow + persona + issue`. Keep the most
|
|
195
264
|
recent occurrence of each composite key.
|
|
196
265
|
|
|
197
|
-
|
|
266
|
+
4. **Read most recent completed sprint** from `.forge/store/sprints/` (status `done` or
|
|
198
267
|
`retrospective-done`), sorted by completion date. Read its task records from
|
|
199
268
|
`.forge/store/tasks/` filtered by the sprint ID.
|
|
200
269
|
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
270
|
+
5. **Synthesize enrichment proposals** — for each friction event, classify the proposed
|
|
271
|
+
change into exactly one of three ops (see `forge/schemas/proposal.schema.json`):
|
|
272
|
+
|
|
273
|
+
| `op` | When to use |
|
|
274
|
+
|-----------------|-----------------------------------------------------------------------------|
|
|
275
|
+
| `insert_skill` | A new skill / persona / kb_docs reference is needed; target file does not yet carry the guidance. |
|
|
276
|
+
| `update_skill` | An existing skill or persona file needs revised guidance — e.g., add a routing pattern reference to `deps.kb_docs`, replace a stale instruction. |
|
|
277
|
+
| `delete_skill` | A skill is unused, redundant, or stale (`skill_unused` / `skill_redundant` / `skill_stale` friction subkinds); target file or section should be removed. |
|
|
278
|
+
|
|
279
|
+
For each proposal capture **at minimum** the schema-required triplet
|
|
280
|
+
`{op, target_path, diff_body}` plus optional `rationale` and `sourceFrictionIds`.
|
|
281
|
+
`sourceFrictionIds` MUST carry the `eventId` of every friction event that
|
|
282
|
+
contributed to the proposal — the next step depends on it to resolve the
|
|
283
|
+
originating task for the recurrence scan.
|
|
284
|
+
For large committed file sets (> 5 files in the sprint), also check whether
|
|
285
|
+
`engineer-skills.md` or `architect-skills.md` should be updated (`update_skill`).
|
|
286
|
+
The op classification is the foundation for the downstream judge (T03),
|
|
287
|
+
delete-candidate detection (T05), compression gate (T06), and queue drain (T07).
|
|
288
|
+
|
|
289
|
+
**Merge with queued proposals (T07).** Concatenate the synthesised
|
|
290
|
+
proposals built in this step with the `queuedProposals` array from
|
|
291
|
+
step 1a, then dedupe the combined array with the same key the drain
|
|
292
|
+
uses (`{op, target_path, sha256(diff_body)}`) so a friction-synthesised
|
|
293
|
+
proposal that happens to be byte-identical to a curator-queued one
|
|
294
|
+
collapses. Use:
|
|
207
295
|
|
|
208
|
-
|
|
296
|
+
```sh
|
|
297
|
+
node -e "
|
|
298
|
+
const { dedupeProposals } = require('./forge/tools/queue-drain.cjs');
|
|
299
|
+
// synthesised = proposals built above from friction events.
|
|
300
|
+
// queued = drained.proposals from step 1a.
|
|
301
|
+
const merged = dedupeProposals(synthesised.concat(queued));
|
|
302
|
+
process.stdout.write(JSON.stringify(merged));
|
|
303
|
+
"
|
|
304
|
+
```
|
|
305
|
+
|
|
306
|
+
The merged array is what feeds steps 5a (recurrence) → 5b (delete
|
|
307
|
+
candidates) → 5b.5 (compression gate) → 5c (judge) — a single batched
|
|
308
|
+
pipeline, never one per task (AC4).
|
|
309
|
+
|
|
310
|
+
5a. **Cross-task replay scoring (recurrence boost)** — before writing the
|
|
311
|
+
artifact, stamp each proposal with `recurrence_count` and
|
|
312
|
+
`recurrence_task_ids` so the T03 judge can score "this friction recurred
|
|
313
|
+
across N tasks" rather than treating every signal as a singleton:
|
|
314
|
+
|
|
315
|
+
```sh
|
|
316
|
+
node -e "
|
|
317
|
+
const { annotateProposals } = require('./forge/tools/replay-scoring.cjs');
|
|
318
|
+
// friction = deduped friction events from step 3, each carrying eventId,
|
|
319
|
+
// taskId, subkind, evidence.skillId (orchestrator-stamped).
|
|
320
|
+
// proposals = array built in step 5.
|
|
321
|
+
// taskOrder = task IDs of the most-recent sprint sorted by completion
|
|
322
|
+
// order — same source as step 4.
|
|
323
|
+
const annotated = annotateProposals(proposals, friction, taskOrder);
|
|
324
|
+
process.stdout.write(JSON.stringify(annotated));
|
|
325
|
+
"
|
|
326
|
+
```
|
|
327
|
+
|
|
328
|
+
Contract (per `forge/tools/replay-scoring.cjs`):
|
|
329
|
+
- `recurrence_count` is the number of distinct tasks (origin task + later
|
|
330
|
+
tasks in `taskOrder`) whose friction events match the proposal's
|
|
331
|
+
originating `(subkind, evidence.skillId)` pair. Always `>= 1`.
|
|
332
|
+
- `recurrence_task_ids` is the `taskOrder`-sorted list of those task IDs.
|
|
333
|
+
- Proposals whose `sourceFrictionIds` cannot be resolved (no matching
|
|
334
|
+
`eventId` in the friction set, or the resolved event lacks
|
|
335
|
+
`subkind`/`evidence.skillId`) receive `recurrence_count: 1` and an empty
|
|
336
|
+
`recurrence_task_ids: []` — neutral signal, not silent failure.
|
|
337
|
+
- The annotator returns new proposal objects; the input array is not
|
|
338
|
+
mutated.
|
|
339
|
+
|
|
340
|
+
5b. **Delete-candidate detection (3-sprint zero-use)** — scan `skill_usage`
|
|
341
|
+
events across the trailing 3 sprints and emit a `delete_skill` proposal
|
|
342
|
+
for every skill with zero retrieval AND zero invocation across the
|
|
343
|
+
window. This is the only mechanism by which the skill repository shrinks:
|
|
344
|
+
|
|
345
|
+
```sh
|
|
346
|
+
node -e "
|
|
347
|
+
const { buildDeleteProposals } = require('./forge/tools/delete-candidate-detector.cjs');
|
|
348
|
+
// skillUsageEvents = all events with type === 'skill_usage' across the
|
|
349
|
+
// sprints in scope (collected via the same Step 1
|
|
350
|
+
// walker, filtered by type instead of friction).
|
|
351
|
+
// sprintOrder = sprint IDs sorted by completion order (oldest →
|
|
352
|
+
// newest). The detector takes the trailing windowSize
|
|
353
|
+
// entries.
|
|
354
|
+
// windowSize = 3 by default; configurable. Defined as the trailing
|
|
355
|
+
// N sprints of sprintOrder.
|
|
356
|
+
// targetPathFor = (skillId) => the on-disk path of the skill file to
|
|
357
|
+
// delete. Workflow chooses the mapping convention.
|
|
358
|
+
const deletes = buildDeleteProposals({
|
|
359
|
+
events: skillUsageEvents,
|
|
360
|
+
sprintOrder,
|
|
361
|
+
windowSize: 3,
|
|
362
|
+
targetPathFor: (skillId) => 'forge/skills/' + skillId + '.md',
|
|
363
|
+
});
|
|
364
|
+
process.stdout.write(JSON.stringify(deletes));
|
|
365
|
+
"
|
|
366
|
+
```
|
|
367
|
+
|
|
368
|
+
Append the resulting `delete_skill` proposals to the proposal array from
|
|
369
|
+
step 5/5a before step 6. Each delete proposal already carries
|
|
370
|
+
`recurrence_count: 1` and `recurrence_task_ids: []` (the annotator from
|
|
371
|
+
step 5a is for friction-derived proposals; delete candidates come from
|
|
372
|
+
usage telemetry, not friction, so recurrence is neutral by construction).
|
|
373
|
+
|
|
374
|
+
5b.5. **Compression gate (reject >20% growth without 3+ frictions)** — a cheap
|
|
375
|
+
deterministic filter that runs BEFORE the LLM judge (step 5c). Any
|
|
376
|
+
`update_skill` proposal that would grow the target file by more than 20%
|
|
377
|
+
(byte-wise, UTF-8) must be backed by at least 3 supporting friction events;
|
|
378
|
+
otherwise it is rejected here and never reaches the judge. `insert_skill`
|
|
379
|
+
and `delete_skill` proposals pass through unconditionally — insert growth
|
|
380
|
+
is handled by the judge's `body_under_2kb` axis and delete only shrinks.
|
|
381
|
+
|
|
382
|
+
Why a pre-judge gate? Judging is expensive. Unbounded skill-body growth is
|
|
383
|
+
the classic SkillOS failure mode — pasting pages of trajectory copy-paste to
|
|
384
|
+
"patch" a friction. It is cheap to detect deterministically and wasteful to
|
|
385
|
+
ask the judge to rule on.
|
|
386
|
+
|
|
387
|
+
```sh
|
|
388
|
+
node -e "
|
|
389
|
+
const fs = require('node:fs');
|
|
390
|
+
const path = require('node:path');
|
|
391
|
+
const { filterProposals } = require('./forge/tools/compression-gate.cjs');
|
|
392
|
+
// proposals = post-5b array (synthesis + recurrence + delete-candidates).
|
|
393
|
+
// PROJECT_ROOT resolves the target_path; forge plugin source is the source
|
|
394
|
+
// of truth for current bodies. The workflow renders the diff via its own
|
|
395
|
+
// applyProposalDiff helper (left abstract here — the gate is body-agnostic).
|
|
396
|
+
const projectRoot = process.env.PROJECT_ROOT;
|
|
397
|
+
const result = filterProposals({
|
|
398
|
+
proposals,
|
|
399
|
+
currentBodyFor: (p) => {
|
|
400
|
+
const abs = path.join(projectRoot, p.target_path);
|
|
401
|
+
try { return fs.readFileSync(abs, 'utf8'); }
|
|
402
|
+
catch (e) { return ''; } // insert_skill or missing file → empty
|
|
403
|
+
},
|
|
404
|
+
newBodyFor: (p) => applyProposalDiff(currentBodyFor(p), p),
|
|
405
|
+
// Default supporting count = proposal.sourceFrictionIds.length. Override
|
|
406
|
+
// if the policy is 'count frictions citing the same skill across the
|
|
407
|
+
// sprint' rather than 'count citations on the proposal itself'.
|
|
408
|
+
});
|
|
409
|
+
const proposalsAfterGate = result.admitted;
|
|
410
|
+
const compressionRejections = result.rejected; // [{ proposal, ...evaluation }]
|
|
411
|
+
process.stdout.write(JSON.stringify({ kept: proposalsAfterGate, rejected: compressionRejections }));
|
|
412
|
+
"
|
|
413
|
+
```
|
|
414
|
+
|
|
415
|
+
**Logging gate rejections.** Append every rejection from this step to the
|
|
416
|
+
same `phase2-<timestamp>-rejections.json` sibling that step 5c uses, with
|
|
417
|
+
the rejection record carrying `{ proposal, admit: false,
|
|
418
|
+
reason: 'compression_gate_growth_unsupported', growthRatio, currentBytes,
|
|
419
|
+
newBytes, supportingFrictionCount, threshold, minSupportingFrictions }`.
|
|
420
|
+
This keeps every drop — gate or judge — traceable in one place.
|
|
421
|
+
|
|
422
|
+
Contract (per `forge/tools/compression-gate.cjs`):
|
|
423
|
+
- `GROWTH_THRESHOLD === 0.20`; comparison is **strict** (`> 0.20`). A
|
|
424
|
+
proposal at exactly 20% growth admits without friction support.
|
|
425
|
+
- `MIN_SUPPORTING_FRICTIONS === 3`. Two or fewer citations is not enough.
|
|
426
|
+
- An update on an empty current body yields `growthRatio: Infinity`; the
|
|
427
|
+
friction-support rule still applies.
|
|
428
|
+
- Negative growth (shrink) admits unconditionally.
|
|
429
|
+
- `filterProposals` partitions the input array preserving order; the
|
|
430
|
+
output `rejected` array carries the structured evaluation alongside the
|
|
431
|
+
original proposal.
|
|
432
|
+
|
|
433
|
+
5c. **LLM-judge gate (Sonnet rubric, drop <3/5)** — score every proposal
|
|
434
|
+
against the 5-axis rubric and drop low-signal proposals before
|
|
435
|
+
presentation. The rubric is single-sourced in
|
|
436
|
+
`forge/tools/judge-proposal.cjs`:
|
|
437
|
+
|
|
438
|
+
| Axis (0..5) | What it measures |
|
|
439
|
+
|---|---|
|
|
440
|
+
| `specificity` | Names a concrete target_path beyond `forge/skills/*` floor; carries a non-trivial rationale; recurrence trail boosts. |
|
|
441
|
+
| `when_not_to_use` | Body contains a literal "When NOT to use" section. |
|
|
442
|
+
| `no_trajectory_copy_paste` | No long verbatim runs or unbroken non-whitespace blocks (>= 400 bytes) that suggest pasted trajectory log. |
|
|
443
|
+
| `body_under_2kb` | `Buffer.byteLength(diff_body, 'utf8') <= 2048`. |
|
|
444
|
+
| `cites_friction` | Proposal carries at least one `sourceFrictionIds` entry; multiple citations or recurrence boost the score. |
|
|
445
|
+
|
|
446
|
+
For each proposal in the post-5b array, the workflow asks Sonnet to
|
|
447
|
+
apply the rubric and emit per-axis 0..5 scores; in the absence of an
|
|
448
|
+
LLM call, the deterministic `scoreProposal(proposal)` helper in
|
|
449
|
+
`judge-proposal.cjs` is used as both the fallback scorer and the
|
|
450
|
+
validation contract for Sonnet-produced scores (single source of truth
|
|
451
|
+
for the rubric definition).
|
|
452
|
+
|
|
453
|
+
```sh
|
|
454
|
+
node -e "
|
|
455
|
+
const {
|
|
456
|
+
scoreProposal,
|
|
457
|
+
decideJudgement,
|
|
458
|
+
} = require('./forge/tools/judge-proposal.cjs');
|
|
459
|
+
// proposals = post-5b array of proposal records.
|
|
460
|
+
const judged = proposals.map((p) => {
|
|
461
|
+
const scored = scoreProposal(p);
|
|
462
|
+
const decision = decideJudgement(scored);
|
|
463
|
+
return { proposal: p, ...decision };
|
|
464
|
+
});
|
|
465
|
+
const kept = judged.filter((j) => j.verdict === 'keep').map((j) => j.proposal);
|
|
466
|
+
const dropped = judged.filter((j) => j.verdict === 'drop');
|
|
467
|
+
process.stdout.write(JSON.stringify({ kept, dropped }));
|
|
468
|
+
"
|
|
469
|
+
```
|
|
470
|
+
|
|
471
|
+
Contract (per `forge/tools/judge-proposal.cjs`):
|
|
472
|
+
- `scoreProposal(proposal)` returns `{ axes, average }` with `axes`
|
|
473
|
+
keyed by every entry in `RUBRIC_AXES` and `average` rounded to one
|
|
474
|
+
decimal place.
|
|
475
|
+
- `decideJudgement({ axes })` returns
|
|
476
|
+
`{ verdict, average, axes, reason }`. `verdict === 'drop'` iff
|
|
477
|
+
`average < 3` (strictly less than); ties at exactly 3.0 keep.
|
|
478
|
+
- `decideJudgement` fails loud on missing or out-of-range axes — the
|
|
479
|
+
judge will NOT silently coerce a malformed score sheet into a verdict.
|
|
480
|
+
|
|
481
|
+
**Logging dropped proposals (AC3).** Every rejection MUST be persisted
|
|
482
|
+
for retro review. Replace the proposal array passed to step 6 with the
|
|
483
|
+
`kept` list, and append the `dropped` list to
|
|
484
|
+
`$PROJECT_ROOT/.forge/enhancement-proposals/phase2-<timestamp>-rejections.json`
|
|
485
|
+
as a sibling artifact. Each rejection record carries the original
|
|
486
|
+
proposal alongside `{ verdict: 'drop', average, axes, reason }`. The
|
|
487
|
+
markdown summary written in step 6 SHOULD include a "Dropped (N)" line
|
|
488
|
+
pointing at the rejections file when N > 0.
|
|
489
|
+
|
|
490
|
+
**Carry-over caveat** — the rubric is deterministic; Sonnet's role is
|
|
491
|
+
to add semantic judgement to axes that the heuristic scorer
|
|
492
|
+
approximates (specificity in particular). When Sonnet is invoked, its
|
|
493
|
+
per-axis scores MUST be validated against the 0..5 range via the same
|
|
494
|
+
`validateAxes` invariant `decideJudgement` enforces. Operators
|
|
495
|
+
investigating an unexpected drop should consult the per-axis trace in
|
|
496
|
+
`reason`.
|
|
497
|
+
|
|
498
|
+
Contract (per `forge/tools/delete-candidate-detector.cjs`):
|
|
499
|
+
- A skill qualifies for deletion iff it has at least one `skill_usage`
|
|
500
|
+
event inside the trailing window AND every in-window observation has
|
|
501
|
+
`retrieved === false` AND `used === false`. Any single `retrieved: true`
|
|
502
|
+
or `used: true` event disqualifies the skill.
|
|
503
|
+
- Skills with zero observations in the window are NOT proposed — this
|
|
504
|
+
case is indistinguishable from a newly-added skill that hasn't been
|
|
505
|
+
loaded yet, so silence is the safe default.
|
|
506
|
+
- Each proposal carries `window_size`, `window_sprint_ids`, and a
|
|
507
|
+
`sourceFrictionIds: []` (delete candidates derive from usage telemetry,
|
|
508
|
+
not friction).
|
|
509
|
+
|
|
510
|
+
**Carry-over caveat** — the trailing-3-sprint window is only meaningful
|
|
511
|
+
once 3 sprints have actually elapsed since `skill_usage` event emission
|
|
512
|
+
landed in FORGE-S24-T01 (forge 0.45.1). During the carry-over period the
|
|
513
|
+
detector still runs over whatever sprintOrder it receives, but the
|
|
514
|
+
signal is noisier: a skill flagged after only one or two sprints of
|
|
515
|
+
history may simply be new or temporarily idle. Operators should treat
|
|
516
|
+
delete proposals from short-history runs as advisory until the full
|
|
517
|
+
window is populated.
|
|
518
|
+
|
|
519
|
+
6. **Write proposal artifact**:
|
|
209
520
|
```sh
|
|
210
521
|
mkdir -p "$PROJECT_ROOT/.forge/enhancement-proposals"
|
|
211
522
|
```
|
|
212
|
-
Write
|
|
213
|
-
|
|
523
|
+
Write **two** outputs for each Phase 2 run (using the `kept` list from
|
|
524
|
+
step 5c — dropped proposals are persisted separately to the
|
|
525
|
+
`phase2-<timestamp>-rejections.json` sibling described in step 5c):
|
|
526
|
+
|
|
527
|
+
- `phase2-<timestamp>.md` — human-readable markdown, one section per proposal,
|
|
528
|
+
showing op + target_path + a fenced diff block.
|
|
529
|
+
- `phase2-<timestamp>.json` — machine-readable array of proposal records, each
|
|
530
|
+
conforming to `forge/schemas/proposal.schema.json` (required keys: `op`,
|
|
531
|
+
`target_path`, `diff_body`; `op` ∈ {insert_skill, update_skill, delete_skill};
|
|
532
|
+
optional `recurrence_count` ≥ 1 and `recurrence_task_ids` populated by step 5a).
|
|
533
|
+
|
|
534
|
+
**Back-compat on read** — pre-0.45.2 proposal records lack `op`. Downstream
|
|
535
|
+
consumers MUST route legacy records through
|
|
536
|
+
`forge/tools/proposal-normalize.cjs:normaliseProposal()` which defaults the
|
|
537
|
+
missing `op` to `insert_skill` (the only op the prior insert-biased flow
|
|
538
|
+
could produce). Do NOT silently coerce — call the helper explicitly so the
|
|
539
|
+
normalisation is auditable.
|
|
214
540
|
|
|
215
|
-
|
|
541
|
+
7. **Present to user**:
|
|
216
542
|
```
|
|
217
543
|
## Phase 2 Enhancement Proposals
|
|
218
544
|
|
|
@@ -221,13 +547,13 @@ Invoked by T09 post-sprint hook or manually via `/forge:enhance --phase 2`.
|
|
|
221
547
|
[A] Apply all [r] Review individually [n] Skip
|
|
222
548
|
```
|
|
223
549
|
|
|
224
|
-
|
|
550
|
+
8. **On approval** — for each approved change:
|
|
225
551
|
- Apply the edit in-place.
|
|
226
552
|
- Call `manage-versions.cjs add-snapshot --source post-sprint:<SPRINT_ID> --enhanced-elements <list>`.
|
|
227
553
|
|
|
228
|
-
|
|
554
|
+
9. **Emit enhancement event** (same schema as Phase 1, with `"phase": "post-sprint"`).
|
|
229
555
|
|
|
230
|
-
|
|
556
|
+
10. **Report**: N changes applied, M skipped, snapshot written or skipped.
|
|
231
557
|
|
|
232
558
|
---
|
|
233
559
|
|