#!/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 { readFileSync, existsSync, mkdirSync } from 'fs'; import { dirname, resolve } from 'path'; import { fileURLToPath } from 'url'; const __dir = dirname(fileURLToPath(import.meta.url)); const cfg = p => { const path = resolve(__dir, p); if (!existsSync(path)) { console.error(`āŒ Missing: ${p}`); process.exit(1); } return JSON.parse(readFileSync(path, 'utf8')); }; async function main() { console.log('šŸ› ļø claw-apply setup\n'); // Check configs console.log('Checking config files...'); const settings = cfg('config/settings.json'); const profile = cfg('config/profile.json'); const searchConfig = cfg('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'); // 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('\nTo run manually:'); console.log(' node job_searcher.mjs — search now'); console.log(' node job_applier.mjs — apply now'); console.log('\nCron schedules (register via OpenClaw):'); console.log(` Search: ${settings.schedules?.search}`); console.log(` Apply: ${settings.schedules?.apply}`); } main().catch(e => { console.error('Setup error:', e.message); process.exit(1); });