@dinyangetoh/codeplug-cli 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/README.md +291 -0
- package/dist/cli/commands/config.d.ts +4 -0
- package/dist/cli/commands/config.d.ts.map +1 -0
- package/dist/cli/commands/config.js +35 -0
- package/dist/cli/commands/config.js.map +1 -0
- package/dist/cli/commands/convention.d.ts +8 -0
- package/dist/cli/commands/convention.d.ts.map +1 -0
- package/dist/cli/commands/convention.js +92 -0
- package/dist/cli/commands/convention.js.map +1 -0
- package/dist/cli/commands/conventionPrompts.d.ts +3 -0
- package/dist/cli/commands/conventionPrompts.d.ts.map +1 -0
- package/dist/cli/commands/conventionPrompts.js +46 -0
- package/dist/cli/commands/conventionPrompts.js.map +1 -0
- package/dist/cli/commands/docs.d.ts +5 -0
- package/dist/cli/commands/docs.d.ts.map +1 -0
- package/dist/cli/commands/docs.js +49 -0
- package/dist/cli/commands/docs.js.map +1 -0
- package/dist/cli/commands/export.d.ts +3 -0
- package/dist/cli/commands/export.d.ts.map +1 -0
- package/dist/cli/commands/export.js +49 -0
- package/dist/cli/commands/export.js.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +114 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/config/ConfigManager.d.ts +15 -0
- package/dist/config/ConfigManager.d.ts.map +1 -0
- package/dist/config/ConfigManager.js +69 -0
- package/dist/config/ConfigManager.js.map +1 -0
- package/dist/config/ConventionSchema.d.ts +85 -0
- package/dist/config/ConventionSchema.d.ts.map +1 -0
- package/dist/config/ConventionSchema.js +22 -0
- package/dist/config/ConventionSchema.js.map +1 -0
- package/dist/config/defaults.d.ts +13 -0
- package/dist/config/defaults.d.ts.map +1 -0
- package/dist/config/defaults.js +28 -0
- package/dist/config/defaults.js.map +1 -0
- package/dist/config/types.d.ts +138 -0
- package/dist/config/types.d.ts.map +1 -0
- package/dist/config/types.js +2 -0
- package/dist/config/types.js.map +1 -0
- package/dist/core/analyzer/AstAnalyzer.d.ts +17 -0
- package/dist/core/analyzer/AstAnalyzer.d.ts.map +1 -0
- package/dist/core/analyzer/AstAnalyzer.js +95 -0
- package/dist/core/analyzer/AstAnalyzer.js.map +1 -0
- package/dist/core/analyzer/ConventionDetector.d.ts +6 -0
- package/dist/core/analyzer/ConventionDetector.d.ts.map +1 -0
- package/dist/core/analyzer/ConventionDetector.js +38 -0
- package/dist/core/analyzer/ConventionDetector.js.map +1 -0
- package/dist/core/analyzer/PatternAggregator.d.ts +11 -0
- package/dist/core/analyzer/PatternAggregator.d.ts.map +1 -0
- package/dist/core/analyzer/PatternAggregator.js +94 -0
- package/dist/core/analyzer/PatternAggregator.js.map +1 -0
- package/dist/core/analyzer/visitors/ComponentVisitor.d.ts +6 -0
- package/dist/core/analyzer/visitors/ComponentVisitor.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/ComponentVisitor.js +80 -0
- package/dist/core/analyzer/visitors/ComponentVisitor.js.map +1 -0
- package/dist/core/analyzer/visitors/ErrorHandlingVisitor.d.ts +6 -0
- package/dist/core/analyzer/visitors/ErrorHandlingVisitor.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/ErrorHandlingVisitor.js +48 -0
- package/dist/core/analyzer/visitors/ErrorHandlingVisitor.js.map +1 -0
- package/dist/core/analyzer/visitors/ImportVisitor.d.ts +6 -0
- package/dist/core/analyzer/visitors/ImportVisitor.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/ImportVisitor.js +68 -0
- package/dist/core/analyzer/visitors/ImportVisitor.js.map +1 -0
- package/dist/core/analyzer/visitors/NamingVisitor.d.ts +7 -0
- package/dist/core/analyzer/visitors/NamingVisitor.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/NamingVisitor.js +88 -0
- package/dist/core/analyzer/visitors/NamingVisitor.js.map +1 -0
- package/dist/core/analyzer/visitors/StructureVisitor.d.ts +6 -0
- package/dist/core/analyzer/visitors/StructureVisitor.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/StructureVisitor.js +8 -0
- package/dist/core/analyzer/visitors/StructureVisitor.js.map +1 -0
- package/dist/core/analyzer/visitors/TestVisitor.d.ts +6 -0
- package/dist/core/analyzer/visitors/TestVisitor.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/TestVisitor.js +47 -0
- package/dist/core/analyzer/visitors/TestVisitor.js.map +1 -0
- package/dist/core/analyzer/visitors/types.d.ts +13 -0
- package/dist/core/analyzer/visitors/types.d.ts.map +1 -0
- package/dist/core/analyzer/visitors/types.js +2 -0
- package/dist/core/analyzer/visitors/types.js.map +1 -0
- package/dist/core/classifier/ConfidenceGate.d.ts +9 -0
- package/dist/core/classifier/ConfidenceGate.d.ts.map +1 -0
- package/dist/core/classifier/ConfidenceGate.js +11 -0
- package/dist/core/classifier/ConfidenceGate.js.map +1 -0
- package/dist/core/classifier/DriftClassifier.d.ts +17 -0
- package/dist/core/classifier/DriftClassifier.d.ts.map +1 -0
- package/dist/core/classifier/DriftClassifier.js +293 -0
- package/dist/core/classifier/DriftClassifier.js.map +1 -0
- package/dist/core/exporter/ExportEngine.d.ts +7 -0
- package/dist/core/exporter/ExportEngine.d.ts.map +1 -0
- package/dist/core/exporter/ExportEngine.js +52 -0
- package/dist/core/exporter/ExportEngine.js.map +1 -0
- package/dist/core/exporter/FreshnessChecker.d.ts +8 -0
- package/dist/core/exporter/FreshnessChecker.d.ts.map +1 -0
- package/dist/core/exporter/FreshnessChecker.js +49 -0
- package/dist/core/exporter/FreshnessChecker.js.map +1 -0
- package/dist/core/exporter/formatters/CiFormatter.d.ts +8 -0
- package/dist/core/exporter/formatters/CiFormatter.d.ts.map +1 -0
- package/dist/core/exporter/formatters/CiFormatter.js +51 -0
- package/dist/core/exporter/formatters/CiFormatter.js.map +1 -0
- package/dist/core/exporter/formatters/ClaudeFormatter.d.ts +9 -0
- package/dist/core/exporter/formatters/ClaudeFormatter.d.ts.map +1 -0
- package/dist/core/exporter/formatters/ClaudeFormatter.js +60 -0
- package/dist/core/exporter/formatters/ClaudeFormatter.js.map +1 -0
- package/dist/core/exporter/formatters/CopilotFormatter.d.ts +9 -0
- package/dist/core/exporter/formatters/CopilotFormatter.d.ts.map +1 -0
- package/dist/core/exporter/formatters/CopilotFormatter.js +50 -0
- package/dist/core/exporter/formatters/CopilotFormatter.js.map +1 -0
- package/dist/core/exporter/formatters/CursorFormatter.d.ts +9 -0
- package/dist/core/exporter/formatters/CursorFormatter.d.ts.map +1 -0
- package/dist/core/exporter/formatters/CursorFormatter.js +51 -0
- package/dist/core/exporter/formatters/CursorFormatter.js.map +1 -0
- package/dist/core/exporter/formatters/JsonFormatter.d.ts +8 -0
- package/dist/core/exporter/formatters/JsonFormatter.d.ts.map +1 -0
- package/dist/core/exporter/formatters/JsonFormatter.js +22 -0
- package/dist/core/exporter/formatters/JsonFormatter.js.map +1 -0
- package/dist/core/exporter/formatters/types.d.ts +7 -0
- package/dist/core/exporter/formatters/types.d.ts.map +1 -0
- package/dist/core/exporter/formatters/types.js +2 -0
- package/dist/core/exporter/formatters/types.js.map +1 -0
- package/dist/core/generator/DocGenerator.d.ts +14 -0
- package/dist/core/generator/DocGenerator.d.ts.map +1 -0
- package/dist/core/generator/DocGenerator.js +136 -0
- package/dist/core/generator/DocGenerator.js.map +1 -0
- package/dist/core/generator/PipelineOrchestrator.d.ts +11 -0
- package/dist/core/generator/PipelineOrchestrator.d.ts.map +1 -0
- package/dist/core/generator/PipelineOrchestrator.js +26 -0
- package/dist/core/generator/PipelineOrchestrator.js.map +1 -0
- package/dist/core/generator/StalenessTracker.d.ts +13 -0
- package/dist/core/generator/StalenessTracker.d.ts.map +1 -0
- package/dist/core/generator/StalenessTracker.js +76 -0
- package/dist/core/generator/StalenessTracker.js.map +1 -0
- package/dist/core/generator/documents/ArchitectureGenerator.d.ts +13 -0
- package/dist/core/generator/documents/ArchitectureGenerator.d.ts.map +1 -0
- package/dist/core/generator/documents/ArchitectureGenerator.js +101 -0
- package/dist/core/generator/documents/ArchitectureGenerator.js.map +1 -0
- package/dist/core/generator/documents/ContributingGenerator.d.ts +10 -0
- package/dist/core/generator/documents/ContributingGenerator.d.ts.map +1 -0
- package/dist/core/generator/documents/ContributingGenerator.js +90 -0
- package/dist/core/generator/documents/ContributingGenerator.js.map +1 -0
- package/dist/core/generator/documents/ConventionsGenerator.d.ts +8 -0
- package/dist/core/generator/documents/ConventionsGenerator.d.ts.map +1 -0
- package/dist/core/generator/documents/ConventionsGenerator.js +65 -0
- package/dist/core/generator/documents/ConventionsGenerator.js.map +1 -0
- package/dist/core/generator/documents/OnboardingGenerator.d.ts +12 -0
- package/dist/core/generator/documents/OnboardingGenerator.d.ts.map +1 -0
- package/dist/core/generator/documents/OnboardingGenerator.js +116 -0
- package/dist/core/generator/documents/OnboardingGenerator.js.map +1 -0
- package/dist/core/generator/documents/ReadmeGenerator.d.ts +12 -0
- package/dist/core/generator/documents/ReadmeGenerator.d.ts.map +1 -0
- package/dist/core/generator/documents/ReadmeGenerator.js +118 -0
- package/dist/core/generator/documents/ReadmeGenerator.js.map +1 -0
- package/dist/core/generator/documents/types.d.ts +18 -0
- package/dist/core/generator/documents/types.d.ts.map +1 -0
- package/dist/core/generator/documents/types.js +2 -0
- package/dist/core/generator/documents/types.js.map +1 -0
- package/dist/core/generator/llm/LlmClient.d.ts +12 -0
- package/dist/core/generator/llm/LlmClient.d.ts.map +1 -0
- package/dist/core/generator/llm/LlmClient.js +26 -0
- package/dist/core/generator/llm/LlmClient.js.map +1 -0
- package/dist/core/generator/llm/healthCheck.d.ts +6 -0
- package/dist/core/generator/llm/healthCheck.d.ts.map +1 -0
- package/dist/core/generator/llm/healthCheck.js +19 -0
- package/dist/core/generator/llm/healthCheck.js.map +1 -0
- package/dist/core/generator/llm/providerPresets.d.ts +8 -0
- package/dist/core/generator/llm/providerPresets.d.ts.map +1 -0
- package/dist/core/generator/llm/providerPresets.js +9 -0
- package/dist/core/generator/llm/providerPresets.js.map +1 -0
- package/dist/core/generator/pipelines/ExtractionPipeline.d.ts +7 -0
- package/dist/core/generator/pipelines/ExtractionPipeline.d.ts.map +1 -0
- package/dist/core/generator/pipelines/ExtractionPipeline.js +13 -0
- package/dist/core/generator/pipelines/ExtractionPipeline.js.map +1 -0
- package/dist/core/generator/pipelines/NerPipeline.d.ts +12 -0
- package/dist/core/generator/pipelines/NerPipeline.d.ts.map +1 -0
- package/dist/core/generator/pipelines/NerPipeline.js +17 -0
- package/dist/core/generator/pipelines/NerPipeline.js.map +1 -0
- package/dist/core/generator/pipelines/SummarizationPipeline.d.ts +7 -0
- package/dist/core/generator/pipelines/SummarizationPipeline.d.ts.map +1 -0
- package/dist/core/generator/pipelines/SummarizationPipeline.js +13 -0
- package/dist/core/generator/pipelines/SummarizationPipeline.js.map +1 -0
- package/dist/core/git/GitIntegration.d.ts +30 -0
- package/dist/core/git/GitIntegration.d.ts.map +1 -0
- package/dist/core/git/GitIntegration.js +92 -0
- package/dist/core/git/GitIntegration.js.map +1 -0
- package/dist/core/git/PreCommitHook.d.ts +9 -0
- package/dist/core/git/PreCommitHook.d.ts.map +1 -0
- package/dist/core/git/PreCommitHook.js +80 -0
- package/dist/core/git/PreCommitHook.js.map +1 -0
- package/dist/core/scorer/AutoFixer.d.ts +12 -0
- package/dist/core/scorer/AutoFixer.d.ts.map +1 -0
- package/dist/core/scorer/AutoFixer.js +121 -0
- package/dist/core/scorer/AutoFixer.js.map +1 -0
- package/dist/core/scorer/ComplianceScorer.d.ts +7 -0
- package/dist/core/scorer/ComplianceScorer.d.ts.map +1 -0
- package/dist/core/scorer/ComplianceScorer.js +76 -0
- package/dist/core/scorer/ComplianceScorer.js.map +1 -0
- package/dist/core/scorer/TrendTracker.d.ts +6 -0
- package/dist/core/scorer/TrendTracker.d.ts.map +1 -0
- package/dist/core/scorer/TrendTracker.js +45 -0
- package/dist/core/scorer/TrendTracker.js.map +1 -0
- package/dist/core/scorer/ViolationDetector.d.ts +17 -0
- package/dist/core/scorer/ViolationDetector.d.ts.map +1 -0
- package/dist/core/scorer/ViolationDetector.js +178 -0
- package/dist/core/scorer/ViolationDetector.js.map +1 -0
- package/dist/models/ModelManager.d.ts +15 -0
- package/dist/models/ModelManager.d.ts.map +1 -0
- package/dist/models/ModelManager.js +72 -0
- package/dist/models/ModelManager.js.map +1 -0
- package/dist/models/ModelRegistry.d.ts +12 -0
- package/dist/models/ModelRegistry.d.ts.map +1 -0
- package/dist/models/ModelRegistry.js +76 -0
- package/dist/models/ModelRegistry.js.map +1 -0
- package/dist/storage/ConventionStore.d.ts +11 -0
- package/dist/storage/ConventionStore.d.ts.map +1 -0
- package/dist/storage/ConventionStore.js +37 -0
- package/dist/storage/ConventionStore.js.map +1 -0
- package/dist/storage/ScoreStore.d.ts +18 -0
- package/dist/storage/ScoreStore.d.ts.map +1 -0
- package/dist/storage/ScoreStore.js +112 -0
- package/dist/storage/ScoreStore.js.map +1 -0
- package/dist/storage/ViolationStore.d.ts +11 -0
- package/dist/storage/ViolationStore.d.ts.map +1 -0
- package/dist/storage/ViolationStore.js +30 -0
- package/dist/storage/ViolationStore.js.map +1 -0
- package/package.json +75 -0
- package/templates/claude/CLAUDE.md.hbs +15 -0
- package/templates/copilot/copilot-instructions.md.hbs +13 -0
- package/templates/cursor/conventions.mdc.hbs +16 -0
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { join, extname } from 'node:path';
|
|
2
|
+
import { readFile } from 'node:fs/promises';
|
|
3
|
+
const SUPPORTED_EXTENSIONS = new Set(['.ts', '.tsx', '.js', '.jsx', '.mjs', '.cjs']);
|
|
4
|
+
const BATCH_SIZE = 50;
|
|
5
|
+
export class AstAnalyzer {
|
|
6
|
+
projectRoot;
|
|
7
|
+
constructor(projectRoot) {
|
|
8
|
+
this.projectRoot = projectRoot;
|
|
9
|
+
}
|
|
10
|
+
async analyze() {
|
|
11
|
+
const start = Date.now();
|
|
12
|
+
const { globby } = await import('globby');
|
|
13
|
+
const files = await globby(['**/*.{ts,tsx,js,jsx,mjs,cjs}'], {
|
|
14
|
+
cwd: this.projectRoot,
|
|
15
|
+
gitignore: true,
|
|
16
|
+
ignore: ['**/node_modules/**', '**/dist/**', '**/build/**', '**/.codeplug/**'],
|
|
17
|
+
absolute: false,
|
|
18
|
+
});
|
|
19
|
+
const sourceFiles = files.filter((f) => SUPPORTED_EXTENSIONS.has(extname(f)));
|
|
20
|
+
const { PatternAggregator } = await import('./PatternAggregator.js');
|
|
21
|
+
const aggregator = new PatternAggregator();
|
|
22
|
+
for await (const batch of this.processBatches(sourceFiles)) {
|
|
23
|
+
aggregator.ingest(batch);
|
|
24
|
+
}
|
|
25
|
+
const folderStructure = this.buildFolderTree(sourceFiles);
|
|
26
|
+
aggregator.ingestStructure(folderStructure);
|
|
27
|
+
return {
|
|
28
|
+
fileCount: sourceFiles.length,
|
|
29
|
+
durationMs: Date.now() - start,
|
|
30
|
+
patterns: aggregator.getPatterns(),
|
|
31
|
+
folderStructure,
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
async *processBatches(files) {
|
|
35
|
+
for (let i = 0; i < files.length; i += BATCH_SIZE) {
|
|
36
|
+
const batch = files.slice(i, i + BATCH_SIZE);
|
|
37
|
+
const results = await Promise.all(batch.map((f) => this.parseFile(f)));
|
|
38
|
+
yield results.filter((r) => r !== null);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
async parseFile(filePath) {
|
|
42
|
+
try {
|
|
43
|
+
const absolutePath = join(this.projectRoot, filePath);
|
|
44
|
+
const code = await readFile(absolutePath, 'utf-8');
|
|
45
|
+
const { parse } = await import('@babel/parser');
|
|
46
|
+
const ast = parse(code, {
|
|
47
|
+
sourceType: 'module',
|
|
48
|
+
plugins: [
|
|
49
|
+
'typescript',
|
|
50
|
+
'jsx',
|
|
51
|
+
'decorators-legacy',
|
|
52
|
+
'classProperties',
|
|
53
|
+
'optionalChaining',
|
|
54
|
+
'nullishCoalescingOperator',
|
|
55
|
+
'dynamicImport',
|
|
56
|
+
],
|
|
57
|
+
errorRecovery: true,
|
|
58
|
+
});
|
|
59
|
+
return { filePath, code, ast };
|
|
60
|
+
}
|
|
61
|
+
catch {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
buildFolderTree(files) {
|
|
66
|
+
const root = {
|
|
67
|
+
name: '.',
|
|
68
|
+
path: '.',
|
|
69
|
+
children: [],
|
|
70
|
+
fileCount: 0,
|
|
71
|
+
};
|
|
72
|
+
for (const file of files) {
|
|
73
|
+
const parts = file.split('/');
|
|
74
|
+
let current = root;
|
|
75
|
+
current.fileCount++;
|
|
76
|
+
for (let i = 0; i < parts.length - 1; i++) {
|
|
77
|
+
const dirName = parts[i];
|
|
78
|
+
let child = current.children.find((c) => c.name === dirName);
|
|
79
|
+
if (!child) {
|
|
80
|
+
child = {
|
|
81
|
+
name: dirName,
|
|
82
|
+
path: parts.slice(0, i + 1).join('/'),
|
|
83
|
+
children: [],
|
|
84
|
+
fileCount: 0,
|
|
85
|
+
};
|
|
86
|
+
current.children.push(child);
|
|
87
|
+
}
|
|
88
|
+
child.fileCount++;
|
|
89
|
+
current = child;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
return root;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
//# sourceMappingURL=AstAnalyzer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"AstAnalyzer.js","sourceRoot":"","sources":["../../../src/core/analyzer/AstAnalyzer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { AnalysisResult, ConventionCandidate } from '../../config/types.js';
|
|
2
|
+
export declare class ConventionDetector {
|
|
3
|
+
detect(analysis: AnalysisResult): Promise<ConventionCandidate[]>;
|
|
4
|
+
private generateId;
|
|
5
|
+
}
|
|
6
|
+
//# sourceMappingURL=ConventionDetector.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ConventionDetector.d.ts","sourceRoot":"","sources":["../../../src/core/analyzer/ConventionDetector.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,mBAAmB,EAAuB,MAAM,uBAAuB,CAAC;AActG,qBAAa,kBAAkB;IACvB,MAAM,CAAC,QAAQ,EAAE,cAAc,GAAG,OAAO,CAAC,mBAAmB,EAAE,CAAC;IAmBtE,OAAO,CAAC,UAAU;CAQnB"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
const SEVERITY_MAP = {
|
|
2
|
+
naming: 'medium',
|
|
3
|
+
structure: 'high',
|
|
4
|
+
component: 'medium',
|
|
5
|
+
testing: 'medium',
|
|
6
|
+
'error-handling': 'high',
|
|
7
|
+
imports: 'low',
|
|
8
|
+
git: 'low',
|
|
9
|
+
state: 'medium',
|
|
10
|
+
api: 'medium',
|
|
11
|
+
};
|
|
12
|
+
export class ConventionDetector {
|
|
13
|
+
async detect(analysis) {
|
|
14
|
+
const candidates = [];
|
|
15
|
+
for (const pattern of analysis.patterns) {
|
|
16
|
+
if (pattern.confidence < 60)
|
|
17
|
+
continue;
|
|
18
|
+
candidates.push({
|
|
19
|
+
id: this.generateId(pattern.dimension, pattern.pattern),
|
|
20
|
+
dimension: pattern.dimension,
|
|
21
|
+
rule: pattern.pattern,
|
|
22
|
+
confidence: pattern.confidence,
|
|
23
|
+
examples: pattern.examples.slice(0, 5),
|
|
24
|
+
severity: SEVERITY_MAP[pattern.dimension] ?? 'medium',
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
return candidates.sort((a, b) => b.confidence - a.confidence);
|
|
28
|
+
}
|
|
29
|
+
generateId(dimension, pattern) {
|
|
30
|
+
const slug = pattern
|
|
31
|
+
.toLowerCase()
|
|
32
|
+
.replace(/[^a-z0-9]+/g, '-')
|
|
33
|
+
.replace(/^-|-$/g, '')
|
|
34
|
+
.slice(0, 40);
|
|
35
|
+
return `${dimension}-${slug}`;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
//# sourceMappingURL=ConventionDetector.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ConventionDetector.js","sourceRoot":"","sources":["../../../src/core/analyzer/ConventionDetector.ts"],"names":[],"mappings":"AAEA,MAAM,YAAY,GAAyC;IACzD,MAAM,EAAE,QAAQ;IAChB,SAAS,EAAE,MAAM;IACjB,SAAS,EAAE,QAAQ;IACnB,OAAO,EAAE,QAAQ;IACjB,gBAAgB,EAAE,MAAM;IACxB,OAAO,EAAE,KAAK;IACd,GAAG,EAAE,KAAK;IACV,KAAK,EAAE,QAAQ;IACf,GAAG,EAAE,QAAQ;CACd,CAAC;AAEF,MAAM,OAAO,kBAAkB;IAC7B,KAAK,CAAC,MAAM,CAAC,QAAwB;QACnC,MAAM,UAAU,GAA0B,EAAE,CAAC;QAE7C,KAAK,MAAM,OAAO,IAAI,QAAQ,CAAC,QAAQ,EAAE,CAAC;YACxC,IAAI,OAAO,CAAC,UAAU,GAAG,EAAE;gBAAE,SAAS;YAEtC,UAAU,CAAC,IAAI,CAAC;gBACd,EAAE,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,OAAO,CAAC;gBACvD,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,IAAI,EAAE,OAAO,CAAC,OAAO;gBACrB,UAAU,EAAE,OAAO,CAAC,UAAU;gBAC9B,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;gBACtC,QAAQ,EAAE,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,QAAQ;aACtD,CAAC,CAAC;QACL,CAAC;QAED,OAAO,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,GAAG,CAAC,CAAC,UAAU,CAAC,CAAC;IAChE,CAAC;IAEO,UAAU,CAAC,SAAiB,EAAE,OAAe;QACnD,MAAM,IAAI,GAAG,OAAO;aACjB,WAAW,EAAE;aACb,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC;aAC3B,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC;aACrB,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAChB,OAAO,GAAG,SAAS,IAAI,IAAI,EAAE,CAAC;IAChC,CAAC;CACF"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { ParsedFile } from './AstAnalyzer.js';
|
|
2
|
+
import type { DetectedPattern, FolderNode } from '../../config/types.js';
|
|
3
|
+
export declare class PatternAggregator {
|
|
4
|
+
private accumulators;
|
|
5
|
+
private visitors;
|
|
6
|
+
ingest(files: ParsedFile[]): void;
|
|
7
|
+
ingestStructure(tree: FolderNode): void;
|
|
8
|
+
getPatterns(): DetectedPattern[];
|
|
9
|
+
private addStructurePattern;
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=PatternAggregator.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"PatternAggregator.d.ts","sourceRoot":"","sources":["../../../src/core/analyzer/PatternAggregator.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AACnD,OAAO,KAAK,EAAE,eAAe,EAAa,UAAU,EAAE,MAAM,uBAAuB,CAAC;AAepF,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,YAAY,CAAyC;IAC7D,OAAO,CAAC,QAAQ,CAMd;IAEF,MAAM,CAAC,KAAK,EAAE,UAAU,EAAE,GAAG,IAAI;IA2BjC,eAAe,CAAC,IAAI,EAAE,UAAU,GAAG,IAAI;IAyBvC,WAAW,IAAI,eAAe,EAAE;IAqBhC,OAAO,CAAC,mBAAmB;CAU5B"}
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { NamingVisitor } from './visitors/NamingVisitor.js';
|
|
2
|
+
import { ComponentVisitor } from './visitors/ComponentVisitor.js';
|
|
3
|
+
import { TestVisitor } from './visitors/TestVisitor.js';
|
|
4
|
+
import { ErrorHandlingVisitor } from './visitors/ErrorHandlingVisitor.js';
|
|
5
|
+
import { ImportVisitor } from './visitors/ImportVisitor.js';
|
|
6
|
+
export class PatternAggregator {
|
|
7
|
+
accumulators = new Map();
|
|
8
|
+
visitors = [
|
|
9
|
+
new NamingVisitor(),
|
|
10
|
+
new ComponentVisitor(),
|
|
11
|
+
new TestVisitor(),
|
|
12
|
+
new ErrorHandlingVisitor(),
|
|
13
|
+
new ImportVisitor(),
|
|
14
|
+
];
|
|
15
|
+
ingest(files) {
|
|
16
|
+
for (const file of files) {
|
|
17
|
+
for (const visitor of this.visitors) {
|
|
18
|
+
const findings = visitor.visit(file);
|
|
19
|
+
for (const f of findings) {
|
|
20
|
+
const key = `${f.dimension}:${f.pattern}`;
|
|
21
|
+
const existing = this.accumulators.get(key);
|
|
22
|
+
if (existing) {
|
|
23
|
+
existing.count += f.count;
|
|
24
|
+
existing.total += f.total;
|
|
25
|
+
if (existing.examples.length < 5 && f.example) {
|
|
26
|
+
existing.examples.push(f.example);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
else {
|
|
30
|
+
this.accumulators.set(key, {
|
|
31
|
+
dimension: f.dimension,
|
|
32
|
+
pattern: f.pattern,
|
|
33
|
+
count: f.count,
|
|
34
|
+
total: f.total,
|
|
35
|
+
examples: f.example ? [f.example] : [],
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
ingestStructure(tree) {
|
|
43
|
+
const topDirs = tree.children.map((c) => c.name);
|
|
44
|
+
const featureLike = ['features', 'modules', 'domains', 'pages'];
|
|
45
|
+
const mvcLike = ['models', 'views', 'controllers'];
|
|
46
|
+
const layeredLike = ['controllers', 'services', 'repositories', 'entities'];
|
|
47
|
+
const isFeatureBased = featureLike.some((d) => topDirs.includes(d));
|
|
48
|
+
const isMvc = mvcLike.filter((d) => topDirs.includes(d)).length >= 2;
|
|
49
|
+
const isLayered = layeredLike.filter((d) => topDirs.includes(d)).length >= 2;
|
|
50
|
+
if (isFeatureBased) {
|
|
51
|
+
this.addStructurePattern('Feature-based folder structure', topDirs);
|
|
52
|
+
}
|
|
53
|
+
else if (isMvc) {
|
|
54
|
+
this.addStructurePattern('MVC folder structure', topDirs);
|
|
55
|
+
}
|
|
56
|
+
else if (isLayered) {
|
|
57
|
+
this.addStructurePattern('Layered architecture folder structure', topDirs);
|
|
58
|
+
}
|
|
59
|
+
const hasSrcDir = topDirs.includes('src');
|
|
60
|
+
if (hasSrcDir) {
|
|
61
|
+
this.addStructurePattern('src/ root directory convention', ['src/']);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
getPatterns() {
|
|
65
|
+
const results = [];
|
|
66
|
+
for (const acc of this.accumulators.values()) {
|
|
67
|
+
if (acc.total === 0)
|
|
68
|
+
continue;
|
|
69
|
+
const confidence = Math.round((acc.count / acc.total) * 100);
|
|
70
|
+
if (confidence < 50)
|
|
71
|
+
continue;
|
|
72
|
+
results.push({
|
|
73
|
+
dimension: acc.dimension,
|
|
74
|
+
pattern: acc.pattern,
|
|
75
|
+
frequency: acc.count,
|
|
76
|
+
total: acc.total,
|
|
77
|
+
confidence,
|
|
78
|
+
examples: acc.examples,
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
return results.sort((a, b) => b.confidence - a.confidence);
|
|
82
|
+
}
|
|
83
|
+
addStructurePattern(pattern, examples) {
|
|
84
|
+
const key = `structure:${pattern}`;
|
|
85
|
+
this.accumulators.set(key, {
|
|
86
|
+
dimension: 'structure',
|
|
87
|
+
pattern,
|
|
88
|
+
count: 1,
|
|
89
|
+
total: 1,
|
|
90
|
+
examples,
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
//# sourceMappingURL=PatternAggregator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"PatternAggregator.js","sourceRoot":"","sources":["../../../src/core/analyzer/PatternAggregator.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { AstVisitor, VisitorFinding } from './types.js';
|
|
2
|
+
import type { ParsedFile } from '../AstAnalyzer.js';
|
|
3
|
+
export declare class ComponentVisitor implements AstVisitor {
|
|
4
|
+
visit(file: ParsedFile): VisitorFinding[];
|
|
5
|
+
}
|
|
6
|
+
//# sourceMappingURL=ComponentVisitor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ComponentVisitor.d.ts","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/ComponentVisitor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC7D,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAIpD,qBAAa,gBAAiB,YAAW,UAAU;IACjD,KAAK,CAAC,IAAI,EAAE,UAAU,GAAG,cAAc,EAAE;CAgF1C"}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import traverse from '@babel/traverse';
|
|
2
|
+
import { extname } from 'node:path';
|
|
3
|
+
export class ComponentVisitor {
|
|
4
|
+
visit(file) {
|
|
5
|
+
const ext = extname(file.filePath);
|
|
6
|
+
if (ext !== '.tsx' && ext !== '.jsx')
|
|
7
|
+
return [];
|
|
8
|
+
const findings = [];
|
|
9
|
+
let hasFunctionalComponent = false;
|
|
10
|
+
let hasClassComponent = false;
|
|
11
|
+
let hasHooks = false;
|
|
12
|
+
try {
|
|
13
|
+
// @ts-expect-error -- @babel/traverse CJS default export not resolved under NodeNext
|
|
14
|
+
traverse(file.ast, {
|
|
15
|
+
FunctionDeclaration(path) {
|
|
16
|
+
if (returnsJSX(path.node.body))
|
|
17
|
+
hasFunctionalComponent = true;
|
|
18
|
+
},
|
|
19
|
+
ArrowFunctionExpression(path) {
|
|
20
|
+
if (path.node.body.type === 'JSXElement' ||
|
|
21
|
+
path.node.body.type === 'JSXFragment' ||
|
|
22
|
+
(path.node.body.type === 'BlockStatement' && returnsJSX(path.node.body))) {
|
|
23
|
+
hasFunctionalComponent = true;
|
|
24
|
+
}
|
|
25
|
+
},
|
|
26
|
+
ClassDeclaration(path) {
|
|
27
|
+
const sc = path.node.superClass;
|
|
28
|
+
if (sc &&
|
|
29
|
+
((sc.type === 'Identifier' && sc.name === 'Component') ||
|
|
30
|
+
(sc.type === 'MemberExpression' && sc.property?.name === 'Component'))) {
|
|
31
|
+
hasClassComponent = true;
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
CallExpression(path) {
|
|
35
|
+
if (path.node.callee.type === 'Identifier' &&
|
|
36
|
+
/^use[A-Z]/.test(path.node.callee.name ?? '')) {
|
|
37
|
+
hasHooks = true;
|
|
38
|
+
}
|
|
39
|
+
},
|
|
40
|
+
noScope: true,
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
catch {
|
|
44
|
+
return findings;
|
|
45
|
+
}
|
|
46
|
+
if (hasFunctionalComponent) {
|
|
47
|
+
findings.push({
|
|
48
|
+
dimension: 'component',
|
|
49
|
+
pattern: 'Functional components',
|
|
50
|
+
count: 1,
|
|
51
|
+
total: 1,
|
|
52
|
+
example: file.filePath,
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
if (hasClassComponent) {
|
|
56
|
+
findings.push({
|
|
57
|
+
dimension: 'component',
|
|
58
|
+
pattern: 'Class components',
|
|
59
|
+
count: 1,
|
|
60
|
+
total: 1,
|
|
61
|
+
example: file.filePath,
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
if (hasHooks) {
|
|
65
|
+
findings.push({
|
|
66
|
+
dimension: 'component',
|
|
67
|
+
pattern: 'Hooks composition pattern',
|
|
68
|
+
count: 1,
|
|
69
|
+
total: 1,
|
|
70
|
+
example: file.filePath,
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
return findings;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
function returnsJSX(body) {
|
|
77
|
+
return body.body?.some((stmt) => stmt.type === 'ReturnStatement' &&
|
|
78
|
+
(stmt.argument?.type === 'JSXElement' || stmt.argument?.type === 'JSXFragment')) ?? false;
|
|
79
|
+
}
|
|
80
|
+
//# sourceMappingURL=ComponentVisitor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ComponentVisitor.js","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/ComponentVisitor.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { AstVisitor, VisitorFinding } from './types.js';
|
|
2
|
+
import type { ParsedFile } from '../AstAnalyzer.js';
|
|
3
|
+
export declare class ErrorHandlingVisitor implements AstVisitor {
|
|
4
|
+
visit(file: ParsedFile): VisitorFinding[];
|
|
5
|
+
}
|
|
6
|
+
//# sourceMappingURL=ErrorHandlingVisitor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ErrorHandlingVisitor.d.ts","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/ErrorHandlingVisitor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC7D,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAGpD,qBAAa,oBAAqB,YAAW,UAAU;IACrD,KAAK,CAAC,IAAI,EAAE,UAAU,GAAG,cAAc,EAAE;CA6C1C"}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import traverse from '@babel/traverse';
|
|
2
|
+
export class ErrorHandlingVisitor {
|
|
3
|
+
visit(file) {
|
|
4
|
+
const findings = [];
|
|
5
|
+
let tryCatchCount = 0;
|
|
6
|
+
let asyncFnCount = 0;
|
|
7
|
+
try {
|
|
8
|
+
// @ts-expect-error -- @babel/traverse CJS default export not resolved under NodeNext
|
|
9
|
+
traverse(file.ast, {
|
|
10
|
+
TryStatement() {
|
|
11
|
+
tryCatchCount++;
|
|
12
|
+
},
|
|
13
|
+
FunctionDeclaration(path) {
|
|
14
|
+
if (path.node.async)
|
|
15
|
+
asyncFnCount++;
|
|
16
|
+
},
|
|
17
|
+
ArrowFunctionExpression(path) {
|
|
18
|
+
if (path.node.async)
|
|
19
|
+
asyncFnCount++;
|
|
20
|
+
},
|
|
21
|
+
noScope: true,
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
catch {
|
|
25
|
+
return findings;
|
|
26
|
+
}
|
|
27
|
+
if (tryCatchCount > 0) {
|
|
28
|
+
findings.push({
|
|
29
|
+
dimension: 'error-handling',
|
|
30
|
+
pattern: 'Try/catch error handling',
|
|
31
|
+
count: tryCatchCount,
|
|
32
|
+
total: Math.max(tryCatchCount, asyncFnCount),
|
|
33
|
+
example: file.filePath,
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
if (asyncFnCount > 0) {
|
|
37
|
+
findings.push({
|
|
38
|
+
dimension: 'api',
|
|
39
|
+
pattern: 'Async/await pattern',
|
|
40
|
+
count: asyncFnCount,
|
|
41
|
+
total: asyncFnCount,
|
|
42
|
+
example: file.filePath,
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
return findings;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
//# sourceMappingURL=ErrorHandlingVisitor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ErrorHandlingVisitor.js","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/ErrorHandlingVisitor.ts"],"names":[],"mappings":"AAEA,OAAO,QAAQ,MAAM,iBAAiB,CAAC;AAEvC,MAAM,OAAO,oBAAoB;IAC/B,KAAK,CAAC,IAAgB;QACpB,MAAM,QAAQ,GAAqB,EAAE,CAAC;QACtC,IAAI,aAAa,GAAG,CAAC,CAAC;QACtB,IAAI,YAAY,GAAG,CAAC,CAAC;QAErB,IAAI,CAAC;YACH,qFAAqF;YACrF,QAAQ,CAAC,IAAI,CAAC,GAAG,EAAE;gBACjB,YAAY;oBACV,aAAa,EAAE,CAAC;gBAClB,CAAC;gBACD,mBAAmB,CAAC,IAAkC;oBACpD,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK;wBAAE,YAAY,EAAE,CAAC;gBACtC,CAAC;gBACD,uBAAuB,CAAC,IAAkC;oBACxD,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK;wBAAE,YAAY,EAAE,CAAC;gBACtC,CAAC;gBACD,OAAO,EAAE,IAAI;aACd,CAAC,CAAC;QACL,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,IAAI,aAAa,GAAG,CAAC,EAAE,CAAC;YACtB,QAAQ,CAAC,IAAI,CAAC;gBACZ,SAAS,EAAE,gBAAgB;gBAC3B,OAAO,EAAE,0BAA0B;gBACnC,KAAK,EAAE,aAAa;gBACpB,KAAK,EAAE,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,YAAY,CAAC;gBAC5C,OAAO,EAAE,IAAI,CAAC,QAAQ;aACvB,CAAC,CAAC;QACL,CAAC;QAED,IAAI,YAAY,GAAG,CAAC,EAAE,CAAC;YACrB,QAAQ,CAAC,IAAI,CAAC;gBACZ,SAAS,EAAE,KAAK;gBAChB,OAAO,EAAE,qBAAqB;gBAC9B,KAAK,EAAE,YAAY;gBACnB,KAAK,EAAE,YAAY;gBACnB,OAAO,EAAE,IAAI,CAAC,QAAQ;aACvB,CAAC,CAAC;QACL,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { AstVisitor, VisitorFinding } from './types.js';
|
|
2
|
+
import type { ParsedFile } from '../AstAnalyzer.js';
|
|
3
|
+
export declare class ImportVisitor implements AstVisitor {
|
|
4
|
+
visit(file: ParsedFile): VisitorFinding[];
|
|
5
|
+
}
|
|
6
|
+
//# sourceMappingURL=ImportVisitor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ImportVisitor.d.ts","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/ImportVisitor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC7D,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAIpD,qBAAa,aAAc,YAAW,UAAU;IAC9C,KAAK,CAAC,IAAI,EAAE,UAAU,GAAG,cAAc,EAAE;CAsE1C"}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import traverse from '@babel/traverse';
|
|
2
|
+
import { basename } from 'node:path';
|
|
3
|
+
export class ImportVisitor {
|
|
4
|
+
visit(file) {
|
|
5
|
+
const findings = [];
|
|
6
|
+
let namedImportCount = 0;
|
|
7
|
+
let defaultImportCount = 0;
|
|
8
|
+
let totalImports = 0;
|
|
9
|
+
let barrelImportCount = 0;
|
|
10
|
+
try {
|
|
11
|
+
// @ts-expect-error -- @babel/traverse CJS default export not resolved under NodeNext
|
|
12
|
+
traverse(file.ast, {
|
|
13
|
+
ImportDeclaration(path) {
|
|
14
|
+
totalImports++;
|
|
15
|
+
const specs = path.node.specifiers;
|
|
16
|
+
const hasDefault = specs.some((s) => s.type === 'ImportDefaultSpecifier');
|
|
17
|
+
const hasNamed = specs.some((s) => s.type === 'ImportSpecifier');
|
|
18
|
+
if (hasDefault)
|
|
19
|
+
defaultImportCount++;
|
|
20
|
+
if (hasNamed)
|
|
21
|
+
namedImportCount++;
|
|
22
|
+
const src = path.node.source.value;
|
|
23
|
+
if (src.endsWith('/index') || /^\.\.?\/[^/]+$/.test(src)) {
|
|
24
|
+
const base = basename(src);
|
|
25
|
+
if (base === 'index' || !base.includes('.')) {
|
|
26
|
+
barrelImportCount++;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
},
|
|
30
|
+
noScope: true,
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
catch {
|
|
34
|
+
return findings;
|
|
35
|
+
}
|
|
36
|
+
if (totalImports > 0) {
|
|
37
|
+
if (namedImportCount > defaultImportCount) {
|
|
38
|
+
findings.push({
|
|
39
|
+
dimension: 'imports',
|
|
40
|
+
pattern: 'Prefer named imports over default imports',
|
|
41
|
+
count: namedImportCount,
|
|
42
|
+
total: totalImports,
|
|
43
|
+
example: file.filePath,
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
else if (defaultImportCount > 0) {
|
|
47
|
+
findings.push({
|
|
48
|
+
dimension: 'imports',
|
|
49
|
+
pattern: 'Default imports prevalent',
|
|
50
|
+
count: defaultImportCount,
|
|
51
|
+
total: totalImports,
|
|
52
|
+
example: file.filePath,
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
if (barrelImportCount > 0) {
|
|
56
|
+
findings.push({
|
|
57
|
+
dimension: 'imports',
|
|
58
|
+
pattern: 'Barrel imports (index re-exports)',
|
|
59
|
+
count: barrelImportCount,
|
|
60
|
+
total: totalImports,
|
|
61
|
+
example: file.filePath,
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
return findings;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
//# sourceMappingURL=ImportVisitor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ImportVisitor.js","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/ImportVisitor.ts"],"names":[],"mappings":"AAEA,OAAO,QAAQ,MAAM,iBAAiB,CAAC;AACvC,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AAErC,MAAM,OAAO,aAAa;IACxB,KAAK,CAAC,IAAgB;QACpB,MAAM,QAAQ,GAAqB,EAAE,CAAC;QACtC,IAAI,gBAAgB,GAAG,CAAC,CAAC;QACzB,IAAI,kBAAkB,GAAG,CAAC,CAAC;QAC3B,IAAI,YAAY,GAAG,CAAC,CAAC;QACrB,IAAI,iBAAiB,GAAG,CAAC,CAAC;QAE1B,IAAI,CAAC;YACH,qFAAqF;YACrF,QAAQ,CAAC,IAAI,CAAC,GAAG,EAAE;gBACjB,iBAAiB,CAAC,IAKjB;oBACC,YAAY,EAAE,CAAC;oBACf,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;oBACnC,MAAM,UAAU,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,wBAAwB,CAAC,CAAC;oBAC1E,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,iBAAiB,CAAC,CAAC;oBAEjE,IAAI,UAAU;wBAAE,kBAAkB,EAAE,CAAC;oBACrC,IAAI,QAAQ;wBAAE,gBAAgB,EAAE,CAAC;oBAEjC,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC;oBACnC,IAAI,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,gBAAgB,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;wBACzD,MAAM,IAAI,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;wBAC3B,IAAI,IAAI,KAAK,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;4BAC5C,iBAAiB,EAAE,CAAC;wBACtB,CAAC;oBACH,CAAC;gBACH,CAAC;gBACD,OAAO,EAAE,IAAI;aACd,CAAC,CAAC;QACL,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,IAAI,YAAY,GAAG,CAAC,EAAE,CAAC;YACrB,IAAI,gBAAgB,GAAG,kBAAkB,EAAE,CAAC;gBAC1C,QAAQ,CAAC,IAAI,CAAC;oBACZ,SAAS,EAAE,SAAS;oBACpB,OAAO,EAAE,2CAA2C;oBACpD,KAAK,EAAE,gBAAgB;oBACvB,KAAK,EAAE,YAAY;oBACnB,OAAO,EAAE,IAAI,CAAC,QAAQ;iBACvB,CAAC,CAAC;YACL,CAAC;iBAAM,IAAI,kBAAkB,GAAG,CAAC,EAAE,CAAC;gBAClC,QAAQ,CAAC,IAAI,CAAC;oBACZ,SAAS,EAAE,SAAS;oBACpB,OAAO,EAAE,2BAA2B;oBACpC,KAAK,EAAE,kBAAkB;oBACzB,KAAK,EAAE,YAAY;oBACnB,OAAO,EAAE,IAAI,CAAC,QAAQ;iBACvB,CAAC,CAAC;YACL,CAAC;YAED,IAAI,iBAAiB,GAAG,CAAC,EAAE,CAAC;gBAC1B,QAAQ,CAAC,IAAI,CAAC;oBACZ,SAAS,EAAE,SAAS;oBACpB,OAAO,EAAE,mCAAmC;oBAC5C,KAAK,EAAE,iBAAiB;oBACxB,KAAK,EAAE,YAAY;oBACnB,OAAO,EAAE,IAAI,CAAC,QAAQ;iBACvB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { AstVisitor, VisitorFinding } from './types.js';
|
|
2
|
+
import type { ParsedFile } from '../AstAnalyzer.js';
|
|
3
|
+
export declare class NamingVisitor implements AstVisitor {
|
|
4
|
+
visit(file: ParsedFile): VisitorFinding[];
|
|
5
|
+
private visitConstants;
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=NamingVisitor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NamingVisitor.d.ts","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/NamingVisitor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC7D,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AASpD,qBAAa,aAAc,YAAW,UAAU;IAC9C,KAAK,CAAC,IAAI,EAAE,UAAU,GAAG,cAAc,EAAE;IA+CzC,OAAO,CAAC,cAAc;CAoCvB"}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import traverse from '@babel/traverse';
|
|
2
|
+
import { basename, extname } from 'node:path';
|
|
3
|
+
const PASCAL_CASE = /^[A-Z][a-zA-Z0-9]*$/;
|
|
4
|
+
const CAMEL_CASE = /^[a-z][a-zA-Z0-9]*$/;
|
|
5
|
+
const SCREAMING_SNAKE = /^[A-Z][A-Z0-9_]*$/;
|
|
6
|
+
const HOOK_PREFIX = /^use[A-Z]/;
|
|
7
|
+
export class NamingVisitor {
|
|
8
|
+
visit(file) {
|
|
9
|
+
const findings = [];
|
|
10
|
+
const fileName = basename(file.filePath, extname(file.filePath));
|
|
11
|
+
const ext = extname(file.filePath);
|
|
12
|
+
const isComponent = ext === '.tsx' || ext === '.jsx';
|
|
13
|
+
if (isComponent) {
|
|
14
|
+
findings.push({
|
|
15
|
+
dimension: 'naming',
|
|
16
|
+
pattern: 'React components use PascalCase file names',
|
|
17
|
+
count: PASCAL_CASE.test(fileName) ? 1 : 0,
|
|
18
|
+
total: 1,
|
|
19
|
+
example: file.filePath,
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
else if (ext === '.ts' || ext === '.js') {
|
|
23
|
+
if (HOOK_PREFIX.test(fileName)) {
|
|
24
|
+
findings.push({
|
|
25
|
+
dimension: 'naming',
|
|
26
|
+
pattern: 'Hooks use "use" prefix with camelCase',
|
|
27
|
+
count: CAMEL_CASE.test(fileName) ? 1 : 0,
|
|
28
|
+
total: 1,
|
|
29
|
+
example: file.filePath,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
else if (PASCAL_CASE.test(fileName)) {
|
|
33
|
+
findings.push({
|
|
34
|
+
dimension: 'naming',
|
|
35
|
+
pattern: 'Class/service files use PascalCase',
|
|
36
|
+
count: 1,
|
|
37
|
+
total: 1,
|
|
38
|
+
example: file.filePath,
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
else if (CAMEL_CASE.test(fileName)) {
|
|
42
|
+
findings.push({
|
|
43
|
+
dimension: 'naming',
|
|
44
|
+
pattern: 'Utility files use camelCase',
|
|
45
|
+
count: 1,
|
|
46
|
+
total: 1,
|
|
47
|
+
example: file.filePath,
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
this.visitConstants(file, findings);
|
|
52
|
+
return findings;
|
|
53
|
+
}
|
|
54
|
+
visitConstants(file, findings) {
|
|
55
|
+
try {
|
|
56
|
+
let screamingCount = 0;
|
|
57
|
+
let constCount = 0;
|
|
58
|
+
// @ts-expect-error -- @babel/traverse CJS default export not resolved under NodeNext
|
|
59
|
+
traverse(file.ast, {
|
|
60
|
+
VariableDeclarator(path) {
|
|
61
|
+
const { node } = path;
|
|
62
|
+
if (node.id.type === 'Identifier' &&
|
|
63
|
+
node.init &&
|
|
64
|
+
(node.init.type === 'StringLiteral' || node.init.type === 'NumericLiteral')) {
|
|
65
|
+
constCount++;
|
|
66
|
+
if (SCREAMING_SNAKE.test(node.id.name)) {
|
|
67
|
+
screamingCount++;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
},
|
|
71
|
+
noScope: true,
|
|
72
|
+
});
|
|
73
|
+
if (constCount > 0) {
|
|
74
|
+
findings.push({
|
|
75
|
+
dimension: 'naming',
|
|
76
|
+
pattern: 'Constants use SCREAMING_SNAKE_CASE',
|
|
77
|
+
count: screamingCount,
|
|
78
|
+
total: constCount,
|
|
79
|
+
example: file.filePath,
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
catch {
|
|
84
|
+
// traverse may fail on some files
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
//# sourceMappingURL=NamingVisitor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NamingVisitor.js","sourceRoot":"","sources":["../../../../src/core/analyzer/visitors/NamingVisitor.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { AstVisitor, VisitorFinding } from './types.js';
|
|
2
|
+
import type { ParsedFile } from '../AstAnalyzer.js';
|
|
3
|
+
export declare class StructureVisitor implements AstVisitor {
|
|
4
|
+
visit(_file: ParsedFile): VisitorFinding[];
|
|
5
|
+
}
|
|
6
|
+
//# sourceMappingURL=StructureVisitor.d.ts.map
|