@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,521 +0,0 @@
1
- import { readFileSync, existsSync } from 'fs';
2
- import { join, basename, dirname } from 'path';
3
- import { AnalysisReport, ProjectSummary, FileNode, WorkspaceInfo } from './types.js';
4
-
5
- /**
6
- * ProjectSummarizer — infers what a project does from its metadata,
7
- * structure, README, package.json, and file naming conventions.
8
- *
9
- * v5.0: Workspace-aware module inference, package.json-first descriptions,
10
- * entry point detection from `bin` fields, and keyword blacklisting.
11
- */
12
- export class ProjectSummarizer {
13
-
14
- /** Keywords that should never appear in project summaries */
15
- private static readonly KEYWORD_BLACKLIST = new Set([
16
- 'node_modules', 'dist', 'build', '.git', '.next', 'coverage',
17
- '__tests__', '__mocks__', 'src', 'lib', 'index', 'main',
18
- 'out', 'tmp', '.cache', 'vendor', '.vscode', '.idea',
19
- ]);
20
-
21
- summarize(projectPath: string, report: AnalysisReport): ProjectSummary {
22
- const packageInfo = this.readPackageJson(projectPath);
23
- const readmeContent = this.readReadme(projectPath);
24
- const modules = this.inferModules(report, projectPath);
25
- const entryPoints = this.findEntryPoints(report, projectPath);
26
- const keywords = this.extractKeywords(packageInfo, readmeContent, modules, report);
27
- const techStack = this.buildTechStack(report, packageInfo);
28
- const description = this.buildDescription(packageInfo, readmeContent, report);
29
- const purpose = this.inferPurpose(keywords, modules, report);
30
-
31
- return {
32
- description,
33
- purpose,
34
- modules,
35
- techStack,
36
- entryPoints,
37
- keywords,
38
- };
39
- }
40
-
41
- // ── Package.json ──
42
-
43
- private readPackageJson(projectPath: string): Record<string, unknown> {
44
- const candidates = [
45
- join(projectPath, 'package.json'),
46
- join(projectPath, 'pyproject.toml'),
47
- join(projectPath, 'pubspec.yaml'),
48
- join(projectPath, 'Cargo.toml'),
49
- join(projectPath, 'pom.xml'),
50
- join(projectPath, 'build.gradle'),
51
- ];
52
-
53
- for (const candidate of candidates) {
54
- if (existsSync(candidate)) {
55
- try {
56
- if (candidate.endsWith('.json')) {
57
- return JSON.parse(readFileSync(candidate, 'utf-8'));
58
- }
59
- // For non-JSON, return raw content as 'raw' field
60
- return { raw: readFileSync(candidate, 'utf-8'), type: basename(candidate) };
61
- } catch {
62
- // ignore
63
- }
64
- }
65
- }
66
- return {};
67
- }
68
-
69
- private readReadme(projectPath: string): string {
70
- const candidates = ['README.md', 'readme.md', 'README.txt', 'README', 'README.rst'];
71
- for (const name of candidates) {
72
- const path = join(projectPath, name);
73
- if (existsSync(path)) {
74
- try {
75
- // Read first 3000 chars — enough for description, skip excessive content
76
- return readFileSync(path, 'utf-8').slice(0, 3000);
77
- } catch {
78
- // ignore
79
- }
80
- }
81
- }
82
- return '';
83
- }
84
-
85
- // ── Module Inference — Workspace-Aware ──
86
-
87
- private inferModules(report: AnalysisReport, projectPath: string): ProjectSummary['modules'] {
88
- const workspaces = report.projectInfo.workspaces;
89
-
90
- // If we have workspaces, use them as the authoritative module list
91
- if (workspaces && workspaces.length > 0) {
92
- return this.inferModulesFromWorkspaces(workspaces, report);
93
- }
94
-
95
- // Fallback: infer from directory structure
96
- return this.inferModulesFromStructure(report);
97
- }
98
-
99
- /**
100
- * Workspace-aware module inference.
101
- * Uses package.json name, description, and file count from each workspace.
102
- */
103
- private inferModulesFromWorkspaces(
104
- workspaces: WorkspaceInfo[],
105
- report: AnalysisReport,
106
- ): ProjectSummary['modules'] {
107
- return workspaces
108
- .map((ws) => {
109
- // Count files belonging to this workspace
110
- const wsPrefix = ws.relativePath + '/';
111
- const fileCount = report.dependencyGraph.nodes.filter(
112
- (n) => n.startsWith(wsPrefix) || n.startsWith(ws.relativePath),
113
- ).length;
114
-
115
- // Get description: prefer package.json description, then README, then heuristic
116
- const description = this.getWorkspaceDescription(ws);
117
-
118
- // Use the short name (last segment of npm scope or dir name)
119
- const displayName = ws.name.includes('/')
120
- ? ws.name.split('/').pop() || ws.name
121
- : basename(ws.relativePath);
122
-
123
- return {
124
- name: displayName,
125
- files: fileCount || this.countFilesInDir(ws.path),
126
- description,
127
- };
128
- })
129
- .filter((m) => m.files > 0)
130
- .sort((a, b) => b.files - a.files);
131
- }
132
-
133
- /**
134
- * Get a meaningful description for a workspace.
135
- * Priority: package.json description > README first line > heuristic
136
- */
137
- private getWorkspaceDescription(ws: WorkspaceInfo): string {
138
- // 1. package.json description (most reliable)
139
- if (ws.description && ws.description.trim().length > 5) {
140
- return ws.description.trim();
141
- }
142
-
143
- // 2. README.md first paragraph
144
- const readmePath = join(ws.path, 'README.md');
145
- if (existsSync(readmePath)) {
146
- try {
147
- const lines = readFileSync(readmePath, 'utf-8').split('\n');
148
- for (const line of lines) {
149
- const trimmed = line.trim();
150
- if (!trimmed || trimmed.startsWith('#') || trimmed.startsWith('[')
151
- || trimmed.startsWith('<') || trimmed.startsWith('!')
152
- || trimmed.length < 15) continue;
153
- return trimmed.slice(0, 200);
154
- }
155
- } catch {
156
- // skip
157
- }
158
- }
159
-
160
- // 3. Heuristic from name and deps
161
- return this.describeModule(basename(ws.relativePath), new Set(Object.keys(ws.dependencies)));
162
- }
163
-
164
- private countFilesInDir(dirPath: string): number {
165
- try {
166
- const { globSync } = require('glob');
167
- return globSync('**/*.{ts,tsx,js,jsx,py,java,go}', {
168
- cwd: dirPath,
169
- ignore: ['**/node_modules/**', '**/dist/**'],
170
- nodir: true,
171
- }).length;
172
- } catch {
173
- return 0;
174
- }
175
- }
176
-
177
- /**
178
- * Fallback module inference from directory structure (non-workspace projects)
179
- */
180
- private inferModulesFromStructure(report: AnalysisReport): ProjectSummary['modules'] {
181
- const modules: Map<string, { files: Set<string>; hints: Set<string> }> = new Map();
182
-
183
- for (const node of report.dependencyGraph.nodes) {
184
- // Safety: skip any node_modules paths that leaked through
185
- if (node.includes('node_modules')) continue;
186
-
187
- const parts = node.split('/');
188
- // Skip root-level files
189
- if (parts.length < 2) continue;
190
-
191
- // Get the module directory (2nd level, or 1st level for flat projects)
192
- let moduleName: string;
193
- const firstDir = parts[0];
194
-
195
- // Common source directories — go one level deeper
196
- if (['src', 'lib', 'app', 'packages', 'modules', 'features', 'apps'].includes(firstDir)) {
197
- moduleName = parts.length > 2 ? parts[1] : firstDir;
198
- } else if (firstDir === 'tests' || firstDir === 'test' || firstDir === '__tests__') {
199
- continue; // Skip test directories for module inference
200
- } else {
201
- moduleName = firstDir;
202
- }
203
-
204
- if (!modules.has(moduleName)) {
205
- modules.set(moduleName, { files: new Set(), hints: new Set() });
206
- }
207
- const mod = modules.get(moduleName)!;
208
- mod.files.add(node);
209
-
210
- // Extract hints from filenames
211
- const filename = basename(node).replace(/\.(ts|js|py|dart|go|java|rb|php|cs|tsx|jsx)$/, '');
212
- if (filename !== 'index' && filename !== 'main' && filename !== '__init__') {
213
- mod.hints.add(filename.toLowerCase());
214
- }
215
- }
216
-
217
- return [...modules.entries()]
218
- .filter(([, data]) => data.files.size > 0)
219
- .sort((a, b) => b[1].files.size - a[1].files.size)
220
- .slice(0, 15) // Top 15 modules
221
- .map(([name, data]) => ({
222
- name,
223
- files: data.files.size,
224
- description: this.describeModule(name, data.hints),
225
- }));
226
- }
227
-
228
- private describeModule(name: string, hints: Set<string>): string {
229
- const n = name.toLowerCase();
230
- const h = [...hints].join(' ');
231
-
232
- // Known module patterns
233
- const patterns: [RegExp, string][] = [
234
- [/auth/, 'Autenticação e autorização'],
235
- [/user/, 'Gestão de usuários'],
236
- [/payment|billing|charge|invoice/, 'Pagamentos e faturamento'],
237
- [/order|cart|checkout/, 'Pedidos e checkout'],
238
- [/product|catalog|item/, 'Catálogo de produtos'],
239
- [/notif/, 'Sistema de notificações'],
240
- [/email|mail|smtp/, 'Envio de emails'],
241
- [/report|analytics|dashboard/, 'Relatórios e analytics'],
242
- [/config|setting/, 'Configuração do sistema'],
243
- [/util|helper|common|shared/, 'Utilitários compartilhados'],
244
- [/middleware|interceptor|guard|pipe/, 'Middleware e interceptors'],
245
- [/database|db|migration|seed/, 'Banco de dados e migrations'],
246
- [/api|controller|route|endpoint/, 'API endpoints'],
247
- [/service|business|domain/, 'Lógica de negócio'],
248
- [/model|entity|schema/, 'Modelos de dados'],
249
- [/test|spec|fixture/, 'Testes'],
250
- [/validator|validation|sanitiz/, 'Validação de dados'],
251
- [/security|crypto|encrypt/, 'Segurança e criptografia'],
252
- [/cache|redis/, 'Cache e performance'],
253
- [/queue|worker|job|task/, 'Processamento assíncrono'],
254
- [/log|monitor|trace|metric/, 'Logging e monitoramento'],
255
- [/file|upload|storage|s3/, 'Armazenamento de arquivos'],
256
- [/search|elastic|index/, 'Busca e indexação'],
257
- [/chat|message|websocket|socket/, 'Comunicação em tempo real'],
258
- [/i18n|locale|translation/, 'Internacionalização'],
259
- [/theme|style|design/, 'Design system e temas'],
260
- [/component|widget|ui/, 'Componentes de UI'],
261
- [/page|screen|view/, 'Páginas/telas'],
262
- [/hook|composable/, 'Hooks/composables reutilizáveis'],
263
- [/store|state|redux|bloc/, 'Gerenciamento de estado'],
264
- [/navigation|router|routing/, 'Navegação e rotas'],
265
- [/bridge/, 'Camada de integração'],
266
- [/core/, 'Núcleo e orquestração'],
267
- [/event/, 'Sistema de eventos'],
268
- [/type/, 'Definições de tipos'],
269
- [/mcp/, 'MCP Server'],
270
- [/cli/, 'Interface de linha de comando'],
271
- [/cloud/, 'Serviço cloud / API'],
272
- [/autonomy/, 'Automação e self-healing'],
273
- [/app/, 'Aplicação principal'],
274
- ];
275
-
276
- const combined = `${n} ${h}`;
277
- for (const [regex, desc] of patterns) {
278
- if (regex.test(combined)) return desc;
279
- }
280
-
281
- return `Módulo ${name}`;
282
- }
283
-
284
- // ── Entry Points — Workspace-Aware ──
285
-
286
- private findEntryPoints(report: AnalysisReport, projectPath: string): string[] {
287
- const entries: string[] = [];
288
-
289
- // 1. From workspace bin fields
290
- const workspaces = report.projectInfo.workspaces;
291
- if (workspaces) {
292
- for (const ws of workspaces) {
293
- if (ws.bin) {
294
- if (typeof ws.bin === 'string') {
295
- entries.push(`${ws.relativePath}/${ws.bin}`);
296
- } else {
297
- for (const [, binPath] of Object.entries(ws.bin)) {
298
- const fullPath = `${ws.relativePath}/${binPath}`;
299
- entries.push(fullPath);
300
- }
301
- }
302
- }
303
- if (ws.main) {
304
- entries.push(`${ws.relativePath}/${ws.main}`);
305
- }
306
- }
307
- }
308
-
309
- // 2. From root package.json bin/main
310
- const rootPkgPath = join(projectPath, 'package.json');
311
- if (existsSync(rootPkgPath)) {
312
- try {
313
- const pkg = JSON.parse(readFileSync(rootPkgPath, 'utf-8'));
314
- if (pkg.bin) {
315
- if (typeof pkg.bin === 'string') {
316
- entries.push(pkg.bin);
317
- } else {
318
- for (const [, binPath] of Object.entries(pkg.bin)) {
319
- entries.push(binPath as string);
320
- }
321
- }
322
- }
323
- if (pkg.main && !entries.includes(pkg.main)) {
324
- entries.push(pkg.main);
325
- }
326
- } catch {
327
- // skip
328
- }
329
- }
330
-
331
- // 3. Pattern-based detection (fallback)
332
- const entryPatterns = [
333
- /^(src\/)?(main|index|app|server|cli)\.(ts|js|py|dart|go|java)$/,
334
- /^(src\/)?bin\//,
335
- /^packages\/[^/]+\/src\/(main|index|app|server|cli)\.(ts|js)$/,
336
- /^packages\/[^/]+\/src\/bin\//,
337
- /manage\.py$/,
338
- /^main\.go$/,
339
- ];
340
-
341
- const patternEntries = report.dependencyGraph.nodes
342
- .filter(node => entryPatterns.some(p => p.test(node)))
343
- .filter(node => !entries.includes(node));
344
-
345
- entries.push(...patternEntries);
346
-
347
- return [...new Set(entries)].slice(0, 10);
348
- }
349
-
350
- // ── Keywords ──
351
-
352
- private extractKeywords(
353
- packageInfo: Record<string, unknown>,
354
- readme: string,
355
- modules: ProjectSummary['modules'],
356
- report: AnalysisReport,
357
- ): string[] {
358
- const keywords = new Set<string>();
359
-
360
- // From package.json keywords
361
- if (Array.isArray(packageInfo.keywords)) {
362
- for (const kw of packageInfo.keywords) {
363
- if (typeof kw === 'string') keywords.add(kw.toLowerCase());
364
- }
365
- }
366
-
367
- // From module names (only clean names)
368
- for (const mod of modules) {
369
- const name = mod.name.toLowerCase();
370
- if (!ProjectSummarizer.KEYWORD_BLACKLIST.has(name)) {
371
- keywords.add(name);
372
- }
373
- }
374
-
375
- // From frameworks detected
376
- for (const fw of report.projectInfo.frameworks) {
377
- keywords.add(fw.toLowerCase());
378
- }
379
-
380
- // From languages
381
- for (const lang of report.projectInfo.primaryLanguages) {
382
- keywords.add(lang.toLowerCase());
383
- }
384
-
385
- // Filter out blacklisted and generic entries
386
- return [...keywords]
387
- .filter(kw => !ProjectSummarizer.KEYWORD_BLACKLIST.has(kw) && kw.length > 1)
388
- .slice(0, 20);
389
- }
390
-
391
- // ── Tech Stack ──
392
-
393
- private buildTechStack(report: AnalysisReport, packageInfo: Record<string, unknown>): string[] {
394
- const stack: string[] = [];
395
-
396
- // Languages
397
- stack.push(...report.projectInfo.primaryLanguages);
398
-
399
- // Frameworks from report
400
- stack.push(...report.projectInfo.frameworks);
401
-
402
- // Dependencies from package.json
403
- const deps = { ...(packageInfo.dependencies as Record<string, string> || {}), ...(packageInfo.devDependencies as Record<string, string> || {}) };
404
- const notable = [
405
- 'express', 'fastify', 'nestjs', '@nestjs/core', 'koa', 'hapi',
406
- 'react', 'next', 'angular', 'vue', 'svelte',
407
- 'prisma', '@prisma/client', 'typeorm', 'sequelize', 'mongoose', 'knex',
408
- 'jest', 'mocha', 'vitest', 'cypress', 'playwright',
409
- 'tailwindcss', 'styled-components', 'emotion',
410
- 'redis', 'ioredis', 'bull', 'bullmq',
411
- 'graphql', 'apollo', '@apollo/server',
412
- 'socket.io', 'ws',
413
- 'passport', 'jsonwebtoken', 'bcrypt',
414
- 'winston', 'pino',
415
- 'swagger', '@nestjs/swagger',
416
- 'docker', 'kubernetes',
417
- 'aws-sdk', '@aws-sdk',
418
- 'stripe', 'paypal',
419
- ];
420
-
421
- for (const dep of Object.keys(deps)) {
422
- const cleaned = dep.replace('@', '').split('/')[0];
423
- if (notable.some(n => dep.includes(n))) {
424
- if (!stack.some(s => s.toLowerCase() === cleaned.toLowerCase())) {
425
- stack.push(dep);
426
- }
427
- }
428
- }
429
-
430
- return [...new Set(stack)].slice(0, 15);
431
- }
432
-
433
- // ── Description Builder ──
434
-
435
- private buildDescription(packageInfo: Record<string, unknown>, readme: string, report: AnalysisReport): string {
436
- // Priority 1: package.json description
437
- if (typeof packageInfo.description === 'string' && packageInfo.description.trim()) {
438
- return packageInfo.description.trim();
439
- }
440
-
441
- // Priority 2: First paragraph of README (skip badges, titles)
442
- if (readme) {
443
- const lines = readme.split('\n');
444
- for (const line of lines) {
445
- const trimmed = line.trim();
446
- // Skip empty lines, headers, badges, links, HTML tags
447
- if (!trimmed) continue;
448
- if (trimmed.startsWith('#')) continue;
449
- if (trimmed.startsWith('[!') || trimmed.startsWith('[![')) continue;
450
- if (trimmed.startsWith('<')) continue;
451
- if (trimmed.startsWith('---') || trimmed.startsWith('===')) continue;
452
- if (trimmed.startsWith('![')) continue;
453
- if (trimmed.length < 20) continue;
454
-
455
- // Found a real text paragraph
456
- return trimmed.slice(0, 300);
457
- }
458
- }
459
-
460
- // Priority 3: Infer from project name and structure
461
- const name = report.projectInfo.name || 'Unknown';
462
- const langs = report.projectInfo.primaryLanguages.join(', ');
463
- const files = report.projectInfo.totalFiles;
464
- return `Projeto ${name} — ${files} arquivos em ${langs}`;
465
- }
466
-
467
- // ── Purpose Inference ──
468
-
469
- private inferPurpose(
470
- keywords: string[],
471
- modules: ProjectSummary['modules'],
472
- report: AnalysisReport,
473
- ): string {
474
- const allSignals = [
475
- ...keywords,
476
- ...modules.map(m => m.name.toLowerCase()),
477
- ...modules.map(m => m.description.toLowerCase()),
478
- // Only use project nodes, never leaked node_modules paths
479
- ...report.dependencyGraph.nodes
480
- .filter(n => !n.includes('node_modules'))
481
- .map(n => n.toLowerCase()),
482
- ].join(' ');
483
-
484
- // Infer project type from signals
485
- const types: [RegExp, string][] = [
486
- [/api.*(rest|graph|endpoint)|controller.*route|swagger|openapi/, 'API Backend'],
487
- [/cli|command.*line|bin\/|yargs|commander/, 'CLI Tool'],
488
- [/component.*ui|react|angular|vue|frontend|page.*screen/, 'Frontend Application'],
489
- [/mobile|flutter|dart|react.native|ionic/, 'Mobile App'],
490
- [/library|lib|package|npm|pub|sdk|module/, 'Library / Package'],
491
- [/test|spec|validator|lint|analyz|check/, 'Tool de Análise / Validação'],
492
- [/microservice|service|worker|queue/, 'Microservice'],
493
- [/monorepo|workspace|packages\//, 'Monorepo'],
494
- [/bot|scraper|crawler|automation/, 'Bot / Automação'],
495
- [/game|canvas|webgl|three/, 'Game / Visualização'],
496
- [/e-?commerce|shop|cart|product|catalog/, 'E-commerce'],
497
- [/blog|cms|content|post|article/, 'CMS / Blog'],
498
- [/auth|login|oauth|sso|identity/, 'Sistema de Autenticação'],
499
- [/chat|message|realtime|socket/, 'Comunicação Real-time'],
500
- [/dashboard|admin|panel|analytics/, 'Dashboard / Admin Panel'],
501
- [/payment|billing|fintech|finance|bank/, 'Fintech / Pagamentos'],
502
- [/health|medical|patient|clinic/, 'Healthcare'],
503
- [/education|course|learn|student/, 'EdTech'],
504
- ];
505
-
506
- const matched: string[] = [];
507
- for (const [regex, type] of types) {
508
- if (regex.test(allSignals)) {
509
- matched.push(type);
510
- }
511
- }
512
-
513
- if (matched.length > 0) {
514
- return matched.slice(0, 3).join(' + ');
515
- }
516
-
517
- // Fallback
518
- const langs = report.projectInfo.primaryLanguages.join('/');
519
- return `Projeto ${langs}`;
520
- }
521
- }
@@ -1,117 +0,0 @@
1
- import { readFileSync } from 'fs';
2
- import { basename, dirname, join, relative } from 'path';
3
- import {
4
- AnalysisReport,
5
- RefactoringPlan,
6
- RefactorStep,
7
- RefactorRule,
8
- FileOperation,
9
- } from './types.js';
10
-
11
- // ── Tier 1 Rules ──
12
- import { HubSplitterRule } from './rules/hub-splitter.js';
13
- import { BarrelOptimizerRule } from './rules/barrel-optimizer.js';
14
- import { ImportOrganizerRule } from './rules/import-organizer.js';
15
- import { ModuleGrouperRule } from './rules/module-grouper.js';
16
- import { DeadCodeDetectorRule } from './rules/dead-code-detector.js';
17
-
18
- /**
19
- * Refactoring Engine v2.0
20
- * Orchestrates Tier 1 (rule-based) and Tier 2 (AST) refactoring rules.
21
- */
22
- export class RefactorEngine {
23
- private rules: RefactorRule[];
24
-
25
- constructor() {
26
- this.rules = [
27
- // Tier 1: Rule Engine (pattern matching)
28
- new HubSplitterRule(),
29
- new BarrelOptimizerRule(),
30
- new ImportOrganizerRule(),
31
- new ModuleGrouperRule(),
32
- new DeadCodeDetectorRule(),
33
- ];
34
- }
35
-
36
- /**
37
- * Analyze a project and generate a refactoring plan.
38
- */
39
- analyze(report: AnalysisReport, projectPath: string): RefactoringPlan {
40
- const allSteps: RefactorStep[] = [];
41
- let stepId = 1;
42
-
43
- // Run each rule
44
- for (const rule of this.rules) {
45
- const steps = rule.analyze(report, projectPath);
46
- for (const step of steps) {
47
- step.id = stepId++;
48
- allSteps.push(step);
49
- }
50
- }
51
-
52
- // Sort by priority
53
- const priorityOrder: Record<string, number> = {
54
- CRITICAL: 0,
55
- HIGH: 1,
56
- MEDIUM: 2,
57
- LOW: 3,
58
- };
59
- allSteps.sort((a, b) => priorityOrder[a.priority] - priorityOrder[b.priority]);
60
-
61
- // Re-number after sorting
62
- allSteps.forEach((s, i) => (s.id = i + 1));
63
-
64
- // Calculate total operations
65
- const totalOperations = allSteps.reduce(
66
- (sum, s) => sum + s.operations.length,
67
- 0
68
- );
69
-
70
- // Estimate score after refactoring
71
- const estimatedScoreAfter = this.estimateScoreAfter(report, allSteps);
72
-
73
- return {
74
- timestamp: new Date().toISOString(),
75
- projectPath,
76
- currentScore: report.score,
77
- estimatedScoreAfter,
78
- steps: allSteps,
79
- totalOperations,
80
- tier1Steps: allSteps.filter((s) => s.tier === 1).length,
81
- tier2Steps: allSteps.filter((s) => s.tier === 2).length,
82
- };
83
- }
84
-
85
- /**
86
- * Estimates the architecture score after applying all refactoring steps.
87
- */
88
- private estimateScoreAfter(
89
- report: AnalysisReport,
90
- steps: RefactorStep[]
91
- ): { overall: number; breakdown: Record<string, number> } {
92
- const breakdown = { ...report.score.breakdown };
93
-
94
- for (const step of steps) {
95
- for (const impact of step.scoreImpact) {
96
- if (impact.metric in breakdown) {
97
- // Use estimated after value, capped at 95
98
- const key = impact.metric as keyof typeof breakdown;
99
- breakdown[key] = Math.min(95, Math.max(breakdown[key], impact.after));
100
- }
101
- }
102
- }
103
-
104
- // Recalculate overall with same weights
105
- const overall = Math.round(
106
- breakdown.modularity * 0.4 +
107
- breakdown.coupling * 0.25 +
108
- breakdown.cohesion * 0.2 +
109
- breakdown.layering * 0.15
110
- );
111
-
112
- return {
113
- overall: Math.min(100, overall),
114
- breakdown,
115
- };
116
- }
117
- }