tasq/node_modules/@claude-flow/mcp/dist/connection-pool.js

273 lines
9.6 KiB
JavaScript

/**
* @claude-flow/mcp - Connection Pool
*
* High-performance connection pooling
*/
import { EventEmitter } from 'events';
const DEFAULT_POOL_CONFIG = {
maxConnections: 10,
minConnections: 2,
idleTimeout: 30000,
acquireTimeout: 5000,
maxWaitingClients: 50,
evictionRunInterval: 10000,
};
class ManagedConnection {
id;
transport;
createdAt;
metadata;
state = 'idle';
lastUsedAt;
useCount = 0;
constructor(id, transport, createdAt = new Date(), metadata) {
this.id = id;
this.transport = transport;
this.createdAt = createdAt;
this.metadata = metadata;
this.lastUsedAt = this.createdAt;
}
acquire() {
this.state = 'busy';
this.lastUsedAt = new Date();
this.useCount++;
}
release() {
this.state = 'idle';
this.lastUsedAt = new Date();
}
isExpired(idleTimeout) {
if (this.state !== 'idle')
return false;
return Date.now() - this.lastUsedAt.getTime() > idleTimeout;
}
isHealthy() {
return this.state !== 'error' && this.state !== 'closed';
}
}
export class ConnectionPool extends EventEmitter {
logger;
transportType;
config;
connections = new Map();
waitingClients = [];
evictionTimer;
connectionCounter = 0;
isShuttingDown = false;
stats = {
totalAcquired: 0,
totalReleased: 0,
totalCreated: 0,
totalDestroyed: 0,
acquireTimeTotal: 0,
acquireCount: 0,
};
constructor(config = {}, logger, transportType = 'in-process') {
super();
this.logger = logger;
this.transportType = transportType;
this.config = { ...DEFAULT_POOL_CONFIG, ...config };
this.startEvictionTimer();
this.initializeMinConnections();
}
async initializeMinConnections() {
const promises = [];
for (let i = 0; i < this.config.minConnections; i++) {
promises.push(this.createConnection());
}
await Promise.all(promises);
this.logger.debug('Connection pool initialized', {
minConnections: this.config.minConnections,
});
}
async createConnection() {
const id = `conn-${++this.connectionCounter}-${Date.now()}`;
const connection = new ManagedConnection(id, this.transportType);
this.connections.set(id, connection);
this.stats.totalCreated++;
this.emit('pool:connection:created', { connectionId: id });
this.logger.debug('Connection created', { id, total: this.connections.size });
return connection;
}
async acquire() {
const startTime = performance.now();
if (this.isShuttingDown) {
throw new Error('Connection pool is shutting down');
}
for (const connection of this.connections.values()) {
if (connection.state === 'idle' && connection.isHealthy()) {
connection.acquire();
this.stats.totalAcquired++;
this.recordAcquireTime(startTime);
this.emit('pool:connection:acquired', { connectionId: connection.id });
this.logger.debug('Connection acquired from pool', { id: connection.id });
return connection;
}
}
if (this.connections.size < this.config.maxConnections) {
const connection = await this.createConnection();
connection.acquire();
this.stats.totalAcquired++;
this.recordAcquireTime(startTime);
this.emit('pool:connection:acquired', { connectionId: connection.id });
return connection;
}
return this.waitForConnection(startTime);
}
waitForConnection(startTime) {
return new Promise((resolve, reject) => {
if (this.waitingClients.length >= this.config.maxWaitingClients) {
reject(new Error('Connection pool exhausted - max waiting clients reached'));
return;
}
const client = {
resolve: (connection) => {
this.recordAcquireTime(startTime);
resolve(connection);
},
reject,
timestamp: Date.now(),
};
this.waitingClients.push(client);
setTimeout(() => {
const index = this.waitingClients.indexOf(client);
if (index !== -1) {
this.waitingClients.splice(index, 1);
reject(new Error(`Connection acquire timeout after ${this.config.acquireTimeout}ms`));
}
}, this.config.acquireTimeout);
});
}
release(connection) {
const managed = this.connections.get(connection.id);
if (!managed) {
this.logger.warn('Attempted to release unknown connection', { id: connection.id });
return;
}
const waitingClient = this.waitingClients.shift();
if (waitingClient) {
managed.acquire();
this.stats.totalAcquired++;
this.emit('pool:connection:acquired', { connectionId: connection.id });
waitingClient.resolve(managed);
return;
}
managed.release();
this.stats.totalReleased++;
this.emit('pool:connection:released', { connectionId: connection.id });
this.logger.debug('Connection released to pool', { id: connection.id });
}
destroy(connection) {
const managed = this.connections.get(connection.id);
if (!managed) {
return;
}
managed.state = 'closed';
this.connections.delete(connection.id);
this.stats.totalDestroyed++;
this.emit('pool:connection:destroyed', { connectionId: connection.id });
this.logger.debug('Connection destroyed', { id: connection.id });
if (this.connections.size < this.config.minConnections && !this.isShuttingDown) {
this.createConnection().catch((err) => {
this.logger.error('Failed to create replacement connection', err);
});
}
}
getStats() {
let idleCount = 0;
let busyCount = 0;
for (const connection of this.connections.values()) {
if (connection.state === 'idle')
idleCount++;
else if (connection.state === 'busy')
busyCount++;
}
return {
totalConnections: this.connections.size,
idleConnections: idleCount,
busyConnections: busyCount,
pendingRequests: this.waitingClients.length,
totalAcquired: this.stats.totalAcquired,
totalReleased: this.stats.totalReleased,
totalCreated: this.stats.totalCreated,
totalDestroyed: this.stats.totalDestroyed,
avgAcquireTime: this.stats.acquireCount > 0
? this.stats.acquireTimeTotal / this.stats.acquireCount
: 0,
};
}
async drain() {
this.isShuttingDown = true;
this.logger.info('Draining connection pool');
while (this.waitingClients.length > 0) {
const client = this.waitingClients.shift();
client?.reject(new Error('Connection pool is draining'));
}
const maxWait = 10000;
const startTime = Date.now();
while (Date.now() - startTime < maxWait) {
let busyCount = 0;
for (const connection of this.connections.values()) {
if (connection.state === 'busy')
busyCount++;
}
if (busyCount === 0)
break;
await new Promise((resolve) => setTimeout(resolve, 100));
}
this.logger.info('Connection pool drained');
}
async clear() {
this.stopEvictionTimer();
await this.drain();
for (const connection of this.connections.values()) {
connection.state = 'closed';
}
this.connections.clear();
this.logger.info('Connection pool cleared');
}
startEvictionTimer() {
this.evictionTimer = setInterval(() => {
this.evictIdleConnections();
}, this.config.evictionRunInterval);
}
stopEvictionTimer() {
if (this.evictionTimer) {
clearInterval(this.evictionTimer);
this.evictionTimer = undefined;
}
}
evictIdleConnections() {
if (this.isShuttingDown)
return;
const toEvict = [];
for (const connection of this.connections.values()) {
if (connection.isExpired(this.config.idleTimeout) &&
this.connections.size > this.config.minConnections) {
toEvict.push(connection);
}
}
for (const connection of toEvict) {
this.destroy(connection);
this.logger.debug('Evicted idle connection', { id: connection.id });
}
if (toEvict.length > 0) {
this.logger.info('Evicted idle connections', { count: toEvict.length });
}
}
recordAcquireTime(startTime) {
const duration = performance.now() - startTime;
this.stats.acquireTimeTotal += duration;
this.stats.acquireCount++;
}
getConnections() {
return Array.from(this.connections.values());
}
isHealthy() {
return !this.isShuttingDown && this.connections.size >= this.config.minConnections;
}
}
export function createConnectionPool(config = {}, logger, transportType = 'in-process') {
return new ConnectionPool(config, logger, transportType);
}
//# sourceMappingURL=connection-pool.js.map