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.
package/dist/chunks/agents.mjs
CHANGED
|
@@ -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.
|
|
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';
|
package/dist/chunks/package.mjs
CHANGED
package/dist/chunks/sessions.mjs
CHANGED
|
@@ -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.
|
|
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 };
|