ccjk 10.0.0 → 10.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -4,7 +4,7 @@ import { nanoid } from 'nanoid';
4
4
  import { EventEmitter } from 'node:events';
5
5
  import { resolve } from 'pathe';
6
6
  import { e as executionTracer } from '../shared/ccjk.BN90X6oc.mjs';
7
- import { t as taskPersistence } from '../shared/ccjk.OYWY25_l.mjs';
7
+ import { t as taskPersistence } from '../shared/ccjk.DtMBiwVG.mjs';
8
8
  import { contextLoader } from './context-loader.mjs';
9
9
  import { g as getGlobalConvoyManager } from './convoy-manager.mjs';
10
10
  import 'better-sqlite3';
@@ -1,4 +1,4 @@
1
- const version = "10.0.0";
1
+ const version = "10.1.0";
2
2
  const homepage = "https://github.com/miounet11/ccjk";
3
3
 
4
4
  export { homepage, version };
@@ -1,5 +1,5 @@
1
1
  import ansis from 'ansis';
2
- import { t as taskPersistence } from '../shared/ccjk.OYWY25_l.mjs';
2
+ import { t as taskPersistence } from '../shared/ccjk.DtMBiwVG.mjs';
3
3
  import 'better-sqlite3';
4
4
  import 'node:fs';
5
5
  import 'pathe';
