@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.
- package/LICENSE +202 -0
- package/README.md +34 -0
- package/drizzle.config.ts +17 -0
- package/lib/api/access-control-builder.d.ts +109 -0
- package/lib/api/access-control-builder.js +181 -0
- package/lib/api/crud-service.d.ts +170 -0
- package/lib/api/crud-service.js +387 -0
- package/lib/api/query-builders.d.ts +74 -0
- package/lib/api/query-builders.js +336 -0
- package/lib/api/reporting-service.d.ts +131 -0
- package/lib/api/reporting-service.js +248 -0
- package/lib/core/query-filters.d.ts +235 -0
- package/lib/core/query-filters.js +23 -0
- package/lib/database/drizzle-schema.d.ts +10043 -0
- package/lib/database/drizzle-schema.js +715 -0
- package/lib/database/index.d.ts +3 -0
- package/lib/database/index.js +22 -0
- package/lib/database/postgres-connection.d.ts +232 -0
- package/lib/database/postgres-connection.js +456 -0
- package/lib/database/retry-strategy.d.ts +68 -0
- package/lib/database/retry-strategy.js +126 -0
- package/lib/index.d.ts +30 -0
- package/lib/index.js +52 -0
- package/package.json +125 -0
|
@@ -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
|
+
}
|