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,622 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Documentation analyzer for detecting AI-generated content patterns.
|
|
3
|
+
*
|
|
4
|
+
* This analyzer examines README files, code comments, and documentation structure
|
|
5
|
+
* to identify characteristics commonly associated with AI-generated documentation.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type {
|
|
9
|
+
CodeFile,
|
|
10
|
+
CodeSubmission,
|
|
11
|
+
DetectedPattern,
|
|
12
|
+
DocumentationResult,
|
|
13
|
+
PatternDetectionResult,
|
|
14
|
+
} from '../types/DetectionTypes';
|
|
15
|
+
import { BaseAnalyzer } from './BaseAnalyzer';
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Analyzer for documentation patterns that may indicate AI generation
|
|
19
|
+
*/
|
|
20
|
+
export class DocumentationAnalyzer extends BaseAnalyzer {
|
|
21
|
+
/**
|
|
22
|
+
* Get analyzer name
|
|
23
|
+
*/
|
|
24
|
+
getAnalyzerName(): string {
|
|
25
|
+
return 'documentation';
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Analyze documentation for AI-generated patterns
|
|
30
|
+
* @param submission Code submission to analyze
|
|
31
|
+
* @returns Documentation analysis result
|
|
32
|
+
*/
|
|
33
|
+
async analyze(submission: CodeSubmission): Promise<DocumentationResult> {
|
|
34
|
+
this.startTimer();
|
|
35
|
+
const patterns: DetectedPattern[] = [];
|
|
36
|
+
|
|
37
|
+
try {
|
|
38
|
+
const { documentation } = submission;
|
|
39
|
+
|
|
40
|
+
// Pattern H2.1: Template README Structure
|
|
41
|
+
if (documentation.readme) {
|
|
42
|
+
const readmeResult = this.analyzeREADMEStructure(documentation.readme);
|
|
43
|
+
if (readmeResult.detected) {
|
|
44
|
+
patterns.push(
|
|
45
|
+
this.createDetectedPattern(
|
|
46
|
+
'H2.1',
|
|
47
|
+
'Template README Structure',
|
|
48
|
+
'high',
|
|
49
|
+
readmeResult.score,
|
|
50
|
+
'README follows AI-generated template structure with standard sections',
|
|
51
|
+
readmeResult.evidence || {},
|
|
52
|
+
),
|
|
53
|
+
);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// Pattern H2.2: Excessive Comment Density
|
|
58
|
+
const commentResult = this.analyzeCommentDensity(documentation.codeFiles);
|
|
59
|
+
if (commentResult.detected) {
|
|
60
|
+
patterns.push(
|
|
61
|
+
this.createDetectedPattern(
|
|
62
|
+
'H2.2',
|
|
63
|
+
'Excessive Comment Density',
|
|
64
|
+
'high',
|
|
65
|
+
commentResult.score,
|
|
66
|
+
'Unusually high and uniform comment density across files',
|
|
67
|
+
commentResult.evidence || {},
|
|
68
|
+
),
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Pattern H2.3: AI-Style Documentation
|
|
73
|
+
if (documentation.readme) {
|
|
74
|
+
const aiStyleResult = this.detectAIStyleDocumentation(documentation.readme);
|
|
75
|
+
if (aiStyleResult.detected) {
|
|
76
|
+
patterns.push(
|
|
77
|
+
this.createDetectedPattern(
|
|
78
|
+
'H2.3',
|
|
79
|
+
'AI-Style Documentation',
|
|
80
|
+
'high',
|
|
81
|
+
aiStyleResult.score,
|
|
82
|
+
'Documentation exhibits characteristics typical of AI-generated content',
|
|
83
|
+
aiStyleResult.evidence || {},
|
|
84
|
+
),
|
|
85
|
+
);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Pattern M2.4: Uniform Comment Patterns
|
|
90
|
+
const uniformCommentResult = this.analyzeUniformCommentPatterns(documentation.codeFiles);
|
|
91
|
+
if (uniformCommentResult.detected) {
|
|
92
|
+
patterns.push(
|
|
93
|
+
this.createDetectedPattern(
|
|
94
|
+
'M2.4',
|
|
95
|
+
'Uniform Comment Patterns',
|
|
96
|
+
'medium',
|
|
97
|
+
uniformCommentResult.score,
|
|
98
|
+
'Comments follow suspiciously uniform patterns across files',
|
|
99
|
+
uniformCommentResult.evidence || {},
|
|
100
|
+
),
|
|
101
|
+
);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
const avgCommentDensity = this.calculateAverageCommentDensity(documentation.codeFiles);
|
|
105
|
+
|
|
106
|
+
return {
|
|
107
|
+
analyzer: 'documentation',
|
|
108
|
+
patterns,
|
|
109
|
+
metadata: {
|
|
110
|
+
filesAnalyzed: documentation.codeFiles.length,
|
|
111
|
+
hasReadme: !!documentation.readme,
|
|
112
|
+
avgCommentDensity,
|
|
113
|
+
},
|
|
114
|
+
};
|
|
115
|
+
} catch (error) {
|
|
116
|
+
console.error('Error in DocumentationAnalyzer:', error);
|
|
117
|
+
return this.createEmptyResult();
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Analyze README structure for template patterns
|
|
123
|
+
* @param readme README content
|
|
124
|
+
* @returns Pattern detection result
|
|
125
|
+
*/
|
|
126
|
+
private analyzeREADMEStructure(readme: string): PatternDetectionResult {
|
|
127
|
+
const standardSections = [
|
|
128
|
+
'installation',
|
|
129
|
+
'usage',
|
|
130
|
+
'api',
|
|
131
|
+
'contributing',
|
|
132
|
+
'license',
|
|
133
|
+
'features',
|
|
134
|
+
'requirements',
|
|
135
|
+
'examples',
|
|
136
|
+
'documentation',
|
|
137
|
+
'getting started',
|
|
138
|
+
'quick start',
|
|
139
|
+
'configuration',
|
|
140
|
+
'support',
|
|
141
|
+
];
|
|
142
|
+
|
|
143
|
+
const sections = this.extractSections(readme);
|
|
144
|
+
const sectionLower = sections.map((s) => s.toLowerCase());
|
|
145
|
+
|
|
146
|
+
const matchedStandard = sectionLower.filter((section) =>
|
|
147
|
+
standardSections.some((std) => section.includes(std) || std.includes(section)),
|
|
148
|
+
);
|
|
149
|
+
|
|
150
|
+
const completeness = matchedStandard.length / standardSections.length;
|
|
151
|
+
|
|
152
|
+
// High completeness (>60%) suggests template usage
|
|
153
|
+
if (completeness > 0.6 && sections.length >= 5) {
|
|
154
|
+
const badges = this.countBadges(readme);
|
|
155
|
+
const genericPhrases = this.countGenericPhrases(readme);
|
|
156
|
+
const templateIndicators = this.countTemplateIndicators(readme);
|
|
157
|
+
|
|
158
|
+
const score = Math.min(
|
|
159
|
+
0.95,
|
|
160
|
+
0.5 +
|
|
161
|
+
completeness * 0.25 +
|
|
162
|
+
Math.min(badges, 5) * 0.03 +
|
|
163
|
+
Math.min(genericPhrases, 10) * 0.02 +
|
|
164
|
+
templateIndicators * 0.05,
|
|
165
|
+
);
|
|
166
|
+
|
|
167
|
+
return this.createPatternResult(true, score, {
|
|
168
|
+
completeness,
|
|
169
|
+
matchedSections: matchedStandard,
|
|
170
|
+
totalSections: sections.length,
|
|
171
|
+
badgeCount: badges,
|
|
172
|
+
genericPhraseCount: genericPhrases,
|
|
173
|
+
templateIndicatorCount: templateIndicators,
|
|
174
|
+
sectionOrder: this.analyzeSectionOrder(sections),
|
|
175
|
+
standardSections,
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
return this.createPatternResult(false, completeness * 0.3);
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
/**
|
|
183
|
+
* Analyze comment density across code files
|
|
184
|
+
* @param codeFiles Array of code files
|
|
185
|
+
* @returns Pattern detection result
|
|
186
|
+
*/
|
|
187
|
+
private analyzeCommentDensity(codeFiles: CodeFile[]): PatternDetectionResult {
|
|
188
|
+
if (codeFiles.length === 0) {
|
|
189
|
+
return this.createPatternResult(false, 0);
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
const densities = codeFiles.map((file) => this.calculateCommentDensity(file.content));
|
|
193
|
+
const averageDensity = densities.reduce((a, b) => a + b, 0) / densities.length;
|
|
194
|
+
const uniformity = this.calculateUniformity(densities);
|
|
195
|
+
|
|
196
|
+
// High density (>40%) with high uniformity (>0.8) suggests AI generation
|
|
197
|
+
if (averageDensity > 0.4 && uniformity > 0.8 && codeFiles.length >= 3) {
|
|
198
|
+
const score = Math.min(0.95, 0.5 + averageDensity * 0.3 + uniformity * 0.2);
|
|
199
|
+
|
|
200
|
+
return this.createPatternResult(true, score, {
|
|
201
|
+
averageDensity,
|
|
202
|
+
uniformity,
|
|
203
|
+
fileCount: codeFiles.length,
|
|
204
|
+
densityDistribution: densities,
|
|
205
|
+
threshold: { density: 0.4, uniformity: 0.8 },
|
|
206
|
+
filesWithHighDensity: densities.filter((d) => d > 0.5).length,
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
return this.createPatternResult(false, Math.max(averageDensity - 0.2, 0) * 0.5);
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
/**
|
|
214
|
+
* Detect AI-style documentation patterns
|
|
215
|
+
* @param readme README content
|
|
216
|
+
* @returns Pattern detection result
|
|
217
|
+
*/
|
|
218
|
+
private detectAIStyleDocumentation(readme: string): PatternDetectionResult {
|
|
219
|
+
const aiIndicators = {
|
|
220
|
+
overlyFormal: this.countOverlyFormalLanguage(readme),
|
|
221
|
+
perfectGrammar: this.checkPerfectGrammar(readme),
|
|
222
|
+
templatePhrases: this.countTemplatePhrases(readme),
|
|
223
|
+
comprehensiveStructure: this.checkComprehensiveStructure(readme),
|
|
224
|
+
technicalPrecision: this.checkTechnicalPrecision(readme),
|
|
225
|
+
lackOfPersonality: this.checkLackOfPersonality(readme),
|
|
226
|
+
};
|
|
227
|
+
|
|
228
|
+
const totalIndicators = Object.keys(aiIndicators).length;
|
|
229
|
+
const positiveIndicators = Object.values(aiIndicators).filter(Boolean).length;
|
|
230
|
+
const ratio = positiveIndicators / totalIndicators;
|
|
231
|
+
|
|
232
|
+
// If most indicators are present, suggest AI generation
|
|
233
|
+
if (ratio > 0.6) {
|
|
234
|
+
const score = Math.min(0.9, 0.5 + ratio * 0.4);
|
|
235
|
+
|
|
236
|
+
return this.createPatternResult(true, score, {
|
|
237
|
+
indicators: aiIndicators,
|
|
238
|
+
positiveIndicators,
|
|
239
|
+
totalIndicators,
|
|
240
|
+
ratio,
|
|
241
|
+
readmeLength: readme.length,
|
|
242
|
+
wordCount: readme.split(/\s+/).length,
|
|
243
|
+
});
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
return this.createPatternResult(false, ratio * 0.4);
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
/**
|
|
250
|
+
* Analyze uniform comment patterns across files
|
|
251
|
+
* @param codeFiles Array of code files
|
|
252
|
+
* @returns Pattern detection result
|
|
253
|
+
*/
|
|
254
|
+
private analyzeUniformCommentPatterns(codeFiles: CodeFile[]): PatternDetectionResult {
|
|
255
|
+
if (codeFiles.length < 3) {
|
|
256
|
+
return this.createPatternResult(false, 0);
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
const commentPatterns = codeFiles.map((file) => this.extractCommentPatterns(file.content));
|
|
260
|
+
const uniformityScore = this.calculateCommentPatternUniformity(commentPatterns);
|
|
261
|
+
|
|
262
|
+
// High uniformity suggests AI-generated comments
|
|
263
|
+
if (uniformityScore > 0.8) {
|
|
264
|
+
const score = Math.min(0.85, 0.4 + uniformityScore * 0.4);
|
|
265
|
+
|
|
266
|
+
return this.createPatternResult(true, score, {
|
|
267
|
+
uniformityScore,
|
|
268
|
+
fileCount: codeFiles.length,
|
|
269
|
+
commonPatterns: this.findCommonCommentPatterns(commentPatterns),
|
|
270
|
+
threshold: 0.8,
|
|
271
|
+
});
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
return this.createPatternResult(false, uniformityScore * 0.3);
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* Calculate comment density for a single file
|
|
279
|
+
* @param content File content
|
|
280
|
+
* @returns Comment density ratio (0.0 to 1.0)
|
|
281
|
+
*/
|
|
282
|
+
private calculateCommentDensity(content: string): number {
|
|
283
|
+
const lines = content.split('\n');
|
|
284
|
+
const codeLines = lines.filter((line) => {
|
|
285
|
+
const trimmed = line.trim();
|
|
286
|
+
return (
|
|
287
|
+
trimmed &&
|
|
288
|
+
!trimmed.startsWith('//') &&
|
|
289
|
+
!trimmed.startsWith('/*') &&
|
|
290
|
+
!trimmed.startsWith('*') &&
|
|
291
|
+
!trimmed.startsWith('#') && // Python/shell comments
|
|
292
|
+
!trimmed.match(/^\s*\/\*/) && // Multi-line comment start
|
|
293
|
+
!trimmed.match(/\*\/\s*$/)
|
|
294
|
+
); // Multi-line comment end
|
|
295
|
+
});
|
|
296
|
+
|
|
297
|
+
const commentLines = lines.length - codeLines.length;
|
|
298
|
+
return lines.length > 0 ? commentLines / lines.length : 0;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
/**
|
|
302
|
+
* Calculate average comment density across files
|
|
303
|
+
* @param codeFiles Array of code files
|
|
304
|
+
* @returns Average comment density
|
|
305
|
+
*/
|
|
306
|
+
private calculateAverageCommentDensity(codeFiles: CodeFile[]): number {
|
|
307
|
+
if (codeFiles.length === 0) return 0;
|
|
308
|
+
|
|
309
|
+
const densities = codeFiles.map((file) => this.calculateCommentDensity(file.content));
|
|
310
|
+
return densities.reduce((sum, density) => sum + density, 0) / densities.length;
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
/**
|
|
314
|
+
* Count template indicators in README
|
|
315
|
+
* @param readme README content
|
|
316
|
+
* @returns Number of template indicators
|
|
317
|
+
*/
|
|
318
|
+
private countTemplateIndicators(readme: string): number {
|
|
319
|
+
const templatePatterns = [
|
|
320
|
+
/\[!\[.*?\]\(.*?\)\]\(.*?\)/g, // Nested badge syntax
|
|
321
|
+
/## Table of Contents/gi,
|
|
322
|
+
/\*\*Note:\*\*/gi,
|
|
323
|
+
/Replace `.*?` with/gi,
|
|
324
|
+
/\$\{.*?\}/g, // Template variables
|
|
325
|
+
/\[\[.*?\]\]/g, // Wiki-style links
|
|
326
|
+
];
|
|
327
|
+
|
|
328
|
+
let count = 0;
|
|
329
|
+
for (const pattern of templatePatterns) {
|
|
330
|
+
const matches = readme.match(pattern);
|
|
331
|
+
if (matches) {
|
|
332
|
+
count += matches.length;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
return count;
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
/**
|
|
340
|
+
* Analyze section ordering for template patterns
|
|
341
|
+
* @param sections Array of section names
|
|
342
|
+
* @returns Section order analysis
|
|
343
|
+
*/
|
|
344
|
+
private analyzeSectionOrder(sections: string[]): any {
|
|
345
|
+
const standardOrder = [
|
|
346
|
+
'title',
|
|
347
|
+
'description',
|
|
348
|
+
'installation',
|
|
349
|
+
'usage',
|
|
350
|
+
'api',
|
|
351
|
+
'examples',
|
|
352
|
+
'contributing',
|
|
353
|
+
'license',
|
|
354
|
+
];
|
|
355
|
+
|
|
356
|
+
let orderScore = 0;
|
|
357
|
+
let lastIndex = -1;
|
|
358
|
+
|
|
359
|
+
sections.forEach((section) => {
|
|
360
|
+
const sectionLower = section.toLowerCase();
|
|
361
|
+
const standardIndex = standardOrder.findIndex(
|
|
362
|
+
(std) => sectionLower.includes(std) || std.includes(sectionLower),
|
|
363
|
+
);
|
|
364
|
+
|
|
365
|
+
if (standardIndex > lastIndex) {
|
|
366
|
+
orderScore++;
|
|
367
|
+
lastIndex = standardIndex;
|
|
368
|
+
}
|
|
369
|
+
});
|
|
370
|
+
|
|
371
|
+
return {
|
|
372
|
+
orderScore,
|
|
373
|
+
maxPossibleScore: Math.min(sections.length, standardOrder.length),
|
|
374
|
+
followsStandardOrder: orderScore / Math.min(sections.length, standardOrder.length) > 0.8,
|
|
375
|
+
};
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
/**
|
|
379
|
+
* Count overly formal language patterns
|
|
380
|
+
* @param content Text content
|
|
381
|
+
* @returns Count of formal language indicators
|
|
382
|
+
*/
|
|
383
|
+
private countOverlyFormalLanguage(content: string): number {
|
|
384
|
+
const formalPatterns = [
|
|
385
|
+
/furthermore/gi,
|
|
386
|
+
/moreover/gi,
|
|
387
|
+
/in addition/gi,
|
|
388
|
+
/consequently/gi,
|
|
389
|
+
/subsequently/gi,
|
|
390
|
+
/comprehensive/gi,
|
|
391
|
+
/sophisticated/gi,
|
|
392
|
+
/facilitate/gi,
|
|
393
|
+
/utilize/gi,
|
|
394
|
+
/implement.*functionality/gi,
|
|
395
|
+
];
|
|
396
|
+
|
|
397
|
+
let count = 0;
|
|
398
|
+
for (const pattern of formalPatterns) {
|
|
399
|
+
const matches = content.match(pattern);
|
|
400
|
+
if (matches) {
|
|
401
|
+
count += matches.length;
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
return count;
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
/**
|
|
409
|
+
* Check for perfect grammar indicators
|
|
410
|
+
* @param content Text content
|
|
411
|
+
* @returns True if grammar seems too perfect
|
|
412
|
+
*/
|
|
413
|
+
private checkPerfectGrammar(content: string): boolean {
|
|
414
|
+
// Simple heuristics for "too perfect" grammar
|
|
415
|
+
const sentences = content.split(/[.!?]+/).filter((s) => s.trim().length > 10);
|
|
416
|
+
if (sentences.length < 5) return false;
|
|
417
|
+
|
|
418
|
+
const indicators = {
|
|
419
|
+
noContractions: !content.match(/\b(don't|won't|can't|it's|you're|we're)\b/i),
|
|
420
|
+
consistentCapitalization: this.hasConsistentCapitalization(content),
|
|
421
|
+
noTypos: !content.match(/\b(teh|recieve|seperate|definately)\b/i),
|
|
422
|
+
perfectPunctuation: this.hasPerfectPunctuation(content),
|
|
423
|
+
};
|
|
424
|
+
|
|
425
|
+
const positiveCount = Object.values(indicators).filter(Boolean).length;
|
|
426
|
+
return positiveCount >= 3;
|
|
427
|
+
}
|
|
428
|
+
|
|
429
|
+
/**
|
|
430
|
+
* Count template phrases
|
|
431
|
+
* @param content Text content
|
|
432
|
+
* @returns Number of template phrases
|
|
433
|
+
*/
|
|
434
|
+
private countTemplatePhrases(content: string): number {
|
|
435
|
+
const templatePhrases = [
|
|
436
|
+
/this project provides/gi,
|
|
437
|
+
/easy to use/gi,
|
|
438
|
+
/getting started is simple/gi,
|
|
439
|
+
/follow these steps/gi,
|
|
440
|
+
/comprehensive solution/gi,
|
|
441
|
+
/powerful and flexible/gi,
|
|
442
|
+
/designed to be/gi,
|
|
443
|
+
/built with.*in mind/gi,
|
|
444
|
+
];
|
|
445
|
+
|
|
446
|
+
let count = 0;
|
|
447
|
+
for (const pattern of templatePhrases) {
|
|
448
|
+
const matches = content.match(pattern);
|
|
449
|
+
if (matches) {
|
|
450
|
+
count += matches.length;
|
|
451
|
+
}
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
return count;
|
|
455
|
+
}
|
|
456
|
+
|
|
457
|
+
/**
|
|
458
|
+
* Check for comprehensive structure
|
|
459
|
+
* @param content Text content
|
|
460
|
+
* @returns True if structure is suspiciously comprehensive
|
|
461
|
+
*/
|
|
462
|
+
private checkComprehensiveStructure(content: string): boolean {
|
|
463
|
+
const sections = this.extractSections(content);
|
|
464
|
+
return sections.length > 8 && content.length > 2000;
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
/**
|
|
468
|
+
* Check for technical precision
|
|
469
|
+
* @param content Text content
|
|
470
|
+
* @returns True if technically precise in AI-like way
|
|
471
|
+
*/
|
|
472
|
+
private checkTechnicalPrecision(content: string): boolean {
|
|
473
|
+
const technicalIndicators = [
|
|
474
|
+
/version \d+\.\d+\.\d+/gi,
|
|
475
|
+
/node\.js \d+\.\d+/gi,
|
|
476
|
+
/typescript \d+\.\d+/gi,
|
|
477
|
+
/npm install/gi,
|
|
478
|
+
/yarn add/gi,
|
|
479
|
+
/pnpm install/gi,
|
|
480
|
+
];
|
|
481
|
+
|
|
482
|
+
let count = 0;
|
|
483
|
+
for (const pattern of technicalIndicators) {
|
|
484
|
+
const matches = content.match(pattern);
|
|
485
|
+
if (matches) {
|
|
486
|
+
count += matches.length;
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
return count > 5;
|
|
491
|
+
}
|
|
492
|
+
|
|
493
|
+
/**
|
|
494
|
+
* Check for lack of personality
|
|
495
|
+
* @param content Text content
|
|
496
|
+
* @returns True if lacks personal touches
|
|
497
|
+
*/
|
|
498
|
+
private checkLackOfPersonality(content: string): boolean {
|
|
499
|
+
const personalityIndicators = [
|
|
500
|
+
/\bi\b/gi, // First person
|
|
501
|
+
/my/gi,
|
|
502
|
+
/personally/gi,
|
|
503
|
+
/in my opinion/gi,
|
|
504
|
+
/i think/gi,
|
|
505
|
+
/i believe/gi,
|
|
506
|
+
/hope/gi,
|
|
507
|
+
/feel free/gi,
|
|
508
|
+
];
|
|
509
|
+
|
|
510
|
+
let count = 0;
|
|
511
|
+
for (const pattern of personalityIndicators) {
|
|
512
|
+
const matches = content.match(pattern);
|
|
513
|
+
if (matches) {
|
|
514
|
+
count += matches.length;
|
|
515
|
+
}
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
return count < 3 && content.length > 1000;
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
/**
|
|
522
|
+
* Extract comment patterns from code
|
|
523
|
+
* @param content Code content
|
|
524
|
+
* @returns Comment pattern analysis
|
|
525
|
+
*/
|
|
526
|
+
private extractCommentPatterns(content: string): any {
|
|
527
|
+
const lines = content.split('\n');
|
|
528
|
+
const commentLines = lines.filter((line) => {
|
|
529
|
+
const trimmed = line.trim();
|
|
530
|
+
return trimmed.startsWith('//') || trimmed.startsWith('/*') || trimmed.startsWith('*');
|
|
531
|
+
});
|
|
532
|
+
|
|
533
|
+
return {
|
|
534
|
+
density: commentLines.length / lines.length,
|
|
535
|
+
avgLength:
|
|
536
|
+
commentLines.reduce((sum, line) => sum + line.length, 0) / (commentLines.length || 1),
|
|
537
|
+
patterns: commentLines.map((line) => line.trim().substring(0, 20)), // First 20 chars
|
|
538
|
+
};
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
/**
|
|
542
|
+
* Calculate uniformity of comment patterns across files
|
|
543
|
+
* @param patterns Array of comment patterns
|
|
544
|
+
* @returns Uniformity score
|
|
545
|
+
*/
|
|
546
|
+
private calculateCommentPatternUniformity(patterns: any[]): number {
|
|
547
|
+
if (patterns.length < 2) return 0;
|
|
548
|
+
|
|
549
|
+
const densities = patterns.map((p) => p.density);
|
|
550
|
+
const avgLengths = patterns.map((p) => p.avgLength);
|
|
551
|
+
|
|
552
|
+
const densityUniformity = this.calculateUniformity(densities);
|
|
553
|
+
const lengthUniformity = this.calculateUniformity(avgLengths);
|
|
554
|
+
|
|
555
|
+
return (densityUniformity + lengthUniformity) / 2;
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
/**
|
|
559
|
+
* Find common comment patterns
|
|
560
|
+
* @param patterns Array of comment patterns
|
|
561
|
+
* @returns Common patterns found
|
|
562
|
+
*/
|
|
563
|
+
private findCommonCommentPatterns(patterns: any[]): string[] {
|
|
564
|
+
const allPatterns = patterns.flatMap((p) => p.patterns);
|
|
565
|
+
const patternCounts = new Map<string, number>();
|
|
566
|
+
|
|
567
|
+
allPatterns.forEach((pattern) => {
|
|
568
|
+
patternCounts.set(pattern, (patternCounts.get(pattern) || 0) + 1);
|
|
569
|
+
});
|
|
570
|
+
|
|
571
|
+
return Array.from(patternCounts.entries())
|
|
572
|
+
.filter(([, count]) => count > 1)
|
|
573
|
+
.map(([pattern]) => pattern)
|
|
574
|
+
.slice(0, 10); // Top 10 common patterns
|
|
575
|
+
}
|
|
576
|
+
|
|
577
|
+
/**
|
|
578
|
+
* Check for consistent capitalization
|
|
579
|
+
* @param content Text content
|
|
580
|
+
* @returns True if capitalization is very consistent
|
|
581
|
+
*/
|
|
582
|
+
private hasConsistentCapitalization(content: string): boolean {
|
|
583
|
+
const sentences = content.split(/[.!?]+/).filter((s) => s.trim().length > 5);
|
|
584
|
+
if (sentences.length < 3) return false;
|
|
585
|
+
|
|
586
|
+
const capitalizedSentences = sentences.filter((s) => {
|
|
587
|
+
const trimmed = s.trim();
|
|
588
|
+
return trimmed.length > 0 && trimmed[0] === trimmed[0].toUpperCase();
|
|
589
|
+
});
|
|
590
|
+
|
|
591
|
+
return capitalizedSentences.length / sentences.length > 0.95;
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
/**
|
|
595
|
+
* Check for perfect punctuation
|
|
596
|
+
* @param content Text content
|
|
597
|
+
* @returns True if punctuation is suspiciously perfect
|
|
598
|
+
*/
|
|
599
|
+
private hasPerfectPunctuation(content: string): boolean {
|
|
600
|
+
// Check for consistent spacing after punctuation
|
|
601
|
+
const punctuationSpacing = content.match(/[.!?]\s{2,}/g);
|
|
602
|
+
const inconsistentSpacing = content.match(/[.!?][^\s]/g);
|
|
603
|
+
|
|
604
|
+
return !inconsistentSpacing && (punctuationSpacing?.length || 0) > 0;
|
|
605
|
+
}
|
|
606
|
+
|
|
607
|
+
/**
|
|
608
|
+
* Create empty result for error cases
|
|
609
|
+
* @returns Empty documentation analysis result
|
|
610
|
+
*/
|
|
611
|
+
private createEmptyResult(): DocumentationResult {
|
|
612
|
+
return {
|
|
613
|
+
analyzer: 'documentation',
|
|
614
|
+
patterns: [],
|
|
615
|
+
metadata: {
|
|
616
|
+
filesAnalyzed: 0,
|
|
617
|
+
hasReadme: false,
|
|
618
|
+
avgCommentDensity: 0,
|
|
619
|
+
},
|
|
620
|
+
};
|
|
621
|
+
}
|
|
622
|
+
}
|