@cluesmith/codev 1.6.2 → 2.0.0-rc.10

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 (121) hide show
  1. package/bin/porch.js +12 -0
  2. package/dist/agent-farm/cli.d.ts.map +1 -1
  3. package/dist/agent-farm/cli.js +25 -14
  4. package/dist/agent-farm/cli.js.map +1 -1
  5. package/dist/agent-farm/commands/index.d.ts +1 -0
  6. package/dist/agent-farm/commands/index.d.ts.map +1 -1
  7. package/dist/agent-farm/commands/index.js +1 -0
  8. package/dist/agent-farm/commands/index.js.map +1 -1
  9. package/dist/agent-farm/commands/kickoff.d.ts +20 -0
  10. package/dist/agent-farm/commands/kickoff.d.ts.map +1 -0
  11. package/dist/agent-farm/commands/kickoff.js +273 -0
  12. package/dist/agent-farm/commands/kickoff.js.map +1 -0
  13. package/dist/agent-farm/commands/spawn.d.ts.map +1 -1
  14. package/dist/agent-farm/commands/spawn.js +30 -96
  15. package/dist/agent-farm/commands/spawn.js.map +1 -1
  16. package/dist/agent-farm/commands/start.d.ts.map +1 -1
  17. package/dist/agent-farm/commands/start.js +8 -50
  18. package/dist/agent-farm/commands/start.js.map +1 -1
  19. package/dist/agent-farm/servers/dashboard-server.js +0 -14
  20. package/dist/agent-farm/servers/dashboard-server.js.map +1 -1
  21. package/dist/agent-farm/state.d.ts +0 -10
  22. package/dist/agent-farm/state.d.ts.map +1 -1
  23. package/dist/agent-farm/state.js +0 -24
  24. package/dist/agent-farm/state.js.map +1 -1
  25. package/dist/cli.d.ts.map +1 -1
  26. package/dist/cli.js +17 -0
  27. package/dist/cli.js.map +1 -1
  28. package/dist/commands/adopt.d.ts.map +1 -1
  29. package/dist/commands/adopt.js +17 -1
  30. package/dist/commands/adopt.js.map +1 -1
  31. package/dist/commands/consult/index.d.ts.map +1 -1
  32. package/dist/commands/consult/index.js +2 -1
  33. package/dist/commands/consult/index.js.map +1 -1
  34. package/dist/commands/init.d.ts.map +1 -1
  35. package/dist/commands/init.js +17 -1
  36. package/dist/commands/init.js.map +1 -1
  37. package/dist/commands/porch/checks.d.ts +29 -0
  38. package/dist/commands/porch/checks.d.ts.map +1 -0
  39. package/dist/commands/porch/checks.js +141 -0
  40. package/dist/commands/porch/checks.js.map +1 -0
  41. package/dist/commands/porch/claude.d.ts +29 -0
  42. package/dist/commands/porch/claude.d.ts.map +1 -0
  43. package/dist/commands/porch/claude.js +79 -0
  44. package/dist/commands/porch/claude.js.map +1 -0
  45. package/dist/commands/porch/index.d.ts +38 -0
  46. package/dist/commands/porch/index.d.ts.map +1 -0
  47. package/dist/commands/porch/index.js +524 -0
  48. package/dist/commands/porch/index.js.map +1 -0
  49. package/dist/commands/porch/plan.d.ts +60 -0
  50. package/dist/commands/porch/plan.d.ts.map +1 -0
  51. package/dist/commands/porch/plan.js +162 -0
  52. package/dist/commands/porch/plan.js.map +1 -0
  53. package/dist/commands/porch/prompts.d.ts +19 -0
  54. package/dist/commands/porch/prompts.d.ts.map +1 -0
  55. package/dist/commands/porch/prompts.js +259 -0
  56. package/dist/commands/porch/prompts.js.map +1 -0
  57. package/dist/commands/porch/protocol.d.ts +57 -0
  58. package/dist/commands/porch/protocol.d.ts.map +1 -0
  59. package/dist/commands/porch/protocol.js +250 -0
  60. package/dist/commands/porch/protocol.js.map +1 -0
  61. package/dist/commands/porch/repl.d.ts +33 -0
  62. package/dist/commands/porch/repl.d.ts.map +1 -0
  63. package/dist/commands/porch/repl.js +206 -0
  64. package/dist/commands/porch/repl.js.map +1 -0
  65. package/dist/commands/porch/run.d.ts +15 -0
  66. package/dist/commands/porch/run.d.ts.map +1 -0
  67. package/dist/commands/porch/run.js +551 -0
  68. package/dist/commands/porch/run.js.map +1 -0
  69. package/dist/commands/porch/signals.d.ts +35 -0
  70. package/dist/commands/porch/signals.d.ts.map +1 -0
  71. package/dist/commands/porch/signals.js +76 -0
  72. package/dist/commands/porch/signals.js.map +1 -0
  73. package/dist/commands/porch/state.d.ts +40 -0
  74. package/dist/commands/porch/state.d.ts.map +1 -0
  75. package/dist/commands/porch/state.js +153 -0
  76. package/dist/commands/porch/state.js.map +1 -0
  77. package/dist/commands/porch/types.d.ts +124 -0
  78. package/dist/commands/porch/types.d.ts.map +1 -0
  79. package/dist/commands/porch/types.js +8 -0
  80. package/dist/commands/porch/types.js.map +1 -0
  81. package/dist/commands/update.d.ts.map +1 -1
  82. package/dist/commands/update.js +19 -0
  83. package/dist/commands/update.js.map +1 -1
  84. package/dist/lib/scaffold.d.ts +24 -0
  85. package/dist/lib/scaffold.d.ts.map +1 -1
  86. package/dist/lib/scaffold.js +78 -0
  87. package/dist/lib/scaffold.js.map +1 -1
  88. package/package.json +6 -2
  89. package/skeleton/porch/prompts/defend.md +103 -0
  90. package/skeleton/porch/prompts/diagnose.md +70 -0
  91. package/skeleton/porch/prompts/evaluate.md +132 -0
  92. package/skeleton/porch/prompts/fix.md +59 -0
  93. package/skeleton/porch/prompts/implement.md +79 -0
  94. package/skeleton/porch/prompts/plan.md +74 -0
  95. package/skeleton/porch/prompts/pr.md +84 -0
  96. package/skeleton/porch/prompts/review.md +179 -0
  97. package/skeleton/porch/prompts/specify.md +53 -0
  98. package/skeleton/porch/prompts/test.md +63 -0
  99. package/skeleton/porch/prompts/understand.md +61 -0
  100. package/skeleton/porch/prompts/verify.md +58 -0
  101. package/skeleton/protocols/bugfix/protocol.json +127 -0
  102. package/skeleton/protocols/protocol-schema.json +237 -0
  103. package/skeleton/protocols/spider/prompts/defend.md +215 -0
  104. package/skeleton/protocols/spider/prompts/evaluate.md +241 -0
  105. package/skeleton/protocols/spider/prompts/implement.md +149 -0
  106. package/skeleton/protocols/spider/prompts/plan.md +214 -0
  107. package/skeleton/protocols/spider/prompts/review.md +217 -0
  108. package/skeleton/protocols/spider/prompts/specify.md +174 -0
  109. package/skeleton/protocols/spider/protocol.json +136 -0
  110. package/skeleton/protocols/spider/templates/plan.md +14 -0
  111. package/skeleton/protocols/tick/protocol.json +151 -0
  112. package/skeleton/roles/architect.md +40 -48
  113. package/skeleton/roles/builder.md +152 -29
  114. package/templates/dashboard/index.html +0 -27
  115. package/templates/dashboard/js/utils.js +0 -86
  116. package/dist/agent-farm/commands/rename.d.ts +0 -13
  117. package/dist/agent-farm/commands/rename.d.ts.map +0 -1
  118. package/dist/agent-farm/commands/rename.js +0 -33
  119. package/dist/agent-farm/commands/rename.js.map +0 -1
  120. package/templates/dashboard/css/activity.css +0 -151
  121. package/templates/dashboard/js/activity.js +0 -112