@@ -0,0 +1,599 @@
1
+ import Database from 'better-sqlite3';
2
+ import { existsSync, mkdirSync } from 'node:fs';
3
+ import { join, dirname } from 'pathe';
4
+
5
+ class TaskPersistence {
6
+ db;
7
+ dbPath;
8
+ constructor(dbPath) {
9
+ this.dbPath = dbPath || join(
10
+ process.env.HOME || process.env.USERPROFILE || ".",
11
+ ".ccjk",
12
+ "brain.db"
13
+ );
14
+ const dir = dirname(this.dbPath);
15
+ if (!existsSync(dir)) {
16
+ mkdirSync(dir, { recursive: true });
17
+ }
18
+ this.db = new Database(this.dbPath);
19
+ this.initSchema();
20
+ }
21
+ /**
22
+ * Initialize database schema
23
+ */
24
+ initSchema() {
25
+ this.db.exec(`
26
+ CREATE TABLE IF NOT EXISTS tasks (
27
+ id TEXT PRIMARY KEY,
28
+ session_id TEXT NOT NULL,
29
+ name TEXT NOT NULL,
30
+ description TEXT,
31
+ status TEXT NOT NULL,
32
+ priority INTEGER NOT NULL DEFAULT 0,
33
+ dependencies TEXT NOT NULL DEFAULT '[]',
34
+ input TEXT NOT NULL,
35
+ output TEXT,
36
+ error TEXT,
37
+ created_at INTEGER NOT NULL,
38
+ started_at INTEGER,
39
+ completed_at INTEGER,
40
+ metadata TEXT NOT NULL DEFAULT '{}'
41
+ );
42
+
43
+ CREATE INDEX IF NOT EXISTS idx_tasks_session ON tasks(session_id);
44
+ CREATE INDEX IF NOT EXISTS idx_tasks_status ON tasks(status);
45
+ CREATE INDEX IF NOT EXISTS idx_tasks_created ON tasks(created_at);
46
+
47
+ CREATE TABLE IF NOT EXISTS sessions (
48
+ id TEXT PRIMARY KEY,
49
+ created_at INTEGER NOT NULL,
50
+ updated_at INTEGER NOT NULL,
51
+ metadata TEXT NOT NULL DEFAULT '{}'
52
+ );
53
+
54
+ CREATE INDEX IF NOT EXISTS idx_sessions_created ON sessions(created_at);
55
+
56
+ CREATE TABLE IF NOT EXISTS task_dependencies (
57
+ task_id TEXT NOT NULL,
58
+ depends_on_id TEXT NOT NULL,
59
+ dependency_type TEXT NOT NULL DEFAULT 'sequential',
60
+ required INTEGER NOT NULL DEFAULT 1,
61
+ created_at INTEGER NOT NULL,
62
+ PRIMARY KEY (task_id, depends_on_id),
63
+ FOREIGN KEY (task_id) REFERENCES tasks(id) ON DELETE CASCADE,
64
+ FOREIGN KEY (depends_on_id) REFERENCES tasks(id) ON DELETE CASCADE
65
+ );
66
+
67
+ CREATE INDEX IF NOT EXISTS idx_task_deps_task ON task_dependencies(task_id);
68
+ CREATE INDEX IF NOT EXISTS idx_task_deps_depends ON task_dependencies(depends_on_id);
69
+
70
+ CREATE TABLE IF NOT EXISTS task_metrics (
71
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
72
+ task_id TEXT NOT NULL,
73
+ session_id TEXT NOT NULL,
74
+ execution_time INTEGER NOT NULL,
75
+ retry_count INTEGER NOT NULL DEFAULT 0,
76
+ success INTEGER NOT NULL,
77
+ error_type TEXT,
78
+ timestamp INTEGER NOT NULL,
79
+ FOREIGN KEY (task_id) REFERENCES tasks(id) ON DELETE CASCADE
80
+ );
81
+
82
+ CREATE INDEX IF NOT EXISTS idx_metrics_task ON task_metrics(task_id);
83
+ CREATE INDEX IF NOT EXISTS idx_metrics_session ON task_metrics(session_id);
84
+ CREATE INDEX IF NOT EXISTS idx_metrics_timestamp ON task_metrics(timestamp);
85
+
86
+ CREATE TABLE IF NOT EXISTS decision_log (
87
+ id TEXT PRIMARY KEY,
88
+ session_id TEXT NOT NULL,
89
+ task_id TEXT,
90
+ decision TEXT NOT NULL,
91
+ reasoning TEXT NOT NULL,
92
+ context TEXT NOT NULL DEFAULT '{}',
93
+ outcome TEXT,
94
+ timestamp INTEGER NOT NULL,
95
+ FOREIGN KEY (task_id) REFERENCES tasks(id) ON DELETE SET NULL
96
+ );
97
+
98
+ CREATE INDEX IF NOT EXISTS idx_decision_session ON decision_log(session_id);
99
+ CREATE INDEX IF NOT EXISTS idx_decision_task ON decision_log(task_id);
100
+ CREATE INDEX IF NOT EXISTS idx_decision_timestamp ON decision_log(timestamp);
101
+ `);
102
+ }
103
+ /**
104
+ * Save a task
105
+ */
106
+ saveTask(task, sessionId) {
107
+ const stmt = this.db.prepare(`
108
+ INSERT OR REPLACE INTO tasks (
109
+ id, session_id, name, description, status, priority,
110
+ dependencies, input, output, error,
111
+ created_at, started_at, completed_at, metadata
112
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
113
+ `);
114
+ stmt.run(
115
+ task.id,
116
+ sessionId,
117
+ task.name,
118
+ task.description || null,
119
+ task.status || "pending",
120
+ task.priority || 0,
121
+ JSON.stringify(task.dependencies || []),
122
+ JSON.stringify(task.input || {}),
123
+ task.output ? JSON.stringify(task.output) : null,
124
+ task.error ? JSON.stringify(task.error) : null,
125
+ Date.now(),
126
+ null,
127
+ null,
128
+ JSON.stringify(task.metadata || {})
129
+ );
130
+ }
131
+ /**
132
+ * Update task status
133
+ */
134
+ updateTaskStatus(taskId, status, output, error) {
135
+ const now = Date.now();
136
+ const updates = ["status = ?"];
137
+ const params = [status];
138
+ if (status === "running") {
139
+ updates.push("started_at = ?");
140
+ params.push(now);
141
+ }
142
+ if (status === "completed" || status === "failed") {
143
+ updates.push("completed_at = ?");
144
+ params.push(now);
145
+ }
146
+ if (output) {
147
+ updates.push("output = ?");
148
+ params.push(JSON.stringify(output));
149
+ }
150
+ if (error) {
151
+ updates.push("error = ?");
152
+ params.push(JSON.stringify({ message: error.message, stack: error.stack }));
153
+ }
154
+ params.push(taskId);
155
+ const stmt = this.db.prepare(`
156
+ UPDATE tasks SET ${updates.join(", ")} WHERE id = ?
157
+ `);
158
+ stmt.run(...params);
159
+ }
160
+ /**
161
+ * Get task by ID
162
+ */
163
+ getTask(taskId) {
164
+ const stmt = this.db.prepare(`
165
+ SELECT * FROM tasks WHERE id = ?
166
+ `);
167
+ const row = stmt.get(taskId);
168
+ if (!row) return void 0;
169
+ return this.rowToTask(row);
170
+ }
171
+ /**
172
+ * Get all tasks for a session
173
+ */
174
+ getSessionTasks(sessionId) {
175
+ const stmt = this.db.prepare(`
176
+ SELECT * FROM tasks WHERE session_id = ? ORDER BY created_at ASC
177
+ `);
178
+ const rows = stmt.all(sessionId);
179
+ return rows.map((row) => this.rowToTask(row));
180
+ }
181
+ /**
182
+ * Get task dependencies
183
+ */
184
+ getDependencies(taskId) {
185
+ const task = this.getTask(taskId);
186
+ if (!task || task.dependencies.length === 0) {
187
+ return [];
188
+ }
189
+ const placeholders = task.dependencies.map(() => "?").join(",");
190
+ const stmt = this.db.prepare(`
191
+ SELECT * FROM tasks WHERE id IN (${placeholders})
192
+ `);
193
+ const rows = stmt.all(...task.dependencies);
194
+ return rows.map((row) => this.rowToTask(row));
195
+ }
196
+ /**
197
+ * Restore context for a session
198
+ */
199
+ restoreContext(sessionId) {
200
+ const tasks = this.getSessionTasks(sessionId);
201
+ if (tasks.length === 0) {
202
+ return null;
203
+ }
204
+ const sessionStmt = this.db.prepare(`
205
+ SELECT * FROM sessions WHERE id = ?
206
+ `);
207
+ const sessionRow = sessionStmt.get(sessionId);
208
+ const metadata = sessionRow ? JSON.parse(sessionRow.metadata) : {};
209
+ return {
210
+ sessionId,
211
+ tasks,
212
+ metadata
213
+ };
214
+ }
215
+ /**
216
+ * Create or update session
217
+ */
218
+ saveSession(sessionId, metadata = {}) {
219
+ const stmt = this.db.prepare(`
220
+ INSERT OR REPLACE INTO sessions (id, created_at, updated_at, metadata)
221
+ VALUES (?, ?, ?, ?)
222
+ `);
223
+ const now = Date.now();
224
+ stmt.run(sessionId, now, now, JSON.stringify(metadata));
225
+ }
226
+ /**
227
+ * List recent sessions
228
+ */
229
+ listSessions(limit = 10) {
230
+ const stmt = this.db.prepare(`
231
+ SELECT * FROM sessions ORDER BY created_at DESC LIMIT ?
232
+ `);
233
+ const rows = stmt.all(limit);
234
+ return rows.map((row) => ({
235
+ id: row.id,
236
+ createdAt: row.created_at,
237
+ metadata: JSON.parse(row.metadata)
238
+ }));
239
+ }
240
+ /**
241
+ * Clean up old sessions
242
+ */
243
+ cleanup(keepDays = 7) {
244
+ const cutoff = Date.now() - keepDays * 24 * 60 * 60 * 1e3;
245
+ const deleteTasksStmt = this.db.prepare(`
246
+ DELETE FROM tasks WHERE created_at < ?
247
+ `);
248
+ const tasksResult = deleteTasksStmt.run(cutoff);
249
+ const deleteSessionsStmt = this.db.prepare(`
250
+ DELETE FROM sessions WHERE created_at < ?
251
+ `);
252
+ const sessionsResult = deleteSessionsStmt.run(cutoff);
253
+ return tasksResult.changes + sessionsResult.changes;
254
+ }
255
+ /**
256
+ * Add task dependency
257
+ */
258
+ addDependency(taskId, dependsOnId, type = "sequential", required = true) {
259
+ const stmt = this.db.prepare(`
260
+ INSERT OR REPLACE INTO task_dependencies (task_id, depends_on_id, dependency_type, required, created_at)
261
+ VALUES (?, ?, ?, ?, ?)
262
+ `);
263
+ stmt.run(taskId, dependsOnId, type, required ? 1 : 0, Date.now());
264
+ }
265
+ /**
266
+ * Remove task dependency
267
+ */
268
+ removeDependency(taskId, dependsOnId) {
269
+ const stmt = this.db.prepare(`
270
+ DELETE FROM task_dependencies WHERE task_id = ? AND depends_on_id = ?
271
+ `);
272
+ stmt.run(taskId, dependsOnId);
273
+ }
274
+ /**
275
+ * Get task dependencies
276
+ */
277
+ getTaskDependencies(taskId) {
278
+ const stmt = this.db.prepare(`
279
+ SELECT * FROM task_dependencies WHERE task_id = ?
280
+ `);
281
+ const rows = stmt.all(taskId);
282
+ return rows.map((row) => ({
283
+ taskId: row.task_id,
284
+ dependsOnId: row.depends_on_id,
285
+ dependencyType: row.dependency_type,
286
+ required: row.required === 1,
287
+ createdAt: row.created_at
288
+ }));
289
+ }
290
+ /**
291
+ * Get tasks that depend on this task
292
+ */
293
+ getDependentTasks(taskId) {
294
+ const stmt = this.db.prepare(`
295
+ SELECT t.* FROM tasks t
296
+ INNER JOIN task_dependencies td ON t.id = td.task_id
297
+ WHERE td.depends_on_id = ?
298
+ `);
299
+ const rows = stmt.all(taskId);
300
+ return rows.map((row) => this.rowToTask(row));
301
+ }
302
+ /**
303
+ * Build dependency graph for session
304
+ */
305
+ buildDependencyGraph(sessionId) {
306
+ const tasks = this.getSessionTasks(sessionId);
307
+ const nodes = [];
308
+ for (const task of tasks) {
309
+ const dependencies = this.getTaskDependencies(task.id);
310
+ const dependents = this.getDependentTasks(task.id);
311
+ nodes.push({
312
+ id: task.id,
313
+ name: task.name,
314
+ status: task.status,
315
+ level: this.calculateTopologicalLevel(task.id, sessionId),
316
+ dependencies: dependencies.map((d) => d.dependsOnId),
317
+ dependents: dependents.map((d) => d.id)
318
+ });
319
+ }
320
+ return nodes.sort((a, b) => a.level - b.level);
321
+ }
322
+ /**
323
+ * Calculate topological level for a task
324
+ */
325
+ calculateTopologicalLevel(taskId, sessionId) {
326
+ const visited = /* @__PURE__ */ new Set();
327
+ const calculating = /* @__PURE__ */ new Set();
328
+ const calculate = (id) => {
329
+ if (visited.has(id)) {
330
+ return 0;
331
+ }
332
+ if (calculating.has(id)) {
333
+ return 0;
334
+ }
335
+ calculating.add(id);
336
+ const deps = this.getTaskDependencies(id);
337
+ if (deps.length === 0) {
338
+ visited.add(id);
339
+ calculating.delete(id);
340
+ return 0;
341
+ }
342
+ let maxLevel = -1;
343
+ for (const dep of deps) {
344
+ const depLevel = calculate(dep.dependsOnId);
345
+ maxLevel = Math.max(maxLevel, depLevel);
346
+ }
347
+ visited.add(id);
348
+ calculating.delete(id);
349
+ return maxLevel + 1;
350
+ };
351
+ return calculate(taskId);
352
+ }
353
+ /**
354
+ * Get tasks in topological order
355
+ */
356
+ getTopologicalOrder(sessionId) {
357
+ const graph = this.buildDependencyGraph(sessionId);
358
+ const tasks = this.getSessionTasks(sessionId);
359
+ const taskMap = new Map(tasks.map((t) => [t.id, t]));
360
+ return graph.sort((a, b) => a.level - b.level).map((node) => taskMap.get(node.id)).filter(Boolean);
361
+ }
362
+ /**
363
+ * Get next executable tasks (no pending dependencies)
364
+ */
365
+ getNextExecutableTasks(sessionId) {
366
+ const tasks = this.getSessionTasks(sessionId);
367
+ const executable = [];
368
+ for (const task of tasks) {
369
+ if (task.status !== "pending") {
370
+ continue;
371
+ }
372
+ const deps = this.getTaskDependencies(task.id);
373
+ const allCompleted = deps.every((dep) => {
374
+ const depTask = this.getTask(dep.dependsOnId);
375
+ return depTask && depTask.status === "completed";
376
+ });
377
+ if (allCompleted) {
378
+ executable.push(task);
379
+ }
380
+ }
381
+ return executable;
382
+ }
383
+ /**
384
+ * Recover execution state for interrupted session
385
+ */
386
+ recoverExecutionState(sessionId) {
387
+ const tasks = this.getSessionTasks(sessionId);
388
+ const graph = this.buildDependencyGraph(sessionId);
389
+ return {
390
+ sessionId,
391
+ pendingTasks: tasks.filter((t) => t.status === "pending"),
392
+ runningTasks: tasks.filter((t) => t.status === "running"),
393
+ completedTasks: tasks.filter((t) => t.status === "completed"),
394
+ failedTasks: tasks.filter((t) => t.status === "failed"),
395
+ dependencyGraph: graph,
396
+ nextExecutable: this.getNextExecutableTasks(sessionId)
397
+ };
398
+ }
399
+ /**
400
+ * Record task metrics
401
+ */
402
+ recordMetrics(metrics) {
403
+ const stmt = this.db.prepare(`
404
+ INSERT INTO task_metrics (task_id, session_id, execution_time, retry_count, success, error_type, timestamp)
405
+ VALUES (?, ?, ?, ?, ?, ?, ?)
406
+ `);
407
+ stmt.run(
408
+ metrics.taskId,
409
+ metrics.sessionId,
410
+ metrics.executionTime,
411
+ metrics.retryCount,
412
+ metrics.success ? 1 : 0,
413
+ metrics.errorType || null,
414
+ Date.now()
415
+ );
416
+ }
417
+ /**
418
+ * Get task metrics
419
+ */
420
+ getTaskMetrics(taskId) {
421
+ const stmt = this.db.prepare(`
422
+ SELECT * FROM task_metrics WHERE task_id = ? ORDER BY timestamp DESC
423
+ `);
424
+ const rows = stmt.all(taskId);
425
+ return rows.map((row) => ({
426
+ taskId: row.task_id,
427
+ sessionId: row.session_id,
428
+ executionTime: row.execution_time,
429
+ retryCount: row.retry_count,
430
+ success: row.success === 1,
431
+ errorType: row.error_type,
432
+ timestamp: row.timestamp
433
+ }));
434
+ }
435
+ /**
436
+ * Get aggregated metrics for session
437
+ */
438
+ getSessionMetrics(sessionId) {
439
+ const stmt = this.db.prepare(`
440
+ SELECT
441
+ COUNT(*) as total,
442
+ SUM(CASE WHEN success = 1 THEN 1 ELSE 0 END) as completed,
443
+ SUM(CASE WHEN success = 0 THEN 1 ELSE 0 END) as failed,
444
+ AVG(execution_time) as avg_time,
445
+ SUM(retry_count) as total_retries
446
+ FROM task_metrics
447
+ WHERE session_id = ?
448
+ `);
449
+ const row = stmt.get(sessionId);
450
+ if (!row || row.total === 0) {
451
+ return {
452
+ totalTasks: 0,
453
+ completedTasks: 0,
454
+ failedTasks: 0,
455
+ avgExecutionTime: 0,
456
+ successRate: 0,
457
+ totalRetries: 0
458
+ };
459
+ }
460
+ return {
461
+ totalTasks: row.total,
462
+ completedTasks: row.completed,
463
+ failedTasks: row.failed,
464
+ avgExecutionTime: Math.round(row.avg_time || 0),
465
+ successRate: row.total > 0 ? row.completed / row.total : 0,
466
+ totalRetries: row.total_retries || 0
467
+ };
468
+ }
469
+ /**
470
+ * Log a decision
471
+ */
472
+ logDecision(log) {
473
+ const stmt = this.db.prepare(`
474
+ INSERT INTO decision_log (id, session_id, task_id, decision, reasoning, context, outcome, timestamp)
475
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?)
476
+ `);
477
+ stmt.run(
478
+ log.id,
479
+ log.sessionId,
480
+ log.taskId || null,
481
+ log.decision,
482
+ log.reasoning,
483
+ log.context,
484
+ log.outcome || null,
485
+ Date.now()
486
+ );
487
+ }
488
+ /**
489
+ * Update decision outcome
490
+ */
491
+ updateDecisionOutcome(decisionId, outcome) {
492
+ const stmt = this.db.prepare(`
493
+ UPDATE decision_log SET outcome = ? WHERE id = ?
494
+ `);
495
+ stmt.run(outcome, decisionId);
496
+ }
497
+ /**
498
+ * Get decision log for session
499
+ */
500
+ getDecisionLog(sessionId) {
501
+ const stmt = this.db.prepare(`
502
+ SELECT * FROM decision_log WHERE session_id = ? ORDER BY timestamp ASC
503
+ `);
504
+ const rows = stmt.all(sessionId);
505
+ return rows.map((row) => ({
506
+ id: row.id,
507
+ sessionId: row.session_id,
508
+ taskId: row.task_id,
509
+ decision: row.decision,
510
+ reasoning: row.reasoning,
511
+ context: row.context,
512
+ outcome: row.outcome,
513
+ timestamp: row.timestamp
514
+ }));
515
+ }
516
+ /**
517
+ * Get decision log for task
518
+ */
519
+ getTaskDecisionLog(taskId) {
520
+ const stmt = this.db.prepare(`
521
+ SELECT * FROM decision_log WHERE task_id = ? ORDER BY timestamp ASC
522
+ `);
523
+ const rows = stmt.all(taskId);
524
+ return rows.map((row) => ({
525
+ id: row.id,
526
+ sessionId: row.session_id,
527
+ taskId: row.task_id,
528
+ decision: row.decision,
529
+ reasoning: row.reasoning,
530
+ context: row.context,
531
+ outcome: row.outcome,
532
+ timestamp: row.timestamp
533
+ }));
534
+ }
535
+ /**
536
+ * Detect circular dependencies
537
+ */
538
+ detectCircularDependency(taskId, dependsOnId) {
539
+ const visited = /* @__PURE__ */ new Set();
540
+ const stack = /* @__PURE__ */ new Set();
541
+ const hasCycle = (currentId) => {
542
+ if (stack.has(currentId)) {
543
+ return true;
544
+ }
545
+ if (visited.has(currentId)) {
546
+ return false;
547
+ }
548
+ visited.add(currentId);
549
+ stack.add(currentId);
550
+ const deps = this.getTaskDependencies(currentId);
551
+ for (const dep of deps) {
552
+ if (hasCycle(dep.dependsOnId)) {
553
+ return true;
554
+ }
555
+ }
556
+ stack.delete(currentId);
557
+ return false;
558
+ };
559
+ const tempDeps = this.getTaskDependencies(taskId);
560
+ tempDeps.push({
561
+ taskId,
562
+ dependsOnId,
563
+ dependencyType: "sequential",
564
+ required: true,
565
+ createdAt: Date.now()
566
+ });
567
+ return hasCycle(taskId);
568
+ }
569
+ /**
570
+ * Close database
571
+ */
572
+ close() {
573
+ this.db.close();
574
+ }
575
+ /**
576
+ * Convert database row to PersistedTask
577
+ */
578
+ rowToTask(row) {
579
+ return {
580
+ id: row.id,
581
+ sessionId: row.session_id,
582
+ name: row.name,
583
+ description: row.description,
584
+ status: row.status,
585
+ priority: row.priority,
586
+ dependencies: JSON.parse(row.dependencies),
587
+ input: row.input,
588
+ output: row.output,
589
+ error: row.error,
590
+ createdAt: row.created_at,
591
+ startedAt: row.started_at,
592
+ completedAt: row.completed_at,
593
+ metadata: row.metadata
594
+ };
595
+ }
596
+ }
597
+ const taskPersistence = new TaskPersistence();
598
+
599
+ export { taskPersistence as t };
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "ccjk",
3
3
  "type": "module",
