@rudderstack/integrations-lib 0.2.48 → 0.2.49
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"manager.d.ts","sourceRoot":"","sources":["../../src/cluster/manager.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AAEtC,OAAO,EACL,qBAAqB,EACrB,oBAAoB,EAMrB,MAAM,SAAS,CAAC;AAGjB;;;;;;;;;GASG;AACH,qBAAa,cAAe,SAAQ,YAAY;IAC9C,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAkC;IAE1D,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAkC;IAE1D,OAAO,CAAC,OAAO,CAAS;IAExB,OAAO,CAAC,cAAc,CAAS;IAE/B,OAAO,CAAC,eAAe,CAA8B;IAErD,OAAO,CAAC,mBAAmB,CAA+B;IAE1D,OAAO,CAAC,cAAc,CAAyC;gBAEnD,OAAO,GAAE,qBAA0B;IAM/C;;;;OAIG;IACH,OAAO,CAAC,UAAU;IAUlB;;;;OAIG;IACU,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IASnC;;OAEG;IACI,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAqB/C;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAU3B;;OAEG;IACH,OAAO,CAAC,oBAAoB;IAO5B;;OAEG;YACW,YAAY;IAqB1B;;OAEG;YACW,WAAW;IAUzB;;OAEG;IACH,OAAO,CAAC,yBAAyB;IAWjC;;OAEG;IACH,OAAO,CAAC,0BAA0B;IAkBlC;;OAEG;IACH,OAAO,CAAC,WAAW;IAuBnB;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAqB3B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAWxB;;OAEG;IACH,OAAO,CAAC,qBAAqB;IAM7B;;OAEG;IACH,OAAO,CAAC,oBAAoB;IAO5B;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAkB1B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAkBxB;;OAEG;IACH,OAAO,CAAC,iBAAiB;IAiCzB;;OAEG;IACH,OAAO,CAAC,gBAAgB;
|
|
1
|
+
{"version":3,"file":"manager.d.ts","sourceRoot":"","sources":["../../src/cluster/manager.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AAEtC,OAAO,EACL,qBAAqB,EACrB,oBAAoB,EAMrB,MAAM,SAAS,CAAC;AAGjB;;;;;;;;;GASG;AACH,qBAAa,cAAe,SAAQ,YAAY;IAC9C,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAkC;IAE1D,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAkC;IAE1D,OAAO,CAAC,OAAO,CAAS;IAExB,OAAO,CAAC,cAAc,CAAS;IAE/B,OAAO,CAAC,eAAe,CAA8B;IAErD,OAAO,CAAC,mBAAmB,CAA+B;IAE1D,OAAO,CAAC,cAAc,CAAyC;gBAEnD,OAAO,GAAE,qBAA0B;IAM/C;;;;OAIG;IACH,OAAO,CAAC,UAAU;IAUlB;;;;OAIG;IACU,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IASnC;;OAEG;IACI,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAqB/C;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAU3B;;OAEG;IACH,OAAO,CAAC,oBAAoB;IAO5B;;OAEG;YACW,YAAY;IAqB1B;;OAEG;YACW,WAAW;IAUzB;;OAEG;IACH,OAAO,CAAC,yBAAyB;IAWjC;;OAEG;IACH,OAAO,CAAC,0BAA0B;IAkBlC;;OAEG;IACH,OAAO,CAAC,WAAW;IAuBnB;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAqB3B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAWxB;;OAEG;IACH,OAAO,CAAC,qBAAqB;IAM7B;;OAEG;IACH,OAAO,CAAC,oBAAoB;IAO5B;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAkB1B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAkBxB;;OAEG;IACH,OAAO,CAAC,iBAAiB;IAiCzB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IA4BxB,OAAO,CAAC,WAAW;IAInB,OAAO,CAAC,kBAAkB;IAM1B;;OAEG;IACH,OAAO,CAAC,0BAA0B;IAsBlC;;OAEG;YACW,eAAe;IA2B7B;;OAEG;YACW,kBAAkB;IAwChC;;OAEG;YACW,uBAAuB;IAOrC;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAU3B;;OAEG;YACW,cAAc;IAwB5B;;OAEG;IACI,cAAc,IAAI,MAAM;IAI/B;;OAEG;IACI,aAAa,IAAI,KAAK,CAAC;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,GAAG,EAAE,MAAM,CAAC;QAAC,YAAY,EAAE,MAAM,CAAA;KAAE,CAAC;IAQhF;;OAEG;IACI,SAAS,IAAI,OAAO;IAI3B;;OAEG;IACI,UAAU,IAAI,OAAO;CAG7B;AAGD,MAAM,WAAW,qBAAqB;IACpC,EAAE,CAAC,CAAC,SAAS,MAAM,oBAAoB,EAAE,KAAK,EAAE,CAAC,EAAE,QAAQ,EAAE,oBAAoB,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC;IAC5F,IAAI,CAAC,CAAC,SAAS,MAAM,oBAAoB,EACvC,KAAK,EAAE,CAAC,EACR,GAAG,IAAI,EAAE,UAAU,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,GAC3C,OAAO,CAAC;CACZ"}
|
package/build/cluster/manager.js
CHANGED
|
@@ -323,16 +323,20 @@ class ClusterManager extends events_1.EventEmitter {
|
|
|
323
323
|
* Handles worker exit events
|
|
324
324
|
*/
|
|
325
325
|
handleWorkerExit(worker, code, signal) {
|
|
326
|
-
|
|
326
|
+
const workerId = this.getWorkerId(worker);
|
|
327
327
|
this.emit('worker:died', worker, code, signal);
|
|
328
328
|
const workerState = this.workers.get(worker.id);
|
|
329
|
-
if (!workerState)
|
|
329
|
+
if (!workerState) {
|
|
330
|
+
logger.warn(`Unknown worker ${workerId} (pid: ${worker.process.pid}) exited with code ${code} and signal ${signal}`);
|
|
330
331
|
return;
|
|
332
|
+
}
|
|
331
333
|
this.workers.delete(worker.id);
|
|
332
334
|
// If we're shutting down or worker was killed intentionally, don't restart
|
|
333
335
|
if (this.isShuttingDown || workerState.isShuttingDown) {
|
|
336
|
+
logger.info(`Worker ${workerId} (pid: ${worker.process.pid}) exited with code ${code} and signal ${signal}`);
|
|
334
337
|
return;
|
|
335
338
|
}
|
|
339
|
+
logger.error(`Worker ${workerId} (pid: ${worker.process.pid}) exited unexpectedly with code ${code} and signal ${signal}`);
|
|
336
340
|
// Handle unexpected exit
|
|
337
341
|
this.handleUnexpectedWorkerExit(workerState);
|
|
338
342
|
}
|
|
@@ -367,7 +371,7 @@ class ClusterManager extends events_1.EventEmitter {
|
|
|
367
371
|
* Shuts down the primary process
|
|
368
372
|
*/
|
|
369
373
|
async shutdownPrimary(signal) {
|
|
370
|
-
logger.
|
|
374
|
+
logger.warn(`Primary process (pid: ${process.pid}) shutting down (signal: ${signal ?? 'manual'})`);
|
|
371
375
|
this.stopHealthMonitoring();
|
|
372
376
|
this.removeSignalHandlers();
|
|
373
377
|
// Shutdown all workers
|
|
@@ -435,7 +439,7 @@ class ClusterManager extends events_1.EventEmitter {
|
|
|
435
439
|
* Shuts down a worker process
|
|
436
440
|
*/
|
|
437
441
|
async shutdownWorker(signal) {
|
|
438
|
-
logger.
|
|
442
|
+
logger.warn(`Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutting down (signal: ${signal ?? 'manual'})`);
|
|
439
443
|
this.removeSignalHandlers();
|
|
440
444
|
// Execute worker shutdown function
|
|
441
445
|
await (0, utils_1.safeExecute)(() => (0, utils_1.timeout)(this.options.workerShutdownFn(signal), this.options.shutdownTimeout, 'worker shutdown timeout'), `Error in worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutdown function`);
|
|
@@ -472,4 +476,4 @@ class ClusterManager extends events_1.EventEmitter {
|
|
|
472
476
|
}
|
|
473
477
|
}
|
|
474
478
|
exports.ClusterManager = ClusterManager;
|
|
475
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"manager.js","sourceRoot":"","sources":["../../src/cluster/manager.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this */\n/* eslint-disable no-param-reassign */\nimport { Worker } from 'cluster';\nimport { EventEmitter } from 'events';\nimport * as logger from '../logger';\nimport {\n  ClusterManagerOptions,\n  ClusterManagerEvents,\n  WorkerState,\n  IPCMessage,\n  PingMessage,\n  PongMessage,\n  ShutdownMessage,\n} from './types';\nimport { delay, timeout, safeExecute, validateAndDefaultOptions } from './utils';\n\n/**\n * ClusterManager - A cluster lifecycle management system\n *\n * The manager supports the following features:\n * - Graceful shutdown with configurable timeout\n * - Worker health monitoring with ping/pong mechanism\n * - Automatic worker restart with configurable limits\n * - Signal handling for shutdown triggers\n * - Flexible primary and worker function handlers\n */\nexport class ClusterManager extends EventEmitter {\n  private readonly options: Required<ClusterManagerOptions>;\n\n  private readonly workers = new Map<number, WorkerState>();\n\n  private started = false;\n\n  private isShuttingDown = false;\n\n  private shutdownPromise: Promise<void> | null = null;\n\n  private healthCheckInterval: NodeJS.Timeout | null = null;\n\n  private signalHandlers = new Map<NodeJS.Signals, () => void>();\n\n  constructor(options: ClusterManagerOptions = {}) {\n    super();\n    this.options = validateAndDefaultOptions(options);\n    this.initialize();\n  }\n\n  /**\n   * Initialize the cluster manager with primary process configuration\n   *\n   * @see {@link https://nodejs.org/docs/latest/api/cluster.html#clustersetupprimaryoptions Node.js cluster.setupPrimary()}\n   */\n  private initialize(): void {\n    // Only call setupPrimary in the primary process\n    if (this.options.cluster.isPrimary) {\n      this.options.cluster.setupPrimary({\n        serialization: this.options.serialization,\n      });\n    }\n    this.setupSignalHandlers();\n  }\n\n  /**\n   * Starts the cluster manager\n   * In primary process: starts workers and health monitoring\n   * In worker process: executes worker function\n   */\n  public async start(): Promise<void> {\n    if (this.options.cluster.isPrimary) {\n      await this.startPrimary();\n    } else {\n      await this.startWorker();\n    }\n    this.started = true;\n  }\n\n  /**\n   * Initiates graceful shutdown of the cluster\n   */\n  public shutdown(signal?: string): Promise<void> {\n    if (!this.started) {\n      return Promise.resolve();\n    }\n\n    if (this.shutdownPromise) {\n      return this.shutdownPromise;\n    }\n\n    this.isShuttingDown = true;\n    this.emit('shutdown:started', signal);\n\n    if (this.options.cluster.isPrimary) {\n      this.shutdownPromise = this.shutdownPrimary(signal);\n    } else {\n      this.shutdownPromise = this.shutdownWorker(signal);\n    }\n\n    return this.shutdownPromise;\n  }\n\n  /**\n   * Sets up signal handlers for graceful shutdown\n   */\n  private setupSignalHandlers(): void {\n    this.options.shutdownSignals.forEach((signal) => {\n      const handler = () => {\n        this.shutdown(signal);\n      };\n      this.signalHandlers.set(signal, handler);\n      process.on(signal, handler);\n    });\n  }\n\n  /**\n   * Removes signal handlers\n   */\n  private removeSignalHandlers(): void {\n    Array.from(this.signalHandlers.entries()).forEach(([signal, handler]) => {\n      process.removeListener(signal, handler);\n    });\n    this.signalHandlers.clear();\n  }\n\n  /**\n   * Starts the primary process\n   */\n  private async startPrimary(): Promise<void> {\n    logger.info(\n      `Primary process (pid: ${process.pid}) starting with ${this.options.numWorkers} workers`,\n    );\n\n    // Execute primary initialization function, any error will stop the cluster from starting\n    await this.options.primaryFn();\n\n    // Set up cluster event handlers\n    this.setupClusterEventHandlers();\n\n    // Spawn initial workers\n    for (let i = 0; i < this.options.numWorkers; i += 1) {\n      const workerId = i + 1; // Worker IDs start from 1\n      this.spawnWorker(workerId);\n    }\n\n    // Start health monitoring\n    this.startHealthMonitoring();\n  }\n\n  /**\n   * Starts a worker process\n   */\n  private async startWorker(): Promise<void> {\n    logger.info(`Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) starting`);\n\n    // Set up IPC message handlers\n    this.setupWorkerMessageHandlers();\n\n    // Execute worker initialization function, any error will be propagated\n    await this.options.workerFn();\n  }\n\n  /**\n   * Sets up cluster event handlers for the primary process\n   */\n  private setupClusterEventHandlers(): void {\n    this.options.cluster.on('exit', (worker: Worker, code: number, signal: string) => {\n      this.handleWorkerExit(worker, code, signal);\n    });\n\n    this.options.cluster.on('online', (worker: Worker) => {\n      logger.info(`Worker ${this.getWorkerId(worker)} (pid: ${worker.process.pid}) is online`);\n      this.emit('worker:started', worker);\n    });\n  }\n\n  /**\n   * Sets up IPC message handlers for worker processes\n   */\n  private setupWorkerMessageHandlers(): void {\n    process.on('message', (message: IPCMessage) => {\n      switch (message.type) {\n        case 'ping':\n          this.handleWorkerPing(message);\n          break;\n        case 'shutdown':\n          this.shutdown(message.signal);\n          break;\n        default:\n          if (!message.type.includes('Metrics')) {\n            logger.warn(`ignoring unknown message type in worker ${process.pid}: ${message.type}`);\n          }\n          break;\n      }\n    });\n  }\n\n  /**\n   * Spawns a new worker and sets up its state\n   */\n  private spawnWorker(id: number): Worker {\n    const worker = this.options.cluster.fork({\n      WORKER_ID: id,\n    });\n    const workerState: WorkerState = {\n      id,\n      worker,\n      restartCount: 0,\n      lastPing: Date.now(),\n      pendingPing: false,\n      isShuttingDown: false,\n    };\n\n    this.workers.set(worker.id, workerState);\n\n    // Set up worker message handler\n    worker.on('message', (message: IPCMessage) => {\n      this.handleWorkerMessage(worker, message);\n    });\n\n    return worker;\n  }\n\n  /**\n   * Handles messages from workers\n   */\n  private handleWorkerMessage(worker: Worker, message: IPCMessage): void {\n    const workerState = this.workers.get(worker.id);\n    if (!workerState) return;\n\n    switch (message.type) {\n      case 'pong':\n        workerState.lastPing = Date.now();\n        workerState.pendingPing = false;\n        break;\n      default:\n        if (!message.type.includes('Metrics')) {\n          logger.warn(\n            `Received unknown message type from worker ${this.getWorkerId(worker)} (pid: ${\n              worker.process.pid\n            }): ${message.type}`,\n          );\n        }\n        break;\n    }\n  }\n\n  /**\n   * Handles ping messages in worker processes\n   */\n  private handleWorkerPing(message: PingMessage): void {\n    const pongMessage: PongMessage = {\n      type: 'pong',\n      timestamp: message.timestamp,\n    };\n\n    if (process.send) {\n      process.send(pongMessage);\n    }\n  }\n\n  /**\n   * Starts health monitoring for all workers\n   */\n  private startHealthMonitoring(): void {\n    this.healthCheckInterval = setInterval(() => {\n      this.performHealthCheck();\n    }, this.options.pingFrequency);\n  }\n\n  /**\n   * Stops health monitoring\n   */\n  private stopHealthMonitoring(): void {\n    if (this.healthCheckInterval) {\n      clearInterval(this.healthCheckInterval);\n      this.healthCheckInterval = null;\n    }\n  }\n\n  /**\n   * Performs health check on all workers\n   */\n  private performHealthCheck(): void {\n    if (this.isShuttingDown) return;\n\n    Array.from(this.workers.values())\n      .filter((workerState) => !workerState.isShuttingDown)\n      .forEach((workerState) => {\n        const timeSinceLastPing = Date.now() - workerState.lastPing;\n\n        if (workerState.pendingPing && timeSinceLastPing > this.options.pingTimeout) {\n          // Worker is stuck, handle it\n          this.handleStuckWorker(workerState);\n        } else if (!workerState.pendingPing) {\n          // Send ping to worker\n          this.sendPingToWorker(workerState);\n        }\n      });\n  }\n\n  /**\n   * Sends a ping message to a worker\n   */\n  private sendPingToWorker(workerState: WorkerState): void {\n    const pingMessage: PingMessage = {\n      type: 'ping',\n      timestamp: Date.now(),\n    };\n\n    workerState.pendingPing = true;\n    try {\n      workerState.worker.send(pingMessage);\n    } catch (error) {\n      logger.error(\n        `Failed to send ping to worker ${workerState.id} (pid: ${\n          workerState.worker.process.pid\n        }): ${error instanceof Error ? error.message : String(error)}`,\n      );\n    }\n  }\n\n  /**\n   * Handles a stuck worker\n   */\n  private handleStuckWorker(workerState: WorkerState): void {\n    logger.error(\n      `Worker ${workerState.id} (pid: ${workerState.worker.process.pid}) is stuck, killing it`,\n    );\n    this.emit('worker:stuck', workerState.worker);\n\n    // Remove worker state\n    this.workers.delete(workerState.worker.id);\n\n    // Kill the worker\n    workerState.worker.kill('SIGKILL');\n\n    // Determine if we should spawn a replacement\n    const shouldSpawn = this.options.stuckWorkerRespawnFunc(workerState.worker);\n\n    if (shouldSpawn && !this.isShuttingDown) {\n      logger.info(\n        `Spawning replacement worker for stuck worker ${workerState.id} (pid: ${workerState.worker.process.pid})`,\n      );\n      const { restartCount } = workerState; // Preserve restart count\n      const newWorker = this.spawnWorker(workerState.id);\n      const newWorkerState = this.workers.get(newWorker.id);\n      if (newWorkerState) {\n        newWorkerState.restartCount = restartCount;\n      }\n    } else {\n      logger.error(\n        `Triggering cluster shutdown due to stuck worker ${workerState.id} (pid: ${workerState.worker.process.pid})`,\n      );\n      this.shutdown('STUCK_WORKER');\n    }\n  }\n\n  /**\n   * Handles worker exit events\n   */\n  private handleWorkerExit(worker: Worker, code: number | null, signal: string | null): void {\n    logger.info(\n      `Worker ${this.getWorkerId(worker)} (pid: ${\n        worker.process.pid\n      }) died with code ${code} and signal ${signal}`,\n    );\n    this.emit('worker:died', worker, code, signal);\n\n    const workerState = this.workers.get(worker.id);\n    if (!workerState) return;\n\n    this.workers.delete(worker.id);\n\n    // If we're shutting down or worker was killed intentionally, don't restart\n    if (this.isShuttingDown || workerState.isShuttingDown) {\n      return;\n    }\n\n    // Handle unexpected exit\n    this.handleUnexpectedWorkerExit(workerState);\n  }\n\n  private getWorkerId(worker: Worker): number {\n    return this.workers.get(worker.id)?.id ?? worker.id;\n  }\n\n  private getCurrentWorkerId(): number {\n    const envWorkerId =\n      process.env.WORKER_ID !== undefined ? Number(process.env.WORKER_ID) : undefined;\n    return envWorkerId ?? this.options.cluster.worker?.id ?? -1;\n  }\n\n  /**\n   * Handles unexpected worker exits with restart logic\n   */\n  private handleUnexpectedWorkerExit(workerState: WorkerState): void {\n    const restartCount = workerState.restartCount + 1;\n\n    if (restartCount <= this.options.restartMaxTimes) {\n      logger.error(\n        `Restarting worker ${workerState.id} (pid: ${workerState.worker.process.pid}) (attempt ${restartCount}/${this.options.restartMaxTimes})`,\n      );\n      const newWorker = this.spawnWorker(workerState.id);\n      const newWorkerState = this.workers.get(newWorker.id);\n      if (newWorkerState) {\n        newWorkerState.restartCount = restartCount;\n      }\n      this.emit('worker:restarted', newWorker, workerState.restartCount);\n    } else {\n      logger.error(\n        `Restart limit (${this.options.restartMaxTimes}) exceeded for worker ${workerState.id} (pid: ${workerState.worker.process.pid}), shutting down cluster`,\n      );\n      this.emit('worker:restart-limit-exceeded', workerState.worker, workerState.restartCount);\n      this.shutdown('RESTART_LIMIT_EXCEEDED');\n    }\n  }\n\n  /**\n   * Shuts down the primary process\n   */\n  private async shutdownPrimary(signal?: string): Promise<void> {\n    logger.info(\n      `Primary process (pid: ${process.pid}) shutting down (signal: ${signal ?? 'manual'})`,\n    );\n\n    this.stopHealthMonitoring();\n    this.removeSignalHandlers();\n\n    // Shutdown all workers\n    await this.shutdownAllWorkers(signal);\n\n    // Execute primary shutdown function\n    await safeExecute(\n      () =>\n        timeout(\n          this.options.primaryShutdownFn(signal),\n          this.options.shutdownTimeout,\n          'primary shutdown timeout',\n        ),\n      'Error in primary shutdown function',\n    );\n\n    logger.info(`Primary process (pid: ${process.pid}) shutdown completed`);\n    this.emit('shutdown:completed');\n    process.exit(0);\n  }\n\n  /**\n   * Shuts down all workers gracefully\n   */\n  private async shutdownAllWorkers(signal?: string): Promise<void> {\n    if (this.workers.size === 0) return;\n\n    logger.info(`Shutting down ${this.workers.size} workers...`);\n\n    // Mark all workers as shutting down and send shutdown message\n    Array.from(this.workers.values()).forEach((workerState) => {\n      workerState.isShuttingDown = true;\n      // Only send shutdown message if worker is still connected\n      if (!workerState.worker.isDead() && workerState.worker.process.connected) {\n        const shutdownMessage: ShutdownMessage = {\n          type: 'shutdown',\n          signal,\n        };\n        try {\n          workerState.worker.send(shutdownMessage);\n        } catch (error) {\n          // Worker IPC channel is already closed, which is fine\n          logger.warn(\n            `Failed to send shutdown message to worker ${workerState.id} (pid: ${\n              workerState.worker.process.pid\n            }): ${error instanceof Error ? error.message : String(error)}`,\n          );\n        }\n      }\n    });\n\n    // Wait for workers to exit gracefully or timeout\n    try {\n      await timeout(\n        this.waitForAllWorkersToExit(),\n        this.options.shutdownTimeout,\n        'Worker shutdown timeout',\n      );\n    } catch (error) {\n      logger.error('Graceful shutdown for workers timed out, forcing shutdown');\n      this.forceKillAllWorkers();\n    }\n  }\n\n  /**\n   * Waits for all workers to exit\n   */\n  private async waitForAllWorkersToExit(): Promise<void> {\n    while (this.workers.size > 0) {\n      // eslint-disable-next-line no-await-in-loop\n      await delay(100);\n    }\n  }\n\n  /**\n   * Force kills all remaining workers\n   */\n  private forceKillAllWorkers(): void {\n    Array.from(this.workers.values()).forEach((workerState) => {\n      logger.error(\n        `Force killing worker ${workerState.id} (pid: ${workerState.worker.process.pid})`,\n      );\n      workerState.worker.kill('SIGKILL');\n    });\n    this.workers.clear();\n  }\n\n  /**\n   * Shuts down a worker process\n   */\n  private async shutdownWorker(signal?: string): Promise<void> {\n    logger.info(\n      `Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutting down (signal: ${\n        signal ?? 'manual'\n      })`,\n    );\n\n    this.removeSignalHandlers();\n\n    // Execute worker shutdown function\n    await safeExecute(\n      () =>\n        timeout(\n          this.options.workerShutdownFn(signal),\n          this.options.shutdownTimeout,\n          'worker shutdown timeout',\n        ),\n      `Error in worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutdown function`,\n    );\n\n    logger.info(`Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutdown completed`);\n    process.exit(0);\n  }\n\n  /**\n   * Gets the current number of active workers\n   */\n  public getWorkerCount(): number {\n    return this.workers.size;\n  }\n\n  /**\n   * Gets information about all workers\n   */\n  public getWorkerInfo(): Array<{ id: number; pid: number; restartCount: number }> {\n    return Array.from(this.workers.values()).map((state) => ({\n      id: state.id,\n      pid: state.worker.process.pid ?? -1,\n      restartCount: state.restartCount,\n    }));\n  }\n\n  /**\n   * Checks if the cluster is currently started\n   */\n  public isStarted(): boolean {\n    return this.started;\n  }\n\n  /**\n   * Checks if the cluster is currently shutting down\n   */\n  public isShutdown(): boolean {\n    return this.isShuttingDown;\n  }\n}\n\n// Type-safe event emitter interface\nexport interface ClusterManagerEmitter {\n  on<K extends keyof ClusterManagerEvents>(event: K, listener: ClusterManagerEvents[K]): this;\n  emit<K extends keyof ClusterManagerEvents>(\n    event: K,\n    ...args: Parameters<ClusterManagerEvents[K]>\n  ): boolean;\n}\n"]}
|
|
479
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"manager.js","sourceRoot":"","sources":["../../src/cluster/manager.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this */\n/* eslint-disable no-param-reassign */\nimport { Worker } from 'cluster';\nimport { EventEmitter } from 'events';\nimport * as logger from '../logger';\nimport {\n  ClusterManagerOptions,\n  ClusterManagerEvents,\n  WorkerState,\n  IPCMessage,\n  PingMessage,\n  PongMessage,\n  ShutdownMessage,\n} from './types';\nimport { delay, timeout, safeExecute, validateAndDefaultOptions } from './utils';\n\n/**\n * ClusterManager - A cluster lifecycle management system\n *\n * The manager supports the following features:\n * - Graceful shutdown with configurable timeout\n * - Worker health monitoring with ping/pong mechanism\n * - Automatic worker restart with configurable limits\n * - Signal handling for shutdown triggers\n * - Flexible primary and worker function handlers\n */\nexport class ClusterManager extends EventEmitter {\n  private readonly options: Required<ClusterManagerOptions>;\n\n  private readonly workers = new Map<number, WorkerState>();\n\n  private started = false;\n\n  private isShuttingDown = false;\n\n  private shutdownPromise: Promise<void> | null = null;\n\n  private healthCheckInterval: NodeJS.Timeout | null = null;\n\n  private signalHandlers = new Map<NodeJS.Signals, () => void>();\n\n  constructor(options: ClusterManagerOptions = {}) {\n    super();\n    this.options = validateAndDefaultOptions(options);\n    this.initialize();\n  }\n\n  /**\n   * Initialize the cluster manager with primary process configuration\n   *\n   * @see {@link https://nodejs.org/docs/latest/api/cluster.html#clustersetupprimaryoptions Node.js cluster.setupPrimary()}\n   */\n  private initialize(): void {\n    // Only call setupPrimary in the primary process\n    if (this.options.cluster.isPrimary) {\n      this.options.cluster.setupPrimary({\n        serialization: this.options.serialization,\n      });\n    }\n    this.setupSignalHandlers();\n  }\n\n  /**\n   * Starts the cluster manager\n   * In primary process: starts workers and health monitoring\n   * In worker process: executes worker function\n   */\n  public async start(): Promise<void> {\n    if (this.options.cluster.isPrimary) {\n      await this.startPrimary();\n    } else {\n      await this.startWorker();\n    }\n    this.started = true;\n  }\n\n  /**\n   * Initiates graceful shutdown of the cluster\n   */\n  public shutdown(signal?: string): Promise<void> {\n    if (!this.started) {\n      return Promise.resolve();\n    }\n\n    if (this.shutdownPromise) {\n      return this.shutdownPromise;\n    }\n\n    this.isShuttingDown = true;\n    this.emit('shutdown:started', signal);\n\n    if (this.options.cluster.isPrimary) {\n      this.shutdownPromise = this.shutdownPrimary(signal);\n    } else {\n      this.shutdownPromise = this.shutdownWorker(signal);\n    }\n\n    return this.shutdownPromise;\n  }\n\n  /**\n   * Sets up signal handlers for graceful shutdown\n   */\n  private setupSignalHandlers(): void {\n    this.options.shutdownSignals.forEach((signal) => {\n      const handler = () => {\n        this.shutdown(signal);\n      };\n      this.signalHandlers.set(signal, handler);\n      process.on(signal, handler);\n    });\n  }\n\n  /**\n   * Removes signal handlers\n   */\n  private removeSignalHandlers(): void {\n    Array.from(this.signalHandlers.entries()).forEach(([signal, handler]) => {\n      process.removeListener(signal, handler);\n    });\n    this.signalHandlers.clear();\n  }\n\n  /**\n   * Starts the primary process\n   */\n  private async startPrimary(): Promise<void> {\n    logger.info(\n      `Primary process (pid: ${process.pid}) starting with ${this.options.numWorkers} workers`,\n    );\n\n    // Execute primary initialization function, any error will stop the cluster from starting\n    await this.options.primaryFn();\n\n    // Set up cluster event handlers\n    this.setupClusterEventHandlers();\n\n    // Spawn initial workers\n    for (let i = 0; i < this.options.numWorkers; i += 1) {\n      const workerId = i + 1; // Worker IDs start from 1\n      this.spawnWorker(workerId);\n    }\n\n    // Start health monitoring\n    this.startHealthMonitoring();\n  }\n\n  /**\n   * Starts a worker process\n   */\n  private async startWorker(): Promise<void> {\n    logger.info(`Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) starting`);\n\n    // Set up IPC message handlers\n    this.setupWorkerMessageHandlers();\n\n    // Execute worker initialization function, any error will be propagated\n    await this.options.workerFn();\n  }\n\n  /**\n   * Sets up cluster event handlers for the primary process\n   */\n  private setupClusterEventHandlers(): void {\n    this.options.cluster.on('exit', (worker: Worker, code: number, signal: string) => {\n      this.handleWorkerExit(worker, code, signal);\n    });\n\n    this.options.cluster.on('online', (worker: Worker) => {\n      logger.info(`Worker ${this.getWorkerId(worker)} (pid: ${worker.process.pid}) is online`);\n      this.emit('worker:started', worker);\n    });\n  }\n\n  /**\n   * Sets up IPC message handlers for worker processes\n   */\n  private setupWorkerMessageHandlers(): void {\n    process.on('message', (message: IPCMessage) => {\n      switch (message.type) {\n        case 'ping':\n          this.handleWorkerPing(message);\n          break;\n        case 'shutdown':\n          this.shutdown(message.signal);\n          break;\n        default:\n          if (!message.type.includes('Metrics')) {\n            logger.warn(`ignoring unknown message type in worker ${process.pid}: ${message.type}`);\n          }\n          break;\n      }\n    });\n  }\n\n  /**\n   * Spawns a new worker and sets up its state\n   */\n  private spawnWorker(id: number): Worker {\n    const worker = this.options.cluster.fork({\n      WORKER_ID: id,\n    });\n    const workerState: WorkerState = {\n      id,\n      worker,\n      restartCount: 0,\n      lastPing: Date.now(),\n      pendingPing: false,\n      isShuttingDown: false,\n    };\n\n    this.workers.set(worker.id, workerState);\n\n    // Set up worker message handler\n    worker.on('message', (message: IPCMessage) => {\n      this.handleWorkerMessage(worker, message);\n    });\n\n    return worker;\n  }\n\n  /**\n   * Handles messages from workers\n   */\n  private handleWorkerMessage(worker: Worker, message: IPCMessage): void {\n    const workerState = this.workers.get(worker.id);\n    if (!workerState) return;\n\n    switch (message.type) {\n      case 'pong':\n        workerState.lastPing = Date.now();\n        workerState.pendingPing = false;\n        break;\n      default:\n        if (!message.type.includes('Metrics')) {\n          logger.warn(\n            `Received unknown message type from worker ${this.getWorkerId(worker)} (pid: ${\n              worker.process.pid\n            }): ${message.type}`,\n          );\n        }\n        break;\n    }\n  }\n\n  /**\n   * Handles ping messages in worker processes\n   */\n  private handleWorkerPing(message: PingMessage): void {\n    const pongMessage: PongMessage = {\n      type: 'pong',\n      timestamp: message.timestamp,\n    };\n\n    if (process.send) {\n      process.send(pongMessage);\n    }\n  }\n\n  /**\n   * Starts health monitoring for all workers\n   */\n  private startHealthMonitoring(): void {\n    this.healthCheckInterval = setInterval(() => {\n      this.performHealthCheck();\n    }, this.options.pingFrequency);\n  }\n\n  /**\n   * Stops health monitoring\n   */\n  private stopHealthMonitoring(): void {\n    if (this.healthCheckInterval) {\n      clearInterval(this.healthCheckInterval);\n      this.healthCheckInterval = null;\n    }\n  }\n\n  /**\n   * Performs health check on all workers\n   */\n  private performHealthCheck(): void {\n    if (this.isShuttingDown) return;\n\n    Array.from(this.workers.values())\n      .filter((workerState) => !workerState.isShuttingDown)\n      .forEach((workerState) => {\n        const timeSinceLastPing = Date.now() - workerState.lastPing;\n\n        if (workerState.pendingPing && timeSinceLastPing > this.options.pingTimeout) {\n          // Worker is stuck, handle it\n          this.handleStuckWorker(workerState);\n        } else if (!workerState.pendingPing) {\n          // Send ping to worker\n          this.sendPingToWorker(workerState);\n        }\n      });\n  }\n\n  /**\n   * Sends a ping message to a worker\n   */\n  private sendPingToWorker(workerState: WorkerState): void {\n    const pingMessage: PingMessage = {\n      type: 'ping',\n      timestamp: Date.now(),\n    };\n\n    workerState.pendingPing = true;\n    try {\n      workerState.worker.send(pingMessage);\n    } catch (error) {\n      logger.error(\n        `Failed to send ping to worker ${workerState.id} (pid: ${\n          workerState.worker.process.pid\n        }): ${error instanceof Error ? error.message : String(error)}`,\n      );\n    }\n  }\n\n  /**\n   * Handles a stuck worker\n   */\n  private handleStuckWorker(workerState: WorkerState): void {\n    logger.error(\n      `Worker ${workerState.id} (pid: ${workerState.worker.process.pid}) is stuck, killing it`,\n    );\n    this.emit('worker:stuck', workerState.worker);\n\n    // Remove worker state\n    this.workers.delete(workerState.worker.id);\n\n    // Kill the worker\n    workerState.worker.kill('SIGKILL');\n\n    // Determine if we should spawn a replacement\n    const shouldSpawn = this.options.stuckWorkerRespawnFunc(workerState.worker);\n\n    if (shouldSpawn && !this.isShuttingDown) {\n      logger.info(\n        `Spawning replacement worker for stuck worker ${workerState.id} (pid: ${workerState.worker.process.pid})`,\n      );\n      const { restartCount } = workerState; // Preserve restart count\n      const newWorker = this.spawnWorker(workerState.id);\n      const newWorkerState = this.workers.get(newWorker.id);\n      if (newWorkerState) {\n        newWorkerState.restartCount = restartCount;\n      }\n    } else {\n      logger.error(\n        `Triggering cluster shutdown due to stuck worker ${workerState.id} (pid: ${workerState.worker.process.pid})`,\n      );\n      this.shutdown('STUCK_WORKER');\n    }\n  }\n\n  /**\n   * Handles worker exit events\n   */\n  private handleWorkerExit(worker: Worker, code: number | null, signal: string | null): void {\n    const workerId = this.getWorkerId(worker);\n    this.emit('worker:died', worker, code, signal);\n\n    const workerState = this.workers.get(worker.id);\n    if (!workerState) {\n      logger.warn(\n        `Unknown worker ${workerId} (pid: ${worker.process.pid}) exited with code ${code} and signal ${signal}`,\n      );\n      return;\n    }\n\n    this.workers.delete(worker.id);\n\n    // If we're shutting down or worker was killed intentionally, don't restart\n    if (this.isShuttingDown || workerState.isShuttingDown) {\n      logger.info(\n        `Worker ${workerId} (pid: ${worker.process.pid}) exited with code ${code} and signal ${signal}`,\n      );\n      return;\n    }\n    logger.error(\n      `Worker ${workerId} (pid: ${worker.process.pid}) exited unexpectedly with code ${code} and signal ${signal}`,\n    );\n    // Handle unexpected exit\n    this.handleUnexpectedWorkerExit(workerState);\n  }\n\n  private getWorkerId(worker: Worker): number {\n    return this.workers.get(worker.id)?.id ?? worker.id;\n  }\n\n  private getCurrentWorkerId(): number {\n    const envWorkerId =\n      process.env.WORKER_ID !== undefined ? Number(process.env.WORKER_ID) : undefined;\n    return envWorkerId ?? this.options.cluster.worker?.id ?? -1;\n  }\n\n  /**\n   * Handles unexpected worker exits with restart logic\n   */\n  private handleUnexpectedWorkerExit(workerState: WorkerState): void {\n    const restartCount = workerState.restartCount + 1;\n\n    if (restartCount <= this.options.restartMaxTimes) {\n      logger.error(\n        `Restarting worker ${workerState.id} (pid: ${workerState.worker.process.pid}) (attempt ${restartCount}/${this.options.restartMaxTimes})`,\n      );\n      const newWorker = this.spawnWorker(workerState.id);\n      const newWorkerState = this.workers.get(newWorker.id);\n      if (newWorkerState) {\n        newWorkerState.restartCount = restartCount;\n      }\n      this.emit('worker:restarted', newWorker, workerState.restartCount);\n    } else {\n      logger.error(\n        `Restart limit (${this.options.restartMaxTimes}) exceeded for worker ${workerState.id} (pid: ${workerState.worker.process.pid}), shutting down cluster`,\n      );\n      this.emit('worker:restart-limit-exceeded', workerState.worker, workerState.restartCount);\n      this.shutdown('RESTART_LIMIT_EXCEEDED');\n    }\n  }\n\n  /**\n   * Shuts down the primary process\n   */\n  private async shutdownPrimary(signal?: string): Promise<void> {\n    logger.warn(\n      `Primary process (pid: ${process.pid}) shutting down (signal: ${signal ?? 'manual'})`,\n    );\n\n    this.stopHealthMonitoring();\n    this.removeSignalHandlers();\n\n    // Shutdown all workers\n    await this.shutdownAllWorkers(signal);\n\n    // Execute primary shutdown function\n    await safeExecute(\n      () =>\n        timeout(\n          this.options.primaryShutdownFn(signal),\n          this.options.shutdownTimeout,\n          'primary shutdown timeout',\n        ),\n      'Error in primary shutdown function',\n    );\n\n    logger.info(`Primary process (pid: ${process.pid}) shutdown completed`);\n    this.emit('shutdown:completed');\n    process.exit(0);\n  }\n\n  /**\n   * Shuts down all workers gracefully\n   */\n  private async shutdownAllWorkers(signal?: string): Promise<void> {\n    if (this.workers.size === 0) return;\n\n    logger.info(`Shutting down ${this.workers.size} workers...`);\n\n    // Mark all workers as shutting down and send shutdown message\n    Array.from(this.workers.values()).forEach((workerState) => {\n      workerState.isShuttingDown = true;\n      // Only send shutdown message if worker is still connected\n      if (!workerState.worker.isDead() && workerState.worker.process.connected) {\n        const shutdownMessage: ShutdownMessage = {\n          type: 'shutdown',\n          signal,\n        };\n        try {\n          workerState.worker.send(shutdownMessage);\n        } catch (error) {\n          // Worker IPC channel is already closed, which is fine\n          logger.warn(\n            `Failed to send shutdown message to worker ${workerState.id} (pid: ${\n              workerState.worker.process.pid\n            }): ${error instanceof Error ? error.message : String(error)}`,\n          );\n        }\n      }\n    });\n\n    // Wait for workers to exit gracefully or timeout\n    try {\n      await timeout(\n        this.waitForAllWorkersToExit(),\n        this.options.shutdownTimeout,\n        'Worker shutdown timeout',\n      );\n    } catch (error) {\n      logger.error('Graceful shutdown for workers timed out, forcing shutdown');\n      this.forceKillAllWorkers();\n    }\n  }\n\n  /**\n   * Waits for all workers to exit\n   */\n  private async waitForAllWorkersToExit(): Promise<void> {\n    while (this.workers.size > 0) {\n      // eslint-disable-next-line no-await-in-loop\n      await delay(100);\n    }\n  }\n\n  /**\n   * Force kills all remaining workers\n   */\n  private forceKillAllWorkers(): void {\n    Array.from(this.workers.values()).forEach((workerState) => {\n      logger.error(\n        `Force killing worker ${workerState.id} (pid: ${workerState.worker.process.pid})`,\n      );\n      workerState.worker.kill('SIGKILL');\n    });\n    this.workers.clear();\n  }\n\n  /**\n   * Shuts down a worker process\n   */\n  private async shutdownWorker(signal?: string): Promise<void> {\n    logger.warn(\n      `Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutting down (signal: ${\n        signal ?? 'manual'\n      })`,\n    );\n\n    this.removeSignalHandlers();\n\n    // Execute worker shutdown function\n    await safeExecute(\n      () =>\n        timeout(\n          this.options.workerShutdownFn(signal),\n          this.options.shutdownTimeout,\n          'worker shutdown timeout',\n        ),\n      `Error in worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutdown function`,\n    );\n\n    logger.info(`Worker ${this.getCurrentWorkerId()} (pid: ${process.pid}) shutdown completed`);\n    process.exit(0);\n  }\n\n  /**\n   * Gets the current number of active workers\n   */\n  public getWorkerCount(): number {\n    return this.workers.size;\n  }\n\n  /**\n   * Gets information about all workers\n   */\n  public getWorkerInfo(): Array<{ id: number; pid: number; restartCount: number }> {\n    return Array.from(this.workers.values()).map((state) => ({\n      id: state.id,\n      pid: state.worker.process.pid ?? -1,\n      restartCount: state.restartCount,\n    }));\n  }\n\n  /**\n   * Checks if the cluster is currently started\n   */\n  public isStarted(): boolean {\n    return this.started;\n  }\n\n  /**\n   * Checks if the cluster is currently shutting down\n   */\n  public isShutdown(): boolean {\n    return this.isShuttingDown;\n  }\n}\n\n// Type-safe event emitter interface\nexport interface ClusterManagerEmitter {\n  on<K extends keyof ClusterManagerEvents>(event: K, listener: ClusterManagerEvents[K]): this;\n  emit<K extends keyof ClusterManagerEvents>(\n    event: K,\n    ...args: Parameters<ClusterManagerEvents[K]>\n  ): boolean;\n}\n"]}
|