@zibby/agent-workflow 0.4.9 → 0.4.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,7 +1,7 @@
1
1
  # @zibby/agent-workflow
2
2
 
3
3
  [![npm version](https://img.shields.io/npm/v/@zibby/agent-workflow.svg)](https://www.npmjs.com/package/@zibby/agent-workflow)
4
- [![CI](https://github.com/ZibbyHQ/agent-workflow/actions/workflows/ci.yml/badge.svg)](https://github.com/ZibbyHQ/agent-workflow/actions/workflows/ci.yml)
4
+ [![CI](https://github.com/ZibbyDev/agent-workflow/actions/workflows/ci.yml/badge.svg)](https://github.com/ZibbyDev/agent-workflow/actions/workflows/ci.yml)
5
5
  [![Types](https://img.shields.io/npm/types/@zibby/agent-workflow.svg)](https://www.npmjs.com/package/@zibby/agent-workflow)
6
6
  [![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](./LICENSE)
7
7
 
@@ -19,6 +19,7 @@ export namespace SKILLS {
19
19
  let JIRA: string;
20
20
  let GITHUB: string;
21
21
  let GITLAB: string;
22
+ let FIGMA: string;
22
23
  let GIT: string;
23
24
  let SLACK: string;
24
25
  let LARK: string;
package/dist/constants.js CHANGED
@@ -1 +1 @@
1
- var o=".zibby/output",t="sessions",I=".session-info.json",_=".zibby-stop",s="result.json",r="raw_stream_output.txt",n="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",REVIEW_MEMORY:"review-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"},e=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];export{e as CI_ENV_VARS,o as DEFAULT_OUTPUT_BASE,n as EVENTS_FILE,r as RAW_OUTPUT_FILE,s as RESULT_FILE,t as SESSIONS_DIR,I as SESSION_INFO_FILE,E as SKILLS,_ as STOP_REQUEST_FILE};
1
+ var o=".zibby/output",t="sessions",I=".session-info.json",_=".zibby-stop",s="result.json",r="raw_stream_output.txt",n="events.json",E={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GITLAB:"gitlab",FIGMA:"figma",GIT:"git",SLACK:"slack",LARK:"lark",CHAT_NOTIFY:"chat_notify",SENTRY:"sentry",MEMORY:"memory",CHAT_MEMORY:"chat-memory",REVIEW_MEMORY:"review-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"},e=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];export{e as CI_ENV_VARS,o as DEFAULT_OUTPUT_BASE,n as EVENTS_FILE,r as RAW_OUTPUT_FILE,s as RESULT_FILE,t as SESSIONS_DIR,I as SESSION_INFO_FILE,E as SKILLS,_ as STOP_REQUEST_FILE};
@@ -1,4 +1,4 @@
1
- var Pt=Object.defineProperty;var we=(o=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(o,{get:(e,t)=>(typeof require<"u"?require:e)[t]}):o)(function(o){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+o+'" is not supported')});var se=(o,e)=>()=>(o&&(e=o(o=0)),e);var We=(o,e)=>{for(var t in e)Pt(o,t,{get:e[t],enumerable:!0})};var Ge,Bt,ce,y,F=se(()=>{Ge=()=>{},Bt={debug:Ge,info:Ge,warn:(...o)=>console.warn("[workflow]",...o),error:(...o)=>console.error("[workflow]",...o)},ce={impl:Bt},y={debug:(...o)=>ce.impl.debug?.(...o),info:(...o)=>ce.impl.info?.(...o),warn:(...o)=>ce.impl.warn?.(...o),error:(...o)=>ce.impl.error?.(...o)}});var et=se(()=>{});var tt={};We(tt,{clearSkills:()=>Wt,getAllSkills:()=>Ut,getSkill:()=>X,hasSkill:()=>jt,listSkillIds:()=>Ft,registerSkill:()=>Lt});function Lt(o){if(!o||typeof o.id!="string")throw new Error("Skill definition must include a string id");K.set(o.id,Object.freeze({...o}))}function X(o){return K.get(o)||null}function jt(o){return K.has(o)}function Ut(){return new Map(K)}function Ft(){return Array.from(K.keys())}function Wt(){K.clear()}var be,K,le=se(()=>{be=Symbol.for("@zibby/agent-workflow.skills");globalThis[be]||(globalThis[be]=new Map);K=globalThis[be]});var Q={};We(Q,{getAgentStrategy:()=>ot,invokeAgent:()=>Jt,listStrategies:()=>Ht,registerStrategy:()=>Gt});function Gt(o){if(!o||typeof o.getName!="function"||typeof o.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let e=W.findIndex(t=>t.getName()===o.getName());e>=0?W[e]=o:W.push(o)}function Ht(){return W.map(o=>o.getName())}function ot(o={}){let{state:e={},preferredAgent:t=null}=o,r=t||e.agentType||process.env.AGENT_TYPE;if(!r){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}`)}y.debug(`[workflow] agent selection: requested=${r}`);let s=W.find(n=>n.getName()===r);if(!s){let n=W.map(a=>a.getName()).join(", ")||"none registered";throw new Error(`Unknown agent '${r}'. Available: ${n}`)}if(!s.canHandle(o))throw new Error(`Agent '${r}' is not available in this environment. Check credentials/environment.`);return y.debug(`[workflow] using agent: ${s.getName()}`),s}async function Jt(o,e={},t={}){let r=ot(e),s=e.state?.config||t.config||{},n=s.models||{},a=t.nodeName&&n[t.nodeName]||null,i=n.default||null,u=s.agent?.[r.name]?.model||null,l=a||i||u||t.model||null,d={...t,model:l,workspace:e.state?.workspace||t.workspace,schema:t.schema||e.schema,images:t.images||e.images||[],skills:t.skills||e.skills||[],config:s},p=o,c=d.skills||[];if(c.length>0&&!t.skipPromptFragments){let w=c.map(_=>{let $=X(_)?.promptFragment;return typeof $=="function"?$():$}).filter(Boolean);w.length>0&&(p+=`
1
+ var Pt=Object.defineProperty;var we=(o=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(o,{get:(e,t)=>(typeof require<"u"?require:e)[t]}):o)(function(o){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+o+'" is not supported')});var se=(o,e)=>()=>(o&&(e=o(o=0)),e);var We=(o,e)=>{for(var t in e)Pt(o,t,{get:e[t],enumerable:!0})};var Ge,Bt,ce,y,F=se(()=>{Ge=()=>{},Bt={debug:Ge,info:Ge,warn:(...o)=>console.warn("[workflow]",...o),error:(...o)=>console.error("[workflow]",...o)},ce={impl:Bt},y={debug:(...o)=>ce.impl.debug?.(...o),info:(...o)=>ce.impl.info?.(...o),warn:(...o)=>ce.impl.warn?.(...o),error:(...o)=>ce.impl.error?.(...o)}});var et=se(()=>{});var tt={};We(tt,{clearSkills:()=>Wt,getAllSkills:()=>Ut,getSkill:()=>X,hasSkill:()=>jt,listSkillIds:()=>Ft,registerSkill:()=>Lt});function Lt(o){if(!o||typeof o.id!="string")throw new Error("Skill definition must include a string id");K.set(o.id,Object.freeze({...o}))}function X(o){return K.get(o)||null}function jt(o){return K.has(o)}function Ut(){return new Map(K)}function Ft(){return Array.from(K.keys())}function Wt(){K.clear()}var be,K,le=se(()=>{be=Symbol.for("@zibby/agent-workflow.skills");globalThis[be]||(globalThis[be]=new Map);K=globalThis[be]});var Q={};We(Q,{getAgentStrategy:()=>ot,invokeAgent:()=>Jt,listStrategies:()=>Ht,registerStrategy:()=>Gt});function Gt(o){if(!o||typeof o.getName!="function"||typeof o.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let e=W.findIndex(t=>t.getName()===o.getName());e>=0?W[e]=o:W.push(o)}function Ht(){return W.map(o=>o.getName())}function ot(o={}){let{state:e={},preferredAgent:t=null}=o,r=t||e.agentType||process.env.AGENT_TYPE;if(!r){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}`)}y.debug(`[workflow] agent selection: requested=${r}`);let s=W.find(n=>n.getName()===r);if(!s){let n=W.map(a=>a.getName()).join(", ")||"none registered";throw new Error(`Unknown agent '${r}'. Available: ${n}`)}if(!s.canHandle(o))throw new Error(`Agent '${r}' is not available in this environment. Check credentials/environment.`);return y.debug(`[workflow] using agent: ${s.getName()}`),s}async function Jt(o,e={},t={}){let r=ot(e),s=e.state?.config||t.config||{},n=s.models||{},a=t.nodeName&&n[t.nodeName]||null,i=n.default||null,u=s.agent?.[r.name]?.model||null,c=a||i||u||t.model||null,d={...t,model:c,workspace:e.state?.workspace||t.workspace,schema:t.schema||e.schema,images:t.images||e.images||[],skills:t.skills||e.skills||[],config:s},p=o,l=d.skills||[];if(l.length>0&&!t.skipPromptFragments){let w=l.map(_=>{let b=X(_)?.promptFragment;return typeof b=="function"?b():b}).filter(Boolean);w.length>0&&(p+=`
2
2
 
3
3
  ${w.join(`
4
4
 
@@ -26,15 +26,15 @@ ${Ze}`,e.col=ze))}return o(a,r,s)}}var Ee=class{constructor(){this._currentNode=
26
26
  `)}stepFail(e){this._origStdoutWrite?this._writeDot(Ie,x.red(e)):process.stdout.write.bind(process.stdout)(`${V} ${Ie} ${x.red(e)}
27
27
  `)}nodeStart(e){this._currentNode=e,this._emitGraphLogMarker({phase:"node_begin",node:e}),this._rawWrite(`${Dt} ${e}`),this._startIntercepting()}nodeComplete(e,t={}){this._stopIntercepting();let{duration:r,details:s}=t;if(s)for(let a of s)this._rawWrite(`${_e} ${a}`);let n=r?x.dim(` ${Ke(r)}`):"";this._rawWrite(`${He} ${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(` ${Ke(s)}`):"";this._rawWrite(`${Ie} ${x.red(t)}`),this._rawWrite(`${He} ${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 Ee;var ue=".zibby/output",Ve="sessions",z=".session-info.json",Xe=".zibby-stop";var Qe=["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=Te(l,z);if(nt(p)){let h=JSON.parse(rt(p,"utf-8"));h.currentNode=this.name,ve(p,JSON.stringify(h,null,2),"utf-8")}let c=Te(l,"..",z);if(nt(c))try{let h=JSON.parse(rt(c,"utf-8"));h.currentNode=this.name,ve(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=Te(l,this.name,"raw_stream_output.txt");Yt(Zt(m),{recursive:!0}),ve(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 qt,existsSync as H,statSync as dt,readdirSync as pt,rmSync as Vt}from"node:fs";import{spawn as lt}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as Xt}from"node:url";import{AsyncLocalStorage as zt}from"node:async_hooks";var xe=new zt;function te(){let o=xe.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 st(o,e){let t=xe.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 xe.run(r,e)}var Ae=new Map,ke=new Map,it=new Map;function at(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");Ae.set(o,e),ke.set(o,"ready"),it.set(o,{...t,cachedAt:Date.now()})}function ct(o,e){ke.set(o,"failed"),it.set(o,{error:e?.message||String(e),failedAt:Date.now()}),Ae.delete(o)}function ut(o){return ke.get(o)==="ready"?Ae.get(o):null}var pe=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function Qt(){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 eo(){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 to({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 oo(o,e){let t=U(e,".ready"),r=U(e,"graph.mjs");if(H(t)&&H(r))return;qt(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=lt("curl",["-fsSL",o],{stdio:["ignore","pipe","inherit"]}),p=lt("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 ro(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(Xt(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 ft(o,e={}){if(!o||typeof o!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let t=te(),r;try{r=eo()}catch(g){throw g}y.debug(`[in-process subgraph] begin '${o}' parent=${t.executionId||"<root>"}`);let s=await to({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=Qt();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=ut(o);if(!h){let g=U(pe,`${d}@${l||"0"}`);try{await oo(i,g);try{so()}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 ro(g),at(o,h,{workflowUuid:d,version:l,runtimeTag:a,cacheDir:g})}catch(m){throw ct(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 st({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 no(o){let e=0,t=[o];for(;t.length;){let r=t.pop(),s;try{s=dt(r)}catch{continue}if(s.isDirectory()){let n;try{n=pt(r)}catch{continue}for(let a of n)t.push(U(r,a))}else e+=s.size}return e}function so({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=pt(pe),t=[],r=0;for(let i of e){let u=U(pe,i),l;try{l=dt(u)}catch{continue}let d=l.isDirectory()?no(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{Vt(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 io=2e3,ao=600*1e3,co=new Set(["completed","failed","canceled","timeout"]);function uo(){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 lo(){let o=process.env.PROJECT_ID;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return o}function po(){let o=process.env.PROJECT_API_TOKEN;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return o}function fo(){return process.env.EXECUTION_ID||null}function ht(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 gt(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 ft(o,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),m=ht(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=uo(),n=lo(),a=po(),i=fo(),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:ao,w=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:io,_=`${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,co.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=ht(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 mt,readFileSync as ho}from"node:fs";import{join as Oe,dirname as St}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=St(Oe(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=Oe(t,u);mt(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=Oe(n,e);if(mt(a))try{s.unshift(await this.loadFile(a))}catch(u){console.warn(`[workflow] could not load ${e} from ${a}: ${u.message}`)}let i=St(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(),c=i.sessionPath;try{if(c){let p=Te(c,z);if(nt(p)){let h=JSON.parse(rt(p,"utf-8"));h.currentNode=this.name,ve(p,JSON.stringify(h,null,2),"utf-8")}let l=Te(c,"..",z);if(nt(l))try{let h=JSON.parse(rt(l,"utf-8"));h.currentNode=this.name,ve(l,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 l=r().config||{},h=l.agents||{},w=this.config.agent??h[this.name]??null,_={state:r()};w&&(_.preferredAgent=w);let b={workspace:u,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:c,config:l,nodeName:this.name,timeout:this.config?.timeout||3e5},$=e?._coreInvokeAgent;$||($=(await Promise.resolve().then(()=>(ee(),Q))).invokeAgent);let f=await $(n,_,b),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),c)try{let m=Te(c,this.name,"raw_stream_output.txt");Yt(Zt(m),{recursive:!0}),ve(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(l){d=l,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 qt,existsSync as H,statSync as dt,readdirSync as pt,rmSync as Vt}from"node:fs";import{spawn as lt}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as Xt}from"node:url";import{AsyncLocalStorage as zt}from"node:async_hooks";var xe=new zt;function te(){let o=xe.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 st(o,e){let t=xe.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 xe.run(r,e)}var Ae=new Map,ke=new Map,it=new Map;function at(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");Ae.set(o,e),ke.set(o,"ready"),it.set(o,{...t,cachedAt:Date.now()})}function ct(o,e){ke.set(o,"failed"),it.set(o,{error:e?.message||String(e),failedAt:Date.now()}),Ae.delete(o)}function ut(o){return ke.get(o)==="ready"?Ae.get(o):null}var pe=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function Qt(){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 eo(){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 to({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 oo(o,e){let t=U(e,".ready"),r=U(e,"graph.mjs");if(H(t)&&H(r))return;qt(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,c)=>{let d=lt("curl",["-fsSL",o],{stdio:["ignore","pipe","inherit"]}),p=lt("tar",["-xzf","-","-C",e],{stdio:["pipe","inherit","inherit"]});d.stdout.pipe(p.stdin);let l,h,w=()=>{if(l!==void 0&&h!==void 0){if(l!==0)return c(new Error(`curl exited ${l}`));if(h!==0)return c(new Error(`tar exited ${h}`));u()}};d.on("close",_=>{l=_,w()}),p.on("close",_=>{h=_,w()}),d.on("error",c),p.on("error",c)});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 ro(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(Xt(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 ft(o,e={}){if(!o||typeof o!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let t=te(),r;try{r=eo()}catch(g){throw g}y.debug(`[in-process subgraph] begin '${o}' parent=${t.executionId||"<root>"}`);let s=await to({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:c,workflowUuid:d,bundleReady:p}=s,l=Qt();if(a&&a!==l)throw await G({apiBase:r.apiBase,authToken:r.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:`runtimeTag mismatch: parent=${l} child=${a}`,code:"RUNTIME_MISMATCH"}}}),new k("runtime-mismatch",`${l} 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=ut(o);if(!h){let g=U(pe,`${d}@${c||"0"}`);try{await oo(i,g);try{so()}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 ro(g),at(o,h,{workflowUuid:d,version:c,runtimeTag:a,cacheDir:g})}catch(m){throw ct(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(),b=await(typeof h=="function"&&h.prototype?.buildGraph?new h:h).buildGraph(),$={...e.input||{}},f,I;try{f=await st({executionId:n,parentExecutionId:t.executionId,conversationId:e.conversationId!==void 0?e.conversationId:t.conversationId,dispatchMode:"inprocess"},()=>b.run(e.parentAgent,$,{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 no(o){let e=0,t=[o];for(;t.length;){let r=t.pop(),s;try{s=dt(r)}catch{continue}if(s.isDirectory()){let n;try{n=pt(r)}catch{continue}for(let a of n)t.push(U(r,a))}else e+=s.size}return e}function so({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=pt(pe),t=[],r=0;for(let i of e){let u=U(pe,i),c;try{c=dt(u)}catch{continue}let d=c.isDirectory()?no(u):c.size;r+=d,t.push({name:i,full:u,size:d,mtimeMs:c.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{Vt(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 io=2e3,ao=600*1e3,co=new Set(["completed","failed","canceled","timeout"]);function uo(){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 lo(){let o=process.env.PROJECT_ID;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return o}function po(){let o=process.env.PROJECT_API_TOKEN;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return o}function fo(){return process.env.EXECUTION_ID||null}function ht(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 gt(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 ft(o,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),m=ht(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=uo(),n=lo(),a=po(),i=fo(),u=`${s}/projects/${encodeURIComponent(n)}/workflows/${encodeURIComponent(o)}/trigger`,c={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(c)});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(),l=p?.data?.jobId||p?.jobId;if(!l)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=${l} (not waiting)`),{jobId:l,status:"accepted",workflow:o};let h=Number.isFinite(e.timeoutMs)?e.timeoutMs:ao,w=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:io,_=`${s}/executions/${encodeURIComponent(l)}`,b=Date.now()+h,$="accepted",f=0;for(;Date.now()<b;){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 ${l} returned ${g.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${l}: ${g.status}`)}let m=await g.json(),R=m?.data||m?.execution||m;if($=R?.status||$,co.has($)){if($!=="completed"){let S=new Error(`Sub-graph '${o}' (${l}) ended in status '${$}'`);throw S.subgraphJobId=l,S.subgraphStatus=$,S}let T=R?.finalState||R?.state||{},M=ht(T,e.output);return y.info(`[sub-graph] '${o}' (${l}) completed after ${f} polls`),M}}let I=new Error(`Sub-graph '${o}' (${l}) timed out after ${Math.round(h/1e3)}s (last status: ${$})`);throw I.subgraphJobId=l,I.subgraphStatus=$,I}import{existsSync as mt,readFileSync as ho}from"node:fs";import{join as Oe,dirname as St}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=St(Oe(t,e));for(let u of n){let c=await this.findAndMergeContextFiles(u,i,t);if(c){let d=u.replace(/\.[^.]+$/,"").toLowerCase();s[d]=c}}}let a=r.discovery||{};for(let[i,u]of Object.entries(a))try{let c=Oe(t,u);mt(c)&&(s[i]=await this.loadFile(c))}catch(c){console.warn(`[workflow] could not load context '${i}' from '${u}': ${c.message}`)}return s}static async findAndMergeContextFiles(e,t,r){let s=[],n=t;for(;n.startsWith(r);){let a=Oe(n,e);if(mt(a))try{s.unshift(await this.loadFile(a))}catch(u){console.warn(`[workflow] could not load ${e} from ${a}: ${u.message}`)}let i=St(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
33
  `):s.every(a=>typeof a=="object")?Object.assign({},...s):s[s.length-1]}static async loadFile(e){let t=ho(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 _t,existsSync as Ne,writeFileSync as wt,unlinkSync as go}from"node:fs";import{join as J,resolve as It}from"node:path";import{config as mo}from"dotenv";import{zodToJsonSchema as yt}from"zod-to-json-schema";import{z as he}from"zod";import So from"handlebars";function wo({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 yo(){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 _o(){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 It(String(e).trim())}catch{return String(e).trim()}}function Io(){yo()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function Eo({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 bo(o={}){let e=Qe.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 $o({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=It(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=bo(e),a="generated")}n=n??Date.now();let d=e.paths?.output||ue;s=J(o,d,Ve,i)}let u=!Ne(s);return u&&_t(s,{recursive:!0}),(u||a!=="initialState.sessionPath")&&wo({traceFrom:r,sessionId:i,sessionPath:s,idSource:a,mkdirFresh:u}),Eo({sessionPath:s,sessionId:i}),{sessionPath:s,sessionId:i,sessionTimestamp:n}}var ge=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={},gt(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 $=typeof he?.toJSONSchema=="function"?he.toJSONSchema(d.outputSchema):yt(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;if(typeof he?.toJSONSchema=="function")try{return he.toJSONSchema(l)}catch{}try{return yt(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();mo({path:J(a,".env")});let i=t.config||{};if(!i||Object.keys(i).length===0)try{let E=J(a,".zibby.config.js");Ne(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=_o(),c=t.sessionPath||p;c||Io();let{sessionPath:h,sessionTimestamp:w,sessionId:_}=$o({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(),tt)),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=[],Be=i?.recursionLimit??100,Tt=0;try{for(;S&&S!=="END";){if(++Tt>Be)throw new Error(`Workflow exceeded recursion limit (${Be}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let O=J(h,Xe);if(Ne(O)){try{go(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 Me=JSON.stringify({sessionPath:h,sessionTimestamp:w,currentNode:S,createdAt:new Date().toISOString(),config:f.get("config")}),xt=J(h,z);wt(xt,Me,"utf-8");let De=f.get("config")?.paths?.output||ue,At=J(a,De,z);_t(J(a,De),{recursive:!0});try{wt(At,Me,"utf-8")}catch{}let Le=t.onPipelineProgress;if(typeof Le=="function")try{Le({cwd:a,sessionPath:h,sessionId:_,outputBase:f.get("config")?.paths?.output||ue,currentNode:S})}catch{}let kt=(this.resolvedToolsMap||{})[S]||null;f.set("_currentNodeTools",kt);let Ot=f.get("nodeConfigs")||{};f.set("_currentNodeConfig",Ot[S]||{}),A.nodeStart(S);let je=Date.now(),re=this.nodePrompts.get(S);if(!this._invokeAgent){let C=await Promise.resolve().then(()=>(ee(),Q));this._invokeAgent=C.invokeAgent}let Nt=this._invokeAgent,Se={},Ct=N.config?.skills||[];for(let C of Ct){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"&&(Se={...Se,...v})}catch(v){console.warn(`[graph] skill '${C}' invokeAgentOptions threw: ${v.message}`)}}let Ue=async(C,P,v={})=>{let B=Nt(C,P,{...Se,...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})})])},Fe={state:f,invokeAgent:async(C={},P={})=>{let v=P.prompt||"";if(re){let B=this._compiledPrompts.get(S);B||(B=So.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 Ue(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:Ue,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(Fe,f),f.getAll(),f):v=await N.execute(Fe,f);let B=Date.now()-je;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()-je}),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 Ce=Symbol.for("@zibby/agent-workflow.nodes");globalThis[Ce]||(globalThis[Ce]=new Map);var Pe=globalThis[Ce];function vo(o,e){Pe.set(o,e)}function Et(o){return Pe.get(o)}function Re(o){return Pe.has(o)}vo("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=To(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 To(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 yo(){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 _o(){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 It(String(e).trim())}catch{return String(e).trim()}}function Io(){yo()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function Eo({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 bo(o={}){let e=Qe.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 $o({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 c=It(String(process.env.ZIBBY_SESSION_PATH));c&&(s=c,a="ZIBBY_SESSION_PATH")}catch{}let i;if(s)i=String(s).split(/[/\\]/).filter(Boolean).pop(),n==null&&(n=Date.now());else{let c=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(c)i=c,a="ZIBBY_SESSION_ID";else{let p=e.sessionId!=null?String(e.sessionId).trim():"";p&&p!=="last"?(i=p,a="config.sessionId"):(i=bo(e),a="generated")}n=n??Date.now();let d=e.paths?.output||ue;s=J(o,d,Ve,i)}let u=!Ne(s);return u&&_t(s,{recursive:!0}),(u||a!=="initialState.sessionPath")&&wo({traceFrom:r,sessionId:i,sessionPath:s,idSource:a,mkdirFresh:u}),Eo({sessionPath:s,sessionId:i}),{sessionPath:s,sessionId:i,sessionTimestamp:n}}var ge=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 c=u?.state&&typeof u.state.getAll=="function"?u.state.getAll():u,d;return typeof n.input=="function"?d=n.input(c):n.input&&typeof n.input=="object"?d=n.input:d={},gt(n.workflow,{input:d,async:n.async===!0,conversationId:typeof n.conversationId=="function"?n.conversationId(c):n.conversationId,output:n.output,timeoutMs:n.timeoutMs,pollIntervalMs:n.pollIntervalMs,signal:c?._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],c=a;a=()=>u(t,c,s,n)}return a()}serialize(){let e=[],t={};for(let[c,d]of this.nodes){let p=this.nodeTypeMap.get(c)||c;e.push({id:c,type:p,data:{nodeType:p,label:c}});let l={};d._isCustomCode&&typeof d.execute=="function"&&(l.customCode=d.execute.toString());let h=this.nodePrompts.get(c);if(h&&(l.prompt=h),typeof d.customExecute=="function"&&(l.executeCode=d.customExecute.toString()),d.outputSchema)if(typeof d.outputSchema._def<"u"){let b=null;if(typeof he?.toJSONSchema=="function")try{b=he.toJSONSchema(d.outputSchema)}catch{}if(!b)try{b=yt(d.outputSchema,{target:"openApi3"})}catch{}l.outputSchema=b?{jsonSchema:b,variables:this._flattenJsonSchemaToVariables(b)}:{schema:d.outputSchema}}else l.outputSchema={schema:d.outputSchema};let w=(this.resolvedToolsMap||{})[c];w?.toolIds&&(l.tools=w.toolIds);let _=Array.isArray(d?.config?.skills)?d.config.skills:Array.isArray(d?.skills)?d.skills:null;_&&_.length>0&&(l.skills=[..._]),Object.keys(l).length>0&&(t[c]=l)}let r=[];for(let[c,d]of this.edges)if(typeof d=="string")r.push({source:c,target:d});else if(d.conditional){let p=this.conditionalCodeMap.get(c)||d.routes.toString(),l=this._inferConditionalTargets(d.routes,d.labels),h=d.labels||{};for(let w of l){let _={source:c,target:w,data:{conditionalCode:p}};h[w]&&(_.label=h[w]),r.push(_)}}let s=c=>{if(!c)return null;if(typeof he?.toJSONSchema=="function")try{return he.toJSONSchema(c)}catch{}try{return yt(c,{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,t){let r=e.toString(),s=new Set,n=/(['"])((?:\\.|(?!\1).)*?)\1|`((?:\\.|[^`$]|\$(?!\{))*?)`/g,a;for(;(a=n.exec(r))!==null;){let c=a[2]!==void 0?a[2]:a[3];c!==void 0&&c!==""&&s.add(c)}let i=new Set(["END","START","__end__","__start__"]);for(let c of this.nodes.keys())i.add(c);if(t&&typeof t=="object")for(let c of Object.keys(t))i.add(c);let u=new Set;for(let c of s)i.has(c)&&u.add(c);if(u.size===0){let c=/return\s+['"]([^'"]+)['"]/g,d;for(;(d=c.exec(r))!==null;)u.add(d[1])}return[...u]}_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(c=>c?.passed===!0).length,u=n.some(c=>c?.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();mo({path:J(a,".env")});let i=t.config||{};if(!i||Object.keys(i).length===0)try{let E=J(a,".zibby.config.js");Ne(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 c=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=_o(),l=t.sessionPath||p;l||Io();let{sessionPath:h,sessionTimestamp:w,sessionId:_}=$o({cwd:a,config:i,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:l,sessionTimestamp:t.sessionTimestamp}});A.step(`Session ${_}`);let b=await fe.loadContext(t.specPath||"",a,c);Object.keys(b).length>0&&A.step(`Context loaded: ${Object.keys(b).join(", ")}`);let $=t.outputPath;!$&&t.specPath&&(e?.calculateOutputPath?$=e.calculateOutputPath(t.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${t.specPath})`));let f=new ie({...t,config:i,agentType:u,outputPath:$,sessionPath:h,sessionTimestamp:w,context:b,resolvedTools:this.resolvedToolsMap||{},_signal:s.signal}),I=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:g}=await Promise.resolve().then(()=>(le(),tt)),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=[],Be=i?.recursionLimit??100,Tt=0;try{for(;S&&S!=="END";){if(++Tt>Be)throw new Error(`Workflow exceeded recursion limit (${Be}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let O=J(h,Xe);if(Ne(O)){try{go(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 Me=JSON.stringify({sessionPath:h,sessionTimestamp:w,currentNode:S,createdAt:new Date().toISOString(),config:f.get("config")}),xt=J(h,z);wt(xt,Me,"utf-8");let De=f.get("config")?.paths?.output||ue,At=J(a,De,z);_t(J(a,De),{recursive:!0});try{wt(At,Me,"utf-8")}catch{}let Le=t.onPipelineProgress;if(typeof Le=="function")try{Le({cwd:a,sessionPath:h,sessionId:_,outputBase:f.get("config")?.paths?.output||ue,currentNode:S})}catch{}let kt=(this.resolvedToolsMap||{})[S]||null;f.set("_currentNodeTools",kt);let Ot=f.get("nodeConfigs")||{};f.set("_currentNodeConfig",Ot[S]||{}),A.nodeStart(S);let je=Date.now(),re=this.nodePrompts.get(S);if(!this._invokeAgent){let C=await Promise.resolve().then(()=>(ee(),Q));this._invokeAgent=C.invokeAgent}let Nt=this._invokeAgent,Se={},Ct=N.config?.skills||[];for(let C of Ct){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"&&(Se={...Se,...v})}catch(v){console.warn(`[graph] skill '${C}' invokeAgentOptions threw: ${v.message}`)}}let Ue=async(C,P,v={})=>{let B=Nt(C,P,{...Se,...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})})])},Fe={state:f,invokeAgent:async(C={},P={})=>{let v=P.prompt||"";if(re){let B=this._compiledPrompts.get(S);B||(B=So.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 Ue(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:Ue,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(Fe,f),f.getAll(),f):v=await N.execute(Fe,f);let B=Date.now()-je;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()-je}),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 Ce=Symbol.for("@zibby/agent-workflow.nodes");globalThis[Ce]||(globalThis[Ce]=new Map);var Pe=globalThis[Ce];function vo(o,e){Pe.set(o,e)}function Et(o){return Pe.get(o)}function Re(o){return Pe.has(o)}vo("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=To(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 To(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 c=i.split(".").reduce((l,h)=>l?.[h],e);if(c===void 0)continue;let d=typeof c=="string"?c:c?.raw??JSON.stringify(c,null,2),p=i.replace(/_/g," ").replace(/\b\w/g,l=>l.toUpperCase());n.push(`## ${p}
38
38
  ${d}`),i.includes(".")||a.add(u)}return n.length===0?o:`${o}
39
39
 
40
40
  ---
@@ -42,4 +42,4 @@ ${d}`),i.includes(".")||a.add(u)}return n.length===0?o:`${o}
42
42
 
43
43
  ${n.join(`
44
44
 
45
- `)}`}le();F();var xo={};function $t(o,e){if(Array.isArray(e))return bt(e);let t=xo[o];return!t||t.length===0?null:bt(t)}function bt(o){if(!Array.isArray(o)||o.length===0)return null;let e=[],t={},r=[];for(let s of o){let n=X(s);if(!n){y.warn(`[workflow] unknown skill "${s}" \u2014 skipping`);continue}r.push(s);for(let a of n.tools||[])e.push({name:a.name,description:a.description,input_schema:a.input_schema||{type:"object",properties:{}}});if(!t[n.serverName])if(typeof n.resolve=="function"){let a=n.resolve();a&&(t[n.serverName]={...a,toolPrefix:s})}else{let a={};for(let i of n.envKeys||[]){let u=process.env[i];u&&(a[i]=u)}t[n.serverName]={command:n.command,args:[...n.args||[]],env:a,toolPrefix:s}}}return r.length===0?null:{toolIds:r,claudeTools:e,mcpServers:t}}F();function Pr(o,e={}){let{nodes:t,edges:r,nodeConfigs:s={}}=o;if(!Array.isArray(t)||t.length===0)throw new D("Graph must have at least one node");if(!Array.isArray(r))throw new D("Graph edges must be an array");let n=new ge(e);e.stateSchema&&n.setStateSchema(e.stateSchema);let a=new Set,i=new Map,u={};for(let c of t){let h=me(c);i.set(c.id,{...c,resolvedType:h}),h==="decision"&&a.add(c.id)}for(let[c,h]of i){if(a.has(c))continue;let w=h.resolvedType,_=s[c]||{},$=$t(w,_.tools);$&&(u[c]=$);let b={};_.prompt&&(b.prompt=_.prompt);let f=Re(w);if(y.debug(`[workflow] compiler: node "${c}" type="${w}" registered=${f}`),_.customCode&&!f)n.addNode(c,vt(c,_.customCode,_),b),n.setNodeType(c,w);else if(f){let I=Et(w);I.factory?n.addNode(c,I.create(c,{..._,resolvedTools:$}),b):n.addNode(c,I,b),n.setNodeType(c,w)}else if(_.executeCode)n.addNode(c,vt(c,_.executeCode,_),b),n.setNodeType(c,w);else throw new D(`Unknown node type "${w}" for node "${c}". Did you forget to register it?`)}n.resolvedToolsMap=u;let l=new Set;for(let c of r)a.has(c.target)||l.add(c.target);let d=t.find(c=>!a.has(c.id)&&!l.has(c.id));if(!d)throw new D("Could not determine entry point: no node without incoming edges found");n.setEntryPoint(d.id);let p=Ao(r,"source");for(let c of r)if(!a.has(c.source))if(a.has(c.target)){let h=c.target,w=p.get(h)||[];if(w.length===0)throw new D(`Decision node "${h}" has no outgoing edges`);let _=ko(h,w,a);n.addConditionalEdges(c.source,_)}else n.addEdge(c.source,c.target);return n}function Rr(o){let e=[];if(!o||typeof o!="object")return{valid:!1,errors:["Config must be a non-null object"]};if((!Array.isArray(o.nodes)||o.nodes.length===0)&&e.push("Graph must have at least one node"),Array.isArray(o.edges)||e.push("Graph edges must be an array"),e.length>0)return{valid:!1,errors:e};let t=o.nodeConfigs||{};for(let i of o.nodes){let u=me(i);if(u==="decision"||Re(u))continue;let l=t[i.id]||{};l.customCode||l.executeCode||e.push(`Unknown node type "${u}" for node "${i.id}". Register it or provide customCode/executeCode.`)}let r=new Set(o.nodes.map(i=>i.id));for(let i of o.edges)r.has(i.source)||e.push(`Edge references unknown source node "${i.source}"`),r.has(i.target)||e.push(`Edge references unknown target node "${i.target}"`);let s=new Set(o.nodes.filter(i=>me(i)==="decision").map(i=>i.id)),n=new Set;for(let i of o.edges)s.has(i.target)||n.add(i.target);let a=o.nodes.filter(i=>!s.has(i.id)&&!n.has(i.id));a.length===0?e.push("No entry point found (every node has incoming edges)"):a.length>1&&e.push(`Multiple entry points found: ${a.map(i=>i.id).join(", ")}`);for(let i of s){let u=o.edges.filter(d=>d.source===i);u.length===0&&e.push(`Decision node "${i}" has no outgoing edges`),u.some(d=>d.data?.conditionalCode||d.conditionalCode)||e.push(`Decision node "${i}" outgoing edges have no conditionalCode`)}return{valid:e.length===0,errors:e}}function Br(o){return!o||!Array.isArray(o.nodes)?[]:o.nodes.filter(e=>me(e)!=="decision").map(e=>e.id)}function me(o){let e=o.data?.nodeType||o.data?.type||o.type;return e==="workflowNode"||e==="custom"||e==="default"?o.id:e}function Ao(o,e){let t=new Map;for(let r of o){let s=r[e];t.has(s)||t.set(s,[]),t.get(s).push(r)}return t}function ko(o,e,t){let r=e.find(i=>i.data?.conditionalCode||i.conditionalCode);if(!r)throw new D(`Decision node "${o}" has no conditionalCode on its outgoing edges`);let s=r.data?.conditionalCode||r.conditionalCode,n=new Set(e.map(i=>i.target).filter(i=>!t.has(i))),a;try{let u=new Function(`return (${s})`)();a=l=>{let d=u(l);return n.has(d)||y.warn(`[workflow] conditional route from "${o}" returned "${d}" which is not in valid targets: ${[...n].join(", ")}`),d}}catch(i){throw new D(`Failed to compile conditionalCode for "${o}": ${i.message}`)}return a}function vt(o,e,t={}){let r;try{r=new Function("invokeAgent","require","console",`return (${e})`)}catch(a){throw new D(`Failed to compile customCode for node "${o}": ${a.message}`)}let s=r(async(...a)=>{let{invokeAgent:i}=await Promise.resolve().then(()=>(ee(),Q));return i(...a)},typeof we<"u"?we:void 0,console),n=null;return t.outputSchema&&(n=t.outputSchema.jsonSchema||t.outputSchema),{name:o,_isCustomCode:!0,outputSchema:n,execute:async a=>{try{let i=await s(a);return typeof i=="object"&&"success"in i?i:{success:!0,output:i,raw:null}}catch(i){return{success:!1,error:i.message,raw:null}}}}}var D=class extends Error{constructor(e){super(e),this.name="CompilationError"}};export{D as CompilationError,Pr as compileGraph,Br as extractSteps,Rr as validateGraphConfig};
45
+ `)}`}le();F();var xo={};function $t(o,e){if(Array.isArray(e))return bt(e);let t=xo[o];return!t||t.length===0?null:bt(t)}function bt(o){if(!Array.isArray(o)||o.length===0)return null;let e=[],t={},r=[];for(let s of o){let n=X(s);if(!n){y.warn(`[workflow] unknown skill "${s}" \u2014 skipping`);continue}r.push(s);for(let a of n.tools||[])e.push({name:a.name,description:a.description,input_schema:a.input_schema||{type:"object",properties:{}}});if(!t[n.serverName])if(typeof n.resolve=="function"){let a=n.resolve();a&&(t[n.serverName]={...a,toolPrefix:s})}else{let a={};for(let i of n.envKeys||[]){let u=process.env[i];u&&(a[i]=u)}t[n.serverName]={command:n.command,args:[...n.args||[]],env:a,toolPrefix:s}}}return r.length===0?null:{toolIds:r,claudeTools:e,mcpServers:t}}F();function Pr(o,e={}){let{nodes:t,edges:r,nodeConfigs:s={}}=o;if(!Array.isArray(t)||t.length===0)throw new D("Graph must have at least one node");if(!Array.isArray(r))throw new D("Graph edges must be an array");let n=new ge(e);e.stateSchema&&n.setStateSchema(e.stateSchema);let a=new Set,i=new Map,u={};for(let l of t){let h=me(l);i.set(l.id,{...l,resolvedType:h}),h==="decision"&&a.add(l.id)}for(let[l,h]of i){if(a.has(l))continue;let w=h.resolvedType,_=s[l]||{},b=$t(w,_.tools);b&&(u[l]=b);let $={};_.prompt&&($.prompt=_.prompt);let f=Re(w);if(y.debug(`[workflow] compiler: node "${l}" type="${w}" registered=${f}`),_.customCode&&!f)n.addNode(l,vt(l,_.customCode,_),$),n.setNodeType(l,w);else if(f){let I=Et(w);I.factory?n.addNode(l,I.create(l,{..._,resolvedTools:b}),$):n.addNode(l,I,$),n.setNodeType(l,w)}else if(_.executeCode)n.addNode(l,vt(l,_.executeCode,_),$),n.setNodeType(l,w);else throw new D(`Unknown node type "${w}" for node "${l}". Did you forget to register it?`)}n.resolvedToolsMap=u;let c=new Set;for(let l of r)a.has(l.target)||c.add(l.target);let d=t.find(l=>!a.has(l.id)&&!c.has(l.id));if(!d)throw new D("Could not determine entry point: no node without incoming edges found");n.setEntryPoint(d.id);let p=Ao(r,"source");for(let l of r)if(!a.has(l.source))if(a.has(l.target)){let h=l.target,w=p.get(h)||[];if(w.length===0)throw new D(`Decision node "${h}" has no outgoing edges`);let _=ko(h,w,a);n.addConditionalEdges(l.source,_)}else n.addEdge(l.source,l.target);return n}function Rr(o){let e=[];if(!o||typeof o!="object")return{valid:!1,errors:["Config must be a non-null object"]};if((!Array.isArray(o.nodes)||o.nodes.length===0)&&e.push("Graph must have at least one node"),Array.isArray(o.edges)||e.push("Graph edges must be an array"),e.length>0)return{valid:!1,errors:e};let t=o.nodeConfigs||{};for(let i of o.nodes){let u=me(i);if(u==="decision"||Re(u))continue;let c=t[i.id]||{};c.customCode||c.executeCode||e.push(`Unknown node type "${u}" for node "${i.id}". Register it or provide customCode/executeCode.`)}let r=new Set(o.nodes.map(i=>i.id));for(let i of o.edges)r.has(i.source)||e.push(`Edge references unknown source node "${i.source}"`),r.has(i.target)||e.push(`Edge references unknown target node "${i.target}"`);let s=new Set(o.nodes.filter(i=>me(i)==="decision").map(i=>i.id)),n=new Set;for(let i of o.edges)s.has(i.target)||n.add(i.target);let a=o.nodes.filter(i=>!s.has(i.id)&&!n.has(i.id));a.length===0?e.push("No entry point found (every node has incoming edges)"):a.length>1&&e.push(`Multiple entry points found: ${a.map(i=>i.id).join(", ")}`);for(let i of s){let u=o.edges.filter(d=>d.source===i);u.length===0&&e.push(`Decision node "${i}" has no outgoing edges`),u.some(d=>d.data?.conditionalCode||d.conditionalCode)||e.push(`Decision node "${i}" outgoing edges have no conditionalCode`)}return{valid:e.length===0,errors:e}}function Br(o){return!o||!Array.isArray(o.nodes)?[]:o.nodes.filter(e=>me(e)!=="decision").map(e=>e.id)}function me(o){let e=o.data?.nodeType||o.data?.type||o.type;return e==="workflowNode"||e==="custom"||e==="default"?o.id:e}function Ao(o,e){let t=new Map;for(let r of o){let s=r[e];t.has(s)||t.set(s,[]),t.get(s).push(r)}return t}function ko(o,e,t){let r=e.find(i=>i.data?.conditionalCode||i.conditionalCode);if(!r)throw new D(`Decision node "${o}" has no conditionalCode on its outgoing edges`);let s=r.data?.conditionalCode||r.conditionalCode,n=new Set(e.map(i=>i.target).filter(i=>!t.has(i))),a;try{let u=new Function(`return (${s})`)();a=c=>{let d=u(c);return n.has(d)||y.warn(`[workflow] conditional route from "${o}" returned "${d}" which is not in valid targets: ${[...n].join(", ")}`),d}}catch(i){throw new D(`Failed to compile conditionalCode for "${o}": ${i.message}`)}return a}function vt(o,e,t={}){let r;try{r=new Function("invokeAgent","require","console",`return (${e})`)}catch(a){throw new D(`Failed to compile customCode for node "${o}": ${a.message}`)}let s=r(async(...a)=>{let{invokeAgent:i}=await Promise.resolve().then(()=>(ee(),Q));return i(...a)},typeof we<"u"?we:void 0,console),n=null;return t.outputSchema&&(n=t.outputSchema.jsonSchema||t.outputSchema),{name:o,_isCustomCode:!0,outputSchema:n,execute:async a=>{try{let i=await s(a);return typeof i=="object"&&"success"in i?i:{success:!0,output:i,raw:null}}catch(i){return{success:!1,error:i.message,raw:null}}}}}var D=class extends Error{constructor(e){super(e),this.name="CompilationError"}};export{D as CompilationError,Pr as compileGraph,Br as extractSteps,Rr as validateGraphConfig};
package/dist/graph.d.ts CHANGED
@@ -119,7 +119,7 @@ export class WorkflowGraph {
119
119
  };
120
120
  });
121
121
  };
122
- _inferConditionalTargets(routeFn: any): any[];
122
+ _inferConditionalTargets(routeFn: any, labels: any): any[];
123
123
  _flattenJsonSchemaToVariables(jsonSchema: any, prefix?: string): any;
124
124
  _flattenSchema(schema: any, prefix?: string): any;
125
125
  _formatLabel(str: any): any;
package/dist/graph.js CHANGED
@@ -1,4 +1,4 @@
1
- var be=Object.defineProperty;var et=(r,t)=>()=>(r&&(t=r(r=0)),t);var Ct=(r,t)=>{for(var e in t)be(r,e,{get:t[e],enumerable:!0})};var Mt,Te,st,w,K=et(()=>{Mt=()=>{},Te={debug:Mt,info:Mt,warn:(...r)=>console.warn("[workflow]",...r),error:(...r)=>console.error("[workflow]",...r)},st={impl:Te},w={debug:(...r)=>st.impl.debug?.(...r),info:(...r)=>st.impl.info?.(...r),warn:(...r)=>st.impl.warn?.(...r),error:(...r)=>st.impl.error?.(...r)}});var Zt=et(()=>{});var zt={};Ct(zt,{clearSkills:()=>Ne,getAllSkills:()=>xe,getSkill:()=>mt,hasSkill:()=>Oe,listSkillIds:()=>Pe,registerSkill:()=>ke});function ke(r){if(!r||typeof r.id!="string")throw new Error("Skill definition must include a string id");Z.set(r.id,Object.freeze({...r}))}function mt(r){return Z.get(r)||null}function Oe(r){return Z.has(r)}function xe(){return new Map(Z)}function Pe(){return Array.from(Z.keys())}function Ne(){Z.clear()}var gt,Z,St=et(()=>{gt=Symbol.for("@zibby/agent-workflow.skills");globalThis[gt]||(globalThis[gt]=new Map);Z=globalThis[gt]});var _t={};Ct(_t,{getAgentStrategy:()=>Kt,invokeAgent:()=>Ce,listStrategies:()=>Be,registerStrategy:()=>Re});function Re(r){if(!r||typeof r.getName!="function"||typeof r.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let t=U.findIndex(e=>e.getName()===r.getName());t>=0?U[t]=r:U.push(r)}function Be(){return U.map(r=>r.getName())}function Kt(r={}){let{state:t={},preferredAgent:e=null}=r,o=e||t.agentType||process.env.AGENT_TYPE;if(!o){let n=U.map(i=>i.getName()).join(", ")||"none registered";throw new Error(`No agent specified. Set agentType in state or AGENT_TYPE env var. Available: ${n}`)}w.debug(`[workflow] agent selection: requested=${o}`);let s=U.find(n=>n.getName()===o);if(!s){let n=U.map(i=>i.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 w.debug(`[workflow] using agent: ${s.getName()}`),s}async function Ce(r,t={},e={}){let o=Kt(t),s=t.state?.config||e.config||{},n=s.models||{},i=e.nodeName&&n[e.nodeName]||null,a=n.default||null,c=s.agent?.[o.name]?.model||null,u=i||a||c||e.model||null,l={...e,model:u,workspace:t.state?.workspace||e.workspace,schema:e.schema||t.schema,images:e.images||t.images||[],skills:e.skills||t.skills||[],config:s},p=r,h=l.skills||[];if(h.length>0&&!e.skipPromptFragments){let I=h.map(b=>{let $=mt(b)?.promptFragment;return typeof $=="function"?$():$}).filter(Boolean);I.length>0&&(p+=`
1
+ var be=Object.defineProperty;var et=(r,t)=>()=>(r&&(t=r(r=0)),t);var Ct=(r,t)=>{for(var e in t)be(r,e,{get:t[e],enumerable:!0})};var Mt,Te,st,w,K=et(()=>{Mt=()=>{},Te={debug:Mt,info:Mt,warn:(...r)=>console.warn("[workflow]",...r),error:(...r)=>console.error("[workflow]",...r)},st={impl:Te},w={debug:(...r)=>st.impl.debug?.(...r),info:(...r)=>st.impl.info?.(...r),warn:(...r)=>st.impl.warn?.(...r),error:(...r)=>st.impl.error?.(...r)}});var Zt=et(()=>{});var zt={};Ct(zt,{clearSkills:()=>Ne,getAllSkills:()=>xe,getSkill:()=>mt,hasSkill:()=>Oe,listSkillIds:()=>Pe,registerSkill:()=>ke});function ke(r){if(!r||typeof r.id!="string")throw new Error("Skill definition must include a string id");Z.set(r.id,Object.freeze({...r}))}function mt(r){return Z.get(r)||null}function Oe(r){return Z.has(r)}function xe(){return new Map(Z)}function Pe(){return Array.from(Z.keys())}function Ne(){Z.clear()}var gt,Z,St=et(()=>{gt=Symbol.for("@zibby/agent-workflow.skills");globalThis[gt]||(globalThis[gt]=new Map);Z=globalThis[gt]});var _t={};Ct(_t,{getAgentStrategy:()=>Kt,invokeAgent:()=>Ce,listStrategies:()=>Be,registerStrategy:()=>Re});function Re(r){if(!r||typeof r.getName!="function"||typeof r.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let t=U.findIndex(e=>e.getName()===r.getName());t>=0?U[t]=r:U.push(r)}function Be(){return U.map(r=>r.getName())}function Kt(r={}){let{state:t={},preferredAgent:e=null}=r,o=e||t.agentType||process.env.AGENT_TYPE;if(!o){let n=U.map(i=>i.getName()).join(", ")||"none registered";throw new Error(`No agent specified. Set agentType in state or AGENT_TYPE env var. Available: ${n}`)}w.debug(`[workflow] agent selection: requested=${o}`);let s=U.find(n=>n.getName()===o);if(!s){let n=U.map(i=>i.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 w.debug(`[workflow] using agent: ${s.getName()}`),s}async function Ce(r,t={},e={}){let o=Kt(t),s=t.state?.config||e.config||{},n=s.models||{},i=e.nodeName&&n[e.nodeName]||null,a=n.default||null,l=s.agent?.[o.name]?.model||null,c=i||a||l||e.model||null,u={...e,model:c,workspace:t.state?.workspace||e.workspace,schema:e.schema||t.schema,images:e.images||t.images||[],skills:e.skills||t.skills||[],config:s},p=r,h=u.skills||[];if(h.length>0&&!e.skipPromptFragments){let I=h.map(b=>{let $=mt(b)?.promptFragment;return typeof $=="function"?$():$}).filter(Boolean);I.length>0&&(p+=`
2
2
 
3
3
  ${I.join(`
4
4
 
@@ -9,10 +9,10 @@ PRIORITY OVERRIDE \u2014 THE FOLLOWING INSTRUCTIONS TAKE PRECEDENCE OVER ALL PRE
9
9
  \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501
10
10
 
11
11
  ${m}
12
- `),w.debug(`[workflow] prompt length: ${p.length} chars`),o.invoke(p,l)}var wt,U,yt=et(()=>{Zt();K();St();wt=Symbol.for("@zibby/agent-workflow.strategies");globalThis[wt]||(globalThis[wt]=[]);U=globalThis[wt]});var $e=new Set(["__proto__","constructor","prototype"]);function pt(r){if($e.has(r))throw new Error(`Invalid state key: "${r}"`)}var rt=class{constructor(t={}){this._state=Object.create(null),Object.assign(this._state,{messages:[],errors:[],artifacts:{},metadata:{},...t}),this._history=[]}get(t){return this._state[t]}set(t,e){pt(t),this._history.push({...this._state}),this._state[t]=e}update(t){let e=Object.getOwnPropertyNames(t);for(let o of e)pt(o);this._history.push({...this._state});for(let o of e)this._state[o]=t[o]}append(t,e){pt(t),this._history.push({...this._state}),Array.isArray(this._state[t])||(this._state[t]=[]),this._state[t].push(e)}getAll(){return{...this._state}}rollback(){this._history.length>0&&(this._state=this._history.pop())}};var ot=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:
12
+ `),w.debug(`[workflow] prompt length: ${p.length} chars`),o.invoke(p,u)}var wt,U,yt=et(()=>{Zt();K();St();wt=Symbol.for("@zibby/agent-workflow.strategies");globalThis[wt]||(globalThis[wt]=[]);U=globalThis[wt]});var $e=new Set(["__proto__","constructor","prototype"]);function pt(r){if($e.has(r))throw new Error(`Invalid state key: "${r}"`)}var rt=class{constructor(t={}){this._state=Object.create(null),Object.assign(this._state,{messages:[],errors:[],artifacts:{},metadata:{},...t}),this._history=[]}get(t){return this._state[t]}set(t,e){pt(t),this._history.push({...this._state}),this._state[t]=e}update(t){let e=Object.getOwnPropertyNames(t);for(let o of e)pt(o);this._history.push({...this._state});for(let o of e)this._state[o]=t[o]}append(t,e){pt(t),this._history.push({...this._state}),Array.isArray(this._state[t])||(this._state[t]=[]),this._state[t].push(e)}getAll(){return{...this._state}}rollback(){this._history.length>0&&(this._state=this._history.pop())}};var ot=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}};K();import{writeFileSync as It,readFileSync as Vt,existsSync as qt,mkdirSync as Me}from"node:fs";import{join as Et,dirname as De}from"node:path";import A from"chalk";var ve="__WORKFLOW_GRAPH_LOG__",V=A.gray("\u2502"),Ae=A.gray("\u250C"),Dt=A.gray("\u2514"),dt=A.green("\u25C6"),Lt=A.hex("#c084fc")("\u25C6"),jt=A.hex("#2dd4bf")("\u25C6"),ht=A.red("\u25C6"),Ut=`${V} `,Wt=2;function Ft(r){return r<1e3?`${r}ms`:`${(r/1e3).toFixed(1)}s`}function Gt(r,t){return(e,o,s)=>{if(typeof e!="string")return r(e,o,s);let n=process.stdout.columns||120,i="";for(let a=0;a<e.length;a++){let c=e[a];t.lineStart&&(i+=Ut,t.col=Wt,t.lineStart=!1),c===`
15
- `?(i+=c,t.lineStart=!0,t.col=0,t.inEsc=!1):c==="\x1B"?(t.inEsc=!0,i+=c):t.inEsc?(i+=c,(c>="A"&&c<="Z"||c>="a"&&c<="z")&&(t.inEsc=!1)):(t.col++,i+=c,t.col>=n&&(i+=`
14
+ `)}`);return t}};K();import{writeFileSync as It,readFileSync as Vt,existsSync as qt,mkdirSync as Me}from"node:fs";import{join as Et,dirname as De}from"node:path";import A from"chalk";var ve="__WORKFLOW_GRAPH_LOG__",V=A.gray("\u2502"),Ae=A.gray("\u250C"),Dt=A.gray("\u2514"),dt=A.green("\u25C6"),Lt=A.hex("#c084fc")("\u25C6"),jt=A.hex("#2dd4bf")("\u25C6"),ht=A.red("\u25C6"),Ut=`${V} `,Wt=2;function Ft(r){return r<1e3?`${r}ms`:`${(r/1e3).toFixed(1)}s`}function Gt(r,t){return(e,o,s)=>{if(typeof e!="string")return r(e,o,s);let n=process.stdout.columns||120,i="";for(let a=0;a<e.length;a++){let l=e[a];t.lineStart&&(i+=Ut,t.col=Wt,t.lineStart=!1),l===`
15
+ `?(i+=l,t.lineStart=!0,t.col=0,t.inEsc=!1):l==="\x1B"?(t.inEsc=!0,i+=l):t.inEsc?(i+=l,(l>="A"&&l<="Z"||l>="a"&&l<="z")&&(t.inEsc=!1)):(t.col++,i+=l,t.col>=n&&(i+=`
16
16
  ${Ut}`,t.col=Wt))}return r(i,o,s)}}var ft=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=Gt(this._origStdoutWrite,t),process.stderr.write=Gt(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}
@@ -26,12 +26,12 @@ ${Ut}`,t.col=Wt))}return r(i,o,s)}}var ft=class{constructor(){this._currentNode=
26
26
  `)}stepFail(t){this._origStdoutWrite?this._writeDot(ht,A.red(t)):process.stdout.write.bind(process.stdout)(`${V} ${ht} ${A.red(t)}
27
27
  `)}nodeStart(t){this._currentNode=t,this._emitGraphLogMarker({phase:"node_begin",node:t}),this._rawWrite(`${Ae} ${t}`),this._startIntercepting()}nodeComplete(t,e={}){this._stopIntercepting();let{duration:o,details:s}=e;if(s)for(let i of s)this._rawWrite(`${dt} ${i}`);let n=o?A.dim(` ${Ft(o)}`):"";this._rawWrite(`${Dt} ${A.green("done")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:t}),this._rawWrite("")}nodeFailed(t,e,o={}){this._stopIntercepting();let{duration:s}=o,n=s?A.dim(` ${Ft(s)}`):"";this._rawWrite(`${ht} ${A.red(e)}`),this._rawWrite(`${Dt} ${A.red("failed")}${n}`),this._emitGraphLogMarker({phase:"node_end",node:t}),this._rawWrite("")}route(t,e){this._rawWrite(A.dim(` ${t} \u2192 ${e}`)),this._rawWrite("")}graphComplete(){}},k=new ft;var nt=".zibby/output",Ht="sessions",Y=".session-info.json",Jt=".zibby-stop";var Yt=["CI_JOB_ID","GITHUB_RUN_ID","CIRCLE_WORKFLOW_ID","BUILD_ID"];var L=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 ot(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=p=>e&&typeof e.get=="function"?e.get(p):t?.[p];if(typeof this.customExecute=="function"){w.debug(`[workflow] node '${this.name}': custom execute (skipping LLM)`);try{let p=await this.customExecute(t);return typeof p=="object"&&p!==null&&p.success===!1?{success:!1,error:p.error||"Node execution failed",raw:p.raw||null}:this.isZodSchema?(w.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 w.error(`[workflow] node '${this.name}' failed: ${p.message}`),p.name==="ZodError"&&w.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(o()):this.prompt,i=s("_skillHints");i&&(n=`${i}
28
28
 
29
- ${n}`);let a=o(),c=a.cwd||process.cwd(),u=a.sessionPath;try{if(u){let p=Et(u,Y);if(qt(p)){let m=JSON.parse(Vt(p,"utf-8"));m.currentNode=this.name,It(p,JSON.stringify(m,null,2),"utf-8")}let h=Et(u,"..",Y);if(qt(h))try{let m=JSON.parse(Vt(h,"utf-8"));m.currentNode=this.name,It(h,JSON.stringify(m,null,2),"utf-8")}catch{}}}catch(p){w.debug(`[workflow] could not update session info: ${p.message}`)}let l=null;for(let p=0;p<=this.retries;p++)try{w.debug(`[workflow] node '${this.name}' attempt ${p}`);let h=o().config||{},m=h.agents||{},I=this.config.agent??m[this.name]??null,b={state:o()};I&&(b.preferredAgent=I);let $={workspace:c,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:u,config:h,nodeName:this.name,timeout:this.config?.timeout||3e5},T=t?._coreInvokeAgent;T||(T=(await Promise.resolve().then(()=>(yt(),_t))).invokeAgent);let d=await T(n,b,$),y,f;if(typeof d=="string"?(y=d,f=null):d.structured?(y=d.raw||JSON.stringify(d.structured,null,2),f=d.structured):(y=d.raw||JSON.stringify(d,null,2),f=d.extracted||null),u)try{let g=Et(u,this.name,"raw_stream_output.txt");Me(De(g),{recursive:!0}),It(g,typeof y=="string"?y:JSON.stringify(y),"utf-8")}catch(g){w.debug(`[workflow] could not save raw output: ${g.message}`)}if(this.isZodSchema&&f){w.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(f,null,2)}`);let g=f;if(typeof this.onComplete=="function")try{g=await this.onComplete(o(),f)}catch(B){w.warn(`[workflow] onComplete hook failed: ${B.message}`)}return{success:!0,output:g,raw:y}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(o(),{raw:y}),raw:y}}catch(g){throw new Error(`onComplete failed: ${g.message}`,{cause:g})}if(this.parser){let g=this.parser.parse(y);return w.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(g,null,2)}`),k.step("Output parsed"),{success:!0,output:g,raw:y}}return{success:!0,output:y,raw:y}}catch(h){l=h,p<this.retries&&w.info(`[workflow] node '${this.name}' failed, retrying (${p+1}/${this.retries})\u2026`)}return{success:!1,error:l.message,raw:null}}},it=class extends L{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}}};K();K();import{mkdirSync as Ue,existsSync as F,statSync as se,readdirSync as ne,rmSync as We}from"node:fs";import{spawn as oe}from"node:child_process";import{join as j}from"node:path";import{pathToFileURL as Fe}from"node:url";import{AsyncLocalStorage as Le}from"node:async_hooks";var bt=new Le;function q(){let r=bt.getStore();return r||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 Xt(r,t){let e=bt.getStore()||q(),o=Object.freeze({executionId:r.executionId,parentExecutionId:r.parentExecutionId??e.executionId??null,depth:(e.depth||0)+(r.executionId!==e.executionId?1:0),conversationId:r.conversationId!==void 0?r.conversationId:e.conversationId??null,dispatchMode:r.dispatchMode??null});return bt.run(o,t)}var $t=new Map,Tt=new Map,Qt=new Map;function te(r,t,e={}){if(!r||typeof r!="string")throw new Error("subgraph-registry.register: name required");if(typeof t!="function")throw new Error("subgraph-registry.register: factory must be a function");$t.set(r,t),Tt.set(r,"ready"),Qt.set(r,{...e,cachedAt:Date.now()})}function ee(r,t){Tt.set(r,"failed"),Qt.set(r,{error:t?.message||String(t),failedAt:Date.now()}),$t.delete(r)}function re(r){return Tt.get(r)==="ready"?$t.get(r):null}var at=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function Ge(){return`node${(process.versions?.node||"").split(".")[0]||"unknown"}-${process.platform}-${process.arch}`}var O=class extends Error{constructor(t,e){super(`in-process sub-graph fallback: ${t}${e?` (${e})`:""}`),this.fallback=!0,this.reason=t,this.detail=e||null,this.name="SubgraphFallback"}};function He(){let r=(process.env.SUBGRAPH_INTERNAL_URL||"").replace(/\/$/,""),t=(process.env.PROGRESS_API_URL||"").replace(/\/executions\/?$/,""),e=r||t,o=process.env.PROJECT_ID,s=process.env.PROJECT_API_TOKEN;if(!e||!o||!s)throw new O("env","SUBGRAPH_INTERNAL_URL/PROGRESS_API_URL/PROJECT_ID/PROJECT_API_TOKEN missing");return{apiBase:e,projectId:o,authToken:s}}async function Je({apiBase:r,authToken:t,body:e}){let o;try{o=await fetch(`${r}/internal/subgraph/begin`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(e)})}catch(n){throw new O("network",`begin fetch failed: ${n.message}`)}let s=null;try{s=await o.json()}catch{}if(!o.ok){if(o.status===404){let n=new Error(`Sub-graph child '${e.childWorkflowType}' not found in project`);throw n.code="SUBGRAPH_NOT_FOUND",n.status=404,n}if(o.status===429){let n=s?.quotaInfo||{},i=new Error(`Sub-graph blocked by quota (${n.used??"?"}/${n.limit??"?"} on ${n.planId||"plan"})`);throw i.code="SUBGRAPH_QUOTA_EXCEEDED",i.status=429,i.quotaInfo=n,i}if(o.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 O("begin-status",`begin returned ${o.status}`)}return s?.data||s}async function W({apiBase:r,authToken:t,payload:e}){try{let o=await fetch(`${r}/internal/subgraph/finalize`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(e)});o.ok||w.warn(`[in-process subgraph] finalize returned ${o.status} for ${e.childExecutionId}`)}catch(o){w.warn(`[in-process subgraph] finalize failed: ${o.message}`)}}async function Ye(r,t){let e=j(t,".ready"),o=j(t,"graph.mjs");if(F(e)&&F(o))return;Ue(t,{recursive:!0});let s=j(t,".lock"),n=!1;try{let{openSync:i,closeSync:a}=await import("node:fs"),c=i(s,"wx");a(c),n=!0}catch(i){if(i.code!=="EEXIST")throw i}if(!n){let i=Date.now()+3e4;for(;Date.now()<i;){if(F(e)&&F(o))return;await new Promise(a=>setTimeout(a,100))}throw new O("bundle-extract-timeout","sibling extract did not complete within 30s")}try{await new Promise((c,u)=>{let l=oe("curl",["-fsSL",r],{stdio:["ignore","pipe","inherit"]}),p=oe("tar",["-xzf","-","-C",t],{stdio:["pipe","inherit","inherit"]});l.stdout.pipe(p.stdin);let h,m,I=()=>{if(h!==void 0&&m!==void 0){if(h!==0)return u(new Error(`curl exited ${h}`));if(m!==0)return u(new Error(`tar exited ${m}`));c()}};l.on("close",b=>{h=b,I()}),p.on("close",b=>{m=b,I()}),l.on("error",u),p.on("error",u)});let{writeFileSync:i,unlinkSync:a}=await import("node:fs");i(e,"");try{a(s)}catch{}}catch(i){try{let{unlinkSync:a}=await import("node:fs");a(s)}catch{}throw new O("bundle-extract-failed",i.message)}}async function Ze(r){let t=j(r,"graph.mjs");if(!F(t))throw new O("entry-missing",`graph.mjs missing under ${r}`);let e;try{e=await import(Fe(t).href)}catch(s){throw new O("import-failed",`${s?.code||s?.name||"unknown"}: ${s.message}`)}let o=e.default||Object.values(e).find(s=>typeof s=="function"&&s.prototype?.buildGraph);if(!o)throw new O("entry-class-missing","no buildGraph() class export found");return o}async function ie(r,t={}){if(!r||typeof r!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let e=q(),o;try{o=He()}catch(f){throw f}w.debug(`[in-process subgraph] begin '${r}' parent=${e.executionId||"<root>"}`);let s=await Je({apiBase:o.apiBase,authToken:o.authToken,body:{parentExecutionId:e.executionId,childWorkflowType:r,input:t.input||{},...t.conversationId?{conversationId:t.conversationId}:{}}}),{childExecutionId:n,runtimeTag:i,bundlePresignedUrl:a,sourcesPresignedUrl:c,workflowVersion:u,workflowUuid:l,bundleReady:p}=s,h=Ge();if(i&&i!==h)throw await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:`runtimeTag mismatch: parent=${h} child=${i}`,code:"RUNTIME_MISMATCH"}}}),new O("runtime-mismatch",`${h} vs ${i}`);if(!p||!a)throw await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:"bundle not ready for in-process; falling back to HTTP",code:"NO_BUNDLE"}}}),new O("no-bundle","workflow bundle not built yet");let m=re(r);if(!m){let f=j(at,`${l}@${u||"0"}`);try{await Ye(a,f);try{Ke()}catch{}}catch(g){throw g.fallback&&await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"failed",error:{message:g.message,code:g.reason}}}),g}try{m=await Ze(f),te(r,m,{workflowUuid:l,version:u,runtimeTag:i,cacheDir:f})}catch(g){throw ee(r,g),await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"failed",error:{message:g.message,code:g.reason||"IMPORT_FAILED"}}}),g.fallback?g:new O("import-failed",g.message)}}let I=Date.now(),$=await(typeof m=="function"&&m.prototype?.buildGraph?new m:m).buildGraph(),T={...t.input||{}},d,y;try{d=await Xt({executionId:n,parentExecutionId:e.executionId,conversationId:t.conversationId!==void 0?t.conversationId:e.conversationId,dispatchMode:"inprocess"},()=>$.run(t.parentAgent,T,{signal:t.signal})),y=d&&typeof d=="object"&&"state"in d?d.state:d}catch(f){throw await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"failed",error:{message:f.message,code:f.code||"CHILD_THREW",stack:f.stack},durationMs:Date.now()-I}}),f}if(d&&typeof d=="object"&&d.stoppedExternally){await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"canceled",finalState:y,durationMs:Date.now()-I}});let f=new Error(`Sub-graph '${r}' canceled by parent abort`);throw f.code="SUBGRAPH_CANCELED",f.subgraphJobId=n,f}return await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"completed",finalState:y,durationMs:Date.now()-I}}),{finalState:y,executionId:n}}function ze(r){let t=0,e=[r];for(;e.length;){let o=e.pop(),s;try{s=se(o)}catch{continue}if(s.isDirectory()){let n;try{n=ne(o)}catch{continue}for(let i of n)e.push(j(o,i))}else t+=s.size}return t}function Ke({cap:r=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!F(at))return{evicted:0,freedBytes:0};let t=ne(at),e=[],o=0;for(let a of t){let c=j(at,a),u;try{u=se(c)}catch{continue}let l=u.isDirectory()?ze(c):u.size;o+=l,e.push({name:a,full:c,size:l,mtimeMs:u.mtimeMs})}if(o<=r)return{evicted:0,freedBytes:0,totalBytes:o};e.sort((a,c)=>a.mtimeMs-c.mtimeMs);let s=Math.floor(r*.7),n=0,i=0;for(let a of e){if(o-n<=s)break;if(!F(j(a.full,".lock")))try{We(a.full,{recursive:!0,force:!0}),n+=a.size,i+=1}catch(c){w.debug(`[sub-graph cache] evict skip ${a.name}: ${c.message}`)}}return i>0&&w.info(`[sub-graph cache] evicted ${i} entr(y/ies), freed ${(n/1024/1024).toFixed(1)}MB`),{evicted:i,freedBytes:n,totalBytes:o-n}}catch(t){return w.debug(`[sub-graph cache] evict failed: ${t.message}`),{evicted:0,freedBytes:0}}}var Ve=2e3,qe=600*1e3,Xe=new Set(["completed","failed","canceled","timeout"]);function Qe(){let r=process.env.PROGRESS_API_URL;if(!r)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 r.replace(/\/executions\/?$/,"")}function tr(){let r=process.env.PROJECT_ID;if(!r)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return r}function er(){let r=process.env.PROJECT_API_TOKEN;if(!r)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return r}function rr(){return process.env.EXECUTION_ID||null}function ae(r,t){return t==null?r:typeof t=="function"?t(r):typeof t=="string"?t.split(".").reduce((e,o)=>e==null?e:e[o],r):r}async function ce(r,t={}){if(!r||typeof r!="string")throw new Error("dispatchSubgraph: workflowName (string) is required");let e=q(),o=Number(process.env.ZIBBY_SUBGRAPH_MAX_DEPTH||10);if((e.depth||0)>=o)throw new Error(`dispatchSubgraph('${r}'): sub-graph depth ${e.depth} reached cap of ${o}. Restructure the graph or raise ZIBBY_SUBGRAPH_MAX_DEPTH.`);if(process.env.ZIBBY_INPROCESS_SUBGRAPH!=="0"&&!t.async)try{w.debug(`[sub-graph] trying in-process for '${r}'`);let{finalState:f}=await ie(r,{input:t.input,conversationId:t.conversationId,signal:t.signal,parentAgent:t.parentAgent}),g=ae(f,t.output);return w.info(`[sub-graph] '${r}' completed in-process`),g}catch(f){if(f instanceof O||f?.fallback)w.info(`[sub-graph] in-process fallback for '${r}': ${f.reason||"unknown"} \u2014 using HTTP`);else throw f}let s=Qe(),n=tr(),i=er(),a=rr(),c=`${s}/projects/${encodeURIComponent(n)}/workflows/${encodeURIComponent(r)}/trigger`,u={input:t.input||{},...a?{parentExecutionId:a}:{},...t.conversationId?{conversationId:t.conversationId}:{}};w.info(`[sub-graph] dispatching '${r}' (${t.async?"async":"sync"}) from parent ${a||"<none>"}`);let l=await fetch(c,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${i}`},body:JSON.stringify(u)});if(!l.ok){let f=null,g="";try{f=await l.json(),g=f?.error||f?.message||JSON.stringify(f)}catch{g=await l.text().catch(()=>"")}if(l.status===429){let v=f?.quotaInfo||{},M=new Error(`Sub-graph '${r}' 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 M.code="SUBGRAPH_QUOTA_EXCEEDED",M.status=429,M.subgraph=r,M.quotaInfo=v,M}if(l.status===400){let v=new Error(`Sub-graph '${r}' rejected input: ${g}`);throw v.code="SUBGRAPH_INVALID_INPUT",v.status=400,v.subgraph=r,v.validationErrors=f?.validationErrors||null,v.missing=f?.missing||null,v}let B=new Error(`Sub-graph '${r}' trigger rejected (${l.status}): ${g}`);throw B.code="SUBGRAPH_TRIGGER_FAILED",B.status=l.status,B.subgraph=r,B}let p=await l.json(),h=p?.data?.jobId||p?.jobId;if(!h)throw new Error(`Sub-graph '${r}' trigger returned no jobId: ${JSON.stringify(p).slice(0,200)}`);if(t.async)return w.info(`[sub-graph] async dispatch of '${r}' \u2192 jobId=${h} (not waiting)`),{jobId:h,status:"accepted",workflow:r};let m=Number.isFinite(t.timeoutMs)?t.timeoutMs:qe,I=Number.isFinite(t.pollIntervalMs)?t.pollIntervalMs:Ve,b=`${s}/executions/${encodeURIComponent(h)}`,$=Date.now()+m,T="accepted",d=0;for(;Date.now()<$;){await new Promise(v=>setTimeout(v,I)),d+=1;let f=await fetch(b,{headers:{Authorization:`Bearer ${i}`}});if(!f.ok){if(f.status>=500){w.warn(`[sub-graph] status poll for ${h} returned ${f.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${h}: ${f.status}`)}let g=await f.json(),B=g?.data||g?.execution||g;if(T=B?.status||T,Xe.has(T)){if(T!=="completed"){let S=new Error(`Sub-graph '${r}' (${h}) ended in status '${T}'`);throw S.subgraphJobId=h,S.subgraphStatus=T,S}let v=B?.finalState||B?.state||{},M=ae(v,t.output);return w.info(`[sub-graph] '${r}' (${h}) completed after ${d} polls`),M}}let y=new Error(`Sub-graph '${r}' (${h}) timed out after ${Math.round(m/1e3)}s (last status: ${T})`);throw y.subgraphJobId=h,y.subgraphStatus=T,y}import{existsSync as ue,readFileSync as or}from"node:fs";import{join as vt,dirname as le}from"node:path";var ct=class{static async loadContext(t,e,o={}){let s={},n=o.filenames||["CONTEXT.md","AGENTS.md"];if(t){let a=le(vt(e,t));for(let c of n){let u=await this.findAndMergeContextFiles(c,a,e);if(u){let l=c.replace(/\.[^.]+$/,"").toLowerCase();s[l]=u}}}let i=o.discovery||{};for(let[a,c]of Object.entries(i))try{let u=vt(e,c);ue(u)&&(s[a]=await this.loadFile(u))}catch(u){console.warn(`[workflow] could not load context '${a}' from '${c}': ${u.message}`)}return s}static async findAndMergeContextFiles(t,e,o){let s=[],n=e;for(;n.startsWith(o);){let i=vt(n,t);if(ue(i))try{s.unshift(await this.loadFile(i))}catch(c){console.warn(`[workflow] could not load ${t} from ${i}: ${c.message}`)}let a=le(n);if(a===n)break;n=a}return s.length===0?null:s.every(i=>typeof i=="string")?s.join(`
29
+ ${n}`);let a=o(),l=a.cwd||process.cwd(),c=a.sessionPath;try{if(c){let p=Et(c,Y);if(qt(p)){let m=JSON.parse(Vt(p,"utf-8"));m.currentNode=this.name,It(p,JSON.stringify(m,null,2),"utf-8")}let h=Et(c,"..",Y);if(qt(h))try{let m=JSON.parse(Vt(h,"utf-8"));m.currentNode=this.name,It(h,JSON.stringify(m,null,2),"utf-8")}catch{}}}catch(p){w.debug(`[workflow] could not update session info: ${p.message}`)}let u=null;for(let p=0;p<=this.retries;p++)try{w.debug(`[workflow] node '${this.name}' attempt ${p}`);let h=o().config||{},m=h.agents||{},I=this.config.agent??m[this.name]??null,b={state:o()};I&&(b.preferredAgent=I);let $={workspace:l,schema:this.isZodSchema?this.outputSchema:null,skills:this.config.skills||[],sessionPath:c,config:h,nodeName:this.name,timeout:this.config?.timeout||3e5},T=t?._coreInvokeAgent;T||(T=(await Promise.resolve().then(()=>(yt(),_t))).invokeAgent);let d=await T(n,b,$),y,f;if(typeof d=="string"?(y=d,f=null):d.structured?(y=d.raw||JSON.stringify(d.structured,null,2),f=d.structured):(y=d.raw||JSON.stringify(d,null,2),f=d.extracted||null),c)try{let g=Et(c,this.name,"raw_stream_output.txt");Me(De(g),{recursive:!0}),It(g,typeof y=="string"?y:JSON.stringify(y),"utf-8")}catch(g){w.debug(`[workflow] could not save raw output: ${g.message}`)}if(this.isZodSchema&&f){w.info(`[workflow] node '${this.name}': output validated: ${JSON.stringify(f,null,2)}`);let g=f;if(typeof this.onComplete=="function")try{g=await this.onComplete(o(),f)}catch(B){w.warn(`[workflow] onComplete hook failed: ${B.message}`)}return{success:!0,output:g,raw:y}}if(typeof this.onComplete=="function")try{return{success:!0,output:await this.onComplete(o(),{raw:y}),raw:y}}catch(g){throw new Error(`onComplete failed: ${g.message}`,{cause:g})}if(this.parser){let g=this.parser.parse(y);return w.info(`[workflow] node '${this.name}': parsed output: ${JSON.stringify(g,null,2)}`),k.step("Output parsed"),{success:!0,output:g,raw:y}}return{success:!0,output:y,raw:y}}catch(h){u=h,p<this.retries&&w.info(`[workflow] node '${this.name}' failed, retrying (${p+1}/${this.retries})\u2026`)}return{success:!1,error:u.message,raw:null}}},it=class extends L{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}}};K();K();import{mkdirSync as Ue,existsSync as F,statSync as se,readdirSync as ne,rmSync as We}from"node:fs";import{spawn as oe}from"node:child_process";import{join as j}from"node:path";import{pathToFileURL as Fe}from"node:url";import{AsyncLocalStorage as Le}from"node:async_hooks";var bt=new Le;function q(){let r=bt.getStore();return r||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 Xt(r,t){let e=bt.getStore()||q(),o=Object.freeze({executionId:r.executionId,parentExecutionId:r.parentExecutionId??e.executionId??null,depth:(e.depth||0)+(r.executionId!==e.executionId?1:0),conversationId:r.conversationId!==void 0?r.conversationId:e.conversationId??null,dispatchMode:r.dispatchMode??null});return bt.run(o,t)}var $t=new Map,Tt=new Map,Qt=new Map;function te(r,t,e={}){if(!r||typeof r!="string")throw new Error("subgraph-registry.register: name required");if(typeof t!="function")throw new Error("subgraph-registry.register: factory must be a function");$t.set(r,t),Tt.set(r,"ready"),Qt.set(r,{...e,cachedAt:Date.now()})}function ee(r,t){Tt.set(r,"failed"),Qt.set(r,{error:t?.message||String(t),failedAt:Date.now()}),$t.delete(r)}function re(r){return Tt.get(r)==="ready"?$t.get(r):null}var at=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function Ge(){return`node${(process.versions?.node||"").split(".")[0]||"unknown"}-${process.platform}-${process.arch}`}var O=class extends Error{constructor(t,e){super(`in-process sub-graph fallback: ${t}${e?` (${e})`:""}`),this.fallback=!0,this.reason=t,this.detail=e||null,this.name="SubgraphFallback"}};function He(){let r=(process.env.SUBGRAPH_INTERNAL_URL||"").replace(/\/$/,""),t=(process.env.PROGRESS_API_URL||"").replace(/\/executions\/?$/,""),e=r||t,o=process.env.PROJECT_ID,s=process.env.PROJECT_API_TOKEN;if(!e||!o||!s)throw new O("env","SUBGRAPH_INTERNAL_URL/PROGRESS_API_URL/PROJECT_ID/PROJECT_API_TOKEN missing");return{apiBase:e,projectId:o,authToken:s}}async function Je({apiBase:r,authToken:t,body:e}){let o;try{o=await fetch(`${r}/internal/subgraph/begin`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(e)})}catch(n){throw new O("network",`begin fetch failed: ${n.message}`)}let s=null;try{s=await o.json()}catch{}if(!o.ok){if(o.status===404){let n=new Error(`Sub-graph child '${e.childWorkflowType}' not found in project`);throw n.code="SUBGRAPH_NOT_FOUND",n.status=404,n}if(o.status===429){let n=s?.quotaInfo||{},i=new Error(`Sub-graph blocked by quota (${n.used??"?"}/${n.limit??"?"} on ${n.planId||"plan"})`);throw i.code="SUBGRAPH_QUOTA_EXCEEDED",i.status=429,i.quotaInfo=n,i}if(o.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 O("begin-status",`begin returned ${o.status}`)}return s?.data||s}async function W({apiBase:r,authToken:t,payload:e}){try{let o=await fetch(`${r}/internal/subgraph/finalize`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(e)});o.ok||w.warn(`[in-process subgraph] finalize returned ${o.status} for ${e.childExecutionId}`)}catch(o){w.warn(`[in-process subgraph] finalize failed: ${o.message}`)}}async function Ye(r,t){let e=j(t,".ready"),o=j(t,"graph.mjs");if(F(e)&&F(o))return;Ue(t,{recursive:!0});let s=j(t,".lock"),n=!1;try{let{openSync:i,closeSync:a}=await import("node:fs"),l=i(s,"wx");a(l),n=!0}catch(i){if(i.code!=="EEXIST")throw i}if(!n){let i=Date.now()+3e4;for(;Date.now()<i;){if(F(e)&&F(o))return;await new Promise(a=>setTimeout(a,100))}throw new O("bundle-extract-timeout","sibling extract did not complete within 30s")}try{await new Promise((l,c)=>{let u=oe("curl",["-fsSL",r],{stdio:["ignore","pipe","inherit"]}),p=oe("tar",["-xzf","-","-C",t],{stdio:["pipe","inherit","inherit"]});u.stdout.pipe(p.stdin);let h,m,I=()=>{if(h!==void 0&&m!==void 0){if(h!==0)return c(new Error(`curl exited ${h}`));if(m!==0)return c(new Error(`tar exited ${m}`));l()}};u.on("close",b=>{h=b,I()}),p.on("close",b=>{m=b,I()}),u.on("error",c),p.on("error",c)});let{writeFileSync:i,unlinkSync:a}=await import("node:fs");i(e,"");try{a(s)}catch{}}catch(i){try{let{unlinkSync:a}=await import("node:fs");a(s)}catch{}throw new O("bundle-extract-failed",i.message)}}async function Ze(r){let t=j(r,"graph.mjs");if(!F(t))throw new O("entry-missing",`graph.mjs missing under ${r}`);let e;try{e=await import(Fe(t).href)}catch(s){throw new O("import-failed",`${s?.code||s?.name||"unknown"}: ${s.message}`)}let o=e.default||Object.values(e).find(s=>typeof s=="function"&&s.prototype?.buildGraph);if(!o)throw new O("entry-class-missing","no buildGraph() class export found");return o}async function ie(r,t={}){if(!r||typeof r!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let e=q(),o;try{o=He()}catch(f){throw f}w.debug(`[in-process subgraph] begin '${r}' parent=${e.executionId||"<root>"}`);let s=await Je({apiBase:o.apiBase,authToken:o.authToken,body:{parentExecutionId:e.executionId,childWorkflowType:r,input:t.input||{},...t.conversationId?{conversationId:t.conversationId}:{}}}),{childExecutionId:n,runtimeTag:i,bundlePresignedUrl:a,sourcesPresignedUrl:l,workflowVersion:c,workflowUuid:u,bundleReady:p}=s,h=Ge();if(i&&i!==h)throw await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:`runtimeTag mismatch: parent=${h} child=${i}`,code:"RUNTIME_MISMATCH"}}}),new O("runtime-mismatch",`${h} vs ${i}`);if(!p||!a)throw await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"canceled",error:{message:"bundle not ready for in-process; falling back to HTTP",code:"NO_BUNDLE"}}}),new O("no-bundle","workflow bundle not built yet");let m=re(r);if(!m){let f=j(at,`${u}@${c||"0"}`);try{await Ye(a,f);try{Ke()}catch{}}catch(g){throw g.fallback&&await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"failed",error:{message:g.message,code:g.reason}}}),g}try{m=await Ze(f),te(r,m,{workflowUuid:u,version:c,runtimeTag:i,cacheDir:f})}catch(g){throw ee(r,g),await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"failed",error:{message:g.message,code:g.reason||"IMPORT_FAILED"}}}),g.fallback?g:new O("import-failed",g.message)}}let I=Date.now(),$=await(typeof m=="function"&&m.prototype?.buildGraph?new m:m).buildGraph(),T={...t.input||{}},d,y;try{d=await Xt({executionId:n,parentExecutionId:e.executionId,conversationId:t.conversationId!==void 0?t.conversationId:e.conversationId,dispatchMode:"inprocess"},()=>$.run(t.parentAgent,T,{signal:t.signal})),y=d&&typeof d=="object"&&"state"in d?d.state:d}catch(f){throw await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"failed",error:{message:f.message,code:f.code||"CHILD_THREW",stack:f.stack},durationMs:Date.now()-I}}),f}if(d&&typeof d=="object"&&d.stoppedExternally){await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"canceled",finalState:y,durationMs:Date.now()-I}});let f=new Error(`Sub-graph '${r}' canceled by parent abort`);throw f.code="SUBGRAPH_CANCELED",f.subgraphJobId=n,f}return await W({apiBase:o.apiBase,authToken:o.authToken,payload:{childExecutionId:n,status:"completed",finalState:y,durationMs:Date.now()-I}}),{finalState:y,executionId:n}}function ze(r){let t=0,e=[r];for(;e.length;){let o=e.pop(),s;try{s=se(o)}catch{continue}if(s.isDirectory()){let n;try{n=ne(o)}catch{continue}for(let i of n)e.push(j(o,i))}else t+=s.size}return t}function Ke({cap:r=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!F(at))return{evicted:0,freedBytes:0};let t=ne(at),e=[],o=0;for(let a of t){let l=j(at,a),c;try{c=se(l)}catch{continue}let u=c.isDirectory()?ze(l):c.size;o+=u,e.push({name:a,full:l,size:u,mtimeMs:c.mtimeMs})}if(o<=r)return{evicted:0,freedBytes:0,totalBytes:o};e.sort((a,l)=>a.mtimeMs-l.mtimeMs);let s=Math.floor(r*.7),n=0,i=0;for(let a of e){if(o-n<=s)break;if(!F(j(a.full,".lock")))try{We(a.full,{recursive:!0,force:!0}),n+=a.size,i+=1}catch(l){w.debug(`[sub-graph cache] evict skip ${a.name}: ${l.message}`)}}return i>0&&w.info(`[sub-graph cache] evicted ${i} entr(y/ies), freed ${(n/1024/1024).toFixed(1)}MB`),{evicted:i,freedBytes:n,totalBytes:o-n}}catch(t){return w.debug(`[sub-graph cache] evict failed: ${t.message}`),{evicted:0,freedBytes:0}}}var Ve=2e3,qe=600*1e3,Xe=new Set(["completed","failed","canceled","timeout"]);function Qe(){let r=process.env.PROGRESS_API_URL;if(!r)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 r.replace(/\/executions\/?$/,"")}function tr(){let r=process.env.PROJECT_ID;if(!r)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return r}function er(){let r=process.env.PROJECT_API_TOKEN;if(!r)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return r}function rr(){return process.env.EXECUTION_ID||null}function ae(r,t){return t==null?r:typeof t=="function"?t(r):typeof t=="string"?t.split(".").reduce((e,o)=>e==null?e:e[o],r):r}async function ce(r,t={}){if(!r||typeof r!="string")throw new Error("dispatchSubgraph: workflowName (string) is required");let e=q(),o=Number(process.env.ZIBBY_SUBGRAPH_MAX_DEPTH||10);if((e.depth||0)>=o)throw new Error(`dispatchSubgraph('${r}'): sub-graph depth ${e.depth} reached cap of ${o}. Restructure the graph or raise ZIBBY_SUBGRAPH_MAX_DEPTH.`);if(process.env.ZIBBY_INPROCESS_SUBGRAPH!=="0"&&!t.async)try{w.debug(`[sub-graph] trying in-process for '${r}'`);let{finalState:f}=await ie(r,{input:t.input,conversationId:t.conversationId,signal:t.signal,parentAgent:t.parentAgent}),g=ae(f,t.output);return w.info(`[sub-graph] '${r}' completed in-process`),g}catch(f){if(f instanceof O||f?.fallback)w.info(`[sub-graph] in-process fallback for '${r}': ${f.reason||"unknown"} \u2014 using HTTP`);else throw f}let s=Qe(),n=tr(),i=er(),a=rr(),l=`${s}/projects/${encodeURIComponent(n)}/workflows/${encodeURIComponent(r)}/trigger`,c={input:t.input||{},...a?{parentExecutionId:a}:{},...t.conversationId?{conversationId:t.conversationId}:{}};w.info(`[sub-graph] dispatching '${r}' (${t.async?"async":"sync"}) from parent ${a||"<none>"}`);let u=await fetch(l,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${i}`},body:JSON.stringify(c)});if(!u.ok){let f=null,g="";try{f=await u.json(),g=f?.error||f?.message||JSON.stringify(f)}catch{g=await u.text().catch(()=>"")}if(u.status===429){let v=f?.quotaInfo||{},M=new Error(`Sub-graph '${r}' 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 M.code="SUBGRAPH_QUOTA_EXCEEDED",M.status=429,M.subgraph=r,M.quotaInfo=v,M}if(u.status===400){let v=new Error(`Sub-graph '${r}' rejected input: ${g}`);throw v.code="SUBGRAPH_INVALID_INPUT",v.status=400,v.subgraph=r,v.validationErrors=f?.validationErrors||null,v.missing=f?.missing||null,v}let B=new Error(`Sub-graph '${r}' trigger rejected (${u.status}): ${g}`);throw B.code="SUBGRAPH_TRIGGER_FAILED",B.status=u.status,B.subgraph=r,B}let p=await u.json(),h=p?.data?.jobId||p?.jobId;if(!h)throw new Error(`Sub-graph '${r}' trigger returned no jobId: ${JSON.stringify(p).slice(0,200)}`);if(t.async)return w.info(`[sub-graph] async dispatch of '${r}' \u2192 jobId=${h} (not waiting)`),{jobId:h,status:"accepted",workflow:r};let m=Number.isFinite(t.timeoutMs)?t.timeoutMs:qe,I=Number.isFinite(t.pollIntervalMs)?t.pollIntervalMs:Ve,b=`${s}/executions/${encodeURIComponent(h)}`,$=Date.now()+m,T="accepted",d=0;for(;Date.now()<$;){await new Promise(v=>setTimeout(v,I)),d+=1;let f=await fetch(b,{headers:{Authorization:`Bearer ${i}`}});if(!f.ok){if(f.status>=500){w.warn(`[sub-graph] status poll for ${h} returned ${f.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${h}: ${f.status}`)}let g=await f.json(),B=g?.data||g?.execution||g;if(T=B?.status||T,Xe.has(T)){if(T!=="completed"){let S=new Error(`Sub-graph '${r}' (${h}) ended in status '${T}'`);throw S.subgraphJobId=h,S.subgraphStatus=T,S}let v=B?.finalState||B?.state||{},M=ae(v,t.output);return w.info(`[sub-graph] '${r}' (${h}) completed after ${d} polls`),M}}let y=new Error(`Sub-graph '${r}' (${h}) timed out after ${Math.round(m/1e3)}s (last status: ${T})`);throw y.subgraphJobId=h,y.subgraphStatus=T,y}import{existsSync as le,readFileSync as or}from"node:fs";import{join as vt,dirname as ue}from"node:path";var ct=class{static async loadContext(t,e,o={}){let s={},n=o.filenames||["CONTEXT.md","AGENTS.md"];if(t){let a=ue(vt(e,t));for(let l of n){let c=await this.findAndMergeContextFiles(l,a,e);if(c){let u=l.replace(/\.[^.]+$/,"").toLowerCase();s[u]=c}}}let i=o.discovery||{};for(let[a,l]of Object.entries(i))try{let c=vt(e,l);le(c)&&(s[a]=await this.loadFile(c))}catch(c){console.warn(`[workflow] could not load context '${a}' from '${l}': ${c.message}`)}return s}static async findAndMergeContextFiles(t,e,o){let s=[],n=e;for(;n.startsWith(o);){let i=vt(n,t);if(le(i))try{s.unshift(await this.loadFile(i))}catch(l){console.warn(`[workflow] could not load ${t} from ${i}: ${l.message}`)}let a=ue(n);if(a===n)break;n=a}return s.length===0?null:s.every(i=>typeof i=="string")?s.join(`
30
30
 
31
31
  ---
32
32
 
33
- `):s.every(i=>typeof i=="object")?Object.assign({},...s):s[s.length-1]}static async loadFile(t){let e=or(t,"utf-8");if(t.endsWith(".json"))return JSON.parse(e);if(t.endsWith(".js")||t.endsWith(".mjs")){let{pathToFileURL:o}=await import("url"),s=await import(o(t).href);return s.default||s}return e}};import{mkdirSync as fe,existsSync as At,writeFileSync as pe,unlinkSync as sr}from"node:fs";import{join as G,resolve as ge}from"node:path";import{config as nr}from"dotenv";import{zodToJsonSchema as de}from"zod-to-json-schema";import{z as ut}from"zod";import ir from"handlebars";function ar({traceFrom:r,sessionId:t,sessionPath:e,idSource:o,mkdirFresh:s}){if(!(process.env.ZIBBY_SESSION_LOG==="1"||process.env.ZIBBY_SESSION_LOG==="true"))return;let i=typeof process.ppid=="number"?process.ppid:"n/a",a=`[zibby:session] from=${r} pid=${process.pid} ppid=${i} sessionId=${t} source=${o} mkdir=${s?"yes":"no"} path=${e}`;if(console.log(a),process.env.ZIBBY_TRACE_SESSION==="1"||process.env.ZIBBY_TRACE_SESSION==="true"){let l=(new Error("session trace").stack||"").split(`
33
+ `):s.every(i=>typeof i=="object")?Object.assign({},...s):s[s.length-1]}static async loadFile(t){let e=or(t,"utf-8");if(t.endsWith(".json"))return JSON.parse(e);if(t.endsWith(".js")||t.endsWith(".mjs")){let{pathToFileURL:o}=await import("url"),s=await import(o(t).href);return s.default||s}return e}};import{mkdirSync as fe,existsSync as At,writeFileSync as pe,unlinkSync as sr}from"node:fs";import{join as G,resolve as ge}from"node:path";import{config as nr}from"dotenv";import{zodToJsonSchema as de}from"zod-to-json-schema";import{z as lt}from"zod";import ir from"handlebars";function ar({traceFrom:r,sessionId:t,sessionPath:e,idSource:o,mkdirFresh:s}){if(!(process.env.ZIBBY_SESSION_LOG==="1"||process.env.ZIBBY_SESSION_LOG==="true"))return;let i=typeof process.ppid=="number"?process.ppid:"n/a",a=`[zibby:session] from=${r} pid=${process.pid} ppid=${i} sessionId=${t} source=${o} mkdir=${s?"yes":"no"} path=${e}`;if(console.log(a),process.env.ZIBBY_TRACE_SESSION==="1"||process.env.ZIBBY_TRACE_SESSION==="true"){let u=(new Error("session trace").stack||"").split(`
34
34
  `).slice(2,14).join(`
35
35
  `);console.log(`[zibby:session] stack (${r}):
36
- ${l}`)}}function cr(){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 ur(){if(!(process.env.ZIBBY_PIN_SESSION_PATH==="1"||process.env.ZIBBY_PIN_SESSION_PATH==="true"))return;let t=process.env.ZIBBY_SESSION_PATH;if(!(t==null||String(t).trim()===""))try{return ge(String(t).trim())}catch{return String(t).trim()}}function lr(){cr()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function pr({sessionPath:r,sessionId:t}){r&&typeof r=="string"&&(process.env.ZIBBY_SESSION_PATH=r),t!=null&&String(t).trim()!==""&&(process.env.ZIBBY_SESSION_ID=String(t).trim())}function dr(r={}){let t=Yt.map(n=>process.env[n]).find(Boolean),e=Math.random().toString(36).slice(2,6),o=t||`${Date.now()}_${e}`,s=r.paths?.sessionPrefix;return s?`${s}_${o}`:o}function hr({cwd:r=process.cwd(),config:t={},initialState:e={},traceFrom:o="resolveWorkflowSession"}={}){let s=e.sessionPath,n=e.sessionTimestamp,i="initialState.sessionPath";if(!s&&process.env.ZIBBY_SESSION_PATH)try{let u=ge(String(process.env.ZIBBY_SESSION_PATH));u&&(s=u,i="ZIBBY_SESSION_PATH")}catch{}let a;if(s)a=String(s).split(/[/\\]/).filter(Boolean).pop(),n==null&&(n=Date.now());else{let u=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(u)a=u,i="ZIBBY_SESSION_ID";else{let p=t.sessionId!=null?String(t.sessionId).trim():"";p&&p!=="last"?(a=p,i="config.sessionId"):(a=dr(t),i="generated")}n=n??Date.now();let l=t.paths?.output||nt;s=G(r,l,Ht,a)}let c=!At(s);return c&&fe(s,{recursive:!0}),(c||i!=="initialState.sessionPath")&&ar({traceFrom:o,sessionId:a,sessionPath:s,idSource:i,mkdirFresh:c}),pr({sessionPath:s,sessionId:a}),{sessionPath:s,sessionId:a,sessionTimestamp:n}}var he=class{constructor(t={}){this.nodes=new Map,this.edges=new Map,this.entryPoint=null,this.middleware=Array.isArray(t.middleware)?[...t.middleware]:[],t.nodeMiddleware&&this.middleware.push(t.nodeMiddleware),this.nodeTypeMap=new Map,this.conditionalCodeMap=new Map,this.stateSchema=t.stateSchema||null,this.inputSchema=t.inputSchema||null,this.contextSchema=t.contextSchema||null,this.nodePrompts=new Map,this.nodeOptions=new Map,this._invokeAgent=t.invokeAgent||null,this._compiledPrompts=new Map}setInputSchema(t){return this.inputSchema=t,this}setContextSchema(t){return this.contextSchema=t,this}setStateSchema(t){return this.stateSchema=t,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(t,e,o={}){if(!(e instanceof L)&&e&&typeof e=="object"&&typeof e.workflow=="string"){let n=e,i={name:t,_isCustomCode:!0,retries:n.retries,onComplete:n.onComplete,execute:async c=>{let u=c?.state&&typeof c.state.getAll=="function"?c.state.getAll():c,l;return typeof n.input=="function"?l=n.input(u):n.input&&typeof n.input=="object"?l=n.input:l={},ce(n.workflow,{input:l,async:n.async===!0,conversationId:typeof n.conversationId=="function"?n.conversationId(u):n.conversationId,output:n.output,timeoutMs:n.timeoutMs,pollIntervalMs:n.pollIntervalMs,signal:u?._signal,parentAgent:c?.agent})}},a=new L(i);return a.name=t,this.nodes.set(t,a),o.prompt&&this.nodePrompts.set(t,o.prompt),Object.keys(o).length>0&&this.nodeOptions.set(t,o),this}let s=e instanceof L?e:new L(e);return s.name=t,this.nodes.set(t,s),o.prompt&&this.nodePrompts.set(t,o.prompt),Object.keys(o).length>0&&this.nodeOptions.set(t,o),this}addConditionalNode(t,e){return this.nodes.set(t,new it({...e,name:t})),this}addEdge(t,e){return this.edges.set(t,e),this}setNodeType(t,e){return this.nodeTypeMap.set(t,e),this}addConditionalEdges(t,e,{labels:o}={}){return this.edges.set(t,{conditional:!0,routes:e,labels:o}),typeof e=="function"&&this.conditionalCodeMap.set(t,e.toString()),this}setEntryPoint(t){return this.entryPoint=t,this}use(t){return typeof t=="function"&&this.middleware.push(t),this}_composeMiddleware(t,e,o,s,n){let i=o;for(let a=t.length-1;a>=0;a--){let c=t[a],u=i;i=()=>c(e,u,s,n)}return i()}serialize(){let t=[],e={};for(let[u,l]of this.nodes){let p=this.nodeTypeMap.get(u)||u;t.push({id:u,type:p,data:{nodeType:p,label:u}});let h={};l._isCustomCode&&typeof l.execute=="function"&&(h.customCode=l.execute.toString());let m=this.nodePrompts.get(u);if(m&&(h.prompt=m),typeof l.customExecute=="function"&&(h.executeCode=l.customExecute.toString()),l.outputSchema)try{if(typeof l.outputSchema._def<"u"){let $=typeof ut?.toJSONSchema=="function"?ut.toJSONSchema(l.outputSchema):de(l.outputSchema,{target:"openApi3"});h.outputSchema={jsonSchema:$,variables:this._flattenJsonSchemaToVariables($)}}else h.outputSchema={schema:l.outputSchema}}catch($){console.warn(`[workflow] failed to convert schema for ${u}:`,$.message)}let I=(this.resolvedToolsMap||{})[u];I?.toolIds&&(h.tools=I.toolIds);let b=Array.isArray(l?.config?.skills)?l.config.skills:Array.isArray(l?.skills)?l.skills:null;b&&b.length>0&&(h.skills=[...b]),Object.keys(h).length>0&&(e[u]=h)}let o=[];for(let[u,l]of this.edges)if(typeof l=="string")o.push({source:u,target:l});else if(l.conditional){let p=this.conditionalCodeMap.get(u)||l.routes.toString(),h=this._inferConditionalTargets(l.routes),m=l.labels||{};for(let I of h){let b={source:u,target:I,data:{conditionalCode:p}};m[I]&&(b.label=m[I]),o.push(b)}}let s=u=>{if(!u)return null;if(typeof ut?.toJSONSchema=="function")try{return ut.toJSONSchema(u)}catch{}try{return de(u,{target:"openApi3"})}catch{return null}},n=this._runtimeSchema(),i=s(n||this.stateSchema),a=s(this.inputSchema),c=s(this.contextSchema);return{nodes:t,edges:o,nodeConfigs:e,stateSchema:i,inputSchema:a,contextSchema:c}}_inferConditionalTargets(t){let e=t.toString(),o=new Set,s=/return\s+['"]([^'"]+)['"]/g,n;for(;(n=s.exec(e))!==null;)o.add(n[1]);return[...o]}_flattenJsonSchemaToVariables(t,e=""){let o=t;if(t.$ref&&t.definitions){let s=t.$ref.replace("#/definitions/","");o=t.definitions[s]||t}return this._flattenSchema(o,e)}_flattenSchema(t,e=""){if(!t||typeof t!="object")return[];let o=[],s=t.properties||{},n=t.required||[];for(let[i,a]of Object.entries(s)){let c=e?`${e}.${i}`:i;o.push({path:c,type:a.type||"unknown",label:a.description||this._formatLabel(i),optional:!n.includes(i)}),a.type==="object"&&a.properties&&o.push(...this._flattenSchema(a,c)),a.type==="array"&&a.items?.type==="object"&&a.items.properties&&o.push(...this._flattenSchema(a.items,`${c}[]`))}return o}_formatLabel(t){return t.replace(/([A-Z])/g," $1").replace(/^./,e=>e.toUpperCase()).trim()}_summarizeNodeOutput(t,e){if(!e||typeof e!="object")return[];let o=[];e.success!==void 0&&o.push(`Result: ${e.success?"passed":"failed"}`);for(let[s,n]of Object.entries(e))if(!(s==="success"||s==="raw"||s==="nextNode")){if(typeof n=="string"&&n.length<=80)o.push(`${s}: ${n}`);else if(Array.isArray(n)){let i=n.length,a=n.filter(u=>u?.passed===!0).length,c=n.some(u=>u?.passed!==void 0);o.push(c?`${s}: ${a}/${i} passed${i-a?`, ${i-a} failed`:""}`:`${s}: ${i} items`)}if(o.length>=4)break}return o}async run(t,e={},o={}){if(!this.entryPoint)throw new Error("No entry point set for graph");let s=new AbortController;o.signal&&(o.signal.aborted?s.abort():o.signal.addEventListener("abort",()=>s.abort(),{once:!0}));let n=o.strategyAbortTimeoutMs??e.config?.strategyAbortTimeoutMs??5e3,i=e.cwd||process.cwd();nr({path:G(i,".env")});let a=e.config||{};if(!a||Object.keys(a).length===0)try{let _=G(i,".zibby.config.js");At(_)&&(a=(await import(_)).default||{})}catch{}process.env.EXECUTION_ID&&!a.agent?.strictMode&&(a.agent={...a.agent,strictMode:!0});let c=e.agentType;if(!c){let _=a?.agent;_?.provider?c=_.provider:_?.gemini?c="gemini":_?.claude?c="claude":_?.cursor?c="cursor":_?.codex?c="codex":c=process.env.AGENT_TYPE||"cursor"}let u=e.contextConfig||t?.config?.contextConfig||t?.config?.context||a?.context||{},l=this._runtimeSchema();if(l){let _=l.safeParse(e);if(!_.success){let x=_.error.issues.map(P=>`${P.path.join(".")}: ${P.message}`);throw console.error("\u274C Initial state validation failed:"),x.forEach(P=>console.error(` - ${P}`)),new Error(`State validation failed: ${x.join(", ")}`)}k.step("State validated against schema")}let p=ur(),h=e.sessionPath||p;h||lr();let{sessionPath:m,sessionTimestamp:I,sessionId:b}=hr({cwd:i,config:a,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:h,sessionTimestamp:e.sessionTimestamp}});k.step(`Session ${b}`);let $=await ct.loadContext(e.specPath||"",i,u);Object.keys($).length>0&&k.step(`Context loaded: ${Object.keys($).join(", ")}`);let T=e.outputPath;!T&&e.specPath&&(t?.calculateOutputPath?T=t.calculateOutputPath(e.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${e.specPath})`));let d=new rt({...e,config:a,agentType:c,outputPath:T,sessionPath:m,sessionTimestamp:I,context:$,resolvedTools:this.resolvedToolsMap||{},_signal:s.signal}),y=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:f}=await Promise.resolve().then(()=>(St(),zt)),g=a.skills&&typeof a.skills=="object"?a.skills:{},B=Object.values(g).filter(_=>_&&typeof _=="object"&&typeof _.id=="string"),v=_=>{for(let x of B)if(x.id===_)return x;return f(_)},M=new Set;for(let[,_]of this.nodes)for(let x of _.config?.skills||[])M.add(x);for(let _ of M){let x=v(_);if(typeof x?.middleware=="function")try{let P=await x.middleware();typeof P=="function"&&y.set(_,P)}catch{}}let S=this.entryPoint,X=[],kt=a?.recursionLimit??100,me=0;try{for(;S&&S!=="END";){if(++me>kt)throw new Error(`Workflow exceeded recursion limit (${kt}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let x=G(m,Jt);if(At(x)){try{sr(x)}catch{}s.abort()}if(s.signal.aborted)return console.warn(`
37
- \u{1F6D1} External stop requested \u2014 ending workflow.`),k.step("Workflow stopped externally"),{success:!0,state:d.getAll(),executionLog:X,stoppedExternally:!0};let P=this.nodes.get(S);if(!P)throw new Error(`Node '${S}' not found in graph`);let Ot=JSON.stringify({sessionPath:m,sessionTimestamp:I,currentNode:S,createdAt:new Date().toISOString(),config:d.get("config")}),Se=G(m,Y);pe(Se,Ot,"utf-8");let xt=d.get("config")?.paths?.output||nt,we=G(i,xt,Y);fe(G(i,xt),{recursive:!0});try{pe(we,Ot,"utf-8")}catch{}let Pt=e.onPipelineProgress;if(typeof Pt=="function")try{Pt({cwd:i,sessionPath:m,sessionId:b,outputBase:d.get("config")?.paths?.output||nt,currentNode:S})}catch{}let _e=(this.resolvedToolsMap||{})[S]||null;d.set("_currentNodeTools",_e);let ye=d.get("nodeConfigs")||{};d.set("_currentNodeConfig",ye[S]||{}),k.nodeStart(S);let Nt=Date.now(),Q=this.nodePrompts.get(S);if(!this._invokeAgent){let N=await Promise.resolve().then(()=>(yt(),_t));this._invokeAgent=N.invokeAgent}let Ie=this._invokeAgent,lt={},Ee=P.config?.skills||[];for(let N of Ee){let R=v(N);if(typeof R?.invokeAgentOptions=="function")try{let E=R.invokeAgentOptions(d.getAll(),{agentType:d.get("agentType"),nodeName:S});E&&typeof E=="object"&&(lt={...lt,...E})}catch(E){console.warn(`[graph] skill '${N}' invokeAgentOptions threw: ${E.message}`)}}let Rt=async(N,R,E={})=>{let C=Ie(N,R,{...lt,...E,signal:s.signal});return C.catch(()=>{}),s.signal.aborted?C:Promise.race([C,new Promise((H,J)=>{let D=()=>{setTimeout(()=>{let z=new Error(`Strategy ignored AbortSignal \u2014 engine deadman fired after ${n}ms`);z.name="AbortError",J(z)},n)};s.signal.addEventListener("abort",D,{once:!0})})])},Bt={state:d,invokeAgent:async(N={},R={})=>{let E=R.prompt||"";if(Q){let C=this._compiledPrompts.get(S);C||(C=ir.compile(Q,{noEscape:!0}),this._compiledPrompts.set(S,C));try{E=C(N)}catch(H){throw console.error(`\u274C Template rendering failed for node '${S}':`,H.message),new Error(`Template rendering failed: ${H.message}`,{cause:H})}}else if(!E)throw new Error(`No prompt template configured for node '${S}' and no prompt provided in options`);return Rt(E,{state:d.getAll(),images:R.images||[]},{model:R.model||d.get("model"),workspace:d.get("workspace"),schema:R.schema,...R,signal:s.signal})},_coreInvokeAgent:Rt,agent:t,nodeId:S,promptTemplate:Q,getPromptTemplate:()=>Q,...d.getAll()};try{let N=(P.config?.skills||[]).map(D=>y.get(D)).filter(Boolean),R=[...this.middleware,...N],E;R.length>0?E=await this._composeMiddleware(R,S,async()=>P.execute(Bt,d),d.getAll(),d):E=await P.execute(Bt,d);let C=Date.now()-Nt;if(X.push({node:S,success:E.success,duration:C,timestamp:new Date().toISOString()}),!E.success){if(s.signal.aborted)return k.step("Workflow stopped externally"),{success:!0,state:d.getAll(),executionLog:X,stoppedExternally:!0};d.append("errors",{node:S,error:E.error});let D=P.config?.retries||0,z=`${S}_retries`,tt=d.getAll()[z]||0;if(tt<D){k.stepInfo(`Retrying (attempt ${tt+1}/${D})`),d.update({[z]:tt+1,[`${S}_raw`]:E.raw});continue}throw k.nodeFailed(S,E.error,{duration:C}),new Error(`Node '${S}' failed after ${tt} attempts: ${E.error}`)}d.update({[S]:E.output});let H=this._summarizeNodeOutput(S,E.output);k.nodeComplete(S,{duration:C,details:H});let J=this.edges.get(S);if(!J)S="END";else if(J.conditional){let D=J.routes(d.getAll());k.route(S,D),S=D}else S=J}catch(N){throw k.isInsideNode&&k.nodeFailed(S,N.message,{duration:Date.now()-Nt}),d.set("failed",!0),d.set("failedAt",S),N}}k.graphComplete();let _={success:!0,state:d.getAll(),executionLog:X};return t&&typeof t.onComplete=="function"&&await t.onComplete(_),_}finally{if(t&&typeof t.cleanup=="function")try{await t.cleanup()}catch(_){console.warn(`[workflow] agent.cleanup() failed: ${_.message}`)}}}};export{he as WorkflowGraph,lr as clearInheritedSessionEnvForFreshRun,dr as generateWorkflowSessionId,ur as readPinnedSessionPathFromEnv,hr as resolveWorkflowSession,cr as shouldTrustInheritedSessionEnv,pr as syncProcessEnvToSession};
36
+ ${u}`)}}function cr(){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 lr(){if(!(process.env.ZIBBY_PIN_SESSION_PATH==="1"||process.env.ZIBBY_PIN_SESSION_PATH==="true"))return;let t=process.env.ZIBBY_SESSION_PATH;if(!(t==null||String(t).trim()===""))try{return ge(String(t).trim())}catch{return String(t).trim()}}function ur(){cr()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function pr({sessionPath:r,sessionId:t}){r&&typeof r=="string"&&(process.env.ZIBBY_SESSION_PATH=r),t!=null&&String(t).trim()!==""&&(process.env.ZIBBY_SESSION_ID=String(t).trim())}function dr(r={}){let t=Yt.map(n=>process.env[n]).find(Boolean),e=Math.random().toString(36).slice(2,6),o=t||`${Date.now()}_${e}`,s=r.paths?.sessionPrefix;return s?`${s}_${o}`:o}function hr({cwd:r=process.cwd(),config:t={},initialState:e={},traceFrom:o="resolveWorkflowSession"}={}){let s=e.sessionPath,n=e.sessionTimestamp,i="initialState.sessionPath";if(!s&&process.env.ZIBBY_SESSION_PATH)try{let c=ge(String(process.env.ZIBBY_SESSION_PATH));c&&(s=c,i="ZIBBY_SESSION_PATH")}catch{}let a;if(s)a=String(s).split(/[/\\]/).filter(Boolean).pop(),n==null&&(n=Date.now());else{let c=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(c)a=c,i="ZIBBY_SESSION_ID";else{let p=t.sessionId!=null?String(t.sessionId).trim():"";p&&p!=="last"?(a=p,i="config.sessionId"):(a=dr(t),i="generated")}n=n??Date.now();let u=t.paths?.output||nt;s=G(r,u,Ht,a)}let l=!At(s);return l&&fe(s,{recursive:!0}),(l||i!=="initialState.sessionPath")&&ar({traceFrom:o,sessionId:a,sessionPath:s,idSource:i,mkdirFresh:l}),pr({sessionPath:s,sessionId:a}),{sessionPath:s,sessionId:a,sessionTimestamp:n}}var he=class{constructor(t={}){this.nodes=new Map,this.edges=new Map,this.entryPoint=null,this.middleware=Array.isArray(t.middleware)?[...t.middleware]:[],t.nodeMiddleware&&this.middleware.push(t.nodeMiddleware),this.nodeTypeMap=new Map,this.conditionalCodeMap=new Map,this.stateSchema=t.stateSchema||null,this.inputSchema=t.inputSchema||null,this.contextSchema=t.contextSchema||null,this.nodePrompts=new Map,this.nodeOptions=new Map,this._invokeAgent=t.invokeAgent||null,this._compiledPrompts=new Map}setInputSchema(t){return this.inputSchema=t,this}setContextSchema(t){return this.contextSchema=t,this}setStateSchema(t){return this.stateSchema=t,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(t,e,o={}){if(!(e instanceof L)&&e&&typeof e=="object"&&typeof e.workflow=="string"){let n=e,i={name:t,_isCustomCode:!0,retries:n.retries,onComplete:n.onComplete,execute:async l=>{let c=l?.state&&typeof l.state.getAll=="function"?l.state.getAll():l,u;return typeof n.input=="function"?u=n.input(c):n.input&&typeof n.input=="object"?u=n.input:u={},ce(n.workflow,{input:u,async:n.async===!0,conversationId:typeof n.conversationId=="function"?n.conversationId(c):n.conversationId,output:n.output,timeoutMs:n.timeoutMs,pollIntervalMs:n.pollIntervalMs,signal:c?._signal,parentAgent:l?.agent})}},a=new L(i);return a.name=t,this.nodes.set(t,a),o.prompt&&this.nodePrompts.set(t,o.prompt),Object.keys(o).length>0&&this.nodeOptions.set(t,o),this}let s=e instanceof L?e:new L(e);return s.name=t,this.nodes.set(t,s),o.prompt&&this.nodePrompts.set(t,o.prompt),Object.keys(o).length>0&&this.nodeOptions.set(t,o),this}addConditionalNode(t,e){return this.nodes.set(t,new it({...e,name:t})),this}addEdge(t,e){return this.edges.set(t,e),this}setNodeType(t,e){return this.nodeTypeMap.set(t,e),this}addConditionalEdges(t,e,{labels:o}={}){return this.edges.set(t,{conditional:!0,routes:e,labels:o}),typeof e=="function"&&this.conditionalCodeMap.set(t,e.toString()),this}setEntryPoint(t){return this.entryPoint=t,this}use(t){return typeof t=="function"&&this.middleware.push(t),this}_composeMiddleware(t,e,o,s,n){let i=o;for(let a=t.length-1;a>=0;a--){let l=t[a],c=i;i=()=>l(e,c,s,n)}return i()}serialize(){let t=[],e={};for(let[c,u]of this.nodes){let p=this.nodeTypeMap.get(c)||c;t.push({id:c,type:p,data:{nodeType:p,label:c}});let h={};u._isCustomCode&&typeof u.execute=="function"&&(h.customCode=u.execute.toString());let m=this.nodePrompts.get(c);if(m&&(h.prompt=m),typeof u.customExecute=="function"&&(h.executeCode=u.customExecute.toString()),u.outputSchema)if(typeof u.outputSchema._def<"u"){let $=null;if(typeof lt?.toJSONSchema=="function")try{$=lt.toJSONSchema(u.outputSchema)}catch{}if(!$)try{$=de(u.outputSchema,{target:"openApi3"})}catch{}h.outputSchema=$?{jsonSchema:$,variables:this._flattenJsonSchemaToVariables($)}:{schema:u.outputSchema}}else h.outputSchema={schema:u.outputSchema};let I=(this.resolvedToolsMap||{})[c];I?.toolIds&&(h.tools=I.toolIds);let b=Array.isArray(u?.config?.skills)?u.config.skills:Array.isArray(u?.skills)?u.skills:null;b&&b.length>0&&(h.skills=[...b]),Object.keys(h).length>0&&(e[c]=h)}let o=[];for(let[c,u]of this.edges)if(typeof u=="string")o.push({source:c,target:u});else if(u.conditional){let p=this.conditionalCodeMap.get(c)||u.routes.toString(),h=this._inferConditionalTargets(u.routes,u.labels),m=u.labels||{};for(let I of h){let b={source:c,target:I,data:{conditionalCode:p}};m[I]&&(b.label=m[I]),o.push(b)}}let s=c=>{if(!c)return null;if(typeof lt?.toJSONSchema=="function")try{return lt.toJSONSchema(c)}catch{}try{return de(c,{target:"openApi3"})}catch{return null}},n=this._runtimeSchema(),i=s(n||this.stateSchema),a=s(this.inputSchema),l=s(this.contextSchema);return{nodes:t,edges:o,nodeConfigs:e,stateSchema:i,inputSchema:a,contextSchema:l}}_inferConditionalTargets(t,e){let o=t.toString(),s=new Set,n=/(['"])((?:\\.|(?!\1).)*?)\1|`((?:\\.|[^`$]|\$(?!\{))*?)`/g,i;for(;(i=n.exec(o))!==null;){let c=i[2]!==void 0?i[2]:i[3];c!==void 0&&c!==""&&s.add(c)}let a=new Set(["END","START","__end__","__start__"]);for(let c of this.nodes.keys())a.add(c);if(e&&typeof e=="object")for(let c of Object.keys(e))a.add(c);let l=new Set;for(let c of s)a.has(c)&&l.add(c);if(l.size===0){let c=/return\s+['"]([^'"]+)['"]/g,u;for(;(u=c.exec(o))!==null;)l.add(u[1])}return[...l]}_flattenJsonSchemaToVariables(t,e=""){let o=t;if(t.$ref&&t.definitions){let s=t.$ref.replace("#/definitions/","");o=t.definitions[s]||t}return this._flattenSchema(o,e)}_flattenSchema(t,e=""){if(!t||typeof t!="object")return[];let o=[],s=t.properties||{},n=t.required||[];for(let[i,a]of Object.entries(s)){let l=e?`${e}.${i}`:i;o.push({path:l,type:a.type||"unknown",label:a.description||this._formatLabel(i),optional:!n.includes(i)}),a.type==="object"&&a.properties&&o.push(...this._flattenSchema(a,l)),a.type==="array"&&a.items?.type==="object"&&a.items.properties&&o.push(...this._flattenSchema(a.items,`${l}[]`))}return o}_formatLabel(t){return t.replace(/([A-Z])/g," $1").replace(/^./,e=>e.toUpperCase()).trim()}_summarizeNodeOutput(t,e){if(!e||typeof e!="object")return[];let o=[];e.success!==void 0&&o.push(`Result: ${e.success?"passed":"failed"}`);for(let[s,n]of Object.entries(e))if(!(s==="success"||s==="raw"||s==="nextNode")){if(typeof n=="string"&&n.length<=80)o.push(`${s}: ${n}`);else if(Array.isArray(n)){let i=n.length,a=n.filter(c=>c?.passed===!0).length,l=n.some(c=>c?.passed!==void 0);o.push(l?`${s}: ${a}/${i} passed${i-a?`, ${i-a} failed`:""}`:`${s}: ${i} items`)}if(o.length>=4)break}return o}async run(t,e={},o={}){if(!this.entryPoint)throw new Error("No entry point set for graph");let s=new AbortController;o.signal&&(o.signal.aborted?s.abort():o.signal.addEventListener("abort",()=>s.abort(),{once:!0}));let n=o.strategyAbortTimeoutMs??e.config?.strategyAbortTimeoutMs??5e3,i=e.cwd||process.cwd();nr({path:G(i,".env")});let a=e.config||{};if(!a||Object.keys(a).length===0)try{let _=G(i,".zibby.config.js");At(_)&&(a=(await import(_)).default||{})}catch{}process.env.EXECUTION_ID&&!a.agent?.strictMode&&(a.agent={...a.agent,strictMode:!0});let l=e.agentType;if(!l){let _=a?.agent;_?.provider?l=_.provider:_?.gemini?l="gemini":_?.claude?l="claude":_?.cursor?l="cursor":_?.codex?l="codex":l=process.env.AGENT_TYPE||"cursor"}let c=e.contextConfig||t?.config?.contextConfig||t?.config?.context||a?.context||{},u=this._runtimeSchema();if(u){let _=u.safeParse(e);if(!_.success){let x=_.error.issues.map(P=>`${P.path.join(".")}: ${P.message}`);throw console.error("\u274C Initial state validation failed:"),x.forEach(P=>console.error(` - ${P}`)),new Error(`State validation failed: ${x.join(", ")}`)}k.step("State validated against schema")}let p=lr(),h=e.sessionPath||p;h||ur();let{sessionPath:m,sessionTimestamp:I,sessionId:b}=hr({cwd:i,config:a,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:h,sessionTimestamp:e.sessionTimestamp}});k.step(`Session ${b}`);let $=await ct.loadContext(e.specPath||"",i,c);Object.keys($).length>0&&k.step(`Context loaded: ${Object.keys($).join(", ")}`);let T=e.outputPath;!T&&e.specPath&&(t?.calculateOutputPath?T=t.calculateOutputPath(e.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${e.specPath})`));let d=new rt({...e,config:a,agentType:l,outputPath:T,sessionPath:m,sessionTimestamp:I,context:$,resolvedTools:this.resolvedToolsMap||{},_signal:s.signal}),y=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:f}=await Promise.resolve().then(()=>(St(),zt)),g=a.skills&&typeof a.skills=="object"?a.skills:{},B=Object.values(g).filter(_=>_&&typeof _=="object"&&typeof _.id=="string"),v=_=>{for(let x of B)if(x.id===_)return x;return f(_)},M=new Set;for(let[,_]of this.nodes)for(let x of _.config?.skills||[])M.add(x);for(let _ of M){let x=v(_);if(typeof x?.middleware=="function")try{let P=await x.middleware();typeof P=="function"&&y.set(_,P)}catch{}}let S=this.entryPoint,X=[],kt=a?.recursionLimit??100,me=0;try{for(;S&&S!=="END";){if(++me>kt)throw new Error(`Workflow exceeded recursion limit (${kt}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let x=G(m,Jt);if(At(x)){try{sr(x)}catch{}s.abort()}if(s.signal.aborted)return console.warn(`
37
+ \u{1F6D1} External stop requested \u2014 ending workflow.`),k.step("Workflow stopped externally"),{success:!0,state:d.getAll(),executionLog:X,stoppedExternally:!0};let P=this.nodes.get(S);if(!P)throw new Error(`Node '${S}' not found in graph`);let Ot=JSON.stringify({sessionPath:m,sessionTimestamp:I,currentNode:S,createdAt:new Date().toISOString(),config:d.get("config")}),Se=G(m,Y);pe(Se,Ot,"utf-8");let xt=d.get("config")?.paths?.output||nt,we=G(i,xt,Y);fe(G(i,xt),{recursive:!0});try{pe(we,Ot,"utf-8")}catch{}let Pt=e.onPipelineProgress;if(typeof Pt=="function")try{Pt({cwd:i,sessionPath:m,sessionId:b,outputBase:d.get("config")?.paths?.output||nt,currentNode:S})}catch{}let _e=(this.resolvedToolsMap||{})[S]||null;d.set("_currentNodeTools",_e);let ye=d.get("nodeConfigs")||{};d.set("_currentNodeConfig",ye[S]||{}),k.nodeStart(S);let Nt=Date.now(),Q=this.nodePrompts.get(S);if(!this._invokeAgent){let N=await Promise.resolve().then(()=>(yt(),_t));this._invokeAgent=N.invokeAgent}let Ie=this._invokeAgent,ut={},Ee=P.config?.skills||[];for(let N of Ee){let R=v(N);if(typeof R?.invokeAgentOptions=="function")try{let E=R.invokeAgentOptions(d.getAll(),{agentType:d.get("agentType"),nodeName:S});E&&typeof E=="object"&&(ut={...ut,...E})}catch(E){console.warn(`[graph] skill '${N}' invokeAgentOptions threw: ${E.message}`)}}let Rt=async(N,R,E={})=>{let C=Ie(N,R,{...ut,...E,signal:s.signal});return C.catch(()=>{}),s.signal.aborted?C:Promise.race([C,new Promise((H,J)=>{let D=()=>{setTimeout(()=>{let z=new Error(`Strategy ignored AbortSignal \u2014 engine deadman fired after ${n}ms`);z.name="AbortError",J(z)},n)};s.signal.addEventListener("abort",D,{once:!0})})])},Bt={state:d,invokeAgent:async(N={},R={})=>{let E=R.prompt||"";if(Q){let C=this._compiledPrompts.get(S);C||(C=ir.compile(Q,{noEscape:!0}),this._compiledPrompts.set(S,C));try{E=C(N)}catch(H){throw console.error(`\u274C Template rendering failed for node '${S}':`,H.message),new Error(`Template rendering failed: ${H.message}`,{cause:H})}}else if(!E)throw new Error(`No prompt template configured for node '${S}' and no prompt provided in options`);return Rt(E,{state:d.getAll(),images:R.images||[]},{model:R.model||d.get("model"),workspace:d.get("workspace"),schema:R.schema,...R,signal:s.signal})},_coreInvokeAgent:Rt,agent:t,nodeId:S,promptTemplate:Q,getPromptTemplate:()=>Q,...d.getAll()};try{let N=(P.config?.skills||[]).map(D=>y.get(D)).filter(Boolean),R=[...this.middleware,...N],E;R.length>0?E=await this._composeMiddleware(R,S,async()=>P.execute(Bt,d),d.getAll(),d):E=await P.execute(Bt,d);let C=Date.now()-Nt;if(X.push({node:S,success:E.success,duration:C,timestamp:new Date().toISOString()}),!E.success){if(s.signal.aborted)return k.step("Workflow stopped externally"),{success:!0,state:d.getAll(),executionLog:X,stoppedExternally:!0};d.append("errors",{node:S,error:E.error});let D=P.config?.retries||0,z=`${S}_retries`,tt=d.getAll()[z]||0;if(tt<D){k.stepInfo(`Retrying (attempt ${tt+1}/${D})`),d.update({[z]:tt+1,[`${S}_raw`]:E.raw});continue}throw k.nodeFailed(S,E.error,{duration:C}),new Error(`Node '${S}' failed after ${tt} attempts: ${E.error}`)}d.update({[S]:E.output});let H=this._summarizeNodeOutput(S,E.output);k.nodeComplete(S,{duration:C,details:H});let J=this.edges.get(S);if(!J)S="END";else if(J.conditional){let D=J.routes(d.getAll());k.route(S,D),S=D}else S=J}catch(N){throw k.isInsideNode&&k.nodeFailed(S,N.message,{duration:Date.now()-Nt}),d.set("failed",!0),d.set("failedAt",S),N}}k.graphComplete();let _={success:!0,state:d.getAll(),executionLog:X};return t&&typeof t.onComplete=="function"&&await t.onComplete(_),_}finally{if(t&&typeof t.cleanup=="function")try{await t.cleanup()}catch(_){console.warn(`[workflow] agent.cleanup() failed: ${_.message}`)}}}};export{he as WorkflowGraph,ur as clearInheritedSessionEnvForFreshRun,dr as generateWorkflowSessionId,lr as readPinnedSessionPathFromEnv,hr as resolveWorkflowSession,cr as shouldTrustInheritedSessionEnv,pr as syncProcessEnvToSession};
package/dist/index.js CHANGED
@@ -1,4 +1,4 @@
1
- var eo=Object.defineProperty;var Ee=(t=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(t,{get:(e,o)=>(typeof require<"u"?require:e)[o]}):t)(function(t){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+t+'" is not supported')});var he=(t,e)=>()=>(t&&(e=t(t=0)),e);var et=(t,e)=>{for(var o in e)eo(t,o,{get:e[o],enumerable:!0})};function no(t){te.impl={...ot,...t}}var tt,ot,te,_,F=he(()=>{tt=()=>{},ot={debug:tt,info:tt,warn:(...t)=>console.warn("[workflow]",...t),error:(...t)=>console.error("[workflow]",...t)},te={impl:ot};_={debug:(...t)=>te.impl.debug?.(...t),info:(...t)=>te.impl.info?.(...t),warn:(...t)=>te.impl.warn?.(...t),error:(...t)=>te.impl.error?.(...t)}});var Ae,Ne=he(()=>{Ae=class{constructor(e,o,n=0){this.name=e,this.description=o,this.priority=n}async invoke(e,o={}){throw new Error(`${this.constructor.name}.invoke() must be implemented`)}canHandle(e){throw new Error(`${this.constructor.name}.canHandle() must be implemented`)}getName(){return this.name}getDescription(){return this.description}getPriority(){return this.priority}}});var mt={};et(mt,{clearSkills:()=>gt,getAllSkills:()=>ft,getSkill:()=>K,hasSkill:()=>pt,listSkillIds:()=>ht,registerSkill:()=>dt});function dt(t){if(!t||typeof t.id!="string")throw new Error("Skill definition must include a string id");V.set(t.id,Object.freeze({...t}))}function K(t){return V.get(t)||null}function pt(t){return V.has(t)}function ft(){return new Map(V)}function ht(){return Array.from(V.keys())}function gt(){V.clear()}var Oe,V,re=he(()=>{Oe=Symbol.for("@zibby/agent-workflow.skills");globalThis[Oe]||(globalThis[Oe]=new Map);V=globalThis[Oe]});var se={};et(se,{getAgentStrategy:()=>Pe,invokeAgent:()=>yt,listStrategies:()=>wt,registerStrategy:()=>St});function St(t){if(!t||typeof t.getName!="function"||typeof t.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let e=G.findIndex(o=>o.getName()===t.getName());e>=0?G[e]=t:G.push(t)}function wt(){return G.map(t=>t.getName())}function Pe(t={}){let{state:e={},preferredAgent:o=null}=t,n=o||e.agentType||process.env.AGENT_TYPE;if(!n){let r=G.map(i=>i.getName()).join(", ")||"none registered";throw new Error(`No agent specified. Set agentType in state or AGENT_TYPE env var. Available: ${r}`)}_.debug(`[workflow] agent selection: requested=${n}`);let s=G.find(r=>r.getName()===n);if(!s){let r=G.map(i=>i.getName()).join(", ")||"none registered";throw new Error(`Unknown agent '${n}'. Available: ${r}`)}if(!s.canHandle(t))throw new Error(`Agent '${n}' is not available in this environment. Check credentials/environment.`);return _.debug(`[workflow] using agent: ${s.getName()}`),s}async function yt(t,e={},o={}){let n=Pe(e),s=e.state?.config||o.config||{},r=s.models||{},i=o.nodeName&&r[o.nodeName]||null,a=r.default||null,c=s.agent?.[n.name]?.model||null,u=i||a||c||o.model||null,d={...o,model:u,workspace:e.state?.workspace||o.workspace,schema:o.schema||e.schema,images:o.images||e.images||[],skills:o.skills||e.skills||[],config:s},p=t,l=d.skills||[];if(l.length>0&&!o.skipPromptFragments){let m=l.map(w=>{let I=K(w)?.promptFragment;return typeof I=="function"?I():I}).filter(Boolean);m.length>0&&(p+=`
1
+ var eo=Object.defineProperty;var Ee=(o=>typeof require<"u"?require:typeof Proxy<"u"?new Proxy(o,{get:(e,t)=>(typeof require<"u"?require:e)[t]}):o)(function(o){if(typeof require<"u")return require.apply(this,arguments);throw Error('Dynamic require of "'+o+'" is not supported')});var he=(o,e)=>()=>(o&&(e=o(o=0)),e);var et=(o,e)=>{for(var t in e)eo(o,t,{get:e[t],enumerable:!0})};function no(o){te.impl={...ot,...o}}var tt,ot,te,_,F=he(()=>{tt=()=>{},ot={debug:tt,info:tt,warn:(...o)=>console.warn("[workflow]",...o),error:(...o)=>console.error("[workflow]",...o)},te={impl:ot};_={debug:(...o)=>te.impl.debug?.(...o),info:(...o)=>te.impl.info?.(...o),warn:(...o)=>te.impl.warn?.(...o),error:(...o)=>te.impl.error?.(...o)}});var Ae,Ne=he(()=>{Ae=class{constructor(e,t,n=0){this.name=e,this.description=t,this.priority=n}async invoke(e,t={}){throw new Error(`${this.constructor.name}.invoke() must be implemented`)}canHandle(e){throw new Error(`${this.constructor.name}.canHandle() must be implemented`)}getName(){return this.name}getDescription(){return this.description}getPriority(){return this.priority}}});var mt={};et(mt,{clearSkills:()=>gt,getAllSkills:()=>ft,getSkill:()=>K,hasSkill:()=>pt,listSkillIds:()=>ht,registerSkill:()=>dt});function dt(o){if(!o||typeof o.id!="string")throw new Error("Skill definition must include a string id");V.set(o.id,Object.freeze({...o}))}function K(o){return V.get(o)||null}function pt(o){return V.has(o)}function ft(){return new Map(V)}function ht(){return Array.from(V.keys())}function gt(){V.clear()}var Oe,V,re=he(()=>{Oe=Symbol.for("@zibby/agent-workflow.skills");globalThis[Oe]||(globalThis[Oe]=new Map);V=globalThis[Oe]});var se={};et(se,{getAgentStrategy:()=>Pe,invokeAgent:()=>yt,listStrategies:()=>wt,registerStrategy:()=>St});function St(o){if(!o||typeof o.getName!="function"||typeof o.invoke!="function")throw new Error("strategy must implement getName() and invoke() (AgentStrategy shape)");let e=G.findIndex(t=>t.getName()===o.getName());e>=0?G[e]=o:G.push(o)}function wt(){return G.map(o=>o.getName())}function Pe(o={}){let{state:e={},preferredAgent:t=null}=o,n=t||e.agentType||process.env.AGENT_TYPE;if(!n){let r=G.map(s=>s.getName()).join(", ")||"none registered";throw new Error(`No agent specified. Set agentType in state or AGENT_TYPE env var. Available: ${r}`)}_.debug(`[workflow] agent selection: requested=${n}`);let i=G.find(r=>r.getName()===n);if(!i){let r=G.map(s=>s.getName()).join(", ")||"none registered";throw new Error(`Unknown agent '${n}'. Available: ${r}`)}if(!i.canHandle(o))throw new Error(`Agent '${n}' is not available in this environment. Check credentials/environment.`);return _.debug(`[workflow] using agent: ${i.getName()}`),i}async function yt(o,e={},t={}){let n=Pe(e),i=e.state?.config||t.config||{},r=i.models||{},s=t.nodeName&&r[t.nodeName]||null,a=r.default||null,c=i.agent?.[n.name]?.model||null,u=s||a||c||t.model||null,d={...t,model:u,workspace:e.state?.workspace||t.workspace,schema:t.schema||e.schema,images:t.images||e.images||[],skills:t.skills||e.skills||[],config:i},p=o,l=d.skills||[];if(l.length>0&&!t.skipPromptFragments){let m=l.map(w=>{let I=K(w)?.promptFragment;return typeof I=="function"?I():I}).filter(Boolean);m.length>0&&(p+=`
2
2
 
3
3
  ${m.join(`
4
4
 
@@ -9,46 +9,46 @@ PRIORITY OVERRIDE \u2014 THE FOLLOWING INSTRUCTIONS TAKE PRECEDENCE OVER ALL PRE
9
9
  \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501
10
10
 
11
11
  ${h}
12
- `),_.debug(`[workflow] prompt length: ${p.length} chars`),n.invoke(p,d)}var Ce,G,q=he(()=>{Ne();F();re();Ce=Symbol.for("@zibby/agent-workflow.strategies");globalThis[Ce]||(globalThis[Ce]=[]);G=globalThis[Ce]});var to=new Set(["__proto__","constructor","prototype"]);function $e(t){if(to.has(t))throw new Error(`Invalid state key: "${t}"`)}var Q=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,o){$e(e),this._history.push({...this._state}),this._state[e]=o}update(e){let o=Object.getOwnPropertyNames(e);for(let n of o)$e(n);this._history.push({...this._state});for(let n of o)this._state[n]=e[n]}append(e,o){$e(e),this._history.push({...this._state}),Array.isArray(this._state[e])||(this._state[e]=[]),this._state[e].push(o)}getAll(){return{...this._state}}rollback(){this._history.length>0&&(this._state=this._history.pop())}};var ee=class{constructor(e){this.schema=e}parse(e){let o=e.match(/```json\s*([\s\S]*?)\s*```/);if(o)return this.validate(JSON.parse(o[1]));let n=[e.match(/\{[\s\S]*?\}/),e.match(/\{[\s\S]*\}/)].filter(Boolean).map(s=>s[0]);for(let s of n)try{return this.validate(JSON.parse(s))}catch(r){if(!(r instanceof SyntaxError))throw r}return this.validate({result:e.trim()})}validate(e){let o=[];for(let[n,s]of Object.entries(this.schema)){if(s.required&&!(n in e)&&o.push(`Missing required field: ${n}`),n in e&&s.type){let r=typeof e[n];r!==s.type&&o.push(`Field '${n}' expected ${s.type}, got ${r}`)}if(s.validate&&n in e){let r=s.validate(e[n]);r&&o.push(`Field '${n}': ${r}`)}}if(o.length>0)throw new Error(`Output validation failed:
13
- ${o.join(`
14
- `)}`);return e}},oo={string:(t=!0)=>({type:"string",required:t}),number:(t=!0)=>({type:"number",required:t}),boolean:(t=!0)=>({type:"boolean",required:t}),array:(t=!0)=>({type:"object",required:t,validate:e=>Array.isArray(e)?null:"must be an array"}),enum:(t,e=!0)=>({type:"string",required:e,validate:o=>t.includes(o)?null:`must be one of: ${t.join(", ")}`})};F();import{writeFileSync as Re,readFileSync as _t,existsSync as It,mkdirSync as uo}from"node:fs";import{join as Be,dirname as lo}from"node:path";import k from"chalk";var lt="__WORKFLOW_GRAPH_LOG__",oe=k.gray("\u2502"),ro=k.gray("\u250C"),nt=k.gray("\u2514"),be=k.green("\u25C6"),rt=k.hex("#c084fc")("\u25C6"),st=k.hex("#2dd4bf")("\u25C6"),Te=k.red("\u25C6"),it=`${oe} `,at=2;function ct(t){return t<1e3?`${t}ms`:`${(t/1e3).toFixed(1)}s`}function ut(t,e){return(o,n,s)=>{if(typeof o!="string")return t(o,n,s);let r=process.stdout.columns||120,i="";for(let a=0;a<o.length;a++){let c=o[a];e.lineStart&&(i+=it,e.col=at,e.lineStart=!1),c===`
15
- `?(i+=c,e.lineStart=!0,e.col=0,e.inEsc=!1):c==="\x1B"?(e.inEsc=!0,i+=c):e.inEsc?(i+=c,(c>="A"&&c<="Z"||c>="a"&&c<="z")&&(e.inEsc=!1)):(e.col++,i+=c,e.col>=r&&(i+=`
16
- ${it}`,e.col=at))}return t(i,n,s)}}var ge=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},o={lineStart:!0,col:0,inEsc:!1};this._outState=e,this._errState=o,process.stdout.write=ut(this._origStdoutWrite,e),process.stderr.write=ut(this._origStderrWrite,o)}_stopIntercepting(){this._origStdoutWrite&&(this._outState&&!this._outState.lineStart&&this._origStdoutWrite(`
12
+ `),_.debug(`[workflow] prompt length: ${p.length} chars`),n.invoke(p,d)}var Ce,G,q=he(()=>{Ne();F();re();Ce=Symbol.for("@zibby/agent-workflow.strategies");globalThis[Ce]||(globalThis[Ce]=[]);G=globalThis[Ce]});var to=new Set(["__proto__","constructor","prototype"]);function $e(o){if(to.has(o))throw new Error(`Invalid state key: "${o}"`)}var Q=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){$e(e),this._history.push({...this._state}),this._state[e]=t}update(e){let t=Object.getOwnPropertyNames(e);for(let n of t)$e(n);this._history.push({...this._state});for(let n of t)this._state[n]=e[n]}append(e,t){$e(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 ee=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 n=[e.match(/\{[\s\S]*?\}/),e.match(/\{[\s\S]*\}/)].filter(Boolean).map(i=>i[0]);for(let i of n)try{return this.validate(JSON.parse(i))}catch(r){if(!(r instanceof SyntaxError))throw r}return this.validate({result:e.trim()})}validate(e){let t=[];for(let[n,i]of Object.entries(this.schema)){if(i.required&&!(n in e)&&t.push(`Missing required field: ${n}`),n in e&&i.type){let r=typeof e[n];r!==i.type&&t.push(`Field '${n}' expected ${i.type}, got ${r}`)}if(i.validate&&n in e){let r=i.validate(e[n]);r&&t.push(`Field '${n}': ${r}`)}}if(t.length>0)throw new Error(`Output validation failed:
13
+ ${t.join(`
14
+ `)}`);return e}},oo={string:(o=!0)=>({type:"string",required:o}),number:(o=!0)=>({type:"number",required:o}),boolean:(o=!0)=>({type:"boolean",required:o}),array:(o=!0)=>({type:"object",required:o,validate:e=>Array.isArray(e)?null:"must be an array"}),enum:(o,e=!0)=>({type:"string",required:e,validate:t=>o.includes(t)?null:`must be one of: ${o.join(", ")}`})};F();import{writeFileSync as Re,readFileSync as _t,existsSync as It,mkdirSync as uo}from"node:fs";import{join as Be,dirname as lo}from"node:path";import k from"chalk";var lt="__WORKFLOW_GRAPH_LOG__",oe=k.gray("\u2502"),ro=k.gray("\u250C"),nt=k.gray("\u2514"),be=k.green("\u25C6"),rt=k.hex("#c084fc")("\u25C6"),st=k.hex("#2dd4bf")("\u25C6"),Te=k.red("\u25C6"),it=`${oe} `,at=2;function ct(o){return o<1e3?`${o}ms`:`${(o/1e3).toFixed(1)}s`}function ut(o,e){return(t,n,i)=>{if(typeof t!="string")return o(t,n,i);let r=process.stdout.columns||120,s="";for(let a=0;a<t.length;a++){let c=t[a];e.lineStart&&(s+=it,e.col=at,e.lineStart=!1),c===`
15
+ `?(s+=c,e.lineStart=!0,e.col=0,e.inEsc=!1):c==="\x1B"?(e.inEsc=!0,s+=c):e.inEsc?(s+=c,(c>="A"&&c<="Z"||c>="a"&&c<="z")&&(e.inEsc=!1)):(e.col++,s+=c,e.col>=r&&(s+=`
16
+ ${it}`,e.col=at))}return o(s,n,i)}}var ge=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=ut(this._origStdoutWrite,e),process.stderr.write=ut(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
- `)}_emitGraphLogMarker(e){if(!this._emitWorkflowGraphMarkers)return;let o=`${lt}${JSON.stringify(e)}
20
- `;this._origStdoutWrite?this._origStdoutWrite(o):process.stdout.write(o)}_writeDot(e,o){this._origStdoutWrite?(this._outState&&!this._outState.lineStart&&(this._origStdoutWrite(`
21
- `),this._outState.lineStart=!0,this._outState.col=0),this._origStdoutWrite(`${e} ${o}
22
- `)):process.stdout.write.bind(process.stdout)(`${e} ${o}
19
+ `)}_emitGraphLogMarker(e){if(!this._emitWorkflowGraphMarkers)return;let t=`${lt}${JSON.stringify(e)}
20
+ `;this._origStdoutWrite?this._origStdoutWrite(t):process.stdout.write(t)}_writeDot(e,t){this._origStdoutWrite?(this._outState&&!this._outState.lineStart&&(this._origStdoutWrite(`
21
+ `),this._outState.lineStart=!0,this._outState.col=0),this._origStdoutWrite(`${e} ${t}
22
+ `)):process.stdout.write.bind(process.stdout)(`${e} ${t}
23
23
  `)}step(e){this._origStdoutWrite?this._writeDot(be,e):process.stdout.write.bind(process.stdout)(`${oe} ${be} ${e}
24
24
  `)}stepInfo(e){this.step(e)}stepTool(e){this._origStdoutWrite?this._writeDot(rt,e):process.stdout.write.bind(process.stdout)(`${oe} ${rt} ${e}
25
- `)}stepMemory(e){let o=k.hex("#2dd4bf")(e);this._origStdoutWrite?this._writeDot(st,o):process.stdout.write.bind(process.stdout)(`${oe} ${st} ${o}
25
+ `)}stepMemory(e){let t=k.hex("#2dd4bf")(e);this._origStdoutWrite?this._writeDot(st,t):process.stdout.write.bind(process.stdout)(`${oe} ${st} ${t}
26
26
  `)}stepFail(e){this._origStdoutWrite?this._writeDot(Te,k.red(e)):process.stdout.write.bind(process.stdout)(`${oe} ${Te} ${k.red(e)}
27
- `)}nodeStart(e){this._currentNode=e,this._emitGraphLogMarker({phase:"node_begin",node:e}),this._rawWrite(`${ro} ${e}`),this._startIntercepting()}nodeComplete(e,o={}){this._stopIntercepting();let{duration:n,details:s}=o;if(s)for(let i of s)this._rawWrite(`${be} ${i}`);let r=n?k.dim(` ${ct(n)}`):"";this._rawWrite(`${nt} ${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(` ${ct(s)}`):"";this._rawWrite(`${Te} ${k.red(o)}`),this._rawWrite(`${nt} ${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",ve="sessions",W=".session-info.json",xe=".zibby-stop",so="result.json",io="raw_stream_output.txt",ao="events.json",co={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",REVIEW_MEMORY:"review-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"},ke=["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(`${ro} ${e}`),this._startIntercepting()}nodeComplete(e,t={}){this._stopIntercepting();let{duration:n,details:i}=t;if(i)for(let s of i)this._rawWrite(`${be} ${s}`);let r=n?k.dim(` ${ct(n)}`):"";this._rawWrite(`${nt} ${k.green("done")}${r}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}nodeFailed(e,t,n={}){this._stopIntercepting();let{duration:i}=n,r=i?k.dim(` ${ct(i)}`):"";this._rawWrite(`${Te} ${k.red(t)}`),this._rawWrite(`${nt} ${k.red("failed")}${r}`),this._emitGraphLogMarker({phase:"node_end",node:e}),this._rawWrite("")}route(e,t){this._rawWrite(k.dim(` ${e} \u2192 ${t}`)),this._rawWrite("")}graphComplete(){}},x=new ge;var ne=".zibby/output",ve="sessions",W=".session-info.json",xe=".zibby-stop",so="result.json",io="raw_stream_output.txt",ao="events.json",co={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GITLAB:"gitlab",FIGMA:"figma",GIT:"git",SLACK:"slack",LARK:"lark",CHAT_NOTIFY:"chat_notify",SENTRY:"sentry",MEMORY:"memory",CHAT_MEMORY:"chat-memory",REVIEW_MEMORY:"review-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"},ke=["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,t){let n=()=>t&&typeof t.getAll=="function"?t.getAll():e,i=p=>t&&typeof t.get=="function"?t.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,s=i("_skillHints");s&&(r=`${s}
28
28
 
29
- ${r}`);let a=n(),c=a.cwd||process.cwd(),u=a.sessionPath;try{if(u){let p=Be(u,W);if(It(p)){let h=JSON.parse(_t(p,"utf-8"));h.currentNode=this.name,Re(p,JSON.stringify(h,null,2),"utf-8")}let l=Be(u,"..",W);if(It(l))try{let h=JSON.parse(_t(l,"utf-8"));h.currentNode=this.name,Re(l,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 l=n().config||{},h=l.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:u,config:l,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),u)try{let S=Be(u,this.name,"raw_stream_output.txt");uo(lo(S),{recursive:!0}),Re(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(l){d=l,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 ho,existsSync as H,statSync as kt,readdirSync as At,rmSync as go}from"node:fs";import{spawn as xt}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as mo}from"node:url";import{AsyncLocalStorage as po}from"node:async_hooks";var Me=new po;function ae(){let t=Me.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 Et(t,e){let o=Me.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 Me.run(n,e)}var De=new Map,Le=new Map,$t=new Map;function bt(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");De.set(t,e),Le.set(t,"ready"),$t.set(t,{...o,cachedAt:Date.now()})}function Tt(t,e){Le.set(t,"failed"),$t.set(t,{error:e?.message||String(e),failedAt:Date.now()}),De.delete(t)}function vt(t){return Le.get(t)==="ready"?De.get(t):null}var me=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function So(){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 wo(){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 yo({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 J({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 _o(t,e){let o=U(e,".ready"),n=U(e,"graph.mjs");if(H(o)&&H(n))return;ho(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(H(o)&&H(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,u)=>{let d=xt("curl",["-fsSL",t],{stdio:["ignore","pipe","inherit"]}),p=xt("tar",["-xzf","-","-C",e],{stdio:["pipe","inherit","inherit"]});d.stdout.pipe(p.stdin);let l,h,m=()=>{if(l!==void 0&&h!==void 0){if(l!==0)return u(new Error(`curl exited ${l}`));if(h!==0)return u(new Error(`tar exited ${h}`));c()}};d.on("close",w=>{l=w,m()}),p.on("close",w=>{h=w,m()}),d.on("error",u),p.on("error",u)});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 Io(t){let e=U(t,"graph.mjs");if(!H(e))throw new A("entry-missing",`graph.mjs missing under ${t}`);let o;try{o=await import(mo(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 Nt(t,e={}){if(!t||typeof t!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let o=ae(),n;try{n=wo()}catch(g){throw g}_.debug(`[in-process subgraph] begin '${t}' parent=${o.executionId||"<root>"}`);let s=await yo({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:u,workflowUuid:d,bundleReady:p}=s,l=So();if(i&&i!==l)throw await J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"canceled",error:{message:`runtimeTag mismatch: parent=${l} child=${i}`,code:"RUNTIME_MISMATCH"}}}),new A("runtime-mismatch",`${l} vs ${i}`);if(!p||!a)throw await J({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=vt(t);if(!h){let g=U(me,`${d}@${u||"0"}`);try{await _o(a,g);try{$o()}catch{}}catch(S){throw S.fallback&&await J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"failed",error:{message:S.message,code:S.reason}}}),S}try{h=await Io(g),bt(t,h,{workflowUuid:d,version:u,runtimeTag:i,cacheDir:g})}catch(S){throw Tt(t,S),await J({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 Et({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 J({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 J({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 J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"completed",finalState:$,durationMs:Date.now()-m}}),{finalState:$,executionId:r}}function Eo(t){let e=0,o=[t];for(;o.length;){let n=o.pop(),s;try{s=kt(n)}catch{continue}if(s.isDirectory()){let r;try{r=At(n)}catch{continue}for(let i of r)o.push(U(n,i))}else e+=s.size}return e}function $o({cap:t=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!H(me))return{evicted:0,freedBytes:0};let e=At(me),o=[],n=0;for(let a of e){let c=U(me,a),u;try{u=kt(c)}catch{continue}let d=u.isDirectory()?Eo(c):u.size;n+=d,o.push({name:a,full:c,size:d,mtimeMs:u.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(!H(U(a.full,".lock")))try{go(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 bo=2e3,To=600*1e3,vo=new Set(["completed","failed","canceled","timeout"]);function xo(){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 ko(){let t=process.env.PROJECT_ID;if(!t)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return t}function Ao(){let t=process.env.PROJECT_API_TOKEN;if(!t)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return t}function No(){return process.env.EXECUTION_ID||null}function Ot(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 je(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 Nt(t,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),S=Ot(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=xo(),r=ko(),i=Ao(),a=No(),c=`${s}/projects/${encodeURIComponent(r)}/workflows/${encodeURIComponent(t)}/trigger`,u={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(u)});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(),l=p?.data?.jobId||p?.jobId;if(!l)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=${l} (not waiting)`),{jobId:l,status:"accepted",workflow:t};let h=Number.isFinite(e.timeoutMs)?e.timeoutMs:To,m=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:bo,w=`${s}/executions/${encodeURIComponent(l)}`,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 ${l} returned ${g.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${l}: ${g.status}`)}let S=await g.json(),R=S?.data||S?.execution||S;if(E=R?.status||E,vo.has(E)){if(E!=="completed"){let y=new Error(`Sub-graph '${t}' (${l}) ended in status '${E}'`);throw y.subgraphJobId=l,y.subgraphStatus=E,y}let v=R?.finalState||R?.state||{},D=Ot(v,e.output);return _.info(`[sub-graph] '${t}' (${l}) completed after ${f} polls`),D}}let $=new Error(`Sub-graph '${t}' (${l}) timed out after ${Math.round(h/1e3)}s (last status: ${E})`);throw $.subgraphJobId=l,$.subgraphStatus=E,$}import{existsSync as Ct,readFileSync as Oo}from"node:fs";import{join as Fe,dirname as Pt}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=Pt(Fe(o,e));for(let c of r){let u=await this.findAndMergeContextFiles(c,a,o);if(u){let d=c.replace(/\.[^.]+$/,"").toLowerCase();s[d]=u}}}let i=n.discovery||{};for(let[a,c]of Object.entries(i))try{let u=Fe(o,c);Ct(u)&&(s[a]=await this.loadFile(u))}catch(u){console.warn(`[workflow] could not load context '${a}' from '${c}': ${u.message}`)}return s}static async findAndMergeContextFiles(e,o,n){let s=[],r=o;for(;r.startsWith(n);){let i=Fe(r,e);if(Ct(i))try{s.unshift(await this.loadFile(i))}catch(c){console.warn(`[workflow] could not load ${e} from ${i}: ${c.message}`)}let a=Pt(r);if(a===r)break;r=a}return s.length===0?null:s.every(i=>typeof i=="string")?s.join(`
29
+ ${r}`);let a=n(),c=a.cwd||process.cwd(),u=a.sessionPath;try{if(u){let p=Be(u,W);if(It(p)){let h=JSON.parse(_t(p,"utf-8"));h.currentNode=this.name,Re(p,JSON.stringify(h,null,2),"utf-8")}let l=Be(u,"..",W);if(It(l))try{let h=JSON.parse(_t(l,"utf-8"));h.currentNode=this.name,Re(l,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 l=n().config||{},h=l.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:u,config:l,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),u)try{let S=Be(u,this.name,"raw_stream_output.txt");uo(lo(S),{recursive:!0}),Re(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(l){d=l,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,t){let n=t&&typeof t.getAll=="function"?t.getAll():e;return{success:!0,output:{nextNode:this.condition(n)},raw:null}}};F();F();import{mkdirSync as ho,existsSync as H,statSync as kt,readdirSync as At,rmSync as go}from"node:fs";import{spawn as xt}from"node:child_process";import{join as U}from"node:path";import{pathToFileURL as mo}from"node:url";import{AsyncLocalStorage as po}from"node:async_hooks";var Me=new po;function ae(){let o=Me.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 Et(o,e){let t=Me.getStore()||ae(),n=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 Me.run(n,e)}var De=new Map,Le=new Map,$t=new Map;function bt(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");De.set(o,e),Le.set(o,"ready"),$t.set(o,{...t,cachedAt:Date.now()})}function Tt(o,e){Le.set(o,"failed"),$t.set(o,{error:e?.message||String(e),failedAt:Date.now()}),De.delete(o)}function vt(o){return Le.get(o)==="ready"?De.get(o):null}var me=process.env.ZIBBY_SUBGRAPH_CACHE_DIR||"/tmp/zibby/subgraphs";function So(){return`node${(process.versions?.node||"").split(".")[0]||"unknown"}-${process.platform}-${process.arch}`}var A=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 wo(){let o=(process.env.SUBGRAPH_INTERNAL_URL||"").replace(/\/$/,""),e=(process.env.PROGRESS_API_URL||"").replace(/\/executions\/?$/,""),t=o||e,n=process.env.PROJECT_ID,i=process.env.PROJECT_API_TOKEN;if(!t||!n||!i)throw new A("env","SUBGRAPH_INTERNAL_URL/PROGRESS_API_URL/PROJECT_ID/PROJECT_API_TOKEN missing");return{apiBase:t,projectId:n,authToken:i}}async function yo({apiBase:o,authToken:e,body:t}){let n;try{n=await fetch(`${o}/internal/subgraph/begin`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(t)})}catch(r){throw new A("network",`begin fetch failed: ${r.message}`)}let i=null;try{i=await n.json()}catch{}if(!n.ok){if(n.status===404){let r=new Error(`Sub-graph child '${t.childWorkflowType}' not found in project`);throw r.code="SUBGRAPH_NOT_FOUND",r.status=404,r}if(n.status===429){let r=i?.quotaInfo||{},s=new Error(`Sub-graph blocked by quota (${r.used??"?"}/${r.limit??"?"} on ${r.planId||"plan"})`);throw s.code="SUBGRAPH_QUOTA_EXCEEDED",s.status=429,s.quotaInfo=r,s}if(n.status===400&&i?.validationErrors){let r=new Error(`Sub-graph rejected input: ${i?.error||i?.message||"validation failed"}`);throw r.code="SUBGRAPH_INVALID_INPUT",r.status=400,r.validationErrors=i.validationErrors,r.missing=i.missing,r}throw new A("begin-status",`begin returned ${n.status}`)}return i?.data||i}async function J({apiBase:o,authToken:e,payload:t}){try{let n=await fetch(`${o}/internal/subgraph/finalize`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${e}`},body:JSON.stringify(t)});n.ok||_.warn(`[in-process subgraph] finalize returned ${n.status} for ${t.childExecutionId}`)}catch(n){_.warn(`[in-process subgraph] finalize failed: ${n.message}`)}}async function _o(o,e){let t=U(e,".ready"),n=U(e,"graph.mjs");if(H(t)&&H(n))return;ho(e,{recursive:!0});let i=U(e,".lock"),r=!1;try{let{openSync:s,closeSync:a}=await import("node:fs"),c=s(i,"wx");a(c),r=!0}catch(s){if(s.code!=="EEXIST")throw s}if(!r){let s=Date.now()+3e4;for(;Date.now()<s;){if(H(t)&&H(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,u)=>{let d=xt("curl",["-fsSL",o],{stdio:["ignore","pipe","inherit"]}),p=xt("tar",["-xzf","-","-C",e],{stdio:["pipe","inherit","inherit"]});d.stdout.pipe(p.stdin);let l,h,m=()=>{if(l!==void 0&&h!==void 0){if(l!==0)return u(new Error(`curl exited ${l}`));if(h!==0)return u(new Error(`tar exited ${h}`));c()}};d.on("close",w=>{l=w,m()}),p.on("close",w=>{h=w,m()}),d.on("error",u),p.on("error",u)});let{writeFileSync:s,unlinkSync:a}=await import("node:fs");s(t,"");try{a(i)}catch{}}catch(s){try{let{unlinkSync:a}=await import("node:fs");a(i)}catch{}throw new A("bundle-extract-failed",s.message)}}async function Io(o){let e=U(o,"graph.mjs");if(!H(e))throw new A("entry-missing",`graph.mjs missing under ${o}`);let t;try{t=await import(mo(e).href)}catch(i){throw new A("import-failed",`${i?.code||i?.name||"unknown"}: ${i.message}`)}let n=t.default||Object.values(t).find(i=>typeof i=="function"&&i.prototype?.buildGraph);if(!n)throw new A("entry-class-missing","no buildGraph() class export found");return n}async function Nt(o,e={}){if(!o||typeof o!="string")throw new Error("runInProcessSubgraph: workflowName (string) is required");let t=ae(),n;try{n=wo()}catch(g){throw g}_.debug(`[in-process subgraph] begin '${o}' parent=${t.executionId||"<root>"}`);let i=await yo({apiBase:n.apiBase,authToken:n.authToken,body:{parentExecutionId:t.executionId,childWorkflowType:o,input:e.input||{},...e.conversationId?{conversationId:e.conversationId}:{}}}),{childExecutionId:r,runtimeTag:s,bundlePresignedUrl:a,sourcesPresignedUrl:c,workflowVersion:u,workflowUuid:d,bundleReady:p}=i,l=So();if(s&&s!==l)throw await J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"canceled",error:{message:`runtimeTag mismatch: parent=${l} child=${s}`,code:"RUNTIME_MISMATCH"}}}),new A("runtime-mismatch",`${l} vs ${s}`);if(!p||!a)throw await J({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=vt(o);if(!h){let g=U(me,`${d}@${u||"0"}`);try{await _o(a,g);try{$o()}catch{}}catch(S){throw S.fallback&&await J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"failed",error:{message:S.message,code:S.reason}}}),S}try{h=await Io(g),bt(o,h,{workflowUuid:d,version:u,runtimeTag:s,cacheDir:g})}catch(S){throw Tt(o,S),await J({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 Et({executionId:r,parentExecutionId:t.executionId,conversationId:e.conversationId!==void 0?e.conversationId:t.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 J({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 J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"canceled",finalState:$,durationMs:Date.now()-m}});let g=new Error(`Sub-graph '${o}' canceled by parent abort`);throw g.code="SUBGRAPH_CANCELED",g.subgraphJobId=r,g}return await J({apiBase:n.apiBase,authToken:n.authToken,payload:{childExecutionId:r,status:"completed",finalState:$,durationMs:Date.now()-m}}),{finalState:$,executionId:r}}function Eo(o){let e=0,t=[o];for(;t.length;){let n=t.pop(),i;try{i=kt(n)}catch{continue}if(i.isDirectory()){let r;try{r=At(n)}catch{continue}for(let s of r)t.push(U(n,s))}else e+=i.size}return e}function $o({cap:o=Number(process.env.ZIBBY_SUBGRAPH_CACHE_CAP_BYTES||2*1024*1024*1024)}={}){try{if(!H(me))return{evicted:0,freedBytes:0};let e=At(me),t=[],n=0;for(let a of e){let c=U(me,a),u;try{u=kt(c)}catch{continue}let d=u.isDirectory()?Eo(c):u.size;n+=d,t.push({name:a,full:c,size:d,mtimeMs:u.mtimeMs})}if(n<=o)return{evicted:0,freedBytes:0,totalBytes:n};t.sort((a,c)=>a.mtimeMs-c.mtimeMs);let i=Math.floor(o*.7),r=0,s=0;for(let a of t){if(n-r<=i)break;if(!H(U(a.full,".lock")))try{go(a.full,{recursive:!0,force:!0}),r+=a.size,s+=1}catch(c){_.debug(`[sub-graph cache] evict skip ${a.name}: ${c.message}`)}}return s>0&&_.info(`[sub-graph cache] evicted ${s} entr(y/ies), freed ${(r/1024/1024).toFixed(1)}MB`),{evicted:s,freedBytes:r,totalBytes:n-r}}catch(e){return _.debug(`[sub-graph cache] evict failed: ${e.message}`),{evicted:0,freedBytes:0}}}var bo=2e3,To=600*1e3,vo=new Set(["completed","failed","canceled","timeout"]);function xo(){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 ko(){let o=process.env.PROJECT_ID;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_ID env var.");return o}function Ao(){let o=process.env.PROJECT_API_TOKEN;if(!o)throw new Error("Sub-graph dispatch requires PROJECT_API_TOKEN env var.");return o}function No(){return process.env.EXECUTION_ID||null}function Ot(o,e){return e==null?o:typeof e=="function"?e(o):typeof e=="string"?e.split(".").reduce((t,n)=>t==null?t:t[n],o):o}async function je(o,e={}){if(!o||typeof o!="string")throw new Error("dispatchSubgraph: workflowName (string) is required");let t=ae(),n=Number(process.env.ZIBBY_SUBGRAPH_MAX_DEPTH||10);if((t.depth||0)>=n)throw new Error(`dispatchSubgraph('${o}'): sub-graph depth ${t.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 '${o}'`);let{finalState:g}=await Nt(o,{input:e.input,conversationId:e.conversationId,signal:e.signal,parentAgent:e.parentAgent}),S=Ot(g,e.output);return _.info(`[sub-graph] '${o}' completed in-process`),S}catch(g){if(g instanceof A||g?.fallback)_.info(`[sub-graph] in-process fallback for '${o}': ${g.reason||"unknown"} \u2014 using HTTP`);else throw g}let i=xo(),r=ko(),s=Ao(),a=No(),c=`${i}/projects/${encodeURIComponent(r)}/workflows/${encodeURIComponent(o)}/trigger`,u={input:e.input||{},...a?{parentExecutionId:a}:{},...e.conversationId?{conversationId:e.conversationId}:{}};_.info(`[sub-graph] dispatching '${o}' (${e.async?"async":"sync"}) from parent ${a||"<none>"}`);let d=await fetch(c,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${s}`},body:JSON.stringify(u)});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 '${o}' 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=o,D.quotaInfo=v,D}if(d.status===400){let v=new Error(`Sub-graph '${o}' rejected input: ${S}`);throw v.code="SUBGRAPH_INVALID_INPUT",v.status=400,v.subgraph=o,v.validationErrors=g?.validationErrors||null,v.missing=g?.missing||null,v}let R=new Error(`Sub-graph '${o}' trigger rejected (${d.status}): ${S}`);throw R.code="SUBGRAPH_TRIGGER_FAILED",R.status=d.status,R.subgraph=o,R}let p=await d.json(),l=p?.data?.jobId||p?.jobId;if(!l)throw new Error(`Sub-graph '${o}' trigger returned no jobId: ${JSON.stringify(p).slice(0,200)}`);if(e.async)return _.info(`[sub-graph] async dispatch of '${o}' \u2192 jobId=${l} (not waiting)`),{jobId:l,status:"accepted",workflow:o};let h=Number.isFinite(e.timeoutMs)?e.timeoutMs:To,m=Number.isFinite(e.pollIntervalMs)?e.pollIntervalMs:bo,w=`${i}/executions/${encodeURIComponent(l)}`,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 ${s}`}});if(!g.ok){if(g.status>=500){_.warn(`[sub-graph] status poll for ${l} returned ${g.status}, will retry`);continue}throw new Error(`Sub-graph status poll failed for ${l}: ${g.status}`)}let S=await g.json(),R=S?.data||S?.execution||S;if(E=R?.status||E,vo.has(E)){if(E!=="completed"){let y=new Error(`Sub-graph '${o}' (${l}) ended in status '${E}'`);throw y.subgraphJobId=l,y.subgraphStatus=E,y}let v=R?.finalState||R?.state||{},D=Ot(v,e.output);return _.info(`[sub-graph] '${o}' (${l}) completed after ${f} polls`),D}}let $=new Error(`Sub-graph '${o}' (${l}) timed out after ${Math.round(h/1e3)}s (last status: ${E})`);throw $.subgraphJobId=l,$.subgraphStatus=E,$}import{existsSync as Ct,readFileSync as Oo}from"node:fs";import{join as Fe,dirname as Pt}from"node:path";var ce=class{static async loadContext(e,t,n={}){let i={},r=n.filenames||["CONTEXT.md","AGENTS.md"];if(e){let a=Pt(Fe(t,e));for(let c of r){let u=await this.findAndMergeContextFiles(c,a,t);if(u){let d=c.replace(/\.[^.]+$/,"").toLowerCase();i[d]=u}}}let s=n.discovery||{};for(let[a,c]of Object.entries(s))try{let u=Fe(t,c);Ct(u)&&(i[a]=await this.loadFile(u))}catch(u){console.warn(`[workflow] could not load context '${a}' from '${c}': ${u.message}`)}return i}static async findAndMergeContextFiles(e,t,n){let i=[],r=t;for(;r.startsWith(n);){let s=Fe(r,e);if(Ct(s))try{i.unshift(await this.loadFile(s))}catch(c){console.warn(`[workflow] could not load ${e} from ${s}: ${c.message}`)}let a=Pt(r);if(a===r)break;r=a}return i.length===0?null:i.every(s=>typeof s=="string")?i.join(`
30
30
 
31
31
  ---
32
32
 
33
- `):s.every(i=>typeof i=="object")?Object.assign({},...s):s[s.length-1]}static async loadFile(e){let o=Oo(e,"utf-8");if(e.endsWith(".json"))return JSON.parse(o);if(e.endsWith(".js")||e.endsWith(".mjs")){let{pathToFileURL:n}=await import("url"),s=await import(n(e).href);return s.default||s}return o}};import{mkdirSync as Mt,existsSync as Ue,writeFileSync as Rt,unlinkSync as Co}from"node:fs";import{join as Y,resolve as Dt}from"node:path";import{config as Po}from"dotenv";import{zodToJsonSchema as Bt}from"zod-to-json-schema";import{z as Se}from"zod";import Ro from"handlebars";function Bo({traceFrom:t,sessionId:e,sessionPath:o,idSource:n,mkdirFresh:s}){if(!(process.env.ZIBBY_SESSION_LOG==="1"||process.env.ZIBBY_SESSION_LOG==="true"))return;let i=typeof process.ppid=="number"?process.ppid:"n/a",a=`[zibby:session] from=${t} pid=${process.pid} ppid=${i} sessionId=${e} source=${n} mkdir=${s?"yes":"no"} path=${o}`;if(console.log(a),process.env.ZIBBY_TRACE_SESSION==="1"||process.env.ZIBBY_TRACE_SESSION==="true"){let d=(new Error("session trace").stack||"").split(`
33
+ `):i.every(s=>typeof s=="object")?Object.assign({},...i):i[i.length-1]}static async loadFile(e){let t=Oo(e,"utf-8");if(e.endsWith(".json"))return JSON.parse(t);if(e.endsWith(".js")||e.endsWith(".mjs")){let{pathToFileURL:n}=await import("url"),i=await import(n(e).href);return i.default||i}return t}};import{mkdirSync as Mt,existsSync as Ue,writeFileSync as Rt,unlinkSync as Co}from"node:fs";import{join as Y,resolve as Dt}from"node:path";import{config as Po}from"dotenv";import{zodToJsonSchema as Bt}from"zod-to-json-schema";import{z as Se}from"zod";import Ro from"handlebars";function Bo({traceFrom:o,sessionId:e,sessionPath:t,idSource:n,mkdirFresh:i}){if(!(process.env.ZIBBY_SESSION_LOG==="1"||process.env.ZIBBY_SESSION_LOG==="true"))return;let s=typeof process.ppid=="number"?process.ppid:"n/a",a=`[zibby:session] from=${o} pid=${process.pid} ppid=${s} sessionId=${e} source=${n} mkdir=${i?"yes":"no"} path=${t}`;if(console.log(a),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
- `);console.log(`[zibby:session] stack (${t}):
36
- ${d}`)}}function Lt(){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 jt(){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 Dt(String(e).trim())}catch{return String(e).trim()}}function Ft(){Lt()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function Ut({sessionPath:t,sessionId:e}){t&&typeof t=="string"&&(process.env.ZIBBY_SESSION_PATH=t),e!=null&&String(e).trim()!==""&&(process.env.ZIBBY_SESSION_ID=String(e).trim())}function Wt(t={}){let e=ke.map(r=>process.env[r]).find(Boolean),o=Math.random().toString(36).slice(2,6),n=e||`${Date.now()}_${o}`,s=t.paths?.sessionPrefix;return s?`${s}_${n}`:n}function Gt({cwd:t=process.cwd(),config:e={},initialState:o={},traceFrom:n="resolveWorkflowSession"}={}){let s=o.sessionPath,r=o.sessionTimestamp,i="initialState.sessionPath";if(!s&&process.env.ZIBBY_SESSION_PATH)try{let u=Dt(String(process.env.ZIBBY_SESSION_PATH));u&&(s=u,i="ZIBBY_SESSION_PATH")}catch{}let a;if(s)a=String(s).split(/[/\\]/).filter(Boolean).pop(),r==null&&(r=Date.now());else{let u=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(u)a=u,i="ZIBBY_SESSION_ID";else{let p=e.sessionId!=null?String(e.sessionId).trim():"";p&&p!=="last"?(a=p,i="config.sessionId"):(a=Wt(e),i="generated")}r=r??Date.now();let d=e.paths?.output||ne;s=Y(t,d,ve,a)}let c=!Ue(s);return c&&Mt(s,{recursive:!0}),(c||i!=="initialState.sessionPath")&&Bo({traceFrom:n,sessionId:a,sessionPath:s,idSource:i,mkdirFresh:c}),Ut({sessionPath:s,sessionId:a}),{sessionPath:s,sessionId:a,sessionTimestamp:r}}var ue=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,o,n={}){if(!(o instanceof j)&&o&&typeof o=="object"&&typeof o.workflow=="string"){let r=o,i={name:e,_isCustomCode:!0,retries:r.retries,onComplete:r.onComplete,execute:async c=>{let u=c?.state&&typeof c.state.getAll=="function"?c.state.getAll():c,d;return typeof r.input=="function"?d=r.input(u):r.input&&typeof r.input=="object"?d=r.input:d={},je(r.workflow,{input:d,async:r.async===!0,conversationId:typeof r.conversationId=="function"?r.conversationId(u):r.conversationId,output:r.output,timeoutMs:r.timeoutMs,pollIntervalMs:r.pollIntervalMs,signal:u?._signal,parentAgent:c?.agent})}},a=new j(i);return a.name=e,this.nodes.set(e,a),n.prompt&&this.nodePrompts.set(e,n.prompt),Object.keys(n).length>0&&this.nodeOptions.set(e,n),this}let s=o instanceof j?o:new j(o);return s.name=e,this.nodes.set(e,s),n.prompt&&this.nodePrompts.set(e,n.prompt),Object.keys(n).length>0&&this.nodeOptions.set(e,n),this}addConditionalNode(e,o){return this.nodes.set(e,new ie({...o,name:e})),this}addEdge(e,o){return this.edges.set(e,o),this}setNodeType(e,o){return this.nodeTypeMap.set(e,o),this}addConditionalEdges(e,o,{labels:n}={}){return this.edges.set(e,{conditional:!0,routes:o,labels:n}),typeof o=="function"&&this.conditionalCodeMap.set(e,o.toString()),this}setEntryPoint(e){return this.entryPoint=e,this}use(e){return typeof e=="function"&&this.middleware.push(e),this}_composeMiddleware(e,o,n,s,r){let i=n;for(let a=e.length-1;a>=0;a--){let c=e[a],u=i;i=()=>c(o,u,s,r)}return i()}serialize(){let e=[],o={};for(let[u,d]of this.nodes){let p=this.nodeTypeMap.get(u)||u;e.push({id:u,type:p,data:{nodeType:p,label:u}});let l={};d._isCustomCode&&typeof d.execute=="function"&&(l.customCode=d.execute.toString());let h=this.nodePrompts.get(u);if(h&&(l.prompt=h),typeof d.customExecute=="function"&&(l.executeCode=d.customExecute.toString()),d.outputSchema)try{if(typeof d.outputSchema._def<"u"){let I=typeof Se?.toJSONSchema=="function"?Se.toJSONSchema(d.outputSchema):Bt(d.outputSchema,{target:"openApi3"});l.outputSchema={jsonSchema:I,variables:this._flattenJsonSchemaToVariables(I)}}else l.outputSchema={schema:d.outputSchema}}catch(I){console.warn(`[workflow] failed to convert schema for ${u}:`,I.message)}let m=(this.resolvedToolsMap||{})[u];m?.toolIds&&(l.tools=m.toolIds);let w=Array.isArray(d?.config?.skills)?d.config.skills:Array.isArray(d?.skills)?d.skills:null;w&&w.length>0&&(l.skills=[...w]),Object.keys(l).length>0&&(o[u]=l)}let n=[];for(let[u,d]of this.edges)if(typeof d=="string")n.push({source:u,target:d});else if(d.conditional){let p=this.conditionalCodeMap.get(u)||d.routes.toString(),l=this._inferConditionalTargets(d.routes),h=d.labels||{};for(let m of l){let w={source:u,target:m,data:{conditionalCode:p}};h[m]&&(w.label=h[m]),n.push(w)}}let s=u=>{if(!u)return null;if(typeof Se?.toJSONSchema=="function")try{return Se.toJSONSchema(u)}catch{}try{return Bt(u,{target:"openApi3"})}catch{return null}},r=this._runtimeSchema(),i=s(r||this.stateSchema),a=s(this.inputSchema),c=s(this.contextSchema);return{nodes:e,edges:n,nodeConfigs:o,stateSchema:i,inputSchema:a,contextSchema:c}}_inferConditionalTargets(e){let o=e.toString(),n=new Set,s=/return\s+['"]([^'"]+)['"]/g,r;for(;(r=s.exec(o))!==null;)n.add(r[1]);return[...n]}_flattenJsonSchemaToVariables(e,o=""){let n=e;if(e.$ref&&e.definitions){let s=e.$ref.replace("#/definitions/","");n=e.definitions[s]||e}return this._flattenSchema(n,o)}_flattenSchema(e,o=""){if(!e||typeof e!="object")return[];let n=[],s=e.properties||{},r=e.required||[];for(let[i,a]of Object.entries(s)){let c=o?`${o}.${i}`:i;n.push({path:c,type:a.type||"unknown",label:a.description||this._formatLabel(i),optional:!r.includes(i)}),a.type==="object"&&a.properties&&n.push(...this._flattenSchema(a,c)),a.type==="array"&&a.items?.type==="object"&&a.items.properties&&n.push(...this._flattenSchema(a.items,`${c}[]`))}return n}_formatLabel(e){return e.replace(/([A-Z])/g," $1").replace(/^./,o=>o.toUpperCase()).trim()}_summarizeNodeOutput(e,o){if(!o||typeof o!="object")return[];let n=[];o.success!==void 0&&n.push(`Result: ${o.success?"passed":"failed"}`);for(let[s,r]of Object.entries(o))if(!(s==="success"||s==="raw"||s==="nextNode")){if(typeof r=="string"&&r.length<=80)n.push(`${s}: ${r}`);else if(Array.isArray(r)){let i=r.length,a=r.filter(u=>u?.passed===!0).length,c=r.some(u=>u?.passed!==void 0);n.push(c?`${s}: ${a}/${i} passed${i-a?`, ${i-a} failed`:""}`:`${s}: ${i} items`)}if(n.length>=4)break}return n}async run(e,o={},n={}){if(!this.entryPoint)throw new Error("No entry point set for graph");let s=new AbortController;n.signal&&(n.signal.aborted?s.abort():n.signal.addEventListener("abort",()=>s.abort(),{once:!0}));let r=n.strategyAbortTimeoutMs??o.config?.strategyAbortTimeoutMs??5e3,i=o.cwd||process.cwd();Po({path:Y(i,".env")});let a=o.config||{};if(!a||Object.keys(a).length===0)try{let b=Y(i,".zibby.config.js");Ue(b)&&(a=(await import(b)).default||{})}catch{}process.env.EXECUTION_ID&&!a.agent?.strictMode&&(a.agent={...a.agent,strictMode:!0});let c=o.agentType;if(!c){let b=a?.agent;b?.provider?c=b.provider:b?.gemini?c="gemini":b?.claude?c="claude":b?.cursor?c="cursor":b?.codex?c="codex":c=process.env.AGENT_TYPE||"cursor"}let u=o.contextConfig||e?.config?.contextConfig||e?.config?.context||a?.context||{},d=this._runtimeSchema();if(d){let b=d.safeParse(o);if(!b.success){let N=b.error.issues.map(O=>`${O.path.join(".")}: ${O.message}`);throw console.error("\u274C Initial state validation failed:"),N.forEach(O=>console.error(` - ${O}`)),new Error(`State validation failed: ${N.join(", ")}`)}x.step("State validated against schema")}let p=jt(),l=o.sessionPath||p;l||Ft();let{sessionPath:h,sessionTimestamp:m,sessionId:w}=Gt({cwd:i,config:a,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:l,sessionTimestamp:o.sessionTimestamp}});x.step(`Session ${w}`);let I=await ce.loadContext(o.specPath||"",i,u);Object.keys(I).length>0&&x.step(`Context loaded: ${Object.keys(I).join(", ")}`);let E=o.outputPath;!E&&o.specPath&&(e?.calculateOutputPath?E=e.calculateOutputPath(o.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${o.specPath})`));let f=new Q({...o,config:a,agentType:c,outputPath:E,sessionPath:h,sessionTimestamp:m,context:I,resolvedTools:this.resolvedToolsMap||{},_signal:s.signal}),$=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:g}=await Promise.resolve().then(()=>(re(),mt)),S=a.skills&&typeof a.skills=="object"?a.skills:{},R=Object.values(S).filter(b=>b&&typeof b=="object"&&typeof b.id=="string"),v=b=>{for(let N of R)if(N.id===b)return N;return g(b)},D=new Set;for(let[,b]of this.nodes)for(let N of b.config?.skills||[])D.add(N);for(let b of D){let N=v(b);if(typeof N?.middleware=="function")try{let O=await N.middleware();typeof O=="function"&&$.set(b,O)}catch{}}let y=this.entryPoint,de=[],ze=a?.recursionLimit??100,zt=0;try{for(;y&&y!=="END";){if(++zt>ze)throw new Error(`Workflow exceeded recursion limit (${ze}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let N=Y(h,xe);if(Ue(N)){try{Co(N)}catch{}s.abort()}if(s.signal.aborted)return console.warn(`
37
- \u{1F6D1} External stop requested \u2014 ending workflow.`),x.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:de,stoppedExternally:!0};let O=this.nodes.get(y);if(!O)throw new Error(`Node '${y}' not found in graph`);let Ze=JSON.stringify({sessionPath:h,sessionTimestamp:m,currentNode:y,createdAt:new Date().toISOString(),config:f.get("config")}),Zt=Y(h,W);Rt(Zt,Ze,"utf-8");let Ve=f.get("config")?.paths?.output||ne,Vt=Y(i,Ve,W);Mt(Y(i,Ve),{recursive:!0});try{Rt(Vt,Ze,"utf-8")}catch{}let Ke=o.onPipelineProgress;if(typeof Ke=="function")try{Ke({cwd:i,sessionPath:h,sessionId:w,outputBase:f.get("config")?.paths?.output||ne,currentNode:y})}catch{}let Kt=(this.resolvedToolsMap||{})[y]||null;f.set("_currentNodeTools",Kt);let qt=f.get("nodeConfigs")||{};f.set("_currentNodeConfig",qt[y]||{}),x.nodeStart(y);let qe=Date.now(),pe=this.nodePrompts.get(y);if(!this._invokeAgent){let C=await Promise.resolve().then(()=>(q(),se));this._invokeAgent=C.invokeAgent}let Xt=this._invokeAgent,Ie={},Qt=O.config?.skills||[];for(let C of Qt){let P=v(C);if(typeof P?.invokeAgentOptions=="function")try{let T=P.invokeAgentOptions(f.getAll(),{agentType:f.get("agentType"),nodeName:y});T&&typeof T=="object"&&(Ie={...Ie,...T})}catch(T){console.warn(`[graph] skill '${C}' invokeAgentOptions threw: ${T.message}`)}}let Xe=async(C,P,T={})=>{let B=Xt(C,P,{...Ie,...T,signal:s.signal});return B.catch(()=>{}),s.signal.aborted?B:Promise.race([B,new Promise((z,Z)=>{let L=()=>{setTimeout(()=>{let X=new Error(`Strategy ignored AbortSignal \u2014 engine deadman fired after ${r}ms`);X.name="AbortError",Z(X)},r)};s.signal.addEventListener("abort",L,{once:!0})})])},Qe={state:f,invokeAgent:async(C={},P={})=>{let T=P.prompt||"";if(pe){let B=this._compiledPrompts.get(y);B||(B=Ro.compile(pe,{noEscape:!0}),this._compiledPrompts.set(y,B));try{T=B(C)}catch(z){throw console.error(`\u274C Template rendering failed for node '${y}':`,z.message),new Error(`Template rendering failed: ${z.message}`,{cause:z})}}else if(!T)throw new Error(`No prompt template configured for node '${y}' and no prompt provided in options`);return Xe(T,{state:f.getAll(),images:P.images||[]},{model:P.model||f.get("model"),workspace:f.get("workspace"),schema:P.schema,...P,signal:s.signal})},_coreInvokeAgent:Xe,agent:e,nodeId:y,promptTemplate:pe,getPromptTemplate:()=>pe,...f.getAll()};try{let C=(O.config?.skills||[]).map(L=>$.get(L)).filter(Boolean),P=[...this.middleware,...C],T;P.length>0?T=await this._composeMiddleware(P,y,async()=>O.execute(Qe,f),f.getAll(),f):T=await O.execute(Qe,f);let B=Date.now()-qe;if(de.push({node:y,success:T.success,duration:B,timestamp:new Date().toISOString()}),!T.success){if(s.signal.aborted)return x.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:de,stoppedExternally:!0};f.append("errors",{node:y,error:T.error});let L=O.config?.retries||0,X=`${y}_retries`,fe=f.getAll()[X]||0;if(fe<L){x.stepInfo(`Retrying (attempt ${fe+1}/${L})`),f.update({[X]:fe+1,[`${y}_raw`]:T.raw});continue}throw x.nodeFailed(y,T.error,{duration:B}),new Error(`Node '${y}' failed after ${fe} attempts: ${T.error}`)}f.update({[y]:T.output});let z=this._summarizeNodeOutput(y,T.output);x.nodeComplete(y,{duration:B,details:z});let Z=this.edges.get(y);if(!Z)y="END";else if(Z.conditional){let L=Z.routes(f.getAll());x.route(y,L),y=L}else y=Z}catch(C){throw x.isInsideNode&&x.nodeFailed(y,C.message,{duration:Date.now()-qe}),f.set("failed",!0),f.set("failedAt",y),C}}x.graphComplete();let b={success:!0,state:f.getAll(),executionLog:de};return e&&typeof e.onComplete=="function"&&await e.onComplete(b),b}finally{if(e&&typeof e.cleanup=="function")try{await e.cleanup()}catch(b){console.warn(`[workflow] agent.cleanup() failed: ${b.message}`)}}}};var We=Symbol.for("@zibby/agent-workflow.nodes");globalThis[We]||(globalThis[We]=new Map);var le=globalThis[We];function Jt(t,e){le.set(t,e)}function Ge(t){return le.get(t)}function we(t){return le.has(t)}function Mo(){return Array.from(le.keys())}function Je(t){let e=le.get(t);return e?e.factory&&typeof e.create=="function"?e.create.toString():typeof e.execute=="function"?e.execute.toString():typeof e=="function"?e.toString():null:null}Jt("ai_agent",{name:"ai_agent",factory:!0,create:(t,e={})=>({name:t,_isCustomCode:!0,execute:async o=>{let n=o?._coreInvokeAgent;n||(n=(await Promise.resolve().then(()=>(q(),se))).invokeAgent);let s=e.extraPromptInstructions||"Execute the task based on the current state.",r=Do(s,o),i=await n(r,{cwd:o.workspace||process.cwd(),model:o.model,tools:e.resolvedTools||null});return{success:!0,output:{raw:i,nodeId:t},raw:typeof i=="string"?i:i.raw}}})});function Do(t,e){let o=/@([\w.]+)/g,n=new Set,s;for(;(s=o.exec(t))!==null;)n.add(s[1]);if(n.size===0)return t;let r=[],i=new Set;for(let a of n){let c=a.split(".")[0];if(i.has(c))continue;let u=a.split(".").reduce((l,h)=>l?.[h],e);if(u===void 0)continue;let d=typeof u=="string"?u:u?.raw??JSON.stringify(u,null,2),p=a.replace(/_/g," ").replace(/\b\w/g,l=>l.toUpperCase());r.push(`## ${p}
38
- ${d}`),a.includes(".")||i.add(c)}return r.length===0?t:`${t}
35
+ `);console.log(`[zibby:session] stack (${o}):
36
+ ${d}`)}}function Lt(){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 jt(){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 Dt(String(e).trim())}catch{return String(e).trim()}}function Ft(){Lt()||(delete process.env.ZIBBY_SESSION_PATH,delete process.env.ZIBBY_SESSION_ID)}function Ut({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 Wt(o={}){let e=ke.map(r=>process.env[r]).find(Boolean),t=Math.random().toString(36).slice(2,6),n=e||`${Date.now()}_${t}`,i=o.paths?.sessionPrefix;return i?`${i}_${n}`:n}function Gt({cwd:o=process.cwd(),config:e={},initialState:t={},traceFrom:n="resolveWorkflowSession"}={}){let i=t.sessionPath,r=t.sessionTimestamp,s="initialState.sessionPath";if(!i&&process.env.ZIBBY_SESSION_PATH)try{let u=Dt(String(process.env.ZIBBY_SESSION_PATH));u&&(i=u,s="ZIBBY_SESSION_PATH")}catch{}let a;if(i)a=String(i).split(/[/\\]/).filter(Boolean).pop(),r==null&&(r=Date.now());else{let u=process.env.ZIBBY_SESSION_ID&&String(process.env.ZIBBY_SESSION_ID).trim();if(u)a=u,s="ZIBBY_SESSION_ID";else{let p=e.sessionId!=null?String(e.sessionId).trim():"";p&&p!=="last"?(a=p,s="config.sessionId"):(a=Wt(e),s="generated")}r=r??Date.now();let d=e.paths?.output||ne;i=Y(o,d,ve,a)}let c=!Ue(i);return c&&Mt(i,{recursive:!0}),(c||s!=="initialState.sessionPath")&&Bo({traceFrom:n,sessionId:a,sessionPath:i,idSource:s,mkdirFresh:c}),Ut({sessionPath:i,sessionId:a}),{sessionPath:i,sessionId:a,sessionTimestamp:r}}var ue=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,n={}){if(!(t instanceof j)&&t&&typeof t=="object"&&typeof t.workflow=="string"){let r=t,s={name:e,_isCustomCode:!0,retries:r.retries,onComplete:r.onComplete,execute:async c=>{let u=c?.state&&typeof c.state.getAll=="function"?c.state.getAll():c,d;return typeof r.input=="function"?d=r.input(u):r.input&&typeof r.input=="object"?d=r.input:d={},je(r.workflow,{input:d,async:r.async===!0,conversationId:typeof r.conversationId=="function"?r.conversationId(u):r.conversationId,output:r.output,timeoutMs:r.timeoutMs,pollIntervalMs:r.pollIntervalMs,signal:u?._signal,parentAgent:c?.agent})}},a=new j(s);return a.name=e,this.nodes.set(e,a),n.prompt&&this.nodePrompts.set(e,n.prompt),Object.keys(n).length>0&&this.nodeOptions.set(e,n),this}let i=t instanceof j?t:new j(t);return i.name=e,this.nodes.set(e,i),n.prompt&&this.nodePrompts.set(e,n.prompt),Object.keys(n).length>0&&this.nodeOptions.set(e,n),this}addConditionalNode(e,t){return this.nodes.set(e,new ie({...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:n}={}){return this.edges.set(e,{conditional:!0,routes:t,labels:n}),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,n,i,r){let s=n;for(let a=e.length-1;a>=0;a--){let c=e[a],u=s;s=()=>c(t,u,i,r)}return s()}serialize(){let e=[],t={};for(let[u,d]of this.nodes){let p=this.nodeTypeMap.get(u)||u;e.push({id:u,type:p,data:{nodeType:p,label:u}});let l={};d._isCustomCode&&typeof d.execute=="function"&&(l.customCode=d.execute.toString());let h=this.nodePrompts.get(u);if(h&&(l.prompt=h),typeof d.customExecute=="function"&&(l.executeCode=d.customExecute.toString()),d.outputSchema)if(typeof d.outputSchema._def<"u"){let I=null;if(typeof Se?.toJSONSchema=="function")try{I=Se.toJSONSchema(d.outputSchema)}catch{}if(!I)try{I=Bt(d.outputSchema,{target:"openApi3"})}catch{}l.outputSchema=I?{jsonSchema:I,variables:this._flattenJsonSchemaToVariables(I)}:{schema:d.outputSchema}}else l.outputSchema={schema:d.outputSchema};let m=(this.resolvedToolsMap||{})[u];m?.toolIds&&(l.tools=m.toolIds);let w=Array.isArray(d?.config?.skills)?d.config.skills:Array.isArray(d?.skills)?d.skills:null;w&&w.length>0&&(l.skills=[...w]),Object.keys(l).length>0&&(t[u]=l)}let n=[];for(let[u,d]of this.edges)if(typeof d=="string")n.push({source:u,target:d});else if(d.conditional){let p=this.conditionalCodeMap.get(u)||d.routes.toString(),l=this._inferConditionalTargets(d.routes,d.labels),h=d.labels||{};for(let m of l){let w={source:u,target:m,data:{conditionalCode:p}};h[m]&&(w.label=h[m]),n.push(w)}}let i=u=>{if(!u)return null;if(typeof Se?.toJSONSchema=="function")try{return Se.toJSONSchema(u)}catch{}try{return Bt(u,{target:"openApi3"})}catch{return null}},r=this._runtimeSchema(),s=i(r||this.stateSchema),a=i(this.inputSchema),c=i(this.contextSchema);return{nodes:e,edges:n,nodeConfigs:t,stateSchema:s,inputSchema:a,contextSchema:c}}_inferConditionalTargets(e,t){let n=e.toString(),i=new Set,r=/(['"])((?:\\.|(?!\1).)*?)\1|`((?:\\.|[^`$]|\$(?!\{))*?)`/g,s;for(;(s=r.exec(n))!==null;){let u=s[2]!==void 0?s[2]:s[3];u!==void 0&&u!==""&&i.add(u)}let a=new Set(["END","START","__end__","__start__"]);for(let u of this.nodes.keys())a.add(u);if(t&&typeof t=="object")for(let u of Object.keys(t))a.add(u);let c=new Set;for(let u of i)a.has(u)&&c.add(u);if(c.size===0){let u=/return\s+['"]([^'"]+)['"]/g,d;for(;(d=u.exec(n))!==null;)c.add(d[1])}return[...c]}_flattenJsonSchemaToVariables(e,t=""){let n=e;if(e.$ref&&e.definitions){let i=e.$ref.replace("#/definitions/","");n=e.definitions[i]||e}return this._flattenSchema(n,t)}_flattenSchema(e,t=""){if(!e||typeof e!="object")return[];let n=[],i=e.properties||{},r=e.required||[];for(let[s,a]of Object.entries(i)){let c=t?`${t}.${s}`:s;n.push({path:c,type:a.type||"unknown",label:a.description||this._formatLabel(s),optional:!r.includes(s)}),a.type==="object"&&a.properties&&n.push(...this._flattenSchema(a,c)),a.type==="array"&&a.items?.type==="object"&&a.items.properties&&n.push(...this._flattenSchema(a.items,`${c}[]`))}return n}_formatLabel(e){return e.replace(/([A-Z])/g," $1").replace(/^./,t=>t.toUpperCase()).trim()}_summarizeNodeOutput(e,t){if(!t||typeof t!="object")return[];let n=[];t.success!==void 0&&n.push(`Result: ${t.success?"passed":"failed"}`);for(let[i,r]of Object.entries(t))if(!(i==="success"||i==="raw"||i==="nextNode")){if(typeof r=="string"&&r.length<=80)n.push(`${i}: ${r}`);else if(Array.isArray(r)){let s=r.length,a=r.filter(u=>u?.passed===!0).length,c=r.some(u=>u?.passed!==void 0);n.push(c?`${i}: ${a}/${s} passed${s-a?`, ${s-a} failed`:""}`:`${i}: ${s} items`)}if(n.length>=4)break}return n}async run(e,t={},n={}){if(!this.entryPoint)throw new Error("No entry point set for graph");let i=new AbortController;n.signal&&(n.signal.aborted?i.abort():n.signal.addEventListener("abort",()=>i.abort(),{once:!0}));let r=n.strategyAbortTimeoutMs??t.config?.strategyAbortTimeoutMs??5e3,s=t.cwd||process.cwd();Po({path:Y(s,".env")});let a=t.config||{};if(!a||Object.keys(a).length===0)try{let b=Y(s,".zibby.config.js");Ue(b)&&(a=(await import(b)).default||{})}catch{}process.env.EXECUTION_ID&&!a.agent?.strictMode&&(a.agent={...a.agent,strictMode:!0});let c=t.agentType;if(!c){let b=a?.agent;b?.provider?c=b.provider:b?.gemini?c="gemini":b?.claude?c="claude":b?.cursor?c="cursor":b?.codex?c="codex":c=process.env.AGENT_TYPE||"cursor"}let u=t.contextConfig||e?.config?.contextConfig||e?.config?.context||a?.context||{},d=this._runtimeSchema();if(d){let b=d.safeParse(t);if(!b.success){let N=b.error.issues.map(O=>`${O.path.join(".")}: ${O.message}`);throw console.error("\u274C Initial state validation failed:"),N.forEach(O=>console.error(` - ${O}`)),new Error(`State validation failed: ${N.join(", ")}`)}x.step("State validated against schema")}let p=jt(),l=t.sessionPath||p;l||Ft();let{sessionPath:h,sessionTimestamp:m,sessionId:w}=Gt({cwd:s,config:a,traceFrom:"WorkflowGraph.run",initialState:{sessionPath:l,sessionTimestamp:t.sessionTimestamp}});x.step(`Session ${w}`);let I=await ce.loadContext(t.specPath||"",s,u);Object.keys(I).length>0&&x.step(`Context loaded: ${Object.keys(I).join(", ")}`);let E=t.outputPath;!E&&t.specPath&&(e?.calculateOutputPath?E=e.calculateOutputPath(t.specPath):console.warn(`\u26A0\uFE0F outputPath not resolved (specPath=${t.specPath})`));let f=new Q({...t,config:a,agentType:c,outputPath:E,sessionPath:h,sessionTimestamp:m,context:I,resolvedTools:this.resolvedToolsMap||{},_signal:i.signal}),$=new Map;try{await import("@zibby/skills")}catch{}let{getSkill:g}=await Promise.resolve().then(()=>(re(),mt)),S=a.skills&&typeof a.skills=="object"?a.skills:{},R=Object.values(S).filter(b=>b&&typeof b=="object"&&typeof b.id=="string"),v=b=>{for(let N of R)if(N.id===b)return N;return g(b)},D=new Set;for(let[,b]of this.nodes)for(let N of b.config?.skills||[])D.add(N);for(let b of D){let N=v(b);if(typeof N?.middleware=="function")try{let O=await N.middleware();typeof O=="function"&&$.set(b,O)}catch{}}let y=this.entryPoint,de=[],ze=a?.recursionLimit??100,zt=0;try{for(;y&&y!=="END";){if(++zt>ze)throw new Error(`Workflow exceeded recursion limit (${ze}) \u2014 likely a cyclic conditional route. Set config.recursionLimit if you need a higher cap.`);let N=Y(h,xe);if(Ue(N)){try{Co(N)}catch{}i.abort()}if(i.signal.aborted)return console.warn(`
37
+ \u{1F6D1} External stop requested \u2014 ending workflow.`),x.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:de,stoppedExternally:!0};let O=this.nodes.get(y);if(!O)throw new Error(`Node '${y}' not found in graph`);let Ze=JSON.stringify({sessionPath:h,sessionTimestamp:m,currentNode:y,createdAt:new Date().toISOString(),config:f.get("config")}),Zt=Y(h,W);Rt(Zt,Ze,"utf-8");let Ve=f.get("config")?.paths?.output||ne,Vt=Y(s,Ve,W);Mt(Y(s,Ve),{recursive:!0});try{Rt(Vt,Ze,"utf-8")}catch{}let Ke=t.onPipelineProgress;if(typeof Ke=="function")try{Ke({cwd:s,sessionPath:h,sessionId:w,outputBase:f.get("config")?.paths?.output||ne,currentNode:y})}catch{}let Kt=(this.resolvedToolsMap||{})[y]||null;f.set("_currentNodeTools",Kt);let qt=f.get("nodeConfigs")||{};f.set("_currentNodeConfig",qt[y]||{}),x.nodeStart(y);let qe=Date.now(),pe=this.nodePrompts.get(y);if(!this._invokeAgent){let C=await Promise.resolve().then(()=>(q(),se));this._invokeAgent=C.invokeAgent}let Xt=this._invokeAgent,Ie={},Qt=O.config?.skills||[];for(let C of Qt){let P=v(C);if(typeof P?.invokeAgentOptions=="function")try{let T=P.invokeAgentOptions(f.getAll(),{agentType:f.get("agentType"),nodeName:y});T&&typeof T=="object"&&(Ie={...Ie,...T})}catch(T){console.warn(`[graph] skill '${C}' invokeAgentOptions threw: ${T.message}`)}}let Xe=async(C,P,T={})=>{let B=Xt(C,P,{...Ie,...T,signal:i.signal});return B.catch(()=>{}),i.signal.aborted?B:Promise.race([B,new Promise((z,Z)=>{let L=()=>{setTimeout(()=>{let X=new Error(`Strategy ignored AbortSignal \u2014 engine deadman fired after ${r}ms`);X.name="AbortError",Z(X)},r)};i.signal.addEventListener("abort",L,{once:!0})})])},Qe={state:f,invokeAgent:async(C={},P={})=>{let T=P.prompt||"";if(pe){let B=this._compiledPrompts.get(y);B||(B=Ro.compile(pe,{noEscape:!0}),this._compiledPrompts.set(y,B));try{T=B(C)}catch(z){throw console.error(`\u274C Template rendering failed for node '${y}':`,z.message),new Error(`Template rendering failed: ${z.message}`,{cause:z})}}else if(!T)throw new Error(`No prompt template configured for node '${y}' and no prompt provided in options`);return Xe(T,{state:f.getAll(),images:P.images||[]},{model:P.model||f.get("model"),workspace:f.get("workspace"),schema:P.schema,...P,signal:i.signal})},_coreInvokeAgent:Xe,agent:e,nodeId:y,promptTemplate:pe,getPromptTemplate:()=>pe,...f.getAll()};try{let C=(O.config?.skills||[]).map(L=>$.get(L)).filter(Boolean),P=[...this.middleware,...C],T;P.length>0?T=await this._composeMiddleware(P,y,async()=>O.execute(Qe,f),f.getAll(),f):T=await O.execute(Qe,f);let B=Date.now()-qe;if(de.push({node:y,success:T.success,duration:B,timestamp:new Date().toISOString()}),!T.success){if(i.signal.aborted)return x.step("Workflow stopped externally"),{success:!0,state:f.getAll(),executionLog:de,stoppedExternally:!0};f.append("errors",{node:y,error:T.error});let L=O.config?.retries||0,X=`${y}_retries`,fe=f.getAll()[X]||0;if(fe<L){x.stepInfo(`Retrying (attempt ${fe+1}/${L})`),f.update({[X]:fe+1,[`${y}_raw`]:T.raw});continue}throw x.nodeFailed(y,T.error,{duration:B}),new Error(`Node '${y}' failed after ${fe} attempts: ${T.error}`)}f.update({[y]:T.output});let z=this._summarizeNodeOutput(y,T.output);x.nodeComplete(y,{duration:B,details:z});let Z=this.edges.get(y);if(!Z)y="END";else if(Z.conditional){let L=Z.routes(f.getAll());x.route(y,L),y=L}else y=Z}catch(C){throw x.isInsideNode&&x.nodeFailed(y,C.message,{duration:Date.now()-qe}),f.set("failed",!0),f.set("failedAt",y),C}}x.graphComplete();let b={success:!0,state:f.getAll(),executionLog:de};return e&&typeof e.onComplete=="function"&&await e.onComplete(b),b}finally{if(e&&typeof e.cleanup=="function")try{await e.cleanup()}catch(b){console.warn(`[workflow] agent.cleanup() failed: ${b.message}`)}}}};var We=Symbol.for("@zibby/agent-workflow.nodes");globalThis[We]||(globalThis[We]=new Map);var le=globalThis[We];function Jt(o,e){le.set(o,e)}function Ge(o){return le.get(o)}function we(o){return le.has(o)}function Mo(){return Array.from(le.keys())}function Je(o){let e=le.get(o);return e?e.factory&&typeof e.create=="function"?e.create.toString():typeof e.execute=="function"?e.execute.toString():typeof e=="function"?e.toString():null:null}Jt("ai_agent",{name:"ai_agent",factory:!0,create:(o,e={})=>({name:o,_isCustomCode:!0,execute:async t=>{let n=t?._coreInvokeAgent;n||(n=(await Promise.resolve().then(()=>(q(),se))).invokeAgent);let i=e.extraPromptInstructions||"Execute the task based on the current state.",r=Do(i,t),s=await n(r,{cwd:t.workspace||process.cwd(),model:t.model,tools:e.resolvedTools||null});return{success:!0,output:{raw:s,nodeId:o},raw:typeof s=="string"?s:s.raw}}})});function Do(o,e){let t=/@([\w.]+)/g,n=new Set,i;for(;(i=t.exec(o))!==null;)n.add(i[1]);if(n.size===0)return o;let r=[],s=new Set;for(let a of n){let c=a.split(".")[0];if(s.has(c))continue;let u=a.split(".").reduce((l,h)=>l?.[h],e);if(u===void 0)continue;let d=typeof u=="string"?u:u?.raw??JSON.stringify(u,null,2),p=a.replace(/_/g," ").replace(/\b\w/g,l=>l.toUpperCase());r.push(`## ${p}
38
+ ${d}`),a.includes(".")||s.add(c)}return r.length===0?o:`${o}
39
39
 
40
40
  ---
41
41
  # Referenced Context
42
42
 
43
43
  ${r.join(`
44
44
 
45
- `)}`}re();F();var ye={};function Ye(t,e){if(Array.isArray(e))return He(e);let o=ye[t];return!o||o.length===0?null:He(o)}function He(t){if(!Array.isArray(t)||t.length===0)return null;let e=[],o={},n=[];for(let s of t){let r=K(s);if(!r){_.warn(`[workflow] unknown skill "${s}" \u2014 skipping`);continue}n.push(s);for(let i of r.tools||[])e.push({name:i.name,description:i.description,input_schema:i.input_schema||{type:"object",properties:{}}});if(!o[r.serverName])if(typeof r.resolve=="function"){let i=r.resolve();i&&(o[r.serverName]={...i,toolPrefix:s})}else{let i={};for(let a of r.envKeys||[]){let c=process.env[a];c&&(i[a]=c)}o[r.serverName]={command:r.command,args:[...r.args||[]],env:i,toolPrefix:s}}}return n.length===0?null:{toolIds:n,claudeTools:e,mcpServers:o}}F();function Lo(t,e={}){let{nodes:o,edges:n,nodeConfigs:s={}}=t;if(!Array.isArray(o)||o.length===0)throw new M("Graph must have at least one node");if(!Array.isArray(n))throw new M("Graph edges must be an array");let r=new ue(e);e.stateSchema&&r.setStateSchema(e.stateSchema);let i=new Set,a=new Map,c={};for(let l of o){let h=_e(l);a.set(l.id,{...l,resolvedType:h}),h==="decision"&&i.add(l.id)}for(let[l,h]of a){if(i.has(l))continue;let m=h.resolvedType,w=s[l]||{},I=Ye(m,w.tools);I&&(c[l]=I);let E={};w.prompt&&(E.prompt=w.prompt);let f=we(m);if(_.debug(`[workflow] compiler: node "${l}" type="${m}" registered=${f}`),w.customCode&&!f)r.addNode(l,Ht(l,w.customCode,w),E),r.setNodeType(l,m);else if(f){let $=Ge(m);$.factory?r.addNode(l,$.create(l,{...w,resolvedTools:I}),E):r.addNode(l,$,E),r.setNodeType(l,m)}else if(w.executeCode)r.addNode(l,Ht(l,w.executeCode,w),E),r.setNodeType(l,m);else throw new M(`Unknown node type "${m}" for node "${l}". Did you forget to register it?`)}r.resolvedToolsMap=c;let u=new Set;for(let l of n)i.has(l.target)||u.add(l.target);let d=o.find(l=>!i.has(l.id)&&!u.has(l.id));if(!d)throw new M("Could not determine entry point: no node without incoming edges found");r.setEntryPoint(d.id);let p=Uo(n,"source");for(let l of n)if(!i.has(l.source))if(i.has(l.target)){let h=l.target,m=p.get(h)||[];if(m.length===0)throw new M(`Decision node "${h}" has no outgoing edges`);let w=Wo(h,m,i);r.addConditionalEdges(l.source,w)}else r.addEdge(l.source,l.target);return r}function jo(t){let e=[];if(!t||typeof t!="object")return{valid:!1,errors:["Config must be a non-null object"]};if((!Array.isArray(t.nodes)||t.nodes.length===0)&&e.push("Graph must have at least one node"),Array.isArray(t.edges)||e.push("Graph edges must be an array"),e.length>0)return{valid:!1,errors:e};let o=t.nodeConfigs||{};for(let a of t.nodes){let c=_e(a);if(c==="decision"||we(c))continue;let u=o[a.id]||{};u.customCode||u.executeCode||e.push(`Unknown node type "${c}" for node "${a.id}". Register it or provide customCode/executeCode.`)}let n=new Set(t.nodes.map(a=>a.id));for(let a of t.edges)n.has(a.source)||e.push(`Edge references unknown source node "${a.source}"`),n.has(a.target)||e.push(`Edge references unknown target node "${a.target}"`);let s=new Set(t.nodes.filter(a=>_e(a)==="decision").map(a=>a.id)),r=new Set;for(let a of t.edges)s.has(a.target)||r.add(a.target);let i=t.nodes.filter(a=>!s.has(a.id)&&!r.has(a.id));i.length===0?e.push("No entry point found (every node has incoming edges)"):i.length>1&&e.push(`Multiple entry points found: ${i.map(a=>a.id).join(", ")}`);for(let a of s){let c=t.edges.filter(d=>d.source===a);c.length===0&&e.push(`Decision node "${a}" has no outgoing edges`),c.some(d=>d.data?.conditionalCode||d.conditionalCode)||e.push(`Decision node "${a}" outgoing edges have no conditionalCode`)}return{valid:e.length===0,errors:e}}function Fo(t){return!t||!Array.isArray(t.nodes)?[]:t.nodes.filter(e=>_e(e)!=="decision").map(e=>e.id)}function _e(t){let e=t.data?.nodeType||t.data?.type||t.type;return e==="workflowNode"||e==="custom"||e==="default"?t.id:e}function Uo(t,e){let o=new Map;for(let n of t){let s=n[e];o.has(s)||o.set(s,[]),o.get(s).push(n)}return o}function Wo(t,e,o){let n=e.find(a=>a.data?.conditionalCode||a.conditionalCode);if(!n)throw new M(`Decision node "${t}" has no conditionalCode on its outgoing edges`);let s=n.data?.conditionalCode||n.conditionalCode,r=new Set(e.map(a=>a.target).filter(a=>!o.has(a))),i;try{let c=new Function(`return (${s})`)();i=u=>{let d=c(u);return r.has(d)||_.warn(`[workflow] conditional route from "${t}" returned "${d}" which is not in valid targets: ${[...r].join(", ")}`),d}}catch(a){throw new M(`Failed to compile conditionalCode for "${t}": ${a.message}`)}return i}function Ht(t,e,o={}){let n;try{n=new Function("invokeAgent","require","console",`return (${e})`)}catch(i){throw new M(`Failed to compile customCode for node "${t}": ${i.message}`)}let s=n(async(...i)=>{let{invokeAgent:a}=await Promise.resolve().then(()=>(q(),se));return a(...i)},typeof Ee<"u"?Ee:void 0,console),r=null;return o.outputSchema&&(r=o.outputSchema.jsonSchema||o.outputSchema),{name:t,_isCustomCode:!0,outputSchema:r,execute:async i=>{try{let a=await s(i);return typeof a=="object"&&"success"in a?a:{success:!0,output:a,raw:null}}catch(a){return{success:!1,error:a.message,raw:null}}}}}var M=class extends Error{constructor(e){super(e),this.name="CompilationError"}};re();Ne();q();function Go(t,e={}){let{nodes:o,edges:n,nodeConfigs:s={}}=t,r=new Set,i=[],a=new Map;for(let I of o){let E=I.data?.nodeType||I.type;a.set(I.id,E),E==="decision"?r.add(I.id):i.push({id:I.id,nodeType:E,label:I.data?.label||I.id})}let c=i.some(I=>{let E=s[I.id]||{};return!E.customCode&&!E.executeCode}),{toolsPerNode:u,toolIdsByVar:d}=qo(i,s),{simpleEdges:p,conditionalEdges:l}=Xo(n,r),h=Qo(i,n,r),m=[],w=e.workflowType||"workflow";return m.push(Ho(e)),m.push(Yo(w,{usesRegisteredNodes:c})),m.push(zo(d)),m.push(Zo(w)),m.push(Vo(i,s)),m.push(Ko(i,h,p,l,u,w)),m.filter(Boolean).join(`
46
- `)}function Jo(t){let e={};for(let[o,n]of Object.entries(t)){let{tools:s,...r}=n;Object.keys(r).length>0&&(e[o]=r)}return e}function Ho(t){let e=t.workflowType||"workflow";return["// Generated workflow",`// ${t.projectId?`Project: ${t.projectId} | `:""}Type: ${e} | Version: ${t.version??0}`,`// Downloaded: ${new Date().toISOString()}`,""].join(`
47
- `)}function Yo(t,{usesRegisteredNodes:e=!0}={}){let o=["import { WorkflowGraph, invokeAgent, getResolvedToolDefinitions } from '@zibby/agent-workflow';"];return e&&o.push("// import './register-nodes.js'; // register custom node types here"),o.push("import { readFileSync, writeFileSync, existsSync, mkdirSync } from 'node:fs';","import { join, dirname } from 'node:path';","import { fileURLToPath } from 'node:url';",""),o.join(`
48
- `)}function zo(t){if(t.size===0)return"";let e=["// \u2500\u2500 Tool Bindings \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500"];for(let[o,n]of t)e.push(`const ${o} = getResolvedToolDefinitions(${JSON.stringify(n)}); // ${n.join(", ")}`);return e.push(""),e.join(`
49
- `)}function Zo(t){return["// \u2500\u2500 Node Configs \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500","const __filename = fileURLToPath(import.meta.url);","const __dirname = dirname(__filename);",`const configPath = join(__dirname, 'workflow-${t}.config.json');`,"const nodeConfigs = existsSync(configPath) ? JSON.parse(readFileSync(configPath, 'utf-8')) : {};",""].join(`
50
- `)}function Vo(t,e){let o=["// \u2500\u2500 Node Implementations \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500",""];for(let n of t){let s=Yt(n.id),r=e[n.id]?.customCode;if(r)o.push(`// @custom \u2014 modified from default "${n.nodeType}" template`),o.push(`const ${s}_execute = ${r};`);else{let i=Je(n.nodeType);i?(o.push(`// Default "${n.nodeType}" implementation`),o.push(`const ${s}_execute = ${i};`)):(o.push(`// No template for "${n.nodeType}" \u2014 passthrough`),o.push(`const ${s}_execute = async (state) => ({ success: true, output: {}, raw: null });`))}o.push("")}return o.join(`
51
- `)}function Ko(t,e,o,n,s,r){let i=["// \u2500\u2500 Graph Builder \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500"];i.push("export function buildGraph(options = {}) {"),i.push(" const graph = new WorkflowGraph(options);",""),i.push(" // Nodes");for(let c of t){let u=Yt(c.id);i.push(` graph.addNode('${c.id}', { name: '${c.id}', execute: ${u}_execute });`),i.push(` graph.setNodeType('${c.id}', '${c.nodeType}');`)}i.push("",` graph.setEntryPoint('${e}');`,""),(o.length>0||n.length>0)&&i.push(" // Edges");for(let c of o)i.push(` graph.addEdge('${c.source}', '${c.target}');`);for(let c of n){let u=c.code.split(`
45
+ `)}`}re();F();var ye={};function Ye(o,e){if(Array.isArray(e))return He(e);let t=ye[o];return!t||t.length===0?null:He(t)}function He(o){if(!Array.isArray(o)||o.length===0)return null;let e=[],t={},n=[];for(let i of o){let r=K(i);if(!r){_.warn(`[workflow] unknown skill "${i}" \u2014 skipping`);continue}n.push(i);for(let s of r.tools||[])e.push({name:s.name,description:s.description,input_schema:s.input_schema||{type:"object",properties:{}}});if(!t[r.serverName])if(typeof r.resolve=="function"){let s=r.resolve();s&&(t[r.serverName]={...s,toolPrefix:i})}else{let s={};for(let a of r.envKeys||[]){let c=process.env[a];c&&(s[a]=c)}t[r.serverName]={command:r.command,args:[...r.args||[]],env:s,toolPrefix:i}}}return n.length===0?null:{toolIds:n,claudeTools:e,mcpServers:t}}F();function Lo(o,e={}){let{nodes:t,edges:n,nodeConfigs:i={}}=o;if(!Array.isArray(t)||t.length===0)throw new M("Graph must have at least one node");if(!Array.isArray(n))throw new M("Graph edges must be an array");let r=new ue(e);e.stateSchema&&r.setStateSchema(e.stateSchema);let s=new Set,a=new Map,c={};for(let l of t){let h=_e(l);a.set(l.id,{...l,resolvedType:h}),h==="decision"&&s.add(l.id)}for(let[l,h]of a){if(s.has(l))continue;let m=h.resolvedType,w=i[l]||{},I=Ye(m,w.tools);I&&(c[l]=I);let E={};w.prompt&&(E.prompt=w.prompt);let f=we(m);if(_.debug(`[workflow] compiler: node "${l}" type="${m}" registered=${f}`),w.customCode&&!f)r.addNode(l,Ht(l,w.customCode,w),E),r.setNodeType(l,m);else if(f){let $=Ge(m);$.factory?r.addNode(l,$.create(l,{...w,resolvedTools:I}),E):r.addNode(l,$,E),r.setNodeType(l,m)}else if(w.executeCode)r.addNode(l,Ht(l,w.executeCode,w),E),r.setNodeType(l,m);else throw new M(`Unknown node type "${m}" for node "${l}". Did you forget to register it?`)}r.resolvedToolsMap=c;let u=new Set;for(let l of n)s.has(l.target)||u.add(l.target);let d=t.find(l=>!s.has(l.id)&&!u.has(l.id));if(!d)throw new M("Could not determine entry point: no node without incoming edges found");r.setEntryPoint(d.id);let p=Uo(n,"source");for(let l of n)if(!s.has(l.source))if(s.has(l.target)){let h=l.target,m=p.get(h)||[];if(m.length===0)throw new M(`Decision node "${h}" has no outgoing edges`);let w=Wo(h,m,s);r.addConditionalEdges(l.source,w)}else r.addEdge(l.source,l.target);return r}function jo(o){let e=[];if(!o||typeof o!="object")return{valid:!1,errors:["Config must be a non-null object"]};if((!Array.isArray(o.nodes)||o.nodes.length===0)&&e.push("Graph must have at least one node"),Array.isArray(o.edges)||e.push("Graph edges must be an array"),e.length>0)return{valid:!1,errors:e};let t=o.nodeConfigs||{};for(let a of o.nodes){let c=_e(a);if(c==="decision"||we(c))continue;let u=t[a.id]||{};u.customCode||u.executeCode||e.push(`Unknown node type "${c}" for node "${a.id}". Register it or provide customCode/executeCode.`)}let n=new Set(o.nodes.map(a=>a.id));for(let a of o.edges)n.has(a.source)||e.push(`Edge references unknown source node "${a.source}"`),n.has(a.target)||e.push(`Edge references unknown target node "${a.target}"`);let i=new Set(o.nodes.filter(a=>_e(a)==="decision").map(a=>a.id)),r=new Set;for(let a of o.edges)i.has(a.target)||r.add(a.target);let s=o.nodes.filter(a=>!i.has(a.id)&&!r.has(a.id));s.length===0?e.push("No entry point found (every node has incoming edges)"):s.length>1&&e.push(`Multiple entry points found: ${s.map(a=>a.id).join(", ")}`);for(let a of i){let c=o.edges.filter(d=>d.source===a);c.length===0&&e.push(`Decision node "${a}" has no outgoing edges`),c.some(d=>d.data?.conditionalCode||d.conditionalCode)||e.push(`Decision node "${a}" outgoing edges have no conditionalCode`)}return{valid:e.length===0,errors:e}}function Fo(o){return!o||!Array.isArray(o.nodes)?[]:o.nodes.filter(e=>_e(e)!=="decision").map(e=>e.id)}function _e(o){let e=o.data?.nodeType||o.data?.type||o.type;return e==="workflowNode"||e==="custom"||e==="default"?o.id:e}function Uo(o,e){let t=new Map;for(let n of o){let i=n[e];t.has(i)||t.set(i,[]),t.get(i).push(n)}return t}function Wo(o,e,t){let n=e.find(a=>a.data?.conditionalCode||a.conditionalCode);if(!n)throw new M(`Decision node "${o}" has no conditionalCode on its outgoing edges`);let i=n.data?.conditionalCode||n.conditionalCode,r=new Set(e.map(a=>a.target).filter(a=>!t.has(a))),s;try{let c=new Function(`return (${i})`)();s=u=>{let d=c(u);return r.has(d)||_.warn(`[workflow] conditional route from "${o}" returned "${d}" which is not in valid targets: ${[...r].join(", ")}`),d}}catch(a){throw new M(`Failed to compile conditionalCode for "${o}": ${a.message}`)}return s}function Ht(o,e,t={}){let n;try{n=new Function("invokeAgent","require","console",`return (${e})`)}catch(s){throw new M(`Failed to compile customCode for node "${o}": ${s.message}`)}let i=n(async(...s)=>{let{invokeAgent:a}=await Promise.resolve().then(()=>(q(),se));return a(...s)},typeof Ee<"u"?Ee:void 0,console),r=null;return t.outputSchema&&(r=t.outputSchema.jsonSchema||t.outputSchema),{name:o,_isCustomCode:!0,outputSchema:r,execute:async s=>{try{let a=await i(s);return typeof a=="object"&&"success"in a?a:{success:!0,output:a,raw:null}}catch(a){return{success:!1,error:a.message,raw:null}}}}}var M=class extends Error{constructor(e){super(e),this.name="CompilationError"}};re();Ne();q();function Go(o,e={}){let{nodes:t,edges:n,nodeConfigs:i={}}=o,r=new Set,s=[],a=new Map;for(let I of t){let E=I.data?.nodeType||I.type;a.set(I.id,E),E==="decision"?r.add(I.id):s.push({id:I.id,nodeType:E,label:I.data?.label||I.id})}let c=s.some(I=>{let E=i[I.id]||{};return!E.customCode&&!E.executeCode}),{toolsPerNode:u,toolIdsByVar:d}=qo(s,i),{simpleEdges:p,conditionalEdges:l}=Xo(n,r),h=Qo(s,n,r),m=[],w=e.workflowType||"workflow";return m.push(Ho(e)),m.push(Yo(w,{usesRegisteredNodes:c})),m.push(zo(d)),m.push(Zo(w)),m.push(Vo(s,i)),m.push(Ko(s,h,p,l,u,w)),m.filter(Boolean).join(`
46
+ `)}function Jo(o){let e={};for(let[t,n]of Object.entries(o)){let{tools:i,...r}=n;Object.keys(r).length>0&&(e[t]=r)}return e}function Ho(o){let e=o.workflowType||"workflow";return["// Generated workflow",`// ${o.projectId?`Project: ${o.projectId} | `:""}Type: ${e} | Version: ${o.version??0}`,`// Downloaded: ${new Date().toISOString()}`,""].join(`
47
+ `)}function Yo(o,{usesRegisteredNodes:e=!0}={}){let t=["import { WorkflowGraph, invokeAgent, getResolvedToolDefinitions } from '@zibby/agent-workflow';"];return e&&t.push("// import './register-nodes.js'; // register custom node types here"),t.push("import { readFileSync, writeFileSync, existsSync, mkdirSync } from 'node:fs';","import { join, dirname } from 'node:path';","import { fileURLToPath } from 'node:url';",""),t.join(`
48
+ `)}function zo(o){if(o.size===0)return"";let e=["// \u2500\u2500 Tool Bindings \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500"];for(let[t,n]of o)e.push(`const ${t} = getResolvedToolDefinitions(${JSON.stringify(n)}); // ${n.join(", ")}`);return e.push(""),e.join(`
49
+ `)}function Zo(o){return["// \u2500\u2500 Node Configs \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500","const __filename = fileURLToPath(import.meta.url);","const __dirname = dirname(__filename);",`const configPath = join(__dirname, 'workflow-${o}.config.json');`,"const nodeConfigs = existsSync(configPath) ? JSON.parse(readFileSync(configPath, 'utf-8')) : {};",""].join(`
50
+ `)}function Vo(o,e){let t=["// \u2500\u2500 Node Implementations \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500",""];for(let n of o){let i=Yt(n.id),r=e[n.id]?.customCode;if(r)t.push(`// @custom \u2014 modified from default "${n.nodeType}" template`),t.push(`const ${i}_execute = ${r};`);else{let s=Je(n.nodeType);s?(t.push(`// Default "${n.nodeType}" implementation`),t.push(`const ${i}_execute = ${s};`)):(t.push(`// No template for "${n.nodeType}" \u2014 passthrough`),t.push(`const ${i}_execute = async (state) => ({ success: true, output: {}, raw: null });`))}t.push("")}return t.join(`
51
+ `)}function Ko(o,e,t,n,i,r){let s=["// \u2500\u2500 Graph Builder \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500"];s.push("export function buildGraph(options = {}) {"),s.push(" const graph = new WorkflowGraph(options);",""),s.push(" // Nodes");for(let c of o){let u=Yt(c.id);s.push(` graph.addNode('${c.id}', { name: '${c.id}', execute: ${u}_execute });`),s.push(` graph.setNodeType('${c.id}', '${c.nodeType}');`)}s.push("",` graph.setEntryPoint('${e}');`,""),(t.length>0||n.length>0)&&s.push(" // Edges");for(let c of t)s.push(` graph.addEdge('${c.source}', '${c.target}');`);for(let c of n){let u=c.code.split(`
52
52
  `).map((d,p)=>p===0?d:` ${d}`).join(`
53
- `);i.push(` graph.addConditionalEdges('${c.source}', ${u});`)}let a=[];for(let c of t){let u=s.get(c.id);u&&a.push(` '${c.id}': ${u},`)}return a.length>0&&i.push(""," graph.resolvedToolsMap = {",...a," };"),i.push(""," return graph;","}",""),i.push("export { nodeConfigs };",""),i.join(`
54
- `)}function qo(t,e){let o=new Map,n=new Map;for(let s of t){let r=e[s.id]?.tools,i;if(Array.isArray(r)&&r.length>0)i=[...r].sort();else{let a=ye[s.nodeType];a?.length>0&&(i=[...a].sort())}if(i){let a=`${i.map(c=>c.replace(/[^a-zA-Z0-9]/g,"")).join("And")}Tools`;o.set(s.id,a),n.has(a)||n.set(a,i)}}return{toolsPerNode:o,toolIdsByVar:n}}function Xo(t,e){let o=[],n=[],s=new Map,r=new Set;for(let i of t)s.has(i.source)||s.set(i.source,[]),s.get(i.source).push(i);for(let i of t)if(!e.has(i.source))if(e.has(i.target)){if(r.has(i.target))continue;r.add(i.target);let c=(s.get(i.target)||[]).find(u=>u.data?.conditionalCode||u.conditionalCode);c&&n.push({source:i.source,code:c.data?.conditionalCode||c.conditionalCode})}else o.push({source:i.source,target:i.target});return{simpleEdges:o,conditionalEdges:n}}function Qo(t,e,o){let n=new Set;for(let r of e)o.has(r.target)||n.add(r.target);let s=t.find(r=>!n.has(r.id));return s?s.id:t[0]?.id}function Yt(t){return t.replace(/[^a-zA-Z0-9]/g,"_")}F();export{Ae as AgentStrategy,ke as CI_ENV_VARS,M as CompilationError,ie as ConditionalNode,ce as ContextLoader,ne as DEFAULT_OUTPUT_BASE,ao as EVENTS_FILE,ye as NODE_DEFAULT_TOOLS,j as Node,ee as OutputParser,io as RAW_OUTPUT_FILE,so as RESULT_FILE,ve as SESSIONS_DIR,W as SESSION_INFO_FILE,co as SKILLS,xe as STOP_REQUEST_FILE,oo as SchemaTypes,ge as Timeline,lt as WORKFLOW_GRAPH_LOG_MARKER_PREFIX,ue as WorkflowGraph,Q as WorkflowState,Ft as clearInheritedSessionEnvForFreshRun,gt as clearSkills,Lo as compileGraph,je as dispatchSubgraph,Fo as extractSteps,Jo as generateNodeConfigsJson,Go as generateWorkflowCode,Wt as generateWorkflowSessionId,Pe as getAgentStrategy,ft as getAllSkills,Ge as getNodeImpl,Je as getNodeTemplate,He as getResolvedToolDefinitions,K as getSkill,we as hasNode,pt as hasSkill,yt as invokeAgent,Mo as listNodeTypes,ht as listSkillIds,wt as listStrategies,jt as readPinnedSessionPathFromEnv,Jt as registerNode,dt as registerSkill,St as registerStrategy,Ye as resolveNodeTools,Gt as resolveWorkflowSession,no as setLogger,Lt as shouldTrustInheritedSessionEnv,Ut as syncProcessEnvToSession,x as timeline,jo as validateGraphConfig};
53
+ `);s.push(` graph.addConditionalEdges('${c.source}', ${u});`)}let a=[];for(let c of o){let u=i.get(c.id);u&&a.push(` '${c.id}': ${u},`)}return a.length>0&&s.push(""," graph.resolvedToolsMap = {",...a," };"),s.push(""," return graph;","}",""),s.push("export { nodeConfigs };",""),s.join(`
54
+ `)}function qo(o,e){let t=new Map,n=new Map;for(let i of o){let r=e[i.id]?.tools,s;if(Array.isArray(r)&&r.length>0)s=[...r].sort();else{let a=ye[i.nodeType];a?.length>0&&(s=[...a].sort())}if(s){let a=`${s.map(c=>c.replace(/[^a-zA-Z0-9]/g,"")).join("And")}Tools`;t.set(i.id,a),n.has(a)||n.set(a,s)}}return{toolsPerNode:t,toolIdsByVar:n}}function Xo(o,e){let t=[],n=[],i=new Map,r=new Set;for(let s of o)i.has(s.source)||i.set(s.source,[]),i.get(s.source).push(s);for(let s of o)if(!e.has(s.source))if(e.has(s.target)){if(r.has(s.target))continue;r.add(s.target);let c=(i.get(s.target)||[]).find(u=>u.data?.conditionalCode||u.conditionalCode);c&&n.push({source:s.source,code:c.data?.conditionalCode||c.conditionalCode})}else t.push({source:s.source,target:s.target});return{simpleEdges:t,conditionalEdges:n}}function Qo(o,e,t){let n=new Set;for(let r of e)t.has(r.target)||n.add(r.target);let i=o.find(r=>!n.has(r.id));return i?i.id:o[0]?.id}function Yt(o){return o.replace(/[^a-zA-Z0-9]/g,"_")}F();export{Ae as AgentStrategy,ke as CI_ENV_VARS,M as CompilationError,ie as ConditionalNode,ce as ContextLoader,ne as DEFAULT_OUTPUT_BASE,ao as EVENTS_FILE,ye as NODE_DEFAULT_TOOLS,j as Node,ee as OutputParser,io as RAW_OUTPUT_FILE,so as RESULT_FILE,ve as SESSIONS_DIR,W as SESSION_INFO_FILE,co as SKILLS,xe as STOP_REQUEST_FILE,oo as SchemaTypes,ge as Timeline,lt as WORKFLOW_GRAPH_LOG_MARKER_PREFIX,ue as WorkflowGraph,Q as WorkflowState,Ft as clearInheritedSessionEnvForFreshRun,gt as clearSkills,Lo as compileGraph,je as dispatchSubgraph,Fo as extractSteps,Jo as generateNodeConfigsJson,Go as generateWorkflowCode,Wt as generateWorkflowSessionId,Pe as getAgentStrategy,ft as getAllSkills,Ge as getNodeImpl,Je as getNodeTemplate,He as getResolvedToolDefinitions,K as getSkill,we as hasNode,pt as hasSkill,yt as invokeAgent,Mo as listNodeTypes,ht as listSkillIds,wt as listStrategies,jt as readPinnedSessionPathFromEnv,Jt as registerNode,dt as registerSkill,St as registerStrategy,Ye as resolveNodeTools,Gt as resolveWorkflowSession,no as setLogger,Lt as shouldTrustInheritedSessionEnv,Ut as syncProcessEnvToSession,x as timeline,jo as validateGraphConfig};
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 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+=`
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 G,nt,k,l,T=N(()=>{G=()=>{},nt={debug:G,info:G,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,7 +11,7 @@ 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 M,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===`
14
+ `)}`);return t}};T();import{writeFileSync as C,readFileSync as tt,existsSync as et,mkdirSync as ht}from"node:fs";import{join as M,dirname as ft}from"node:path";import c from"chalk";var at="__WORKFLOW_GRAPH_LOG__",I=c.gray("\u2502"),lt=c.gray("\u250C"),D=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
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(`
@@ -24,6 +24,6 @@ ${B}`,t.col=H))}return r(a,o,s)}}var x=class{constructor(){this._currentNode=nul
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
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
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 F=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}
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(`${D} ${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(`${D} ${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 F=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
29
  ${n}`);let g=o(),p=g.cwd||process.cwd(),_=g.sessionPath;try{if(_){let i=M(_,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=M(_,"..",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=M(_,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 F{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,F as Node};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@zibby/agent-workflow",
3
- "version": "0.4.9",
3
+ "version": "0.4.10",
4
4
  "description": "Graph-based AI agent workflow orchestration. Bring your own agent strategies.",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",