@zibby/agent-workflow 0.4.6 → 0.4.7

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.
@@ -22,6 +22,7 @@ export namespace SKILLS {
22
22
  let GIT: string;
23
23
  let SLACK: string;
24
24
  let LARK: string;
25
+ let CHAT_NOTIFY: string;
25
26
  let SENTRY: string;
26
27
  let MEMORY: string;
27
28
  let CHAT_MEMORY: string;
package/dist/constants.js CHANGED
@@ -1 +1 @@
1
- var o=".zibby/output",t="sessions",I=".session-info.json",s=".zibby-stop",_="result.json",n="raw_stream_output.txt",r="events.json",E={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GITLAB:"gitlab",GIT:"git",SLACK:"slack",LARK:"lark",SENTRY:"sentry",MEMORY:"memory",CHAT_MEMORY:"chat-memory",RUNNER:"runner",SKILL_INSTALLER:"skill-installer",CORE_TOOLS:"core-tools",WORKFLOW_BUILDER:"workflow-builder",SESSION:"session",OPENAI_BILLING:"openai_billing",ANTHROPIC_BILLING:"anthropic_billing",CURSOR_ADMIN:"cursor_admin",NOTION:"notion"},L=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];export{L as CI_ENV_VARS,o as DEFAULT_OUTPUT_BASE,r as EVENTS_FILE,n as RAW_OUTPUT_FILE,_ as RESULT_FILE,t as SESSIONS_DIR,I as SESSION_INFO_FILE,E as SKILLS,s as STOP_REQUEST_FILE};
1
+ var o=".zibby/output",t="sessions",I=".session-info.json",_=".zibby-stop",s="result.json",n="raw_stream_output.txt",r="events.json",E={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GITLAB:"gitlab",GIT:"git",SLACK:"slack",LARK:"lark",CHAT_NOTIFY:"chat_notify",SENTRY:"sentry",MEMORY:"memory",CHAT_MEMORY:"chat-memory",RUNNER:"runner",SKILL_INSTALLER:"skill-installer",CORE_TOOLS:"core-tools",WORKFLOW_BUILDER:"workflow-builder",SESSION:"session",OPENAI_BILLING:"openai_billing",ANTHROPIC_BILLING:"anthropic_billing",CURSOR_ADMIN:"cursor_admin",NOTION:"notion"},L=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];export{L as CI_ENV_VARS,o as DEFAULT_OUTPUT_BASE,r as EVENTS_FILE,n as RAW_OUTPUT_FILE,s as RESULT_FILE,t as SESSIONS_DIR,I as SESSION_INFO_FILE,E as SKILLS,_ as STOP_REQUEST_FILE};
@@ -11,9 +11,9 @@ PRIORITY OVERRIDE \u2014 THE FOLLOWING INSTRUCTIONS TAKE PRECEDENCE OVER ALL PRE
11
11
  ${h}
12
12
  `),y.debug(`[workflow] prompt length: ${p.length} chars`),r.invoke(p,d)}var be,W,ee=se(()=>{Qe();F();le();be=Symbol.for("@zibby/agent-workflow.strategies");globalThis[be]||(globalThis[be]=[]);W=globalThis[be]});var Pt=new Set(["__proto__","constructor","prototype"]);function we(o){if(Pt.has(o))throw new Error(`Invalid state key: "${o}"`)}var ie=class{constructor(e={}){this._state=Object.create(null),Object.assign(this._state,{messages:[],errors:[],artifacts:{},metadata:{},...e}),this._history=[]}get(e){return this._state[e]}set(e,t){we(e),this._history.push({...this._state}),this._state[e]=t}update(e){let t=Object.getOwnPropertyNames(e);for(let r of t)we(r);this._history.push({...this._state});for(let r of t)this._state[r]=e[r]}append(e,t){we(e),this._history.push({...this._state}),Array.isArray(this._state[e])||(this._state[e]=[]),this._state[e].push(t)}getAll(){return{...this._state}}rollback(){this._history.length>0&&(this._state=this._history.pop())}};var ae=class{constructor(e){this.schema=e}parse(e){let t=e.match(/```json\s*([\s\S]*?)\s*```/);if(t)return this.validate(JSON.parse(t[1]));let r=[e.match(/\{[\s\S]*?\}/),e.match(/\{[\s\S]*\}/)].filter(Boolean).map(s=>s[0]);for(let s of r)try{return this.validate(JSON.parse(s))}catch(n){if(!(n instanceof SyntaxError))throw n}return this.validate({result:e.trim()})}validate(e){let t=[];for(let[r,s]of Object.entries(this.schema)){if(s.required&&!(r in e)&&t.push(`Missing required field: ${r}`),r in e&&s.type){let n=typeof e[r];n!==s.type&&t.push(`Field '${r}' expected ${s.type}, got ${n}`)}if(s.validate&&r in e){let n=s.validate(e[r]);n&&t.push(`Field '${r}': ${n}`)}}if(t.length>0)throw new Error(`Output validation failed:
