2021-03-18 03:17:05 +01:00
|
|
|
// https://gist.github.com/nfantone/1eaa803772025df69d07f4dbf5df7e58
|
2024-02-21 19:14:26 +01:00
|
|
|
import { inspect } from "node:util";
|
2023-12-05 08:12:10 +01:00
|
|
|
|
2021-03-18 03:17:05 +01:00
|
|
|
/**
|
|
|
|
* @callback BeforeShutdownListener
|
|
|
|
* @param {string} [signalOrEvent] The exit signal or event name received on the process.
|
|
|
|
*/
|
2024-02-27 18:02:59 +01:00
|
|
|
type BeforeShutdownListener = (signalOrEvent: string) => PromiseLike<void>;
|
2021-03-18 03:17:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* System signals the app will listen to initiate shutdown.
|
|
|
|
* @const {string[]}
|
|
|
|
*/
|
2023-01-13 05:40:33 +01:00
|
|
|
const SHUTDOWN_SIGNALS = ["SIGINT", "SIGTERM"];
|
2021-03-18 03:17:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Time in milliseconds to wait before forcing shutdown.
|
|
|
|
* @const {number}
|
|
|
|
*/
|
|
|
|
const SHUTDOWN_TIMEOUT = 15000;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A queue of listener callbacks to execute before shutting
|
|
|
|
* down the process.
|
|
|
|
* @type {BeforeShutdownListener[]}
|
|
|
|
*/
|
2024-02-27 18:02:59 +01:00
|
|
|
const shutdownListeners: BeforeShutdownListener[] = [];
|
2021-03-18 03:17:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Listen for signals and execute given `fn` function once.
|
|
|
|
* @param {string[]} signals System signals to listen to.
|
|
|
|
* @param {function(string)} fn Function to execute on shutdown.
|
|
|
|
*/
|
2023-01-13 05:40:33 +01:00
|
|
|
const processOnce = (
|
|
|
|
signals: string[],
|
|
|
|
fn: (signalOrEvent: string) => void,
|
|
|
|
) => {
|
2021-03-18 05:33:14 +01:00
|
|
|
for (const sig of signals) {
|
|
|
|
process.once(sig, fn);
|
|
|
|
}
|
2021-03-18 03:17:05 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets a forced shutdown mechanism that will exit the process after `timeout` milliseconds.
|
|
|
|
* @param {number} timeout Time to wait before forcing shutdown (milliseconds)
|
|
|
|
*/
|
2022-02-04 03:10:53 +01:00
|
|
|
const forceExitAfter = (timeout: number) => () => {
|
2021-03-18 03:17:05 +01:00
|
|
|
setTimeout(() => {
|
|
|
|
// Force shutdown after timeout
|
2023-01-13 05:40:33 +01:00
|
|
|
console.warn(
|
|
|
|
`Could not close resources gracefully after ${timeout}ms: forcing shutdown`,
|
|
|
|
);
|
2021-03-18 03:17:05 +01:00
|
|
|
return process.exit(1);
|
|
|
|
}, timeout).unref();
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Main process shutdown handler. Will invoke every previously registered async shutdown listener
|
|
|
|
* in the queue and exit with a code of `0`. Any `Promise` rejections from any listener will
|
|
|
|
* be logged out as a warning, but won't prevent other callbacks from executing.
|
|
|
|
* @param {string} signalOrEvent The exit signal or event name received on the process.
|
|
|
|
*/
|
2022-02-04 03:10:53 +01:00
|
|
|
async function shutdownHandler(signalOrEvent: string) {
|
2023-01-13 05:40:33 +01:00
|
|
|
if (process.env.NODE_ENV === "test") return process.exit(0);
|
2021-06-05 07:54:07 +02:00
|
|
|
|
2021-03-18 03:17:05 +01:00
|
|
|
console.warn(`Shutting down: received [${signalOrEvent}] signal`);
|
|
|
|
|
|
|
|
for (const listener of shutdownListeners) {
|
|
|
|
try {
|
|
|
|
await listener(signalOrEvent);
|
|
|
|
} catch (err) {
|
2022-02-04 03:10:53 +01:00
|
|
|
if (err instanceof Error) {
|
2023-01-13 05:40:33 +01:00
|
|
|
console.warn(
|
2024-02-21 14:23:24 +01:00
|
|
|
`A shutdown handler failed before completing with:\n${inspect(err)}`,
|
2023-01-13 05:40:33 +01:00
|
|
|
);
|
2022-02-04 03:10:53 +01:00
|
|
|
}
|
2021-03-18 03:17:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return process.exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Registers a new shutdown listener to be invoked before exiting
|
|
|
|
* the main process. Listener handlers are guaranteed to be called in the order
|
|
|
|
* they were registered.
|
|
|
|
* @param {BeforeShutdownListener} listener The shutdown listener to register.
|
|
|
|
* @returns {BeforeShutdownListener} Echoes back the supplied `listener`.
|
|
|
|
*/
|
2024-02-27 18:02:59 +01:00
|
|
|
export function beforeShutdown(
|
|
|
|
listener: BeforeShutdownListener,
|
|
|
|
): BeforeShutdownListener {
|
2021-03-18 03:17:05 +01:00
|
|
|
shutdownListeners.push(listener);
|
|
|
|
return listener;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Register shutdown callback that kills the process after `SHUTDOWN_TIMEOUT` milliseconds
|
|
|
|
// This prevents custom shutdown handlers from hanging the process indefinitely
|
|
|
|
processOnce(SHUTDOWN_SIGNALS, forceExitAfter(SHUTDOWN_TIMEOUT));
|
|
|
|
|
|
|
|
// Register process shutdown callback
|
|
|
|
// Will listen to incoming signal events and execute all registered handlers in the stack
|
|
|
|
processOnce(SHUTDOWN_SIGNALS, shutdownHandler);
|