@levironexe/architect 0.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 (210) hide show
  1. package/CHANGELOG.md +32 -0
  2. package/CONTRIBUTING.md +55 -0
  3. package/README.md +341 -0
  4. package/dist/analyzers/ast-parser.d.ts +3 -0
  5. package/dist/analyzers/ast-parser.js +305 -0
  6. package/dist/analyzers/ast-parser.js.map +1 -0
  7. package/dist/analyzers/dependency-graph.d.ts +2 -0
  8. package/dist/analyzers/dependency-graph.js +67 -0
  9. package/dist/analyzers/dependency-graph.js.map +1 -0
  10. package/dist/analyzers/duplication.d.ts +2 -0
  11. package/dist/analyzers/duplication.js +56 -0
  12. package/dist/analyzers/duplication.js.map +1 -0
  13. package/dist/analyzers/file-walker.d.ts +3 -0
  14. package/dist/analyzers/file-walker.js +80 -0
  15. package/dist/analyzers/file-walker.js.map +1 -0
  16. package/dist/cli/context-runner.d.ts +1 -0
  17. package/dist/cli/context-runner.js +16 -0
  18. package/dist/cli/context-runner.js.map +1 -0
  19. package/dist/cli/index.d.ts +24 -0
  20. package/dist/cli/index.js +217 -0
  21. package/dist/cli/index.js.map +1 -0
  22. package/dist/cli/init-runner.d.ts +25 -0
  23. package/dist/cli/init-runner.js +152 -0
  24. package/dist/cli/init-runner.js.map +1 -0
  25. package/dist/cli/scan-runner.d.ts +8 -0
  26. package/dist/cli/scan-runner.js +133 -0
  27. package/dist/cli/scan-runner.js.map +1 -0
  28. package/dist/formatters/plan-json.d.ts +2 -0
  29. package/dist/formatters/plan-json.js +4 -0
  30. package/dist/formatters/plan-json.js.map +1 -0
  31. package/dist/formatters/plan-markdown.d.ts +2 -0
  32. package/dist/formatters/plan-markdown.js +42 -0
  33. package/dist/formatters/plan-markdown.js.map +1 -0
  34. package/dist/formatters/plan-prompt.d.ts +4 -0
  35. package/dist/formatters/plan-prompt.js +5 -0
  36. package/dist/formatters/plan-prompt.js.map +1 -0
  37. package/dist/formatters/plan-terminal.d.ts +5 -0
  38. package/dist/formatters/plan-terminal.js +62 -0
  39. package/dist/formatters/plan-terminal.js.map +1 -0
  40. package/dist/generators/blueprint-renderer.d.ts +3 -0
  41. package/dist/generators/blueprint-renderer.js +27 -0
  42. package/dist/generators/blueprint-renderer.js.map +1 -0
  43. package/dist/generators/claudeWriter.d.ts +3 -0
  44. package/dist/generators/claudeWriter.js +9 -0
  45. package/dist/generators/claudeWriter.js.map +1 -0
  46. package/dist/generators/copilotWriter.d.ts +3 -0
  47. package/dist/generators/copilotWriter.js +11 -0
  48. package/dist/generators/copilotWriter.js.map +1 -0
  49. package/dist/generators/cursorWriter.d.ts +3 -0
  50. package/dist/generators/cursorWriter.js +14 -0
  51. package/dist/generators/cursorWriter.js.map +1 -0
  52. package/dist/generators/genericWriter.d.ts +3 -0
  53. package/dist/generators/genericWriter.js +9 -0
  54. package/dist/generators/genericWriter.js.map +1 -0
  55. package/dist/generators/template-context.d.ts +18 -0
  56. package/dist/generators/template-context.js +126 -0
  57. package/dist/generators/template-context.js.map +1 -0
  58. package/dist/generators/templateRenderer.d.ts +2 -0
  59. package/dist/generators/templateRenderer.js +19 -0
  60. package/dist/generators/templateRenderer.js.map +1 -0
  61. package/dist/generators/windsurfWriter.d.ts +3 -0
  62. package/dist/generators/windsurfWriter.js +14 -0
  63. package/dist/generators/windsurfWriter.js.map +1 -0
  64. package/dist/generators/writer-types.d.ts +11 -0
  65. package/dist/generators/writer-types.js +40 -0
  66. package/dist/generators/writer-types.js.map +1 -0
  67. package/dist/llm/claude-provider.d.ts +8 -0
  68. package/dist/llm/claude-provider.js +22 -0
  69. package/dist/llm/claude-provider.js.map +1 -0
  70. package/dist/llm/concern-classifier.d.ts +15 -0
  71. package/dist/llm/concern-classifier.js +61 -0
  72. package/dist/llm/concern-classifier.js.map +1 -0
  73. package/dist/llm/config.d.ts +11 -0
  74. package/dist/llm/config.js +120 -0
  75. package/dist/llm/config.js.map +1 -0
  76. package/dist/llm/ollama-provider.d.ts +8 -0
  77. package/dist/llm/ollama-provider.js +27 -0
  78. package/dist/llm/ollama-provider.js.map +1 -0
  79. package/dist/llm/openai-provider.d.ts +8 -0
  80. package/dist/llm/openai-provider.js +19 -0
  81. package/dist/llm/openai-provider.js.map +1 -0
  82. package/dist/llm/prompt-builder.d.ts +12 -0
  83. package/dist/llm/prompt-builder.js +132 -0
  84. package/dist/llm/prompt-builder.js.map +1 -0
  85. package/dist/llm/provider.d.ts +17 -0
  86. package/dist/llm/provider.js +2 -0
  87. package/dist/llm/provider.js.map +1 -0
  88. package/dist/llm/response-parser.d.ts +6 -0
  89. package/dist/llm/response-parser.js +128 -0
  90. package/dist/llm/response-parser.js.map +1 -0
  91. package/dist/planner/plan-generator.d.ts +7 -0
  92. package/dist/planner/plan-generator.js +275 -0
  93. package/dist/planner/plan-generator.js.map +1 -0
  94. package/dist/planner/plan-prompt-builder.d.ts +9 -0
  95. package/dist/planner/plan-prompt-builder.js +92 -0
  96. package/dist/planner/plan-prompt-builder.js.map +1 -0
  97. package/dist/planner/plan-response-parser.d.ts +7 -0
  98. package/dist/planner/plan-response-parser.js +21 -0
  99. package/dist/planner/plan-response-parser.js.map +1 -0
  100. package/dist/planner/plan-validator.d.ts +3 -0
  101. package/dist/planner/plan-validator.js +49 -0
  102. package/dist/planner/plan-validator.js.map +1 -0
  103. package/dist/reporters/scan-json.d.ts +13 -0
  104. package/dist/reporters/scan-json.js +26 -0
  105. package/dist/reporters/scan-json.js.map +1 -0
  106. package/dist/reporters/terminal.d.ts +6 -0
  107. package/dist/reporters/terminal.js +224 -0
  108. package/dist/reporters/terminal.js.map +1 -0
  109. package/dist/scoring/consistency-score.d.ts +3 -0
  110. package/dist/scoring/consistency-score.js +23 -0
  111. package/dist/scoring/consistency-score.js.map +1 -0
  112. package/dist/scoring/duplication-score.d.ts +3 -0
  113. package/dist/scoring/duplication-score.js +16 -0
  114. package/dist/scoring/duplication-score.js.map +1 -0
  115. package/dist/scoring/health-score.d.ts +4 -0
  116. package/dist/scoring/health-score.js +20 -0
  117. package/dist/scoring/health-score.js.map +1 -0
  118. package/dist/scoring/issue-builder.d.ts +4 -0
  119. package/dist/scoring/issue-builder.js +62 -0
  120. package/dist/scoring/issue-builder.js.map +1 -0
  121. package/dist/scoring/modularity-score.d.ts +3 -0
  122. package/dist/scoring/modularity-score.js +56 -0
  123. package/dist/scoring/modularity-score.js.map +1 -0
  124. package/dist/scoring/pattern-analysis.d.ts +3 -0
  125. package/dist/scoring/pattern-analysis.js +74 -0
  126. package/dist/scoring/pattern-analysis.js.map +1 -0
  127. package/dist/scoring/separation-score.d.ts +3 -0
  128. package/dist/scoring/separation-score.js +35 -0
  129. package/dist/scoring/separation-score.js.map +1 -0
  130. package/dist/skills/detector.d.ts +4 -0
  131. package/dist/skills/detector.js +104 -0
  132. package/dist/skills/detector.js.map +1 -0
  133. package/dist/skills/lister.d.ts +9 -0
  134. package/dist/skills/lister.js +35 -0
  135. package/dist/skills/lister.js.map +1 -0
  136. package/dist/skills/loader.d.ts +6 -0
  137. package/dist/skills/loader.js +76 -0
  138. package/dist/skills/loader.js.map +1 -0
  139. package/dist/skills/structure-check.d.ts +2 -0
  140. package/dist/skills/structure-check.js +37 -0
  141. package/dist/skills/structure-check.js.map +1 -0
  142. package/dist/skills/validator.d.ts +6 -0
  143. package/dist/skills/validator.js +229 -0
  144. package/dist/skills/validator.js.map +1 -0
  145. package/dist/types/analysis.d.ts +130 -0
  146. package/dist/types/analysis.js +41 -0
  147. package/dist/types/analysis.js.map +1 -0
  148. package/dist/types/concern.d.ts +48 -0
  149. package/dist/types/concern.js +16 -0
  150. package/dist/types/concern.js.map +1 -0
  151. package/dist/types/generation.d.ts +32 -0
  152. package/dist/types/generation.js +2 -0
  153. package/dist/types/generation.js.map +1 -0
  154. package/dist/types/issue.d.ts +12 -0
  155. package/dist/types/issue.js +2 -0
  156. package/dist/types/issue.js.map +1 -0
  157. package/dist/types/pattern.d.ts +15 -0
  158. package/dist/types/pattern.js +2 -0
  159. package/dist/types/pattern.js.map +1 -0
  160. package/dist/types/plan.d.ts +56 -0
  161. package/dist/types/plan.js +2 -0
  162. package/dist/types/plan.js.map +1 -0
  163. package/dist/types/scan-output.d.ts +84 -0
  164. package/dist/types/scan-output.js +2 -0
  165. package/dist/types/scan-output.js.map +1 -0
  166. package/dist/types/scoring.d.ts +15 -0
  167. package/dist/types/scoring.js +2 -0
  168. package/dist/types/scoring.js.map +1 -0
  169. package/dist/types/skill.d.ts +97 -0
  170. package/dist/types/skill.js +2 -0
  171. package/dist/types/skill.js.map +1 -0
  172. package/dist/utils/agent-detector.d.ts +2 -0
  173. package/dist/utils/agent-detector.js +22 -0
  174. package/dist/utils/agent-detector.js.map +1 -0
  175. package/dist/utils/interactive.d.ts +6 -0
  176. package/dist/utils/interactive.js +15 -0
  177. package/dist/utils/interactive.js.map +1 -0
  178. package/dist/utils/path.d.ts +5 -0
  179. package/dist/utils/path.js +31 -0
  180. package/dist/utils/path.js.map +1 -0
  181. package/dist/utils/progress.d.ts +17 -0
  182. package/dist/utils/progress.js +48 -0
  183. package/dist/utils/progress.js.map +1 -0
  184. package/dist/utils/thresholds.d.ts +6 -0
  185. package/dist/utils/thresholds.js +48 -0
  186. package/dist/utils/thresholds.js.map +1 -0
  187. package/package.json +63 -0
  188. package/skills/meta/general-js.skill.yaml +131 -0
  189. package/skills/patterns/clerk-auth.skill.yaml +349 -0
  190. package/skills/patterns/docker-deploy.skill.yaml +214 -0
  191. package/skills/patterns/drizzle.skill.yaml +277 -0
  192. package/skills/patterns/mongoose.skill.yaml +290 -0
  193. package/skills/patterns/nextauth.skill.yaml +308 -0
  194. package/skills/patterns/playwright-e2e.skill.yaml +265 -0
  195. package/skills/patterns/prisma.skill.yaml +255 -0
  196. package/skills/patterns/s3-storage.skill.yaml +235 -0
  197. package/skills/patterns/selenium-e2e.skill.yaml +276 -0
  198. package/skills/patterns/supabase-auth.skill.yaml +298 -0
  199. package/skills/patterns/supabase.skill.yaml +304 -0
  200. package/skills/patterns/vercel-deploy.skill.yaml +219 -0
  201. package/skills/patterns/vitest-testing.skill.yaml +262 -0
  202. package/skills/stacks/express-api.skill.yaml +155 -0
  203. package/skills/stacks/fastify-api.skill.yaml +119 -0
  204. package/skills/stacks/hono-api.skill.yaml +130 -0
  205. package/skills/stacks/nestjs.skill.yaml +135 -0
  206. package/skills/stacks/nextjs-app-router.skill.yaml +176 -0
  207. package/skills/stacks/react-spa.skill.yaml +153 -0
  208. package/skills/stacks/vue-nuxt.skill.yaml +115 -0
  209. package/templates/architect-plan.md +139 -0
  210. package/templates/architect-refactor.md +119 -0