13
13
  ${t.join(`
14
- `)}`);return e}};F();import{writeFileSync as $e,readFileSync as ot,existsSync as rt,mkdirSync as Jt}from"node:fs";import{join as ve,dirname as Yt}from"node:path";import x from"chalk";var Bt="__WORKFLOW_GRAPH_LOG__",V=x.gray("\u2502"),Mt=x.gray("\u250C"),Ge=x.gray("\u2514"),ye=x.green("\u25C6"),He=x.hex("#c084fc")("\u25C6"),Je=x.hex("#2dd4bf")("\u25C6"),_e=x.red("\u25C6"),Ye=`${V} `,Ze=2;function ze(o){return o<1e3?`${o}ms`:`${(o/1e3).toFixed(1)}s`}function Ke(o,e){return(t,r,s)=>{if(typeof t!="string")return o(t,r,s);let n=process.stdout.columns||120,a="";for(let i=0;i<t.length;i++){let u=t[i];e.lineStart&&(a+=Ye,e.col=Ze,e.lineStart=!1),u===`
14
+ `)}`);return e}};F();import{writeFileSync as $e,readFileSync as ot,existsSync as rt,mkdirSync as Yt}from"node:fs";import{join as ve,dirname as Jt}from"node:path";import x from"chalk";var Bt="__WORKFLOW_GRAPH_LOG__",V=x.gray("\u2502"),Mt=x.gray("\u250C"),Ge=x.gray("\u2514"),ye=x.green("\u25C6"),He=x.hex("#c084fc")("\u25C6"),Ye=x.hex("#2dd4bf")("\u25C6"),_e=x.red("\u25C6"),Je=`${V} `,Ze=2;function ze(o){return o<1e3?`${o}ms`:`${(o/1e3).toFixed(1)}s`}function Ke(o,e){return(t,r,s)=>{if(typeof t!="string")return o(t,r,s);let n=process.stdout.columns||120,a="";for(let i=0;i<t.length;i++){let u=t[i];e.lineStart&&(a+=Je,e.col=Ze,e.lineStart=!1),u===`
15
15
  `?(a+=u,e.lineStart=!0,e.col=0,e.inEsc=!1):u==="\x1B"?(e.inEsc=!0,a+=u):e.inEsc?(a+=u,(u>="A"&&u<="Z"||u>="a"&&u<="z")&&(e.inEsc=!1)):(e.col++,a+=u,e.col>=n&&(a+=`
16
- ${Ye}`,e.col=Ze))}return o(a,r,s)}}var Ie=class{constructor(){this._currentNode=null,this._origStdoutWrite=null,this._origStderrWrite=null,this._emitWorkflowGraphMarkers=String(process.env.ZIBBY_EMIT_GRAPH_MARKERS||"").trim()==="1"||String(process.env.ZIBBY_WORKFLOW_GRAPH_LOG_MARKERS||"").trim()==="1"}get isInsideNode(){return this._currentNode!==null}_startIntercepting(){this._origStdoutWrite=process.stdout.write.bind(process.stdout),this._origStderrWrite=process.stderr.write.bind(process.stderr);let e={lineStart:!0,col:0,inEsc:!1},t={lineStart:!0,col:0,inEsc:!1};this._outState=e,this._errState=t,process.stdout.write=Ke(this._origStdoutWrite,e),process.stderr.write=Ke(this._origStderrWrite,t)}_stopIntercepting(){this._origStdoutWrite&&(this._outState&&!this._outState.lineStart&&this._origStdoutWrite(`
16
+ ${Je}`,e.col=Ze))}return o(a,r,s)}}var Ie=class{constructor(){this._currentNode=null,this._origStdoutWrite=null,this._origStderrWrite=null,this._emitWorkflowGraphMarkers=String(process.env.ZIBBY_EMIT_GRAPH_MARKERS||"").trim()==="1"||String(process.env.ZIBBY_WORKFLOW_GRAPH_LOG_MARKERS||"").trim()==="1"}get isInsideNode(){return this._currentNode!==null}_startIntercepting(){this._origStdoutWrite=process.stdout.write.bind(process.stdout),this._origStderrWrite=process.stderr.write.bind(process.stderr);let e={lineStart:!0,col:0,inEsc:!1},t={lineStart:!0,col:0,inEsc:!1};this._outState=e,this._errState=t,process.stdout.write=Ke(this._origStdoutWrite,e),process.stderr.write=Ke(this._origStderrWrite,t)}_stopIntercepting(){this._origStdoutWrite&&(this._outState&&!this._outState.lineStart&&this._origStdoutWrite(`
17
17
  `),process.stdout.write=this._origStdoutWrite),this._origStderrWrite&&(this._errState&&!this._errState.lineStart&&this._origStderrWrite(`
18
18
  `),process.stderr.write=this._origStderrWrite),this._origStdoutWrite=null,this._origStderrWrite=null}_rawWrite(e){(this._origStdoutWrite||process.stdout.write.bind(process.stdout))(`${e}
19
19
  `)}_emitGraphLogMarker(e){if(!this._emitWorkflowGraphMarkers)return;let t=`${Bt}${JSON.stringify(e)}
@@ -22,19 +22,19 @@ ${Ye}`,e.col=Ze))}return o(a,r,s)}}var Ie=class{constructor(){this._currentNode=
22
22
  `)):process.stdout.write.bind(process.stdout)(`${e} ${t}
23
23
  `)}step(e){this._origStdoutWrite?this._writeDot(ye,e):process.stdout.write.bind(process.stdout)(`${V} ${ye} ${e}
24
24
  `)}stepInfo(e){this.step(e)}stepTool(e){this._origStdoutWrite?this._writeDot(He,e):process.stdout.write.bind(process.stdout)(`${V} ${He} ${e}
25
- `)}stepMemory(e){let t=x.hex("#2dd4bf")(e);this._origStdoutWrite?this._writeDot(Je,t):process.stdout.write.bind(process.stdout)(`${V} ${Je} ${t}
25
+ `)}stepMemory(e){let t=x.hex("#2dd4bf")(e);this._origStdoutWrite?this._writeDot(Ye,t):process.stdout.write.bind(process.stdout)(`${V} ${Ye} ${t}
26
26
  `)}stepFail(e){this._origStdoutWrite?this._writeDot(_e,x.red(e)):process.stdout.write.bind(process.stdout)(`${V} ${_e} ${x.red(e)}
27
27
  `)}nodeStart(e){this._currentNode=e,this._emitGraphLogMarker({phase:"node_begin",node:e}),this._rawWrite(`${Mt} ${e}`),this._startIntercepting()}nodeComplete(e,t={}){this._stopIntercepting();let{duration:r,details:s}=t;if(s)for(let a of s)this._rawWrite(`${ye} ${a}`);let n=r?x.dim(` ${ze(r)}`):"";this._rawWrite(`${Ge} ${x.green("done")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}nodeFailed(e,t,r={}){this._stopIntercepting();let{duration:s}=r,n=s?x.dim(` ${ze(s)}`):"";this._rawWrite(`${_e} ${x.red(t)}`),this._rawWrite(`${Ge} ${x.red("failed")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}route(e,t){this._rawWrite(x.dim(` ${e} \u2192 ${t}`)),this._rawWrite("")}graphComplete(){}},A=new Ie;var ue=".zibby/output",qe="sessions",z=".session-info.json",Ve=".zibby-stop";var Xe=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];var j=class{constructor(e){if(this.config=e,this.name=e.name,this.prompt=e.prompt,this.outputSchema=e.outputSchema,!this.outputSchema&&!e._isCustomCode)throw new Error(`Node '${this.name}' must define outputSchema (Zod schema). This defines the contract for what the node returns to state.`);this.isZodSchema=this.outputSchema&&typeof this.outputSchema._def<"u",this.parser=e.outputSchema&&!this.isZodSchema?new ae(e.outputSchema):null,this.retries=e.retries||0,this.onComplete=e.onComplete,this.customExecute=e.execute}async execute(e,t){let r=()=>t&&typeof t.getAll=="function"?t.getAll():e,s=p=>t&&typeof t.get=="function"?t.get(p):e?.[p];if(typeof this.customExecute=="function"){y.debug(`[workflow] node '${this.name}': custom execute (skipping LLM)`);try{let p=await this.customExecute(e);return typeof p=="object"&&p!==null&&p.success===!1?{success:!1,error:p.error||"Node execution failed",raw:p.raw||null}:this.isZodSchema?(y.debug(`[workflow] node '${this.name}': validating output schema`),{success:!0,output:this.outputSchema.parse(p),raw:null}):{success:!0,output:p,raw:null}}catch(p){return y.error(`[workflow] node '${this.name}' failed: ${p.message}`),p.name==="ZodError"&&y.error(`Schema errors: ${JSON.stringify(p.issues||p.errors,null,2)}`),{success:!1,error:p.message,raw:null}}}let n=typeof this.prompt=="function"?this.prompt(r()):this.prompt,a=s("_skillHints");a&&(n=`${a}
28
28
 
29
- ${n}`);let i=r(),u=i.cwd||process.cwd(),l=i.sessionPath;try{if(l){let p=ve(l,z);if(rt(p)){let h=JSON.parse(ot(p,"utf-8"));h.currentNode=this.name,$e(p,JSON.stringify(h,null,2),"utf-8")}let c=ve(l,"..",z);if(rt(c))try{let h=JSON.parse(ot(c,"utf-8"));h.currentNode=this.name,$e(c,JSON.stringify(h,null,2),"utf-8")}catch{}}}catch(p){y.debug(`[workflow] could not update session info: ${p.message}`)}let d=null;for(let p=0;p<=this.retries;p++)try{y.debug(`[workflow] node '${this.name}' attempt ${p}`);let c=r().config||{},h=c.agents||{},w=this.config.agent??h[this.name]??null,_={state:r()};w&&(_.preferredAgent=w);let $={workspace:u,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:l,config:c,nodeName:this.name,timeout:this.config?.timeout||3e5},b=e?._coreInvokeAgent;b||(b=(await Promise.resolve().then(()=>(ee(),Q))).invokeAgent);let f=await b(n,_,$),I,g;if(typeof f=="string"?(I=f,g=null):f.structured?(I=f.raw||JSON.stringify(f.structured,null,2),g=f.structured):(I=f.raw||JSON.stringify(f,null,2),g=f.extracted||null),l)try{let m=ve(l,this.name,"raw_stream_output.txt");Jt(Yt(m),{recursive:!0}),$e(m,typeof I=="string"?I:JSON.stringify(I),"utf-8")}catch(m){y.debug(`[workflow] could not save raw output: ${m.message}`)}if(this.isZodSchema&&g){y.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(g,null,2)}`);let m=g;if(typeof this.onComplete=="function")try{m=await this.onComplete(r(),g)}catch(R){y.warn(`[workflow] onComplete hook failed: ${R.message}`)}return{success:!0,output:m,raw:I}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(r(),{raw:I}),raw:I}}catch(m){throw new Error(`onComplete failed: ${m.message}`,{cause:m})}if(this.parser){let m=this.parser.parse(I);return y.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(m,null,2)}`),A.step("Output parsed"),{success:!0,output:m,raw:I}}return{success:!0,output:I,raw:I}}catch(c){d=c,p<this.retries&&y.info(`[workflow] node '${this.name}' failed, retrying (${p+1}/${this.retries})\u2026`)}return{success:!1,error:d.message,raw:null}}},de=class extends j{constructor(e){super({...e,_isCustomCode:!0}),this.condition=e.condition}async execute(e,t){let r=t&&typeof t.getAll=="function"?t.getAll():e;return{success:!0,output:{nextNode:this.condition(r)},raw:null}}};F();F();import{mkdirSync as Kt,existsSync as H,statSync as lt,readdirSync as dt,rmSync as qt}from"node:fs";import{spawn as ut}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as Vt}from"node:url";import{AsyncLocalStorage as Zt}from"node:async_hooks";var Te=new Zt;function te(){let o=Te.getStore();return o||Object.freeze({executionId:process.env.EXECUTION_ID||null,parentExecutionId:process.env.PARENT_EXECUTION_ID||null,depth:0,conversationId:process.env.ZIBBY_CONVERSATION_ID||null,dispatchMode:process.env.DISPATCH_MODE||null})}function nt(o,e){let t=Te.getStore()||te(),r=Object.freeze({executionId:o.executionId,parentExecutionId:o.parentExecutionId??t.executionId??null,depth:(t.depth||0)+(o.executionId!==t.executionId?1:0),conversationId:o.conversationId!==void 0?o.conversationId:t.conversationId??null,dispatchMode:o.dispatchMode??null});return Te.run(r,e)}var xe=new Map,Ae=new Map,st=new Map;function it(o,e,t={}){if(!o||typeof o!="string")throw new Error("subgraph-registry.register: name required");if(typeof e!="function")throw new Error("subgraph-registry.register: factory must be a function");xe.set(o,e),Ae.set(o,"ready"),st.set(o,{...t,cachedAt:Date.now()})}function at(o,e){Ae.set(o,"failed"),st.set(o,{error:e?.message||String(e),failedAt:Date.now()}),xe.delete(o)}function ct(o){return Ae.get(o)==="ready"?xe.get(o):null}var pe=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function Xt(){return`node${(process.versions?.node||"").split(".")[0]||"unknown"}-${process.platform}-${process.arch}`}var k=class extends Error{constructor(e,t){super(`in-process sub-graph fallback: ${e}${t?` (${t})`:""}`),this.fallback=!0,this.reason=e,this.detail=t||null,this.name="SubgraphFallback"}};function Qt(){let o=(process.env.SUBGRAPH_INTERNAL_URL||"").replace(/\/$/,""),e=(process.env.PROGRESS_API_URL||"").replace(/\/executions\/?$/,""),t=o||e,r=process.env.PROJECT_ID,s=process.env.PROJECT_API_TOKEN;if(!t||!r||!s)throw new k("env","SUBGRAPH_INTERNAL_URL/PROGRESS_API_URL/PROJECT_ID/PROJECT_API_TOKEN missing");return{apiBase:t,projectId:r,authToken:s}}async function eo({apiBase:o,authToken:e,body:t}){let r;try{r=await fetch(`${o}/internal/subgraph/begin`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(t)})}catch(n){throw new k("network",`begin fetch failed: ${n.message}`)}let s=null;try{s=await r.json()}catch{}if(!r.ok){if(r.status===404){let n=new Error(`Sub-graph child '${t.childWorkflowType}' not found in project`);throw n.code="SUBGRAPH_NOT_FOUND",n.status=404,n}if(r.status===429){let n=s?.quotaInfo||{},a=new Error(`Sub-graph blocked by quota (${n.used??"?"}/${n.limit??"?"} on ${n.planId||"plan"})`);throw a.code="SUBGRAPH_QUOTA_EXCEEDED",a.status=429,a.quotaInfo=n,a}if(r.status===400&&s?.validationErrors){let n=new Error(`Sub-graph rejected input: ${s?.error||s?.message||"validation failed"}`);throw n.code="SUBGRAPH_INVALID_INPUT",n.status=400,n.validationErrors=s.validationErrors,n.missing=s.missing,n}throw new k("begin-status",`begin returned ${r.status}`)}return s?.data||s}async function G({apiBase:o,authToken:e,payload:t}){try{let r=await fetch(`${o}/internal/subgraph/finalize`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(t)});r.ok||y.warn(`[in-process subgraph] finalize returned ${r.status} for ${t.childExecutionId}`)}catch(r){y.warn(`[in-process subgraph] finalize failed: ${r.message}`)}}async function to(o,e){let t=U(e,".ready"),r=U(e,"graph.mjs");if(H(t)&&H(r))return;Kt(e,{recursive:!0});let s=U(e,".lock"),n=!1;try{let{openSync:a,closeSync:i}=await import("node:fs"),u=a(s,"wx");i(u),n=!0}catch(a){if(a.code!=="EEXIST")throw a}if(!n){let a=Date.now()+3e4;for(;Date.now()<a;){if(H(t)&&H(r))return;await new Promise(i=>setTimeout(i,100))}throw new k("bundle-extract-timeout","sibling extract did not complete within 30s")}try{await new Promise((u,l)=>{let d=ut("curl",["-fsSL",o],{stdio:["ignore","pipe","inherit"]}),p=ut("tar",["-xzf","-","-C",e],{stdio:["pipe","inherit","inherit"]});d.stdout.pipe(p.stdin);let c,h,w=()=>{if(c!==void 0&&h!==void 0){if(c!==0)return l(new Error(`curl exited ${c}`));if(h!==0)return l(new Error(`tar exited ${h}`));u()}};d.on("close",_=>{c=_,w()}),p.on("close",_=>{h=_,w()}),d.on("error",l),p.on("error",l)});let{writeFileSync:a,unlinkSync:i}=await import("node:fs");a(t,"");try{i(s)}catch{}}catch(a){try{let{unlinkSync:i}=await import("node:fs");i(s)}catch{}throw new k("bundle-extract-failed",a.message)}}async function oo(o){let e=U(o,"graph.mjs");if(!H(e))throw new k("entry-missing",`graph.mjs missing under ${o}`);let t;try{t=await import(Vt(e).href)}catch(s){throw new k("import-failed",`${s?.code||s?.name||"unknown"}: ${s.message}`)}let r=t.default||Object.values(t).find(s=>typeof s=="function"&&s.prototype?.buildGraph);if(!r)throw new k("entry-class-missing","no buildGraph() class export found");return r}async function pt(o,e={}){if(!o||typeof o!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let t=te(),r;try{r=Qt()}catch(g){throw g}y.debug(`[in-process subgraph] begin '${o}' parent=${t.executionId||"<root>"}`);let s=await eo({apiBase:r.apiBase,authToken:r.authToken,body:{parentExecutionId:t.executionId,childWorkflowType:o,input:e.input||{},...e.conversationId?{conversationId:e.conversationId}:{}}}),{childExecutionId:n,runtimeTag:a,bundlePresignedUrl:i,sourcesPresignedUrl:u,workflowVersion:l,workflowUuid:d,bundleReady:p}=s,c=Xt();if(a&&a!==c)throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:`runtimeTag mismatch: parent=${c} child=${a}`,code:"RUNTIME_MISMATCH"}}}),new k("runtime-mismatch",`${c} vs ${a}`);if(!p||!i)throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:"bundle not ready for in-process; falling back to HTTP",code:"NO_BUNDLE"}}}),new k("no-bundle","workflow bundle not built yet");let h=ct(o);if(!h){let g=U(pe,`${d}@${l||"0"}`);try{await to(i,g);try{no()}catch{}}catch(m){throw m.fallback&&await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"failed",error:{message:m.message,code:m.reason}}}),m}try{h=await oo(g),it(o,h,{workflowUuid:d,version:l,runtimeTag:a,cacheDir:g})}catch(m){throw at(o,m),await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"failed",error:{message:m.message,code:m.reason||"IMPORT_FAILED"}}}),m.fallback?m:new k("import-failed",m.message)}}let w=Date.now(),$=await(typeof h=="function"&&h.prototype?.buildGraph?new h:h).buildGraph(),b={...e.input||{}},f,I;try{f=await nt({executionId:n,parentExecutionId:t.executionId,conversationId:e.conversationId!==void 0?e.conversationId:t.conversationId,dispatchMode:"inprocess"},()=>$.run(e.parentAgent,b,{signal:e.signal})),I=f&&typeof f=="object"&&"state"in f?f.state:f}catch(g){throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"failed",error:{message:g.message,code:g.code||"CHILD_THREW",stack:g.stack},durationMs:Date.now()-w}}),g}if(f&&typeof f=="object"&&f.stoppedExternally){await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",finalState:I,durationMs:Date.now()-w}});let g=new Error(`Sub-graph '${o}' canceled by parent abort`);throw g.code="SUBGRAPH_CANCELED",g.subgraphJobId=n,g}return await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"completed",finalState:I,durationMs:Date.now()-w}}),{finalState:I,executionId:n}}function ro(o){let e=0,t=[o];for(;t.length;){let r=t.pop(),s;try{s=lt(r)}catch{continue}if(s.isDirectory()){let n;try{n=dt(r)}catch{continue}for(let a of n)t.push(U(r,a))}else e+=s.size}return e}function no({cap:o=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!H(pe))return{evicted:0,freedBytes:0};let e=dt(pe),t=[],r=0;for(let i of e){let u=U(pe,i),l;try{l=lt(u)}catch{continue}let d=l.isDirectory()?ro(u):l.size;r+=d,t.push({name:i,full:u,size:d,mtimeMs:l.mtimeMs})}if(r<=o)return{evicted:0,freedBytes:0,totalBytes:r};t.sort((i,u)=>i.mtimeMs-u.mtimeMs);let s=Math.floor(o*.7),n=0,a=0;for(let i of t){if(r-n<=s)break;if(!H(U(i.full,".lock")))try{qt(i.full,{recursive:!0,force:!0}),n+=i.size,a+=1}catch(u){y.debug(`[sub-graph cache] evict skip ${i.name}: ${u.message}`)}}return a>0&&y.info(`[sub-graph cache] evicted ${a} entr(y/ies), freed ${(n/1024/1024).toFixed(1)}MB`),{evicted:a,freedBytes:n,totalBytes:r-n}}catch(e){return y.debug(`[sub-graph cache] evict failed: ${e.message}`),{evicted:0,freedBytes:0}}}var so=2e3,io=600*1e3,ao=new Set(["completed","failed","canceled","timeout"]);function co(){let o=process.env.PROGRESS_API_URL;if(!o)throw new Error("Sub-graph dispatch requires PROGRESS_API_URL env var (set automatically on cloud runs). Sub-graphs are not supported in local in-process runs yet \u2014 deploy the parent and child to cloud.");return o.replace(/\/executions\/?$/,"")}function uo(){let o=process.env.PROJECT_ID;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return o}function lo(){let o=process.env.PROJECT_API_TOKEN;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return o}function po(){return process.env.EXECUTION_ID||null}function ft(o,e){return e==null?o:typeof e=="function"?e(o):typeof e=="string"?e.split(".").reduce((t,r)=>t==null?t:t[r],o):o}async function ht(o,e={}){if(!o||typeof o!="string")throw new Error("dispatchSubgraph: workflowName (string) is required");let t=te(),r=Number(process.env.ZIBBY_SUBGRAPH_MAX_DEPTH||10);if((t.depth||0)>=r)throw new Error(`dispatchSubgraph('${o}'): sub-graph depth ${t.depth} reached cap of ${r}. Restructure the graph or raise ZIBBY_SUBGRAPH_MAX_DEPTH.`);if(process.env.ZIBBY_INPROCESS_SUBGRAPH!=="0"&&!e.async)try{y.debug(`[sub-graph] trying in-process for '${o}'`);let{finalState:g}=await pt(o,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),m=ft(g,e.output);return y.info(`[sub-graph] '${o}' completed in-process`),m}catch(g){if(g instanceof k||g?.fallback)y.info(`[sub-graph] in-process fallback for '${o}': ${g.reason||"unknown"} \u2014 using HTTP`);else throw g}let s=co(),n=uo(),a=lo(),i=po(),u=`${s}/projects/${encodeURIComponent(n)}/workflows/${encodeURIComponent(o)}/trigger`,l={input:e.input||{},...i?{parentExecutionId:i}:{},...e.conversationId?{conversationId:e.conversationId}:{}};y.info(`[sub-graph] dispatching '${o}' (${e.async?"async":"sync"}) from parent ${i||"<none>"}`);let d=await fetch(u,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${a}`},body:JSON.stringify(l)});if(!d.ok){let g=null,m="";try{g=await d.json(),m=g?.error||g?.message||JSON.stringify(g)}catch{m=await d.text().catch(()=>"")}if(d.status===429){let T=g?.quotaInfo||{},M=new Error(`Sub-graph '${o}' blocked by execution quota (${T.used??"?"}/${T.limit??"?"} on plan ${T.planId||"unknown"}). Sub-workflow runs count toward the same monthly cap as user-triggered runs.`);throw M.code="SUBGRAPH_QUOTA_EXCEEDED",M.status=429,M.subgraph=o,M.quotaInfo=T,M}if(d.status===400){let T=new Error(`Sub-graph '${o}' rejected input: ${m}`);throw T.code="SUBGRAPH_INVALID_INPUT",T.status=400,T.subgraph=o,T.validationErrors=g?.validationErrors||null,T.missing=g?.missing||null,T}let R=new Error(`Sub-graph '${o}' trigger rejected (${d.status}): ${m}`);throw R.code="SUBGRAPH_TRIGGER_FAILED",R.status=d.status,R.subgraph=o,R}let p=await d.json(),c=p?.data?.jobId||p?.jobId;if(!c)throw new Error(`Sub-graph '${o}' trigger returned no jobId: ${JSON.stringify(p).slice(0,200)}`);if(e.async)return y.info(`[sub-graph] async dispatch of '${o}' \u2192 jobId=${c} (not waiting)`),{jobId:c,status:"accepted",workflow:o};let h=Number.isFinite(e.timeoutMs)?e.timeoutMs:io,w=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:so,_=`${s}/executions/${encodeURIComponent(c)}`,$=Date.now()+h,b="accepted",f=0;for(;Date.now()<$;){await new Promise(T=>setTimeout(T,w)),f+=1;let g=await fetch(_,{headers:{Authorization:`Bearer ${a}`}});if(!g.ok){if(g.status>=500){y.warn(`[sub-graph] status poll for ${c} returned ${g.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${c}: ${g.status}`)}let m=await g.json(),R=m?.data||m?.execution||m;if(b=R?.status||b,ao.has(b)){if(b!=="completed"){let S=new Error(`Sub-graph '${o}' (${c}) ended in status '${b}'`);throw S.subgraphJobId=c,S.subgraphStatus=b,S}let T=R?.finalState||R?.state||{},M=ft(T,e.output);return y.info(`[sub-graph] '${o}' (${c}) completed after ${f} polls`),M}}let I=new Error(`Sub-graph '${o}' (${c}) timed out after ${Math.round(h/1e3)}s (last status: ${b})`);throw I.subgraphJobId=c,I.subgraphStatus=b,I}import{existsSync as gt,readFileSync as fo}from"node:fs";import{join as ke,dirname as mt}from"node:path";var fe=class{static async loadContext(e,t,r={}){let s={},n=r.filenames||["CONTEXT.md","AGENTS.md"];if(e){let i=mt(ke(t,e));for(let u of n){let l=await this.findAndMergeContextFiles(u,i,t);if(l){let d=u.replace(/\.[^.]+$/,"").toLowerCase();s[d]=l}}}let a=r.discovery||{};for(let[i,u]of Object.entries(a))try{let l=ke(t,u);gt(l)&&(s[i]=await this.loadFile(l))}catch(l){console.warn(`[workflow] could not load context '${i}' from '${u}': ${l.message}`)}return s}static async findAndMergeContextFiles(e,t,r){let s=[],n=t;for(;n.startsWith(r);){let a=ke(n,e);if(gt(a))try{s.unshift(await this.loadFile(a))}catch(u){console.warn(`[workflow] could not load ${e} from ${a}: ${u.message}`)}let i=mt(n);if(i===n)break;n=i}return s.length===0?null:s.every(a=>typeof a=="string")?s.join(`
29
+ ${n}`);let i=r(),u=i.cwd||process.cwd(),l=i.sessionPath;try{if(l){let p=ve(l,z);if(rt(p)){let h=JSON.parse(ot(p,"utf-8"));h.currentNode=this.name,$e(p,JSON.stringify(h,null,2),"utf-8")}let c=ve(l,"..",z);if(rt(c))try{let h=JSON.parse(ot(c,"utf-8"));h.currentNode=this.name,$e(c,JSON.stringify(h,null,2),"utf-8")}catch{}}}catch(p){y.debug(`[workflow] could not update session info: ${p.message}`)}let d=null;for(let p=0;p<=this.retries;p++)try{y.debug(`[workflow] node '${this.name}' attempt ${p}`);let c=r().config||{},h=c.agents||{},w=this.config.agent??h[this.name]??null,_={state:r()};w&&(_.preferredAgent=w);let $={workspace:u,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:l,config:c,nodeName:this.name,timeout:this.config?.timeout||3e5},b=e?._coreInvokeAgent;b||(b=(await Promise.resolve().then(()=>(ee(),Q))).invokeAgent);let f=await b(n,_,$),I,g;if(typeof f=="string"?(I=f,g=null):f.structured?(I=f.raw||JSON.stringify(f.structured,null,2),g=f.structured):(I=f.raw||JSON.stringify(f,null,2),g=f.extracted||null),l)try{let m=ve(l,this.name,"raw_stream_output.txt");Yt(Jt(m),{recursive:!0}),$e(m,typeof I=="string"?I:JSON.stringify(I),"utf-8")}catch(m){y.debug(`[workflow] could not save raw output: ${m.message}`)}if(this.isZodSchema&&g){y.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(g,null,2)}`);let m=g;if(typeof this.onComplete=="function")try{m=await this.onComplete(r(),g)}catch(R){y.warn(`[workflow] onComplete hook failed: ${R.message}`)}return{success:!0,output:m,raw:I}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(r(),{raw:I}),raw:I}}catch(m){throw new Error(`onComplete failed: ${m.message}`,{cause:m})}if(this.parser){let m=this.parser.parse(I);return y.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(m,null,2)}`),A.step("Output parsed"),{success:!0,output:m,raw:I}}return{success:!0,output:I,raw:I}}catch(c){d=c,p<this.retries&&y.info(`[workflow] node '${this.name}' failed, retrying (${p+1}/${this.retries})\u2026`)}return{success:!1,error:d.message,raw:null}}},de=class extends j{constructor(e){super({...e,_isCustomCode:!0}),this.condition=e.condition}async execute(e,t){let r=t&&typeof t.getAll=="function"?t.getAll():e;return{success:!0,output:{nextNode:this.condition(r)},raw:null}}};F();F();import{mkdirSync as Kt,existsSync as H,statSync as lt,readdirSync as dt,rmSync as qt}from"node:fs";import{spawn as ut}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as Vt}from"node:url";import{AsyncLocalStorage as Zt}from"node:async_hooks";var Te=new Zt;function te(){let o=Te.getStore();return o||Object.freeze({executionId:process.env.EXECUTION_ID||null,parentExecutionId:process.env.PARENT_EXECUTION_ID||null,depth:0,conversationId:process.env.ZIBBY_CONVERSATION_ID||null,dispatchMode:process.env.DISPATCH_MODE||null})}function nt(o,e){let t=Te.getStore()||te(),r=Object.freeze({executionId:o.executionId,parentExecutionId:o.parentExecutionId??t.executionId??null,depth:(t.depth||0)+(o.executionId!==t.executionId?1:0),conversationId:o.conversationId!==void 0?o.conversationId:t.conversationId??null,dispatchMode:o.dispatchMode??null});return Te.run(r,e)}var xe=new Map,Ae=new Map,st=new Map;function it(o,e,t={}){if(!o||typeof o!="string")throw new Error("subgraph-registry.register: name required");if(typeof e!="function")throw new Error("subgraph-registry.register: factory must be a function");xe.set(o,e),Ae.set(o,"ready"),st.set(o,{...t,cachedAt:Date.now()})}function at(o,e){Ae.set(o,"failed"),st.set(o,{error:e?.message||String(e),failedAt:Date.now()}),xe.delete(o)}function ct(o){return Ae.get(o)==="ready"?xe.get(o):null}var pe=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function Xt(){return`node${(process.versions?.node||"").split(".")[0]||"unknown"}-${process.platform}-${process.arch}`}var k=class extends Error{constructor(e,t){super(`in-process sub-graph fallback: ${e}${t?` (${t})`:""}`),this.fallback=!0,this.reason=e,this.detail=t||null,this.name="SubgraphFallback"}};function Qt(){let o=(process.env.SUBGRAPH_INTERNAL_URL||"").replace(/\/$/,""),e=(process.env.PROGRESS_API_URL||"").replace(/\/executions\/?$/,""),t=o||e,r=process.env.PROJECT_ID,s=process.env.PROJECT_API_TOKEN;if(!t||!r||!s)throw new k("env","SUBGRAPH_INTERNAL_URL/PROGRESS_API_URL/PROJECT_ID/PROJECT_API_TOKEN missing");return{apiBase:t,projectId:r,authToken:s}}async function eo({apiBase:o,authToken:e,body:t}){let r;try{r=await fetch(`${o}/internal/subgraph/begin`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(t)})}catch(n){throw new k("network",`begin fetch failed: ${n.message}`)}let s=null;try{s=await r.json()}catch{}if(!r.ok){if(r.status===404){let n=new Error(`Sub-graph child '${t.childWorkflowType}' not found in project`);throw n.code="SUBGRAPH_NOT_FOUND",n.status=404,n}if(r.status===429){let n=s?.quotaInfo||{},a=new Error(`Sub-graph blocked by quota (${n.used??"?"}/${n.limit??"?"} on ${n.planId||"plan"})`);throw a.code="SUBGRAPH_QUOTA_EXCEEDED",a.status=429,a.quotaInfo=n,a}if(r.status===400&&s?.validationErrors){let n=new Error(`Sub-graph rejected input: ${s?.error||s?.message||"validation failed"}`);throw n.code="SUBGRAPH_INVALID_INPUT",n.status=400,n.validationErrors=s.validationErrors,n.missing=s.missing,n}throw new k("begin-status",`begin returned ${r.status}`)}return s?.data||s}async function G({apiBase:o,authToken:e,payload:t}){try{let r=await fetch(`${o}/internal/subgraph/finalize`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(t)});r.ok||y.warn(`[in-process subgraph] finalize returned ${r.status} for ${t.childExecutionId}`)}catch(r){y.warn(`[in-process subgraph] finalize failed: ${r.message}`)}}async function to(o,e){let t=U(e,".ready"),r=U(e,"graph.mjs");if(H(t)&&H(r))return;Kt(e,{recursive:!0});let s=U(e,".lock"),n=!1;try{let{openSync:a,closeSync:i}=await import("node:fs"),u=a(s,"wx");i(u),n=!0}catch(a){if(a.code!=="EEXIST")throw a}if(!n){let a=Date.now()+3e4;for(;Date.now()<a;){if(H(t)&&H(r))return;await new Promise(i=>setTimeout(i,100))}throw new k("bundle-extract-timeout","sibling extract did not complete within 30s")}try{await new Promise((u,l)=>{let d=ut("curl",["-fsSL",o],{stdio:["ignore","pipe","inherit"]}),p=ut("tar",["-xzf","-","-C",e],{stdio:["pipe","inherit","inherit"]});d.stdout.pipe(p.stdin);let c,h,w=()=>{if(c!==void 0&&h!==void 0){if(c!==0)return l(new Error(`curl exited ${c}`));if(h!==0)return l(new Error(`tar exited ${h}`));u()}};d.on("close",_=>{c=_,w()}),p.on("close",_=>{h=_,w()}),d.on("error",l),p.on("error",l)});let{writeFileSync:a,unlinkSync:i}=await import("node:fs");a(t,"");try{i(s)}catch{}}catch(a){try{let{unlinkSync:i}=await import("node:fs");i(s)}catch{}throw new k("bundle-extract-failed",a.message)}}async function oo(o){let e=U(o,"graph.mjs");if(!H(e))throw new k("entry-missing",`graph.mjs missing under ${o}`);let t;try{t=await import(Vt(e).href)}catch(s){throw new k("import-failed",`${s?.code||s?.name||"unknown"}: ${s.message}`)}let r=t.default||Object.values(t).find(s=>typeof s=="function"&&s.prototype?.buildGraph);if(!r)throw new k("entry-class-missing","no buildGraph() class export found");return r}async function pt(o,e={}){if(!o||typeof o!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let t=te(),r;try{r=Qt()}catch(g){throw g}y.debug(`[in-process subgraph] begin '${o}' parent=${t.executionId||"<root>"}`);let s=await eo({apiBase:r.apiBase,authToken:r.authToken,body:{parentExecutionId:t.executionId,childWorkflowType:o,input:e.input||{},...e.conversationId?{conversationId:e.conversationId}:{}}}),{childExecutionId:n,runtimeTag:a,bundlePresignedUrl:i,sourcesPresignedUrl:u,workflowVersion:l,workflowUuid:d,bundleReady:p}=s,c=Xt();if(a&&a!==c)throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:`runtimeTag mismatch: parent=${c} child=${a}`,code:"RUNTIME_MISMATCH"}}}),new k("runtime-mismatch",`${c} vs ${a}`);if(!p||!i)throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:"bundle not ready for in-process; falling back to HTTP",code:"NO_BUNDLE"}}}),new k("no-bundle","workflow bundle not built yet");let h=ct(o);if(!h){let g=U(pe,`${d}@${l||"0"}`);try{await to(i,g);try{no()}catch{}}catch(m){throw m.fallback&&await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"failed",error:{message:m.message,code:m.reason}}}),m}try{h=await oo(g),it(o,h,{workflowUuid:d,version:l,runtimeTag:a,cacheDir:g})}catch(m){throw at(o,m),await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"failed",error:{message:m.message,code:m.reason||"IMPORT_FAILED"}}}),m.fallback?m:new k("import-failed",m.message)}}let w=Date.now(),$=await(typeof h=="function"&&h.prototype?.buildGraph?new h:h).buildGraph(),b={...e.input||{}},f,I;try{f=await nt({executionId:n,parentExecutionId:t.executionId,conversationId:e.conversationId!==void 0?e.conversationId:t.conversationId,dispatchMode:"inprocess"},()=>$.run(e.parentAgent,b,{signal:e.signal})),I=f&&typeof f=="object"&&"state"in f?f.state:f}catch(g){throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"failed",error:{message:g.message,code:g.code||"CHILD_THREW",stack:g.stack},durationMs:Date.now()-w}}),g}if(f&&typeof f=="object"&&f.stoppedExternally){await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",finalState:I,durationMs:Date.now()-w}});let g=new Error(`Sub-graph '${o}' canceled by parent abort`);throw g.code="SUBGRAPH_CANCELED",g.subgraphJobId=n,g}return await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"completed",finalState:I,durationMs:Date.now()-w}}),{finalState:I,executionId:n}}function ro(o){let e=0,t=[o];for(;t.length;){let r=t.pop(),s;try{s=lt(r)}catch{continue}if(s.isDirectory()){let n;try{n=dt(r)}catch{continue}for(let a of n)t.push(U(r,a))}else e+=s.size}return e}function no({cap:o=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!H(pe))return{evicted:0,freedBytes:0};let e=dt(pe),t=[],r=0;for(let i of e){let u=U(pe,i),l;try{l=lt(u)}catch{continue}let d=l.isDirectory()?ro(u):l.size;r+=d,t.push({name:i,full:u,size:d,mtimeMs:l.mtimeMs})}if(r<=o)return{evicted:0,freedBytes:0,totalBytes:r};t.sort((i,u)=>i.mtimeMs-u.mtimeMs);let s=Math.floor(o*.7),n=0,a=0;for(let i of t){if(r-n<=s)break;if(!H(U(i.full,".lock")))try{qt(i.full,{recursive:!0,force:!0}),n+=i.size,a+=1}catch(u){y.debug(`[sub-graph cache] evict skip ${i.name}: ${u.message}`)}}return a>0&&y.info(`[sub-graph cache] evicted ${a} entr(y/ies), freed ${(n/1024/1024).toFixed(1)}MB`),{evicted:a,freedBytes:n,totalBytes:r-n}}catch(e){return y.debug(`[sub-graph cache] evict failed: ${e.message}`),{evicted:0,freedBytes:0}}}var so=2e3,io=600*1e3,ao=new Set(["completed","failed","canceled","timeout"]);function co(){let o=process.env.PROGRESS_API_URL;if(!o)throw new Error("Sub-graph dispatch requires PROGRESS_API_URL env var (set automatically on cloud runs). Sub-graphs are not supported in local in-process runs yet \u2014 deploy the parent and child to cloud.");return o.replace(/\/executions\/?$/,"")}function uo(){let o=process.env.PROJECT_ID;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return o}function lo(){let o=process.env.PROJECT_API_TOKEN;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return o}function po(){return process.env.EXECUTION_ID||null}function ft(o,e){return e==null?o:typeof e=="function"?e(o):typeof e=="string"?e.split(".").reduce((t,r)=>t==null?t:t[r],o):o}async function ht(o,e={}){if(!o||typeof o!="string")throw new Error("dispatchSubgraph: workflowName (string) is required");let t=te(),r=Number(process.env.ZIBBY_SUBGRAPH_MAX_DEPTH||10);if((t.depth||0)>=r)throw new Error(`dispatchSubgraph('${o}'): sub-graph depth ${t.depth} reached cap of ${r}. Restructure the graph or raise ZIBBY_SUBGRAPH_MAX_DEPTH.`);if(process.env.ZIBBY_INPROCESS_SUBGRAPH!=="0"&&!e.async)try{y.debug(`[sub-graph] trying in-process for '${o}'`);let{finalState:g}=await pt(o,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),m=ft(g,e.output);return y.info(`[sub-graph] '${o}' completed in-process`),m}catch(g){if(g instanceof k||g?.fallback)y.info(`[sub-graph] in-process fallback for '${o}': ${g.reason||"unknown"} \u2014 using HTTP`);else throw g}let s=co(),n=uo(),a=lo(),i=po(),u=`${s}/projects/${encodeURIComponent(n)}/workflows/${encodeURIComponent(o)}/trigger`,l={input:e.input||{},...i?{parentExecutionId:i}:{},...e.conversationId?{conversationId:e.conversationId}:{}};y.info(`[sub-graph] dispatching '${o}' (${e.async?"async":"sync"}) from parent ${i||"<none>"}`);let d=await fetch(u,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${a}`},body:JSON.stringify(l)});if(!d.ok){let g=null,m="";try{g=await d.json(),m=g?.error||g?.message||JSON.stringify(g)}catch{m=await d.text().catch(()=>"")}if(d.status===429){let T=g?.quotaInfo||{},M=new Error(`Sub-graph '${o}' blocked by execution quota (${T.used??"?"}/${T.limit??"?"} on plan ${T.planId||"unknown"}). Sub-workflow runs count toward the same monthly cap as user-triggered runs.`);throw M.code="SUBGRAPH_QUOTA_EXCEEDED",M.status=429,M.subgraph=o,M.quotaInfo=T,M}if(d.status===400){let T=new Error(`Sub-graph '${o}' rejected input: ${m}`);throw T.code="SUBGRAPH_INVALID_INPUT",T.status=400,T.subgraph=o,T.validationErrors=g?.validationErrors||null,T.missing=g?.missing||null,T}let R=new Error(`Sub-graph '${o}' trigger rejected (${d.status}): ${m}`);throw R.code="SUBGRAPH_TRIGGER_FAILED",R.status=d.status,R.subgraph=o,R}let p=await d.json(),c=p?.data?.jobId||p?.jobId;if(!c)throw new Error(`Sub-graph '${o}' trigger returned no jobId: ${JSON.stringify(p).slice(0,200)}`);if(e.async)return y.info(`[sub-graph] async dispatch of '${o}' \u2192 jobId=${c} (not waiting)`),{jobId:c,status:"accepted",workflow:o};let h=Number.isFinite(e.timeoutMs)?e.timeoutMs:io,w=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:so,_=`${s}/executions/${encodeURIComponent(c)}`,$=Date.now()+h,b="accepted",f=0;for(;Date.now()<$;){await new Promise(T=>setTimeout(T,w)),f+=1;let g=await fetch(_,{headers:{Authorization:`Bearer ${a}`}});if(!g.ok){if(g.status>=500){y.warn(`[sub-graph] status poll for ${c} returned ${g.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${c}: ${g.status}`)}let m=await g.json(),R=m?.data||m?.execution||m;if(b=R?.status||b,ao.has(b)){if(b!=="completed"){let S=new Error(`Sub-graph '${o}' (${c}) ended in status '${b}'`);throw S.subgraphJobId=c,S.subgraphStatus=b,S}let T=R?.finalState||R?.state||{},M=ft(T,e.output);return y.info(`[sub-graph] '${o}' (${c}) completed after ${f} polls`),M}}let I=new Error(`Sub-graph '${o}' (${c}) timed out after ${Math.round(h/1e3)}s (last status: ${b})`);throw I.subgraphJobId=c,I.subgraphStatus=b,I}import{existsSync as gt,readFileSync as fo}from"node:fs";import{join as ke,dirname as mt}from"node:path";var fe=class{static async loadContext(e,t,r={}){let s={},n=r.filenames||["CONTEXT.md","AGENTS.md"];if(e){let i=mt(ke(t,e));for(let u of n){let l=await this.findAndMergeContextFiles(u,i,t);if(l){let d=u.replace(/\.[^.]+$/,"").toLowerCase();s[d]=l}}}let a=r.discovery||{};for(let[i,u]of Object.entries(a))try{let l=ke(t,u);gt(l)&&(s[i]=await this.loadFile(l))}catch(l){console.warn(`[workflow] could not load context '${i}' from '${u}': ${l.message}`)}return s}static async findAndMergeContextFiles(e,t,r){let s=[],n=t;for(;n.startsWith(r);){let a=ke(n,e);if(gt(a))try{s.unshift(await this.loadFile(a))}catch(u){console.warn(`[workflow] could not load ${e} from ${a}: ${u.message}`)}let i=mt(n);if(i===n)break;n=i}return s.length===0?null:s.every(a=>typeof a=="string")?s.join(`
30
30
 
31
31
  ---
32
32
 
33
- `):s.every(a=>typeof a=="object")?Object.assign({},...s):s[s.length-1]}static async loadFile(e){let t=fo(e,"utf-8");if(e.endsWith(".json"))return JSON.parse(t);if(e.endsWith(".js")||e.endsWith(".mjs")){let{pathToFileURL:r}=await import("url"),s=await import(r(e).href);return s.default||s}return t}};import{mkdirSync as yt,existsSync as Oe,writeFileSync as St,unlinkSync as ho}from"node:fs";import{join as J,resolve as _t}from"node:path";import{config as go}from"dotenv";import{zodToJsonSchema as wt}from"zod-to-json-schema";import mo from"handlebars";function So({traceFrom:o,sessionId:e,sessionPath:t,idSource:r,mkdirFresh:s}){if(!(process.env.ZIBBY_SESSION_LOG==="1"||process.env.ZIBBY_SESSION_LOG==="true"))return;let a=typeof process.ppid=="number"?process.ppid:"n/a",i=`[zibby:session] from=${o} pid=${process.pid} ppid=${a} sessionId=${e} source=${r} mkdir=${s?"yes":"no"} path=${t}`;if(console.log(i),process.env.ZIBBY_TRACE_SESSION==="1"||process.env.ZIBBY_TRACE_SESSION==="true"){let d=(new Error("session trace").stack||"").split(`
33
+ `):s.every(a=>typeof a=="object")?Object.assign({},...s):s[s.length-1]}static async loadFile(e){let t=fo(e,"utf-8");if(e.endsWith(".json"))return JSON.parse(t);if(e.endsWith(".js")||e.endsWith(".mjs")){let{pathToFileURL:r}=await import("url"),s=await import(r(e).href);return s.default||s}return t}};import{mkdirSync as yt,existsSync as Oe,writeFileSync as St,unlinkSync as ho}from"node:fs";import{join as Y,resolve as _t}from"node:path";import{config as go}from"dotenv";import{zodToJsonSchema as wt}from"zod-to-json-schema";import mo from"handlebars";function So({traceFrom:o,sessionId:e,sessionPath:t,idSource:r,mkdirFresh:s}){if(!(process.env.ZIBBY_SESSION_LOG==="1"||process.env.ZIBBY_SESSION_LOG==="true"))return;let a=typeof process.ppid=="number"?process.ppid:"n/a",i=`[zibby:session] from=${o} pid=${process.pid} ppid=${a} sessionId=${e} source=${r} mkdir=${s?"yes":"no"} path=${t}`;if(console.log(i),process.env.ZIBBY_TRACE_SESSION==="1"||process.env.ZIBBY_TRACE_SESSION==="true"){let d=(new Error("session trace").stack||"").split(`
34
34
  `).slice(2,14).join(`
35
35
  `);console.log(`[zibby:session] stack (${o}):
36
- ${d}`)}}function wo(){return process.env.ZIBBY_TRUST_SESSION_ENV==="1"||process.env.ZIBBY_TRUST_SESSION_ENV==="true"||process.env.ZIBBY_KEEP_SESSION_ENV==="1"||process.env.ZIBBY_KEEP_SESSION_ENV==="true"}function yo(){if(!(process.env.ZIBBY_PIN_SESSION_PATH==="1"||process.env.ZIBBY_PIN_SESSION_PATH==="true"))return;let e=process.env.ZIBBY_SESSION_PATH;if(!(e==null||String(e).trim()===""))try{return _t(String(e).trim())}catch{return String(e).trim()}}function _o(){wo()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function Io({sessionPath:o,sessionId:e}){o&&typeof o=="string"&&(process.env.ZIBBY_SESSION_PATH=o),e!=null&&String(e).trim()!==""&&(process.env.ZIBBY_SESSION_ID=String(e).trim())}function Eo(o={}){let e=Xe.map(n=>process.env[n]).find(Boolean),t=Math.random().toString(36).slice(2,6),r=e||`${Date.now()}_${t}`,s=o.paths?.sessionPrefix;return s?`${s}_${r}`:r}function bo({cwd:o=process.cwd(),config:e={},initialState:t={},traceFrom:r="resolveWorkflowSession"}={}){let s=t.sessionPath,n=t.sessionTimestamp,a="initialState.sessionPath";if(!s&&process.env.ZIBBY_SESSION_PATH)try{let l=_t(String(process.env.ZIBBY_SESSION_PATH));l&&(s=l,a="ZIBBY_SESSION_PATH")}catch{}let i;if(s)i=String(s).split(/[/\\]/).filter(Boolean).pop(),n==null&&(n=Date.now());else{let l=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(l)i=l,a="ZIBBY_SESSION_ID";else{let p=e.sessionId!=null?String(e.sessionId).trim():"";p&&p!=="last"?(i=p,a="config.sessionId"):(i=Eo(e),a="generated")}n=n??Date.now();let d=e.paths?.output||ue;s=J(o,d,qe,i)}let u=!Oe(s);return u&&yt(s,{recursive:!0}),(u||a!=="initialState.sessionPath")&&So({traceFrom:r,sessionId:i,sessionPath:s,idSource:a,mkdirFresh:u}),Io({sessionPath:s,sessionId:i}),{sessionPath:s,sessionId:i,sessionTimestamp:n}}var he=class{constructor(e={}){this.nodes=new Map,this.edges=new Map,this.entryPoint=null,this.middleware=Array.isArray(e.middleware)?[...e.middleware]:[],e.nodeMiddleware&&this.middleware.push(e.nodeMiddleware),this.nodeTypeMap=new Map,this.conditionalCodeMap=new Map,this.stateSchema=e.stateSchema||null,this.inputSchema=e.inputSchema||null,this.contextSchema=e.contextSchema||null,this.nodePrompts=new Map,this.nodeOptions=new Map,this._invokeAgent=e.invokeAgent||null,this._compiledPrompts=new Map}setInputSchema(e){return this.inputSchema=e,this}setContextSchema(e){return this.contextSchema=e,this}setStateSchema(e){return this.stateSchema=e,this}getInputSchema(){return this.inputSchema}getContextSchema(){return this.contextSchema}getStateSchema(){return this.stateSchema}_runtimeSchema(){if(this.inputSchema&&this.contextSchema)try{return this.inputSchema.merge(this.contextSchema)}catch{}return this.inputSchema&&!this.contextSchema?this.inputSchema:this.stateSchema}addNode(e,t,r={}){if(!(t instanceof j)&&t&&typeof t=="object"&&typeof t.workflow=="string"){let n=t,a={name:e,_isCustomCode:!0,retries:n.retries,onComplete:n.onComplete,execute:async u=>{let l=u?.state&&typeof u.state.getAll=="function"?u.state.getAll():u,d;return typeof n.input=="function"?d=n.input(l):n.input&&typeof n.input=="object"?d=n.input:d={},ht(n.workflow,{input:d,async:n.async===!0,conversationId:typeof n.conversationId=="function"?n.conversationId(l):n.conversationId,output:n.output,timeoutMs:n.timeoutMs,pollIntervalMs:n.pollIntervalMs,signal:l?._signal,parentAgent:u?.agent})}},i=new j(a);return i.name=e,this.nodes.set(e,i),r.prompt&&this.nodePrompts.set(e,r.prompt),Object.keys(r).length>0&&this.nodeOptions.set(e,r),this}let s=t instanceof j?t:new j(t);return s.name=e,this.nodes.set(e,s),r.prompt&&this.nodePrompts.set(e,r.prompt),Object.keys(r).length>0&&this.nodeOptions.set(e,r),this}addConditionalNode(e,t){return this.nodes.set(e,new de({...t,name:e})),this}addEdge(e,t){return this.edges.set(e,t),this}setNodeType(e,t){return this.nodeTypeMap.set(e,t),this}addConditionalEdges(e,t,{labels:r}={}){return this.edges.set(e,{conditional:!0,routes:t,labels:r}),typeof t=="function"&&this.conditionalCodeMap.set(e,t.toString()),this}setEntryPoint(e){return this.entryPoint=e,this}use(e){return typeof e=="function"&&this.middleware.push(e),this}_composeMiddleware(e,t,r,s,n){let a=r;for(let i=e.length-1;i>=0;i--){let u=e[i],l=a;a=()=>u(t,l,s,n)}return a()}serialize(){let e=[],t={};for(let[l,d]of this.nodes){let p=this.nodeTypeMap.get(l)||l;e.push({id:l,type:p,data:{nodeType:p,label:l}});let c={};d._isCustomCode&&typeof d.execute=="function"&&(c.customCode=d.execute.toString());let h=this.nodePrompts.get(l);if(h&&(c.prompt=h),typeof d.customExecute=="function"&&(c.executeCode=d.customExecute.toString()),d.outputSchema)try{if(typeof d.outputSchema._def<"u"){let $=wt(d.outputSchema,{target:"openApi3"});c.outputSchema={jsonSchema:$,variables:this._flattenJsonSchemaToVariables($)}}else c.outputSchema={schema:d.outputSchema}}catch($){console.warn(`[workflow] failed to convert schema for ${l}:`,$.message)}let w=(this.resolvedToolsMap||{})[l];w?.toolIds&&(c.tools=w.toolIds);let _=Array.isArray(d?.config?.skills)?d.config.skills:Array.isArray(d?.skills)?d.skills:null;_&&_.length>0&&(c.skills=[..._]),Object.keys(c).length>0&&(t[l]=c)}let r=[];for(let[l,d]of this.edges)if(typeof d=="string")r.push({source:l,target:d});else if(d.conditional){let p=this.conditionalCodeMap.get(l)||d.routes.toString(),c=this._inferConditionalTargets(d.routes),h=d.labels||{};for(let w of c){let _={source:l,target:w,data:{conditionalCode:p}};h[w]&&(_.label=h[w]),r.push(_)}}let s=l=>{if(!l)return null;try{return wt(l,{target:"openApi3"})}catch{return null}},n=this._runtimeSchema(),a=s(n||this.stateSchema),i=s(this.inputSchema),u=s(this.contextSchema);return{nodes:e,edges:r,nodeConfigs:t,stateSchema:a,inputSchema:i,contextSchema:u}}_inferConditionalTargets(e){let t=e.toString(),r=new Set,s=/return\s+['"]([^'"]+)['"]/g,n;for(;(n=s.exec(t))!==null;)r.add(n[1]);return[...r]}_flattenJsonSchemaToVariables(e,t=""){let r=e;if(e.$ref&&e.definitions){let s=e.$ref.replace("#/definitions/","");r=e.definitions[s]||e}return this._flattenSchema(r,t)}_flattenSchema(e,t=""){if(!e||typeof e!="object")return[];let r=[],s=e.properties||{},n=e.required||[];for(let[a,i]of Object.entries(s)){let u=t?`${t}.${a}`:a;r.push({path:u,type:i.type||"unknown",label:i.description||this._formatLabel(a),optional:!n.includes(a)}),i.type==="object"&&i.properties&&r.push(...this._flattenSchema(i,u)),i.type==="array"&&i.items?.type==="object"&&i.items.properties&&r.push(...this._flattenSchema(i.items,`${u}[]`))}return r}_formatLabel(e){return e.replace(/([A-Z])/g," $1").replace(/^./,t=>t.toUpperCase()).trim()}_summarizeNodeOutput(e,t){if(!t||typeof t!="object")return[];let r=[];t.success!==void 0&&r.push(`Result: ${t.success?"passed":"failed"}`);for(let[s,n]of Object.entries(t))if(!(s==="success"||s==="raw"||s==="nextNode")){if(typeof n=="string"&&n.length<=80)r.push(`${s}: ${n}`);else if(Array.isArray(n)){let a=n.length,i=n.filter(l=>l?.passed===!0).length,u=n.some(l=>l?.passed!==void 0);r.push(u?`${s}: ${i}/${a} passed${a-i?`, ${a-i} failed`:""}`:`${s}: ${a} items`)}if(r.length>=4)break}return r}async run(e,t={},r={}){if(!this.entryPoint)throw new Error("No entry point set for graph");let s=new AbortController;r.signal&&(r.signal.aborted?s.abort():r.signal.addEventListener("abort",()=>s.abort(),{once:!0}));let n=r.strategyAbortTimeoutMs??t.config?.strategyAbortTimeoutMs??5e3,a=t.cwd||process.cwd();go({path:J(a,".env")});let i=t.config||{};if(!i||Object.keys(i).length===0)try{let E=J(a,".zibby.config.js");Oe(E)&&(i=(await import(E)).default||{})}catch{}process.env.EXECUTION_ID&&!i.agent?.strictMode&&(i.agent={...i.agent,strictMode:!0});let u=t.agentType;if(!u){let E=i?.agent;E?.provider?u=E.provider:E?.gemini?u="gemini":E?.claude?u="claude":E?.cursor?u="cursor":E?.codex?u="codex":u=process.env.AGENT_TYPE||"cursor"}let l=t.contextConfig||e?.config?.contextConfig||e?.config?.context||i?.context||{},d=this._runtimeSchema();if(d){let E=d.safeParse(t);if(!E.success){let O=E.error.issues.map(N=>`${N.path.join(".")}: ${N.message}`);throw console.error("\u274C Initial state validation failed:"),O.forEach(N=>console.error(` - ${N}`)),new Error(`State validation failed: ${O.join(", ")}`)}A.step("State validated against schema")}let p=yo(),c=t.sessionPath||p;c||_o();let{sessionPath:h,sessionTimestamp:w,sessionId:_}=bo({cwd:a,config:i,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:c,sessionTimestamp:t.sessionTimestamp}});A.step(`Session ${_}`);let $=await fe.loadContext(t.specPath||"",a,l);Object.keys($).length>0&&A.step(`Context loaded: ${Object.keys($).join(", ")}`);let b=t.outputPath;!b&&t.specPath&&(e?.calculateOutputPath?b=e.calculateOutputPath(t.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${t.specPath})`));let f=new ie({...t,config:i,agentType:u,outputPath:b,sessionPath:h,sessionTimestamp:w,context:$,resolvedTools:this.resolvedToolsMap||{},_signal:s.signal}),I=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:g}=await Promise.resolve().then(()=>(le(),et)),m=i.skills&&typeof i.skills=="object"?i.skills:{},R=Object.values(m).filter(E=>E&&typeof E=="object"&&typeof E.id=="string"),T=E=>{for(let O of R)if(O.id===E)return O;return g(E)},M=new Set;for(let[,E]of this.nodes)for(let O of E.config?.skills||[])M.add(O);for(let E of M){let O=T(E);if(typeof O?.middleware=="function")try{let N=await O.middleware();typeof N=="function"&&I.set(E,N)}catch{}}let S=this.entryPoint,oe=[],Re=i?.recursionLimit??100,vt=0;try{for(;S&&S!=="END";){if(++vt>Re)throw new Error(`Workflow exceeded recursion limit (${Re}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let O=J(h,Ve);if(Oe(O)){try{ho(O)}catch{}s.abort()}if(s.signal.aborted)return console.warn(`
37
- \u{1F6D1} External stop requested \u2014 ending workflow.`),A.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:oe,stoppedExternally:!0};let N=this.nodes.get(S);if(!N)throw new Error(`Node '${S}' not found in graph`);let Be=JSON.stringify({sessionPath:h,sessionTimestamp:w,currentNode:S,createdAt:new Date().toISOString(),config:f.get("config")}),Tt=J(h,z);St(Tt,Be,"utf-8");let Me=f.get("config")?.paths?.output||ue,xt=J(a,Me,z);yt(J(a,Me),{recursive:!0});try{St(xt,Be,"utf-8")}catch{}let De=t.onPipelineProgress;if(typeof De=="function")try{De({cwd:a,sessionPath:h,sessionId:_,outputBase:f.get("config")?.paths?.output||ue,currentNode:S})}catch{}let At=(this.resolvedToolsMap||{})[S]||null;f.set("_currentNodeTools",At);let kt=f.get("nodeConfigs")||{};f.set("_currentNodeConfig",kt[S]||{}),A.nodeStart(S);let Le=Date.now(),re=this.nodePrompts.get(S);if(!this._invokeAgent){let C=await Promise.resolve().then(()=>(ee(),Q));this._invokeAgent=C.invokeAgent}let Ot=this._invokeAgent,me={},Nt=N.config?.skills||[];for(let C of Nt){let P=T(C);if(typeof P?.invokeAgentOptions=="function")try{let v=P.invokeAgentOptions(f.getAll(),{agentType:f.get("agentType"),nodeName:S});v&&typeof v=="object"&&(me={...me,...v})}catch(v){console.warn(`[graph] skill '${C}' invokeAgentOptions threw: ${v.message}`)}}let je=async(C,P,v={})=>{let B=Ot(C,P,{...me,...v,signal:s.signal});return B.catch(()=>{}),s.signal.aborted?B:Promise.race([B,new Promise((Y,Z)=>{let L=()=>{setTimeout(()=>{let q=new Error(`Strategy ignored AbortSignal \u2014 engine deadman fired after ${n}ms`);q.name="AbortError",Z(q)},n)};s.signal.addEventListener("abort",L,{once:!0})})])},Ue={state:f,invokeAgent:async(C={},P={})=>{let v=P.prompt||"";if(re){let B=this._compiledPrompts.get(S);B||(B=mo.compile(re,{noEscape:!0}),this._compiledPrompts.set(S,B));try{v=B(C)}catch(Y){throw console.error(`\u274C Template rendering failed for node '${S}':`,Y.message),new Error(`Template rendering failed: ${Y.message}`,{cause:Y})}}else if(!v)throw new Error(`No prompt template configured for node '${S}' and no prompt provided in options`);return je(v,{state:f.getAll(),images:P.images||[]},{model:P.model||f.get("model"),workspace:f.get("workspace"),schema:P.schema,...P,signal:s.signal})},_coreInvokeAgent:je,agent:e,nodeId:S,promptTemplate:re,getPromptTemplate:()=>re,...f.getAll()};try{let C=(N.config?.skills||[]).map(L=>I.get(L)).filter(Boolean),P=[...this.middleware,...C],v;P.length>0?v=await this._composeMiddleware(P,S,async()=>N.execute(Ue,f),f.getAll(),f):v=await N.execute(Ue,f);let B=Date.now()-Le;if(oe.push({node:S,success:v.success,duration:B,timestamp:new Date().toISOString()}),!v.success){if(s.signal.aborted)return A.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:oe,stoppedExternally:!0};f.append("errors",{node:S,error:v.error});let L=N.config?.retries||0,q=`${S}_retries`,ne=f.getAll()[q]||0;if(ne<L){A.stepInfo(`Retrying (attempt ${ne+1}/${L})`),f.update({[q]:ne+1,[`${S}_raw`]:v.raw});continue}throw A.nodeFailed(S,v.error,{duration:B}),new Error(`Node '${S}' failed after ${ne} attempts: ${v.error}`)}f.update({[S]:v.output});let Y=this._summarizeNodeOutput(S,v.output);A.nodeComplete(S,{duration:B,details:Y});let Z=this.edges.get(S);if(!Z)S="END";else if(Z.conditional){let L=Z.routes(f.getAll());A.route(S,L),S=L}else S=Z}catch(C){throw A.isInsideNode&&A.nodeFailed(S,C.message,{duration:Date.now()-Le}),f.set("failed",!0),f.set("failedAt",S),C}}A.graphComplete();let E={success:!0,state:f.getAll(),executionLog:oe};return e&&typeof e.onComplete=="function"&&await e.onComplete(E),E}finally{if(e&&typeof e.cleanup=="function")try{await e.cleanup()}catch(E){console.warn(`[workflow] agent.cleanup() failed: ${E.message}`)}}}};var Ne=Symbol.for("@zibby/agent-workflow.nodes");globalThis[Ne]||(globalThis[Ne]=new Map);var Ce=globalThis[Ne];function $o(o,e){Ce.set(o,e)}function It(o){return Ce.get(o)}function Pe(o){return Ce.has(o)}$o("ai_agent",{name:"ai_agent",factory:!0,create:(o,e={})=>({name:o,_isCustomCode:!0,execute:async t=>{let r=t?._coreInvokeAgent;r||(r=(await Promise.resolve().then(()=>(ee(),Q))).invokeAgent);let s=e.extraPromptInstructions||"Execute the task based on the current state.",n=vo(s,t),a=await r(n,{cwd:t.workspace||process.cwd(),model:t.model,tools:e.resolvedTools||null});return{success:!0,output:{raw:a,nodeId:o},raw:typeof a=="string"?a:a.raw}}})});function vo(o,e){let t=/@([\w.]+)/g,r=new Set,s;for(;(s=t.exec(o))!==null;)r.add(s[1]);if(r.size===0)return o;let n=[],a=new Set;for(let i of r){let u=i.split(".")[0];if(a.has(u))continue;let l=i.split(".").reduce((c,h)=>c?.[h],e);if(l===void 0)continue;let d=typeof l=="string"?l:l?.raw??JSON.stringify(l,null,2),p=i.replace(/_/g," ").replace(/\b\w/g,c=>c.toUpperCase());n.push(`## ${p}
36
+ ${d}`)}}function wo(){return process.env.ZIBBY_TRUST_SESSION_ENV==="1"||process.env.ZIBBY_TRUST_SESSION_ENV==="true"||process.env.ZIBBY_KEEP_SESSION_ENV==="1"||process.env.ZIBBY_KEEP_SESSION_ENV==="true"}function yo(){if(!(process.env.ZIBBY_PIN_SESSION_PATH==="1"||process.env.ZIBBY_PIN_SESSION_PATH==="true"))return;let e=process.env.ZIBBY_SESSION_PATH;if(!(e==null||String(e).trim()===""))try{return _t(String(e).trim())}catch{return String(e).trim()}}function _o(){wo()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function Io({sessionPath:o,sessionId:e}){o&&typeof o=="string"&&(process.env.ZIBBY_SESSION_PATH=o),e!=null&&String(e).trim()!==""&&(process.env.ZIBBY_SESSION_ID=String(e).trim())}function Eo(o={}){let e=Xe.map(n=>process.env[n]).find(Boolean),t=Math.random().toString(36).slice(2,6),r=e||`${Date.now()}_${t}`,s=o.paths?.sessionPrefix;return s?`${s}_${r}`:r}function bo({cwd:o=process.cwd(),config:e={},initialState:t={},traceFrom:r="resolveWorkflowSession"}={}){let s=t.sessionPath,n=t.sessionTimestamp,a="initialState.sessionPath";if(!s&&process.env.ZIBBY_SESSION_PATH)try{let l=_t(String(process.env.ZIBBY_SESSION_PATH));l&&(s=l,a="ZIBBY_SESSION_PATH")}catch{}let i;if(s)i=String(s).split(/[/\\]/).filter(Boolean).pop(),n==null&&(n=Date.now());else{let l=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(l)i=l,a="ZIBBY_SESSION_ID";else{let p=e.sessionId!=null?String(e.sessionId).trim():"";p&&p!=="last"?(i=p,a="config.sessionId"):(i=Eo(e),a="generated")}n=n??Date.now();let d=e.paths?.output||ue;s=Y(o,d,qe,i)}let u=!Oe(s);return u&&yt(s,{recursive:!0}),(u||a!=="initialState.sessionPath")&&So({traceFrom:r,sessionId:i,sessionPath:s,idSource:a,mkdirFresh:u}),Io({sessionPath:s,sessionId:i}),{sessionPath:s,sessionId:i,sessionTimestamp:n}}var he=class{constructor(e={}){this.nodes=new Map,this.edges=new Map,this.entryPoint=null,this.middleware=Array.isArray(e.middleware)?[...e.middleware]:[],e.nodeMiddleware&&this.middleware.push(e.nodeMiddleware),this.nodeTypeMap=new Map,this.conditionalCodeMap=new Map,this.stateSchema=e.stateSchema||null,this.inputSchema=e.inputSchema||null,this.contextSchema=e.contextSchema||null,this.nodePrompts=new Map,this.nodeOptions=new Map,this._invokeAgent=e.invokeAgent||null,this._compiledPrompts=new Map}setInputSchema(e){return this.inputSchema=e,this}setContextSchema(e){return this.contextSchema=e,this}setStateSchema(e){return this.stateSchema=e,this}getInputSchema(){return this.inputSchema}getContextSchema(){return this.contextSchema}getStateSchema(){return this.stateSchema}_runtimeSchema(){if(this.inputSchema&&this.contextSchema)try{return this.inputSchema.merge(this.contextSchema)}catch{}return this.inputSchema&&!this.contextSchema?this.inputSchema:this.stateSchema}addNode(e,t,r={}){if(!(t instanceof j)&&t&&typeof t=="object"&&typeof t.workflow=="string"){let n=t,a={name:e,_isCustomCode:!0,retries:n.retries,onComplete:n.onComplete,execute:async u=>{let l=u?.state&&typeof u.state.getAll=="function"?u.state.getAll():u,d;return typeof n.input=="function"?d=n.input(l):n.input&&typeof n.input=="object"?d=n.input:d={},ht(n.workflow,{input:d,async:n.async===!0,conversationId:typeof n.conversationId=="function"?n.conversationId(l):n.conversationId,output:n.output,timeoutMs:n.timeoutMs,pollIntervalMs:n.pollIntervalMs,signal:l?._signal,parentAgent:u?.agent})}},i=new j(a);return i.name=e,this.nodes.set(e,i),r.prompt&&this.nodePrompts.set(e,r.prompt),Object.keys(r).length>0&&this.nodeOptions.set(e,r),this}let s=t instanceof j?t:new j(t);return s.name=e,this.nodes.set(e,s),r.prompt&&this.nodePrompts.set(e,r.prompt),Object.keys(r).length>0&&this.nodeOptions.set(e,r),this}addConditionalNode(e,t){return this.nodes.set(e,new de({...t,name:e})),this}addEdge(e,t){return this.edges.set(e,t),this}setNodeType(e,t){return this.nodeTypeMap.set(e,t),this}addConditionalEdges(e,t,{labels:r}={}){return this.edges.set(e,{conditional:!0,routes:t,labels:r}),typeof t=="function"&&this.conditionalCodeMap.set(e,t.toString()),this}setEntryPoint(e){return this.entryPoint=e,this}use(e){return typeof e=="function"&&this.middleware.push(e),this}_composeMiddleware(e,t,r,s,n){let a=r;for(let i=e.length-1;i>=0;i--){let u=e[i],l=a;a=()=>u(t,l,s,n)}return a()}serialize(){let e=[],t={};for(let[l,d]of this.nodes){let p=this.nodeTypeMap.get(l)||l;e.push({id:l,type:p,data:{nodeType:p,label:l}});let c={};d._isCustomCode&&typeof d.execute=="function"&&(c.customCode=d.execute.toString());let h=this.nodePrompts.get(l);if(h&&(c.prompt=h),typeof d.customExecute=="function"&&(c.executeCode=d.customExecute.toString()),d.outputSchema)try{if(typeof d.outputSchema._def<"u"){let $=wt(d.outputSchema,{target:"openApi3"});c.outputSchema={jsonSchema:$,variables:this._flattenJsonSchemaToVariables($)}}else c.outputSchema={schema:d.outputSchema}}catch($){console.warn(`[workflow] failed to convert schema for ${l}:`,$.message)}let w=(this.resolvedToolsMap||{})[l];w?.toolIds&&(c.tools=w.toolIds);let _=Array.isArray(d?.config?.skills)?d.config.skills:Array.isArray(d?.skills)?d.skills:null;_&&_.length>0&&(c.skills=[..._]),Object.keys(c).length>0&&(t[l]=c)}let r=[];for(let[l,d]of this.edges)if(typeof d=="string")r.push({source:l,target:d});else if(d.conditional){let p=this.conditionalCodeMap.get(l)||d.routes.toString(),c=this._inferConditionalTargets(d.routes),h=d.labels||{};for(let w of c){let _={source:l,target:w,data:{conditionalCode:p}};h[w]&&(_.label=h[w]),r.push(_)}}let s=l=>{if(!l)return null;try{return wt(l,{target:"openApi3"})}catch{return null}},n=this._runtimeSchema(),a=s(n||this.stateSchema),i=s(this.inputSchema),u=s(this.contextSchema);return{nodes:e,edges:r,nodeConfigs:t,stateSchema:a,inputSchema:i,contextSchema:u}}_inferConditionalTargets(e){let t=e.toString(),r=new Set,s=/return\s+['"]([^'"]+)['"]/g,n;for(;(n=s.exec(t))!==null;)r.add(n[1]);return[...r]}_flattenJsonSchemaToVariables(e,t=""){let r=e;if(e.$ref&&e.definitions){let s=e.$ref.replace("#/definitions/","");r=e.definitions[s]||e}return this._flattenSchema(r,t)}_flattenSchema(e,t=""){if(!e||typeof e!="object")return[];let r=[],s=e.properties||{},n=e.required||[];for(let[a,i]of Object.entries(s)){let u=t?`${t}.${a}`:a;r.push({path:u,type:i.type||"unknown",label:i.description||this._formatLabel(a),optional:!n.includes(a)}),i.type==="object"&&i.properties&&r.push(...this._flattenSchema(i,u)),i.type==="array"&&i.items?.type==="object"&&i.items.properties&&r.push(...this._flattenSchema(i.items,`${u}[]`))}return r}_formatLabel(e){return e.replace(/([A-Z])/g," $1").replace(/^./,t=>t.toUpperCase()).trim()}_summarizeNodeOutput(e,t){if(!t||typeof t!="object")return[];let r=[];t.success!==void 0&&r.push(`Result: ${t.success?"passed":"failed"}`);for(let[s,n]of Object.entries(t))if(!(s==="success"||s==="raw"||s==="nextNode")){if(typeof n=="string"&&n.length<=80)r.push(`${s}: ${n}`);else if(Array.isArray(n)){let a=n.length,i=n.filter(l=>l?.passed===!0).length,u=n.some(l=>l?.passed!==void 0);r.push(u?`${s}: ${i}/${a} passed${a-i?`, ${a-i} failed`:""}`:`${s}: ${a} items`)}if(r.length>=4)break}return r}async run(e,t={},r={}){if(!this.entryPoint)throw new Error("No entry point set for graph");let s=new AbortController;r.signal&&(r.signal.aborted?s.abort():r.signal.addEventListener("abort",()=>s.abort(),{once:!0}));let n=r.strategyAbortTimeoutMs??t.config?.strategyAbortTimeoutMs??5e3,a=t.cwd||process.cwd();go({path:Y(a,".env")});let i=t.config||{};if(!i||Object.keys(i).length===0)try{let E=Y(a,".zibby.config.js");Oe(E)&&(i=(await import(E)).default||{})}catch{}process.env.EXECUTION_ID&&!i.agent?.strictMode&&(i.agent={...i.agent,strictMode:!0});let u=t.agentType;if(!u){let E=i?.agent;E?.provider?u=E.provider:E?.gemini?u="gemini":E?.claude?u="claude":E?.cursor?u="cursor":E?.codex?u="codex":u=process.env.AGENT_TYPE||"cursor"}let l=t.contextConfig||e?.config?.contextConfig||e?.config?.context||i?.context||{},d=this._runtimeSchema();if(d){let E=d.safeParse(t);if(!E.success){let O=E.error.issues.map(N=>`${N.path.join(".")}: ${N.message}`);throw console.error("\u274C Initial state validation failed:"),O.forEach(N=>console.error(` - ${N}`)),new Error(`State validation failed: ${O.join(", ")}`)}A.step("State validated against schema")}let p=yo(),c=t.sessionPath||p;c||_o();let{sessionPath:h,sessionTimestamp:w,sessionId:_}=bo({cwd:a,config:i,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:c,sessionTimestamp:t.sessionTimestamp}});A.step(`Session ${_}`);let $=await fe.loadContext(t.specPath||"",a,l);Object.keys($).length>0&&A.step(`Context loaded: ${Object.keys($).join(", ")}`);let b=t.outputPath;!b&&t.specPath&&(e?.calculateOutputPath?b=e.calculateOutputPath(t.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${t.specPath})`));let f=new ie({...t,config:i,agentType:u,outputPath:b,sessionPath:h,sessionTimestamp:w,context:$,resolvedTools:this.resolvedToolsMap||{},_signal:s.signal}),I=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:g}=await Promise.resolve().then(()=>(le(),et)),m=i.skills&&typeof i.skills=="object"?i.skills:{},R=Object.values(m).filter(E=>E&&typeof E=="object"&&typeof E.id=="string"),T=E=>{for(let O of R)if(O.id===E)return O;return g(E)},M=new Set;for(let[,E]of this.nodes)for(let O of E.config?.skills||[])M.add(O);for(let E of M){let O=T(E);if(typeof O?.middleware=="function")try{let N=await O.middleware();typeof N=="function"&&I.set(E,N)}catch{}}let S=this.entryPoint,oe=[],Re=i?.recursionLimit??100,vt=0;try{for(;S&&S!=="END";){if(++vt>Re)throw new Error(`Workflow exceeded recursion limit (${Re}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let O=Y(h,Ve);if(Oe(O)){try{ho(O)}catch{}s.abort()}if(s.signal.aborted)return console.warn(`
37
+ \u{1F6D1} External stop requested \u2014 ending workflow.`),A.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:oe,stoppedExternally:!0};let N=this.nodes.get(S);if(!N)throw new Error(`Node '${S}' not found in graph`);let Be=JSON.stringify({sessionPath:h,sessionTimestamp:w,currentNode:S,createdAt:new Date().toISOString(),config:f.get("config")}),Tt=Y(h,z);St(Tt,Be,"utf-8");let Me=f.get("config")?.paths?.output||ue,xt=Y(a,Me,z);yt(Y(a,Me),{recursive:!0});try{St(xt,Be,"utf-8")}catch{}let De=t.onPipelineProgress;if(typeof De=="function")try{De({cwd:a,sessionPath:h,sessionId:_,outputBase:f.get("config")?.paths?.output||ue,currentNode:S})}catch{}let At=(this.resolvedToolsMap||{})[S]||null;f.set("_currentNodeTools",At);let kt=f.get("nodeConfigs")||{};f.set("_currentNodeConfig",kt[S]||{}),A.nodeStart(S);let Le=Date.now(),re=this.nodePrompts.get(S);if(!this._invokeAgent){let C=await Promise.resolve().then(()=>(ee(),Q));this._invokeAgent=C.invokeAgent}let Ot=this._invokeAgent,me={},Nt=N.config?.skills||[];for(let C of Nt){let P=T(C);if(typeof P?.invokeAgentOptions=="function")try{let v=P.invokeAgentOptions(f.getAll(),{agentType:f.get("agentType"),nodeName:S});v&&typeof v=="object"&&(me={...me,...v})}catch(v){console.warn(`[graph] skill '${C}' invokeAgentOptions threw: ${v.message}`)}}let je=async(C,P,v={})=>{let B=Ot(C,P,{...me,...v,signal:s.signal});return B.catch(()=>{}),s.signal.aborted?B:Promise.race([B,new Promise((J,Z)=>{let L=()=>{setTimeout(()=>{let q=new Error(`Strategy ignored AbortSignal \u2014 engine deadman fired after ${n}ms`);q.name="AbortError",Z(q)},n)};s.signal.addEventListener("abort",L,{once:!0})})])},Ue={state:f,invokeAgent:async(C={},P={})=>{let v=P.prompt||"";if(re){let B=this._compiledPrompts.get(S);B||(B=mo.compile(re,{noEscape:!0}),this._compiledPrompts.set(S,B));try{v=B(C)}catch(J){throw console.error(`\u274C Template rendering failed for node '${S}':`,J.message),new Error(`Template rendering failed: ${J.message}`,{cause:J})}}else if(!v)throw new Error(`No prompt template configured for node '${S}' and no prompt provided in options`);return je(v,{state:f.getAll(),images:P.images||[]},{model:P.model||f.get("model"),workspace:f.get("workspace"),schema:P.schema,...P,signal:s.signal})},_coreInvokeAgent:je,agent:e,nodeId:S,promptTemplate:re,getPromptTemplate:()=>re,...f.getAll()};try{let C=(N.config?.skills||[]).map(L=>I.get(L)).filter(Boolean),P=[...this.middleware,...C],v;P.length>0?v=await this._composeMiddleware(P,S,async()=>N.execute(Ue,f),f.getAll(),f):v=await N.execute(Ue,f);let B=Date.now()-Le;if(oe.push({node:S,success:v.success,duration:B,timestamp:new Date().toISOString()}),!v.success){if(s.signal.aborted)return A.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:oe,stoppedExternally:!0};f.append("errors",{node:S,error:v.error});let L=N.config?.retries||0,q=`${S}_retries`,ne=f.getAll()[q]||0;if(ne<L){A.stepInfo(`Retrying (attempt ${ne+1}/${L})`),f.update({[q]:ne+1,[`${S}_raw`]:v.raw});continue}throw A.nodeFailed(S,v.error,{duration:B}),new Error(`Node '${S}' failed after ${ne} attempts: ${v.error}`)}f.update({[S]:v.output});let J=this._summarizeNodeOutput(S,v.output);A.nodeComplete(S,{duration:B,details:J});let Z=this.edges.get(S);if(!Z)S="END";else if(Z.conditional){let L=Z.routes(f.getAll());A.route(S,L),S=L}else S=Z}catch(C){throw A.isInsideNode&&A.nodeFailed(S,C.message,{duration:Date.now()-Le}),f.set("failed",!0),f.set("failedAt",S),C}}A.graphComplete();let E={success:!0,state:f.getAll(),executionLog:oe};return e&&typeof e.onComplete=="function"&&await e.onComplete(E),E}finally{if(e&&typeof e.cleanup=="function")try{await e.cleanup()}catch(E){console.warn(`[workflow] agent.cleanup() failed: ${E.message}`)}}}};var Ne=Symbol.for("@zibby/agent-workflow.nodes");globalThis[Ne]||(globalThis[Ne]=new Map);var Ce=globalThis[Ne];function $o(o,e){Ce.set(o,e)}function It(o){return Ce.get(o)}function Pe(o){return Ce.has(o)}$o("ai_agent",{name:"ai_agent",factory:!0,create:(o,e={})=>({name:o,_isCustomCode:!0,execute:async t=>{let r=t?._coreInvokeAgent;r||(r=(await Promise.resolve().then(()=>(ee(),Q))).invokeAgent);let s=e.extraPromptInstructions||"Execute the task based on the current state.",n=vo(s,t),a=await r(n,{cwd:t.workspace||process.cwd(),model:t.model,tools:e.resolvedTools||null});return{success:!0,output:{raw:a,nodeId:o},raw:typeof a=="string"?a:a.raw}}})});function vo(o,e){let t=/@([\w.]+)/g,r=new Set,s;for(;(s=t.exec(o))!==null;)r.add(s[1]);if(r.size===0)return o;let n=[],a=new Set;for(let i of r){let u=i.split(".")[0];if(a.has(u))continue;let l=i.split(".").reduce((c,h)=>c?.[h],e);if(l===void 0)continue;let d=typeof l=="string"?l:l?.raw??JSON.stringify(l,null,2),p=i.replace(/_/g," ").replace(/\b\w/g,c=>c.toUpperCase());n.push(`## ${p}
38
38
  ${d}`),i.includes(".")||a.add(u)}return n.length===0?o:`${o}
39
39
 
40
40
  ---
package/dist/index.js CHANGED
@@ -24,7 +24,7 @@ ${st}`,e.col=it))}return t(i,n,s)}}var ge=class{constructor(){this._currentNode=
24
24
  `)}stepInfo(e){this.step(e)}stepTool(e){this._origStdoutWrite?this._writeDot(nt,e):process.stdout.write.bind(process.stdout)(`${oe} ${nt} ${e}
25
25
  `)}stepMemory(e){let o=k.hex("#2dd4bf")(e);this._origStdoutWrite?this._writeDot(rt,o):process.stdout.write.bind(process.stdout)(`${oe} ${rt} ${o}
26
26
  `)}stepFail(e){this._origStdoutWrite?this._writeDot(be,k.red(e)):process.stdout.write.bind(process.stdout)(`${oe} ${be} ${k.red(e)}
27
- `)}nodeStart(e){this._currentNode=e,this._emitGraphLogMarker({phase:"node_begin",node:e}),this._rawWrite(`${no} ${e}`),this._startIntercepting()}nodeComplete(e,o={}){this._stopIntercepting();let{duration:n,details:s}=o;if(s)for(let i of s)this._rawWrite(`${$e} ${i}`);let r=n?k.dim(` ${at(n)}`):"";this._rawWrite(`${ot} ${k.green("done")}${r}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}nodeFailed(e,o,n={}){this._stopIntercepting();let{duration:s}=n,r=s?k.dim(` ${at(s)}`):"";this._rawWrite(`${be} ${k.red(o)}`),this._rawWrite(`${ot} ${k.red("failed")}${r}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}route(e,o){this._rawWrite(k.dim(` ${e} \u2192 ${o}`)),this._rawWrite("")}graphComplete(){}},x=new ge;var ne=".zibby/output",Te="sessions",W=".session-info.json",ve=".zibby-stop",ro="result.json",so="raw_stream_output.txt",io="events.json",ao={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GITLAB:"gitlab",GIT:"git",SLACK:"slack",LARK:"lark",SENTRY:"sentry",MEMORY:"memory",CHAT_MEMORY:"chat-memory",RUNNER:"runner",SKILL_INSTALLER:"skill-installer",CORE_TOOLS:"core-tools",WORKFLOW_BUILDER:"workflow-builder",SESSION:"session",OPENAI_BILLING:"openai_billing",ANTHROPIC_BILLING:"anthropic_billing",CURSOR_ADMIN:"cursor_admin",NOTION:"notion"},xe=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];var j=class{constructor(e){if(this.config=e,this.name=e.name,this.prompt=e.prompt,this.outputSchema=e.outputSchema,!this.outputSchema&&!e._isCustomCode)throw new Error(`Node '${this.name}' must define outputSchema (Zod schema). This defines the contract for what the node returns to state.`);this.isZodSchema=this.outputSchema&&typeof this.outputSchema._def<"u",this.parser=e.outputSchema&&!this.isZodSchema?new ee(e.outputSchema):null,this.retries=e.retries||0,this.onComplete=e.onComplete,this.customExecute=e.execute}async execute(e,o){let n=()=>o&&typeof o.getAll=="function"?o.getAll():e,s=p=>o&&typeof o.get=="function"?o.get(p):e?.[p];if(typeof this.customExecute=="function"){_.debug(`[workflow] node '${this.name}': custom execute (skipping LLM)`);try{let p=await this.customExecute(e);return typeof p=="object"&&p!==null&&p.success===!1?{success:!1,error:p.error||"Node execution failed",raw:p.raw||null}:this.isZodSchema?(_.debug(`[workflow] node '${this.name}': validating output schema`),{success:!0,output:this.outputSchema.parse(p),raw:null}):{success:!0,output:p,raw:null}}catch(p){return _.error(`[workflow] node '${this.name}' failed: ${p.message}`),p.name==="ZodError"&&_.error(`Schema errors: ${JSON.stringify(p.issues||p.errors,null,2)}`),{success:!1,error:p.message,raw:null}}}let r=typeof this.prompt=="function"?this.prompt(n()):this.prompt,i=s("_skillHints");i&&(r=`${i}
27
+ `)}nodeStart(e){this._currentNode=e,this._emitGraphLogMarker({phase:"node_begin",node:e}),this._rawWrite(`${no} ${e}`),this._startIntercepting()}nodeComplete(e,o={}){this._stopIntercepting();let{duration:n,details:s}=o;if(s)for(let i of s)this._rawWrite(`${$e} ${i}`);let r=n?k.dim(` ${at(n)}`):"";this._rawWrite(`${ot} ${k.green("done")}${r}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}nodeFailed(e,o,n={}){this._stopIntercepting();let{duration:s}=n,r=s?k.dim(` ${at(s)}`):"";this._rawWrite(`${be} ${k.red(o)}`),this._rawWrite(`${ot} ${k.red("failed")}${r}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}route(e,o){this._rawWrite(k.dim(` ${e} \u2192 ${o}`)),this._rawWrite("")}graphComplete(){}},x=new ge;var ne=".zibby/output",Te="sessions",W=".session-info.json",ve=".zibby-stop",ro="result.json",so="raw_stream_output.txt",io="events.json",ao={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GITLAB:"gitlab",GIT:"git",SLACK:"slack",LARK:"lark",CHAT_NOTIFY:"chat_notify",SENTRY:"sentry",MEMORY:"memory",CHAT_MEMORY:"chat-memory",RUNNER:"runner",SKILL_INSTALLER:"skill-installer",CORE_TOOLS:"core-tools",WORKFLOW_BUILDER:"workflow-builder",SESSION:"session",OPENAI_BILLING:"openai_billing",ANTHROPIC_BILLING:"anthropic_billing",CURSOR_ADMIN:"cursor_admin",NOTION:"notion"},xe=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];var j=class{constructor(e){if(this.config=e,this.name=e.name,this.prompt=e.prompt,this.outputSchema=e.outputSchema,!this.outputSchema&&!e._isCustomCode)throw new Error(`Node '${this.name}' must define outputSchema (Zod schema). This defines the contract for what the node returns to state.`);this.isZodSchema=this.outputSchema&&typeof this.outputSchema._def<"u",this.parser=e.outputSchema&&!this.isZodSchema?new ee(e.outputSchema):null,this.retries=e.retries||0,this.onComplete=e.onComplete,this.customExecute=e.execute}async execute(e,o){let n=()=>o&&typeof o.getAll=="function"?o.getAll():e,s=p=>o&&typeof o.get=="function"?o.get(p):e?.[p];if(typeof this.customExecute=="function"){_.debug(`[workflow] node '${this.name}': custom execute (skipping LLM)`);try{let p=await this.customExecute(e);return typeof p=="object"&&p!==null&&p.success===!1?{success:!1,error:p.error||"Node execution failed",raw:p.raw||null}:this.isZodSchema?(_.debug(`[workflow] node '${this.name}': validating output schema`),{success:!0,output:this.outputSchema.parse(p),raw:null}):{success:!0,output:p,raw:null}}catch(p){return _.error(`[workflow] node '${this.name}' failed: ${p.message}`),p.name==="ZodError"&&_.error(`Schema errors: ${JSON.stringify(p.issues||p.errors,null,2)}`),{success:!1,error:p.message,raw:null}}}let r=typeof this.prompt=="function"?this.prompt(n()):this.prompt,i=s("_skillHints");i&&(r=`${i}
28
28
 
29
29
  ${r}`);let a=n(),c=a.cwd||process.cwd(),l=a.sessionPath;try{if(l){let p=Re(l,W);if(_t(p)){let h=JSON.parse(yt(p,"utf-8"));h.currentNode=this.name,Pe(p,JSON.stringify(h,null,2),"utf-8")}let u=Re(l,"..",W);if(_t(u))try{let h=JSON.parse(yt(u,"utf-8"));h.currentNode=this.name,Pe(u,JSON.stringify(h,null,2),"utf-8")}catch{}}}catch(p){_.debug(`[workflow] could not update session info: ${p.message}`)}let d=null;for(let p=0;p<=this.retries;p++)try{_.debug(`[workflow] node '${this.name}' attempt ${p}`);let u=n().config||{},h=u.agents||{},m=this.config.agent??h[this.name]??null,w={state:n()};m&&(w.preferredAgent=m);let I={workspace:c,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:l,config:u,nodeName:this.name,timeout:this.config?.timeout||3e5},E=e?._coreInvokeAgent;E||(E=(await Promise.resolve().then(()=>(q(),se))).invokeAgent);let f=await E(r,w,I),$,g;if(typeof f=="string"?($=f,g=null):f.structured?($=f.raw||JSON.stringify(f.structured,null,2),g=f.structured):($=f.raw||JSON.stringify(f,null,2),g=f.extracted||null),l)try{let S=Re(l,this.name,"raw_stream_output.txt");co(lo(S),{recursive:!0}),Pe(S,typeof $=="string"?$:JSON.stringify($),"utf-8")}catch(S){_.debug(`[workflow] could not save raw output: ${S.message}`)}if(this.isZodSchema&&g){_.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(g,null,2)}`);let S=g;if(typeof this.onComplete=="function")try{S=await this.onComplete(n(),g)}catch(R){_.warn(`[workflow] onComplete hook failed: ${R.message}`)}return{success:!0,output:S,raw:$}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(n(),{raw:$}),raw:$}}catch(S){throw new Error(`onComplete failed: ${S.message}`,{cause:S})}if(this.parser){let S=this.parser.parse($);return _.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(S,null,2)}`),x.step("Output parsed"),{success:!0,output:S,raw:$}}return{success:!0,output:$,raw:$}}catch(u){d=u,p<this.retries&&_.info(`[workflow] node '${this.name}' failed, retrying (${p+1}/${this.retries})\u2026`)}return{success:!1,error:d.message,raw:null}}},ie=class extends j{constructor(e){super({...e,_isCustomCode:!0}),this.condition=e.condition}async execute(e,o){let n=o&&typeof o.getAll=="function"?o.getAll():e;return{success:!0,output:{nextNode:this.condition(n)},raw:null}}};F();F();import{mkdirSync as fo,existsSync as J,statSync as xt,readdirSync as kt,rmSync as ho}from"node:fs";import{spawn as vt}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as go}from"node:url";import{AsyncLocalStorage as uo}from"node:async_hooks";var Be=new uo;function ae(){let t=Be.getStore();return t||Object.freeze({executionId:process.env.EXECUTION_ID||null,parentExecutionId:process.env.PARENT_EXECUTION_ID||null,depth:0,conversationId:process.env.ZIBBY_CONVERSATION_ID||null,dispatchMode:process.env.DISPATCH_MODE||null})}function It(t,e){let o=Be.getStore()||ae(),n=Object.freeze({executionId:t.executionId,parentExecutionId:t.parentExecutionId??o.executionId??null,depth:(o.depth||0)+(t.executionId!==o.executionId?1:0),conversationId:t.conversationId!==void 0?t.conversationId:o.conversationId??null,dispatchMode:t.dispatchMode??null});return Be.run(n,e)}var Me=new Map,De=new Map,Et=new Map;function $t(t,e,o={}){if(!t||typeof t!="string")throw new Error("subgraph-registry.register: name required");if(typeof e!="function")throw new Error("subgraph-registry.register: factory must be a function");Me.set(t,e),De.set(t,"ready"),Et.set(t,{...o,cachedAt:Date.now()})}function bt(t,e){De.set(t,"failed"),Et.set(t,{error:e?.message||String(e),failedAt:Date.now()}),Me.delete(t)}function Tt(t){return De.get(t)==="ready"?Me.get(t):null}var me=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function mo(){return`node${(process.versions?.node||"").split(".")[0]||"unknown"}-${process.platform}-${process.arch}`}var A=class extends Error{constructor(e,o){super(`in-process sub-graph fallback: ${e}${o?` (${o})`:""}`),this.fallback=!0,this.reason=e,this.detail=o||null,this.name="SubgraphFallback"}};function So(){let t=(process.env.SUBGRAPH_INTERNAL_URL||"").replace(/\/$/,""),e=(process.env.PROGRESS_API_URL||"").replace(/\/executions\/?$/,""),o=t||e,n=process.env.PROJECT_ID,s=process.env.PROJECT_API_TOKEN;if(!o||!n||!s)throw new A("env","SUBGRAPH_INTERNAL_URL/PROGRESS_API_URL/PROJECT_ID/PROJECT_API_TOKEN missing");return{apiBase:o,projectId:n,authToken:s}}async function wo({apiBase:t,authToken:e,body:o}){let n;try{n=await fetch(`${t}/internal/subgraph/begin`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(o)})}catch(r){throw new A("network",`begin fetch failed: ${r.message}`)}let s=null;try{s=await n.json()}catch{}if(!n.ok){if(n.status===404){let r=new Error(`Sub-graph child '${o.childWorkflowType}' not found in project`);throw r.code="SUBGRAPH_NOT_FOUND",r.status=404,r}if(n.status===429){let r=s?.quotaInfo||{},i=new Error(`Sub-graph blocked by quota (${r.used??"?"}/${r.limit??"?"} on ${r.planId||"plan"})`);throw i.code="SUBGRAPH_QUOTA_EXCEEDED",i.status=429,i.quotaInfo=r,i}if(n.status===400&&s?.validationErrors){let r=new Error(`Sub-graph rejected input: ${s?.error||s?.message||"validation failed"}`);throw r.code="SUBGRAPH_INVALID_INPUT",r.status=400,r.validationErrors=s.validationErrors,r.missing=s.missing,r}throw new A("begin-status",`begin returned ${n.status}`)}return s?.data||s}async function H({apiBase:t,authToken:e,payload:o}){try{let n=await fetch(`${t}/internal/subgraph/finalize`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(o)});n.ok||_.warn(`[in-process subgraph] finalize returned ${n.status} for ${o.childExecutionId}`)}catch(n){_.warn(`[in-process subgraph] finalize failed: ${n.message}`)}}async function yo(t,e){let o=U(e,".ready"),n=U(e,"graph.mjs");if(J(o)&&J(n))return;fo(e,{recursive:!0});let s=U(e,".lock"),r=!1;try{let{openSync:i,closeSync:a}=await import("node:fs"),c=i(s,"wx");a(c),r=!0}catch(i){if(i.code!=="EEXIST")throw i}if(!r){let i=Date.now()+3e4;for(;Date.now()<i;){if(J(o)&&J(n))return;await new Promise(a=>setTimeout(a,100))}throw new A("bundle-extract-timeout","sibling extract did not complete within 30s")}try{await new Promise((c,l)=>{let d=vt("curl",["-fsSL",t],{stdio:["ignore","pipe","inherit"]}),p=vt("tar",["-xzf","-","-C",e],{stdio:["pipe","inherit","inherit"]});d.stdout.pipe(p.stdin);let u,h,m=()=>{if(u!==void 0&&h!==void 0){if(u!==0)return l(new Error(`curl exited ${u}`));if(h!==0)return l(new Error(`tar exited ${h}`));c()}};d.on("close",w=>{u=w,m()}),p.on("close",w=>{h=w,m()}),d.on("error",l),p.on("error",l)});let{writeFileSync:i,unlinkSync:a}=await import("node:fs");i(o,"");try{a(s)}catch{}}catch(i){try{let{unlinkSync:a}=await import("node:fs");a(s)}catch{}throw new A("bundle-extract-failed",i.message)}}async function _o(t){let e=U(t,"graph.mjs");if(!J(e))throw new A("entry-missing",`graph.mjs missing under ${t}`);let o;try{o=await import(go(e).href)}catch(s){throw new A("import-failed",`${s?.code||s?.name||"unknown"}: ${s.message}`)}let n=o.default||Object.values(o).find(s=>typeof s=="function"&&s.prototype?.buildGraph);if(!n)throw new A("entry-class-missing","no buildGraph() class export found");return n}async function At(t,e={}){if(!t||typeof t!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let o=ae(),n;try{n=So()}catch(g){throw g}_.debug(`[in-process subgraph] begin '${t}' parent=${o.executionId||"<root>"}`);let s=await wo({apiBase:n.apiBase,authToken:n.authToken,body:{parentExecutionId:o.executionId,childWorkflowType:t,input:e.input||{},...e.conversationId?{conversationId:e.conversationId}:{}}}),{childExecutionId:r,runtimeTag:i,bundlePresignedUrl:a,sourcesPresignedUrl:c,workflowVersion:l,workflowUuid:d,bundleReady:p}=s,u=mo();if(i&&i!==u)throw await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"canceled",error:{message:`runtimeTag mismatch: parent=${u} child=${i}`,code:"RUNTIME_MISMATCH"}}}),new A("runtime-mismatch",`${u} vs ${i}`);if(!p||!a)throw await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"canceled",error:{message:"bundle not ready for in-process; falling back to HTTP",code:"NO_BUNDLE"}}}),new A("no-bundle","workflow bundle not built yet");let h=Tt(t);if(!h){let g=U(me,`${d}@${l||"0"}`);try{await yo(a,g);try{Eo()}catch{}}catch(S){throw S.fallback&&await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"failed",error:{message:S.message,code:S.reason}}}),S}try{h=await _o(g),$t(t,h,{workflowUuid:d,version:l,runtimeTag:i,cacheDir:g})}catch(S){throw bt(t,S),await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"failed",error:{message:S.message,code:S.reason||"IMPORT_FAILED"}}}),S.fallback?S:new A("import-failed",S.message)}}let m=Date.now(),I=await(typeof h=="function"&&h.prototype?.buildGraph?new h:h).buildGraph(),E={...e.input||{}},f,$;try{f=await It({executionId:r,parentExecutionId:o.executionId,conversationId:e.conversationId!==void 0?e.conversationId:o.conversationId,dispatchMode:"inprocess"},()=>I.run(e.parentAgent,E,{signal:e.signal})),$=f&&typeof f=="object"&&"state"in f?f.state:f}catch(g){throw await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"failed",error:{message:g.message,code:g.code||"CHILD_THREW",stack:g.stack},durationMs:Date.now()-m}}),g}if(f&&typeof f=="object"&&f.stoppedExternally){await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"canceled",finalState:$,durationMs:Date.now()-m}});let g=new Error(`Sub-graph '${t}' canceled by parent abort`);throw g.code="SUBGRAPH_CANCELED",g.subgraphJobId=r,g}return await H({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"completed",finalState:$,durationMs:Date.now()-m}}),{finalState:$,executionId:r}}function Io(t){let e=0,o=[t];for(;o.length;){let n=o.pop(),s;try{s=xt(n)}catch{continue}if(s.isDirectory()){let r;try{r=kt(n)}catch{continue}for(let i of r)o.push(U(n,i))}else e+=s.size}return e}function Eo({cap:t=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!J(me))return{evicted:0,freedBytes:0};let e=kt(me),o=[],n=0;for(let a of e){let c=U(me,a),l;try{l=xt(c)}catch{continue}let d=l.isDirectory()?Io(c):l.size;n+=d,o.push({name:a,full:c,size:d,mtimeMs:l.mtimeMs})}if(n<=t)return{evicted:0,freedBytes:0,totalBytes:n};o.sort((a,c)=>a.mtimeMs-c.mtimeMs);let s=Math.floor(t*.7),r=0,i=0;for(let a of o){if(n-r<=s)break;if(!J(U(a.full,".lock")))try{ho(a.full,{recursive:!0,force:!0}),r+=a.size,i+=1}catch(c){_.debug(`[sub-graph cache] evict skip ${a.name}: ${c.message}`)}}return i>0&&_.info(`[sub-graph cache] evicted ${i} entr(y/ies), freed ${(r/1024/1024).toFixed(1)}MB`),{evicted:i,freedBytes:r,totalBytes:n-r}}catch(e){return _.debug(`[sub-graph cache] evict failed: ${e.message}`),{evicted:0,freedBytes:0}}}var $o=2e3,bo=600*1e3,To=new Set(["completed","failed","canceled","timeout"]);function vo(){let t=process.env.PROGRESS_API_URL;if(!t)throw new Error("Sub-graph dispatch requires PROGRESS_API_URL env var (set automatically on cloud runs). Sub-graphs are not supported in local in-process runs yet \u2014 deploy the parent and child to cloud.");return t.replace(/\/executions\/?$/,"")}function xo(){let t=process.env.PROJECT_ID;if(!t)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return t}function ko(){let t=process.env.PROJECT_API_TOKEN;if(!t)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return t}function Ao(){return process.env.EXECUTION_ID||null}function Nt(t,e){return e==null?t:typeof e=="function"?e(t):typeof e=="string"?e.split(".").reduce((o,n)=>o==null?o:o[n],t):t}async function Le(t,e={}){if(!t||typeof t!="string")throw new Error("dispatchSubgraph: workflowName (string) is required");let o=ae(),n=Number(process.env.ZIBBY_SUBGRAPH_MAX_DEPTH||10);if((o.depth||0)>=n)throw new Error(`dispatchSubgraph('${t}'): sub-graph depth ${o.depth} reached cap of ${n}. Restructure the graph or raise ZIBBY_SUBGRAPH_MAX_DEPTH.`);if(process.env.ZIBBY_INPROCESS_SUBGRAPH!=="0"&&!e.async)try{_.debug(`[sub-graph] trying in-process for '${t}'`);let{finalState:g}=await At(t,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),S=Nt(g,e.output);return _.info(`[sub-graph] '${t}' completed in-process`),S}catch(g){if(g instanceof A||g?.fallback)_.info(`[sub-graph] in-process fallback for '${t}': ${g.reason||"unknown"} \u2014 using HTTP`);else throw g}let s=vo(),r=xo(),i=ko(),a=Ao(),c=`${s}/projects/${encodeURIComponent(r)}/workflows/${encodeURIComponent(t)}/trigger`,l={input:e.input||{},...a?{parentExecutionId:a}:{},...e.conversationId?{conversationId:e.conversationId}:{}};_.info(`[sub-graph] dispatching '${t}' (${e.async?"async":"sync"}) from parent ${a||"<none>"}`);let d=await fetch(c,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${i}`},body:JSON.stringify(l)});if(!d.ok){let g=null,S="";try{g=await d.json(),S=g?.error||g?.message||JSON.stringify(g)}catch{S=await d.text().catch(()=>"")}if(d.status===429){let v=g?.quotaInfo||{},D=new Error(`Sub-graph '${t}' blocked by execution quota (${v.used??"?"}/${v.limit??"?"} on plan ${v.planId||"unknown"}). Sub-workflow runs count toward the same monthly cap as user-triggered runs.`);throw D.code="SUBGRAPH_QUOTA_EXCEEDED",D.status=429,D.subgraph=t,D.quotaInfo=v,D}if(d.status===400){let v=new Error(`Sub-graph '${t}' rejected input: ${S}`);throw v.code="SUBGRAPH_INVALID_INPUT",v.status=400,v.subgraph=t,v.validationErrors=g?.validationErrors||null,v.missing=g?.missing||null,v}let R=new Error(`Sub-graph '${t}' trigger rejected (${d.status}): ${S}`);throw R.code="SUBGRAPH_TRIGGER_FAILED",R.status=d.status,R.subgraph=t,R}let p=await d.json(),u=p?.data?.jobId||p?.jobId;if(!u)throw new Error(`Sub-graph '${t}' trigger returned no jobId: ${JSON.stringify(p).slice(0,200)}`);if(e.async)return _.info(`[sub-graph] async dispatch of '${t}' \u2192 jobId=${u} (not waiting)`),{jobId:u,status:"accepted",workflow:t};let h=Number.isFinite(e.timeoutMs)?e.timeoutMs:bo,m=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:$o,w=`${s}/executions/${encodeURIComponent(u)}`,I=Date.now()+h,E="accepted",f=0;for(;Date.now()<I;){await new Promise(v=>setTimeout(v,m)),f+=1;let g=await fetch(w,{headers:{Authorization:`Bearer ${i}`}});if(!g.ok){if(g.status>=500){_.warn(`[sub-graph] status poll for ${u} returned ${g.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${u}: ${g.status}`)}let S=await g.json(),R=S?.data||S?.execution||S;if(E=R?.status||E,To.has(E)){if(E!=="completed"){let y=new Error(`Sub-graph '${t}' (${u}) ended in status '${E}'`);throw y.subgraphJobId=u,y.subgraphStatus=E,y}let v=R?.finalState||R?.state||{},D=Nt(v,e.output);return _.info(`[sub-graph] '${t}' (${u}) completed after ${f} polls`),D}}let $=new Error(`Sub-graph '${t}' (${u}) timed out after ${Math.round(h/1e3)}s (last status: ${E})`);throw $.subgraphJobId=u,$.subgraphStatus=E,$}import{existsSync as Ot,readFileSync as No}from"node:fs";import{join as je,dirname as Ct}from"node:path";var ce=class{static async loadContext(e,o,n={}){let s={},r=n.filenames||["CONTEXT.md","AGENTS.md"];if(e){let a=Ct(je(o,e));for(let c of r){let l=await this.findAndMergeContextFiles(c,a,o);if(l){let d=c.replace(/\.[^.]+$/,"").toLowerCase();s[d]=l}}}let i=n.discovery||{};for(let[a,c]of Object.entries(i))try{let l=je(o,c);Ot(l)&&(s[a]=await this.loadFile(l))}catch(l){console.warn(`[workflow] could not load context '${a}' from '${c}': ${l.message}`)}return s}static async findAndMergeContextFiles(e,o,n){let s=[],r=o;for(;r.startsWith(n);){let i=je(r,e);if(Ot(i))try{s.unshift(await this.loadFile(i))}catch(c){console.warn(`[workflow] could not load ${e} from ${i}: ${c.message}`)}let a=Ct(r);if(a===r)break;r=a}return s.length===0?null:s.every(i=>typeof i=="string")?s.join(`
30
30
 
package/dist/node.js CHANGED
@@ -1,4 +1,4 @@
1
- var it=Object.defineProperty;var N=(r,t)=>()=>(r&&(t=r(r=0)),t);var st=(r,t)=>{for(var e in t)it(r,e,{get:t[e],enumerable:!0})};var D,nt,k,l,T=N(()=>{D=()=>{},nt={debug:D,info:D,warn:(...r)=>console.warn("[workflow]",...r),error:(...r)=>console.error("[workflow]",...r)},k={impl:nt},l={debug:(...r)=>k.impl.debug?.(...r),info:(...r)=>k.impl.info?.(...r),warn:(...r)=>k.impl.warn?.(...r),error:(...r)=>k.impl.error?.(...r)}});var Z=N(()=>{});function V(r){return ut.get(r)||null}var v,ut,z=N(()=>{v=Symbol.for("@zibby/agent-workflow.skills");globalThis[v]||(globalThis[v]=new Map);ut=globalThis[v]});var X={};st(X,{getAgentStrategy:()=>q,invokeAgent:()=>dt,listStrategies:()=>pt,registerStrategy:()=>ct});function ct(r){if(!r||typeof r.getName!="function"||typeof r.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let t=w.findIndex(e=>e.getName()===r.getName());t>=0?w[t]=r:w.push(r)}function pt(){return w.map(r=>r.getName())}function q(r={}){let{state:t={},preferredAgent:e=null}=r,o=e||t.agentType||process.env.AGENT_TYPE;if(!o){let n=w.map(a=>a.getName()).join(", ")||"none registered";throw new Error(`No agent specified. Set agentType in state or AGENT_TYPE env var. Available: ${n}`)}l.debug(`[workflow] agent selection: requested=${o}`);let s=w.find(n=>n.getName()===o);if(!s){let n=w.map(a=>a.getName()).join(", ")||"none registered";throw new Error(`Unknown agent '${o}'. Available: ${n}`)}if(!s.canHandle(r))throw new Error(`Agent '${o}' is not available in this environment. Check credentials/environment.`);return l.debug(`[workflow] using agent: ${s.getName()}`),s}async function dt(r,t={},e={}){let o=q(t),s=t.state?.config||e.config||{},n=s.models||{},a=e.nodeName&&n[e.nodeName]||null,g=n.default||null,p=s.agent?.[o.name]?.model||null,_=a||g||p||e.model||null,E={...e,model:_,workspace:t.state?.workspace||e.workspace,schema:e.schema||t.schema,images:e.images||t.images||[],skills:e.skills||t.skills||[],config:s},i=r,h=E.skills||[];if(h.length>0&&!e.skipPromptFragments){let y=h.map(O=>{let $=V(O)?.promptFragment;return typeof $=="function"?$():$}).filter(Boolean);y.length>0&&(i+=`
1
+ var it=Object.defineProperty;var N=(r,t)=>()=>(r&&(t=r(r=0)),t);var st=(r,t)=>{for(var e in t)it(r,e,{get:t[e],enumerable:!0})};var D,nt,k,l,T=N(()=>{D=()=>{},nt={debug:D,info:D,warn:(...r)=>console.warn("[workflow]",...r),error:(...r)=>console.error("[workflow]",...r)},k={impl:nt},l={debug:(...r)=>k.impl.debug?.(...r),info:(...r)=>k.impl.info?.(...r),warn:(...r)=>k.impl.warn?.(...r),error:(...r)=>k.impl.error?.(...r)}});var Z=N(()=>{});function V(r){return ct.get(r)||null}var v,ct,z=N(()=>{v=Symbol.for("@zibby/agent-workflow.skills");globalThis[v]||(globalThis[v]=new Map);ct=globalThis[v]});var X={};st(X,{getAgentStrategy:()=>q,invokeAgent:()=>dt,listStrategies:()=>pt,registerStrategy:()=>ut});function ut(r){if(!r||typeof r.getName!="function"||typeof r.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let t=w.findIndex(e=>e.getName()===r.getName());t>=0?w[t]=r:w.push(r)}function pt(){return w.map(r=>r.getName())}function q(r={}){let{state:t={},preferredAgent:e=null}=r,o=e||t.agentType||process.env.AGENT_TYPE;if(!o){let n=w.map(a=>a.getName()).join(", ")||"none registered";throw new Error(`No agent specified. Set agentType in state or AGENT_TYPE env var. Available: ${n}`)}l.debug(`[workflow] agent selection: requested=${o}`);let s=w.find(n=>n.getName()===o);if(!s){let n=w.map(a=>a.getName()).join(", ")||"none registered";throw new Error(`Unknown agent '${o}'. Available: ${n}`)}if(!s.canHandle(r))throw new Error(`Agent '${o}' is not available in this environment. Check credentials/environment.`);return l.debug(`[workflow] using agent: ${s.getName()}`),s}async function dt(r,t={},e={}){let o=q(t),s=t.state?.config||e.config||{},n=s.models||{},a=e.nodeName&&n[e.nodeName]||null,g=n.default||null,p=s.agent?.[o.name]?.model||null,_=a||g||p||e.model||null,E={...e,model:_,workspace:t.state?.workspace||e.workspace,schema:e.schema||t.schema,images:e.images||t.images||[],skills:e.skills||t.skills||[],config:s},i=r,h=E.skills||[];if(h.length>0&&!e.skipPromptFragments){let y=h.map(O=>{let $=V(O)?.promptFragment;return typeof $=="function"?$():$}).filter(Boolean);y.length>0&&(i+=`
2
2
 
3
3
  ${y.join(`
4
4
 
@@ -11,9 +11,9 @@ PRIORITY OVERRIDE \u2014 THE FOLLOWING INSTRUCTIONS TAKE PRECEDENCE OVER ALL PRE
11
11
  ${f}
12
12
  `),l.debug(`[workflow] prompt length: ${i.length} chars`),o.invoke(i,E)}var P,w,Q=N(()=>{Z();T();z();P=Symbol.for("@zibby/agent-workflow.strategies");globalThis[P]||(globalThis[P]=[]);w=globalThis[P]});var b=class{constructor(t){this.schema=t}parse(t){let e=t.match(/```json\s*([\s\S]*?)\s*```/);if(e)return this.validate(JSON.parse(e[1]));let o=[t.match(/\{[\s\S]*?\}/),t.match(/\{[\s\S]*\}/)].filter(Boolean).map(s=>s[0]);for(let s of o)try{return this.validate(JSON.parse(s))}catch(n){if(!(n instanceof SyntaxError))throw n}return this.validate({result:t.trim()})}validate(t){let e=[];for(let[o,s]of Object.entries(this.schema)){if(s.required&&!(o in t)&&e.push(`Missing required field: ${o}`),o in t&&s.type){let n=typeof t[o];n!==s.type&&e.push(`Field '${o}' expected ${s.type}, got ${n}`)}if(s.validate&&o in t){let n=s.validate(t[o]);n&&e.push(`Field '${o}': ${n}`)}}if(e.length>0)throw new Error(`Output validation failed:
13
13
  ${e.join(`
14
- `)}`);return t}};T();import{writeFileSync as C,readFileSync as tt,existsSync as et,mkdirSync as ht}from"node:fs";import{join as F,dirname as ft}from"node:path";import u from"chalk";var at="__WORKFLOW_GRAPH_LOG__",I=u.gray("\u2502"),lt=u.gray("\u250C"),G=u.gray("\u2514"),W=u.green("\u25C6"),U=u.hex("#c084fc")("\u25C6"),j=u.hex("#2dd4bf")("\u25C6"),A=u.red("\u25C6"),B=`${I} `,J=2;function H(r){return r<1e3?`${r}ms`:`${(r/1e3).toFixed(1)}s`}function K(r,t){return(e,o,s)=>{if(typeof e!="string")return r(e,o,s);let n=process.stdout.columns||120,a="";for(let g=0;g<e.length;g++){let p=e[g];t.lineStart&&(a+=B,t.col=J,t.lineStart=!1),p===`
14
+ `)}`);return t}};T();import{writeFileSync as C,readFileSync as tt,existsSync as et,mkdirSync as ht}from"node:fs";import{join as F,dirname as ft}from"node:path";import c from"chalk";var at="__WORKFLOW_GRAPH_LOG__",I=c.gray("\u2502"),lt=c.gray("\u250C"),G=c.gray("\u2514"),W=c.green("\u25C6"),U=c.hex("#c084fc")("\u25C6"),j=c.hex("#2dd4bf")("\u25C6"),A=c.red("\u25C6"),B=`${I} `,H=2;function J(r){return r<1e3?`${r}ms`:`${(r/1e3).toFixed(1)}s`}function K(r,t){return(e,o,s)=>{if(typeof e!="string")return r(e,o,s);let n=process.stdout.columns||120,a="";for(let g=0;g<e.length;g++){let p=e[g];t.lineStart&&(a+=B,t.col=H,t.lineStart=!1),p===`
15
15
  `?(a+=p,t.lineStart=!0,t.col=0,t.inEsc=!1):p==="\x1B"?(t.inEsc=!0,a+=p):t.inEsc?(a+=p,(p>="A"&&p<="Z"||p>="a"&&p<="z")&&(t.inEsc=!1)):(t.col++,a+=p,t.col>=n&&(a+=`
16
- ${B}`,t.col=J))}return r(a,o,s)}}var x=class{constructor(){this._currentNode=null,this._origStdoutWrite=null,this._origStderrWrite=null,this._emitWorkflowGraphMarkers=String(process.env.ZIBBY_EMIT_GRAPH_MARKERS||"").trim()==="1"||String(process.env.ZIBBY_WORKFLOW_GRAPH_LOG_MARKERS||"").trim()==="1"}get isInsideNode(){return this._currentNode!==null}_startIntercepting(){this._origStdoutWrite=process.stdout.write.bind(process.stdout),this._origStderrWrite=process.stderr.write.bind(process.stderr);let t={lineStart:!0,col:0,inEsc:!1},e={lineStart:!0,col:0,inEsc:!1};this._outState=t,this._errState=e,process.stdout.write=K(this._origStdoutWrite,t),process.stderr.write=K(this._origStderrWrite,e)}_stopIntercepting(){this._origStdoutWrite&&(this._outState&&!this._outState.lineStart&&this._origStdoutWrite(`
16
+ ${B}`,t.col=H))}return r(a,o,s)}}var x=class{constructor(){this._currentNode=null,this._origStdoutWrite=null,this._origStderrWrite=null,this._emitWorkflowGraphMarkers=String(process.env.ZIBBY_EMIT_GRAPH_MARKERS||"").trim()==="1"||String(process.env.ZIBBY_WORKFLOW_GRAPH_LOG_MARKERS||"").trim()==="1"}get isInsideNode(){return this._currentNode!==null}_startIntercepting(){this._origStdoutWrite=process.stdout.write.bind(process.stdout),this._origStderrWrite=process.stderr.write.bind(process.stderr);let t={lineStart:!0,col:0,inEsc:!1},e={lineStart:!0,col:0,inEsc:!1};this._outState=t,this._errState=e,process.stdout.write=K(this._origStdoutWrite,t),process.stderr.write=K(this._origStderrWrite,e)}_stopIntercepting(){this._origStdoutWrite&&(this._outState&&!this._outState.lineStart&&this._origStdoutWrite(`
17
17
  `),process.stdout.write=this._origStdoutWrite),this._origStderrWrite&&(this._errState&&!this._errState.lineStart&&this._origStderrWrite(`
18
18
  `),process.stderr.write=this._origStderrWrite),this._origStdoutWrite=null,this._origStderrWrite=null}_rawWrite(t){(this._origStdoutWrite||process.stdout.write.bind(process.stdout))(`${t}
19
19
  `)}_emitGraphLogMarker(t){if(!this._emitWorkflowGraphMarkers)return;let e=`${at}${JSON.stringify(t)}
@@ -22,8 +22,8 @@ ${B}`,t.col=J))}return r(a,o,s)}}var x=class{constructor(){this._currentNode=nul
22
22
  `)):process.stdout.write.bind(process.stdout)(`${t} ${e}
23
23
  `)}step(t){this._origStdoutWrite?this._writeDot(W,t):process.stdout.write.bind(process.stdout)(`${I} ${W} ${t}
24
24
  `)}stepInfo(t){this.step(t)}stepTool(t){this._origStdoutWrite?this._writeDot(U,t):process.stdout.write.bind(process.stdout)(`${I} ${U} ${t}
25
- `)}stepMemory(t){let e=u.hex("#2dd4bf")(t);this._origStdoutWrite?this._writeDot(j,e):process.stdout.write.bind(process.stdout)(`${I} ${j} ${e}
26
- `)}stepFail(t){this._origStdoutWrite?this._writeDot(A,u.red(t)):process.stdout.write.bind(process.stdout)(`${I} ${A} ${u.red(t)}
27
- `)}nodeStart(t){this._currentNode=t,this._emitGraphLogMarker({phase:"node_begin",node:t}),this._rawWrite(`${lt} ${t}`),this._startIntercepting()}nodeComplete(t,e={}){this._stopIntercepting();let{duration:o,details:s}=e;if(s)for(let a of s)this._rawWrite(`${W} ${a}`);let n=o?u.dim(` ${H(o)}`):"";this._rawWrite(`${G} ${u.green("done")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:t}),this._rawWrite("")}nodeFailed(t,e,o={}){this._stopIntercepting();let{duration:s}=o,n=s?u.dim(` ${H(s)}`):"";this._rawWrite(`${A} ${u.red(e)}`),this._rawWrite(`${G} ${u.red("failed")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:t}),this._rawWrite("")}route(t,e){this._rawWrite(u.dim(` ${t} \u2192 ${e}`)),this._rawWrite("")}graphComplete(){}},Y=new x;var L=".session-info.json";var M=class{constructor(t){if(this.config=t,this.name=t.name,this.prompt=t.prompt,this.outputSchema=t.outputSchema,!this.outputSchema&&!t._isCustomCode)throw new Error(`Node '${this.name}' must define outputSchema (Zod schema). This defines the contract for what the node returns to state.`);this.isZodSchema=this.outputSchema&&typeof this.outputSchema._def<"u",this.parser=t.outputSchema&&!this.isZodSchema?new b(t.outputSchema):null,this.retries=t.retries||0,this.onComplete=t.onComplete,this.customExecute=t.execute}async execute(t,e){let o=()=>e&&typeof e.getAll=="function"?e.getAll():t,s=i=>e&&typeof e.get=="function"?e.get(i):t?.[i];if(typeof this.customExecute=="function"){l.debug(`[workflow] node '${this.name}': custom execute (skipping LLM)`);try{let i=await this.customExecute(t);return typeof i=="object"&&i!==null&&i.success===!1?{success:!1,error:i.error||"Node execution failed",raw:i.raw||null}:this.isZodSchema?(l.debug(`[workflow] node '${this.name}': validating output schema`),{success:!0,output:this.outputSchema.parse(i),raw:null}):{success:!0,output:i,raw:null}}catch(i){return l.error(`[workflow] node '${this.name}' failed: ${i.message}`),i.name==="ZodError"&&l.error(`Schema errors: ${JSON.stringify(i.issues||i.errors,null,2)}`),{success:!1,error:i.message,raw:null}}}let n=typeof this.prompt=="function"?this.prompt(o()):this.prompt,a=s("_skillHints");a&&(n=`${a}
25
+ `)}stepMemory(t){let e=c.hex("#2dd4bf")(t);this._origStdoutWrite?this._writeDot(j,e):process.stdout.write.bind(process.stdout)(`${I} ${j} ${e}
26
+ `)}stepFail(t){this._origStdoutWrite?this._writeDot(A,c.red(t)):process.stdout.write.bind(process.stdout)(`${I} ${A} ${c.red(t)}
27
+ `)}nodeStart(t){this._currentNode=t,this._emitGraphLogMarker({phase:"node_begin",node:t}),this._rawWrite(`${lt} ${t}`),this._startIntercepting()}nodeComplete(t,e={}){this._stopIntercepting();let{duration:o,details:s}=e;if(s)for(let a of s)this._rawWrite(`${W} ${a}`);let n=o?c.dim(` ${J(o)}`):"";this._rawWrite(`${G} ${c.green("done")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:t}),this._rawWrite("")}nodeFailed(t,e,o={}){this._stopIntercepting();let{duration:s}=o,n=s?c.dim(` ${J(s)}`):"";this._rawWrite(`${A} ${c.red(e)}`),this._rawWrite(`${G} ${c.red("failed")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:t}),this._rawWrite("")}route(t,e){this._rawWrite(c.dim(` ${t} \u2192 ${e}`)),this._rawWrite("")}graphComplete(){}},Y=new x;var L=".session-info.json";var M=class{constructor(t){if(this.config=t,this.name=t.name,this.prompt=t.prompt,this.outputSchema=t.outputSchema,!this.outputSchema&&!t._isCustomCode)throw new Error(`Node '${this.name}' must define outputSchema (Zod schema). This defines the contract for what the node returns to state.`);this.isZodSchema=this.outputSchema&&typeof this.outputSchema._def<"u",this.parser=t.outputSchema&&!this.isZodSchema?new b(t.outputSchema):null,this.retries=t.retries||0,this.onComplete=t.onComplete,this.customExecute=t.execute}async execute(t,e){let o=()=>e&&typeof e.getAll=="function"?e.getAll():t,s=i=>e&&typeof e.get=="function"?e.get(i):t?.[i];if(typeof this.customExecute=="function"){l.debug(`[workflow] node '${this.name}': custom execute (skipping LLM)`);try{let i=await this.customExecute(t);return typeof i=="object"&&i!==null&&i.success===!1?{success:!1,error:i.error||"Node execution failed",raw:i.raw||null}:this.isZodSchema?(l.debug(`[workflow] node '${this.name}': validating output schema`),{success:!0,output:this.outputSchema.parse(i),raw:null}):{success:!0,output:i,raw:null}}catch(i){return l.error(`[workflow] node '${this.name}' failed: ${i.message}`),i.name==="ZodError"&&l.error(`Schema errors: ${JSON.stringify(i.issues||i.errors,null,2)}`),{success:!1,error:i.message,raw:null}}}let n=typeof this.prompt=="function"?this.prompt(o()):this.prompt,a=s("_skillHints");a&&(n=`${a}
28
28
 
29
- ${n}`);let g=o(),p=g.cwd||process.cwd(),_=g.sessionPath;try{if(_){let i=F(_,L);if(et(i)){let f=JSON.parse(tt(i,"utf-8"));f.currentNode=this.name,C(i,JSON.stringify(f,null,2),"utf-8")}let h=F(_,"..",L);if(et(h))try{let f=JSON.parse(tt(h,"utf-8"));f.currentNode=this.name,C(h,JSON.stringify(f,null,2),"utf-8")}catch{}}}catch(i){l.debug(`[workflow] could not update session info: ${i.message}`)}let E=null;for(let i=0;i<=this.retries;i++)try{l.debug(`[workflow] node '${this.name}' attempt ${i}`);let h=o().config||{},f=h.agents||{},y=this.config.agent??f[this.name]??null,O={state:o()};y&&(O.preferredAgent=y);let $={workspace:p,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:_,config:h,nodeName:this.name,timeout:this.config?.timeout||3e5},R=t?._coreInvokeAgent;R||(R=(await Promise.resolve().then(()=>(Q(),X))).invokeAgent);let m=await R(n,O,$),d,S;if(typeof m=="string"?(d=m,S=null):m.structured?(d=m.raw||JSON.stringify(m.structured,null,2),S=m.structured):(d=m.raw||JSON.stringify(m,null,2),S=m.extracted||null),_)try{let c=F(_,this.name,"raw_stream_output.txt");ht(ft(c),{recursive:!0}),C(c,typeof d=="string"?d:JSON.stringify(d),"utf-8")}catch(c){l.debug(`[workflow] could not save raw output: ${c.message}`)}if(this.isZodSchema&&S){l.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(S,null,2)}`);let c=S;if(typeof this.onComplete=="function")try{c=await this.onComplete(o(),S)}catch(ot){l.warn(`[workflow] onComplete hook failed: ${ot.message}`)}return{success:!0,output:c,raw:d}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(o(),{raw:d}),raw:d}}catch(c){throw new Error(`onComplete failed: ${c.message}`,{cause:c})}if(this.parser){let c=this.parser.parse(d);return l.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(c,null,2)}`),Y.step("Output parsed"),{success:!0,output:c,raw:d}}return{success:!0,output:d,raw:d}}catch(h){E=h,i<this.retries&&l.info(`[workflow] node '${this.name}' failed, retrying (${i+1}/${this.retries})\u2026`)}return{success:!1,error:E.message,raw:null}}},rt=class extends M{constructor(t){super({...t,_isCustomCode:!0}),this.condition=t.condition}async execute(t,e){let o=e&&typeof e.getAll=="function"?e.getAll():t;return{success:!0,output:{nextNode:this.condition(o)},raw:null}}};export{rt as ConditionalNode,M as Node};
29
+ ${n}`);let g=o(),p=g.cwd||process.cwd(),_=g.sessionPath;try{if(_){let i=F(_,L);if(et(i)){let f=JSON.parse(tt(i,"utf-8"));f.currentNode=this.name,C(i,JSON.stringify(f,null,2),"utf-8")}let h=F(_,"..",L);if(et(h))try{let f=JSON.parse(tt(h,"utf-8"));f.currentNode=this.name,C(h,JSON.stringify(f,null,2),"utf-8")}catch{}}}catch(i){l.debug(`[workflow] could not update session info: ${i.message}`)}let E=null;for(let i=0;i<=this.retries;i++)try{l.debug(`[workflow] node '${this.name}' attempt ${i}`);let h=o().config||{},f=h.agents||{},y=this.config.agent??f[this.name]??null,O={state:o()};y&&(O.preferredAgent=y);let $={workspace:p,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:_,config:h,nodeName:this.name,timeout:this.config?.timeout||3e5},R=t?._coreInvokeAgent;R||(R=(await Promise.resolve().then(()=>(Q(),X))).invokeAgent);let m=await R(n,O,$),d,S;if(typeof m=="string"?(d=m,S=null):m.structured?(d=m.raw||JSON.stringify(m.structured,null,2),S=m.structured):(d=m.raw||JSON.stringify(m,null,2),S=m.extracted||null),_)try{let u=F(_,this.name,"raw_stream_output.txt");ht(ft(u),{recursive:!0}),C(u,typeof d=="string"?d:JSON.stringify(d),"utf-8")}catch(u){l.debug(`[workflow] could not save raw output: ${u.message}`)}if(this.isZodSchema&&S){l.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(S,null,2)}`);let u=S;if(typeof this.onComplete=="function")try{u=await this.onComplete(o(),S)}catch(ot){l.warn(`[workflow] onComplete hook failed: ${ot.message}`)}return{success:!0,output:u,raw:d}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(o(),{raw:d}),raw:d}}catch(u){throw new Error(`onComplete failed: ${u.message}`,{cause:u})}if(this.parser){let u=this.parser.parse(d);return l.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(u,null,2)}`),Y.step("Output parsed"),{success:!0,output:u,raw:d}}return{success:!0,output:d,raw:d}}catch(h){E=h,i<this.retries&&l.info(`[workflow] node '${this.name}' failed, retrying (${i+1}/${this.retries})\u2026`)}return{success:!1,error:E.message,raw:null}}},rt=class extends M{constructor(t){super({...t,_isCustomCode:!0}),this.condition=t.condition}async execute(t,e){let o=e&&typeof e.getAll=="function"?e.getAll():t;return{success:!0,output:{nextNode:this.condition(o)},raw:null}}};export{rt as ConditionalNode,M as Node};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@zibby/agent-workflow",
3
- "version": "0.4.6",
3
+ "version": "0.4.7",
4
4
  "description": "Graph-based AI agent workflow orchestration. Bring your own agent strategies.",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",