|
import os from 'node:os'; |
|
import {onExit} from 'signal-exit'; |
|
|
|
const DEFAULT_FORCE_KILL_TIMEOUT = 1000 * 5; |
|
|
|
|
|
export const spawnedKill = (kill, signal = 'SIGTERM', options = {}) => { |
|
const killResult = kill(signal); |
|
setKillTimeout(kill, signal, options, killResult); |
|
return killResult; |
|
}; |
|
|
|
const setKillTimeout = (kill, signal, options, killResult) => { |
|
if (!shouldForceKill(signal, options, killResult)) { |
|
return; |
|
} |
|
|
|
const timeout = getForceKillAfterTimeout(options); |
|
const t = setTimeout(() => { |
|
kill('SIGKILL'); |
|
}, timeout); |
|
|
|
|
|
|
|
|
|
|
|
if (t.unref) { |
|
t.unref(); |
|
} |
|
}; |
|
|
|
const shouldForceKill = (signal, {forceKillAfterTimeout}, killResult) => isSigterm(signal) && forceKillAfterTimeout !== false && killResult; |
|
|
|
const isSigterm = signal => signal === os.constants.signals.SIGTERM |
|
|| (typeof signal === 'string' && signal.toUpperCase() === 'SIGTERM'); |
|
|
|
const getForceKillAfterTimeout = ({forceKillAfterTimeout = true}) => { |
|
if (forceKillAfterTimeout === true) { |
|
return DEFAULT_FORCE_KILL_TIMEOUT; |
|
} |
|
|
|
if (!Number.isFinite(forceKillAfterTimeout) || forceKillAfterTimeout < 0) { |
|
throw new TypeError(`Expected the \`forceKillAfterTimeout\` option to be a non-negative integer, got \`${forceKillAfterTimeout}\` (${typeof forceKillAfterTimeout})`); |
|
} |
|
|
|
return forceKillAfterTimeout; |
|
}; |
|
|
|
|
|
export const spawnedCancel = (spawned, context) => { |
|
const killResult = spawned.kill(); |
|
|
|
if (killResult) { |
|
context.isCanceled = true; |
|
} |
|
}; |
|
|
|
const timeoutKill = (spawned, signal, reject) => { |
|
spawned.kill(signal); |
|
reject(Object.assign(new Error('Timed out'), {timedOut: true, signal})); |
|
}; |
|
|
|
|
|
export const setupTimeout = (spawned, {timeout, killSignal = 'SIGTERM'}, spawnedPromise) => { |
|
if (timeout === 0 || timeout === undefined) { |
|
return spawnedPromise; |
|
} |
|
|
|
let timeoutId; |
|
const timeoutPromise = new Promise((resolve, reject) => { |
|
timeoutId = setTimeout(() => { |
|
timeoutKill(spawned, killSignal, reject); |
|
}, timeout); |
|
}); |
|
|
|
const safeSpawnedPromise = spawnedPromise.finally(() => { |
|
clearTimeout(timeoutId); |
|
}); |
|
|
|
return Promise.race([timeoutPromise, safeSpawnedPromise]); |
|
}; |
|
|
|
export const validateTimeout = ({timeout}) => { |
|
if (timeout !== undefined && (!Number.isFinite(timeout) || timeout < 0)) { |
|
throw new TypeError(`Expected the \`timeout\` option to be a non-negative integer, got \`${timeout}\` (${typeof timeout})`); |
|
} |
|
}; |
|
|
|
|
|
export const setExitHandler = async (spawned, {cleanup, detached}, timedPromise) => { |
|
if (!cleanup || detached) { |
|
return timedPromise; |
|
} |
|
|
|
const removeExitHandler = onExit(() => { |
|
spawned.kill(); |
|
}); |
|
|
|
return timedPromise.finally(() => { |
|
removeExitHandler(); |
|
}); |
|
}; |
|
|