@zibby/core 0.4.1 → 0.4.3

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.
@@ -1,4 +1,4 @@
1
- import{AgentStrategy as It,getSkill as wt}from"@zibby/agent-workflow";import{execSync as bt,spawn as vt}from"node:child_process";import{zodToJsonSchema as Tt}from"zod-to-json-schema";import{existsSync as st,mkdirSync as ot,readFileSync as it,rmSync as At,writeFileSync as lt}from"node:fs";import{join as b}from"node:path";import G from"chalk";var g={debug:0,info:1,warn:2,error:3,silent:4},V=class{constructor(){this._level=this._getLogLevel()}_getLogLevel(){if(process.env.ZIBBY_DEBUG==="true")return g.debug;if(process.env.ZIBBY_VERBOSE==="true")return g.info;let t=process.env.LOG_LEVEL?.toLowerCase();return t&&t in g?g[t]:g.info}_shouldLog(t){return g[t]>=this._level}_formatMessage(t,e,r={}){let s=new Date().toISOString(),n=`${this._getPrefix(t)} ${e}`;return Object.keys(r).length>0&&(n+=G.dim(` ${JSON.stringify(r)}`)),n}_getPrefix(t){return{debug:G.gray("[DEBUG]"),info:G.cyan("[INFO]"),warn:G.yellow("[WARN]"),error:G.red("\u274C [ERROR]")}[t]||""}debug(t,e){this._shouldLog("debug")&&console.log(this._formatMessage("debug",t,e))}info(t,e){this._shouldLog("info")&&console.log(this._formatMessage("info",t,e))}warn(t,e){this._shouldLog("warn")&&console.warn(this._formatMessage("warn",t,e))}error(t,e){this._shouldLog("error")&&console.error(this._formatMessage("error",t,e))}setLevel(t){t in g&&(this._level=g[t])}getLevel(){return Object.keys(g).find(t=>g[t]===this._level)}},u=new V;var tt={ASSISTANT:"gpt-5.4-nano-2026-03-17",CLAUDE:"claude-sonnet-4-6",CURSOR:"auto",CODEX:"o4-mini",GEMINI:"gemini-2.5-pro",OPENAI_POSTPROCESSING:"gpt-4o-mini"};var et={auto:"gemini-2.5-pro","gemini-2.5-pro":"gemini-2.5-pro","gemini-2.5-flash":"gemini-2.5-flash"},kt={CURSOR_AGENT_DEFAULT:1200*1e3,OPENAI_REQUEST:18e4};import{zodToJsonSchema as dt}from"zod-to-json-schema";var M=class{static generateFileOutputInstructions(t,e){let r;typeof t?.parse=="function"?r=dt(t,{target:"openApi3"}):r=t;let s=this._buildExample(r);return`
1
+ import{AgentStrategy as pt,getSkill as mt}from"@zibby/agent-workflow";import{execSync as dt,spawn as ht}from"node:child_process";import{zodToJsonSchema as gt}from"zod-to-json-schema";import{existsSync as et,mkdirSync as rt,readFileSync as st,rmSync as yt,writeFileSync as nt}from"node:fs";import{join as w}from"node:path";import P from"chalk";var g={debug:0,info:1,warn:2,error:3,silent:4},W=class{constructor(){this._level=this._getLogLevel()}_getLogLevel(){if(process.env.ZIBBY_DEBUG==="true")return g.debug;if(process.env.ZIBBY_VERBOSE==="true")return g.info;let t=process.env.LOG_LEVEL?.toLowerCase();return t&&t in g?g[t]:g.info}_shouldLog(t){return g[t]>=this._level}_formatMessage(t,e,s={}){let n=new Date().toISOString(),r=`${this._getPrefix(t)} ${e}`;return Object.keys(s).length>0&&(r+=P.dim(` ${JSON.stringify(s)}`)),r}_getPrefix(t){return{debug:P.gray("[DEBUG]"),info:P.cyan("[INFO]"),warn:P.yellow("[WARN]"),error:P.red("\u274C [ERROR]")}[t]||""}debug(t,e){this._shouldLog("debug")&&console.log(this._formatMessage("debug",t,e))}info(t,e){this._shouldLog("info")&&console.log(this._formatMessage("info",t,e))}warn(t,e){this._shouldLog("warn")&&console.warn(this._formatMessage("warn",t,e))}error(t,e){this._shouldLog("error")&&console.error(this._formatMessage("error",t,e))}setLevel(t){t in g&&(this._level=g[t])}getLevel(){return Object.keys(g).find(t=>g[t]===this._level)}},u=new W;var Q={ASSISTANT:"gpt-5.4-nano-2026-03-17",CLAUDE:"claude-sonnet-4-6",CURSOR:"auto",CODEX:"o4-mini",GEMINI:"gemini-2.5-pro",OPENAI_POSTPROCESSING:"gpt-4o-mini"};var tt={auto:"gemini-2.5-pro","gemini-2.5-pro":"gemini-2.5-pro","gemini-2.5-flash":"gemini-2.5-flash"},It={CURSOR_AGENT_DEFAULT:1200*1e3,OPENAI_REQUEST:18e4};import{zodToJsonSchema as ft}from"zod-to-json-schema";var $=class{static generateFileOutputInstructions(t,e){let s;typeof t?.parse=="function"?s=ft(t,{target:"openApi3"}):s=t;let n=this._buildExample(s);return`
2
2
  \u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550
3
3
  \u{1F6A8} MANDATORY: WRITE RESULT TO FILE \u{1F6A8}
4
4
  \u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550
@@ -11,34 +11,33 @@ DO NOT just output JSON to stdout. The file MUST exist when you finish.
11
11
  DO NOT skip this step. The workflow WILL FAIL if the file is missing.
12
12
 
13
13
  Required JSON structure:
14
- ${JSON.stringify(s,null,2)}
14
+ ${JSON.stringify(n,null,2)}
15
15
 
16
16
  JSON types (strict \u2014 validators reject wrong types):
17
17
  - Use bare JSON numbers for numeric fields (e.g. 3000). Do NOT quote them as strings (wrong: "3000").
18
18
  - Use true/false without quotes for booleans.
19
19
  - Use unquoted null where a field may be null.
20
20
 
21
- Rules: valid JSON only, no markdown wrapping, no extra text in the file.`}static _buildExample(t){if(!t)return{};let e=t.type;if(e==="object"&&t.properties){let r={};for(let[s,o]of Object.entries(t.properties))r[s]=this._buildExample(o);return r}if(e==="array"&&t.items)return[this._buildExample(t.items)];if(t.enum&&Array.isArray(t.enum)&&t.enum.length>0)return`<${t.enum.join("|")}>`;if(e==="string")return"<string>";if(e==="number"||e==="integer")return 0;if(e==="boolean")return!1;if(t.description)return`<${t.description}>`;if(t.nullable||t.oneOf||t.anyOf){let r=t.oneOf?.find(s=>s.type!=="null")||t.anyOf?.find(s=>s.type!=="null");return r?this._buildExample(r):null}return"<value>"}};var C=class l{constructor(t=""){this.userPrompt=t,this.systemInstructions=[],this.metadata={}}setUserPrompt(t){return this.userPrompt=t,this}appendUserPrompt(t){return this.userPrompt?this.userPrompt+=`
21
+ Rules: valid JSON only, no markdown wrapping, no extra text in the file.`}static _buildExample(t){if(!t)return{};let e=t.type;if(e==="object"&&t.properties){let s={};for(let[n,i]of Object.entries(t.properties))s[n]=this._buildExample(i);return s}if(e==="array"&&t.items)return[this._buildExample(t.items)];if(t.enum&&Array.isArray(t.enum)&&t.enum.length>0)return`<${t.enum.join("|")}>`;if(e==="string")return"<string>";if(e==="number"||e==="integer")return 0;if(e==="boolean")return!1;if(t.description)return`<${t.description}>`;if(t.nullable||t.oneOf||t.anyOf){let s=t.oneOf?.find(n=>n.type!=="null")||t.anyOf?.find(n=>n.type!=="null");return s?this._buildExample(s):null}return"<value>"}};var C=class c{constructor(t=""){this.userPrompt=t,this.systemInstructions=[],this.metadata={}}setUserPrompt(t){return this.userPrompt=t,this}appendUserPrompt(t){return this.userPrompt?this.userPrompt+=`
22
22
 
23
- ${t}`:this.userPrompt=t,this}addSkillHints(t){return t&&this.systemInstructions.push({type:"skill_hints",content:t,position:"prepend"}),this}addStructuredOutput(t,e){if(t&&e){let r=M.generateFileOutputInstructions(t,e);this.systemInstructions.push({type:"structured_output",content:r,position:"append"}),this.metadata.structuredOutputPath=e}return this}addExtraInstructions(t){if(t?.trim()){let e=`\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
23
+ ${t}`:this.userPrompt=t,this}addSkillHints(t){return t&&this.systemInstructions.push({type:"skill_hints",content:t,position:"prepend"}),this}addStructuredOutput(t,e){if(t&&e){let s=$.generateFileOutputInstructions(t,e);this.systemInstructions.push({type:"structured_output",content:s,position:"append"}),this.metadata.structuredOutputPath=e}return this}addExtraInstructions(t){if(t?.trim()){let e=`\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
24
24
  \u26A0\uFE0F PRIORITY OVERRIDE \u2014 THE FOLLOWING INSTRUCTIONS TAKE PRECEDENCE OVER ALL PREVIOUS CONTENT
25
25
  \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
26
26
 
27
27
  ${t.trim()}`;this.systemInstructions.push({type:"extra_instructions",content:e,position:"append"})}return this}addSkillFragments(t){return t&&t.length>0&&this.systemInstructions.push({type:"skill_fragments",content:t.join(`
28
28
 
29
- `),position:"append"}),this}addSystemInstruction(t,e="custom",r="append"){return t&&this.systemInstructions.push({type:e,content:t,position:r}),this}getUserPrompt(){return this.userPrompt}getSystemInstructions(){return this.systemInstructions}getMetadata(){return this.metadata}build(){let t=this.systemInstructions.filter(s=>s.position==="prepend").map(s=>s.content),e=this.systemInstructions.filter(s=>s.position==="append").map(s=>s.content);return[...t,this.userPrompt,...e].filter(Boolean).join(`
30
-
31
- `)}getStats(){let t=this.userPrompt.length,e=this.build().length;return{userPromptLength:t,fullPromptLength:e,systemInstructionsLength:e-t,instructionCount:this.systemInstructions.length,instructionTypes:this.systemInstructions.map(r=>r.type)}}clone(){let t=new l(this.userPrompt);return t.systemInstructions=[...this.systemInstructions],t.metadata={...this.metadata},t}};var D=class l{constructor(){this.buffer="",this.extractedResult=null,this.rawText="",this.zodSchema=null,this.lastOutputLength=0,this.onToolCall=null,this._lastToolEmit=null}processChunk(t){if(!t)return null;this.buffer+=t;let e=this.buffer.split(`
32
- `);this.buffer=e.pop()||"";let r="";for(let s of e)if(s.trim())try{let o=JSON.parse(s);this._emitToolCalls(o);let n=this.extractText(o);if(n){if(this.rawText&&n.startsWith(this.rawText)){let i=n.substring(this.rawText.length);this.rawText=n,r+=i}else(!this.rawText.includes(n)||n.length<20)&&(this.rawText+=n,r+=n);this.tryExtractResult(this.rawText)}else this.isValidResult(o)&&(this.rawText+=`${s}
33
- `,r+=`${s}
34
- `,this.extractedResult=o)}catch{if(s.includes('"text"')||s.includes('"content"')){let n=s.match(/"text"\s*:\s*"([^"]*)/),i=s.match(/"content"\s*:\s*"([^"]*)/),c=n?n[1]:i?i[1]:null;c&&!this.rawText.includes(c)&&(r+=c,this.rawText+=c)}}return r||null}flush(){if(!this.buffer.trim())return null;let t="";try{let e=JSON.parse(this.buffer);this._emitToolCalls(e);let r=this.extractText(e);r&&(this.rawText+=r,t+=r,this.tryExtractResult(r))}catch{this.rawText+=this.buffer,t+=this.buffer,this.tryExtractResult(this.buffer)}return this.buffer="",t||null}_emitToolCalls(t){if(!this.onToolCall)return;let e=(n,i)=>{if(!n)return;let c=`${n}:${JSON.stringify(i??{})}`;this._lastToolEmit!==c&&(this._lastToolEmit=c,this.onToolCall(n,i??void 0))},r=n=>{if(n!=null){if(typeof n=="object"&&!Array.isArray(n))return n;if(typeof n=="string")try{return JSON.parse(n)}catch{return}}};if(t.type==="tool_use"||t.type==="tool_call"){if(t.name){e(t.name,r(t.input??t.arguments));return}let n=t.tool_call;if(n&&typeof n=="object"&&!Array.isArray(n)){let i=Object.keys(n);if(i.length===1){let c=i[0],a=n[c],p=a&&typeof a=="object"?a.args??a.input??a:void 0;e(c,r(p))}return}return}if(Array.isArray(t.tool_calls)){for(let n of t.tool_calls)e(n.name,r(n.input??n.arguments));return}let s=t.message??t;if(Array.isArray(s?.tool_calls)){for(let n of s.tool_calls)e(n.name,r(n.input??n.arguments));return}let o=s?.content??t.content;if(Array.isArray(o))for(let n of o)(n.type==="tool_use"||n.type==="tool_call")&&n.name&&e(n.name,r(n.input??n.arguments))}extractText(t){if(t.type==="assistant"&&t.message?.content){let e=t.message.content;if(Array.isArray(e))return e.filter(r=>r.type==="text"&&r.text).map(r=>r.text).join("")}return t.type==="thinking"&&t.text||t.text?t.text:t.content&&typeof t.content=="string"?t.content:t.delta?t.delta:null}tryExtractResult(t){if(!t||typeof t!="string")return;let e=[],r=/```json\s*\n?([\s\S]*?)\n?```/g,s;for(;(s=r.exec(t))!==null;){let d=s[1].trim();try{JSON.parse(d),e.push({text:d,source:"markdown"})}catch{}}let o=0,n=0;for(;o<t.length&&(o=t.indexOf("{",o),o!==-1);){let d=0,m=o;for(let h=o;h<t.length;h++)if(t[h]==="{")d++;else if(t[h]==="}"&&(d--,d===0)){m=h,e.push({text:t.substring(o,m+1),source:"brace"}),n++;break}o=m+1}let i=this.extractedResult,c=i?JSON.stringify(i).length:0,a=0,p=-1;for(let d=0;d<e.length;d++){let m=e[d];try{let h=m.text.replace(/,(\s*[}\]])/g,"$1"),v=JSON.parse(h);this.isValidResult(v)&&(a++,c=JSON.stringify(v).length,i=v,p=d)}catch{}}i&&(this.extractedResult=i)}isValidResult(t){if(!t||typeof t!="object"||Array.isArray(t)||t.session_id||t.timestamp_ms||t.type||t.call_id||t.tool_call||t.result&&typeof t.result=="object"&&(t.result.success&&typeof t.result.success=="object"||t.result.error&&typeof t.result.error=="object")||t.args&&typeof t.args=="object")return!1;if(this.zodSchema)try{return this.zodSchema.parse(t),!0}catch{return!1}return!0}getResult(){return this.extractedResult}getRawText(){return this.rawText}static extractResult(t,e=null){let r=new l;r.zodSchema=e,r.processChunk(t),r.flush();let s=r.getResult();return!s&&process.env.LOG_LEVEL==="debug"&&console.error("[StreamingParser] No result extracted from",t?.length||0,"chars"),s}};function W(l){let t=`${l}_POOL`,e=process.env[t];if(!e||typeof e!="string")return{picked:null,count:0,dispose:()=>{}};let r=e.split(/[,\n]+/).map(n=>n.trim()).filter(Boolean);if(r.length===0)return{picked:null,count:0,dispose:()=>{}};let s=r[Math.floor(Math.random()*r.length)],o=process.env[l];return process.env[l]=s,u.debug(`[auth-pool] ${l}: picked 1 of ${r.length} from pool (***${s.slice(-4)})`),{picked:s,count:r.length,dispose(){o===void 0?delete process.env[l]:process.env[l]=o}}}import{execSync as zt}from"node:child_process";import{existsSync as ht,mkdirSync as Ht}from"node:fs";import{join as rt,resolve as gt}from"node:path";var yt="/workspace/repos",Et=".zibby/repos";function _t(){return process.env.REPOS?yt:gt(process.cwd(),Et)}function Ot(l){return String(l).replace(/\//g,"-")}function St(){let l=process.env.REPOS;if(!l)return[];try{let t=JSON.parse(l);return Array.isArray(t)?t:[]}catch{return[]}}function xt(l={}){let t=l.baseDir||_t();return St().map((r,s)=>{let o=rt(t,Ot(r.name));return{name:r.name,url:r.cloneUrl||r.url||null,branch:r.branch||null,provider:r.provider||"github",localPath:o,isCheckedOut:ht(rt(o,".git")),primary:s===0}})}function nt(l={}){let t=xt(l);if(t.length===0)return null;let e=["## Available repositories",""];for(let r of t){let s=r.isCheckedOut?"already checked out at":"will be checked out to",o=r.primary?" (primary)":"";e.push(`- \`${r.name}\`${o} \u2014 ${s} \`${r.localPath}\``)}return e.push(""),e.push(`To materialize a repo: call the \`git_checkout\` skill with the repo name (e.g. \`git_checkout({ url: "${t[0].name}" })\`). In code, \`import { ensureRepo } from "@zibby/core"\` and \`await ensureRepo()\` returns the local path (clones on first call, no-op after).`),e.join(`
35
- `)}function Pt(l){if(!l)return null;let t=String(l),e=t.match(/```(?:json)?\s*([\s\S]*?)```/i);if(e?.[1])try{return JSON.parse(e[1].trim())}catch{}let r=t.indexOf("{");if(r<0)return null;let s=0,o=!1,n=!1,i=-1;for(let c=r;c<t.length;c++){let a=t[c];if(o){n?n=!1:a==="\\"?n=!0:a==='"'&&(o=!1);continue}if(a==='"'){o=!0;continue}if(a==="{"){s===0&&(i=c),s+=1;continue}if(a==="}"){if(s===0)continue;if(s-=1,s===0&&i>=0){let p=t.slice(i,c+1);try{return JSON.parse(p)}catch{i=-1}}}}return null}function Nt(l){let t=String(l||"").trim();if(!t)return null;try{return JSON.parse(t)}catch{return Pt(t)}}function Gt(l){try{let t=JSON.parse(l);if(typeof t=="string")return t;if(typeof t?.response=="string")return t.response;if(typeof t?.text=="string")return t.text;if(typeof t?.output=="string")return t.output;if(Array.isArray(t?.candidates)&&t.candidates.length>0){let e=t.candidates[0];if(typeof e?.content=="string")return e.content;if(Array.isArray(e?.content?.parts)){let r=e.content.parts.map(s=>typeof s?.text=="string"?s.text:"").join("");if(r.trim())return r}}}catch{}return l}var ct=class extends It{constructor(){super("gemini","Gemini (Google)",70)}canHandle(t){if(!!!(process.env.GEMINI_API_KEY||process.env.GEMINI_API_KEY_POOL||process.env.GOOGLE_API_KEY||process.env.GOOGLE_API_KEY_POOL))return u.debug("GeminiAgentStrategy: GEMINI_API_KEY or GOOGLE_API_KEY not set"),!1;try{return bt("gemini --version",{encoding:"utf-8",timeout:5e3,stdio:"pipe"}),!0}catch{return u.warn("[Gemini] gemini CLI not found. Install: npm install -g @google/gemini-cli"),!1}}async invoke(t,e={}){let{model:r,workspace:s=process.cwd(),schema:o=null,skills:n=null,sessionPath:i=null,nodeName:c=null,timeout:a=600*1e3,signal:p=null}=e,d=[W("GEMINI_API_KEY"),W("GOOGLE_API_KEY")];try{let m=r;(!m||m==="auto")&&(m=tt.GEMINI);let h=et[m]||m,v=String(process.env.GEMINI_API_KEY||"").trim(),J=String(process.env.GOOGLE_API_KEY||"").trim(),R=this._resolveSkillsToMcp(n,{sessionPath:i,workspace:s,nodeName:c}),ut=Object.keys(R).length>0,x=new C(t),j=nt();j&&x.addSystemInstruction(j,"repo_context","prepend");let U=o&&typeof o.parse=="function",T=null;if(o){let f;try{let E=U?Tt(o,{target:"openAi"}):o;f=JSON.stringify(E,null,2)}catch{f="{}"}if(ut){x.addSystemInstruction(`Write valid JSON that matches this schema:
36
- ${f}`,"schema_instruction","append");let E=`zibby-result-${Date.now()}.json`,_=b(s,".zibby","tmp");T=b(_,E),ot(_,{recursive:!0}),x.addStructuredOutput(o,T)}else x.addSystemInstruction(`Return ONLY valid JSON (no markdown, no commentary) that matches this schema:
37
- ${f}`,"json_instruction","append")}let at=x.build(),pt=x.getUserPrompt(),$=x.getStats(),q=String(process.env.GEMINI_API_KEY||process.env.GOOGLE_API_KEY||"").trim(),ft=q?` | key: ***${q.slice(-4)}`:" | key: not set";console.log(`
38
- \u25C6 Model: ${h||"auto"}${ft}
39
- `);let k=(await import("chalk")).default;console.log(`
40
- ${k.bold("Prompt sent to LLM:")}`),console.log(k.dim("\u2500".repeat(60))),console.log(k.dim(pt)),console.log(k.dim("\u2500".repeat(60)));let K=this._createGeminiConfigDir(s,R),I=["--output-format","json"];h&&h!=="auto"&&I.push("--model",h);let F=Object.keys(R);if(F.length>0){I.push("--approval-mode","yolo");for(let f of F)I.push("--allowed-mcp-server-names",f);u.info(`[Gemini] Enabling MCP servers: ${F.join(", ")}`)}else n&&n.length>0&&u.warn(`[Gemini] Skills requested but no MCP servers configured: ${n.join(", ")}`);I.push("-p",at);let A={...process.env,GEMINI_CLI_HOME:K};v?(A.GEMINI_API_KEY=v,delete A.GOOGLE_API_KEY):J&&(A.GOOGLE_API_KEY=J,delete A.GEMINI_API_KEY),u.debug(`[Gemini] Command: gemini ${I.slice(0,8).join(" ")}... (${I.length} total args)`),u.debug(`[Gemini] User prompt: ${$.userPromptLength} chars, System instructions: ${$.systemInstructionsLength} chars (${$.instructionCount} blocks), Full: ${$.fullPromptLength} chars`),u.debug(`[Gemini] Config home: ${K}`),u.debug(`[Gemini] GEMINI_CLI_HOME env: ${A.GEMINI_CLI_HOME}`);let Y="",P=null;try{Y=await new Promise((E,_)=>{let O=vt("gemini",I,{cwd:s,env:A,stdio:["ignore","pipe","pipe"],...p&&{signal:p}}),z="",X="",B=!1,N=null,H=!1,Z=setTimeout(()=>{try{O.kill("SIGTERM")}catch{}},a),L=()=>{H||B||(H=!0,N=setTimeout(()=>{if(!B)try{O.kill("SIGKILL")}catch{}},5e3))};p&&(p.aborted?L():p.addEventListener("abort",L,{once:!0})),O.stdout.on("data",S=>{z+=S.toString()}),O.stderr.on("data",S=>{X+=S.toString()}),O.on("error",S=>{if(clearTimeout(Z),N&&clearTimeout(N),p&&!p.aborted)try{p.removeEventListener("abort",L)}catch{}_(S)}),O.on("close",S=>{if(B=!0,clearTimeout(Z),N&&clearTimeout(N),p&&!p.aborted)try{p.removeEventListener("abort",L)}catch{}if(H){let Q=new Error("Aborted via signal");return Q.name="AbortError",_(Q)}if(S===0)return E(z.trim());_(new Error(`gemini failed with code ${S}: ${(X||z).trim()}`))})})}catch(f){P=f}finally{try{At(K,{recursive:!0,force:!0})}catch{}}let y=Gt(Y).trim();if(!o){if(P)throw P;return y}if(T){let f=st(T);if(u.info(`[Gemini] Result file: ${f?"present":"missing"} at ${T}`),f)try{let E=it(T,"utf-8").trim(),_=JSON.parse(E),O=U?o.parse(_):_;return u.info("[Gemini] Structured output recovered from result file"),{raw:y,structured:O}}catch(E){u.warn(`[Gemini] Result file parse/validation failed: ${E.message}`)}else P||u.warn("[Gemini] Result file missing; falling back to stream-parsed JSON")}let w=null;if(o){let f=new D;f.zodSchema=o,f.processChunk(y),f.flush(),w=f.getResult()}if(u.info(`[Gemini] Raw stdout length: ${Y.length} chars`),u.info(`[Gemini] Extracted text length: ${y.length} chars`),u.info(`[Gemini] StreamParser result: ${w?"extracted":"null"}`),w||(y.length<2e3?u.info(`[Gemini] Raw text preview:
29
+ `),position:"append"}),this}addSystemInstruction(t,e="custom",s="append"){return t&&this.systemInstructions.push({type:e,content:t,position:s}),this}getUserPrompt(){return this.userPrompt}getSystemInstructions(){return this.systemInstructions}getMetadata(){return this.metadata}build(){let t=this.systemInstructions.filter(n=>n.position==="prepend").map(n=>n.content),e=this.systemInstructions.filter(n=>n.position==="append").map(n=>n.content);return[...t,this.userPrompt,...e].filter(Boolean).join(`
30
+
31
+ `)}getStats(){let t=this.userPrompt.length,e=this.build().length;return{userPromptLength:t,fullPromptLength:e,systemInstructionsLength:e-t,instructionCount:this.systemInstructions.length,instructionTypes:this.systemInstructions.map(s=>s.type)}}clone(){let t=new c(this.userPrompt);return t.systemInstructions=[...this.systemInstructions],t.metadata={...this.metadata},t}};var J=class c{constructor(){this.buffer="",this.extractedResult=null,this.rawText="",this.zodSchema=null,this.lastOutputLength=0,this.onToolCall=null,this._lastToolEmit=null}processChunk(t){if(!t)return null;this.buffer+=t;let e=this.buffer.split(`
32
+ `);this.buffer=e.pop()||"";let s="";for(let n of e)if(n.trim())try{let i=JSON.parse(n);this._emitToolCalls(i);let r=this.extractText(i);if(r){if(this.rawText&&r.startsWith(this.rawText)){let o=r.substring(this.rawText.length);this.rawText=r,s+=o}else(!this.rawText.includes(r)||r.length<20)&&(this.rawText+=r,s+=r);this.tryExtractResult(this.rawText)}else this.isValidResult(i)&&(this.rawText+=`${n}
33
+ `,s+=`${n}
34
+ `,this.extractedResult=i)}catch{if(n.includes('"text"')||n.includes('"content"')){let r=n.match(/"text"\s*:\s*"([^"]*)/),o=n.match(/"content"\s*:\s*"([^"]*)/),l=r?r[1]:o?o[1]:null;l&&!this.rawText.includes(l)&&(s+=l,this.rawText+=l)}}return s||null}flush(){if(!this.buffer.trim())return null;let t="";try{let e=JSON.parse(this.buffer);this._emitToolCalls(e);let s=this.extractText(e);s&&(this.rawText+=s,t+=s,this.tryExtractResult(s))}catch{this.rawText+=this.buffer,t+=this.buffer,this.tryExtractResult(this.buffer)}return this.buffer="",t||null}_emitToolCalls(t){if(!this.onToolCall)return;let e=(r,o)=>{if(!r)return;let l=`${r}:${JSON.stringify(o??{})}`;this._lastToolEmit!==l&&(this._lastToolEmit=l,this.onToolCall(r,o??void 0))},s=r=>{if(r!=null){if(typeof r=="object"&&!Array.isArray(r))return r;if(typeof r=="string")try{return JSON.parse(r)}catch{return}}};if(t.type==="tool_use"||t.type==="tool_call"){if(t.name){e(t.name,s(t.input??t.arguments));return}let r=t.tool_call;if(r&&typeof r=="object"&&!Array.isArray(r)){let o=Object.keys(r);if(o.length===1){let l=o[0],a=r[l],f=a&&typeof a=="object"?a.args??a.input??a:void 0;e(l,s(f))}return}return}if(Array.isArray(t.tool_calls)){for(let r of t.tool_calls)e(r.name,s(r.input??r.arguments));return}let n=t.message??t;if(Array.isArray(n?.tool_calls)){for(let r of n.tool_calls)e(r.name,s(r.input??r.arguments));return}let i=n?.content??t.content;if(Array.isArray(i))for(let r of i)(r.type==="tool_use"||r.type==="tool_call")&&r.name&&e(r.name,s(r.input??r.arguments))}extractText(t){if(t.type==="assistant"&&t.message?.content){let e=t.message.content;if(Array.isArray(e))return e.filter(s=>s.type==="text"&&s.text).map(s=>s.text).join("")}return t.type==="thinking"&&t.text||t.text?t.text:t.content&&typeof t.content=="string"?t.content:t.delta?t.delta:null}tryExtractResult(t){if(!t||typeof t!="string")return;let e=[],s=/```json\s*\n?([\s\S]*?)\n?```/g,n;for(;(n=s.exec(t))!==null;){let d=n[1].trim();try{JSON.parse(d),e.push({text:d,source:"markdown"})}catch{}}let i=0,r=0;for(;i<t.length&&(i=t.indexOf("{",i),i!==-1);){let d=0,m=i;for(let h=i;h<t.length;h++)if(t[h]==="{")d++;else if(t[h]==="}"&&(d--,d===0)){m=h,e.push({text:t.substring(i,m+1),source:"brace"}),r++;break}i=m+1}let o=this.extractedResult,l=o?JSON.stringify(o).length:0,a=0,f=-1;for(let d=0;d<e.length;d++){let m=e[d];try{let h=m.text.replace(/,(\s*[}\]])/g,"$1"),T=JSON.parse(h);this.isValidResult(T)&&(a++,l=JSON.stringify(T).length,o=T,f=d)}catch{}}o&&(this.extractedResult=o)}isValidResult(t){if(!t||typeof t!="object"||Array.isArray(t)||t.session_id||t.timestamp_ms||t.type||t.call_id||t.tool_call||t.result&&typeof t.result=="object"&&(t.result.success&&typeof t.result.success=="object"||t.result.error&&typeof t.result.error=="object")||t.args&&typeof t.args=="object")return!1;if(this.zodSchema)try{return this.zodSchema.parse(t),!0}catch{return!1}return!0}getResult(){return this.extractedResult}getRawText(){return this.rawText}static extractResult(t,e=null){let s=new c;s.zodSchema=e,s.processChunk(t),s.flush();let n=s.getResult();return!n&&process.env.LOG_LEVEL==="debug"&&console.error("[StreamingParser] No result extracted from",t?.length||0,"chars"),n}};function H(c){let t=`${c}_POOL`,e=process.env[t];if(!e||typeof e!="string")return{picked:null,count:0,dispose:()=>{}};let s=e.split(/[,\n]+/).map(r=>r.trim()).filter(Boolean);if(s.length===0)return{picked:null,count:0,dispose:()=>{}};let n=s[Math.floor(Math.random()*s.length)],i=process.env[c];return process.env[c]=n,u.debug(`[auth-pool] ${c}: picked 1 of ${s.length} from pool (***${n.slice(-4)})`),{picked:n,count:s.length,dispose(){i===void 0?delete process.env[c]:process.env[c]=i}}}function Et(c){if(!c)return null;let t=String(c),e=t.match(/```(?:json)?\s*([\s\S]*?)```/i);if(e?.[1])try{return JSON.parse(e[1].trim())}catch{}let s=t.indexOf("{");if(s<0)return null;let n=0,i=!1,r=!1,o=-1;for(let l=s;l<t.length;l++){let a=t[l];if(i){r?r=!1:a==="\\"?r=!0:a==='"'&&(i=!1);continue}if(a==='"'){i=!0;continue}if(a==="{"){n===0&&(o=l),n+=1;continue}if(a==="}"){if(n===0)continue;if(n-=1,n===0&&o>=0){let f=t.slice(o,l+1);try{return JSON.parse(f)}catch{o=-1}}}}return null}function _t(c){let t=String(c||"").trim();if(!t)return null;try{return JSON.parse(t)}catch{return Et(t)}}function Ot(c){try{let t=JSON.parse(c);if(typeof t=="string")return t;if(typeof t?.response=="string")return t.response;if(typeof t?.text=="string")return t.text;if(typeof t?.output=="string")return t.output;if(Array.isArray(t?.candidates)&&t.candidates.length>0){let e=t.candidates[0];if(typeof e?.content=="string")return e.content;if(Array.isArray(e?.content?.parts)){let s=e.content.parts.map(n=>typeof n?.text=="string"?n.text:"").join("");if(s.trim())return s}}}catch{}return c}var it=class extends pt{constructor(){super("gemini","Gemini (Google)",70)}canHandle(t){if(!!!(process.env.GEMINI_API_KEY||process.env.GEMINI_API_KEY_POOL||process.env.GOOGLE_API_KEY||process.env.GOOGLE_API_KEY_POOL))return u.debug("GeminiAgentStrategy: GEMINI_API_KEY or GOOGLE_API_KEY not set"),!1;try{return dt("gemini --version",{encoding:"utf-8",timeout:5e3,stdio:"pipe"}),!0}catch{return u.warn("[Gemini] gemini CLI not found. Install: npm install -g @google/gemini-cli"),!1}}async invoke(t,e={}){let{model:s,workspace:n=process.cwd(),schema:i=null,skills:r=null,sessionPath:o=null,nodeName:l=null,timeout:a=600*1e3,signal:f=null}=e,d=[H("GEMINI_API_KEY"),H("GOOGLE_API_KEY")];try{let m=s;(!m||m==="auto")&&(m=Q.GEMINI);let h=tt[m]||m,T=String(process.env.GEMINI_API_KEY||"").trim(),D=String(process.env.GOOGLE_API_KEY||"").trim(),R=this._resolveSkillsToMcp(r,{sessionPath:o,workspace:n,nodeName:l}),ot=Object.keys(R).length>0,v=new C(t),U=i&&typeof i.parse=="function",b=null;if(i){let p;try{let E=U?gt(i,{target:"openAi"}):i;p=JSON.stringify(E,null,2)}catch{p="{}"}if(ot){v.addSystemInstruction(`Write valid JSON that matches this schema:
35
+ ${p}`,"schema_instruction","append");let E=`zibby-result-${Date.now()}.json`,_=w(n,".zibby","tmp");b=w(_,E),rt(_,{recursive:!0}),v.addStructuredOutput(i,b)}else v.addSystemInstruction(`Return ONLY valid JSON (no markdown, no commentary) that matches this schema:
36
+ ${p}`,"json_instruction","append")}let lt=v.build(),ut=v.getUserPrompt(),M=v.getStats(),q=String(process.env.GEMINI_API_KEY||process.env.GOOGLE_API_KEY||"").trim(),ct=q?` | key: ***${q.slice(-4)}`:" | key: not set";console.log(`
37
+ \u25C6 Model: ${h||"auto"}${ct}
38
+ `);let L=(await import("chalk")).default;console.log(`
39
+ ${L.bold("Prompt sent to LLM:")}`),console.log(L.dim("\u2500".repeat(60))),console.log(L.dim(ut)),console.log(L.dim("\u2500".repeat(60)));let K=this._createGeminiConfigDir(n,R),x=["--output-format","json"];h&&h!=="auto"&&x.push("--model",h);let Y=Object.keys(R);if(Y.length>0){x.push("--approval-mode","yolo");for(let p of Y)x.push("--allowed-mcp-server-names",p);u.info(`[Gemini] Enabling MCP servers: ${Y.join(", ")}`)}else r&&r.length>0&&u.warn(`[Gemini] Skills requested but no MCP servers configured: ${r.join(", ")}`);x.push("-p",lt);let A={...process.env,GEMINI_CLI_HOME:K};T?(A.GEMINI_API_KEY=T,delete A.GOOGLE_API_KEY):D&&(A.GOOGLE_API_KEY=D,delete A.GEMINI_API_KEY),u.debug(`[Gemini] Command: gemini ${x.slice(0,8).join(" ")}... (${x.length} total args)`),u.debug(`[Gemini] User prompt: ${M.userPromptLength} chars, System instructions: ${M.systemInstructionsLength} chars (${M.instructionCount} blocks), Full: ${M.fullPromptLength} chars`),u.debug(`[Gemini] Config home: ${K}`),u.debug(`[Gemini] GEMINI_CLI_HOME env: ${A.GEMINI_CLI_HOME}`);let F="",N=null;try{F=await new Promise((E,_)=>{let O=ht("gemini",x,{cwd:n,env:A,stdio:["ignore","pipe","pipe"],...f&&{signal:f}}),z="",X="",B=!1,G=null,V=!1,Z=setTimeout(()=>{try{O.kill("SIGTERM")}catch{}},a),k=()=>{V||B||(V=!0,G=setTimeout(()=>{if(!B)try{O.kill("SIGKILL")}catch{}},5e3))};f&&(f.aborted?k():f.addEventListener("abort",k,{once:!0})),O.stdout.on("data",S=>{z+=S.toString()}),O.stderr.on("data",S=>{X+=S.toString()}),O.on("error",S=>{if(clearTimeout(Z),G&&clearTimeout(G),f&&!f.aborted)try{f.removeEventListener("abort",k)}catch{}_(S)}),O.on("close",S=>{if(B=!0,clearTimeout(Z),G&&clearTimeout(G),f&&!f.aborted)try{f.removeEventListener("abort",k)}catch{}if(V){let j=new Error("Aborted via signal");return j.name="AbortError",_(j)}if(S===0)return E(z.trim());_(new Error(`gemini failed with code ${S}: ${(X||z).trim()}`))})})}catch(p){N=p}finally{try{yt(K,{recursive:!0,force:!0})}catch{}}let y=Ot(F).trim();if(!i){if(N)throw N;return y}if(b){let p=et(b);if(u.info(`[Gemini] Result file: ${p?"present":"missing"} at ${b}`),p)try{let E=st(b,"utf-8").trim(),_=JSON.parse(E),O=U?i.parse(_):_;return u.info("[Gemini] Structured output recovered from result file"),{raw:y,structured:O}}catch(E){u.warn(`[Gemini] Result file parse/validation failed: ${E.message}`)}else N||u.warn("[Gemini] Result file missing; falling back to stream-parsed JSON")}let I=null;if(i){let p=new J;p.zodSchema=i,p.processChunk(y),p.flush(),I=p.getResult()}if(u.info(`[Gemini] Raw stdout length: ${F.length} chars`),u.info(`[Gemini] Extracted text length: ${y.length} chars`),u.info(`[Gemini] StreamParser result: ${I?"extracted":"null"}`),I||(y.length<2e3?u.info(`[Gemini] Raw text preview:
41
40
  ${y}`):u.info(`[Gemini] Raw text preview (first 1000 chars):
42
- ${y.slice(0,1e3)}`),w=Nt(y)),!w)throw P||(u.error("[Gemini] Failed to extract valid JSON from output"),u.error("\u{1F4A1} Tip: Set strictMode=true in .zibby.config.js for OpenAI proxy fallback"),new Error("Gemini did not return valid JSON for structured output. Enable strictMode for proxy fallback."));let mt=U?o.parse(w):w;return{raw:y,structured:mt}}finally{for(let m of d)m.dispose()}}_resolveSkillsToMcp(t,e={}){if(!Array.isArray(t)||t.length===0)return{};let r={};for(let s of t){let o=wt(s);if(!o||typeof o.resolve!="function")continue;let n=o.resolve(e);if(!n)continue;let i=o.cursorKey||o.serverName||s,c={command:n.command};n.args?.length&&(c.args=n.args),n.env&&Object.keys(n.env).length>0&&(c.env=n.env),n.cwd&&(c.cwd=n.cwd),r[i]=c}return r}_createGeminiConfigDir(t,e){let r=`${Date.now()}-${Math.random().toString(16).slice(2,10)}`,s=b(t||process.cwd(),".zibby","tmp",`gemini-home-${r}`),o=b(s,".gemini");ot(o,{recursive:!0});let n=b(o,"settings.json"),i={},c=b(process.env.HOME||"",".gemini","settings.json");if(st(c))try{i=JSON.parse(it(c,"utf-8"))}catch{i={}}let a={...i,mcpServers:{...i.mcpServers&&typeof i.mcpServers=="object"?i.mcpServers:{},...e||{}}};lt(n,`${JSON.stringify(a,null,2)}
43
- `,"utf-8");let p=b(t||process.cwd(),".zibby","tmp","gemini-settings-debug.json");try{lt(p,`${JSON.stringify(a,null,2)}
44
- `,"utf-8")}catch{}return u.debug(`[Gemini] Created isolated config with ${Object.keys(a.mcpServers||{}).length} MCP servers`),u.debug(`[Gemini] MCP servers: ${JSON.stringify(Object.keys(a.mcpServers||{}),null,2)}`),s}};export{ct as GeminiAgentStrategy};
41
+ ${y.slice(0,1e3)}`),I=_t(y)),!I)throw N||(u.error("[Gemini] Failed to extract valid JSON from output"),u.error("\u{1F4A1} Tip: Set strictMode=true in .zibby.config.js for OpenAI proxy fallback"),new Error("Gemini did not return valid JSON for structured output. Enable strictMode for proxy fallback."));let at=U?i.parse(I):I;return{raw:y,structured:at}}finally{for(let m of d)m.dispose()}}_resolveSkillsToMcp(t,e={}){if(!Array.isArray(t)||t.length===0)return{};let s={};for(let n of t){let i=mt(n);if(!i||typeof i.resolve!="function")continue;let r=i.resolve(e);if(!r)continue;let o=i.cursorKey||i.serverName||n,l={command:r.command};r.args?.length&&(l.args=r.args),r.env&&Object.keys(r.env).length>0&&(l.env=r.env),r.cwd&&(l.cwd=r.cwd),s[o]=l}return s}_createGeminiConfigDir(t,e){let s=`${Date.now()}-${Math.random().toString(16).slice(2,10)}`,n=w(t||process.cwd(),".zibby","tmp",`gemini-home-${s}`),i=w(n,".gemini");rt(i,{recursive:!0});let r=w(i,"settings.json"),o={},l=w(process.env.HOME||"",".gemini","settings.json");if(et(l))try{o=JSON.parse(st(l,"utf-8"))}catch{o={}}let a={...o,mcpServers:{...o.mcpServers&&typeof o.mcpServers=="object"?o.mcpServers:{},...e||{}}};nt(r,`${JSON.stringify(a,null,2)}
42
+ `,"utf-8");let f=w(t||process.cwd(),".zibby","tmp","gemini-settings-debug.json");try{nt(f,`${JSON.stringify(a,null,2)}
43
+ `,"utf-8")}catch{}return u.debug(`[Gemini] Created isolated config with ${Object.keys(a.mcpServers||{}).length} MCP servers`),u.debug(`[Gemini] MCP servers: ${JSON.stringify(Object.keys(a.mcpServers||{}),null,2)}`),n}};export{it as GeminiAgentStrategy};