@crewx/sdk 0.8.2 → 0.8.3-rc.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (33) hide show
  1. package/dist/conversation/index.d.ts +0 -1
  2. package/dist/conversation/sqlite-provider.d.ts +2 -3
  3. package/dist/esm/index.js +48 -56
  4. package/dist/esm/plugins/index.js +109 -51
  5. package/dist/esm/repository/index.js +103 -0
  6. package/dist/index.js +48 -56
  7. package/dist/migrations/0000_init.sql +156 -0
  8. package/dist/migrations/meta/0000_snapshot.json +1091 -0
  9. package/dist/migrations/meta/_journal.json +13 -0
  10. package/dist/plugins/index.js +109 -51
  11. package/dist/plugins/sqlite-tracing.d.ts +0 -1
  12. package/dist/repository/base-sqlite-repository.d.ts +6 -0
  13. package/dist/repository/db.d.ts +9 -0
  14. package/dist/repository/errors.d.ts +6 -0
  15. package/dist/repository/index.d.ts +21 -0
  16. package/dist/repository/index.js +103 -0
  17. package/dist/repository/migrate.d.ts +3 -0
  18. package/dist/repository/request-log.repository.d.ts +31 -0
  19. package/dist/repository/span.repository.d.ts +16 -0
  20. package/dist/repository/task.repository.d.ts +115 -0
  21. package/dist/repository/thread-box.repository.d.ts +20 -0
  22. package/dist/repository/thread.repository.d.ts +56 -0
  23. package/dist/repository/tool-call.repository.d.ts +19 -0
  24. package/dist/repository/workspace.repository.d.ts +64 -0
  25. package/dist/schema/index.d.ts +7 -0
  26. package/dist/schema/request-logs.d.ts +309 -0
  27. package/dist/schema/spans.d.ts +254 -0
  28. package/dist/schema/tasks.d.ts +660 -0
  29. package/dist/schema/thread-boxes.d.ts +210 -0
  30. package/dist/schema/threads.d.ts +214 -0
  31. package/dist/schema/tool-calls.d.ts +178 -0
  32. package/dist/schema/workspaces.d.ts +159 -0
  33. package/package.json +14 -3
