@girardelli/architect 5.0.0 → 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.
Files changed (335) hide show
  1. package/dist/{cli.d.ts → src/adapters/cli.d.ts} +1 -2
  2. package/dist/{cli.js → src/adapters/cli.js} +191 -213
  3. package/dist/src/adapters/cli.js.map +1 -0
  4. package/dist/src/adapters/github-action.d.ts +9 -0
  5. package/dist/src/adapters/github-action.js +94 -0
  6. package/dist/src/adapters/github-action.js.map +1 -0
  7. package/dist/src/adapters/html-reporter/scripts.d.ts +5 -0
  8. package/dist/src/adapters/html-reporter/scripts.js +400 -0
  9. package/dist/src/adapters/html-reporter/scripts.js.map +1 -0
  10. package/dist/src/adapters/html-reporter/sections/agents.d.ts +2 -0
  11. package/dist/src/adapters/html-reporter/sections/agents.js +260 -0
  12. package/dist/src/adapters/html-reporter/sections/agents.js.map +1 -0
  13. package/dist/src/adapters/html-reporter/sections/anti-patterns.d.ts +13 -0
  14. package/dist/src/adapters/html-reporter/sections/anti-patterns.js +64 -0
  15. package/dist/src/adapters/html-reporter/sections/anti-patterns.js.map +1 -0
  16. package/dist/src/adapters/html-reporter/sections/header.d.ts +3 -0
  17. package/dist/src/adapters/html-reporter/sections/header.js +30 -0
  18. package/dist/src/adapters/html-reporter/sections/header.js.map +1 -0
  19. package/dist/src/adapters/html-reporter/sections/layers.d.ts +9 -0
  20. package/dist/src/adapters/html-reporter/sections/layers.js +143 -0
  21. package/dist/src/adapters/html-reporter/sections/layers.js.map +1 -0
  22. package/dist/src/adapters/html-reporter/sections/overview.d.ts +2 -0
  23. package/dist/src/adapters/html-reporter/sections/overview.js +58 -0
  24. package/dist/src/adapters/html-reporter/sections/overview.js.map +1 -0
  25. package/dist/src/adapters/html-reporter/sections/refactoring-plan.d.ts +3 -0
  26. package/dist/src/adapters/html-reporter/sections/refactoring-plan.js +151 -0
  27. package/dist/src/adapters/html-reporter/sections/refactoring-plan.js.map +1 -0
  28. package/dist/src/adapters/html-reporter/sections/score.d.ts +7 -0
  29. package/dist/src/adapters/html-reporter/sections/score.js +70 -0
  30. package/dist/src/adapters/html-reporter/sections/score.js.map +1 -0
  31. package/dist/src/adapters/html-reporter/sections/suggestions.d.ts +7 -0
  32. package/dist/src/adapters/html-reporter/sections/suggestions.js +34 -0
  33. package/dist/src/adapters/html-reporter/sections/suggestions.js.map +1 -0
  34. package/dist/src/adapters/html-reporter/styles.d.ts +1 -0
  35. package/dist/src/adapters/html-reporter/styles.js +526 -0
  36. package/dist/src/adapters/html-reporter/styles.js.map +1 -0
  37. package/dist/src/adapters/html-reporter/utils_adapters.d.ts +20 -0
  38. package/dist/src/adapters/html-reporter/utils_adapters.js +32 -0
  39. package/dist/src/adapters/html-reporter/utils_adapters.js.map +1 -0
  40. package/dist/src/adapters/html-reporter/utils_sections.d.ts +7 -0
  41. package/dist/src/adapters/html-reporter/utils_sections.js +58 -0
  42. package/dist/src/adapters/html-reporter/utils_sections.js.map +1 -0
  43. package/dist/src/adapters/html-reporter.d.ts +10 -0
  44. package/dist/src/adapters/html-reporter.js +97 -0
  45. package/dist/src/adapters/html-reporter.js.map +1 -0
  46. package/dist/src/adapters/progress-logger.d.ts +55 -0
  47. package/dist/src/adapters/progress-logger.js +200 -0
  48. package/dist/src/adapters/progress-logger.js.map +1 -0
  49. package/dist/{refactor-reporter.d.ts → src/adapters/refactor-reporter.d.ts} +1 -2
  50. package/dist/{refactor-reporter.js → src/adapters/refactor-reporter.js} +1 -1
  51. package/dist/src/adapters/refactor-reporter.js.map +1 -0
  52. package/dist/{reporter.d.ts → src/adapters/reporter.d.ts} +1 -2
  53. package/dist/src/adapters/reporter.js.map +1 -0
  54. package/dist/src/core/GenesisTerminal.d.ts +8 -0
  55. package/dist/src/core/GenesisTerminal.js +105 -0
  56. package/dist/src/core/GenesisTerminal.js.map +1 -0
  57. package/dist/{index.d.ts → src/core/architect.d.ts} +4 -18
  58. package/dist/{index.js → src/core/architect.js} +22 -21
  59. package/dist/src/core/architect.js.map +1 -0
  60. package/dist/tests/architect-adapter-enrichment.test.d.ts +1 -0
  61. package/dist/tests/architect-adapter-enrichment.test.js +11 -0
  62. package/dist/tests/architect-adapter-enrichment.test.js.map +1 -0
  63. package/dist/tests/github-action.test.d.ts +1 -0
  64. package/dist/tests/github-action.test.js +92 -0
  65. package/dist/tests/github-action.test.js.map +1 -0
  66. package/package.json +15 -65
  67. package/src/adapters/cli.ts +492 -0
  68. package/src/adapters/github-action.ts +109 -0
  69. package/src/adapters/html-reporter/scripts.ts +402 -0
  70. package/src/adapters/html-reporter/sections/agents.ts +267 -0
  71. package/src/adapters/html-reporter/sections/anti-patterns.ts +81 -0
  72. package/src/adapters/html-reporter/sections/header.ts +35 -0
  73. package/src/adapters/html-reporter/sections/layers.ts +165 -0
  74. package/src/adapters/html-reporter/sections/overview.ts +64 -0
  75. package/src/adapters/html-reporter/sections/refactoring-plan.ts +166 -0
  76. package/src/adapters/html-reporter/sections/score.ts +80 -0
  77. package/src/adapters/html-reporter/sections/suggestions.ts +39 -0
  78. package/src/adapters/html-reporter/styles.ts +525 -0
  79. package/src/adapters/html-reporter/utils_adapters.ts +39 -0
  80. package/src/adapters/html-reporter/utils_sections.ts +55 -0
  81. package/src/adapters/html-reporter.ts +102 -0
  82. package/src/adapters/progress-logger.ts +236 -0
  83. package/src/{refactor-reporter.ts → adapters/refactor-reporter.ts} +2 -2
  84. package/src/{reporter.ts → adapters/reporter.ts} +1 -1
  85. package/src/core/GenesisTerminal.ts +127 -0
  86. package/src/{index.ts → core/architect.ts} +27 -45
  87. package/tests/github-action.test.ts +109 -0
  88. package/tsconfig.json +12 -19
  89. package/CONTRIBUTING.md +0 -140
  90. package/LICENSE +0 -21
  91. package/PROJECT_STRUCTURE.txt +0 -168
  92. package/README.md +0 -257
  93. package/architect-run.sh +0 -431
  94. package/assets/banner-v3.html +0 -561
  95. package/dist/agent-generator/context-enricher.d.ts +0 -58
  96. package/dist/agent-generator/context-enricher.d.ts.map +0 -1
  97. package/dist/agent-generator/context-enricher.js +0 -613
  98. package/dist/agent-generator/context-enricher.js.map +0 -1
  99. package/dist/agent-generator/domain-inferrer.d.ts +0 -52
  100. package/dist/agent-generator/domain-inferrer.d.ts.map +0 -1
  101. package/dist/agent-generator/domain-inferrer.js +0 -585
  102. package/dist/agent-generator/domain-inferrer.js.map +0 -1
  103. package/dist/agent-generator/framework-detector.d.ts +0 -40
  104. package/dist/agent-generator/framework-detector.d.ts.map +0 -1
  105. package/dist/agent-generator/framework-detector.js +0 -611
  106. package/dist/agent-generator/framework-detector.js.map +0 -1
  107. package/dist/agent-generator/index.d.ts +0 -47
  108. package/dist/agent-generator/index.d.ts.map +0 -1
  109. package/dist/agent-generator/index.js +0 -545
  110. package/dist/agent-generator/index.js.map +0 -1
  111. package/dist/agent-generator/stack-detector.d.ts +0 -14
  112. package/dist/agent-generator/stack-detector.d.ts.map +0 -1
  113. package/dist/agent-generator/stack-detector.js +0 -124
  114. package/dist/agent-generator/stack-detector.js.map +0 -1
  115. package/dist/agent-generator/templates/core/agents.d.ts +0 -17
  116. package/dist/agent-generator/templates/core/agents.d.ts.map +0 -1
  117. package/dist/agent-generator/templates/core/agents.js +0 -1256
  118. package/dist/agent-generator/templates/core/agents.js.map +0 -1
  119. package/dist/agent-generator/templates/core/architecture-rules.d.ts +0 -7
  120. package/dist/agent-generator/templates/core/architecture-rules.d.ts.map +0 -1
  121. package/dist/agent-generator/templates/core/architecture-rules.js +0 -274
  122. package/dist/agent-generator/templates/core/architecture-rules.js.map +0 -1
  123. package/dist/agent-generator/templates/core/general-rules.d.ts +0 -8
  124. package/dist/agent-generator/templates/core/general-rules.d.ts.map +0 -1
  125. package/dist/agent-generator/templates/core/general-rules.js +0 -301
  126. package/dist/agent-generator/templates/core/general-rules.js.map +0 -1
  127. package/dist/agent-generator/templates/core/hooks-generator.d.ts +0 -21
  128. package/dist/agent-generator/templates/core/hooks-generator.d.ts.map +0 -1
  129. package/dist/agent-generator/templates/core/hooks-generator.js +0 -233
  130. package/dist/agent-generator/templates/core/hooks-generator.js.map +0 -1
  131. package/dist/agent-generator/templates/core/index-md.d.ts +0 -7
  132. package/dist/agent-generator/templates/core/index-md.d.ts.map +0 -1
  133. package/dist/agent-generator/templates/core/index-md.js +0 -246
  134. package/dist/agent-generator/templates/core/index-md.js.map +0 -1
  135. package/dist/agent-generator/templates/core/orchestrator.d.ts +0 -8
  136. package/dist/agent-generator/templates/core/orchestrator.d.ts.map +0 -1
  137. package/dist/agent-generator/templates/core/orchestrator.js +0 -422
  138. package/dist/agent-generator/templates/core/orchestrator.js.map +0 -1
  139. package/dist/agent-generator/templates/core/preflight.d.ts +0 -8
  140. package/dist/agent-generator/templates/core/preflight.d.ts.map +0 -1
  141. package/dist/agent-generator/templates/core/preflight.js +0 -213
  142. package/dist/agent-generator/templates/core/preflight.js.map +0 -1
  143. package/dist/agent-generator/templates/core/quality-gates.d.ts +0 -11
  144. package/dist/agent-generator/templates/core/quality-gates.d.ts.map +0 -1
  145. package/dist/agent-generator/templates/core/quality-gates.js +0 -254
  146. package/dist/agent-generator/templates/core/quality-gates.js.map +0 -1
  147. package/dist/agent-generator/templates/core/security-rules.d.ts +0 -7
  148. package/dist/agent-generator/templates/core/security-rules.d.ts.map +0 -1
  149. package/dist/agent-generator/templates/core/security-rules.js +0 -528
  150. package/dist/agent-generator/templates/core/security-rules.js.map +0 -1
  151. package/dist/agent-generator/templates/core/skills-generator.d.ts +0 -19
  152. package/dist/agent-generator/templates/core/skills-generator.d.ts.map +0 -1
  153. package/dist/agent-generator/templates/core/skills-generator.js +0 -546
  154. package/dist/agent-generator/templates/core/skills-generator.js.map +0 -1
  155. package/dist/agent-generator/templates/core/workflow-fix-bug.d.ts +0 -7
  156. package/dist/agent-generator/templates/core/workflow-fix-bug.d.ts.map +0 -1
  157. package/dist/agent-generator/templates/core/workflow-fix-bug.js +0 -237
  158. package/dist/agent-generator/templates/core/workflow-fix-bug.js.map +0 -1
  159. package/dist/agent-generator/templates/core/workflow-new-feature.d.ts +0 -8
  160. package/dist/agent-generator/templates/core/workflow-new-feature.d.ts.map +0 -1
  161. package/dist/agent-generator/templates/core/workflow-new-feature.js +0 -321
  162. package/dist/agent-generator/templates/core/workflow-new-feature.js.map +0 -1
  163. package/dist/agent-generator/templates/core/workflow-review.d.ts +0 -7
  164. package/dist/agent-generator/templates/core/workflow-review.d.ts.map +0 -1
  165. package/dist/agent-generator/templates/core/workflow-review.js +0 -104
  166. package/dist/agent-generator/templates/core/workflow-review.js.map +0 -1
  167. package/dist/agent-generator/templates/domain/index.d.ts +0 -22
  168. package/dist/agent-generator/templates/domain/index.d.ts.map +0 -1
  169. package/dist/agent-generator/templates/domain/index.js +0 -1176
  170. package/dist/agent-generator/templates/domain/index.js.map +0 -1
  171. package/dist/agent-generator/templates/stack/index.d.ts +0 -8
  172. package/dist/agent-generator/templates/stack/index.d.ts.map +0 -1
  173. package/dist/agent-generator/templates/stack/index.js +0 -695
  174. package/dist/agent-generator/templates/stack/index.js.map +0 -1
  175. package/dist/agent-generator/templates/template-helpers.d.ts +0 -75
  176. package/dist/agent-generator/templates/template-helpers.d.ts.map +0 -1
  177. package/dist/agent-generator/templates/template-helpers.js +0 -726
  178. package/dist/agent-generator/templates/template-helpers.js.map +0 -1
  179. package/dist/agent-generator/types.d.ts +0 -196
  180. package/dist/agent-generator/types.d.ts.map +0 -1
  181. package/dist/agent-generator/types.js +0 -27
  182. package/dist/agent-generator/types.js.map +0 -1
  183. package/dist/analyzer.d.ts +0 -38
  184. package/dist/analyzer.d.ts.map +0 -1
  185. package/dist/analyzer.js +0 -383
  186. package/dist/analyzer.js.map +0 -1
  187. package/dist/analyzers/forecast.d.ts +0 -85
  188. package/dist/analyzers/forecast.d.ts.map +0 -1
  189. package/dist/analyzers/forecast.js +0 -337
  190. package/dist/analyzers/forecast.js.map +0 -1
  191. package/dist/analyzers/git-cache.d.ts +0 -7
  192. package/dist/analyzers/git-cache.d.ts.map +0 -1
  193. package/dist/analyzers/git-cache.js +0 -41
  194. package/dist/analyzers/git-cache.js.map +0 -1
  195. package/dist/analyzers/git-history.d.ts +0 -113
  196. package/dist/analyzers/git-history.d.ts.map +0 -1
  197. package/dist/analyzers/git-history.js +0 -333
  198. package/dist/analyzers/git-history.js.map +0 -1
  199. package/dist/analyzers/index.d.ts +0 -10
  200. package/dist/analyzers/index.d.ts.map +0 -1
  201. package/dist/analyzers/index.js +0 -7
  202. package/dist/analyzers/index.js.map +0 -1
  203. package/dist/analyzers/temporal-scorer.d.ts +0 -72
  204. package/dist/analyzers/temporal-scorer.d.ts.map +0 -1
  205. package/dist/analyzers/temporal-scorer.js +0 -140
  206. package/dist/analyzers/temporal-scorer.js.map +0 -1
  207. package/dist/anti-patterns.d.ts +0 -24
  208. package/dist/anti-patterns.d.ts.map +0 -1
  209. package/dist/anti-patterns.js +0 -230
  210. package/dist/anti-patterns.js.map +0 -1
  211. package/dist/cli.d.ts.map +0 -1
  212. package/dist/cli.js.map +0 -1
  213. package/dist/config.d.ts +0 -12
  214. package/dist/config.d.ts.map +0 -1
  215. package/dist/config.js +0 -110
  216. package/dist/config.js.map +0 -1
  217. package/dist/diagram.d.ts +0 -9
  218. package/dist/diagram.d.ts.map +0 -1
  219. package/dist/diagram.js +0 -116
  220. package/dist/diagram.js.map +0 -1
  221. package/dist/html-reporter.d.ts +0 -47
  222. package/dist/html-reporter.d.ts.map +0 -1
  223. package/dist/html-reporter.js +0 -1747
  224. package/dist/html-reporter.js.map +0 -1
  225. package/dist/index.d.ts.map +0 -1
  226. package/dist/index.js.map +0 -1
  227. package/dist/project-summarizer.d.ts +0 -38
  228. package/dist/project-summarizer.d.ts.map +0 -1
  229. package/dist/project-summarizer.js +0 -463
  230. package/dist/project-summarizer.js.map +0 -1
  231. package/dist/refactor-engine.d.ts +0 -18
  232. package/dist/refactor-engine.d.ts.map +0 -1
  233. package/dist/refactor-engine.js +0 -86
  234. package/dist/refactor-engine.js.map +0 -1
  235. package/dist/refactor-reporter.d.ts.map +0 -1
  236. package/dist/refactor-reporter.js.map +0 -1
  237. package/dist/reporter.d.ts.map +0 -1
  238. package/dist/reporter.js.map +0 -1
  239. package/dist/rules/barrel-optimizer.d.ts +0 -13
  240. package/dist/rules/barrel-optimizer.d.ts.map +0 -1
  241. package/dist/rules/barrel-optimizer.js +0 -77
  242. package/dist/rules/barrel-optimizer.js.map +0 -1
  243. package/dist/rules/dead-code-detector.d.ts +0 -21
  244. package/dist/rules/dead-code-detector.d.ts.map +0 -1
  245. package/dist/rules/dead-code-detector.js +0 -117
  246. package/dist/rules/dead-code-detector.js.map +0 -1
  247. package/dist/rules/hub-splitter.d.ts +0 -13
  248. package/dist/rules/hub-splitter.d.ts.map +0 -1
  249. package/dist/rules/hub-splitter.js +0 -110
  250. package/dist/rules/hub-splitter.js.map +0 -1
  251. package/dist/rules/import-organizer.d.ts +0 -13
  252. package/dist/rules/import-organizer.d.ts.map +0 -1
  253. package/dist/rules/import-organizer.js +0 -85
  254. package/dist/rules/import-organizer.js.map +0 -1
  255. package/dist/rules/module-grouper.d.ts +0 -13
  256. package/dist/rules/module-grouper.d.ts.map +0 -1
  257. package/dist/rules/module-grouper.js +0 -110
  258. package/dist/rules/module-grouper.js.map +0 -1
  259. package/dist/scanner.d.ts +0 -31
  260. package/dist/scanner.d.ts.map +0 -1
  261. package/dist/scanner.js +0 -328
  262. package/dist/scanner.js.map +0 -1
  263. package/dist/scorer.d.ts +0 -27
  264. package/dist/scorer.d.ts.map +0 -1
  265. package/dist/scorer.js +0 -229
  266. package/dist/scorer.js.map +0 -1
  267. package/dist/types.d.ts +0 -186
  268. package/dist/types.d.ts.map +0 -1
  269. package/dist/types.js +0 -2
  270. package/dist/types.js.map +0 -1
  271. package/examples/sample-report.md +0 -207
  272. package/jest.config.js +0 -18
  273. package/src/agent-generator/context-enricher.ts +0 -672
  274. package/src/agent-generator/domain-inferrer.ts +0 -635
  275. package/src/agent-generator/framework-detector.ts +0 -669
  276. package/src/agent-generator/index.ts +0 -634
  277. package/src/agent-generator/stack-detector.ts +0 -115
  278. package/src/agent-generator/templates/core/agents.ts +0 -1296
  279. package/src/agent-generator/templates/core/architecture-rules.ts +0 -287
  280. package/src/agent-generator/templates/core/general-rules.ts +0 -306
  281. package/src/agent-generator/templates/core/hooks-generator.ts +0 -242
  282. package/src/agent-generator/templates/core/index-md.ts +0 -260
  283. package/src/agent-generator/templates/core/orchestrator.ts +0 -459
  284. package/src/agent-generator/templates/core/preflight.ts +0 -215
  285. package/src/agent-generator/templates/core/quality-gates.ts +0 -256
  286. package/src/agent-generator/templates/core/security-rules.ts +0 -543
  287. package/src/agent-generator/templates/core/skills-generator.ts +0 -585
  288. package/src/agent-generator/templates/core/workflow-fix-bug.ts +0 -239
  289. package/src/agent-generator/templates/core/workflow-new-feature.ts +0 -323
  290. package/src/agent-generator/templates/core/workflow-review.ts +0 -106
  291. package/src/agent-generator/templates/domain/index.ts +0 -1201
  292. package/src/agent-generator/templates/stack/index.ts +0 -705
  293. package/src/agent-generator/templates/template-helpers.ts +0 -776
  294. package/src/agent-generator/types.ts +0 -232
  295. package/src/analyzer.ts +0 -447
  296. package/src/analyzers/forecast.ts +0 -496
  297. package/src/analyzers/git-cache.ts +0 -52
  298. package/src/analyzers/git-history.ts +0 -488
  299. package/src/analyzers/index.ts +0 -33
  300. package/src/analyzers/temporal-scorer.ts +0 -227
  301. package/src/anti-patterns.ts +0 -287
  302. package/src/cli.ts +0 -517
  303. package/src/config.ts +0 -123
  304. package/src/diagram.ts +0 -144
  305. package/src/html-reporter.ts +0 -1830
  306. package/src/project-summarizer.ts +0 -521
  307. package/src/refactor-engine.ts +0 -117
  308. package/src/rules/barrel-optimizer.ts +0 -97
  309. package/src/rules/dead-code-detector.ts +0 -132
  310. package/src/rules/hub-splitter.ts +0 -123
  311. package/src/rules/import-organizer.ts +0 -98
  312. package/src/rules/module-grouper.ts +0 -124
  313. package/src/scanner.ts +0 -344
  314. package/src/scorer.ts +0 -254
  315. package/src/types.ts +0 -193
  316. package/tests/agent-generator.test.ts +0 -427
  317. package/tests/analyzers-integration.test.ts +0 -174
  318. package/tests/anti-patterns.test.ts +0 -94
  319. package/tests/context-enricher.test.ts +0 -971
  320. package/tests/fixtures/monorepo/package.json +0 -6
  321. package/tests/fixtures/monorepo/packages/app/package.json +0 -12
  322. package/tests/fixtures/monorepo/packages/app/src/index.ts +0 -6
  323. package/tests/fixtures/monorepo/packages/core/package.json +0 -7
  324. package/tests/fixtures/monorepo/packages/core/src/index.ts +0 -7
  325. package/tests/forecast.test.ts +0 -509
  326. package/tests/framework-detector.test.ts +0 -1172
  327. package/tests/git-history.test.ts +0 -254
  328. package/tests/monorepo-scan.test.ts +0 -170
  329. package/tests/scanner.test.ts +0 -54
  330. package/tests/scorer.test.ts +0 -674
  331. package/tests/stack-detector.test.ts +0 -241
  332. package/tests/template-generation.test.ts +0 -706
  333. package/tests/template-helpers.test.ts +0 -1152
  334. package/tests/temporal-scorer.test.ts +0 -307
  335. /package/dist/{reporter.js → src/adapters/reporter.js} +0 -0