@@ -0,0 +1,224 @@
1
+ import path from 'node:path';
2
+ import { Chalk } from 'chalk';
3
+ import { createEmptyDependencyGraphSummary, createEmptyDuplicationSummary } from '../types/analysis.js';
4
+ export function renderDiscoveryReport(targetDirectory, files) {
5
+ if (files.length === 0) {
6
+ process.stdout.write(`No supported JS/TS files found in ${targetDirectory}\n`);
7
+ return;
8
+ }
9
+ process.stdout.write(`Architect scan: ${targetDirectory}\n\n`);
10
+ for (const filePath of files) {
11
+ process.stdout.write(`${path.relative(targetDirectory, filePath)}\n`);
12
+ }
13
+ }
14
+ export function renderScanReport(result, options = {}) {
15
+ const chalk = new Chalk({ level: options.color === false ? 0 : 1 });
16
+ const dependencyGraph = result.dependencyGraph ?? createEmptyDependencyGraphSummary(false);
17
+ const duplication = result.duplication ?? createEmptyDuplicationSummary(false);
18
+ const dependencyHotspots = result.summary.dependencyHotspots ?? 0;
19
+ const circularDependencies = result.summary.circularDependencies ?? 0;
20
+ const duplicateFindings = result.summary.duplicateFindings ?? 0;
21
+ const duplicatedLines = result.summary.duplicatedLines ?? 0;
22
+ if (result.files.length === 0 && result.parseErrors.length === 0) {
23
+ process.stdout.write(`No supported JS/TS files found in ${result.summary.targetDir}\n`);
24
+ return;
25
+ }
26
+ process.stdout.write(`Architect scan: ${result.summary.targetDir}\n\n`);
27
+ renderProjectOverview(result);
28
+ renderDetectedArchitecture(result);
29
+ renderStructureComparison(result);
30
+ process.stdout.write(`${pad('FILE', 28)} ${pad('LOC', 5)} ${pad('FN', 4)} ${pad('CL', 4)} ${pad('IMP', 5)} ${pad('EXP', 5)} STATUS\n`);
31
+ for (const file of result.files) {
32
+ const status = formatStatus(file, chalk);
33
+ process.stdout.write(`${pad(file.relativePath, 28)} ${pad(String(file.loc), 5)} ${pad(String(file.functions.length), 4)} ${pad(String(file.classes.length), 4)} ${pad(String(file.imports.length), 5)} ${pad(String(file.exports.length), 5)} ${status}\n`);
34
+ }
35
+ const flaggedFunctions = result.files.flatMap((file) => file.functions
36
+ .filter((item) => item.isFlagged)
37
+ .map((item) => `${file.relativePath} :: ${item.name} (complexity ${item.complexity})`));
38
+ if (flaggedFunctions.length > 0) {
39
+ process.stdout.write(`\nCritical functions:\n`);
40
+ for (const entry of flaggedFunctions) {
41
+ process.stdout.write(`- ${entry}\n`);
42
+ }
43
+ }
44
+ process.stdout.write(`\nDependency insights:\n`);
45
+ if (dependencyGraph.hotspots.length > 0) {
46
+ for (const hotspot of dependencyGraph.hotspots) {
47
+ process.stdout.write(`- Hotspot: ${hotspot.relativePath} (depended on by ${hotspot.dependentCount} files)\n`);
48
+ }
49
+ }
50
+ if (dependencyGraph.circularDependencies.length > 0) {
51
+ for (const cycle of dependencyGraph.circularDependencies) {
52
+ process.stdout.write(`- Circular dependency: ${cycle.files.join(' -> ')}\n`);
53
+ }
54
+ }
55
+ if (dependencyGraph.unreferencedFiles.length > 0) {
56
+ for (const filePath of dependencyGraph.unreferencedFiles) {
57
+ process.stdout.write(`- Unreferenced: ${filePath}\n`);
58
+ }
59
+ }
60
+ if (dependencyGraph.hotspots.length === 0
61
+ && dependencyGraph.circularDependencies.length === 0
62
+ && dependencyGraph.unreferencedFiles.length === 0) {
63
+ process.stdout.write(`- No dependency risks detected\n`);
64
+ }
65
+ process.stdout.write(`\nDuplication findings:\n`);
66
+ if (duplication.findings.length > 0) {
67
+ for (const finding of duplication.findings) {
68
+ process.stdout.write(`- ${formatDuplicationFinding(finding)}\n`);
69
+ }
70
+ }
71
+ else {
72
+ process.stdout.write(`- No significant duplication findings detected\n`);
73
+ }
74
+ renderHealthReport(result);
75
+ renderIssues(result);
76
+ renderGuidance(result);
77
+ if (options.verbose) {
78
+ renderVerboseDiagnostics(result.diagnostics ?? []);
79
+ }
80
+ process.stdout.write(`\nSummary:\n`);
81
+ process.stdout.write(`- Files scanned: ${result.summary.totalFiles}\n`);
82
+ process.stdout.write(`- Total LOC: ${result.summary.totalLoc}\n`);
83
+ process.stdout.write(`- Flagged files: ${result.summary.flaggedFiles}\n`);
84
+ process.stdout.write(`- Flagged functions: ${result.summary.flaggedFunctions}\n`);
85
+ process.stdout.write(`- Dependency hotspots: ${dependencyHotspots}\n`);
86
+ process.stdout.write(`- Circular dependencies: ${circularDependencies}\n`);
87
+ process.stdout.write(`- Duplicate findings: ${duplicateFindings}\n`);
88
+ process.stdout.write(`- Duplicated lines: ${duplicatedLines}\n`);
89
+ process.stdout.write(`- Skipped files: ${result.summary.skippedFiles}\n`);
90
+ if (result.summary.flaggedFiles === 0
91
+ && result.summary.flaggedFunctions === 0
92
+ && dependencyHotspots === 0
93
+ && circularDependencies === 0
94
+ && duplicateFindings === 0) {
95
+ process.stdout.write(`- No critical findings detected\n`);
96
+ }
97
+ for (const parseError of result.parseErrors) {
98
+ process.stderr.write(`WARN Failed to parse ${parseError.relativePath}: ${parseError.error}\n`);
99
+ }
100
+ if (result.parseErrors.length > 0) {
101
+ const noun = result.parseErrors.length === 1 ? 'file' : 'files';
102
+ process.stderr.write(`WARN Skipped ${result.parseErrors.length} ${noun} due to syntax errors. Run with --verbose to see file paths and parser reasons.\n`);
103
+ }
104
+ for (const warning of result.skillLoadWarnings ?? []) {
105
+ process.stderr.write(`WARN Ignored invalid architecture skill ${warning.file}: ${warning.message}\n`);
106
+ }
107
+ if ((dependencyGraph.isPartial || duplication.isPartial) && result.summary.skippedFiles > 0) {
108
+ const noun = result.summary.skippedFiles === 1 ? 'file was' : 'files were';
109
+ process.stderr.write(`WARN Dependency and duplication findings may be partial because ${result.summary.skippedFiles} ${noun} skipped\n`);
110
+ }
111
+ }
112
+ function renderVerboseDiagnostics(diagnostics) {
113
+ process.stdout.write(`\nVerbose diagnostics:\n`);
114
+ if (diagnostics.length === 0) {
115
+ process.stdout.write(`- No additional diagnostics recorded\n`);
116
+ return;
117
+ }
118
+ for (const diagnostic of diagnostics) {
119
+ const details = diagnostic.details ? ` ${JSON.stringify(diagnostic.details)}` : '';
120
+ process.stdout.write(`- ${diagnostic.phase}: ${diagnostic.message}${details}\n`);
121
+ }
122
+ }
123
+ function renderProjectOverview(result) {
124
+ process.stdout.write(`Project overview:\n`);
125
+ process.stdout.write(`- Files scanned: ${result.summary.totalFiles}\n`);
126
+ process.stdout.write(`- Total LOC: ${result.summary.totalLoc}\n`);
127
+ process.stdout.write(`- Languages: ${formatLanguages(result.files)}\n\n`);
128
+ }
129
+ function renderDetectedArchitecture(result) {
130
+ const matches = result.matchedSkills ?? [];
131
+ const primary = matches.find((match) => match.primary);
132
+ const secondary = matches.filter((match) => !match.primary);
133
+ process.stdout.write(`Detected architecture:\n`);
134
+ if (primary) {
135
+ process.stdout.write(`- Primary: ${primary.skill.name} (${primary.skill.id}) [${primary.confidence} confidence]\n`);
136
+ }
137
+ else {
138
+ process.stdout.write(`- No confident primary architecture skill detected\n`);
139
+ }
140
+ for (const match of secondary) {
141
+ process.stdout.write(`- Secondary: ${match.skill.name} (${match.skill.id})\n`);
142
+ }
143
+ process.stdout.write(`\n`);
144
+ }
145
+ function renderStructureComparison(result) {
146
+ const comparison = result.structureComparison;
147
+ process.stdout.write(`Structure comparison:\n`);
148
+ if (!comparison?.isAvailable) {
149
+ process.stdout.write(`- Unavailable because no primary architecture skill was detected\n\n`);
150
+ return;
151
+ }
152
+ if (comparison.entries.length === 0) {
153
+ process.stdout.write(`- No expected structural areas declared by ${comparison.skillId}\n\n`);
154
+ return;
155
+ }
156
+ for (const entry of comparison.entries) {
157
+ process.stdout.write(`- ${pad(entry.status, 8)} ${pad(entry.path, 18)} ${entry.purpose}\n`);
158
+ }
159
+ process.stdout.write(`\n`);
160
+ }
161
+ function renderHealthReport(result) {
162
+ process.stdout.write(`\nHealth report:\n`);
163
+ const scores = result.scores;
164
+ if (!scores) {
165
+ process.stdout.write(`- Unavailable\n`);
166
+ return;
167
+ }
168
+ process.stdout.write(`- Overall score: ${scores.overall} ${scores.label}\n`);
169
+ process.stdout.write(`- Dimensions:\n`);
170
+ process.stdout.write(` - modularity: ${scores.modularity.score} ${scores.modularity.label} - ${scores.modularity.reasons.join('; ')}\n`);
171
+ process.stdout.write(` - duplication: ${scores.duplication.score} ${scores.duplication.label} - ${scores.duplication.reasons.join('; ')}\n`);
172
+ }
173
+ function renderIssues(result) {
174
+ const issues = result.issues ?? [];
175
+ process.stdout.write(`\nRanked issues:\n`);
176
+ if (issues.length === 0) {
177
+ process.stdout.write(`- No ranked structural issues detected\n`);
178
+ return;
179
+ }
180
+ for (const issue of issues) {
181
+ const location = issue.location ? ` [${issue.location}]` : '';
182
+ process.stdout.write(`- ${issue.severity.toUpperCase()} ${issue.category}${location}: ${issue.message} ${issue.suggestion}\n`);
183
+ }
184
+ }
185
+ function renderGuidance(result) {
186
+ const guidance = result.guidance;
187
+ if (!guidance) {
188
+ return;
189
+ }
190
+ process.stdout.write(`\nNext step:\n`);
191
+ process.stdout.write(`- ${guidance.message}`);
192
+ if (guidance.command) {
193
+ process.stdout.write(` Run \`${guidance.command}\` to generate a refactoring roadmap.`);
194
+ }
195
+ process.stdout.write(`\n`);
196
+ }
197
+ function formatDuplicationFinding(finding) {
198
+ const [left, right] = finding.occurrences;
199
+ return `Duplicate block (${finding.duplicatedLines} lines): ${left?.relativePath}:${left?.startLine}-${left?.endLine} <-> ${right?.relativePath}:${right?.startLine}-${right?.endLine}`;
200
+ }
201
+ function formatStatus(file, chalk) {
202
+ if (file.isOversized) {
203
+ return chalk.red('OVERSIZED');
204
+ }
205
+ if (file.hasCriticalComplexity) {
206
+ return chalk.yellow('COMPLEX');
207
+ }
208
+ return chalk.green('OK');
209
+ }
210
+ function pad(value, width) {
211
+ return value.length >= width ? value.slice(0, width) : value.padEnd(width, ' ');
212
+ }
213
+ function formatLanguages(files) {
214
+ const extensions = new Set(files.map((file) => path.extname(file.relativePath)));
215
+ const labels = [];
216
+ if (extensions.has('.ts') || extensions.has('.tsx')) {
217
+ labels.push('TypeScript');
218
+ }
219
+ if (extensions.has('.js') || extensions.has('.jsx')) {
220
+ labels.push('JavaScript');
221
+ }
222
+ return labels.length > 0 ? labels.join(', ') : 'Unknown';
223
+ }
224
+ //# sourceMappingURL=terminal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"terminal.js","sourceRoot":"","sources":["../../src/reporters/terminal.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { PatternFinding } from '../types/pattern.js';
2
+ import type { DimensionScore } from '../types/scoring.js';
3
+ export declare function scoreConsistency(findings: PatternFinding[] | undefined): DimensionScore | null;
@@ -0,0 +1,23 @@
1
+ import { clampScore, labelForScore } from './health-score.js';
2
+ const WEIGHT = 25;
3
+ export function scoreConsistency(findings) {
4
+ const usable = (findings ?? []).filter((finding) => finding.confidence !== 'insufficient');
5
+ if (usable.length === 0) {
6
+ return null;
7
+ }
8
+ const scores = usable.map(scoreFinding);
9
+ const score = clampScore(scores.reduce((total, item) => total + item, 0) / scores.length);
10
+ const reasons = usable.map((finding) => finding.reason);
11
+ return {
12
+ score,
13
+ weight: WEIGHT,
14
+ label: labelForScore(score),
15
+ reasons
16
+ };
17
+ }
18
+ function scoreFinding(finding) {
19
+ const base = finding.patternCount <= 1 ? 100 : finding.patternCount === 2 ? 70 : finding.patternCount === 3 ? 40 : 10;
20
+ const deviationPenalty = Math.min(20, finding.deviations.length * 5);
21
+ return clampScore(base - deviationPenalty);
22
+ }
23
+ //# sourceMappingURL=consistency-score.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"consistency-score.js","sourceRoot":"","sources":["../../src/scoring/consistency-score.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAE9D,MAAM,MAAM,GAAG,EAAE,CAAC;AAElB,MAAM,UAAU,gBAAgB,CAAC,QAAsC;IACrE,MAAM,MAAM,GAAG,CAAC,QAAQ,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,UAAU,KAAK,cAAc,CAAC,CAAC;IAE3F,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACxB,OAAO,IAAI,CAAC;IACd,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;IACxC,MAAM,KAAK,GAAG,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,CAAC,KAAK,GAAG,IAAI,EAAE,CAAC,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;IAC1F,MAAM,OAAO,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IAExD,OAAO;QACL,KAAK;QACL,MAAM,EAAE,MAAM;QACd,KAAK,EAAE,aAAa,CAAC,KAAK,CAAC;QAC3B,OAAO;KACR,CAAC;AACJ,CAAC;AAED,SAAS,YAAY,CAAC,OAAuB;IAC3C,MAAM,IAAI,GAAG,OAAO,CAAC,YAAY,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,OAAO,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IACtH,MAAM,gBAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,OAAO,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACrE,OAAO,UAAU,CAAC,IAAI,GAAG,gBAAgB,CAAC,CAAC;AAC7C,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { DuplicationSummary } from '../types/analysis.js';
2
+ import type { DimensionScore } from '../types/scoring.js';
3
+ export declare function scoreDuplication(duplication: DuplicationSummary): DimensionScore;
@@ -0,0 +1,16 @@
1
+ const WEIGHT = 20;
2
+ export function scoreDuplication(duplication) {
3
+ const percentage = duplication.duplicationPercentage;
4
+ const score = percentage <= 5 ? 100 : percentage <= 15 ? 70 : percentage <= 30 ? 40 : 10;
5
+ const reasons = [`${percentage.toFixed(1)}% duplication`];
6
+ if (duplication.isPartial) {
7
+ reasons.push('duplication data may be partial');
8
+ }
9
+ return {
10
+ score,
11
+ weight: WEIGHT,
12
+ label: score >= 80 ? 'healthy' : score >= 50 ? 'warning' : 'critical',
13
+ reasons
14
+ };
15
+ }
16
+ //# sourceMappingURL=duplication-score.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"duplication-score.js","sourceRoot":"","sources":["../../src/scoring/duplication-score.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,GAAG,EAAE,CAAC;AAElB,MAAM,UAAU,gBAAgB,CAAC,WAA+B;IAC9D,MAAM,UAAU,GAAG,WAAW,CAAC,qBAAqB,CAAC;IACrD,MAAM,KAAK,GAAG,UAAU,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IACzF,MAAM,OAAO,GAAG,CAAC,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC;IAE1D,IAAI,WAAW,CAAC,SAAS,EAAE,CAAC;QAC1B,OAAO,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;IAClD,CAAC;IAED,OAAO;QACL,KAAK;QACL,MAAM,EAAE,MAAM;QACd,KAAK,EAAE,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU;QACrE,OAAO;KACR,CAAC;AACJ,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { DimensionScore, HealthLabel, ScoreBreakdown } from '../types/scoring.js';
2
+ export declare function calculateHealthScore(modularity: DimensionScore, duplication: DimensionScore): ScoreBreakdown;
3
+ export declare function clampScore(score: number): number;
4
+ export declare function labelForScore(score: number): Exclude<HealthLabel, 'unavailable'>;
@@ -0,0 +1,20 @@
1
+ export function calculateHealthScore(modularity, duplication) {
2
+ const overall = clampScore(modularity.score * 0.5 + duplication.score * 0.5);
3
+ return {
4
+ modularity,
5
+ duplication,
6
+ overall,
7
+ label: labelForScore(overall)
8
+ };
9
+ }
10
+ export function clampScore(score) {
11
+ return Math.max(0, Math.min(100, Math.round(score)));
12
+ }
13
+ export function labelForScore(score) {
14
+ if (score >= 80)
15
+ return 'healthy';
16
+ if (score >= 50)
17
+ return 'warning';
18
+ return 'critical';
19
+ }
20
+ //# sourceMappingURL=health-score.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"health-score.js","sourceRoot":"","sources":["../../src/scoring/health-score.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,oBAAoB,CAAC,UAA0B,EAAE,WAA2B;IAC1F,MAAM,OAAO,GAAG,UAAU,CAAC,UAAU,CAAC,KAAK,GAAG,GAAG,GAAG,WAAW,CAAC,KAAK,GAAG,GAAG,CAAC,CAAC;IAC7E,OAAO;QACL,UAAU;QACV,WAAW;QACX,OAAO;QACP,KAAK,EAAE,aAAa,CAAC,OAAO,CAAC;KAC9B,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,KAAa;IACtC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;AACvD,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAa;IACzC,IAAI,KAAK,IAAI,EAAE;QAAE,OAAO,SAAS,CAAC;IAClC,IAAI,KAAK,IAAI,EAAE;QAAE,OAAO,SAAS,CAAC;IAClC,OAAO,UAAU,CAAC;AACpB,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { ScanResult } from '../types/analysis.js';
2
+ import type { ReportGuidance, ReportIssue } from '../types/issue.js';
3
+ export declare function buildIssues(result: ScanResult): ReportIssue[];
4
+ export declare function createReportGuidance(result: ScanResult): ReportGuidance;
@@ -0,0 +1,62 @@
1
+ const SEVERITY_ORDER = {
2
+ critical: 0,
3
+ warning: 1,
4
+ info: 2
5
+ };
6
+ export function buildIssues(result) {
7
+ const issues = [];
8
+ for (const file of result.files) {
9
+ if (file.isOversized) {
10
+ issues.push({
11
+ severity: 'critical',
12
+ category: 'modularity',
13
+ location: file.relativePath,
14
+ message: `${file.relativePath} is oversized at ${file.loc} LOC.`,
15
+ suggestion: 'Split this file around its dominant responsibilities before adding more features.'
16
+ });
17
+ }
18
+ for (const fn of file.functions.filter((item) => item.isFlagged)) {
19
+ issues.push({
20
+ severity: 'warning',
21
+ category: 'complexity',
22
+ location: `${file.relativePath} :: ${fn.name}`,
23
+ message: `${fn.name} has high complexity (${fn.complexity}).`,
24
+ suggestion: 'Extract smaller functions or simplify branching logic.'
25
+ });
26
+ }
27
+ }
28
+ if ((result.duplication.duplicationPercentage ?? 0) > 15) {
29
+ issues.push({
30
+ severity: result.duplication.duplicationPercentage > 30 ? 'critical' : 'warning',
31
+ category: 'duplication',
32
+ message: `${result.duplication.duplicationPercentage.toFixed(1)}% duplicated code detected.`,
33
+ suggestion: 'Consolidate duplicated blocks behind shared helpers or services.'
34
+ });
35
+ }
36
+ for (const hotspot of result.dependencyGraph.hotspots) {
37
+ issues.push({
38
+ severity: 'warning',
39
+ category: 'dependency',
40
+ location: hotspot.relativePath,
41
+ message: `${hotspot.relativePath} is depended on by ${hotspot.dependentCount} files.`,
42
+ suggestion: 'Review whether this module has too many responsibilities.'
43
+ });
44
+ }
45
+ return issues.sort(compareIssues);
46
+ }
47
+ export function createReportGuidance(result) {
48
+ const hasCritical = (result.issues ?? []).some((issue) => issue.severity === 'critical');
49
+ return {
50
+ message: hasCritical
51
+ ? 'Critical structural issues found. Generate a refactoring roadmap before adding features.'
52
+ : 'Use the health report to guide the next refactoring pass.',
53
+ command: undefined
54
+ };
55
+ }
56
+ function compareIssues(left, right) {
57
+ return (SEVERITY_ORDER[left.severity] - SEVERITY_ORDER[right.severity]
58
+ || left.category.localeCompare(right.category)
59
+ || (left.location ?? '').localeCompare(right.location ?? '')
60
+ || left.message.localeCompare(right.message));
61
+ }
62
+ //# sourceMappingURL=issue-builder.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"issue-builder.js","sourceRoot":"","sources":["../../src/scoring/issue-builder.ts"],"names":[],"mappings":"AAGA,MAAM,cAAc,GAA4C;IAC9D,QAAQ,EAAE,CAAC;IACX,OAAO,EAAE,CAAC;IACV,IAAI,EAAE,CAAC;CACR,CAAC;AAEF,MAAM,UAAU,WAAW,CAAC,MAAkB;IAC5C,MAAM,MAAM,GAAkB,EAAE,CAAC;IAEjC,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;QAChC,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,MAAM,CAAC,IAAI,CAAC;gBACV,QAAQ,EAAE,UAAU;gBACpB,QAAQ,EAAE,YAAY;gBACtB,QAAQ,EAAE,IAAI,CAAC,YAAY;gBAC3B,OAAO,EAAE,GAAG,IAAI,CAAC,YAAY,oBAAoB,IAAI,CAAC,GAAG,OAAO;gBAChE,UAAU,EAAE,mFAAmF;aAChG,CAAC,CAAC;QACL,CAAC;QAED,KAAK,MAAM,EAAE,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC;YACjE,MAAM,CAAC,IAAI,CAAC;gBACV,QAAQ,EAAE,SAAS;gBACnB,QAAQ,EAAE,YAAY;gBACtB,QAAQ,EAAE,GAAG,IAAI,CAAC,YAAY,OAAO,EAAE,CAAC,IAAI,EAAE;gBAC9C,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,yBAAyB,EAAE,CAAC,UAAU,IAAI;gBAC7D,UAAU,EAAE,wDAAwD;aACrE,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,qBAAqB,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC;QACzD,MAAM,CAAC,IAAI,CAAC;YACV,QAAQ,EAAE,MAAM,CAAC,WAAW,CAAC,qBAAqB,GAAG,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,SAAS;YAChF,QAAQ,EAAE,aAAa;YACvB,OAAO,EAAE,GAAG,MAAM,CAAC,WAAW,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC,CAAC,6BAA6B;YAC5F,UAAU,EAAE,kEAAkE;SAC/E,CAAC,CAAC;IACL,CAAC;IAED,KAAK,MAAM,OAAO,IAAI,MAAM,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC;QACtD,MAAM,CAAC,IAAI,CAAC;YACV,QAAQ,EAAE,SAAS;YACnB,QAAQ,EAAE,YAAY;YACtB,QAAQ,EAAE,OAAO,CAAC,YAAY;YAC9B,OAAO,EAAE,GAAG,OAAO,CAAC,YAAY,sBAAsB,OAAO,CAAC,cAAc,SAAS;YACrF,UAAU,EAAE,2DAA2D;SACxE,CAAC,CAAC;IACL,CAAC;IAED,OAAO,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;AACpC,CAAC;AAED,MAAM,UAAU,oBAAoB,CAAC,MAAkB;IACrD,MAAM,WAAW,GAAG,CAAC,MAAM,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,QAAQ,KAAK,UAAU,CAAC,CAAC;IAEzF,OAAO;QACL,OAAO,EAAE,WAAW;YAClB,CAAC,CAAC,0FAA0F;YAC5F,CAAC,CAAC,2DAA2D;QAC/D,OAAO,EAAE,SAAS;KACnB,CAAC;AACJ,CAAC;AAED,SAAS,aAAa,CAAC,IAAiB,EAAE,KAAkB;IAC1D,OAAO,CACL,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,cAAc,CAAC,KAAK,CAAC,QAAQ,CAAC;WAC3D,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,QAAQ,CAAC;WAC3C,CAAC,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC,CAAC,aAAa,CAAC,KAAK,CAAC,QAAQ,IAAI,EAAE,CAAC;WACzD,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,OAAO,CAAC,CAC7C,CAAC;AACJ,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { FileAnalysis } from '../types/analysis.js';
2
+ import type { DimensionScore } from '../types/scoring.js';
3
+ export declare function scoreModularity(files: FileAnalysis[]): DimensionScore;
@@ -0,0 +1,56 @@
1
+ const WEIGHT = 25;
2
+ export function scoreModularity(files) {
3
+ if (files.length === 0) {
4
+ return {
5
+ score: 100,
6
+ weight: WEIGHT,
7
+ label: 'healthy',
8
+ reasons: ['No analyzable files found']
9
+ };
10
+ }
11
+ const totalLoc = files.reduce((total, file) => total + file.loc, 0);
12
+ const oversizedFiles = files.filter((file) => file.loc > 300).length;
13
+ const functions = files.flatMap((file) => file.functions);
14
+ const oversizedFunctions = functions.filter((fn) => fn.loc > 50).length;
15
+ const averageFunctionsPerFile = functions.length / files.length;
16
+ const largestFileLoc = Math.max(...files.map((file) => file.loc));
17
+ const singleFileRatio = totalLoc === 0 ? 0 : largestFileLoc / totalLoc;
18
+ const reasons = [];
19
+ let score = 100;
20
+ if (oversizedFiles > 0) {
21
+ score -= Math.min(30, (oversizedFiles / files.length) * 40);
22
+ reasons.push(`${oversizedFiles} oversized file(s)`);
23
+ }
24
+ if (oversizedFunctions > 0) {
25
+ score -= Math.min(25, (oversizedFunctions / Math.max(functions.length, 1)) * 35);
26
+ reasons.push(`${oversizedFunctions} oversized function(s)`);
27
+ }
28
+ if (averageFunctionsPerFile > 8) {
29
+ score -= 15;
30
+ reasons.push('high average functions per file');
31
+ }
32
+ if (totalLoc > 100 && singleFileRatio > 0.5) {
33
+ score -= singleFileRatio > 0.8 ? 45 : 30;
34
+ reasons.push('single file holds most project code');
35
+ }
36
+ const finalScore = clampScore(score);
37
+ return {
38
+ score: finalScore,
39
+ weight: WEIGHT,
40
+ label: labelForScore(finalScore),
41
+ reasons: reasons.length > 0 ? reasons : ['Files and functions are well distributed']
42
+ };
43
+ }
44
+ function clampScore(score) {
45
+ return Math.max(0, Math.min(100, Math.round(score)));
46
+ }
47
+ function labelForScore(score) {
48
+ if (score >= 80) {
49
+ return 'healthy';
50
+ }
51
+ if (score >= 50) {
52
+ return 'warning';
53
+ }
54
+ return 'critical';
55
+ }
56
+ //# sourceMappingURL=modularity-score.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"modularity-score.js","sourceRoot":"","sources":["../../src/scoring/modularity-score.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { ConcernClassification } from '../types/concern.js';
2
+ import type { PatternFinding } from '../types/pattern.js';
3
+ export declare function analyzePatterns(classifications: ConcernClassification[] | undefined): PatternFinding[];
@@ -0,0 +1,74 @@
1
+ const MIN_EVIDENCE = 2;
2
+ export function analyzePatterns(classifications) {
3
+ if (!classifications || classifications.length === 0) {
4
+ return [];
5
+ }
6
+ const grouped = new Map();
7
+ for (const classification of classifications) {
8
+ for (const fn of classification.functions) {
9
+ if (fn.concern === 'unclassified') {
10
+ continue;
11
+ }
12
+ const items = grouped.get(fn.concern) ?? [];
13
+ items.push({
14
+ location: `${classification.file} :: ${fn.name}`,
15
+ pattern: inferPattern(classification.file, fn.name)
16
+ });
17
+ grouped.set(fn.concern, items);
18
+ }
19
+ }
20
+ return Array.from(grouped.entries()).map(([concern, items]) => createFinding(concern, items));
21
+ }
22
+ function createFinding(concern, items) {
23
+ if (items.length < MIN_EVIDENCE) {
24
+ return {
25
+ concern,
26
+ dominantPattern: null,
27
+ patternCount: 0,
28
+ deviations: [],
29
+ confidence: 'insufficient',
30
+ reason: 'Insufficient evidence to infer a dominant pattern'
31
+ };
32
+ }
33
+ const counts = new Map();
34
+ for (const item of items) {
35
+ counts.set(item.pattern, (counts.get(item.pattern) ?? 0) + 1);
36
+ }
37
+ const sorted = Array.from(counts.entries()).sort((left, right) => right[1] - left[1] || left[0].localeCompare(right[0]));
38
+ const dominantPattern = sorted[0]?.[0] ?? 'unknown';
39
+ const deviations = items
40
+ .filter((item) => item.pattern !== dominantPattern)
41
+ .map((item) => ({
42
+ location: item.location,
43
+ pattern: item.pattern,
44
+ expectedPattern: dominantPattern
45
+ }));
46
+ return {
47
+ concern,
48
+ dominantPattern,
49
+ patternCount: counts.size,
50
+ deviations,
51
+ confidence: counts.size === 1 ? 'high' : counts.size === 2 ? 'medium' : 'low',
52
+ reason: counts.size === 1 ? `Consistent ${dominantPattern} pattern` : `${counts.size} patterns detected for ${concern}`
53
+ };
54
+ }
55
+ function inferPattern(filePath, functionName) {
56
+ const normalized = `${filePath}/${functionName}`.toLowerCase();
57
+ if (normalized.includes('controller')) {
58
+ return 'controller';
59
+ }
60
+ if (normalized.includes('route')) {
61
+ return 'route-handler';
62
+ }
63
+ if (normalized.includes('service')) {
64
+ return 'service';
65
+ }
66
+ if (normalized.includes('model') || normalized.includes('repository') || normalized.includes('find') || normalized.includes('create')) {
67
+ return 'repository';
68
+ }
69
+ if (normalized.includes('schema') || normalized.includes('valid')) {
70
+ return 'schema-validation';
71
+ }
72
+ return 'inline';
73
+ }
74
+ //# sourceMappingURL=pattern-analysis.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pattern-analysis.js","sourceRoot":"","sources":["../../src/scoring/pattern-analysis.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { ConcernClassification } from '../types/concern.js';
2
+ import type { DimensionScore } from '../types/scoring.js';
3
+ export declare function scoreSeparation(classifications: ConcernClassification[] | undefined): DimensionScore | null;
@@ -0,0 +1,35 @@
1
+ import { clampScore, labelForScore } from './health-score.js';
2
+ const WEIGHT = 30;
3
+ export function scoreSeparation(classifications) {
4
+ if (!classifications || classifications.length === 0) {
5
+ return null;
6
+ }
7
+ let warningFiles = 0;
8
+ let criticalFiles = 0;
9
+ for (const classification of classifications) {
10
+ const concernCount = new Set(classification.functions.map((fn) => fn.concern).filter((concern) => concern !== 'unclassified')).size;
11
+ if (concernCount >= 3) {
12
+ criticalFiles += 1;
13
+ }
14
+ else if (concernCount === 2) {
15
+ warningFiles += 1;
16
+ }
17
+ }
18
+ const fileCount = Math.max(classifications.length, 1);
19
+ const penalty = (warningFiles / fileCount) * 35 + (criticalFiles / fileCount) * 70;
20
+ const score = clampScore(100 - penalty);
21
+ const reasons = [];
22
+ if (warningFiles > 0) {
23
+ reasons.push(`${warningFiles} file(s) contain two concerns`);
24
+ }
25
+ if (criticalFiles > 0) {
26
+ reasons.push(`${criticalFiles} file(s) contain three or more concerns`);
27
+ }
28
+ return {
29
+ score,
30
+ weight: WEIGHT,
31
+ label: labelForScore(score),
32
+ reasons: reasons.length > 0 ? reasons : ['Files keep concerns separated']
33
+ };
34
+ }
35
+ //# sourceMappingURL=separation-score.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"separation-score.js","sourceRoot":"","sources":["../../src/scoring/separation-score.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAE9D,MAAM,MAAM,GAAG,EAAE,CAAC;AAElB,MAAM,UAAU,eAAe,CAAC,eAAoD;IAClF,IAAI,CAAC,eAAe,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACrD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,IAAI,YAAY,GAAG,CAAC,CAAC;IACrB,IAAI,aAAa,GAAG,CAAC,CAAC;IAEtB,KAAK,MAAM,cAAc,IAAI,eAAe,EAAE,CAAC;QAC7C,MAAM,YAAY,GAAG,IAAI,GAAG,CAC1B,cAAc,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,KAAK,cAAc,CAAC,CACjG,CAAC,IAAI,CAAC;QAEP,IAAI,YAAY,IAAI,CAAC,EAAE,CAAC;YACtB,aAAa,IAAI,CAAC,CAAC;QACrB,CAAC;aAAM,IAAI,YAAY,KAAK,CAAC,EAAE,CAAC;YAC9B,YAAY,IAAI,CAAC,CAAC;QACpB,CAAC;IACH,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IACtD,MAAM,OAAO,GAAG,CAAC,YAAY,GAAG,SAAS,CAAC,GAAG,EAAE,GAAG,CAAC,aAAa,GAAG,SAAS,CAAC,GAAG,EAAE,CAAC;IACnF,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,GAAG,OAAO,CAAC,CAAC;IACxC,MAAM,OAAO,GAAa,EAAE,CAAC;IAE7B,IAAI,YAAY,GAAG,CAAC,EAAE,CAAC;QACrB,OAAO,CAAC,IAAI,CAAC,GAAG,YAAY,+BAA+B,CAAC,CAAC;IAC/D,CAAC;IAED,IAAI,aAAa,GAAG,CAAC,EAAE,CAAC;QACtB,OAAO,CAAC,IAAI,CAAC,GAAG,aAAa,yCAAyC,CAAC,CAAC;IAC1E,CAAC;IAED,OAAO;QACL,KAAK;QACL,MAAM,EAAE,MAAM;QACd,KAAK,EAAE,aAAa,CAAC,KAAK,CAAC;QAC3B,OAAO,EAAE,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,+BAA+B,CAAC;KAC1E,CAAC;AACJ,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { FileAnalysis } from '../types/analysis.js';
2
+ import type { ArchitectureSkill, ProjectCharacteristics, SkillMatch } from '../types/skill.js';
3
+ export declare function collectProjectCharacteristics(rootDir: string, filePaths: string[], analyses: FileAnalysis[]): Promise<ProjectCharacteristics>;
4
+ export declare function detectSkills(characteristics: ProjectCharacteristics, skills: ArchitectureSkill[]): SkillMatch[];