codereview-aia 0.1.0
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/.cr-aia.yml +23 -0
- package/.crignore +0 -0
- package/dist/index.js +27 -0
- package/package.json +85 -0
- package/src/analysis/FindingsExtractor.ts +431 -0
- package/src/analysis/ai-detection/analyzers/BaseAnalyzer.ts +267 -0
- package/src/analysis/ai-detection/analyzers/DocumentationAnalyzer.ts +622 -0
- package/src/analysis/ai-detection/analyzers/GitHistoryAnalyzer.ts +430 -0
- package/src/analysis/ai-detection/core/AIDetectionEngine.ts +467 -0
- package/src/analysis/ai-detection/types/DetectionTypes.ts +406 -0
- package/src/analysis/ai-detection/utils/SubmissionConverter.ts +390 -0
- package/src/analysis/context/ReviewContext.ts +378 -0
- package/src/analysis/context/index.ts +7 -0
- package/src/analysis/index.ts +8 -0
- package/src/analysis/tokens/TokenAnalysisFormatter.ts +154 -0
- package/src/analysis/tokens/TokenAnalyzer.ts +747 -0
- package/src/analysis/tokens/index.ts +8 -0
- package/src/clients/base/abstractClient.ts +190 -0
- package/src/clients/base/httpClient.ts +160 -0
- package/src/clients/base/index.ts +12 -0
- package/src/clients/base/modelDetection.ts +107 -0
- package/src/clients/base/responseProcessor.ts +586 -0
- package/src/clients/factory/clientFactory.ts +55 -0
- package/src/clients/factory/index.ts +8 -0
- package/src/clients/implementations/index.ts +8 -0
- package/src/clients/implementations/openRouterClient.ts +411 -0
- package/src/clients/openRouterClient.ts +863 -0
- package/src/clients/openRouterClientWrapper.ts +44 -0
- package/src/clients/utils/directoryStructure.ts +52 -0
- package/src/clients/utils/index.ts +11 -0
- package/src/clients/utils/languageDetection.ts +44 -0
- package/src/clients/utils/promptFormatter.ts +105 -0
- package/src/clients/utils/promptLoader.ts +53 -0
- package/src/clients/utils/tokenCounter.ts +297 -0
- package/src/core/ApiClientSelector.ts +37 -0
- package/src/core/ConfigurationService.ts +591 -0
- package/src/core/ConsolidationService.ts +423 -0
- package/src/core/InteractiveDisplayManager.ts +81 -0
- package/src/core/OutputManager.ts +275 -0
- package/src/core/ReviewGenerator.ts +140 -0
- package/src/core/fileDiscovery.ts +237 -0
- package/src/core/handlers/EstimationHandler.ts +104 -0
- package/src/core/handlers/FileProcessingHandler.ts +204 -0
- package/src/core/handlers/OutputHandler.ts +125 -0
- package/src/core/handlers/ReviewExecutor.ts +104 -0
- package/src/core/reviewOrchestrator.ts +333 -0
- package/src/core/utils/ModelInfoUtils.ts +56 -0
- package/src/formatters/outputFormatter.ts +62 -0
- package/src/formatters/utils/IssueFormatters.ts +83 -0
- package/src/formatters/utils/JsonFormatter.ts +77 -0
- package/src/formatters/utils/MarkdownFormatters.ts +609 -0
- package/src/formatters/utils/MetadataFormatter.ts +269 -0
- package/src/formatters/utils/ModelInfoExtractor.ts +115 -0
- package/src/index.ts +27 -0
- package/src/plugins/PluginInterface.ts +50 -0
- package/src/plugins/PluginManager.ts +126 -0
- package/src/prompts/PromptManager.ts +69 -0
- package/src/prompts/cache/PromptCache.ts +50 -0
- package/src/prompts/promptText/common/variables/css-frameworks.json +33 -0
- package/src/prompts/promptText/common/variables/framework-versions.json +45 -0
- package/src/prompts/promptText/frameworks/react/comprehensive.hbs +19 -0
- package/src/prompts/promptText/languages/css/comprehensive.hbs +18 -0
- package/src/prompts/promptText/languages/generic/comprehensive.hbs +20 -0
- package/src/prompts/promptText/languages/html/comprehensive.hbs +18 -0
- package/src/prompts/promptText/languages/javascript/comprehensive.hbs +18 -0
- package/src/prompts/promptText/languages/python/comprehensive.hbs +18 -0
- package/src/prompts/promptText/languages/typescript/comprehensive.hbs +18 -0
- package/src/runtime/auth/service.ts +58 -0
- package/src/runtime/auth/session.ts +103 -0
- package/src/runtime/auth/types.ts +11 -0
- package/src/runtime/cliEntry.ts +196 -0
- package/src/runtime/errors.ts +13 -0
- package/src/runtime/fileCollector.ts +188 -0
- package/src/runtime/manifest.ts +64 -0
- package/src/runtime/openrouterProxy.ts +45 -0
- package/src/runtime/proxyConfig.ts +94 -0
- package/src/runtime/proxyEnvironment.ts +71 -0
- package/src/runtime/reportMerge.ts +102 -0
- package/src/runtime/reporting/markdownReportBuilder.ts +138 -0
- package/src/runtime/reporting/reportDataCollector.ts +234 -0
- package/src/runtime/reporting/summaryGenerator.ts +86 -0
- package/src/runtime/reviewPipeline.ts +155 -0
- package/src/runtime/runAiCodeReview.ts +153 -0
- package/src/runtime/runtimeConfig.ts +5 -0
- package/src/runtime/ui/Layout.tsx +57 -0
- package/src/runtime/ui/RuntimeApp.tsx +150 -0
- package/src/runtime/ui/inkModules.ts +73 -0
- package/src/runtime/ui/screens/AuthScreen.tsx +128 -0
- package/src/runtime/ui/screens/ModeSelection.tsx +52 -0
- package/src/runtime/ui/screens/ProgressScreen.tsx +55 -0
- package/src/runtime/ui/screens/ResultsScreen.tsx +76 -0
- package/src/strategies/ArchitecturalReviewStrategy.ts +54 -0
- package/src/strategies/CodingTestReviewStrategy.ts +920 -0
- package/src/strategies/ConsolidatedReviewStrategy.ts +59 -0
- package/src/strategies/ExtractPatternsReviewStrategy.ts +64 -0
- package/src/strategies/MultiPassReviewStrategy.ts +785 -0
- package/src/strategies/ReviewStrategy.ts +64 -0
- package/src/strategies/StrategyFactory.ts +79 -0
- package/src/strategies/index.ts +14 -0
- package/src/tokenizers/baseTokenizer.ts +61 -0
- package/src/tokenizers/gptTokenizer.ts +27 -0
- package/src/tokenizers/index.ts +8 -0
- package/src/types/apiResponses.ts +40 -0
- package/src/types/cli.ts +24 -0
- package/src/types/common.ts +39 -0
- package/src/types/configuration.ts +201 -0
- package/src/types/handlebars.d.ts +5 -0
- package/src/types/patch.d.ts +25 -0
- package/src/types/review.ts +294 -0
- package/src/types/reviewContext.d.ts +65 -0
- package/src/types/reviewSchema.ts +181 -0
- package/src/types/structuredReview.ts +167 -0
- package/src/types/tokenAnalysis.ts +56 -0
- package/src/utils/FileReader.ts +93 -0
- package/src/utils/FileWriter.ts +76 -0
- package/src/utils/PathGenerator.ts +97 -0
- package/src/utils/api/apiUtils.ts +14 -0
- package/src/utils/api/index.ts +1 -0
- package/src/utils/apiErrorHandler.ts +287 -0
- package/src/utils/ciDataCollector.ts +252 -0
- package/src/utils/codingTestConfigLoader.ts +466 -0
- package/src/utils/dependencies/aiDependencyAnalyzer.ts +454 -0
- package/src/utils/detection/frameworkDetector.ts +879 -0
- package/src/utils/detection/index.ts +10 -0
- package/src/utils/detection/projectTypeDetector.ts +518 -0
- package/src/utils/diagramGenerator.ts +206 -0
- package/src/utils/errorLogger.ts +60 -0
- package/src/utils/estimationUtils.ts +407 -0
- package/src/utils/fileFilters.ts +373 -0
- package/src/utils/fileSystem.ts +57 -0
- package/src/utils/index.ts +36 -0
- package/src/utils/logger.ts +240 -0
- package/src/utils/pathValidator.ts +98 -0
- package/src/utils/priorityFilter.ts +59 -0
- package/src/utils/projectDocs.ts +189 -0
- package/src/utils/promptPaths.ts +29 -0
- package/src/utils/promptTemplateManager.ts +157 -0
- package/src/utils/review/consolidateReview.ts +553 -0
- package/src/utils/review/fixDisplay.ts +100 -0
- package/src/utils/review/fixImplementation.ts +61 -0
- package/src/utils/review/index.ts +36 -0
- package/src/utils/review/interactiveProcessing.ts +294 -0
- package/src/utils/review/progressTracker.ts +296 -0
- package/src/utils/review/reviewExtraction.ts +382 -0
- package/src/utils/review/types.ts +46 -0
- package/src/utils/reviewActionHandler.ts +18 -0
- package/src/utils/reviewParser.ts +253 -0
- package/src/utils/sanitizer.ts +238 -0
- package/src/utils/smartFileSelector.ts +255 -0
- package/src/utils/templateLoader.ts +514 -0
- package/src/utils/treeGenerator.ts +153 -0
- package/tsconfig.build.json +14 -0
- package/tsconfig.json +59 -0
|
@@ -0,0 +1,333 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Review orchestrator module.
|
|
3
|
+
*
|
|
4
|
+
* This module is responsible for coordinating the review process,
|
|
5
|
+
* selecting the appropriate API client, and managing the review workflow.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import * as path from 'node:path';
|
|
9
|
+
import type { ProgrammingLanguage } from '../types/common';
|
|
10
|
+
import type { ReviewOptions } from '../types/review';
|
|
11
|
+
import { getApiKeyType } from '../utils/api/apiUtils';
|
|
12
|
+
import { getConfig } from '../core/ConfigurationService';
|
|
13
|
+
import { configurationService } from '../core/ConfigurationService';
|
|
14
|
+
import { createDirectory } from '../utils/fileSystem';
|
|
15
|
+
import logger from '../utils/logger';
|
|
16
|
+
import { readProjectDocs } from '../utils/projectDocs';
|
|
17
|
+
// Import other dependencies
|
|
18
|
+
import { selectApiClient } from './ApiClientSelector';
|
|
19
|
+
import { performEstimation } from './handlers/EstimationHandler';
|
|
20
|
+
// Import handlers
|
|
21
|
+
import { discoverFilesForReview, readFilesForReview } from './handlers/FileProcessingHandler';
|
|
22
|
+
import { createOutputDirectory, handleReviewOutput } from './handlers/OutputHandler';
|
|
23
|
+
import { executeReview } from './handlers/ReviewExecutor';
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Orchestrate the code review process
|
|
28
|
+
*
|
|
29
|
+
* This function is the main entry point for the code review process. It coordinates
|
|
30
|
+
* the entire review workflow, including:
|
|
31
|
+
* - Validating inputs and environment variables
|
|
32
|
+
* - Selecting the appropriate API client based on available API keys
|
|
33
|
+
* - Discovering files to review
|
|
34
|
+
* - Handling different review types (consolidated, individual, architectural)
|
|
35
|
+
* - Managing output directories and file generation
|
|
36
|
+
* - Supporting interactive mode for real-time feedback
|
|
37
|
+
*
|
|
38
|
+
* @param target Path to the file or directory to review
|
|
39
|
+
* @param options Review options including type, output format, and interactive mode
|
|
40
|
+
* @throws Error if the review process fails for any reason
|
|
41
|
+
*/
|
|
42
|
+
export async function orchestrateReview(target: string, options: ReviewOptions): Promise<void> {
|
|
43
|
+
// Initialize configuration
|
|
44
|
+
getConfig();
|
|
45
|
+
try {
|
|
46
|
+
// Validate input parameters
|
|
47
|
+
if (options === undefined) {
|
|
48
|
+
throw new Error('Review options object must be provided');
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// Validate that options contains a review type
|
|
52
|
+
if (!options.type) {
|
|
53
|
+
throw new Error('Review type must be specified in options');
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Ensure target is defined with a default of "." for current directory
|
|
57
|
+
const effectiveTarget = target || '.';
|
|
58
|
+
|
|
59
|
+
// Log if we're using the default target
|
|
60
|
+
if (!target || target.trim() === '') {
|
|
61
|
+
logger.info('No target path provided, defaulting to current directory (".")');
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// Add debug information if debug mode is enabled
|
|
65
|
+
if (options.debug) {
|
|
66
|
+
logger.debug(`[ORCHESTRATOR] Effective target: "${effectiveTarget}"`);
|
|
67
|
+
logger.debug(`[ORCHESTRATOR] Review type: "${options.type}"`);
|
|
68
|
+
logger.debug(`Review options: ${JSON.stringify(options, null, 2)}`);
|
|
69
|
+
logger.debug(
|
|
70
|
+
`Target path: ${effectiveTarget}${!target || target.trim() === '' ? ' (defaulted to ".")' : ''}`,
|
|
71
|
+
);
|
|
72
|
+
logger.debug(`Selected model: ${process.env.AI_CODE_REVIEW_MODEL || 'not set'}`);
|
|
73
|
+
logger.debug(`API key type: ${getApiKeyType() || 'None'}`);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Test API connections if requested (removed - no longer supported)
|
|
77
|
+
if (options.testApi) {
|
|
78
|
+
logger.warn('API connection tests are not available in this build.');
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// Log the review type
|
|
82
|
+
if (options.type === 'architectural') {
|
|
83
|
+
logger.info(`Starting architectural review for ${effectiveTarget}...`);
|
|
84
|
+
} else if (options.type === 'coding-test') {
|
|
85
|
+
logger.info(`Starting coding test evaluation for ${effectiveTarget}...`);
|
|
86
|
+
} else if (options.type === 'extract-patterns') {
|
|
87
|
+
logger.info(`Starting pattern extraction for ${effectiveTarget}...`);
|
|
88
|
+
} else if (options.type === 'unused-code') {
|
|
89
|
+
logger.info(`Starting unused code review for ${effectiveTarget}...`);
|
|
90
|
+
} else {
|
|
91
|
+
logger.info(`Starting ${options.type} review for ${effectiveTarget}...`);
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// Determine the project path
|
|
95
|
+
const projectPath = process.cwd();
|
|
96
|
+
const projectName = path.basename(projectPath);
|
|
97
|
+
|
|
98
|
+
// Create output directory using the centralized function
|
|
99
|
+
const configOutputDir = configurationService.getPathsConfig().outputDir;
|
|
100
|
+
const outputBaseDir = createOutputDirectory(projectPath, {
|
|
101
|
+
outputDir: options.outputDir,
|
|
102
|
+
configOutputDir: configOutputDir,
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
// Create the directory
|
|
106
|
+
await createDirectory(outputBaseDir);
|
|
107
|
+
|
|
108
|
+
// Log project information
|
|
109
|
+
logger.info(`Project: ${projectName}`);
|
|
110
|
+
logger.info(`Project path: ${projectPath}`);
|
|
111
|
+
|
|
112
|
+
// Detect language and framework
|
|
113
|
+
let frameworkDetectionResult = null;
|
|
114
|
+
if (!options.language) {
|
|
115
|
+
try {
|
|
116
|
+
const { detectFramework } = await import('../utils/detection');
|
|
117
|
+
frameworkDetectionResult = await detectFramework(projectPath);
|
|
118
|
+
|
|
119
|
+
if (frameworkDetectionResult) {
|
|
120
|
+
options.language = frameworkDetectionResult.language as ProgrammingLanguage;
|
|
121
|
+
options.framework = frameworkDetectionResult.framework;
|
|
122
|
+
|
|
123
|
+
if (
|
|
124
|
+
frameworkDetectionResult.framework !== 'none' &&
|
|
125
|
+
frameworkDetectionResult.confidence > 0.6
|
|
126
|
+
) {
|
|
127
|
+
logger.info(
|
|
128
|
+
`Detected language: ${frameworkDetectionResult.language}, framework: ${frameworkDetectionResult.framework} (confidence: ${frameworkDetectionResult.confidence.toFixed(2)})`,
|
|
129
|
+
);
|
|
130
|
+
|
|
131
|
+
if (frameworkDetectionResult.frameworkVersion) {
|
|
132
|
+
logger.info(`Framework version: ${frameworkDetectionResult.frameworkVersion}`);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
if (
|
|
136
|
+
frameworkDetectionResult.additionalFrameworks &&
|
|
137
|
+
frameworkDetectionResult.additionalFrameworks.length > 0
|
|
138
|
+
) {
|
|
139
|
+
logger.info(
|
|
140
|
+
`Additional frameworks detected: ${frameworkDetectionResult.additionalFrameworks.join(', ')}`,
|
|
141
|
+
);
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
if (
|
|
145
|
+
frameworkDetectionResult.cssFrameworks &&
|
|
146
|
+
frameworkDetectionResult.cssFrameworks.length > 0
|
|
147
|
+
) {
|
|
148
|
+
const cssFrameworksStr = frameworkDetectionResult.cssFrameworks
|
|
149
|
+
.map((cf) => (cf.version ? `${cf.name} (${cf.version})` : cf.name))
|
|
150
|
+
.join(', ');
|
|
151
|
+
logger.info(`CSS frameworks detected: ${cssFrameworksStr}`);
|
|
152
|
+
}
|
|
153
|
+
} else {
|
|
154
|
+
logger.info(
|
|
155
|
+
`Detected language: ${frameworkDetectionResult.language}, no specific framework detected`,
|
|
156
|
+
);
|
|
157
|
+
|
|
158
|
+
// Still log CSS frameworks if detected
|
|
159
|
+
if (
|
|
160
|
+
frameworkDetectionResult.cssFrameworks &&
|
|
161
|
+
frameworkDetectionResult.cssFrameworks.length > 0
|
|
162
|
+
) {
|
|
163
|
+
const cssFrameworksStr = frameworkDetectionResult.cssFrameworks
|
|
164
|
+
.map((cf) => (cf.version ? `${cf.name} (${cf.version})` : cf.name))
|
|
165
|
+
.join(', ');
|
|
166
|
+
logger.info(`CSS frameworks detected: ${cssFrameworksStr}`);
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
} catch (error) {
|
|
171
|
+
logger.debug(
|
|
172
|
+
`Error detecting language/framework: ${error instanceof Error ? error.message : String(error)}`,
|
|
173
|
+
);
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// Discover files to review
|
|
178
|
+
const filesToReview = await discoverFilesForReview(effectiveTarget, projectPath, options);
|
|
179
|
+
|
|
180
|
+
if (filesToReview.length === 0) {
|
|
181
|
+
return; // No files to review, exit early
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
// If estimate flag is set, calculate and display token usage and cost estimates
|
|
185
|
+
if (options.estimate) {
|
|
186
|
+
// Get the model name from options or environment variables
|
|
187
|
+
const modelName =
|
|
188
|
+
options.model || process.env.AI_CODE_REVIEW_MODEL || 'openrouter:anthropic/claude-3-opus-20240229';
|
|
189
|
+
|
|
190
|
+
try {
|
|
191
|
+
// Read file contents for token analysis
|
|
192
|
+
const { fileInfos, errors } = await readFilesForReview(filesToReview, projectPath);
|
|
193
|
+
|
|
194
|
+
// If we have errors reading files, report them but continue
|
|
195
|
+
if (errors.length > 0) {
|
|
196
|
+
console.warn(`Warning: Failed to read ${errors.length} file(s):`);
|
|
197
|
+
for (const error of errors) {
|
|
198
|
+
console.warn(` - ${error.path}: ${error.error}`);
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
// Ensure we have at least some files to analyze
|
|
203
|
+
if (fileInfos.length === 0) {
|
|
204
|
+
throw new Error(
|
|
205
|
+
'No files could be read for review. Please check file permissions and paths.',
|
|
206
|
+
);
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
// Perform estimation
|
|
210
|
+
await performEstimation(fileInfos, filesToReview, options, modelName);
|
|
211
|
+
} catch (error) {
|
|
212
|
+
logger.error(
|
|
213
|
+
`Estimation failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
214
|
+
);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
return; // Exit after displaying the estimation
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
// Read file contents
|
|
221
|
+
const { fileInfos } = await readFilesForReview(filesToReview, projectPath);
|
|
222
|
+
|
|
223
|
+
// Read project documentation if enabled
|
|
224
|
+
let projectDocs = null;
|
|
225
|
+
if (options.includeProjectDocs) {
|
|
226
|
+
logger.info('Reading project documentation...');
|
|
227
|
+
projectDocs = await readProjectDocs(projectPath);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
// Get the API client configuration
|
|
231
|
+
const apiClientConfig = await selectApiClient(options);
|
|
232
|
+
|
|
233
|
+
// Log writer model if configured
|
|
234
|
+
const config = getConfig(options as any);
|
|
235
|
+
logger.debug(`Config writerModel: ${config.writerModel}`);
|
|
236
|
+
if (config.writerModel) {
|
|
237
|
+
logger.info(`Using writer model for consolidation: ${config.writerModel}`);
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
// Perform token analysis to check if content exceeds context window
|
|
241
|
+
let tokenAnalysis = null;
|
|
242
|
+
|
|
243
|
+
if (!options.multiPass) {
|
|
244
|
+
try {
|
|
245
|
+
logger.info('Analyzing token usage to determine review strategy...');
|
|
246
|
+
|
|
247
|
+
// Use the new TokenAnalyzer for more comprehensive analysis
|
|
248
|
+
const { TokenAnalyzer } = await import('../analysis/tokens');
|
|
249
|
+
|
|
250
|
+
const tokenAnalysisOptions = {
|
|
251
|
+
reviewType: options.type,
|
|
252
|
+
modelName: apiClientConfig.modelName,
|
|
253
|
+
contextMaintenanceFactor: options.contextMaintenanceFactor || 0.15,
|
|
254
|
+
forceSinglePass: options.forceSinglePass,
|
|
255
|
+
batchTokenLimit: options.batchTokenLimit,
|
|
256
|
+
};
|
|
257
|
+
|
|
258
|
+
// Log if forceSinglePass is enabled
|
|
259
|
+
if (options.forceSinglePass) {
|
|
260
|
+
logger.info(
|
|
261
|
+
'Force single-pass mode is enabled. This will override the chunking recommendation.',
|
|
262
|
+
);
|
|
263
|
+
logger.info(
|
|
264
|
+
"Note: This may result in token limit errors if the content exceeds the model's context window.",
|
|
265
|
+
);
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
tokenAnalysis = TokenAnalyzer.analyzeFiles(fileInfos, tokenAnalysisOptions);
|
|
269
|
+
|
|
270
|
+
} catch (error) {
|
|
271
|
+
logger.warn(
|
|
272
|
+
`Token analysis failed: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
273
|
+
);
|
|
274
|
+
logger.info('Proceeding with review without token analysis');
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
// Execute the review
|
|
279
|
+
const reviewResult = await executeReview(
|
|
280
|
+
fileInfos,
|
|
281
|
+
options,
|
|
282
|
+
apiClientConfig as any,
|
|
283
|
+
projectDocs,
|
|
284
|
+
tokenAnalysis,
|
|
285
|
+
);
|
|
286
|
+
|
|
287
|
+
// Handle review output
|
|
288
|
+
await handleReviewOutput(reviewResult, options, outputBaseDir);
|
|
289
|
+
} catch (error) {
|
|
290
|
+
// Import TokenLimitError for specific handling
|
|
291
|
+
const { TokenLimitError } = await import('../utils/apiErrorHandler');
|
|
292
|
+
|
|
293
|
+
// Handle token limit errors with helpful guidance
|
|
294
|
+
if (error instanceof TokenLimitError) {
|
|
295
|
+
logger.error('');
|
|
296
|
+
logger.error('==========================================');
|
|
297
|
+
logger.error('TOKEN LIMIT EXCEEDED');
|
|
298
|
+
logger.error('==========================================');
|
|
299
|
+
logger.error(`The codebase is too large for single-pass review.`);
|
|
300
|
+
if (error.tokenCount) {
|
|
301
|
+
logger.error(`Content size: ${error.tokenCount.toLocaleString()} tokens`);
|
|
302
|
+
}
|
|
303
|
+
logger.error('');
|
|
304
|
+
logger.error(
|
|
305
|
+
'SOLUTION: Use the --multi-pass flag to automatically split the review into multiple passes:',
|
|
306
|
+
);
|
|
307
|
+
logger.error('');
|
|
308
|
+
logger.error(' cr-aia --multi-pass');
|
|
309
|
+
logger.error('');
|
|
310
|
+
logger.error(
|
|
311
|
+
'This will intelligently chunk your codebase and maintain context between passes.',
|
|
312
|
+
);
|
|
313
|
+
logger.error('==========================================');
|
|
314
|
+
logger.error('');
|
|
315
|
+
|
|
316
|
+
if (options.debug && error.stack) {
|
|
317
|
+
logger.debug(`Error stack trace: ${error.stack}`);
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
// Re-throw with a cleaner message
|
|
321
|
+
throw new Error('Token limit exceeded. Please use --multi-pass flag for large codebases.');
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
// Handle any other errors
|
|
325
|
+
logger.error(`Review failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
326
|
+
|
|
327
|
+
if (error instanceof Error && error.stack && options.debug) {
|
|
328
|
+
logger.debug(`Error stack trace: ${error.stack}`);
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
throw error;
|
|
332
|
+
}
|
|
333
|
+
}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Utilities for handling model information and display
|
|
3
|
+
*
|
|
4
|
+
* This module provides helper functions for parsing and displaying
|
|
5
|
+
* provider and model information in a user-friendly format.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Parse and display provider and model information
|
|
10
|
+
*
|
|
11
|
+
* @param modelName The full model name (e.g., 'openai:gpt-4.1')
|
|
12
|
+
* @returns An object with provider and model display information
|
|
13
|
+
*/
|
|
14
|
+
export function getProviderDisplayInfo(modelName: string): { provider: string; model: string } {
|
|
15
|
+
const getFallbackInfo = () => {
|
|
16
|
+
const parts = modelName.split(':');
|
|
17
|
+
|
|
18
|
+
if (parts.length === 2 && parts[1]) {
|
|
19
|
+
const providerPart = parts[0].toLowerCase();
|
|
20
|
+
return {
|
|
21
|
+
provider: providerPart.charAt(0).toUpperCase() + providerPart.slice(1), // Capitalize provider name
|
|
22
|
+
model: parts[1],
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
return {
|
|
27
|
+
provider: 'Unknown',
|
|
28
|
+
model: modelName,
|
|
29
|
+
};
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
// If the model name doesn't contain a colon, it's not in the expected format
|
|
33
|
+
if (!modelName.includes(':')) {
|
|
34
|
+
return {
|
|
35
|
+
provider: 'Unknown',
|
|
36
|
+
model: modelName,
|
|
37
|
+
};
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
try {
|
|
41
|
+
const [providerPart, modelPart] = modelName.split(':', 2);
|
|
42
|
+
if (modelPart) {
|
|
43
|
+
const provider =
|
|
44
|
+
providerPart.charAt(0).toUpperCase() + providerPart.slice(1).toLowerCase();
|
|
45
|
+
return {
|
|
46
|
+
provider,
|
|
47
|
+
model: modelPart,
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
} catch (_error) {
|
|
51
|
+
// If parsing fails, use a fallback approach
|
|
52
|
+
return getFallbackInfo();
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
return getFallbackInfo();
|
|
56
|
+
}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Formatter for code review output in different formats.
|
|
3
|
+
*
|
|
4
|
+
* This module provides formatting utilities for code review results, supporting
|
|
5
|
+
* multiple output formats including Markdown and JSON. It handles the transformation
|
|
6
|
+
* of raw review data into well-structured, readable formats suitable for different
|
|
7
|
+
* consumption patterns.
|
|
8
|
+
*
|
|
9
|
+
* Key responsibilities:
|
|
10
|
+
* - Converting review results to Markdown format with proper headings and sections
|
|
11
|
+
* - Converting review results to JSON format for programmatic consumption
|
|
12
|
+
* - Sanitizing content to prevent rendering issues
|
|
13
|
+
* - Adding metadata like review date, model used, and cost information
|
|
14
|
+
* - Formatting code snippets and recommendations consistently
|
|
15
|
+
*
|
|
16
|
+
* The formatter ensures that review outputs are consistent, readable, and properly
|
|
17
|
+
* structured regardless of the review type or content.
|
|
18
|
+
*/
|
|
19
|
+
|
|
20
|
+
import type { ReviewResult } from '../types/review';
|
|
21
|
+
import { formatAsJson } from './utils/JsonFormatter';
|
|
22
|
+
import { formatAsMarkdown } from './utils/MarkdownFormatters';
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Format the review output based on the specified format
|
|
26
|
+
* @param review Review result to format
|
|
27
|
+
* @param format Output format (markdown or json)
|
|
28
|
+
* @returns Formatted review output
|
|
29
|
+
*/
|
|
30
|
+
export function formatReviewOutput(review: ReviewResult, format: string): string {
|
|
31
|
+
// Debug logging to help diagnose issues with missing fields
|
|
32
|
+
if (!review.filePath) {
|
|
33
|
+
console.warn('Warning: filePath is undefined or empty in ReviewResult');
|
|
34
|
+
}
|
|
35
|
+
if (!review.modelUsed) {
|
|
36
|
+
console.warn('Warning: modelUsed is undefined or empty in ReviewResult');
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
// Ensure costInfo is set if only cost is available
|
|
40
|
+
if (review.cost && !review.costInfo) {
|
|
41
|
+
review.costInfo = review.cost;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
if (format === 'json') {
|
|
45
|
+
return formatAsJson(review);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
return formatAsMarkdown(review);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export { formatIssue, formatSchemaIssue } from './utils/IssueFormatters';
|
|
52
|
+
export { formatAsJson } from './utils/JsonFormatter';
|
|
53
|
+
// Re-export utility functions for use by other modules
|
|
54
|
+
export { formatAsMarkdown } from './utils/MarkdownFormatters';
|
|
55
|
+
export {
|
|
56
|
+
createEnhancedMetadata,
|
|
57
|
+
formatCostInfo,
|
|
58
|
+
formatMetadataSection,
|
|
59
|
+
parseCostInfo,
|
|
60
|
+
parseMetadata,
|
|
61
|
+
} from './utils/MetadataFormatter';
|
|
62
|
+
export { extractModelInfo, extractModelInfoFromString } from './utils/ModelInfoExtractor';
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Formatters for review issues in different formats.
|
|
3
|
+
*
|
|
4
|
+
* This module provides functions to format review issues in different formats,
|
|
5
|
+
* including structured issues and schema-based issues.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { ReviewIssue } from '../../types/structuredReview';
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Format a single issue from the schema format
|
|
12
|
+
* @param issue Issue object from schema
|
|
13
|
+
* @param index Issue number
|
|
14
|
+
* @returns Formatted issue string
|
|
15
|
+
*/
|
|
16
|
+
export function formatSchemaIssue(issue: any, index: number): string {
|
|
17
|
+
let issueMarkdown = `### ${index}. ${issue.description}\n\n`;
|
|
18
|
+
|
|
19
|
+
if (issue.filePath) {
|
|
20
|
+
issueMarkdown += `**File**: \`${issue.filePath}\`\n`;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
if (issue.location) {
|
|
24
|
+
issueMarkdown += `**Location**: Lines ${issue.location.startLine}-${issue.location.endLine}\n\n`;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
if (issue.currentCode) {
|
|
28
|
+
issueMarkdown += `**Current Code**:\n\`\`\`\n${issue.currentCode}\n\`\`\`\n\n`;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
if (issue.suggestedCode) {
|
|
32
|
+
issueMarkdown += `**Suggested Fix**:\n\`\`\`\n${issue.suggestedCode}\n\`\`\`\n\n`;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if (issue.explanation) {
|
|
36
|
+
issueMarkdown += `**Explanation**: ${issue.explanation}\n\n`;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
issueMarkdown += `---\n\n`;
|
|
40
|
+
|
|
41
|
+
return issueMarkdown;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Format a single issue as Markdown
|
|
46
|
+
* @param issue Review issue
|
|
47
|
+
* @returns Markdown string
|
|
48
|
+
*/
|
|
49
|
+
export function formatIssue(issue: ReviewIssue): string {
|
|
50
|
+
// Guard against null or undefined issues
|
|
51
|
+
if (!issue) {
|
|
52
|
+
return '#### [Error: Issue data missing]';
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const { title, type, filePath, lineNumbers, description, codeSnippet, suggestedFix, impact } =
|
|
56
|
+
issue;
|
|
57
|
+
|
|
58
|
+
let issueMarkdown = `#### ${title || '[Untitled Issue]'}\n`;
|
|
59
|
+
|
|
60
|
+
if (filePath) {
|
|
61
|
+
issueMarkdown += `- **Location**: \`${filePath}${lineNumbers ? `:${lineNumbers}` : ''}\`\n`;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
if (type) {
|
|
65
|
+
issueMarkdown += `- **Type**: ${type}\n`;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
issueMarkdown += `- **Description**: ${description || 'No description provided'}\n`;
|
|
69
|
+
|
|
70
|
+
if (codeSnippet) {
|
|
71
|
+
issueMarkdown += `- **Code**:\n\`\`\`\n${codeSnippet}\n\`\`\`\n`;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
if (suggestedFix) {
|
|
75
|
+
issueMarkdown += `- **Suggested Fix**:\n\`\`\`\n${suggestedFix}\n\`\`\`\n`;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
if (impact) {
|
|
79
|
+
issueMarkdown += `- **Impact**: ${impact}\n`;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
return issueMarkdown;
|
|
83
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Formatter for JSON output of code reviews.
|
|
3
|
+
*
|
|
4
|
+
* This module provides functions to format code review results as JSON,
|
|
5
|
+
* including metadata, structured data, and content.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { ReviewResult } from '../../types/review';
|
|
9
|
+
import { sanitizeContent } from '../../utils/sanitizer';
|
|
10
|
+
import { createEnhancedMetadata, parseMetadata } from './MetadataFormatter';
|
|
11
|
+
import { extractModelInfo } from './ModelInfoExtractor';
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Format the review as JSON
|
|
15
|
+
* @param review Review result to format
|
|
16
|
+
* @returns JSON string
|
|
17
|
+
*/
|
|
18
|
+
export function formatAsJson(review: ReviewResult): string {
|
|
19
|
+
// Extract model information
|
|
20
|
+
const { modelVendor, modelName, modelInfo } = extractModelInfo(review.modelUsed);
|
|
21
|
+
|
|
22
|
+
// Sanitize the content to prevent XSS attacks
|
|
23
|
+
const sanitizedContent = sanitizeContent(review.content);
|
|
24
|
+
|
|
25
|
+
// Parse structured data if available
|
|
26
|
+
let parsedStructuredData = review.structuredData;
|
|
27
|
+
if (typeof review.structuredData === 'string') {
|
|
28
|
+
try {
|
|
29
|
+
parsedStructuredData = JSON.parse(review.structuredData);
|
|
30
|
+
} catch (error) {
|
|
31
|
+
console.error('Error parsing structured review data:', error);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// Parse additional metadata if available
|
|
36
|
+
const additionalMetadata = parseMetadata(review.metadata);
|
|
37
|
+
|
|
38
|
+
// Format path for display
|
|
39
|
+
let displayPath = review.filePath || '';
|
|
40
|
+
if (!displayPath || displayPath === review.reviewType || displayPath === 'consolidated') {
|
|
41
|
+
displayPath = `${process.cwd()} (Current Directory)`;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// Create enhanced metadata with detection info
|
|
45
|
+
const enhancedMetadata = createEnhancedMetadata(
|
|
46
|
+
modelVendor,
|
|
47
|
+
modelName,
|
|
48
|
+
modelInfo,
|
|
49
|
+
review.reviewType,
|
|
50
|
+
displayPath,
|
|
51
|
+
review.timestamp,
|
|
52
|
+
review.costInfo || review.cost,
|
|
53
|
+
review.toolVersion,
|
|
54
|
+
review.commandOptions,
|
|
55
|
+
additionalMetadata,
|
|
56
|
+
review.detectedLanguage,
|
|
57
|
+
review.detectedFramework,
|
|
58
|
+
review.frameworkVersion,
|
|
59
|
+
review.cssFrameworks,
|
|
60
|
+
);
|
|
61
|
+
|
|
62
|
+
// Create a copy of the review with enhanced metadata
|
|
63
|
+
const reviewWithMeta = {
|
|
64
|
+
...review,
|
|
65
|
+
content: sanitizedContent,
|
|
66
|
+
structuredData: parsedStructuredData,
|
|
67
|
+
meta: enhancedMetadata,
|
|
68
|
+
// Legacy metadata field for backward compatibility
|
|
69
|
+
metadata: {
|
|
70
|
+
model: modelInfo,
|
|
71
|
+
generatedAt: new Date(review.timestamp).toISOString(),
|
|
72
|
+
costEstimation: review.cost,
|
|
73
|
+
},
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
return JSON.stringify(reviewWithMeta, null, 2);
|
|
77
|
+
}
|