@vibecheckai/cli 3.4.0 → 3.5.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 (166) hide show
  1. package/bin/registry.js +243 -152
  2. package/bin/runners/cli-utils.js +2 -33
  3. package/bin/runners/context/generators/cursor.js +49 -2
  4. package/bin/runners/lib/agent-firewall/learning/learning-engine.js +849 -0
  5. package/bin/runners/lib/analyzers.js +544 -19
  6. package/bin/runners/lib/audit-logger.js +532 -0
  7. package/bin/runners/lib/authority/authorities/architecture.js +364 -0
  8. package/bin/runners/lib/authority/authorities/compliance.js +341 -0
  9. package/bin/runners/lib/authority/authorities/human.js +343 -0
  10. package/bin/runners/lib/authority/authorities/quality.js +420 -0
  11. package/bin/runners/lib/authority/authorities/security.js +228 -0
  12. package/bin/runners/lib/authority/index.js +293 -0
  13. package/bin/runners/lib/authority-badge.js +425 -425
  14. package/bin/runners/lib/bundle/bundle-intelligence.js +846 -0
  15. package/bin/runners/lib/cli-charts.js +368 -0
  16. package/bin/runners/lib/cli-config-display.js +405 -0
  17. package/bin/runners/lib/cli-demo.js +275 -0
  18. package/bin/runners/lib/cli-errors.js +438 -0
  19. package/bin/runners/lib/cli-help-formatter.js +439 -0
  20. package/bin/runners/lib/cli-interactive-menu.js +509 -0
  21. package/bin/runners/lib/cli-prompts.js +441 -0
  22. package/bin/runners/lib/cli-scan-cards.js +362 -0
  23. package/bin/runners/lib/compliance-reporter.js +710 -0
  24. package/bin/runners/lib/conductor/index.js +671 -0
  25. package/bin/runners/lib/easy/README.md +123 -0
  26. package/bin/runners/lib/easy/index.js +140 -0
  27. package/bin/runners/lib/easy/interactive-wizard.js +788 -0
  28. package/bin/runners/lib/easy/one-click-firewall.js +564 -0
  29. package/bin/runners/lib/easy/zero-config-reality.js +714 -0
  30. package/bin/runners/lib/engines/accessibility-engine.js +218 -18
  31. package/bin/runners/lib/engines/api-consistency-engine.js +335 -30
  32. package/bin/runners/lib/engines/async-patterns-engine.js +444 -0
  33. package/bin/runners/lib/engines/bundle-size-engine.js +433 -0
  34. package/bin/runners/lib/engines/confidence-scoring.js +276 -0
  35. package/bin/runners/lib/engines/context-detection.js +264 -0
  36. package/bin/runners/lib/engines/cross-file-analysis-engine.js +292 -27
  37. package/bin/runners/lib/engines/database-patterns-engine.js +429 -0
  38. package/bin/runners/lib/engines/duplicate-code-engine.js +354 -0
  39. package/bin/runners/lib/engines/empty-catch-engine.js +127 -17
  40. package/bin/runners/lib/engines/env-variables-engine.js +458 -0
  41. package/bin/runners/lib/engines/error-handling-engine.js +437 -0
  42. package/bin/runners/lib/engines/false-positive-prevention.js +630 -0
  43. package/bin/runners/lib/engines/framework-adapters/index.js +607 -0
  44. package/bin/runners/lib/engines/framework-detection.js +508 -0
  45. package/bin/runners/lib/engines/import-order-engine.js +429 -0
  46. package/bin/runners/lib/engines/mock-data-engine.js +53 -10
  47. package/bin/runners/lib/engines/naming-conventions-engine.js +544 -0
  48. package/bin/runners/lib/engines/noise-reduction-engine.js +452 -0
  49. package/bin/runners/lib/engines/orchestrator.js +334 -0
  50. package/bin/runners/lib/engines/performance-issues-engine.js +176 -36
  51. package/bin/runners/lib/engines/react-patterns-engine.js +457 -0
  52. package/bin/runners/lib/engines/security-vulnerabilities-engine.js +382 -54
  53. package/bin/runners/lib/engines/type-aware-engine.js +263 -39
  54. package/bin/runners/lib/engines/vibecheck-engines/index.js +122 -13
  55. package/bin/runners/lib/engines/vibecheck-engines/lib/ai-hallucination-engine.js +806 -0
  56. package/bin/runners/lib/engines/vibecheck-engines/lib/hardcoded-secrets-engine.js +373 -73
  57. package/bin/runners/lib/engines/vibecheck-engines/lib/smart-fix-engine.js +577 -0
  58. package/bin/runners/lib/engines/vibecheck-engines/lib/vibe-score-engine.js +543 -0
  59. package/bin/runners/lib/engines/vibecheck-engines.js +514 -0
  60. package/bin/runners/lib/enhanced-features/index.js +305 -0
  61. package/bin/runners/lib/enhanced-output.js +631 -0
  62. package/bin/runners/lib/enterprise.js +300 -0
  63. package/bin/runners/lib/entitlements-v2.js +103 -11
  64. package/bin/runners/lib/firewall/command-validator.js +351 -0
  65. package/bin/runners/lib/firewall/config.js +341 -0
  66. package/bin/runners/lib/firewall/content-validator.js +519 -0
  67. package/bin/runners/lib/firewall/index.js +101 -0
  68. package/bin/runners/lib/firewall/path-validator.js +256 -0
  69. package/bin/runners/lib/html-proof-report.js +350 -700
  70. package/bin/runners/lib/intelligence/cross-repo-intelligence.js +817 -0
  71. package/bin/runners/lib/mcp-utils.js +425 -0
  72. package/bin/runners/lib/missions/plan.js +46 -6
  73. package/bin/runners/lib/missions/templates.js +232 -0
  74. package/bin/runners/lib/output/index.js +1022 -0
  75. package/bin/runners/lib/policy-engine.js +652 -0
  76. package/bin/runners/lib/polish/autofix/accessibility-fixes.js +333 -0
  77. package/bin/runners/lib/polish/autofix/async-handlers.js +273 -0
  78. package/bin/runners/lib/polish/autofix/dead-code.js +280 -0
  79. package/bin/runners/lib/polish/autofix/imports-optimizer.js +344 -0
  80. package/bin/runners/lib/polish/autofix/index.js +200 -0
  81. package/bin/runners/lib/polish/autofix/remove-consoles.js +209 -0
  82. package/bin/runners/lib/polish/autofix/strengthen-types.js +245 -0
  83. package/bin/runners/lib/polish/backend-checks.js +148 -0
  84. package/bin/runners/lib/polish/documentation-checks.js +111 -0
  85. package/bin/runners/lib/polish/frontend-checks.js +168 -0
  86. package/bin/runners/lib/polish/index.js +71 -0
  87. package/bin/runners/lib/polish/infrastructure-checks.js +131 -0
  88. package/bin/runners/lib/polish/library-detection.js +175 -0
  89. package/bin/runners/lib/polish/performance-checks.js +100 -0
  90. package/bin/runners/lib/polish/security-checks.js +148 -0
  91. package/bin/runners/lib/polish/utils.js +203 -0
  92. package/bin/runners/lib/prompt-builder.js +540 -0
  93. package/bin/runners/lib/proof-certificate.js +634 -0
  94. package/bin/runners/lib/reality/accessibility-audit.js +946 -0
  95. package/bin/runners/lib/reality/api-contract-validator.js +1012 -0
  96. package/bin/runners/lib/reality/chaos-engineering.js +1084 -0
  97. package/bin/runners/lib/reality/performance-tracker.js +1077 -0
  98. package/bin/runners/lib/reality/scenario-generator.js +1404 -0
  99. package/bin/runners/lib/reality/visual-regression.js +852 -0
  100. package/bin/runners/lib/reality-profiler.js +717 -0
  101. package/bin/runners/lib/replay/flight-recorder-viewer.js +1160 -0
  102. package/bin/runners/lib/review/ai-code-review.js +832 -0
  103. package/bin/runners/lib/rules/custom-rule-engine.js +985 -0
  104. package/bin/runners/lib/sbom-generator.js +641 -0
  105. package/bin/runners/lib/scan-output-enhanced.js +512 -0
  106. package/bin/runners/lib/scan-output.js +47 -0
  107. package/bin/runners/lib/security/owasp-scanner.js +939 -0
  108. package/bin/runners/lib/terminal-ui.js +113 -1
  109. package/bin/runners/lib/unified-cli-output.js +603 -430
  110. package/bin/runners/lib/validators/contract-validator.js +283 -0
  111. package/bin/runners/lib/validators/dead-export-detector.js +279 -0
  112. package/bin/runners/lib/validators/dep-audit.js +245 -0
  113. package/bin/runners/lib/validators/env-validator.js +319 -0
  114. package/bin/runners/lib/validators/index.js +120 -0
  115. package/bin/runners/lib/validators/license-checker.js +252 -0
  116. package/bin/runners/lib/validators/route-validator.js +290 -0
  117. package/bin/runners/runAIAgent.js +5 -10
  118. package/bin/runners/runAgent.js +3 -0
  119. package/bin/runners/runApprove.js +1233 -1200
  120. package/bin/runners/runAuth.js +22 -1
  121. package/bin/runners/runAuthority.js +528 -0
  122. package/bin/runners/runCheckpoint.js +4 -24
  123. package/bin/runners/runClassify.js +862 -859
  124. package/bin/runners/runConductor.js +772 -0
  125. package/bin/runners/runContainer.js +366 -0
  126. package/bin/runners/runContext.js +3 -0
  127. package/bin/runners/runDoctor.js +28 -41
  128. package/bin/runners/runEasy.js +410 -0
  129. package/bin/runners/runFirewall.js +3 -0
  130. package/bin/runners/runFirewallHook.js +3 -0
  131. package/bin/runners/runFix.js +76 -66
  132. package/bin/runners/runGuard.js +411 -18
  133. package/bin/runners/runIaC.js +372 -0
  134. package/bin/runners/runInit.js +10 -60
  135. package/bin/runners/runMcp.js +11 -12
  136. package/bin/runners/runPolish.js +240 -64
  137. package/bin/runners/runPromptFirewall.js +5 -12
  138. package/bin/runners/runProve.js +20 -55
  139. package/bin/runners/runReality.js +68 -59
  140. package/bin/runners/runReport.js +31 -5
  141. package/bin/runners/runRuntime.js +5 -8
  142. package/bin/runners/runScan.js +194 -1286
  143. package/bin/runners/runShip.js +695 -47
  144. package/bin/runners/runTruth.js +3 -0
  145. package/bin/runners/runValidate.js +7 -11
  146. package/bin/runners/runVibe.js +791 -0
  147. package/bin/runners/runWatch.js +14 -23
  148. package/bin/vibecheck.js +175 -56
  149. package/mcp-server/index.js +190 -14
  150. package/mcp-server/package.json +1 -1
  151. package/mcp-server/tools-v3.js +397 -64
  152. package/mcp-server/tools.js +495 -0
  153. package/package.json +1 -1
  154. package/bin/runners/lib/engines/vibecheck-engines/lib/ast-cache.js +0 -164
  155. package/bin/runners/lib/engines/vibecheck-engines/lib/code-quality-engine.js +0 -291
  156. package/bin/runners/lib/engines/vibecheck-engines/lib/console-logs-engine.js +0 -83
  157. package/bin/runners/lib/engines/vibecheck-engines/lib/dead-code-engine.js +0 -198
  158. package/bin/runners/lib/engines/vibecheck-engines/lib/deprecated-api-engine.js +0 -275
  159. package/bin/runners/lib/engines/vibecheck-engines/lib/empty-catch-engine.js +0 -167
  160. package/bin/runners/lib/engines/vibecheck-engines/lib/file-filter.js +0 -217
  161. package/bin/runners/lib/engines/vibecheck-engines/lib/mock-data-engine.js +0 -140
  162. package/bin/runners/lib/engines/vibecheck-engines/lib/parallel-processor.js +0 -164
  163. package/bin/runners/lib/engines/vibecheck-engines/lib/performance-issues-engine.js +0 -234
  164. package/bin/runners/lib/engines/vibecheck-engines/lib/type-aware-engine.js +0 -217
  165. package/bin/runners/lib/engines/vibecheck-engines/lib/unsafe-regex-engine.js +0 -78
  166. package/mcp-server/index-v1.js +0 -698
