@girardelli/architect 2.2.0 → 4.0.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 (296) hide show
  1. package/README.md +105 -116
  2. package/__test_agent_output__/INDEX.md +1 -0
  3. package/__test_agent_output__/agents/AGENT-ORCHESTRATOR.md +1 -0
  4. package/__test_agent_output__/agents/DATABASE-ENGINEER.md +174 -0
  5. package/__test_agent_output__/agents/QA-TEST-ENGINEER.md +138 -0
  6. package/__test_agent_output__/agents/SECURITY-AUDITOR.md +106 -0
  7. package/__test_agent_output__/agents/TECH-DEBT-CONTROLLER.md +104 -0
  8. package/__test_agent_output__/agents/TYPESCRIPT-BACKEND-DEVELOPER.md +135 -0
  9. package/__test_agent_output__/guards/CODE-REVIEW-CHECKLIST.md +95 -0
  10. package/__test_agent_output__/guards/PREFLIGHT.md +200 -0
  11. package/__test_agent_output__/guards/QUALITY-GATES.md +1 -0
  12. package/__test_agent_output__/rules/00-general.md +229 -0
  13. package/__test_agent_output__/rules/01-architecture.md +191 -0
  14. package/__test_agent_output__/rules/02-security.md +402 -0
  15. package/__test_agent_output__/rules/03-nestjs.md +124 -0
  16. package/__test_agent_output__/templates/ADR.md +95 -0
  17. package/__test_agent_output__/templates/BDD.md +58 -0
  18. package/__test_agent_output__/templates/C4.md +68 -0
  19. package/__test_agent_output__/templates/TDD.md +86 -0
  20. package/__test_agent_output__/templates/THREAT-MODEL.md +82 -0
  21. package/__test_agent_output__/workflows/fix-bug.md +228 -0
  22. package/__test_agent_output__/workflows/new-feature.md +311 -0
  23. package/__test_agent_output__/workflows/review.md +95 -0
  24. package/__test_context_7RvUrO/src/modules/empty/empty.ts +0 -0
  25. package/__test_context_Rf5fNJ/src/modules/mixed/mixed.ts +5 -0
  26. package/__test_context_WRCnYH/src/modules/test/test.ts +10 -0
  27. package/__test_context_YsnVS3/src/modules/test/test.ts +10 -0
  28. package/__test_context_w7XZeH/src/modules/mixed/mixed.ts +5 -0
  29. package/__test_context_y5noh6/src/modules/empty/empty.ts +0 -0
  30. package/__test_framework__24OjAu/package.json +1 -0
  31. package/__test_framework__3ZDZsx/pyproject.toml +8 -0
  32. package/__test_framework__4T54Jn/package.json +1 -0
  33. package/__test_framework__4tlXu9/pyproject.toml +8 -0
  34. package/__test_framework__6boWqQ/Pipfile +6 -0
  35. package/__test_framework__6gygMU/pom.xml +10 -0
  36. package/__test_framework__6kxj0N/go.mod +8 -0
  37. package/__test_framework__7CEoXw/pom.xml +10 -0
  38. package/__test_framework__85DDz0/Pipfile +6 -0
  39. package/__test_framework__9WrRIr/pom.xml +7 -0
  40. package/__test_framework__ANqGKl/Gemfile +5 -0
  41. package/__test_framework__BCXTEM/go.mod +3 -0
  42. package/__test_framework__BHiPNq/setup.py +2 -0
  43. package/__test_framework__BqkiKv/package.json +1 -0
  44. package/__test_framework__C5yd8X/Pipfile.lock +1 -0
  45. package/__test_framework__C5yd8X/requirements.txt +1 -0
  46. package/__test_framework__C87d3a/manage.py +1 -0
  47. package/__test_framework__C87d3a/requirements.txt +2 -0
  48. package/__test_framework__DXNwc5/build.gradle +7 -0
  49. package/__test_framework__GhHSt3/build.gradle.kts +4 -0
  50. package/__test_framework__GzklJP/Cargo.toml +7 -0
  51. package/__test_framework__H4hd13/go.mod +8 -0
  52. package/__test_framework__HKjOXO/composer.json +1 -0
  53. package/__test_framework__HaDN45/Gemfile +3 -0
  54. package/__test_framework__IBO7YG/pyproject.toml +9 -0
  55. package/__test_framework__JwSOyF/pyproject.toml +6 -0
  56. package/__test_framework__K6HrCr/build.gradle +2 -0
  57. package/__test_framework__KzRPlh/pubspec.yaml +9 -0
  58. package/__test_framework__L6uIym/pyproject.toml +6 -0
  59. package/__test_framework__LOdoGK/requirements.txt +4 -0
  60. package/__test_framework__LgHzss/package.json +1 -0
  61. package/__test_framework__M76M6q/Gemfile +5 -0
  62. package/__test_framework__Mr9vWW/composer.json +1 -0
  63. package/__test_framework__N03Gnv/package.json +1 -0
  64. package/__test_framework__Num4UE/requirements +1 -0
  65. package/__test_framework__OAGw3Y/build.gradle +7 -0
  66. package/__test_framework__OQc8yG/pubspec.yaml +9 -0
  67. package/__test_framework__OwKZcd/requirements.txt +3 -0
  68. package/__test_framework__P0gFv7/requirements +1 -0
  69. package/__test_framework__PN55Rq/package.json +1 -0
  70. package/__test_framework__PQiqX8/pubspec.yaml +3 -0
  71. package/__test_framework__RBHsg7/composer.json +1 -0
  72. package/__test_framework__RHxif4/Cargo.toml +7 -0
  73. package/__test_framework__T0v0p1/Cargo.toml +4 -0
  74. package/__test_framework__Tu0clt/Pipfile.lock +1 -0
  75. package/__test_framework__Tu0clt/requirements.txt +1 -0
  76. package/__test_framework__TwDj9P/Cargo.toml +4 -0
  77. package/__test_framework__VQJNC4/pom.xml +7 -0
  78. package/__test_framework__W6sm05/package.json +1 -0
  79. package/__test_framework__W7vBLy/pyproject.toml +4 -0
  80. package/__test_framework__WNJOWT/setup.py +2 -0
  81. package/__test_framework__WSJs7U/package.json +1 -0
  82. package/__test_framework__YQ5VpA/build.gradle.kts +4 -0
  83. package/__test_framework__ZNEUEs/package.json +1 -0
  84. package/__test_framework__Znt922/pom.xml +7 -0
  85. package/__test_framework__azyg0h/pom.xml +7 -0
  86. package/__test_framework__c6otLr/package.json +1 -0
  87. package/__test_framework__cl9S9G/build.gradle +2 -0
  88. package/__test_framework__eilvV4/composer.json +1 -0
  89. package/__test_framework__gQZxXO/manage.py +1 -0
  90. package/__test_framework__gQZxXO/requirements.txt +2 -0
  91. package/__test_framework__ghvl26/poetry.lock +1 -0
  92. package/__test_framework__ghvl26/pyproject.toml +2 -0
  93. package/__test_framework__hR7b9U/Makefile +11 -0
  94. package/__test_framework__iESVsi/composer.json +1 -0
  95. package/__test_framework__jm6TJy/package.json +1 -0
  96. package/__test_framework__kBUpjs/pyproject.toml +9 -0
  97. package/__test_framework__kqoZrw/requirements.txt +4 -0
  98. package/__test_framework__lWkoyO/pyproject.toml +4 -0
  99. package/__test_framework__mTKnUO/package.json +1 -0
  100. package/__test_framework__nCeZwe/Makefile +11 -0
  101. package/__test_framework__oljsU0/package.json +1 -0
  102. package/__test_framework__osRG4q/go.mod +3 -0
  103. package/__test_framework__pCHH4F/package.json +1 -0
  104. package/__test_framework__pExx6E/Gemfile +3 -0
  105. package/__test_framework__pyBoGd/pyproject.toml +5 -0
  106. package/__test_framework__qw16VQ/package.json +1 -0
  107. package/__test_framework__rRayrG/package.json +1 -0
  108. package/__test_framework__s82zO5/package.json +1 -0
  109. package/__test_framework__tp8MFK/pyproject.toml +5 -0
  110. package/__test_framework__w44k4w/composer.json +1 -0
  111. package/__test_framework__yefPZY/poetry.lock +1 -0
  112. package/__test_framework__yefPZY/pyproject.toml +2 -0
  113. package/__test_framework__zCiyDT/requirements.txt +3 -0
  114. package/__test_framework__zGZN3j/pubspec.yaml +3 -0
  115. package/__test_framework__zXpnxL/package.json +1 -0
  116. package/architect-run.sh +431 -0
  117. package/assets/banner-v3.html +561 -0
  118. package/dist/agent-generator/context-enricher.d.ts +58 -0
  119. package/dist/agent-generator/context-enricher.d.ts.map +1 -0
  120. package/dist/agent-generator/context-enricher.js +581 -0
  121. package/dist/agent-generator/context-enricher.js.map +1 -0
  122. package/dist/agent-generator/domain-inferrer.d.ts +52 -0
  123. package/dist/agent-generator/domain-inferrer.d.ts.map +1 -0
  124. package/dist/agent-generator/domain-inferrer.js +575 -0
  125. package/dist/agent-generator/domain-inferrer.js.map +1 -0
  126. package/dist/agent-generator/framework-detector.d.ts +40 -0
  127. package/dist/agent-generator/framework-detector.d.ts.map +1 -0
  128. package/dist/agent-generator/framework-detector.js +611 -0
  129. package/dist/agent-generator/framework-detector.js.map +1 -0
  130. package/dist/agent-generator/index.d.ts +33 -0
  131. package/dist/agent-generator/index.d.ts.map +1 -0
  132. package/dist/agent-generator/index.js +477 -0
  133. package/dist/agent-generator/index.js.map +1 -0
  134. package/dist/agent-generator/stack-detector.d.ts +12 -0
  135. package/dist/agent-generator/stack-detector.d.ts.map +1 -0
  136. package/dist/agent-generator/stack-detector.js +128 -0
  137. package/dist/agent-generator/stack-detector.js.map +1 -0
  138. package/dist/agent-generator/templates/core/agents.d.ts +17 -0
  139. package/dist/agent-generator/templates/core/agents.d.ts.map +1 -0
  140. package/dist/agent-generator/templates/core/agents.js +1252 -0
  141. package/dist/agent-generator/templates/core/agents.js.map +1 -0
  142. package/dist/agent-generator/templates/core/architecture-rules.d.ts +7 -0
  143. package/dist/agent-generator/templates/core/architecture-rules.d.ts.map +1 -0
  144. package/dist/agent-generator/templates/core/architecture-rules.js +274 -0
  145. package/dist/agent-generator/templates/core/architecture-rules.js.map +1 -0
  146. package/dist/agent-generator/templates/core/general-rules.d.ts +8 -0
  147. package/dist/agent-generator/templates/core/general-rules.d.ts.map +1 -0
  148. package/dist/agent-generator/templates/core/general-rules.js +301 -0
  149. package/dist/agent-generator/templates/core/general-rules.js.map +1 -0
  150. package/dist/agent-generator/templates/core/index-md.d.ts +7 -0
  151. package/dist/agent-generator/templates/core/index-md.d.ts.map +1 -0
  152. package/dist/agent-generator/templates/core/index-md.js +246 -0
  153. package/dist/agent-generator/templates/core/index-md.js.map +1 -0
  154. package/dist/agent-generator/templates/core/orchestrator.d.ts +8 -0
  155. package/dist/agent-generator/templates/core/orchestrator.d.ts.map +1 -0
  156. package/dist/agent-generator/templates/core/orchestrator.js +422 -0
  157. package/dist/agent-generator/templates/core/orchestrator.js.map +1 -0
  158. package/dist/agent-generator/templates/core/preflight.d.ts +8 -0
  159. package/dist/agent-generator/templates/core/preflight.d.ts.map +1 -0
  160. package/dist/agent-generator/templates/core/preflight.js +213 -0
  161. package/dist/agent-generator/templates/core/preflight.js.map +1 -0
  162. package/dist/agent-generator/templates/core/quality-gates.d.ts +11 -0
  163. package/dist/agent-generator/templates/core/quality-gates.d.ts.map +1 -0
  164. package/dist/agent-generator/templates/core/quality-gates.js +254 -0
  165. package/dist/agent-generator/templates/core/quality-gates.js.map +1 -0
  166. package/dist/agent-generator/templates/core/security-rules.d.ts +7 -0
  167. package/dist/agent-generator/templates/core/security-rules.d.ts.map +1 -0
  168. package/dist/agent-generator/templates/core/security-rules.js +528 -0
  169. package/dist/agent-generator/templates/core/security-rules.js.map +1 -0
  170. package/dist/agent-generator/templates/core/skills-generator.d.ts +6 -0
  171. package/dist/agent-generator/templates/core/skills-generator.d.ts.map +1 -0
  172. package/dist/agent-generator/templates/core/skills-generator.js +207 -0
  173. package/dist/agent-generator/templates/core/skills-generator.js.map +1 -0
  174. package/dist/agent-generator/templates/core/workflow-fix-bug.d.ts +7 -0
  175. package/dist/agent-generator/templates/core/workflow-fix-bug.d.ts.map +1 -0
  176. package/dist/agent-generator/templates/core/workflow-fix-bug.js +237 -0
  177. package/dist/agent-generator/templates/core/workflow-fix-bug.js.map +1 -0
  178. package/dist/agent-generator/templates/core/workflow-new-feature.d.ts +8 -0
  179. package/dist/agent-generator/templates/core/workflow-new-feature.d.ts.map +1 -0
  180. package/dist/agent-generator/templates/core/workflow-new-feature.js +321 -0
  181. package/dist/agent-generator/templates/core/workflow-new-feature.js.map +1 -0
  182. package/dist/agent-generator/templates/core/workflow-review.d.ts +7 -0
  183. package/dist/agent-generator/templates/core/workflow-review.d.ts.map +1 -0
  184. package/dist/agent-generator/templates/core/workflow-review.js +104 -0
  185. package/dist/agent-generator/templates/core/workflow-review.js.map +1 -0
  186. package/dist/agent-generator/templates/domain/index.d.ts +22 -0
  187. package/dist/agent-generator/templates/domain/index.d.ts.map +1 -0
  188. package/dist/agent-generator/templates/domain/index.js +1176 -0
  189. package/dist/agent-generator/templates/domain/index.js.map +1 -0
  190. package/dist/agent-generator/templates/stack/index.d.ts +8 -0
  191. package/dist/agent-generator/templates/stack/index.d.ts.map +1 -0
  192. package/dist/agent-generator/templates/stack/index.js +695 -0
  193. package/dist/agent-generator/templates/stack/index.js.map +1 -0
  194. package/dist/agent-generator/templates/template-helpers.d.ts +75 -0
  195. package/dist/agent-generator/templates/template-helpers.d.ts.map +1 -0
  196. package/dist/agent-generator/templates/template-helpers.js +726 -0
  197. package/dist/agent-generator/templates/template-helpers.js.map +1 -0
  198. package/dist/agent-generator/types.d.ts +196 -0
  199. package/dist/agent-generator/types.d.ts.map +1 -0
  200. package/dist/agent-generator/types.js +27 -0
  201. package/dist/agent-generator/types.js.map +1 -0
  202. package/dist/analyzer.d.ts +5 -0
  203. package/dist/analyzer.d.ts.map +1 -1
  204. package/dist/analyzer.js +35 -4
  205. package/dist/analyzer.js.map +1 -1
  206. package/dist/analyzers/forecast.d.ts +85 -0
  207. package/dist/analyzers/forecast.d.ts.map +1 -0
  208. package/dist/analyzers/forecast.js +337 -0
  209. package/dist/analyzers/forecast.js.map +1 -0
  210. package/dist/analyzers/git-cache.d.ts +7 -0
  211. package/dist/analyzers/git-cache.d.ts.map +1 -0
  212. package/dist/analyzers/git-cache.js +41 -0
  213. package/dist/analyzers/git-cache.js.map +1 -0
  214. package/dist/analyzers/git-history.d.ts +113 -0
  215. package/dist/analyzers/git-history.d.ts.map +1 -0
  216. package/dist/analyzers/git-history.js +333 -0
  217. package/dist/analyzers/git-history.js.map +1 -0
  218. package/dist/analyzers/index.d.ts +10 -0
  219. package/dist/analyzers/index.d.ts.map +1 -0
  220. package/dist/analyzers/index.js +7 -0
  221. package/dist/analyzers/index.js.map +1 -0
  222. package/dist/analyzers/temporal-scorer.d.ts +72 -0
  223. package/dist/analyzers/temporal-scorer.d.ts.map +1 -0
  224. package/dist/analyzers/temporal-scorer.js +140 -0
  225. package/dist/analyzers/temporal-scorer.js.map +1 -0
  226. package/dist/cli.d.ts +2 -3
  227. package/dist/cli.d.ts.map +1 -1
  228. package/dist/cli.js +275 -113
  229. package/dist/cli.js.map +1 -1
  230. package/dist/html-reporter.d.ts +3 -1
  231. package/dist/html-reporter.d.ts.map +1 -1
  232. package/dist/html-reporter.js +248 -12
  233. package/dist/html-reporter.js.map +1 -1
  234. package/dist/index.d.ts +16 -3
  235. package/dist/index.d.ts.map +1 -1
  236. package/dist/index.js +63 -4
  237. package/dist/index.js.map +1 -1
  238. package/dist/project-summarizer.d.ts +18 -0
  239. package/dist/project-summarizer.d.ts.map +1 -0
  240. package/dist/project-summarizer.js +306 -0
  241. package/dist/project-summarizer.js.map +1 -0
  242. package/dist/refactor-reporter.js +1 -1
  243. package/dist/types.d.ts +13 -0
  244. package/dist/types.d.ts.map +1 -1
  245. package/package.json +12 -3
  246. package/src/agent-generator/context-enricher.ts +643 -0
  247. package/src/agent-generator/domain-inferrer.ts +625 -0
  248. package/src/agent-generator/framework-detector.ts +669 -0
  249. package/src/agent-generator/index.ts +555 -0
  250. package/src/agent-generator/stack-detector.ts +103 -0
  251. package/src/agent-generator/templates/core/agents.ts +1293 -0
  252. package/src/agent-generator/templates/core/architecture-rules.ts +287 -0
  253. package/src/agent-generator/templates/core/general-rules.ts +306 -0
  254. package/src/agent-generator/templates/core/index-md.ts +260 -0
  255. package/src/agent-generator/templates/core/orchestrator.ts +459 -0
  256. package/src/agent-generator/templates/core/preflight.ts +215 -0
  257. package/src/agent-generator/templates/core/quality-gates.ts +256 -0
  258. package/src/agent-generator/templates/core/security-rules.ts +543 -0
  259. package/src/agent-generator/templates/core/skills-generator.ts +236 -0
  260. package/src/agent-generator/templates/core/workflow-fix-bug.ts +239 -0
  261. package/src/agent-generator/templates/core/workflow-new-feature.ts +323 -0
  262. package/src/agent-generator/templates/core/workflow-review.ts +106 -0
  263. package/src/agent-generator/templates/domain/index.ts +1201 -0
  264. package/src/agent-generator/templates/stack/index.ts +705 -0
  265. package/src/agent-generator/templates/template-helpers.ts +776 -0
  266. package/src/agent-generator/types.ts +232 -0
  267. package/src/analyzer.ts +38 -4
  268. package/src/analyzers/forecast.ts +496 -0
  269. package/src/analyzers/git-cache.ts +52 -0
  270. package/src/analyzers/git-history.ts +488 -0
  271. package/src/analyzers/index.ts +33 -0
  272. package/src/analyzers/temporal-scorer.ts +227 -0
  273. package/src/cli.ts +316 -117
  274. package/src/html-reporter.ts +263 -13
  275. package/src/index.ts +92 -9
  276. package/src/project-summarizer.ts +347 -0
  277. package/src/refactor-reporter.ts +1 -1
  278. package/src/types.ts +10 -0
  279. package/tests/agent-generator.test.ts +411 -0
  280. package/tests/analyzers-integration.test.ts +174 -0
  281. package/tests/architect-adapter-enrichment.test.ts +9 -0
  282. package/tests/context-enricher.test.ts +971 -0
  283. package/tests/forecast.test.ts +509 -0
  284. package/tests/framework-detector.test.ts +1172 -0
  285. package/tests/git-history.test.ts +254 -0
  286. package/tests/scanner.test.ts +7 -8
  287. package/tests/scorer.test.ts +588 -0
  288. package/tests/stack-detector.test.ts +241 -0
  289. package/tests/template-generation.test.ts +706 -0
  290. package/tests/template-helpers.test.ts +1152 -0
  291. package/tests/temporal-scorer.test.ts +307 -0
  292. package/dist/agent-generator.d.ts +0 -106
  293. package/dist/agent-generator.d.ts.map +0 -1
  294. package/dist/agent-generator.js +0 -1398
  295. package/dist/agent-generator.js.map +0 -1
  296. package/src/agent-generator.ts +0 -1526
