@rudderstack/integrations-lib 0.2.42 → 0.2.44
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.
- package/build/cluster/manager.d.ts.map +1 -1
- package/build/cluster/manager.js +7 -4
- package/build/feature-flags/core/config.d.ts +14 -0
- package/build/feature-flags/core/config.d.ts.map +1 -0
- package/build/feature-flags/core/config.js +111 -0
- package/build/feature-flags/core/index.d.ts +4 -0
- package/build/feature-flags/core/index.d.ts.map +1 -0
- package/build/feature-flags/core/index.js +10 -0
- package/build/feature-flags/core/service.d.ts +19 -0
- package/build/feature-flags/core/service.d.ts.map +1 -0
- package/build/feature-flags/core/service.js +106 -0
- package/build/feature-flags/core/utils.d.ts +8 -0
- package/build/feature-flags/core/utils.d.ts.map +1 -0
- package/build/feature-flags/core/utils.js +40 -0
- package/build/feature-flags/flags/defaults.d.ts +3 -0
- package/build/feature-flags/flags/defaults.d.ts.map +1 -0
- package/build/feature-flags/flags/defaults.js +24 -0
- package/build/feature-flags/flags/index.d.ts +5 -0
- package/build/feature-flags/flags/index.d.ts.map +1 -0
- package/build/feature-flags/flags/index.js +10 -0
- package/build/feature-flags/flags/interfaces.d.ts +12 -0
- package/build/feature-flags/flags/interfaces.d.ts.map +1 -0
- package/build/feature-flags/flags/interfaces.js +3 -0
- package/build/feature-flags/flags/loader.d.ts +14 -0
- package/build/feature-flags/flags/loader.d.ts.map +1 -0
- package/build/feature-flags/flags/loader.js +37 -0
- package/build/feature-flags/flags/registry.d.ts +12 -0
- package/build/feature-flags/flags/registry.d.ts.map +1 -0
- package/build/feature-flags/flags/registry.js +38 -0
- package/build/feature-flags/index.d.ts +9 -0
- package/build/feature-flags/index.d.ts.map +1 -0
- package/build/feature-flags/index.js +19 -0
- package/build/feature-flags/providers/factory.d.ts +7 -0
- package/build/feature-flags/providers/factory.d.ts.map +1 -0
- package/build/feature-flags/providers/factory.js +25 -0
- package/build/feature-flags/providers/flagsmith/index.d.ts +2 -0
- package/build/feature-flags/providers/flagsmith/index.d.ts.map +1 -0
- package/build/feature-flags/providers/flagsmith/index.js +6 -0
- package/build/feature-flags/providers/flagsmith/provider.d.ts +13 -0
- package/build/feature-flags/providers/flagsmith/provider.d.ts.map +1 -0
- package/build/feature-flags/providers/flagsmith/provider.js +73 -0
- package/build/feature-flags/providers/flagsmith/types.d.ts +1 -0
- package/build/feature-flags/providers/flagsmith/types.d.ts.map +1 -0
- package/build/feature-flags/providers/flagsmith/types.js +4 -0
- package/build/feature-flags/providers/index.d.ts +5 -0
- package/build/feature-flags/providers/index.d.ts.map +1 -0
- package/build/feature-flags/providers/index.js +10 -0
- package/build/feature-flags/providers/interfaces.d.ts +11 -0
- package/build/feature-flags/providers/interfaces.d.ts.map +1 -0
- package/build/feature-flags/providers/interfaces.js +3 -0
- package/build/feature-flags/providers/local/config.d.ts +4 -0
- package/build/feature-flags/providers/local/config.d.ts.map +1 -0
- package/build/feature-flags/providers/local/config.js +11 -0
- package/build/feature-flags/providers/local/index.d.ts +3 -0
- package/build/feature-flags/providers/local/index.d.ts.map +1 -0
- package/build/feature-flags/providers/local/index.js +6 -0
- package/build/feature-flags/providers/local/provider.d.ts +17 -0
- package/build/feature-flags/providers/local/provider.d.ts.map +1 -0
- package/build/feature-flags/providers/local/provider.js +105 -0
- package/build/feature-flags/providers/local/types.d.ts +4 -0
- package/build/feature-flags/providers/local/types.d.ts.map +1 -0
- package/build/feature-flags/providers/local/types.js +3 -0
- package/build/feature-flags/types.d.ts +96 -0
- package/build/feature-flags/types.d.ts.map +1 -0
- package/build/feature-flags/types.js +11 -0
- package/build/index.d.ts +1 -0
- package/build/index.d.ts.map +1 -1
- package/build/index.js +2 -1
- package/package.json +3 -2
|
@@ -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;
|
|
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;IAsBxB,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
|
@@ -65,9 +65,12 @@ class ClusterManager extends events_1.EventEmitter {
|
|
|
65
65
|
* @see {@link https://nodejs.org/docs/latest/api/cluster.html#clustersetupprimaryoptions Node.js cluster.setupPrimary()}
|
|
66
66
|
*/
|
|
67
67
|
initialize() {
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
68
|
+
// Only call setupPrimary in the primary process
|
|
69
|
+
if (this.options.cluster.isPrimary) {
|
|
70
|
+
this.options.cluster.setupPrimary({
|
|
71
|
+
serialization: this.options.serialization,
|
|
72
|
+
});
|
|
73
|
+
}
|
|
71
74
|
this.setupSignalHandlers();
|
|
72
75
|
}
|
|
73
76
|
/**
|
|
@@ -469,4 +472,4 @@ class ClusterManager extends events_1.EventEmitter {
|
|
|
469
472
|
}
|
|
470
473
|
}
|
|
471
474
|
exports.ClusterManager = ClusterManager;
|
|
472
|
-
//# 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    this.options.cluster.setupPrimary({\n      serialization: this.options.serialization,\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"]}
|
|
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"]}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import type { FeatureFlagConfig, ResolvedFeatureFlagConfig } from '../types';
|
|
2
|
+
export declare class ConfigResolver {
|
|
3
|
+
static resolveConfig(config: Partial<FeatureFlagConfig>): ResolvedFeatureFlagConfig;
|
|
4
|
+
static initializeFlagsmithSDK(config: ResolvedFeatureFlagConfig): void;
|
|
5
|
+
private static getConfigValue;
|
|
6
|
+
private static parseBoolean;
|
|
7
|
+
private static parseNumber;
|
|
8
|
+
static parseEnvKey(envKey: string): {
|
|
9
|
+
prefix: string;
|
|
10
|
+
workspace?: string;
|
|
11
|
+
flag?: string;
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
//# sourceMappingURL=config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../../src/feature-flags/core/config.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,iBAAiB,EAAE,yBAAyB,EAAE,MAAM,UAAU,CAAC;AAE7E,qBAAa,cAAc;IACzB,MAAM,CAAC,aAAa,CAAC,MAAM,EAAE,OAAO,CAAC,iBAAiB,CAAC,GAAG,yBAAyB;IA2CnF,MAAM,CAAC,sBAAsB,CAAC,MAAM,EAAE,yBAAyB,GAAG,IAAI;IAuBtE,OAAO,CAAC,MAAM,CAAC,cAAc;IAsB7B,OAAO,CAAC,MAAM,CAAC,YAAY;IAW3B,OAAO,CAAC,MAAM,CAAC,WAAW;IAY1B,MAAM,CAAC,WAAW,CAAC,MAAM,EAAE,MAAM,GAAG;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,SAAS,CAAC,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE;CA4C1F"}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ConfigResolver = void 0;
|
|
4
|
+
const featureflag_sdk_node_1 = require("@rudderstack/featureflag-sdk-node");
|
|
5
|
+
class ConfigResolver {
|
|
6
|
+
static resolveConfig(config) {
|
|
7
|
+
return {
|
|
8
|
+
provider: this.getConfigValue(config.provider, 'FEATURE_FLAG_PROVIDER', 'local'),
|
|
9
|
+
apiKey: this.getConfigValue(config.apiKey, 'FEATURE_FLAG_API_KEY', undefined),
|
|
10
|
+
enableLocalEvaluation: this.getConfigValue(config.enableLocalEvaluation, 'FEATURE_FLAG_ENABLE_LOCAL_EVALUATION', false, this.parseBoolean),
|
|
11
|
+
enableCache: this.getConfigValue(config.enableCache, 'FEATURE_FLAG_ENABLE_CACHE', true, this.parseBoolean),
|
|
12
|
+
cacheTtlSeconds: this.getConfigValue(config.cacheTtlSeconds, 'FEATURE_FLAG_CACHE_TTL_SECONDS', 60, this.parseNumber),
|
|
13
|
+
timeoutSeconds: this.getConfigValue(config.timeoutSeconds, 'FEATURE_FLAG_TIMEOUT_SECONDS', 60, this.parseNumber),
|
|
14
|
+
retryAttempts: this.getConfigValue(config.retryAttempts, 'FEATURE_FLAG_RETRY_ATTEMPTS', 3, this.parseNumber),
|
|
15
|
+
enableAnalytics: this.getConfigValue(config.enableAnalytics, 'FEATURE_FLAG_ENABLE_ANALYTICS', true, this.parseBoolean),
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
static initializeFlagsmithSDK(config) {
|
|
19
|
+
if (config.provider !== 'flagsmith' || !config.apiKey) {
|
|
20
|
+
throw new Error('API key is required for Flagsmith provider. Set it via config.apiKey or FEATURE_FLAG_API_KEY environment variable.');
|
|
21
|
+
}
|
|
22
|
+
(0, featureflag_sdk_node_1.initFeatureFlagClient)({
|
|
23
|
+
provider: {
|
|
24
|
+
type: 'flagsmith',
|
|
25
|
+
apiKey: config.apiKey,
|
|
26
|
+
timeoutInSeconds: config.timeoutSeconds,
|
|
27
|
+
retryAttempts: config.retryAttempts,
|
|
28
|
+
enableLocalEvaluation: config.enableLocalEvaluation,
|
|
29
|
+
enableAnalytics: config.enableAnalytics,
|
|
30
|
+
},
|
|
31
|
+
cache: {
|
|
32
|
+
enabled: config.enableCache,
|
|
33
|
+
ttlInSeconds: config.cacheTtlSeconds,
|
|
34
|
+
},
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
static getConfigValue(configValue, envKey, defaultValue, parser) {
|
|
38
|
+
const envValue = process.env[envKey];
|
|
39
|
+
if (envValue !== undefined) {
|
|
40
|
+
if (parser) {
|
|
41
|
+
try {
|
|
42
|
+
return parser(envValue);
|
|
43
|
+
}
|
|
44
|
+
catch {
|
|
45
|
+
// Fall back to config/default on parse error
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
return envValue;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
return configValue !== undefined ? configValue : defaultValue;
|
|
53
|
+
}
|
|
54
|
+
static parseBoolean(value) {
|
|
55
|
+
const lowerValue = value.toLowerCase().trim();
|
|
56
|
+
if (lowerValue === 'true' || lowerValue === '1' || lowerValue === 'yes') {
|
|
57
|
+
return true;
|
|
58
|
+
}
|
|
59
|
+
if (lowerValue === 'false' || lowerValue === '0' || lowerValue === 'no') {
|
|
60
|
+
return false;
|
|
61
|
+
}
|
|
62
|
+
throw new Error(`Invalid boolean value: "${value}"`);
|
|
63
|
+
}
|
|
64
|
+
static parseNumber(value) {
|
|
65
|
+
const numValue = Number(value.trim());
|
|
66
|
+
if (Number.isNaN(numValue)) {
|
|
67
|
+
throw new Error(`Invalid number value: "${value}"`);
|
|
68
|
+
}
|
|
69
|
+
return numValue;
|
|
70
|
+
}
|
|
71
|
+
// Utility for parsing environment variables
|
|
72
|
+
// Supports both patterns:
|
|
73
|
+
// - FEATURE_FLAG_LOCAL_ENABLE_FEATURE (global flag with single underscore)
|
|
74
|
+
// - FEATURE_FLAG_LOCAL__workspace123__API__TIMEOUT__MS (workspace flag with double underscore)
|
|
75
|
+
static parseEnvKey(envKey) {
|
|
76
|
+
// Check for workspace-specific pattern first (contains double underscore)
|
|
77
|
+
if (envKey.includes('__')) {
|
|
78
|
+
const parts = envKey.split('__');
|
|
79
|
+
if (parts.length < 3) {
|
|
80
|
+
throw new Error(`Invalid workspace environment key format: "${envKey}". Expected format: PREFIX__WORKSPACE__FLAG`);
|
|
81
|
+
}
|
|
82
|
+
const prefix = parts[0];
|
|
83
|
+
const workspace = parts[1];
|
|
84
|
+
const flag = parts.slice(2).join('__');
|
|
85
|
+
return {
|
|
86
|
+
prefix,
|
|
87
|
+
workspace,
|
|
88
|
+
flag,
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
// Global pattern (single underscore)
|
|
92
|
+
const underscoreIndex = envKey.indexOf('_');
|
|
93
|
+
if (underscoreIndex === -1) {
|
|
94
|
+
throw new Error(`Invalid environment key format: "${envKey}". Expected format: PREFIX_FLAG or PREFIX__WORKSPACE__FLAG`);
|
|
95
|
+
}
|
|
96
|
+
// Find the last part after the prefix
|
|
97
|
+
const prefixParts = envKey.split('_');
|
|
98
|
+
if (prefixParts.length < 4) {
|
|
99
|
+
throw new Error(`Invalid global environment key format: "${envKey}". Expected format: FEATURE_FLAG_LOCAL_FLAG`);
|
|
100
|
+
}
|
|
101
|
+
// For FEATURE_FLAG_LOCAL_FLAG_NAME, prefix is "FEATURE_FLAG_LOCAL", flag is "FLAG_NAME"
|
|
102
|
+
const prefix = prefixParts.slice(0, 3).join('_'); // FEATURE_FLAG_LOCAL
|
|
103
|
+
const flag = prefixParts.slice(3).join('_'); // Everything after
|
|
104
|
+
return {
|
|
105
|
+
prefix,
|
|
106
|
+
flag,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
exports.ConfigResolver = ConfigResolver;
|
|
111
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../src/feature-flags/core/config.ts"],"names":[],"mappings":"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","sourcesContent":["import { initFeatureFlagClient } from '@rudderstack/featureflag-sdk-node';\nimport type { FeatureFlagConfig, ResolvedFeatureFlagConfig } from '../types';\n\nexport class ConfigResolver {\n  static resolveConfig(config: Partial<FeatureFlagConfig>): ResolvedFeatureFlagConfig {\n    return {\n      provider: this.getConfigValue(config.provider, 'FEATURE_FLAG_PROVIDER', 'local'),\n      apiKey: this.getConfigValue(config.apiKey, 'FEATURE_FLAG_API_KEY', undefined),\n      enableLocalEvaluation: this.getConfigValue(\n        config.enableLocalEvaluation,\n        'FEATURE_FLAG_ENABLE_LOCAL_EVALUATION',\n        false,\n        this.parseBoolean,\n      ),\n      enableCache: this.getConfigValue(\n        config.enableCache,\n        'FEATURE_FLAG_ENABLE_CACHE',\n        true,\n        this.parseBoolean,\n      ),\n      cacheTtlSeconds: this.getConfigValue(\n        config.cacheTtlSeconds,\n        'FEATURE_FLAG_CACHE_TTL_SECONDS',\n        60,\n        this.parseNumber,\n      ),\n      timeoutSeconds: this.getConfigValue(\n        config.timeoutSeconds,\n        'FEATURE_FLAG_TIMEOUT_SECONDS',\n        60,\n        this.parseNumber,\n      ),\n      retryAttempts: this.getConfigValue(\n        config.retryAttempts,\n        'FEATURE_FLAG_RETRY_ATTEMPTS',\n        3,\n        this.parseNumber,\n      ),\n      enableAnalytics: this.getConfigValue(\n        config.enableAnalytics,\n        'FEATURE_FLAG_ENABLE_ANALYTICS',\n        true,\n        this.parseBoolean,\n      ),\n    };\n  }\n\n  static initializeFlagsmithSDK(config: ResolvedFeatureFlagConfig): void {\n    if (config.provider !== 'flagsmith' || !config.apiKey) {\n      throw new Error(\n        'API key is required for Flagsmith provider. Set it via config.apiKey or FEATURE_FLAG_API_KEY environment variable.',\n      );\n    }\n\n    initFeatureFlagClient({\n      provider: {\n        type: 'flagsmith',\n        apiKey: config.apiKey,\n        timeoutInSeconds: config.timeoutSeconds,\n        retryAttempts: config.retryAttempts,\n        enableLocalEvaluation: config.enableLocalEvaluation,\n        enableAnalytics: config.enableAnalytics,\n      },\n      cache: {\n        enabled: config.enableCache,\n        ttlInSeconds: config.cacheTtlSeconds,\n      },\n    });\n  }\n\n  private static getConfigValue<T>(\n    configValue: T | undefined,\n    envKey: string,\n    defaultValue: T,\n    parser?: (value: string) => T,\n  ): T {\n    const envValue = process.env[envKey];\n    if (envValue !== undefined) {\n      if (parser) {\n        try {\n          return parser(envValue);\n        } catch {\n          // Fall back to config/default on parse error\n        }\n      } else {\n        return envValue as T;\n      }\n    }\n\n    return configValue !== undefined ? configValue : defaultValue;\n  }\n\n  private static parseBoolean(value: string): boolean {\n    const lowerValue = value.toLowerCase().trim();\n    if (lowerValue === 'true' || lowerValue === '1' || lowerValue === 'yes') {\n      return true;\n    }\n    if (lowerValue === 'false' || lowerValue === '0' || lowerValue === 'no') {\n      return false;\n    }\n    throw new Error(`Invalid boolean value: \"${value}\"`);\n  }\n\n  private static parseNumber(value: string): number {\n    const numValue = Number(value.trim());\n    if (Number.isNaN(numValue)) {\n      throw new Error(`Invalid number value: \"${value}\"`);\n    }\n    return numValue;\n  }\n\n  // Utility for parsing environment variables\n  // Supports both patterns:\n  // - FEATURE_FLAG_LOCAL_ENABLE_FEATURE (global flag with single underscore)\n  // - FEATURE_FLAG_LOCAL__workspace123__API__TIMEOUT__MS (workspace flag with double underscore)\n  static parseEnvKey(envKey: string): { prefix: string; workspace?: string; flag?: string } {\n    // Check for workspace-specific pattern first (contains double underscore)\n    if (envKey.includes('__')) {\n      const parts = envKey.split('__');\n      if (parts.length < 3) {\n        throw new Error(\n          `Invalid workspace environment key format: \"${envKey}\". Expected format: PREFIX__WORKSPACE__FLAG`,\n        );\n      }\n      const prefix = parts[0];\n      const workspace = parts[1];\n      const flag = parts.slice(2).join('__');\n      return {\n        prefix,\n        workspace,\n        flag,\n      };\n    }\n\n    // Global pattern (single underscore)\n    const underscoreIndex = envKey.indexOf('_');\n    if (underscoreIndex === -1) {\n      throw new Error(\n        `Invalid environment key format: \"${envKey}\". Expected format: PREFIX_FLAG or PREFIX__WORKSPACE__FLAG`,\n      );\n    }\n\n    // Find the last part after the prefix\n    const prefixParts = envKey.split('_');\n    if (prefixParts.length < 4) {\n      throw new Error(\n        `Invalid global environment key format: \"${envKey}\". Expected format: FEATURE_FLAG_LOCAL_FLAG`,\n      );\n    }\n\n    // For FEATURE_FLAG_LOCAL_FLAG_NAME, prefix is \"FEATURE_FLAG_LOCAL\", flag is \"FLAG_NAME\"\n    const prefix = prefixParts.slice(0, 3).join('_'); // FEATURE_FLAG_LOCAL\n    const flag = prefixParts.slice(3).join('_'); // Everything after\n\n    return {\n      prefix,\n      flag,\n    };\n  }\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/feature-flags/core/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,WAAW,CAAC;AAC/C,OAAO,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAC1C,OAAO,EAAE,gBAAgB,EAAE,MAAM,SAAS,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.FeatureFlagUtils = exports.ConfigResolver = exports.FeatureFlagService = void 0;
|
|
4
|
+
var service_1 = require("./service");
|
|
5
|
+
Object.defineProperty(exports, "FeatureFlagService", { enumerable: true, get: function () { return service_1.FeatureFlagService; } });
|
|
6
|
+
var config_1 = require("./config");
|
|
7
|
+
Object.defineProperty(exports, "ConfigResolver", { enumerable: true, get: function () { return config_1.ConfigResolver; } });
|
|
8
|
+
var utils_1 = require("./utils");
|
|
9
|
+
Object.defineProperty(exports, "FeatureFlagUtils", { enumerable: true, get: function () { return utils_1.FeatureFlagUtils; } });
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZmVhdHVyZS1mbGFncy9jb3JlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHFDQUErQztBQUF0Qyw2R0FBQSxrQkFBa0IsT0FBQTtBQUMzQixtQ0FBMEM7QUFBakMsd0dBQUEsY0FBYyxPQUFBO0FBQ3ZCLGlDQUEyQztBQUFsQyx5R0FBQSxnQkFBZ0IsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IEZlYXR1cmVGbGFnU2VydmljZSB9IGZyb20gJy4vc2VydmljZSc7XG5leHBvcnQgeyBDb25maWdSZXNvbHZlciB9IGZyb20gJy4vY29uZmlnJztcbmV4cG9ydCB7IEZlYXR1cmVGbGFnVXRpbHMgfSBmcm9tICcuL3V0aWxzJztcbiJdfQ==
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { FeatureFlagConfig, FeatureFlagUser, FeatureFlagResponse, FeatureFlagDefinition, IFeatureFlagService, IFeatureFlagRegistry } from '../types';
|
|
2
|
+
import { ErrorBehaviour } from '../types';
|
|
3
|
+
export declare class FeatureFlagService implements IFeatureFlagService {
|
|
4
|
+
private provider;
|
|
5
|
+
private flagRegistry;
|
|
6
|
+
constructor(registry?: IFeatureFlagRegistry);
|
|
7
|
+
static create(config?: Partial<FeatureFlagConfig>, flags?: FeatureFlagDefinition[]): Promise<FeatureFlagService>;
|
|
8
|
+
static create(config?: Partial<FeatureFlagConfig>, registry?: IFeatureFlagRegistry): Promise<FeatureFlagService>;
|
|
9
|
+
initialize(config?: Partial<FeatureFlagConfig>): Promise<void>;
|
|
10
|
+
isFeatureEnabled(user: FeatureFlagUser, flagName: string, onErrorBehaviour?: ErrorBehaviour): Promise<FeatureFlagResponse>;
|
|
11
|
+
isFeatureEnabledLatest(user: FeatureFlagUser, flagName: string, onErrorBehaviour?: ErrorBehaviour): Promise<FeatureFlagResponse>;
|
|
12
|
+
getFeatureValue(user: FeatureFlagUser, flagName: string, onErrorBehaviour?: ErrorBehaviour): Promise<FeatureFlagResponse>;
|
|
13
|
+
getFeatureValueLatest(user: FeatureFlagUser, flagName: string, onErrorBehaviour?: ErrorBehaviour): Promise<FeatureFlagResponse>;
|
|
14
|
+
registerFlags(flags: FeatureFlagDefinition[]): void;
|
|
15
|
+
getRegisteredFlag(key: string): FeatureFlagDefinition | undefined;
|
|
16
|
+
private processProviderResult;
|
|
17
|
+
private createDefaultResponse;
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=service.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"service.d.ts","sourceRoot":"","sources":["../../../src/feature-flags/core/service.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,iBAAiB,EACjB,eAAe,EACf,mBAAmB,EACnB,qBAAqB,EACrB,mBAAmB,EAEnB,oBAAoB,EAErB,MAAM,UAAU,CAAC;AAClB,OAAO,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAM1C,qBAAa,kBAAmB,YAAW,mBAAmB;IAC5D,OAAO,CAAC,QAAQ,CAAqC;IAErD,OAAO,CAAC,YAAY,CAAuB;gBAE/B,QAAQ,CAAC,EAAE,oBAAoB;WAOvB,MAAM,CACxB,MAAM,CAAC,EAAE,OAAO,CAAC,iBAAiB,CAAC,EACnC,KAAK,CAAC,EAAE,qBAAqB,EAAE,GAC9B,OAAO,CAAC,kBAAkB,CAAC;WACV,MAAM,CACxB,MAAM,CAAC,EAAE,OAAO,CAAC,iBAAiB,CAAC,EACnC,QAAQ,CAAC,EAAE,oBAAoB,GAC9B,OAAO,CAAC,kBAAkB,CAAC;IAyBjB,UAAU,CAAC,MAAM,GAAE,OAAO,CAAC,iBAAiB,CAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAWlE,gBAAgB,CAC3B,IAAI,EAAE,eAAe,EACrB,QAAQ,EAAE,MAAM,EAChB,gBAAgB,GAAE,cAAiD,GAClE,OAAO,CAAC,mBAAmB,CAAC;IAKlB,sBAAsB,CACjC,IAAI,EAAE,eAAe,EACrB,QAAQ,EAAE,MAAM,EAChB,gBAAgB,GAAE,cAAiD,GAClE,OAAO,CAAC,mBAAmB,CAAC;IAKlB,eAAe,CAC1B,IAAI,EAAE,eAAe,EACrB,QAAQ,EAAE,MAAM,EAChB,gBAAgB,GAAE,cAAiD,GAClE,OAAO,CAAC,mBAAmB,CAAC;IAKlB,qBAAqB,CAChC,IAAI,EAAE,eAAe,EACrB,QAAQ,EAAE,MAAM,EAChB,gBAAgB,GAAE,cAAiD,GAClE,OAAO,CAAC,mBAAmB,CAAC;IAMxB,aAAa,CAAC,KAAK,EAAE,qBAAqB,EAAE,GAAG,IAAI;IAInD,iBAAiB,CAAC,GAAG,EAAE,MAAM,GAAG,qBAAqB,GAAG,SAAS;IAIxE,OAAO,CAAC,qBAAqB;IAqC7B,OAAO,CAAC,qBAAqB;CAU9B"}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.FeatureFlagService = void 0;
|
|
4
|
+
const types_1 = require("../types");
|
|
5
|
+
const registry_1 = require("../flags/registry");
|
|
6
|
+
const factory_1 = require("../providers/factory");
|
|
7
|
+
const config_1 = require("./config");
|
|
8
|
+
const defaults_1 = require("../flags/defaults");
|
|
9
|
+
class FeatureFlagService {
|
|
10
|
+
constructor(registry) {
|
|
11
|
+
this.provider = null;
|
|
12
|
+
this.flagRegistry = registry || new registry_1.FeatureFlagRegistry();
|
|
13
|
+
// Load default flags
|
|
14
|
+
this.flagRegistry.register(defaults_1.DEFAULT_FLAGS);
|
|
15
|
+
}
|
|
16
|
+
static async create(config = {}, flagsOrRegistry) {
|
|
17
|
+
let registry;
|
|
18
|
+
if (flagsOrRegistry) {
|
|
19
|
+
// Check if it's a registry (has register method) or flag definitions array
|
|
20
|
+
if (Array.isArray(flagsOrRegistry)) {
|
|
21
|
+
// It's an array of flag definitions
|
|
22
|
+
registry = new registry_1.FeatureFlagRegistry();
|
|
23
|
+
registry.register(flagsOrRegistry);
|
|
24
|
+
}
|
|
25
|
+
else {
|
|
26
|
+
// It's a registry instance
|
|
27
|
+
registry = flagsOrRegistry;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
const service = new FeatureFlagService(registry);
|
|
31
|
+
await service.initialize(config);
|
|
32
|
+
return service;
|
|
33
|
+
}
|
|
34
|
+
// Strategy Pattern: Initialize provider once, use polymorphism
|
|
35
|
+
async initialize(config = {}) {
|
|
36
|
+
const resolvedConfig = config_1.ConfigResolver.resolveConfig(config);
|
|
37
|
+
// Factory creates the appropriate provider - NO CONDITIONALS
|
|
38
|
+
const factory = new factory_1.FeatureFlagProviderFactory(this.flagRegistry);
|
|
39
|
+
this.provider = factory.create(resolvedConfig);
|
|
40
|
+
// Initialize the provider
|
|
41
|
+
await this.provider.initialize();
|
|
42
|
+
}
|
|
43
|
+
async isFeatureEnabled(user, flagName, onErrorBehaviour = types_1.ErrorBehaviour.RETURN_RICH_ERROR) {
|
|
44
|
+
const providerResult = await this.provider.isFeatureEnabled(user, flagName);
|
|
45
|
+
return this.processProviderResult(providerResult, flagName, onErrorBehaviour);
|
|
46
|
+
}
|
|
47
|
+
async isFeatureEnabledLatest(user, flagName, onErrorBehaviour = types_1.ErrorBehaviour.RETURN_RICH_ERROR) {
|
|
48
|
+
const providerResult = await this.provider.isFeatureEnabledLatest(user, flagName);
|
|
49
|
+
return this.processProviderResult(providerResult, flagName, onErrorBehaviour);
|
|
50
|
+
}
|
|
51
|
+
async getFeatureValue(user, flagName, onErrorBehaviour = types_1.ErrorBehaviour.RETURN_RICH_ERROR) {
|
|
52
|
+
const providerResult = await this.provider.getFeatureValue(user, flagName);
|
|
53
|
+
return this.processProviderResult(providerResult, flagName, onErrorBehaviour);
|
|
54
|
+
}
|
|
55
|
+
async getFeatureValueLatest(user, flagName, onErrorBehaviour = types_1.ErrorBehaviour.RETURN_RICH_ERROR) {
|
|
56
|
+
const providerResult = await this.provider.getFeatureValueLatest(user, flagName);
|
|
57
|
+
return this.processProviderResult(providerResult, flagName, onErrorBehaviour);
|
|
58
|
+
}
|
|
59
|
+
// Registry methods
|
|
60
|
+
registerFlags(flags) {
|
|
61
|
+
this.flagRegistry.register(flags);
|
|
62
|
+
}
|
|
63
|
+
getRegisteredFlag(key) {
|
|
64
|
+
return this.flagRegistry.get(key);
|
|
65
|
+
}
|
|
66
|
+
processProviderResult(providerResult, flagName, onErrorBehaviour) {
|
|
67
|
+
if (providerResult.error) {
|
|
68
|
+
switch (onErrorBehaviour) {
|
|
69
|
+
case types_1.ErrorBehaviour.THROW_ERROR:
|
|
70
|
+
throw providerResult.error;
|
|
71
|
+
case types_1.ErrorBehaviour.RETURN_RICH_ERROR:
|
|
72
|
+
// Return the rich error response as-is with isDefault = false
|
|
73
|
+
return {
|
|
74
|
+
...providerResult,
|
|
75
|
+
isDefault: false,
|
|
76
|
+
};
|
|
77
|
+
case types_1.ErrorBehaviour.RETURN_DEFAULT:
|
|
78
|
+
// Return default value from registry
|
|
79
|
+
return this.createDefaultResponse(flagName, providerResult);
|
|
80
|
+
default:
|
|
81
|
+
// Default to RETURN_RICH_ERROR behavior
|
|
82
|
+
return {
|
|
83
|
+
...providerResult,
|
|
84
|
+
isDefault: false,
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
// Success case - pass through rich provider data
|
|
89
|
+
return {
|
|
90
|
+
...providerResult,
|
|
91
|
+
isDefault: false,
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
createDefaultResponse(flagName, errorResult) {
|
|
95
|
+
const defaultValue = this.flagRegistry.getDefaultValue(flagName);
|
|
96
|
+
return {
|
|
97
|
+
name: flagName,
|
|
98
|
+
enabled: typeof defaultValue === 'boolean' ? defaultValue : Boolean(defaultValue),
|
|
99
|
+
value: defaultValue,
|
|
100
|
+
error: errorResult.error,
|
|
101
|
+
isDefault: true,
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
exports.FeatureFlagService = FeatureFlagService;
|
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../../../src/feature-flags/core/service.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  FeatureFlagConfig,\n  FeatureFlagUser,\n  FeatureFlagResponse,\n  FeatureFlagDefinition,\n  IFeatureFlagService,\n  IFeatureFlagProvider,\n  IFeatureFlagRegistry,\n  FeatureValue,\n} from '../types';\nimport { ErrorBehaviour } from '../types';\nimport { FeatureFlagRegistry } from '../flags/registry';\nimport { FeatureFlagProviderFactory } from '../providers/factory';\nimport { ConfigResolver } from './config';\nimport { DEFAULT_FLAGS } from '../flags/defaults';\n\nexport class FeatureFlagService implements IFeatureFlagService {\n  private provider: IFeatureFlagProvider | null = null;\n\n  private flagRegistry: IFeatureFlagRegistry;\n\n  constructor(registry?: IFeatureFlagRegistry) {\n    this.flagRegistry = registry || new FeatureFlagRegistry();\n    // Load default flags\n    this.flagRegistry.register(DEFAULT_FLAGS);\n  }\n\n  // Factory method overloads: Create and initialize service in one step\n  public static async create(\n    config?: Partial<FeatureFlagConfig>,\n    flags?: FeatureFlagDefinition[],\n  ): Promise<FeatureFlagService>;\n  public static async create(\n    config?: Partial<FeatureFlagConfig>,\n    registry?: IFeatureFlagRegistry,\n  ): Promise<FeatureFlagService>;\n  public static async create(\n    config: Partial<FeatureFlagConfig> = {},\n    flagsOrRegistry?: FeatureFlagDefinition[] | IFeatureFlagRegistry,\n  ): Promise<FeatureFlagService> {\n    let registry: IFeatureFlagRegistry | undefined;\n\n    if (flagsOrRegistry) {\n      // Check if it's a registry (has register method) or flag definitions array\n      if (Array.isArray(flagsOrRegistry)) {\n        // It's an array of flag definitions\n        registry = new FeatureFlagRegistry();\n        registry.register(flagsOrRegistry);\n      } else {\n        // It's a registry instance\n        registry = flagsOrRegistry;\n      }\n    }\n\n    const service = new FeatureFlagService(registry);\n    await service.initialize(config);\n    return service;\n  }\n\n  // Strategy Pattern: Initialize provider once, use polymorphism\n  public async initialize(config: Partial<FeatureFlagConfig> = {}): Promise<void> {\n    const resolvedConfig = ConfigResolver.resolveConfig(config);\n\n    // Factory creates the appropriate provider - NO CONDITIONALS\n    const factory = new FeatureFlagProviderFactory(this.flagRegistry);\n    this.provider = factory.create(resolvedConfig);\n\n    // Initialize the provider\n    await this.provider.initialize();\n  }\n\n  public async isFeatureEnabled(\n    user: FeatureFlagUser,\n    flagName: string,\n    onErrorBehaviour: ErrorBehaviour = ErrorBehaviour.RETURN_RICH_ERROR,\n  ): Promise<FeatureFlagResponse> {\n    const providerResult = await this.provider!.isFeatureEnabled(user, flagName);\n    return this.processProviderResult(providerResult, flagName, onErrorBehaviour);\n  }\n\n  public async isFeatureEnabledLatest(\n    user: FeatureFlagUser,\n    flagName: string,\n    onErrorBehaviour: ErrorBehaviour = ErrorBehaviour.RETURN_RICH_ERROR,\n  ): Promise<FeatureFlagResponse> {\n    const providerResult = await this.provider!.isFeatureEnabledLatest(user, flagName);\n    return this.processProviderResult(providerResult, flagName, onErrorBehaviour);\n  }\n\n  public async getFeatureValue(\n    user: FeatureFlagUser,\n    flagName: string,\n    onErrorBehaviour: ErrorBehaviour = ErrorBehaviour.RETURN_RICH_ERROR,\n  ): Promise<FeatureFlagResponse> {\n    const providerResult = await this.provider!.getFeatureValue(user, flagName);\n    return this.processProviderResult(providerResult, flagName, onErrorBehaviour);\n  }\n\n  public async getFeatureValueLatest(\n    user: FeatureFlagUser,\n    flagName: string,\n    onErrorBehaviour: ErrorBehaviour = ErrorBehaviour.RETURN_RICH_ERROR,\n  ): Promise<FeatureFlagResponse> {\n    const providerResult = await this.provider!.getFeatureValueLatest(user, flagName);\n    return this.processProviderResult(providerResult, flagName, onErrorBehaviour);\n  }\n\n  // Registry methods\n  public registerFlags(flags: FeatureFlagDefinition[]): void {\n    this.flagRegistry.register(flags);\n  }\n\n  public getRegisteredFlag(key: string): FeatureFlagDefinition | undefined {\n    return this.flagRegistry.get(key);\n  }\n\n  private processProviderResult(\n    providerResult: FeatureValue,\n    flagName: string,\n    onErrorBehaviour: ErrorBehaviour,\n  ): FeatureFlagResponse {\n    if (providerResult.error) {\n      switch (onErrorBehaviour) {\n        case ErrorBehaviour.THROW_ERROR:\n          throw providerResult.error;\n\n        case ErrorBehaviour.RETURN_RICH_ERROR:\n          // Return the rich error response as-is with isDefault = false\n          return {\n            ...providerResult,\n            isDefault: false,\n          };\n\n        case ErrorBehaviour.RETURN_DEFAULT:\n          // Return default value from registry\n          return this.createDefaultResponse(flagName, providerResult);\n\n        default:\n          // Default to RETURN_RICH_ERROR behavior\n          return {\n            ...providerResult,\n            isDefault: false,\n          };\n      }\n    }\n\n    // Success case - pass through rich provider data\n    return {\n      ...providerResult,\n      isDefault: false,\n    };\n  }\n\n  private createDefaultResponse(flagName: string, errorResult: FeatureValue): FeatureFlagResponse {\n    const defaultValue = this.flagRegistry.getDefaultValue(flagName);\n    return {\n      name: flagName,\n      enabled: typeof defaultValue === 'boolean' ? defaultValue : Boolean(defaultValue),\n      value: defaultValue,\n      error: errorResult.error,\n      isDefault: true,\n    };\n  }\n}\n"]}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { FeatureFlagDefinition } from '../types';
|
|
2
|
+
export declare class FeatureFlagUtils {
|
|
3
|
+
static normalizeKey(key: string): string;
|
|
4
|
+
static validateDefinition(definition: FeatureFlagDefinition): void;
|
|
5
|
+
static createWorkspaceEnvKey(flagName: string, workspaceId: string): string;
|
|
6
|
+
static createGlobalEnvKey(flagName: string): string;
|
|
7
|
+
}
|
|
8
|
+
//# sourceMappingURL=utils.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/feature-flags/core/utils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,UAAU,CAAC;AAEtD,qBAAa,gBAAgB;IAC3B,MAAM,CAAC,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM;IAIxC,MAAM,CAAC,kBAAkB,CAAC,UAAU,EAAE,qBAAqB,GAAG,IAAI;IA0BlE,MAAM,CAAC,qBAAqB,CAAC,QAAQ,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,GAAG,MAAM;IAK3E,MAAM,CAAC,kBAAkB,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM;CAIpD"}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.FeatureFlagUtils = void 0;
|
|
4
|
+
class FeatureFlagUtils {
|
|
5
|
+
static normalizeKey(key) {
|
|
6
|
+
return key.toLowerCase().replace(/[^a-z0-9-_]/g, '-');
|
|
7
|
+
}
|
|
8
|
+
static validateDefinition(definition) {
|
|
9
|
+
if (!definition.key || typeof definition.key !== 'string') {
|
|
10
|
+
throw new Error('Flag definition must have a valid key');
|
|
11
|
+
}
|
|
12
|
+
if (!definition.name || typeof definition.name !== 'string') {
|
|
13
|
+
throw new Error('Flag definition must have a valid name');
|
|
14
|
+
}
|
|
15
|
+
if (!definition.description || typeof definition.description !== 'string') {
|
|
16
|
+
throw new Error('Flag definition must have a valid description');
|
|
17
|
+
}
|
|
18
|
+
if (definition.defaultValue === undefined) {
|
|
19
|
+
throw new Error('Flag definition must have a defaultValue');
|
|
20
|
+
}
|
|
21
|
+
if (!['boolean', 'string', 'number'].includes(definition.type)) {
|
|
22
|
+
throw new Error('Flag definition type must be boolean, string, or number');
|
|
23
|
+
}
|
|
24
|
+
// Validate defaultValue matches type
|
|
25
|
+
const actualType = typeof definition.defaultValue;
|
|
26
|
+
if (actualType !== definition.type) {
|
|
27
|
+
throw new Error(`Flag defaultValue type (${actualType}) does not match declared type (${definition.type})`);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
static createWorkspaceEnvKey(flagName, workspaceId) {
|
|
31
|
+
const normalizedFlag = flagName.toUpperCase().replace(/-/g, '_');
|
|
32
|
+
return `FEATURE_FLAG_LOCAL__${workspaceId}__${normalizedFlag}`;
|
|
33
|
+
}
|
|
34
|
+
static createGlobalEnvKey(flagName) {
|
|
35
|
+
const normalizedFlag = flagName.toUpperCase().replace(/-/g, '_');
|
|
36
|
+
return `FEATURE_FLAG_LOCAL_${normalizedFlag}`;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
exports.FeatureFlagUtils = FeatureFlagUtils;
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZmVhdHVyZS1mbGFncy9jb3JlL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLE1BQWEsZ0JBQWdCO0lBQzNCLE1BQU0sQ0FBQyxZQUFZLENBQUMsR0FBVztRQUM3QixPQUFPLEdBQUcsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxPQUFPLENBQUMsY0FBYyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ3hELENBQUM7SUFFRCxNQUFNLENBQUMsa0JBQWtCLENBQUMsVUFBaUM7UUFDekQsSUFBSSxDQUFDLFVBQVUsQ0FBQyxHQUFHLElBQUksT0FBTyxVQUFVLENBQUMsR0FBRyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzFELE1BQU0sSUFBSSxLQUFLLENBQUMsdUNBQXVDLENBQUMsQ0FBQztRQUMzRCxDQUFDO1FBQ0QsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLElBQUksT0FBTyxVQUFVLENBQUMsSUFBSSxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzVELE1BQU0sSUFBSSxLQUFLLENBQUMsd0NBQXdDLENBQUMsQ0FBQztRQUM1RCxDQUFDO1FBQ0QsSUFBSSxDQUFDLFVBQVUsQ0FBQyxXQUFXLElBQUksT0FBTyxVQUFVLENBQUMsV0FBVyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzFFLE1BQU0sSUFBSSxLQUFLLENBQUMsK0NBQStDLENBQUMsQ0FBQztRQUNuRSxDQUFDO1FBQ0QsSUFBSSxVQUFVLENBQUMsWUFBWSxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQzFDLE1BQU0sSUFBSSxLQUFLLENBQUMsMENBQTBDLENBQUMsQ0FBQztRQUM5RCxDQUFDO1FBQ0QsSUFBSSxDQUFDLENBQUMsU0FBUyxFQUFFLFFBQVEsRUFBRSxRQUFRLENBQUMsQ0FBQyxRQUFRLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUM7WUFDL0QsTUFBTSxJQUFJLEtBQUssQ0FBQyx5REFBeUQsQ0FBQyxDQUFDO1FBQzdFLENBQUM7UUFFRCxxQ0FBcUM7UUFDckMsTUFBTSxVQUFVLEdBQUcsT0FBTyxVQUFVLENBQUMsWUFBWSxDQUFDO1FBQ2xELElBQUksVUFBVSxLQUFLLFVBQVUsQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNuQyxNQUFNLElBQUksS0FBSyxDQUNiLDJCQUEyQixVQUFVLG1DQUFtQyxVQUFVLENBQUMsSUFBSSxHQUFHLENBQzNGLENBQUM7UUFDSixDQUFDO0lBQ0gsQ0FBQztJQUVELE1BQU0sQ0FBQyxxQkFBcUIsQ0FBQyxRQUFnQixFQUFFLFdBQW1CO1FBQ2hFLE1BQU0sY0FBYyxHQUFHLFFBQVEsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQ2pFLE9BQU8sdUJBQXVCLFdBQVcsS0FBSyxjQUFjLEVBQUUsQ0FBQztJQUNqRSxDQUFDO0lBRUQsTUFBTSxDQUFDLGtCQUFrQixDQUFDLFFBQWdCO1FBQ3hDLE1BQU0sY0FBYyxHQUFHLFFBQVEsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQ2pFLE9BQU8sc0JBQXNCLGNBQWMsRUFBRSxDQUFDO0lBQ2hELENBQUM7Q0FDRjtBQXhDRCw0Q0F3Q0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IEZlYXR1cmVGbGFnRGVmaW5pdGlvbiB9IGZyb20gJy4uL3R5cGVzJztcblxuZXhwb3J0IGNsYXNzIEZlYXR1cmVGbGFnVXRpbHMge1xuICBzdGF0aWMgbm9ybWFsaXplS2V5KGtleTogc3RyaW5nKTogc3RyaW5nIHtcbiAgICByZXR1cm4ga2V5LnRvTG93ZXJDYXNlKCkucmVwbGFjZSgvW15hLXowLTktX10vZywgJy0nKTtcbiAgfVxuXG4gIHN0YXRpYyB2YWxpZGF0ZURlZmluaXRpb24oZGVmaW5pdGlvbjogRmVhdHVyZUZsYWdEZWZpbml0aW9uKTogdm9pZCB7XG4gICAgaWYgKCFkZWZpbml0aW9uLmtleSB8fCB0eXBlb2YgZGVmaW5pdGlvbi5rZXkgIT09ICdzdHJpbmcnKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0ZsYWcgZGVmaW5pdGlvbiBtdXN0IGhhdmUgYSB2YWxpZCBrZXknKTtcbiAgICB9XG4gICAgaWYgKCFkZWZpbml0aW9uLm5hbWUgfHwgdHlwZW9mIGRlZmluaXRpb24ubmFtZSAhPT0gJ3N0cmluZycpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignRmxhZyBkZWZpbml0aW9uIG11c3QgaGF2ZSBhIHZhbGlkIG5hbWUnKTtcbiAgICB9XG4gICAgaWYgKCFkZWZpbml0aW9uLmRlc2NyaXB0aW9uIHx8IHR5cGVvZiBkZWZpbml0aW9uLmRlc2NyaXB0aW9uICE9PSAnc3RyaW5nJykge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdGbGFnIGRlZmluaXRpb24gbXVzdCBoYXZlIGEgdmFsaWQgZGVzY3JpcHRpb24nKTtcbiAgICB9XG4gICAgaWYgKGRlZmluaXRpb24uZGVmYXVsdFZhbHVlID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignRmxhZyBkZWZpbml0aW9uIG11c3QgaGF2ZSBhIGRlZmF1bHRWYWx1ZScpO1xuICAgIH1cbiAgICBpZiAoIVsnYm9vbGVhbicsICdzdHJpbmcnLCAnbnVtYmVyJ10uaW5jbHVkZXMoZGVmaW5pdGlvbi50eXBlKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdGbGFnIGRlZmluaXRpb24gdHlwZSBtdXN0IGJlIGJvb2xlYW4sIHN0cmluZywgb3IgbnVtYmVyJyk7XG4gICAgfVxuXG4gICAgLy8gVmFsaWRhdGUgZGVmYXVsdFZhbHVlIG1hdGNoZXMgdHlwZVxuICAgIGNvbnN0IGFjdHVhbFR5cGUgPSB0eXBlb2YgZGVmaW5pdGlvbi5kZWZhdWx0VmFsdWU7XG4gICAgaWYgKGFjdHVhbFR5cGUgIT09IGRlZmluaXRpb24udHlwZSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgRmxhZyBkZWZhdWx0VmFsdWUgdHlwZSAoJHthY3R1YWxUeXBlfSkgZG9lcyBub3QgbWF0Y2ggZGVjbGFyZWQgdHlwZSAoJHtkZWZpbml0aW9uLnR5cGV9KWAsXG4gICAgICApO1xuICAgIH1cbiAgfVxuXG4gIHN0YXRpYyBjcmVhdGVXb3Jrc3BhY2VFbnZLZXkoZmxhZ05hbWU6IHN0cmluZywgd29ya3NwYWNlSWQ6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgY29uc3Qgbm9ybWFsaXplZEZsYWcgPSBmbGFnTmFtZS50b1VwcGVyQ2FzZSgpLnJlcGxhY2UoLy0vZywgJ18nKTtcbiAgICByZXR1cm4gYEZFQVRVUkVfRkxBR19MT0NBTF9fJHt3b3Jrc3BhY2VJZH1fXyR7bm9ybWFsaXplZEZsYWd9YDtcbiAgfVxuXG4gIHN0YXRpYyBjcmVhdGVHbG9iYWxFbnZLZXkoZmxhZ05hbWU6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgY29uc3Qgbm9ybWFsaXplZEZsYWcgPSBmbGFnTmFtZS50b1VwcGVyQ2FzZSgpLnJlcGxhY2UoLy0vZywgJ18nKTtcbiAgICByZXR1cm4gYEZFQVRVUkVfRkxBR19MT0NBTF8ke25vcm1hbGl6ZWRGbGFnfWA7XG4gIH1cbn1cbiJdfQ==
|