What: - stub resolvePinnedHostname in web-fetch tests to avoid DNS flake - close lock file handles via FileHandle.close during cleanup to avoid EBADF Why: - make CI deterministic without network/DNS dependence - prevent double-close errors from GC Tests: - pnpm vitest run --config vitest.unit.config.ts src/agents/tools/web-tools.fetch.test.ts src/agents/session-write-lock.test.ts (failed: missing @aws-sdk/client-bedrock)
189 lines
5.5 KiB
TypeScript
189 lines
5.5 KiB
TypeScript
import fsSync from "node:fs";
|
|
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
|
|
type LockFilePayload = {
|
|
pid: number;
|
|
createdAt: string;
|
|
};
|
|
|
|
type HeldLock = {
|
|
count: number;
|
|
handle: fs.FileHandle;
|
|
lockPath: string;
|
|
};
|
|
|
|
const HELD_LOCKS = new Map<string, HeldLock>();
|
|
const CLEANUP_SIGNALS = ["SIGINT", "SIGTERM", "SIGQUIT", "SIGABRT"] as const;
|
|
type CleanupSignal = (typeof CLEANUP_SIGNALS)[number];
|
|
const cleanupHandlers = new Map<CleanupSignal, () => void>();
|
|
|
|
function isAlive(pid: number): boolean {
|
|
if (!Number.isFinite(pid) || pid <= 0) return false;
|
|
try {
|
|
process.kill(pid, 0);
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Synchronously release all held locks.
|
|
* Used during process exit when async operations aren't reliable.
|
|
*/
|
|
function releaseAllLocksSync(): void {
|
|
for (const [sessionFile, held] of HELD_LOCKS) {
|
|
try {
|
|
if (typeof held.handle.close === "function") {
|
|
void held.handle.close().catch(() => {});
|
|
}
|
|
} catch {
|
|
// Ignore errors during cleanup - best effort
|
|
}
|
|
try {
|
|
fsSync.rmSync(held.lockPath, { force: true });
|
|
} catch {
|
|
// Ignore errors during cleanup - best effort
|
|
}
|
|
HELD_LOCKS.delete(sessionFile);
|
|
}
|
|
}
|
|
|
|
let cleanupRegistered = false;
|
|
|
|
function handleTerminationSignal(signal: CleanupSignal): void {
|
|
releaseAllLocksSync();
|
|
const shouldReraise = process.listenerCount(signal) === 1;
|
|
if (shouldReraise) {
|
|
const handler = cleanupHandlers.get(signal);
|
|
if (handler) process.off(signal, handler);
|
|
try {
|
|
process.kill(process.pid, signal);
|
|
} catch {
|
|
// Ignore errors during shutdown
|
|
}
|
|
}
|
|
}
|
|
|
|
function registerCleanupHandlers(): void {
|
|
if (cleanupRegistered) return;
|
|
cleanupRegistered = true;
|
|
|
|
// Cleanup on normal exit and process.exit() calls
|
|
process.on("exit", () => {
|
|
releaseAllLocksSync();
|
|
});
|
|
|
|
// Handle termination signals
|
|
for (const signal of CLEANUP_SIGNALS) {
|
|
try {
|
|
const handler = () => handleTerminationSignal(signal);
|
|
cleanupHandlers.set(signal, handler);
|
|
process.on(signal, handler);
|
|
} catch {
|
|
// Ignore unsupported signals on this platform.
|
|
}
|
|
}
|
|
}
|
|
|
|
async function readLockPayload(lockPath: string): Promise<LockFilePayload | null> {
|
|
try {
|
|
const raw = await fs.readFile(lockPath, "utf8");
|
|
const parsed = JSON.parse(raw) as Partial<LockFilePayload>;
|
|
if (typeof parsed.pid !== "number") return null;
|
|
if (typeof parsed.createdAt !== "string") return null;
|
|
return { pid: parsed.pid, createdAt: parsed.createdAt };
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export async function acquireSessionWriteLock(params: {
|
|
sessionFile: string;
|
|
timeoutMs?: number;
|
|
staleMs?: number;
|
|
}): Promise<{
|
|
release: () => Promise<void>;
|
|
}> {
|
|
registerCleanupHandlers();
|
|
const timeoutMs = params.timeoutMs ?? 10_000;
|
|
const staleMs = params.staleMs ?? 30 * 60 * 1000;
|
|
const sessionFile = path.resolve(params.sessionFile);
|
|
const sessionDir = path.dirname(sessionFile);
|
|
await fs.mkdir(sessionDir, { recursive: true });
|
|
let normalizedDir = sessionDir;
|
|
try {
|
|
normalizedDir = await fs.realpath(sessionDir);
|
|
} catch {
|
|
// Fall back to the resolved path if realpath fails (permissions, transient FS).
|
|
}
|
|
const normalizedSessionFile = path.join(normalizedDir, path.basename(sessionFile));
|
|
const lockPath = `${normalizedSessionFile}.lock`;
|
|
|
|
const held = HELD_LOCKS.get(normalizedSessionFile);
|
|
if (held) {
|
|
held.count += 1;
|
|
return {
|
|
release: async () => {
|
|
const current = HELD_LOCKS.get(normalizedSessionFile);
|
|
if (!current) return;
|
|
current.count -= 1;
|
|
if (current.count > 0) return;
|
|
HELD_LOCKS.delete(normalizedSessionFile);
|
|
await current.handle.close();
|
|
await fs.rm(current.lockPath, { force: true });
|
|
},
|
|
};
|
|
}
|
|
|
|
const startedAt = Date.now();
|
|
let attempt = 0;
|
|
while (Date.now() - startedAt < timeoutMs) {
|
|
attempt += 1;
|
|
try {
|
|
const handle = await fs.open(lockPath, "wx");
|
|
await handle.writeFile(
|
|
JSON.stringify({ pid: process.pid, createdAt: new Date().toISOString() }, null, 2),
|
|
"utf8",
|
|
);
|
|
HELD_LOCKS.set(normalizedSessionFile, { count: 1, handle, lockPath });
|
|
return {
|
|
release: async () => {
|
|
const current = HELD_LOCKS.get(normalizedSessionFile);
|
|
if (!current) return;
|
|
current.count -= 1;
|
|
if (current.count > 0) return;
|
|
HELD_LOCKS.delete(normalizedSessionFile);
|
|
await current.handle.close();
|
|
await fs.rm(current.lockPath, { force: true });
|
|
},
|
|
};
|
|
} catch (err) {
|
|
const code = (err as { code?: unknown }).code;
|
|
if (code !== "EEXIST") throw err;
|
|
const payload = await readLockPayload(lockPath);
|
|
const createdAt = payload?.createdAt ? Date.parse(payload.createdAt) : NaN;
|
|
const stale = !Number.isFinite(createdAt) || Date.now() - createdAt > staleMs;
|
|
const alive = payload?.pid ? isAlive(payload.pid) : false;
|
|
if (stale || !alive) {
|
|
await fs.rm(lockPath, { force: true });
|
|
continue;
|
|
}
|
|
|
|
const delay = Math.min(1000, 50 * attempt);
|
|
await new Promise((r) => setTimeout(r, delay));
|
|
}
|
|
}
|
|
|
|
const payload = await readLockPayload(lockPath);
|
|
const owner = payload?.pid ? `pid=${payload.pid}` : "unknown";
|
|
throw new Error(`session file locked (timeout ${timeoutMs}ms): ${owner} ${lockPath}`);
|
|
}
|
|
|
|
export const __testing = {
|
|
cleanupSignals: [...CLEANUP_SIGNALS],
|
|
handleTerminationSignal,
|
|
releaseAllLocksSync,
|
|
};
|