claude-flow-novice 2.15.2 → 2.15.3

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.
Files changed (98) hide show
  1. package/.claude/hooks/cfn-BACKUP_USAGE.md +243 -243
  2. package/.claude/hooks/cfn-invoke-security-validation.sh +69 -69
  3. package/.claude/hooks/cfn-post-edit-cfn-retrospective.sh +78 -78
  4. package/.claude/hooks/cfn-post-edit.config.json +44 -44
  5. package/.claude/skills/agent-lifecycle/SKILL.md +60 -0
  6. package/.claude/skills/agent-lifecycle/execute-lifecycle-hook.sh +573 -0
  7. package/.claude/skills/agent-lifecycle/simple-audit.sh +31 -0
  8. package/.claude/skills/cfn-hybrid-routing/check-dependencies.sh +51 -51
  9. package/.claude/skills/cfn-loop-validation/orchestrate-cfn-loop.sh +252 -252
  10. package/.claude/skills/cfn-redis-coordination/agent-recovery.sh +74 -74
  11. package/.claude/skills/cfn-redis-coordination/get-context.sh +112 -112
  12. package/.claude/skills/cfn-transparency-middleware/middleware-config.sh +28 -28
  13. package/.claude/skills/cfn-transparency-middleware/performance-benchmark.sh +78 -78
  14. package/.claude/skills/cfn-transparency-middleware/test-integration.sh +161 -161
  15. package/.claude/skills/cfn-transparency-middleware/test-transparency-skill.sh +367 -367
  16. package/.claude/skills/cfn-transparency-middleware/tests/input-validation.sh +92 -92
  17. package/.claude/skills/cfn-transparency-middleware/wrap-agent.sh +131 -131
  18. package/claude-assets/hooks/cfn-BACKUP_USAGE.md +243 -243
  19. package/claude-assets/hooks/cfn-invoke-security-validation.sh +69 -69
  20. package/claude-assets/hooks/cfn-post-edit-cfn-retrospective.sh +78 -78
  21. package/claude-assets/hooks/cfn-post-edit.config.json +44 -44
  22. package/claude-assets/hooks/cfn-post-execution/memory-cleanup.sh +19 -19
  23. package/claude-assets/hooks/cfn-pre-execution/memory-check.sh +19 -19
  24. package/claude-assets/skills/agent-lifecycle/execute-lifecycle-hook.sh +572 -572
  25. package/claude-assets/skills/agent-lifecycle/simple-audit.sh +30 -30
  26. package/claude-assets/skills/cfn-automatic-memory-persistence/persist-agent-output.sh +48 -48
  27. package/claude-assets/skills/cfn-automatic-memory-persistence/query-agent-history.sh +34 -34
  28. package/claude-assets/skills/cfn-deliverable-validation/confidence-calculator.sh +261 -261
  29. package/claude-assets/skills/cfn-expert-update/update-expert.sh +345 -345
  30. package/claude-assets/skills/cfn-hybrid-routing/check-dependencies.sh +51 -51
  31. package/claude-assets/skills/cfn-intervention-detector/detect-intervention.sh +110 -110
  32. package/claude-assets/skills/cfn-intervention-orchestrator/execute-intervention.sh +58 -58
  33. package/claude-assets/skills/cfn-loop-validation/orchestrate-cfn-loop.sh +252 -252
  34. package/claude-assets/skills/cfn-loop2-output-processing/process-validator-output.sh +275 -275
  35. package/claude-assets/skills/cfn-memory-management/check-memory.sh +159 -159
  36. package/claude-assets/skills/cfn-memory-management/cleanup-memory.sh +196 -196
  37. package/claude-assets/skills/cfn-node-heap-sizer/task-mode-heap-limiter.sh +325 -325
  38. package/claude-assets/skills/cfn-playbook-auto-update/auto-update-playbook.sh +85 -85
  39. package/claude-assets/skills/cfn-redis-coordination/agent-recovery.sh +74 -74
  40. package/claude-assets/skills/cfn-redis-coordination/get-context.sh +112 -112
  41. package/claude-assets/skills/cfn-scope-simplifier/simplify-scope.sh +67 -67
  42. package/claude-assets/skills/cfn-specialist-injection/recommend-specialist.sh +56 -56
  43. package/claude-assets/skills/cfn-standardized-error-handling/capture-agent-error.sh +86 -86
  44. package/claude-assets/skills/cfn-standardized-error-handling/test-error-handling.sh +165 -165
  45. package/claude-assets/skills/cfn-task-config-init/initialize-config.sh +264 -264
  46. package/claude-assets/skills/cfn-task-decomposition/task-decomposer.sh +278 -278
  47. package/claude-assets/skills/cfn-transparency-middleware/middleware-config.sh +28 -28
  48. package/claude-assets/skills/cfn-transparency-middleware/performance-benchmark.sh +78 -78
  49. package/claude-assets/skills/cfn-transparency-middleware/test-integration.sh +161 -161
  50. package/claude-assets/skills/cfn-transparency-middleware/test-transparency-skill.sh +367 -367
  51. package/claude-assets/skills/cfn-transparency-middleware/tests/input-validation.sh +92 -92
  52. package/claude-assets/skills/cfn-transparency-middleware/wrap-agent.sh +131 -131
  53. package/claude-assets/skills/docker-build/SKILL.md +96 -203
  54. package/claude-assets/skills/docker-build/build.sh +73 -73
  55. package/claude-assets/skills/integration/agent-handoff.sh +494 -0
  56. package/claude-assets/skills/integration/file-operations.sh +414 -0
  57. package/claude-assets/skills/workflow-codification/APPROVAL_WORKFLOW.md +806 -0
  58. package/claude-assets/skills/workflow-codification/COST_TRACKING.md +637 -0
  59. package/claude-assets/skills/workflow-codification/EDGE_CASE_TRACKING.md +404 -0
  60. package/claude-assets/skills/workflow-codification/README_PHASE4.md +457 -0
  61. package/claude-assets/skills/workflow-codification/SKILL.md +110 -0
  62. package/claude-assets/skills/workflow-codification/analyze-patterns.sh +899 -0
  63. package/claude-assets/skills/workflow-codification/approval-workflow.sh +514 -0
  64. package/claude-assets/skills/workflow-codification/generate-skill-update.sh +525 -0
  65. package/claude-assets/skills/workflow-codification/review-skill.sh +643 -0
  66. package/claude-assets/skills/workflow-codification/templates/email-notification.txt +114 -0
  67. package/claude-assets/skills/workflow-codification/templates/slack-notification.md +85 -0
  68. package/claude-assets/skills/workflow-codification/test-integration.sh +281 -0
  69. package/claude-assets/skills/workflow-codification/track-cost-savings.sh +445 -0
  70. package/claude-assets/skills/workflow-codification/track-edge-case.sh +323 -0
  71. package/dist/cli/config-manager.js +91 -109
  72. package/dist/cli/config-manager.js.map +1 -1
  73. package/dist/integration/DatabaseHandoff.js +507 -0
  74. package/dist/integration/DatabaseHandoff.js.map +1 -0
  75. package/dist/integration/StandardAdapter.js +291 -0
  76. package/dist/integration/StandardAdapter.js.map +1 -0
  77. package/dist/lib/agent-output-parser.js +518 -0
  78. package/dist/lib/agent-output-parser.js.map +1 -0
  79. package/dist/lib/agent-output-validator.js +950 -0
  80. package/dist/lib/agent-output-validator.js.map +1 -0
  81. package/dist/lib/artifact-registry.js +443 -0
  82. package/dist/lib/artifact-registry.js.map +1 -0
  83. package/dist/lib/config-validator.js +687 -0
  84. package/dist/lib/config-validator.js.map +1 -0
  85. package/dist/types/agent-output.js +44 -0
  86. package/dist/types/agent-output.js.map +1 -0
  87. package/dist/types/config.js +28 -0
  88. package/dist/types/config.js.map +1 -0
  89. package/package.json +2 -1
  90. package/scripts/artifact-cleanup.sh +392 -0
  91. package/scripts/deploy-production.sh +355 -355
  92. package/scripts/docker-playwright-fix.sh +311 -311
  93. package/scripts/docker-rebuild-all-agents.sh +127 -127
  94. package/scripts/memory-leak-prevention.sh +305 -305
  95. package/scripts/migrate-artifacts.sh +563 -0
  96. package/scripts/migrate-yaml-to-json.sh +465 -0
  97. package/scripts/run-marketing-tests.sh +42 -42
  98. package/scripts/update_paths.sh +46 -46
