@pipeline-builder/pipeline-data 3.1.0

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.
@@ -0,0 +1,456 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.db = exports.Connection = void 0;
6
+ exports.getConnection = getConnection;
7
+ exports.closeConnection = closeConnection;
8
+ exports.testConnection = testConnection;
9
+ exports.initializeDatabase = initializeDatabase;
10
+ const api_core_1 = require("@pipeline-builder/api-core");
11
+ const drizzle_orm_1 = require("drizzle-orm");
12
+ const node_postgres_1 = require("drizzle-orm/node-postgres");
13
+ const pg_1 = require("pg");
14
+ const drizzle_schema_1 = require("./drizzle-schema");
15
+ const retry_strategy_1 = require("./retry-strategy");
16
+ const logger = (0, api_core_1.createLogger)('Database');
17
+ /**
18
+ * Get database configuration from environment variables
19
+ * Note: Uses environment variables directly to avoid circular dependency with pipeline-core
20
+ */
21
+ function parseIntEnv(value, fallback) {
22
+ if (!value)
23
+ return fallback;
24
+ const parsed = parseInt(value, 10);
25
+ return Number.isNaN(parsed) ? fallback : parsed;
26
+ }
27
+ /** Detect Lambda environment for pool-size tuning */
28
+ const isLambda = !!process.env.AWS_LAMBDA_FUNCTION_NAME;
29
+ function getDatabaseConfig() {
30
+ // Lambda: small pool (each invocation is short-lived, many concurrent instances)
31
+ // ECS/long-running: larger pool for sustained concurrency
32
+ const defaultPoolSize = isLambda ? 2 : 20;
33
+ const defaultIdleTimeout = isLambda ? 10000 : 30000;
34
+ return {
35
+ host: process.env.DB_HOST || 'postgres',
36
+ port: parseIntEnv(process.env.DB_PORT, 5432),
37
+ database: process.env.DATABASE || 'pipeline_builder',
38
+ user: process.env.DB_USER || 'postgres',
39
+ password: process.env.DB_PASSWORD || (process.env.NODE_ENV === 'production' ? (() => { throw new Error('DB_PASSWORD is required in production'); })() : ''),
40
+ maxPoolSize: parseIntEnv(process.env.DRIZZLE_MAX_POOL_SIZE, defaultPoolSize),
41
+ idleTimeoutMillis: parseIntEnv(process.env.DRIZZLE_IDLE_TIMEOUT_MILLIS, defaultIdleTimeout),
42
+ connectionTimeoutMillis: parseIntEnv(process.env.DRIZZLE_CONNECTION_TIMEOUT_MILLIS, 5000),
43
+ };
44
+ }
45
+ /**
46
+ * Singleton database connection class.
47
+ * Manages PostgreSQL connection pooling and Drizzle ORM instance.
48
+ *
49
+ * Features:
50
+ * - Singleton pattern for single connection pool
51
+ * - Automatic connection retry with backoff
52
+ * - Connection health monitoring
53
+ * - Graceful shutdown handling
54
+ * - Comprehensive error handling
55
+ * - Connection statistics tracking
56
+ *
57
+ * @example
58
+ * ```typescript
59
+ * import { Connection } from './connection';
60
+ *
61
+ * const connection = Connection.getInstance();
62
+ * const plugins = await connection.db.select().from(schema.plugin);
63
+ *
64
+ * // During shutdown
65
+ * await connection.close();
66
+ * ```
67
+ */
68
+ class Connection {
69
+ static instance = null;
70
+ /**
71
+ * Drizzle ORM database instance with schema
72
+ */
73
+ db;
74
+ pool;
75
+ options;
76
+ retryStrategy;
77
+ isShuttingDown = false;
78
+ /**
79
+ * Private constructor to enforce singleton pattern.
80
+ * Initializes PostgreSQL connection pool and Drizzle ORM instance.
81
+ *
82
+ * @param options - Optional configuration for the connection
83
+ * @throws {Error} If database initialization fails after all retries
84
+ */
85
+ constructor(options = {}) {
86
+ this.options = {
87
+ enableLogging: options.enableLogging ?? true,
88
+ enableAutoRetry: options.enableAutoRetry ?? true,
89
+ maxRetries: options.maxRetries ?? parseInt(process.env.DB_MAX_RETRIES || '3', 10),
90
+ retryDelay: options.retryDelay ?? parseInt(process.env.DB_RETRY_DELAY_MS || '1000', 10),
91
+ ssl: options.ssl ?? false,
92
+ };
93
+ // Initialize retry strategy
94
+ this.retryStrategy = new retry_strategy_1.ConnectionRetryStrategy({
95
+ maxRetries: this.options.maxRetries,
96
+ baseDelay: this.options.retryDelay,
97
+ });
98
+ try {
99
+ const config = getDatabaseConfig();
100
+ const poolConfig = {
101
+ host: config.host,
102
+ port: config.port,
103
+ database: config.database,
104
+ user: config.user,
105
+ password: config.password,
106
+ max: config.maxPoolSize,
107
+ idleTimeoutMillis: config.idleTimeoutMillis,
108
+ connectionTimeoutMillis: config.connectionTimeoutMillis,
109
+ ssl: this.options.ssl,
110
+ allowExitOnIdle: true,
111
+ };
112
+ this.pool = new pg_1.Pool(poolConfig);
113
+ this.setupEventHandlers();
114
+ this.db = (0, node_postgres_1.drizzle)(this.pool, { schema: drizzle_schema_1.schema });
115
+ if (this.options.enableLogging) {
116
+ logger.info('Database connection initialized successfully');
117
+ this.logConnectionConfig(poolConfig);
118
+ }
119
+ }
120
+ catch (error) {
121
+ logger.error('Failed to initialize database connection:', error);
122
+ throw new Error('Database initialization failed');
123
+ }
124
+ }
125
+ /**
126
+ * Gets the singleton instance of the Connection class.
127
+ * Creates a new instance if one doesn't exist.
128
+ *
129
+ * @param options - Optional configuration (only used on first call)
130
+ * @returns The singleton Connection instance
131
+ */
132
+ static getInstance(options) {
133
+ if (!Connection.instance) {
134
+ Connection.instance = new Connection(options);
135
+ }
136
+ return Connection.instance;
137
+ }
138
+ /**
139
+ * Resets the singleton instance.
140
+ * Useful for testing or reconfiguring the connection.
141
+ *
142
+ * @param closeExisting - Whether to close existing connection before reset
143
+ */
144
+ static async reset(closeExisting = true) {
145
+ if (Connection.instance && closeExisting) {
146
+ await Connection.instance.close();
147
+ }
148
+ Connection.instance = null;
149
+ }
150
+ /**
151
+ * Tests the database connection
152
+ *
153
+ * @returns Promise that resolves to true if connection is healthy
154
+ */
155
+ async testConnection() {
156
+ try {
157
+ const client = await this.pool.connect();
158
+ const result = await client.query('SELECT 1');
159
+ client.release();
160
+ if (this.options.enableLogging) {
161
+ logger.info('Database connection test successful');
162
+ }
163
+ return result.rows.length > 0;
164
+ }
165
+ catch (error) {
166
+ logger.error('Database connection test failed:', error);
167
+ return false;
168
+ }
169
+ }
170
+ /**
171
+ * Gets connection pool statistics
172
+ *
173
+ * @returns Current connection pool statistics
174
+ */
175
+ getStats() {
176
+ return {
177
+ totalCount: this.pool.totalCount,
178
+ idleCount: this.pool.idleCount,
179
+ waitingCount: this.pool.waitingCount,
180
+ };
181
+ }
182
+ /**
183
+ * Executes a database transaction
184
+ *
185
+ * @param callback - Function to execute within the transaction
186
+ * @returns Result of the transaction
187
+ *
188
+ * @example
189
+ * ```typescript
190
+ * const result = await connection.transaction(async (tx) => {
191
+ * await tx.insert(schema.plugin).values({ ... });
192
+ * await tx.insert(schema.metadata).values({ ... });
193
+ * return { success: true };
194
+ * });
195
+ * ```
196
+ */
197
+ async transaction(callback, timeoutMs = parseIntEnv(process.env.DB_TRANSACTION_TIMEOUT_MS, 30000)) {
198
+ let timer;
199
+ // Wrap callback to set PostgreSQL statement_timeout as a server-side guard.
200
+ // The Promise.race timeout below handles the JS side, but statement_timeout
201
+ // ensures the DB itself cancels long-running queries if the JS timeout fires
202
+ // but the connection isn't cleaned up.
203
+ const wrappedCallback = async (tx) => {
204
+ await tx.execute((0, drizzle_orm_1.sql) `SET LOCAL statement_timeout = ${String(timeoutMs)}`);
205
+ return callback(tx);
206
+ };
207
+ const txPromise = this.db.transaction(wrappedCallback);
208
+ const timeoutPromise = new Promise((_, reject) => {
209
+ timer = setTimeout(() => reject(new Error(`Transaction timeout after ${timeoutMs}ms`)), timeoutMs);
210
+ });
211
+ try {
212
+ return await Promise.race([txPromise, timeoutPromise]);
213
+ }
214
+ finally {
215
+ clearTimeout(timer);
216
+ }
217
+ }
218
+ /**
219
+ * Acquires a client from the pool for manual query execution
220
+ * Remember to release the client when done
221
+ *
222
+ * @returns PostgreSQL client from the pool
223
+ *
224
+ * @example
225
+ * ```typescript
226
+ * const client = await connection.getClient();
227
+ * try {
228
+ * await client.query('BEGIN');
229
+ * await client.query('INSERT INTO ...');
230
+ * await client.query('COMMIT');
231
+ * } finally {
232
+ * client.release();
233
+ * }
234
+ * ```
235
+ */
236
+ async getClient() {
237
+ return this.pool.connect();
238
+ }
239
+ /**
240
+ * Closes the database connection pool gracefully.
241
+ * Should be called during application shutdown.
242
+ *
243
+ * @param timeout - Maximum time to wait for connections to close (ms)
244
+ * @returns Promise that resolves when pool is closed
245
+ */
246
+ async close(timeout = parseIntEnv(process.env.DB_CLOSE_TIMEOUT_MS, 5000)) {
247
+ if (this.isShuttingDown) {
248
+ logger.warn('Connection is already shutting down');
249
+ return;
250
+ }
251
+ this.isShuttingDown = true;
252
+ try {
253
+ if (this.options.enableLogging) {
254
+ logger.info('Closing database connection pool...');
255
+ const stats = this.getStats();
256
+ logger.info(`Pool stats - Total: ${stats.totalCount}, Idle: ${stats.idleCount}, Waiting: ${stats.waitingCount}`);
257
+ }
258
+ // Set a timeout for graceful shutdown
259
+ const closePromise = this.pool.end();
260
+ const timeoutPromise = new Promise((_, reject) => setTimeout(() => reject(new Error('Connection close timeout')), timeout));
261
+ await Promise.race([closePromise, timeoutPromise]);
262
+ if (this.options.enableLogging) {
263
+ logger.info('Database connection closed successfully');
264
+ }
265
+ }
266
+ catch (error) {
267
+ logger.error('Error closing database connection:', error);
268
+ throw error;
269
+ }
270
+ finally {
271
+ this.isShuttingDown = false;
272
+ }
273
+ }
274
+ /**
275
+ * Checks if the connection is shutting down
276
+ *
277
+ * @returns true if connection is in shutdown state
278
+ */
279
+ isClosing() {
280
+ return this.isShuttingDown;
281
+ }
282
+ /**
283
+ * Sets up event handlers for the connection pool
284
+ */
285
+ setupEventHandlers() {
286
+ this.pool.on('error', (err) => {
287
+ logger.error('Unexpected error on idle client:', err);
288
+ if (this.options.enableAutoRetry && this.retryStrategy.getAttempts() < this.options.maxRetries) {
289
+ void this.retryStrategy.handleConnectionError(err, () => this.testConnection()).catch((retryErr) => {
290
+ logger.error('Connection retry error:', retryErr);
291
+ });
292
+ }
293
+ });
294
+ this.pool.on('connect', () => {
295
+ this.retryStrategy.reset(); // Reset on successful connection
296
+ if (this.options.enableLogging) {
297
+ logger.debug('New database connection established');
298
+ }
299
+ });
300
+ this.pool.on('remove', () => {
301
+ if (this.options.enableLogging) {
302
+ logger.debug('Client removed from pool');
303
+ }
304
+ });
305
+ }
306
+ /**
307
+ * Logs connection configuration (sanitized)
308
+ */
309
+ logConnectionConfig(config) {
310
+ logger.info('Database Configuration:', {
311
+ host: `${config.host}:${config.port}`,
312
+ database: config.database,
313
+ user: config.user,
314
+ maxPoolSize: config.max,
315
+ idleTimeoutMs: config.idleTimeoutMillis,
316
+ connectionTimeoutMs: config.connectionTimeoutMillis,
317
+ ssl: config.ssl ? 'enabled' : 'disabled',
318
+ });
319
+ }
320
+ }
321
+ exports.Connection = Connection;
322
+ /**
323
+ * Singleton database instance for use throughout the application.
324
+ *
325
+ * @example
326
+ * ```typescript
327
+ * import { db } from './connection';
328
+ *
329
+ * // Select queries
330
+ * const plugins = await db.select().from(schema.plugin);
331
+ *
332
+ * // Insert queries
333
+ * await db.insert(schema.plugin).values({ name: 'my-plugin' });
334
+ *
335
+ * // Transactions
336
+ * await db.transaction(async (tx) => {
337
+ * await tx.insert(schema.plugin).values({ ... });
338
+ * await tx.update(schema.plugin).set({ ... });
339
+ * });
340
+ * ```
341
+ */
342
+ // Lazy initialization to avoid race condition on module load
343
+ let _dbInstance = null;
344
+ /**
345
+ * Get the database instance with lazy initialization
346
+ * This avoids the race condition where the module is loaded before environment is configured
347
+ */
348
+ function getDbInstance() {
349
+ if (!_dbInstance) {
350
+ _dbInstance = Connection.getInstance().db;
351
+ }
352
+ return _dbInstance;
353
+ }
354
+ /**
355
+ * Proxy-based lazy database instance
356
+ * The actual connection is only created when first accessed
357
+ *
358
+ * @example
359
+ * ```typescript
360
+ * import { db } from './connection';
361
+ *
362
+ * // Connection is created here on first use, not on import
363
+ * const plugins = await db.select().from(schema.plugin);
364
+ * ```
365
+ */
366
+ exports.db = new Proxy({}, {
367
+ get(_, prop) {
368
+ const instance = getDbInstance();
369
+ const value = instance[prop];
370
+ // Bind methods to the instance to preserve 'this' context
371
+ if (typeof value === 'function') {
372
+ return value.bind(instance);
373
+ }
374
+ return value;
375
+ },
376
+ });
377
+ /**
378
+ * Gets the Connection instance for advanced operations
379
+ *
380
+ * @example
381
+ * ```typescript
382
+ * import { getConnection } from './connection';
383
+ *
384
+ * const connection = getConnection();
385
+ * const stats = connection.getStats();
386
+ * console.log(`Active connections: ${stats.totalCount}`);
387
+ * ```
388
+ */
389
+ function getConnection() {
390
+ return Connection.getInstance();
391
+ }
392
+ /**
393
+ * Closes the database connection
394
+ * Should be called during application shutdown
395
+ *
396
+ * @example
397
+ * ```typescript
398
+ * import { closeConnection } from './connection';
399
+ *
400
+ * process.on('SIGTERM', async () => {
401
+ * await closeConnection();
402
+ * process.exit(0);
403
+ * });
404
+ * ```
405
+ */
406
+ async function closeConnection() {
407
+ const connection = Connection.getInstance();
408
+ await connection.close();
409
+ _dbInstance = null; // Reset lazy instance
410
+ }
411
+ /**
412
+ * Tests the database connection
413
+ *
414
+ * @returns Promise that resolves to true if connection is healthy
415
+ *
416
+ * @example
417
+ * ```typescript
418
+ * import { testConnection } from './connection';
419
+ *
420
+ * if (await testConnection()) {
421
+ * console.log('Database is ready');
422
+ * } else {
423
+ * console.error('Database connection failed');
424
+ * process.exit(1);
425
+ * }
426
+ * ```
427
+ */
428
+ async function testConnection() {
429
+ const connection = Connection.getInstance();
430
+ return connection.testConnection();
431
+ }
432
+ /**
433
+ * Initialize the database connection explicitly
434
+ * Call this during application startup after environment is configured
435
+ *
436
+ * @example
437
+ * ```typescript
438
+ * import { initializeDatabase } from './connection';
439
+ *
440
+ * async function bootstrap() {
441
+ * // Load environment variables first
442
+ * dotenv.config();
443
+ *
444
+ * // Then initialize database
445
+ * await initializeDatabase();
446
+ * }
447
+ * ```
448
+ */
449
+ async function initializeDatabase() {
450
+ const connection = Connection.getInstance();
451
+ const healthy = await connection.testConnection();
452
+ if (!healthy) {
453
+ throw new Error('Database connection failed during initialization');
454
+ }
455
+ }
456
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"postgres-connection.js","sourceRoot":"","sources":["../../src/database/postgres-connection.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createLogger } from '@pipeline-builder/api-core';\nimport { sql } from 'drizzle-orm';\nimport { drizzle } from 'drizzle-orm/node-postgres';\nimport { Pool, PoolConfig, PoolClient } from 'pg';\nimport { schema } from './drizzle-schema';\nimport { ConnectionRetryStrategy } from './retry-strategy';\n\nconst logger = createLogger('Database');\n\n/**\n * Get database configuration from environment variables\n * Note: Uses environment variables directly to avoid circular dependency with pipeline-core\n */\nfunction parseIntEnv(value: string | undefined, fallback: number): number {\n  if (!value) return fallback;\n  const parsed = parseInt(value, 10);\n  return Number.isNaN(parsed) ? fallback : parsed;\n}\n\n/** Detect Lambda environment for pool-size tuning */\nconst isLambda = !!process.env.AWS_LAMBDA_FUNCTION_NAME;\n\nfunction getDatabaseConfig() {\n  // Lambda: small pool (each invocation is short-lived, many concurrent instances)\n  // ECS/long-running: larger pool for sustained concurrency\n  const defaultPoolSize = isLambda ? 2 : 20;\n  const defaultIdleTimeout = isLambda ? 10000 : 30000;\n\n  return {\n    host: process.env.DB_HOST || 'postgres',\n    port: parseIntEnv(process.env.DB_PORT, 5432),\n    database: process.env.DATABASE || 'pipeline_builder',\n    user: process.env.DB_USER || 'postgres',\n    password: process.env.DB_PASSWORD || (process.env.NODE_ENV === 'production' ? (() => { throw new Error('DB_PASSWORD is required in production'); })() as string : ''),\n    maxPoolSize: parseIntEnv(process.env.DRIZZLE_MAX_POOL_SIZE, defaultPoolSize),\n    idleTimeoutMillis: parseIntEnv(process.env.DRIZZLE_IDLE_TIMEOUT_MILLIS, defaultIdleTimeout),\n    connectionTimeoutMillis: parseIntEnv(process.env.DRIZZLE_CONNECTION_TIMEOUT_MILLIS, 5000),\n  };\n}\n\n/**\n * Database connection statistics for monitoring\n */\nexport interface ConnectionStats {\n  totalCount: number;\n  idleCount: number;\n  waitingCount: number;\n}\n\n/**\n * Options for configuring the database connection\n */\nexport interface ConnectionOptions {\n  /** Whether to enable connection logging */\n  enableLogging?: boolean;\n\n  /** Whether to automatically retry failed connections */\n  enableAutoRetry?: boolean;\n\n  /** Maximum number of connection retry attempts */\n  maxRetries?: number;\n\n  /** Delay between retry attempts in milliseconds */\n  retryDelay?: number;\n\n  /** SSL configuration */\n  ssl?: boolean | { rejectUnauthorized: boolean };\n}\n\n/**\n * Singleton database connection class.\n * Manages PostgreSQL connection pooling and Drizzle ORM instance.\n *\n * Features:\n * - Singleton pattern for single connection pool\n * - Automatic connection retry with backoff\n * - Connection health monitoring\n * - Graceful shutdown handling\n * - Comprehensive error handling\n * - Connection statistics tracking\n *\n * @example\n * ```typescript\n * import { Connection } from './connection';\n *\n * const connection = Connection.getInstance();\n * const plugins = await connection.db.select().from(schema.plugin);\n *\n * // During shutdown\n * await connection.close();\n * ```\n */\nexport class Connection {\n  private static instance: Connection | null = null;\n\n  /**\n   * Drizzle ORM database instance with schema\n   */\n  public readonly db: ReturnType<typeof drizzle>;\n\n  private readonly pool: Pool;\n  private readonly options: Required<ConnectionOptions>;\n  private readonly retryStrategy: ConnectionRetryStrategy;\n  private isShuttingDown = false;\n\n  /**\n   * Private constructor to enforce singleton pattern.\n   * Initializes PostgreSQL connection pool and Drizzle ORM instance.\n   *\n   * @param options - Optional configuration for the connection\n   * @throws {Error} If database initialization fails after all retries\n   */\n  private constructor(options: ConnectionOptions = {}) {\n    this.options = {\n      enableLogging: options.enableLogging ?? true,\n      enableAutoRetry: options.enableAutoRetry ?? true,\n      maxRetries: options.maxRetries ?? parseInt(process.env.DB_MAX_RETRIES || '3', 10),\n      retryDelay: options.retryDelay ?? parseInt(process.env.DB_RETRY_DELAY_MS || '1000', 10),\n      ssl: options.ssl ?? false,\n    };\n\n    // Initialize retry strategy\n    this.retryStrategy = new ConnectionRetryStrategy({\n      maxRetries: this.options.maxRetries,\n      baseDelay: this.options.retryDelay,\n    });\n\n    try {\n      const config = getDatabaseConfig();\n\n      const poolConfig: PoolConfig = {\n        host: config.host,\n        port: config.port,\n        database: config.database,\n        user: config.user,\n        password: config.password,\n        max: config.maxPoolSize,\n        idleTimeoutMillis: config.idleTimeoutMillis,\n        connectionTimeoutMillis: config.connectionTimeoutMillis,\n        ssl: this.options.ssl,\n        allowExitOnIdle: true,\n      };\n\n      this.pool = new Pool(poolConfig);\n      this.setupEventHandlers();\n\n      this.db = drizzle(this.pool, { schema });\n\n      if (this.options.enableLogging) {\n        logger.info('Database connection initialized successfully');\n        this.logConnectionConfig(poolConfig);\n      }\n    } catch (error) {\n      logger.error('Failed to initialize database connection:', error);\n      throw new Error('Database initialization failed');\n    }\n  }\n\n  /**\n   * Gets the singleton instance of the Connection class.\n   * Creates a new instance if one doesn't exist.\n   *\n   * @param options - Optional configuration (only used on first call)\n   * @returns The singleton Connection instance\n   */\n  public static getInstance(options?: ConnectionOptions): Connection {\n    if (!Connection.instance) {\n      Connection.instance = new Connection(options);\n    }\n    return Connection.instance;\n  }\n\n  /**\n   * Resets the singleton instance.\n   * Useful for testing or reconfiguring the connection.\n   *\n   * @param closeExisting - Whether to close existing connection before reset\n   */\n  public static async reset(closeExisting: boolean = true): Promise<void> {\n    if (Connection.instance && closeExisting) {\n      await Connection.instance.close();\n    }\n    Connection.instance = null;\n  }\n\n  /**\n   * Tests the database connection\n   *\n   * @returns Promise that resolves to true if connection is healthy\n   */\n  public async testConnection(): Promise<boolean> {\n    try {\n      const client = await this.pool.connect();\n      const result = await client.query('SELECT 1');\n      client.release();\n\n      if (this.options.enableLogging) {\n        logger.info('Database connection test successful');\n      }\n\n      return result.rows.length > 0;\n    } catch (error) {\n      logger.error('Database connection test failed:', error);\n      return false;\n    }\n  }\n\n  /**\n   * Gets connection pool statistics\n   *\n   * @returns Current connection pool statistics\n   */\n  public getStats(): ConnectionStats {\n    return {\n      totalCount: this.pool.totalCount,\n      idleCount: this.pool.idleCount,\n      waitingCount: this.pool.waitingCount,\n    };\n  }\n\n  /**\n   * Executes a database transaction\n   *\n   * @param callback - Function to execute within the transaction\n   * @returns Result of the transaction\n   *\n   * @example\n   * ```typescript\n   * const result = await connection.transaction(async (tx) => {\n   *   await tx.insert(schema.plugin).values({ ... });\n   *   await tx.insert(schema.metadata).values({ ... });\n   *   return { success: true };\n   * });\n   * ```\n   */\n  public async transaction<T>(\n    callback: Parameters<typeof this.db.transaction>[0],\n    timeoutMs: number = parseIntEnv(process.env.DB_TRANSACTION_TIMEOUT_MS, 30000),\n  ): Promise<T> {\n    let timer: ReturnType<typeof setTimeout>;\n    // Wrap callback to set PostgreSQL statement_timeout as a server-side guard.\n    // The Promise.race timeout below handles the JS side, but statement_timeout\n    // ensures the DB itself cancels long-running queries if the JS timeout fires\n    // but the connection isn't cleaned up.\n    const wrappedCallback: typeof callback = async (tx) => {\n      await tx.execute(sql`SET LOCAL statement_timeout = ${String(timeoutMs)}`);\n      return callback(tx);\n    };\n    const txPromise = this.db.transaction(wrappedCallback) as Promise<T>;\n    const timeoutPromise = new Promise<never>((_, reject) => {\n      timer = setTimeout(() => reject(new Error(`Transaction timeout after ${timeoutMs}ms`)), timeoutMs);\n    });\n    try {\n      return await Promise.race([txPromise, timeoutPromise]);\n    } finally {\n      clearTimeout(timer!);\n    }\n  }\n\n  /**\n   * Acquires a client from the pool for manual query execution\n   * Remember to release the client when done\n   *\n   * @returns PostgreSQL client from the pool\n   *\n   * @example\n   * ```typescript\n   * const client = await connection.getClient();\n   * try {\n   *   await client.query('BEGIN');\n   *   await client.query('INSERT INTO ...');\n   *   await client.query('COMMIT');\n   * } finally {\n   *   client.release();\n   * }\n   * ```\n   */\n  public async getClient(): Promise<PoolClient> {\n    return this.pool.connect();\n  }\n\n  /**\n   * Closes the database connection pool gracefully.\n   * Should be called during application shutdown.\n   *\n   * @param timeout - Maximum time to wait for connections to close (ms)\n   * @returns Promise that resolves when pool is closed\n   */\n  public async close(timeout: number = parseIntEnv(process.env.DB_CLOSE_TIMEOUT_MS, 5000)): Promise<void> {\n    if (this.isShuttingDown) {\n      logger.warn('Connection is already shutting down');\n      return;\n    }\n\n    this.isShuttingDown = true;\n\n    try {\n      if (this.options.enableLogging) {\n        logger.info('Closing database connection pool...');\n        const stats = this.getStats();\n        logger.info(`Pool stats - Total: ${stats.totalCount}, Idle: ${stats.idleCount}, Waiting: ${stats.waitingCount}`);\n      }\n\n      // Set a timeout for graceful shutdown\n      const closePromise = this.pool.end();\n      const timeoutPromise = new Promise((_, reject) =>\n        setTimeout(() => reject(new Error('Connection close timeout')), timeout),\n      );\n\n      await Promise.race([closePromise, timeoutPromise]);\n\n      if (this.options.enableLogging) {\n        logger.info('Database connection closed successfully');\n      }\n    } catch (error) {\n      logger.error('Error closing database connection:', error);\n      throw error;\n    } finally {\n      this.isShuttingDown = false;\n    }\n  }\n\n  /**\n   * Checks if the connection is shutting down\n   *\n   * @returns true if connection is in shutdown state\n   */\n  public isClosing(): boolean {\n    return this.isShuttingDown;\n  }\n\n  /**\n   * Sets up event handlers for the connection pool\n   */\n  private setupEventHandlers(): void {\n    this.pool.on('error', (err) => {\n      logger.error('Unexpected error on idle client:', err);\n\n      if (this.options.enableAutoRetry && this.retryStrategy.getAttempts() < this.options.maxRetries) {\n        void this.retryStrategy.handleConnectionError(err, () => this.testConnection()).catch((retryErr) => {\n          logger.error('Connection retry error:', retryErr);\n        });\n      }\n    });\n\n    this.pool.on('connect', () => {\n      this.retryStrategy.reset(); // Reset on successful connection\n\n      if (this.options.enableLogging) {\n        logger.debug('New database connection established');\n      }\n    });\n\n    this.pool.on('remove', () => {\n      if (this.options.enableLogging) {\n        logger.debug('Client removed from pool');\n      }\n    });\n  }\n\n  /**\n   * Logs connection configuration (sanitized)\n   */\n  private logConnectionConfig(config: PoolConfig): void {\n    logger.info('Database Configuration:', {\n      host: `${config.host}:${config.port}`,\n      database: config.database,\n      user: config.user,\n      maxPoolSize: config.max,\n      idleTimeoutMs: config.idleTimeoutMillis,\n      connectionTimeoutMs: config.connectionTimeoutMillis,\n      ssl: config.ssl ? 'enabled' : 'disabled',\n    });\n  }\n}\n\n/**\n * Singleton database instance for use throughout the application.\n *\n * @example\n * ```typescript\n * import { db } from './connection';\n *\n * // Select queries\n * const plugins = await db.select().from(schema.plugin);\n *\n * // Insert queries\n * await db.insert(schema.plugin).values({ name: 'my-plugin' });\n *\n * // Transactions\n * await db.transaction(async (tx) => {\n *   await tx.insert(schema.plugin).values({ ... });\n *   await tx.update(schema.plugin).set({ ... });\n * });\n * ```\n */\n\n// Lazy initialization to avoid race condition on module load\nlet _dbInstance: ReturnType<typeof drizzle> | null = null;\n\n/**\n * Get the database instance with lazy initialization\n * This avoids the race condition where the module is loaded before environment is configured\n */\nfunction getDbInstance(): ReturnType<typeof drizzle> {\n  if (!_dbInstance) {\n    _dbInstance = Connection.getInstance().db;\n  }\n  return _dbInstance;\n}\n\n/**\n * Proxy-based lazy database instance\n * The actual connection is only created when first accessed\n *\n * @example\n * ```typescript\n * import { db } from './connection';\n *\n * // Connection is created here on first use, not on import\n * const plugins = await db.select().from(schema.plugin);\n * ```\n */\nexport const db = new Proxy({} as ReturnType<typeof drizzle>, {\n  get(_, prop: string | symbol) {\n    const instance = getDbInstance();\n    const value = instance[prop as keyof typeof instance];\n    // Bind methods to the instance to preserve 'this' context\n    if (typeof value === 'function') {\n      return value.bind(instance);\n    }\n    return value;\n  },\n});\n\n/**\n * Gets the Connection instance for advanced operations\n *\n * @example\n * ```typescript\n * import { getConnection } from './connection';\n *\n * const connection = getConnection();\n * const stats = connection.getStats();\n * console.log(`Active connections: ${stats.totalCount}`);\n * ```\n */\nexport function getConnection(): Connection {\n  return Connection.getInstance();\n}\n\n/**\n * Closes the database connection\n * Should be called during application shutdown\n *\n * @example\n * ```typescript\n * import { closeConnection } from './connection';\n *\n * process.on('SIGTERM', async () => {\n *   await closeConnection();\n *   process.exit(0);\n * });\n * ```\n */\nexport async function closeConnection(): Promise<void> {\n  const connection = Connection.getInstance();\n  await connection.close();\n  _dbInstance = null; // Reset lazy instance\n}\n\n/**\n * Tests the database connection\n *\n * @returns Promise that resolves to true if connection is healthy\n *\n * @example\n * ```typescript\n * import { testConnection } from './connection';\n *\n * if (await testConnection()) {\n *   console.log('Database is ready');\n * } else {\n *   console.error('Database connection failed');\n *   process.exit(1);\n * }\n * ```\n */\nexport async function testConnection(): Promise<boolean> {\n  const connection = Connection.getInstance();\n  return connection.testConnection();\n}\n\n/**\n * Initialize the database connection explicitly\n * Call this during application startup after environment is configured\n *\n * @example\n * ```typescript\n * import { initializeDatabase } from './connection';\n *\n * async function bootstrap() {\n *   // Load environment variables first\n *   dotenv.config();\n *\n *   // Then initialize database\n *   await initializeDatabase();\n * }\n * ```\n */\nexport async function initializeDatabase(): Promise<void> {\n  const connection = Connection.getInstance();\n  const healthy = await connection.testConnection();\n  if (!healthy) {\n    throw new Error('Database connection failed during initialization');\n  }\n}"]}
@@ -0,0 +1,68 @@
1
+ /**
2
+ * Configuration for connection retry strategy
3
+ */
4
+ export interface RetryConfig {
5
+ /** Maximum number of retry attempts */
6
+ maxRetries: number;
7
+ /** Base delay between retries in milliseconds */
8
+ baseDelay: number;
9
+ }
10
+ /**
11
+ * Implements exponential backoff retry strategy for database connections.
12
+ *
13
+ * Features:
14
+ * - Exponential backoff with configurable base delay
15
+ * - Attempt tracking and logging
16
+ * - Graceful failure after max retries
17
+ *
18
+ * @example
19
+ * ```typescript
20
+ * const strategy = new ConnectionRetryStrategy({ maxRetries: 3, baseDelay: 1000 });
21
+ *
22
+ * const result = await strategy.execute(async () => {
23
+ * return await db.query('SELECT 1');
24
+ * });
25
+ * ```
26
+ */
27
+ export declare class ConnectionRetryStrategy {
28
+ private readonly config;
29
+ private attempts;
30
+ constructor(config: RetryConfig);
31
+ /**
32
+ * Executes an operation with retry logic
33
+ *
34
+ * @param operation - Async function to execute with retries
35
+ * @returns Promise resolving to operation result
36
+ * @throws Error if all retry attempts fail
37
+ */
38
+ execute<T>(operation: () => Promise<T>): Promise<T>;
39
+ /**
40
+ * Handles connection errors with retry tracking
41
+ *
42
+ * @param error - Error that occurred
43
+ * @param testConnection - Function to test if connection is restored
44
+ */
45
+ handleConnectionError(error: Error, testConnection: () => Promise<boolean>): Promise<void>;
46
+ /**
47
+ * Resets the attempt counter
48
+ * Call this after a successful operation
49
+ */
50
+ reset(): void;
51
+ /**
52
+ * Gets the current attempt count
53
+ */
54
+ getAttempts(): number;
55
+ /**
56
+ * Calculates exponential backoff delay
57
+ *
58
+ * @param attempt - Current attempt number (1-indexed)
59
+ * @returns Delay in milliseconds
60
+ */
61
+ private calculateBackoff;
62
+ /**
63
+ * Sleeps for the specified duration
64
+ *
65
+ * @param ms - Milliseconds to sleep
66
+ */
67
+ private sleep;
68
+ }