@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,97 +0,0 @@
1
- import { basename, dirname } from 'path';
2
- import { AnalysisReport, RefactorRule, RefactorStep, FileOperation } from '../types.js';
3
-
4
- /**
5
- * Barrel Optimizer Rule (Tier 1)
6
- * Analyzes barrel files (__init__.py, index.ts) and suggests optimization.
7
- * Barrel files that re-export everything create unnecessary coupling.
8
- */
9
- export class BarrelOptimizerRule implements RefactorRule {
10
- name = 'barrel-optimizer';
11
- tier = 1 as const;
12
-
13
- private static readonly BARREL_FILES = new Set([
14
- '__init__.py', 'index.ts', 'index.js', 'index.tsx', 'index.jsx',
15
- ]);
16
-
17
- analyze(report: AnalysisReport, projectPath: string): RefactorStep[] {
18
- const steps: RefactorStep[] = [];
19
-
20
- // Find barrel files in the dependency graph
21
- const barrelNodes = report.dependencyGraph.nodes.filter((n) =>
22
- BarrelOptimizerRule.BARREL_FILES.has(basename(n))
23
- );
24
-
25
- for (const barrel of barrelNodes) {
26
- // Count how many things this barrel re-exports (outgoing edges)
27
- const outgoing = report.dependencyGraph.edges.filter(
28
- (e) => e.from === barrel
29
- );
30
- const incoming = report.dependencyGraph.edges.filter(
31
- (e) => e.to === barrel
32
- );
33
-
34
- if (outgoing.length < 3) continue;
35
-
36
- // Check for "pass-through" pattern: files import from barrel
37
- // but barrel just re-exports from siblings
38
- const siblingDir = dirname(barrel);
39
- const siblingExports = outgoing.filter(
40
- (e) => dirname(e.to) === siblingDir
41
- );
42
-
43
- const operations: FileOperation[] = [];
44
-
45
- // Suggest direct imports instead of barrel
46
- for (const consumer of incoming) {
47
- const consumedModules = outgoing
48
- .filter((e) => {
49
- // Check if consumer actually needs this module
50
- return report.dependencyGraph.edges.some(
51
- (edge) => edge.from === consumer.from && edge.to === e.to
52
- );
53
- })
54
- .map((e) => e.to);
55
-
56
- if (consumedModules.length > 0) {
57
- operations.push({
58
- type: 'MODIFY',
59
- path: consumer.from,
60
- description: `Replace barrel import from \`${basename(barrel)}\` with direct imports: ${consumedModules.map((m) => basename(m)).join(', ')}`,
61
- });
62
- }
63
- }
64
-
65
- // Suggest simplifying the barrel
66
- if (siblingExports.length > 5) {
67
- operations.push({
68
- type: 'MODIFY',
69
- path: barrel,
70
- description: `Simplify ${basename(barrel)}: only re-export public API (${siblingExports.length} re-exports detected, consider reducing)`,
71
- });
72
- }
73
-
74
- if (operations.length > 0) {
75
- steps.push({
76
- id: 0,
77
- tier: 1,
78
- rule: this.name,
79
- priority: outgoing.length >= 8 ? 'HIGH' : 'MEDIUM',
80
- title: `Optimize barrel: ${barrel}`,
81
- description: `\`${barrel}\` re-exports ${outgoing.length} modules. ` +
82
- `This creates a "Shotgun Surgery" risk — any change propagates widely.`,
83
- rationale: `Barrel files that re-export everything make it hard to tree-shake unused code ` +
84
- `and create implicit dependencies. Direct imports make dependency relationships explicit ` +
85
- `and reduce the blast radius of changes.`,
86
- operations,
87
- scoreImpact: [
88
- { metric: 'coupling', before: report.score.breakdown.coupling, after: Math.min(95, report.score.breakdown.coupling + 10) },
89
- { metric: 'layering', before: report.score.breakdown.layering, after: Math.min(95, report.score.breakdown.layering + 5) },
90
- ],
91
- });
92
- }
93
- }
94
-
95
- return steps;
96
- }
97
- }
@@ -1,132 +0,0 @@
1
- import { basename } from 'path';
2
- import { AnalysisReport, RefactorRule, RefactorStep, FileOperation } from '../types.js';
3
-
4
- /**
5
- * Dead Code Detector Rule (Tier 1)
6
- * Finds files with no incoming edges (nobody imports them)
7
- * and exports that are never used.
8
- *
9
- * Handles both path-style (deepguard/report.py) and
10
- * dot-notation (deepguard.report) references.
11
- */
12
- export class DeadCodeDetectorRule implements RefactorRule {
13
- name = 'dead-code-detector';
14
- tier = 1 as const;
15
-
16
- private static readonly ENTRY_POINTS = new Set([
17
- 'main.py', 'cli.py', 'app.py', 'manage.py', 'wsgi.py', 'asgi.py',
18
- 'main.ts', 'main.js', 'app.ts', 'app.js', 'server.ts', 'server.js',
19
- 'index.html', 'setup.py', 'setup.cfg', 'pyproject.toml',
20
- ]);
21
-
22
- analyze(report: AnalysisReport, projectPath: string): RefactorStep[] {
23
- const steps: RefactorStep[] = [];
24
- const edges = report.dependencyGraph.edges;
25
-
26
- // Build a set of ALL referenced targets (both path and dot-notation)
27
- const allTargets = new Set<string>();
28
- const allSources = new Set<string>();
29
- for (const edge of edges) {
30
- allTargets.add(edge.to);
31
- allSources.add(edge.from);
32
- }
33
-
34
- // Only consider actual files (with path separators) as candidates
35
- const fileNodes = report.dependencyGraph.nodes.filter(
36
- (n) => n.includes('/') || n.includes('\\')
37
- );
38
-
39
- // Build incoming edge count considering dot-notation matches
40
- const incomingCount: Record<string, number> = {};
41
-
42
- for (const file of fileNodes) {
43
- incomingCount[file] = 0;
44
-
45
- // Direct incoming edges
46
- for (const edge of edges) {
47
- if (edge.to === file) {
48
- incomingCount[file]++;
49
- }
50
- }
51
-
52
- // Check dot-notation references:
53
- // deepguard/report.py might be referenced as deepguard.report or .report
54
- const dotVariants = this.getDotVariants(file);
55
- for (const variant of dotVariants) {
56
- if (allTargets.has(variant)) {
57
- incomingCount[file]++;
58
- }
59
- }
60
- }
61
-
62
- // Find orphan files
63
- const orphans: string[] = [];
64
-
65
- for (const [file, count] of Object.entries(incomingCount)) {
66
- const fileName = basename(file);
67
-
68
- // Skip entry points and config files
69
- if (DeadCodeDetectorRule.ENTRY_POINTS.has(fileName)) continue;
70
- if (fileName.startsWith('__')) continue;
71
- if (fileName.startsWith('.')) continue;
72
- if (fileName.endsWith('.test.ts') || fileName.endsWith('.spec.ts')) continue;
73
- if (fileName.endsWith('_test.py') || fileName.endsWith('.test.py')) continue;
74
-
75
- // Also skip if the file has outgoing edges (it's active code)
76
- if (allSources.has(file)) continue;
77
-
78
- if (count === 0) {
79
- orphans.push(file);
80
- }
81
- }
82
-
83
- if (orphans.length > 0) {
84
- const operations: FileOperation[] = orphans.map((file) => ({
85
- type: 'DELETE' as const,
86
- path: file,
87
- description: `\`${basename(file)}\` has no incoming dependencies — verify if still needed`,
88
- }));
89
-
90
- steps.push({
91
- id: 0,
92
- tier: 1,
93
- rule: this.name,
94
- priority: orphans.length >= 3 ? 'MEDIUM' : 'LOW',
95
- title: `Review ${orphans.length} potentially unused file(s)`,
96
- description: `Found ${orphans.length} file(s) with no incoming dependencies: ` +
97
- `${orphans.map((f) => `\`${basename(f)}\``).join(', ')}. ` +
98
- `These may be dead code or missing from the module's public API.`,
99
- rationale: `Files with zero incoming edges are either entry points (already excluded), ` +
100
- `or potentially dead code. Removing dead code reduces maintenance burden ` +
101
- `and improves modularity scores.`,
102
- operations,
103
- scoreImpact: [
104
- { metric: 'modularity', before: report.score.breakdown.modularity, after: Math.min(95, report.score.breakdown.modularity + 5) },
105
- ],
106
- });
107
- }
108
-
109
- return steps;
110
- }
111
-
112
- /**
113
- * Generate dot-notation variants for a file path.
114
- * "deepguard/report.py" → ["deepguard.report", ".report"]
115
- */
116
- private getDotVariants(filePath: string): string[] {
117
- const variants: string[] = [];
118
- const withoutExt = filePath.replace(/\.[^.]+$/, '');
119
- const dotPath = withoutExt.replace(/[/\\]/g, '.');
120
-
121
- variants.push(dotPath);
122
-
123
- // Relative dot-notation: .report
124
- const parts = filePath.split('/');
125
- if (parts.length >= 2) {
126
- const lastPart = parts[parts.length - 1].replace(/\.[^.]+$/, '');
127
- variants.push(`.${lastPart}`);
128
- }
129
-
130
- return variants;
131
- }
132
- }
@@ -1,123 +0,0 @@
1
- import { basename, dirname, join } from 'path';
2
- import { AnalysisReport, RefactorRule, RefactorStep, FileOperation } from '../types.js';
3
-
4
- /**
5
- * Hub Splitter Rule (Tier 1)
6
- * Detects files with many connections and generates split plans.
7
- * A "hub" is a file that many other files depend on, creating tight coupling.
8
- */
9
- export class HubSplitterRule implements RefactorRule {
10
- name = 'hub-splitter';
11
- tier = 1 as const;
12
-
13
- analyze(report: AnalysisReport, projectPath: string): RefactorStep[] {
14
- const steps: RefactorStep[] = [];
15
-
16
- // Count connections per node
17
- const connectionCount: Record<string, { incoming: string[]; outgoing: string[] }> = {};
18
-
19
- for (const edge of report.dependencyGraph.edges) {
20
- if (!connectionCount[edge.from]) connectionCount[edge.from] = { incoming: [], outgoing: [] };
21
- if (!connectionCount[edge.to]) connectionCount[edge.to] = { incoming: [], outgoing: [] };
22
- connectionCount[edge.from].outgoing.push(edge.to);
23
- connectionCount[edge.to].incoming.push(edge.from);
24
- }
25
-
26
- // Find hubs (5+ incoming connections, not barrel files)
27
- const barrelFiles = new Set(['__init__.py', 'index.ts', 'index.js', 'index.tsx', 'mod.rs']);
28
-
29
- for (const [file, connections] of Object.entries(connectionCount)) {
30
- const fileName = basename(file);
31
- if (barrelFiles.has(fileName)) continue;
32
- if (connections.incoming.length < 5) continue;
33
-
34
- const operations: FileOperation[] = [];
35
-
36
- // Determine if this is a dot-notation module or a real file
37
- const isDotNotation = !file.includes('/') && !file.includes('\\');
38
- const moduleName = isDotNotation
39
- ? file.split('.').pop() || file
40
- : fileName.replace(/\.[^.]+$/, '');
41
- const moduleDir = isDotNotation
42
- ? file.split('.').slice(0, -1).join('/')
43
- : dirname(file);
44
- const ext = isDotNotation ? 'py' : (fileName.split('.').pop() || 'py');
45
-
46
- // Analyze what dependents import to suggest groupings
47
- const dependentGroups = this.groupDependents(connections.incoming);
48
-
49
- // Suggest splitting into domain modules
50
- if (dependentGroups.length >= 2) {
51
- for (const group of dependentGroups) {
52
- const newFileName = `${moduleName}_${group.name}.${ext}`;
53
- const newPath = moduleDir ? `${moduleDir}/${newFileName}` : newFileName;
54
-
55
- operations.push({
56
- type: 'CREATE',
57
- path: newPath,
58
- description: `Create \`${newFileName}\` with functionality used by: ${group.dependents.join(', ')}`,
59
- content: ext === 'py'
60
- ? `"""${moduleName}_${group.name} — extracted from ${moduleName}."""\n# Used by: ${group.dependents.join(', ')}\n`
61
- : `// ${moduleName}_${group.name} — extracted from ${moduleName}\n// Used by: ${group.dependents.join(', ')}\n`,
62
- });
63
- }
64
-
65
- // Update imports in all dependents
66
- for (const dependent of connections.incoming) {
67
- operations.push({
68
- type: 'MODIFY',
69
- path: dependent,
70
- description: `Update imports in \`${basename(dependent)}\` to use new split modules`,
71
- });
72
- }
73
-
74
- // Mark original for refactoring
75
- operations.push({
76
- type: 'MODIFY',
77
- path: isDotNotation ? `${moduleDir}/${moduleName}.${ext}` : file,
78
- description: `Refactor \`${moduleName}.${ext}\` — extract grouped functionality to new modules`,
79
- });
80
- }
81
-
82
- if (operations.length > 0) {
83
- steps.push({
84
- id: 0,
85
- tier: 1,
86
- rule: this.name,
87
- priority: connections.incoming.length >= 8 ? 'CRITICAL' : 'HIGH',
88
- title: `Split hub file: ${moduleName}.${ext}`,
89
- description: `\`${file}\` has ${connections.incoming.length} incoming connections. ` +
90
- `Split into ${dependentGroups.length} focused modules to reduce coupling.`,
91
- rationale: `High fan-in (${connections.incoming.length} files depend on this) creates a bottleneck. ` +
92
- `Changes to this file ripple to ${connections.incoming.length} other files. ` +
93
- `Splitting by usage pattern reduces blast radius.`,
94
- operations,
95
- scoreImpact: [
96
- { metric: 'coupling', before: report.score.breakdown.coupling, after: Math.min(95, report.score.breakdown.coupling + 15) },
97
- ],
98
- });
99
- }
100
- }
101
-
102
- return steps;
103
- }
104
-
105
- private groupDependents(
106
- dependents: string[]
107
- ): Array<{ name: string; dependents: string[] }> {
108
- // Group by top-level directory
109
- const groups: Record<string, string[]> = {};
110
-
111
- for (const dep of dependents) {
112
- const parts = dep.includes('/') ? dep.split('/') : dep.split('.');
113
- const groupName = parts.length >= 2 ? parts[parts.length - 2] : 'core';
114
- if (!groups[groupName]) groups[groupName] = [];
115
- groups[groupName].push(basename(dep));
116
- }
117
-
118
- return Object.entries(groups).map(([name, deps]) => ({
119
- name,
120
- dependents: deps,
121
- }));
122
- }
123
- }
@@ -1,98 +0,0 @@
1
- import { basename, dirname } from 'path';
2
- import { AnalysisReport, RefactorRule, RefactorStep, FileOperation } from '../types.js';
3
-
4
- /**
5
- * Import Organizer Rule (Tier 1)
6
- * Detects files that import from too many different modules (cross-boundary).
7
- * Suggests dependency injection or facade patterns.
8
- */
9
- export class ImportOrganizerRule implements RefactorRule {
10
- name = 'import-organizer';
11
- tier = 1 as const;
12
-
13
- analyze(report: AnalysisReport, projectPath: string): RefactorStep[] {
14
- const steps: RefactorStep[] = [];
15
-
16
- // Find files that import from many different directories
17
- const crossBoundary: Record<string, { targets: Set<string>; dirs: Set<string> }> = {};
18
-
19
- for (const edge of report.dependencyGraph.edges) {
20
- const fromDir = dirname(edge.from);
21
- const toDir = dirname(edge.to);
22
-
23
- if (!crossBoundary[edge.from]) {
24
- crossBoundary[edge.from] = { targets: new Set(), dirs: new Set() };
25
- }
26
- crossBoundary[edge.from].targets.add(edge.to);
27
-
28
- if (fromDir !== toDir) {
29
- crossBoundary[edge.from].dirs.add(toDir);
30
- }
31
- }
32
-
33
- // Files importing from 3+ different directories
34
- const violators = Object.entries(crossBoundary)
35
- .filter(([_, data]) => data.dirs.size >= 3)
36
- .sort((a, b) => b[1].dirs.size - a[1].dirs.size);
37
-
38
- for (const [file, data] of violators) {
39
- const operations: FileOperation[] = [];
40
- const fileName = basename(file);
41
- const fileDir = dirname(file);
42
-
43
- // Suggest creating a facade/service layer
44
- const ext = fileName.split('.').pop() || 'py';
45
- const nameBase = fileName.replace(/\.[^.]+$/, '');
46
- const facadePath = `${fileDir}/${nameBase}_deps.${ext}`;
47
-
48
- operations.push({
49
- type: 'CREATE',
50
- path: facadePath,
51
- description: `Create dependency facade \`${basename(facadePath)}\` — centralizes ${data.dirs.size} cross-module imports`,
52
- content: this.generateFacadeContent(ext, Array.from(data.targets), Array.from(data.dirs)),
53
- });
54
-
55
- operations.push({
56
- type: 'MODIFY',
57
- path: file,
58
- description: `Refactor \`${fileName}\` to import from local facade instead of ${data.dirs.size} different modules`,
59
- });
60
-
61
- steps.push({
62
- id: 0,
63
- tier: 1,
64
- rule: this.name,
65
- priority: data.dirs.size >= 5 ? 'HIGH' : 'MEDIUM',
66
- title: `Reduce cross-boundary imports: ${fileName}`,
67
- description: `\`${file}\` imports from ${data.dirs.size} different modules: ` +
68
- `${Array.from(data.dirs).map((d) => `\`${d}\``).join(', ')}. ` +
69
- `Consider using a facade or dependency injection.`,
70
- rationale: `Files with imports scattered across many modules have high afferent coupling. ` +
71
- `A facade centralizes these dependencies, making the file easier to test (mock one facade) ` +
72
- `and reducing the impact of changes in dependent modules.`,
73
- operations,
74
- scoreImpact: [
75
- { metric: 'cohesion', before: report.score.breakdown.cohesion, after: Math.min(95, report.score.breakdown.cohesion + 5) },
76
- { metric: 'coupling', before: report.score.breakdown.coupling, after: Math.min(95, report.score.breakdown.coupling + 5) },
77
- ],
78
- });
79
- }
80
-
81
- return steps;
82
- }
83
-
84
- private generateFacadeContent(ext: string, targets: string[], dirs: string[]): string {
85
- if (ext === 'py') {
86
- const imports = targets
87
- .map((t) => `# from ${t.replace(/\//g, '.')} import ...`)
88
- .join('\n');
89
- return `"""Dependency facade — centralizes cross-module imports."""\n\n${imports}\n\n# Re-export what ${dirs.length} modules need\n`;
90
- }
91
-
92
- // JS/TS
93
- const imports = targets
94
- .map((t) => `// export { ... } from '${t}';`)
95
- .join('\n');
96
- return `/**\n * Dependency facade — centralizes cross-module imports.\n */\n\n${imports}\n`;
97
- }
98
- }
@@ -1,124 +0,0 @@
1
- import { basename, dirname, join } from 'path';
2
- import { AnalysisReport, RefactorRule, RefactorStep, FileOperation } from '../types.js';
3
-
4
- /**
5
- * Module Grouper Rule (Tier 1)
6
- * Analyzes which files are frequently imported together and suggests
7
- * grouping them into cohesive modules/packages.
8
- */
9
- export class ModuleGrouperRule implements RefactorRule {
10
- name = 'module-grouper';
11
- tier = 1 as const;
12
-
13
- analyze(report: AnalysisReport, projectPath: string): RefactorStep[] {
14
- const steps: RefactorStep[] = [];
15
-
16
- // Build co-import matrix: which files are imported together?
17
- const coImportCount: Record<string, Record<string, number>> = {};
18
-
19
- // For each source file, see what it imports
20
- const importsBySource: Record<string, string[]> = {};
21
- for (const edge of report.dependencyGraph.edges) {
22
- if (!importsBySource[edge.from]) importsBySource[edge.from] = [];
23
- importsBySource[edge.from].push(edge.to);
24
- }
25
-
26
- // Count co-imports
27
- for (const [source, targets] of Object.entries(importsBySource)) {
28
- for (let i = 0; i < targets.length; i++) {
29
- for (let j = i + 1; j < targets.length; j++) {
30
- const a = targets[i];
31
- const b = targets[j];
32
- if (!coImportCount[a]) coImportCount[a] = {};
33
- if (!coImportCount[b]) coImportCount[b] = {};
34
- coImportCount[a][b] = (coImportCount[a][b] || 0) + 1;
35
- coImportCount[b][a] = (coImportCount[b][a] || 0) + 1;
36
- }
37
- }
38
- }
39
-
40
- // Find clusters: files that are always imported together
41
- const clusters: Array<{ files: string[]; coImportScore: number }> = [];
42
- const visited = new Set<string>();
43
-
44
- for (const [fileA, partners] of Object.entries(coImportCount)) {
45
- if (visited.has(fileA)) continue;
46
-
47
- const strongPartners = Object.entries(partners)
48
- .filter(([_, count]) => count >= 2)
49
- .sort((a, b) => b[1] - a[1]);
50
-
51
- if (strongPartners.length >= 2) {
52
- const cluster = [fileA, ...strongPartners.map(([f]) => f)];
53
- const inSameDir = cluster.every(
54
- (f) => dirname(f) === dirname(cluster[0])
55
- );
56
-
57
- // Only suggest if NOT already in the same directory
58
- if (!inSameDir) {
59
- const score = strongPartners.reduce((sum, [_, c]) => sum + c, 0);
60
- clusters.push({ files: cluster, coImportScore: score });
61
- cluster.forEach((f) => visited.add(f));
62
- }
63
- }
64
- }
65
-
66
- // Generate steps for each cluster
67
- for (const cluster of clusters.slice(0, 3)) {
68
- const operations: FileOperation[] = [];
69
- const clusterName = this.suggestModuleName(cluster.files);
70
- const targetDir = `${dirname(cluster.files[0])}/${clusterName}`;
71
-
72
- // Create new module directory
73
- operations.push({
74
- type: 'CREATE',
75
- path: `${targetDir}/__init__.py`,
76
- description: `Create new module \`${clusterName}/\` to group ${cluster.files.length} co-dependent files`,
77
- content: `"""Module ${clusterName} — grouped by co-import pattern."""\n`,
78
- });
79
-
80
- // Move files
81
- for (const file of cluster.files) {
82
- const newPath = join(targetDir, basename(file));
83
- operations.push({
84
- type: 'MOVE',
85
- path: file,
86
- newPath,
87
- description: `Move \`${basename(file)}\` → \`${clusterName}/${basename(file)}\``,
88
- });
89
- }
90
-
91
- steps.push({
92
- id: 0,
93
- tier: 1,
94
- rule: this.name,
95
- priority: 'MEDIUM',
96
- title: `Group co-dependent files into \`${clusterName}/\``,
97
- description: `Files ${cluster.files.map((f) => `\`${basename(f)}\``).join(', ')} ` +
98
- `are frequently imported together (co-import score: ${cluster.coImportScore}). ` +
99
- `Grouping them improves cohesion.`,
100
- rationale: `Files that are frequently imported together belong in the same module. ` +
101
- `This improves discoverability and reduces the cognitive load of understanding ` +
102
- `which files work together.`,
103
- operations,
104
- scoreImpact: [
105
- { metric: 'cohesion', before: report.score.breakdown.cohesion, after: Math.min(95, report.score.breakdown.cohesion + 10) },
106
- { metric: 'modularity', before: report.score.breakdown.modularity, after: Math.min(95, report.score.breakdown.modularity + 5) },
107
- ],
108
- });
109
- }
110
-
111
- return steps;
112
- }
113
-
114
- private suggestModuleName(files: string[]): string {
115
- // Try to infer a common theme from filenames
116
- const names = files.map((f) => basename(f).replace(/\.[^.]+$/, '').toLowerCase());
117
- const commonParts = names[0].split(/[_-]/).filter((part) =>
118
- names.every((n) => n.includes(part))
119
- );
120
-
121
- if (commonParts.length > 0) return commonParts[0];
122
- return 'shared';
123
- }
124
- }