@@ -0,0 +1,337 @@
1
+ /**
2
+ * Architecture Weather Forecast — Predictive analysis
3
+ *
4
+ * Combines temporal scores + velocity vectors + change coupling
5
+ * to predict which modules will become anti-patterns.
6
+ *
7
+ * Key concept: Pre-Anti-Pattern
8
+ * A module isn't an anti-pattern yet, but its trajectory says it will be.
9
+ * "Your code doesn't have a problem — it WILL have one in 3 months."
10
+ *
11
+ * @author Camilo Girardelli — Girardelli Tecnologia
12
+ * @license MIT
13
+ */
14
+ const DEFAULT_CONFIG = {
15
+ antiPatternThreshold: 40,
16
+ godClassChurnThreshold: 150,
17
+ shotgunCouplingThreshold: 5,
18
+ busFatorRiskThreshold: 1,
19
+ forecastWeeks: 26,
20
+ };
21
+ // ═══════════════════════════════════════════════════════════════
22
+ // FORECAST ENGINE
23
+ // ═══════════════════════════════════════════════════════════════
24
+ export class ForecastEngine {
25
+ constructor(config) {
26
+ this.config = { ...DEFAULT_CONFIG, ...config };
27
+ }
28
+ /**
29
+ * Generate architecture weather forecast.
30
+ */
31
+ forecast(gitReport, temporalReport) {
32
+ const preAntiPatterns = [];
33
+ const moduleForecastMap = new Map();
34
+ // Detect pre-anti-patterns for each module
35
+ for (const module of gitReport.modules) {
36
+ const temporal = temporalReport.modules.find(m => m.module === module.modulePath);
37
+ if (!temporal)
38
+ continue;
39
+ const patterns = this.detectPreAntiPatterns(module, temporal, gitReport.changeCouplings);
40
+ preAntiPatterns.push(...patterns);
41
+ const forecast = this.forecastModule(module, temporal, patterns);
42
+ moduleForecastMap.set(module.modulePath, forecast);
43
+ }
44
+ // Sort by risk
45
+ const modules = Array.from(moduleForecastMap.values())
46
+ .sort((a, b) => b.bottleneckProbability - a.bottleneckProbability);
47
+ const outlook = this.classifyOutlook(temporalReport, preAntiPatterns);
48
+ const headline = this.generateHeadline(outlook, preAntiPatterns, modules);
49
+ const topRisks = this.identifyTopRisks(modules, preAntiPatterns);
50
+ const recommendations = this.generateRecommendations(modules, preAntiPatterns);
51
+ return {
52
+ projectPath: gitReport.projectPath,
53
+ generatedAt: new Date().toISOString(),
54
+ overallOutlook: outlook,
55
+ headline,
56
+ modules,
57
+ preAntiPatterns: preAntiPatterns.sort((a, b) => a.weeksToThreshold - b.weeksToThreshold),
58
+ topRisks,
59
+ recommendations,
60
+ };
61
+ }
62
+ // ── Pre-Anti-Pattern Detection ──
63
+ detectPreAntiPatterns(module, temporal, couplings) {
64
+ const patterns = [];
65
+ patterns.push(...this.detectEmergingGodClass(module, temporal));
66
+ patterns.push(...this.detectEmergingShotgunSurgery(module, couplings));
67
+ patterns.push(...this.detectBusFactorRisk(module, temporal));
68
+ patterns.push(...this.detectComplexitySpiral(module, temporal));
69
+ patterns.push(...this.detectCouplingMagnet(module, couplings, temporal));
70
+ return patterns;
71
+ }
72
+ detectEmergingGodClass(module, temporal) {
73
+ const patterns = [];
74
+ for (const file of module.files) {
75
+ if (file.churnRate < this.config.godClassChurnThreshold)
76
+ continue;
77
+ if (temporal.velocity.churnTrend <= 0)
78
+ continue;
79
+ // Project when churn will exceed critical threshold
80
+ const weeklyGrowth = file.churnRate * (temporal.velocity.churnTrend / 100) / 4;
81
+ const criticalChurn = this.config.godClassChurnThreshold * 2;
82
+ const weeksToThreshold = weeklyGrowth > 0
83
+ ? Math.ceil((criticalChurn - file.churnRate) / weeklyGrowth)
84
+ : Infinity;
85
+ if (weeksToThreshold <= this.config.forecastWeeks) {
86
+ patterns.push({
87
+ type: 'emerging-god-class',
88
+ module: module.modulePath,
89
+ severity: weeksToThreshold <= 8 ? 'alert' : weeksToThreshold <= 16 ? 'warning' : 'watch',
90
+ currentScore: temporal.staticScore,
91
+ projectedScore: temporal.projectedScore,
92
+ weeksToThreshold,
93
+ threshold: criticalChurn,
94
+ description: `File '${file.path}' has churn rate ${Math.round(file.churnRate)} lines/commit and growing ${Math.round(temporal.velocity.churnTrend)}%`,
95
+ evidence: [
96
+ `Current churn: ${Math.round(file.churnRate)} lines/commit`,
97
+ `Growth rate: ${Math.round(temporal.velocity.churnTrend)}%`,
98
+ `${file.commits} commits in analysis period`,
99
+ `${file.authors.size} contributor(s)`,
100
+ ],
101
+ recommendation: 'Split into smaller, focused modules before complexity makes refactoring prohibitively expensive.',
102
+ confidence: Math.min(0.9, 0.5 + (file.commits / 50)),
103
+ });
104
+ }
105
+ }
106
+ return patterns;
107
+ }
108
+ detectEmergingShotgunSurgery(module, couplings) {
109
+ // Count how many files this module is coupled with
110
+ const moduleCouplings = couplings.filter(c => c.fileA.startsWith(module.modulePath) || c.fileB.startsWith(module.modulePath));
111
+ if (moduleCouplings.length < this.config.shotgunCouplingThreshold)
112
+ return [];
113
+ const avgConfidence = moduleCouplings.reduce((s, c) => s + c.confidence, 0) / moduleCouplings.length;
114
+ return [{
115
+ type: 'emerging-shotgun-surgery',
116
+ module: module.modulePath,
117
+ severity: moduleCouplings.length > 10 ? 'alert' : 'warning',
118
+ currentScore: 0,
119
+ projectedScore: 0,
120
+ weeksToThreshold: 4,
121
+ threshold: this.config.shotgunCouplingThreshold,
122
+ description: `Module '${module.modulePath}' has ${moduleCouplings.length} change-coupled files — changes here ripple across the codebase`,
123
+ evidence: [
124
+ `${moduleCouplings.length} files change together with this module`,
125
+ `Average coupling confidence: ${Math.round(avgConfidence * 100)}%`,
126
+ ...moduleCouplings.slice(0, 3).map(c => `${c.fileA} ↔ ${c.fileB} (${c.cochangeCount} co-changes)`),
127
+ ],
128
+ recommendation: 'Extract shared concerns into a dedicated module. Introduce interfaces to decouple.',
129
+ confidence: avgConfidence,
130
+ }];
131
+ }
132
+ detectBusFactorRisk(module, temporal) {
133
+ if (module.busFactor > this.config.busFatorRiskThreshold)
134
+ return [];
135
+ if (module.aggregateCommits < 5)
136
+ return []; // too few commits to judge
137
+ return [{
138
+ type: 'bus-factor-risk',
139
+ module: module.modulePath,
140
+ severity: temporal.trend === 'degrading' ? 'alert' : 'warning',
141
+ currentScore: temporal.staticScore,
142
+ projectedScore: temporal.projectedScore,
143
+ weeksToThreshold: this.config.forecastWeeks,
144
+ threshold: 2,
145
+ description: `Module '${module.modulePath}' has bus factor of ${module.busFactor} — all knowledge in one person`,
146
+ evidence: [
147
+ `Only ${module.busFactor} contributor(s)`,
148
+ `${module.aggregateCommits} total commits`,
149
+ `${module.files.length} files in module`,
150
+ ],
151
+ recommendation: 'Pair programming or code review rotation to spread knowledge. Document critical decisions.',
152
+ confidence: 0.8,
153
+ }];
154
+ }
155
+ detectComplexitySpiral(module, temporal) {
156
+ if (temporal.velocity.churnTrend <= 20)
157
+ return [];
158
+ if (temporal.velocity.direction !== 'accelerating')
159
+ return [];
160
+ // Accelerating churn + increasing commit rate = complexity spiral
161
+ const weeklyScoreDecay = (temporal.staticScore - temporal.projectedScore) / temporal.projectionWeeks;
162
+ const weeksToThreshold = weeklyScoreDecay > 0
163
+ ? Math.ceil((temporal.temporalScore - this.config.antiPatternThreshold) / weeklyScoreDecay)
164
+ : Infinity;
165
+ if (weeksToThreshold > this.config.forecastWeeks)
166
+ return [];
167
+ return [{
168
+ type: 'complexity-spiral',
169
+ module: module.modulePath,
170
+ severity: weeksToThreshold <= 8 ? 'alert' : 'warning',
171
+ currentScore: temporal.temporalScore,
172
+ projectedScore: temporal.projectedScore,
173
+ weeksToThreshold,
174
+ threshold: this.config.antiPatternThreshold,
175
+ description: `Module '${module.modulePath}' is in a complexity spiral — accelerating churn with increasing commit frequency`,
176
+ evidence: [
177
+ `Churn trend: +${Math.round(temporal.velocity.churnTrend)}%`,
178
+ `Commit acceleration: +${Math.round(temporal.velocity.commitAcceleration)}%`,
179
+ `Current temporal score: ${temporal.temporalScore}/100`,
180
+ `Projected score in ${temporal.projectionWeeks} weeks: ${temporal.projectedScore}/100`,
181
+ ],
182
+ recommendation: 'Stop adding features to this module. Invest in refactoring and test coverage first.',
183
+ confidence: temporal.projectionConfidence,
184
+ }];
185
+ }
186
+ detectCouplingMagnet(module, couplings, temporal) {
187
+ // Files that are increasingly coupled with many others
188
+ const inboundCouplings = couplings.filter(c => c.fileB.startsWith(module.modulePath) && c.confidence > 0.5);
189
+ if (inboundCouplings.length < 3)
190
+ return [];
191
+ if (temporal.velocity.commitAcceleration <= 0)
192
+ return [];
193
+ return [{
194
+ type: 'coupling-magnet',
195
+ module: module.modulePath,
196
+ severity: 'watch',
197
+ currentScore: temporal.staticScore,
198
+ projectedScore: temporal.projectedScore,
199
+ weeksToThreshold: 12,
200
+ threshold: 10,
201
+ description: `Module '${module.modulePath}' is becoming a coupling magnet — ${inboundCouplings.length} high-confidence inbound dependencies`,
202
+ evidence: [
203
+ `${inboundCouplings.length} modules depend on changes here`,
204
+ `Module commit rate accelerating: +${Math.round(temporal.velocity.commitAcceleration)}%`,
205
+ ],
206
+ recommendation: 'Extract stable interfaces. Consider the Dependency Inversion Principle to break inbound coupling.',
207
+ confidence: 0.6,
208
+ }];
209
+ }
210
+ // ── Module Forecast ──
211
+ forecastModule(module, temporal, patterns) {
212
+ const health = this.classifyHealth(temporal);
213
+ const forecast6m = this.classify6MonthForecast(temporal, patterns);
214
+ const bottleneckProb = this.calculateBottleneckProbability(temporal, patterns, module);
215
+ const riskFactors = [];
216
+ if (temporal.trend === 'degrading')
217
+ riskFactors.push('Score degrading');
218
+ if (module.busFactor <= 1)
219
+ riskFactors.push('Single contributor');
220
+ if (temporal.velocity.churnTrend > 30)
221
+ riskFactors.push('Churn increasing');
222
+ if (patterns.length > 0)
223
+ riskFactors.push(`${patterns.length} pre-anti-pattern(s)`);
224
+ const topAction = patterns.length > 0
225
+ ? patterns[0].recommendation
226
+ : temporal.trend === 'degrading'
227
+ ? 'Review recent changes and stabilize'
228
+ : 'No action needed';
229
+ return {
230
+ module: module.modulePath,
231
+ currentHealth: health,
232
+ forecast6Months: forecast6m,
233
+ preAntiPatterns: patterns,
234
+ bottleneckProbability: bottleneckProb,
235
+ riskFactors,
236
+ topAction,
237
+ };
238
+ }
239
+ classifyHealth(temporal) {
240
+ if (temporal.temporalScore < 30)
241
+ return 'critical';
242
+ if (temporal.temporalScore < 50 || temporal.trend === 'degrading')
243
+ return 'degrading';
244
+ if (temporal.temporalScore < 70)
245
+ return 'at-risk';
246
+ return 'healthy';
247
+ }
248
+ classify6MonthForecast(temporal, patterns) {
249
+ const alerts = patterns.filter(p => p.severity === 'alert');
250
+ if (alerts.length > 0 || temporal.projectedScore < 30)
251
+ return 'breakdown';
252
+ if (temporal.trend === 'degrading' || patterns.length > 0)
253
+ return 'declining';
254
+ return 'stable';
255
+ }
256
+ calculateBottleneckProbability(temporal, patterns, module) {
257
+ let prob = 0;
258
+ // Low score → higher probability
259
+ if (temporal.temporalScore < 50)
260
+ prob += 0.3;
261
+ else if (temporal.temporalScore < 70)
262
+ prob += 0.1;
263
+ // Degrading trend
264
+ if (temporal.trend === 'degrading')
265
+ prob += 0.2;
266
+ // Pre-anti-patterns
267
+ prob += Math.min(0.3, patterns.length * 0.1);
268
+ // Low bus factor
269
+ if (module.busFactor <= 1)
270
+ prob += 0.1;
271
+ // High churn
272
+ if (temporal.velocity.churnTrend > 30)
273
+ prob += 0.1;
274
+ return Math.min(1, Math.round(prob * 100) / 100);
275
+ }
276
+ // ── Overall Analysis ──
277
+ classifyOutlook(temporal, patterns) {
278
+ const alerts = patterns.filter(p => p.severity === 'alert');
279
+ if (alerts.length >= 2 || temporal.overallTrend === 'degrading')
280
+ return 'stormy';
281
+ if (alerts.length >= 1 || patterns.length >= 3)
282
+ return 'cloudy';
283
+ return 'sunny';
284
+ }
285
+ generateHeadline(outlook, patterns, modules) {
286
+ const critical = modules.filter(m => m.currentHealth === 'critical').length;
287
+ const degrading = modules.filter(m => m.currentHealth === 'degrading').length;
288
+ switch (outlook) {
289
+ case 'stormy':
290
+ return `${critical + degrading} module(s) at risk. ${patterns.length} pre-anti-pattern(s) detected. Immediate action recommended.`;
291
+ case 'cloudy':
292
+ return `Architecture trending stable with ${patterns.length} emerging concern(s). Proactive refactoring recommended.`;
293
+ case 'sunny':
294
+ return 'Architecture is healthy and stable. Continue current practices.';
295
+ }
296
+ }
297
+ identifyTopRisks(modules, patterns) {
298
+ const risks = [];
299
+ const breakdowns = modules.filter(m => m.forecast6Months === 'breakdown');
300
+ if (breakdowns.length > 0) {
301
+ risks.push(`${breakdowns.length} module(s) projected to break down within 6 months: ${breakdowns.map(m => m.module).join(', ')}`);
302
+ }
303
+ const busRisks = patterns.filter(p => p.type === 'bus-factor-risk');
304
+ if (busRisks.length > 0) {
305
+ risks.push(`Bus factor risk in ${busRisks.length} module(s) — knowledge concentrated in single contributors`);
306
+ }
307
+ const spirals = patterns.filter(p => p.type === 'complexity-spiral');
308
+ if (spirals.length > 0) {
309
+ risks.push(`Complexity spiral detected in: ${spirals.map(p => p.module).join(', ')}`);
310
+ }
311
+ return risks.slice(0, 5);
312
+ }
313
+ generateRecommendations(modules, patterns) {
314
+ const recs = [];
315
+ const critical = modules.filter(m => m.currentHealth === 'critical');
316
+ if (critical.length > 0) {
317
+ recs.push(`Immediate: Stabilize ${critical.map(m => m.module).join(', ')} — freeze features, invest in refactoring`);
318
+ }
319
+ const godClasses = patterns.filter(p => p.type === 'emerging-god-class');
320
+ if (godClasses.length > 0) {
321
+ recs.push(`Split growing files before they become god classes: ${godClasses.map(p => p.module).join(', ')}`);
322
+ }
323
+ const shotgun = patterns.filter(p => p.type === 'emerging-shotgun-surgery');
324
+ if (shotgun.length > 0) {
325
+ recs.push(`Decouple modules with high change coupling to prevent shotgun surgery`);
326
+ }
327
+ const busRisks = patterns.filter(p => p.type === 'bus-factor-risk');
328
+ if (busRisks.length > 0) {
329
+ recs.push(`Spread knowledge: pair programming or rotation for ${busRisks.map(p => p.module).join(', ')}`);
330
+ }
331
+ if (recs.length === 0) {
332
+ recs.push('Architecture is healthy. Continue monitoring temporal trends.');
333
+ }
334
+ return recs.slice(0, 5);
335
+ }
336
+ }
337
+ //# sourceMappingURL=forecast.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"forecast.js","sourceRoot":"","sources":["../../src/analyzers/forecast.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Git History Cache — Serialize/deserialize GitHistoryReport
3
+ */
4
+ import type { GitHistoryReport } from './git-history.js';
5
+ export declare function saveToCache(report: GitHistoryReport, projectPath: string, cacheDir?: string): void;
6
+ export declare function loadFromCache(projectPath: string, cacheDir?: string, maxAgeMs?: number): GitHistoryReport | null;
7
+ //# sourceMappingURL=git-cache.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git-cache.d.ts","sourceRoot":"","sources":["../../src/analyzers/git-cache.ts"],"names":[],"mappings":"AAAA;;GAEG;AAIH,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAEzD,wBAAgB,WAAW,CACzB,MAAM,EAAE,gBAAgB,EACxB,WAAW,EAAE,MAAM,EACnB,QAAQ,SAAqB,GAC5B,IAAI,CAsBN;AAED,wBAAgB,aAAa,CAC3B,WAAW,EAAE,MAAM,EACnB,QAAQ,SAAqB,EAC7B,QAAQ,SAAU,GACjB,gBAAgB,GAAG,IAAI,CAWzB"}
@@ -0,0 +1,41 @@
1
+ /**
2
+ * Git History Cache — Serialize/deserialize GitHistoryReport
3
+ */
4
+ import * as fs from 'fs';
5
+ import * as path from 'path';
6
+ export function saveToCache(report, projectPath, cacheDir = '.architect-cache') {
7
+ const dir = path.join(projectPath, cacheDir);
8
+ fs.mkdirSync(dir, { recursive: true });
9
+ const serializable = {
10
+ ...report,
11
+ modules: report.modules.map(m => ({
12
+ ...m,
13
+ files: m.files.map(f => ({
14
+ ...f,
15
+ authors: Array.from(f.authors),
16
+ lastModified: f.lastModified.toISOString(),
17
+ })),
18
+ })),
19
+ hotspots: report.hotspots.map(f => ({
20
+ ...f,
21
+ authors: Array.from(f.authors),
22
+ lastModified: f.lastModified.toISOString(),
23
+ })),
24
+ };
25
+ fs.writeFileSync(path.join(dir, 'git-history.json'), JSON.stringify(serializable, null, 2));
26
+ }
27
+ export function loadFromCache(projectPath, cacheDir = '.architect-cache', maxAgeMs = 3600000) {
28
+ const cachePath = path.join(projectPath, cacheDir, 'git-history.json');
29
+ if (!fs.existsSync(cachePath))
30
+ return null;
31
+ try {
32
+ const raw = JSON.parse(fs.readFileSync(cachePath, 'utf-8'));
33
+ if (Date.now() - new Date(raw.analyzedAt).getTime() > maxAgeMs)
34
+ return null;
35
+ return raw;
36
+ }
37
+ catch {
38
+ return null;
39
+ }
40
+ }
41
+ //# sourceMappingURL=git-cache.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git-cache.js","sourceRoot":"","sources":["../../src/analyzers/git-cache.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,MAAM,IAAI,CAAC;AACzB,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAG7B,MAAM,UAAU,WAAW,CACzB,MAAwB,EACxB,WAAmB,EACnB,QAAQ,GAAG,kBAAkB;IAE7B,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;IAC7C,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAEvC,MAAM,YAAY,GAAG;QACnB,GAAG,MAAM;QACT,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAChC,GAAG,CAAC;YACJ,KAAK,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBACvB,GAAG,CAAC;gBACJ,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC;gBAC9B,YAAY,EAAE,CAAC,CAAC,YAAY,CAAC,WAAW,EAAE;aAC3C,CAAC,CAAC;SACJ,CAAC,CAAC;QACH,QAAQ,EAAE,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAClC,GAAG,CAAC;YACJ,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC;YAC9B,YAAY,EAAE,CAAC,CAAC,YAAY,CAAC,WAAW,EAAE;SAC3C,CAAC,CAAC;KACJ,CAAC;IAEF,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,kBAAkB,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;AAC9F,CAAC;AAED,MAAM,UAAU,aAAa,CAC3B,WAAmB,EACnB,QAAQ,GAAG,kBAAkB,EAC7B,QAAQ,GAAG,OAAO;IAElB,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,QAAQ,EAAE,kBAAkB,CAAC,CAAC;IACvE,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,IAAI,CAAC;IAE3C,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;QAC5D,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,OAAO,EAAE,GAAG,QAAQ;YAAE,OAAO,IAAI,CAAC;QAC5E,OAAO,GAAG,CAAC;IACb,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC"}
@@ -0,0 +1,113 @@
1
+ /**
2
+ * Git History Analyzer — Temporal analysis of codebase evolution
3
+ *
4
+ * Reads git log to build velocity vectors, churn rates, and hotspot maps.
5
+ * Enables Architect v4's predictive capabilities.
6
+ *
7
+ * Key metrics per module:
8
+ * - Commit frequency (commits/week rolling 4-week average)
9
+ * - Churn rate (lines added + deleted per commit)
10
+ * - Author diversity (bus factor proxy)
11
+ * - Change coupling (files that change together)
12
+ *
13
+ * @author Camilo Girardelli — Girardelli Tecnologia
14
+ * @license MIT
15
+ */
16
+ export interface GitCommit {
17
+ hash: string;
18
+ author: string;
19
+ date: Date;
20
+ message: string;
21
+ files: FileChange[];
22
+ }
23
+ export interface FileChange {
24
+ path: string;
25
+ additions: number;
26
+ deletions: number;
27
+ }
28
+ export interface FileHistory {
29
+ path: string;
30
+ commits: number;
31
+ totalAdditions: number;
32
+ totalDeletions: number;
33
+ churnRate: number;
34
+ authors: Set<string>;
35
+ busFactor: number;
36
+ lastModified: Date;
37
+ weeklyCommitRate: number;
38
+ isHotspot: boolean;
39
+ }
40
+ export interface ModuleHistory {
41
+ modulePath: string;
42
+ files: FileHistory[];
43
+ aggregateCommits: number;
44
+ aggregateChurn: number;
45
+ avgWeeklyRate: number;
46
+ topHotspots: FileHistory[];
47
+ velocityVector: VelocityVector;
48
+ busFactor: number;
49
+ }
50
+ export interface VelocityVector {
51
+ /** Commit rate trend: positive = accelerating, negative = decelerating */
52
+ commitAcceleration: number;
53
+ /** Churn trend: positive = increasing complexity, negative = stabilizing */
54
+ churnTrend: number;
55
+ /** Overall direction: "accelerating" | "stable" | "decelerating" */
56
+ direction: 'accelerating' | 'stable' | 'decelerating';
57
+ }
58
+ export interface ChangeCoupling {
59
+ fileA: string;
60
+ fileB: string;
61
+ cochangeCount: number;
62
+ confidence: number;
63
+ }
64
+ export interface GitHistoryReport {
65
+ projectPath: string;
66
+ analyzedAt: string;
67
+ periodWeeks: number;
68
+ totalCommits: number;
69
+ totalAuthors: number;
70
+ modules: ModuleHistory[];
71
+ hotspots: FileHistory[];
72
+ changeCouplings: ChangeCoupling[];
73
+ commitTimeline: WeeklySnapshot[];
74
+ }
75
+ export interface WeeklySnapshot {
76
+ weekStart: string;
77
+ commits: number;
78
+ churn: number;
79
+ activeFiles: number;
80
+ }
81
+ export interface GitAnalyzerConfig {
82
+ /** How many weeks of history to analyze (default: 24) */
83
+ periodWeeks?: number;
84
+ /** Rolling window for averages (default: 4 weeks) */
85
+ rollingWindowWeeks?: number;
86
+ /** Churn threshold to flag as hotspot (default: 500 lines) */
87
+ hotspotChurnThreshold?: number;
88
+ /** Minimum co-change count for coupling (default: 3) */
89
+ couplingMinCochanges?: number;
90
+ /** Path to cache dir (default: .architect-cache/) */
91
+ cacheDir?: string;
92
+ }
93
+ export declare class GitHistoryAnalyzer {
94
+ private config;
95
+ constructor(config?: GitAnalyzerConfig);
96
+ /**
97
+ * Analyze git history for the project at the given path.
98
+ * Returns a comprehensive GitHistoryReport.
99
+ */
100
+ analyze(projectPath: string): GitHistoryReport;
101
+ private parseGitLog;
102
+ private parseLogOutput;
103
+ private buildFileHistories;
104
+ private groupByModule;
105
+ private calculateVelocity;
106
+ private detectHotspots;
107
+ private detectChangeCoupling;
108
+ private buildTimeline;
109
+ private validateGitRepo;
110
+ private getSinceDate;
111
+ private getModulePath;
112
+ }
113
+ //# sourceMappingURL=git-history.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git-history.d.ts","sourceRoot":"","sources":["../../src/analyzers/git-history.ts"],"names":[],"mappings":"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"}