@@ -0,0 +1,368 @@
1
+ /**
2
+ * CLI Charts - ASCII charts and visual data display for terminal output
3
+ *
4
+ * Provides:
5
+ * - Horizontal bar charts
6
+ * - Vertical bar charts (sparklines)
7
+ * - Trend indicators
8
+ * - Distribution displays
9
+ * - Comparison tables
10
+ */
11
+
12
+ "use strict";
13
+
14
+ const {
15
+ ansi,
16
+ sym,
17
+ box,
18
+ visibleLength,
19
+ padRight,
20
+ padLeft,
21
+ center,
22
+ SUPPORTS_UNICODE,
23
+ } = require("./unified-cli-output");
24
+
25
+ // ═══════════════════════════════════════════════════════════════════════════════
26
+ // CONFIGURATION
27
+ // ═══════════════════════════════════════════════════════════════════════════════
28
+
29
+ const CHART_WIDTH = 40;
30
+ const SPARKLINE_CHARS = SUPPORTS_UNICODE
31
+ ? ['▁', '▂', '▃', '▄', '▅', '▆', '▇', '█']
32
+ : ['_', '.', '-', '=', '#'];
33
+
34
+ // ═══════════════════════════════════════════════════════════════════════════════
35
+ // HORIZONTAL BAR CHART
36
+ // ═══════════════════════════════════════════════════════════════════════════════
37
+
38
+ /**
39
+ * Render a horizontal bar chart
40
+ * @param {Array<{label: string, value: number, color?: string}>} data
41
+ * @param {Object} options
42
+ */
43
+ function horizontalBarChart(data, options = {}) {
44
+ const {
45
+ width = CHART_WIDTH,
46
+ showValues = true,
47
+ showPercent = false,
48
+ title = null,
49
+ maxValue = null,
50
+ } = options;
51
+
52
+ const lines = [];
53
+ const max = maxValue || Math.max(...data.map(d => d.value), 1);
54
+ const maxLabelLen = Math.max(...data.map(d => d.label.length), 8);
55
+
56
+ if (title) {
57
+ lines.push(` ${ansi.bold}${title}${ansi.reset}`);
58
+ lines.push(` ${ansi.gray}${box.horizontal.repeat(maxLabelLen + width + 10)}${ansi.reset}`);
59
+ }
60
+
61
+ for (const item of data) {
62
+ const barLen = Math.round((item.value / max) * width);
63
+ const bar = sym.filled.repeat(barLen) + ansi.gray + sym.empty.repeat(width - barLen) + ansi.reset;
64
+ const color = item.color || ansi.cyan;
65
+ const label = padRight(item.label, maxLabelLen);
66
+
67
+ let valueStr = '';
68
+ if (showValues) {
69
+ valueStr = ` ${ansi.bold}${item.value}${ansi.reset}`;
70
+ if (showPercent) {
71
+ const pct = Math.round((item.value / max) * 100);
72
+ valueStr += ` ${ansi.gray}(${pct}%)${ansi.reset}`;
73
+ }
74
+ }
75
+
76
+ lines.push(` ${ansi.gray}${label}${ansi.reset} ${color}${bar}${ansi.reset}${valueStr}`);
77
+ }
78
+
79
+ return lines.join('\n');
80
+ }
81
+
82
+ // ═══════════════════════════════════════════════════════════════════════════════
83
+ // SPARKLINE
84
+ // ═══════════════════════════════════════════════════════════════════════════════
85
+
86
+ /**
87
+ * Render a sparkline (mini inline chart)
88
+ * @param {number[]} values - Array of values
89
+ * @param {Object} options
90
+ */
91
+ function sparkline(values, options = {}) {
92
+ const {
93
+ min = Math.min(...values),
94
+ max = Math.max(...values),
95
+ color = ansi.cyan,
96
+ } = options;
97
+
98
+ if (values.length === 0) return '';
99
+
100
+ const range = max - min || 1;
101
+ const chars = values.map(v => {
102
+ const normalized = (v - min) / range;
103
+ const idx = Math.min(Math.floor(normalized * SPARKLINE_CHARS.length), SPARKLINE_CHARS.length - 1);
104
+ return SPARKLINE_CHARS[idx];
105
+ });
106
+
107
+ return `${color}${chars.join('')}${ansi.reset}`;
108
+ }
109
+
110
+ // ═══════════════════════════════════════════════════════════════════════════════
111
+ // TREND INDICATOR
112
+ // ═══════════════════════════════════════════════════════════════════════════════
113
+
114
+ /**
115
+ * Render a trend indicator (up/down/stable)
116
+ * @param {number} current - Current value
117
+ * @param {number} previous - Previous value
118
+ * @param {Object} options
119
+ */
120
+ function trendIndicator(current, previous, options = {}) {
121
+ const { reverse = false, showDelta = true } = options;
122
+
123
+ const diff = current - previous;
124
+ const pct = previous !== 0 ? Math.round((diff / previous) * 100) : 0;
125
+
126
+ let icon, color;
127
+ if (diff > 0) {
128
+ icon = SUPPORTS_UNICODE ? '↑' : '^';
129
+ color = reverse ? ansi.red : ansi.green;
130
+ } else if (diff < 0) {
131
+ icon = SUPPORTS_UNICODE ? '↓' : 'v';
132
+ color = reverse ? ansi.green : ansi.red;
133
+ } else {
134
+ icon = SUPPORTS_UNICODE ? '→' : '-';
135
+ color = ansi.gray;
136
+ }
137
+
138
+ let result = `${color}${icon}${ansi.reset}`;
139
+ if (showDelta && diff !== 0) {
140
+ const sign = diff > 0 ? '+' : '';
141
+ result += ` ${color}${sign}${pct}%${ansi.reset}`;
142
+ }
143
+
144
+ return result;
145
+ }
146
+
147
+ // ═══════════════════════════════════════════════════════════════════════════════
148
+ // DISTRIBUTION DISPLAY
149
+ // ═══════════════════════════════════════════════════════════════════════════════
150
+
151
+ /**
152
+ * Render a distribution display (like severity breakdown)
153
+ * @param {Object} counts - {label: count} object
154
+ * @param {Object} options
155
+ */
156
+ function distributionDisplay(counts, options = {}) {
157
+ const {
158
+ colors = {},
159
+ icons = {},
160
+ total = null,
161
+ title = null,
162
+ } = options;
163
+
164
+ const lines = [];
165
+ const sum = total || Object.values(counts).reduce((a, b) => a + b, 0);
166
+
167
+ if (title) {
168
+ lines.push(` ${ansi.bold}${title}${ansi.reset}`);
169
+ }
170
+
171
+ const entries = Object.entries(counts).filter(([_, v]) => v > 0);
172
+
173
+ if (entries.length === 0) {
174
+ lines.push(` ${ansi.green}${sym.check}${ansi.reset} None found`);
175
+ return lines.join('\n');
176
+ }
177
+
178
+ for (const [label, count] of entries) {
179
+ const pct = Math.round((count / sum) * 100);
180
+ const color = colors[label] || ansi.gray;
181
+ const icon = icons[label] || sym.bullet;
182
+
183
+ // Visual bar
184
+ const barWidth = 20;
185
+ const filledLen = Math.round((count / sum) * barWidth);
186
+ const bar = `${color}${sym.filled.repeat(filledLen)}${ansi.gray}${sym.empty.repeat(barWidth - filledLen)}${ansi.reset}`;
187
+
188
+ lines.push(` ${color}${icon}${ansi.reset} ${padRight(label, 12)} ${bar} ${ansi.bold}${count}${ansi.reset} ${ansi.gray}(${pct}%)${ansi.reset}`);
189
+ }
190
+
191
+ return lines.join('\n');
192
+ }
193
+
194
+ // ═══════════════════════════════════════════════════════════════════════════════
195
+ // SEVERITY BREAKDOWN
196
+ // ═══════════════════════════════════════════════════════════════════════════════
197
+
198
+ /**
199
+ * Render severity breakdown (specialized distribution)
200
+ * @param {Object} counts - {critical, high, medium, low, info}
201
+ */
202
+ function severityBreakdown(counts, options = {}) {
203
+ const severityConfig = {
204
+ critical: { color: ansi.red, icon: sym.critical, order: 0 },
205
+ high: { color: ansi.yellow, icon: sym.high, order: 1 },
206
+ medium: { color: ansi.yellow, icon: sym.medium, order: 2 },
207
+ low: { color: ansi.blue, icon: sym.low, order: 3 },
208
+ info: { color: ansi.gray, icon: sym.info, order: 4 },
209
+ };
210
+
211
+ const colors = {};
212
+ const icons = {};
213
+
214
+ for (const [sev, config] of Object.entries(severityConfig)) {
215
+ colors[sev] = config.color;
216
+ icons[sev] = config.icon;
217
+ }
218
+
219
+ return distributionDisplay(counts, {
220
+ colors,
221
+ icons,
222
+ title: options.title || 'Issues by Severity',
223
+ ...options,
224
+ });
225
+ }
226
+
227
+ // ═══════════════════════════════════════════════════════════════════════════════
228
+ // COMPARISON TABLE
229
+ // ═══════════════════════════════════════════════════════════════════════════════
230
+
231
+ /**
232
+ * Render a comparison table (before/after, current/previous)
233
+ * @param {Array<{label: string, before: number, after: number}>} data
234
+ */
235
+ function comparisonTable(data, options = {}) {
236
+ const {
237
+ beforeLabel = 'Before',
238
+ afterLabel = 'After',
239
+ title = null,
240
+ showTrend = true,
241
+ } = options;
242
+
243
+ const lines = [];
244
+ const maxLabelLen = Math.max(...data.map(d => d.label.length), 10);
245
+
246
+ if (title) {
247
+ lines.push(` ${ansi.bold}${title}${ansi.reset}`);
248
+ }
249
+
250
+ // Header
251
+ lines.push(` ${padRight('', maxLabelLen)} ${padLeft(beforeLabel, 10)} ${padLeft(afterLabel, 10)} ${showTrend ? 'Change' : ''}`);
252
+ lines.push(` ${ansi.gray}${box.horizontal.repeat(maxLabelLen + 35)}${ansi.reset}`);
253
+
254
+ for (const item of data) {
255
+ const label = padRight(item.label, maxLabelLen);
256
+ const before = padLeft(String(item.before), 10);
257
+ const after = padLeft(String(item.after), 10);
258
+ const trend = showTrend ? trendIndicator(item.after, item.before, { reverse: item.reverse }) : '';
259
+
260
+ lines.push(` ${label} ${ansi.gray}${before}${ansi.reset} ${ansi.bold}${after}${ansi.reset} ${trend}`);
261
+ }
262
+
263
+ return lines.join('\n');
264
+ }
265
+
266
+ // ═══════════════════════════════════════════════════════════════════════════════
267
+ // STATS SUMMARY
268
+ // ═══════════════════════════════════════════════════════════════════════════════
269
+
270
+ /**
271
+ * Render a compact stats summary
272
+ * @param {Array<{label: string, value: string|number, icon?: string, color?: string}>} stats
273
+ */
274
+ function statsSummary(stats, options = {}) {
275
+ const { separator = ' │ ', inline = true } = options;
276
+
277
+ const items = stats.map(s => {
278
+ const icon = s.icon ? `${s.icon} ` : '';
279
+ const color = s.color || ansi.reset;
280
+ return `${ansi.gray}${s.label}:${ansi.reset} ${color}${ansi.bold}${s.value}${ansi.reset}`;
281
+ });
282
+
283
+ if (inline) {
284
+ return ` ${items.join(`${ansi.gray}${separator}${ansi.reset}`)}`;
285
+ }
286
+
287
+ return items.map(item => ` ${item}`).join('\n');
288
+ }
289
+
290
+ // ═══════════════════════════════════════════════════════════════════════════════
291
+ // HEALTH GAUGE
292
+ // ═══════════════════════════════════════════════════════════════════════════════
293
+
294
+ /**
295
+ * Render a health gauge (circular-style progress)
296
+ * @param {number} score - Score 0-100
297
+ */
298
+ function healthGauge(score, options = {}) {
299
+ const { width = 30, showLabel = true } = options;
300
+
301
+ const clamped = Math.max(0, Math.min(100, score));
302
+ const filled = Math.round((clamped / 100) * width);
303
+
304
+ let color;
305
+ if (clamped >= 80) color = ansi.green;
306
+ else if (clamped >= 60) color = ansi.yellow;
307
+ else if (clamped >= 40) color = ansi.yellow;
308
+ else color = ansi.red;
309
+
310
+ const bar = `${color}${sym.filled.repeat(filled)}${ansi.gray}${sym.empty.repeat(width - filled)}${ansi.reset}`;
311
+ const scoreStr = `${color}${ansi.bold}${clamped}${ansi.reset}${ansi.gray}/100${ansi.reset}`;
312
+
313
+ const lines = [];
314
+ if (showLabel) {
315
+ lines.push(` ${ansi.gray}Health Score${ansi.reset}`);
316
+ }
317
+ lines.push(` [${bar}] ${scoreStr}`);
318
+
319
+ return lines.join('\n');
320
+ }
321
+
322
+ // ═══════════════════════════════════════════════════════════════════════════════
323
+ // ACTIVITY HEATMAP (simplified)
324
+ // ═══════════════════════════════════════════════════════════════════════════════
325
+
326
+ /**
327
+ * Render a simplified activity heatmap (like GitHub contribution graph)
328
+ * @param {number[]} values - Array of activity values (7 days)
329
+ */
330
+ function activityHeatmap(values, options = {}) {
331
+ const { labels = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'] } = options;
332
+
333
+ const max = Math.max(...values, 1);
334
+ const chars = SUPPORTS_UNICODE
335
+ ? ['░', '▒', '▓', '█']
336
+ : ['.', ':', '#', '@'];
337
+
338
+ const lines = [];
339
+ lines.push(` ${ansi.gray}Activity${ansi.reset}`);
340
+
341
+ const blocks = values.map((v, i) => {
342
+ const normalized = v / max;
343
+ const idx = Math.min(Math.floor(normalized * chars.length), chars.length - 1);
344
+ const color = normalized > 0.75 ? ansi.green : normalized > 0.25 ? ansi.cyan : ansi.gray;
345
+ return `${color}${chars[idx]}${ansi.reset}`;
346
+ });
347
+
348
+ lines.push(` ${blocks.join(' ')}`);
349
+ lines.push(` ${ansi.gray}${labels.map(l => l[0]).join(' ')}${ansi.reset}`);
350
+
351
+ return lines.join('\n');
352
+ }
353
+
354
+ // ═══════════════════════════════════════════════════════════════════════════════
355
+ // EXPORTS
356
+ // ═══════════════════════════════════════════════════════════════════════════════
357
+
358
+ module.exports = {
359
+ horizontalBarChart,
360
+ sparkline,
361
+ trendIndicator,
362
+ distributionDisplay,
363
+ severityBreakdown,
364
+ comparisonTable,
365
+ statsSummary,
366
+ healthGauge,
367
+ activityHeatmap,
368
+ };