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.
Files changed (105) hide show
  1. package/.claude/commands/dev.md +314 -0
  2. package/.claude/commands/plan.md +389 -0
  3. package/.claude/commands/premerge.md +179 -0
  4. package/.claude/commands/research.md +42 -0
  5. package/.claude/commands/review.md +442 -0
  6. package/.claude/commands/rollback.md +721 -0
  7. package/.claude/commands/ship.md +134 -0
  8. package/.claude/commands/sonarcloud.md +152 -0
  9. package/.claude/commands/status.md +77 -0
  10. package/.claude/commands/validate.md +237 -0
  11. package/.claude/commands/verify.md +221 -0
  12. package/.claude/rules/greptile-review-process.md +285 -0
  13. package/.claude/rules/workflow.md +105 -0
  14. package/.claude/scripts/greptile-resolve.sh +526 -0
  15. package/.claude/scripts/load-env.sh +32 -0
  16. package/.forge/hooks/check-tdd.js +240 -0
  17. package/.github/PLUGIN_TEMPLATE.json +32 -0
  18. package/.mcp.json.example +12 -0
  19. package/AGENTS.md +169 -0
  20. package/CLAUDE.md +99 -0
  21. package/LICENSE +21 -0
  22. package/README.md +414 -0
  23. package/bin/forge-cmd.js +313 -0
  24. package/bin/forge-validate.js +303 -0
  25. package/bin/forge.js +4228 -0
  26. package/docs/AGENT_INSTALL_PROMPT.md +342 -0
  27. package/docs/ENHANCED_ONBOARDING.md +602 -0
  28. package/docs/EXAMPLES.md +482 -0
  29. package/docs/GREPTILE_SETUP.md +400 -0
  30. package/docs/MANUAL_REVIEW_GUIDE.md +106 -0
  31. package/docs/ROADMAP.md +359 -0
  32. package/docs/SETUP.md +632 -0
  33. package/docs/TOOLCHAIN.md +849 -0
  34. package/docs/VALIDATION.md +363 -0
  35. package/docs/WORKFLOW.md +400 -0
  36. package/docs/planning/PROGRESS.md +396 -0
  37. package/docs/plans/.gitkeep +0 -0
  38. package/docs/plans/2026-02-27-forge-test-suite-v2-decisions.md +21 -0
  39. package/docs/plans/2026-02-27-forge-test-suite-v2-design.md +362 -0
  40. package/docs/plans/2026-02-27-forge-test-suite-v2-tasks.md +343 -0
  41. package/docs/plans/2026-03-02-superpowers-gaps-decisions.md +26 -0
  42. package/docs/plans/2026-03-02-superpowers-gaps-design.md +239 -0
  43. package/docs/plans/2026-03-02-superpowers-gaps-tasks.md +260 -0
  44. package/docs/plans/2026-03-04-agent-command-parity-design.md +163 -0
  45. package/docs/plans/2026-03-04-verify-worktree-cleanup-decisions.md +7 -0
  46. package/docs/plans/2026-03-04-verify-worktree-cleanup-design.md +165 -0
  47. package/docs/plans/2026-03-05-forge-uto-decisions.md +6 -0
  48. package/docs/plans/2026-03-05-forge-uto-design.md +116 -0
  49. package/docs/plans/2026-03-05-forge-uto-tasks.md +244 -0
  50. package/docs/plans/2026-03-10-command-creator-and-eval-decisions.md +52 -0
  51. package/docs/plans/2026-03-10-command-creator-and-eval-design.md +350 -0
  52. package/docs/plans/2026-03-10-command-creator-and-eval-tasks.md +426 -0
  53. package/docs/plans/2026-03-10-stale-workflow-refs-decisions.md +8 -0
  54. package/docs/plans/2026-03-10-stale-workflow-refs-design.md +80 -0
  55. package/docs/plans/2026-03-10-stale-workflow-refs-tasks.md +90 -0
  56. package/docs/plans/2026-03-14-beads-plan-context-decisions.md +9 -0
  57. package/docs/plans/2026-03-14-beads-plan-context-design.md +171 -0
  58. package/docs/plans/2026-03-14-beads-plan-context-tasks.md +160 -0
  59. package/docs/plans/2026-03-14-skill-eval-loop-decisions.md +33 -0
  60. package/docs/plans/2026-03-14-skill-eval-loop-design.md +118 -0
  61. package/docs/plans/2026-03-14-skill-eval-loop-results.md +78 -0
  62. package/docs/plans/2026-03-14-skill-eval-loop-tasks.md +160 -0
  63. package/docs/plans/2026-03-15-agent-command-parity-v2-decisions.md +11 -0
  64. package/docs/plans/2026-03-15-agent-command-parity-v2-design.md +145 -0
  65. package/docs/plans/2026-03-15-agent-command-parity-v2-tasks.md +211 -0
  66. package/docs/research/TEMPLATE.md +292 -0
  67. package/docs/research/advanced-testing.md +297 -0
  68. package/docs/research/agent-permissions.md +167 -0
  69. package/docs/research/dependency-chain.md +328 -0
  70. package/docs/research/forge-workflow-v2.md +550 -0
  71. package/docs/research/plugin-architecture.md +772 -0
  72. package/docs/research/pr4-cli-automation.md +326 -0
  73. package/docs/research/premerge-verify-restructure.md +205 -0
  74. package/docs/research/skills-restructure.md +508 -0
  75. package/docs/research/sonarcloud-perfection-plan.md +166 -0
  76. package/docs/research/sonarcloud-quality-gate.md +184 -0
  77. package/docs/research/superpowers-integration.md +403 -0
  78. package/docs/research/superpowers.md +319 -0
  79. package/docs/research/test-environment.md +519 -0
  80. package/install.sh +1062 -0
  81. package/lefthook.yml +39 -0
  82. package/lib/agents/README.md +198 -0
  83. package/lib/agents/claude.plugin.json +28 -0
  84. package/lib/agents/cline.plugin.json +22 -0
  85. package/lib/agents/codex.plugin.json +19 -0
  86. package/lib/agents/copilot.plugin.json +24 -0
  87. package/lib/agents/cursor.plugin.json +25 -0
  88. package/lib/agents/kilocode.plugin.json +22 -0
  89. package/lib/agents/opencode.plugin.json +20 -0
  90. package/lib/agents/roo.plugin.json +23 -0
  91. package/lib/agents-config.js +2112 -0
  92. package/lib/commands/dev.js +513 -0
  93. package/lib/commands/plan.js +696 -0
  94. package/lib/commands/recommend.js +119 -0
  95. package/lib/commands/ship.js +377 -0
  96. package/lib/commands/status.js +378 -0
  97. package/lib/commands/validate.js +602 -0
  98. package/lib/context-merge.js +359 -0
  99. package/lib/plugin-catalog.js +360 -0
  100. package/lib/plugin-manager.js +166 -0
  101. package/lib/plugin-recommender.js +141 -0
  102. package/lib/project-discovery.js +491 -0
  103. package/lib/setup.js +118 -0
  104. package/lib/workflow-profiles.js +203 -0
  105. 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
+ };