@entelligentsia/forgecli 0.10.1 → 0.11.3
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 +88 -0
- package/README.md +21 -3
- package/dist/CHANGELOG-forge-plugin.md +49 -0
- package/dist/bin/forge.js +0 -0
- package/dist/extensions/forgecli/add-pipeline.d.ts +19 -0
- package/dist/extensions/forgecli/add-pipeline.js +143 -0
- package/dist/extensions/forgecli/add-pipeline.js.map +1 -0
- package/dist/extensions/forgecli/add-task.d.ts +20 -0
- package/dist/extensions/forgecli/add-task.js +154 -0
- package/dist/extensions/forgecli/add-task.js.map +1 -0
- 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/calibrate.d.ts +61 -0
- package/dist/extensions/forgecli/calibrate.js +488 -0
- package/dist/extensions/forgecli/calibrate.js.map +1 -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/fix-bug.d.ts +9 -1
- package/dist/extensions/forgecli/fix-bug.js +101 -9
- package/dist/extensions/forgecli/fix-bug.js.map +1 -1
- package/dist/extensions/forgecli/forge-commands.js +15 -22
- package/dist/extensions/forgecli/forge-commands.js.map +1 -1
- package/dist/extensions/forgecli/forge-subagent.js +34 -7
- package/dist/extensions/forgecli/forge-subagent.js.map +1 -1
- package/dist/extensions/forgecli/forge-update-command.d.ts +9 -0
- package/dist/extensions/forgecli/forge-update-command.js +106 -7
- package/dist/extensions/forgecli/forge-update-command.js.map +1 -1
- package/dist/extensions/forgecli/health-check.d.ts +32 -1
- package/dist/extensions/forgecli/health-check.js +337 -12
- package/dist/extensions/forgecli/health-check.js.map +1 -1
- package/dist/extensions/forgecli/hook-dispatcher.d.ts +25 -1
- package/dist/extensions/forgecli/hook-dispatcher.js +108 -11
- package/dist/extensions/forgecli/hook-dispatcher.js.map +1 -1
- package/dist/extensions/forgecli/hooks/check-update.d.ts +81 -0
- package/dist/extensions/forgecli/hooks/check-update.js +308 -0
- package/dist/extensions/forgecli/hooks/check-update.js.map +1 -0
- package/dist/extensions/forgecli/hooks/forge-permissions.d.ts +32 -0
- package/dist/extensions/forgecli/hooks/forge-permissions.js +119 -0
- package/dist/extensions/forgecli/hooks/forge-permissions.js.map +1 -0
- package/dist/extensions/forgecli/hooks/triage-error.d.ts +23 -0
- package/dist/extensions/forgecli/hooks/triage-error.js +62 -0
- package/dist/extensions/forgecli/hooks/triage-error.js.map +1 -0
- package/dist/extensions/forgecli/hooks/write-guard.d.ts +28 -0
- package/dist/extensions/forgecli/hooks/write-guard.js +229 -0
- package/dist/extensions/forgecli/hooks/write-guard.js.map +1 -0
- package/dist/extensions/forgecli/index.js +60 -0
- package/dist/extensions/forgecli/index.js.map +1 -1
- package/dist/extensions/forgecli/init-context.d.ts +1 -1
- package/dist/extensions/forgecli/init-context.js +21 -6
- package/dist/extensions/forgecli/init-context.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/materialize.d.ts +16 -0
- package/dist/extensions/forgecli/materialize.js +195 -0
- package/dist/extensions/forgecli/materialize.js.map +1 -0
- package/dist/extensions/forgecli/migrate.d.ts +19 -0
- package/dist/extensions/forgecli/migrate.js +258 -0
- package/dist/extensions/forgecli/migrate.js.map +1 -0
- package/dist/extensions/forgecli/migration-engine.d.ts +111 -0
- package/dist/extensions/forgecli/migration-engine.js +533 -0
- package/dist/extensions/forgecli/migration-engine.js.map +1 -0
- package/dist/extensions/forgecli/quiz-agent.d.ts +17 -0
- package/dist/extensions/forgecli/quiz-agent.js +98 -0
- package/dist/extensions/forgecli/quiz-agent.js.map +1 -0
- package/dist/extensions/forgecli/remove-command.d.ts +17 -0
- package/dist/extensions/forgecli/remove-command.js +124 -0
- package/dist/extensions/forgecli/remove-command.js.map +1 -0
- package/dist/extensions/forgecli/report-bug.d.ts +25 -0
- package/dist/extensions/forgecli/report-bug.js +159 -0
- package/dist/extensions/forgecli/report-bug.js.map +1 -0
- package/dist/extensions/forgecli/retrospective.d.ts +19 -0
- package/dist/extensions/forgecli/retrospective.js +156 -0
- package/dist/extensions/forgecli/retrospective.js.map +1 -0
- package/dist/extensions/forgecli/run-sprint.js +50 -1
- package/dist/extensions/forgecli/run-sprint.js.map +1 -1
- package/dist/extensions/forgecli/run-task.d.ts +9 -1
- package/dist/extensions/forgecli/run-task.js +94 -18
- package/dist/extensions/forgecli/run-task.js.map +1 -1
- package/dist/extensions/forgecli/session-registry.d.ts +27 -2
- package/dist/extensions/forgecli/session-registry.js +52 -1
- package/dist/extensions/forgecli/session-registry.js.map +1 -1
- package/dist/extensions/forgecli/status-command.d.ts +19 -0
- package/dist/extensions/forgecli/status-command.js +140 -0
- package/dist/extensions/forgecli/status-command.js.map +1 -0
- package/dist/extensions/forgecli/store-query.d.ts +22 -0
- package/dist/extensions/forgecli/store-query.js +107 -0
- package/dist/extensions/forgecli/store-query.js.map +1 -0
- package/dist/extensions/forgecli/store-repair.d.ts +17 -0
- package/dist/extensions/forgecli/store-repair.js +123 -0
- package/dist/extensions/forgecli/store-repair.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/extensions/forgecli/thread-switcher.js +213 -28
- package/dist/extensions/forgecli/thread-switcher.js.map +1 -1
- package/dist/extensions/forgecli/update-tools.d.ts +23 -0
- package/dist/extensions/forgecli/update-tools.js +136 -0
- package/dist/extensions/forgecli/update-tools.js.map +1 -0
- package/dist/extensions/forgecli/viewport-theme.js +4 -0
- package/dist/extensions/forgecli/viewport-theme.js.map +1 -1
- package/dist/forge-payload/.base-pack/personas/supervisor.md +9 -0
- package/dist/forge-payload/.base-pack/workflows/enhance.md +17 -11
- package/dist/forge-payload/.claude-plugin/plugin.json +1 -1
- package/dist/forge-payload/.schemas/config.schema.json +83 -0
- package/dist/forge-payload/.schemas/migrations.json +2065 -0
- package/dist/forge-payload/commands/regenerate.md +17 -1
- package/dist/forge-payload/meta/personas/README.md +16 -0
- package/dist/forge-payload/meta/personas/meta-architect.md +70 -0
- package/dist/forge-payload/meta/personas/meta-bug-fixer.md +73 -0
- package/dist/forge-payload/meta/personas/meta-collator.md +72 -0
- package/dist/forge-payload/meta/personas/meta-engineer.md +70 -0
- package/dist/forge-payload/meta/personas/meta-orchestrator.md +71 -0
- package/dist/forge-payload/meta/personas/meta-product-manager.md +82 -0
- package/dist/forge-payload/meta/personas/meta-qa-engineer.md +91 -0
- package/dist/forge-payload/meta/personas/meta-supervisor.md +92 -0
- package/dist/forge-payload/meta/skill-recommendations.md +154 -0
- package/dist/forge-payload/meta/skills/meta-architect-skills.md +43 -0
- package/dist/forge-payload/meta/skills/meta-bug-fixer-skills.md +43 -0
- package/dist/forge-payload/meta/skills/meta-collator-skills.md +41 -0
- package/dist/forge-payload/meta/skills/meta-engineer-skills.md +43 -0
- package/dist/forge-payload/meta/skills/meta-generic-skills.md +58 -0
- package/dist/forge-payload/meta/skills/meta-qa-engineer-skills.md +46 -0
- package/dist/forge-payload/meta/skills/meta-supervisor-skills.md +43 -0
- package/dist/forge-payload/meta/store-schema/bug.schema.md +71 -0
- package/dist/forge-payload/meta/store-schema/event.schema.md +76 -0
- package/dist/forge-payload/meta/store-schema/feature.schema.md +65 -0
- package/dist/forge-payload/meta/store-schema/sprint.schema.md +64 -0
- package/dist/forge-payload/meta/store-schema/task.schema.md +78 -0
- package/dist/forge-payload/meta/templates/meta-code-review.md +26 -0
- package/dist/forge-payload/meta/templates/meta-plan-review.md +28 -0
- package/dist/forge-payload/meta/templates/meta-plan.md +28 -0
- package/dist/forge-payload/meta/templates/meta-progress.md +25 -0
- package/dist/forge-payload/meta/templates/meta-retrospective.md +28 -0
- package/dist/forge-payload/meta/templates/meta-sprint-manifest.md +26 -0
- package/dist/forge-payload/meta/templates/meta-sprint-requirements.md +91 -0
- package/dist/forge-payload/meta/templates/meta-task-prompt.md +26 -0
- package/dist/forge-payload/meta/tool-specs/collate.spec.md +88 -0
- package/dist/forge-payload/meta/tool-specs/generation-manifest.spec.md +139 -0
- package/dist/forge-payload/meta/tool-specs/manage-config.spec.md +143 -0
- package/dist/forge-payload/meta/tool-specs/seed-store.spec.md +91 -0
- package/dist/forge-payload/meta/tool-specs/store-cli.spec.md +328 -0
- package/dist/forge-payload/meta/tool-specs/validate-store.spec.md +191 -0
- package/dist/forge-payload/meta/workflows/_fragments/context-injection.md +75 -0
- package/dist/forge-payload/meta/workflows/_fragments/event-emission-schema.md +73 -0
- package/dist/forge-payload/meta/workflows/_fragments/finalize.md +13 -0
- package/dist/forge-payload/meta/workflows/_fragments/friction-emit.md +73 -0
- package/dist/forge-payload/meta/workflows/_fragments/progress-reporting.md +38 -0
- package/dist/forge-payload/meta/workflows/_fragments/store-cli-verbs.md +39 -0
- package/dist/forge-payload/meta/workflows/meta-approve.md +119 -0
- package/dist/forge-payload/meta/workflows/meta-collate.md +89 -0
- package/dist/forge-payload/meta/workflows/meta-commit.md +93 -0
- package/dist/forge-payload/meta/workflows/meta-enhance.md +292 -0
- package/dist/forge-payload/meta/workflows/meta-fix-bug.md +501 -0
- package/dist/forge-payload/meta/workflows/meta-implement.md +132 -0
- package/dist/forge-payload/meta/workflows/meta-migrate.md +455 -0
- package/dist/forge-payload/meta/workflows/meta-orchestrate.md +993 -0
- package/dist/forge-payload/meta/workflows/meta-plan-task.md +133 -0
- package/dist/forge-payload/meta/workflows/meta-quiz-agent.md +135 -0
- package/dist/forge-payload/meta/workflows/meta-retrospective.md +65 -0
- package/dist/forge-payload/meta/workflows/meta-review-implementation.md +119 -0
- package/dist/forge-payload/meta/workflows/meta-review-plan.md +108 -0
- package/dist/forge-payload/meta/workflows/meta-review-sprint-completion.md +65 -0
- package/dist/forge-payload/meta/workflows/meta-sprint-intake.md +76 -0
- package/dist/forge-payload/meta/workflows/meta-sprint-plan.md +147 -0
- package/dist/forge-payload/meta/workflows/meta-update-implementation.md +76 -0
- package/dist/forge-payload/meta/workflows/meta-update-plan.md +76 -0
- package/dist/forge-payload/meta/workflows/meta-validate.md +111 -0
- package/dist/forge-payload/tools/build-persona-pack.cjs +120 -11
- package/dist/forge-payload/tools/check-structure.cjs +344 -0
- package/dist/forge-payload/tools/list-skills.js +76 -0
- package/dist/forge-payload/tools/store-cli.cjs +27 -1
- package/dist/forge-payload/tools/substitute-placeholders.cjs +60 -8
- package/dist/forge-payload/tools/verify-integrity.cjs +86 -0
- package/package.json +2 -2
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
export interface RemediationResult {
|
|
2
|
+
/** One-line user-facing hint explaining the error and the fix. */
|
|
3
|
+
hint: string;
|
|
4
|
+
/** Optional copy-pasteable store-cli command. Empty string if N/A. */
|
|
5
|
+
command: string;
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* Infer the entity type from an entity name/ID string.
|
|
9
|
+
* Returns "task" as default if no match.
|
|
10
|
+
*/
|
|
11
|
+
export declare function inferEntityType(entity: string): string;
|
|
12
|
+
/** Parsed breakdown of a validation error string. */
|
|
13
|
+
interface ParsedError {
|
|
14
|
+
field: string;
|
|
15
|
+
errorKind: "enum" | "required" | "undeclared" | "type" | "pattern" | "datetime" | "length" | "other";
|
|
16
|
+
observed?: string;
|
|
17
|
+
enumValues?: string[];
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Parse a single validation error line from validate.js / store-cli.
|
|
21
|
+
*
|
|
22
|
+
* Examples:
|
|
23
|
+
* 'status: value "verified" not in [reported, triaged, in-progress, fixed]'
|
|
24
|
+
* 'taskId: missing required field'
|
|
25
|
+
* 'xyz: undeclared field'
|
|
26
|
+
* 'sprintId: expected string, got number'
|
|
27
|
+
* 'title: value length 0 is below minLength 1'
|
|
28
|
+
* 'createdAt: value "today" is not a valid date-time'
|
|
29
|
+
* 'prefix: value "AB" does not match pattern ^[A-Z]+-[A-Z]$'
|
|
30
|
+
*/
|
|
31
|
+
export declare function parseValidationError(line: string): ParsedError;
|
|
32
|
+
/**
|
|
33
|
+
* Given a single validation error line (from validate.js or store-cli),
|
|
34
|
+
* return a user-facing remediation hint and optional copy-pasteable command.
|
|
35
|
+
*
|
|
36
|
+
* @param errorLine A single error line from validate.js / store-cli output.
|
|
37
|
+
* @param entityType The entity type (task, sprint, bug, feature, event).
|
|
38
|
+
* @param entityId The entity ID (e.g. "FORGE-S18-T02") — used for commands.
|
|
39
|
+
* @returns RemediationResult with hint and command.
|
|
40
|
+
*/
|
|
41
|
+
export declare function remediateError(errorLine: string, entityType: string, entityId: string): RemediationResult;
|
|
42
|
+
/**
|
|
43
|
+
* Parse a multi-line validation output (e.g. from store-cli validate or
|
|
44
|
+
* validate-store --dry-run) into individual error lines and return
|
|
45
|
+
* remediation for each.
|
|
46
|
+
*
|
|
47
|
+
* @param output Raw multi-line output from validation tool.
|
|
48
|
+
* @param entityType Override entity type (inferred from output if not provided).
|
|
49
|
+
* @returns Array of { errorLine, remediation } objects.
|
|
50
|
+
*/
|
|
51
|
+
export declare function remediateValidationOutput(output: string, entityType?: string): Array<{
|
|
52
|
+
errorLine: string;
|
|
53
|
+
remediation: RemediationResult;
|
|
54
|
+
}>;
|
|
55
|
+
/**
|
|
56
|
+
* Format a block message for a write-guard violation, appending remediation hints.
|
|
57
|
+
* Used by hook-dispatcher.ts for store-cli intercept blocks.
|
|
58
|
+
*
|
|
59
|
+
* @param rawReason The raw reason string from validateStoreCLIPayload or checkWriteGuard.
|
|
60
|
+
* @param entityType The entity type if known.
|
|
61
|
+
* @param entityId The entity ID if known.
|
|
62
|
+
* @returns Enhanced reason string with remediation hints appended.
|
|
63
|
+
*/
|
|
64
|
+
export declare function enhanceBlockMessage(rawReason: string, entityType?: string, entityId?: string): string;
|
|
65
|
+
export {};
|
|
@@ -0,0 +1,298 @@
|
|
|
1
|
+
// Store-error remediation — forge-cli#24
|
|
2
|
+
//
|
|
3
|
+
// Shared remediation-hint surface for store validation errors.
|
|
4
|
+
// Consumed by:
|
|
5
|
+
// - health-check.ts (per-error row in store-integrity output)
|
|
6
|
+
// - hooks/write-guard.ts (block-message body for schema violations)
|
|
7
|
+
// - hook-dispatcher.ts (store-cli intercept block messages)
|
|
8
|
+
// - store-validator.ts (structured result for hook callers)
|
|
9
|
+
//
|
|
10
|
+
// Error sources:
|
|
11
|
+
// 1. validate.js (in-process via write-guard) — produces error strings like:
|
|
12
|
+
// "status: value "verified" not in [reported, triaged, in-progress, fixed]"
|
|
13
|
+
// "taskId: missing required field"
|
|
14
|
+
// "xyz: undeclared field"
|
|
15
|
+
// 2. store-cli.cjs validate (subprocess via store-validator.ts) — same format,
|
|
16
|
+
// piped through stderr.
|
|
17
|
+
// 3. validate-store.cjs --dry-run (subprocess via health-check.ts) — lines like:
|
|
18
|
+
// "ERROR FORGE-S18-T02: status: value "verified" not in [reported, triaged, ...]"
|
|
19
|
+
// "WARN FORGE-S18-T02: ..."
|
|
20
|
+
//
|
|
21
|
+
// This module parses those formats and returns a one-line user-facing hint
|
|
22
|
+
// plus an optional copy-pasteable store-cli command.
|
|
23
|
+
// ── Entity-type inference from entity name / ID ────────────────────────────────
|
|
24
|
+
/** Map entity type to the corresponding schema's status enum values. */
|
|
25
|
+
const STATUS_ENUMS = {
|
|
26
|
+
task: [
|
|
27
|
+
"draft", "planned", "plan-approved", "implementing", "implemented",
|
|
28
|
+
"review-approved", "approved", "committed", "plan-revision-required",
|
|
29
|
+
"code-revision-required", "blocked", "escalated", "abandoned",
|
|
30
|
+
],
|
|
31
|
+
sprint: [
|
|
32
|
+
"planning", "active", "completed", "retrospective-done",
|
|
33
|
+
"partially-completed", "blocked", "abandoned",
|
|
34
|
+
],
|
|
35
|
+
bug: ["reported", "triaged", "in-progress", "fixed"],
|
|
36
|
+
feature: ["proposed", "accepted", "in-progress", "delivered", "declined"],
|
|
37
|
+
event: [], // no status field
|
|
38
|
+
};
|
|
39
|
+
/** Known entity types that support write via store-cli. */
|
|
40
|
+
const ENTITY_TYPES = new Set(["task", "sprint", "bug", "feature", "event"]);
|
|
41
|
+
/**
|
|
42
|
+
* Infer the entity type from an entity name/ID string.
|
|
43
|
+
* Returns "task" as default if no match.
|
|
44
|
+
*/
|
|
45
|
+
export function inferEntityType(entity) {
|
|
46
|
+
const lower = entity.toLowerCase();
|
|
47
|
+
// Bug patterns: BUG-015, FORGE-BUG-015, bug-031
|
|
48
|
+
if (/\bbug/i.test(entity))
|
|
49
|
+
return "bug";
|
|
50
|
+
// Sprint patterns: S18 (standalone), sprint-01 — but NOT inside task IDs like FORGE-S18-T02
|
|
51
|
+
if (/^[A-Z]+-S\d+$/i.test(entity) || /\bsprint/i.test(entity))
|
|
52
|
+
return "sprint";
|
|
53
|
+
// Feature patterns: FORGE-F01
|
|
54
|
+
if (/\bf\d+\b/i.test(entity) || lower.includes("feat"))
|
|
55
|
+
return "feature";
|
|
56
|
+
// Event patterns
|
|
57
|
+
if (lower.includes("event"))
|
|
58
|
+
return "event";
|
|
59
|
+
// Task IDs: FORGE-S18-T02, PROJECT-T01
|
|
60
|
+
if (/\bt\d+\b/i.test(entity))
|
|
61
|
+
return "task";
|
|
62
|
+
// Default
|
|
63
|
+
return "task";
|
|
64
|
+
}
|
|
65
|
+
const STATUS_REMEDIATION = {
|
|
66
|
+
hint: "Set status to one of the legal values for this entity type.",
|
|
67
|
+
command: (entityType, entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" update-status ${entityType} ${entityId} status <legal-value>`,
|
|
68
|
+
};
|
|
69
|
+
const REQUIRED_FIELD_REMEDIATION = {
|
|
70
|
+
hint: "Add the missing field with a valid value. Use the template command to see the canonical shape.",
|
|
71
|
+
command: (entityType, _entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" template ${entityType}`,
|
|
72
|
+
};
|
|
73
|
+
const UNDECLARED_FIELD_REMEDIATION = {
|
|
74
|
+
hint: "Remove the undeclared field, or check the schema for the correct property name.",
|
|
75
|
+
command: (entityType, _entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" describe ${entityType}`,
|
|
76
|
+
};
|
|
77
|
+
const TYPE_MISMATCH_REMEDIATION = {
|
|
78
|
+
hint: "Use the correct type for this field. Use the describe command to see the expected type.",
|
|
79
|
+
command: (entityType, _entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" describe ${entityType}`,
|
|
80
|
+
};
|
|
81
|
+
const PATTERN_REMEDIATION = {
|
|
82
|
+
hint: "The value must match the expected pattern (e.g. a date-time or ID format).",
|
|
83
|
+
command: (entityType, _entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" describe ${entityType}`,
|
|
84
|
+
};
|
|
85
|
+
const DATE_TIME_REMEDIATION = {
|
|
86
|
+
hint: "Use an ISO 8601 date-time string (e.g. 2026-05-21T12:00:00Z).",
|
|
87
|
+
command: (entityType, _entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" describe ${entityType}`,
|
|
88
|
+
};
|
|
89
|
+
const LENGTH_REMEDIATION = {
|
|
90
|
+
hint: "Adjust the value length to satisfy the schema constraint.",
|
|
91
|
+
command: (entityType, _entityId) => `node "$FORGE_ROOT/tools/store-cli.cjs" describe ${entityType}`,
|
|
92
|
+
};
|
|
93
|
+
/**
|
|
94
|
+
* Parse a single validation error line from validate.js / store-cli.
|
|
95
|
+
*
|
|
96
|
+
* Examples:
|
|
97
|
+
* 'status: value "verified" not in [reported, triaged, in-progress, fixed]'
|
|
98
|
+
* 'taskId: missing required field'
|
|
99
|
+
* 'xyz: undeclared field'
|
|
100
|
+
* 'sprintId: expected string, got number'
|
|
101
|
+
* 'title: value length 0 is below minLength 1'
|
|
102
|
+
* 'createdAt: value "today" is not a valid date-time'
|
|
103
|
+
* 'prefix: value "AB" does not match pattern ^[A-Z]+-[A-Z]$'
|
|
104
|
+
*/
|
|
105
|
+
export function parseValidationError(line) {
|
|
106
|
+
// Enum violation: 'field: value "X" not in [a, b, c]'
|
|
107
|
+
const enumMatch = line.match(/^(\w+):\s+value\s+"([^"]+)"\s+not in \[([^\]]+)\]/);
|
|
108
|
+
if (enumMatch) {
|
|
109
|
+
return {
|
|
110
|
+
field: enumMatch[1],
|
|
111
|
+
errorKind: "enum",
|
|
112
|
+
observed: enumMatch[2],
|
|
113
|
+
enumValues: enumMatch[3].split(",").map((s) => s.trim()),
|
|
114
|
+
};
|
|
115
|
+
}
|
|
116
|
+
// Required field: 'field: missing required field'
|
|
117
|
+
if (/\bmissing required field\b/.test(line)) {
|
|
118
|
+
const fieldMatch = line.match(/^(\w+):/);
|
|
119
|
+
return { field: fieldMatch?.[1] ?? "unknown", errorKind: "required" };
|
|
120
|
+
}
|
|
121
|
+
// Undeclared field: 'xyz: undeclared field'
|
|
122
|
+
if (/\bundeclared field\b/.test(line)) {
|
|
123
|
+
const fieldMatch = line.match(/^(\w+):/);
|
|
124
|
+
return { field: fieldMatch?.[1] ?? "unknown", errorKind: "undeclared" };
|
|
125
|
+
}
|
|
126
|
+
// Type mismatch: 'field: expected string, got number'
|
|
127
|
+
const typeMatch = line.match(/^(\w+):\s+expected\s+\S+,?\s+got\s+\S+/);
|
|
128
|
+
if (typeMatch) {
|
|
129
|
+
return { field: typeMatch[1], errorKind: "type" };
|
|
130
|
+
}
|
|
131
|
+
// Date-time format: 'field: value "..." is not a valid date-time'
|
|
132
|
+
if (/\bis not a valid date-time\b/.test(line)) {
|
|
133
|
+
const fieldMatch = line.match(/^(\w+):/);
|
|
134
|
+
return { field: fieldMatch?.[1] ?? "unknown", errorKind: "datetime" };
|
|
135
|
+
}
|
|
136
|
+
// Pattern mismatch: 'field: value "..." does not match pattern ...'
|
|
137
|
+
if (/\bdoes not match pattern\b/.test(line)) {
|
|
138
|
+
const fieldMatch = line.match(/^(\w+):/);
|
|
139
|
+
return { field: fieldMatch?.[1] ?? "unknown", errorKind: "pattern" };
|
|
140
|
+
}
|
|
141
|
+
// Length violations
|
|
142
|
+
if (/\blength\b.*\b(exceeds|below)\b/.test(line) || /\b(exceeds|below)\b.*\blength\b/.test(line)) {
|
|
143
|
+
const fieldMatch = line.match(/^(\w+):/);
|
|
144
|
+
return { field: fieldMatch?.[1] ?? "unknown", errorKind: "length" };
|
|
145
|
+
}
|
|
146
|
+
// Generic: capture leading field name if present
|
|
147
|
+
const genericMatch = line.match(/^(\w+):/);
|
|
148
|
+
return { field: genericMatch?.[1] ?? "unknown", errorKind: "other" };
|
|
149
|
+
}
|
|
150
|
+
// ── Public API ──────────────────────────────────────────────────────────────────
|
|
151
|
+
/**
|
|
152
|
+
* Given a single validation error line (from validate.js or store-cli),
|
|
153
|
+
* return a user-facing remediation hint and optional copy-pasteable command.
|
|
154
|
+
*
|
|
155
|
+
* @param errorLine A single error line from validate.js / store-cli output.
|
|
156
|
+
* @param entityType The entity type (task, sprint, bug, feature, event).
|
|
157
|
+
* @param entityId The entity ID (e.g. "FORGE-S18-T02") — used for commands.
|
|
158
|
+
* @returns RemediationResult with hint and command.
|
|
159
|
+
*/
|
|
160
|
+
export function remediateError(errorLine, entityType, entityId) {
|
|
161
|
+
const parsed = parseValidationError(errorLine);
|
|
162
|
+
switch (parsed.errorKind) {
|
|
163
|
+
case "enum": {
|
|
164
|
+
const legalValues = STATUS_ENUMS[entityType] ?? parsed.enumValues ?? [];
|
|
165
|
+
const isStatusField = parsed.field === "status" || parsed.field.toLowerCase().includes("status");
|
|
166
|
+
if (isStatusField && legalValues.length > 0) {
|
|
167
|
+
return {
|
|
168
|
+
hint: `"${parsed.observed}" is not a legal ${entityType} status. Legal values: ${legalValues.join(", ")}.`,
|
|
169
|
+
command: `node "$FORGE_ROOT/tools/store-cli.cjs" update-status ${entityType} ${entityId} status <${legalValues.join("|")}>`,
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
if (legalValues.length > 0) {
|
|
173
|
+
return {
|
|
174
|
+
hint: `Invalid value for "${parsed.field}". Allowed: ${legalValues.join(", ")}.`,
|
|
175
|
+
command: `node "$FORGE_ROOT/tools/store-cli.cjs" template ${entityType}`,
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
return {
|
|
179
|
+
hint: STATUS_REMEDIATION.hint,
|
|
180
|
+
command: STATUS_REMEDIATION.command(entityType, entityId),
|
|
181
|
+
};
|
|
182
|
+
}
|
|
183
|
+
case "required":
|
|
184
|
+
return {
|
|
185
|
+
hint: REQUIRED_FIELD_REMEDIATION.hint,
|
|
186
|
+
command: REQUIRED_FIELD_REMEDIATION.command(entityType, entityId),
|
|
187
|
+
};
|
|
188
|
+
case "undeclared":
|
|
189
|
+
return {
|
|
190
|
+
hint: UNDECLARED_FIELD_REMEDIATION.hint,
|
|
191
|
+
command: UNDECLARED_FIELD_REMEDIATION.command(entityType, entityId),
|
|
192
|
+
};
|
|
193
|
+
case "type":
|
|
194
|
+
return {
|
|
195
|
+
hint: TYPE_MISMATCH_REMEDIATION.hint,
|
|
196
|
+
command: TYPE_MISMATCH_REMEDIATION.command(entityType, entityId),
|
|
197
|
+
};
|
|
198
|
+
case "datetime":
|
|
199
|
+
return {
|
|
200
|
+
hint: DATE_TIME_REMEDIATION.hint,
|
|
201
|
+
command: DATE_TIME_REMEDIATION.command(entityType, entityId),
|
|
202
|
+
};
|
|
203
|
+
case "pattern":
|
|
204
|
+
return {
|
|
205
|
+
hint: PATTERN_REMEDIATION.hint,
|
|
206
|
+
command: PATTERN_REMEDIATION.command(entityType, entityId),
|
|
207
|
+
};
|
|
208
|
+
case "length":
|
|
209
|
+
return {
|
|
210
|
+
hint: LENGTH_REMEDIATION.hint,
|
|
211
|
+
command: LENGTH_REMEDIATION.command(entityType, entityId),
|
|
212
|
+
};
|
|
213
|
+
case "other":
|
|
214
|
+
default:
|
|
215
|
+
return {
|
|
216
|
+
hint: "Check the schema for the expected shape.",
|
|
217
|
+
command: `node "$FORGE_ROOT/tools/store-cli.cjs" template ${entityType}`,
|
|
218
|
+
};
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Parse a multi-line validation output (e.g. from store-cli validate or
|
|
223
|
+
* validate-store --dry-run) into individual error lines and return
|
|
224
|
+
* remediation for each.
|
|
225
|
+
*
|
|
226
|
+
* @param output Raw multi-line output from validation tool.
|
|
227
|
+
* @param entityType Override entity type (inferred from output if not provided).
|
|
228
|
+
* @returns Array of { errorLine, remediation } objects.
|
|
229
|
+
*/
|
|
230
|
+
export function remediateValidationOutput(output, entityType) {
|
|
231
|
+
const lines = output
|
|
232
|
+
.split("\n")
|
|
233
|
+
.map((l) => l.trim())
|
|
234
|
+
.filter((l) => l.length > 0);
|
|
235
|
+
const results = [];
|
|
236
|
+
for (const line of lines) {
|
|
237
|
+
// Strip "ERROR" / "WARN" prefix from validate-store --dry-run output
|
|
238
|
+
const cleaned = line.replace(/^(ERROR|WARN)\s+/, "");
|
|
239
|
+
if (!cleaned)
|
|
240
|
+
continue;
|
|
241
|
+
// Extract entity ID from lines like "FORGE-S18-T02: status: ..."
|
|
242
|
+
const entityIdMatch = cleaned.match(/^([A-Z]+-S?\d+-T?\d+|[A-Z]+-BUG-\d+|[A-Z]+-F\d+):/);
|
|
243
|
+
const entityId = entityIdMatch?.[1] ?? "unknown";
|
|
244
|
+
// If entity type not overridden, try to infer
|
|
245
|
+
const inferredType = entityType ?? inferEntityType(entityId);
|
|
246
|
+
// Skip non-error lines (hint lines from validate.js, blank lines, etc.)
|
|
247
|
+
if (cleaned.startsWith("(") || cleaned.startsWith("hint:") || cleaned.startsWith("#")) {
|
|
248
|
+
continue;
|
|
249
|
+
}
|
|
250
|
+
// Skip lines that are just an entity name without an error field
|
|
251
|
+
// (the real error is on the same line after the entity prefix)
|
|
252
|
+
const errorPart = entityIdMatch ? cleaned.slice(entityIdMatch[0].length).trim() : cleaned;
|
|
253
|
+
if (!errorPart || errorPart.length < 3)
|
|
254
|
+
continue;
|
|
255
|
+
results.push({
|
|
256
|
+
errorLine: cleaned,
|
|
257
|
+
remediation: remediateError(errorPart, inferredType, entityId),
|
|
258
|
+
});
|
|
259
|
+
}
|
|
260
|
+
return results;
|
|
261
|
+
}
|
|
262
|
+
/**
|
|
263
|
+
* Format a block message for a write-guard violation, appending remediation hints.
|
|
264
|
+
* Used by hook-dispatcher.ts for store-cli intercept blocks.
|
|
265
|
+
*
|
|
266
|
+
* @param rawReason The raw reason string from validateStoreCLIPayload or checkWriteGuard.
|
|
267
|
+
* @param entityType The entity type if known.
|
|
268
|
+
* @param entityId The entity ID if known.
|
|
269
|
+
* @returns Enhanced reason string with remediation hints appended.
|
|
270
|
+
*/
|
|
271
|
+
export function enhanceBlockMessage(rawReason, entityType, entityId) {
|
|
272
|
+
// Parse the raw reason into lines and try to add remediation to each error line.
|
|
273
|
+
const lines = rawReason.split("\n");
|
|
274
|
+
const enhanced = [];
|
|
275
|
+
for (const line of lines) {
|
|
276
|
+
// Strip leading list markers and bullet points
|
|
277
|
+
const stripped = line.replace(/^\s*[-•]\s*/, "").trim();
|
|
278
|
+
// Check if this line contains a recognizable validation error pattern
|
|
279
|
+
const parsed = parseValidationError(stripped);
|
|
280
|
+
// Also match lines that contain validation error patterns not caught by parseValidationError
|
|
281
|
+
const hasKnownError = parsed.errorKind !== "other" || /\bmissing required\b|\bnot in \[|\bundeclared\b|\bexpected\b.*\bgot\b/.test(stripped);
|
|
282
|
+
if (hasKnownError) {
|
|
283
|
+
const type = entityType ?? "task";
|
|
284
|
+
const id = entityId ?? "unknown";
|
|
285
|
+
const remediation = remediateError(stripped, type, id);
|
|
286
|
+
enhanced.push(line);
|
|
287
|
+
enhanced.push(` 💡 ${remediation.hint}`);
|
|
288
|
+
if (remediation.command) {
|
|
289
|
+
enhanced.push(` → ${remediation.command}`);
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
else {
|
|
293
|
+
enhanced.push(line);
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
return enhanced.join("\n");
|
|
297
|
+
}
|
|
298
|
+
//# sourceMappingURL=store-error-remediation.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"store-error-remediation.js","sourceRoot":"","sources":["../../../../src/extensions/forgecli/lib/store-error-remediation.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type { ExtensionAPI } from "@earendil-works/pi-coding-agent";
|
|
2
|
+
export interface ParsedMaterializeArgs {
|
|
3
|
+
mode: "all" | "single-workflow";
|
|
4
|
+
workflowId?: string;
|
|
5
|
+
}
|
|
6
|
+
/**
|
|
7
|
+
* Parse /forge:materialize arguments.
|
|
8
|
+
*
|
|
9
|
+
* Recognised forms:
|
|
10
|
+
* (empty) → { mode: "all" }
|
|
11
|
+
* --all → { mode: "all" }
|
|
12
|
+
* workflows <id> → { mode: "single-workflow", workflowId: "<id>" }
|
|
13
|
+
* workflows:<id> → { mode: "single-workflow", workflowId: "<id>" }
|
|
14
|
+
*/
|
|
15
|
+
export declare function parseMaterializeArgs(args: string): ParsedMaterializeArgs;
|
|
16
|
+
export declare function registerMaterialize(pi: ExtensionAPI): void;
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
// materialize.ts — native handler for /forge:materialize (FORGE-S23-T09)
|
|
2
|
+
//
|
|
3
|
+
// Ports the deterministic subset of the plugin's /forge:materialize:
|
|
4
|
+
// - Full warm-up (no args or --all): runs substitute-placeholders.cjs,
|
|
5
|
+
// build-persona-pack.cjs, and build-context-pack.cjs from the bundled payload.
|
|
6
|
+
// - Single-workflow mode: deferred — notify + return (follow-up task).
|
|
7
|
+
//
|
|
8
|
+
// Mode-neutral invariant: never writes .forge/config.json mode field.
|
|
9
|
+
// Uses spawn("node", [toolPath, ...argv]) — argv-array, no shell (IL6).
|
|
10
|
+
//
|
|
11
|
+
// References: regenerate.ts (substitute-placeholders.cjs argv),
|
|
12
|
+
// forge-init.ts (build-persona-pack / build-context-pack argv).
|
|
13
|
+
import { spawn } from "node:child_process";
|
|
14
|
+
import * as fs from "node:fs";
|
|
15
|
+
import * as path from "node:path";
|
|
16
|
+
import { getBundledPayloadRoot, getBundledToolsRoot } from "./forge-init.js";
|
|
17
|
+
// ── Argument parser ─────────────────────────────────────────────────────────
|
|
18
|
+
/**
|
|
19
|
+
* Parse /forge:materialize arguments.
|
|
20
|
+
*
|
|
21
|
+
* Recognised forms:
|
|
22
|
+
* (empty) → { mode: "all" }
|
|
23
|
+
* --all → { mode: "all" }
|
|
24
|
+
* workflows <id> → { mode: "single-workflow", workflowId: "<id>" }
|
|
25
|
+
* workflows:<id> → { mode: "single-workflow", workflowId: "<id>" }
|
|
26
|
+
*/
|
|
27
|
+
export function parseMaterializeArgs(args) {
|
|
28
|
+
const trimmed = args.trim();
|
|
29
|
+
if (!trimmed || trimmed === "--all") {
|
|
30
|
+
return { mode: "all" };
|
|
31
|
+
}
|
|
32
|
+
// "workflows plan_task" or "workflows:plan_task"
|
|
33
|
+
const workflowsMatch = trimmed.match(/^workflows[: ](.+)$/);
|
|
34
|
+
if (workflowsMatch) {
|
|
35
|
+
return { mode: "single-workflow", workflowId: workflowsMatch[1].trim() };
|
|
36
|
+
}
|
|
37
|
+
// Unrecognised flags/forms: treat as full warm-up (fail-open)
|
|
38
|
+
return { mode: "all" };
|
|
39
|
+
}
|
|
40
|
+
// ── Tool runner ─────────────────────────────────────────────────────────────
|
|
41
|
+
/**
|
|
42
|
+
* Spawn `node <toolPath> [...argv]` and await completion.
|
|
43
|
+
* Returns `{ ok: boolean, stderr: string }`.
|
|
44
|
+
*
|
|
45
|
+
* Non-zero exit is FATAL when `fatal === true` (default), ADVISORY otherwise.
|
|
46
|
+
*/
|
|
47
|
+
async function runTool(toolPath, argv, cwd, label, timeoutMs = 60_000) {
|
|
48
|
+
return new Promise((resolve) => {
|
|
49
|
+
const child = spawn("node", [toolPath, ...argv], {
|
|
50
|
+
cwd,
|
|
51
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
52
|
+
});
|
|
53
|
+
let stderrBuf = "";
|
|
54
|
+
const timer = setTimeout(() => {
|
|
55
|
+
child.kill("SIGKILL");
|
|
56
|
+
resolve({ ok: false, stderr: `${label} timed out after ${timeoutMs}ms` });
|
|
57
|
+
}, timeoutMs);
|
|
58
|
+
child.stderr?.on("data", (d) => {
|
|
59
|
+
stderrBuf += d.toString();
|
|
60
|
+
});
|
|
61
|
+
child.on("close", (code) => {
|
|
62
|
+
clearTimeout(timer);
|
|
63
|
+
resolve({ ok: code === 0, stderr: stderrBuf });
|
|
64
|
+
});
|
|
65
|
+
child.on("error", (err) => {
|
|
66
|
+
clearTimeout(timer);
|
|
67
|
+
resolve({ ok: false, stderr: err.message });
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
// ── Register ────────────────────────────────────────────────────────────────
|
|
72
|
+
export function registerMaterialize(pi) {
|
|
73
|
+
pi.registerCommand("forge:materialize", {
|
|
74
|
+
description: "Fill missing or stubbed Forge scaffolding without overwriting pristine files. " +
|
|
75
|
+
"Deterministic complement to /forge:init --fast. Mode-neutral: never writes config mode.",
|
|
76
|
+
async handler(args, ctx) {
|
|
77
|
+
const cwd = process.cwd();
|
|
78
|
+
const parsed = parseMaterializeArgs(args);
|
|
79
|
+
// Single-workflow mode: deferred in forge-cli — notify and return
|
|
80
|
+
if (parsed.mode === "single-workflow") {
|
|
81
|
+
ctx.ui.notify(`〇 forge:materialize — single-workflow mode ('${parsed.workflowId ?? "?"}') is not yet supported in forge-cli. ` +
|
|
82
|
+
"Use the Forge plugin directly or run /forge:materialize --all for a full warm-up.", "info");
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
// Config guard
|
|
86
|
+
const configPath = path.join(cwd, ".forge", "config.json");
|
|
87
|
+
if (!fs.existsSync(configPath)) {
|
|
88
|
+
ctx.ui.notify("× forge:materialize — no .forge/config.json found. Run /forge:init first.", "error");
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
// Resolve tool paths
|
|
92
|
+
let bundleRoot;
|
|
93
|
+
let toolsRoot;
|
|
94
|
+
try {
|
|
95
|
+
bundleRoot = getBundledPayloadRoot();
|
|
96
|
+
toolsRoot = getBundledToolsRoot();
|
|
97
|
+
}
|
|
98
|
+
catch (err) {
|
|
99
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
100
|
+
ctx.ui.notify(`× forge:materialize — could not resolve bundle paths: ${msg}`, "error");
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
const substituteTool = path.join(toolsRoot, "substitute-placeholders.cjs");
|
|
104
|
+
const basePackDir = path.join(bundleRoot, ".base-pack");
|
|
105
|
+
const configJsonPath = path.join(cwd, ".forge", "config.json");
|
|
106
|
+
// Guard: tools must exist
|
|
107
|
+
if (!fs.existsSync(substituteTool)) {
|
|
108
|
+
ctx.ui.notify(`× forge:materialize — substitute-placeholders.cjs not found at ${substituteTool}`, "error");
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
if (!fs.existsSync(basePackDir)) {
|
|
112
|
+
ctx.ui.notify(`× forge:materialize — .base-pack not found at ${basePackDir}`, "error");
|
|
113
|
+
return;
|
|
114
|
+
}
|
|
115
|
+
ctx.ui.setStatus?.("forge:materialize", "Materializing .forge/ artifacts…");
|
|
116
|
+
// Step 1: substitute-placeholders.cjs (fatal on failure)
|
|
117
|
+
// argv shape mirrors regenerate.ts usage
|
|
118
|
+
const initContextJson = path.join(cwd, ".forge", "init-context.json");
|
|
119
|
+
const subArgv = [
|
|
120
|
+
"--forge-root",
|
|
121
|
+
bundleRoot,
|
|
122
|
+
"--base-pack",
|
|
123
|
+
basePackDir,
|
|
124
|
+
"--config",
|
|
125
|
+
configJsonPath,
|
|
126
|
+
"--out",
|
|
127
|
+
cwd,
|
|
128
|
+
];
|
|
129
|
+
if (fs.existsSync(initContextJson)) {
|
|
130
|
+
subArgv.push("--context", initContextJson);
|
|
131
|
+
}
|
|
132
|
+
const subResult = await runTool(substituteTool, subArgv, cwd, "substitute-placeholders", 60_000);
|
|
133
|
+
if (!subResult.ok) {
|
|
134
|
+
const errMsg = subResult.stderr.trim().split("\n").slice(-3).join(" | ") || "unknown error";
|
|
135
|
+
ctx.ui.notify(`× forge:materialize — substitute-placeholders.cjs failed: ${errMsg}`, "error");
|
|
136
|
+
ctx.ui.setStatus?.("forge:materialize", undefined);
|
|
137
|
+
return;
|
|
138
|
+
}
|
|
139
|
+
ctx.ui.notify("〇 forge:materialize — scaffold filled", "info");
|
|
140
|
+
// Step 2: build-persona-pack.cjs (advisory — non-fatal)
|
|
141
|
+
const buildPersonaPackTool = path.join(toolsRoot, "build-persona-pack.cjs");
|
|
142
|
+
if (fs.existsSync(buildPersonaPackTool)) {
|
|
143
|
+
fs.mkdirSync(path.join(cwd, ".forge", "cache"), { recursive: true });
|
|
144
|
+
const personaResult = await runTool(buildPersonaPackTool, ["--out", path.join(cwd, ".forge", "cache", "persona-pack.json")], cwd, "build-persona-pack", 30_000);
|
|
145
|
+
if (!personaResult.ok) {
|
|
146
|
+
const msg = personaResult.stderr.trim().split("\n").at(-1) ?? "unknown";
|
|
147
|
+
ctx.ui.notify(`△ forge:materialize — build-persona-pack.cjs failed (non-fatal): ${msg}`, "warning");
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
// Step 3: build-context-pack.cjs (advisory — non-fatal)
|
|
151
|
+
const buildContextPackTool = path.join(toolsRoot, "build-context-pack.cjs");
|
|
152
|
+
if (fs.existsSync(buildContextPackTool)) {
|
|
153
|
+
// Read engineering path from config
|
|
154
|
+
let kbPath = "engineering";
|
|
155
|
+
try {
|
|
156
|
+
const raw = fs.readFileSync(configPath, "utf8");
|
|
157
|
+
const cfg = JSON.parse(raw);
|
|
158
|
+
const p = cfg.paths;
|
|
159
|
+
if (p && typeof p.engineering === "string")
|
|
160
|
+
kbPath = p.engineering;
|
|
161
|
+
}
|
|
162
|
+
catch {
|
|
163
|
+
// use default
|
|
164
|
+
}
|
|
165
|
+
const ctxResult = await runTool(buildContextPackTool, [
|
|
166
|
+
"--arch-dir",
|
|
167
|
+
path.join(cwd, kbPath, "architecture"),
|
|
168
|
+
"--out-md",
|
|
169
|
+
path.join(cwd, ".forge", "cache", "context-pack.md"),
|
|
170
|
+
"--out-json",
|
|
171
|
+
path.join(cwd, ".forge", "cache", "context-pack.json"),
|
|
172
|
+
], cwd, "build-context-pack", 30_000);
|
|
173
|
+
if (!ctxResult.ok) {
|
|
174
|
+
const msg = ctxResult.stderr.trim().split("\n").at(-1) ?? "unknown";
|
|
175
|
+
ctx.ui.notify(`△ forge:materialize — build-context-pack.cjs failed (non-fatal): ${msg}`, "warning");
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
// Complete
|
|
179
|
+
ctx.ui.setStatus?.("forge:materialize", undefined);
|
|
180
|
+
ctx.ui.notify("〇 forge:materialize complete — gaps filled (mode unchanged)", "info");
|
|
181
|
+
// Promotion hint (mode-neutral: only read, never write)
|
|
182
|
+
try {
|
|
183
|
+
const raw = fs.readFileSync(configPath, "utf8");
|
|
184
|
+
const cfg = JSON.parse(raw);
|
|
185
|
+
if (cfg.mode === "fast") {
|
|
186
|
+
ctx.ui.notify("〇 To declare the project fully generated: /forge:config mode full", "info");
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
catch {
|
|
190
|
+
// non-fatal — promotion hint is cosmetic
|
|
191
|
+
}
|
|
192
|
+
},
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
//# sourceMappingURL=materialize.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"materialize.js","sourceRoot":"","sources":["../../../src/extensions/forgecli/materialize.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { ExtensionAPI } from "@earendil-works/pi-coding-agent";
|
|
2
|
+
export interface ParsedMigrateArgs {
|
|
3
|
+
/** True when --structural flag present OR .forge/structure-versions.json absent. */
|
|
4
|
+
structural: boolean;
|
|
5
|
+
/** True when --dry-run flag present (schema branch only). */
|
|
6
|
+
dryRun: boolean;
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* Parse /forge:migrate arguments.
|
|
10
|
+
*
|
|
11
|
+
* Also checks filesystem for .forge/structure-versions.json to determine
|
|
12
|
+
* structural vs schema branch (absence → structural).
|
|
13
|
+
*/
|
|
14
|
+
export declare function parseMigrateArgs(args: string, cwd: string): ParsedMigrateArgs;
|
|
15
|
+
export interface RegisterMigrateOptions {
|
|
16
|
+
/** Override bundleRoot — for testing only. Production: always undefined (resolved via getBundledPayloadRoot). */
|
|
17
|
+
_testBundleRoot?: string;
|
|
18
|
+
}
|
|
19
|
+
export declare function registerMigrate(pi: ExtensionAPI, opts?: RegisterMigrateOptions): void;
|