@vibecheckai/cli 3.0.3 → 3.0.5

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 (119) hide show
  1. package/bin/cli-hygiene.js +241 -0
  2. package/bin/dev/run-v2-torture.js +30 -0
  3. package/bin/guardrail.js +843 -0
  4. package/bin/runners/cli-utils.js +1070 -0
  5. package/bin/runners/context/ai-task-decomposer.js +337 -0
  6. package/bin/runners/context/analyzer.js +462 -0
  7. package/bin/runners/context/api-contracts.js +427 -0
  8. package/bin/runners/context/context-diff.js +342 -0
  9. package/bin/runners/context/context-pruner.js +291 -0
  10. package/bin/runners/context/dependency-graph.js +414 -0
  11. package/bin/runners/context/generators/claude.js +107 -0
  12. package/bin/runners/context/generators/codex.js +108 -0
  13. package/bin/runners/context/generators/copilot.js +119 -0
  14. package/bin/runners/context/generators/cursor.js +514 -0
  15. package/bin/runners/context/generators/mcp.js +151 -0
  16. package/bin/runners/context/generators/windsurf.js +180 -0
  17. package/bin/runners/context/git-context.js +302 -0
  18. package/bin/runners/context/index.js +1042 -0
  19. package/bin/runners/context/insights.js +173 -0
  20. package/bin/runners/context/mcp-server/generate-rules.js +337 -0
  21. package/bin/runners/context/mcp-server/index.js +1176 -0
  22. package/bin/runners/context/mcp-server/package.json +24 -0
  23. package/bin/runners/context/memory.js +200 -0
  24. package/bin/runners/context/monorepo.js +215 -0
  25. package/bin/runners/context/multi-repo-federation.js +404 -0
  26. package/bin/runners/context/patterns.js +253 -0
  27. package/bin/runners/context/proof-context.js +972 -0
  28. package/bin/runners/context/security-scanner.js +303 -0
  29. package/bin/runners/context/semantic-search.js +350 -0
  30. package/bin/runners/context/shared.js +264 -0
  31. package/bin/runners/context/team-conventions.js +310 -0
  32. package/bin/runners/lib/ai-bridge.js +416 -0
  33. package/bin/runners/lib/analysis-core.js +271 -0
  34. package/bin/runners/lib/analyzers.js +579 -0
  35. package/bin/runners/lib/assets/vibecheck-logo.png +0 -0
  36. package/bin/runners/lib/audit-bridge.js +391 -0
  37. package/bin/runners/lib/auth-truth.js +193 -0
  38. package/bin/runners/lib/auth.js +215 -0
  39. package/bin/runners/lib/backup.js +62 -0
  40. package/bin/runners/lib/billing.js +107 -0
  41. package/bin/runners/lib/claims.js +118 -0
  42. package/bin/runners/lib/cli-ui.js +540 -0
  43. package/bin/runners/lib/compliance-bridge-new.js +0 -0
  44. package/bin/runners/lib/compliance-bridge.js +165 -0
  45. package/bin/runners/lib/contracts/auth-contract.js +202 -0
  46. package/bin/runners/lib/contracts/env-contract.js +181 -0
  47. package/bin/runners/lib/contracts/external-contract.js +206 -0
  48. package/bin/runners/lib/contracts/guard.js +168 -0
  49. package/bin/runners/lib/contracts/index.js +89 -0
  50. package/bin/runners/lib/contracts/plan-validator.js +311 -0
  51. package/bin/runners/lib/contracts/route-contract.js +199 -0
  52. package/bin/runners/lib/contracts.js +804 -0
  53. package/bin/runners/lib/detect.js +89 -0
  54. package/bin/runners/lib/detectors-v2.js +703 -0
  55. package/bin/runners/lib/doctor/autofix.js +254 -0
  56. package/bin/runners/lib/doctor/index.js +37 -0
  57. package/bin/runners/lib/doctor/modules/dependencies.js +325 -0
  58. package/bin/runners/lib/doctor/modules/index.js +46 -0
  59. package/bin/runners/lib/doctor/modules/network.js +250 -0
  60. package/bin/runners/lib/doctor/modules/project.js +312 -0
  61. package/bin/runners/lib/doctor/modules/runtime.js +224 -0
  62. package/bin/runners/lib/doctor/modules/security.js +348 -0
  63. package/bin/runners/lib/doctor/modules/system.js +213 -0
  64. package/bin/runners/lib/doctor/modules/vibecheck.js +394 -0
  65. package/bin/runners/lib/doctor/reporter.js +262 -0
  66. package/bin/runners/lib/doctor/service.js +262 -0
  67. package/bin/runners/lib/doctor/types.js +113 -0
  68. package/bin/runners/lib/doctor/ui.js +263 -0
  69. package/bin/runners/lib/doctor-enhanced.js +233 -0
  70. package/bin/runners/lib/doctor-v2.js +608 -0
  71. package/bin/runners/lib/drift.js +425 -0
  72. package/bin/runners/lib/enforcement.js +72 -0
  73. package/bin/runners/lib/entitlements.js +8 -3
  74. package/bin/runners/lib/env-resolver.js +417 -0
  75. package/bin/runners/lib/extractors/client-calls.js +990 -0
  76. package/bin/runners/lib/extractors/fastify-route-dump.js +573 -0
  77. package/bin/runners/lib/extractors/fastify-routes.js +426 -0
  78. package/bin/runners/lib/extractors/index.js +363 -0
  79. package/bin/runners/lib/extractors/next-routes.js +524 -0
  80. package/bin/runners/lib/extractors/proof-graph.js +431 -0
  81. package/bin/runners/lib/extractors/route-matcher.js +451 -0
  82. package/bin/runners/lib/extractors/truthpack-v2.js +377 -0
  83. package/bin/runners/lib/extractors/ui-bindings.js +547 -0
  84. package/bin/runners/lib/findings-schema.js +281 -0
  85. package/bin/runners/lib/html-report.js +650 -0
  86. package/bin/runners/lib/missions/templates.js +45 -0
  87. package/bin/runners/lib/policy.js +295 -0
  88. package/bin/runners/lib/reality/correlation-detectors.js +359 -0
  89. package/bin/runners/lib/reality/index.js +318 -0
  90. package/bin/runners/lib/reality/request-hashing.js +416 -0
  91. package/bin/runners/lib/reality/request-mapper.js +453 -0
  92. package/bin/runners/lib/reality/safety-rails.js +463 -0
  93. package/bin/runners/lib/reality/semantic-snapshot.js +408 -0
  94. package/bin/runners/lib/reality/toast-detector.js +393 -0
  95. package/bin/runners/lib/route-truth.js +10 -10
  96. package/bin/runners/lib/schema-validator.js +350 -0
  97. package/bin/runners/lib/schemas/contracts.schema.json +160 -0
  98. package/bin/runners/lib/schemas/finding.schema.json +100 -0
  99. package/bin/runners/lib/schemas/mission-pack.schema.json +206 -0
  100. package/bin/runners/lib/schemas/proof-graph.schema.json +176 -0
  101. package/bin/runners/lib/schemas/reality-report.schema.json +162 -0
  102. package/bin/runners/lib/schemas/share-pack.schema.json +180 -0
  103. package/bin/runners/lib/schemas/ship-report.schema.json +117 -0
  104. package/bin/runners/lib/schemas/truthpack-v2.schema.json +303 -0
  105. package/bin/runners/lib/schemas/validator.js +438 -0
  106. package/bin/runners/lib/verdict-engine.js +628 -0
  107. package/bin/runners/runAIAgent.js +228 -1
  108. package/bin/runners/runBadge.js +181 -1
  109. package/bin/runners/runCtxDiff.js +301 -0
  110. package/bin/runners/runInitGha.js +78 -15
  111. package/bin/runners/runLaunch.js +180 -1
  112. package/bin/runners/runProve.js +23 -0
  113. package/bin/runners/runReplay.js +114 -84
  114. package/bin/runners/runScan.js +111 -32
  115. package/bin/runners/runShip.js +23 -2
  116. package/bin/runners/runTruthpack.js +9 -7
  117. package/bin/runners/runValidate.js +161 -1
  118. package/bin/vibecheck.js +6 -1
  119. package/package.json +1 -1