4
- "version": "10.0.0",
4
+ "version": "10.1.0",
5
5
  "packageManager": "pnpm@10.17.1",
6
6
  "description": "CLI toolkit for Claude Code and Codex setup. Simplifies MCP service installation, API configuration, workflow management, and multi-provider support with guided interactive setup.",
7
7
  "author": {
@@ -1,239 +0,0 @@
1
- import Database from 'better-sqlite3';
2
- import { existsSync, mkdirSync } from 'node:fs';
3
- import { join, dirname } from 'pathe';
4
-
5
- class TaskPersistence {
6
- db;
7
- dbPath;
8
- constructor(dbPath) {
9
- this.dbPath = dbPath || join(
10
- process.env.HOME || process.env.USERPROFILE || ".",
11
- ".ccjk",
12
- "brain.db"
13
- );
14
- const dir = dirname(this.dbPath);
15
- if (!existsSync(dir)) {
16
- mkdirSync(dir, { recursive: true });
17
- }
18
- this.db = new Database(this.dbPath);
19
- this.initSchema();
20
- }
21
- /**
22
- * Initialize database schema
23
- */
24
- initSchema() {
25
- this.db.exec(`
26
- CREATE TABLE IF NOT EXISTS tasks (
27
- id TEXT PRIMARY KEY,
28
- session_id TEXT NOT NULL,
29
- name TEXT NOT NULL,
30
- description TEXT,
31
- status TEXT NOT NULL,
32
- priority INTEGER NOT NULL DEFAULT 0,
33
- dependencies TEXT NOT NULL DEFAULT '[]',
34
- input TEXT NOT NULL,
35
- output TEXT,
36
- error TEXT,
37
- created_at INTEGER NOT NULL,
38
- started_at INTEGER,
39
- completed_at INTEGER,
40
- metadata TEXT NOT NULL DEFAULT '{}'
41
- );
42
-
43
- CREATE INDEX IF NOT EXISTS idx_tasks_session ON tasks(session_id);
44
- CREATE INDEX IF NOT EXISTS idx_tasks_status ON tasks(status);
45
- CREATE INDEX IF NOT EXISTS idx_tasks_created ON tasks(created_at);
46
-
47
- CREATE TABLE IF NOT EXISTS sessions (
48
- id TEXT PRIMARY KEY,
49
- created_at INTEGER NOT NULL,
50
- updated_at INTEGER NOT NULL,
51
- metadata TEXT NOT NULL DEFAULT '{}'
52
- );
53
-
54
- CREATE INDEX IF NOT EXISTS idx_sessions_created ON sessions(created_at);
55
- `);
56
- }
57
- /**
58
- * Save a task
59
- */
60
- saveTask(task, sessionId) {
61
- const stmt = this.db.prepare(`
62
- INSERT OR REPLACE INTO tasks (
63
- id, session_id, name, description, status, priority,
64
- dependencies, input, output, error,
65
- created_at, started_at, completed_at, metadata
66
- ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
67
- `);
68
- stmt.run(
69
- task.id,
70
- sessionId,
71
- task.name,
72
- task.description || null,
73
- task.status || "pending",
74
- task.priority || 0,
75
- JSON.stringify(task.dependencies || []),
76
- JSON.stringify(task.input || {}),
77
- task.output ? JSON.stringify(task.output) : null,
78
- task.error ? JSON.stringify(task.error) : null,
79
- Date.now(),
80
- null,
81
- null,
82
- JSON.stringify(task.metadata || {})
83
- );
84
- }
85
- /**
86
- * Update task status
87
- */
88
- updateTaskStatus(taskId, status, output, error) {
89
- const now = Date.now();
90
- const updates = ["status = ?"];
91
- const params = [status];
92
- if (status === "running") {
93
- updates.push("started_at = ?");
94
- params.push(now);
95
- }
96
- if (status === "completed" || status === "failed") {
97
- updates.push("completed_at = ?");
98
- params.push(now);
99
- }
100
- if (output) {
101
- updates.push("output = ?");
102
- params.push(JSON.stringify(output));
103
- }
104
- if (error) {
105
- updates.push("error = ?");
106
- params.push(JSON.stringify({ message: error.message, stack: error.stack }));
107
- }
108
- params.push(taskId);
109
- const stmt = this.db.prepare(`
110
- UPDATE tasks SET ${updates.join(", ")} WHERE id = ?
111
- `);
112
- stmt.run(...params);
113
- }
114
- /**
115
- * Get task by ID
116
- */
117
- getTask(taskId) {
118
- const stmt = this.db.prepare(`
119
- SELECT * FROM tasks WHERE id = ?
120
- `);
121
- const row = stmt.get(taskId);
122
- if (!row) return void 0;
123
- return this.rowToTask(row);
124
- }
125
- /**
126
- * Get all tasks for a session
127
- */
128
- getSessionTasks(sessionId) {
129
- const stmt = this.db.prepare(`
130
- SELECT * FROM tasks WHERE session_id = ? ORDER BY created_at ASC
131
- `);
132
- const rows = stmt.all(sessionId);
133
- return rows.map((row) => this.rowToTask(row));
134
- }
135
- /**
136
- * Get task dependencies
137
- */
138
- getDependencies(taskId) {
139
- const task = this.getTask(taskId);
140
- if (!task || task.dependencies.length === 0) {
141
- return [];
142
- }
143
- const placeholders = task.dependencies.map(() => "?").join(",");
144
- const stmt = this.db.prepare(`
145
- SELECT * FROM tasks WHERE id IN (${placeholders})
146
- `);
147
- const rows = stmt.all(...task.dependencies);
148
- return rows.map((row) => this.rowToTask(row));
149
- }
150
- /**
151
- * Restore context for a session
152
- */
153
- restoreContext(sessionId) {
154
- const tasks = this.getSessionTasks(sessionId);
155
- if (tasks.length === 0) {
156
- return null;
157
- }
158
- const sessionStmt = this.db.prepare(`
159
- SELECT * FROM sessions WHERE id = ?
160
- `);
161
- const sessionRow = sessionStmt.get(sessionId);
162
- const metadata = sessionRow ? JSON.parse(sessionRow.metadata) : {};
163
- return {
164
- sessionId,
165
- tasks,
166
- metadata
167
- };
168
- }
169
- /**
170
- * Create or update session
171
- */
172
- saveSession(sessionId, metadata = {}) {
173
- const stmt = this.db.prepare(`
174
- INSERT OR REPLACE INTO sessions (id, created_at, updated_at, metadata)
175
- VALUES (?, ?, ?, ?)
176
- `);
177
- const now = Date.now();
178
- stmt.run(sessionId, now, now, JSON.stringify(metadata));
179
- }
180
- /**
181
- * List recent sessions
182
- */
183
- listSessions(limit = 10) {
184
- const stmt = this.db.prepare(`
185
- SELECT * FROM sessions ORDER BY created_at DESC LIMIT ?
186
- `);
187
- const rows = stmt.all(limit);
188
- return rows.map((row) => ({
189
- id: row.id,
190
- createdAt: row.created_at,
191
- metadata: JSON.parse(row.metadata)
192
- }));
193
- }
194
- /**
195
- * Clean up old sessions
196
- */
197
- cleanup(keepDays = 7) {
198
- const cutoff = Date.now() - keepDays * 24 * 60 * 60 * 1e3;
199
- const deleteTasksStmt = this.db.prepare(`
200
- DELETE FROM tasks WHERE created_at < ?
201
- `);
202
- const tasksResult = deleteTasksStmt.run(cutoff);
203
- const deleteSessionsStmt = this.db.prepare(`
204
- DELETE FROM sessions WHERE created_at < ?
205
- `);
206
- const sessionsResult = deleteSessionsStmt.run(cutoff);
207
- return tasksResult.changes + sessionsResult.changes;
208
- }
209
- /**
210
- * Close database
211
- */
212
- close() {
213
- this.db.close();
214
- }
215
- /**
216
- * Convert database row to PersistedTask
217
- */
218
- rowToTask(row) {
219
- return {
220
- id: row.id,
221
- sessionId: row.session_id,
222
- name: row.name,
223
- description: row.description,
224
- status: row.status,
225
- priority: row.priority,
226
- dependencies: JSON.parse(row.dependencies),
227
- input: row.input,
228
- output: row.output,
229
- error: row.error,
230
- createdAt: row.created_at,
231
- startedAt: row.started_at,
232
- completedAt: row.completed_at,
233
- metadata: row.metadata
234
- };
235
- }
236
- }
237
- const taskPersistence = new TaskPersistence();
238
-
239
- export { taskPersistence as t };