@@ -0,0 +1,13 @@
1
+ {
2
+ "version": "7",
3
+ "dialect": "sqlite",
4
+ "entries": [
5
+ {
6
+ "idx": 0,
7
+ "version": "6",
8
+ "when": 1777950449379,
9
+ "tag": "0000_init",
10
+ "breakpoints": true
11
+ }
12
+ ]
13
+ }
@@ -1,53 +1,111 @@
1
- 'use strict';var fs=require('fs'),k=require('path'),os=require('os'),U=require('better-sqlite3'),crypto=require('crypto');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}function _interopNamespace(e){if(e&&e.__esModule)return e;var n=Object.create(null);if(e){Object.keys(e).forEach(function(k){if(k!=='default'){var d=Object.getOwnPropertyDescriptor(e,k);Object.defineProperty(n,k,d.get?d:{enumerable:true,get:function(){return e[k]}});}})}n.default=e;return Object.freeze(n)}var k__namespace=/*#__PURE__*/_interopNamespace(k);var U__default=/*#__PURE__*/_interopDefault(U);var l=class{detach(s){}};function S(i){let s=t=>String(t).padStart(2,"0");return `${i.getFullYear()}${s(i.getMonth()+1)}${s(i.getDate())}T${s(i.getHours())}${s(i.getMinutes())}${s(i.getSeconds())}`}var f=class extends l{name="file-logger";unsubs=[];logFiles=new Map;logsDir;version;constructor(s){super(),this.logsDir=k.join(s?.workspaceRoot??process.cwd(),".crewx","logs"),this.version=s?.version??"unknown";}attach(s){this.unsubs.push(s.on("task:start",t=>{try{fs.existsSync(this.logsDir)||fs.mkdirSync(this.logsDir,{recursive:!0});let r=S(t.timestamp),a=k.join(this.logsDir,`${r}_${t.traceId}.log`);this.logFiles.set(t.traceId,a);let o=`=== TASK LOG: ${t.traceId} ===
1
+ 'use strict';var fs=require('fs'),ut=require('path'),At=require('os'),crypto=require('crypto'),drizzleOrm=require('drizzle-orm'),sqliteCore=require('drizzle-orm/sqlite-core');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}function _interopNamespace(e){if(e&&e.__esModule)return e;var n=Object.create(null);if(e){Object.keys(e).forEach(function(k){if(k!=='default'){var d=Object.getOwnPropertyDescriptor(e,k);Object.defineProperty(n,k,d.get?d:{enumerable:true,get:function(){return e[k]}});}})}n.default=e;return Object.freeze(n)}var ut__namespace=/*#__PURE__*/_interopNamespace(ut);var At__default=/*#__PURE__*/_interopDefault(At);var Q=(d=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(d,{get:(t,e)=>(typeof require<"u"?require:t)[e]}):d)(function(d){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+d+'" is not supported')});var A=class{detach(t){}};function it(d){let t=e=>String(e).padStart(2,"0");return `${d.getFullYear()}${t(d.getMonth()+1)}${t(d.getDate())}T${t(d.getHours())}${t(d.getMinutes())}${t(d.getSeconds())}`}var Z=class extends A{name="file-logger";unsubs=[];logFiles=new Map;logsDir;version;constructor(t){super(),this.logsDir=ut.join(t?.workspaceRoot??process.cwd(),".crewx","logs"),this.version=t?.version??"unknown";}attach(t){this.unsubs.push(t.on("task:start",e=>{try{fs.existsSync(this.logsDir)||fs.mkdirSync(this.logsDir,{recursive:!0});let s=it(e.timestamp),n=ut.join(this.logsDir,`${s}_${e.traceId}.log`);this.logFiles.set(e.traceId,n);let r=`=== TASK LOG: ${e.traceId} ===
2
2
  CrewX Version: ${this.version}
3
- Mode: ${t.mode}
4
- Agent: ${t.agentRef}
5
- Started: ${t.timestamp.toLocaleString()}
6
- Message: ${t.message}
3
+ Mode: ${e.mode}
4
+ Agent: ${e.agentRef}
5
+ Started: ${e.timestamp.toLocaleString()}
6
+ Message: ${e.message}
7
7
 
8
- `;fs.writeFileSync(a,o,{encoding:"utf8",mode:384});}catch{}}),s.on("task:output",t=>{try{let r=this.logFiles.get(t.traceId);if(!r)return;let a=new Date().toISOString();fs.appendFileSync(r,`[${a}] STDOUT: ${t.output}
9
- `,"utf8");}catch{}}),s.on("task:end",t=>{try{let r=this.logFiles.get(t.traceId);if(!r)return;let a=new Date().toLocaleString(),o=t.error?`failed: ${t.error.message}`:"completed successfully",p=`[${a}] INFO: Task ${o} in ${t.durationMs}ms
10
- [${a}] INFO: Process closed with exit code: ${t.error?1:0}
11
- `;fs.appendFileSync(r,p,"utf8"),this.logFiles.delete(t.traceId);}catch{}}));}detach(s){this.unsubs.forEach(t=>t()),this.unsubs=[],this.logFiles.clear();}};function O(i){let s=k__namespace.resolve(i);return process.platform==="win32"&&(s=s.replace(/\\/g,"/"),s=s.replace(/^([A-Z]):/,(t,r)=>`${r.toLowerCase()}:`)),s.length>1&&!/^[a-zA-Z]:\/$/.test(s)&&(s=s.replace(/\/+$/,"")),s}function x(i){let s=O(i);return crypto.createHash("sha256").update(s).digest("hex")}var _=class extends l{name="sqlite-tracing";db=null;unsubs=[];dbPath;version;constructor(s){super(),this.dbPath=k.join(s?.dbRoot??os.homedir(),".crewx","crewx.db"),this.version=s?.version??"unknown";}attach(s){let t=k.dirname(this.dbPath);if(fs.existsSync(t)||fs.mkdirSync(t,{recursive:true}),this.db=new U__default.default(this.dbPath),this.db.prepare("SELECT name FROM sqlite_master WHERE type='table' AND name='tasks'").get()!==void 0){let e=this.db.prepare("PRAGMA table_info(tasks)").all(),n=d=>e.some(u=>u.name===d);n("parent_task_id")||this.db.exec("ALTER TABLE tasks ADD COLUMN parent_task_id TEXT"),n("caller_agent_id")||this.db.exec("ALTER TABLE tasks ADD COLUMN caller_agent_id TEXT"),n("trace_id")||this.db.exec("ALTER TABLE tasks ADD COLUMN trace_id TEXT");}let a=process.cwd(),o=fs.existsSync(k.join(a,"crewx.yaml"))||fs.existsSync(k.join(a,"crewx.yml")),p=o?x(a):null,b=o?k.basename(a):null,g=process.argv.join(" ");this.unsubs.push(s.on("task:start",e=>{try{let n=process.env.CREWX_CALLER_AGENT_ID||null,d=process.env.CREWX_PARENT_TASK_ID||null,u=process.env.CREWX_TRACE_ID||e.traceId,C=e.metadata?JSON.stringify(e.metadata):JSON.stringify({provider:e.provider??"cli/claude"});this.db.prepare(`INSERT OR IGNORE INTO tasks
12
- (id, agent_id, prompt, mode, status, pid, started_at,
13
- crewx_version, platform,
14
- model, rendered_prompt, command, coding_agent_command,
15
- workspace_id, workspace_name,
16
- caller_agent_id, parent_task_id, trace_id, metadata, thread_id)
17
- VALUES (?, ?, ?, ?, 'running', ?, ?,
18
- ?, ?,
19
- ?, ?, ?, ?,
20
- ?, ?,
21
- ?, ?, ?, ?, ?)`).run(e.traceId,e.agentRef.replace(/^@/,""),e.message,e.mode,e.pid??null,e.timestamp.toISOString(),this.version,e.platform??"cli",e.model??null,e.renderedPrompt??null,g,e.codingAgentCommand??null,e.workspaceId??p,e.workspaceName??b,n,d,u,C,e.threadId??null);}catch{}}),s.on("task:output",e=>{try{let n=this.db.prepare("SELECT logs FROM tasks WHERE id=?").get(e.traceId),d=n?.logs?JSON.parse(n.logs):[];d.push({timestamp:e.timestamp.toISOString(),level:e.level??"stdout",message:e.output}),this.db.prepare("UPDATE tasks SET logs=? WHERE id=?").run(JSON.stringify(d),e.traceId);}catch{}}),s.on("task:end",e=>{try{let n=e.error?"failed":"success";this.db.prepare(`UPDATE tasks
22
- SET status=?, result=?, error=?,
23
- completed_at=?, duration_ms=?,
24
- exit_code=?,
25
- input_tokens=?, output_tokens=?, cached_input_tokens=?,
26
- cost_usd=?,
27
- model=COALESCE(?, model)
28
- WHERE id=?`).run(n,e.result??null,e.error?JSON.stringify(e.error):null,e.timestamp.toISOString(),e.durationMs,e.exitCode??null,e.inputTokens??0,e.outputTokens??0,e.cachedInputTokens??0,e.costUsd??0,e.model??null,e.traceId);}catch{}}));}detach(s){this.unsubs.forEach(t=>t()),this.unsubs=[],this.db?.close(),this.db=null;}};var W=`
29
- CREATE TABLE IF NOT EXISTS threads (
30
- id TEXT PRIMARY KEY,
31
- workspace_id TEXT,
32
- platform TEXT NOT NULL DEFAULT 'cli',
33
- title TEXT,
34
- first_message TEXT,
35
- last_message TEXT,
36
- message_count INTEGER NOT NULL DEFAULT 0,
37
- created_at TEXT NOT NULL,
38
- updated_at TEXT NOT NULL,
39
- metadata TEXT
40
- )
41
- `,H=["id","thread_id","prompt","result","started_at","trace_id","status","parent_task_id","agent_id"];function X(i){return i.replace(/<conversation_history[^>]*>[\s\S]*?<\/conversation_history>/g,"").split(`
42
- `).filter(a=>!(a.startsWith("Loaded ")&&a.includes("layouts from")||a.includes("[dotenv@")||a.includes("[Nest]")&&a.includes("DEBUG")||a.startsWith("Registered custom layout:")||a.startsWith("Updated custom layout:"))).join(`
43
- `).trim()}function q(i){if(!i)return "";let s=i;try{let t=JSON.parse(s);Array.isArray(t)?s=t.filter(r=>r?.type==="text"&&r?.text).map(r=>r.text).join(`
44
- `):t&&typeof t=="object"&&t.result!==void 0&&(s=t.result||"");}catch{s=X(s);}return s}var m=class{db;constructor(s){let t=s??k.join(os.homedir(),".crewx","crewx.db");this.db=new U__default.default(t),this.init();}init(){this.db.pragma("journal_mode = WAL"),this.db.exec(W),this.migrateThreadsTable();}migrateThreadsTable(){this.db.prepare("PRAGMA table_info(threads)").all().some(t=>t.name==="title_locked")||this.db.exec("ALTER TABLE threads ADD COLUMN title_locked INTEGER NOT NULL DEFAULT 0");}updateThread(s,t){t.title!==void 0&&this.db.prepare("UPDATE threads SET title = ?, title_locked = 1, updated_at = ? WHERE id = ?").run(t.title,new Date().toISOString(),s);}async ensureThread(s,t,r){let a=this.db.prepare("SELECT platform FROM threads WHERE id = ?").get(s);if(a){if(a.platform!==t)throw new Error(`Thread '${s}' already exists with platform '${a.platform}' \u2014 cannot change to '${t}' (platform is immutable)`);r&&this.db.prepare("UPDATE threads SET workspace_id = COALESCE(workspace_id, ?) WHERE id = ?").run(r,s);return}let o=new Date().toISOString();this.db.prepare("INSERT INTO threads (id, platform, workspace_id, message_count, created_at, updated_at) VALUES (?, ?, ?, 0, ?, ?)").run(s,t,r??null,o,o);}async fetchHistory(s,t){let r=t?.limit??100,a=["thread_id = ?","(parent_task_id IS NULL OR parent_task_id = '')","(status IN ('done', 'completed', 'success') OR status IS NULL)"],o=[s];t?.currentTraceId&&(a.push("trace_id != ?"),o.push(t.currentTraceId));let p=a.join(" AND "),g=`SELECT ${H.join(", ")} FROM tasks WHERE ${p} ORDER BY started_at ASC LIMIT ?`;o.push(r);let e=this.db.prepare(g).all(...o),n=this.db.prepare("SELECT platform, title, first_message, last_message, message_count, updated_at FROM threads WHERE id = ?").get(s),d=n?.platform??"cli",u=this.rowsToMessages(e);return {threadId:s,platform:d,messages:u,metadata:{title:n?.title??void 0,firstMessage:n?.first_message??void 0,lastMessage:n?.last_message??void 0,messageCount:n?.message_count??0,updatedAt:n?.updated_at?new Date(n.updated_at).getTime():void 0}}}async saveUserMessage(s,t,r,a){let o=new Date().toISOString();this.db.prepare(`UPDATE threads
45
- SET first_message = COALESCE(first_message, ?),
46
- title = CASE WHEN title_locked = 0 AND title IS NULL THEN substr(?, 1, 60) ELSE title END,
47
- last_message = ?,
48
- message_count = message_count + 1,
49
- updated_at = ?
50
- WHERE id = ?`).run(t,t,t,o,s);}async saveAssistantMessage(s,t,r,a){let o=new Date().toISOString();this.db.prepare(`UPDATE threads
51
- SET last_message = ?,
52
- updated_at = ?
53
- WHERE id = ?`).run(t,o,s);}close(){this.db.close();}rowsToMessages(s){let t=[];for(let r of s){r.prompt&&t.push({id:`${r.id}-user`,text:r.prompt,isAssistant:false,timestamp:new Date(r.started_at).getTime()});let a=q(r.result);a&&t.push({id:`${r.id}-assistant`,text:a,isAssistant:true,timestamp:new Date(r.started_at).getTime()});}return t}};var T=class extends l{name="conversation";_provider;unsubStart=null;unsubEnd=null;constructor(s){super(),this._provider=new m(s?.dbPath);}get conversationProvider(){return this._provider}attach(s){this.unsubStart=s.on("task:start",async t=>{if(!t.threadId)return;let r=t.platform??"cli";try{await this._provider.ensureThread(t.threadId,r,t.workspaceId),await this._provider.saveUserMessage(t.threadId,t.message??"");}catch{}}),this.unsubEnd=s.on("task:end",async t=>{if(!t.result)return;let r=t.metadata?.threadId;if(!r)return;let a=t.agentRef?.replace(/^@/,"")??"";try{await this._provider.saveAssistantMessage(r,t.result,a);}catch{}});}detach(s){this.unsubStart?.(),this.unsubStart=null,this.unsubEnd?.(),this.unsubEnd=null,this._provider.close?.();}};exports.ConversationPlugin=T;exports.FileLoggerPlugin=f;exports.SqliteTracingPlugin=_;
8
+ `;fs.writeFileSync(n,r,{encoding:"utf8",mode:384});}catch{}}),t.on("task:output",e=>{try{let s=this.logFiles.get(e.traceId);if(!s)return;let n=new Date().toISOString();fs.appendFileSync(s,`[${n}] STDOUT: ${e.output}
9
+ `,"utf8");}catch{}}),t.on("task:end",e=>{try{let s=this.logFiles.get(e.traceId);if(!s)return;let n=new Date().toLocaleString(),r=e.error?`failed: ${e.error.message}`:"completed successfully",i=`[${n}] INFO: Task ${r} in ${e.durationMs}ms
10
+ [${n}] INFO: Process closed with exit code: ${e.error?1:0}
11
+ `;fs.appendFileSync(s,i,"utf8"),this.logFiles.delete(e.traceId);}catch{}}));}detach(t){this.unsubs.forEach(e=>e()),this.unsubs=[],this.logFiles.clear();}};function Tt(d){let t=ut__namespace.resolve(d);return process.platform==="win32"&&(t=t.replace(/\\/g,"/"),t=t.replace(/^([A-Z]):/,(e,s)=>`${s.toLowerCase()}:`)),t.length>1&&!/^[a-zA-Z]:\/$/.test(t)&&(t=t.replace(/\/+$/,"")),t}function ct(d){let t=Tt(d);return crypto.createHash("sha256").update(t).digest("hex")}var F=class{resolveDbPath(){return process.env.CREWX_DB?process.env.CREWX_DB:process.env.CREWX_TRACES_DB?process.env.CREWX_TRACES_DB:ut.join(At__default.default.homedir(),".crewx","crewx.db")}resolveDbPaths(){return [this.resolveDbPath()]}isMissingTableError(t){return t instanceof Error&&/no such table:/i.test(t.message)}dbExists(t){return fs.existsSync(t??this.resolveDbPath())}};function w(d){let t=Q("better-sqlite3"),{drizzle:e}=Q("drizzle-orm/better-sqlite3"),s=new t(d);return s.exec("PRAGMA journal_mode = WAL"),s.exec("PRAGMA busy_timeout = 5000"),s.exec("PRAGMA foreign_keys = ON"),{db:e(s),runRaw:(n,r=[])=>s.prepare(n).run(...r),close:()=>s.close()}}var pt=new Set;function vt(d){let{migrate:t}=Q("drizzle-orm/better-sqlite3/migrator"),e=ut__namespace.default.join(__dirname,"../migrations"),s=fs.existsSync(e)?e:ut__namespace.default.join(__dirname,"../../../../drizzle/migrations");t(d,{migrationsFolder:s});}function z(d,t){pt.has(t)||(vt(d),pt.add(t));}var c=class extends Error{code;cause;constructor(t,e,s){super(e),this.name="RepositoryError",this.code=t,this.cause=s,Object.setPrototypeOf(this,new.target.prototype);}};var $=sqliteCore.sqliteTable("workspaces",{id:sqliteCore.text("id").primaryKey(),slug:sqliteCore.text("slug").notNull().unique(),name:sqliteCore.text("name").notNull(),workspace_path:sqliteCore.text("workspace_path"),description:sqliteCore.text("description"),is_active:sqliteCore.integer("is_active").notNull().default(1),created_at:sqliteCore.text("created_at").notNull(),updated_at:sqliteCore.text("updated_at").notNull()});var o=sqliteCore.sqliteTable("tasks",{id:sqliteCore.text("id").primaryKey(),agent_id:sqliteCore.text("agent_id").notNull(),user_id:sqliteCore.text("user_id"),prompt:sqliteCore.text("prompt").notNull(),mode:sqliteCore.text("mode").notNull().default("execute"),status:sqliteCore.text("status").notNull().default("running"),result:sqliteCore.text("result"),error:sqliteCore.text("error"),started_at:sqliteCore.text("started_at").notNull(),completed_at:sqliteCore.text("completed_at"),duration_ms:sqliteCore.integer("duration_ms"),metadata:sqliteCore.text("metadata"),workspace_id:sqliteCore.text("workspace_id"),workspace_name:sqliteCore.text("workspace_name"),trace_id:sqliteCore.text("trace_id"),parent_task_id:sqliteCore.text("parent_task_id"),caller_agent_id:sqliteCore.text("caller_agent_id"),model:sqliteCore.text("model"),platform:sqliteCore.text("platform").default("cli"),crewx_version:sqliteCore.text("crewx_version"),input_tokens:sqliteCore.integer("input_tokens").default(0),output_tokens:sqliteCore.integer("output_tokens").default(0),cost_usd:sqliteCore.real("cost_usd").default(0),pid:sqliteCore.integer("pid"),rendered_prompt:sqliteCore.text("rendered_prompt"),command:sqliteCore.text("command"),coding_agent_command:sqliteCore.text("coding_agent_command"),exit_code:sqliteCore.integer("exit_code"),logs:sqliteCore.text("logs"),thread_id:sqliteCore.text("thread_id"),workspace_ref:sqliteCore.text("workspace_ref"),project_id:sqliteCore.text("project_id"),project_name:sqliteCore.text("project_name"),project_ref:sqliteCore.text("project_ref"),cached_input_tokens:sqliteCore.integer("cached_input_tokens").default(0)},d=>({idx_tasks_agent_id:sqliteCore.index("idx_tasks_agent_id").on(d.agent_id),idx_tasks_status:sqliteCore.index("idx_tasks_status").on(d.status),idx_tasks_started_at:sqliteCore.index("idx_tasks_started_at").on(d.started_at),idx_tasks_trace_id:sqliteCore.index("idx_tasks_trace_id").on(d.trace_id),idx_tasks_parent_task_id:sqliteCore.index("idx_tasks_parent_task_id").on(d.parent_task_id),idx_tasks_crewx_version:sqliteCore.index("idx_tasks_crewx_version").on(d.crewx_version),idx_tasks_pid:sqliteCore.index("idx_tasks_pid").on(d.pid),idx_tasks_thread_id:sqliteCore.index("idx_tasks_thread_id").on(d.thread_id),idx_tasks_workspace_id:sqliteCore.index("idx_tasks_workspace_id").on(d.workspace_id),idx_tasks_workspace_ref:sqliteCore.index("idx_tasks_workspace_ref").on(d.workspace_ref),idx_tasks_project_id:sqliteCore.index("idx_tasks_project_id").on(d.project_id),idx_tasks_ws_started:sqliteCore.index("idx_tasks_ws_started").on(d.workspace_id,d.started_at)}));var u=sqliteCore.sqliteTable("threads",{id:sqliteCore.text("id").primaryKey(),workspace_id:sqliteCore.text("workspace_id").references(()=>$.id,{onDelete:"set null"}),platform:sqliteCore.text("platform").notNull().default("cli"),title:sqliteCore.text("title"),first_message:sqliteCore.text("first_message"),last_message:sqliteCore.text("last_message"),message_count:sqliteCore.integer("message_count").notNull().default(0),created_at:sqliteCore.text("created_at").notNull(),updated_at:sqliteCore.text("updated_at").notNull(),metadata:sqliteCore.text("metadata"),title_locked:sqliteCore.integer("title_locked").notNull().default(0)},d=>({idx_threads_updated_at:sqliteCore.index("idx_threads_updated_at").on(d.updated_at),idx_threads_workspace_id:sqliteCore.index("idx_threads_workspace_id").on(d.workspace_id)}));var Ft=sqliteCore.sqliteTable("spans",{id:sqliteCore.text("id").primaryKey(),task_id:sqliteCore.text("task_id").references(()=>o.id,{onDelete:"set null"}),parent_span_id:sqliteCore.text("parent_span_id").references(()=>Ft.id,{onDelete:"set null"}),name:sqliteCore.text("name").notNull(),kind:sqliteCore.text("kind").notNull().default("internal"),status:sqliteCore.text("status").notNull().default("ok"),started_at:sqliteCore.text("started_at").notNull(),completed_at:sqliteCore.text("completed_at"),duration_ms:sqliteCore.integer("duration_ms"),input:sqliteCore.text("input"),output:sqliteCore.text("output"),error:sqliteCore.text("error"),attributes:sqliteCore.text("attributes")},d=>({idx_spans_task_id:sqliteCore.index("idx_spans_task_id").on(d.task_id),idx_spans_parent_span_id:sqliteCore.index("idx_spans_parent_span_id").on(d.parent_span_id)}));sqliteCore.sqliteTable("tool_calls",{id:sqliteCore.text("id").primaryKey(),task_id:sqliteCore.text("task_id").notNull().references(()=>o.id,{onDelete:"cascade"}),session_id:sqliteCore.text("session_id"),tool_name:sqliteCore.text("tool_name").notNull(),files:sqliteCore.text("files"),input:sqliteCore.text("input"),output:sqliteCore.text("output"),duration_ms:sqliteCore.integer("duration_ms"),timestamp:sqliteCore.text("timestamp").notNull()},d=>({idx_tool_calls_task_id:sqliteCore.index("idx_tool_calls_task_id").on(d.task_id),idx_tool_calls_tool_name:sqliteCore.index("idx_tool_calls_tool_name").on(d.tool_name),idx_tool_calls_timestamp:sqliteCore.index("idx_tool_calls_timestamp").on(d.timestamp)}));sqliteCore.sqliteTable("thread_boxes",{id:sqliteCore.text("id").primaryKey(),thread_id:sqliteCore.text("thread_id").notNull().references(()=>u.id,{onDelete:"cascade"}),seq:sqliteCore.integer("seq").notNull(),first_task_id:sqliteCore.text("first_task_id").notNull(),mid_task_id:sqliteCore.text("mid_task_id").notNull(),last_task_id:sqliteCore.text("last_task_id").notNull(),task_count:sqliteCore.integer("task_count").notNull(),summary:sqliteCore.text("summary"),source_tokens:sqliteCore.integer("source_tokens").notNull(),summary_tokens:sqliteCore.integer("summary_tokens"),created_at:sqliteCore.text("created_at").notNull()},d=>({idx_thread_boxes_thread_id:sqliteCore.index("idx_thread_boxes_thread_id").on(d.thread_id),idx_thread_boxes_seq:sqliteCore.index("idx_thread_boxes_seq").on(d.thread_id,d.seq),uniq_thread_boxes_thread_seq:sqliteCore.unique().on(d.thread_id,d.seq)}));sqliteCore.sqliteTable("request_logs",{id:sqliteCore.text("id").primaryKey(),path:sqliteCore.text("path").notNull(),method:sqliteCore.text("method").notNull(),status_code:sqliteCore.integer("status_code").notNull(),duration_ms:sqliteCore.integer("duration_ms").notNull(),ip:sqliteCore.text("ip"),request_headers:sqliteCore.text("request_headers"),response_headers:sqliteCore.text("response_headers"),request_body:sqliteCore.text("request_body"),response_body:sqliteCore.text("response_body"),query:sqliteCore.text("query"),user_id:sqliteCore.text("user_id"),project_id:sqliteCore.text("project_id"),partition_key:sqliteCore.text("partition_key").notNull(),timestamp:sqliteCore.text("timestamp").notNull().default(drizzleOrm.sql`(datetime('now'))`),metadata:sqliteCore.text("metadata")},d=>({idx_request_logs_timestamp:sqliteCore.index("idx_request_logs_timestamp").on(d.timestamp),idx_request_logs_path:sqliteCore.index("idx_request_logs_path").on(d.path),idx_request_logs_status_code:sqliteCore.index("idx_request_logs_status_code").on(d.status_code),idx_request_logs_partition_key:sqliteCore.index("idx_request_logs_partition_key").on(d.partition_key)}));var K=class extends F{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=ut.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let n=ut.dirname(e);fs.existsSync(n)||fs.mkdirSync(n,{recursive:true});}else if(!fs.existsSync(e))throw new c("NOT_FOUND","Database not found");let s=w(e);if(t)try{z(s.db,e);}catch(n){throw s.close(),n}return s}startTask(t){let e=this.openHandle(true);try{e.db.insert(o).values({id:t.id,agent_id:t.agentId,prompt:t.prompt,mode:t.mode,status:t.status,started_at:t.startedAt,pid:t.pid??null,parent_task_id:t.parentTaskId??null,caller_agent_id:t.callerAgentId??null,trace_id:t.traceId??null,command:t.command??null,metadata:t.metadata??null,workspace_id:t.workspaceId??null,workspace_name:t.workspaceName??null,platform:t.platform??"cli",crewx_version:t.crewxVersion??null,thread_id:t.threadId??null,model:t.model??null,rendered_prompt:t.renderedPrompt??null,coding_agent_command:t.codingAgentCommand??null}).onConflictDoNothing().run();}catch(s){throw s instanceof c?s:new c("DB_ERROR","Failed to start task",s)}finally{e.close();}}finishTask(t){let e=this.openHandle(true);try{e.runRaw(`UPDATE tasks SET status=?, result=?, error=?, completed_at=?, duration_ms=?,
12
+ exit_code=?, input_tokens=?, output_tokens=?, cached_input_tokens=?, cost_usd=?,
13
+ model=COALESCE(?, model) WHERE id=?`,[t.status,t.result??null,t.error??null,t.completedAt,t.durationMs??null,t.exitCode??null,t.inputTokens??0,t.outputTokens??0,t.cachedInputTokens??0,t.costUsd??0,t.model??null,t.id]);}catch(s){throw s instanceof c?s:new c("DB_ERROR","Failed to finish task",s)}finally{e.close();}}appendLog(t,e){let s=this.openHandle(true);try{s.db.transaction(n=>{let r=n.select({logs:o.logs}).from(o).where(drizzleOrm.eq(o.id,t)).limit(1).get(),i=r?.logs?JSON.parse(r.logs):[];i.push(e),n.update(o).set({logs:JSON.stringify(i)}).where(drizzleOrm.eq(o.id,t)).run();},{behavior:"immediate"});}catch(n){throw n instanceof c?n:new c("DB_ERROR","Failed to append log",n)}finally{s.close();}}getRunningTasks(){if(!this.dbExists())return [];let t=this.openHandle(false);try{return t.db.select().from(o).where(drizzleOrm.eq(o.status,"running")).orderBy(drizzleOrm.desc(o.started_at)).all()}catch(e){throw new c("DB_ERROR","Failed to get running tasks",e)}finally{t.close();}}getAllTasks(){if(!this.dbExists())return [];let t=this.openHandle(false);try{return t.db.select().from(o).orderBy(drizzleOrm.desc(o.started_at)).limit(100).all()}catch(e){throw new c("DB_ERROR","Failed to get all tasks",e)}finally{t.close();}}getTask(t){if(!this.dbExists())return;let e=this.openHandle(false);try{return e.db.select().from(o).where(drizzleOrm.eq(o.id,t)).limit(1).get()??void 0}catch(s){throw new c("DB_ERROR","Failed to get task",s)}finally{e.close();}}killTask(t){if(!this.dbExists())return {killed:false};let e=this.openHandle(true);try{let s=e.db.select({id:o.id,status:o.status,pid:o.pid}).from(o).where(drizzleOrm.eq(o.id,t)).limit(1).get();if(!s||s.status!=="running")return {killed:!1};if(s.pid)try{process.kill(s.pid,"SIGTERM");}catch{}return e.db.update(o).set({status:"failed",error:"Killed by user",completed_at:new Date().toISOString()}).where(drizzleOrm.and(drizzleOrm.eq(o.id,t),drizzleOrm.eq(o.status,"running"))).run(),{killed:!0,pid:s.pid??void 0}}catch(s){throw s instanceof c?s:new c("DB_ERROR","Failed to kill task",s)}finally{e.close();}}findTaskStatus(t,e){let s=this.resolveDbPaths();for(let n of s){if(!fs.existsSync(n))continue;let r=w(n);try{let i=e?drizzleOrm.eq(o.workspace_id,e):void 0,l=i?drizzleOrm.and(drizzleOrm.eq(o.id,t),i):drizzleOrm.eq(o.id,t),a=r.db.select().from(o).where(l).limit(1).get()??void 0;if(!a){let _=drizzleOrm.or(drizzleOrm.eq(o.thread_id,t),drizzleOrm.and(drizzleOrm.isNull(o.thread_id),drizzleOrm.like(o.command,`%--thread=${t}%`))),p=i?drizzleOrm.and(_,i):_;a=r.db.select().from(o).where(p).orderBy(drizzleOrm.desc(o.started_at)).limit(1).get()??void 0;}if(a)return a}catch(i){throw new c("DB_ERROR","Failed to find task status",i)}finally{r.close();}}}findChildTasks(t,e){let s=this.resolveDbPaths(),n=new Set,r=[];for(let i of s){if(!fs.existsSync(i))continue;let l=w(i);try{let a=e?drizzleOrm.and(drizzleOrm.eq(o.parent_task_id,t),drizzleOrm.eq(o.workspace_id,e)):drizzleOrm.eq(o.parent_task_id,t),_=l.db.select().from(o).where(a).orderBy(drizzleOrm.asc(o.started_at)).all();for(let p of _)n.has(p.id)||(n.add(p.id),r.push(p));}catch(a){throw new c("DB_ERROR","Failed to find child tasks",a)}finally{l.close();}}return r}getWorkspaceUsageSummary(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.all(t?drizzleOrm.sql`
14
+ SELECT
15
+ COALESCE(workspace_id, 'unknown') AS workspace_id,
16
+ COALESCE(workspace_name, 'Unknown Workspace') AS workspace_name,
17
+ COALESCE(SUM(input_tokens), 0) AS input_tokens,
18
+ COALESCE(SUM(output_tokens), 0) AS output_tokens,
19
+ COALESCE(SUM(cost_usd), 0) AS cost_usd,
20
+ COUNT(*) AS task_count
21
+ FROM tasks
22
+ WHERE workspace_id = ${t}
23
+ GROUP BY workspace_id, workspace_name
24
+ ORDER BY (COALESCE(SUM(input_tokens), 0) + COALESCE(SUM(output_tokens), 0)) DESC
25
+ `:drizzleOrm.sql`
26
+ SELECT
27
+ COALESCE(workspace_id, 'unknown') AS workspace_id,
28
+ COALESCE(workspace_name, 'Unknown Workspace') AS workspace_name,
29
+ COALESCE(SUM(input_tokens), 0) AS input_tokens,
30
+ COALESCE(SUM(output_tokens), 0) AS output_tokens,
31
+ COALESCE(SUM(cost_usd), 0) AS cost_usd,
32
+ COUNT(*) AS task_count
33
+ FROM tasks
34
+ GROUP BY workspace_id, workspace_name
35
+ ORDER BY (COALESCE(SUM(input_tokens), 0) + COALESCE(SUM(output_tokens), 0)) DESC
36
+ `)}catch(s){throw new c("DB_ERROR","Failed to get workspace usage summary",s)}finally{e.close();}}getThreadTokenUsage(t,e){let s=this.resolveDbPaths(),n=new Set,r=0,i=0,l=0;for(let a of s){if(!fs.existsSync(a))continue;let _=w(a);try{let p=drizzleOrm.or(drizzleOrm.eq(o.thread_id,t),drizzleOrm.and(drizzleOrm.isNull(o.thread_id),drizzleOrm.like(o.command,`%--thread=${t}%`))),g=e?drizzleOrm.and(p,drizzleOrm.eq(o.workspace_id,e)):p,D=_.db.select({id:o.id,input_tokens:o.input_tokens,output_tokens:o.output_tokens,cost_usd:o.cost_usd}).from(o).where(g).all();for(let v of D)n.has(v.id)||(n.add(v.id),r+=v.input_tokens??0,i+=v.output_tokens??0,l+=v.cost_usd??0);}catch(p){throw new c("DB_ERROR","Failed to get thread token usage",p)}finally{_.close();}}return {inputTokens:r,outputTokens:i,costUsd:l}}findTasksByThread(t,e){let s=this.resolveDbPaths(),n=new Set,r=[];for(let i of s){if(!fs.existsSync(i))continue;let l=w(i);try{let a=drizzleOrm.or(drizzleOrm.eq(o.thread_id,t),drizzleOrm.and(drizzleOrm.isNull(o.thread_id),drizzleOrm.like(o.command,`%--thread=${t}%`))),_=e?drizzleOrm.and(a,drizzleOrm.eq(o.workspace_id,e)):a,p=l.db.select().from(o).where(_).orderBy(drizzleOrm.asc(o.started_at)).all();for(let g of p)n.has(g.id)||(n.add(g.id),r.push(g));}catch(a){throw new c("DB_ERROR","Failed to find tasks by thread",a)}finally{l.close();}}return r}findAllTasks(t){if(!this.dbExists())return {rows:[],total:0};let e=this.openHandle(false);try{let s=[];t.workspaceId&&s.push(drizzleOrm.eq(o.workspace_id,t.workspaceId));let n=t.agents&&t.agents.length>0?t.agents:t.agentId?[t.agentId]:null;n&&s.push(drizzleOrm.inArray(o.agent_id,n));let r=t.statuses&&t.statuses.length>0?t.statuses:t.status?[t.status]:null;r&&s.push(drizzleOrm.inArray(o.status,r));let i=t.q??t.search;i&&s.push(drizzleOrm.like(o.prompt,`%${i}%`)),t.from&&s.push(drizzleOrm.gte(o.started_at,t.from)),t.to&&s.push(drizzleOrm.lt(o.started_at,t.to));let l=s.length>0?drizzleOrm.and(...s):void 0,a=e.db.select({count:drizzleOrm.sql`count(*)`}).from(o).where(l).get(),_=(t.sortDir??"DESC")==="ASC"?drizzleOrm.asc(o.started_at):drizzleOrm.desc(o.started_at);return {rows:e.db.select().from(o).where(l).orderBy(_).limit(t.limit).offset(t.offset).all(),total:a?.count??0}}catch(s){throw new c("DB_ERROR","Failed to find all tasks",s)}finally{e.close();}}getAgentUsage(t,e,s){if(!this.dbExists())return [];let n=this.openHandle(false);try{return n.db.all(s?drizzleOrm.sql`
37
+ SELECT
38
+ t.agent_id,
39
+ COUNT(*) AS total_tasks,
40
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
41
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
42
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
43
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
44
+ FROM tasks t
45
+ WHERE t.status IN ('completed', 'success')
46
+ AND t.started_at >= ${t}
47
+ AND t.started_at < ${e}
48
+ AND t.workspace_name = ${s}
49
+ GROUP BY t.agent_id
50
+ ORDER BY (COALESCE(SUM(t.input_tokens), 0) + COALESCE(SUM(t.output_tokens), 0)) DESC
51
+ `:drizzleOrm.sql`
52
+ SELECT
53
+ t.agent_id,
54
+ COUNT(*) AS total_tasks,
55
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
56
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
57
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
58
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
59
+ FROM tasks t
60
+ WHERE t.status IN ('completed', 'success')
61
+ AND t.started_at >= ${t}
62
+ AND t.started_at < ${e}
63
+ GROUP BY t.agent_id
64
+ ORDER BY (COALESCE(SUM(t.input_tokens), 0) + COALESCE(SUM(t.output_tokens), 0)) DESC
65
+ `).map(i=>({agentId:i.agent_id,totalTasks:i.total_tasks,inputTokens:i.input_tokens,outputTokens:i.output_tokens,cachedInputTokens:i.cached_input_tokens,costUsd:i.cost_usd}))}catch(r){throw new c("DB_ERROR","Failed to get agent usage",r)}finally{n.close();}}getAgentUsageTrendRaw(t,e,s){if(!this.dbExists())return [];let n=this.openHandle(false);try{return n.db.all(s?drizzleOrm.sql`
66
+ SELECT
67
+ date(t.started_at) AS date,
68
+ t.agent_id,
69
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
70
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
71
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
72
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
73
+ FROM tasks t
74
+ WHERE t.status IN ('completed', 'success')
75
+ AND t.started_at >= ${t}
76
+ AND t.started_at < ${e}
77
+ AND t.workspace_name = ${s}
78
+ GROUP BY date(t.started_at), t.agent_id
79
+ ORDER BY date(t.started_at) ASC
80
+ `:drizzleOrm.sql`
81
+ SELECT
82
+ date(t.started_at) AS date,
83
+ t.agent_id,
84
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
85
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
86
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
87
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
88
+ FROM tasks t
89
+ WHERE t.status IN ('completed', 'success')
90
+ AND t.started_at >= ${t}
91
+ AND t.started_at < ${e}
92
+ GROUP BY date(t.started_at), t.agent_id
93
+ ORDER BY date(t.started_at) ASC
94
+ `).map(i=>({date:i.date,agentId:i.agent_id,inputTokens:i.input_tokens,outputTokens:i.output_tokens,cachedInputTokens:i.cached_input_tokens,costUsd:i.cost_usd}))}catch(r){throw new c("DB_ERROR","Failed to get agent usage trend",r)}finally{n.close();}}findTaskForStop(t,e){if(!this.dbExists())return;let s=this.openHandle(false);try{return s.db.select().from(o).where(drizzleOrm.and(drizzleOrm.eq(o.id,t),drizzleOrm.eq(o.workspace_id,e))).limit(1).get()??void 0}catch(n){throw new c("DB_ERROR","Failed to find task for stop",n)}finally{s.close();}}markTaskFailed(t,e,s){if(!this.dbExists())return;let n=this.openHandle(true);try{let r=new Date().toISOString(),i=s?drizzleOrm.and(drizzleOrm.eq(o.id,t),drizzleOrm.eq(o.status,"running"),drizzleOrm.eq(o.workspace_id,s)):drizzleOrm.and(drizzleOrm.eq(o.id,t),drizzleOrm.eq(o.status,"running"));n.db.update(o).set({status:"failed",error:e,completed_at:r}).where(i).run();}catch(r){throw r instanceof c?r:new c("DB_ERROR","Failed to mark task failed",r)}finally{n.close();}}findTasksByPromptHint(t,e){let s=this.resolveDbPaths(),n=new Set,r=[];for(let i of s){if(!fs.existsSync(i))continue;let l=w(i);try{let a=e?drizzleOrm.and(drizzleOrm.like(o.prompt,`%${t}%`),drizzleOrm.eq(o.workspace_id,e)):drizzleOrm.like(o.prompt,`%${t}%`),_=l.db.select().from(o).where(a).orderBy(drizzleOrm.asc(o.started_at)).all();for(let p of _)n.has(p.id)||(n.add(p.id),r.push(p));}catch(a){throw new c("DB_ERROR","Failed to find tasks by prompt hint",a)}finally{l.close();}}return r}};var rt=class extends A{name="sqlite-tracing";unsubs=[];dbPath;version;constructor(t){super(),this.dbPath=ut.join(t?.dbRoot??At.homedir(),".crewx","crewx.db"),this.version=t?.version??"unknown";}attach(t){let e=new K({dbPath:this.dbPath}),s=process.cwd(),n=fs.existsSync(ut.join(s,"crewx.yaml"))||fs.existsSync(ut.join(s,"crewx.yml")),r=n?ct(s):null,i=n?ut.basename(s):null,l=process.argv.join(" ");this.unsubs.push(t.on("task:start",a=>{try{let _=process.env.CREWX_CALLER_AGENT_ID||null,p=process.env.CREWX_PARENT_TASK_ID||null,g=process.env.CREWX_TRACE_ID||a.traceId,D=a.metadata?JSON.stringify(a.metadata):JSON.stringify({provider:a.provider??"cli/claude"});e.startTask({id:a.traceId,agentId:a.agentRef.replace(/^@/,""),prompt:a.message,mode:a.mode,status:"running",pid:a.pid??null,startedAt:a.timestamp.toISOString(),crewxVersion:this.version,platform:a.platform??"cli",model:a.model??null,renderedPrompt:a.renderedPrompt??null,command:l,codingAgentCommand:a.codingAgentCommand??null,workspaceId:a.workspaceId??r,workspaceName:a.workspaceName??i,callerAgentId:_,parentTaskId:p,traceId:g,metadata:D,threadId:a.threadId??null});}catch{}}),t.on("task:output",a=>{try{e.appendLog(a.traceId,{timestamp:a.timestamp.toISOString(),level:a.level??"stdout",message:a.output});}catch{}}),t.on("task:end",a=>{try{e.finishTask({id:a.traceId,status:a.error?"failed":"success",result:a.result??null,error:a.error?JSON.stringify(a.error):null,completedAt:a.timestamp.toISOString(),durationMs:a.durationMs,exitCode:a.exitCode??null,inputTokens:a.inputTokens??0,outputTokens:a.outputTokens??0,cachedInputTokens:a.cachedInputTokens??0,costUsd:a.costUsd??0,model:a.model??null});}catch{}}));}detach(t){this.unsubs.forEach(e=>e()),this.unsubs=[];}};var J=class extends F{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=ut.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let n=ut.dirname(e);fs.existsSync(n)||fs.mkdirSync(n,{recursive:true});}else if(!fs.existsSync(e))throw new c("NOT_FOUND","Database not found");let s=w(e);if(t)try{z(s.db,e);}catch(n){throw s.close(),n}return s}validateWorkspaceId(t,e){return t.db.select({id:$.id}).from($).where(drizzleOrm.eq($.id,e)).limit(1).get()?e:null}findAllThreads(t){let e=this.resolveDbPaths(),s=new Set,n=[];for(let r of e){if(!fs.existsSync(r))continue;let i=w(r);try{let l=t?drizzleOrm.or(drizzleOrm.eq(u.workspace_id,t),drizzleOrm.isNull(u.workspace_id)):void 0,a=i.db.select().from(u).where(l).orderBy(drizzleOrm.desc(u.updated_at)).all();for(let _ of a)s.has(_.id)||(s.add(_.id),n.push(_));}catch(l){throw new c("DB_ERROR","Failed to find all threads",l)}finally{i.close();}}return n}findThreadById(t,e){let s=this.resolveDbPaths();for(let n of s){if(!fs.existsSync(n))continue;let r=w(n);try{let i=drizzleOrm.eq(u.id,t),l=e?drizzleOrm.and(i,drizzleOrm.or(drizzleOrm.eq(u.workspace_id,e),drizzleOrm.isNull(u.workspace_id))):i,a=r.db.select().from(u).where(l).limit(1).get()??void 0;if(a)return a}catch(i){throw new c("DB_ERROR","Failed to find thread by id",i)}finally{r.close();}}}threadExists(t,e){let s=this.resolveDbPaths();for(let n of s){if(!fs.existsSync(n))continue;let r=w(n);try{let i=drizzleOrm.eq(u.id,t),l=e?drizzleOrm.and(i,drizzleOrm.or(drizzleOrm.eq(u.workspace_id,e),drizzleOrm.isNull(u.workspace_id))):i;if(r.db.select({id:u.id}).from(u).where(l).limit(1).get())return !0}catch(i){throw new c("DB_ERROR","Failed to check thread existence",i)}finally{r.close();}}return false}aggregateTaskStats(t,e){let s=this.resolveDbPaths(),n=0,r=0,i=0,l=0,a=0,_=new Set;for(let p of s){if(!fs.existsSync(p))continue;let g=w(p);try{let D=drizzleOrm.and(drizzleOrm.eq(o.thread_id,t),drizzleOrm.or(drizzleOrm.isNull(o.parent_task_id),drizzleOrm.eq(o.parent_task_id,""))),v=e?drizzleOrm.and(D,drizzleOrm.eq(o.workspace_id,e)):D,B=g.db.select({cnt:drizzleOrm.sql`count(*)`,total_input:drizzleOrm.sql`COALESCE(SUM(input_tokens), 0)`,total_output:drizzleOrm.sql`COALESCE(SUM(output_tokens), 0)`,total_cached:drizzleOrm.sql`COALESCE(SUM(cached_input_tokens), 0)`,total_cost:drizzleOrm.sql`COALESCE(SUM(cost_usd), 0)`}).from(o).where(v).get();B&&(n+=B.cnt,r+=B.total_input,i+=B.total_output,l+=B.total_cached,a+=B.total_cost);let bt=g.db.all(drizzleOrm.sql`
95
+ SELECT DISTINCT agent_id FROM tasks
96
+ WHERE thread_id = ${t}
97
+ AND agent_id IS NOT NULL AND agent_id != ''
98
+ AND (parent_task_id IS NULL OR parent_task_id = '')
99
+ ${e?drizzleOrm.sql`AND workspace_id = ${e}`:drizzleOrm.sql``}
100
+ `);for(let yt of bt)_.add(yt.agent_id);}catch(D){throw new c("DB_ERROR","Failed to aggregate task stats",D)}finally{g.close();}}return {taskCount:n,inputTokens:r,outputTokens:i,cachedInputTokens:l,costUsd:a,agentIds:Array.from(_)}}findTopLevelTasks(t,e){let s=this.resolveDbPaths(),n=new Set,r=[];for(let i of s){if(!fs.existsSync(i))continue;let l=w(i);try{let a=drizzleOrm.and(drizzleOrm.eq(o.thread_id,t),drizzleOrm.or(drizzleOrm.isNull(o.parent_task_id),drizzleOrm.eq(o.parent_task_id,""))),_=e?drizzleOrm.and(a,drizzleOrm.eq(o.workspace_id,e)):a,p=l.db.select().from(o).where(_).orderBy(drizzleOrm.asc(o.started_at)).all();for(let g of p)n.has(g.id)||(n.add(g.id),r.push(g));}catch(a){throw new c("DB_ERROR","Failed to find top-level tasks",a)}finally{l.close();}}return r}findAllTasks(t,e){let s=this.resolveDbPaths(),n=new Set,r=[];for(let i of s){if(!fs.existsSync(i))continue;let l=w(i);try{let a=drizzleOrm.eq(o.thread_id,t),_=e?drizzleOrm.and(a,drizzleOrm.eq(o.workspace_id,e)):a,p=l.db.select().from(o).where(_).orderBy(drizzleOrm.asc(o.started_at)).all();for(let g of p)n.has(g.id)||(n.add(g.id),r.push(g));}catch(a){throw new c("DB_ERROR","Failed to find all tasks for thread",a)}finally{l.close();}}return r}findTaskById(t,e,s){let n=this.resolveDbPaths();for(let r of n){if(!fs.existsSync(r))continue;let i=w(r);try{let l=drizzleOrm.and(drizzleOrm.eq(o.id,e),drizzleOrm.eq(o.thread_id,t)),a=s?drizzleOrm.and(l,drizzleOrm.eq(o.workspace_id,s)):l,_=i.db.select().from(o).where(a).limit(1).get();if(!_)continue;let p=i.db.select().from(o).where(drizzleOrm.eq(o.parent_task_id,_.id)).orderBy(drizzleOrm.asc(o.started_at)).all();return {task:_,children:p}}catch(l){throw new c("DB_ERROR","Failed to find task by id",l)}finally{i.close();}}}batchFetchTasks(t,e){let s=new Map;if(t.length===0)return s;let n=this.resolveDbPaths();for(let r of n){if(!fs.existsSync(r))continue;let i=w(r);try{let l=drizzleOrm.and(drizzleOrm.inArray(o.thread_id,t),drizzleOrm.or(drizzleOrm.isNull(o.parent_task_id),drizzleOrm.eq(o.parent_task_id,""))),a=e?drizzleOrm.and(l,drizzleOrm.eq(o.workspace_id,e)):l,_=i.db.select().from(o).where(a).orderBy(drizzleOrm.asc(o.started_at)).all();for(let p of _){let g=p.thread_id;s.has(g)||s.set(g,[]),s.get(g).push(p);}}catch(l){throw new c("DB_ERROR","Failed to batch fetch tasks",l)}finally{i.close();}}return s}updateThreadTitle(t,e,s){if(!this.dbExists())return;let n=this.openHandle(true);try{let r=drizzleOrm.eq(u.id,t),i=s?drizzleOrm.and(r,drizzleOrm.or(drizzleOrm.eq(u.workspace_id,s),drizzleOrm.isNull(u.workspace_id))):r;if(!n.db.select({id:u.id}).from(u).where(i).limit(1).get())return;n.db.update(u).set({title:e,title_locked:1,updated_at:new Date().toISOString()}).where(drizzleOrm.eq(u.id,t)).run();}catch(r){throw r instanceof c?r:new c("DB_ERROR","Failed to update thread title",r)}finally{n.close();}}upsertThread(t,e){let s=this.openHandle(true);try{let n=e.workspaceId?this.validateWorkspaceId(s,e.workspaceId):null,r=new Date().toISOString();if(s.db.select({id:u.id,message_count:u.message_count}).from(u).where(drizzleOrm.eq(u.id,t)).limit(1).get()){let l={updated_at:r};e.title!==void 0&&(l.title=e.title),e.titleLocked!==void 0&&(l.title_locked=e.titleLocked?1:0),s.db.update(u).set(l).where(drizzleOrm.eq(u.id,t)).run();}else s.db.insert(u).values({id:t,platform:e.platform,workspace_id:n,title:e.title??null,title_locked:e.titleLocked?1:0,message_count:0,created_at:r,updated_at:r}).run();}catch(n){throw n instanceof c?n:new c("DB_ERROR","Failed to upsert thread",n)}finally{s.close();}}ensureThread(t,e,s){let n=this.openHandle(true);try{let r=s?this.validateWorkspaceId(n,s):null,i=n.db.select({id:u.id,platform:u.platform,workspace_id:u.workspace_id}).from(u).where(drizzleOrm.eq(u.id,t)).limit(1).get();if(i){r&&!i.workspace_id&&n.db.update(u).set({workspace_id:r}).where(drizzleOrm.eq(u.id,t)).run();return}let l=new Date().toISOString();n.db.insert(u).values({id:t,platform:e,workspace_id:r,message_count:0,created_at:l,updated_at:l}).run();}catch(r){throw r instanceof c?r:new c("DB_ERROR","Failed to ensure thread",r)}finally{n.close();}}saveUserMessage(t,e,s){if(!this.dbExists())return;let n=this.openHandle(true);try{let r=new Date().toISOString();n.db.run(drizzleOrm.sql`
101
+ UPDATE threads
102
+ SET first_message = COALESCE(first_message, ${e}),
103
+ title = CASE WHEN title_locked = 0 AND title IS NULL THEN substr(${e}, 1, 60) ELSE title END,
104
+ last_message = ${e},
105
+ message_count = message_count + 1,
106
+ updated_at = ${r}
107
+ WHERE id = ${t}
108
+ `);}catch(r){throw r instanceof c?r:new c("DB_ERROR","Failed to save user message",r)}finally{n.close();}}saveAssistantMessage(t,e,s){if(!this.dbExists())return;let n=this.openHandle(true);try{let r=new Date().toISOString();n.db.update(u).set({last_message:e,updated_at:r}).where(drizzleOrm.eq(u.id,t)).run();}catch(r){throw r instanceof c?r:new c("DB_ERROR","Failed to save assistant message",r)}finally{n.close();}}updateThread(t,e){if(!this.dbExists())return;let s=this.openHandle(true);try{let n={updated_at:new Date().toISOString()};e.title!==void 0&&(n.title=e.title,n.title_locked=1),e.titleLocked!==void 0&&(n.title_locked=e.titleLocked?1:0),s.db.update(u).set(n).where(drizzleOrm.eq(u.id,t)).run();}catch(n){throw n instanceof c?n:new c("DB_ERROR","Failed to update thread",n)}finally{s.close();}}};function ie(d){return d.replace(/<conversation_history[^>]*>[\s\S]*?<\/conversation_history>/g,"").split(`
109
+ `).filter(n=>!(n.startsWith("Loaded ")&&n.includes("layouts from")||n.includes("[dotenv@")||n.includes("[Nest]")&&n.includes("DEBUG")||n.startsWith("Registered custom layout:")||n.startsWith("Updated custom layout:"))).join(`
110
+ `).trim()}function ae(d){if(!d)return "";let t=d;try{let e=JSON.parse(t);Array.isArray(e)?t=e.filter(s=>s?.type==="text"&&s?.text).map(s=>s.text).join(`
111
+ `):e&&typeof e=="object"&&e.result!==void 0&&(t=e.result||"");}catch{t=ie(t);}return t}var V=class{dbPath;constructor(t){this.dbPath=t??ut.join(At.homedir(),".crewx","crewx.db");}getThreadRepo(){return new J({dbPath:this.dbPath})}updateThread(t,e){this.getThreadRepo().updateThread(t,{title:e.title});}async ensureThread(t,e,s){let n=this.getThreadRepo(),r=n.findThreadById(t);if(r){if(r.platform!==e)throw new Error(`Thread '${t}' already exists with platform '${r.platform}' \u2014 cannot change to '${e}' (platform is immutable)`);return}n.ensureThread(t,e,s);}async fetchHistory(t,e){let s=e?.limit??100,n=this.getThreadRepo(),r=n.findThreadById(t),i=n.findTopLevelTasks(t),l=new Set(["done","completed","success"]);i=i.filter(p=>(!p.status||l.has(p.status))&&(!e?.currentTraceId||p.trace_id!==e.currentTraceId)),i=i.slice(0,s);let a=r?.platform??"cli",_=this.rowsToMessages(i);return {threadId:t,platform:a,messages:_,metadata:{title:r?.title??void 0,firstMessage:r?.first_message??void 0,lastMessage:r?.last_message??void 0,messageCount:r?.message_count??0,updatedAt:r?.updated_at?new Date(r.updated_at).getTime():void 0}}}async saveUserMessage(t,e,s,n){this.getThreadRepo().saveUserMessage(t,e);}async saveAssistantMessage(t,e,s,n){this.getThreadRepo().saveAssistantMessage(t,e);}close(){}rowsToMessages(t){let e=[];for(let s of t){s.prompt&&e.push({id:`${s.id}-user`,text:s.prompt,isAssistant:false,timestamp:new Date(s.started_at).getTime()});let n=ae(s.result);n&&e.push({id:`${s.id}-assistant`,text:n,isAssistant:true,timestamp:new Date(s.started_at).getTime()});}return e}};var ot=class extends A{name="conversation";_provider;unsubStart=null;unsubEnd=null;constructor(t){super(),this._provider=new V(t?.dbPath);}get conversationProvider(){return this._provider}attach(t){this.unsubStart=t.on("task:start",async e=>{if(!e.threadId)return;let s=e.platform??"cli";try{await this._provider.ensureThread(e.threadId,s,e.workspaceId),await this._provider.saveUserMessage(e.threadId,e.message??"");}catch{}}),this.unsubEnd=t.on("task:end",async e=>{if(!e.result)return;let s=e.metadata?.threadId;if(!s)return;let n=e.agentRef?.replace(/^@/,"")??"";try{await this._provider.saveAssistantMessage(s,e.result,n);}catch{}});}detach(t){this.unsubStart?.(),this.unsubStart=null,this.unsubEnd?.(),this.unsubEnd=null,this._provider.close?.();}};exports.ConversationPlugin=ot;exports.FileLoggerPlugin=Z;exports.SqliteTracingPlugin=rt;
@@ -6,7 +6,6 @@ export interface SqliteTracingPluginOptions {
6
6
  }