@@ -0,0 +1,507 @@
1
+ /**
2
+ * DatabaseHandoff.ts - Standard database handoff patterns with cross-database correlation
3
+ *
4
+ * Features:
5
+ * - Cross-database correlation via task_id
6
+ * - Transaction management (begin/commit/rollback)
7
+ * - Query builder with standard correlation
8
+ * - Connection pooling (PostgreSQL, SQLite)
9
+ * - Automatic retry on transient failures
10
+ */ import { Pool as PgPool } from 'pg';
11
+ import sqlite3 from 'sqlite3';
12
+ import { StandardAdapter, JSONLogger } from './StandardAdapter';
13
+ /**
14
+ * DatabaseHandoff - Reference implementation for cross-database correlation
15
+ *
16
+ * @example
17
+ * ```typescript
18
+ * // PostgreSQL example
19
+ * const pgHandoff = new DatabaseHandoff({
20
+ * type: 'postgresql',
21
+ * pg: {
22
+ * host: 'localhost',
23
+ * port: 5432,
24
+ * database: 'cfn_db',
25
+ * user: 'cfn_user',
26
+ * password: 'secret',
27
+ * },
28
+ * }, {
29
+ * task_id: 'task-123',
30
+ * agent_id: 'agent-456',
31
+ * });
32
+ *
33
+ * await pgHandoff.initialize();
34
+ *
35
+ * // Create handoff with automatic correlation
36
+ * const handoff = await pgHandoff.createHandoff({
37
+ * source_agent_id: 'agent-456',
38
+ * target_agent_id: 'agent-789',
39
+ * payload: { data: 'example' },
40
+ * });
41
+ *
42
+ * // Query by task_id (cross-database correlation)
43
+ * const handoffs = await pgHandoff.getHandoffsByTaskId('task-123');
44
+ *
45
+ * // Transaction example
46
+ * await pgHandoff.withTransaction(async (tx) => {
47
+ * await tx.query('INSERT INTO tasks ...');
48
+ * await tx.query('UPDATE agents ...');
49
+ * // Automatic commit on success, rollback on error
50
+ * });
51
+ * ```
52
+ */ export class DatabaseHandoff {
53
+ config;
54
+ adapter;
55
+ logger;
56
+ // Connection pools
57
+ pg_pool;
58
+ sqlite_db;
59
+ // Initialization state
60
+ initialized = false;
61
+ constructor(config, context){
62
+ this.config = config;
63
+ this.logger = context.logger || new JSONLogger();
64
+ this.adapter = new StandardAdapter({
65
+ task_id: context.task_id,
66
+ agent_id: context.agent_id,
67
+ logger: this.logger
68
+ });
69
+ }
70
+ /**
71
+ * Initialize database connection and schema
72
+ */ async initialize() {
73
+ if (this.initialized) {
74
+ return;
75
+ }
76
+ try {
77
+ if (this.config.type === 'postgresql') {
78
+ await this.initializePostgreSQL();
79
+ } else if (this.config.type === 'sqlite') {
80
+ await this.initializeSQLite();
81
+ } else {
82
+ throw new Error(`Unsupported database type: ${this.config.type}`);
83
+ }
84
+ await this.ensureSchema();
85
+ this.initialized = true;
86
+ this.logger.info('Database handoff initialized', {
87
+ task_id: this.adapter.getContext().task_id,
88
+ database_type: this.config.type
89
+ });
90
+ } catch (error) {
91
+ this.logger.error('Failed to initialize database', {
92
+ task_id: this.adapter.getContext().task_id,
93
+ error: error instanceof Error ? error.message : String(error)
94
+ });
95
+ throw error;
96
+ }
97
+ }
98
+ /**
99
+ * Create a new handoff record
100
+ */ async createHandoff(params) {
101
+ this.ensureInitialized();
102
+ const handoff_id = this.generateHandoffId();
103
+ const { task_id } = this.adapter.getContext();
104
+ const handoff = {
105
+ handoff_id,
106
+ task_id,
107
+ source_agent_id: params.source_agent_id,
108
+ target_agent_id: params.target_agent_id,
109
+ status: 'pending',
110
+ payload: params.payload,
111
+ metadata: params.metadata,
112
+ created_at: new Date(),
113
+ updated_at: new Date()
114
+ };
115
+ return await this.adapter.withRetry(async ()=>{
116
+ if (this.config.type === 'postgresql') {
117
+ return await this.createHandoffPostgreSQL(handoff);
118
+ } else {
119
+ return await this.createHandoffSQLite(handoff);
120
+ }
121
+ });
122
+ }
123
+ /**
124
+ * Get handoff by ID
125
+ */ async getHandoff(handoff_id) {
126
+ this.ensureInitialized();
127
+ return await this.adapter.withRetry(async ()=>{
128
+ if (this.config.type === 'postgresql') {
129
+ return await this.getHandoffPostgreSQL(handoff_id);
130
+ } else {
131
+ return await this.getHandoffSQLite(handoff_id);
132
+ }
133
+ });
134
+ }
135
+ /**
136
+ * Get all handoffs for a task (cross-database correlation)
137
+ */ async getHandoffsByTaskId(task_id) {
138
+ this.ensureInitialized();
139
+ return await this.adapter.withRetry(async ()=>{
140
+ if (this.config.type === 'postgresql') {
141
+ return await this.getHandoffsByTaskIdPostgreSQL(task_id);
142
+ } else {
143
+ return await this.getHandoffsByTaskIdSQLite(task_id);
144
+ }
145
+ });
146
+ }
147
+ /**
148
+ * Update handoff status
149
+ */ async updateHandoffStatus(handoff_id, status, metadata) {
150
+ this.ensureInitialized();
151
+ await this.adapter.withRetry(async ()=>{
152
+ if (this.config.type === 'postgresql') {
153
+ await this.updateHandoffStatusPostgreSQL(handoff_id, status, metadata);
154
+ } else {
155
+ await this.updateHandoffStatusSQLite(handoff_id, status, metadata);
156
+ }
157
+ });
158
+ this.logger.info('Handoff status updated', {
159
+ handoff_id,
160
+ status,
161
+ task_id: this.adapter.getContext().task_id
162
+ });
163
+ }
164
+ /**
165
+ * Execute queries within a transaction
166
+ * Automatically commits on success, rolls back on error
167
+ */ async withTransaction(callback) {
168
+ this.ensureInitialized();
169
+ if (this.config.type === 'postgresql') {
170
+ return await this.withTransactionPostgreSQL(callback);
171
+ } else {
172
+ return await this.withTransactionSQLite(callback);
173
+ }
174
+ }
175
+ /**
176
+ * Close all database connections
177
+ */ async close() {
178
+ try {
179
+ if (this.pg_pool) {
180
+ await this.pg_pool.end();
181
+ this.logger.info('PostgreSQL connection pool closed');
182
+ }
183
+ if (this.sqlite_db) {
184
+ await this.sqlite_db.close();
185
+ this.logger.info('SQLite connection closed');
186
+ }
187
+ this.initialized = false;
188
+ } catch (error) {
189
+ this.logger.error('Error closing database connections', {
190
+ error: error instanceof Error ? error.message : String(error)
191
+ });
192
+ throw error;
193
+ }
194
+ }
195
+ // --- PostgreSQL Implementation ---
196
+ async initializePostgreSQL() {
197
+ if (!this.config.pg) {
198
+ throw new Error('PostgreSQL configuration missing');
199
+ }
200
+ this.pg_pool = new PgPool({
201
+ host: this.config.pg.host,
202
+ port: this.config.pg.port,
203
+ database: this.config.pg.database,
204
+ user: this.config.pg.user,
205
+ password: this.config.pg.password,
206
+ max: this.config.pg.max_connections || 10,
207
+ idleTimeoutMillis: 30000,
208
+ connectionTimeoutMillis: 5000
209
+ });
210
+ // Test connection
211
+ const client = await this.pg_pool.connect();
212
+ client.release();
213
+ }
214
+ async createHandoffPostgreSQL(handoff) {
215
+ const query = `
216
+ INSERT INTO handoffs (
217
+ handoff_id, task_id, source_agent_id, target_agent_id,
218
+ status, payload, metadata, created_at, updated_at
219
+ ) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9)
220
+ RETURNING *
221
+ `;
222
+ const values = [
223
+ handoff.handoff_id,
224
+ handoff.task_id,
225
+ handoff.source_agent_id,
226
+ handoff.target_agent_id || null,
227
+ handoff.status,
228
+ JSON.stringify(handoff.payload),
229
+ handoff.metadata ? JSON.stringify(handoff.metadata) : null,
230
+ handoff.created_at,
231
+ handoff.updated_at
232
+ ];
233
+ const result = await this.pg_pool.query(query, values);
234
+ return this.rowToHandoff(result.rows[0]);
235
+ }
236
+ async getHandoffPostgreSQL(handoff_id) {
237
+ const query = 'SELECT * FROM handoffs WHERE handoff_id = $1';
238
+ const result = await this.pg_pool.query(query, [
239
+ handoff_id
240
+ ]);
241
+ return result.rows.length > 0 ? this.rowToHandoff(result.rows[0]) : null;
242
+ }
243
+ async getHandoffsByTaskIdPostgreSQL(task_id) {
244
+ const query = 'SELECT * FROM handoffs WHERE task_id = $1 ORDER BY created_at DESC';
245
+ const result = await this.pg_pool.query(query, [
246
+ task_id
247
+ ]);
248
+ return result.rows.map((row)=>this.rowToHandoff(row));
249
+ }
250
+ async updateHandoffStatusPostgreSQL(handoff_id, status, metadata) {
251
+ const updates = [
252
+ 'status = $2',
253
+ 'updated_at = $3'
254
+ ];
255
+ const values = [
256
+ handoff_id,
257
+ status,
258
+ new Date()
259
+ ];
260
+ if (status === 'completed') {
261
+ updates.push('completed_at = $4');
262
+ values.push(new Date());
263
+ }
264
+ if (metadata) {
265
+ const idx = values.length + 1;
266
+ updates.push(`metadata = $${idx}`);
267
+ values.push(JSON.stringify(metadata));
268
+ }
269
+ const query = `UPDATE handoffs SET ${updates.join(', ')} WHERE handoff_id = $1`;
270
+ await this.pg_pool.query(query, values);
271
+ }
272
+ async withTransactionPostgreSQL(callback) {
273
+ const client = await this.pg_pool.connect();
274
+ try {
275
+ await client.query('BEGIN');
276
+ this.logger.debug('Transaction started (PostgreSQL)');
277
+ const tx = new TransactionClient(client, this.logger);
278
+ const result = await callback(tx);
279
+ await client.query('COMMIT');
280
+ this.logger.debug('Transaction committed (PostgreSQL)');
281
+ return result;
282
+ } catch (error) {
283
+ await client.query('ROLLBACK');
284
+ this.logger.warn('Transaction rolled back (PostgreSQL)', {
285
+ error: error instanceof Error ? error.message : String(error)
286
+ });
287
+ throw error;
288
+ } finally{
289
+ client.release();
290
+ }
291
+ }
292
+ // --- SQLite Implementation ---
293
+ async initializeSQLite() {
294
+ if (!this.config.sqlite) {
295
+ throw new Error('SQLite configuration missing');
296
+ }
297
+ const sqlite = await import('sqlite');
298
+ this.sqlite_db = await sqlite.open({
299
+ filename: this.config.sqlite.filepath,
300
+ driver: sqlite3.Database
301
+ });
302
+ }
303
+ async createHandoffSQLite(handoff) {
304
+ const query = `
305
+ INSERT INTO handoffs (
306
+ handoff_id, task_id, source_agent_id, target_agent_id,
307
+ status, payload, metadata, created_at, updated_at
308
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)
309
+ `;
310
+ await this.sqlite_db.run(query, [
311
+ handoff.handoff_id,
312
+ handoff.task_id,
313
+ handoff.source_agent_id,
314
+ handoff.target_agent_id || null,
315
+ handoff.status,
316
+ JSON.stringify(handoff.payload),
317
+ handoff.metadata ? JSON.stringify(handoff.metadata) : null,
318
+ handoff.created_at.toISOString(),
319
+ handoff.updated_at.toISOString()
320
+ ]);
321
+ return handoff;
322
+ }
323
+ async getHandoffSQLite(handoff_id) {
324
+ const query = 'SELECT * FROM handoffs WHERE handoff_id = ?';
325
+ const row = await this.sqlite_db.get(query, [
326
+ handoff_id
327
+ ]);
328
+ return row ? this.rowToHandoff(row) : null;
329
+ }
330
+ async getHandoffsByTaskIdSQLite(task_id) {
331
+ const query = 'SELECT * FROM handoffs WHERE task_id = ? ORDER BY created_at DESC';
332
+ const rows = await this.sqlite_db.all(query, [
333
+ task_id
334
+ ]);
335
+ return rows.map((row)=>this.rowToHandoff(row));
336
+ }
337
+ async updateHandoffStatusSQLite(handoff_id, status, metadata) {
338
+ let query = 'UPDATE handoffs SET status = ?, updated_at = ?';
339
+ const values = [
340
+ status,
341
+ new Date().toISOString()
342
+ ];
343
+ if (status === 'completed') {
344
+ query += ', completed_at = ?';
345
+ values.push(new Date().toISOString());
346
+ }
347
+ if (metadata) {
348
+ query += ', metadata = ?';
349
+ values.push(JSON.stringify(metadata));
350
+ }
351
+ query += ' WHERE handoff_id = ?';
352
+ values.push(handoff_id);
353
+ await this.sqlite_db.run(query, values);
354
+ }
355
+ async withTransactionSQLite(callback) {
356
+ try {
357
+ await this.sqlite_db.run('BEGIN TRANSACTION');
358
+ this.logger.debug('Transaction started (SQLite)');
359
+ const tx = new TransactionClient(this.sqlite_db, this.logger);
360
+ const result = await callback(tx);
361
+ await this.sqlite_db.run('COMMIT');
362
+ this.logger.debug('Transaction committed (SQLite)');
363
+ return result;
364
+ } catch (error) {
365
+ await this.sqlite_db.run('ROLLBACK');
366
+ this.logger.warn('Transaction rolled back (SQLite)', {
367
+ error: error instanceof Error ? error.message : String(error)
368
+ });
369
+ throw error;
370
+ }
371
+ }
372
+ // --- Schema Management ---
373
+ async ensureSchema() {
374
+ const schema = `
375
+ CREATE TABLE IF NOT EXISTS handoffs (
376
+ handoff_id TEXT PRIMARY KEY,
377
+ task_id TEXT NOT NULL,
378
+ source_agent_id TEXT NOT NULL,
379
+ target_agent_id TEXT,
380
+ status TEXT NOT NULL,
381
+ payload TEXT NOT NULL,
382
+ metadata TEXT,
383
+ created_at TEXT NOT NULL,
384
+ updated_at TEXT NOT NULL,
385
+ completed_at TEXT
386
+ );
387
+ CREATE INDEX IF NOT EXISTS idx_handoffs_task_id ON handoffs(task_id);
388
+ CREATE INDEX IF NOT EXISTS idx_handoffs_status ON handoffs(status);
389
+ CREATE INDEX IF NOT EXISTS idx_handoffs_created_at ON handoffs(created_at);
390
+ `;
391
+ if (this.config.type === 'postgresql') {
392
+ // PostgreSQL schema (adjust types)
393
+ const pgSchema = schema.replace(/TEXT/g, 'VARCHAR(255)').replace(/payload VARCHAR\(255\)/g, 'payload JSONB').replace(/metadata VARCHAR\(255\)/g, 'metadata JSONB').replace(/created_at VARCHAR\(255\)/g, 'created_at TIMESTAMP').replace(/updated_at VARCHAR\(255\)/g, 'updated_at TIMESTAMP').replace(/completed_at VARCHAR\(255\)/g, 'completed_at TIMESTAMP');
394
+ const statements = pgSchema.split(';').filter((s)=>s.trim());
395
+ for (const stmt of statements){
396
+ await this.pg_pool.query(stmt);
397
+ }
398
+ } else {
399
+ // SQLite schema
400
+ const statements = schema.split(';').filter((s)=>s.trim());
401
+ for (const stmt of statements){
402
+ await this.sqlite_db.run(stmt);
403
+ }
404
+ }
405
+ }
406
+ // --- Helper Methods ---
407
+ ensureInitialized() {
408
+ if (!this.initialized) {
409
+ throw new Error('DatabaseHandoff not initialized. Call initialize() first.');
410
+ }
411
+ }
412
+ generateHandoffId() {
413
+ const timestamp = Date.now();
414
+ const random = Math.random().toString(36).substring(2, 10);
415
+ return `handoff-${timestamp}-${random}`;
416
+ }
417
+ rowToHandoff(row) {
418
+ return {
419
+ handoff_id: row.handoff_id,
420
+ task_id: row.task_id,
421
+ source_agent_id: row.source_agent_id,
422
+ target_agent_id: row.target_agent_id,
423
+ status: row.status,
424
+ payload: typeof row.payload === 'string' ? JSON.parse(row.payload) : row.payload,
425
+ metadata: row.metadata ? typeof row.metadata === 'string' ? JSON.parse(row.metadata) : row.metadata : undefined,
426
+ created_at: new Date(row.created_at),
427
+ updated_at: new Date(row.updated_at),
428
+ completed_at: row.completed_at ? new Date(row.completed_at) : undefined
429
+ };
430
+ }
431
+ }
432
+ /**
433
+ * Transaction client for safe query execution within transactions
434
+ */ export class TransactionClient {
435
+ client;
436
+ logger;
437
+ constructor(client, logger){
438
+ this.client = client;
439
+ this.logger = logger;
440
+ }
441
+ async query(sql, params) {
442
+ this.logger.debug('Executing query in transaction', {
443
+ sql
444
+ });
445
+ if ('query' in this.client) {
446
+ // PostgreSQL
447
+ const result = await this.client.query(sql, params);
448
+ return result;
449
+ } else {
450
+ // SQLite
451
+ if (sql.trim().toUpperCase().startsWith('SELECT')) {
452
+ return await this.client.all(sql, params);
453
+ } else {
454
+ return await this.client.run(sql, params);
455
+ }
456
+ }
457
+ }
458
+ } /**
459
+ * USAGE EXAMPLE - Before (Ad-hoc database access):
460
+ *
461
+ * ```typescript
462
+ * // ❌ No correlation, no transaction safety, manual connection management
463
+ * import { Pool } from 'pg';
464
+ *
465
+ * const pool = new Pool({ ... });
466
+ *
467
+ * async function createTask(data: any) {
468
+ * const client = await pool.connect();
469
+ * try {
470
+ * await client.query('BEGIN');
471
+ * await client.query('INSERT INTO tasks ...');
472
+ * await client.query('INSERT INTO task_metadata ...');
473
+ * await client.query('COMMIT');
474
+ * } catch (err) {
475
+ * await client.query('ROLLBACK');
476
+ * throw err;
477
+ * } finally {
478
+ * client.release();
479
+ * }
480
+ * }
481
+ * ```
482
+ *
483
+ * USAGE EXAMPLE - After (Standardized with correlation):
484
+ *
485
+ * ```typescript
486
+ * // ✅ Automatic correlation, transaction safety, retry logic
487
+ * const handoff = new DatabaseHandoff({
488
+ * type: 'postgresql',
489
+ * pg: { host: 'localhost', port: 5432, database: 'cfn', user: 'user', password: 'pass' },
490
+ * }, {
491
+ * task_id: 'task-123',
492
+ * agent_id: 'agent-456',
493
+ * });
494
+ *
495
+ * await handoff.initialize();
496
+ *
497
+ * async function createTask(data: any) {
498
+ * await handoff.withTransaction(async (tx) => {
499
+ * await tx.query('INSERT INTO tasks (task_id, data) VALUES ($1, $2)', ['task-123', data]);
500
+ * await tx.query('INSERT INTO task_metadata (task_id, source) VALUES ($1, $2)', ['task-123', 'agent-456']);
501
+ * // Auto-commit on success, auto-rollback on error
502
+ * });
503
+ * }
504
+ * ```
505
+ */
506
+
507
+ //# sourceMappingURL=DatabaseHandoff.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/integration/DatabaseHandoff.ts"],"sourcesContent":["/**\r\n * DatabaseHandoff.ts - Standard database handoff patterns with cross-database correlation\r\n *\r\n * Features:\r\n * - Cross-database correlation via task_id\r\n * - Transaction management (begin/commit/rollback)\r\n * - Query builder with standard correlation\r\n * - Connection pooling (PostgreSQL, SQLite)\r\n * - Automatic retry on transient failures\r\n */\r\n\r\nimport { Pool as PgPool, PoolClient, QueryResult } from 'pg';\r\nimport sqlite3 from 'sqlite3';\r\nimport { Database as SqliteDB } from 'sqlite';\r\nimport { StandardAdapter, Logger, JSONLogger } from './StandardAdapter';\r\n\r\n/**\r\n * Standard handoff record structure\r\n */\r\nexport interface HandoffRecord {\r\n /** Unique handoff ID */\r\n handoff_id: string;\r\n /** Task correlation ID */\r\n task_id: string;\r\n /** Source agent ID */\r\n source_agent_id: string;\r\n /** Target agent ID (if known) */\r\n target_agent_id?: string;\r\n /** Handoff status: pending, in_progress, completed, failed */\r\n status: 'pending' | 'in_progress' | 'completed' | 'failed';\r\n /** Handoff payload (JSON) */\r\n payload: Record<string, unknown>;\r\n /** Optional metadata */\r\n metadata?: Record<string, unknown>;\r\n /** Creation timestamp */\r\n created_at: Date;\r\n /** Update timestamp */\r\n updated_at: Date;\r\n /** Completion timestamp */\r\n completed_at?: Date;\r\n}\r\n\r\n/**\r\n * Database connection configuration\r\n */\r\nexport interface DatabaseConfig {\r\n type: 'postgresql' | 'sqlite';\r\n /** PostgreSQL connection config */\r\n pg?: {\r\n host: string;\r\n port: number;\r\n database: string;\r\n user: string;\r\n password: string;\r\n max_connections?: number;\r\n };\r\n /** SQLite connection config */\r\n sqlite?: {\r\n filepath: string;\r\n mode?: number;\r\n };\r\n}\r\n\r\n/**\r\n * Transaction context for safe rollback\r\n */\r\ninterface TransactionContext {\r\n client?: PoolClient;\r\n in_transaction: boolean;\r\n}\r\n\r\n/**\r\n * DatabaseHandoff - Reference implementation for cross-database correlation\r\n *\r\n * @example\r\n * ```typescript\r\n * // PostgreSQL example\r\n * const pgHandoff = new DatabaseHandoff({\r\n * type: 'postgresql',\r\n * pg: {\r\n * host: 'localhost',\r\n * port: 5432,\r\n * database: 'cfn_db',\r\n * user: 'cfn_user',\r\n * password: 'secret',\r\n * },\r\n * }, {\r\n * task_id: 'task-123',\r\n * agent_id: 'agent-456',\r\n * });\r\n *\r\n * await pgHandoff.initialize();\r\n *\r\n * // Create handoff with automatic correlation\r\n * const handoff = await pgHandoff.createHandoff({\r\n * source_agent_id: 'agent-456',\r\n * target_agent_id: 'agent-789',\r\n * payload: { data: 'example' },\r\n * });\r\n *\r\n * // Query by task_id (cross-database correlation)\r\n * const handoffs = await pgHandoff.getHandoffsByTaskId('task-123');\r\n *\r\n * // Transaction example\r\n * await pgHandoff.withTransaction(async (tx) => {\r\n * await tx.query('INSERT INTO tasks ...');\r\n * await tx.query('UPDATE agents ...');\r\n * // Automatic commit on success, rollback on error\r\n * });\r\n * ```\r\n */\r\nexport class DatabaseHandoff {\r\n private config: DatabaseConfig;\r\n private adapter: StandardAdapter;\r\n private logger: Logger;\r\n\r\n // Connection pools\r\n private pg_pool?: PgPool;\r\n private sqlite_db?: SqliteDB;\r\n\r\n // Initialization state\r\n private initialized = false;\r\n\r\n constructor(\r\n config: DatabaseConfig,\r\n context: { task_id: string; agent_id?: string; logger?: Logger }\r\n ) {\r\n this.config = config;\r\n this.logger = context.logger || new JSONLogger();\r\n this.adapter = new StandardAdapter({\r\n task_id: context.task_id,\r\n agent_id: context.agent_id,\r\n logger: this.logger,\r\n });\r\n }\r\n\r\n /**\r\n * Initialize database connection and schema\r\n */\r\n async initialize(): Promise<void> {\r\n if (this.initialized) {\r\n return;\r\n }\r\n\r\n try {\r\n if (this.config.type === 'postgresql') {\r\n await this.initializePostgreSQL();\r\n } else if (this.config.type === 'sqlite') {\r\n await this.initializeSQLite();\r\n } else {\r\n throw new Error(`Unsupported database type: ${this.config.type}`);\r\n }\r\n\r\n await this.ensureSchema();\r\n this.initialized = true;\r\n\r\n this.logger.info('Database handoff initialized', {\r\n task_id: this.adapter.getContext().task_id,\r\n database_type: this.config.type,\r\n });\r\n } catch (error) {\r\n this.logger.error('Failed to initialize database', {\r\n task_id: this.adapter.getContext().task_id,\r\n error: error instanceof Error ? error.message : String(error),\r\n });\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Create a new handoff record\r\n */\r\n async createHandoff(params: {\r\n source_agent_id: string;\r\n target_agent_id?: string;\r\n payload: Record<string, unknown>;\r\n metadata?: Record<string, unknown>;\r\n }): Promise<HandoffRecord> {\r\n this.ensureInitialized();\r\n\r\n const handoff_id = this.generateHandoffId();\r\n const { task_id } = this.adapter.getContext();\r\n\r\n const handoff: HandoffRecord = {\r\n handoff_id,\r\n task_id,\r\n source_agent_id: params.source_agent_id,\r\n target_agent_id: params.target_agent_id,\r\n status: 'pending',\r\n payload: params.payload,\r\n metadata: params.metadata,\r\n created_at: new Date(),\r\n updated_at: new Date(),\r\n };\r\n\r\n return await this.adapter.withRetry(async () => {\r\n if (this.config.type === 'postgresql') {\r\n return await this.createHandoffPostgreSQL(handoff);\r\n } else {\r\n return await this.createHandoffSQLite(handoff);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Get handoff by ID\r\n */\r\n async getHandoff(handoff_id: string): Promise<HandoffRecord | null> {\r\n this.ensureInitialized();\r\n\r\n return await this.adapter.withRetry(async () => {\r\n if (this.config.type === 'postgresql') {\r\n return await this.getHandoffPostgreSQL(handoff_id);\r\n } else {\r\n return await this.getHandoffSQLite(handoff_id);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Get all handoffs for a task (cross-database correlation)\r\n */\r\n async getHandoffsByTaskId(task_id: string): Promise<HandoffRecord[]> {\r\n this.ensureInitialized();\r\n\r\n return await this.adapter.withRetry(async () => {\r\n if (this.config.type === 'postgresql') {\r\n return await this.getHandoffsByTaskIdPostgreSQL(task_id);\r\n } else {\r\n return await this.getHandoffsByTaskIdSQLite(task_id);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Update handoff status\r\n */\r\n async updateHandoffStatus(\r\n handoff_id: string,\r\n status: HandoffRecord['status'],\r\n metadata?: Record<string, unknown>\r\n ): Promise<void> {\r\n this.ensureInitialized();\r\n\r\n await this.adapter.withRetry(async () => {\r\n if (this.config.type === 'postgresql') {\r\n await this.updateHandoffStatusPostgreSQL(handoff_id, status, metadata);\r\n } else {\r\n await this.updateHandoffStatusSQLite(handoff_id, status, metadata);\r\n }\r\n });\r\n\r\n this.logger.info('Handoff status updated', {\r\n handoff_id,\r\n status,\r\n task_id: this.adapter.getContext().task_id,\r\n });\r\n }\r\n\r\n /**\r\n * Execute queries within a transaction\r\n * Automatically commits on success, rolls back on error\r\n */\r\n async withTransaction<T>(\r\n callback: (tx: TransactionClient) => Promise<T>\r\n ): Promise<T> {\r\n this.ensureInitialized();\r\n\r\n if (this.config.type === 'postgresql') {\r\n return await this.withTransactionPostgreSQL(callback);\r\n } else {\r\n return await this.withTransactionSQLite(callback);\r\n }\r\n }\r\n\r\n /**\r\n * Close all database connections\r\n */\r\n async close(): Promise<void> {\r\n try {\r\n if (this.pg_pool) {\r\n await this.pg_pool.end();\r\n this.logger.info('PostgreSQL connection pool closed');\r\n }\r\n\r\n if (this.sqlite_db) {\r\n await this.sqlite_db.close();\r\n this.logger.info('SQLite connection closed');\r\n }\r\n\r\n this.initialized = false;\r\n } catch (error) {\r\n this.logger.error('Error closing database connections', {\r\n error: error instanceof Error ? error.message : String(error),\r\n });\r\n throw error;\r\n }\r\n }\r\n\r\n // --- PostgreSQL Implementation ---\r\n\r\n private async initializePostgreSQL(): Promise<void> {\r\n if (!this.config.pg) {\r\n throw new Error('PostgreSQL configuration missing');\r\n }\r\n\r\n this.pg_pool = new PgPool({\r\n host: this.config.pg.host,\r\n port: this.config.pg.port,\r\n database: this.config.pg.database,\r\n user: this.config.pg.user,\r\n password: this.config.pg.password,\r\n max: this.config.pg.max_connections || 10,\r\n idleTimeoutMillis: 30000,\r\n connectionTimeoutMillis: 5000,\r\n });\r\n\r\n // Test connection\r\n const client = await this.pg_pool.connect();\r\n client.release();\r\n }\r\n\r\n private async createHandoffPostgreSQL(handoff: HandoffRecord): Promise<HandoffRecord> {\r\n const query = `\r\n INSERT INTO handoffs (\r\n handoff_id, task_id, source_agent_id, target_agent_id,\r\n status, payload, metadata, created_at, updated_at\r\n ) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9)\r\n RETURNING *\r\n `;\r\n\r\n const values = [\r\n handoff.handoff_id,\r\n handoff.task_id,\r\n handoff.source_agent_id,\r\n handoff.target_agent_id || null,\r\n handoff.status,\r\n JSON.stringify(handoff.payload),\r\n handoff.metadata ? JSON.stringify(handoff.metadata) : null,\r\n handoff.created_at,\r\n handoff.updated_at,\r\n ];\r\n\r\n const result = await this.pg_pool!.query(query, values);\r\n return this.rowToHandoff(result.rows[0]);\r\n }\r\n\r\n private async getHandoffPostgreSQL(handoff_id: string): Promise<HandoffRecord | null> {\r\n const query = 'SELECT * FROM handoffs WHERE handoff_id = $1';\r\n const result = await this.pg_pool!.query(query, [handoff_id]);\r\n return result.rows.length > 0 ? this.rowToHandoff(result.rows[0]) : null;\r\n }\r\n\r\n private async getHandoffsByTaskIdPostgreSQL(task_id: string): Promise<HandoffRecord[]> {\r\n const query = 'SELECT * FROM handoffs WHERE task_id = $1 ORDER BY created_at DESC';\r\n const result = await this.pg_pool!.query(query, [task_id]);\r\n return result.rows.map(row => this.rowToHandoff(row));\r\n }\r\n\r\n private async updateHandoffStatusPostgreSQL(\r\n handoff_id: string,\r\n status: HandoffRecord['status'],\r\n metadata?: Record<string, unknown>\r\n ): Promise<void> {\r\n const updates: string[] = ['status = $2', 'updated_at = $3'];\r\n const values: any[] = [handoff_id, status, new Date()];\r\n\r\n if (status === 'completed') {\r\n updates.push('completed_at = $4');\r\n values.push(new Date());\r\n }\r\n\r\n if (metadata) {\r\n const idx = values.length + 1;\r\n updates.push(`metadata = $${idx}`);\r\n values.push(JSON.stringify(metadata));\r\n }\r\n\r\n const query = `UPDATE handoffs SET ${updates.join(', ')} WHERE handoff_id = $1`;\r\n await this.pg_pool!.query(query, values);\r\n }\r\n\r\n private async withTransactionPostgreSQL<T>(\r\n callback: (tx: TransactionClient) => Promise<T>\r\n ): Promise<T> {\r\n const client = await this.pg_pool!.connect();\r\n\r\n try {\r\n await client.query('BEGIN');\r\n this.logger.debug('Transaction started (PostgreSQL)');\r\n\r\n const tx = new TransactionClient(client, this.logger);\r\n const result = await callback(tx);\r\n\r\n await client.query('COMMIT');\r\n this.logger.debug('Transaction committed (PostgreSQL)');\r\n\r\n return result;\r\n } catch (error) {\r\n await client.query('ROLLBACK');\r\n this.logger.warn('Transaction rolled back (PostgreSQL)', {\r\n error: error instanceof Error ? error.message : String(error),\r\n });\r\n throw error;\r\n } finally {\r\n client.release();\r\n }\r\n }\r\n\r\n // --- SQLite Implementation ---\r\n\r\n private async initializeSQLite(): Promise<void> {\r\n if (!this.config.sqlite) {\r\n throw new Error('SQLite configuration missing');\r\n }\r\n\r\n const sqlite = await import('sqlite');\r\n this.sqlite_db = await sqlite.open({\r\n filename: this.config.sqlite.filepath,\r\n driver: sqlite3.Database,\r\n });\r\n }\r\n\r\n private async createHandoffSQLite(handoff: HandoffRecord): Promise<HandoffRecord> {\r\n const query = `\r\n INSERT INTO handoffs (\r\n handoff_id, task_id, source_agent_id, target_agent_id,\r\n status, payload, metadata, created_at, updated_at\r\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\r\n `;\r\n\r\n await this.sqlite_db!.run(query, [\r\n handoff.handoff_id,\r\n handoff.task_id,\r\n handoff.source_agent_id,\r\n handoff.target_agent_id || null,\r\n handoff.status,\r\n JSON.stringify(handoff.payload),\r\n handoff.metadata ? JSON.stringify(handoff.metadata) : null,\r\n handoff.created_at.toISOString(),\r\n handoff.updated_at.toISOString(),\r\n ]);\r\n\r\n return handoff;\r\n }\r\n\r\n private async getHandoffSQLite(handoff_id: string): Promise<HandoffRecord | null> {\r\n const query = 'SELECT * FROM handoffs WHERE handoff_id = ?';\r\n const row = await this.sqlite_db!.get(query, [handoff_id]);\r\n return row ? this.rowToHandoff(row) : null;\r\n }\r\n\r\n private async getHandoffsByTaskIdSQLite(task_id: string): Promise<HandoffRecord[]> {\r\n const query = 'SELECT * FROM handoffs WHERE task_id = ? ORDER BY created_at DESC';\r\n const rows = await this.sqlite_db!.all(query, [task_id]);\r\n return rows.map(row => this.rowToHandoff(row));\r\n }\r\n\r\n private async updateHandoffStatusSQLite(\r\n handoff_id: string,\r\n status: HandoffRecord['status'],\r\n metadata?: Record<string, unknown>\r\n ): Promise<void> {\r\n let query = 'UPDATE handoffs SET status = ?, updated_at = ?';\r\n const values: any[] = [status, new Date().toISOString()];\r\n\r\n if (status === 'completed') {\r\n query += ', completed_at = ?';\r\n values.push(new Date().toISOString());\r\n }\r\n\r\n if (metadata) {\r\n query += ', metadata = ?';\r\n values.push(JSON.stringify(metadata));\r\n }\r\n\r\n query += ' WHERE handoff_id = ?';\r\n values.push(handoff_id);\r\n\r\n await this.sqlite_db!.run(query, values);\r\n }\r\n\r\n private async withTransactionSQLite<T>(\r\n callback: (tx: TransactionClient) => Promise<T>\r\n ): Promise<T> {\r\n try {\r\n await this.sqlite_db!.run('BEGIN TRANSACTION');\r\n this.logger.debug('Transaction started (SQLite)');\r\n\r\n const tx = new TransactionClient(this.sqlite_db!, this.logger);\r\n const result = await callback(tx);\r\n\r\n await this.sqlite_db!.run('COMMIT');\r\n this.logger.debug('Transaction committed (SQLite)');\r\n\r\n return result;\r\n } catch (error) {\r\n await this.sqlite_db!.run('ROLLBACK');\r\n this.logger.warn('Transaction rolled back (SQLite)', {\r\n error: error instanceof Error ? error.message : String(error),\r\n });\r\n throw error;\r\n }\r\n }\r\n\r\n // --- Schema Management ---\r\n\r\n private async ensureSchema(): Promise<void> {\r\n const schema = `\r\n CREATE TABLE IF NOT EXISTS handoffs (\r\n handoff_id TEXT PRIMARY KEY,\r\n task_id TEXT NOT NULL,\r\n source_agent_id TEXT NOT NULL,\r\n target_agent_id TEXT,\r\n status TEXT NOT NULL,\r\n payload TEXT NOT NULL,\r\n metadata TEXT,\r\n created_at TEXT NOT NULL,\r\n updated_at TEXT NOT NULL,\r\n completed_at TEXT\r\n );\r\n CREATE INDEX IF NOT EXISTS idx_handoffs_task_id ON handoffs(task_id);\r\n CREATE INDEX IF NOT EXISTS idx_handoffs_status ON handoffs(status);\r\n CREATE INDEX IF NOT EXISTS idx_handoffs_created_at ON handoffs(created_at);\r\n `;\r\n\r\n if (this.config.type === 'postgresql') {\r\n // PostgreSQL schema (adjust types)\r\n const pgSchema = schema\r\n .replace(/TEXT/g, 'VARCHAR(255)')\r\n .replace(/payload VARCHAR\\(255\\)/g, 'payload JSONB')\r\n .replace(/metadata VARCHAR\\(255\\)/g, 'metadata JSONB')\r\n .replace(/created_at VARCHAR\\(255\\)/g, 'created_at TIMESTAMP')\r\n .replace(/updated_at VARCHAR\\(255\\)/g, 'updated_at TIMESTAMP')\r\n .replace(/completed_at VARCHAR\\(255\\)/g, 'completed_at TIMESTAMP');\r\n\r\n const statements = pgSchema.split(';').filter(s => s.trim());\r\n for (const stmt of statements) {\r\n await this.pg_pool!.query(stmt);\r\n }\r\n } else {\r\n // SQLite schema\r\n const statements = schema.split(';').filter(s => s.trim());\r\n for (const stmt of statements) {\r\n await this.sqlite_db!.run(stmt);\r\n }\r\n }\r\n }\r\n\r\n // --- Helper Methods ---\r\n\r\n private ensureInitialized(): void {\r\n if (!this.initialized) {\r\n throw new Error('DatabaseHandoff not initialized. Call initialize() first.');\r\n }\r\n }\r\n\r\n private generateHandoffId(): string {\r\n const timestamp = Date.now();\r\n const random = Math.random().toString(36).substring(2, 10);\r\n return `handoff-${timestamp}-${random}`;\r\n }\r\n\r\n private rowToHandoff(row: any): HandoffRecord {\r\n return {\r\n handoff_id: row.handoff_id,\r\n task_id: row.task_id,\r\n source_agent_id: row.source_agent_id,\r\n target_agent_id: row.target_agent_id,\r\n status: row.status,\r\n payload: typeof row.payload === 'string' ? JSON.parse(row.payload) : row.payload,\r\n metadata: row.metadata\r\n ? typeof row.metadata === 'string'\r\n ? JSON.parse(row.metadata)\r\n : row.metadata\r\n : undefined,\r\n created_at: new Date(row.created_at),\r\n updated_at: new Date(row.updated_at),\r\n completed_at: row.completed_at ? new Date(row.completed_at) : undefined,\r\n };\r\n }\r\n}\r\n\r\n/**\r\n * Transaction client for safe query execution within transactions\r\n */\r\nexport class TransactionClient {\r\n constructor(\r\n private client: PoolClient | SqliteDB,\r\n private logger: Logger\r\n ) {}\r\n\r\n async query(sql: string, params?: any[]): Promise<any> {\r\n this.logger.debug('Executing query in transaction', { sql });\r\n\r\n if ('query' in this.client) {\r\n // PostgreSQL\r\n const result = await this.client.query(sql, params);\r\n return result;\r\n } else {\r\n // SQLite\r\n if (sql.trim().toUpperCase().startsWith('SELECT')) {\r\n return await this.client.all(sql, params);\r\n } else {\r\n return await this.client.run(sql, params);\r\n }\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * USAGE EXAMPLE - Before (Ad-hoc database access):\r\n *\r\n * ```typescript\r\n * // ❌ No correlation, no transaction safety, manual connection management\r\n * import { Pool } from 'pg';\r\n *\r\n * const pool = new Pool({ ... });\r\n *\r\n * async function createTask(data: any) {\r\n * const client = await pool.connect();\r\n * try {\r\n * await client.query('BEGIN');\r\n * await client.query('INSERT INTO tasks ...');\r\n * await client.query('INSERT INTO task_metadata ...');\r\n * await client.query('COMMIT');\r\n * } catch (err) {\r\n * await client.query('ROLLBACK');\r\n * throw err;\r\n * } finally {\r\n * client.release();\r\n * }\r\n * }\r\n * ```\r\n *\r\n * USAGE EXAMPLE - After (Standardized with correlation):\r\n *\r\n * ```typescript\r\n * // ✅ Automatic correlation, transaction safety, retry logic\r\n * const handoff = new DatabaseHandoff({\r\n * type: 'postgresql',\r\n * pg: { host: 'localhost', port: 5432, database: 'cfn', user: 'user', password: 'pass' },\r\n * }, {\r\n * task_id: 'task-123',\r\n * agent_id: 'agent-456',\r\n * });\r\n *\r\n * await handoff.initialize();\r\n *\r\n * async function createTask(data: any) {\r\n * await handoff.withTransaction(async (tx) => {\r\n * await tx.query('INSERT INTO tasks (task_id, data) VALUES ($1, $2)', ['task-123', data]);\r\n * await tx.query('INSERT INTO task_metadata (task_id, source) VALUES ($1, $2)', ['task-123', 'agent-456']);\r\n * // Auto-commit on success, auto-rollback on error\r\n * });\r\n * }\r\n * ```\r\n */\r\n"],"names":["Pool","PgPool","sqlite3","StandardAdapter","JSONLogger","DatabaseHandoff","config","adapter","logger","pg_pool","sqlite_db","initialized","context","task_id","agent_id","initialize","type","initializePostgreSQL","initializeSQLite","Error","ensureSchema","info","getContext","database_type","error","message","String","createHandoff","params","ensureInitialized","handoff_id","generateHandoffId","handoff","source_agent_id","target_agent_id","status","payload","metadata","created_at","Date","updated_at","withRetry","createHandoffPostgreSQL","createHandoffSQLite","getHandoff","getHandoffPostgreSQL","getHandoffSQLite","getHandoffsByTaskId","getHandoffsByTaskIdPostgreSQL","getHandoffsByTaskIdSQLite","updateHandoffStatus","updateHandoffStatusPostgreSQL","updateHandoffStatusSQLite","withTransaction","callback","withTransactionPostgreSQL","withTransactionSQLite","close","end","pg","host","port","database","user","password","max","max_connections","idleTimeoutMillis","connectionTimeoutMillis","client","connect","release","query","values","JSON","stringify","result","rowToHandoff","rows","length","map","row","updates","push","idx","join","debug","tx","TransactionClient","warn","sqlite","open","filename","filepath","driver","Database","run","toISOString","get","all","schema","pgSchema","replace","statements","split","filter","s","trim","stmt","timestamp","now","random","Math","toString","substring","parse","undefined","completed_at","sql","toUpperCase","startsWith"],"mappings":"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"}