edsger 0.2.3 → 0.2.5

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 (89) hide show
  1. package/README.md +17 -0
  2. package/dist/api/features/batch-operations.d.ts +16 -0
  3. package/dist/api/features/batch-operations.js +100 -0
  4. package/dist/api/features/index.d.ts +1 -0
  5. package/dist/api/features/index.js +1 -0
  6. package/dist/api/features/test-cases.d.ts +8 -0
  7. package/dist/api/features/test-cases.js +45 -0
  8. package/dist/api/features/user-stories.d.ts +8 -0
  9. package/dist/api/features/user-stories.js +45 -0
  10. package/dist/cli/commands/refactor-command.d.ts +2 -0
  11. package/dist/cli/commands/refactor-command.js +123 -0
  12. package/dist/cli/formatters/formatter-utils.d.ts +23 -0
  13. package/dist/cli/formatters/formatter-utils.js +67 -0
  14. package/dist/cli/index.js +7 -0
  15. package/dist/cli/utils/command-handler.d.ts +23 -0
  16. package/dist/cli/utils/command-handler.js +39 -0
  17. package/dist/cli.d.ts +2 -2
  18. package/dist/cli.js +4 -99
  19. package/dist/phases/code-implementation/analyzer-helpers.d.ts +28 -0
  20. package/dist/phases/code-implementation/analyzer-helpers.js +177 -0
  21. package/dist/phases/code-implementation/analyzer.d.ts +2 -0
  22. package/dist/phases/code-implementation/analyzer.js +304 -175
  23. package/dist/phases/code-implementation-verification/index.d.ts +1 -0
  24. package/dist/phases/code-implementation-verification/index.js +1 -0
  25. package/dist/phases/code-implementation-verification/verifier.d.ts +31 -0
  26. package/dist/phases/code-implementation-verification/verifier.js +196 -0
  27. package/dist/phases/feature-analysis/analyzer-helpers.d.ts +62 -0
  28. package/dist/phases/feature-analysis/analyzer-helpers.js +450 -0
  29. package/dist/phases/feature-analysis/analyzer.d.ts +1 -0
  30. package/dist/phases/feature-analysis/analyzer.js +132 -219
  31. package/dist/phases/feature-analysis-verification/index.d.ts +1 -0
  32. package/dist/phases/feature-analysis-verification/index.js +1 -0
  33. package/dist/phases/feature-analysis-verification/verifier.d.ts +37 -0
  34. package/dist/phases/feature-analysis-verification/verifier.js +147 -0
  35. package/dist/phases/pull-request/creator.js +2 -1
  36. package/dist/phases/technical-design/analyzer-helpers.d.ts +37 -0
  37. package/dist/phases/technical-design/analyzer-helpers.js +144 -0
  38. package/dist/phases/technical-design/analyzer.d.ts +3 -0
  39. package/dist/phases/technical-design/analyzer.js +282 -318
  40. package/dist/phases/technical-design-verification/index.d.ts +1 -0
  41. package/dist/phases/technical-design-verification/index.js +1 -0
  42. package/dist/phases/technical-design-verification/verifier.d.ts +36 -0
  43. package/dist/phases/technical-design-verification/verifier.js +147 -0
  44. package/dist/prompts/checklist-verification.d.ts +11 -0
  45. package/dist/prompts/checklist-verification.js +153 -0
  46. package/dist/prompts/code-implementation-improvement.d.ts +5 -0
  47. package/dist/prompts/code-implementation-improvement.js +108 -0
  48. package/dist/prompts/code-implementation-verification.d.ts +3 -0
  49. package/dist/prompts/code-implementation-verification.js +176 -0
  50. package/dist/prompts/feature-analysis-improvement.d.ts +8 -0
  51. package/dist/prompts/feature-analysis-improvement.js +109 -0
  52. package/dist/prompts/feature-analysis.js +1 -1
  53. package/dist/prompts/technical-design-improvement.d.ts +5 -0
  54. package/dist/prompts/technical-design-improvement.js +93 -0
  55. package/dist/prompts/technical-design-verification.d.ts +11 -0
  56. package/dist/prompts/technical-design-verification.js +134 -0
  57. package/dist/prompts/technical-design.js +1 -1
  58. package/dist/services/audit-logs.d.ts +60 -0
  59. package/dist/services/audit-logs.js +115 -0
  60. package/dist/services/checklist.d.ts +1 -0
  61. package/dist/types/index.d.ts +19 -0
  62. package/dist/workflow-runner/executors/phase-executor.js +56 -12
  63. package/package.json +1 -1
  64. package/dist/api/features.d.ts +0 -100
  65. package/dist/api/features.js +0 -219
  66. package/dist/logger.d.ts +0 -19
  67. package/dist/logger.js +0 -52
  68. package/dist/types.d.ts +0 -99
  69. package/dist/types.js +0 -1
  70. package/dist/utils/image-processor.d.ts +0 -5
  71. package/dist/utils/image-processor.js +0 -55
  72. package/dist/workflow-runner/config/stage-configs.d.ts +0 -5
  73. package/dist/workflow-runner/config/stage-configs.js +0 -34
  74. package/dist/workflow-runner/core/feature-filter.test.d.ts +0 -4
  75. package/dist/workflow-runner/core/feature-filter.test.js +0 -127
  76. package/dist/workflow-runner/executors/stage-executor.d.ts +0 -8
  77. package/dist/workflow-runner/executors/stage-executor.js +0 -49
  78. package/dist/workflow-runner/feature-fetcher.d.ts +0 -41
  79. package/dist/workflow-runner/feature-fetcher.js +0 -121
  80. package/dist/workflow-runner/feature-service.d.ts +0 -17
  81. package/dist/workflow-runner/feature-service.js +0 -60
  82. package/dist/workflow-runner/pipeline.d.ts +0 -18
  83. package/dist/workflow-runner/pipeline.js +0 -197
  84. package/dist/workflow-runner/processor.d.ts +0 -40
  85. package/dist/workflow-runner/processor.js +0 -191
  86. package/dist/workflow-runner/status-updater.d.ts +0 -27
  87. package/dist/workflow-runner/status-updater.js +0 -80
  88. package/dist/workflow-runner/types.d.ts +0 -48
  89. package/dist/workflow-runner/types.js +0 -4