@@ -0,0 +1,263 @@
1
+ /**
2
+ * Doctor CLI UI Components
3
+ *
4
+ * Beautiful terminal output for the Doctor service
5
+ */
6
+
7
+ const { SEVERITY, CATEGORY_META } = require('./types');
8
+
9
+ // ANSI color codes
10
+ const c = {
11
+ reset: '\x1b[0m',
12
+ bold: '\x1b[1m',
13
+ dim: '\x1b[2m',
14
+ italic: '\x1b[3m',
15
+ underline: '\x1b[4m',
16
+
17
+ black: '\x1b[30m',
18
+ red: '\x1b[31m',
19
+ green: '\x1b[32m',
20
+ yellow: '\x1b[33m',
21
+ blue: '\x1b[34m',
22
+ magenta: '\x1b[35m',
23
+ cyan: '\x1b[36m',
24
+ white: '\x1b[37m',
25
+
26
+ bgRed: '\x1b[41m',
27
+ bgGreen: '\x1b[42m',
28
+ bgYellow: '\x1b[43m',
29
+ bgBlue: '\x1b[44m',
30
+ bgMagenta: '\x1b[45m',
31
+ bgCyan: '\x1b[46m',
32
+ };
33
+
34
+ const SEVERITY_CONFIG = {
35
+ [SEVERITY.PASS]: { icon: '✓', color: c.green, label: 'PASS' },
36
+ [SEVERITY.INFO]: { icon: 'ℹ', color: c.blue, label: 'INFO' },
37
+ [SEVERITY.WARNING]: { icon: '!', color: c.yellow, label: 'WARN' },
38
+ [SEVERITY.ERROR]: { icon: '✗', color: c.red, label: 'ERROR' },
39
+ [SEVERITY.CRITICAL]: { icon: '⛔', color: `${c.bgRed}${c.white}`, label: 'CRIT' },
40
+ };
41
+
42
+ const BOX_CHARS = {
43
+ topLeft: '╔',
44
+ topRight: '╗',
45
+ bottomLeft: '╚',
46
+ bottomRight: '╝',
47
+ horizontal: '═',
48
+ vertical: '║',
49
+ lightHorizontal: '─',
50
+ };
51
+
52
+ function box(width = 74) {
53
+ return {
54
+ top: `${c.cyan}${BOX_CHARS.topLeft}${BOX_CHARS.horizontal.repeat(width)}${BOX_CHARS.topRight}${c.reset}`,
55
+ bottom: `${c.cyan}${BOX_CHARS.bottomLeft}${BOX_CHARS.horizontal.repeat(width)}${BOX_CHARS.bottomRight}${c.reset}`,
56
+ line: (content, align = 'left') => {
57
+ const stripped = stripAnsi(content);
58
+ const padding = width - stripped.length;
59
+ const left = align === 'center' ? Math.floor(padding / 2) : 0;
60
+ const right = padding - left;
61
+ return `${c.cyan}${BOX_CHARS.vertical}${c.reset}${' '.repeat(left)}${content}${' '.repeat(right)}${c.cyan}${BOX_CHARS.vertical}${c.reset}`;
62
+ },
63
+ empty: `${c.cyan}${BOX_CHARS.vertical}${c.reset}${' '.repeat(width)}${c.cyan}${BOX_CHARS.vertical}${c.reset}`,
64
+ };
65
+ }
66
+
67
+ function stripAnsi(str) {
68
+ return str.replace(/\x1b\[[0-9;]*m/g, '');
69
+ }
70
+
71
+ function printHeader() {
72
+ const b = box();
73
+ console.log('');
74
+ console.log(b.top);
75
+ console.log(b.empty);
76
+ console.log(b.line(`${c.bold}🩺 VIBECHECK DOCTOR${c.reset}`, 'center'));
77
+ console.log(b.line(`${c.dim}Enterprise Environment Diagnostics${c.reset}`, 'center'));
78
+ console.log(b.empty);
79
+ console.log(b.bottom);
80
+ console.log('');
81
+ }
82
+
83
+ function printCategoryHeader(category) {
84
+ const meta = CATEGORY_META[category] || { name: category, icon: '📋' };
85
+ console.log(`${c.cyan}${meta.icon} ${c.bold}${meta.name}${c.reset}`);
86
+ console.log(`${c.dim}${BOX_CHARS.lightHorizontal.repeat(50)}${c.reset}`);
87
+ }
88
+
89
+ function printDiagnostic(diagnostic, options = {}) {
90
+ const { showDetail = true, showFixes = true, indent = ' ' } = options;
91
+ const config = SEVERITY_CONFIG[diagnostic.severity] || SEVERITY_CONFIG[SEVERITY.INFO];
92
+
93
+ // Main line
94
+ console.log(`${indent}${config.color}${config.icon}${c.reset} ${diagnostic.message}`);
95
+
96
+ // Detail
97
+ if (showDetail && diagnostic.detail) {
98
+ console.log(`${indent} ${c.dim}${diagnostic.detail}${c.reset}`);
99
+ }
100
+
101
+ // First fix hint
102
+ if (showFixes && diagnostic.fixes && diagnostic.fixes.length > 0) {
103
+ const fix = diagnostic.fixes[0];
104
+ if (fix.command) {
105
+ console.log(`${indent} ${c.cyan}→ ${fix.command}${c.reset}`);
106
+ } else if (fix.url) {
107
+ console.log(`${indent} ${c.cyan}→ ${fix.url}${c.reset}`);
108
+ } else if (fix.description) {
109
+ console.log(`${indent} ${c.cyan}→ ${fix.description}${c.reset}`);
110
+ }
111
+ }
112
+ }
113
+
114
+ function printProgress(name, total, current) {
115
+ const percent = Math.round((current / total) * 100);
116
+ const barWidth = 20;
117
+ const filled = Math.round((current / total) * barWidth);
118
+ const bar = '█'.repeat(filled) + '░'.repeat(barWidth - filled);
119
+
120
+ process.stdout.write(`\r${c.dim}${name}${c.reset} [${c.cyan}${bar}${c.reset}] ${percent}%`);
121
+ }
122
+
123
+ function clearProgress() {
124
+ process.stdout.write('\r' + ' '.repeat(60) + '\r');
125
+ }
126
+
127
+ function printSpinner(message, frame = 0) {
128
+ const frames = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
129
+ process.stdout.write(`\r${c.cyan}${frames[frame % frames.length]}${c.reset} ${message}`);
130
+ return (frame + 1) % frames.length;
131
+ }
132
+
133
+ function printIssuesSummary(diagnostics) {
134
+ const critical = diagnostics.filter(d => d.severity === SEVERITY.CRITICAL);
135
+ const errors = diagnostics.filter(d => d.severity === SEVERITY.ERROR);
136
+ const warnings = diagnostics.filter(d => d.severity === SEVERITY.WARNING);
137
+
138
+ if (critical.length === 0 && errors.length === 0 && warnings.length === 0) {
139
+ return;
140
+ }
141
+
142
+ const b = box();
143
+ console.log(b.top);
144
+ console.log(b.line(`${c.bold}ISSUES FOUND${c.reset}`));
145
+ console.log(b.bottom);
146
+ console.log('');
147
+
148
+ if (critical.length > 0) {
149
+ console.log(`${c.bgRed}${c.white}${c.bold} CRITICAL (${critical.length}) ${c.reset} — Must fix immediately\n`);
150
+ for (const d of critical) {
151
+ console.log(` ${c.red}●${c.reset} ${c.bold}${d.name}${c.reset}`);
152
+ console.log(` ${c.dim}${d.message}${c.reset}`);
153
+ if (d.fixes?.[0]?.command) {
154
+ console.log(` ${c.cyan}→ ${d.fixes[0].command}${c.reset}`);
155
+ }
156
+ console.log('');
157
+ }
158
+ }
159
+
160
+ if (errors.length > 0) {
161
+ console.log(`${c.red}${c.bold}❌ ERRORS (${errors.length})${c.reset} — Must fix before using vibecheck\n`);
162
+ for (const d of errors) {
163
+ console.log(` ${c.red}●${c.reset} ${c.bold}${d.name}${c.reset}`);
164
+ console.log(` ${c.dim}${d.message}${c.reset}`);
165
+ if (d.fixes?.[0]?.command) {
166
+ console.log(` ${c.cyan}→ ${d.fixes[0].command}${c.reset}`);
167
+ }
168
+ console.log('');
169
+ }
170
+ }
171
+
172
+ if (warnings.length > 0) {
173
+ console.log(`${c.yellow}${c.bold}⚠️ WARNINGS (${warnings.length})${c.reset} — Recommended fixes\n`);
174
+ for (const d of warnings) {
175
+ console.log(` ${c.yellow}●${c.reset} ${c.bold}${d.name}${c.reset}`);
176
+ console.log(` ${c.dim}${d.message}${c.reset}`);
177
+ if (d.fixes?.[0]?.command) {
178
+ console.log(` ${c.cyan}→ ${d.fixes[0].command}${c.reset}`);
179
+ }
180
+ console.log('');
181
+ }
182
+ }
183
+ }
184
+
185
+ function printFinalSummary(summary, durationMs) {
186
+ const b = box();
187
+
188
+ const healthColor = summary.healthScore >= 80 ? c.green
189
+ : summary.healthScore >= 60 ? c.yellow
190
+ : c.red;
191
+ const healthIcon = summary.healthScore >= 80 ? '💚'
192
+ : summary.healthScore >= 60 ? '💛'
193
+ : '❤️';
194
+
195
+ const verdictColor = {
196
+ HEALTHY: c.green,
197
+ DEGRADED: c.yellow,
198
+ UNHEALTHY: c.red,
199
+ CRITICAL: `${c.bgRed}${c.white}`,
200
+ }[summary.verdict] || c.white;
201
+
202
+ console.log(b.top);
203
+ console.log(b.empty);
204
+ console.log(b.line(`${healthIcon} ${c.bold}Health Score: ${healthColor}${summary.healthScore}%${c.reset}`));
205
+ console.log(b.line(` ${c.bold}Verdict: ${verdictColor}${summary.verdict}${c.reset}`));
206
+ console.log(b.empty);
207
+ console.log(b.line(`${c.green}✓ ${summary.passed} passed${c.reset} ${c.blue}ℹ ${summary.info} info${c.reset} ${c.yellow}! ${summary.warnings} warnings${c.reset} ${c.red}✗ ${summary.errors} errors${c.reset} ${summary.critical > 0 ? `${c.bgRed}${c.white}⛔ ${summary.critical}${c.reset}` : ''}`));
208
+ console.log(b.line(`${c.dim}Completed ${summary.total} checks in ${(durationMs / 1000).toFixed(1)}s${c.reset}`));
209
+ console.log(b.empty);
210
+ console.log(b.bottom);
211
+ console.log('');
212
+
213
+ // Final message
214
+ if (summary.critical > 0) {
215
+ console.log(`${c.bgRed}${c.white}${c.bold} ⛔ CRITICAL ISSUES DETECTED ${c.reset}`);
216
+ console.log(`${c.red}Fix critical issues immediately before proceeding.${c.reset}\n`);
217
+ } else if (summary.errors > 0) {
218
+ console.log(`${c.red}${c.bold}⚠ Fix the errors above before running vibecheck.${c.reset}\n`);
219
+ } else if (summary.warnings > 0) {
220
+ console.log(`${c.yellow}Your environment is ready, but consider fixing the warnings.${c.reset}\n`);
221
+ } else {
222
+ console.log(`${c.green}${c.bold}✓ Your environment is perfectly configured!${c.reset}\n`);
223
+ }
224
+ }
225
+
226
+ function printAutoFixResults(results) {
227
+ console.log('');
228
+ console.log(`${c.cyan}${c.bold}🔧 Auto-Fix Results${c.reset}`);
229
+ console.log(`${c.dim}${BOX_CHARS.lightHorizontal.repeat(50)}${c.reset}`);
230
+
231
+ console.log(` Attempted: ${results.attempted}`);
232
+ console.log(` ${c.green}Succeeded: ${results.succeeded}${c.reset}`);
233
+ console.log(` ${c.red}Failed: ${results.failed}${c.reset}`);
234
+ console.log(` ${c.dim}Skipped: ${results.skipped}${c.reset}`);
235
+ console.log('');
236
+
237
+ for (const r of results.results) {
238
+ const icon = r.status === 'success' ? `${c.green}✓${c.reset}`
239
+ : r.status === 'failed' ? `${c.red}✗${c.reset}`
240
+ : `${c.dim}○${c.reset}`;
241
+ console.log(` ${icon} ${r.diagnosticName}: ${r.status}`);
242
+ if (r.error) {
243
+ console.log(` ${c.dim}${r.error}${c.reset}`);
244
+ }
245
+ }
246
+ console.log('');
247
+ }
248
+
249
+ module.exports = {
250
+ c,
251
+ SEVERITY_CONFIG,
252
+ box,
253
+ stripAnsi,
254
+ printHeader,
255
+ printCategoryHeader,
256
+ printDiagnostic,
257
+ printProgress,
258
+ clearProgress,
259
+ printSpinner,
260
+ printIssuesSummary,
261
+ printFinalSummary,
262
+ printAutoFixResults,
263
+ };
@@ -0,0 +1,233 @@
1
+ /**
2
+ * Enhanced Doctor Command
3
+ *
4
+ * Self-diagnosing CLI that catches 90% of setup issues.
5
+ * Goal: kill support tickets by being brutally helpful.
6
+ */
7
+
8
+ const fs = require('fs');
9
+ const path = require('path');
10
+ const { execSync } = require('child_process');
11
+
12
+ const c = {
13
+ reset: '\x1b[0m',
14
+ bold: '\x1b[1m',
15
+ dim: '\x1b[2m',
16
+ red: '\x1b[31m',
17
+ green: '\x1b[32m',
18
+ yellow: '\x1b[33m',
19
+ cyan: '\x1b[36m',
20
+ };
21
+
22
+ class DoctorEnhanced {
23
+ constructor(projectPath) {
24
+ this.projectPath = projectPath;
25
+ this.issues = [];
26
+ this.fixes = [];
27
+ }
28
+
29
+ async diagnose() {
30
+ console.log('\n🔍 vibecheck Doctor\n');
31
+ console.log('Checking your environment...\n');
32
+
33
+ await this.checkNodeVersion();
34
+ await this.checkPackageManager();
35
+ await this.checkRequiredBinaries();
36
+ await this.checkEnvVars();
37
+ await this.checkPermissions();
38
+ await this.checkProjectStructure();
39
+ await this.checkCanBuild();
40
+ await this.checkCanRun();
41
+
42
+ this.printReport();
43
+ return this.issues.length === 0 ? 0 : 1;
44
+ }
45
+
46
+ async checkNodeVersion() {
47
+ try {
48
+ const version = process.version;
49
+ const major = parseInt(version.slice(1).split('.')[0]);
50
+
51
+ if (major < 18) {
52
+ this.issue('Node version too old', `You're running Node ${version}. vibecheck requires Node 18+.`, `nvm install 18 && nvm use 18`);
53
+ } else {
54
+ this.pass(`Node version: ${version}`);
55
+ }
56
+ } catch {
57
+ this.issue('Cannot detect Node version', 'Node.js may not be installed.', 'Install Node.js from nodejs.org');
58
+ }
59
+ }
60
+
61
+ async checkPackageManager() {
62
+ const managers = ['pnpm', 'npm', 'yarn'];
63
+ let found = null;
64
+
65
+ for (const mgr of managers) {
66
+ try {
67
+ execSync(`${mgr} --version`, { stdio: 'ignore' });
68
+ found = mgr;
69
+ break;
70
+ } catch {
71
+ // Not found
72
+ }
73
+ }
74
+
75
+ if (!found) {
76
+ this.issue('No package manager found', 'Install pnpm, npm, or yarn.', 'npm install -g pnpm');
77
+ } else {
78
+ this.pass(`Package manager: ${found}`);
79
+ }
80
+ }
81
+
82
+ async checkRequiredBinaries() {
83
+ const binaries = [
84
+ { name: 'git', required: true },
85
+ { name: 'playwright', required: false, check: 'npx playwright --version' },
86
+ ];
87
+
88
+ for (const bin of binaries) {
89
+ try {
90
+ if (bin.check) {
91
+ execSync(bin.check, { stdio: 'ignore' });
92
+ } else {
93
+ execSync(`which ${bin.name}`, { stdio: 'ignore' });
94
+ }
95
+ this.pass(`${bin.name} found`);
96
+ } catch {
97
+ if (bin.required) {
98
+ this.issue(`${bin.name} not found`, `${bin.name} is required.`, `Install ${bin.name}`);
99
+ } else {
100
+ this.warn(`${bin.name} not found`, `Optional: ${bin.name} enables runtime verification.`, `npx playwright install`);
101
+ }
102
+ }
103
+ }
104
+ }
105
+
106
+ async checkEnvVars() {
107
+ const required = process.env.VIBECHECK_API_KEY ? [] : ['VIBECHECK_API_KEY (optional)'];
108
+ const sensitive = ['API_KEY', 'SECRET', 'TOKEN', 'PASSWORD'];
109
+
110
+ // Check for common missing env vars
111
+ const envFile = path.join(this.projectPath, '.env');
112
+ if (!fs.existsSync(envFile)) {
113
+ this.warn('No .env file', 'Consider creating .env for local configuration.', 'touch .env');
114
+ }
115
+
116
+ // Check for exposed secrets (but never print them)
117
+ const envExample = path.join(this.projectPath, '.env.example');
118
+ if (fs.existsSync(envExample)) {
119
+ const content = fs.readFileSync(envExample, 'utf8');
120
+ const hasSecrets = sensitive.some(s => content.includes(s));
121
+ if (hasSecrets) {
122
+ this.pass('.env.example found');
123
+ }
124
+ }
125
+ }
126
+
127
+ async checkPermissions() {
128
+ try {
129
+ const testFile = path.join(this.projectPath, '.vibecheck', '.test-write');
130
+ fs.mkdirSync(path.dirname(testFile), { recursive: true });
131
+ fs.writeFileSync(testFile, 'test');
132
+ fs.unlinkSync(testFile);
133
+ this.pass('Write permissions OK');
134
+ } catch {
135
+ this.issue('No write permissions', 'Cannot write to .vibecheck directory.', `chmod -R u+w ${this.projectPath}`);
136
+ }
137
+ }
138
+
139
+ async checkProjectStructure() {
140
+ const packageJson = path.join(this.projectPath, 'package.json');
141
+ if (!fs.existsSync(packageJson)) {
142
+ this.issue('No package.json', 'This doesn\'t look like a Node.js project.', 'Run vibecheck init');
143
+ return;
144
+ }
145
+
146
+ this.pass('package.json found');
147
+
148
+ // Check for common project structures
149
+ const hasSrc = fs.existsSync(path.join(this.projectPath, 'src')) ||
150
+ fs.existsSync(path.join(this.projectPath, 'app')) ||
151
+ fs.existsSync(path.join(this.projectPath, 'pages'));
152
+
153
+ if (!hasSrc) {
154
+ this.warn('No source directory found', 'vibecheck works best with standard project structures.', 'Consider organizing code in src/ or app/');
155
+ }
156
+ }
157
+
158
+ async checkCanBuild() {
159
+ try {
160
+ const packageJson = JSON.parse(fs.readFileSync(path.join(this.projectPath, 'package.json'), 'utf8'));
161
+ if (packageJson.scripts?.build) {
162
+ try {
163
+ execSync('npm run build', { cwd: this.projectPath, stdio: 'ignore', timeout: 10000 });
164
+ this.pass('Project builds successfully');
165
+ } catch {
166
+ this.warn('Build failed', 'Project may have build errors.', 'Run: npm run build');
167
+ }
168
+ }
169
+ } catch {
170
+ // Can't check build
171
+ }
172
+ }
173
+
174
+ async checkCanRun() {
175
+ // Minimal runtime check would go here
176
+ // For now, just pass
177
+ this.pass('Runtime check skipped (use --runtime for full check)');
178
+ }
179
+
180
+ issue(title, description, fix) {
181
+ this.issues.push({ type: 'error', title, description, fix });
182
+ }
183
+
184
+ warn(title, description, fix) {
185
+ this.issues.push({ type: 'warning', title, description, fix });
186
+ }
187
+
188
+ pass(message) {
189
+ // Silent pass - only show issues
190
+ }
191
+
192
+ printReport() {
193
+ if (this.issues.length === 0) {
194
+ console.log(`${c.green}${c.bold}✓ All checks passed!${c.reset}\n`);
195
+ return;
196
+ }
197
+
198
+ console.log('\n' + '═'.repeat(70));
199
+ console.log(`${c.bold}DIAGNOSIS REPORT${c.reset}`);
200
+ console.log('═'.repeat(70) + '\n');
201
+
202
+ const errors = this.issues.filter(i => i.type === 'error');
203
+ const warnings = this.issues.filter(i => i.type === 'warning');
204
+
205
+ if (errors.length > 0) {
206
+ console.log(`${c.red}${c.bold}❌ ERRORS (${errors.length}):${c.reset}\n`);
207
+ for (const issue of errors) {
208
+ console.log(` ${c.bold}${issue.title}${c.reset}`);
209
+ console.log(` ${c.dim}${issue.description}${c.reset}`);
210
+ if (issue.fix) {
211
+ console.log(` ${c.cyan}→ Fix: ${issue.fix}${c.reset}`);
212
+ }
213
+ console.log('');
214
+ }
215
+ }
216
+
217
+ if (warnings.length > 0) {
218
+ console.log(`${c.yellow}${c.bold}⚠️ WARNINGS (${warnings.length}):${c.reset}\n`);
219
+ for (const issue of warnings) {
220
+ console.log(` ${c.bold}${issue.title}${c.reset}`);
221
+ console.log(` ${c.dim}${issue.description}${c.reset}`);
222
+ if (issue.fix) {
223
+ console.log(` ${c.cyan}→ Fix: ${issue.fix}${c.reset}`);
224
+ }
225
+ console.log('');
226
+ }
227
+ }
228
+
229
+ console.log('═'.repeat(70) + '\n');
230
+ }
231
+ }
232
+
233
+ module.exports = { DoctorEnhanced };