@pipeline-builder/api-server 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 +39 -0
- package/lib/api/app-factory.d.ts +71 -0
- package/lib/api/app-factory.js +212 -0
- package/lib/api/check-quota.d.ts +13 -0
- package/lib/api/check-quota.js +67 -0
- package/lib/api/context-middleware.d.ts +34 -0
- package/lib/api/context-middleware.js +45 -0
- package/lib/api/etag-middleware.d.ts +7 -0
- package/lib/api/etag-middleware.js +37 -0
- package/lib/api/get-context.d.ts +22 -0
- package/lib/api/get-context.js +31 -0
- package/lib/api/health-checks.d.ts +25 -0
- package/lib/api/health-checks.js +36 -0
- package/lib/api/idempotency-middleware.d.ts +9 -0
- package/lib/api/idempotency-middleware.js +64 -0
- package/lib/api/index.d.ts +15 -0
- package/lib/api/index.js +43 -0
- package/lib/api/metrics.d.ts +12 -0
- package/lib/api/metrics.js +83 -0
- package/lib/api/middleware-factory.d.ts +47 -0
- package/lib/api/middleware-factory.js +66 -0
- package/lib/api/middleware.d.ts +1 -0
- package/lib/api/middleware.js +14 -0
- package/lib/api/quota-helpers.d.ts +23 -0
- package/lib/api/quota-helpers.js +25 -0
- package/lib/api/request-types.d.ts +55 -0
- package/lib/api/request-types.js +62 -0
- package/lib/api/require-org-id.d.ts +14 -0
- package/lib/api/require-org-id.js +31 -0
- package/lib/api/route-wrapper.d.ts +50 -0
- package/lib/api/route-wrapper.js +62 -0
- package/lib/api/server.d.ts +79 -0
- package/lib/api/server.js +144 -0
- package/lib/api/tracing.d.ts +15 -0
- package/lib/api/tracing.js +53 -0
- package/lib/http/index.d.ts +2 -0
- package/lib/http/index.js +21 -0
- package/lib/http/sse-connection-manager.d.ts +145 -0
- package/lib/http/sse-connection-manager.js +329 -0
- package/lib/http/ws-manager.d.ts +37 -0
- package/lib/http/ws-manager.js +105 -0
- package/lib/index.d.ts +30 -0
- package/lib/index.js +51 -0
- package/package.json +143 -0
|
@@ -0,0 +1,329 @@
|
|
|
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.SSEManager = void 0;
|
|
6
|
+
const api_core_1 = require("@pipeline-builder/api-core");
|
|
7
|
+
const pipeline_core_1 = require("@pipeline-builder/pipeline-core");
|
|
8
|
+
const uuid_1 = require("uuid");
|
|
9
|
+
const logger = (0, api_core_1.createLogger)('SSEManager');
|
|
10
|
+
/**
|
|
11
|
+
* SSE helper class with memory leak protection
|
|
12
|
+
*
|
|
13
|
+
* Features:
|
|
14
|
+
* - Client limits per request ID
|
|
15
|
+
* - Automatic timeout for idle connections
|
|
16
|
+
* - Periodic cleanup of stale connections
|
|
17
|
+
* - Connection statistics
|
|
18
|
+
*
|
|
19
|
+
* @example
|
|
20
|
+
* ```typescript
|
|
21
|
+
* const sseManager = new SSEManager({ maxClientsPerRequest: 5 });
|
|
22
|
+
* app.get('/logs/:requestId', sseManager.middleware());
|
|
23
|
+
*
|
|
24
|
+
* // Send events
|
|
25
|
+
* sseManager.send('request-123', 'INFO', 'Processing...');
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
class SSEManager {
|
|
29
|
+
clients = new Map();
|
|
30
|
+
maxClientsPerRequest;
|
|
31
|
+
clientTimeoutMs;
|
|
32
|
+
cleanupInterval = null;
|
|
33
|
+
constructor(options = {}) {
|
|
34
|
+
this.maxClientsPerRequest = options.maxClientsPerRequest ?? parseInt(process.env.SSE_MAX_CLIENTS_PER_REQUEST || '10', 10);
|
|
35
|
+
this.clientTimeoutMs = options.clientTimeoutMs ?? parseInt(process.env.SSE_CLIENT_TIMEOUT_MS || '1800000', 10); // 30 minutes
|
|
36
|
+
const cleanupIntervalMs = options.cleanupIntervalMs ?? parseInt(process.env.SSE_CLEANUP_INTERVAL_MS || '300000', 10); // 5 minutes
|
|
37
|
+
this.startCleanupInterval(cleanupIntervalMs);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Adds a client to the SSE manager
|
|
41
|
+
*
|
|
42
|
+
* @param requestId - Unique request ID
|
|
43
|
+
* @param res - Express Response object
|
|
44
|
+
* @returns true if client was added, false if rejected (limit reached)
|
|
45
|
+
*/
|
|
46
|
+
addClient(requestId, res) {
|
|
47
|
+
const existing = this.clients.get(requestId) || [];
|
|
48
|
+
// Check client limit
|
|
49
|
+
if (existing.length >= this.maxClientsPerRequest) {
|
|
50
|
+
logger.warn(`Client limit reached for request ${requestId} (max: ${this.maxClientsPerRequest})`);
|
|
51
|
+
return false;
|
|
52
|
+
}
|
|
53
|
+
// Create timeout for this client
|
|
54
|
+
const clientId = (0, uuid_1.v7)();
|
|
55
|
+
const timeout = setTimeout(() => {
|
|
56
|
+
logger.debug(`Client ${clientId} timed out for request ${requestId}`);
|
|
57
|
+
this.removeClient(requestId, clientId);
|
|
58
|
+
try {
|
|
59
|
+
res.end();
|
|
60
|
+
}
|
|
61
|
+
catch (err) {
|
|
62
|
+
logger.debug('Response already closed on timeout', { requestId, clientId, error: err instanceof Error ? err.message : String(err) });
|
|
63
|
+
}
|
|
64
|
+
}, this.clientTimeoutMs);
|
|
65
|
+
const client = {
|
|
66
|
+
id: clientId,
|
|
67
|
+
res,
|
|
68
|
+
connectedAt: Date.now(),
|
|
69
|
+
timeout,
|
|
70
|
+
backpressureCount: 0,
|
|
71
|
+
};
|
|
72
|
+
// Handle disconnection
|
|
73
|
+
res.on('close', () => {
|
|
74
|
+
clearTimeout(timeout);
|
|
75
|
+
this.removeClient(requestId, clientId);
|
|
76
|
+
});
|
|
77
|
+
res.on('error', (err) => {
|
|
78
|
+
logger.error(`SSE client error for request ${requestId}:`, err);
|
|
79
|
+
clearTimeout(timeout);
|
|
80
|
+
this.removeClient(requestId, clientId);
|
|
81
|
+
});
|
|
82
|
+
existing.push(client);
|
|
83
|
+
this.clients.set(requestId, existing);
|
|
84
|
+
logger.debug(`Client ${clientId} connected for request ${requestId} (total: ${existing.length})`);
|
|
85
|
+
return true;
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Removes a client from the manager
|
|
89
|
+
*/
|
|
90
|
+
removeClient(requestId, clientId) {
|
|
91
|
+
const clients = this.clients.get(requestId);
|
|
92
|
+
if (!clients)
|
|
93
|
+
return;
|
|
94
|
+
const remaining = clients.filter(c => {
|
|
95
|
+
if (c.id === clientId) {
|
|
96
|
+
clearTimeout(c.timeout);
|
|
97
|
+
return false;
|
|
98
|
+
}
|
|
99
|
+
return true;
|
|
100
|
+
});
|
|
101
|
+
if (remaining.length === 0) {
|
|
102
|
+
this.clients.delete(requestId);
|
|
103
|
+
logger.debug(`All clients disconnected for request ${requestId}`);
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
this.clients.set(requestId, remaining);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Sends a message to all SSE clients for a requestId
|
|
111
|
+
*
|
|
112
|
+
* @param requestId - Request ID
|
|
113
|
+
* @param type - Event type
|
|
114
|
+
* @param message - Message string
|
|
115
|
+
* @param data - Optional additional data
|
|
116
|
+
* @returns Number of clients the message was sent to
|
|
117
|
+
*/
|
|
118
|
+
send(requestId, type, message, data) {
|
|
119
|
+
const payload = {
|
|
120
|
+
ts: new Date().toISOString(),
|
|
121
|
+
type,
|
|
122
|
+
message,
|
|
123
|
+
data,
|
|
124
|
+
};
|
|
125
|
+
const clients = [...(this.clients.get(requestId) || [])];
|
|
126
|
+
let sentCount = 0;
|
|
127
|
+
const serialized = `data: ${JSON.stringify(payload)}\n\n`;
|
|
128
|
+
for (const client of clients) {
|
|
129
|
+
try {
|
|
130
|
+
// Backpressure: skip clients whose write buffer is full
|
|
131
|
+
if (client.res.writableEnded) {
|
|
132
|
+
this.removeClient(requestId, client.id);
|
|
133
|
+
continue;
|
|
134
|
+
}
|
|
135
|
+
const canWrite = client.res.write(serialized);
|
|
136
|
+
if (!canWrite) {
|
|
137
|
+
client.backpressureCount++;
|
|
138
|
+
// Disconnect clients that consistently can't keep up (10 consecutive backpressure events)
|
|
139
|
+
if (client.backpressureCount >= pipeline_core_1.CoreConstants.SSE_BACKPRESSURE_THRESHOLD) {
|
|
140
|
+
logger.warn(`Disconnecting slow client ${client.id} for request ${requestId} (${client.backpressureCount} backpressure events)`);
|
|
141
|
+
this.removeClient(requestId, client.id);
|
|
142
|
+
try {
|
|
143
|
+
client.res.end();
|
|
144
|
+
}
|
|
145
|
+
catch { /* already closed */ }
|
|
146
|
+
continue;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
client.backpressureCount = 0; // Reset on successful write
|
|
151
|
+
}
|
|
152
|
+
sentCount++;
|
|
153
|
+
}
|
|
154
|
+
catch (error) {
|
|
155
|
+
logger.error(`Failed to send to client ${client.id}:`, error);
|
|
156
|
+
this.removeClient(requestId, client.id);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
return sentCount;
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Broadcast a message to all connected clients across all requests
|
|
163
|
+
*
|
|
164
|
+
* @param type - Event type
|
|
165
|
+
* @param message - Message string
|
|
166
|
+
* @param data - Optional additional data
|
|
167
|
+
* @returns Total number of clients the message was sent to
|
|
168
|
+
*/
|
|
169
|
+
broadcast(type, message, data) {
|
|
170
|
+
let totalSent = 0;
|
|
171
|
+
for (const requestId of this.clients.keys()) {
|
|
172
|
+
totalSent += this.send(requestId, type, message, data);
|
|
173
|
+
}
|
|
174
|
+
return totalSent;
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Close all clients for a specific request
|
|
178
|
+
*
|
|
179
|
+
* @param requestId - Request ID to close
|
|
180
|
+
* @param finalMessage - Optional final message to send before closing
|
|
181
|
+
*/
|
|
182
|
+
closeRequest(requestId, finalMessage) {
|
|
183
|
+
const clients = this.clients.get(requestId);
|
|
184
|
+
if (!clients)
|
|
185
|
+
return;
|
|
186
|
+
if (finalMessage) {
|
|
187
|
+
this.send(requestId, 'COMPLETED', finalMessage);
|
|
188
|
+
}
|
|
189
|
+
for (const client of clients) {
|
|
190
|
+
clearTimeout(client.timeout);
|
|
191
|
+
try {
|
|
192
|
+
client.res.end();
|
|
193
|
+
}
|
|
194
|
+
catch (err) {
|
|
195
|
+
logger.debug('Response already closed on request close', { requestId, clientId: client.id, error: err instanceof Error ? err.message : String(err) });
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
this.clients.delete(requestId);
|
|
199
|
+
logger.debug(`Closed all clients for request ${requestId}`);
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Get statistics about current connections
|
|
203
|
+
*/
|
|
204
|
+
getStats() {
|
|
205
|
+
let totalClients = 0;
|
|
206
|
+
let oldestConnection = null;
|
|
207
|
+
const now = Date.now();
|
|
208
|
+
for (const clients of this.clients.values()) {
|
|
209
|
+
totalClients += clients.length;
|
|
210
|
+
for (const client of clients) {
|
|
211
|
+
const age = now - client.connectedAt;
|
|
212
|
+
if (oldestConnection === null || age > oldestConnection) {
|
|
213
|
+
oldestConnection = age;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
return {
|
|
218
|
+
totalRequests: this.clients.size,
|
|
219
|
+
totalClients,
|
|
220
|
+
oldestConnectionMs: oldestConnection,
|
|
221
|
+
};
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Check if a request has any connected clients
|
|
225
|
+
*/
|
|
226
|
+
hasClients(requestId) {
|
|
227
|
+
const clients = this.clients.get(requestId);
|
|
228
|
+
return clients !== undefined && clients.length > 0;
|
|
229
|
+
}
|
|
230
|
+
/**
|
|
231
|
+
* Get the number of clients for a specific request
|
|
232
|
+
*/
|
|
233
|
+
getClientCount(requestId) {
|
|
234
|
+
return this.clients.get(requestId)?.length ?? 0;
|
|
235
|
+
}
|
|
236
|
+
/**
|
|
237
|
+
* Middleware to initialize SSE connection
|
|
238
|
+
*
|
|
239
|
+
* @example
|
|
240
|
+
* ```typescript
|
|
241
|
+
* app.get('/logs/:requestId', sseManager.middleware());
|
|
242
|
+
* ```
|
|
243
|
+
*/
|
|
244
|
+
middleware() {
|
|
245
|
+
const UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;
|
|
246
|
+
return (req, res) => {
|
|
247
|
+
const { requestId } = req.params;
|
|
248
|
+
if (!UUID_RE.test(requestId)) {
|
|
249
|
+
res.status(400).end('Invalid requestId format');
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
// Check client limit BEFORE flushing headers, so we can still send 429
|
|
253
|
+
const existing = this.clients.get(requestId) || [];
|
|
254
|
+
if (existing.length >= this.maxClientsPerRequest) {
|
|
255
|
+
logger.warn(`Client limit reached for request ${requestId} (max: ${this.maxClientsPerRequest})`);
|
|
256
|
+
res.status(429).end('Too many connections for this request');
|
|
257
|
+
return;
|
|
258
|
+
}
|
|
259
|
+
res.setHeader('Content-Type', 'text/event-stream');
|
|
260
|
+
res.setHeader('Cache-Control', 'no-cache');
|
|
261
|
+
res.setHeader('Connection', 'keep-alive');
|
|
262
|
+
res.setHeader('X-Accel-Buffering', 'no'); // Disable nginx buffering
|
|
263
|
+
res.flushHeaders();
|
|
264
|
+
this.addClient(requestId, res);
|
|
265
|
+
};
|
|
266
|
+
}
|
|
267
|
+
/**
|
|
268
|
+
* Start periodic cleanup of stale connections
|
|
269
|
+
*/
|
|
270
|
+
startCleanupInterval(intervalMs) {
|
|
271
|
+
this.cleanupInterval = setInterval(() => {
|
|
272
|
+
this.cleanup();
|
|
273
|
+
}, intervalMs);
|
|
274
|
+
// Don't prevent process exit
|
|
275
|
+
this.cleanupInterval.unref();
|
|
276
|
+
}
|
|
277
|
+
/**
|
|
278
|
+
* Clean up stale connections using single-pass partition.
|
|
279
|
+
* O(R × C) instead of O(R × C²), and avoids mutation-during-iteration.
|
|
280
|
+
*/
|
|
281
|
+
cleanup() {
|
|
282
|
+
const now = Date.now();
|
|
283
|
+
let cleaned = 0;
|
|
284
|
+
for (const [requestId, clients] of this.clients.entries()) {
|
|
285
|
+
const stale = [];
|
|
286
|
+
const active = [];
|
|
287
|
+
for (const client of clients) {
|
|
288
|
+
if (now - client.connectedAt > this.clientTimeoutMs) {
|
|
289
|
+
stale.push(client);
|
|
290
|
+
}
|
|
291
|
+
else {
|
|
292
|
+
active.push(client);
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
for (const client of stale) {
|
|
296
|
+
clearTimeout(client.timeout);
|
|
297
|
+
try {
|
|
298
|
+
client.res.end();
|
|
299
|
+
}
|
|
300
|
+
catch { /* already closed */ }
|
|
301
|
+
cleaned++;
|
|
302
|
+
}
|
|
303
|
+
if (active.length === 0) {
|
|
304
|
+
this.clients.delete(requestId);
|
|
305
|
+
}
|
|
306
|
+
else if (stale.length > 0) {
|
|
307
|
+
this.clients.set(requestId, active);
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
if (cleaned > 0) {
|
|
311
|
+
logger.info(`Cleaned up ${cleaned} stale SSE connections`);
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
/**
|
|
315
|
+
* Shutdown the SSE manager and close all connections
|
|
316
|
+
*/
|
|
317
|
+
shutdown() {
|
|
318
|
+
if (this.cleanupInterval) {
|
|
319
|
+
clearInterval(this.cleanupInterval);
|
|
320
|
+
this.cleanupInterval = null;
|
|
321
|
+
}
|
|
322
|
+
for (const requestId of [...this.clients.keys()]) {
|
|
323
|
+
this.closeRequest(requestId, 'Server shutting down');
|
|
324
|
+
}
|
|
325
|
+
logger.info('SSE Manager shut down');
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
exports.SSEManager = SSEManager;
|
|
329
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sse-connection-manager.js","sourceRoot":"","sources":["../../src/http/sse-connection-manager.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 { CoreConstants } from '@pipeline-builder/pipeline-core';\nimport { Response } from 'express';\nimport { v7 as uuid } from 'uuid';\n\nconst logger = createLogger('SSEManager');\n\n/**\n * Event types for SSE logging\n */\nexport type SSEEventType = 'INFO' | 'WARN' | 'ERROR' | 'COMPLETED' | 'ROLLBACK' | 'MESSAGE';\n\n/**\n * SSE payload structure\n */\nexport interface SSEPayload {\n  ts: string;\n  type: SSEEventType;\n  message: string;\n  data?: unknown;\n}\n\n/**\n * SSE client with connection tracking\n */\nexport interface SSEClient {\n  id: string;\n  res: Response;\n  connectedAt: number;\n  timeout: NodeJS.Timeout;\n  /** Number of consecutive backpressure events (write returned false). */\n  backpressureCount: number;\n}\n\n/**\n * SSE Manager configuration options\n */\nexport interface SSEManagerOptions {\n  /** Maximum clients allowed per request ID (default: 10) */\n  maxClientsPerRequest?: number;\n  /** Client timeout in milliseconds (default: 30 minutes) */\n  clientTimeoutMs?: number;\n  /** Interval for cleanup checks in milliseconds (default: 5 minutes) */\n  cleanupIntervalMs?: number;\n}\n\n/**\n * SSE Manager statistics\n */\nexport interface SSEManagerStats {\n  totalRequests: number;\n  totalClients: number;\n  oldestConnectionMs: number | null;\n}\n\n/**\n * SSE helper class with memory leak protection\n *\n * Features:\n * - Client limits per request ID\n * - Automatic timeout for idle connections\n * - Periodic cleanup of stale connections\n * - Connection statistics\n *\n * @example\n * ```typescript\n * const sseManager = new SSEManager({ maxClientsPerRequest: 5 });\n * app.get('/logs/:requestId', sseManager.middleware());\n *\n * // Send events\n * sseManager.send('request-123', 'INFO', 'Processing...');\n * ```\n */\nexport class SSEManager {\n  private clients = new Map<string, SSEClient[]>();\n  private readonly maxClientsPerRequest: number;\n  private readonly clientTimeoutMs: number;\n  private cleanupInterval: NodeJS.Timeout | null = null;\n\n  constructor(options: SSEManagerOptions = {}) {\n    this.maxClientsPerRequest = options.maxClientsPerRequest ?? parseInt(process.env.SSE_MAX_CLIENTS_PER_REQUEST || '10', 10);\n    this.clientTimeoutMs = options.clientTimeoutMs ?? parseInt(process.env.SSE_CLIENT_TIMEOUT_MS || '1800000', 10); // 30 minutes\n\n    const cleanupIntervalMs = options.cleanupIntervalMs ?? parseInt(process.env.SSE_CLEANUP_INTERVAL_MS || '300000', 10); // 5 minutes\n    this.startCleanupInterval(cleanupIntervalMs);\n  }\n\n  /**\n   * Adds a client to the SSE manager\n   *\n   * @param requestId - Unique request ID\n   * @param res - Express Response object\n   * @returns true if client was added, false if rejected (limit reached)\n   */\n  addClient(requestId: string, res: Response): boolean {\n    const existing = this.clients.get(requestId) || [];\n\n    // Check client limit\n    if (existing.length >= this.maxClientsPerRequest) {\n      logger.warn(`Client limit reached for request ${requestId} (max: ${this.maxClientsPerRequest})`);\n      return false;\n    }\n\n    // Create timeout for this client\n    const clientId = uuid();\n    const timeout = setTimeout(() => {\n      logger.debug(`Client ${clientId} timed out for request ${requestId}`);\n      this.removeClient(requestId, clientId);\n      try {\n        res.end();\n      } catch (err) {\n        logger.debug('Response already closed on timeout', { requestId, clientId, error: err instanceof Error ? err.message : String(err) });\n      }\n    }, this.clientTimeoutMs);\n\n    const client: SSEClient = {\n      id: clientId,\n      res,\n      connectedAt: Date.now(),\n      timeout,\n      backpressureCount: 0,\n    };\n\n    // Handle disconnection\n    res.on('close', () => {\n      clearTimeout(timeout);\n      this.removeClient(requestId, clientId);\n    });\n\n    res.on('error', (err) => {\n      logger.error(`SSE client error for request ${requestId}:`, err);\n      clearTimeout(timeout);\n      this.removeClient(requestId, clientId);\n    });\n\n    existing.push(client);\n    this.clients.set(requestId, existing);\n\n    logger.debug(`Client ${clientId} connected for request ${requestId} (total: ${existing.length})`);\n    return true;\n  }\n\n  /**\n   * Removes a client from the manager\n   */\n  private removeClient(requestId: string, clientId: string): void {\n    const clients = this.clients.get(requestId);\n    if (!clients) return;\n\n    const remaining = clients.filter(c => {\n      if (c.id === clientId) {\n        clearTimeout(c.timeout);\n        return false;\n      }\n      return true;\n    });\n\n    if (remaining.length === 0) {\n      this.clients.delete(requestId);\n      logger.debug(`All clients disconnected for request ${requestId}`);\n    } else {\n      this.clients.set(requestId, remaining);\n    }\n  }\n\n  /**\n   * Sends a message to all SSE clients for a requestId\n   *\n   * @param requestId - Request ID\n   * @param type - Event type\n   * @param message - Message string\n   * @param data - Optional additional data\n   * @returns Number of clients the message was sent to\n   */\n  send(requestId: string, type: SSEEventType, message: string, data?: unknown): number {\n    const payload: SSEPayload = {\n      ts: new Date().toISOString(),\n      type,\n      message,\n      data,\n    };\n\n    const clients = [...(this.clients.get(requestId) || [])];\n    let sentCount = 0;\n    const serialized = `data: ${JSON.stringify(payload)}\\n\\n`;\n\n    for (const client of clients) {\n      try {\n        // Backpressure: skip clients whose write buffer is full\n        if (client.res.writableEnded) {\n          this.removeClient(requestId, client.id);\n          continue;\n        }\n        const canWrite = client.res.write(serialized);\n        if (!canWrite) {\n          client.backpressureCount++;\n          // Disconnect clients that consistently can't keep up (10 consecutive backpressure events)\n          if (client.backpressureCount >= CoreConstants.SSE_BACKPRESSURE_THRESHOLD) {\n            logger.warn(`Disconnecting slow client ${client.id} for request ${requestId} (${client.backpressureCount} backpressure events)`);\n            this.removeClient(requestId, client.id);\n            try { client.res.end(); } catch { /* already closed */ }\n            continue;\n          }\n        } else {\n          client.backpressureCount = 0; // Reset on successful write\n        }\n        sentCount++;\n      } catch (error) {\n        logger.error(`Failed to send to client ${client.id}:`, error);\n        this.removeClient(requestId, client.id);\n      }\n    }\n\n    return sentCount;\n  }\n\n  /**\n   * Broadcast a message to all connected clients across all requests\n   *\n   * @param type - Event type\n   * @param message - Message string\n   * @param data - Optional additional data\n   * @returns Total number of clients the message was sent to\n   */\n  broadcast(type: SSEEventType, message: string, data?: unknown): number {\n    let totalSent = 0;\n    for (const requestId of this.clients.keys()) {\n      totalSent += this.send(requestId, type, message, data);\n    }\n    return totalSent;\n  }\n\n  /**\n   * Close all clients for a specific request\n   *\n   * @param requestId - Request ID to close\n   * @param finalMessage - Optional final message to send before closing\n   */\n  closeRequest(requestId: string, finalMessage?: string): void {\n    const clients = this.clients.get(requestId);\n    if (!clients) return;\n\n    if (finalMessage) {\n      this.send(requestId, 'COMPLETED', finalMessage);\n    }\n\n    for (const client of clients) {\n      clearTimeout(client.timeout);\n      try {\n        client.res.end();\n      } catch (err) {\n        logger.debug('Response already closed on request close', { requestId, clientId: client.id, error: err instanceof Error ? err.message : String(err) });\n      }\n    }\n\n    this.clients.delete(requestId);\n    logger.debug(`Closed all clients for request ${requestId}`);\n  }\n\n  /**\n   * Get statistics about current connections\n   */\n  getStats(): SSEManagerStats {\n    let totalClients = 0;\n    let oldestConnection: number | null = null;\n    const now = Date.now();\n\n    for (const clients of this.clients.values()) {\n      totalClients += clients.length;\n      for (const client of clients) {\n        const age = now - client.connectedAt;\n        if (oldestConnection === null || age > oldestConnection) {\n          oldestConnection = age;\n        }\n      }\n    }\n\n    return {\n      totalRequests: this.clients.size,\n      totalClients,\n      oldestConnectionMs: oldestConnection,\n    };\n  }\n\n  /**\n   * Check if a request has any connected clients\n   */\n  hasClients(requestId: string): boolean {\n    const clients = this.clients.get(requestId);\n    return clients !== undefined && clients.length > 0;\n  }\n\n  /**\n   * Get the number of clients for a specific request\n   */\n  getClientCount(requestId: string): number {\n    return this.clients.get(requestId)?.length ?? 0;\n  }\n\n  /**\n   * Middleware to initialize SSE connection\n   *\n   * @example\n   * ```typescript\n   * app.get('/logs/:requestId', sseManager.middleware());\n   * ```\n   */\n  middleware() {\n    const UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;\n\n    return (req: { params: { requestId: string } }, res: Response) => {\n      const { requestId } = req.params;\n\n      if (!UUID_RE.test(requestId)) {\n        res.status(400).end('Invalid requestId format');\n        return;\n      }\n\n      // Check client limit BEFORE flushing headers, so we can still send 429\n      const existing = this.clients.get(requestId) || [];\n      if (existing.length >= this.maxClientsPerRequest) {\n        logger.warn(`Client limit reached for request ${requestId} (max: ${this.maxClientsPerRequest})`);\n        res.status(429).end('Too many connections for this request');\n        return;\n      }\n\n      res.setHeader('Content-Type', 'text/event-stream');\n      res.setHeader('Cache-Control', 'no-cache');\n      res.setHeader('Connection', 'keep-alive');\n      res.setHeader('X-Accel-Buffering', 'no'); // Disable nginx buffering\n      res.flushHeaders();\n\n      this.addClient(requestId, res);\n    };\n  }\n\n  /**\n   * Start periodic cleanup of stale connections\n   */\n  private startCleanupInterval(intervalMs: number): void {\n    this.cleanupInterval = setInterval(() => {\n      this.cleanup();\n    }, intervalMs);\n\n    // Don't prevent process exit\n    this.cleanupInterval.unref();\n  }\n\n  /**\n   * Clean up stale connections using single-pass partition.\n   * O(R × C) instead of O(R × C²), and avoids mutation-during-iteration.\n   */\n  private cleanup(): void {\n    const now = Date.now();\n    let cleaned = 0;\n\n    for (const [requestId, clients] of this.clients.entries()) {\n      const stale: SSEClient[] = [];\n      const active: SSEClient[] = [];\n\n      for (const client of clients) {\n        if (now - client.connectedAt > this.clientTimeoutMs) {\n          stale.push(client);\n        } else {\n          active.push(client);\n        }\n      }\n\n      for (const client of stale) {\n        clearTimeout(client.timeout);\n        try { client.res.end(); } catch { /* already closed */ }\n        cleaned++;\n      }\n\n      if (active.length === 0) {\n        this.clients.delete(requestId);\n      } else if (stale.length > 0) {\n        this.clients.set(requestId, active);\n      }\n    }\n\n    if (cleaned > 0) {\n      logger.info(`Cleaned up ${cleaned} stale SSE connections`);\n    }\n  }\n\n  /**\n   * Shutdown the SSE manager and close all connections\n   */\n  shutdown(): void {\n    if (this.cleanupInterval) {\n      clearInterval(this.cleanupInterval);\n      this.cleanupInterval = null;\n    }\n\n    for (const requestId of [...this.clients.keys()]) {\n      this.closeRequest(requestId, 'Server shutting down');\n    }\n\n    logger.info('SSE Manager shut down');\n  }\n}\n"]}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
export interface WSClient {
|
|
2
|
+
id: string;
|
|
3
|
+
orgId: string;
|
|
4
|
+
send(data: string): void;
|
|
5
|
+
close(): void;
|
|
6
|
+
}
|
|
7
|
+
type MessageHandler = (client: WSClient, message: Record<string, unknown>) => void;
|
|
8
|
+
/**
|
|
9
|
+
* WebSocket connection manager.
|
|
10
|
+
* Manages authenticated WebSocket connections per org.
|
|
11
|
+
* Works alongside SSE for backward compatibility.
|
|
12
|
+
*
|
|
13
|
+
* Requires a WebSocket library (ws) to be wired in at server startup.
|
|
14
|
+
* This module provides the management layer only.
|
|
15
|
+
*/
|
|
16
|
+
export declare class WSManager {
|
|
17
|
+
private clients;
|
|
18
|
+
private handlers;
|
|
19
|
+
private maxClientsPerOrg;
|
|
20
|
+
constructor(options?: {
|
|
21
|
+
maxClientsPerOrg?: number;
|
|
22
|
+
});
|
|
23
|
+
addClient(client: WSClient): boolean;
|
|
24
|
+
removeClient(client: WSClient): void;
|
|
25
|
+
onMessage(type: string, handler: MessageHandler): void;
|
|
26
|
+
handleMessage(client: WSClient, raw: string): void;
|
|
27
|
+
sendToOrg(orgId: string, type: string, data: unknown): number;
|
|
28
|
+
broadcast(type: string, data: unknown): number;
|
|
29
|
+
getStats(): {
|
|
30
|
+
orgs: number;
|
|
31
|
+
clients: number;
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
export declare function createWSManager(options?: {
|
|
35
|
+
maxClientsPerOrg?: number;
|
|
36
|
+
}): WSManager;
|
|
37
|
+
export {};
|
|
@@ -0,0 +1,105 @@
|
|
|
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.WSManager = void 0;
|
|
6
|
+
exports.createWSManager = createWSManager;
|
|
7
|
+
const api_core_1 = require("@pipeline-builder/api-core");
|
|
8
|
+
const logger = (0, api_core_1.createLogger)('ws-manager');
|
|
9
|
+
/**
|
|
10
|
+
* WebSocket connection manager.
|
|
11
|
+
* Manages authenticated WebSocket connections per org.
|
|
12
|
+
* Works alongside SSE for backward compatibility.
|
|
13
|
+
*
|
|
14
|
+
* Requires a WebSocket library (ws) to be wired in at server startup.
|
|
15
|
+
* This module provides the management layer only.
|
|
16
|
+
*/
|
|
17
|
+
class WSManager {
|
|
18
|
+
clients = new Map();
|
|
19
|
+
handlers = new Map();
|
|
20
|
+
maxClientsPerOrg;
|
|
21
|
+
constructor(options = {}) {
|
|
22
|
+
this.maxClientsPerOrg = options.maxClientsPerOrg ?? 50;
|
|
23
|
+
}
|
|
24
|
+
addClient(client) {
|
|
25
|
+
if (!this.clients.has(client.orgId)) {
|
|
26
|
+
this.clients.set(client.orgId, new Set());
|
|
27
|
+
}
|
|
28
|
+
const orgClients = this.clients.get(client.orgId);
|
|
29
|
+
if (orgClients.size >= this.maxClientsPerOrg) {
|
|
30
|
+
logger.warn('Max WebSocket clients reached for org', { orgId: client.orgId });
|
|
31
|
+
return false;
|
|
32
|
+
}
|
|
33
|
+
orgClients.add(client);
|
|
34
|
+
logger.debug('WebSocket client connected', { orgId: client.orgId, clientId: client.id, total: orgClients.size });
|
|
35
|
+
return true;
|
|
36
|
+
}
|
|
37
|
+
removeClient(client) {
|
|
38
|
+
const orgClients = this.clients.get(client.orgId);
|
|
39
|
+
if (orgClients) {
|
|
40
|
+
orgClients.delete(client);
|
|
41
|
+
if (orgClients.size === 0)
|
|
42
|
+
this.clients.delete(client.orgId);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
onMessage(type, handler) {
|
|
46
|
+
this.handlers.set(type, handler);
|
|
47
|
+
}
|
|
48
|
+
handleMessage(client, raw) {
|
|
49
|
+
try {
|
|
50
|
+
const msg = JSON.parse(raw);
|
|
51
|
+
if (!msg.type)
|
|
52
|
+
return;
|
|
53
|
+
const handler = this.handlers.get(msg.type);
|
|
54
|
+
if (handler)
|
|
55
|
+
handler(client, msg);
|
|
56
|
+
}
|
|
57
|
+
catch {
|
|
58
|
+
logger.debug('Invalid WebSocket message', { clientId: client.id });
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
sendToOrg(orgId, type, data) {
|
|
62
|
+
const orgClients = this.clients.get(orgId);
|
|
63
|
+
if (!orgClients)
|
|
64
|
+
return 0;
|
|
65
|
+
const payload = JSON.stringify({ type, data, ts: new Date().toISOString() });
|
|
66
|
+
let sent = 0;
|
|
67
|
+
for (const client of orgClients) {
|
|
68
|
+
try {
|
|
69
|
+
client.send(payload);
|
|
70
|
+
sent++;
|
|
71
|
+
}
|
|
72
|
+
catch {
|
|
73
|
+
this.removeClient(client);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
return sent;
|
|
77
|
+
}
|
|
78
|
+
broadcast(type, data) {
|
|
79
|
+
const payload = JSON.stringify({ type, data, ts: new Date().toISOString() });
|
|
80
|
+
let sent = 0;
|
|
81
|
+
for (const [, orgClients] of this.clients) {
|
|
82
|
+
for (const client of orgClients) {
|
|
83
|
+
try {
|
|
84
|
+
client.send(payload);
|
|
85
|
+
sent++;
|
|
86
|
+
}
|
|
87
|
+
catch {
|
|
88
|
+
this.removeClient(client);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
return sent;
|
|
93
|
+
}
|
|
94
|
+
getStats() {
|
|
95
|
+
let clients = 0;
|
|
96
|
+
for (const [, orgClients] of this.clients)
|
|
97
|
+
clients += orgClients.size;
|
|
98
|
+
return { orgs: this.clients.size, clients };
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
exports.WSManager = WSManager;
|
|
102
|
+
function createWSManager(options) {
|
|
103
|
+
return new WSManager(options);
|
|
104
|
+
}
|
|
105
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ws-manager.js","sourceRoot":"","sources":["../../src/http/ws-manager.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';\n\nconst logger = createLogger('ws-manager');\n\nexport interface WSClient {\n  id: string;\n  orgId: string;\n  send(data: string): void;\n  close(): void;\n}\n\ntype MessageHandler = (client: WSClient, message: Record<string, unknown>) => void;\n\n/**\n * WebSocket connection manager.\n * Manages authenticated WebSocket connections per org.\n * Works alongside SSE for backward compatibility.\n *\n * Requires a WebSocket library (ws) to be wired in at server startup.\n * This module provides the management layer only.\n */\nexport class WSManager {\n  private clients = new Map<string, Set<WSClient>>();\n  private handlers = new Map<string, MessageHandler>();\n  private maxClientsPerOrg: number;\n\n  constructor(options: { maxClientsPerOrg?: number } = {}) {\n    this.maxClientsPerOrg = options.maxClientsPerOrg ?? 50;\n  }\n\n  addClient(client: WSClient): boolean {\n    if (!this.clients.has(client.orgId)) {\n      this.clients.set(client.orgId, new Set());\n    }\n    const orgClients = this.clients.get(client.orgId)!;\n    if (orgClients.size >= this.maxClientsPerOrg) {\n      logger.warn('Max WebSocket clients reached for org', { orgId: client.orgId });\n      return false;\n    }\n    orgClients.add(client);\n    logger.debug('WebSocket client connected', { orgId: client.orgId, clientId: client.id, total: orgClients.size });\n    return true;\n  }\n\n  removeClient(client: WSClient): void {\n    const orgClients = this.clients.get(client.orgId);\n    if (orgClients) {\n      orgClients.delete(client);\n      if (orgClients.size === 0) this.clients.delete(client.orgId);\n    }\n  }\n\n  onMessage(type: string, handler: MessageHandler): void {\n    this.handlers.set(type, handler);\n  }\n\n  handleMessage(client: WSClient, raw: string): void {\n    try {\n      const msg = JSON.parse(raw) as { type?: string; [key: string]: unknown };\n      if (!msg.type) return;\n      const handler = this.handlers.get(msg.type);\n      if (handler) handler(client, msg);\n    } catch {\n      logger.debug('Invalid WebSocket message', { clientId: client.id });\n    }\n  }\n\n  sendToOrg(orgId: string, type: string, data: unknown): number {\n    const orgClients = this.clients.get(orgId);\n    if (!orgClients) return 0;\n    const payload = JSON.stringify({ type, data, ts: new Date().toISOString() });\n    let sent = 0;\n    for (const client of orgClients) {\n      try { client.send(payload); sent++; } catch { this.removeClient(client); }\n    }\n    return sent;\n  }\n\n  broadcast(type: string, data: unknown): number {\n    const payload = JSON.stringify({ type, data, ts: new Date().toISOString() });\n    let sent = 0;\n    for (const [, orgClients] of this.clients) {\n      for (const client of orgClients) {\n        try { client.send(payload); sent++; } catch { this.removeClient(client); }\n      }\n    }\n    return sent;\n  }\n\n  getStats(): { orgs: number; clients: number } {\n    let clients = 0;\n    for (const [, orgClients] of this.clients) clients += orgClients.size;\n    return { orgs: this.clients.size, clients };\n  }\n}\n\nexport function createWSManager(options?: { maxClientsPerOrg?: number }): WSManager {\n  return new WSManager(options);\n}\n"]}
|
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module @pipeline-builder/api-server
|
|
3
|
+
*
|
|
4
|
+
* Express server infrastructure and request lifecycle utilities.
|
|
5
|
+
*
|
|
6
|
+
* **App Factory**
|
|
7
|
+
* - createApp — creates a configured Express app with CORS, Helmet, rate limiting
|
|
8
|
+
* - runServer, startServer — server lifecycle with graceful shutdown
|
|
9
|
+
*
|
|
10
|
+
* **Middleware**
|
|
11
|
+
* - attachRequestContext / createRequestContext — attaches identity + logging to each request
|
|
12
|
+
* - requireOrgId — validates organization ID is present on the request
|
|
13
|
+
* - checkQuota — quota enforcement middleware
|
|
14
|
+
* - etagMiddleware — ETag-based conditional response support
|
|
15
|
+
* - idempotencyMiddleware — idempotent request handling
|
|
16
|
+
* - middlewareFactory — composable middleware builder
|
|
17
|
+
*
|
|
18
|
+
* **Route Helpers**
|
|
19
|
+
* - withRoute — wraps async route handlers with context extraction, orgId validation, and error handling
|
|
20
|
+
* - getContext — retrieves RequestContext from the Express request
|
|
21
|
+
* - RouteContext, RequestContext — route and request context types
|
|
22
|
+
*
|
|
23
|
+
* **SSE**
|
|
24
|
+
* - SSEManager — Server-Sent Events connection manager
|
|
25
|
+
*
|
|
26
|
+
* **Observability**
|
|
27
|
+
* - Tracing and metrics collection utilities
|
|
28
|
+
*/
|
|
29
|
+
export * from './api';
|
|
30
|
+
export * from './http';
|
package/lib/index.js
ADDED
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright 2026 Pipeline Builder Contributors
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
5
|
+
if (k2 === undefined) k2 = k;
|
|
6
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
7
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
8
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
9
|
+
}
|
|
10
|
+
Object.defineProperty(o, k2, desc);
|
|
11
|
+
}) : (function(o, m, k, k2) {
|
|
12
|
+
if (k2 === undefined) k2 = k;
|
|
13
|
+
o[k2] = m[k];
|
|
14
|
+
}));
|
|
15
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
16
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
17
|
+
};
|
|
18
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
19
|
+
/**
|
|
20
|
+
* @module @pipeline-builder/api-server
|
|
21
|
+
*
|
|
22
|
+
* Express server infrastructure and request lifecycle utilities.
|
|
23
|
+
*
|
|
24
|
+
* **App Factory**
|
|
25
|
+
* - createApp — creates a configured Express app with CORS, Helmet, rate limiting
|
|
26
|
+
* - runServer, startServer — server lifecycle with graceful shutdown
|
|
27
|
+
*
|
|
28
|
+
* **Middleware**
|
|
29
|
+
* - attachRequestContext / createRequestContext — attaches identity + logging to each request
|
|
30
|
+
* - requireOrgId — validates organization ID is present on the request
|
|
31
|
+
* - checkQuota — quota enforcement middleware
|
|
32
|
+
* - etagMiddleware — ETag-based conditional response support
|
|
33
|
+
* - idempotencyMiddleware — idempotent request handling
|
|
34
|
+
* - middlewareFactory — composable middleware builder
|
|
35
|
+
*
|
|
36
|
+
* **Route Helpers**
|
|
37
|
+
* - withRoute — wraps async route handlers with context extraction, orgId validation, and error handling
|
|
38
|
+
* - getContext — retrieves RequestContext from the Express request
|
|
39
|
+
* - RouteContext, RequestContext — route and request context types
|
|
40
|
+
*
|
|
41
|
+
* **SSE**
|
|
42
|
+
* - SSEManager — Server-Sent Events connection manager
|
|
43
|
+
*
|
|
44
|
+
* **Observability**
|
|
45
|
+
* - Tracing and metrics collection utilities
|
|
46
|
+
*/
|
|
47
|
+
// API Infrastructure
|
|
48
|
+
__exportStar(require("./api"), exports);
|
|
49
|
+
// HTTP Utilities
|
|
50
|
+
__exportStar(require("./http"), exports);
|
|
51
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFdEM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQTJCRztBQUVILHFCQUFxQjtBQUNyQix3Q0FBc0I7QUFFdEIsaUJBQWlCO0FBQ2pCLHlDQUF1QiIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG4vKipcbiAqIEBtb2R1bGUgQHBpcGVsaW5lLWJ1aWxkZXIvYXBpLXNlcnZlclxuICpcbiAqIEV4cHJlc3Mgc2VydmVyIGluZnJhc3RydWN0dXJlIGFuZCByZXF1ZXN0IGxpZmVjeWNsZSB1dGlsaXRpZXMuXG4gKlxuICogKipBcHAgRmFjdG9yeSoqXG4gKiAtIGNyZWF0ZUFwcCDigJQgY3JlYXRlcyBhIGNvbmZpZ3VyZWQgRXhwcmVzcyBhcHAgd2l0aCBDT1JTLCBIZWxtZXQsIHJhdGUgbGltaXRpbmdcbiAqIC0gcnVuU2VydmVyLCBzdGFydFNlcnZlciDigJQgc2VydmVyIGxpZmVjeWNsZSB3aXRoIGdyYWNlZnVsIHNodXRkb3duXG4gKlxuICogKipNaWRkbGV3YXJlKipcbiAqIC0gYXR0YWNoUmVxdWVzdENvbnRleHQgLyBjcmVhdGVSZXF1ZXN0Q29udGV4dCDigJQgYXR0YWNoZXMgaWRlbnRpdHkgKyBsb2dnaW5nIHRvIGVhY2ggcmVxdWVzdFxuICogLSByZXF1aXJlT3JnSWQg4oCUIHZhbGlkYXRlcyBvcmdhbml6YXRpb24gSUQgaXMgcHJlc2VudCBvbiB0aGUgcmVxdWVzdFxuICogLSBjaGVja1F1b3RhIOKAlCBxdW90YSBlbmZvcmNlbWVudCBtaWRkbGV3YXJlXG4gKiAtIGV0YWdNaWRkbGV3YXJlIOKAlCBFVGFnLWJhc2VkIGNvbmRpdGlvbmFsIHJlc3BvbnNlIHN1cHBvcnRcbiAqIC0gaWRlbXBvdGVuY3lNaWRkbGV3YXJlIOKAlCBpZGVtcG90ZW50IHJlcXVlc3QgaGFuZGxpbmdcbiAqIC0gbWlkZGxld2FyZUZhY3Rvcnkg4oCUIGNvbXBvc2FibGUgbWlkZGxld2FyZSBidWlsZGVyXG4gKlxuICogKipSb3V0ZSBIZWxwZXJzKipcbiAqIC0gd2l0aFJvdXRlIOKAlCB3cmFwcyBhc3luYyByb3V0ZSBoYW5kbGVycyB3aXRoIGNvbnRleHQgZXh0cmFjdGlvbiwgb3JnSWQgdmFsaWRhdGlvbiwgYW5kIGVycm9yIGhhbmRsaW5nXG4gKiAtIGdldENvbnRleHQg4oCUIHJldHJpZXZlcyBSZXF1ZXN0Q29udGV4dCBmcm9tIHRoZSBFeHByZXNzIHJlcXVlc3RcbiAqIC0gUm91dGVDb250ZXh0LCBSZXF1ZXN0Q29udGV4dCDigJQgcm91dGUgYW5kIHJlcXVlc3QgY29udGV4dCB0eXBlc1xuICpcbiAqICoqU1NFKipcbiAqIC0gU1NFTWFuYWdlciDigJQgU2VydmVyLVNlbnQgRXZlbnRzIGNvbm5lY3Rpb24gbWFuYWdlclxuICpcbiAqICoqT2JzZXJ2YWJpbGl0eSoqXG4gKiAtIFRyYWNpbmcgYW5kIG1ldHJpY3MgY29sbGVjdGlvbiB1dGlsaXRpZXNcbiAqL1xuXG4vLyBBUEkgSW5mcmFzdHJ1Y3R1cmVcbmV4cG9ydCAqIGZyb20gJy4vYXBpJztcblxuLy8gSFRUUCBVdGlsaXRpZXNcbmV4cG9ydCAqIGZyb20gJy4vaHR0cCc7Il19
|