@aiready/context-analyzer 0.22.9 → 0.22.11
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/.turbo/turbo-build.log +33 -32
- package/.turbo/turbo-format-check.log +1 -1
- package/.turbo/turbo-lint.log +8 -0
- package/dist/chunk-HD4Y3GYL.mjs +124 -0
- package/dist/{chunk-M2EGQ36M.mjs → chunk-ISWPFG2C.mjs} +43 -98
- package/dist/{chunk-BQCISA2F.mjs → chunk-QW6ULRML.mjs} +32 -6
- package/dist/{chunk-WHB7QI7N.mjs → chunk-T733LS62.mjs} +23 -4
- package/dist/{cli-action-CXIHOVAC.mjs → cli-action-FV4G4OB6.mjs} +3 -3
- package/dist/{cli-action-3CWN7PBE.mjs → cli-action-HYSWE7F4.mjs} +3 -3
- package/dist/{cli-action-MLFCIW2O.mjs → cli-action-VIQZSWSP.mjs} +3 -3
- package/dist/{cli-action-E7UGP4KE.mjs → cli-action-W7TESWAV.mjs} +6 -8
- package/dist/cli.js +236 -191
- package/dist/cli.mjs +1 -1
- package/dist/console-report-AP4JYNQY.mjs +74 -0
- package/dist/html-report-ULELSIYG.mjs +73 -0
- package/dist/index.d.mts +69 -337
- package/dist/index.d.ts +69 -337
- package/dist/index.js +90 -452
- package/dist/index.mjs +9 -157
- package/dist/interactive-setup-JGFBFI3M.mjs +75 -0
- package/dist/{orchestrator-3ERQS3NW.mjs → orchestrator-QNE2E4TE.mjs} +2 -2
- package/dist/summary-GQRWW3A2.mjs +7 -0
- package/dist/summary-JTBS7CPM.mjs +7 -0
- package/dist/summary-TZFB6ZFM.mjs +7 -0
- package/package.json +2 -2
- package/src/__tests__/file-classification.fixtures.ts +57 -0
- package/src/__tests__/file-classification.test.ts +24 -929
- package/src/__tests__/provider.test.ts +6 -72
- package/src/classifier.ts +42 -42
- package/src/cli-action.ts +11 -6
- package/src/index.ts +30 -27
- package/src/provider.ts +1 -1
- package/src/summary.ts +65 -10
- package/dist/__tests__/analyzer.test.d.ts +0 -2
- package/dist/__tests__/analyzer.test.d.ts.map +0 -1
- package/dist/__tests__/analyzer.test.js +0 -198
- package/dist/__tests__/analyzer.test.js.map +0 -1
- package/dist/__tests__/auto-detection.test.d.ts +0 -2
- package/dist/__tests__/auto-detection.test.d.ts.map +0 -1
- package/dist/__tests__/auto-detection.test.js +0 -132
- package/dist/__tests__/auto-detection.test.js.map +0 -1
- package/dist/__tests__/cluster-detector.test.d.ts +0 -2
- package/dist/__tests__/cluster-detector.test.d.ts.map +0 -1
- package/dist/__tests__/cluster-detector.test.js +0 -121
- package/dist/__tests__/cluster-detector.test.js.map +0 -1
- package/dist/__tests__/contract.test.d.ts +0 -2
- package/dist/__tests__/contract.test.d.ts.map +0 -1
- package/dist/__tests__/contract.test.js +0 -59
- package/dist/__tests__/contract.test.js.map +0 -1
- package/dist/__tests__/enhanced-cohesion.test.d.ts +0 -2
- package/dist/__tests__/enhanced-cohesion.test.d.ts.map +0 -1
- package/dist/__tests__/enhanced-cohesion.test.js +0 -119
- package/dist/__tests__/enhanced-cohesion.test.js.map +0 -1
- package/dist/__tests__/file-classification.test.d.ts +0 -2
- package/dist/__tests__/file-classification.test.d.ts.map +0 -1
- package/dist/__tests__/file-classification.test.js +0 -749
- package/dist/__tests__/file-classification.test.js.map +0 -1
- package/dist/__tests__/fragmentation-advanced.test.d.ts +0 -2
- package/dist/__tests__/fragmentation-advanced.test.d.ts.map +0 -1
- package/dist/__tests__/fragmentation-advanced.test.js +0 -44
- package/dist/__tests__/fragmentation-advanced.test.js.map +0 -1
- package/dist/__tests__/fragmentation-coupling.test.d.ts +0 -2
- package/dist/__tests__/fragmentation-coupling.test.d.ts.map +0 -1
- package/dist/__tests__/fragmentation-coupling.test.js +0 -52
- package/dist/__tests__/fragmentation-coupling.test.js.map +0 -1
- package/dist/__tests__/fragmentation-log.test.d.ts +0 -2
- package/dist/__tests__/fragmentation-log.test.d.ts.map +0 -1
- package/dist/__tests__/fragmentation-log.test.js +0 -29
- package/dist/__tests__/fragmentation-log.test.js.map +0 -1
- package/dist/__tests__/provider.test.d.ts +0 -2
- package/dist/__tests__/provider.test.d.ts.map +0 -1
- package/dist/__tests__/provider.test.js +0 -72
- package/dist/__tests__/provider.test.js.map +0 -1
- package/dist/__tests__/remediation.test.d.ts +0 -2
- package/dist/__tests__/remediation.test.d.ts.map +0 -1
- package/dist/__tests__/remediation.test.js +0 -61
- package/dist/__tests__/remediation.test.js.map +0 -1
- package/dist/__tests__/scoring.test.d.ts +0 -2
- package/dist/__tests__/scoring.test.d.ts.map +0 -1
- package/dist/__tests__/scoring.test.js +0 -298
- package/dist/__tests__/scoring.test.js.map +0 -1
- package/dist/__tests__/structural-cohesion.test.d.ts +0 -2
- package/dist/__tests__/structural-cohesion.test.d.ts.map +0 -1
- package/dist/__tests__/structural-cohesion.test.js +0 -35
- package/dist/__tests__/structural-cohesion.test.js.map +0 -1
- package/dist/analyzer.d.ts +0 -37
- package/dist/analyzer.d.ts.map +0 -1
- package/dist/analyzer.js +0 -283
- package/dist/analyzer.js.map +0 -1
- package/dist/analyzers/python-context.d.ts +0 -38
- package/dist/analyzers/python-context.d.ts.map +0 -1
- package/dist/analyzers/python-context.js +0 -234
- package/dist/analyzers/python-context.js.map +0 -1
- package/dist/ast-utils.d.ts +0 -16
- package/dist/ast-utils.d.ts.map +0 -1
- package/dist/ast-utils.js +0 -81
- package/dist/ast-utils.js.map +0 -1
- package/dist/chunk-22ZO4EKZ.mjs +0 -1297
- package/dist/chunk-2HE27YEV.mjs +0 -1739
- package/dist/chunk-45P4RDYP.mjs +0 -607
- package/dist/chunk-474DEGWW.mjs +0 -1792
- package/dist/chunk-4SYIJ7CU.mjs +0 -1538
- package/dist/chunk-4U4LDWGF.mjs +0 -360
- package/dist/chunk-4XQVYYPC.mjs +0 -1470
- package/dist/chunk-5CLU3HYU.mjs +0 -1475
- package/dist/chunk-5K73Q3OQ.mjs +0 -1520
- package/dist/chunk-5N5DCJOV.mjs +0 -583
- package/dist/chunk-6AVS4KTM.mjs +0 -1536
- package/dist/chunk-6FQYIG6I.mjs +0 -1298
- package/dist/chunk-6I4552YB.mjs +0 -1467
- package/dist/chunk-6LPITDKG.mjs +0 -1539
- package/dist/chunk-72QC5QUS.mjs +0 -549
- package/dist/chunk-736QSHJP.mjs +0 -1807
- package/dist/chunk-7LUSCLGR.mjs +0 -2058
- package/dist/chunk-7VK3XTSH.mjs +0 -1756
- package/dist/chunk-7ZEJGWLN.mjs +0 -1363
- package/dist/chunk-AECWO7NQ.mjs +0 -1539
- package/dist/chunk-AEK3MZC5.mjs +0 -709
- package/dist/chunk-AJC3FR6G.mjs +0 -1509
- package/dist/chunk-AMPK6SWS.mjs +0 -1754
- package/dist/chunk-BA7QGUHN.mjs +0 -1722
- package/dist/chunk-BCEZGRXI.mjs +0 -1297
- package/dist/chunk-BD4NWUVG.mjs +0 -1242
- package/dist/chunk-BEZPBI5C.mjs +0 -1829
- package/dist/chunk-BHCRDEE4.mjs +0 -1745
- package/dist/chunk-BW463GQB.mjs +0 -1767
- package/dist/chunk-CAX2MOUZ.mjs +0 -1801
- package/dist/chunk-CBWM3EK5.mjs +0 -1854
- package/dist/chunk-CCBNKQYB.mjs +0 -1812
- package/dist/chunk-CDIVYADN.mjs +0 -2110
- package/dist/chunk-CVGIDSMN.mjs +0 -1522
- package/dist/chunk-D25B5LZR.mjs +0 -1739
- package/dist/chunk-D3SIHB2V.mjs +0 -2118
- package/dist/chunk-DD7UVNE3.mjs +0 -678
- package/dist/chunk-DMRZMS2U.mjs +0 -964
- package/dist/chunk-DXG5NIYL.mjs +0 -1527
- package/dist/chunk-EBXG2Q5Y.mjs +0 -2059
- package/dist/chunk-EH3PMNZQ.mjs +0 -569
- package/dist/chunk-EMYD7NS6.mjs +0 -137
- package/dist/chunk-EVX2W2BK.mjs +0 -1896
- package/dist/chunk-EWFR366Y.mjs +0 -1740
- package/dist/chunk-EX7HCWAO.mjs +0 -625
- package/dist/chunk-FNPSK3CG.mjs +0 -1760
- package/dist/chunk-FO6YT6RG.mjs +0 -1751
- package/dist/chunk-FYI56A5M.mjs +0 -1892
- package/dist/chunk-G3CCJCBI.mjs +0 -1521
- package/dist/chunk-G7PO3DNK.mjs +0 -1072
- package/dist/chunk-GFADGYXZ.mjs +0 -1752
- package/dist/chunk-GTRIBVS6.mjs +0 -1467
- package/dist/chunk-GXTGOLZT.mjs +0 -92
- package/dist/chunk-H4HWBQU6.mjs +0 -1530
- package/dist/chunk-HDFXSEFW.mjs +0 -605
- package/dist/chunk-HOUDVRG2.mjs +0 -1422
- package/dist/chunk-HQNHM2X7.mjs +0 -997
- package/dist/chunk-I54HL4FZ.mjs +0 -781
- package/dist/chunk-I77HFFZU.mjs +0 -1876
- package/dist/chunk-IKRP7ECY.mjs +0 -1754
- package/dist/chunk-ILMLGJGI.mjs +0 -1295
- package/dist/chunk-IPIE5TXJ.mjs +0 -1741
- package/dist/chunk-IRWCPDWD.mjs +0 -779
- package/dist/chunk-J3MUOWHC.mjs +0 -1747
- package/dist/chunk-J5TA3AZU.mjs +0 -1795
- package/dist/chunk-JH535NPP.mjs +0 -1619
- package/dist/chunk-JUHHOSHG.mjs +0 -1808
- package/dist/chunk-JZ2SE4DB.mjs +0 -1116
- package/dist/chunk-K2WFOBAZ.mjs +0 -1821
- package/dist/chunk-K6U64EL3.mjs +0 -517
- package/dist/chunk-KDUUZQBK.mjs +0 -1692
- package/dist/chunk-KGFWKSGJ.mjs +0 -1442
- package/dist/chunk-KGVMS4R5.mjs +0 -1750
- package/dist/chunk-KWIS5FQP.mjs +0 -1739
- package/dist/chunk-KYSZF5N6.mjs +0 -1876
- package/dist/chunk-LERPI33Y.mjs +0 -2060
- package/dist/chunk-M64RHH4D.mjs +0 -1896
- package/dist/chunk-MBE4AQP5.mjs +0 -1362
- package/dist/chunk-MR7WXHIE.mjs +0 -1833
- package/dist/chunk-MZP3G7TF.mjs +0 -2118
- package/dist/chunk-N2GQWNFG.mjs +0 -1527
- package/dist/chunk-N6XBOOVA.mjs +0 -564
- package/dist/chunk-NJUW6VED.mjs +0 -610
- package/dist/chunk-NOHK5DLU.mjs +0 -2173
- package/dist/chunk-NQA3F2HJ.mjs +0 -1532
- package/dist/chunk-NXXQ2U73.mjs +0 -1467
- package/dist/chunk-OP4G6GLN.mjs +0 -1876
- package/dist/chunk-ORLC5Y4J.mjs +0 -1787
- package/dist/chunk-OTCQL7DY.mjs +0 -2045
- package/dist/chunk-OUYSZZ7X.mjs +0 -1846
- package/dist/chunk-OZE3FVZT.mjs +0 -1089
- package/dist/chunk-P3T3H27S.mjs +0 -1895
- package/dist/chunk-P5YV5WIX.mjs +0 -1803
- package/dist/chunk-P74BO725.mjs +0 -1296
- package/dist/chunk-PDN74MG3.mjs +0 -1834
- package/dist/chunk-PJD4VCIH.mjs +0 -1722
- package/dist/chunk-PVVCCE6W.mjs +0 -755
- package/dist/chunk-PVVMK56C.mjs +0 -1793
- package/dist/chunk-Q2GDZ2FZ.mjs +0 -1794
- package/dist/chunk-QDGPR3L6.mjs +0 -1518
- package/dist/chunk-RQCIJO5Z.mjs +0 -1116
- package/dist/chunk-RRB2C34Q.mjs +0 -1738
- package/dist/chunk-RYIB5CWD.mjs +0 -781
- package/dist/chunk-S6OPP4L5.mjs +0 -1791
- package/dist/chunk-SAVOSPM3.mjs +0 -1522
- package/dist/chunk-SFK6XTJE.mjs +0 -2110
- package/dist/chunk-SIX4KMF2.mjs +0 -1468
- package/dist/chunk-SPAM2YJE.mjs +0 -1537
- package/dist/chunk-T6ZCOPPI.mjs +0 -538
- package/dist/chunk-TPF75CNP.mjs +0 -581
- package/dist/chunk-TWWPY7FD.mjs +0 -1754
- package/dist/chunk-U5R2FTCR.mjs +0 -1803
- package/dist/chunk-UG7OPVHB.mjs +0 -1521
- package/dist/chunk-UMZTAWDA.mjs +0 -1812
- package/dist/chunk-UU4HZ7ZT.mjs +0 -1849
- package/dist/chunk-UXC6QUZ7.mjs +0 -1801
- package/dist/chunk-VBWXHKGD.mjs +0 -1895
- package/dist/chunk-VIJTZPBI.mjs +0 -1470
- package/dist/chunk-VLV6MXPL.mjs +0 -1750
- package/dist/chunk-VTALAPQZ.mjs +0 -1241
- package/dist/chunk-W2KNBN6W.mjs +0 -1849
- package/dist/chunk-W37E7MW5.mjs +0 -1403
- package/dist/chunk-W76FEISE.mjs +0 -1538
- package/dist/chunk-WCFQYXQA.mjs +0 -1532
- package/dist/chunk-WKOZOHOU.mjs +0 -2060
- package/dist/chunk-WTQJNY4U.mjs +0 -1785
- package/dist/chunk-XBFM2Z4O.mjs +0 -1792
- package/dist/chunk-XIXAWCMS.mjs +0 -1760
- package/dist/chunk-XTAXUNQN.mjs +0 -1742
- package/dist/chunk-XY77XABG.mjs +0 -1545
- package/dist/chunk-XZ645X5U.mjs +0 -1425
- package/dist/chunk-Y6FXYEAI.mjs +0 -10
- package/dist/chunk-YCGDIGOG.mjs +0 -1467
- package/dist/chunk-YYB6NZE3.mjs +0 -1869
- package/dist/chunk-Z5WY6A4P.mjs +0 -1754
- package/dist/chunk-ZAMFWKIB.mjs +0 -1842
- package/dist/classifier.d.ts +0 -114
- package/dist/classifier.d.ts.map +0 -1
- package/dist/classifier.js +0 -439
- package/dist/classifier.js.map +0 -1
- package/dist/cli-action-SA7SCYNV.mjs +0 -95
- package/dist/cli-action-VCXBZGZP.mjs +0 -95
- package/dist/cli-action-YAJOJCXJ.mjs +0 -95
- package/dist/cli.d.ts.map +0 -1
- package/dist/cli.js.map +0 -1
- package/dist/cluster-detector.d.ts +0 -8
- package/dist/cluster-detector.d.ts.map +0 -1
- package/dist/cluster-detector.js +0 -70
- package/dist/cluster-detector.js.map +0 -1
- package/dist/defaults.d.ts +0 -7
- package/dist/defaults.d.ts.map +0 -1
- package/dist/defaults.js +0 -54
- package/dist/defaults.js.map +0 -1
- package/dist/graph-builder.d.ts +0 -33
- package/dist/graph-builder.d.ts.map +0 -1
- package/dist/graph-builder.js +0 -225
- package/dist/graph-builder.js.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/metrics.d.ts +0 -34
- package/dist/metrics.d.ts.map +0 -1
- package/dist/metrics.js +0 -170
- package/dist/metrics.js.map +0 -1
- package/dist/orchestrator-3L3NAZYP.mjs +0 -10
- package/dist/orchestrator-5AL3XBPZ.mjs +0 -10
- package/dist/orchestrator-KMAKMHTD.mjs +0 -10
- package/dist/orchestrator-MONOZHVW.mjs +0 -10
- package/dist/orchestrator-R6MZT4Z2.mjs +0 -10
- package/dist/orchestrator-ZR7JSKWI.mjs +0 -10
- package/dist/provider.d.ts +0 -6
- package/dist/provider.d.ts.map +0 -1
- package/dist/provider.js +0 -48
- package/dist/provider.js.map +0 -1
- package/dist/python-context-3GZKN3LR.mjs +0 -162
- package/dist/python-context-GOH747QU.mjs +0 -202
- package/dist/python-context-O2EN3M6Z.mjs +0 -162
- package/dist/python-context-PAETRLDY.mjs +0 -185
- package/dist/python-context-TBI5FVFY.mjs +0 -203
- package/dist/python-context-UOPTQH44.mjs +0 -192
- package/dist/remediation.d.ts +0 -25
- package/dist/remediation.d.ts.map +0 -1
- package/dist/remediation.js +0 -98
- package/dist/remediation.js.map +0 -1
- package/dist/scoring.d.ts +0 -9
- package/dist/scoring.d.ts.map +0 -1
- package/dist/scoring.js +0 -142
- package/dist/scoring.js.map +0 -1
- package/dist/semantic-analysis.d.ts +0 -33
- package/dist/semantic-analysis.d.ts.map +0 -1
- package/dist/semantic-analysis.js +0 -303
- package/dist/semantic-analysis.js.map +0 -1
- package/dist/summary-7PZVW72O.mjs +0 -7
- package/dist/summary-LKUCJAIS.mjs +0 -7
- package/dist/summary.d.ts +0 -6
- package/dist/summary.d.ts.map +0 -1
- package/dist/summary.js +0 -92
- package/dist/summary.js.map +0 -1
- package/dist/types.d.ts +0 -124
- package/dist/types.d.ts.map +0 -1
- package/dist/types.js +0 -2
- package/dist/types.js.map +0 -1
- package/dist/utils/output-formatter.d.ts +0 -14
- package/dist/utils/output-formatter.d.ts.map +0 -1
- package/dist/utils/output-formatter.js +0 -338
- package/dist/utils/output-formatter.js.map +0 -1
|
@@ -5,933 +5,28 @@ import {
|
|
|
5
5
|
adjustCohesionForClassification,
|
|
6
6
|
getClassificationRecommendations,
|
|
7
7
|
} from '../index';
|
|
8
|
-
import
|
|
9
|
-
|
|
10
|
-
describe('file classification', () => {
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
const classification = classifyFile(node, 0.5, [
|
|
34
|
-
'module1',
|
|
35
|
-
'module2',
|
|
36
|
-
'module3',
|
|
37
|
-
]);
|
|
38
|
-
expect(classification).toBe('barrel-export');
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
it('should classify type definition files correctly', () => {
|
|
42
|
-
const node = createNode({
|
|
43
|
-
file: 'src/types.ts',
|
|
44
|
-
exports: [
|
|
45
|
-
{ name: 'User', type: 'interface', inferredDomain: 'user' },
|
|
46
|
-
{ name: 'Order', type: 'interface', inferredDomain: 'order' },
|
|
47
|
-
{ name: 'Product', type: 'type', inferredDomain: 'product' },
|
|
48
|
-
{ name: 'Status', type: 'type', inferredDomain: 'unknown' },
|
|
49
|
-
],
|
|
50
|
-
linesOfCode: 100,
|
|
51
|
-
});
|
|
52
|
-
|
|
53
|
-
const classification = classifyFile(node, 0.5, [
|
|
54
|
-
'user',
|
|
55
|
-
'order',
|
|
56
|
-
'product',
|
|
57
|
-
]);
|
|
58
|
-
expect(classification).toBe('type-definition');
|
|
59
|
-
});
|
|
60
|
-
|
|
61
|
-
it('should classify files in /types/ directory as type-definition', () => {
|
|
62
|
-
const node = createNode({
|
|
63
|
-
file: 'shared/src/types/audit/parser.ts',
|
|
64
|
-
exports: [
|
|
65
|
-
{
|
|
66
|
-
name: 'AuditParserConfig',
|
|
67
|
-
type: 'interface',
|
|
68
|
-
inferredDomain: 'audit',
|
|
69
|
-
},
|
|
70
|
-
{ name: 'ParseResult', type: 'type', inferredDomain: 'parse' },
|
|
71
|
-
],
|
|
72
|
-
linesOfCode: 50,
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
const classification = classifyFile(node, 0.5, ['audit', 'parse']);
|
|
76
|
-
expect(classification).toBe('type-definition');
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
it('should classify files in nested /types/ subdirectories as type-definition', () => {
|
|
80
|
-
const node = createNode({
|
|
81
|
-
file: 'shared/src/types/audit/status.ts',
|
|
82
|
-
exports: [
|
|
83
|
-
{ name: 'AuditStatus', type: 'type', inferredDomain: 'audit' },
|
|
84
|
-
{
|
|
85
|
-
name: 'StatusMapping',
|
|
86
|
-
type: 'interface',
|
|
87
|
-
inferredDomain: 'status',
|
|
88
|
-
},
|
|
89
|
-
],
|
|
90
|
-
linesOfCode: 30,
|
|
91
|
-
});
|
|
92
|
-
|
|
93
|
-
const classification = classifyFile(node, 0.5, ['audit', 'status']);
|
|
94
|
-
expect(classification).toBe('type-definition');
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
it('should classify pure type files (only type/interface exports) as type-definition', () => {
|
|
98
|
-
const node = createNode({
|
|
99
|
-
file: 'src/models/user-models.ts', // NOT in /types/ but only type exports
|
|
100
|
-
exports: [
|
|
101
|
-
{
|
|
102
|
-
name: 'UserCreateInput',
|
|
103
|
-
type: 'interface',
|
|
104
|
-
inferredDomain: 'user',
|
|
105
|
-
},
|
|
106
|
-
{
|
|
107
|
-
name: 'UserUpdateInput',
|
|
108
|
-
type: 'interface',
|
|
109
|
-
inferredDomain: 'user',
|
|
110
|
-
},
|
|
111
|
-
{ name: 'UserFilter', type: 'type', inferredDomain: 'user' },
|
|
112
|
-
],
|
|
113
|
-
linesOfCode: 80,
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
const classification = classifyFile(node, 0.5, ['user']);
|
|
117
|
-
expect(classification).toBe('type-definition');
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
it('should classify cohesive module files correctly', () => {
|
|
121
|
-
const node = createNode({
|
|
122
|
-
file: 'src/calculator.ts',
|
|
123
|
-
exports: [
|
|
124
|
-
{ name: 'calculate', type: 'function', inferredDomain: 'calc' },
|
|
125
|
-
{ name: 'format', type: 'function', inferredDomain: 'calc' },
|
|
126
|
-
{ name: 'validate', type: 'function', inferredDomain: 'calc' },
|
|
127
|
-
],
|
|
128
|
-
imports: ['../utils'],
|
|
129
|
-
linesOfCode: 300,
|
|
130
|
-
});
|
|
131
|
-
|
|
132
|
-
const classification = classifyFile(node, 0.8, ['calc']);
|
|
133
|
-
expect(classification).toBe('cohesive-module');
|
|
134
|
-
});
|
|
135
|
-
|
|
136
|
-
it('should classify mixed concerns files correctly', () => {
|
|
137
|
-
const node = createNode({
|
|
138
|
-
file: 'src/audit.ts',
|
|
139
|
-
exports: [
|
|
140
|
-
{ name: 'auditStatus', type: 'function', inferredDomain: 'audit' },
|
|
141
|
-
{ name: 'createJob', type: 'function', inferredDomain: 'job' },
|
|
142
|
-
{ name: 'LineItem', type: 'interface', inferredDomain: 'order' },
|
|
143
|
-
{ name: 'SupportingDoc', type: 'type', inferredDomain: 'doc' },
|
|
144
|
-
],
|
|
145
|
-
imports: ['../auth', '../job', '../order'],
|
|
146
|
-
linesOfCode: 384,
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
const classification = classifyFile(node, 0.3, [
|
|
150
|
-
'audit',
|
|
151
|
-
'job',
|
|
152
|
-
'order',
|
|
153
|
-
'doc',
|
|
154
|
-
]);
|
|
155
|
-
expect(classification).toBe('mixed-concerns');
|
|
156
|
-
});
|
|
157
|
-
|
|
158
|
-
it('should classify files with multiple domains and very low cohesion as mixed-concerns', () => {
|
|
159
|
-
const node = createNode({
|
|
160
|
-
file: 'src/modules/mixed-module.ts', // NOT a utility/config/service path
|
|
161
|
-
exports: [
|
|
162
|
-
{ name: 'DateCalculator', type: 'class', inferredDomain: 'date' }, // Use class to avoid utility detection
|
|
163
|
-
{ name: 'ReportBuilder', type: 'class', inferredDomain: 'report' },
|
|
164
|
-
{ name: 'AuditLogger', type: 'class', inferredDomain: 'audit' },
|
|
165
|
-
],
|
|
166
|
-
imports: [],
|
|
167
|
-
linesOfCode: 150,
|
|
168
|
-
});
|
|
169
|
-
|
|
170
|
-
// Multiple domains + very low cohesion (< 0.4) = mixed concerns
|
|
171
|
-
// Note: NOT in /utils/ or /helpers/ or /services/ path
|
|
172
|
-
const classification = classifyFile(node, 0.3, [
|
|
173
|
-
'date',
|
|
174
|
-
'report',
|
|
175
|
-
'audit',
|
|
176
|
-
]);
|
|
177
|
-
expect(classification).toBe('mixed-concerns');
|
|
178
|
-
});
|
|
179
|
-
|
|
180
|
-
it('should classify single domain files as cohesive-module regardless of cohesion', () => {
|
|
181
|
-
const node = createNode({
|
|
182
|
-
file: 'src/component.ts',
|
|
183
|
-
exports: [
|
|
184
|
-
{ name: 'Component', type: 'function', inferredDomain: 'ui' },
|
|
185
|
-
],
|
|
186
|
-
imports: ['react'],
|
|
187
|
-
linesOfCode: 100,
|
|
188
|
-
});
|
|
189
|
-
|
|
190
|
-
// Single domain = cohesive module (even with medium cohesion)
|
|
191
|
-
const classification = classifyFile(node, 0.6, ['ui']);
|
|
192
|
-
expect(classification).toBe('cohesive-module');
|
|
193
|
-
});
|
|
194
|
-
|
|
195
|
-
it('should classify utility files as cohesive-module by design', () => {
|
|
196
|
-
const node = createNode({
|
|
197
|
-
file: 'src/utils/helpers.ts',
|
|
198
|
-
exports: [
|
|
199
|
-
{ name: 'formatDate', type: 'function', inferredDomain: 'date' },
|
|
200
|
-
{ name: 'stringifyJSON', type: 'function', inferredDomain: 'json' },
|
|
201
|
-
{ name: 'validateEmail', type: 'function', inferredDomain: 'email' },
|
|
202
|
-
],
|
|
203
|
-
imports: [],
|
|
204
|
-
linesOfCode: 150,
|
|
205
|
-
});
|
|
206
|
-
|
|
207
|
-
// Utility files are classified as cohesive by design
|
|
208
|
-
const classification = classifyFile(node, 0.4, ['date', 'json', 'email']);
|
|
209
|
-
expect(classification).toBe('utility-module');
|
|
210
|
-
});
|
|
211
|
-
|
|
212
|
-
it('should classify config/schema files as cohesive-module', () => {
|
|
213
|
-
const node = createNode({
|
|
214
|
-
file: 'src/db-schema.ts',
|
|
215
|
-
exports: [
|
|
216
|
-
{ name: 'userTable', type: 'const', inferredDomain: 'db' },
|
|
217
|
-
{ name: 'userSchema', type: 'const', inferredDomain: 'schema' },
|
|
218
|
-
],
|
|
219
|
-
imports: ['../db'],
|
|
220
|
-
linesOfCode: 81,
|
|
221
|
-
});
|
|
222
|
-
|
|
223
|
-
// Config/schema files are classified as cohesive
|
|
224
|
-
const classification = classifyFile(node, 0.4, ['db', 'schema']);
|
|
225
|
-
expect(classification).toBe('cohesive-module');
|
|
226
|
-
});
|
|
227
|
-
});
|
|
228
|
-
|
|
229
|
-
describe('adjustFragmentationForClassification', () => {
|
|
230
|
-
it('should return 0 fragmentation for barrel exports', () => {
|
|
231
|
-
const result = adjustFragmentationForClassification(0.8, 'barrel-export');
|
|
232
|
-
expect(result).toBe(0);
|
|
233
|
-
});
|
|
234
|
-
|
|
235
|
-
it('should return 0 fragmentation for type definitions', () => {
|
|
236
|
-
const result = adjustFragmentationForClassification(
|
|
237
|
-
0.9,
|
|
238
|
-
'type-definition'
|
|
239
|
-
);
|
|
240
|
-
expect(result).toBe(0);
|
|
241
|
-
});
|
|
242
|
-
|
|
243
|
-
it('should reduce fragmentation by 70% for cohesive modules', () => {
|
|
244
|
-
const result = adjustFragmentationForClassification(
|
|
245
|
-
0.6,
|
|
246
|
-
'cohesive-module'
|
|
247
|
-
);
|
|
248
|
-
expect(result).toBeCloseTo(0.18, 2); // 0.6 * 0.3
|
|
249
|
-
});
|
|
250
|
-
|
|
251
|
-
it('should keep full fragmentation for mixed concerns', () => {
|
|
252
|
-
const result = adjustFragmentationForClassification(
|
|
253
|
-
0.7,
|
|
254
|
-
'mixed-concerns'
|
|
255
|
-
);
|
|
256
|
-
expect(result).toBe(0.7);
|
|
257
|
-
});
|
|
258
|
-
|
|
259
|
-
it('should reduce fragmentation by 30% for unknown classification', () => {
|
|
260
|
-
const result = adjustFragmentationForClassification(0.5, 'unknown');
|
|
261
|
-
expect(result).toBeCloseTo(0.35, 2); // 0.5 * 0.7
|
|
262
|
-
});
|
|
263
|
-
});
|
|
264
|
-
|
|
265
|
-
describe('getClassificationRecommendations', () => {
|
|
266
|
-
it('should provide barrel export recommendations', () => {
|
|
267
|
-
const recommendations = getClassificationRecommendations(
|
|
268
|
-
'barrel-export',
|
|
269
|
-
'src/index.ts',
|
|
270
|
-
['High fragmentation']
|
|
271
|
-
);
|
|
272
|
-
expect(recommendations).toContain(
|
|
273
|
-
'Barrel export file detected - multiple domains are expected here'
|
|
274
|
-
);
|
|
275
|
-
});
|
|
276
|
-
|
|
277
|
-
it('should provide type definition recommendations', () => {
|
|
278
|
-
const recommendations = getClassificationRecommendations(
|
|
279
|
-
'type-definition',
|
|
280
|
-
'src/types.ts',
|
|
281
|
-
['High fragmentation']
|
|
282
|
-
);
|
|
283
|
-
expect(recommendations).toContain(
|
|
284
|
-
'Type definition file - centralized types improve consistency'
|
|
285
|
-
);
|
|
286
|
-
});
|
|
287
|
-
|
|
288
|
-
it('should provide cohesive module recommendations', () => {
|
|
289
|
-
const recommendations = getClassificationRecommendations(
|
|
290
|
-
'cohesive-module',
|
|
291
|
-
'src/calculator.ts',
|
|
292
|
-
[]
|
|
293
|
-
);
|
|
294
|
-
expect(recommendations).toContain(
|
|
295
|
-
'Module has good cohesion despite its size'
|
|
296
|
-
);
|
|
297
|
-
});
|
|
298
|
-
|
|
299
|
-
it('should provide mixed concerns recommendations', () => {
|
|
300
|
-
const recommendations = getClassificationRecommendations(
|
|
301
|
-
'mixed-concerns',
|
|
302
|
-
'src/audit.ts',
|
|
303
|
-
['Multiple domains detected']
|
|
304
|
-
);
|
|
305
|
-
expect(recommendations).toContain(
|
|
306
|
-
'Consider splitting this file by domain'
|
|
307
|
-
);
|
|
308
|
-
});
|
|
309
|
-
});
|
|
310
|
-
|
|
311
|
-
describe('integration: barrel export detection edge cases', () => {
|
|
312
|
-
it('should detect barrel export even for non-index files with re-export patterns', () => {
|
|
313
|
-
const node = createNode({
|
|
314
|
-
file: 'src/exports.ts',
|
|
315
|
-
imports: [
|
|
316
|
-
'../module1',
|
|
317
|
-
'../module2',
|
|
318
|
-
'../module3',
|
|
319
|
-
'../module4',
|
|
320
|
-
'../module5',
|
|
321
|
-
],
|
|
322
|
-
exports: [
|
|
323
|
-
{ name: 'a', type: 'function' },
|
|
324
|
-
{ name: 'b', type: 'function' },
|
|
325
|
-
{ name: 'c', type: 'function' },
|
|
326
|
-
{ name: 'd', type: 'function' },
|
|
327
|
-
{ name: 'e', type: 'function' },
|
|
328
|
-
],
|
|
329
|
-
linesOfCode: 25, // Very sparse - mostly re-exports
|
|
330
|
-
});
|
|
331
|
-
|
|
332
|
-
const classification = classifyFile(node, 0.5, ['module1', 'module2']);
|
|
333
|
-
expect(classification).toBe('barrel-export');
|
|
334
|
-
});
|
|
335
|
-
|
|
336
|
-
it('should not misclassify large component files as barrel exports', () => {
|
|
337
|
-
const node = createNode({
|
|
338
|
-
file: 'src/components/Calculator.tsx', // NOT an index file
|
|
339
|
-
imports: ['react', '../hooks', '../utils'],
|
|
340
|
-
exports: [{ name: 'Calculator', type: 'function' }],
|
|
341
|
-
linesOfCode: 346, // Substantial code
|
|
342
|
-
});
|
|
343
|
-
|
|
344
|
-
// Single domain, high cohesion
|
|
345
|
-
const classification = classifyFile(node, 0.9, ['calculator']);
|
|
346
|
-
expect(classification).toBe('cohesive-module');
|
|
347
|
-
});
|
|
348
|
-
});
|
|
349
|
-
|
|
350
|
-
describe('new classification patterns', () => {
|
|
351
|
-
describe('lambda-handler detection', () => {
|
|
352
|
-
it('should classify files in handlers directory as lambda-handler', () => {
|
|
353
|
-
const node = createNode({
|
|
354
|
-
file: 'src/handlers/tier1-immediate.ts',
|
|
355
|
-
exports: [
|
|
356
|
-
{ name: 'handler', type: 'function', inferredDomain: 'match' },
|
|
357
|
-
],
|
|
358
|
-
imports: ['../services/matcher', '../services/db'],
|
|
359
|
-
linesOfCode: 150,
|
|
360
|
-
});
|
|
361
|
-
|
|
362
|
-
const classification = classifyFile(node, 0.21, ['match', 'db']);
|
|
363
|
-
expect(classification).toBe('lambda-handler');
|
|
364
|
-
});
|
|
365
|
-
|
|
366
|
-
it('should classify files with handler export as lambda-handler', () => {
|
|
367
|
-
const node = createNode({
|
|
368
|
-
file: 'src/api/process.ts',
|
|
369
|
-
exports: [
|
|
370
|
-
{
|
|
371
|
-
name: 'processHandler',
|
|
372
|
-
type: 'function',
|
|
373
|
-
inferredDomain: 'process',
|
|
374
|
-
},
|
|
375
|
-
],
|
|
376
|
-
imports: ['../services/queue'],
|
|
377
|
-
linesOfCode: 80,
|
|
378
|
-
});
|
|
379
|
-
|
|
380
|
-
const classification = classifyFile(node, 0.3, ['process']);
|
|
381
|
-
expect(classification).toBe('lambda-handler');
|
|
382
|
-
});
|
|
383
|
-
|
|
384
|
-
it('should classify files in lambdas directory as lambda-handler', () => {
|
|
385
|
-
const node = createNode({
|
|
386
|
-
file: 'src/lambdas/match-single-document.ts',
|
|
387
|
-
exports: [
|
|
388
|
-
{ name: 'handler', type: 'function', inferredDomain: 'match' },
|
|
389
|
-
],
|
|
390
|
-
imports: ['../services/matcher'],
|
|
391
|
-
linesOfCode: 100,
|
|
392
|
-
});
|
|
393
|
-
|
|
394
|
-
const classification = classifyFile(node, 0.22, ['match']);
|
|
395
|
-
expect(classification).toBe('lambda-handler');
|
|
396
|
-
});
|
|
397
|
-
|
|
398
|
-
it('should classify single export functions as lambda-handler', () => {
|
|
399
|
-
const node = createNode({
|
|
400
|
-
file: 'src/functions/process.ts',
|
|
401
|
-
exports: [
|
|
402
|
-
{ name: 'default', type: 'default', inferredDomain: 'process' },
|
|
403
|
-
],
|
|
404
|
-
imports: ['../services/processor'],
|
|
405
|
-
linesOfCode: 60,
|
|
406
|
-
});
|
|
407
|
-
|
|
408
|
-
const classification = classifyFile(node, 0.4, ['process']);
|
|
409
|
-
expect(classification).toBe('lambda-handler');
|
|
410
|
-
});
|
|
411
|
-
});
|
|
412
|
-
|
|
413
|
-
describe('service-file detection', () => {
|
|
414
|
-
it('should classify files with -service.ts pattern as service-file', () => {
|
|
415
|
-
const node = createNode({
|
|
416
|
-
file: 'src/services/email-service.ts',
|
|
417
|
-
exports: [
|
|
418
|
-
{ name: 'EmailService', type: 'class', inferredDomain: 'email' },
|
|
419
|
-
],
|
|
420
|
-
imports: ['../utils/smtp', '../templates'],
|
|
421
|
-
linesOfCode: 200,
|
|
422
|
-
});
|
|
423
|
-
|
|
424
|
-
const classification = classifyFile(node, 0.07, [
|
|
425
|
-
'email',
|
|
426
|
-
'smtp',
|
|
427
|
-
'templates',
|
|
428
|
-
]);
|
|
429
|
-
expect(classification).toBe('service-file');
|
|
430
|
-
});
|
|
431
|
-
|
|
432
|
-
it('should classify files in services directory as service-file', () => {
|
|
433
|
-
const node = createNode({
|
|
434
|
-
file: 'src/services/notification.ts',
|
|
435
|
-
exports: [
|
|
436
|
-
{
|
|
437
|
-
name: 'sendNotification',
|
|
438
|
-
type: 'function',
|
|
439
|
-
inferredDomain: 'notification',
|
|
440
|
-
},
|
|
441
|
-
{
|
|
442
|
-
name: 'queueNotification',
|
|
443
|
-
type: 'function',
|
|
444
|
-
inferredDomain: 'notification',
|
|
445
|
-
},
|
|
446
|
-
],
|
|
447
|
-
imports: ['../db', '../email'],
|
|
448
|
-
linesOfCode: 120,
|
|
449
|
-
});
|
|
450
|
-
|
|
451
|
-
const classification = classifyFile(node, 0.35, ['notification']);
|
|
452
|
-
expect(classification).toBe('service-file');
|
|
453
|
-
});
|
|
454
|
-
|
|
455
|
-
it('should classify class exports in services directory as service-file', () => {
|
|
456
|
-
const node = createNode({
|
|
457
|
-
file: 'src/api/user-service.ts',
|
|
458
|
-
exports: [
|
|
459
|
-
{ name: 'UserService', type: 'class', inferredDomain: 'user' },
|
|
460
|
-
],
|
|
461
|
-
imports: ['../db', '../auth'],
|
|
462
|
-
linesOfCode: 180,
|
|
463
|
-
});
|
|
464
|
-
|
|
465
|
-
const classification = classifyFile(node, 0.25, ['user']);
|
|
466
|
-
expect(classification).toBe('service-file');
|
|
467
|
-
});
|
|
468
|
-
});
|
|
469
|
-
|
|
470
|
-
describe('email-template detection', () => {
|
|
471
|
-
it('should classify receipt-writer files as email-template', () => {
|
|
472
|
-
const node = createNode({
|
|
473
|
-
file: 'src/emails/receipt-writer.ts',
|
|
474
|
-
exports: [
|
|
475
|
-
{
|
|
476
|
-
name: 'generateReceipt',
|
|
477
|
-
type: 'function',
|
|
478
|
-
inferredDomain: 'receipt',
|
|
479
|
-
},
|
|
480
|
-
],
|
|
481
|
-
imports: ['../templates/base', '../services/db'],
|
|
482
|
-
linesOfCode: 150,
|
|
483
|
-
});
|
|
484
|
-
|
|
485
|
-
const classification = classifyFile(node, 0.08, [
|
|
486
|
-
'receipt',
|
|
487
|
-
'templates',
|
|
488
|
-
'db',
|
|
489
|
-
]);
|
|
490
|
-
expect(classification).toBe('email-template');
|
|
491
|
-
});
|
|
492
|
-
|
|
493
|
-
it('should classify files in emails directory as email-template', () => {
|
|
494
|
-
const node = createNode({
|
|
495
|
-
file: 'src/emails/welcome.ts',
|
|
496
|
-
exports: [
|
|
497
|
-
{
|
|
498
|
-
name: 'renderWelcomeEmail',
|
|
499
|
-
type: 'function',
|
|
500
|
-
inferredDomain: 'email',
|
|
501
|
-
},
|
|
502
|
-
],
|
|
503
|
-
imports: ['../templates/layout'],
|
|
504
|
-
linesOfCode: 80,
|
|
505
|
-
});
|
|
506
|
-
|
|
507
|
-
const classification = classifyFile(node, 0.15, ['email']);
|
|
508
|
-
expect(classification).toBe('email-template');
|
|
509
|
-
});
|
|
510
|
-
|
|
511
|
-
it('should classify files with template patterns as email-template', () => {
|
|
512
|
-
const node = createNode({
|
|
513
|
-
file: 'src/templates/invoice-template.ts',
|
|
514
|
-
exports: [
|
|
515
|
-
{
|
|
516
|
-
name: 'renderInvoice',
|
|
517
|
-
type: 'function',
|
|
518
|
-
inferredDomain: 'invoice',
|
|
519
|
-
},
|
|
520
|
-
],
|
|
521
|
-
imports: ['../services/pdf'],
|
|
522
|
-
linesOfCode: 100,
|
|
523
|
-
});
|
|
524
|
-
|
|
525
|
-
const classification = classifyFile(node, 0.2, ['invoice']);
|
|
526
|
-
expect(classification).toBe('email-template');
|
|
527
|
-
});
|
|
528
|
-
});
|
|
529
|
-
|
|
530
|
-
describe('parser-file detection', () => {
|
|
531
|
-
it('should classify parser files correctly', () => {
|
|
532
|
-
const node = createNode({
|
|
533
|
-
file: 'src/parsers/base-parser-deterministic.ts',
|
|
534
|
-
exports: [
|
|
535
|
-
{
|
|
536
|
-
name: 'parseDeterministic',
|
|
537
|
-
type: 'function',
|
|
538
|
-
inferredDomain: 'parse',
|
|
539
|
-
},
|
|
540
|
-
{
|
|
541
|
-
name: 'parseNonDeterministic',
|
|
542
|
-
type: 'function',
|
|
543
|
-
inferredDomain: 'parse',
|
|
544
|
-
},
|
|
545
|
-
],
|
|
546
|
-
imports: ['../utils/transform'],
|
|
547
|
-
linesOfCode: 120,
|
|
548
|
-
});
|
|
549
|
-
|
|
550
|
-
const classification = classifyFile(node, 0.15, ['parse']);
|
|
551
|
-
expect(classification).toBe('parser-file');
|
|
552
|
-
});
|
|
553
|
-
|
|
554
|
-
it('should classify files with parser in name as parser-file', () => {
|
|
555
|
-
const node = createNode({
|
|
556
|
-
file: 'src/parsers/data-parser.ts',
|
|
557
|
-
exports: [
|
|
558
|
-
{ name: 'parseData', type: 'function', inferredDomain: 'data' },
|
|
559
|
-
{ name: 'transformData', type: 'function', inferredDomain: 'data' },
|
|
560
|
-
],
|
|
561
|
-
imports: [],
|
|
562
|
-
linesOfCode: 90,
|
|
563
|
-
});
|
|
564
|
-
|
|
565
|
-
const classification = classifyFile(node, 0.25, ['data']);
|
|
566
|
-
expect(classification).toBe('parser-file');
|
|
567
|
-
});
|
|
568
|
-
|
|
569
|
-
it('should classify converter files as parser-file', () => {
|
|
570
|
-
const node = createNode({
|
|
571
|
-
file: 'src/converters/xml-converter.ts',
|
|
572
|
-
exports: [
|
|
573
|
-
{
|
|
574
|
-
name: 'convertXmlToJson',
|
|
575
|
-
type: 'function',
|
|
576
|
-
inferredDomain: 'xml',
|
|
577
|
-
},
|
|
578
|
-
{
|
|
579
|
-
name: 'convertJsonToXml',
|
|
580
|
-
type: 'function',
|
|
581
|
-
inferredDomain: 'xml',
|
|
582
|
-
},
|
|
583
|
-
],
|
|
584
|
-
imports: ['xml2js'],
|
|
585
|
-
linesOfCode: 60,
|
|
586
|
-
});
|
|
587
|
-
|
|
588
|
-
const classification = classifyFile(node, 0.3, ['xml']);
|
|
589
|
-
expect(classification).toBe('parser-file');
|
|
590
|
-
});
|
|
591
|
-
});
|
|
592
|
-
|
|
593
|
-
describe('utility-module detection', () => {
|
|
594
|
-
it('should classify dynamodb-utils.ts as utility-module', () => {
|
|
595
|
-
const node = createNode({
|
|
596
|
-
file: 'src/utils/dynamodb-utils.ts',
|
|
597
|
-
exports: [
|
|
598
|
-
{ name: 'getItem', type: 'function', inferredDomain: 'db' },
|
|
599
|
-
{ name: 'putItem', type: 'function', inferredDomain: 'db' },
|
|
600
|
-
{ name: 'queryItems', type: 'function', inferredDomain: 'db' },
|
|
601
|
-
],
|
|
602
|
-
imports: ['aws-sdk'],
|
|
603
|
-
linesOfCode: 100,
|
|
604
|
-
});
|
|
605
|
-
|
|
606
|
-
const classification = classifyFile(node, 0.21, ['db']);
|
|
607
|
-
expect(classification).toBe('utility-module');
|
|
608
|
-
});
|
|
609
|
-
|
|
610
|
-
it('should classify s3-utils.ts as utility-module', () => {
|
|
611
|
-
const node = createNode({
|
|
612
|
-
file: 'src/utils/s3-utils.ts',
|
|
613
|
-
exports: [
|
|
614
|
-
{ name: 'uploadFile', type: 'function', inferredDomain: 's3' },
|
|
615
|
-
{ name: 'downloadFile', type: 'function', inferredDomain: 's3' },
|
|
616
|
-
{ name: 'deleteFile', type: 'function', inferredDomain: 's3' },
|
|
617
|
-
],
|
|
618
|
-
imports: ['aws-sdk'],
|
|
619
|
-
linesOfCode: 80,
|
|
620
|
-
});
|
|
621
|
-
|
|
622
|
-
const classification = classifyFile(node, 0.26, ['s3']);
|
|
623
|
-
expect(classification).toBe('utility-module');
|
|
624
|
-
});
|
|
625
|
-
|
|
626
|
-
it('should classify files ending with -utils.ts as utility-module', () => {
|
|
627
|
-
const node = createNode({
|
|
628
|
-
file: 'src/helpers/date-utils.ts',
|
|
629
|
-
exports: [
|
|
630
|
-
{ name: 'formatDate', type: 'function', inferredDomain: 'date' },
|
|
631
|
-
{ name: 'validateDate', type: 'function', inferredDomain: 'date' },
|
|
632
|
-
],
|
|
633
|
-
imports: [],
|
|
634
|
-
linesOfCode: 50,
|
|
635
|
-
});
|
|
636
|
-
|
|
637
|
-
const classification = classifyFile(node, 0.3, ['date']);
|
|
638
|
-
expect(classification).toBe('utility-module');
|
|
639
|
-
});
|
|
640
|
-
});
|
|
641
|
-
|
|
642
|
-
describe('session file detection', () => {
|
|
643
|
-
it('should classify session.ts as cohesive-module', () => {
|
|
644
|
-
const node = createNode({
|
|
645
|
-
file: 'src/session.ts',
|
|
646
|
-
exports: [
|
|
647
|
-
{
|
|
648
|
-
name: 'createSession',
|
|
649
|
-
type: 'function',
|
|
650
|
-
inferredDomain: 'session',
|
|
651
|
-
},
|
|
652
|
-
{ name: 'getSession', type: 'function', inferredDomain: 'session' },
|
|
653
|
-
{
|
|
654
|
-
name: 'destroySession',
|
|
655
|
-
type: 'function',
|
|
656
|
-
inferredDomain: 'session',
|
|
657
|
-
},
|
|
658
|
-
],
|
|
659
|
-
imports: ['../db', '../auth'],
|
|
660
|
-
linesOfCode: 100,
|
|
661
|
-
});
|
|
662
|
-
|
|
663
|
-
const classification = classifyFile(node, 0.26, ['session']);
|
|
664
|
-
expect(classification).toBe('cohesive-module');
|
|
665
|
-
});
|
|
666
|
-
});
|
|
667
|
-
});
|
|
668
|
-
|
|
669
|
-
describe('adjustCohesionForClassification', () => {
|
|
670
|
-
it('should return 1 for barrel exports', () => {
|
|
671
|
-
const result = adjustCohesionForClassification(0.3, 'barrel-export');
|
|
672
|
-
expect(result).toBe(1);
|
|
673
|
-
});
|
|
674
|
-
|
|
675
|
-
it('should return 1 for type definitions', () => {
|
|
676
|
-
const result = adjustCohesionForClassification(0.2, 'type-definition');
|
|
677
|
-
expect(result).toBe(1);
|
|
678
|
-
});
|
|
679
|
-
|
|
680
|
-
it('should boost cohesion for utility modules', () => {
|
|
681
|
-
const result = adjustCohesionForClassification(0.21, 'utility-module');
|
|
682
|
-
expect(result).toBeGreaterThan(0.21);
|
|
683
|
-
expect(result).toBeLessThanOrEqual(1);
|
|
684
|
-
});
|
|
685
|
-
|
|
686
|
-
it('should boost cohesion for service files', () => {
|
|
687
|
-
const result = adjustCohesionForClassification(0.07, 'service-file');
|
|
688
|
-
expect(result).toBeGreaterThan(0.07);
|
|
689
|
-
expect(result).toBeLessThanOrEqual(1);
|
|
690
|
-
});
|
|
691
|
-
|
|
692
|
-
it('should boost cohesion for lambda handlers', () => {
|
|
693
|
-
const result = adjustCohesionForClassification(0.21, 'lambda-handler');
|
|
694
|
-
expect(result).toBeGreaterThan(0.21);
|
|
695
|
-
expect(result).toBeLessThanOrEqual(1);
|
|
696
|
-
});
|
|
697
|
-
|
|
698
|
-
it('should boost cohesion for email templates', () => {
|
|
699
|
-
const result = adjustCohesionForClassification(0.08, 'email-template');
|
|
700
|
-
expect(result).toBeGreaterThan(0.08);
|
|
701
|
-
expect(result).toBeLessThanOrEqual(1);
|
|
702
|
-
});
|
|
703
|
-
|
|
704
|
-
it('should boost cohesion for parser files', () => {
|
|
705
|
-
const result = adjustCohesionForClassification(0.15, 'parser-file');
|
|
706
|
-
expect(result).toBeGreaterThan(0.15);
|
|
707
|
-
expect(result).toBeLessThanOrEqual(1);
|
|
708
|
-
});
|
|
709
|
-
|
|
710
|
-
it('should keep original cohesion for mixed-concerns', () => {
|
|
711
|
-
const result = adjustCohesionForClassification(0.3, 'mixed-concerns');
|
|
712
|
-
expect(result).toBe(0.3);
|
|
713
|
-
});
|
|
714
|
-
|
|
715
|
-
it('should boost cohesion for utility module with related export names', () => {
|
|
716
|
-
const node = createNode({
|
|
717
|
-
file: 'src/utils/dynamodb-utils.ts',
|
|
718
|
-
exports: [
|
|
719
|
-
{ name: 'getItem', type: 'function', inferredDomain: 'db' },
|
|
720
|
-
{ name: 'putItem', type: 'function', inferredDomain: 'db' },
|
|
721
|
-
{ name: 'deleteItem', type: 'function', inferredDomain: 'db' },
|
|
722
|
-
],
|
|
723
|
-
imports: ['aws-sdk'],
|
|
724
|
-
linesOfCode: 100,
|
|
725
|
-
});
|
|
726
|
-
|
|
727
|
-
// Related names (all Item operations) should get higher boost
|
|
728
|
-
const result = adjustCohesionForClassification(
|
|
729
|
-
0.21,
|
|
730
|
-
'utility-module',
|
|
731
|
-
node
|
|
732
|
-
);
|
|
733
|
-
expect(result).toBeGreaterThan(0.5); // Significant boost for related names
|
|
734
|
-
});
|
|
735
|
-
|
|
736
|
-
it('should boost cohesion for lambda handler with single entry point', () => {
|
|
737
|
-
const node = createNode({
|
|
738
|
-
file: 'src/handlers/process.ts',
|
|
739
|
-
exports: [
|
|
740
|
-
{ name: 'handler', type: 'function', inferredDomain: 'process' },
|
|
741
|
-
],
|
|
742
|
-
imports: ['../services/queue'],
|
|
743
|
-
linesOfCode: 80,
|
|
744
|
-
});
|
|
745
|
-
|
|
746
|
-
// Single entry point should get higher boost
|
|
747
|
-
const result = adjustCohesionForClassification(
|
|
748
|
-
0.22,
|
|
749
|
-
'lambda-handler',
|
|
750
|
-
node
|
|
751
|
-
);
|
|
752
|
-
expect(result).toBeGreaterThan(0.5); // Significant boost for single entry
|
|
753
|
-
});
|
|
754
|
-
|
|
755
|
-
it('should boost cohesion for class-based service files', () => {
|
|
756
|
-
const node = createNode({
|
|
757
|
-
file: 'src/services/email-service.ts',
|
|
758
|
-
exports: [
|
|
759
|
-
{ name: 'EmailService', type: 'class', inferredDomain: 'email' },
|
|
760
|
-
],
|
|
761
|
-
imports: ['../utils/smtp'],
|
|
762
|
-
linesOfCode: 200,
|
|
763
|
-
});
|
|
764
|
-
|
|
765
|
-
// Class-based service should get higher boost
|
|
766
|
-
const result = adjustCohesionForClassification(
|
|
767
|
-
0.15,
|
|
768
|
-
'service-file',
|
|
769
|
-
node
|
|
770
|
-
);
|
|
771
|
-
expect(result).toBeGreaterThan(0.45); // Significant boost for class-based
|
|
772
|
-
});
|
|
773
|
-
});
|
|
774
|
-
|
|
775
|
-
describe('new classification recommendations', () => {
|
|
776
|
-
it('should provide utility-module recommendations', () => {
|
|
777
|
-
const recommendations = getClassificationRecommendations(
|
|
778
|
-
'utility-module',
|
|
779
|
-
'src/utils/helpers.ts',
|
|
780
|
-
['Low cohesion']
|
|
781
|
-
);
|
|
782
|
-
expect(recommendations).toContain(
|
|
783
|
-
'Utility module detected - multiple domains are acceptable here'
|
|
784
|
-
);
|
|
785
|
-
});
|
|
786
|
-
|
|
787
|
-
it('should provide service-file recommendations', () => {
|
|
788
|
-
const recommendations = getClassificationRecommendations(
|
|
789
|
-
'service-file',
|
|
790
|
-
'src/services/email.ts',
|
|
791
|
-
['Multiple domains']
|
|
792
|
-
);
|
|
793
|
-
expect(recommendations).toContain(
|
|
794
|
-
'Service file detected - orchestration of multiple dependencies is expected'
|
|
795
|
-
);
|
|
796
|
-
});
|
|
797
|
-
|
|
798
|
-
it('should provide lambda-handler recommendations', () => {
|
|
799
|
-
const recommendations = getClassificationRecommendations(
|
|
800
|
-
'lambda-handler',
|
|
801
|
-
'src/handlers/process.ts',
|
|
802
|
-
['Low cohesion']
|
|
803
|
-
);
|
|
804
|
-
expect(recommendations).toContain(
|
|
805
|
-
'Lambda handler detected - coordination of services is expected'
|
|
806
|
-
);
|
|
807
|
-
});
|
|
808
|
-
|
|
809
|
-
it('should provide email-template recommendations', () => {
|
|
810
|
-
const recommendations = getClassificationRecommendations(
|
|
811
|
-
'email-template',
|
|
812
|
-
'src/emails/receipt.ts',
|
|
813
|
-
['Multiple domains']
|
|
814
|
-
);
|
|
815
|
-
expect(recommendations).toContain(
|
|
816
|
-
'Email template detected - references multiple domains for rendering'
|
|
817
|
-
);
|
|
818
|
-
});
|
|
819
|
-
|
|
820
|
-
it('should provide parser-file recommendations', () => {
|
|
821
|
-
const recommendations = getClassificationRecommendations(
|
|
822
|
-
'parser-file',
|
|
823
|
-
'src/parsers/data.ts',
|
|
824
|
-
['Multiple domains']
|
|
825
|
-
);
|
|
826
|
-
expect(recommendations).toContain(
|
|
827
|
-
'Parser/transformer file detected - handles multiple data sources'
|
|
828
|
-
);
|
|
829
|
-
});
|
|
830
|
-
});
|
|
831
|
-
|
|
832
|
-
describe('Next.js page detection', () => {
|
|
833
|
-
it('should classify Next.js calculator pages as nextjs-page', () => {
|
|
834
|
-
const node = createNode({
|
|
835
|
-
file: 'app/cents-per-km-calculator/page.tsx',
|
|
836
|
-
exports: [
|
|
837
|
-
{ name: 'metadata', type: 'const', inferredDomain: 'seo' },
|
|
838
|
-
{ name: 'faqJsonLd', type: 'const', inferredDomain: 'jsonld' },
|
|
839
|
-
{ name: 'default', type: 'default', inferredDomain: 'page' },
|
|
840
|
-
{ name: 'icon', type: 'const', inferredDomain: 'ui' },
|
|
841
|
-
],
|
|
842
|
-
imports: ['../components/Calculator', '../lib/utils'],
|
|
843
|
-
linesOfCode: 208,
|
|
844
|
-
});
|
|
845
|
-
|
|
846
|
-
const classification = classifyFile(node, 0.25, [
|
|
847
|
-
'seo',
|
|
848
|
-
'jsonld',
|
|
849
|
-
'page',
|
|
850
|
-
'ui',
|
|
851
|
-
]);
|
|
852
|
-
expect(classification).toBe('nextjs-page');
|
|
853
|
-
});
|
|
854
|
-
|
|
855
|
-
it('should classify investment property calculator page as nextjs-page', () => {
|
|
856
|
-
const node = createNode({
|
|
857
|
-
file: 'app/investment-property-tax-calculator/page.tsx',
|
|
858
|
-
exports: [
|
|
859
|
-
{ name: 'metadata', type: 'const', inferredDomain: 'seo' },
|
|
860
|
-
{ name: 'faqJsonLd', type: 'const', inferredDomain: 'jsonld' },
|
|
861
|
-
{ name: 'default', type: 'default', inferredDomain: 'page' },
|
|
862
|
-
],
|
|
863
|
-
imports: ['../components/Form'],
|
|
864
|
-
linesOfCode: 204,
|
|
865
|
-
});
|
|
866
|
-
|
|
867
|
-
const classification = classifyFile(node, 0.3, ['seo', 'jsonld', 'page']);
|
|
868
|
-
expect(classification).toBe('nextjs-page');
|
|
869
|
-
});
|
|
870
|
-
|
|
871
|
-
it('should not classify non-page.tsx files in /app/ as nextjs-page', () => {
|
|
872
|
-
const node = createNode({
|
|
873
|
-
file: 'app/components/Header.tsx',
|
|
874
|
-
exports: [{ name: 'Header', type: 'function', inferredDomain: 'ui' }],
|
|
875
|
-
imports: ['react'],
|
|
876
|
-
linesOfCode: 50,
|
|
877
|
-
});
|
|
878
|
-
|
|
879
|
-
const classification = classifyFile(node, 0.8, ['ui']);
|
|
880
|
-
expect(classification).toBe('cohesive-module');
|
|
881
|
-
});
|
|
882
|
-
|
|
883
|
-
it('should not classify page.tsx files outside /app/ as nextjs-page', () => {
|
|
884
|
-
const node = createNode({
|
|
885
|
-
file: 'src/pages/page.tsx', // Pages Router, not App Router
|
|
886
|
-
exports: [{ name: 'default', type: 'default', inferredDomain: 'page' }],
|
|
887
|
-
imports: ['react'],
|
|
888
|
-
linesOfCode: 100,
|
|
889
|
-
});
|
|
890
|
-
|
|
891
|
-
const classification = classifyFile(node, 0.5, ['page']);
|
|
892
|
-
expect(classification).not.toBe('nextjs-page');
|
|
893
|
-
});
|
|
894
|
-
|
|
895
|
-
it('should classify Next.js page with generateMetadata as nextjs-page', () => {
|
|
896
|
-
const node = createNode({
|
|
897
|
-
file: 'app/dynamic-page/page.tsx',
|
|
898
|
-
exports: [
|
|
899
|
-
{ name: 'generateMetadata', type: 'function', inferredDomain: 'seo' },
|
|
900
|
-
{ name: 'default', type: 'default', inferredDomain: 'page' },
|
|
901
|
-
],
|
|
902
|
-
imports: ['../lib/api'],
|
|
903
|
-
linesOfCode: 150,
|
|
904
|
-
});
|
|
905
|
-
|
|
906
|
-
const classification = classifyFile(node, 0.4, ['seo', 'page']);
|
|
907
|
-
expect(classification).toBe('nextjs-page');
|
|
908
|
-
});
|
|
909
|
-
});
|
|
910
|
-
|
|
911
|
-
describe('nextjs-page cohesion adjustment', () => {
|
|
912
|
-
it('should return 1 for nextjs-page', () => {
|
|
913
|
-
const result = adjustCohesionForClassification(0.25, 'nextjs-page');
|
|
914
|
-
expect(result).toBe(1);
|
|
915
|
-
});
|
|
916
|
-
});
|
|
917
|
-
|
|
918
|
-
describe('nextjs-page fragmentation adjustment', () => {
|
|
919
|
-
it('should reduce fragmentation by 80% for nextjs-page', () => {
|
|
920
|
-
const result = adjustFragmentationForClassification(0.5, 'nextjs-page');
|
|
921
|
-
expect(result).toBe(0.1); // 0.5 * 0.2
|
|
922
|
-
});
|
|
923
|
-
});
|
|
924
|
-
|
|
925
|
-
describe('nextjs-page recommendations', () => {
|
|
926
|
-
it('should provide nextjs-page recommendations', () => {
|
|
927
|
-
const recommendations = getClassificationRecommendations(
|
|
928
|
-
'nextjs-page',
|
|
929
|
-
'app/calculator/page.tsx',
|
|
930
|
-
['Low cohesion']
|
|
931
|
-
);
|
|
932
|
-
expect(recommendations).toContain(
|
|
933
|
-
'Next.js App Router page detected - metadata/JSON-LD/component pattern is cohesive'
|
|
934
|
-
);
|
|
935
|
-
});
|
|
936
|
-
});
|
|
8
|
+
import { createNode } from './file-classification.fixtures';
|
|
9
|
+
|
|
10
|
+
describe('file classification core', () => {
|
|
11
|
+
it('should classify barrel export files correctly', () => {
|
|
12
|
+
const node = createNode({
|
|
13
|
+
file: 'src/index.ts',
|
|
14
|
+
imports: ['../module1', '../module2', '../module3'],
|
|
15
|
+
exports: [
|
|
16
|
+
{ name: 'func1', type: 'function', inferredDomain: 'module1' },
|
|
17
|
+
{ name: 'func2', type: 'function', inferredDomain: 'module2' },
|
|
18
|
+
{ name: 'func3', type: 'function', inferredDomain: 'module3' },
|
|
19
|
+
],
|
|
20
|
+
linesOfCode: 20,
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
const classification = classifyFile(node, 0.5, [
|
|
24
|
+
'module1',
|
|
25
|
+
'module2',
|
|
26
|
+
'module3',
|
|
27
|
+
]);
|
|
28
|
+
expect(classification).toBe('barrel-export');
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
// ... (keeping only a few core tests here)
|
|
937
32
|
});
|