forge-workflow 0.0.1
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/.claude/commands/dev.md +314 -0
- package/.claude/commands/plan.md +389 -0
- package/.claude/commands/premerge.md +179 -0
- package/.claude/commands/research.md +42 -0
- package/.claude/commands/review.md +442 -0
- package/.claude/commands/rollback.md +721 -0
- package/.claude/commands/ship.md +134 -0
- package/.claude/commands/sonarcloud.md +152 -0
- package/.claude/commands/status.md +77 -0
- package/.claude/commands/validate.md +237 -0
- package/.claude/commands/verify.md +221 -0
- package/.claude/rules/greptile-review-process.md +285 -0
- package/.claude/rules/workflow.md +105 -0
- package/.claude/scripts/greptile-resolve.sh +526 -0
- package/.claude/scripts/load-env.sh +32 -0
- package/.forge/hooks/check-tdd.js +240 -0
- package/.github/PLUGIN_TEMPLATE.json +32 -0
- package/.mcp.json.example +12 -0
- package/AGENTS.md +169 -0
- package/CLAUDE.md +99 -0
- package/LICENSE +21 -0
- package/README.md +414 -0
- package/bin/forge-cmd.js +313 -0
- package/bin/forge-validate.js +303 -0
- package/bin/forge.js +4228 -0
- package/docs/AGENT_INSTALL_PROMPT.md +342 -0
- package/docs/ENHANCED_ONBOARDING.md +602 -0
- package/docs/EXAMPLES.md +482 -0
- package/docs/GREPTILE_SETUP.md +400 -0
- package/docs/MANUAL_REVIEW_GUIDE.md +106 -0
- package/docs/ROADMAP.md +359 -0
- package/docs/SETUP.md +632 -0
- package/docs/TOOLCHAIN.md +849 -0
- package/docs/VALIDATION.md +363 -0
- package/docs/WORKFLOW.md +400 -0
- package/docs/planning/PROGRESS.md +396 -0
- package/docs/plans/.gitkeep +0 -0
- package/docs/plans/2026-02-27-forge-test-suite-v2-decisions.md +21 -0
- package/docs/plans/2026-02-27-forge-test-suite-v2-design.md +362 -0
- package/docs/plans/2026-02-27-forge-test-suite-v2-tasks.md +343 -0
- package/docs/plans/2026-03-02-superpowers-gaps-decisions.md +26 -0
- package/docs/plans/2026-03-02-superpowers-gaps-design.md +239 -0
- package/docs/plans/2026-03-02-superpowers-gaps-tasks.md +260 -0
- package/docs/plans/2026-03-04-agent-command-parity-design.md +163 -0
- package/docs/plans/2026-03-04-verify-worktree-cleanup-decisions.md +7 -0
- package/docs/plans/2026-03-04-verify-worktree-cleanup-design.md +165 -0
- package/docs/plans/2026-03-05-forge-uto-decisions.md +6 -0
- package/docs/plans/2026-03-05-forge-uto-design.md +116 -0
- package/docs/plans/2026-03-05-forge-uto-tasks.md +244 -0
- package/docs/plans/2026-03-10-command-creator-and-eval-decisions.md +52 -0
- package/docs/plans/2026-03-10-command-creator-and-eval-design.md +350 -0
- package/docs/plans/2026-03-10-command-creator-and-eval-tasks.md +426 -0
- package/docs/plans/2026-03-10-stale-workflow-refs-decisions.md +8 -0
- package/docs/plans/2026-03-10-stale-workflow-refs-design.md +80 -0
- package/docs/plans/2026-03-10-stale-workflow-refs-tasks.md +90 -0
- package/docs/plans/2026-03-14-beads-plan-context-decisions.md +9 -0
- package/docs/plans/2026-03-14-beads-plan-context-design.md +171 -0
- package/docs/plans/2026-03-14-beads-plan-context-tasks.md +160 -0
- package/docs/plans/2026-03-14-skill-eval-loop-decisions.md +33 -0
- package/docs/plans/2026-03-14-skill-eval-loop-design.md +118 -0
- package/docs/plans/2026-03-14-skill-eval-loop-results.md +78 -0
- package/docs/plans/2026-03-14-skill-eval-loop-tasks.md +160 -0
- package/docs/plans/2026-03-15-agent-command-parity-v2-decisions.md +11 -0
- package/docs/plans/2026-03-15-agent-command-parity-v2-design.md +145 -0
- package/docs/plans/2026-03-15-agent-command-parity-v2-tasks.md +211 -0
- package/docs/research/TEMPLATE.md +292 -0
- package/docs/research/advanced-testing.md +297 -0
- package/docs/research/agent-permissions.md +167 -0
- package/docs/research/dependency-chain.md +328 -0
- package/docs/research/forge-workflow-v2.md +550 -0
- package/docs/research/plugin-architecture.md +772 -0
- package/docs/research/pr4-cli-automation.md +326 -0
- package/docs/research/premerge-verify-restructure.md +205 -0
- package/docs/research/skills-restructure.md +508 -0
- package/docs/research/sonarcloud-perfection-plan.md +166 -0
- package/docs/research/sonarcloud-quality-gate.md +184 -0
- package/docs/research/superpowers-integration.md +403 -0
- package/docs/research/superpowers.md +319 -0
- package/docs/research/test-environment.md +519 -0
- package/install.sh +1062 -0
- package/lefthook.yml +39 -0
- package/lib/agents/README.md +198 -0
- package/lib/agents/claude.plugin.json +28 -0
- package/lib/agents/cline.plugin.json +22 -0
- package/lib/agents/codex.plugin.json +19 -0
- package/lib/agents/copilot.plugin.json +24 -0
- package/lib/agents/cursor.plugin.json +25 -0
- package/lib/agents/kilocode.plugin.json +22 -0
- package/lib/agents/opencode.plugin.json +20 -0
- package/lib/agents/roo.plugin.json +23 -0
- package/lib/agents-config.js +2112 -0
- package/lib/commands/dev.js +513 -0
- package/lib/commands/plan.js +696 -0
- package/lib/commands/recommend.js +119 -0
- package/lib/commands/ship.js +377 -0
- package/lib/commands/status.js +378 -0
- package/lib/commands/validate.js +602 -0
- package/lib/context-merge.js +359 -0
- package/lib/plugin-catalog.js +360 -0
- package/lib/plugin-manager.js +166 -0
- package/lib/plugin-recommender.js +141 -0
- package/lib/project-discovery.js +491 -0
- package/lib/setup.js +118 -0
- package/lib/workflow-profiles.js +203 -0
- package/package.json +115 -0
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Recommend Command
|
|
3
|
+
*
|
|
4
|
+
* Displays tool recommendations based on detected tech stack and budget mode.
|
|
5
|
+
* Read-only — no installations, no side effects.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const { recommend } = require('../plugin-recommender');
|
|
9
|
+
const { detectTechStack } = require('../project-discovery');
|
|
10
|
+
const { BUDGET_MODES, STAGES } = require('../plugin-catalog');
|
|
11
|
+
|
|
12
|
+
const TIER_LABELS = {
|
|
13
|
+
free: '[F]',
|
|
14
|
+
'free-public': '[FP]',
|
|
15
|
+
'free-limited': '[FL]',
|
|
16
|
+
paid: '[P]',
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
const STAGE_NAMES = {
|
|
20
|
+
research: 'Research',
|
|
21
|
+
plan: 'Plan',
|
|
22
|
+
dev: 'Dev',
|
|
23
|
+
check: 'Check',
|
|
24
|
+
ship: 'Ship',
|
|
25
|
+
review: 'Review',
|
|
26
|
+
merge: 'Merge',
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Format the skipped tools section.
|
|
31
|
+
* @param {Object[]} skipped
|
|
32
|
+
* @returns {string[]}
|
|
33
|
+
*/
|
|
34
|
+
function formatSkippedSection(skipped) {
|
|
35
|
+
if (skipped.length === 0) return [];
|
|
36
|
+
const lines = [
|
|
37
|
+
'',
|
|
38
|
+
` Skipped (${skipped.length} tools not matching budget):`,
|
|
39
|
+
...skipped.slice(0, 5).map((tool) => ` ${tool.name}: ${tool.reason}`),
|
|
40
|
+
];
|
|
41
|
+
if (skipped.length > 5) {
|
|
42
|
+
lines.push(` ... and ${skipped.length - 5} more`);
|
|
43
|
+
}
|
|
44
|
+
return lines;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Format recommendations into displayable text.
|
|
49
|
+
* @param {{ recommended: Object[], skipped: Object[] }} recommendations
|
|
50
|
+
* @returns {string}
|
|
51
|
+
*/
|
|
52
|
+
function formatRecommendations(recommendations) {
|
|
53
|
+
const { recommended, skipped } = recommendations;
|
|
54
|
+
|
|
55
|
+
if (recommended.length === 0) {
|
|
56
|
+
return 'No tools recommended for current configuration.';
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
// Group by stage
|
|
60
|
+
const byStage = {};
|
|
61
|
+
for (const stage of Object.values(STAGES)) {
|
|
62
|
+
byStage[stage] = recommended.filter((t) => t.stage === stage);
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
const lines = [];
|
|
66
|
+
for (const [stage, tools] of Object.entries(byStage)) {
|
|
67
|
+
if (tools.length === 0) continue;
|
|
68
|
+
lines.push(
|
|
69
|
+
'',
|
|
70
|
+
` ${STAGE_NAMES[stage] || stage}`,
|
|
71
|
+
` ${'─'.repeat(40)}`,
|
|
72
|
+
);
|
|
73
|
+
for (const tool of tools) {
|
|
74
|
+
const tier = TIER_LABELS[tool.tier] || `[${tool.tier}]`;
|
|
75
|
+
if (tool.install.cmdCurl) {
|
|
76
|
+
lines.push(
|
|
77
|
+
` ${tier.padEnd(5)} ${tool.name}`,
|
|
78
|
+
` CLI (recommended): ${tool.install.cmd}`,
|
|
79
|
+
` Curl (no install): ${tool.install.cmdCurl}`,
|
|
80
|
+
);
|
|
81
|
+
} else {
|
|
82
|
+
lines.push(` ${tier.padEnd(5)} ${tool.name.padEnd(25)} ${tool.install.cmd}`);
|
|
83
|
+
}
|
|
84
|
+
if (tool.alternatives && tool.alternatives.length > 0) {
|
|
85
|
+
const altNames = tool.alternatives.map((a) => `${a.tool} (${a.tier})`).join(', ');
|
|
86
|
+
lines.push(` Free alternatives: ${altNames}`);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
lines.push(...formatSkippedSection(skipped), '');
|
|
92
|
+
return lines.join('\n');
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Handle the recommend command.
|
|
97
|
+
* @param {Object} flags - Parsed CLI flags
|
|
98
|
+
* @param {string} [projectPath] - Project path (defaults to cwd)
|
|
99
|
+
* @returns {{ budgetMode: string, recommendations: Object, error?: string }}
|
|
100
|
+
*/
|
|
101
|
+
function handleRecommend(flags, projectPath) {
|
|
102
|
+
const budgetMode = flags.budget || 'startup';
|
|
103
|
+
|
|
104
|
+
// Validate budget mode
|
|
105
|
+
if (!BUDGET_MODES[budgetMode]) {
|
|
106
|
+
return {
|
|
107
|
+
budgetMode,
|
|
108
|
+
recommendations: { recommended: [], skipped: [] },
|
|
109
|
+
error: `Invalid budget mode: '${budgetMode}'. Valid: ${Object.keys(BUDGET_MODES).join(', ')}`,
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
const techStack = detectTechStack(projectPath || process.cwd());
|
|
114
|
+
const recommendations = recommend(techStack, budgetMode);
|
|
115
|
+
|
|
116
|
+
return { budgetMode, recommendations };
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
module.exports = { formatRecommendations, handleRecommend };
|
|
@@ -0,0 +1,377 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Ship Command - PR Creation with Auto-Generated Documentation
|
|
3
|
+
* Creates pull requests with comprehensive body generated from research and metrics
|
|
4
|
+
*
|
|
5
|
+
* Security: Uses execFileSync for command execution to prevent injection
|
|
6
|
+
* Automation: Extracts key decisions, test scenarios, and coverage metrics
|
|
7
|
+
*
|
|
8
|
+
* @module commands/ship
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
const { execFileSync } = require('node:child_process');
|
|
12
|
+
const fs = require('node:fs');
|
|
13
|
+
const path = require('node:path');
|
|
14
|
+
|
|
15
|
+
function getExecOptions() {
|
|
16
|
+
return { encoding: 'utf8', cwd: process.cwd(), timeout: 120000 };
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
function getGhCheckOptions() {
|
|
20
|
+
return { encoding: 'utf8', cwd: process.cwd(), timeout: 3000 };
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const VALID_PR_PREFIXES = ['feat:', 'fix:', 'docs:', 'refactor:', 'test:', 'chore:', 'perf:', 'ci:', 'build:', 'revert:'];
|
|
24
|
+
|
|
25
|
+
const MAX_SLUG_LENGTH = 100;
|
|
26
|
+
const MAX_FILE_SIZE = 5 * 1024 * 1024; // 5MB
|
|
27
|
+
|
|
28
|
+
function isCommandNotFound(error) {
|
|
29
|
+
return error.message.includes('ENOENT') || error.message.includes('not found');
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Validate feature slug format
|
|
34
|
+
* Ensures slug matches expected pattern and doesn't contain path traversal
|
|
35
|
+
* @private
|
|
36
|
+
*/
|
|
37
|
+
function validateFeatureSlug(slug) {
|
|
38
|
+
if (!slug || typeof slug !== 'string') {
|
|
39
|
+
return { valid: false, error: 'Feature slug must be a non-empty string' };
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Length limits (prevent DoS)
|
|
43
|
+
const MIN_SLUG_LENGTH = 3;
|
|
44
|
+
|
|
45
|
+
if (slug.length < MIN_SLUG_LENGTH) {
|
|
46
|
+
return { valid: false, error: `Slug too short (minimum ${MIN_SLUG_LENGTH} characters)` };
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
if (slug.length > MAX_SLUG_LENGTH) {
|
|
50
|
+
return { valid: false, error: `Slug too long (maximum ${MAX_SLUG_LENGTH} characters)` };
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// Only allow lowercase alphanumeric and hyphens; must start and end with alphanumeric
|
|
54
|
+
const slugPattern = /^[a-z0-9](?:[a-z0-9-]*[a-z0-9])?$/; // NOSONAR S5852 - no backtracking: anchored, alternation is possessive
|
|
55
|
+
if (!slugPattern.test(slug)) {
|
|
56
|
+
return {
|
|
57
|
+
valid: false,
|
|
58
|
+
error: `Invalid slug format '${slug}'. Only lowercase letters, numbers, and hyphens allowed.`,
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
return { valid: true };
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function extractKeyDecisions(researchContent) { // NOSONAR S3776
|
|
66
|
+
if (!researchContent || typeof researchContent !== 'string') return [];
|
|
67
|
+
const decisions = [];
|
|
68
|
+
|
|
69
|
+
// Split by sections to avoid ReDoS - safer than complex regex
|
|
70
|
+
const lines = researchContent.split('\n');
|
|
71
|
+
let inDecisionsSection = false;
|
|
72
|
+
let sectionContent = [];
|
|
73
|
+
|
|
74
|
+
for (const line of lines) {
|
|
75
|
+
if (/^##\s+Key Decisions/i.test(line)) {
|
|
76
|
+
inDecisionsSection = true;
|
|
77
|
+
continue;
|
|
78
|
+
}
|
|
79
|
+
if (inDecisionsSection && /^##\s+/.test(line)) {
|
|
80
|
+
// Hit next section, stop
|
|
81
|
+
break;
|
|
82
|
+
}
|
|
83
|
+
if (inDecisionsSection) {
|
|
84
|
+
sectionContent.push(line);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
// Extract bullet format: "- **Decision N**: value"
|
|
89
|
+
for (const line of sectionContent) {
|
|
90
|
+
const boldStart = line.indexOf('**Decision');
|
|
91
|
+
if (boldStart === -1) continue;
|
|
92
|
+
const afterDecision = line.slice(boldStart + 10); // after '**Decision'
|
|
93
|
+
const closingBold = afterDecision.indexOf('**:');
|
|
94
|
+
if (closingBold === -1) continue;
|
|
95
|
+
const value = afterDecision.slice(closingBold + 3).trim();
|
|
96
|
+
if (value) decisions.push(value);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
// Extract heading format:
|
|
100
|
+
// "### Decision N: Title" (+ optional "**Reasoning**: ...")
|
|
101
|
+
for (let i = 0; i < sectionContent.length; i++) {
|
|
102
|
+
const headingMatch = /^###\s*Decision\s*\d+:\s*([^\n]+)/i.exec(sectionContent[i]); // NOSONAR S5852 - anchored with bounded capture
|
|
103
|
+
if (!headingMatch) continue;
|
|
104
|
+
|
|
105
|
+
const title = headingMatch[1].trim();
|
|
106
|
+
if (!title) continue;
|
|
107
|
+
|
|
108
|
+
let reasoning = '';
|
|
109
|
+
for (let j = i + 1; j < sectionContent.length; j++) {
|
|
110
|
+
const nextLine = sectionContent[j];
|
|
111
|
+
|
|
112
|
+
// Stop when the next decision heading or a new top-level section starts
|
|
113
|
+
if (/^###\s*Decision\s*\d+:/i.test(nextLine) || /^##\s+/.test(nextLine)) break;
|
|
114
|
+
|
|
115
|
+
const reasoningMatch = /^\*\*Reasoning\*\*:\s*([^\n]+)/i.exec(nextLine); // NOSONAR S5852 - anchored with bounded capture
|
|
116
|
+
if (reasoningMatch) {
|
|
117
|
+
reasoning = reasoningMatch[1].trim();
|
|
118
|
+
break;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
const formattedDecision = reasoning ? `${title} - Reasoning: ${reasoning}` : title;
|
|
123
|
+
decisions.push(formattedDecision);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
// Deduplicate if documents include both representations for the same decision
|
|
127
|
+
return [...new Set(decisions)];
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
function extractTestScenarios(researchContent) {
|
|
131
|
+
if (!researchContent || typeof researchContent !== 'string') return [];
|
|
132
|
+
const scenarios = [];
|
|
133
|
+
|
|
134
|
+
// Split by sections to avoid ReDoS - safer than complex regex
|
|
135
|
+
const lines = researchContent.split('\n');
|
|
136
|
+
let inScenariosSection = false;
|
|
137
|
+
let sectionContent = [];
|
|
138
|
+
|
|
139
|
+
for (const line of lines) {
|
|
140
|
+
if (/^##\s+(?:TDD\s+)?Test Scenarios/i.test(line)) {
|
|
141
|
+
inScenariosSection = true;
|
|
142
|
+
continue;
|
|
143
|
+
}
|
|
144
|
+
if (inScenariosSection && /^##\s+/.test(line)) {
|
|
145
|
+
// Hit next section, stop
|
|
146
|
+
break;
|
|
147
|
+
}
|
|
148
|
+
if (inScenariosSection) {
|
|
149
|
+
sectionContent.push(line);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
// Extract scenarios from both supported formats:
|
|
154
|
+
// 1) "1. Scenario description"
|
|
155
|
+
// 2) "### Scenario N: Scenario description"
|
|
156
|
+
for (const line of sectionContent) {
|
|
157
|
+
const numberedMatch = /^\d+\.\s+([^\n]+)$/.exec(line); // NOSONAR S5852 - anchored with bounded capture
|
|
158
|
+
if (numberedMatch) {
|
|
159
|
+
scenarios.push(numberedMatch[1].trim());
|
|
160
|
+
continue;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
const headingMatch = /^###\s*Scenario\s*\d+:\s*([^\n]+)$/i.exec(line); // NOSONAR S5852 - anchored with bounded capture
|
|
164
|
+
if (headingMatch) {
|
|
165
|
+
scenarios.push(headingMatch[1].trim());
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
return scenarios;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
async function getTestCoverage() {
|
|
172
|
+
try {
|
|
173
|
+
const coveragePath = path.join(process.cwd(), 'coverage', 'coverage-summary.json');
|
|
174
|
+
if (!fs.existsSync(coveragePath)) {
|
|
175
|
+
// Return object indicating no coverage instead of null
|
|
176
|
+
return { available: false };
|
|
177
|
+
}
|
|
178
|
+
const coverageData = JSON.parse(fs.readFileSync(coveragePath, 'utf8'));
|
|
179
|
+
const totals = coverageData.total;
|
|
180
|
+
return {
|
|
181
|
+
available: true,
|
|
182
|
+
lines: totals.lines.pct,
|
|
183
|
+
branches: totals.branches.pct,
|
|
184
|
+
functions: totals.functions.pct,
|
|
185
|
+
statements: totals.statements.pct,
|
|
186
|
+
};
|
|
187
|
+
} catch (error_) { // NOSONAR S2486 - intentional: file read failure returns unavailable
|
|
188
|
+
void error_;
|
|
189
|
+
// Return object indicating error instead of null
|
|
190
|
+
return { available: false, error: true };
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
function generatePRBody(context) {
|
|
195
|
+
const { featureName, researchDoc, decisions = [], testScenarios = [], coverage } = context;
|
|
196
|
+
let body = `## Summary
|
|
197
|
+
|
|
198
|
+
Implements **${featureName}**
|
|
199
|
+
|
|
200
|
+
`;
|
|
201
|
+
if (researchDoc) {
|
|
202
|
+
body += `**Research:** [${path.basename(researchDoc)}](${researchDoc})
|
|
203
|
+
|
|
204
|
+
`;
|
|
205
|
+
}
|
|
206
|
+
if (decisions.length > 0) {
|
|
207
|
+
body += `## Key Decisions
|
|
208
|
+
|
|
209
|
+
`;
|
|
210
|
+
decisions.forEach(decision => { body += `- ${decision}\n`; });
|
|
211
|
+
body += '\n';
|
|
212
|
+
}
|
|
213
|
+
if (testScenarios.length > 0) {
|
|
214
|
+
body += `## Test Scenarios
|
|
215
|
+
|
|
216
|
+
`;
|
|
217
|
+
testScenarios.forEach(scenario => { body += `- ${scenario}\n`; });
|
|
218
|
+
body += '\n';
|
|
219
|
+
}
|
|
220
|
+
// Include coverage if available (handles both old and new format)
|
|
221
|
+
if (coverage && (coverage.available !== false) && coverage.lines !== undefined) {
|
|
222
|
+
body += `## Test Coverage
|
|
223
|
+
|
|
224
|
+
- **Lines:** ${coverage.lines}%
|
|
225
|
+
- **Branches:** ${coverage.branches}%
|
|
226
|
+
- **Functions:** ${coverage.functions}%
|
|
227
|
+
- **Statements:** ${coverage.statements}%
|
|
228
|
+
|
|
229
|
+
`;
|
|
230
|
+
}
|
|
231
|
+
body += `## Development Approach
|
|
232
|
+
|
|
233
|
+
✅ **TDD (Test-Driven Development)**
|
|
234
|
+
- Tests written before implementation
|
|
235
|
+
- RED-GREEN-REFACTOR cycles
|
|
236
|
+
- All tests passing
|
|
237
|
+
|
|
238
|
+
---
|
|
239
|
+
|
|
240
|
+
🤖 Generated with [Forge Workflow](https://github.com/anthropics/forge)
|
|
241
|
+
`;
|
|
242
|
+
return body;
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
function validatePRTitle(title) {
|
|
246
|
+
if (!title || typeof title !== 'string') {
|
|
247
|
+
return { valid: false, error: 'PR title is required' };
|
|
248
|
+
}
|
|
249
|
+
const hasValidPrefix = VALID_PR_PREFIXES.some(prefix => title.startsWith(prefix));
|
|
250
|
+
if (!hasValidPrefix) {
|
|
251
|
+
return { valid: false, error: `PR title must start with a valid prefix: ${VALID_PR_PREFIXES.join(', ')}` };
|
|
252
|
+
}
|
|
253
|
+
if (title.length < 10) return { valid: false, error: 'PR title too short (minimum 10 characters)' };
|
|
254
|
+
if (title.length > 100) return { valid: false, error: 'PR title too long (maximum 100 characters)' };
|
|
255
|
+
return { valid: true };
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
async function createPR(options) { // NOSONAR S3776
|
|
259
|
+
const { title, body, dryRun = false } = options;
|
|
260
|
+
const titleValidation = validatePRTitle(title);
|
|
261
|
+
if (!titleValidation.valid) return { success: false, error: titleValidation.error };
|
|
262
|
+
try {
|
|
263
|
+
execFileSync('gh', ['--version'], getGhCheckOptions()); // NOSONAR S4036 - hardcoded CLI command, no user input, developer tool context
|
|
264
|
+
} catch (error) {
|
|
265
|
+
if (isCommandNotFound(error)) {
|
|
266
|
+
return { success: false, error: 'GitHub CLI (gh) not found. Install from: https://cli.github.com/' };
|
|
267
|
+
}
|
|
268
|
+
if (error.killed && error.signal === 'SIGTERM') {
|
|
269
|
+
return { success: false, error: 'GitHub CLI version check timed out. Check gh installation.' };
|
|
270
|
+
}
|
|
271
|
+
// Catch-all: other errors (permissions, corrupted binary, etc.)
|
|
272
|
+
return { success: false, error: `GitHub CLI check failed: ${error.message}` };
|
|
273
|
+
}
|
|
274
|
+
try {
|
|
275
|
+
execFileSync('git', ['rev-parse', '--git-dir'], getExecOptions()); // NOSONAR S4036 - hardcoded CLI command, no user input, developer tool context
|
|
276
|
+
} catch (error_) { // NOSONAR S2486 - intentional: not-a-git-repo is the expected failure signal
|
|
277
|
+
void error_;
|
|
278
|
+
return { success: false, error: 'Not in a git repository. Initialize with: git init' };
|
|
279
|
+
}
|
|
280
|
+
try {
|
|
281
|
+
execFileSync('git', ['remote', 'get-url', 'origin'], getExecOptions()); // NOSONAR S4036 - hardcoded CLI command, no user input, developer tool context
|
|
282
|
+
} catch (error_) { // NOSONAR S2486 - intentional: no-remote is the expected failure signal
|
|
283
|
+
void error_;
|
|
284
|
+
return { success: false, error: 'No git remote configured. Add with: git remote add origin <url>' };
|
|
285
|
+
}
|
|
286
|
+
if (dryRun) {
|
|
287
|
+
return { success: true, message: '[DRY RUN] Would create PR with title: ' + title, prUrl: 'https://github.com/owner/repo/pull/1' };
|
|
288
|
+
}
|
|
289
|
+
try {
|
|
290
|
+
const result = execFileSync('gh', ['pr', 'create', '--title', title, '--body', body], getExecOptions()); // NOSONAR S4036 - hardcoded CLI command, no user input, developer tool context
|
|
291
|
+
const urlMatch = /https:\/\/github\.com\/[^\s]+/.exec(result);
|
|
292
|
+
const prUrl = urlMatch ? urlMatch[0] : null;
|
|
293
|
+
const numberMatch = /\/pull\/(\d+)/.exec(result);
|
|
294
|
+
const prNumber = numberMatch ? Number.parseInt(numberMatch[1], 10) : null;
|
|
295
|
+
return { success: true, prUrl, prNumber, output: result };
|
|
296
|
+
} catch (error) {
|
|
297
|
+
// Check for timeout
|
|
298
|
+
if (error.killed && error.signal === 'SIGTERM') {
|
|
299
|
+
return { success: false, error: 'GitHub CLI command timed out after 2 minutes. Check network connection.' };
|
|
300
|
+
}
|
|
301
|
+
return { success: false, error: `Failed to create PR: ${error.message}`, output: error.stdout || error.message };
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
async function executeShip(options) {
|
|
306
|
+
const { featureSlug, title, dryRun = false } = options || {};
|
|
307
|
+
|
|
308
|
+
// Validate feature slug
|
|
309
|
+
if (!featureSlug || typeof featureSlug !== 'string' || featureSlug.trim() === '') {
|
|
310
|
+
return { success: false, error: 'Feature slug is required and must be a non-empty string' };
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
const slugValidation = validateFeatureSlug(featureSlug);
|
|
314
|
+
if (!slugValidation.valid) {
|
|
315
|
+
return { success: false, error: slugValidation.error };
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
// Validate PR title
|
|
319
|
+
if (!title || typeof title !== 'string' || title.trim() === '') {
|
|
320
|
+
return { success: false, error: 'PR title is required and must be a non-empty string' };
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
const titleValidation = validatePRTitle(title);
|
|
324
|
+
if (!titleValidation.valid) {
|
|
325
|
+
return { success: false, error: titleValidation.error };
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
try {
|
|
329
|
+
const researchPath = path.join(process.cwd(), 'docs', 'research', `${featureSlug}.md`);
|
|
330
|
+
let researchContent = null;
|
|
331
|
+
let decisions = [];
|
|
332
|
+
let testScenarios = [];
|
|
333
|
+
if (fs.existsSync(researchPath)) {
|
|
334
|
+
// Check file size to prevent resource exhaustion (max 5MB)
|
|
335
|
+
const stats = fs.statSync(researchPath);
|
|
336
|
+
if (stats.size > MAX_FILE_SIZE) {
|
|
337
|
+
return {
|
|
338
|
+
success: false,
|
|
339
|
+
error: `Research document too large (${Math.round(stats.size / 1024 / 1024)}MB). Maximum size: 5MB`,
|
|
340
|
+
};
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
researchContent = fs.readFileSync(researchPath, 'utf8');
|
|
344
|
+
decisions = extractKeyDecisions(researchContent);
|
|
345
|
+
testScenarios = extractTestScenarios(researchContent);
|
|
346
|
+
}
|
|
347
|
+
const coverage = await getTestCoverage();
|
|
348
|
+
const featureName = featureSlug.split('-').map(word => word.charAt(0).toUpperCase() + word.slice(1)).join(' ');
|
|
349
|
+
const prBody = generatePRBody({
|
|
350
|
+
featureName,
|
|
351
|
+
researchDoc: researchContent ? `docs/research/${featureSlug}.md` : null,
|
|
352
|
+
decisions,
|
|
353
|
+
testScenarios,
|
|
354
|
+
coverage,
|
|
355
|
+
});
|
|
356
|
+
const result = await createPR({ title, body: prBody, dryRun });
|
|
357
|
+
if (!result.success) return result;
|
|
358
|
+
return {
|
|
359
|
+
success: true,
|
|
360
|
+
prUrl: result.prUrl,
|
|
361
|
+
prNumber: result.prNumber,
|
|
362
|
+
message: dryRun ? 'Dry run successful - PR body generated' : `Pull request created successfully: ${result.prUrl}`,
|
|
363
|
+
};
|
|
364
|
+
} catch (error) {
|
|
365
|
+
return { success: false, error: `Failed to execute ship command: ${error.message}` };
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
module.exports = {
|
|
370
|
+
extractKeyDecisions,
|
|
371
|
+
extractTestScenarios,
|
|
372
|
+
getTestCoverage,
|
|
373
|
+
generatePRBody,
|
|
374
|
+
validatePRTitle,
|
|
375
|
+
createPR,
|
|
376
|
+
executeShip,
|
|
377
|
+
};
|