- filter.mjs: loadProfile now async, uses loadJSON - telegram_answers.mjs: answers read/write through storage layer - status.mjs: uses initQueue + loadQueue for S3 support - setup.mjs: await all loadConfig calls - storage.mjs: more robust getS3Key using URL parsing Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
106 lines
4.3 KiB
JavaScript
106 lines
4.3 KiB
JavaScript
#!/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); });
|