Files
claw-apply/lib/session.mjs
Matthew Jackson b1528ac0ad refactor: extract magic numbers to constants, fix audit issues
- Centralize all magic numbers/strings in lib/constants.mjs
- Fix double-replaced import names in filter.mjs
- Consolidate duplicate fs imports in job_applier/job_searcher
- Remove empty JSDoc block in job_searcher
- Update keywords.mjs model from claude-3-haiku to claude-haiku-4-5
- Extract Anthropic API URLs to constants
- Convert :has-text() selectors to page.locator() API
- Fix SIGTERM handler conflict — move partial-run notification into lock.onShutdown
- Remove unused exports (LOCAL_USER_AGENT, DEFAULT_REVIEW_WINDOW_MINUTES)
- Fix variable shadowing (b -> v) in job_filter reduce callback
- Replace SKILL.md PM2 references with system cron

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 08:45:17 -08:00

68 lines
2.5 KiB
JavaScript

/**
* session.mjs — Kernel Managed Auth session refresh
* Call refreshSession() before creating a browser to ensure the profile is fresh
*/
import { createRequire } from 'module';
import {
KERNEL_SDK_PATH, SESSION_REFRESH_POLL_TIMEOUT, SESSION_REFRESH_POLL_WAIT,
SESSION_LOGIN_VERIFY_WAIT
} from './constants.mjs';
const require = createRequire(import.meta.url);
export async function refreshSession(platform, apiKey, connectionIds = {}) {
const connectionId = connectionIds[platform];
if (!connectionId) throw new Error(`No Kernel connection ID configured for platform: ${platform} — add it to settings.json under kernel.connection_ids`);
const Kernel = require(KERNEL_SDK_PATH);
const kernel = new Kernel({ apiKey });
console.log(` 🔄 Refreshing ${platform} session...`);
// Trigger re-auth (uses stored credentials automatically)
const loginResp = await kernel.auth.connections.login(connectionId);
if (loginResp.status === 'SUCCESS') {
console.log(`${platform} session refreshed`);
return true;
}
// If not immediately successful, poll for up to 30s
console.log(`${platform} session pending (status: ${loginResp.status}), polling...`);
const start = Date.now();
let pollCount = 0;
while (Date.now() - start < SESSION_REFRESH_POLL_TIMEOUT) {
await new Promise(r => setTimeout(r, SESSION_REFRESH_POLL_WAIT));
pollCount++;
const conn = await kernel.auth.connections.retrieve(connectionId);
if (conn.status === 'SUCCESS') {
console.log(`${platform} session refreshed (after ${pollCount} polls)`);
return true;
}
if (['FAILED', 'EXPIRED', 'CANCELED'].includes(conn.status)) {
console.warn(` ⚠️ ${platform} session refresh failed: ${conn.status} (after ${pollCount} polls)`);
return false;
}
}
console.warn(` ⚠️ ${platform} session refresh timed out`);
return false;
}
/**
* Verify login after browser connects — if not logged in, trigger refresh and retry
*/
export async function ensureLoggedIn(page, verifyFn, platform, apiKey, connectionIds = {}, maxAttempts = 2) {
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
const loggedIn = await verifyFn(page);
if (loggedIn) return true;
if (attempt < maxAttempts) {
console.warn(` ⚠️ ${platform} not logged in (attempt ${attempt}), refreshing session...`);
await refreshSession(platform, apiKey, connectionIds);
await page.reload({ waitUntil: 'domcontentloaded' });
await page.waitForTimeout(SESSION_LOGIN_VERIFY_WAIT);
}
}
return false;
}