@girardelli/architect-core 8.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/dist/src/core/analyzer.d.ts +42 -0
- package/dist/src/core/analyzer.js +431 -0
- package/dist/src/core/analyzer.js.map +1 -0
- package/dist/src/core/analyzers/forecast.d.ts +84 -0
- package/dist/src/core/analyzers/forecast.js +338 -0
- package/dist/src/core/analyzers/forecast.js.map +1 -0
- package/dist/src/core/analyzers/index.d.ts +9 -0
- package/dist/src/core/analyzers/index.js +7 -0
- package/dist/src/core/analyzers/index.js.map +1 -0
- package/dist/src/core/analyzers/temporal-scorer.d.ts +71 -0
- package/dist/src/core/analyzers/temporal-scorer.js +141 -0
- package/dist/src/core/analyzers/temporal-scorer.js.map +1 -0
- package/dist/src/core/anti-patterns.d.ts +28 -0
- package/dist/src/core/anti-patterns.js +264 -0
- package/dist/src/core/anti-patterns.js.map +1 -0
- package/dist/src/core/ast/ast-parser.interface.d.ts +20 -0
- package/dist/src/core/ast/ast-parser.interface.js +2 -0
- package/dist/src/core/ast/ast-parser.interface.js.map +1 -0
- package/dist/src/core/ast/path-resolver.d.ts +13 -0
- package/dist/src/core/ast/path-resolver.js +54 -0
- package/dist/src/core/ast/path-resolver.js.map +1 -0
- package/dist/src/core/ast/tree-sitter-parser.d.ts +10 -0
- package/dist/src/core/ast/tree-sitter-parser.js +142 -0
- package/dist/src/core/ast/tree-sitter-parser.js.map +1 -0
- package/dist/src/core/config.d.ts +11 -0
- package/dist/src/core/config.js +112 -0
- package/dist/src/core/config.js.map +1 -0
- package/dist/src/core/diagram.d.ts +9 -0
- package/dist/src/core/diagram.js +101 -0
- package/dist/src/core/diagram.js.map +1 -0
- package/dist/src/core/i18n.d.ts +14 -0
- package/dist/src/core/i18n.js +54 -0
- package/dist/src/core/i18n.js.map +1 -0
- package/dist/src/core/locales/en.d.ts +2 -0
- package/dist/src/core/locales/en.js +337 -0
- package/dist/src/core/locales/en.js.map +1 -0
- package/dist/src/core/locales/pt-BR.d.ts +172 -0
- package/dist/src/core/locales/pt-BR.js +337 -0
- package/dist/src/core/locales/pt-BR.js.map +1 -0
- package/dist/src/core/locales/types.d.ts +86 -0
- package/dist/src/core/locales/types.js +2 -0
- package/dist/src/core/locales/types.js.map +1 -0
- package/dist/src/core/plugin-loader.d.ts +11 -0
- package/dist/src/core/plugin-loader.js +67 -0
- package/dist/src/core/plugin-loader.js.map +1 -0
- package/dist/src/core/project-summarizer.d.ts +16 -0
- package/dist/src/core/project-summarizer.js +37 -0
- package/dist/src/core/project-summarizer.js.map +1 -0
- package/dist/src/core/refactor-engine.d.ts +18 -0
- package/dist/src/core/refactor-engine.js +87 -0
- package/dist/src/core/refactor-engine.js.map +1 -0
- package/dist/src/core/rules/barrel-optimizer.d.ts +13 -0
- package/dist/src/core/rules/barrel-optimizer.js +76 -0
- package/dist/src/core/rules/barrel-optimizer.js.map +1 -0
- package/dist/src/core/rules/dead-code-detector.d.ts +21 -0
- package/dist/src/core/rules/dead-code-detector.js +116 -0
- package/dist/src/core/rules/dead-code-detector.js.map +1 -0
- package/dist/src/core/rules/hub-splitter.d.ts +13 -0
- package/dist/src/core/rules/hub-splitter.js +117 -0
- package/dist/src/core/rules/hub-splitter.js.map +1 -0
- package/dist/src/core/rules/import-organizer.d.ts +13 -0
- package/dist/src/core/rules/import-organizer.js +84 -0
- package/dist/src/core/rules/import-organizer.js.map +1 -0
- package/dist/src/core/rules/module-grouper.d.ts +13 -0
- package/dist/src/core/rules/module-grouper.js +116 -0
- package/dist/src/core/rules/module-grouper.js.map +1 -0
- package/dist/src/core/rules-engine.d.ts +7 -0
- package/dist/src/core/rules-engine.js +89 -0
- package/dist/src/core/rules-engine.js.map +1 -0
- package/dist/src/core/scorer.d.ts +15 -0
- package/dist/src/core/scorer.js +165 -0
- package/dist/src/core/scorer.js.map +1 -0
- package/dist/src/core/summarizer/keyword-extractor.d.ts +6 -0
- package/dist/src/core/summarizer/keyword-extractor.js +38 -0
- package/dist/src/core/summarizer/keyword-extractor.js.map +1 -0
- package/dist/src/core/summarizer/module-inferrer.d.ts +11 -0
- package/dist/src/core/summarizer/module-inferrer.js +171 -0
- package/dist/src/core/summarizer/module-inferrer.js.map +1 -0
- package/dist/src/core/summarizer/package-reader.d.ts +3 -0
- package/dist/src/core/summarizer/package-reader.js +33 -0
- package/dist/src/core/summarizer/package-reader.js.map +1 -0
- package/dist/src/core/summarizer/purpose-inferrer.d.ts +8 -0
- package/dist/src/core/summarizer/purpose-inferrer.js +179 -0
- package/dist/src/core/summarizer/purpose-inferrer.js.map +1 -0
- package/dist/src/core/summarizer/readme-reader.d.ts +3 -0
- package/dist/src/core/summarizer/readme-reader.js +24 -0
- package/dist/src/core/summarizer/readme-reader.js.map +1 -0
- package/dist/src/core/types/architect-rules.d.ts +27 -0
- package/dist/src/core/types/architect-rules.js +2 -0
- package/dist/src/core/types/architect-rules.js.map +1 -0
- package/dist/src/core/types/core.d.ts +87 -0
- package/dist/src/core/types/core.js +2 -0
- package/dist/src/core/types/core.js.map +1 -0
- package/dist/src/core/types/infrastructure.d.ts +38 -0
- package/dist/src/core/types/infrastructure.js +2 -0
- package/dist/src/core/types/infrastructure.js.map +1 -0
- package/dist/src/core/types/plugin.d.ts +12 -0
- package/dist/src/core/types/plugin.js +2 -0
- package/dist/src/core/types/plugin.js.map +1 -0
- package/dist/src/core/types/rules.d.ts +53 -0
- package/dist/src/core/types/rules.js +2 -0
- package/dist/src/core/types/rules.js.map +1 -0
- package/dist/src/core/types/summarizer.d.ts +12 -0
- package/dist/src/core/types/summarizer.js +2 -0
- package/dist/src/core/types/summarizer.js.map +1 -0
- package/dist/src/infrastructure/git-cache.d.ts +6 -0
- package/dist/src/infrastructure/git-cache.js +41 -0
- package/dist/src/infrastructure/git-cache.js.map +1 -0
- package/dist/src/infrastructure/git-history.d.ts +112 -0
- package/dist/src/infrastructure/git-history.js +340 -0
- package/dist/src/infrastructure/git-history.js.map +1 -0
- package/dist/src/infrastructure/logger.d.ts +20 -0
- package/dist/src/infrastructure/logger.js +57 -0
- package/dist/src/infrastructure/logger.js.map +1 -0
- package/dist/src/infrastructure/scanner.d.ts +31 -0
- package/dist/src/infrastructure/scanner.js +334 -0
- package/dist/src/infrastructure/scanner.js.map +1 -0
- package/dist/tests/analyzers-integration.test.d.ts +7 -0
- package/dist/tests/analyzers-integration.test.js +140 -0
- package/dist/tests/analyzers-integration.test.js.map +1 -0
- package/dist/tests/anti-patterns.test.d.ts +1 -0
- package/dist/tests/anti-patterns.test.js +81 -0
- package/dist/tests/anti-patterns.test.js.map +1 -0
- package/dist/tests/ast-parser.test.d.ts +1 -0
- package/dist/tests/ast-parser.test.js +94 -0
- package/dist/tests/ast-parser.test.js.map +1 -0
- package/dist/tests/fixtures/monorepo/packages/app/src/index.d.ts +1 -0
- package/dist/tests/fixtures/monorepo/packages/app/src/index.js +9 -0
- package/dist/tests/fixtures/monorepo/packages/app/src/index.js.map +1 -0
- package/dist/tests/fixtures/monorepo/packages/core/src/index.d.ts +2 -0
- package/dist/tests/fixtures/monorepo/packages/core/src/index.js +11 -0
- package/dist/tests/fixtures/monorepo/packages/core/src/index.js.map +1 -0
- package/dist/tests/forecast.test.d.ts +7 -0
- package/dist/tests/forecast.test.js +380 -0
- package/dist/tests/forecast.test.js.map +1 -0
- package/dist/tests/git-history.test.d.ts +7 -0
- package/dist/tests/git-history.test.js +193 -0
- package/dist/tests/git-history.test.js.map +1 -0
- package/dist/tests/i18n.test.d.ts +1 -0
- package/dist/tests/i18n.test.js +39 -0
- package/dist/tests/i18n.test.js.map +1 -0
- package/dist/tests/monorepo-scan.test.d.ts +11 -0
- package/dist/tests/monorepo-scan.test.js +143 -0
- package/dist/tests/monorepo-scan.test.js.map +1 -0
- package/dist/tests/plugin-loader.test.d.ts +1 -0
- package/dist/tests/plugin-loader.test.js +31 -0
- package/dist/tests/plugin-loader.test.js.map +1 -0
- package/dist/tests/rules-engine.test.d.ts +1 -0
- package/dist/tests/rules-engine.test.js +112 -0
- package/dist/tests/rules-engine.test.js.map +1 -0
- package/dist/tests/scanner.test.d.ts +1 -0
- package/dist/tests/scanner.test.js +44 -0
- package/dist/tests/scanner.test.js.map +1 -0
- package/dist/tests/scorer.test.d.ts +1 -0
- package/dist/tests/scorer.test.js +610 -0
- package/dist/tests/scorer.test.js.map +1 -0
- package/dist/tests/temporal-scorer.test.d.ts +7 -0
- package/dist/tests/temporal-scorer.test.js +239 -0
- package/dist/tests/temporal-scorer.test.js.map +1 -0
- package/package.json +29 -0
- package/src/core/analyzer.ts +499 -0
- package/src/core/analyzers/forecast.ts +497 -0
- package/src/core/analyzers/index.ts +33 -0
- package/src/core/analyzers/temporal-scorer.ts +227 -0
- package/src/core/anti-patterns.ts +324 -0
- package/src/core/ast/ast-parser.interface.ts +21 -0
- package/src/core/ast/path-resolver.ts +61 -0
- package/src/core/ast/tree-sitter-parser.ts +158 -0
- package/src/core/config.ts +125 -0
- package/src/core/diagram.ts +129 -0
- package/src/core/i18n.ts +64 -0
- package/src/core/locales/en.ts +340 -0
- package/src/core/locales/pt-BR.ts +341 -0
- package/src/core/locales/types.ts +95 -0
- package/src/core/plugin-loader.ts +80 -0
- package/src/core/project-summarizer.ts +42 -0
- package/src/core/refactor-engine.ts +112 -0
- package/src/core/rules/barrel-optimizer.ts +99 -0
- package/src/core/rules/dead-code-detector.ts +134 -0
- package/src/core/rules/hub-splitter.ts +135 -0
- package/src/core/rules/import-organizer.ts +100 -0
- package/src/core/rules/module-grouper.ts +133 -0
- package/src/core/rules-engine.ts +100 -0
- package/src/core/scorer.ts +181 -0
- package/src/core/summarizer/keyword-extractor.ts +53 -0
- package/src/core/summarizer/module-inferrer.ts +194 -0
- package/src/core/summarizer/package-reader.ts +34 -0
- package/src/core/summarizer/purpose-inferrer.ts +197 -0
- package/src/core/summarizer/readme-reader.ts +24 -0
- package/src/core/types/architect-rules.ts +29 -0
- package/src/core/types/core.ts +94 -0
- package/src/core/types/infrastructure.ts +41 -0
- package/src/core/types/plugin.ts +19 -0
- package/src/core/types/rules.ts +51 -0
- package/src/core/types/summarizer.ts +8 -0
- package/src/infrastructure/git-cache.ts +52 -0
- package/src/infrastructure/git-history.ts +496 -0
- package/src/infrastructure/logger.ts +68 -0
- package/src/infrastructure/scanner.ts +349 -0
- package/tests/analyzers-integration.test.ts +174 -0
- package/tests/anti-patterns.test.ts +95 -0
- package/tests/ast-parser.test.ts +102 -0
- package/tests/fixtures/monorepo/package.json +6 -0
- package/tests/fixtures/monorepo/packages/app/package.json +12 -0
- package/tests/fixtures/monorepo/packages/app/src/index.ts +6 -0
- package/tests/fixtures/monorepo/packages/core/package.json +7 -0
- package/tests/fixtures/monorepo/packages/core/src/index.ts +7 -0
- package/tests/forecast.test.ts +504 -0
- package/tests/git-history.test.ts +254 -0
- package/tests/i18n.test.ts +47 -0
- package/tests/monorepo-scan.test.ts +170 -0
- package/tests/plugin-loader.test.ts +40 -0
- package/tests/rules-engine.test.ts +131 -0
- package/tests/scanner.test.ts +54 -0
- package/tests/scorer.test.ts +675 -0
- package/tests/temporal-scorer.test.ts +306 -0
- package/tsconfig.json +9 -0
|
@@ -0,0 +1,338 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Architecture Weather Forecast — Predictive analysis
|
|
3
|
+
*
|
|
4
|
+
* Combines temporal scores + velocity vectors + change coupling
|
|
5
|
+
* to predict which modules will become anti-patterns.
|
|
6
|
+
*
|
|
7
|
+
* Key concept: Pre-Anti-Pattern
|
|
8
|
+
* A module isn't an anti-pattern yet, but its trajectory says it will be.
|
|
9
|
+
* "Your code doesn't have a problem — it WILL have one in 3 months."
|
|
10
|
+
*
|
|
11
|
+
* @author Camilo Girardelli — Girardelli Tecnologia
|
|
12
|
+
* @license MIT
|
|
13
|
+
*/
|
|
14
|
+
const DEFAULT_CONFIG = {
|
|
15
|
+
antiPatternThreshold: 40,
|
|
16
|
+
godClassChurnThreshold: 150,
|
|
17
|
+
shotgunCouplingThreshold: 5,
|
|
18
|
+
busFatorRiskThreshold: 1,
|
|
19
|
+
forecastWeeks: 26,
|
|
20
|
+
};
|
|
21
|
+
// ═══════════════════════════════════════════════════════════════
|
|
22
|
+
// FORECAST ENGINE
|
|
23
|
+
// ═══════════════════════════════════════════════════════════════
|
|
24
|
+
export class ForecastEngine {
|
|
25
|
+
config;
|
|
26
|
+
constructor(config) {
|
|
27
|
+
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Generate architecture weather forecast.
|
|
31
|
+
*/
|
|
32
|
+
forecast(gitReport, temporalReport) {
|
|
33
|
+
const preAntiPatterns = [];
|
|
34
|
+
const moduleForecastMap = new Map();
|
|
35
|
+
// Detect pre-anti-patterns for each module
|
|
36
|
+
for (const module of gitReport.modules) {
|
|
37
|
+
const temporal = temporalReport.modules.find(m => m.module === module.modulePath);
|
|
38
|
+
if (!temporal)
|
|
39
|
+
continue;
|
|
40
|
+
const patterns = this.detectPreAntiPatterns(module, temporal, gitReport.changeCouplings);
|
|
41
|
+
preAntiPatterns.push(...patterns);
|
|
42
|
+
const forecast = this.forecastModule(module, temporal, patterns);
|
|
43
|
+
moduleForecastMap.set(module.modulePath, forecast);
|
|
44
|
+
}
|
|
45
|
+
// Sort by risk
|
|
46
|
+
const modules = Array.from(moduleForecastMap.values())
|
|
47
|
+
.sort((a, b) => b.bottleneckProbability - a.bottleneckProbability);
|
|
48
|
+
const outlook = this.classifyOutlook(temporalReport, preAntiPatterns);
|
|
49
|
+
const headline = this.generateHeadline(outlook, preAntiPatterns, modules);
|
|
50
|
+
const topRisks = this.identifyTopRisks(modules, preAntiPatterns);
|
|
51
|
+
const recommendations = this.generateRecommendations(modules, preAntiPatterns);
|
|
52
|
+
return {
|
|
53
|
+
projectPath: gitReport.projectPath,
|
|
54
|
+
generatedAt: new Date().toISOString(),
|
|
55
|
+
overallOutlook: outlook,
|
|
56
|
+
headline,
|
|
57
|
+
modules,
|
|
58
|
+
preAntiPatterns: preAntiPatterns.sort((a, b) => a.weeksToThreshold - b.weeksToThreshold),
|
|
59
|
+
topRisks,
|
|
60
|
+
recommendations,
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
// ── Pre-Anti-Pattern Detection ──
|
|
64
|
+
detectPreAntiPatterns(module, temporal, couplings) {
|
|
65
|
+
const patterns = [];
|
|
66
|
+
patterns.push(...this.detectEmergingGodClass(module, temporal));
|
|
67
|
+
patterns.push(...this.detectEmergingShotgunSurgery(module, couplings));
|
|
68
|
+
patterns.push(...this.detectBusFactorRisk(module, temporal));
|
|
69
|
+
patterns.push(...this.detectComplexitySpiral(module, temporal));
|
|
70
|
+
patterns.push(...this.detectCouplingMagnet(module, couplings, temporal));
|
|
71
|
+
return patterns;
|
|
72
|
+
}
|
|
73
|
+
detectEmergingGodClass(module, temporal) {
|
|
74
|
+
const patterns = [];
|
|
75
|
+
for (const file of module.files) {
|
|
76
|
+
if (file.churnRate < this.config.godClassChurnThreshold)
|
|
77
|
+
continue;
|
|
78
|
+
if (temporal.velocity.churnTrend <= 0)
|
|
79
|
+
continue;
|
|
80
|
+
// Project when churn will exceed critical threshold
|
|
81
|
+
const weeklyGrowth = file.churnRate * (temporal.velocity.churnTrend / 100) / 4;
|
|
82
|
+
const criticalChurn = this.config.godClassChurnThreshold * 2;
|
|
83
|
+
const weeksToThreshold = weeklyGrowth > 0
|
|
84
|
+
? Math.ceil((criticalChurn - file.churnRate) / weeklyGrowth)
|
|
85
|
+
: Infinity;
|
|
86
|
+
if (weeksToThreshold <= this.config.forecastWeeks) {
|
|
87
|
+
patterns.push({
|
|
88
|
+
type: 'emerging-god-class',
|
|
89
|
+
module: module.modulePath,
|
|
90
|
+
severity: weeksToThreshold <= 8 ? 'alert' : weeksToThreshold <= 16 ? 'warning' : 'watch',
|
|
91
|
+
currentScore: temporal.staticScore,
|
|
92
|
+
projectedScore: temporal.projectedScore,
|
|
93
|
+
weeksToThreshold,
|
|
94
|
+
threshold: criticalChurn,
|
|
95
|
+
description: `File '${file.path}' has churn rate ${Math.round(file.churnRate)} lines/commit and growing ${Math.round(temporal.velocity.churnTrend)}%`,
|
|
96
|
+
evidence: [
|
|
97
|
+
`Current churn: ${Math.round(file.churnRate)} lines/commit`,
|
|
98
|
+
`Growth rate: ${Math.round(temporal.velocity.churnTrend)}%`,
|
|
99
|
+
`${file.commits} commits in analysis period`,
|
|
100
|
+
`${file.authors.size} contributor(s)`,
|
|
101
|
+
],
|
|
102
|
+
recommendation: 'Split into smaller, focused modules before complexity makes refactoring prohibitively expensive.',
|
|
103
|
+
confidence: Math.min(0.9, 0.5 + (file.commits / 50)),
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return patterns;
|
|
108
|
+
}
|
|
109
|
+
detectEmergingShotgunSurgery(module, couplings) {
|
|
110
|
+
// Count how many files this module is coupled with
|
|
111
|
+
const moduleCouplings = couplings.filter(c => c.fileA.startsWith(module.modulePath) || c.fileB.startsWith(module.modulePath));
|
|
112
|
+
if (moduleCouplings.length < this.config.shotgunCouplingThreshold)
|
|
113
|
+
return [];
|
|
114
|
+
const avgConfidence = moduleCouplings.reduce((s, c) => s + c.confidence, 0) / moduleCouplings.length;
|
|
115
|
+
return [{
|
|
116
|
+
type: 'emerging-shotgun-surgery',
|
|
117
|
+
module: module.modulePath,
|
|
118
|
+
severity: moduleCouplings.length > 10 ? 'alert' : 'warning',
|
|
119
|
+
currentScore: 0,
|
|
120
|
+
projectedScore: 0,
|
|
121
|
+
weeksToThreshold: 4,
|
|
122
|
+
threshold: this.config.shotgunCouplingThreshold,
|
|
123
|
+
description: `Module '${module.modulePath}' has ${moduleCouplings.length} change-coupled files — changes here ripple across the codebase`,
|
|
124
|
+
evidence: [
|
|
125
|
+
`${moduleCouplings.length} files change together with this module`,
|
|
126
|
+
`Average coupling confidence: ${Math.round(avgConfidence * 100)}%`,
|
|
127
|
+
...moduleCouplings.slice(0, 3).map(c => `${c.fileA} ↔ ${c.fileB} (${c.cochangeCount} co-changes)`),
|
|
128
|
+
],
|
|
129
|
+
recommendation: 'Extract shared concerns into a dedicated module. Introduce interfaces to decouple.',
|
|
130
|
+
confidence: avgConfidence,
|
|
131
|
+
}];
|
|
132
|
+
}
|
|
133
|
+
detectBusFactorRisk(module, temporal) {
|
|
134
|
+
if (module.busFactor > this.config.busFatorRiskThreshold)
|
|
135
|
+
return [];
|
|
136
|
+
if (module.aggregateCommits < 5)
|
|
137
|
+
return []; // too few commits to judge
|
|
138
|
+
return [{
|
|
139
|
+
type: 'bus-factor-risk',
|
|
140
|
+
module: module.modulePath,
|
|
141
|
+
severity: temporal.trend === 'degrading' ? 'alert' : 'warning',
|
|
142
|
+
currentScore: temporal.staticScore,
|
|
143
|
+
projectedScore: temporal.projectedScore,
|
|
144
|
+
weeksToThreshold: this.config.forecastWeeks,
|
|
145
|
+
threshold: 2,
|
|
146
|
+
description: `Module '${module.modulePath}' has bus factor of ${module.busFactor} — all knowledge in one person`,
|
|
147
|
+
evidence: [
|
|
148
|
+
`Only ${module.busFactor} contributor(s)`,
|
|
149
|
+
`${module.aggregateCommits} total commits`,
|
|
150
|
+
`${module.files.length} files in module`,
|
|
151
|
+
],
|
|
152
|
+
recommendation: 'Pair programming or code review rotation to spread knowledge. Document critical decisions.',
|
|
153
|
+
confidence: 0.8,
|
|
154
|
+
}];
|
|
155
|
+
}
|
|
156
|
+
detectComplexitySpiral(module, temporal) {
|
|
157
|
+
if (temporal.velocity.churnTrend <= 20)
|
|
158
|
+
return [];
|
|
159
|
+
if (temporal.velocity.direction !== 'accelerating')
|
|
160
|
+
return [];
|
|
161
|
+
// Accelerating churn + increasing commit rate = complexity spiral
|
|
162
|
+
const weeklyScoreDecay = (temporal.staticScore - temporal.projectedScore) / temporal.projectionWeeks;
|
|
163
|
+
const weeksToThreshold = weeklyScoreDecay > 0
|
|
164
|
+
? Math.ceil((temporal.temporalScore - this.config.antiPatternThreshold) / weeklyScoreDecay)
|
|
165
|
+
: Infinity;
|
|
166
|
+
if (weeksToThreshold > this.config.forecastWeeks)
|
|
167
|
+
return [];
|
|
168
|
+
return [{
|
|
169
|
+
type: 'complexity-spiral',
|
|
170
|
+
module: module.modulePath,
|
|
171
|
+
severity: weeksToThreshold <= 8 ? 'alert' : 'warning',
|
|
172
|
+
currentScore: temporal.temporalScore,
|
|
173
|
+
projectedScore: temporal.projectedScore,
|
|
174
|
+
weeksToThreshold,
|
|
175
|
+
threshold: this.config.antiPatternThreshold,
|
|
176
|
+
description: `Module '${module.modulePath}' is in a complexity spiral — accelerating churn with increasing commit frequency`,
|
|
177
|
+
evidence: [
|
|
178
|
+
`Churn trend: +${Math.round(temporal.velocity.churnTrend)}%`,
|
|
179
|
+
`Commit acceleration: +${Math.round(temporal.velocity.commitAcceleration)}%`,
|
|
180
|
+
`Current temporal score: ${temporal.temporalScore}/100`,
|
|
181
|
+
`Projected score in ${temporal.projectionWeeks} weeks: ${temporal.projectedScore}/100`,
|
|
182
|
+
],
|
|
183
|
+
recommendation: 'Stop adding features to this module. Invest in refactoring and test coverage first.',
|
|
184
|
+
confidence: temporal.projectionConfidence,
|
|
185
|
+
}];
|
|
186
|
+
}
|
|
187
|
+
detectCouplingMagnet(module, couplings, temporal) {
|
|
188
|
+
// Files that are increasingly coupled with many others
|
|
189
|
+
const inboundCouplings = couplings.filter(c => c.fileB.startsWith(module.modulePath) && c.confidence > 0.5);
|
|
190
|
+
if (inboundCouplings.length < 3)
|
|
191
|
+
return [];
|
|
192
|
+
if (temporal.velocity.commitAcceleration <= 0)
|
|
193
|
+
return [];
|
|
194
|
+
return [{
|
|
195
|
+
type: 'coupling-magnet',
|
|
196
|
+
module: module.modulePath,
|
|
197
|
+
severity: 'watch',
|
|
198
|
+
currentScore: temporal.staticScore,
|
|
199
|
+
projectedScore: temporal.projectedScore,
|
|
200
|
+
weeksToThreshold: 12,
|
|
201
|
+
threshold: 10,
|
|
202
|
+
description: `Module '${module.modulePath}' is becoming a coupling magnet — ${inboundCouplings.length} high-confidence inbound dependencies`,
|
|
203
|
+
evidence: [
|
|
204
|
+
`${inboundCouplings.length} modules depend on changes here`,
|
|
205
|
+
`Module commit rate accelerating: +${Math.round(temporal.velocity.commitAcceleration)}%`,
|
|
206
|
+
],
|
|
207
|
+
recommendation: 'Extract stable interfaces. Consider the Dependency Inversion Principle to break inbound coupling.',
|
|
208
|
+
confidence: 0.6,
|
|
209
|
+
}];
|
|
210
|
+
}
|
|
211
|
+
// ── Module Forecast ──
|
|
212
|
+
forecastModule(module, temporal, patterns) {
|
|
213
|
+
const health = this.classifyHealth(temporal);
|
|
214
|
+
const forecast6m = this.classify6MonthForecast(temporal, patterns);
|
|
215
|
+
const bottleneckProb = this.calculateBottleneckProbability(temporal, patterns, module);
|
|
216
|
+
const riskFactors = [];
|
|
217
|
+
if (temporal.trend === 'degrading')
|
|
218
|
+
riskFactors.push('Score degrading');
|
|
219
|
+
if (module.busFactor <= 1)
|
|
220
|
+
riskFactors.push('Single contributor');
|
|
221
|
+
if (temporal.velocity.churnTrend > 30)
|
|
222
|
+
riskFactors.push('Churn increasing');
|
|
223
|
+
if (patterns.length > 0)
|
|
224
|
+
riskFactors.push(`${patterns.length} pre-anti-pattern(s)`);
|
|
225
|
+
const topAction = patterns.length > 0
|
|
226
|
+
? patterns[0].recommendation
|
|
227
|
+
: temporal.trend === 'degrading'
|
|
228
|
+
? 'Review recent changes and stabilize'
|
|
229
|
+
: 'No action needed';
|
|
230
|
+
return {
|
|
231
|
+
module: module.modulePath,
|
|
232
|
+
currentHealth: health,
|
|
233
|
+
forecast6Months: forecast6m,
|
|
234
|
+
preAntiPatterns: patterns,
|
|
235
|
+
bottleneckProbability: bottleneckProb,
|
|
236
|
+
riskFactors,
|
|
237
|
+
topAction,
|
|
238
|
+
};
|
|
239
|
+
}
|
|
240
|
+
classifyHealth(temporal) {
|
|
241
|
+
if (temporal.temporalScore < 30)
|
|
242
|
+
return 'critical';
|
|
243
|
+
if (temporal.temporalScore < 50 || temporal.trend === 'degrading')
|
|
244
|
+
return 'degrading';
|
|
245
|
+
if (temporal.temporalScore < 70)
|
|
246
|
+
return 'at-risk';
|
|
247
|
+
return 'healthy';
|
|
248
|
+
}
|
|
249
|
+
classify6MonthForecast(temporal, patterns) {
|
|
250
|
+
const alerts = patterns.filter(p => p.severity === 'alert');
|
|
251
|
+
if (alerts.length > 0 || temporal.projectedScore < 30)
|
|
252
|
+
return 'breakdown';
|
|
253
|
+
if (temporal.trend === 'degrading' || patterns.length > 0)
|
|
254
|
+
return 'declining';
|
|
255
|
+
return 'stable';
|
|
256
|
+
}
|
|
257
|
+
calculateBottleneckProbability(temporal, patterns, module) {
|
|
258
|
+
let prob = 0;
|
|
259
|
+
// Low score → higher probability
|
|
260
|
+
if (temporal.temporalScore < 50)
|
|
261
|
+
prob += 0.3;
|
|
262
|
+
else if (temporal.temporalScore < 70)
|
|
263
|
+
prob += 0.1;
|
|
264
|
+
// Degrading trend
|
|
265
|
+
if (temporal.trend === 'degrading')
|
|
266
|
+
prob += 0.2;
|
|
267
|
+
// Pre-anti-patterns
|
|
268
|
+
prob += Math.min(0.3, patterns.length * 0.1);
|
|
269
|
+
// Low bus factor
|
|
270
|
+
if (module.busFactor <= 1)
|
|
271
|
+
prob += 0.1;
|
|
272
|
+
// High churn
|
|
273
|
+
if (temporal.velocity.churnTrend > 30)
|
|
274
|
+
prob += 0.1;
|
|
275
|
+
return Math.min(1, Math.round(prob * 100) / 100);
|
|
276
|
+
}
|
|
277
|
+
// ── Overall Analysis ──
|
|
278
|
+
classifyOutlook(temporal, patterns) {
|
|
279
|
+
const alerts = patterns.filter(p => p.severity === 'alert');
|
|
280
|
+
if (alerts.length >= 2 || temporal.overallTrend === 'degrading')
|
|
281
|
+
return 'stormy';
|
|
282
|
+
if (alerts.length >= 1 || patterns.length >= 3)
|
|
283
|
+
return 'cloudy';
|
|
284
|
+
return 'sunny';
|
|
285
|
+
}
|
|
286
|
+
generateHeadline(outlook, patterns, modules) {
|
|
287
|
+
const critical = modules.filter(m => m.currentHealth === 'critical').length;
|
|
288
|
+
const degrading = modules.filter(m => m.currentHealth === 'degrading').length;
|
|
289
|
+
switch (outlook) {
|
|
290
|
+
case 'stormy':
|
|
291
|
+
return `${critical + degrading} module(s) at risk. ${patterns.length} pre-anti-pattern(s) detected. Immediate action recommended.`;
|
|
292
|
+
case 'cloudy':
|
|
293
|
+
return `Architecture trending stable with ${patterns.length} emerging concern(s). Proactive refactoring recommended.`;
|
|
294
|
+
case 'sunny':
|
|
295
|
+
return 'Architecture is healthy and stable. Continue current practices.';
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
identifyTopRisks(modules, patterns) {
|
|
299
|
+
const risks = [];
|
|
300
|
+
const breakdowns = modules.filter(m => m.forecast6Months === 'breakdown');
|
|
301
|
+
if (breakdowns.length > 0) {
|
|
302
|
+
risks.push(`${breakdowns.length} module(s) projected to break down within 6 months: ${breakdowns.map(m => m.module).join(', ')}`);
|
|
303
|
+
}
|
|
304
|
+
const busRisks = patterns.filter(p => p.type === 'bus-factor-risk');
|
|
305
|
+
if (busRisks.length > 0) {
|
|
306
|
+
risks.push(`Bus factor risk in ${busRisks.length} module(s) — knowledge concentrated in single contributors`);
|
|
307
|
+
}
|
|
308
|
+
const spirals = patterns.filter(p => p.type === 'complexity-spiral');
|
|
309
|
+
if (spirals.length > 0) {
|
|
310
|
+
risks.push(`Complexity spiral detected in: ${spirals.map(p => p.module).join(', ')}`);
|
|
311
|
+
}
|
|
312
|
+
return risks.slice(0, 5);
|
|
313
|
+
}
|
|
314
|
+
generateRecommendations(modules, patterns) {
|
|
315
|
+
const recs = [];
|
|
316
|
+
const critical = modules.filter(m => m.currentHealth === 'critical');
|
|
317
|
+
if (critical.length > 0) {
|
|
318
|
+
recs.push(`Immediate: Stabilize ${critical.map(m => m.module).join(', ')} — freeze features, invest in refactoring`);
|
|
319
|
+
}
|
|
320
|
+
const godClasses = patterns.filter(p => p.type === 'emerging-god-class');
|
|
321
|
+
if (godClasses.length > 0) {
|
|
322
|
+
recs.push(`Split growing files before they become god classes: ${godClasses.map(p => p.module).join(', ')}`);
|
|
323
|
+
}
|
|
324
|
+
const shotgun = patterns.filter(p => p.type === 'emerging-shotgun-surgery');
|
|
325
|
+
if (shotgun.length > 0) {
|
|
326
|
+
recs.push(`Decouple modules with high change coupling to prevent shotgun surgery`);
|
|
327
|
+
}
|
|
328
|
+
const busRisks = patterns.filter(p => p.type === 'bus-factor-risk');
|
|
329
|
+
if (busRisks.length > 0) {
|
|
330
|
+
recs.push(`Spread knowledge: pair programming or rotation for ${busRisks.map(p => p.module).join(', ')}`);
|
|
331
|
+
}
|
|
332
|
+
if (recs.length === 0) {
|
|
333
|
+
recs.push('Architecture is healthy. Continue monitoring temporal trends.');
|
|
334
|
+
}
|
|
335
|
+
return recs.slice(0, 5);
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
//# sourceMappingURL=forecast.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"forecast.js","sourceRoot":"","sources":["../../../../src/core/analyzers/forecast.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Architect v4.0 Analyzers — Temporal & Predictive
|
|
3
|
+
*/
|
|
4
|
+
export { GitHistoryAnalyzer } from '../../infrastructure/git-history.js';
|
|
5
|
+
export type { GitCommit, FileChange, FileHistory, ModuleHistory, VelocityVector, ChangeCoupling, GitHistoryReport, WeeklySnapshot, GitAnalyzerConfig, } from '../../infrastructure/git-history.js';
|
|
6
|
+
export { TemporalScorer } from './temporal-scorer.js';
|
|
7
|
+
export type { Trend, TemporalScore, TemporalReport, TemporalScorerConfig, } from './temporal-scorer.js';
|
|
8
|
+
export { ForecastEngine } from './forecast.js';
|
|
9
|
+
export type { PreAntiPatternType, PreAntiPattern, ModuleForecast, WeatherForecast, ForecastConfig, } from './forecast.js';
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Architect v4.0 Analyzers — Temporal & Predictive
|
|
3
|
+
*/
|
|
4
|
+
export { GitHistoryAnalyzer } from '../../infrastructure/git-history.js';
|
|
5
|
+
export { TemporalScorer } from './temporal-scorer.js';
|
|
6
|
+
export { ForecastEngine } from './forecast.js';
|
|
7
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/core/analyzers/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,kBAAkB,EAAE,MAAM,qCAAqC,CAAC;AAazE,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AAQtD,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC"}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Temporal Score Dimension — Adds time-series awareness to architecture scoring
|
|
3
|
+
*
|
|
4
|
+
* Combines current static score with historical velocity to produce:
|
|
5
|
+
* - Trend per module (improving / stable / degrading)
|
|
6
|
+
* - Temporal risk score (static score penalized by negative velocity)
|
|
7
|
+
* - Projected score in N weeks
|
|
8
|
+
*
|
|
9
|
+
* @author Camilo Girardelli — Girardelli Tecnologia
|
|
10
|
+
* @license MIT
|
|
11
|
+
*/
|
|
12
|
+
import type { VelocityVector, GitHistoryReport } from '../../infrastructure/git-history.js';
|
|
13
|
+
export type Trend = 'improving' | 'stable' | 'degrading';
|
|
14
|
+
export interface TemporalScore {
|
|
15
|
+
/** Module or file path */
|
|
16
|
+
module: string;
|
|
17
|
+
/** Current static score (from ArchitectureScorer) */
|
|
18
|
+
staticScore: number;
|
|
19
|
+
/** Temporal-adjusted score (penalizes degrading trends) */
|
|
20
|
+
temporalScore: number;
|
|
21
|
+
/** Direction of change */
|
|
22
|
+
trend: Trend;
|
|
23
|
+
/** Projected score in projectionWeeks */
|
|
24
|
+
projectedScore: number;
|
|
25
|
+
/** Confidence in the projection (0-1) */
|
|
26
|
+
projectionConfidence: number;
|
|
27
|
+
/** Weeks used for projection */
|
|
28
|
+
projectionWeeks: number;
|
|
29
|
+
/** Risk level derived from temporal analysis */
|
|
30
|
+
riskLevel: 'low' | 'medium' | 'high' | 'critical';
|
|
31
|
+
/** Velocity data */
|
|
32
|
+
velocity: VelocityVector;
|
|
33
|
+
}
|
|
34
|
+
export interface TemporalReport {
|
|
35
|
+
projectPath: string;
|
|
36
|
+
analyzedAt: string;
|
|
37
|
+
overallTrend: Trend;
|
|
38
|
+
overallTemporalScore: number;
|
|
39
|
+
modules: TemporalScore[];
|
|
40
|
+
degradingModules: TemporalScore[];
|
|
41
|
+
improvingModules: TemporalScore[];
|
|
42
|
+
}
|
|
43
|
+
export interface TemporalScorerConfig {
|
|
44
|
+
/** Weeks ahead to project (default: 12) */
|
|
45
|
+
projectionWeeks?: number;
|
|
46
|
+
/** Weight of churn trend in temporal penalty (0-1, default: 0.6) */
|
|
47
|
+
churnWeight?: number;
|
|
48
|
+
/** Weight of commit acceleration in temporal penalty (0-1, default: 0.4) */
|
|
49
|
+
commitWeight?: number;
|
|
50
|
+
/** Threshold for trend classification: accelerating if > threshold % */
|
|
51
|
+
acceleratingThreshold?: number;
|
|
52
|
+
/** Threshold for trend classification: decelerating if < -threshold % */
|
|
53
|
+
deceleratingThreshold?: number;
|
|
54
|
+
}
|
|
55
|
+
export declare class TemporalScorer {
|
|
56
|
+
private config;
|
|
57
|
+
constructor(config?: TemporalScorerConfig);
|
|
58
|
+
/**
|
|
59
|
+
* Score modules temporally using git history + static scores.
|
|
60
|
+
*
|
|
61
|
+
* @param gitReport - Output from GitHistoryAnalyzer
|
|
62
|
+
* @param staticScores - Map of modulePath → static score (0-100)
|
|
63
|
+
*/
|
|
64
|
+
score(gitReport: GitHistoryReport, staticScores: Map<string, number>): TemporalReport;
|
|
65
|
+
private scoreModule;
|
|
66
|
+
private classifyTrend;
|
|
67
|
+
private classifyRisk;
|
|
68
|
+
private classifyOverallTrend;
|
|
69
|
+
/** Infer a static score when none is provided (based on churn metrics) */
|
|
70
|
+
private inferStaticScore;
|
|
71
|
+
}
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Temporal Score Dimension — Adds time-series awareness to architecture scoring
|
|
3
|
+
*
|
|
4
|
+
* Combines current static score with historical velocity to produce:
|
|
5
|
+
* - Trend per module (improving / stable / degrading)
|
|
6
|
+
* - Temporal risk score (static score penalized by negative velocity)
|
|
7
|
+
* - Projected score in N weeks
|
|
8
|
+
*
|
|
9
|
+
* @author Camilo Girardelli — Girardelli Tecnologia
|
|
10
|
+
* @license MIT
|
|
11
|
+
*/
|
|
12
|
+
const DEFAULT_CONFIG = {
|
|
13
|
+
projectionWeeks: 12,
|
|
14
|
+
churnWeight: 0.6,
|
|
15
|
+
commitWeight: 0.4,
|
|
16
|
+
acceleratingThreshold: 15,
|
|
17
|
+
deceleratingThreshold: -15,
|
|
18
|
+
};
|
|
19
|
+
// ═══════════════════════════════════════════════════════════════
|
|
20
|
+
// TEMPORAL SCORER
|
|
21
|
+
// ═══════════════════════════════════════════════════════════════
|
|
22
|
+
export class TemporalScorer {
|
|
23
|
+
config;
|
|
24
|
+
constructor(config) {
|
|
25
|
+
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Score modules temporally using git history + static scores.
|
|
29
|
+
*
|
|
30
|
+
* @param gitReport - Output from GitHistoryAnalyzer
|
|
31
|
+
* @param staticScores - Map of modulePath → static score (0-100)
|
|
32
|
+
*/
|
|
33
|
+
score(gitReport, staticScores) {
|
|
34
|
+
const modules = [];
|
|
35
|
+
for (const moduleHistory of gitReport.modules) {
|
|
36
|
+
const staticScore = staticScores.get(moduleHistory.modulePath)
|
|
37
|
+
?? this.inferStaticScore(moduleHistory);
|
|
38
|
+
const ts = this.scoreModule(moduleHistory, staticScore);
|
|
39
|
+
modules.push(ts);
|
|
40
|
+
}
|
|
41
|
+
// Sort by risk (worst first)
|
|
42
|
+
modules.sort((a, b) => a.temporalScore - b.temporalScore);
|
|
43
|
+
const degrading = modules.filter(m => m.trend === 'degrading');
|
|
44
|
+
const improving = modules.filter(m => m.trend === 'improving');
|
|
45
|
+
const overallTrend = this.classifyOverallTrend(modules);
|
|
46
|
+
const overallScore = modules.length > 0
|
|
47
|
+
? Math.round(modules.reduce((s, m) => s + m.temporalScore, 0) / modules.length)
|
|
48
|
+
: 0;
|
|
49
|
+
return {
|
|
50
|
+
projectPath: gitReport.projectPath,
|
|
51
|
+
analyzedAt: new Date().toISOString(),
|
|
52
|
+
overallTrend: overallTrend,
|
|
53
|
+
overallTemporalScore: overallScore,
|
|
54
|
+
modules,
|
|
55
|
+
degradingModules: degrading,
|
|
56
|
+
improvingModules: improving,
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
scoreModule(module, staticScore) {
|
|
60
|
+
const velocity = module.velocityVector;
|
|
61
|
+
// Calculate temporal penalty based on velocity
|
|
62
|
+
const churnPenalty = velocity.churnTrend > 0
|
|
63
|
+
? velocity.churnTrend * this.config.churnWeight * 0.3 // 30% impact per 100% churn increase
|
|
64
|
+
: velocity.churnTrend * this.config.churnWeight * 0.1; // 10% bonus per 100% churn decrease
|
|
65
|
+
const commitPenalty = velocity.commitAcceleration > 20
|
|
66
|
+
? (velocity.commitAcceleration - 20) * this.config.commitWeight * 0.2 // penalty for excessive churn
|
|
67
|
+
: 0;
|
|
68
|
+
const totalPenalty = Math.max(-20, Math.min(30, churnPenalty + commitPenalty));
|
|
69
|
+
const temporalScore = Math.max(0, Math.min(100, Math.round(staticScore - totalPenalty)));
|
|
70
|
+
// Trend classification
|
|
71
|
+
const trend = this.classifyTrend(velocity);
|
|
72
|
+
// Linear projection
|
|
73
|
+
const weeklyDelta = totalPenalty / Math.max(this.config.projectionWeeks, 1);
|
|
74
|
+
const projectedScore = Math.max(0, Math.min(100, Math.round(temporalScore - (weeklyDelta * this.config.projectionWeeks))));
|
|
75
|
+
// Confidence decreases with projection distance and instability
|
|
76
|
+
const instability = Math.abs(velocity.churnTrend) + Math.abs(velocity.commitAcceleration);
|
|
77
|
+
const projectionConfidence = Math.max(0.1, Math.min(1, 1 - (instability / 200) - (this.config.projectionWeeks / 52)));
|
|
78
|
+
const riskLevel = this.classifyRisk(temporalScore, trend, module.busFactor);
|
|
79
|
+
return {
|
|
80
|
+
module: module.modulePath,
|
|
81
|
+
staticScore,
|
|
82
|
+
temporalScore,
|
|
83
|
+
trend,
|
|
84
|
+
projectedScore,
|
|
85
|
+
projectionConfidence: Math.round(projectionConfidence * 100) / 100,
|
|
86
|
+
projectionWeeks: this.config.projectionWeeks,
|
|
87
|
+
riskLevel,
|
|
88
|
+
velocity,
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
classifyTrend(velocity) {
|
|
92
|
+
// Degrading: churn increasing significantly or commit acceleration very high
|
|
93
|
+
if (velocity.churnTrend > 30 || velocity.commitAcceleration > 50) {
|
|
94
|
+
return 'degrading';
|
|
95
|
+
}
|
|
96
|
+
// Improving: churn decreasing and stable or decelerating
|
|
97
|
+
if (velocity.churnTrend < -10 && velocity.direction !== 'accelerating') {
|
|
98
|
+
return 'improving';
|
|
99
|
+
}
|
|
100
|
+
return 'stable';
|
|
101
|
+
}
|
|
102
|
+
classifyRisk(temporalScore, trend, busFactor) {
|
|
103
|
+
if (temporalScore < 30 || (temporalScore < 50 && trend === 'degrading')) {
|
|
104
|
+
return 'critical';
|
|
105
|
+
}
|
|
106
|
+
if (temporalScore < 50 || (trend === 'degrading' && busFactor <= 1)) {
|
|
107
|
+
return 'high';
|
|
108
|
+
}
|
|
109
|
+
if (temporalScore < 70 || trend === 'degrading') {
|
|
110
|
+
return 'medium';
|
|
111
|
+
}
|
|
112
|
+
return 'low';
|
|
113
|
+
}
|
|
114
|
+
classifyOverallTrend(modules) {
|
|
115
|
+
if (modules.length === 0)
|
|
116
|
+
return 'stable';
|
|
117
|
+
const degrading = modules.filter(m => m.trend === 'degrading').length;
|
|
118
|
+
const improving = modules.filter(m => m.trend === 'improving').length;
|
|
119
|
+
const degradingRatio = degrading / modules.length;
|
|
120
|
+
const improvingRatio = improving / modules.length;
|
|
121
|
+
if (degradingRatio > 0.3)
|
|
122
|
+
return 'degrading';
|
|
123
|
+
if (improvingRatio > 0.3)
|
|
124
|
+
return 'improving';
|
|
125
|
+
return 'stable';
|
|
126
|
+
}
|
|
127
|
+
/** Infer a static score when none is provided (based on churn metrics) */
|
|
128
|
+
inferStaticScore(module) {
|
|
129
|
+
const avgChurn = module.aggregateChurn / Math.max(module.aggregateCommits, 1);
|
|
130
|
+
if (avgChurn < 20)
|
|
131
|
+
return 85;
|
|
132
|
+
if (avgChurn < 50)
|
|
133
|
+
return 75;
|
|
134
|
+
if (avgChurn < 100)
|
|
135
|
+
return 65;
|
|
136
|
+
if (avgChurn < 200)
|
|
137
|
+
return 50;
|
|
138
|
+
return 35;
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
//# sourceMappingURL=temporal-scorer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"temporal-scorer.js","sourceRoot":"","sources":["../../../../src/core/analyzers/temporal-scorer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { AntiPattern, ArchitectConfig } from './types/core.js';
|
|
2
|
+
import { FileNode } from './types/infrastructure.js';
|
|
3
|
+
import type { CustomAntiPatternDetector } from './types/plugin.js';
|
|
4
|
+
export declare class AntiPatternDetector {
|
|
5
|
+
private config;
|
|
6
|
+
private dependencyGraph;
|
|
7
|
+
/** Paths that indicate third-party or build artifacts — never report anti-patterns here */
|
|
8
|
+
private static readonly EXCLUDED_PATH_SEGMENTS;
|
|
9
|
+
private customDetectors;
|
|
10
|
+
private pluginContext?;
|
|
11
|
+
constructor(config: ArchitectConfig);
|
|
12
|
+
setCustomDetectors(detectors: CustomAntiPatternDetector[]): void;
|
|
13
|
+
/**
|
|
14
|
+
* Check if a file path belongs to the project's own source code.
|
|
15
|
+
* Returns false for node_modules, dist, build artifacts, etc.
|
|
16
|
+
*/
|
|
17
|
+
private isProjectFile;
|
|
18
|
+
detect(fileTree: FileNode, dependencies: Map<string, Set<string>>): Promise<AntiPattern[]>;
|
|
19
|
+
private detectGodClasses;
|
|
20
|
+
private detectCircularDependencies;
|
|
21
|
+
private findCycle;
|
|
22
|
+
private detectLeakyAbstractions;
|
|
23
|
+
private detectFeatureEnvy;
|
|
24
|
+
private detectShotgunSurgery;
|
|
25
|
+
private countMethods;
|
|
26
|
+
private countInternalExports;
|
|
27
|
+
private walkFileTree;
|
|
28
|
+
}
|