7
7
  export declare class SqliteTracingPlugin extends CrewxPlugin {
8
8
  readonly name = "sqlite-tracing";
9
- private db;
10
9
  private unsubs;
11
10
  private readonly dbPath;
12
11
  private readonly version;
@@ -0,0 +1,6 @@
1
+ export declare abstract class BaseSqliteRepository {
2
+ resolveDbPath(): string;
3
+ resolveDbPaths(): string[];
4
+ protected isMissingTableError(error: unknown): boolean;
5
+ protected dbExists(dbPath?: string): boolean;
6
+ }
@@ -0,0 +1,9 @@
1
+ import type { BetterSQLite3Database } from 'drizzle-orm/better-sqlite3';
2
+ import type * as schema from '../schema/index.js';
3
+ export type DrizzleDb = BetterSQLite3Database<typeof schema>;
4
+ export interface DrizzleHandle {
5
+ db: DrizzleDb;
6
+ runRaw(sqlStr: string, params?: unknown[]): void;
7
+ close(): void;
8
+ }
9
+ export declare function openDrizzleDb(dbPath: string): DrizzleHandle;
@@ -0,0 +1,6 @@
1
+ export type RepositoryErrorCode = 'VALIDATION' | 'NOT_FOUND' | 'CONFLICT' | 'DB_ERROR';
2
+ export declare class RepositoryError extends Error {
3
+ readonly code: RepositoryErrorCode;
4
+ readonly cause?: unknown;
5
+ constructor(code: RepositoryErrorCode, message: string, cause?: unknown);
6
+ }
@@ -0,0 +1,21 @@
1
+ export { BaseSqliteRepository } from './base-sqlite-repository.js';
2
+ export { RepositoryError } from './errors.js';
3
+ export type { RepositoryErrorCode } from './errors.js';
4
+ export { WorkspaceRepository } from './workspace.repository.js';
5
+ export type { ProjectRow, ThreadRow as WorkspaceThreadRow, ThreadWithAgentRow } from './workspace.repository.js';
6
+ export { openDrizzleDb } from './db.js';
7
+ export type { DrizzleDb, DrizzleHandle } from './db.js';
8
+ export { runMigrations, runMigrationsOnce } from './migrate.js';
9
+ export type { Workspace, NewWorkspace } from '../schema/index.js';
10
+ export { TaskRepository } from './task.repository.js';
11
+ export type { TaskRow, NewTask, AgentUsageRow, TrendRow } from './task.repository.js';
12
+ export { ThreadRepository } from './thread.repository.js';
13
+ export type { ThreadRow, NewThread } from './thread.repository.js';
14
+ export { SpanRepository } from './span.repository.js';
15
+ export type { SpanRow, NewSpan } from './span.repository.js';
16
+ export { ToolCallRepository } from './tool-call.repository.js';
17
+ export type { ToolCallRow, NewToolCall } from './tool-call.repository.js';
18
+ export { ThreadBoxRepository } from './thread-box.repository.js';
19
+ export type { ThreadBoxRow, NewThreadBox } from './thread-box.repository.js';
20
+ export { RequestLogRepository } from './request-log.repository.js';
21
+ export type { RequestLogRow, NewRequestLog, RequestLogEntry } from './request-log.repository.js';
@@ -0,0 +1,103 @@
1
+ 'use strict';var fs=require('fs'),gt=require('path'),Ft=require('os'),drizzleOrm=require('drizzle-orm'),sqliteCore=require('drizzle-orm/sqlite-core'),crypto=require('crypto');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}function _interopNamespace(e){if(e&&e.__esModule)return e;var n=Object.create(null);if(e){Object.keys(e).forEach(function(k){if(k!=='default'){var d=Object.getOwnPropertyDescriptor(e,k);Object.defineProperty(n,k,d.get?d:{enumerable:true,get:function(){return e[k]}});}})}n.default=e;return Object.freeze(n)}var gt__namespace=/*#__PURE__*/_interopNamespace(gt);var Ft__default=/*#__PURE__*/_interopDefault(Ft);var ot=(l=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(l,{get:(t,e)=>(typeof require<"u"?require:t)[e]}):l)(function(l){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+l+'" is not supported')});var R=class{resolveDbPath(){return process.env.CREWX_DB?process.env.CREWX_DB:process.env.CREWX_TRACES_DB?process.env.CREWX_TRACES_DB:gt.join(Ft__default.default.homedir(),".crewx","crewx.db")}resolveDbPaths(){return [this.resolveDbPath()]}isMissingTableError(t){return t instanceof Error&&/no such table:/i.test(t.message)}dbExists(t){return fs.existsSync(t??this.resolveDbPath())}};var i=class extends Error{code;cause;constructor(t,e,n){super(e),this.name="RepositoryError",this.code=t,this.cause=n,Object.setPrototypeOf(this,new.target.prototype);}};function m(l){let t=ot("better-sqlite3"),{drizzle:e}=ot("drizzle-orm/better-sqlite3"),n=new t(l);return n.exec("PRAGMA journal_mode = WAL"),n.exec("PRAGMA busy_timeout = 5000"),n.exec("PRAGMA foreign_keys = ON"),{db:e(n),runRaw:(s,o=[])=>n.prepare(s).run(...o),close:()=>n.close()}}var kt=new Set;function Q(l){let{migrate:t}=ot("drizzle-orm/better-sqlite3/migrator"),e=gt__namespace.default.join(__dirname,"../migrations"),n=fs.existsSync(e)?e:gt__namespace.default.join(__dirname,"../../../../drizzle/migrations");t(l,{migrationsFolder:n});}function x(l,t){kt.has(t)||(Q(l),kt.add(t));}var p=sqliteCore.sqliteTable("workspaces",{id:sqliteCore.text("id").primaryKey(),slug:sqliteCore.text("slug").notNull().unique(),name:sqliteCore.text("name").notNull(),workspace_path:sqliteCore.text("workspace_path"),description:sqliteCore.text("description"),is_active:sqliteCore.integer("is_active").notNull().default(1),created_at:sqliteCore.text("created_at").notNull(),updated_at:sqliteCore.text("updated_at").notNull()});var r=sqliteCore.sqliteTable("tasks",{id:sqliteCore.text("id").primaryKey(),agent_id:sqliteCore.text("agent_id").notNull(),user_id:sqliteCore.text("user_id"),prompt:sqliteCore.text("prompt").notNull(),mode:sqliteCore.text("mode").notNull().default("execute"),status:sqliteCore.text("status").notNull().default("running"),result:sqliteCore.text("result"),error:sqliteCore.text("error"),started_at:sqliteCore.text("started_at").notNull(),completed_at:sqliteCore.text("completed_at"),duration_ms:sqliteCore.integer("duration_ms"),metadata:sqliteCore.text("metadata"),workspace_id:sqliteCore.text("workspace_id"),workspace_name:sqliteCore.text("workspace_name"),trace_id:sqliteCore.text("trace_id"),parent_task_id:sqliteCore.text("parent_task_id"),caller_agent_id:sqliteCore.text("caller_agent_id"),model:sqliteCore.text("model"),platform:sqliteCore.text("platform").default("cli"),crewx_version:sqliteCore.text("crewx_version"),input_tokens:sqliteCore.integer("input_tokens").default(0),output_tokens:sqliteCore.integer("output_tokens").default(0),cost_usd:sqliteCore.real("cost_usd").default(0),pid:sqliteCore.integer("pid"),rendered_prompt:sqliteCore.text("rendered_prompt"),command:sqliteCore.text("command"),coding_agent_command:sqliteCore.text("coding_agent_command"),exit_code:sqliteCore.integer("exit_code"),logs:sqliteCore.text("logs"),thread_id:sqliteCore.text("thread_id"),workspace_ref:sqliteCore.text("workspace_ref"),project_id:sqliteCore.text("project_id"),project_name:sqliteCore.text("project_name"),project_ref:sqliteCore.text("project_ref"),cached_input_tokens:sqliteCore.integer("cached_input_tokens").default(0)},l=>({idx_tasks_agent_id:sqliteCore.index("idx_tasks_agent_id").on(l.agent_id),idx_tasks_status:sqliteCore.index("idx_tasks_status").on(l.status),idx_tasks_started_at:sqliteCore.index("idx_tasks_started_at").on(l.started_at),idx_tasks_trace_id:sqliteCore.index("idx_tasks_trace_id").on(l.trace_id),idx_tasks_parent_task_id:sqliteCore.index("idx_tasks_parent_task_id").on(l.parent_task_id),idx_tasks_crewx_version:sqliteCore.index("idx_tasks_crewx_version").on(l.crewx_version),idx_tasks_pid:sqliteCore.index("idx_tasks_pid").on(l.pid),idx_tasks_thread_id:sqliteCore.index("idx_tasks_thread_id").on(l.thread_id),idx_tasks_workspace_id:sqliteCore.index("idx_tasks_workspace_id").on(l.workspace_id),idx_tasks_workspace_ref:sqliteCore.index("idx_tasks_workspace_ref").on(l.workspace_ref),idx_tasks_project_id:sqliteCore.index("idx_tasks_project_id").on(l.project_id),idx_tasks_ws_started:sqliteCore.index("idx_tasks_ws_started").on(l.workspace_id,l.started_at)}));var u=sqliteCore.sqliteTable("threads",{id:sqliteCore.text("id").primaryKey(),workspace_id:sqliteCore.text("workspace_id").references(()=>p.id,{onDelete:"set null"}),platform:sqliteCore.text("platform").notNull().default("cli"),title:sqliteCore.text("title"),first_message:sqliteCore.text("first_message"),last_message:sqliteCore.text("last_message"),message_count:sqliteCore.integer("message_count").notNull().default(0),created_at:sqliteCore.text("created_at").notNull(),updated_at:sqliteCore.text("updated_at").notNull(),metadata:sqliteCore.text("metadata"),title_locked:sqliteCore.integer("title_locked").notNull().default(0)},l=>({idx_threads_updated_at:sqliteCore.index("idx_threads_updated_at").on(l.updated_at),idx_threads_workspace_id:sqliteCore.index("idx_threads_workspace_id").on(l.workspace_id)}));var z=sqliteCore.sqliteTable("spans",{id:sqliteCore.text("id").primaryKey(),task_id:sqliteCore.text("task_id").references(()=>r.id,{onDelete:"set null"}),parent_span_id:sqliteCore.text("parent_span_id").references(()=>z.id,{onDelete:"set null"}),name:sqliteCore.text("name").notNull(),kind:sqliteCore.text("kind").notNull().default("internal"),status:sqliteCore.text("status").notNull().default("ok"),started_at:sqliteCore.text("started_at").notNull(),completed_at:sqliteCore.text("completed_at"),duration_ms:sqliteCore.integer("duration_ms"),input:sqliteCore.text("input"),output:sqliteCore.text("output"),error:sqliteCore.text("error"),attributes:sqliteCore.text("attributes")},l=>({idx_spans_task_id:sqliteCore.index("idx_spans_task_id").on(l.task_id),idx_spans_parent_span_id:sqliteCore.index("idx_spans_parent_span_id").on(l.parent_span_id)}));var B=sqliteCore.sqliteTable("tool_calls",{id:sqliteCore.text("id").primaryKey(),task_id:sqliteCore.text("task_id").notNull().references(()=>r.id,{onDelete:"cascade"}),session_id:sqliteCore.text("session_id"),tool_name:sqliteCore.text("tool_name").notNull(),files:sqliteCore.text("files"),input:sqliteCore.text("input"),output:sqliteCore.text("output"),duration_ms:sqliteCore.integer("duration_ms"),timestamp:sqliteCore.text("timestamp").notNull()},l=>({idx_tool_calls_task_id:sqliteCore.index("idx_tool_calls_task_id").on(l.task_id),idx_tool_calls_tool_name:sqliteCore.index("idx_tool_calls_tool_name").on(l.tool_name),idx_tool_calls_timestamp:sqliteCore.index("idx_tool_calls_timestamp").on(l.timestamp)}));var C=sqliteCore.sqliteTable("thread_boxes",{id:sqliteCore.text("id").primaryKey(),thread_id:sqliteCore.text("thread_id").notNull().references(()=>u.id,{onDelete:"cascade"}),seq:sqliteCore.integer("seq").notNull(),first_task_id:sqliteCore.text("first_task_id").notNull(),mid_task_id:sqliteCore.text("mid_task_id").notNull(),last_task_id:sqliteCore.text("last_task_id").notNull(),task_count:sqliteCore.integer("task_count").notNull(),summary:sqliteCore.text("summary"),source_tokens:sqliteCore.integer("source_tokens").notNull(),summary_tokens:sqliteCore.integer("summary_tokens"),created_at:sqliteCore.text("created_at").notNull()},l=>({idx_thread_boxes_thread_id:sqliteCore.index("idx_thread_boxes_thread_id").on(l.thread_id),idx_thread_boxes_seq:sqliteCore.index("idx_thread_boxes_seq").on(l.thread_id,l.seq),uniq_thread_boxes_thread_seq:sqliteCore.unique().on(l.thread_id,l.seq)}));var J=sqliteCore.sqliteTable("request_logs",{id:sqliteCore.text("id").primaryKey(),path:sqliteCore.text("path").notNull(),method:sqliteCore.text("method").notNull(),status_code:sqliteCore.integer("status_code").notNull(),duration_ms:sqliteCore.integer("duration_ms").notNull(),ip:sqliteCore.text("ip"),request_headers:sqliteCore.text("request_headers"),response_headers:sqliteCore.text("response_headers"),request_body:sqliteCore.text("request_body"),response_body:sqliteCore.text("response_body"),query:sqliteCore.text("query"),user_id:sqliteCore.text("user_id"),project_id:sqliteCore.text("project_id"),partition_key:sqliteCore.text("partition_key").notNull(),timestamp:sqliteCore.text("timestamp").notNull().default(drizzleOrm.sql`(datetime('now'))`),metadata:sqliteCore.text("metadata")},l=>({idx_request_logs_timestamp:sqliteCore.index("idx_request_logs_timestamp").on(l.timestamp),idx_request_logs_path:sqliteCore.index("idx_request_logs_path").on(l.path),idx_request_logs_status_code:sqliteCore.index("idx_request_logs_status_code").on(l.status_code),idx_request_logs_partition_key:sqliteCore.index("idx_request_logs_partition_key").on(l.partition_key)}));function it(l){let t=gt__namespace.resolve(l);return process.platform==="win32"&&(t=t.replace(/\\/g,"/"),t=t.replace(/^([A-Z]):/,(e,n)=>`${n.toLowerCase()}:`)),t.length>1&&!/^[a-zA-Z]:\/$/.test(t)&&(t=t.replace(/\/+$/,"")),t}function St(l){let t=it(l);return crypto.createHash("sha256").update(t).digest("hex")}var lt=class extends R{dbRoot;constructor(t={}){super(),this.dbRoot=t.dbRoot;}resolveDbPath(){return this.dbRoot?gt.join(this.dbRoot,".crewx","crewx.db"):super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{Q(n.db);}catch(s){throw n.close(),s}return n}resolveSlug(t,e,n){let s=gt.basename(n),o=`${gt.basename(gt.dirname(n))}/${s}`,a=[s,o];try{let d=c=>t.select({id:p.id}).from(p).where(drizzleOrm.and(drizzleOrm.eq(p.slug,c),drizzleOrm.ne(p.id,e))).limit(1).all().length>0;for(let c of a)if(!d(c))return c;for(let c=2;c<1e3;c+=1){let _=`${o}-${c}`;if(!d(_))return _}}catch{}return s}ensureRow(t,e){let{id:n,slug:s,name:o,workspacePath:a}=e,d=new Date().toISOString();t.insert(p).values({id:n,slug:s,name:o,workspace_path:a,is_active:1,created_at:d,updated_at:d}).onConflictDoNothing().run(),t.update(p).set({workspace_path:a,updated_at:d}).where(drizzleOrm.and(drizzleOrm.eq(p.id,n),drizzleOrm.isNull(p.workspace_path))).run();}registerWorkspace(t){let e=it(t),n=this.openHandle(true);try{let s=St(e),o=gt.basename(e),a=this.resolveSlug(n.db,s,e);return this.ensureRow(n.db,{id:s,slug:a,name:o,workspacePath:e}),{id:s,slug:a}}catch(s){throw s instanceof i?s:new i("DB_ERROR","Failed to register workspace",s)}finally{n.close();}}listProjects(t){if(!this.dbExists())return {rows:[],total:0};let e=this.openHandle(false);try{let n=t.isActive!==void 0?drizzleOrm.eq(p.is_active,t.isActive?1:0):void 0,s=e.db.select({count:drizzleOrm.sql`count(*)`}).from(p).where(n).get();return {rows:e.db.select().from(p).where(n).orderBy(drizzleOrm.desc(p.updated_at)).limit(t.limit).offset(t.offset).all(),total:s?.count??0}}catch(n){throw new i("DB_ERROR","Failed to list projects",n)}finally{e.close();}}findById(t){if(!this.dbExists())return;let e=this.openHandle(false);try{return e.db.select().from(p).where(drizzleOrm.eq(p.id,t)).limit(1).get()??void 0}catch(n){throw new i("DB_ERROR","Failed to find workspace",n)}finally{e.close();}}findAgentsByProject(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.all(drizzleOrm.sql`SELECT DISTINCT agent_id FROM tasks WHERE workspace_id = ${t} AND agent_id IS NOT NULL ORDER BY agent_id`).map(s=>s.agent_id)}catch(n){throw new i("DB_ERROR","Failed to find agents by project",n)}finally{e.close();}}findThreadsByProject(t,e){if(!this.dbExists())return {rows:[],total:0};let n=this.openHandle(false);try{let s=n.db.get(drizzleOrm.sql`SELECT COUNT(*) as count FROM threads WHERE workspace_id = ${t}`);return {rows:n.db.all(drizzleOrm.sql`SELECT t.*,
2
+ (SELECT agent_id FROM tasks tk WHERE tk.thread_id = t.id AND tk.agent_id IS NOT NULL ORDER BY tk.started_at ASC LIMIT 1) AS agent_id
3
+ FROM threads t
4
+ WHERE t.workspace_id = ${t}
5
+ ORDER BY t.updated_at DESC
6
+ LIMIT ${e.limit} OFFSET ${e.offset}`),total:s?.count??0}}catch(s){throw new i("DB_ERROR","Failed to find threads by project",s)}finally{n.close();}}findBySlug(t){if(!this.dbExists())return;let e=this.openHandle(false);try{return e.db.select().from(p).where(drizzleOrm.eq(p.slug,t)).limit(1).get()??void 0}catch(n){throw new i("DB_ERROR","Failed to find workspace by slug",n)}finally{e.close();}}slugExists(t,e){if(!this.dbExists())return false;let n=this.openHandle(false);try{let s=e?drizzleOrm.and(drizzleOrm.eq(p.slug,t),drizzleOrm.ne(p.id,e)):drizzleOrm.eq(p.slug,t);return !!n.db.select({id:p.id}).from(p).where(s).limit(1).get()}catch(s){throw new i("DB_ERROR","Failed to check slug",s)}finally{n.close();}}insert(t,e,n,s){let o=this.openHandle(true);try{let a=new Date().toISOString();o.db.insert(p).values({id:t,slug:e,name:n,workspace_path:s,is_active:1,created_at:a,updated_at:a}).run();let d=o.db.select().from(p).where(drizzleOrm.eq(p.id,t)).limit(1).get();if(!d)throw new i("DB_ERROR","Insert did not return a row");return d}catch(a){throw a instanceof i?a:new i("DB_ERROR","Failed to insert workspace",a)}finally{o.close();}}update(t,e,n){let s=this.openHandle(true);try{s.runRaw(`UPDATE workspaces SET ${e.join(", ")} WHERE id = ?`,n);let o=s.db.select().from(p).where(drizzleOrm.eq(p.id,t)).limit(1).get();if(!o)throw new i("NOT_FOUND",`Workspace ${t} not found`);return o}catch(o){throw o instanceof i?o:new i("DB_ERROR","Failed to update workspace",o)}finally{s.close();}}cleanupOrphanWorkspaces(){if(!this.dbExists())return {softDeleted:0,checked:0};let t=this.openHandle(true);try{let e=t.db.select({id:p.id,workspace_path:p.workspace_path}).from(p).where(drizzleOrm.and(drizzleOrm.eq(p.is_active,1),drizzleOrm.isNotNull(p.workspace_path))).all(),n=0;for(let s of e){let o=s.workspace_path;fs.existsSync(gt.join(o,"crewx.yaml"))||fs.existsSync(gt.join(o,"crewx.yml"))||(t.db.update(p).set({is_active:0,updated_at:new Date().toISOString()}).where(drizzleOrm.eq(p.id,s.id)).run(),n+=1);}return {softDeleted:n,checked:e.length}}catch(e){throw new i("DB_ERROR","Failed to cleanup orphan workspaces",e)}finally{t.close();}}delete(t){let e=this.openHandle(true);try{e.db.run(drizzleOrm.sql`UPDATE threads SET workspace_id = NULL WHERE workspace_id = ${t}`),e.db.delete(p).where(drizzleOrm.eq(p.id,t)).run();}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to delete workspace",n)}finally{e.close();}}};var pt=class extends R{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=gt.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{x(n.db,e);}catch(s){throw n.close(),s}return n}startTask(t){let e=this.openHandle(true);try{e.db.insert(r).values({id:t.id,agent_id:t.agentId,prompt:t.prompt,mode:t.mode,status:t.status,started_at:t.startedAt,pid:t.pid??null,parent_task_id:t.parentTaskId??null,caller_agent_id:t.callerAgentId??null,trace_id:t.traceId??null,command:t.command??null,metadata:t.metadata??null,workspace_id:t.workspaceId??null,workspace_name:t.workspaceName??null,platform:t.platform??"cli",crewx_version:t.crewxVersion??null,thread_id:t.threadId??null,model:t.model??null,rendered_prompt:t.renderedPrompt??null,coding_agent_command:t.codingAgentCommand??null}).onConflictDoNothing().run();}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to start task",n)}finally{e.close();}}finishTask(t){let e=this.openHandle(true);try{e.runRaw(`UPDATE tasks SET status=?, result=?, error=?, completed_at=?, duration_ms=?,
7
+ exit_code=?, input_tokens=?, output_tokens=?, cached_input_tokens=?, cost_usd=?,
8
+ model=COALESCE(?, model) WHERE id=?`,[t.status,t.result??null,t.error??null,t.completedAt,t.durationMs??null,t.exitCode??null,t.inputTokens??0,t.outputTokens??0,t.cachedInputTokens??0,t.costUsd??0,t.model??null,t.id]);}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to finish task",n)}finally{e.close();}}appendLog(t,e){let n=this.openHandle(true);try{n.db.transaction(s=>{let o=s.select({logs:r.logs}).from(r).where(drizzleOrm.eq(r.id,t)).limit(1).get(),a=o?.logs?JSON.parse(o.logs):[];a.push(e),s.update(r).set({logs:JSON.stringify(a)}).where(drizzleOrm.eq(r.id,t)).run();},{behavior:"immediate"});}catch(s){throw s instanceof i?s:new i("DB_ERROR","Failed to append log",s)}finally{n.close();}}getRunningTasks(){if(!this.dbExists())return [];let t=this.openHandle(false);try{return t.db.select().from(r).where(drizzleOrm.eq(r.status,"running")).orderBy(drizzleOrm.desc(r.started_at)).all()}catch(e){throw new i("DB_ERROR","Failed to get running tasks",e)}finally{t.close();}}getAllTasks(){if(!this.dbExists())return [];let t=this.openHandle(false);try{return t.db.select().from(r).orderBy(drizzleOrm.desc(r.started_at)).limit(100).all()}catch(e){throw new i("DB_ERROR","Failed to get all tasks",e)}finally{t.close();}}getTask(t){if(!this.dbExists())return;let e=this.openHandle(false);try{return e.db.select().from(r).where(drizzleOrm.eq(r.id,t)).limit(1).get()??void 0}catch(n){throw new i("DB_ERROR","Failed to get task",n)}finally{e.close();}}killTask(t){if(!this.dbExists())return {killed:false};let e=this.openHandle(true);try{let n=e.db.select({id:r.id,status:r.status,pid:r.pid}).from(r).where(drizzleOrm.eq(r.id,t)).limit(1).get();if(!n||n.status!=="running")return {killed:!1};if(n.pid)try{process.kill(n.pid,"SIGTERM");}catch{}return e.db.update(r).set({status:"failed",error:"Killed by user",completed_at:new Date().toISOString()}).where(drizzleOrm.and(drizzleOrm.eq(r.id,t),drizzleOrm.eq(r.status,"running"))).run(),{killed:!0,pid:n.pid??void 0}}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to kill task",n)}finally{e.close();}}findTaskStatus(t,e){let n=this.resolveDbPaths();for(let s of n){if(!fs.existsSync(s))continue;let o=m(s);try{let a=e?drizzleOrm.eq(r.workspace_id,e):void 0,d=a?drizzleOrm.and(drizzleOrm.eq(r.id,t),a):drizzleOrm.eq(r.id,t),c=o.db.select().from(r).where(d).limit(1).get()??void 0;if(!c){let _=drizzleOrm.or(drizzleOrm.eq(r.thread_id,t),drizzleOrm.and(drizzleOrm.isNull(r.thread_id),drizzleOrm.like(r.command,`%--thread=${t}%`))),h=a?drizzleOrm.and(_,a):_;c=o.db.select().from(r).where(h).orderBy(drizzleOrm.desc(r.started_at)).limit(1).get()??void 0;}if(c)return c}catch(a){throw new i("DB_ERROR","Failed to find task status",a)}finally{o.close();}}}findChildTasks(t,e){let n=this.resolveDbPaths(),s=new Set,o=[];for(let a of n){if(!fs.existsSync(a))continue;let d=m(a);try{let c=e?drizzleOrm.and(drizzleOrm.eq(r.parent_task_id,t),drizzleOrm.eq(r.workspace_id,e)):drizzleOrm.eq(r.parent_task_id,t),_=d.db.select().from(r).where(c).orderBy(drizzleOrm.asc(r.started_at)).all();for(let h of _)s.has(h.id)||(s.add(h.id),o.push(h));}catch(c){throw new i("DB_ERROR","Failed to find child tasks",c)}finally{d.close();}}return o}getWorkspaceUsageSummary(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.all(t?drizzleOrm.sql`
9
+ SELECT
10
+ COALESCE(workspace_id, 'unknown') AS workspace_id,
11
+ COALESCE(workspace_name, 'Unknown Workspace') AS workspace_name,
12
+ COALESCE(SUM(input_tokens), 0) AS input_tokens,
13
+ COALESCE(SUM(output_tokens), 0) AS output_tokens,
14
+ COALESCE(SUM(cost_usd), 0) AS cost_usd,
15
+ COUNT(*) AS task_count
16
+ FROM tasks
17
+ WHERE workspace_id = ${t}
18
+ GROUP BY workspace_id, workspace_name
19
+ ORDER BY (COALESCE(SUM(input_tokens), 0) + COALESCE(SUM(output_tokens), 0)) DESC
20
+ `:drizzleOrm.sql`
21
+ SELECT
22
+ COALESCE(workspace_id, 'unknown') AS workspace_id,
23
+ COALESCE(workspace_name, 'Unknown Workspace') AS workspace_name,
24
+ COALESCE(SUM(input_tokens), 0) AS input_tokens,
25
+ COALESCE(SUM(output_tokens), 0) AS output_tokens,
26
+ COALESCE(SUM(cost_usd), 0) AS cost_usd,
27
+ COUNT(*) AS task_count
28
+ FROM tasks
29
+ GROUP BY workspace_id, workspace_name
30
+ ORDER BY (COALESCE(SUM(input_tokens), 0) + COALESCE(SUM(output_tokens), 0)) DESC
31
+ `)}catch(n){throw new i("DB_ERROR","Failed to get workspace usage summary",n)}finally{e.close();}}getThreadTokenUsage(t,e){let n=this.resolveDbPaths(),s=new Set,o=0,a=0,d=0;for(let c of n){if(!fs.existsSync(c))continue;let _=m(c);try{let h=drizzleOrm.or(drizzleOrm.eq(r.thread_id,t),drizzleOrm.and(drizzleOrm.isNull(r.thread_id),drizzleOrm.like(r.command,`%--thread=${t}%`))),k=e?drizzleOrm.and(h,drizzleOrm.eq(r.workspace_id,e)):h,$=_.db.select({id:r.id,input_tokens:r.input_tokens,output_tokens:r.output_tokens,cost_usd:r.cost_usd}).from(r).where(k).all();for(let L of $)s.has(L.id)||(s.add(L.id),o+=L.input_tokens??0,a+=L.output_tokens??0,d+=L.cost_usd??0);}catch(h){throw new i("DB_ERROR","Failed to get thread token usage",h)}finally{_.close();}}return {inputTokens:o,outputTokens:a,costUsd:d}}findTasksByThread(t,e){let n=this.resolveDbPaths(),s=new Set,o=[];for(let a of n){if(!fs.existsSync(a))continue;let d=m(a);try{let c=drizzleOrm.or(drizzleOrm.eq(r.thread_id,t),drizzleOrm.and(drizzleOrm.isNull(r.thread_id),drizzleOrm.like(r.command,`%--thread=${t}%`))),_=e?drizzleOrm.and(c,drizzleOrm.eq(r.workspace_id,e)):c,h=d.db.select().from(r).where(_).orderBy(drizzleOrm.asc(r.started_at)).all();for(let k of h)s.has(k.id)||(s.add(k.id),o.push(k));}catch(c){throw new i("DB_ERROR","Failed to find tasks by thread",c)}finally{d.close();}}return o}findAllTasks(t){if(!this.dbExists())return {rows:[],total:0};let e=this.openHandle(false);try{let n=[];t.workspaceId&&n.push(drizzleOrm.eq(r.workspace_id,t.workspaceId));let s=t.agents&&t.agents.length>0?t.agents:t.agentId?[t.agentId]:null;s&&n.push(drizzleOrm.inArray(r.agent_id,s));let o=t.statuses&&t.statuses.length>0?t.statuses:t.status?[t.status]:null;o&&n.push(drizzleOrm.inArray(r.status,o));let a=t.q??t.search;a&&n.push(drizzleOrm.like(r.prompt,`%${a}%`)),t.from&&n.push(drizzleOrm.gte(r.started_at,t.from)),t.to&&n.push(drizzleOrm.lt(r.started_at,t.to));let d=n.length>0?drizzleOrm.and(...n):void 0,c=e.db.select({count:drizzleOrm.sql`count(*)`}).from(r).where(d).get(),_=(t.sortDir??"DESC")==="ASC"?drizzleOrm.asc(r.started_at):drizzleOrm.desc(r.started_at);return {rows:e.db.select().from(r).where(d).orderBy(_).limit(t.limit).offset(t.offset).all(),total:c?.count??0}}catch(n){throw new i("DB_ERROR","Failed to find all tasks",n)}finally{e.close();}}getAgentUsage(t,e,n){if(!this.dbExists())return [];let s=this.openHandle(false);try{return s.db.all(n?drizzleOrm.sql`
32
+ SELECT
33
+ t.agent_id,
34
+ COUNT(*) AS total_tasks,
35
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
36
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
37
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
38
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
39
+ FROM tasks t
40
+ WHERE t.status IN ('completed', 'success')
41
+ AND t.started_at >= ${t}
42
+ AND t.started_at < ${e}
43
+ AND t.workspace_name = ${n}
44
+ GROUP BY t.agent_id
45
+ ORDER BY (COALESCE(SUM(t.input_tokens), 0) + COALESCE(SUM(t.output_tokens), 0)) DESC
46
+ `:drizzleOrm.sql`
47
+ SELECT
48
+ t.agent_id,
49
+ COUNT(*) AS total_tasks,
50
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
51
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
52
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
53
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
54
+ FROM tasks t
55
+ WHERE t.status IN ('completed', 'success')
56
+ AND t.started_at >= ${t}
57
+ AND t.started_at < ${e}
58
+ GROUP BY t.agent_id
59
+ ORDER BY (COALESCE(SUM(t.input_tokens), 0) + COALESCE(SUM(t.output_tokens), 0)) DESC
60
+ `).map(a=>({agentId:a.agent_id,totalTasks:a.total_tasks,inputTokens:a.input_tokens,outputTokens:a.output_tokens,cachedInputTokens:a.cached_input_tokens,costUsd:a.cost_usd}))}catch(o){throw new i("DB_ERROR","Failed to get agent usage",o)}finally{s.close();}}getAgentUsageTrendRaw(t,e,n){if(!this.dbExists())return [];let s=this.openHandle(false);try{return s.db.all(n?drizzleOrm.sql`
61
+ SELECT
62
+ date(t.started_at) AS date,
63
+ t.agent_id,
64
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
65
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
66
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
67
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
68
+ FROM tasks t
69
+ WHERE t.status IN ('completed', 'success')
70
+ AND t.started_at >= ${t}
71
+ AND t.started_at < ${e}
72
+ AND t.workspace_name = ${n}
73
+ GROUP BY date(t.started_at), t.agent_id
74
+ ORDER BY date(t.started_at) ASC
75
+ `:drizzleOrm.sql`
76
+ SELECT
77
+ date(t.started_at) AS date,
78
+ t.agent_id,
79
+ COALESCE(SUM(t.input_tokens), 0) AS input_tokens,
80
+ COALESCE(SUM(t.output_tokens), 0) AS output_tokens,
81
+ COALESCE(SUM(t.cached_input_tokens), 0) AS cached_input_tokens,
82
+ COALESCE(SUM(t.cost_usd), 0) AS cost_usd
83
+ FROM tasks t
84
+ WHERE t.status IN ('completed', 'success')
85
+ AND t.started_at >= ${t}
86
+ AND t.started_at < ${e}
87
+ GROUP BY date(t.started_at), t.agent_id
88
+ ORDER BY date(t.started_at) ASC
89
+ `).map(a=>({date:a.date,agentId:a.agent_id,inputTokens:a.input_tokens,outputTokens:a.output_tokens,cachedInputTokens:a.cached_input_tokens,costUsd:a.cost_usd}))}catch(o){throw new i("DB_ERROR","Failed to get agent usage trend",o)}finally{s.close();}}findTaskForStop(t,e){if(!this.dbExists())return;let n=this.openHandle(false);try{return n.db.select().from(r).where(drizzleOrm.and(drizzleOrm.eq(r.id,t),drizzleOrm.eq(r.workspace_id,e))).limit(1).get()??void 0}catch(s){throw new i("DB_ERROR","Failed to find task for stop",s)}finally{n.close();}}markTaskFailed(t,e,n){if(!this.dbExists())return;let s=this.openHandle(true);try{let o=new Date().toISOString(),a=n?drizzleOrm.and(drizzleOrm.eq(r.id,t),drizzleOrm.eq(r.status,"running"),drizzleOrm.eq(r.workspace_id,n)):drizzleOrm.and(drizzleOrm.eq(r.id,t),drizzleOrm.eq(r.status,"running"));s.db.update(r).set({status:"failed",error:e,completed_at:o}).where(a).run();}catch(o){throw o instanceof i?o:new i("DB_ERROR","Failed to mark task failed",o)}finally{s.close();}}findTasksByPromptHint(t,e){let n=this.resolveDbPaths(),s=new Set,o=[];for(let a of n){if(!fs.existsSync(a))continue;let d=m(a);try{let c=e?drizzleOrm.and(drizzleOrm.like(r.prompt,`%${t}%`),drizzleOrm.eq(r.workspace_id,e)):drizzleOrm.like(r.prompt,`%${t}%`),_=d.db.select().from(r).where(c).orderBy(drizzleOrm.asc(r.started_at)).all();for(let h of _)s.has(h.id)||(s.add(h.id),o.push(h));}catch(c){throw new i("DB_ERROR","Failed to find tasks by prompt hint",c)}finally{d.close();}}return o}};var _t=class extends R{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=gt.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{x(n.db,e);}catch(s){throw n.close(),s}return n}validateWorkspaceId(t,e){return t.db.select({id:p.id}).from(p).where(drizzleOrm.eq(p.id,e)).limit(1).get()?e:null}findAllThreads(t){let e=this.resolveDbPaths(),n=new Set,s=[];for(let o of e){if(!fs.existsSync(o))continue;let a=m(o);try{let d=t?drizzleOrm.or(drizzleOrm.eq(u.workspace_id,t),drizzleOrm.isNull(u.workspace_id)):void 0,c=a.db.select().from(u).where(d).orderBy(drizzleOrm.desc(u.updated_at)).all();for(let _ of c)n.has(_.id)||(n.add(_.id),s.push(_));}catch(d){throw new i("DB_ERROR","Failed to find all threads",d)}finally{a.close();}}return s}findThreadById(t,e){let n=this.resolveDbPaths();for(let s of n){if(!fs.existsSync(s))continue;let o=m(s);try{let a=drizzleOrm.eq(u.id,t),d=e?drizzleOrm.and(a,drizzleOrm.or(drizzleOrm.eq(u.workspace_id,e),drizzleOrm.isNull(u.workspace_id))):a,c=o.db.select().from(u).where(d).limit(1).get()??void 0;if(c)return c}catch(a){throw new i("DB_ERROR","Failed to find thread by id",a)}finally{o.close();}}}threadExists(t,e){let n=this.resolveDbPaths();for(let s of n){if(!fs.existsSync(s))continue;let o=m(s);try{let a=drizzleOrm.eq(u.id,t),d=e?drizzleOrm.and(a,drizzleOrm.or(drizzleOrm.eq(u.workspace_id,e),drizzleOrm.isNull(u.workspace_id))):a;if(o.db.select({id:u.id}).from(u).where(d).limit(1).get())return !0}catch(a){throw new i("DB_ERROR","Failed to check thread existence",a)}finally{o.close();}}return false}aggregateTaskStats(t,e){let n=this.resolveDbPaths(),s=0,o=0,a=0,d=0,c=0,_=new Set;for(let h of n){if(!fs.existsSync(h))continue;let k=m(h);try{let $=drizzleOrm.and(drizzleOrm.eq(r.thread_id,t),drizzleOrm.or(drizzleOrm.isNull(r.parent_task_id),drizzleOrm.eq(r.parent_task_id,""))),L=e?drizzleOrm.and($,drizzleOrm.eq(r.workspace_id,e)):$,j=k.db.select({cnt:drizzleOrm.sql`count(*)`,total_input:drizzleOrm.sql`COALESCE(SUM(input_tokens), 0)`,total_output:drizzleOrm.sql`COALESCE(SUM(output_tokens), 0)`,total_cached:drizzleOrm.sql`COALESCE(SUM(cached_input_tokens), 0)`,total_cost:drizzleOrm.sql`COALESCE(SUM(cost_usd), 0)`}).from(r).where(L).get();j&&(s+=j.cnt,o+=j.total_input,a+=j.total_output,d+=j.total_cached,c+=j.total_cost);let qt=k.db.all(drizzleOrm.sql`
90
+ SELECT DISTINCT agent_id FROM tasks
91
+ WHERE thread_id = ${t}
92
+ AND agent_id IS NOT NULL AND agent_id != ''
93
+ AND (parent_task_id IS NULL OR parent_task_id = '')
94
+ ${e?drizzleOrm.sql`AND workspace_id = ${e}`:drizzleOrm.sql``}
95
+ `);for(let zt of qt)_.add(zt.agent_id);}catch($){throw new i("DB_ERROR","Failed to aggregate task stats",$)}finally{k.close();}}return {taskCount:s,inputTokens:o,outputTokens:a,cachedInputTokens:d,costUsd:c,agentIds:Array.from(_)}}findTopLevelTasks(t,e){let n=this.resolveDbPaths(),s=new Set,o=[];for(let a of n){if(!fs.existsSync(a))continue;let d=m(a);try{let c=drizzleOrm.and(drizzleOrm.eq(r.thread_id,t),drizzleOrm.or(drizzleOrm.isNull(r.parent_task_id),drizzleOrm.eq(r.parent_task_id,""))),_=e?drizzleOrm.and(c,drizzleOrm.eq(r.workspace_id,e)):c,h=d.db.select().from(r).where(_).orderBy(drizzleOrm.asc(r.started_at)).all();for(let k of h)s.has(k.id)||(s.add(k.id),o.push(k));}catch(c){throw new i("DB_ERROR","Failed to find top-level tasks",c)}finally{d.close();}}return o}findAllTasks(t,e){let n=this.resolveDbPaths(),s=new Set,o=[];for(let a of n){if(!fs.existsSync(a))continue;let d=m(a);try{let c=drizzleOrm.eq(r.thread_id,t),_=e?drizzleOrm.and(c,drizzleOrm.eq(r.workspace_id,e)):c,h=d.db.select().from(r).where(_).orderBy(drizzleOrm.asc(r.started_at)).all();for(let k of h)s.has(k.id)||(s.add(k.id),o.push(k));}catch(c){throw new i("DB_ERROR","Failed to find all tasks for thread",c)}finally{d.close();}}return o}findTaskById(t,e,n){let s=this.resolveDbPaths();for(let o of s){if(!fs.existsSync(o))continue;let a=m(o);try{let d=drizzleOrm.and(drizzleOrm.eq(r.id,e),drizzleOrm.eq(r.thread_id,t)),c=n?drizzleOrm.and(d,drizzleOrm.eq(r.workspace_id,n)):d,_=a.db.select().from(r).where(c).limit(1).get();if(!_)continue;let h=a.db.select().from(r).where(drizzleOrm.eq(r.parent_task_id,_.id)).orderBy(drizzleOrm.asc(r.started_at)).all();return {task:_,children:h}}catch(d){throw new i("DB_ERROR","Failed to find task by id",d)}finally{a.close();}}}batchFetchTasks(t,e){let n=new Map;if(t.length===0)return n;let s=this.resolveDbPaths();for(let o of s){if(!fs.existsSync(o))continue;let a=m(o);try{let d=drizzleOrm.and(drizzleOrm.inArray(r.thread_id,t),drizzleOrm.or(drizzleOrm.isNull(r.parent_task_id),drizzleOrm.eq(r.parent_task_id,""))),c=e?drizzleOrm.and(d,drizzleOrm.eq(r.workspace_id,e)):d,_=a.db.select().from(r).where(c).orderBy(drizzleOrm.asc(r.started_at)).all();for(let h of _){let k=h.thread_id;n.has(k)||n.set(k,[]),n.get(k).push(h);}}catch(d){throw new i("DB_ERROR","Failed to batch fetch tasks",d)}finally{a.close();}}return n}updateThreadTitle(t,e,n){if(!this.dbExists())return;let s=this.openHandle(true);try{let o=drizzleOrm.eq(u.id,t),a=n?drizzleOrm.and(o,drizzleOrm.or(drizzleOrm.eq(u.workspace_id,n),drizzleOrm.isNull(u.workspace_id))):o;if(!s.db.select({id:u.id}).from(u).where(a).limit(1).get())return;s.db.update(u).set({title:e,title_locked:1,updated_at:new Date().toISOString()}).where(drizzleOrm.eq(u.id,t)).run();}catch(o){throw o instanceof i?o:new i("DB_ERROR","Failed to update thread title",o)}finally{s.close();}}upsertThread(t,e){let n=this.openHandle(true);try{let s=e.workspaceId?this.validateWorkspaceId(n,e.workspaceId):null,o=new Date().toISOString();if(n.db.select({id:u.id,message_count:u.message_count}).from(u).where(drizzleOrm.eq(u.id,t)).limit(1).get()){let d={updated_at:o};e.title!==void 0&&(d.title=e.title),e.titleLocked!==void 0&&(d.title_locked=e.titleLocked?1:0),n.db.update(u).set(d).where(drizzleOrm.eq(u.id,t)).run();}else n.db.insert(u).values({id:t,platform:e.platform,workspace_id:s,title:e.title??null,title_locked:e.titleLocked?1:0,message_count:0,created_at:o,updated_at:o}).run();}catch(s){throw s instanceof i?s:new i("DB_ERROR","Failed to upsert thread",s)}finally{n.close();}}ensureThread(t,e,n){let s=this.openHandle(true);try{let o=n?this.validateWorkspaceId(s,n):null,a=s.db.select({id:u.id,platform:u.platform,workspace_id:u.workspace_id}).from(u).where(drizzleOrm.eq(u.id,t)).limit(1).get();if(a){o&&!a.workspace_id&&s.db.update(u).set({workspace_id:o}).where(drizzleOrm.eq(u.id,t)).run();return}let d=new Date().toISOString();s.db.insert(u).values({id:t,platform:e,workspace_id:o,message_count:0,created_at:d,updated_at:d}).run();}catch(o){throw o instanceof i?o:new i("DB_ERROR","Failed to ensure thread",o)}finally{s.close();}}saveUserMessage(t,e,n){if(!this.dbExists())return;let s=this.openHandle(true);try{let o=new Date().toISOString();s.db.run(drizzleOrm.sql`
96
+ UPDATE threads
97
+ SET first_message = COALESCE(first_message, ${e}),
98
+ title = CASE WHEN title_locked = 0 AND title IS NULL THEN substr(${e}, 1, 60) ELSE title END,
99
+ last_message = ${e},
100
+ message_count = message_count + 1,
101
+ updated_at = ${o}
102
+ WHERE id = ${t}
103
+ `);}catch(o){throw o instanceof i?o:new i("DB_ERROR","Failed to save user message",o)}finally{s.close();}}saveAssistantMessage(t,e,n){if(!this.dbExists())return;let s=this.openHandle(true);try{let o=new Date().toISOString();s.db.update(u).set({last_message:e,updated_at:o}).where(drizzleOrm.eq(u.id,t)).run();}catch(o){throw o instanceof i?o:new i("DB_ERROR","Failed to save assistant message",o)}finally{s.close();}}updateThread(t,e){if(!this.dbExists())return;let n=this.openHandle(true);try{let s={updated_at:new Date().toISOString()};e.title!==void 0&&(s.title=e.title,s.title_locked=1),e.titleLocked!==void 0&&(s.title_locked=e.titleLocked?1:0),n.db.update(u).set(s).where(drizzleOrm.eq(u.id,t)).run();}catch(s){throw s instanceof i?s:new i("DB_ERROR","Failed to update thread",s)}finally{n.close();}}};var ht=class extends R{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=gt.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{x(n.db,e);}catch(s){throw n.close(),s}return n}insertSpan(t){let e=this.openHandle(true);try{e.db.insert(z).values(t).run();}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to insert span",n)}finally{e.close();}}findByTaskId(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.select().from(z).where(drizzleOrm.eq(z.task_id,t)).all()}catch(n){throw new i("DB_ERROR","Failed to find spans by task id",n)}finally{e.close();}}findById(t){if(!this.dbExists())return;let e=this.openHandle(false);try{return e.db.select().from(z).where(drizzleOrm.eq(z.id,t)).limit(1).get()??void 0}catch(n){throw new i("DB_ERROR","Failed to find span by id",n)}finally{e.close();}}};var ft=class extends R{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=gt.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{x(n.db,e);}catch(s){throw n.close(),s}return n}insertToolCall(t){let e=this.openHandle(true);try{e.db.insert(B).values(t).run();}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to insert tool call",n)}finally{e.close();}}findByTaskId(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.select().from(B).where(drizzleOrm.eq(B.task_id,t)).all()}catch(n){throw new i("DB_ERROR","Failed to find tool calls by task id",n)}finally{e.close();}}aggregateByName(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.select({toolName:B.tool_name,count:drizzleOrm.sql`count(*)`}).from(B).where(drizzleOrm.eq(B.task_id,t)).groupBy(B.tool_name).all().map(n=>({toolName:n.toolName,count:n.count}))}catch(n){throw new i("DB_ERROR","Failed to aggregate tool calls by name",n)}finally{e.close();}}};var mt=class extends R{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=gt.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{x(n.db,e);}catch(s){throw n.close(),s}return n}ensureThreadExists(t,e){if(!t.db.select({id:u.id}).from(u).where(drizzleOrm.eq(u.id,e)).limit(1).get())throw new i("NOT_FOUND",`Thread not found: ${e}`)}findByThreadId(t){if(!this.dbExists())return [];let e=this.openHandle(false);try{return this.ensureThreadExists(e,t),e.db.select().from(C).where(drizzleOrm.eq(C.thread_id,t)).orderBy(drizzleOrm.asc(C.seq)).all()}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to find boxes by thread id",n)}finally{e.close();}}findById(t,e){if(!this.dbExists())return;let n=this.openHandle(false);try{return n.db.select().from(C).where(drizzleOrm.and(drizzleOrm.eq(C.id,e),drizzleOrm.eq(C.thread_id,t))).limit(1).get()??void 0}catch(s){throw s instanceof i?s:new i("DB_ERROR","Failed to find box by id",s)}finally{n.close();}}insert(t){let e=this.openHandle(true);try{this.ensureThreadExists(e,t.threadId);try{e.db.insert(C).values({id:t.id,thread_id:t.threadId,seq:t.seq,first_task_id:t.first_task_id,mid_task_id:t.mid_task_id,last_task_id:t.last_task_id,task_count:t.task_count,summary:t.summary??null,source_tokens:t.source_tokens,summary_tokens:t.summary_tokens??null,created_at:t.created_at}).run();}catch(s){throw s instanceof Error&&/UNIQUE constraint failed/i.test(s.message)?new i("CONFLICT",`Duplicate seq ${String(t.seq)} for thread ${t.threadId}`,s):s}let n=e.db.select().from(C).where(drizzleOrm.eq(C.id,t.id)).limit(1).get();if(!n)throw new i("DB_ERROR","Insert did not return a row");return n}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to insert thread box",n)}finally{e.close();}}};var wt=class extends R{dbPath;constructor(t={}){super(),t.dbPath?this.dbPath=t.dbPath:t.dbRoot&&(this.dbPath=gt.join(t.dbRoot,".crewx","crewx.db"));}resolveDbPath(){return this.dbPath?this.dbPath:super.resolveDbPath()}openHandle(t){let e=this.resolveDbPath();if(t){let s=gt.dirname(e);fs.existsSync(s)||fs.mkdirSync(s,{recursive:true});}else if(!fs.existsSync(e))throw new i("NOT_FOUND","Database not found");let n=m(e);if(t)try{x(n.db,e);}catch(s){throw n.close(),s}return n}bulkInsert(t){if(t.length===0)return;let e=this.openHandle(true);try{let n=new Date().toISOString();e.db.transaction(s=>{for(let o of t)s.insert(J).values({id:crypto.randomUUID(),path:o.path,method:o.method,status_code:o.statusCode,duration_ms:o.durationMs,ip:o.ip??null,request_headers:o.requestHeaders??null,response_headers:o.responseHeaders??null,request_body:o.requestBody??null,response_body:o.responseBody??null,query:o.query??null,user_id:o.userId??null,project_id:o.projectId??null,partition_key:o.partitionKey??"default",timestamp:n,metadata:o.metadata??null}).run();});}catch(n){throw n instanceof i?n:new i("DB_ERROR","Failed to bulk insert request logs",n)}finally{e.close();}}findRecent(t=100){if(!this.dbExists())return [];let e=this.openHandle(false);try{return e.db.select().from(J).orderBy(drizzleOrm.desc(J.timestamp)).limit(t).all()}catch(n){throw new i("DB_ERROR","Failed to find recent request logs",n)}finally{e.close();}}};exports.BaseSqliteRepository=R;exports.RepositoryError=i;exports.RequestLogRepository=wt;exports.SpanRepository=ht;exports.TaskRepository=pt;exports.ThreadBoxRepository=mt;exports.ThreadRepository=_t;exports.ToolCallRepository=ft;exports.WorkspaceRepository=lt;exports.openDrizzleDb=m;exports.runMigrations=Q;exports.runMigrationsOnce=x;
@@ -0,0 +1,3 @@
1
+ import type { DrizzleDb } from './db.js';
2
+ export declare function runMigrations(db: DrizzleDb): void;
3
+ export declare function runMigrationsOnce(db: DrizzleDb, dbPath: string): void;
@@ -0,0 +1,31 @@
1
+ import { BaseSqliteRepository } from './base-sqlite-repository.js';
2
+ import { request_logs } from '../schema/index.js';
3
+ export type RequestLogRow = typeof request_logs.$inferSelect;
4
+ export type NewRequestLog = typeof request_logs.$inferInsert;
5
+ export interface RequestLogEntry {
6
+ path: string;
7
+ method: string;
8
+ statusCode: number;
9
+ durationMs: number;
10
+ ip?: string;
11
+ requestHeaders?: string;
12
+ responseHeaders?: string;
13
+ requestBody?: string;
14
+ responseBody?: string;
15
+ query?: string;
16
+ userId?: string;
17
+ projectId?: string;
18
+ partitionKey?: string;
19
+ metadata?: string;
20
+ }
21
+ export declare class RequestLogRepository extends BaseSqliteRepository {
22
+ private readonly dbPath?;
23
+ constructor(opts?: {
24
+ dbPath?: string;
25
+ dbRoot?: string;
26
+ });
27
+ resolveDbPath(): string;
28
+ private openHandle;
29
+ bulkInsert(entries: RequestLogEntry[]): void;
30
+ findRecent(limit?: number): RequestLogRow[];
31
+ }
@@ -0,0 +1,16 @@
1
+ import { BaseSqliteRepository } from './base-sqlite-repository.js';
2
+ import { spans } from '../schema/index.js';
3
+ export type SpanRow = typeof spans.$inferSelect;
4
+ export type NewSpan = typeof spans.$inferInsert;
5
+ export declare class SpanRepository extends BaseSqliteRepository {
6
+ private readonly dbPath?;
7
+ constructor(opts?: {
8
+ dbPath?: string;
9
+ dbRoot?: string;
10
+ });
11
+ resolveDbPath(): string;
12
+ private openHandle;
13
+ insertSpan(opts: NewSpan): void;
14
+ findByTaskId(taskId: string): SpanRow[];
15
+ findById(spanId: string): SpanRow | undefined;
16
+ }