@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.
- package/bin/cli-hygiene.js +241 -0
- package/bin/dev/run-v2-torture.js +30 -0
- package/bin/guardrail.js +843 -0
- package/bin/runners/cli-utils.js +1070 -0
- package/bin/runners/context/ai-task-decomposer.js +337 -0
- package/bin/runners/context/analyzer.js +462 -0
- package/bin/runners/context/api-contracts.js +427 -0
- package/bin/runners/context/context-diff.js +342 -0
- package/bin/runners/context/context-pruner.js +291 -0
- package/bin/runners/context/dependency-graph.js +414 -0
- package/bin/runners/context/generators/claude.js +107 -0
- package/bin/runners/context/generators/codex.js +108 -0
- package/bin/runners/context/generators/copilot.js +119 -0
- package/bin/runners/context/generators/cursor.js +514 -0
- package/bin/runners/context/generators/mcp.js +151 -0
- package/bin/runners/context/generators/windsurf.js +180 -0
- package/bin/runners/context/git-context.js +302 -0
- package/bin/runners/context/index.js +1042 -0
- package/bin/runners/context/insights.js +173 -0
- package/bin/runners/context/mcp-server/generate-rules.js +337 -0
- package/bin/runners/context/mcp-server/index.js +1176 -0
- package/bin/runners/context/mcp-server/package.json +24 -0
- package/bin/runners/context/memory.js +200 -0
- package/bin/runners/context/monorepo.js +215 -0
- package/bin/runners/context/multi-repo-federation.js +404 -0
- package/bin/runners/context/patterns.js +253 -0
- package/bin/runners/context/proof-context.js +972 -0
- package/bin/runners/context/security-scanner.js +303 -0
- package/bin/runners/context/semantic-search.js +350 -0
- package/bin/runners/context/shared.js +264 -0
- package/bin/runners/context/team-conventions.js +310 -0
- package/bin/runners/lib/ai-bridge.js +416 -0
- package/bin/runners/lib/analysis-core.js +271 -0
- package/bin/runners/lib/analyzers.js +579 -0
- package/bin/runners/lib/assets/vibecheck-logo.png +0 -0
- package/bin/runners/lib/audit-bridge.js +391 -0
- package/bin/runners/lib/auth-truth.js +193 -0
- package/bin/runners/lib/auth.js +215 -0
- package/bin/runners/lib/backup.js +62 -0
- package/bin/runners/lib/billing.js +107 -0
- package/bin/runners/lib/claims.js +118 -0
- package/bin/runners/lib/cli-ui.js +540 -0
- package/bin/runners/lib/compliance-bridge-new.js +0 -0
- package/bin/runners/lib/compliance-bridge.js +165 -0
- package/bin/runners/lib/contracts/auth-contract.js +202 -0
- package/bin/runners/lib/contracts/env-contract.js +181 -0
- package/bin/runners/lib/contracts/external-contract.js +206 -0
- package/bin/runners/lib/contracts/guard.js +168 -0
- package/bin/runners/lib/contracts/index.js +89 -0
- package/bin/runners/lib/contracts/plan-validator.js +311 -0
- package/bin/runners/lib/contracts/route-contract.js +199 -0
- package/bin/runners/lib/contracts.js +804 -0
- package/bin/runners/lib/detect.js +89 -0
- package/bin/runners/lib/detectors-v2.js +703 -0
- package/bin/runners/lib/doctor/autofix.js +254 -0
- package/bin/runners/lib/doctor/index.js +37 -0
- package/bin/runners/lib/doctor/modules/dependencies.js +325 -0
- package/bin/runners/lib/doctor/modules/index.js +46 -0
- package/bin/runners/lib/doctor/modules/network.js +250 -0
- package/bin/runners/lib/doctor/modules/project.js +312 -0
- package/bin/runners/lib/doctor/modules/runtime.js +224 -0
- package/bin/runners/lib/doctor/modules/security.js +348 -0
- package/bin/runners/lib/doctor/modules/system.js +213 -0
- package/bin/runners/lib/doctor/modules/vibecheck.js +394 -0
- package/bin/runners/lib/doctor/reporter.js +262 -0
- package/bin/runners/lib/doctor/service.js +262 -0
- package/bin/runners/lib/doctor/types.js +113 -0
- package/bin/runners/lib/doctor/ui.js +263 -0
- package/bin/runners/lib/doctor-enhanced.js +233 -0
- package/bin/runners/lib/doctor-v2.js +608 -0
- package/bin/runners/lib/drift.js +425 -0
- package/bin/runners/lib/enforcement.js +72 -0
- package/bin/runners/lib/entitlements.js +8 -3
- package/bin/runners/lib/env-resolver.js +417 -0
- package/bin/runners/lib/extractors/client-calls.js +990 -0
- package/bin/runners/lib/extractors/fastify-route-dump.js +573 -0
- package/bin/runners/lib/extractors/fastify-routes.js +426 -0
- package/bin/runners/lib/extractors/index.js +363 -0
- package/bin/runners/lib/extractors/next-routes.js +524 -0
- package/bin/runners/lib/extractors/proof-graph.js +431 -0
- package/bin/runners/lib/extractors/route-matcher.js +451 -0
- package/bin/runners/lib/extractors/truthpack-v2.js +377 -0
- package/bin/runners/lib/extractors/ui-bindings.js +547 -0
- package/bin/runners/lib/findings-schema.js +281 -0
- package/bin/runners/lib/html-report.js +650 -0
- package/bin/runners/lib/missions/templates.js +45 -0
- package/bin/runners/lib/policy.js +295 -0
- package/bin/runners/lib/reality/correlation-detectors.js +359 -0
- package/bin/runners/lib/reality/index.js +318 -0
- package/bin/runners/lib/reality/request-hashing.js +416 -0
- package/bin/runners/lib/reality/request-mapper.js +453 -0
- package/bin/runners/lib/reality/safety-rails.js +463 -0
- package/bin/runners/lib/reality/semantic-snapshot.js +408 -0
- package/bin/runners/lib/reality/toast-detector.js +393 -0
- package/bin/runners/lib/route-truth.js +10 -10
- package/bin/runners/lib/schema-validator.js +350 -0
- package/bin/runners/lib/schemas/contracts.schema.json +160 -0
- package/bin/runners/lib/schemas/finding.schema.json +100 -0
- package/bin/runners/lib/schemas/mission-pack.schema.json +206 -0
- package/bin/runners/lib/schemas/proof-graph.schema.json +176 -0
- package/bin/runners/lib/schemas/reality-report.schema.json +162 -0
- package/bin/runners/lib/schemas/share-pack.schema.json +180 -0
- package/bin/runners/lib/schemas/ship-report.schema.json +117 -0
- package/bin/runners/lib/schemas/truthpack-v2.schema.json +303 -0
- package/bin/runners/lib/schemas/validator.js +438 -0
- package/bin/runners/lib/verdict-engine.js +628 -0
- package/bin/runners/runAIAgent.js +228 -1
- package/bin/runners/runBadge.js +181 -1
- package/bin/runners/runCtxDiff.js +301 -0
- package/bin/runners/runInitGha.js +78 -15
- package/bin/runners/runLaunch.js +180 -1
- package/bin/runners/runProve.js +23 -0
- package/bin/runners/runReplay.js +114 -84
- package/bin/runners/runScan.js +111 -32
- package/bin/runners/runShip.js +23 -2
- package/bin/runners/runTruthpack.js +9 -7
- package/bin/runners/runValidate.js +161 -1
- package/bin/vibecheck.js +6 -1
- 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 };
|