#!/usr/bin/env node /** * setup.mjs — claw-apply setup wizard * Verifies config, tests logins, registers cron jobs * Run once after configuring: node setup.mjs */ import { existsSync, mkdirSync, writeFileSync, chmodSync } from 'fs'; import { dirname, resolve } from 'path'; import { fileURLToPath } from 'url'; import { loadConfig } from './lib/queue.mjs'; import { loadEnv } from './lib/env.mjs'; loadEnv(); const __dir = dirname(fileURLToPath(import.meta.url)); async function main() { console.log('šŸ› ļø claw-apply setup\n'); // Check configs console.log('Checking config files...'); const settings = await loadConfig(resolve(__dir, 'config/settings.json')); const profile = await loadConfig(resolve(__dir, 'config/profile.json')); const searchConfig = await loadConfig(resolve(__dir, 'config/search_config.json')); const checks = [ [profile.name?.first && profile.name?.last, 'profile.json: name'], [profile.email && profile.email !== 'jane@example.com', 'profile.json: email'], [profile.phone, 'profile.json: phone'], [profile.resume_path && existsSync(profile.resume_path), 'profile.json: resume_path (file must exist)'], [settings.notifications?.telegram_user_id !== 'YOUR_TELEGRAM_USER_ID', 'settings.json: telegram_user_id'], [settings.notifications?.bot_token !== 'YOUR_TELEGRAM_BOT_TOKEN', 'settings.json: bot_token'], [settings.kernel?.proxy_id !== 'YOUR_KERNEL_PROXY_ID', 'settings.json: kernel.proxy_id'], [searchConfig.searches?.length > 0, 'search_config.json: at least one search'], ]; let ok = true; for (const [pass, label] of checks) { console.log(` ${pass ? 'āœ…' : 'āŒ'} ${label}`); if (!pass) ok = false; } if (!ok) { console.log('\nāš ļø Fix the above before continuing.\n'); process.exit(1); } // Create data directory mkdirSync(resolve(__dir, 'data'), { recursive: true }); console.log('\nāœ… Data directory ready'); // Write .env file with API keys (gitignored — never embedded in cron payloads) const envPath = resolve(__dir, '.env'); const envLines = []; if (process.env.KERNEL_API_KEY) envLines.push(`KERNEL_API_KEY=${process.env.KERNEL_API_KEY}`); if (process.env.ANTHROPIC_API_KEY) envLines.push(`ANTHROPIC_API_KEY=${process.env.ANTHROPIC_API_KEY}`); if (envLines.length) { writeFileSync(envPath, envLines.join('\n') + '\n', { mode: 0o600 }); console.log('āœ… API keys written to .env (mode 600, gitignored)'); } // Test Telegram if (settings.notifications?.bot_token && settings.notifications?.telegram_user_id) { const { sendTelegram } = await import('./lib/notify.mjs'); await sendTelegram(settings, 'šŸ¤– *claw-apply setup complete!* Job search and auto-apply is ready to run.'); console.log('āœ… Telegram test message sent'); } // Test LinkedIn login console.log('\nTesting LinkedIn login...'); const { createBrowser } = await import('./lib/browser.mjs'); const { verifyLogin: liLogin } = await import('./lib/linkedin.mjs'); let liBrowser; try { liBrowser = await createBrowser(settings, 'linkedin'); const loggedIn = await liLogin(liBrowser.page); console.log(loggedIn ? 'āœ… LinkedIn login OK' : 'āŒ LinkedIn not logged in — check Kernel Managed Auth'); } catch (e) { console.log(`āŒ LinkedIn browser error: ${e.message}`); } finally { await liBrowser?.browser?.close().catch(() => {}); } // Test Wellfound login console.log('\nTesting Wellfound login...'); const { verifyLogin: wfLogin } = await import('./lib/wellfound.mjs'); let wfBrowser; try { wfBrowser = await createBrowser(settings, 'wellfound'); const loggedIn = await wfLogin(wfBrowser.page); console.log(loggedIn ? 'āœ… Wellfound login OK' : 'āš ļø Wellfound login unconfirmed'); } catch (e) { console.log(`āŒ Wellfound browser error: ${e.message}`); } finally { await wfBrowser?.browser?.close().catch(() => {}); } console.log('\nšŸŽ‰ Setup complete. claw-apply is ready.'); console.log('\nScheduling is managed by OpenClaw crons. Run manually:'); console.log(' node job_searcher.mjs — search now'); console.log(' node job_filter.mjs — filter/score jobs'); console.log(' node job_applier.mjs — apply now'); console.log(' node status.mjs — queue + run status'); } main().catch(e => { console.error('Setup error:', e.message); process.exit(1); });