@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,306 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tests for TemporalScorer
|
|
3
|
+
*
|
|
4
|
+
* Validates trend classification, temporal score penalties,
|
|
5
|
+
* projection, risk levels, and overall trend aggregation.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { TemporalScorer } from '../src/core/analyzers/temporal-scorer.js';
|
|
9
|
+
import type {
|
|
10
|
+
GitHistoryReport,
|
|
11
|
+
ModuleHistory,
|
|
12
|
+
VelocityVector,
|
|
13
|
+
FileHistory,
|
|
14
|
+
} from '../src/infrastructure/git-history.js';
|
|
15
|
+
|
|
16
|
+
// ═══════════════════════════════════════════════════════════════
|
|
17
|
+
// TEST HELPERS
|
|
18
|
+
// ═══════════════════════════════════════════════════════════════
|
|
19
|
+
|
|
20
|
+
function makeFileHistory(overrides: Partial<FileHistory> = {}): FileHistory {
|
|
21
|
+
return {
|
|
22
|
+
path: 'test.ts',
|
|
23
|
+
commits: 10,
|
|
24
|
+
totalAdditions: 200,
|
|
25
|
+
totalDeletions: 50,
|
|
26
|
+
churnRate: 25,
|
|
27
|
+
authors: new Set(['alice']),
|
|
28
|
+
busFactor: 1,
|
|
29
|
+
lastModified: new Date(),
|
|
30
|
+
weeklyCommitRate: 2,
|
|
31
|
+
isHotspot: false,
|
|
32
|
+
...overrides,
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function makeVelocity(overrides: Partial<VelocityVector> = {}): VelocityVector {
|
|
37
|
+
return {
|
|
38
|
+
commitAcceleration: 0,
|
|
39
|
+
churnTrend: 0,
|
|
40
|
+
direction: 'stable',
|
|
41
|
+
...overrides,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
function makeModule(
|
|
46
|
+
modulePath: string,
|
|
47
|
+
velocity: Partial<VelocityVector> = {},
|
|
48
|
+
overrides: Partial<ModuleHistory> = {},
|
|
49
|
+
): ModuleHistory {
|
|
50
|
+
const files = overrides.files ?? [makeFileHistory()];
|
|
51
|
+
return {
|
|
52
|
+
modulePath,
|
|
53
|
+
files,
|
|
54
|
+
aggregateCommits: files.reduce((s, f) => s + f.commits, 0),
|
|
55
|
+
aggregateChurn: files.reduce((s, f) => s + f.totalAdditions + f.totalDeletions, 0),
|
|
56
|
+
avgWeeklyRate: 2,
|
|
57
|
+
topHotspots: [],
|
|
58
|
+
velocityVector: makeVelocity(velocity),
|
|
59
|
+
busFactor: 2,
|
|
60
|
+
...overrides,
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
function makeGitReport(modules: ModuleHistory[]): GitHistoryReport {
|
|
65
|
+
return {
|
|
66
|
+
projectPath: '/test',
|
|
67
|
+
analyzedAt: new Date().toISOString(),
|
|
68
|
+
periodWeeks: 24,
|
|
69
|
+
totalCommits: 100,
|
|
70
|
+
totalAuthors: 5,
|
|
71
|
+
modules,
|
|
72
|
+
hotspots: [],
|
|
73
|
+
changeCouplings: [],
|
|
74
|
+
commitTimeline: [],
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// ═══════════════════════════════════════════════════════════════
|
|
79
|
+
// TESTS
|
|
80
|
+
// ═══════════════════════════════════════════════════════════════
|
|
81
|
+
|
|
82
|
+
describe('TemporalScorer', () => {
|
|
83
|
+
describe('score()', () => {
|
|
84
|
+
it('should score all modules from git report', () => {
|
|
85
|
+
const scorer = new TemporalScorer();
|
|
86
|
+
const modules = [makeModule('src'), makeModule('lib')];
|
|
87
|
+
const gitReport = makeGitReport(modules);
|
|
88
|
+
const staticScores = new Map([['src', 80], ['lib', 70]]);
|
|
89
|
+
|
|
90
|
+
const report = scorer.score(gitReport, staticScores);
|
|
91
|
+
|
|
92
|
+
expect(report.modules.length).toBe(2);
|
|
93
|
+
expect(report.projectPath).toBe('/test');
|
|
94
|
+
expect(report.analyzedAt).toBeDefined();
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
it('should infer static score when not provided', () => {
|
|
98
|
+
const scorer = new TemporalScorer();
|
|
99
|
+
const modules = [makeModule('unknown-module')];
|
|
100
|
+
const gitReport = makeGitReport(modules);
|
|
101
|
+
const staticScores = new Map<string, number>(); // empty
|
|
102
|
+
|
|
103
|
+
const report = scorer.score(gitReport, staticScores);
|
|
104
|
+
|
|
105
|
+
expect(report.modules[0].staticScore).toBeGreaterThan(0);
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
it('should sort modules by temporal score ascending (worst first)', () => {
|
|
109
|
+
const scorer = new TemporalScorer();
|
|
110
|
+
const modules = [
|
|
111
|
+
makeModule('good', { churnTrend: -20, direction: 'decelerating' }),
|
|
112
|
+
makeModule('bad', { churnTrend: 50, commitAcceleration: 60, direction: 'accelerating' }),
|
|
113
|
+
];
|
|
114
|
+
const gitReport = makeGitReport(modules);
|
|
115
|
+
const staticScores = new Map([['good', 80], ['bad', 80]]);
|
|
116
|
+
|
|
117
|
+
const report = scorer.score(gitReport, staticScores);
|
|
118
|
+
|
|
119
|
+
expect(report.modules[0].temporalScore).toBeLessThanOrEqual(report.modules[1].temporalScore);
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
describe('trend classification', () => {
|
|
124
|
+
it('should classify degrading when churn trend is high', () => {
|
|
125
|
+
const scorer = new TemporalScorer();
|
|
126
|
+
const modules = [makeModule('src', { churnTrend: 40, commitAcceleration: 10, direction: 'stable' })];
|
|
127
|
+
const gitReport = makeGitReport(modules);
|
|
128
|
+
const staticScores = new Map([['src', 70]]);
|
|
129
|
+
|
|
130
|
+
const report = scorer.score(gitReport, staticScores);
|
|
131
|
+
|
|
132
|
+
expect(report.modules[0].trend).toBe('degrading');
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('should classify degrading when commit acceleration is very high', () => {
|
|
136
|
+
const scorer = new TemporalScorer();
|
|
137
|
+
const modules = [makeModule('src', { churnTrend: 5, commitAcceleration: 60, direction: 'accelerating' })];
|
|
138
|
+
const gitReport = makeGitReport(modules);
|
|
139
|
+
const staticScores = new Map([['src', 70]]);
|
|
140
|
+
|
|
141
|
+
const report = scorer.score(gitReport, staticScores);
|
|
142
|
+
|
|
143
|
+
expect(report.modules[0].trend).toBe('degrading');
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
it('should classify improving when churn is decreasing', () => {
|
|
147
|
+
const scorer = new TemporalScorer();
|
|
148
|
+
const modules = [makeModule('src', { churnTrend: -15, commitAcceleration: 0, direction: 'stable' })];
|
|
149
|
+
const gitReport = makeGitReport(modules);
|
|
150
|
+
const staticScores = new Map([['src', 70]]);
|
|
151
|
+
|
|
152
|
+
const report = scorer.score(gitReport, staticScores);
|
|
153
|
+
|
|
154
|
+
expect(report.modules[0].trend).toBe('improving');
|
|
155
|
+
});
|
|
156
|
+
|
|
157
|
+
it('should classify stable for moderate metrics', () => {
|
|
158
|
+
const scorer = new TemporalScorer();
|
|
159
|
+
const modules = [makeModule('src', { churnTrend: 5, commitAcceleration: 5, direction: 'stable' })];
|
|
160
|
+
const gitReport = makeGitReport(modules);
|
|
161
|
+
const staticScores = new Map([['src', 70]]);
|
|
162
|
+
|
|
163
|
+
const report = scorer.score(gitReport, staticScores);
|
|
164
|
+
|
|
165
|
+
expect(report.modules[0].trend).toBe('stable');
|
|
166
|
+
});
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
describe('temporal score calculation', () => {
|
|
170
|
+
it('should penalize degrading modules', () => {
|
|
171
|
+
const scorer = new TemporalScorer();
|
|
172
|
+
const modules = [makeModule('src', { churnTrend: 50, commitAcceleration: 30, direction: 'accelerating' })];
|
|
173
|
+
const gitReport = makeGitReport(modules);
|
|
174
|
+
const staticScores = new Map([['src', 80]]);
|
|
175
|
+
|
|
176
|
+
const report = scorer.score(gitReport, staticScores);
|
|
177
|
+
|
|
178
|
+
expect(report.modules[0].temporalScore).toBeLessThan(80);
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
it('should give slight bonus for improving modules', () => {
|
|
182
|
+
const scorer = new TemporalScorer();
|
|
183
|
+
const modules = [makeModule('src', { churnTrend: -50, commitAcceleration: -10, direction: 'decelerating' })];
|
|
184
|
+
const gitReport = makeGitReport(modules);
|
|
185
|
+
const staticScores = new Map([['src', 70]]);
|
|
186
|
+
|
|
187
|
+
const report = scorer.score(gitReport, staticScores);
|
|
188
|
+
|
|
189
|
+
// The bonus comes from negative churn penalty
|
|
190
|
+
expect(report.modules[0].temporalScore).toBeGreaterThanOrEqual(70);
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
it('should clamp temporal score between 0 and 100', () => {
|
|
194
|
+
const scorer = new TemporalScorer();
|
|
195
|
+
const modules = [makeModule('src', { churnTrend: 500, commitAcceleration: 200, direction: 'accelerating' })];
|
|
196
|
+
const gitReport = makeGitReport(modules);
|
|
197
|
+
const staticScores = new Map([['src', 10]]);
|
|
198
|
+
|
|
199
|
+
const report = scorer.score(gitReport, staticScores);
|
|
200
|
+
|
|
201
|
+
expect(report.modules[0].temporalScore).toBeGreaterThanOrEqual(0);
|
|
202
|
+
expect(report.modules[0].temporalScore).toBeLessThanOrEqual(100);
|
|
203
|
+
});
|
|
204
|
+
});
|
|
205
|
+
|
|
206
|
+
describe('risk classification', () => {
|
|
207
|
+
it('should classify critical for very low scores', () => {
|
|
208
|
+
const scorer = new TemporalScorer();
|
|
209
|
+
const modules = [makeModule('src', { churnTrend: 100, commitAcceleration: 80, direction: 'accelerating' })];
|
|
210
|
+
const gitReport = makeGitReport(modules);
|
|
211
|
+
const staticScores = new Map([['src', 25]]);
|
|
212
|
+
|
|
213
|
+
const report = scorer.score(gitReport, staticScores);
|
|
214
|
+
|
|
215
|
+
expect(report.modules[0].riskLevel).toBe('critical');
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
it('should classify low for healthy modules', () => {
|
|
219
|
+
const scorer = new TemporalScorer();
|
|
220
|
+
const modules = [makeModule('src', { churnTrend: -5, commitAcceleration: 0, direction: 'stable' })];
|
|
221
|
+
const gitReport = makeGitReport(modules);
|
|
222
|
+
const staticScores = new Map([['src', 85]]);
|
|
223
|
+
|
|
224
|
+
const report = scorer.score(gitReport, staticScores);
|
|
225
|
+
|
|
226
|
+
expect(report.modules[0].riskLevel).toBe('low');
|
|
227
|
+
});
|
|
228
|
+
});
|
|
229
|
+
|
|
230
|
+
describe('projection', () => {
|
|
231
|
+
it('should project future score', () => {
|
|
232
|
+
const scorer = new TemporalScorer({ projectionWeeks: 12 });
|
|
233
|
+
const modules = [makeModule('src', { churnTrend: 30, commitAcceleration: 10, direction: 'stable' })];
|
|
234
|
+
const gitReport = makeGitReport(modules);
|
|
235
|
+
const staticScores = new Map([['src', 70]]);
|
|
236
|
+
|
|
237
|
+
const report = scorer.score(gitReport, staticScores);
|
|
238
|
+
|
|
239
|
+
expect(report.modules[0].projectedScore).toBeDefined();
|
|
240
|
+
expect(report.modules[0].projectionWeeks).toBe(12);
|
|
241
|
+
expect(report.modules[0].projectionConfidence).toBeGreaterThan(0);
|
|
242
|
+
expect(report.modules[0].projectionConfidence).toBeLessThanOrEqual(1);
|
|
243
|
+
});
|
|
244
|
+
});
|
|
245
|
+
|
|
246
|
+
describe('overall trend', () => {
|
|
247
|
+
it('should return degrading when >30% modules are degrading', () => {
|
|
248
|
+
const scorer = new TemporalScorer();
|
|
249
|
+
const modules = [
|
|
250
|
+
makeModule('a', { churnTrend: 50, direction: 'accelerating' }),
|
|
251
|
+
makeModule('b', { churnTrend: 50, direction: 'accelerating' }),
|
|
252
|
+
makeModule('c', { churnTrend: 0, direction: 'stable' }),
|
|
253
|
+
];
|
|
254
|
+
const gitReport = makeGitReport(modules);
|
|
255
|
+
const staticScores = new Map([['a', 70], ['b', 70], ['c', 70]]);
|
|
256
|
+
|
|
257
|
+
const report = scorer.score(gitReport, staticScores);
|
|
258
|
+
|
|
259
|
+
expect(report.overallTrend).toBe('degrading');
|
|
260
|
+
});
|
|
261
|
+
|
|
262
|
+
it('should return improving when >30% modules are improving', () => {
|
|
263
|
+
const scorer = new TemporalScorer();
|
|
264
|
+
const modules = [
|
|
265
|
+
makeModule('a', { churnTrend: -20, commitAcceleration: 0, direction: 'stable' }),
|
|
266
|
+
makeModule('b', { churnTrend: -20, commitAcceleration: 0, direction: 'stable' }),
|
|
267
|
+
makeModule('c', { churnTrend: 0, direction: 'stable' }),
|
|
268
|
+
];
|
|
269
|
+
const gitReport = makeGitReport(modules);
|
|
270
|
+
const staticScores = new Map([['a', 70], ['b', 70], ['c', 70]]);
|
|
271
|
+
|
|
272
|
+
const report = scorer.score(gitReport, staticScores);
|
|
273
|
+
|
|
274
|
+
expect(report.overallTrend).toBe('improving');
|
|
275
|
+
});
|
|
276
|
+
|
|
277
|
+
it('should return stable for empty modules', () => {
|
|
278
|
+
const scorer = new TemporalScorer();
|
|
279
|
+
const gitReport = makeGitReport([]);
|
|
280
|
+
const staticScores = new Map<string, number>();
|
|
281
|
+
|
|
282
|
+
const report = scorer.score(gitReport, staticScores);
|
|
283
|
+
|
|
284
|
+
expect(report.overallTrend).toBe('stable');
|
|
285
|
+
expect(report.overallTemporalScore).toBe(0);
|
|
286
|
+
});
|
|
287
|
+
});
|
|
288
|
+
|
|
289
|
+
describe('degrading/improving lists', () => {
|
|
290
|
+
it('should separate degrading and improving modules', () => {
|
|
291
|
+
const scorer = new TemporalScorer();
|
|
292
|
+
const modules = [
|
|
293
|
+
makeModule('bad', { churnTrend: 50, direction: 'accelerating' }),
|
|
294
|
+
makeModule('good', { churnTrend: -20, commitAcceleration: 0, direction: 'decelerating' }),
|
|
295
|
+
makeModule('ok', { churnTrend: 0, direction: 'stable' }),
|
|
296
|
+
];
|
|
297
|
+
const gitReport = makeGitReport(modules);
|
|
298
|
+
const staticScores = new Map([['bad', 70], ['good', 70], ['ok', 70]]);
|
|
299
|
+
|
|
300
|
+
const report = scorer.score(gitReport, staticScores);
|
|
301
|
+
|
|
302
|
+
expect(report.degradingModules.some(m => m.module === 'bad')).toBe(true);
|
|
303
|
+
expect(report.improvingModules.some(m => m.module === 'good')).toBe(true);
|
|
304
|
+
});
|
|
305
|
+
});
|
|
306
|
+
});
|