package/dist/cli.js CHANGED
@@ -1,100 +1,5 @@
1
1
  #!/usr/bin/env node
2
- var _a, _b, _c;
3
- import { Command } from 'commander';
4
- import { readFileSync } from 'fs';
5
- import { join, dirname } from 'path';
6
- import { fileURLToPath } from 'url';
7
- import { config as dotenvConfig } from 'dotenv';
8
- import { logError } from './utils/logger.js';
9
- import { runWorkflow } from './cli/commands/workflow-command.js';
10
- import { runFeatureAnalysis } from './cli/commands/feature-analysis-command.js';
11
- import { runTechnicalDesign } from './cli/commands/technical-design-command.js';
12
- import { runCodeImplementation } from './cli/commands/code-implementation-command.js';
13
- import { runFunctionalTestingCommand } from './cli/commands/functional-testing-command.js';
14
- import { runCodeReview } from './cli/commands/code-review-command.js';
15
- // Get package.json version dynamically
16
- const __filename = fileURLToPath(import.meta.url);
17
- const __dirname = dirname(__filename);
18
- const packageJson = JSON.parse(readFileSync(join(__dirname, '../package.json'), 'utf-8'));
19
- const version = packageJson.version;
20
- // Load environment variables from .env file
21
- // Load from current working directory (where the command is run)
22
- dotenvConfig({ path: join(process.cwd(), '.env') });
23
- const program = new Command();
24
- program
25
- .name('edsger')
26
- .description('AI-powered code review CLI tool using Claude Code SDK')
27
- .version(version);
28
- program
29
- .option('-r, --review', 'Review code changes')
30
- .option('-s, --staged', 'Review only staged changes')
31
- .option('-f, --files <patterns...>', 'Review specific file patterns')
32
- .option('--feature-analysis <featureId>', 'Analyze feature and generate user stories/test cases')
33
- .option('--technical-design <featureId>', 'Generate technical design for a feature')
34
- .option('--implement <featureId>', 'Implement code for a feature based on its specifications')
35
- .option('--test <featureId>', 'Run functional tests for a feature using Playwright')
36
- .option('--workflow', 'Run continuous workflow processor to handle ready_for_dev features')
37
- .option('-c, --config <path>', 'Path to config file')
38
- .option('-v, --verbose', 'Verbose output');
39
- program.action(async (options) => {
40
- try {
41
- await runEdsger(options);
42
- }
43
- catch (error) {
44
- logError(error instanceof Error ? error.message : String(error));
45
- process.exit(1);
46
- }
47
- });
48
- export const runEdsger = async (options) => {
49
- // Handle workflow mode
50
- if (options.workflow) {
51
- await runWorkflow(options);
52
- return;
53
- }
54
- // Handle feature analysis mode
55
- if (options.featureAnalysis) {
56
- await runFeatureAnalysis(options);
57
- return;
58
- }
59
- // Handle technical design mode
60
- if (options.technicalDesign) {
61
- await runTechnicalDesign(options);
62
- return;
63
- }
64
- // Handle code implementation mode
65
- if (options.implement) {
66
- await runCodeImplementation(options);
67
- return;
68
- }
69
- // Handle functional testing mode
70
- if (options.test) {
71
- await runFunctionalTestingCommand(options);
72
- return;
73
- }
74
- // Handle code review mode (explicit --review flag or as fallback)
75
- if (options.review || options.staged || options.files) {
76
- await runCodeReview(options);
77
- return;
78
- }
79
- // Default to workflow mode if environment is configured, otherwise code review
80
- const productId = process.env.EDSGER_PRODUCT_ID;
81
- const mcpToken = process.env.EDSGER_MCP_TOKEN;
82
- if (productId && mcpToken) {
83
- // Environment is configured for workflow, use that as default
84
- options.workflow = true;
85
- await runWorkflow(options);
86
- }
87
- else {
88
- // Default to code review mode
89
- await runCodeReview(options);
90
- }
91
- };
92
- // Only parse when this file is run directly (not imported as module)
93
- // Check if this is the main module being executed
94
- const isMainModule = import.meta.url === `file://${process.argv[1]}` ||
95
- ((_a = process.argv[1]) === null || _a === void 0 ? void 0 : _a.endsWith('/edsger')) ||
96
- ((_b = process.argv[1]) === null || _b === void 0 ? void 0 : _b.endsWith('\\edsger')) ||
97
- ((_c = process.argv[1]) === null || _c === void 0 ? void 0 : _c.endsWith('cli.js'));
98
- if (isMainModule) {
99
- program.parse();
100
- }
2
+ // Re-export the CLI module
3
+ export * from './cli/index.js';
4
+ // Import and trigger the CLI parsing if this file is run directly
5
+ import './cli/index.js';
@@ -0,0 +1,28 @@
1
+ import { EdsgerConfig } from '../../types/index.js';
2
+ import { ChecklistPhaseContext } from '../../services/checklist.js';
3
+ import { ChecklistVerificationResult } from '../code-implementation-verification/index.js';
4
+ import { CodeImplementationResult } from './analyzer.js';
5
+ export interface VerificationCycleResult {
6
+ passed: boolean;
7
+ verificationResult: ChecklistVerificationResult | null;
8
+ nextPrompt: string | null;
9
+ }
10
+ /**
11
+ * Perform a complete verification cycle:
12
+ * 1. Verify code implementation against checklist
13
+ * 2. Log results to audit logs
14
+ * 3. Create improvement prompt if verification failed
15
+ */
16
+ export declare function performVerificationCycle(branchName: string, baseBranch: string, checklistContext: ChecklistPhaseContext | null, featureId: string, featureName: string, featureDescription: string | undefined, config: EdsgerConfig, currentIteration: number, maxIterations: number, mcpServerUrl: string, mcpToken: string, verbose?: boolean): Promise<VerificationCycleResult>;
17
+ /**
18
+ * Build successful implementation result
19
+ */
20
+ export declare function buildImplementationResult(featureId: string, branchName: string, implementationSummary: string, filesModified: string[], commitHash: string, iterations: number, checklistResults?: any[], checklistItemResults?: any[]): CodeImplementationResult;
21
+ /**
22
+ * Build verification failure result
23
+ */
24
+ export declare function buildVerificationFailureResult(featureId: string, branchName: string, implementationSummary: string, filesModified: string[], commitHash: string, verificationResult: ChecklistVerificationResult, iterations: number): CodeImplementationResult;
25
+ /**
26
+ * Build error result for no implementation
27
+ */
28
+ export declare function buildNoResultsError(featureId: string, branchName: string): CodeImplementationResult;
@@ -0,0 +1,177 @@
1
+ import { logInfo, logError } from '../../utils/logger.js';
2
+ import { logFeatureVerificationEvent } from '../../services/audit-logs.js';
3
+ import { formatChecklistsForContext, processChecklistItemResultsFromResponse, } from '../../services/checklist.js';
4
+ import { verifyCodeImplementationCompliance, } from '../code-implementation-verification/index.js';
5
+ import { createCodeImplementationImprovementPrompt } from '../../prompts/code-implementation-improvement.js';
6
+ /**
7
+ * Perform a complete verification cycle:
8
+ * 1. Verify code implementation against checklist
9
+ * 2. Log results to audit logs
10
+ * 3. Create improvement prompt if verification failed
11
+ */
12
+ export async function performVerificationCycle(branchName, baseBranch, checklistContext, featureId, featureName, featureDescription, config, currentIteration, maxIterations, mcpServerUrl, mcpToken, verbose) {
13
+ // If no checklist, verification passes automatically
14
+ if (!checklistContext || checklistContext.checklists.length === 0) {
15
+ if (verbose) {
16
+ logInfo('ā„¹ļø No checklists to verify - skipping verification');
17
+ }
18
+ return {
19
+ passed: true,
20
+ verificationResult: null,
21
+ nextPrompt: null,
22
+ };
23
+ }
24
+ if (verbose) {
25
+ logInfo(`\nšŸ” Verifying code implementation (iteration ${currentIteration}/${maxIterations})...`);
26
+ }
27
+ // Format checklist context for verification
28
+ const checklistInfo = formatChecklistsForContext(checklistContext);
29
+ // Perform verification
30
+ const verificationResult = await verifyCodeImplementationCompliance({
31
+ featureId,
32
+ branchName,
33
+ baseBranch,
34
+ featureName,
35
+ featureDescription,
36
+ checklistContext: checklistInfo,
37
+ verbose,
38
+ }, config);
39
+ // Determine if verification passed
40
+ const verificationPassed = verificationResult.rejected_count === 0;
41
+ // Log verification result
42
+ if (verificationPassed) {
43
+ if (verbose) {
44
+ logInfo(`āœ… Code verification passed: ${verificationResult.confirmed_count} confirmed, ${verificationResult.uncertain_count} uncertain`);
45
+ }
46
+ await logFeatureVerificationEvent(mcpServerUrl, mcpToken, {
47
+ featureId,
48
+ phase: 'code_implementation',
49
+ iteration: currentIteration,
50
+ result: 'success',
51
+ verificationData: {
52
+ confirmed_count: verificationResult.confirmed_count,
53
+ rejected_count: verificationResult.rejected_count,
54
+ uncertain_count: verificationResult.uncertain_count,
55
+ summary: verificationResult.summary,
56
+ },
57
+ }, verbose);
58
+ // Create checklist results based on verification result
59
+ const checklistItemResults = verificationResult.item_verifications.map((item) => ({
60
+ checklist_item_id: item.checklist_item_id,
61
+ is_passed: item.is_satisfied && item.verification_status === 'confirmed',
62
+ value: item.verification_status,
63
+ notes: item.verification_reason || undefined,
64
+ }));
65
+ await processChecklistItemResultsFromResponse({ featureId, mcpServerUrl, mcpToken }, 'code_implementation', checklistItemResults, verbose);
66
+ }
67
+ else {
68
+ logError(`āŒ Iteration ${currentIteration}: Code verification FAILED - ${verificationResult.rejected_count} items rejected, ${verificationResult.uncertain_count} uncertain`);
69
+ await logFeatureVerificationEvent(mcpServerUrl, mcpToken, {
70
+ featureId,
71
+ phase: 'code_implementation',
72
+ iteration: currentIteration,
73
+ result: 'error',
74
+ verificationData: {
75
+ confirmed_count: verificationResult.confirmed_count,
76
+ rejected_count: verificationResult.rejected_count,
77
+ uncertain_count: verificationResult.uncertain_count,
78
+ rejected_items: verificationResult.item_verifications
79
+ .filter((item) => item.verification_status === 'rejected')
80
+ .map((item) => ({
81
+ checklist_item_id: item.checklist_item_id,
82
+ reason: item.verification_reason || 'No reason provided',
83
+ concerns: item.concerns || [],
84
+ improvement_suggestions: item.improvement_suggestions || [],
85
+ })),
86
+ uncertain_items: verificationResult.item_verifications
87
+ .filter((item) => item.verification_status === 'uncertain')
88
+ .map((item) => ({
89
+ checklist_item_id: item.checklist_item_id,
90
+ reason: item.verification_reason || 'No reason provided',
91
+ concerns: item.concerns || [],
92
+ improvement_suggestions: item.improvement_suggestions || [],
93
+ })),
94
+ summary: verificationResult.summary,
95
+ overall_suggestions: verificationResult.overall_suggestions || [],
96
+ },
97
+ }, verbose);
98
+ }
99
+ // If verification failed and we have iterations left, create improvement prompt
100
+ let nextPrompt = null;
101
+ if (!verificationPassed && currentIteration < maxIterations) {
102
+ nextPrompt = createCodeImplementationImprovementPrompt(verificationResult, branchName, baseBranch);
103
+ if (verbose) {
104
+ logInfo(`\nšŸ“ Created improvement prompt for next iteration (${maxIterations - currentIteration} attempts remaining)`);
105
+ }
106
+ }
107
+ else if (!verificationPassed) {
108
+ // Max iterations reached - create checklist results based on final verification
109
+ if (verbose) {
110
+ logInfo(`\nšŸ“‹ Creating checklist results for final iteration (max iterations reached)`);
111
+ }
112
+ // Create checklist results for all items (both passed and failed)
113
+ const checklistItemResults = verificationResult.item_verifications.map((item) => ({
114
+ checklist_item_id: item.checklist_item_id,
115
+ is_passed: item.is_satisfied && item.verification_status === 'confirmed',
116
+ value: item.verification_status,
117
+ notes: item.verification_reason || undefined,
118
+ }));
119
+ await processChecklistItemResultsFromResponse({ featureId, mcpServerUrl, mcpToken }, 'code_implementation', checklistItemResults, verbose);
120
+ }
121
+ return {
122
+ passed: verificationPassed,
123
+ verificationResult,
124
+ nextPrompt,
125
+ };
126
+ }
127
+ /**
128
+ * Build successful implementation result
129
+ */
130
+ export function buildImplementationResult(featureId, branchName, implementationSummary, filesModified, commitHash, iterations, checklistResults, checklistItemResults) {
131
+ return {
132
+ featureId,
133
+ branchName,
134
+ implementationSummary,
135
+ status: 'success',
136
+ message: 'Code implementation completed successfully',
137
+ filesModified,
138
+ commitHash,
139
+ data: {
140
+ checklist_results: checklistResults,
141
+ checklist_item_results: checklistItemResults,
142
+ },
143
+ };
144
+ }
145
+ /**
146
+ * Build verification failure result
147
+ */
148
+ export function buildVerificationFailureResult(featureId, branchName, implementationSummary, filesModified, commitHash, verificationResult, iterations) {
149
+ return {
150
+ featureId,
151
+ branchName,
152
+ implementationSummary,
153
+ status: 'error',
154
+ message: `Code implementation verification failed after ${iterations} iterations. Code committed for manual review.`,
155
+ filesModified,
156
+ commitHash,
157
+ data: {
158
+ verification_failed: true,
159
+ verification_summary: verificationResult.summary,
160
+ rejected_count: verificationResult.rejected_count,
161
+ uncertain_count: verificationResult.uncertain_count,
162
+ improvement_suggestions: verificationResult.overall_suggestions || [],
163
+ },
164
+ };
165
+ }
166
+ /**
167
+ * Build error result for no implementation
168
+ */
169
+ export function buildNoResultsError(featureId, branchName) {
170
+ return {
171
+ featureId,
172
+ branchName,
173
+ implementationSummary: null,
174
+ status: 'error',
175
+ message: 'Code implementation failed - no results produced',
176
+ };
177
+ }
@@ -6,6 +6,7 @@ export interface CodeImplementationOptions {
6
6
  mcpToken: string;
7
7
  verbose?: boolean;
8
8
  baseBranch?: string;
9
+ maxVerificationIterations?: number;
9
10
  }
10
11
  export interface CodeImplementationResult {
11
12
  featureId: string;
@@ -27,6 +28,7 @@ export interface CodeImplementationResult {
27
28
  value?: any;
28
29
  notes?: string;
29
30
  }>;
31
+ [key: string]: any;
30
32
  };
31
33
  }
32
34
  export declare const implementFeatureCode: (options: CodeImplementationOptions, config: EdsgerConfig, checklistContext?: ChecklistPhaseContext | null) => Promise<CodeImplementationResult>;