@@ -0,0 +1,162 @@
1
+ /**
2
+ * Porch Plan Parsing
3
+ *
4
+ * Extracts implementation phases from plan.md files.
5
+ * Looks for `### Phase N: <title>` headers or JSON phases block.
6
+ *
7
+ * Plan phases are simple: pending → in_progress → complete.
8
+ * All checks (implement, defend, evaluate) run together at the end.
9
+ */
10
+ import * as fs from 'node:fs';
11
+ import * as path from 'node:path';
12
+ // ============================================================================
13
+ // Plan File Discovery
14
+ // ============================================================================
15
+ /**
16
+ * Find the plan file for a project
17
+ * Searches both legacy (codev/plans/NNNN-name.md) and new (codev/projects/NNNN-name/plan.md) locations
18
+ */
19
+ export function findPlanFile(projectRoot, projectId, projectName) {
20
+ const searchPaths = [];
21
+ // New structure: codev/projects/<id>-<name>/plan.md
22
+ if (projectName) {
23
+ searchPaths.push(path.join(projectRoot, 'codev/projects', `${projectId}-${projectName}`, 'plan.md'));
24
+ }
25
+ // Legacy structure: codev/plans/<id>-*.md
26
+ const plansDir = path.join(projectRoot, 'codev/plans');
27
+ if (fs.existsSync(plansDir)) {
28
+ const files = fs.readdirSync(plansDir);
29
+ const match = files.find(f => f.startsWith(`${projectId}-`) && f.endsWith('.md'));
30
+ if (match) {
31
+ searchPaths.push(path.join(plansDir, match));
32
+ }
33
+ }
34
+ for (const planPath of searchPaths) {
35
+ if (fs.existsSync(planPath)) {
36
+ return planPath;
37
+ }
38
+ }
39
+ return null;
40
+ }
41
+ // ============================================================================
42
+ // Phase Extraction
43
+ // ============================================================================
44
+ /**
45
+ * Extract phases from plan markdown content
46
+ * Returns phases with status 'pending'
47
+ *
48
+ * Looks for a JSON code block:
49
+ * ```json
50
+ * {"phases": [{"id": "phase_1", "title": "..."}, ...]}
51
+ * ```
52
+ */
53
+ export function extractPlanPhases(planContent) {
54
+ // Look for JSON code block with phases
55
+ const jsonMatch = planContent.match(/```json\s*\n([\s\S]*?)\n```/);
56
+ if (jsonMatch) {
57
+ try {
58
+ const parsed = JSON.parse(jsonMatch[1]);
59
+ if (parsed.phases && Array.isArray(parsed.phases)) {
60
+ return parsed.phases.map((p, index) => ({
61
+ id: p.id,
62
+ title: p.title,
63
+ status: index === 0 ? 'in_progress' : 'pending',
64
+ }));
65
+ }
66
+ }
67
+ catch (e) {
68
+ // JSON parse failed, fall through to default
69
+ console.warn('Failed to parse phases JSON from plan:', e);
70
+ }
71
+ }
72
+ // No JSON block found - return single default phase
73
+ return [{
74
+ id: 'phase_1',
75
+ title: 'Implementation',
76
+ status: 'in_progress',
77
+ }];
78
+ }
79
+ /**
80
+ * Extract phases from a plan file
81
+ * Fails loudly if file doesn't exist
82
+ */
83
+ export function extractPhasesFromFile(planFilePath) {
84
+ if (!fs.existsSync(planFilePath)) {
85
+ throw new Error(`Plan file not found: ${planFilePath}`);
86
+ }
87
+ const content = fs.readFileSync(planFilePath, 'utf-8');
88
+ return extractPlanPhases(content);
89
+ }
90
+ // ============================================================================
91
+ // Phase Navigation
92
+ // ============================================================================
93
+ /**
94
+ * Check if a plan phase is complete
95
+ */
96
+ export function isPlanPhaseComplete(phase) {
97
+ return phase.status === 'complete';
98
+ }
99
+ /**
100
+ * Get the current plan phase (first non-complete phase)
101
+ */
102
+ export function getCurrentPlanPhase(phases) {
103
+ for (const phase of phases) {
104
+ if (phase.status !== 'complete') {
105
+ return phase;
106
+ }
107
+ }
108
+ return null; // All phases complete
109
+ }
110
+ /**
111
+ * Get the next plan phase after a given phase
112
+ */
113
+ export function getNextPlanPhase(phases, currentPhaseId) {
114
+ const currentIndex = phases.findIndex(p => p.id === currentPhaseId);
115
+ if (currentIndex >= 0 && currentIndex < phases.length - 1) {
116
+ return phases[currentIndex + 1];
117
+ }
118
+ return null;
119
+ }
120
+ /**
121
+ * Check if all plan phases are complete
122
+ */
123
+ export function allPlanPhasesComplete(phases) {
124
+ return phases.every(isPlanPhaseComplete);
125
+ }
126
+ /**
127
+ * Advance to the next plan phase
128
+ * Marks current phase complete and next phase in_progress
129
+ * Returns updated phases array and whether to move to review
130
+ */
131
+ export function advancePlanPhase(phases, currentPhaseId) {
132
+ const phaseIndex = phases.findIndex(p => p.id === currentPhaseId);
133
+ if (phaseIndex < 0) {
134
+ return { phases, moveToReview: false };
135
+ }
136
+ const updatedPhases = phases.map((p, i) => {
137
+ if (i === phaseIndex) {
138
+ return { ...p, status: 'complete' };
139
+ }
140
+ if (i === phaseIndex + 1) {
141
+ return { ...p, status: 'in_progress' };
142
+ }
143
+ return p;
144
+ });
145
+ // Check if all phases are now complete
146
+ const moveToReview = updatedPhases.every(p => p.status === 'complete');
147
+ return { phases: updatedPhases, moveToReview };
148
+ }
149
+ /**
150
+ * Get phase content from plan (the text under the phase header)
151
+ */
152
+ export function getPhaseContent(planContent, phaseId) {
153
+ // Extract phase number from id
154
+ const numMatch = phaseId.match(/phase_(\d+)/);
155
+ if (!numMatch)
156
+ return null;
157
+ const phaseNum = numMatch[1];
158
+ const regex = new RegExp(`###\\s*Phase\\s+${phaseNum}:\\s*[^\\n]+\\n([\\s\\S]*?)(?=\\n###\\s*Phase|\\n##\\s|$)`, 'i');
159
+ const match = planContent.match(regex);
160
+ return match ? match[1].trim() : null;
161
+ }
162
+ //# sourceMappingURL=plan.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plan.js","sourceRoot":"","sources":["../../../src/commands/porch/plan.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ /**
2
+ * Phase prompts for Porch
3
+ *
4
+ * Loads phase-specific prompts from the protocol's prompts/ directory.
5
+ * Prompts are markdown files with {{variable}} placeholders.
6
+ *
7
+ * For build-verify cycles, when iteration > 1, previous build outputs
8
+ * and review files are listed so Claude can read them for context.
9
+ */
10
+ import type { ProjectState, Protocol } from './types.js';
11
+ /**
12
+ * Build a prompt for the current phase.
13
+ * Loads from protocol's prompts/ directory if available, otherwise uses fallback.
14
+ *
15
+ * For build-verify phases with iteration > 1, lists previous build outputs
16
+ * and review files so Claude can read them for context.
17
+ */
18
+ export declare function buildPhasePrompt(projectRoot: string, state: ProjectState, protocol: Protocol): string;
19
+ //# sourceMappingURL=prompts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompts.d.ts","sourceRoot":"","sources":["../../../src/commands/porch/prompts.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAIH,OAAO,KAAK,EAAE,YAAY,EAAE,QAAQ,EAA6C,MAAM,YAAY,CAAC;AAoJpG;;;;;;GAMG;AACH,wBAAgB,gBAAgB,CAC9B,WAAW,EAAE,MAAM,EACnB,KAAK,EAAE,YAAY,EACnB,QAAQ,EAAE,QAAQ,GACjB,MAAM,CA8DR"}
@@ -0,0 +1,259 @@
1
+ /**
2
+ * Phase prompts for Porch
3
+ *
4
+ * Loads phase-specific prompts from the protocol's prompts/ directory.
5
+ * Prompts are markdown files with {{variable}} placeholders.
6
+ *
7
+ * For build-verify cycles, when iteration > 1, previous build outputs
8
+ * and review files are listed so Claude can read them for context.
9
+ */
10
+ import * as fs from 'node:fs';
11
+ import * as path from 'node:path';
12
+ import { getPhaseConfig, isPhased, isBuildVerify } from './protocol.js';
13
+ import { findPlanFile, getCurrentPlanPhase, getPhaseContent } from './plan.js';
14
+ /** Locations to search for protocol prompts */
15
+ const PROTOCOL_PATHS = [
16
+ 'codev/protocols',
17
+ 'codev-skeleton/protocols',
18
+ 'skeleton/protocols', // Used in packages/codev
19
+ ];
20
+ /**
21
+ * Get project summary from projectlist.md.
22
+ * Returns the summary field for the given project ID.
23
+ */
24
+ function getProjectSummary(projectRoot, projectId) {
25
+ const projectlistPath = path.join(projectRoot, 'codev', 'projectlist.md');
26
+ if (!fs.existsSync(projectlistPath)) {
27
+ return null;
28
+ }
29
+ try {
30
+ const content = fs.readFileSync(projectlistPath, 'utf-8');
31
+ // Look for the project entry by ID
32
+ // Format: - id: "0076" followed by summary: "..."
33
+ const idPattern = new RegExp(`- id: ["']?${projectId}["']?`, 'i');
34
+ const idMatch = content.match(idPattern);
35
+ if (!idMatch) {
36
+ return null;
37
+ }
38
+ // Find the summary after this ID (within the next ~500 chars)
39
+ const afterId = content.slice(idMatch.index, idMatch.index + 500);
40
+ const summaryMatch = afterId.match(/summary:\s*["'](.+?)["']/);
41
+ if (summaryMatch) {
42
+ return summaryMatch[1];
43
+ }
44
+ return null;
45
+ }
46
+ catch {
47
+ return null;
48
+ }
49
+ }
50
+ /**
51
+ * Find the prompts directory for a protocol.
52
+ */
53
+ function findPromptsDir(projectRoot, protocolName) {
54
+ for (const basePath of PROTOCOL_PATHS) {
55
+ const promptsDir = path.join(projectRoot, basePath, protocolName, 'prompts');
56
+ if (fs.existsSync(promptsDir)) {
57
+ return promptsDir;
58
+ }
59
+ }
60
+ return null;
61
+ }
62
+ /**
63
+ * Load a prompt file from the protocol's prompts directory.
64
+ */
65
+ function loadPromptFile(promptsDir, promptFile) {
66
+ const promptPath = path.join(promptsDir, promptFile);
67
+ if (fs.existsSync(promptPath)) {
68
+ return fs.readFileSync(promptPath, 'utf-8');
69
+ }
70
+ return null;
71
+ }
72
+ /**
73
+ * Substitute template variables in a prompt.
74
+ */
75
+ function substituteVariables(prompt, state, planPhase, summary) {
76
+ const variables = {
77
+ project_id: state.id,
78
+ title: state.title,
79
+ current_state: state.phase,
80
+ protocol: state.protocol,
81
+ };
82
+ // Add summary/goal if available
83
+ if (summary) {
84
+ variables.summary = summary;
85
+ variables.goal = summary; // Alias for convenience
86
+ }
87
+ if (planPhase) {
88
+ variables.plan_phase_id = planPhase.id;
89
+ variables.plan_phase_title = planPhase.title;
90
+ }
91
+ // Replace {{variable}} with values
92
+ return prompt.replace(/\{\{(\w+)\}\}/g, (match, varName) => {
93
+ return variables[varName] || match;
94
+ });
95
+ }
96
+ /**
97
+ * Build a header listing all previous iteration files.
98
+ * Claude can read these files to understand the history and feedback.
99
+ */
100
+ function buildHistoryHeader(history, currentIteration) {
101
+ const lines = [
102
+ '# ⚠️ REVISION REQUIRED',
103
+ '',
104
+ `This is iteration ${currentIteration}. Previous iterations received feedback from reviewers.`,
105
+ '',
106
+ '**Read the files below to understand the history and address the feedback.**',
107
+ '',
108
+ '## Previous Iterations',
109
+ '',
110
+ ];
111
+ for (const record of history) {
112
+ lines.push(`### Iteration ${record.iteration}`);
113
+ lines.push('');
114
+ if (record.build_output) {
115
+ lines.push(`**Build Output:** \`${record.build_output}\``);
116
+ }
117
+ if (record.reviews.length > 0) {
118
+ lines.push('');
119
+ lines.push('**Reviews:**');
120
+ for (const review of record.reviews) {
121
+ const icon = review.verdict === 'APPROVE' ? '✓' :
122
+ review.verdict === 'COMMENT' ? '💬' : '✗';
123
+ lines.push(`- ${review.model} (${icon} ${review.verdict}): \`${review.file}\``);
124
+ }
125
+ }
126
+ lines.push('');
127
+ }
128
+ lines.push('## Instructions');
129
+ lines.push('');
130
+ lines.push('1. Read the review files above to understand the feedback');
131
+ lines.push('2. Address any REQUEST_CHANGES issues');
132
+ lines.push('3. Consider suggestions from COMMENT and APPROVE reviews');
133
+ lines.push('');
134
+ return lines.join('\n');
135
+ }
136
+ /**
137
+ * Build a prompt for the current phase.
138
+ * Loads from protocol's prompts/ directory if available, otherwise uses fallback.
139
+ *
140
+ * For build-verify phases with iteration > 1, lists previous build outputs
141
+ * and review files so Claude can read them for context.
142
+ */
143
+ export function buildPhasePrompt(projectRoot, state, protocol) {
144
+ const phaseConfig = getPhaseConfig(protocol, state.phase);
145
+ if (!phaseConfig) {
146
+ return buildFallbackPrompt(state, 'unknown');
147
+ }
148
+ // Get project summary from projectlist.md
149
+ const summary = getProjectSummary(projectRoot, state.id);
150
+ // Get current plan phase for phased protocols
151
+ let currentPlanPhase = null;
152
+ if (isPhased(protocol, state.phase) && state.plan_phases.length > 0) {
153
+ currentPlanPhase = getCurrentPlanPhase(state.plan_phases);
154
+ }
155
+ // Build history header if this is a retry iteration
156
+ let historyHeader = '';
157
+ if (isBuildVerify(protocol, state.phase) && state.iteration > 1 && state.history.length > 0) {
158
+ historyHeader = buildHistoryHeader(state.history, state.iteration);
159
+ }
160
+ // Try to load prompt from protocol directory
161
+ const promptsDir = findPromptsDir(projectRoot, state.protocol);
162
+ if (promptsDir) {
163
+ const promptFileName = `${state.phase}.md`;
164
+ const promptContent = loadPromptFile(promptsDir, promptFileName);
165
+ if (promptContent) {
166
+ let result = substituteVariables(promptContent, state, currentPlanPhase, summary);
167
+ // Add goal/summary header if available
168
+ if (summary) {
169
+ result = `## Goal\n\n${summary}\n\n---\n\n` + result;
170
+ }
171
+ // Add plan phase context if applicable
172
+ if (currentPlanPhase) {
173
+ result = addPlanPhaseContext(projectRoot, state, currentPlanPhase, result);
174
+ }
175
+ // Prepend history if this is a retry
176
+ if (historyHeader) {
177
+ result = historyHeader + '\n\n---\n\n' + result;
178
+ }
179
+ // Add signal instructions footer
180
+ result += buildSignalFooter();
181
+ return result;
182
+ }
183
+ }
184
+ // Fallback to generic prompt if no protocol prompt found
185
+ let fallback = buildFallbackPrompt(state, phaseConfig.name, currentPlanPhase, summary);
186
+ // Prepend history if this is a retry
187
+ if (historyHeader) {
188
+ fallback = historyHeader + '\n\n---\n\n' + fallback;
189
+ }
190
+ return fallback;
191
+ }
192
+ /**
193
+ * Add plan phase context from the plan file.
194
+ */
195
+ function addPlanPhaseContext(projectRoot, state, planPhase, prompt) {
196
+ const planPath = findPlanFile(projectRoot, state.id, state.title);
197
+ if (!planPath) {
198
+ return prompt;
199
+ }
200
+ try {
201
+ const planContent = fs.readFileSync(planPath, 'utf-8');
202
+ const phaseContent = getPhaseContent(planContent, planPhase.id);
203
+ if (phaseContent) {
204
+ return prompt + `\n\n## Current Plan Phase Details\n\n**${planPhase.id}: ${planPhase.title}**\n\n${phaseContent}\n`;
205
+ }
206
+ }
207
+ catch {
208
+ // Ignore errors reading plan
209
+ }
210
+ return prompt;
211
+ }
212
+ /**
213
+ * Build signal instructions footer.
214
+ */
215
+ function buildSignalFooter() {
216
+ return `
217
+
218
+ ## Completion Signals
219
+
220
+ When you complete your work, output one of these signals:
221
+
222
+ - **Phase complete**: \`PHASE_COMPLETE\`
223
+ - **Need human approval**: \`GATE_NEEDED\`
224
+ - **Blocked on something**: \`BLOCKED: <reason>\`
225
+
226
+ Output the signal on its own line when appropriate.
227
+ `;
228
+ }
229
+ /**
230
+ * Build a fallback prompt when no protocol prompt is found.
231
+ */
232
+ function buildFallbackPrompt(state, phaseName, planPhase, summary) {
233
+ let prompt = `# Phase: ${phaseName}
234
+
235
+ You are executing the ${phaseName} phase of the ${state.protocol.toUpperCase()} protocol.
236
+
237
+ ## Context
238
+
239
+ - **Project ID**: ${state.id}
240
+ - **Project Title**: ${state.title}
241
+ - **Protocol**: ${state.protocol}
242
+ `;
243
+ if (planPhase) {
244
+ prompt += `- **Plan Phase**: ${planPhase.id} - ${planPhase.title}\n`;
245
+ }
246
+ // Add goal from projectlist.md summary
247
+ if (summary) {
248
+ prompt += `\n## Goal\n\n${summary}\n`;
249
+ }
250
+ prompt += `
251
+ ## Task
252
+
253
+ Complete the work for this phase according to the protocol.
254
+
255
+ `;
256
+ prompt += buildSignalFooter();
257
+ return prompt;
258
+ }
259
+ //# sourceMappingURL=prompts.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompts.js","sourceRoot":"","sources":["../../../src/commands/porch/prompts.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,57 @@
1
+ /**
2
+ * Porch Protocol Loading
3
+ *
4
+ * Loads protocol definitions from JSON files.
5
+ * Fails loudly if protocol not found or invalid.
6
+ */
7
+ import type { Protocol, ProtocolPhase, BuildConfig, VerifyConfig, OnCompleteConfig } from './types.js';
8
+ /**
9
+ * Find and load a protocol by name
10
+ * Fails loudly if not found or invalid.
11
+ */
12
+ export declare function loadProtocol(projectRoot: string, protocolName: string): Protocol;
13
+ /**
14
+ * Get phase configuration by id
15
+ */
16
+ export declare function getPhaseConfig(protocol: Protocol, phaseId: string): ProtocolPhase | null;
17
+ /**
18
+ * Get the next phase after the given phase
19
+ */
20
+ export declare function getNextPhase(protocol: Protocol, currentPhaseId: string): ProtocolPhase | null;
21
+ /**
22
+ * Get check commands for a phase
23
+ */
24
+ export declare function getPhaseChecks(protocol: Protocol, phaseId: string): Record<string, string>;
25
+ /**
26
+ * Get gate name for a phase (if any)
27
+ */
28
+ export declare function getPhaseGate(protocol: Protocol, phaseId: string): string | null;
29
+ /**
30
+ * Check if a phase runs per plan phase
31
+ */
32
+ export declare function isPhased(protocol: Protocol, phaseId: string): boolean;
33
+ /**
34
+ * Get checks to run when a plan phase completes (after evaluate stage)
35
+ */
36
+ export declare function getPhaseCompletionChecks(protocol: Protocol): Record<string, string>;
37
+ /**
38
+ * Check if a phase is a build_verify phase
39
+ */
40
+ export declare function isBuildVerify(protocol: Protocol, phaseId: string): boolean;
41
+ /**
42
+ * Get build config for a phase
43
+ */
44
+ export declare function getBuildConfig(protocol: Protocol, phaseId: string): BuildConfig | null;
45
+ /**
46
+ * Get verify config for a phase
47
+ */
48
+ export declare function getVerifyConfig(protocol: Protocol, phaseId: string): VerifyConfig | null;
49
+ /**
50
+ * Get max iterations for a build_verify phase
51
+ */
52
+ export declare function getMaxIterations(protocol: Protocol, phaseId: string): number;
53
+ /**
54
+ * Get on_complete config for a phase
55
+ */
56
+ export declare function getOnCompleteConfig(protocol: Protocol, phaseId: string): OnCompleteConfig | null;
57
+ //# sourceMappingURL=protocol.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"protocol.d.ts","sourceRoot":"","sources":["../../../src/commands/porch/protocol.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAIH,OAAO,KAAK,EAAE,QAAQ,EAAE,aAAa,EAAE,WAAW,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAYvG;;;GAGG;AACH,wBAAgB,YAAY,CAAC,WAAW,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,QAAQ,CAoBhF;AA+ID;;GAEG;AACH,wBAAgB,cAAc,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,aAAa,GAAG,IAAI,CAExF;AAED;;GAEG;AACH,wBAAgB,YAAY,CAAC,QAAQ,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,GAAG,aAAa,GAAG,IAAI,CAM7F;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAa1F;AAED;;GAEG;AACH,wBAAgB,YAAY,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAG/E;AAED;;GAEG;AACH,wBAAgB,QAAQ,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,OAAO,CAGrE;AAED;;GAEG;AACH,wBAAgB,wBAAwB,CAAC,QAAQ,EAAE,QAAQ,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAEnF;AAED;;GAEG;AACH,wBAAgB,aAAa,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,OAAO,CAG1E;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,WAAW,GAAG,IAAI,CAGtF;AAED;;GAEG;AACH,wBAAgB,eAAe,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,IAAI,CAGxF;AAED;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,CAG5E;AAED;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,gBAAgB,GAAG,IAAI,CAGhG"}