@crewx/sdk 0.8.8-rc.31 → 0.8.8-rc.33

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.
@@ -110,4 +110,4 @@ ${s.join(`
110
110
  `),p},{behavior:"immediate"})}}catch(r){throw r instanceof u?r:new u("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(l).set({last_message:e,updated_at:r}).where(drizzleOrm.eq(l.id,t)).run();}catch(r){throw r instanceof u?r:new u("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(l).set(n).where(drizzleOrm.eq(l.id,t)).run();}catch(n){throw n instanceof u?n:new u("DB_ERROR","Failed to update thread",n)}finally{s.close();}}togglePin(t,e){let s=this.openHandle(true);try{let n=e?drizzleOrm.and(drizzleOrm.eq(l.id,t),drizzleOrm.eq(l.workspace_id,e)):drizzleOrm.eq(l.id,t),r=s.db.select({pinned:l.pinned,metadata:l.metadata}).from(l).where(n).get();if(!r)return null;let i=r.pinned?0:1,a=r.metadata?JSON.parse(r.metadata):{};if(i){let c=e?drizzleOrm.and(drizzleOrm.eq(l.pinned,1),drizzleOrm.eq(l.workspace_id,e)):drizzleOrm.eq(l.pinned,1),p=s.db.select({metadata:l.metadata}).from(l).where(c).all(),m=0;for(let f of p){let E=f.metadata?JSON.parse(f.metadata):{};typeof E.pinOrder=="number"&&E.pinOrder>m&&(m=E.pinOrder);}a.pinOrder=m+1;}else delete a.pinOrder;return s.db.update(l).set({pinned:i,metadata:Object.keys(a).length>0?JSON.stringify(a):null}).where(n).run(),{pinned:!!i}}catch(n){throw n instanceof u?n:new u("DB_ERROR","Failed to toggle pin",n)}finally{s.close();}}reorderPins(t,e){let s=this.openHandle(true);try{for(let n=0;n<t.length;n++){let r=e?drizzleOrm.and(drizzleOrm.eq(l.id,t[n]),drizzleOrm.eq(l.workspace_id,e)):drizzleOrm.eq(l.id,t[n]),i=s.db.select({metadata:l.metadata}).from(l).where(r).get();if(!i)continue;let a=i.metadata?JSON.parse(i.metadata):{};a.pinOrder=n+1,s.db.update(l).set({metadata:JSON.stringify(a)}).where(r).run();}}catch(n){throw n instanceof u?n:new u("DB_ERROR","Failed to reorder pins",n)}finally{s.close();}}toggleStar(t,e){let s=this.openHandle(true);try{let n=e?drizzleOrm.and(drizzleOrm.eq(l.id,t),drizzleOrm.eq(l.workspace_id,e)):drizzleOrm.eq(l.id,t),r=s.db.select({starred:l.starred}).from(l).where(n).get();if(!r)return null;let i=r.starred?0:1;return s.db.update(l).set({starred:i}).where(n).run(),{starred:!!i}}catch(n){throw n instanceof u?n:new u("DB_ERROR","Failed to toggle star",n)}finally{s.close();}}};function pe(d){return d.replace(/<conversation_history[^>]*>[\s\S]*?<\/conversation_history>/g,"").split(`
111
111
  `).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(`
112
112
  `).trim()}function _e(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(`
113
- `):e&&typeof e=="object"&&e.result!==void 0&&(t=e.result||"");}catch{t=pe(t);}return t}var Z=class{dbPath;constructor(t){this.dbPath=t??L.join(Pt.homedir(),".crewx","crewx.db");}getThreadRepo(){return new Q({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 {created:false}}return n.ensureThread(t,e,s),{created:true}}async fetchHistory(t,e){let s=e?.limit??100,n=this.getThreadRepo(),r=n.findThreadById(t),i=n.findTopLevelTasks(t),a=new Set(["done","completed","success"]);i=i.filter(m=>(!m.status||a.has(m.status))&&(!e?.currentTraceId||m.trace_id!==e.currentTraceId)),i=i.slice(0,s);let c=r?.platform??"cli",p=this.rowsToMessages(i);return {threadId:t,platform:c,messages:p,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){let{firstMessage:r}=this.getThreadRepo().saveUserMessage(t,e);return {id:t,firstMessage:r}}async saveAssistantMessage(t,e,s,n){return this.getThreadRepo().saveAssistantMessage(t,e),{id:t}}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=_e(s.result);n&&e.push({id:`${s.id}-assistant`,text:n,isAssistant:true,timestamp:new Date(s.started_at).getTime(),metadata:{agent_id:s.agent_id}});}return e}};var lt=class extends A{name="conversation";_provider;unsubStart=null;unsubEnd=null;constructor(t){super(),this._provider=new Z(t?.dbPath);}get conversationProvider(){return this._provider}async afterUserMessage(t,e,s,n){}async afterAssistantMessage(t,e,s){}attach(t){this.unsubStart=t.on("task:start",async e=>{if(!e.threadId)return;let s=e.platform??"cli";try{let n=await this._provider.ensureThread(e.threadId,s,e.workspaceId),r=await this._provider.saveUserMessage(e.threadId,e.message??"");await this.afterUserMessage(e.threadId,r.id,n.created||r.firstMessage,e);}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{let{id:r}=await this._provider.saveAssistantMessage(s,e.result,n);await this.afterAssistantMessage(s,r,e);}catch{}});}detach(t){this.unsubStart?.(),this.unsubStart=null,this.unsubEnd?.(),this.unsubEnd=null,this._provider.close?.();}};exports.ConversationPlugin=lt;exports.FileLoggerPlugin=et;exports.SqliteTracingPlugin=it;
113
+ `):e&&typeof e=="object"&&e.result!==void 0&&(t=e.result||"");}catch{t=pe(t);}return t}var Z=class{dbPath;constructor(t){this.dbPath=t??L.join(Pt.homedir(),".crewx","crewx.db");}getThreadRepo(){return new Q({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 {created:false}}return n.ensureThread(t,e,s),{created:true}}async fetchHistory(t,e){let s=e?.limit??100,n=this.getThreadRepo(),r=n.findThreadById(t),i=n.findTopLevelTasks(t),a=new Set(["done","completed","success"]);i=i.filter(m=>(!m.status||a.has(m.status))&&(!e?.currentTraceId||m.trace_id!==e.currentTraceId)),i=i.slice(0,s);let c=r?.platform??"cli",p=this.rowsToMessages(i);return {threadId:t,platform:c,messages:p,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){let{firstMessage:r}=this.getThreadRepo().saveUserMessage(t,e);return {id:t,firstMessage:r}}async saveAssistantMessage(t,e,s,n){return this.getThreadRepo().saveAssistantMessage(t,e),{id:t}}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=_e(s.result);n&&e.push({id:`${s.id}-assistant`,text:n,isAssistant:true,timestamp:new Date(s.started_at).getTime(),metadata:{agent_id:s.agent_id,task_id:s.id}});}return e}};var lt=class extends A{name="conversation";_provider;unsubStart=null;unsubEnd=null;constructor(t){super(),this._provider=new Z(t?.dbPath);}get conversationProvider(){return this._provider}async afterUserMessage(t,e,s,n){}async afterAssistantMessage(t,e,s){}attach(t){this.unsubStart=t.on("task:start",async e=>{if(!e.threadId)return;let s=e.platform??"cli";try{let n=await this._provider.ensureThread(e.threadId,s,e.workspaceId),r=await this._provider.saveUserMessage(e.threadId,e.message??"");await this.afterUserMessage(e.threadId,r.id,n.created||r.firstMessage,e);}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{let{id:r}=await this._provider.saveAssistantMessage(s,e.result,n);await this.afterAssistantMessage(s,r,e);}catch{}});}detach(t){this.unsubStart?.(),this.unsubStart=null,this.unsubEnd?.(),this.unsubEnd=null,this._provider.close?.();}};exports.ConversationPlugin=lt;exports.FileLoggerPlugin=et;exports.SqliteTracingPlugin=it;
@@ -4,5 +4,8 @@ export interface FormatConversationBlockContext {
4
4
  messagesCount: number;
5
5
  agentIds: string[];
6
6
  primaryAgentId: string;
7
+ currentAgentId: string;
8
+ threadId: string;
9
+ now: string;
7
10
  }
8
- export declare function formatConversationHelper(this: any, messages: any, platform: any, options?: any): string;
11
+ export declare function formatConversationHelper(this: any, ...args: any[]): string;
@@ -0,0 +1,2 @@
1
+ export declare function toIsoWithOffset(date: Date): string;
2
+ export declare function formatRelativeTime(deltaMs: number): string;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@crewx/sdk",
3
- "version": "0.8.8-rc.31",
3
+ "version": "0.8.8-rc.33",
4
4
  "license": "UNLICENSED",
5
5
  "engines": {
6
6
  "node": ">=20.19.0"
@@ -304,8 +304,15 @@ layouts:
304
304
 
305
305
  {{#if props.showConversationHistory}}
306
306
  {{#if messages.length}}
307
+ {{#if thread_id}}
308
+ <conversation_context thread_id="{{thread_id}}" agent_id="{{agent.id}}" now="{{now}}">
309
+ To review your previous work in this conversation:
310
+ crewx dreaming --thread={{thread_id}} --agent={{agent.id}} --continue
311
+ </conversation_context>
312
+
313
+ {{/if}}
307
314
  <conversation_history platform="{{platform}}">
308
- {{#formatConversation messages platform}}
315
+ {{#formatConversation messages platform agent.id thread_id}}
309
316
  {{#if primaryAgentId}}
310
317
  Primary agent: @{{primaryAgentId}}
311
318
  {{/if}}
@@ -319,15 +326,17 @@ layouts:
319
326
  {{/if}}
320
327
  {{#each messages}}
321
328
  {{#if isAssistant}}
322
- **{{#if metadata.slack}}{{#with metadata.slack}}{{#if bot_username}}{{bot_username}}{{else}}Assistant{{/if}}{{#if bot_user_id}} (<@{{bot_user_id}}>){{/if}}{{/with}}{{else}}Assistant{{/if}}{{#if metadata.agent_id}} (@{{metadata.agent_id}}){{/if}}**
329
+ <message role="assistant"{{#if metadata.agent_id}} agent="{{metadata.agent_id}}"{{/if}}{{#if ../currentAgentId}}{{#if (eq metadata.agent_id ../currentAgentId)}}{{#if metadata.task_id}} task_id="{{metadata.task_id}}"{{/if}}{{/if}}{{/if}}{{#if created_at}} created_at="{{created_at}}"{{/if}}{{#if relative_time}} relative_time="{{relative_time}}"{{/if}}{{#if metadata.slack}}{{#with metadata.slack}}{{#if bot_username}} name="{{bot_username}}"{{/if}}{{#if bot_user_id}} slack_user="{{bot_user_id}}"{{/if}}{{/with}}{{/if}}>
323
330
  {{else}}
324
- **{{#if metadata.slack}}{{#with metadata.slack}}{{#if user_profile.display_name}}{{user_profile.display_name}}{{else if username}}{{username}}{{else if user_id}}User{{/if}}{{#if user_id}} (<@{{user_id}}>){{/if}}{{/with}}{{else}}User{{/if}}**
325
- {{/if}}: {{{escapeHandlebars text}}}
331
+ <message role="user"{{#if created_at}} created_at="{{created_at}}"{{/if}}{{#if relative_time}} relative_time="{{relative_time}}"{{/if}}{{#if metadata.slack}}{{#with metadata.slack}}{{#if user_profile.display_name}} name="{{user_profile.display_name}}"{{else if username}} name="{{username}}"{{/if}}{{#if user_id}} slack_user="{{user_id}}"{{/if}}{{/with}}{{/if}}>
332
+ {{/if}}
333
+ {{{escapeHandlebars text}}}
326
334
  {{#if files}}
327
335
  {{#each files}}
328
336
  📎 {{name}}{{#if size}} ({{formatFileSize size}}){{/if}}{{#if localPath}} - Local: {{localPath}}{{/if}}
329
337
  {{/each}}
330
338
  {{/if}}
339
+ </message>
331
340
  {{/each}}
332
341
  {{/formatConversation}}
333
342