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.
- package/.claude/hooks/cfn-BACKUP_USAGE.md +243 -243
- package/.claude/hooks/cfn-invoke-security-validation.sh +69 -69
- package/.claude/hooks/cfn-post-edit-cfn-retrospective.sh +78 -78
- package/.claude/hooks/cfn-post-edit.config.json +44 -44
- package/.claude/skills/agent-lifecycle/SKILL.md +60 -0
- package/.claude/skills/agent-lifecycle/execute-lifecycle-hook.sh +573 -0
- package/.claude/skills/agent-lifecycle/simple-audit.sh +31 -0
- package/.claude/skills/cfn-hybrid-routing/check-dependencies.sh +51 -51
- package/.claude/skills/cfn-loop-validation/orchestrate-cfn-loop.sh +252 -252
- package/.claude/skills/cfn-redis-coordination/agent-recovery.sh +74 -74
- package/.claude/skills/cfn-redis-coordination/get-context.sh +112 -112
- package/.claude/skills/cfn-transparency-middleware/middleware-config.sh +28 -28
- package/.claude/skills/cfn-transparency-middleware/performance-benchmark.sh +78 -78
- package/.claude/skills/cfn-transparency-middleware/test-integration.sh +161 -161
- package/.claude/skills/cfn-transparency-middleware/test-transparency-skill.sh +367 -367
- package/.claude/skills/cfn-transparency-middleware/tests/input-validation.sh +92 -92
- package/.claude/skills/cfn-transparency-middleware/wrap-agent.sh +131 -131
- package/claude-assets/hooks/cfn-BACKUP_USAGE.md +243 -243
- package/claude-assets/hooks/cfn-invoke-security-validation.sh +69 -69
- package/claude-assets/hooks/cfn-post-edit-cfn-retrospective.sh +78 -78
- package/claude-assets/hooks/cfn-post-edit.config.json +44 -44
- package/claude-assets/hooks/cfn-post-execution/memory-cleanup.sh +19 -19
- package/claude-assets/hooks/cfn-pre-execution/memory-check.sh +19 -19
- package/claude-assets/skills/agent-lifecycle/execute-lifecycle-hook.sh +572 -572
- package/claude-assets/skills/agent-lifecycle/simple-audit.sh +30 -30
- package/claude-assets/skills/cfn-automatic-memory-persistence/persist-agent-output.sh +48 -48
- package/claude-assets/skills/cfn-automatic-memory-persistence/query-agent-history.sh +34 -34
- package/claude-assets/skills/cfn-deliverable-validation/confidence-calculator.sh +261 -261
- package/claude-assets/skills/cfn-expert-update/update-expert.sh +345 -345
- package/claude-assets/skills/cfn-hybrid-routing/check-dependencies.sh +51 -51
- package/claude-assets/skills/cfn-intervention-detector/detect-intervention.sh +110 -110
- package/claude-assets/skills/cfn-intervention-orchestrator/execute-intervention.sh +58 -58
- package/claude-assets/skills/cfn-loop-validation/orchestrate-cfn-loop.sh +252 -252
- package/claude-assets/skills/cfn-loop2-output-processing/process-validator-output.sh +275 -275
- package/claude-assets/skills/cfn-memory-management/check-memory.sh +159 -159
- package/claude-assets/skills/cfn-memory-management/cleanup-memory.sh +196 -196
- package/claude-assets/skills/cfn-node-heap-sizer/task-mode-heap-limiter.sh +325 -325
- package/claude-assets/skills/cfn-playbook-auto-update/auto-update-playbook.sh +85 -85
- package/claude-assets/skills/cfn-redis-coordination/agent-recovery.sh +74 -74
- package/claude-assets/skills/cfn-redis-coordination/get-context.sh +112 -112
- package/claude-assets/skills/cfn-scope-simplifier/simplify-scope.sh +67 -67
- package/claude-assets/skills/cfn-specialist-injection/recommend-specialist.sh +56 -56
- package/claude-assets/skills/cfn-standardized-error-handling/capture-agent-error.sh +86 -86
- package/claude-assets/skills/cfn-standardized-error-handling/test-error-handling.sh +165 -165
- package/claude-assets/skills/cfn-task-config-init/initialize-config.sh +264 -264
- package/claude-assets/skills/cfn-task-decomposition/task-decomposer.sh +278 -278
- package/claude-assets/skills/cfn-transparency-middleware/middleware-config.sh +28 -28
- package/claude-assets/skills/cfn-transparency-middleware/performance-benchmark.sh +78 -78
- package/claude-assets/skills/cfn-transparency-middleware/test-integration.sh +161 -161
- package/claude-assets/skills/cfn-transparency-middleware/test-transparency-skill.sh +367 -367
- package/claude-assets/skills/cfn-transparency-middleware/tests/input-validation.sh +92 -92
- package/claude-assets/skills/cfn-transparency-middleware/wrap-agent.sh +131 -131
- package/claude-assets/skills/docker-build/SKILL.md +96 -203
- package/claude-assets/skills/docker-build/build.sh +73 -73
- package/claude-assets/skills/integration/agent-handoff.sh +494 -0
- package/claude-assets/skills/integration/file-operations.sh +414 -0
- package/claude-assets/skills/workflow-codification/APPROVAL_WORKFLOW.md +806 -0
- package/claude-assets/skills/workflow-codification/COST_TRACKING.md +637 -0
- package/claude-assets/skills/workflow-codification/EDGE_CASE_TRACKING.md +404 -0
- package/claude-assets/skills/workflow-codification/README_PHASE4.md +457 -0
- package/claude-assets/skills/workflow-codification/SKILL.md +110 -0
- package/claude-assets/skills/workflow-codification/analyze-patterns.sh +899 -0
- package/claude-assets/skills/workflow-codification/approval-workflow.sh +514 -0
- package/claude-assets/skills/workflow-codification/generate-skill-update.sh +525 -0
- package/claude-assets/skills/workflow-codification/review-skill.sh +643 -0
- package/claude-assets/skills/workflow-codification/templates/email-notification.txt +114 -0
- package/claude-assets/skills/workflow-codification/templates/slack-notification.md +85 -0
- package/claude-assets/skills/workflow-codification/test-integration.sh +281 -0
- package/claude-assets/skills/workflow-codification/track-cost-savings.sh +445 -0
- package/claude-assets/skills/workflow-codification/track-edge-case.sh +323 -0
- package/dist/cli/config-manager.js +91 -109
- package/dist/cli/config-manager.js.map +1 -1
- package/dist/integration/DatabaseHandoff.js +507 -0
- package/dist/integration/DatabaseHandoff.js.map +1 -0
- package/dist/integration/StandardAdapter.js +291 -0
- package/dist/integration/StandardAdapter.js.map +1 -0
- package/dist/lib/agent-output-parser.js +518 -0
- package/dist/lib/agent-output-parser.js.map +1 -0
- package/dist/lib/agent-output-validator.js +950 -0
- package/dist/lib/agent-output-validator.js.map +1 -0
- package/dist/lib/artifact-registry.js +443 -0
- package/dist/lib/artifact-registry.js.map +1 -0
- package/dist/lib/config-validator.js +687 -0
- package/dist/lib/config-validator.js.map +1 -0
- package/dist/types/agent-output.js +44 -0
- package/dist/types/agent-output.js.map +1 -0
- package/dist/types/config.js +28 -0
- package/dist/types/config.js.map +1 -0
- package/package.json +2 -1
- package/scripts/artifact-cleanup.sh +392 -0
- package/scripts/deploy-production.sh +355 -355
- package/scripts/docker-playwright-fix.sh +311 -311
- package/scripts/docker-rebuild-all-agents.sh +127 -127
- package/scripts/memory-leak-prevention.sh +305 -305
- package/scripts/migrate-artifacts.sh +563 -0
- package/scripts/migrate-yaml-to-json.sh +465 -0
- package/scripts/run-marketing-tests.sh +42 -42
- 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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"}
|