@@ -1,674 +0,0 @@
1
- import { ArchitectureScorer } from '../src/scorer.js';
2
- import { DependencyEdge, AntiPattern } from '../src/types.js';
3
-
4
- describe('ArchitectureScorer', () => {
5
- const scorer = new ArchitectureScorer();
6
-
7
- const mockEdges: DependencyEdge[] = [
8
- { from: 'src/a.ts', to: 'src/b.ts', type: 'import', weight: 1 },
9
- { from: 'src/b.ts', to: 'src/c.ts', type: 'import', weight: 1 },
10
- { from: 'src/c.ts', to: 'src/d.ts', type: 'import', weight: 1 },
11
- ];
12
-
13
- const mockAntiPatterns: AntiPattern[] = [
14
- {
15
- name: 'God Class',
16
- severity: 'CRITICAL',
17
- location: 'src/Manager.ts',
18
- description: 'Test',
19
- suggestion: 'Test',
20
- },
21
- ];
22
-
23
- describe('score', () => {
24
- it('should return a score between 0 and 100', () => {
25
- const result = scorer.score(mockEdges, mockAntiPatterns, 50);
26
-
27
- expect(result.overall).toBeGreaterThanOrEqual(0);
28
- expect(result.overall).toBeLessThanOrEqual(100);
29
- });
30
-
31
- it('should calculate component scores', () => {
32
- const result = scorer.score(mockEdges, mockAntiPatterns, 50);
33
-
34
- expect(result.components).toBeDefined();
35
- expect(result.components.length).toBeGreaterThan(0);
36
-
37
- for (const component of result.components) {
38
- expect(component.score).toBeGreaterThanOrEqual(0);
39
- expect(component.score).toBeLessThanOrEqual(100);
40
- expect(component.weight).toBeGreaterThan(0);
41
- }
42
- });
43
-
44
- it('should provide breakdown of component scores', () => {
45
- const result = scorer.score(mockEdges, mockAntiPatterns, 50);
46
-
47
- expect(result.breakdown).toBeDefined();
48
- expect(result.breakdown.modularity).toBeGreaterThanOrEqual(0);
49
- expect(result.breakdown.coupling).toBeGreaterThanOrEqual(0);
50
- expect(result.breakdown.cohesion).toBeGreaterThanOrEqual(0);
51
- expect(result.breakdown.layering).toBeGreaterThanOrEqual(0);
52
- });
53
-
54
- it('should handle empty edges', () => {
55
- const result = scorer.score([], mockAntiPatterns, 10);
56
-
57
- expect(result.overall).toBeGreaterThanOrEqual(0);
58
- expect(result.overall).toBeLessThanOrEqual(100);
59
- });
60
-
61
- it('should penalize anti-patterns in layering score', () => {
62
- const lotsOfPatterns: AntiPattern[] = Array(10)
63
- .fill(null)
64
- .map((_, i) => ({
65
- name: 'Test Pattern',
66
- severity: 'HIGH',
67
- location: `src/file${i}.ts`,
68
- description: 'Test',
69
- suggestion: 'Test',
70
- }));
71
-
72
- const resultWithPatterns = scorer.score(mockEdges, lotsOfPatterns, 50);
73
- const resultWithoutPatterns = scorer.score(mockEdges, [], 50);
74
-
75
- expect(resultWithPatterns.breakdown.layering).toBeLessThanOrEqual(
76
- resultWithoutPatterns.breakdown.layering
77
- );
78
- });
79
- });
80
-
81
- // ═══════════════════════════════════════════════════════════════════════
82
- // MODULARITY TESTS
83
- // ═══════════════════════════════════════════════════════════════════════
84
- describe('calculateModularity', () => {
85
- it('should score 50 when totalFiles is 0', () => {
86
- const result = scorer.score([], [], 0);
87
- expect(result.breakdown.modularity).toBe(50);
88
- });
89
-
90
- it('should score 95 when avgEdgesPerFile < 2', () => {
91
- // 1 file, 1 edge → avgEdgesPerFile = 1
92
- const edges: DependencyEdge[] = [
93
- { from: 'src/a.ts', to: 'src/b.ts', type: 'import', weight: 1 },
94
- ];
95
- const result = scorer.score(edges, [], 1);
96
- expect(result.breakdown.modularity).toBe(95);
97
- });
98
-
99
- it('should score 85 when avgEdgesPerFile is 2-4', () => {
100
- // 2 files, 5 edges → avgEdgesPerFile = 2.5
101
- const edges: DependencyEdge[] = [
102
- { from: 'src/a.ts', to: 'src/b.ts', type: 'import', weight: 1 },
103
- { from: 'src/b.ts', to: 'src/c.ts', type: 'import', weight: 1 },
104
- { from: 'src/c.ts', to: 'src/a.ts', type: 'import', weight: 1 },
105
- { from: 'src/a.ts', to: 'src/c.ts', type: 'import', weight: 1 },
106
- { from: 'src/b.ts', to: 'src/a.ts', type: 'import', weight: 1 },
107
- ];
108
- const result = scorer.score(edges, [], 2);
109
- expect(result.breakdown.modularity).toBe(85);
110
- });
111
-
112
- it('should score 70 when avgEdgesPerFile is 4-6', () => {
113
- // 2 files, 10 edges → avgEdgesPerFile = 5
114
- const edges: DependencyEdge[] = Array(10)
115
- .fill(null)
116
- .map((_, i) => ({
117
- from: `src/a.ts`,
118
- to: `src/b.ts`,
119
- type: 'import' as const,
120
- weight: 1,
121
- }));
122
- const result = scorer.score(edges, [], 2);
123
- expect(result.breakdown.modularity).toBe(70);
124
- });
125
-
126
- it('should score 50 when avgEdgesPerFile is 6-10', () => {
127
- // 2 files, 18 edges → avgEdgesPerFile = 9
128
- const edges: DependencyEdge[] = Array(18)
129
- .fill(null)
130
- .map((_, i) => ({
131
- from: i % 2 === 0 ? 'src/a.ts' : 'src/b.ts',
132
- to: i % 2 === 0 ? 'src/b.ts' : 'src/a.ts',
133
- type: 'import' as const,
134
- weight: 1,
135
- }));
136
- const result = scorer.score(edges, [], 2);
137
- expect(result.breakdown.modularity).toBe(50);
138
- });
139
-
140
- it('should score 30 when avgEdgesPerFile >= 10', () => {
141
- // 1 file, 10+ edges
142
- const edges: DependencyEdge[] = Array(11)
143
- .fill(null)
144
- .map((_, i) => ({
145
- from: 'src/a.ts',
146
- to: `src/file${i}.ts`,
147
- type: 'import' as const,
148
- weight: 1,
149
- }));
150
- const result = scorer.score(edges, [], 1);
151
- expect(result.breakdown.modularity).toBe(30);
152
- });
153
- });
154
-
155
- // ═══════════════════════════════════════════════════════════════════════
156
- // COUPLING TESTS
157
- // ═══════════════════════════════════════════════════════════════════════
158
- describe('calculateCoupling', () => {
159
- it('should score 50 when totalFiles is 0', () => {
160
- const result = scorer.score([], [], 0);
161
- expect(result.breakdown.coupling).toBe(50);
162
- });
163
-
164
- it('should score 50 when totalFiles is 1', () => {
165
- const edges: DependencyEdge[] = [
166
- { from: 'src/a.ts', to: 'src/b.ts', type: 'import', weight: 1 },
167
- ];
168
- const result = scorer.score(edges, [], 1);
169
- expect(result.breakdown.coupling).toBe(50);
170
- });
171
-
172
- it('should score 95 when couplingRatio < 0.15', () => {
173
- // 20 files, 1 file has 2 edges → ratio = 2/19 ≈ 0.105
174
- const edges: DependencyEdge[] = [
175
- { from: 'src/hub.ts', to: 'src/a.ts', type: 'import', weight: 1 },
176
- { from: 'src/hub.ts', to: 'src/b.ts', type: 'import', weight: 1 },
177
- { from: 'src/c.ts', to: 'src/d.ts', type: 'import', weight: 1 },
178
- ];
179
- const result = scorer.score(edges, [], 20);
180
- expect(result.breakdown.coupling).toBe(95);
181
- });
182
-
183
- it('should score 85 when couplingRatio is 0.15-0.25', () => {
184
- // 10 files, hub has 2 edges → ratio = 2/9 ≈ 0.222
185
- const edges: DependencyEdge[] = [
186
- { from: 'src/hub.ts', to: 'src/a.ts', type: 'import', weight: 1 },
187
- { from: 'src/hub.ts', to: 'src/b.ts', type: 'import', weight: 1 },
188
- ];
189
- const result = scorer.score(edges, [], 10);
190
- expect(result.breakdown.coupling).toBe(85);
191
- });
192
-
193
- it('should score 75 when couplingRatio is 0.25-0.35', () => {
194
- // 10 files, hub has 3 edges → ratio = 3/9 ≈ 0.333
195
- const edges: DependencyEdge[] = [
196
- { from: 'src/hub.ts', to: 'src/a.ts', type: 'import', weight: 1 },
197
- { from: 'src/hub.ts', to: 'src/b.ts', type: 'import', weight: 1 },
198
- { from: 'src/hub.ts', to: 'src/c.ts', type: 'import', weight: 1 },
199
- ];
200
- const result = scorer.score(edges, [], 10);
201
- expect(result.breakdown.coupling).toBe(75);
202
- });
203
-
204
- it('should score 65 when couplingRatio is 0.35-0.5', () => {
205
- // 10 files, hub has 4 edges → ratio = 4/9 ≈ 0.444
206
- const edges: DependencyEdge[] = [
207
- { from: 'src/hub.ts', to: 'src/a.ts', type: 'import', weight: 1 },
208
- { from: 'src/hub.ts', to: 'src/b.ts', type: 'import', weight: 1 },
209
- { from: 'src/hub.ts', to: 'src/c.ts', type: 'import', weight: 1 },
210
- { from: 'src/hub.ts', to: 'src/d.ts', type: 'import', weight: 1 },
211
- ];
212
- const result = scorer.score(edges, [], 10);
213
- expect(result.breakdown.coupling).toBe(65);
214
- });
215
-
216
- it('should score 50 when couplingRatio is 0.5-0.7', () => {
217
- // 10 files, hub has 6 edges → ratio = 6/9 ≈ 0.667
218
- const edges: DependencyEdge[] = Array(6)
219
- .fill(null)
220
- .map((_, i) => ({
221
- from: 'src/hub.ts',
222
- to: `src/file${i}.ts`,
223
- type: 'import' as const,
224
- weight: 1,
225
- }));
226
- const result = scorer.score(edges, [], 10);
227
- expect(result.breakdown.coupling).toBe(50);
228
- });
229
-
230
- it('should score 35 when couplingRatio is 0.7-0.85', () => {
231
- // 10 files, hub has 7 edges → ratio = 7/9 ≈ 0.778
232
- const edges: DependencyEdge[] = Array(7)
233
- .fill(null)
234
- .map((_, i) => ({
235
- from: 'src/hub.ts',
236
- to: `src/file${i}.ts`,
237
- type: 'import' as const,
238
- weight: 1,
239
- }));
240
- const result = scorer.score(edges, [], 10);
241
- expect(result.breakdown.coupling).toBe(35);
242
- });
243
-
244
- it('should score 20 when couplingRatio >= 0.85', () => {
245
- // 10 files, hub has 8 edges → ratio = 8/9 ≈ 0.889
246
- const edges: DependencyEdge[] = Array(8)
247
- .fill(null)
248
- .map((_, i) => ({
249
- from: 'src/hub.ts',
250
- to: `src/file${i}.ts`,
251
- type: 'import' as const,
252
- weight: 1,
253
- }));
254
- const result = scorer.score(edges, [], 10);
255
- expect(result.breakdown.coupling).toBe(20);
256
- });
257
-
258
- it('should exclude barrel files (index.ts) from coupling calculation', () => {
259
- // Barrel files should not count toward maxEdgeCount
260
- const edges: DependencyEdge[] = [
261
- { from: 'src/index.ts', to: 'src/a.ts', type: 'import', weight: 1 },
262
- { from: 'src/index.ts', to: 'src/b.ts', type: 'import', weight: 1 },
263
- { from: 'src/index.ts', to: 'src/c.ts', type: 'import', weight: 1 },
264
- { from: 'src/hub.ts', to: 'src/x.ts', type: 'import', weight: 1 },
265
- ];
266
- const result = scorer.score(edges, [], 10);
267
- // Only hub.ts (1 edge) should count, ratio = 1/9
268
- expect(result.breakdown.coupling).toBe(95);
269
- });
270
-
271
- it('should exclude barrel files (__init__.py) from coupling calculation', () => {
272
- const edges: DependencyEdge[] = [
273
- { from: 'src/__init__.py', to: 'src/a.py', type: 'import', weight: 1 },
274
- { from: 'src/__init__.py', to: 'src/b.py', type: 'import', weight: 1 },
275
- { from: 'src/__init__.py', to: 'src/c.py', type: 'import', weight: 1 },
276
- { from: 'src/__init__.py', to: 'src/d.py', type: 'import', weight: 1 },
277
- ];
278
- const result = scorer.score(edges, [], 10);
279
- // All edges from __init__.py should be filtered
280
- expect(result.breakdown.coupling).toBe(95);
281
- });
282
-
283
- it('should exclude barrel files (mod.rs) from coupling calculation', () => {
284
- const edges: DependencyEdge[] = [
285
- { from: 'src/mod.rs', to: 'src/a.rs', type: 'import', weight: 1 },
286
- { from: 'src/mod.rs', to: 'src/b.rs', type: 'import', weight: 1 },
287
- ];
288
- const result = scorer.score(edges, [], 10);
289
- expect(result.breakdown.coupling).toBe(95);
290
- });
291
-
292
- it('should exclude barrel files (__init__.pyi) from coupling calculation', () => {
293
- const edges: DependencyEdge[] = [
294
- { from: 'src/__init__.pyi', to: 'src/a.pyi', type: 'import', weight: 1 },
295
- { from: 'src/__init__.pyi', to: 'src/b.pyi', type: 'import', weight: 1 },
296
- ];
297
- const result = scorer.score(edges, [], 10);
298
- expect(result.breakdown.coupling).toBe(95);
299
- });
300
-
301
- it('should exclude barrel files as destinations', () => {
302
- // When the destination is index.ts, it should not count
303
- const edges: DependencyEdge[] = [
304
- { from: 'src/a.ts', to: 'src/index.ts', type: 'import', weight: 1 },
305
- { from: 'src/b.ts', to: 'src/index.ts', type: 'import', weight: 1 },
306
- { from: 'src/hub.ts', to: 'src/c.ts', type: 'import', weight: 1 },
307
- ];
308
- const result = scorer.score(edges, [], 10);
309
- expect(result.breakdown.coupling).toBe(95);
310
- });
311
- });
312
-
313
- // ═══════════════════════════════════════════════════════════════════════
314
- // COHESION TESTS
315
- // ═══════════════════════════════════════════════════════════════════════
316
- describe('calculateCohesion', () => {
317
- it('should score 50 when there are no edges', () => {
318
- const result = scorer.score([], [], 10);
319
- expect(result.breakdown.cohesion).toBe(50);
320
- });
321
-
322
- it('should score 95 when cohesionRatio > 0.8', () => {
323
- // 10 edges, 9 internal → ratio = 0.9
324
- const edges: DependencyEdge[] = [
325
- // Same package → internal
326
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
327
- { from: 'api/b.ts', to: 'api/c.ts', type: 'import', weight: 1 },
328
- { from: 'api/c.ts', to: 'api/d.ts', type: 'import', weight: 1 },
329
- { from: 'api/d.ts', to: 'api/e.ts', type: 'import', weight: 1 },
330
- { from: 'api/e.ts', to: 'api/f.ts', type: 'import', weight: 1 },
331
- { from: 'api/f.ts', to: 'api/g.ts', type: 'import', weight: 1 },
332
- { from: 'api/g.ts', to: 'api/h.ts', type: 'import', weight: 1 },
333
- { from: 'api/h.ts', to: 'api/i.ts', type: 'import', weight: 1 },
334
- { from: 'api/i.ts', to: 'api/j.ts', type: 'import', weight: 1 },
335
- // 1 external
336
- { from: 'api/k.ts', to: 'service/x.ts', type: 'import', weight: 1 },
337
- ];
338
- const result = scorer.score(edges, [], 50);
339
- expect(result.breakdown.cohesion).toBe(95);
340
- });
341
-
342
- it('should score 85 when cohesionRatio is 0.6-0.8', () => {
343
- // 10 edges, 7 internal → ratio = 0.7
344
- const edges: DependencyEdge[] = [
345
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
346
- { from: 'api/b.ts', to: 'api/c.ts', type: 'import', weight: 1 },
347
- { from: 'api/c.ts', to: 'api/d.ts', type: 'import', weight: 1 },
348
- { from: 'api/d.ts', to: 'api/e.ts', type: 'import', weight: 1 },
349
- { from: 'api/e.ts', to: 'api/f.ts', type: 'import', weight: 1 },
350
- { from: 'api/f.ts', to: 'api/g.ts', type: 'import', weight: 1 },
351
- { from: 'api/g.ts', to: 'api/h.ts', type: 'import', weight: 1 },
352
- // 3 external
353
- { from: 'api/x.ts', to: 'service/a.ts', type: 'import', weight: 1 },
354
- { from: 'api/y.ts', to: 'service/b.ts', type: 'import', weight: 1 },
355
- { from: 'api/z.ts', to: 'service/c.ts', type: 'import', weight: 1 },
356
- ];
357
- const result = scorer.score(edges, [], 50);
358
- expect(result.breakdown.cohesion).toBe(85);
359
- });
360
-
361
- it('should score 75 when cohesionRatio is 0.45-0.6', () => {
362
- // 10 edges, 5 internal → ratio = 0.5
363
- const edges: DependencyEdge[] = [
364
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
365
- { from: 'api/b.ts', to: 'api/c.ts', type: 'import', weight: 1 },
366
- { from: 'api/c.ts', to: 'api/d.ts', type: 'import', weight: 1 },
367
- { from: 'api/d.ts', to: 'api/e.ts', type: 'import', weight: 1 },
368
- { from: 'api/e.ts', to: 'api/f.ts', type: 'import', weight: 1 },
369
- // 5 external
370
- { from: 'api/w.ts', to: 'service/a.ts', type: 'import', weight: 1 },
371
- { from: 'api/x.ts', to: 'service/b.ts', type: 'import', weight: 1 },
372
- { from: 'api/y.ts', to: 'service/c.ts', type: 'import', weight: 1 },
373
- { from: 'api/z.ts', to: 'service/d.ts', type: 'import', weight: 1 },
374
- { from: 'api/q.ts', to: 'service/e.ts', type: 'import', weight: 1 },
375
- ];
376
- const result = scorer.score(edges, [], 50);
377
- expect(result.breakdown.cohesion).toBe(75);
378
- });
379
-
380
- it('should score 65 when cohesionRatio is 0.3-0.45', () => {
381
- // 10 edges, 4 internal → ratio = 0.4
382
- const edges: DependencyEdge[] = [
383
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
384
- { from: 'api/b.ts', to: 'api/c.ts', type: 'import', weight: 1 },
385
- { from: 'api/c.ts', to: 'api/d.ts', type: 'import', weight: 1 },
386
- { from: 'api/d.ts', to: 'api/e.ts', type: 'import', weight: 1 },
387
- // 6 external
388
- { from: 'api/w.ts', to: 'service/a.ts', type: 'import', weight: 1 },
389
- { from: 'api/x.ts', to: 'service/b.ts', type: 'import', weight: 1 },
390
- { from: 'api/y.ts', to: 'service/c.ts', type: 'import', weight: 1 },
391
- { from: 'api/z.ts', to: 'service/d.ts', type: 'import', weight: 1 },
392
- { from: 'api/q.ts', to: 'service/e.ts', type: 'import', weight: 1 },
393
- { from: 'api/r.ts', to: 'service/f.ts', type: 'import', weight: 1 },
394
- ];
395
- const result = scorer.score(edges, [], 50);
396
- expect(result.breakdown.cohesion).toBe(65);
397
- });
398
-
399
- it('should score 50 when cohesionRatio is 0.15-0.3', () => {
400
- // 10 edges, 2 internal → ratio = 0.2
401
- const edges: DependencyEdge[] = [
402
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
403
- { from: 'api/b.ts', to: 'api/c.ts', type: 'import', weight: 1 },
404
- // 8 external
405
- { from: 'api/v.ts', to: 'service/a.ts', type: 'import', weight: 1 },
406
- { from: 'api/w.ts', to: 'service/b.ts', type: 'import', weight: 1 },
407
- { from: 'api/x.ts', to: 'service/c.ts', type: 'import', weight: 1 },
408
- { from: 'api/y.ts', to: 'service/d.ts', type: 'import', weight: 1 },
409
- { from: 'api/z.ts', to: 'service/e.ts', type: 'import', weight: 1 },
410
- { from: 'api/q.ts', to: 'service/f.ts', type: 'import', weight: 1 },
411
- { from: 'api/r.ts', to: 'service/g.ts', type: 'import', weight: 1 },
412
- { from: 'api/s.ts', to: 'service/h.ts', type: 'import', weight: 1 },
413
- ];
414
- const result = scorer.score(edges, [], 50);
415
- expect(result.breakdown.cohesion).toBe(50);
416
- });
417
-
418
- it('should score 30 when cohesionRatio <= 0.15', () => {
419
- // 10 edges, 1 internal → ratio = 0.1
420
- const edges: DependencyEdge[] = [
421
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
422
- // 9 external
423
- { from: 'api/c.ts', to: 'service/a.ts', type: 'import', weight: 1 },
424
- { from: 'api/d.ts', to: 'service/b.ts', type: 'import', weight: 1 },
425
- { from: 'api/e.ts', to: 'service/c.ts', type: 'import', weight: 1 },
426
- { from: 'api/f.ts', to: 'service/d.ts', type: 'import', weight: 1 },
427
- { from: 'api/g.ts', to: 'service/e.ts', type: 'import', weight: 1 },
428
- { from: 'api/h.ts', to: 'service/f.ts', type: 'import', weight: 1 },
429
- { from: 'api/i.ts', to: 'service/g.ts', type: 'import', weight: 1 },
430
- { from: 'api/j.ts', to: 'service/h.ts', type: 'import', weight: 1 },
431
- { from: 'api/k.ts', to: 'service/i.ts', type: 'import', weight: 1 },
432
- ];
433
- const result = scorer.score(edges, [], 50);
434
- expect(result.breakdown.cohesion).toBe(30);
435
- });
436
- });
437
-
438
- // ═══════════════════════════════════════════════════════════════════════
439
- // INTERNAL DEPENDENCY TESTS
440
- // ═══════════════════════════════════════════════════════════════════════
441
- describe('isInternalDependency', () => {
442
- it('should return true when both files are root files (no directory)', () => {
443
- const edges: DependencyEdge[] = [
444
- { from: 'app.ts', to: 'main.ts', type: 'import', weight: 1 },
445
- ];
446
- const result = scorer.score(edges, [], 2);
447
- // Both root → internal → cohesion should be high
448
- expect(result.breakdown.cohesion).toBe(95);
449
- });
450
-
451
- it('should return true when files share same top-level directory', () => {
452
- const edges: DependencyEdge[] = [
453
- { from: 'api/routes.ts', to: 'api/handlers.ts', type: 'import', weight: 1 },
454
- ];
455
- const result = scorer.score(edges, [], 2);
456
- // Same package (api) → internal
457
- expect(result.breakdown.cohesion).toBe(95);
458
- });
459
-
460
- it('should return false when files are in different top-level directories', () => {
461
- const edges: DependencyEdge[] = [
462
- { from: 'api/routes.ts', to: 'service/handler.ts', type: 'import', weight: 1 },
463
- ];
464
- const result = scorer.score(edges, [], 2);
465
- // Different packages → external
466
- expect(result.breakdown.cohesion).toBe(30);
467
- });
468
-
469
- it('should handle Python package structure', () => {
470
- const edges: DependencyEdge[] = [
471
- { from: 'deepguard/cli.py', to: 'deepguard/analyzer.py', type: 'import', weight: 1 },
472
- ];
473
- const result = scorer.score(edges, [], 2);
474
- // Same package → internal
475
- expect(result.breakdown.cohesion).toBe(95);
476
- });
477
-
478
- it('should handle deeply nested paths - same top-level package', () => {
479
- const edges: DependencyEdge[] = [
480
- { from: 'mypackage/domain/entities/user.ts', to: 'mypackage/application/services/user.service.ts', type: 'import', weight: 1 },
481
- ];
482
- const result = scorer.score(edges, [], 2);
483
- // Both start with mypackage → internal
484
- expect(result.breakdown.cohesion).toBe(95);
485
- });
486
-
487
- it('should handle mixed package structures', () => {
488
- const edges: DependencyEdge[] = [
489
- { from: 'pkg1/file.ts', to: 'pkg2/file.ts', type: 'import', weight: 1 },
490
- { from: 'pkg1/a.ts', to: 'pkg1/b.ts', type: 'import', weight: 1 },
491
- ];
492
- const result = scorer.score(edges, [], 4);
493
- // 50% internal
494
- expect(result.breakdown.cohesion).toBe(75);
495
- });
496
- });
497
-
498
- // ═══════════════════════════════════════════════════════════════════════
499
- // LAYERING TESTS
500
- // ═══════════════════════════════════════════════════════════════════════
501
- describe('calculateLayering', () => {
502
- it('should score 95 when there are 0 violations', () => {
503
- const antiPatterns: AntiPattern[] = [
504
- {
505
- name: 'God Class',
506
- severity: 'CRITICAL',
507
- location: 'src/Manager.ts',
508
- description: 'Test',
509
- suggestion: 'Test',
510
- },
511
- ];
512
- const result = scorer.score([], antiPatterns, 10);
513
- expect(result.breakdown.layering).toBe(95);
514
- });
515
-
516
- it('should score 90 when there is 1 violation in a large project', () => {
517
- const antiPatterns: AntiPattern[] = [
518
- {
519
- name: 'Leaky Abstraction',
520
- severity: 'HIGH',
521
- location: 'src/api.ts',
522
- description: 'Test',
523
- suggestion: 'Test',
524
- },
525
- ];
526
- // 1 violation / 100 files = 1% ratio → score 90
527
- const result = scorer.score([], antiPatterns, 100);
528
- expect(result.breakdown.layering).toBe(90);
529
- });
530
-
531
- it('should score 80 when ratio is between 2-5%', () => {
532
- const antiPatterns: AntiPattern[] = [
533
- {
534
- name: 'Leaky Abstraction',
535
- severity: 'HIGH',
536
- location: 'src/api.ts',
537
- description: 'Test',
538
- suggestion: 'Test',
539
- },
540
- {
541
- name: 'Shotgun Surgery',
542
- severity: 'MEDIUM',
543
- location: 'src/service.ts',
544
- description: 'Test',
545
- suggestion: 'Test',
546
- },
547
- ];
548
- // 2 violations / 100 files = 2% ratio → score 80
549
- const result = scorer.score([], antiPatterns, 100);
550
- expect(result.breakdown.layering).toBe(80);
551
- });
552
-
553
- it('should score 65 when ratio is between 5-10%', () => {
554
- const antiPatterns: AntiPattern[] = [
555
- {
556
- name: 'Leaky Abstraction',
557
- severity: 'HIGH',
558
- location: 'src/api.ts',
559
- description: 'Test',
560
- suggestion: 'Test',
561
- },
562
- {
563
- name: 'Shotgun Surgery',
564
- severity: 'MEDIUM',
565
- location: 'src/service.ts',
566
- description: 'Test',
567
- suggestion: 'Test',
568
- },
569
- {
570
- name: 'Circular Dependency',
571
- severity: 'CRITICAL',
572
- location: 'src/model.ts',
573
- description: 'Test',
574
- suggestion: 'Test',
575
- },
576
- ];
577
- // 3 violations / 50 files = 6% ratio → score 65
578
- const result = scorer.score([], antiPatterns, 50);
579
- expect(result.breakdown.layering).toBe(65);
580
- });
581
-
582
- it('should score 50 when ratio is between 10-20%', () => {
583
- const antiPatterns: AntiPattern[] = [
584
- { name: 'Leaky Abstraction', severity: 'HIGH', location: 'src/api.ts', description: 'Test', suggestion: 'Test' },
585
- { name: 'Shotgun Surgery', severity: 'MEDIUM', location: 'src/service.ts', description: 'Test', suggestion: 'Test' },
586
- { name: 'Circular Dependency', severity: 'CRITICAL', location: 'src/model.ts', description: 'Test', suggestion: 'Test' },
587
- { name: 'Leaky Abstraction', severity: 'HIGH', location: 'src/data.ts', description: 'Test', suggestion: 'Test' },
588
- ];
589
- // 4 violations / 30 files = 13.3% ratio → score 50
590
- const result = scorer.score([], antiPatterns, 30);
591
- expect(result.breakdown.layering).toBe(50);
592
- });
593
-
594
- it('should score 20 when ratio exceeds 35%', () => {
595
- const antiPatterns: AntiPattern[] = [
596
- { name: 'Leaky Abstraction', severity: 'HIGH', location: 'src/a.ts', description: 'Test', suggestion: 'Test' },
597
- { name: 'Shotgun Surgery', severity: 'MEDIUM', location: 'src/b.ts', description: 'Test', suggestion: 'Test' },
598
- { name: 'Circular Dependency', severity: 'CRITICAL', location: 'src/c.ts', description: 'Test', suggestion: 'Test' },
599
- { name: 'Leaky Abstraction', severity: 'HIGH', location: 'src/d.ts', description: 'Test', suggestion: 'Test' },
600
- { name: 'Shotgun Surgery', severity: 'MEDIUM', location: 'src/e.ts', description: 'Test', suggestion: 'Test' },
601
- { name: 'Circular Dependency', severity: 'CRITICAL', location: 'src/f.ts', description: 'Test', suggestion: 'Test' },
602
- ];
603
- // 6 violations / 10 files = 60% ratio → score 20
604
- const result = scorer.score([], antiPatterns, 10);
605
- expect(result.breakdown.layering).toBe(20);
606
- });
607
-
608
- it('should only count specific violation types', () => {
609
- // Only count: 'Leaky Abstraction', 'Shotgun Surgery', 'Circular Dependency'
610
- const antiPatterns: AntiPattern[] = [
611
- { name: 'God Class', severity: 'CRITICAL', location: 'src/a.ts', description: 'Test', suggestion: 'Test' },
612
- { name: 'Leaky Abstraction', severity: 'HIGH', location: 'src/b.ts', description: 'Test', suggestion: 'Test' },
613
- { name: 'Feature Envy', severity: 'MEDIUM', location: 'src/c.ts', description: 'Test', suggestion: 'Test' },
614
- { name: 'Shotgun Surgery', severity: 'MEDIUM', location: 'src/d.ts', description: 'Test', suggestion: 'Test' },
615
- { name: 'Long Method', severity: 'LOW', location: 'src/e.ts', description: 'Test', suggestion: 'Test' },
616
- { name: 'Circular Dependency', severity: 'CRITICAL', location: 'src/f.ts', description: 'Test', suggestion: 'Test' },
617
- ];
618
- // Only Leaky Abstraction, Shotgun Surgery, Circular Dependency count = 3
619
- // 3 violations / 100 files = 3% ratio → score 80
620
- const result = scorer.score([], antiPatterns, 100);
621
- expect(result.breakdown.layering).toBe(80);
622
- });
623
- });
624
-
625
- // ═══════════════════════════════════════════════════════════════════════
626
- // OVERALL SCORE TESTS
627
- // ═══════════════════════════════════════════════════════════════════════
628
- describe('overall scoring and weighting', () => {
629
- it('should compute weighted average correctly', () => {
630
- // Create edges for known scores
631
- const edges: DependencyEdge[] = [
632
- { from: 'api/a.ts', to: 'api/b.ts', type: 'import', weight: 1 },
633
- ];
634
- const antiPatterns: AntiPattern[] = [];
635
-
636
- const result = scorer.score(edges, antiPatterns, 1);
637
-
638
- // With 1 file, 1 edge:
639
- // modularity = 95 (avgEdgesPerFile = 1)
640
- // coupling = 50 (1 totalFile)
641
- // cohesion = 95 (internal dependency)
642
- // layering = 95 (no violations)
643
- // overall = 95*0.4 + 50*0.25 + 95*0.2 + 95*0.15 = 38 + 12.5 + 19 + 14.25 = 83.75 ≈ 84
644
-
645
- expect(result.overall).toBeGreaterThanOrEqual(83);
646
- expect(result.overall).toBeLessThanOrEqual(84);
647
- });
648
-
649
- it('should clamp overall score to [0, 100]', () => {
650
- const result = scorer.score([], [], 0);
651
- expect(result.overall).toBeGreaterThanOrEqual(0);
652
- expect(result.overall).toBeLessThanOrEqual(100);
653
- });
654
-
655
- it('should round component scores', () => {
656
- const edges: DependencyEdge[] = Array(3)
657
- .fill(null)
658
- .map((_, i) => ({
659
- from: 'a.ts',
660
- to: `b${i}.ts`,
661
- type: 'import' as const,
662
- weight: 1,
663
- }));
664
-
665
- const result = scorer.score(edges, [], 1);
666
-
667
- // All component scores should be integers
668
- expect(Number.isInteger(result.breakdown.modularity)).toBe(true);
669
- expect(Number.isInteger(result.breakdown.coupling)).toBe(true);
670
- expect(Number.isInteger(result.breakdown.cohesion)).toBe(true);
671
- expect(Number.isInteger(result.breakdown.layering)).toBe(true);
672
- });
673
- });
674
- });