@zibby/skills 0.1.27 → 0.1.28
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/dist/chat-notify.js +4 -4
- package/dist/index.js +23 -23
- package/dist/package.json +1 -1
- package/docs/apps/agent-ops.md +114 -0
- package/docs/apps/deploy.md +120 -0
- package/docs/apps/index.md +74 -0
- package/docs/apps/managing.md +121 -0
- package/docs/cli-reference.md +105 -0
- package/docs/intro.md +12 -0
- package/docs/recipes/index.md +1 -0
- package/docs/recipes/sentry-triage.md +93 -0
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -21,7 +21,7 @@ ${"#".repeat(i)} ${n.trim()}
|
|
|
21
21
|
${n}
|
|
22
22
|
`):e.push(n)}return e.join("").replace(/\n{3,}/g,`
|
|
23
23
|
|
|
24
|
-
`)}function Q(s){return String(s||"").toLowerCase().replace(/\s+/g,"").replace(/[()\-_::"'`]/g,"")}function Se(s){return Q(s).replace(/[a-z0-9]+/g,"")}function le(s,t){let e=Q(s),r=Q(t);if(!e||!r)return 0;if(e===r)return 1;if(e.length===1||r.length===1)return e===r?1:0;let n=l=>{let p=new Map;for(let d=0;d<l.length-1;d++){let m=l.slice(d,d+2);p.set(m,(p.get(m)||0)+1)}return p},i=n(e),o=n(r),c=0,a=0,u=0;for(let l of i.values())a+=l;for(let l of o.values())u+=l;for(let[l,p]of i.entries()){let d=o.get(l)||0;c+=Math.min(p,d)}return 2*c/Math.max(1,a+u)}function
|
|
24
|
+
`)}function Q(s){return String(s||"").toLowerCase().replace(/\s+/g,"").replace(/[()\-_::"'`]/g,"")}function Se(s){return Q(s).replace(/[a-z0-9]+/g,"")}function le(s,t){let e=Q(s),r=Q(t);if(!e||!r)return 0;if(e===r)return 1;if(e.length===1||r.length===1)return e===r?1:0;let n=l=>{let p=new Map;for(let d=0;d<l.length-1;d++){let m=l.slice(d,d+2);p.set(m,(p.get(m)||0)+1)}return p},i=n(e),o=n(r),c=0,a=0,u=0;for(let l of i.values())a+=l;for(let l of o.values())u+=l;for(let[l,p]of i.entries()){let d=o.get(l)||0;c+=Math.min(p,d)}return 2*c/Math.max(1,a+u)}function F(s){return String(s||"").toLowerCase().replace(/\s+/g,"").replace(/[()\-_::"'`]/g,"")}function Er(s,t=[]){let e=Array.isArray(t)?t:[];if(e.length===0)return{requested:s||null,resolved:null,strategy:"none"};let r=e.filter(c=>!c.subtask),n=r.length>0?r:e,i=F(s);if(i){let c=n.find(l=>F(l.name)===i);if(c)return{requested:s,resolved:c,strategy:"exact"};let a={task:["task","\u4EFB\u52A1","\u4E8B\u9879","to do","todo"],story:["story","\u7528\u6237\u6545\u4E8B","\u9700\u6C42"],bug:["bug","\u7F3A\u9677","\u95EE\u9898"],improvement:["improvement","\u4F18\u5316","\u6539\u8FDB"],epic:["epic","\u53F2\u8BD7"]};for(let l of Object.values(a)){if(!l.some(d=>F(d)===i))continue;let p=n.find(d=>l.some(m=>F(m)===F(d.name)));if(p)return{requested:s,resolved:p,strategy:"alias"}}let u=n.map(l=>({t:l,score:le(s,l.name)})).sort((l,p)=>p.score-l.score);if(u[0]&&u[0].score>=.5)return{requested:s,resolved:u[0].t,strategy:"fuzzy"}}let o=["task","story","bug","improvement","epic"];for(let c of o){let a=n.find(u=>F(u.name)===c);if(a)return{requested:s||null,resolved:a,strategy:"default-preferred"}}return{requested:s||null,resolved:n[0],strategy:"default-first"}}async function We(s){let t=`projectKeys=${encodeURIComponent(s)}&expand=projects.issuetypes`,e=await O(`/rest/api/3/issue/createmeta?${t}`),r=Array.isArray(e?.projects)?e.projects:[],i=r.find(c=>String(c?.key||"").toUpperCase()===String(s||"").toUpperCase())||r[0]||null;return(Array.isArray(i?.issuetypes)?i.issuetypes:[]).map(c=>({id:c.id,name:c.name,subtask:!!c.subtask,description:c.description||null}))}async function He(s,t){if(!s)throw new Error("projectKey is required");let e="sprint is not EMPTY";t==="active"?e="sprint in openSprints()":t==="closed"?e="sprint in closedSprints()":t==="future"&&(e="sprint in futureSprints()");let r=`project = ${s} AND ${e} ORDER BY updated DESC`,n=`jql=${encodeURIComponent(r)}&maxResults=100&fields=customfield_10020`,i=await O(`/rest/api/3/search/jql?${n}`),o=new Map;for(let c of i.issues||[])for(let a of c.fields?.customfield_10020||[])a&&!o.has(a.id)&&o.set(a.id,{id:a.id,name:a.name,state:a.state,boardId:a.boardId||null,startDate:a.startDate||null,endDate:a.endDate||null,goal:a.goal||null});return[...o.values()].sort((c,a)=>{let u={active:0,future:1,closed:2},l=(u[c.state]??3)-(u[a.state]??3);return l!==0?l:String(a.startDate||"").localeCompare(String(c.startDate||""))})}function Cr(s,{sprintId:t,sprintName:e,target:r}={}){let n=Array.isArray(s)?s:[];if(!n.length)return{sprint:null,selectedBy:"none"};if(t!=null&&String(t).trim()!=="")return{sprint:n.find(c=>String(c.id)===String(t))||null,selectedBy:"id"};if(e&&String(e).trim()){let o=String(e).trim(),c=n.find(u=>String(u.name||"").toLowerCase()===o.toLowerCase());if(c)return{sprint:c,selectedBy:"name-exact"};let a=n.map(u=>({s:u,score:le(o,u.name||"")})).sort((u,l)=>l.score-u.score);return a[0]&&a[0].score>=.5?{sprint:a[0].s,selectedBy:"name-fuzzy"}:{sprint:null,selectedBy:"name-none"}}let i=String(r||"current").trim().toLowerCase();return i==="active"||i==="current"||i==="latest"?{sprint:n[0],selectedBy:i}:{sprint:n[0],selectedBy:"default"}}function Lr(s,t){let e=s?.fields?.customfield_10020;return Array.isArray(e)?e.some(r=>String(r?.id)===String(t)):!1}async function Jr({issueKey:s,projectKey:t,sprintId:e,attempts:r=3,delayMs:n=450}){let i=[];for(let o=0;o<r;o++){try{let c=`project = ${t} AND key = ${s} AND sprint = ${e}`,a=`jql=${encodeURIComponent(c)}&maxResults=1&fields=key,status`,u=await O(`/rest/api/3/search/jql?${a}`);if(Number(u?.total||0)>0)return i.push({attempt:o+1,jql:!0,issueField:null}),{ok:!0,method:"jql",traces:i};let p=await O(`/rest/api/3/issue/${s}?fields=customfield_10020,status`),d=Lr(p,e);if(i.push({attempt:o+1,jql:!1,issueField:d}),d)return{ok:!0,method:"issue_field",traces:i}}catch(c){i.push({attempt:o+1,error:String(c?.message||c)})}o<r-1&&await new Promise(c=>setTimeout(c,n))}return{ok:!1,method:"none",traces:i}}async function ve({issueKey:s,projectKey:t,sprintId:e,sprintName:r,target:n}){if(!s)return{ok:!1,error:"issueKey is required"};let i=t;if(!i&&(i=(await O(`/rest/api/3/issue/${s}?fields=project`))?.fields?.project?.key||null,!i))return{ok:!1,error:`Could not resolve project for ${s}`};let o=await He(i,"active");if(!o.length)return{ok:!1,error:`No assignable active sprint found for project ${i}`};let{sprint:c,selectedBy:a}=Cr(o,{sprintId:e,sprintName:r,target:n});if(!c)return{ok:!1,error:`No matching sprint found in ${i}`,requested:{sprintId:e??null,sprintName:r??null,target:n??"current"},availableSprints:o.map(p=>({id:p.id,name:p.name,state:p.state}))};await O(`/rest/api/3/issue/${s}`,{method:"PUT",body:{fields:{customfield_10020:Number(c.id)}}});let u=await Jr({issueKey:s,projectKey:i,sprintId:c.id}),l=u.ok;return{ok:l,issueKey:s,projectKey:i,sprintId:c.id,sprintName:c.name,selectedBy:a,verifiedBy:u.method,verified:l,verificationTrace:u.traces,warning:l?null:`Sprint assignment attempted but verification did not find ${s} in sprint ${c.id}`}}async function O(s,t={}){let e=async()=>{let{token:r,cloudId:n}=await Rr("jira");if(typeof r!="string"||!r)throw new Error(`Invalid jira token type: ${typeof r}`);if(!n)throw new Error("Invalid jira cloudId: missing");let i=`https://api.atlassian.com/ex/jira/${n}${s}`,o=await fetch(i,{method:t.method||"GET",headers:{Authorization:`Bearer ${r}`,Accept:"application/json",...t.body?{"Content-Type":"application/json"}:{},...t.headers},body:t.body?JSON.stringify(t.body):void 0});if(!o.ok){let a=await o.text().catch(()=>"");throw new Error(`Jira API ${o.status}: ${a.slice(0,300)}`)}let c=await o.text().catch(()=>"");if(!c||!c.trim())return{};try{return JSON.parse(c)}catch{return{raw:c}}};try{return await e()}catch(r){let n=String(r?.message||r||"").toLowerCase();if(!(n.includes("token")||n.includes("401")||n.includes("403")||n.includes("substring")))throw r;return $r("jira"),e()}}var Ye={id:"jira",serverName:"jira",allowedTools:["mcp__jira__*"],requiresIntegration:A.JIRA,envKeys:["ATLASSIAN_ACCESS_TOKEN","ATLASSIAN_CLOUD_ID"],description:"Zibby Jira MCP Server (OAuth Bearer)",promptFragment:`## Jira (connected)
|
|
25
25
|
You have direct access to the user's Jira. Use these tools proactively:
|
|
26
26
|
|
|
27
27
|
### Issue tools
|
|
@@ -71,7 +71,7 @@ When user asks to move/transition ticket status:
|
|
|
71
71
|
3. Pick the correct transition from returned list (match by "to" status name, not guesswork), then call jira_transition_issue with transitionId.
|
|
72
72
|
4. Call jira_get_issue(issueKey) to verify final status before claiming success.
|
|
73
73
|
5. If target wording differs (e.g. \u5DF2\u7ECF\u9A8C\u6536 vs \u5DF2\u9A8C\u6536), try toStatus first; only ask user to confirm when no reasonable match exists.
|
|
74
|
-
6. IMPORTANT: When target is clear, complete transition + verification in SAME turn. Do NOT stop after listing options.`,resolve(){let s=xr();if(!s)return null;let t={};for(let e of this.envKeys)process.env[e]&&(t[e]=process.env[e]);return process.env.ATLASSIAN_INSTANCE_URL&&(t.ATLASSIAN_INSTANCE_URL=process.env.ATLASSIAN_INSTANCE_URL),{command:"node",args:[s],env:t,description:this.description}},async handleToolCall(s,t){try{switch(s){case"jira_list_projects":{let e=await O("/rest/api/3/project"),r=(Array.isArray(e)?e:[]).map(n=>({id:n.id,key:n.key,name:n.name,style:n.style}));return JSON.stringify({count:r.length,projects:r})}case"jira_list_statuses":{let{projectKey:e}=t||{};if(e){let i=await O(`/rest/api/3/project/${encodeURIComponent(e)}/statuses`),o=Array.isArray(i)?i:[],c=new Map;for(let u of o)for(let l of u.statuses||[])l?.id&&(c.has(l.id)||c.set(l.id,{id:l.id,name:l.name,category:l.statusCategory?.name||null}));let a=[...c.values()].sort((u,l)=>String(u.name).localeCompare(String(l.name)));return JSON.stringify({scope:"project",projectKey:e,count:a.length,statuses:a})}let r=await O("/rest/api/3/status"),n=(Array.isArray(r)?r:[]).map(i=>({id:i.id,name:i.name,category:i.statusCategory?.name||null})).sort((i,o)=>String(i.name).localeCompare(String(o.name)));return JSON.stringify({scope:"global",count:n.length,statuses:n})}case"jira_list_issue_types":{let{projectKey:e}=t||{};if(!e)return JSON.stringify({error:"projectKey is required"});let r=await We(e);return JSON.stringify({projectKey:e,count:r.length,issueTypes:r})}case"jira_search":{let e=t.jql||"",r=t.maxResults||20;e.replace(/\s*ORDER\s+BY\s+.*/i,"").trim()||(e=`created >= -365d ${e}`.trim());let i=`jql=${encodeURIComponent(e)}&maxResults=${r}&fields=summary,status,assignee,priority,updated,issuetype,project`,c=((await O(`/rest/api/3/search/jql?${i}`)).issues||[]).map(a=>({key:a.key,project:a.fields?.project?.key,summary:a.fields?.summary,status:a.fields?.status?.name,assignee:a.fields?.assignee?.displayName||"Unassigned",priority:a.fields?.priority?.name,type:a.fields?.issuetype?.name}));return JSON.stringify({count:c.length,issues:c})}case"jira_get_issue":{let e=t.issueKey;if(!e)return JSON.stringify({error:"issueKey is required"});let r=await O(`/rest/api/3/issue/${e}`);return JSON.stringify({key:r.key,project:r.fields?.project?.key,summary:r.fields?.summary,description:r.fields?.description,status:r.fields?.status?.name,assignee:r.fields?.assignee?.displayName||"Unassigned",priority:r.fields?.priority?.name,type:r.fields?.issuetype?.name,labels:r.fields?.labels,created:r.fields?.created,updated:r.fields?.updated})}case"jira_create_issue":{let{projectKey:e,summary:r,issueType:n,description:i,priority:o,labels:c,assigneeId:a,moveToSprint:u,moveToActiveSprint:l,sprintId:p,sprintName:d,target:m}=t;if(!e||!r)return JSON.stringify({error:"projectKey and summary are required"});let h={requested:n||null,resolved:null,strategy:"none"},_=[];try{_=await We(e),h=Er(n,_)}catch{}let g={project:{key:e},summary:r,issuetype:h?.resolved?.id?{id:h.resolved.id}:{name:n||"Task"}};i&&(g.description={type:"doc",version:1,content:[{type:"paragraph",content:[{type:"text",text:i}]}]}),o&&(g.priority={name:o}),c?.length&&(g.labels=c),a&&(g.assignee={id:a});let y=await O("/rest/api/3/issue",{method:"POST",body:{fields:g}}),k={ok:!0,key:y.key,id:y.id,self:y.self};return h?.resolved&&(k.issueType=h.resolved.name,k.issueTypeResolution=h.strategy,h.strategy!=="exact"&&h.requested&&B(h.requested)!==B(h.resolved.name)&&(k.issueTypeWarning=`Requested "${h.requested}" is not available in ${e}; used "${h.resolved.name}" instead.`)),_.length>0&&(k.availableIssueTypes=_.map(f=>f.name)),(u||l)&&(k.sprintMove=await ve({issueKey:y.key,projectKey:e,sprintId:p,sprintName:d,target:m})),JSON.stringify(k)}case"jira_list_sprints":{let{projectKey:e,state:r}=t,n=await He(e,r);return JSON.stringify({count:n.length,sprints:n})}case"jira_move_to_active_sprint":{let{issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o}=t||{},c=await ve({issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o||"current"});return JSON.stringify(c)}case"jira_move_issue_to_sprint":{let{issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o}=t||{},c=await ve({issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o});return JSON.stringify(c)}case"jira_get_sprint_issues":{let{sprintName:e,sprintId:r,projectKey:n,status:i,maxResults:o}=t;if(!e&&!r)return JSON.stringify({error:"sprintName or sprintId is required"});let c=o||50,a=r?`sprint = ${r}`:`sprint = "${e}"`,u=n?`project = ${n} AND `:"",l=i?` AND status = "${i}"`:"",p=`${u}${a}${l} ORDER BY status ASC, priority DESC`,d=`jql=${encodeURIComponent(p)}&maxResults=${c}&fields=summary,status,assignee,priority,issuetype,project`,m=await O(`/rest/api/3/search/jql?${d}`),h=(m.issues||[]).map(g=>({key:g.key,project:g.fields?.project?.key,summary:g.fields?.summary,status:g.fields?.status?.name,assignee:g.fields?.assignee?.displayName||"Unassigned",priority:g.fields?.priority?.name,type:g.fields?.issuetype?.name})),_={};for(let g of h)_[g.status]=(_[g.status]||0)+1;return JSON.stringify({count:h.length,total:m.total||h.length,statusCounts:_,issues:h})}case"jira_get_comments":{let{issueKey:e,maxResults:r}=t;if(!e)return JSON.stringify({error:"issueKey is required"});let i=await O(`/rest/api/3/issue/${e}/comment?maxResults=${r||50}&orderBy=-created`),o=(i.comments||[]).map(c=>{let a="";return c.body?.content&&(a=ce(c.body.content)),{id:c.id,author:c.author?.displayName||"Unknown",body:a,created:c.created,updated:c.updated}});return JSON.stringify({count:o.length,total:i.total||o.length,comments:o})}case"jira_add_comment":{let{issueKey:e,body:r}=t;return!e||!r?JSON.stringify({error:"issueKey and body are required"}):(await O(`/rest/api/3/issue/${e}/comment`,{method:"POST",body:{body:{type:"doc",version:1,content:[{type:"paragraph",content:[{type:"text",text:r}]}]}}}),JSON.stringify({ok:!0,issueKey:e}))}case"jira_edit_issue":{let{issueKey:e,fields:r}=t;return!e||!r?JSON.stringify({error:"issueKey and fields are required"}):(await O(`/rest/api/3/issue/${e}`,{method:"PUT",body:{fields:r}}),JSON.stringify({ok:!0,issueKey:e}))}case"jira_transition_issue":{let{issueKey:e,transitionId:r,toStatus:n,statusName:i,status:o}=t;if(!e)return JSON.stringify({error:"issueKey is required"});let c=String(n||i||o||"").trim();if(!r&&!c){let p=((await O(`/rest/api/3/issue/${e}/transitions`)).transitions||[]).map(d=>({id:d.id,name:d.name,to:d.to?.name}));return JSON.stringify({ok:!1,error:"transitionId or toStatus is required",issueKey:e,availableTransitions:p})}let a=r;if(!a){let p=(await O(`/rest/api/3/issue/${e}/transitions`)).transitions||[],d=Q(c),m=p.find(h=>Q(h?.name||"")===d||Q(h?.to?.name||"")===d);if(!m){let h=Se(c);h.length>=2&&(m=p.find(_=>{let g=Se(_?.name||""),y=Se(_?.to?.name||""),k=g.length>=2&&(g.includes(h)||h.includes(g)),f=y.length>=2&&(y.includes(h)||h.includes(y));return k||f}))}if(!m){let h=p.map(k=>{let f=le(c,k?.name||""),S=le(c,k?.to?.name||"");return{t:k,score:Math.max(f,S)}}).sort((k,f)=>f.score-k.score),_=h[0],g=h[1];_&&_.score>=.45&&(!g||_.score-g.score>=.12)&&(m=_.t)}if(!m?.id)return JSON.stringify({ok:!1,error:`No transition matches target status: "${c}"`,issueKey:e,availableTransitions:p.map(h=>({id:h.id,name:h.name,to:h.to?.name}))});a=m.id}await O(`/rest/api/3/issue/${e}/transitions`,{method:"POST",body:{transition:{id:a}}});let u=await O(`/rest/api/3/issue/${e}?fields=status`);return JSON.stringify({ok:!0,issueKey:e,transitionId:a,statusAfter:u?.fields?.status?.name||null})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},tools:[{name:"jira_list_projects",description:"List all Jira projects accessible to the user",input_schema:{type:"object",properties:{}}},{name:"jira_list_statuses",description:"List Jira statuses. Use projectKey to get statuses applicable in that project workflow.",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Optional project key (e.g. PROJ). If omitted, returns global status catalog."}}}},{name:"jira_list_issue_types",description:"List issue types allowed for issue creation in the given project.",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Project key, e.g. PROJ"}},required:["projectKey"]}},{name:"jira_search",description:"Search Jira issues using JQL",input_schema:{type:"object",properties:{jql:{type:"string",description:'JQL query string, e.g. "project = PROJ AND status = Open"'},maxResults:{type:"number",description:"Max results to return (default 20)"}},required:["jql"]}},{name:"jira_get_issue",description:"Get details of a specific Jira issue",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"}},required:["issueKey"]}},{name:"jira_create_issue",description:"Create a new Jira issue",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Project key, e.g. PROJ"},summary:{type:"string",description:"Issue title/summary"},issueType:{type:"string",description:"Issue type (default: Task). Common: Task, Bug, Story, Epic"},description:{type:"string",description:"Issue description (plain text)"},priority:{type:"string",description:"Priority name, e.g. High, Medium, Low"},labels:{type:"array",items:{type:"string"},description:"Array of label strings"},assigneeId:{type:"string",description:"Atlassian account ID to assign to"},moveToSprint:{type:"boolean",description:"If true, move created issue to a sprint and verify."},moveToActiveSprint:{type:"boolean",description:"Backward-compatible alias for moveToSprint."},sprintId:{type:"number",description:"Optional sprint id for placement."},sprintName:{type:"string",description:"Optional sprint name for placement."},target:{type:"string",description:"Placement target when sprintId/sprintName omitted: current|active|latest (default: current)."}},required:["projectKey","summary"]}},{name:"jira_list_sprints",description:"List sprints for a Jira project (returns sprint names, IDs, states, dates)",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Project key, e.g. PROJ"},state:{type:"string",description:"Filter: active, closed, future. Omit for all."}},required:["projectKey"]}},{name:"jira_get_sprint_issues",description:"Get all issues in a sprint, optionally filtered by status column name",input_schema:{type:"object",properties:{sprintName:{type:"string",description:"Sprint name (from jira_list_sprints). Use this OR sprintId."},sprintId:{type:"number",description:"Sprint ID (from jira_list_sprints). Use this OR sprintName."},projectKey:{type:"string",description:"Project key to scope the search (optional)"},status:{type:"string",description:'Filter by status name (e.g. "\u8FDB\u884C\u4E2D", "\u6D4B\u8BD5", "Done")'},maxResults:{type:"number",description:"Max issues to return (default 50)"}}}},{name:"jira_move_to_active_sprint",description:"Backward-compatible alias: move issue to sprint target and verify membership.",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},projectKey:{type:"string",description:"Optional project key. If omitted, inferred from issue."},sprintId:{type:"number",description:"Optional sprint id."},sprintName:{type:"string",description:"Optional sprint name."},target:{type:"string",description:"Target when sprintId/sprintName omitted: current|active|latest (default: current)."}},required:["issueKey"]}},{name:"jira_move_issue_to_sprint",description:"Move an issue to a sprint by id/name/target and verify membership.",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},projectKey:{type:"string",description:"Optional project key. If omitted, inferred from issue."},sprintId:{type:"number",description:"Optional sprint id."},sprintName:{type:"string",description:"Optional sprint name."},target:{type:"string",description:"Target when sprintId/sprintName omitted: current|active|latest (default: current)."}},required:["issueKey"]}},{name:"jira_get_comments",description:"Get comments on a Jira issue (newest first)",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},maxResults:{type:"number",description:"Max comments to return (default 50)"}},required:["issueKey"]}},{name:"jira_add_comment",description:"Add a comment to a Jira issue",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},body:{type:"string",description:"Comment text (plain text)"}},required:["issueKey","body"]}},{name:"jira_edit_issue",description:"Update fields on a Jira issue (summary, story points, labels, priority)",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},fields:{type:"object",description:"Object of field names to values",additionalProperties:!0}},required:["issueKey","fields"]}},{name:"jira_transition_issue",description:"Move a Jira issue to a different status. Always pass toStatus when user gave a target; only pass issueKey alone when you explicitly need to list transitions.",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},transitionId:{type:"string",description:"Transition ID to perform (optional if toStatus is provided)"},toStatus:{type:"string",description:'Target status/column name (e.g. "\u5DF2\u7ECF\u9A8C\u6536", "Done", "In Progress"). If provided, tool resolves matching transition automatically.'}},required:["issueKey"]}}]};import{resolveIntegrationToken as Ze}from"@zibby/core/backend-client.js";async function j(s,t={}){let{token:e}=await Ze("github"),r=s.startsWith("https://")?s:`https://api.github.com${s}`,n={Authorization:`Bearer ${e}`,Accept:t.accept||"application/vnd.github.v3+json","User-Agent":"Zibby-App",...t.body?{"Content-Type":"application/json"}:{}},i=await fetch(r,{method:t.method||"GET",headers:n,body:t.body?JSON.stringify(t.body):void 0});if(!i.ok){let o=await i.text().catch(()=>"");throw new Error(`GitHub API ${i.status}: ${o.slice(0,300)}`)}return t.raw?i.text():i.json()}var Ve={id:"github",serverName:"github",allowedTools:["mcp__github__*"],requiresIntegration:A.GITHUB,envKeys:["GITHUB_TOKEN"],description:"GitHub \u2014 issues, PRs, commits, code search, file reading",promptFragment:`## GitHub (connected)
|
|
74
|
+
6. IMPORTANT: When target is clear, complete transition + verification in SAME turn. Do NOT stop after listing options.`,resolve(){let s=xr();if(!s)return null;let t={};for(let e of this.envKeys)process.env[e]&&(t[e]=process.env[e]);return process.env.ATLASSIAN_INSTANCE_URL&&(t.ATLASSIAN_INSTANCE_URL=process.env.ATLASSIAN_INSTANCE_URL),{command:"node",args:[s],env:t,description:this.description}},async handleToolCall(s,t){try{switch(s){case"jira_list_projects":{let e=await O("/rest/api/3/project"),r=(Array.isArray(e)?e:[]).map(n=>({id:n.id,key:n.key,name:n.name,style:n.style}));return JSON.stringify({count:r.length,projects:r})}case"jira_list_statuses":{let{projectKey:e}=t||{};if(e){let i=await O(`/rest/api/3/project/${encodeURIComponent(e)}/statuses`),o=Array.isArray(i)?i:[],c=new Map;for(let u of o)for(let l of u.statuses||[])l?.id&&(c.has(l.id)||c.set(l.id,{id:l.id,name:l.name,category:l.statusCategory?.name||null}));let a=[...c.values()].sort((u,l)=>String(u.name).localeCompare(String(l.name)));return JSON.stringify({scope:"project",projectKey:e,count:a.length,statuses:a})}let r=await O("/rest/api/3/status"),n=(Array.isArray(r)?r:[]).map(i=>({id:i.id,name:i.name,category:i.statusCategory?.name||null})).sort((i,o)=>String(i.name).localeCompare(String(o.name)));return JSON.stringify({scope:"global",count:n.length,statuses:n})}case"jira_list_issue_types":{let{projectKey:e}=t||{};if(!e)return JSON.stringify({error:"projectKey is required"});let r=await We(e);return JSON.stringify({projectKey:e,count:r.length,issueTypes:r})}case"jira_search":{let e=t.jql||"",r=t.maxResults||20;e.replace(/\s*ORDER\s+BY\s+.*/i,"").trim()||(e=`created >= -365d ${e}`.trim());let i=`jql=${encodeURIComponent(e)}&maxResults=${r}&fields=summary,status,assignee,priority,updated,issuetype,project`,c=((await O(`/rest/api/3/search/jql?${i}`)).issues||[]).map(a=>({key:a.key,project:a.fields?.project?.key,summary:a.fields?.summary,status:a.fields?.status?.name,assignee:a.fields?.assignee?.displayName||"Unassigned",priority:a.fields?.priority?.name,type:a.fields?.issuetype?.name}));return JSON.stringify({count:c.length,issues:c})}case"jira_get_issue":{let e=t.issueKey;if(!e)return JSON.stringify({error:"issueKey is required"});let r=await O(`/rest/api/3/issue/${e}`);return JSON.stringify({key:r.key,project:r.fields?.project?.key,summary:r.fields?.summary,description:r.fields?.description,status:r.fields?.status?.name,assignee:r.fields?.assignee?.displayName||"Unassigned",priority:r.fields?.priority?.name,type:r.fields?.issuetype?.name,labels:r.fields?.labels,created:r.fields?.created,updated:r.fields?.updated})}case"jira_create_issue":{let{projectKey:e,summary:r,issueType:n,description:i,priority:o,labels:c,assigneeId:a,moveToSprint:u,moveToActiveSprint:l,sprintId:p,sprintName:d,target:m}=t;if(!e||!r)return JSON.stringify({error:"projectKey and summary are required"});let h={requested:n||null,resolved:null,strategy:"none"},_=[];try{_=await We(e),h=Er(n,_)}catch{}let g={project:{key:e},summary:r,issuetype:h?.resolved?.id?{id:h.resolved.id}:{name:n||"Task"}};i&&(g.description={type:"doc",version:1,content:[{type:"paragraph",content:[{type:"text",text:i}]}]}),o&&(g.priority={name:o}),c?.length&&(g.labels=c),a&&(g.assignee={id:a});let y=await O("/rest/api/3/issue",{method:"POST",body:{fields:g}}),k={ok:!0,key:y.key,id:y.id,self:y.self};return h?.resolved&&(k.issueType=h.resolved.name,k.issueTypeResolution=h.strategy,h.strategy!=="exact"&&h.requested&&F(h.requested)!==F(h.resolved.name)&&(k.issueTypeWarning=`Requested "${h.requested}" is not available in ${e}; used "${h.resolved.name}" instead.`)),_.length>0&&(k.availableIssueTypes=_.map(f=>f.name)),(u||l)&&(k.sprintMove=await ve({issueKey:y.key,projectKey:e,sprintId:p,sprintName:d,target:m})),JSON.stringify(k)}case"jira_list_sprints":{let{projectKey:e,state:r}=t,n=await He(e,r);return JSON.stringify({count:n.length,sprints:n})}case"jira_move_to_active_sprint":{let{issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o}=t||{},c=await ve({issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o||"current"});return JSON.stringify(c)}case"jira_move_issue_to_sprint":{let{issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o}=t||{},c=await ve({issueKey:e,projectKey:r,sprintId:n,sprintName:i,target:o});return JSON.stringify(c)}case"jira_get_sprint_issues":{let{sprintName:e,sprintId:r,projectKey:n,status:i,maxResults:o}=t;if(!e&&!r)return JSON.stringify({error:"sprintName or sprintId is required"});let c=o||50,a=r?`sprint = ${r}`:`sprint = "${e}"`,u=n?`project = ${n} AND `:"",l=i?` AND status = "${i}"`:"",p=`${u}${a}${l} ORDER BY status ASC, priority DESC`,d=`jql=${encodeURIComponent(p)}&maxResults=${c}&fields=summary,status,assignee,priority,issuetype,project`,m=await O(`/rest/api/3/search/jql?${d}`),h=(m.issues||[]).map(g=>({key:g.key,project:g.fields?.project?.key,summary:g.fields?.summary,status:g.fields?.status?.name,assignee:g.fields?.assignee?.displayName||"Unassigned",priority:g.fields?.priority?.name,type:g.fields?.issuetype?.name})),_={};for(let g of h)_[g.status]=(_[g.status]||0)+1;return JSON.stringify({count:h.length,total:m.total||h.length,statusCounts:_,issues:h})}case"jira_get_comments":{let{issueKey:e,maxResults:r}=t;if(!e)return JSON.stringify({error:"issueKey is required"});let i=await O(`/rest/api/3/issue/${e}/comment?maxResults=${r||50}&orderBy=-created`),o=(i.comments||[]).map(c=>{let a="";return c.body?.content&&(a=ce(c.body.content)),{id:c.id,author:c.author?.displayName||"Unknown",body:a,created:c.created,updated:c.updated}});return JSON.stringify({count:o.length,total:i.total||o.length,comments:o})}case"jira_add_comment":{let{issueKey:e,body:r}=t;return!e||!r?JSON.stringify({error:"issueKey and body are required"}):(await O(`/rest/api/3/issue/${e}/comment`,{method:"POST",body:{body:{type:"doc",version:1,content:[{type:"paragraph",content:[{type:"text",text:r}]}]}}}),JSON.stringify({ok:!0,issueKey:e}))}case"jira_edit_issue":{let{issueKey:e,fields:r}=t;return!e||!r?JSON.stringify({error:"issueKey and fields are required"}):(await O(`/rest/api/3/issue/${e}`,{method:"PUT",body:{fields:r}}),JSON.stringify({ok:!0,issueKey:e}))}case"jira_transition_issue":{let{issueKey:e,transitionId:r,toStatus:n,statusName:i,status:o}=t;if(!e)return JSON.stringify({error:"issueKey is required"});let c=String(n||i||o||"").trim();if(!r&&!c){let p=((await O(`/rest/api/3/issue/${e}/transitions`)).transitions||[]).map(d=>({id:d.id,name:d.name,to:d.to?.name}));return JSON.stringify({ok:!1,error:"transitionId or toStatus is required",issueKey:e,availableTransitions:p})}let a=r;if(!a){let p=(await O(`/rest/api/3/issue/${e}/transitions`)).transitions||[],d=Q(c),m=p.find(h=>Q(h?.name||"")===d||Q(h?.to?.name||"")===d);if(!m){let h=Se(c);h.length>=2&&(m=p.find(_=>{let g=Se(_?.name||""),y=Se(_?.to?.name||""),k=g.length>=2&&(g.includes(h)||h.includes(g)),f=y.length>=2&&(y.includes(h)||h.includes(y));return k||f}))}if(!m){let h=p.map(k=>{let f=le(c,k?.name||""),S=le(c,k?.to?.name||"");return{t:k,score:Math.max(f,S)}}).sort((k,f)=>f.score-k.score),_=h[0],g=h[1];_&&_.score>=.45&&(!g||_.score-g.score>=.12)&&(m=_.t)}if(!m?.id)return JSON.stringify({ok:!1,error:`No transition matches target status: "${c}"`,issueKey:e,availableTransitions:p.map(h=>({id:h.id,name:h.name,to:h.to?.name}))});a=m.id}await O(`/rest/api/3/issue/${e}/transitions`,{method:"POST",body:{transition:{id:a}}});let u=await O(`/rest/api/3/issue/${e}?fields=status`);return JSON.stringify({ok:!0,issueKey:e,transitionId:a,statusAfter:u?.fields?.status?.name||null})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},tools:[{name:"jira_list_projects",description:"List all Jira projects accessible to the user",input_schema:{type:"object",properties:{}}},{name:"jira_list_statuses",description:"List Jira statuses. Use projectKey to get statuses applicable in that project workflow.",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Optional project key (e.g. PROJ). If omitted, returns global status catalog."}}}},{name:"jira_list_issue_types",description:"List issue types allowed for issue creation in the given project.",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Project key, e.g. PROJ"}},required:["projectKey"]}},{name:"jira_search",description:"Search Jira issues using JQL",input_schema:{type:"object",properties:{jql:{type:"string",description:'JQL query string, e.g. "project = PROJ AND status = Open"'},maxResults:{type:"number",description:"Max results to return (default 20)"}},required:["jql"]}},{name:"jira_get_issue",description:"Get details of a specific Jira issue",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"}},required:["issueKey"]}},{name:"jira_create_issue",description:"Create a new Jira issue",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Project key, e.g. PROJ"},summary:{type:"string",description:"Issue title/summary"},issueType:{type:"string",description:"Issue type (default: Task). Common: Task, Bug, Story, Epic"},description:{type:"string",description:"Issue description (plain text)"},priority:{type:"string",description:"Priority name, e.g. High, Medium, Low"},labels:{type:"array",items:{type:"string"},description:"Array of label strings"},assigneeId:{type:"string",description:"Atlassian account ID to assign to"},moveToSprint:{type:"boolean",description:"If true, move created issue to a sprint and verify."},moveToActiveSprint:{type:"boolean",description:"Backward-compatible alias for moveToSprint."},sprintId:{type:"number",description:"Optional sprint id for placement."},sprintName:{type:"string",description:"Optional sprint name for placement."},target:{type:"string",description:"Placement target when sprintId/sprintName omitted: current|active|latest (default: current)."}},required:["projectKey","summary"]}},{name:"jira_list_sprints",description:"List sprints for a Jira project (returns sprint names, IDs, states, dates)",input_schema:{type:"object",properties:{projectKey:{type:"string",description:"Project key, e.g. PROJ"},state:{type:"string",description:"Filter: active, closed, future. Omit for all."}},required:["projectKey"]}},{name:"jira_get_sprint_issues",description:"Get all issues in a sprint, optionally filtered by status column name",input_schema:{type:"object",properties:{sprintName:{type:"string",description:"Sprint name (from jira_list_sprints). Use this OR sprintId."},sprintId:{type:"number",description:"Sprint ID (from jira_list_sprints). Use this OR sprintName."},projectKey:{type:"string",description:"Project key to scope the search (optional)"},status:{type:"string",description:'Filter by status name (e.g. "\u8FDB\u884C\u4E2D", "\u6D4B\u8BD5", "Done")'},maxResults:{type:"number",description:"Max issues to return (default 50)"}}}},{name:"jira_move_to_active_sprint",description:"Backward-compatible alias: move issue to sprint target and verify membership.",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},projectKey:{type:"string",description:"Optional project key. If omitted, inferred from issue."},sprintId:{type:"number",description:"Optional sprint id."},sprintName:{type:"string",description:"Optional sprint name."},target:{type:"string",description:"Target when sprintId/sprintName omitted: current|active|latest (default: current)."}},required:["issueKey"]}},{name:"jira_move_issue_to_sprint",description:"Move an issue to a sprint by id/name/target and verify membership.",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},projectKey:{type:"string",description:"Optional project key. If omitted, inferred from issue."},sprintId:{type:"number",description:"Optional sprint id."},sprintName:{type:"string",description:"Optional sprint name."},target:{type:"string",description:"Target when sprintId/sprintName omitted: current|active|latest (default: current)."}},required:["issueKey"]}},{name:"jira_get_comments",description:"Get comments on a Jira issue (newest first)",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},maxResults:{type:"number",description:"Max comments to return (default 50)"}},required:["issueKey"]}},{name:"jira_add_comment",description:"Add a comment to a Jira issue",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},body:{type:"string",description:"Comment text (plain text)"}},required:["issueKey","body"]}},{name:"jira_edit_issue",description:"Update fields on a Jira issue (summary, story points, labels, priority)",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},fields:{type:"object",description:"Object of field names to values",additionalProperties:!0}},required:["issueKey","fields"]}},{name:"jira_transition_issue",description:"Move a Jira issue to a different status. Always pass toStatus when user gave a target; only pass issueKey alone when you explicitly need to list transitions.",input_schema:{type:"object",properties:{issueKey:{type:"string",description:"Issue key, e.g. PROJ-123"},transitionId:{type:"string",description:"Transition ID to perform (optional if toStatus is provided)"},toStatus:{type:"string",description:'Target status/column name (e.g. "\u5DF2\u7ECF\u9A8C\u6536", "Done", "In Progress"). If provided, tool resolves matching transition automatically.'}},required:["issueKey"]}}]};import{resolveIntegrationToken as Ve}from"@zibby/core/backend-client.js";async function j(s,t={}){let{token:e}=await Ve("github"),r=s.startsWith("https://")?s:`https://api.github.com${s}`,n={Authorization:`Bearer ${e}`,Accept:t.accept||"application/vnd.github.v3+json","User-Agent":"Zibby-App",...t.body?{"Content-Type":"application/json"}:{}},i=await fetch(r,{method:t.method||"GET",headers:n,body:t.body?JSON.stringify(t.body):void 0});if(!i.ok){let o=await i.text().catch(()=>"");throw new Error(`GitHub API ${i.status}: ${o.slice(0,300)}`)}return t.raw?i.text():i.json()}var Ze={id:"github",serverName:"github",allowedTools:["mcp__github__*"],requiresIntegration:A.GITHUB,envKeys:["GITHUB_TOKEN"],description:"GitHub \u2014 issues, PRs, commits, code search, file reading",promptFragment:`## GitHub (connected)
|
|
75
75
|
You have access to the user's GitHub repositories. Available tools:
|
|
76
76
|
|
|
77
77
|
### Discovery
|
|
@@ -102,26 +102,26 @@ When user says "check out repo-name" or "clone repo-name":
|
|
|
102
102
|
3. STOP. Do not offer to inspect files or ask what to do next.
|
|
103
103
|
|
|
104
104
|
When user just wants to "look at" or "read" files (not clone):
|
|
105
|
-
- Use github_get_file to read individual files via API`,resolve(){let s={};for(let t of this.envKeys)process.env[t]&&(s[t]=process.env[t]);return{command:"npx",args:["-y","@modelcontextprotocol/server-github@latest"],env:s}},async handleToolCall(s,t){try{switch(s){case"github_search_issues":{let e=t.query;if(!e)return JSON.stringify({error:"query is required"});let r=await j(`/search/issues?q=${encodeURIComponent(e)}&per_page=${t.limit||20}`),n=(r.items||[]).map(i=>({number:i.number,title:i.title,state:i.state,repo:i.repository_url?.split("/").slice(-2).join("/"),url:i.html_url,user:i.user?.login,isPR:!!i.pull_request,labels:(i.labels||[]).map(o=>o.name),createdAt:i.created_at}));return JSON.stringify({total:r.total_count,items:n})}case"github_search_code":{let e=t.query;if(!e)return JSON.stringify({error:"query is required"});let r=t.repo?`+repo:${t.repo}`:"",n=t.language?`+language:${t.language}`:"",i=await j(`/search/code?q=${encodeURIComponent(e)}${r}${n}&per_page=${t.limit||15}`),o=(i.items||[]).map(c=>({name:c.name,path:c.path,repo:c.repository?.full_name,url:c.html_url,score:c.score}));return JSON.stringify({total:i.total_count,items:o})}case"github_get_pr":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}`);return JSON.stringify({number:i.number,title:i.title,state:i.state,merged:i.merged,body:i.body?.slice(0,5e3),user:i.user?.login,branch:i.head?.ref,base:i.base?.ref,changedFiles:i.changed_files,additions:i.additions,deletions:i.deletions,createdAt:i.created_at,mergedAt:i.merged_at,url:i.html_url,labels:(i.labels||[]).map(o=>o.name)})}case"github_get_pr_diff":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}`,{accept:"application/vnd.github.v3.diff",raw:!0}),o=i.length>15e3;return JSON.stringify({number:n,diff:o?i.slice(0,15e3):i,truncated:o,totalLength:i.length})}case"github_list_pr_files":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}/files?per_page=100`);return JSON.stringify({total:i.length,files:i.map(o=>({filename:o.filename,status:o.status,additions:o.additions,deletions:o.deletions,patch:o.patch?.slice(0,3e3)}))})}case"github_list_pr_comments":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}/comments?per_page=50`),o=await j(`/repos/${e}/${r}/issues/${n}/comments?per_page=50`),c=[...i.map(a=>({type:"review",user:a.user?.login,body:a.body?.slice(0,1e3),path:a.path,line:a.line,createdAt:a.created_at})),...o.map(a=>({type:"issue",user:a.user?.login,body:a.body?.slice(0,1e3),createdAt:a.created_at}))].sort((a,u)=>new Date(a.createdAt)-new Date(u.createdAt));return JSON.stringify({total:c.length,comments:c})}case"github_list_commits":{let{owner:e,repo:r,branch:n,path:i,limit:o}=t;if(!e||!r)return JSON.stringify({error:"owner and repo are required"});let c=`/repos/${e}/${r}/commits?per_page=${o||20}`;n&&(c+=`&sha=${encodeURIComponent(n)}`),i&&(c+=`&path=${encodeURIComponent(i)}`);let a=await j(c);return JSON.stringify({total:a.length,commits:a.map(u=>({sha:u.sha?.slice(0,8),fullSha:u.sha,message:u.commit?.message?.slice(0,300),author:u.commit?.author?.name,date:u.commit?.author?.date,url:u.html_url}))})}case"github_get_commit":{let{owner:e,repo:r,sha:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and sha are required"});let i=await j(`/repos/${e}/${r}/commits/${n}`);return JSON.stringify({sha:i.sha?.slice(0,8),message:i.commit?.message,author:i.commit?.author?.name,date:i.commit?.author?.date,stats:i.stats,files:(i.files||[]).map(o=>({filename:o.filename,status:o.status,additions:o.additions,deletions:o.deletions,patch:o.patch?.slice(0,3e3)}))})}case"github_get_file":{let{owner:e,repo:r,path:n,ref:i}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and path are required"});let o=`/repos/${e}/${r}/contents/${encodeURIComponent(n)}`;i&&(o+=`?ref=${encodeURIComponent(i)}`);let c=await j(o);if(c.type!=="file")return Array.isArray(c)?JSON.stringify({type:"directory",path:n,entries:c.map(l=>({name:l.name,type:l.type,size:l.size,path:l.path}))}):JSON.stringify({error:`Not a file: ${c.type}`});let a=Buffer.from(c.content||"","base64").toString("utf-8"),u=a.length>2e4;return JSON.stringify({path:c.path,size:c.size,sha:c.sha?.slice(0,8),content:u?a.slice(0,2e4):a,truncated:u})}case"github_get_user":try{let e=await j("/installation/repositories?per_page=1");if(e.repositories&&e.repositories.length>0){let r=e.repositories[0],n=r.owner.login,i=r.owner.type,o=i==="Organization"?`/orgs/${n}`:`/users/${n}`,c=await j(o);return JSON.stringify({login:c.login,name:c.name||c.login,avatar:c.avatar_url,bio:c.bio||c.description,type:i,isOrg:i==="Organization",publicRepos:c.public_repos,message:"Showing GitHub App installation owner (GitHub Apps cannot access /user endpoint)"})}return JSON.stringify({error:"No repositories accessible to this GitHub App installation"})}catch(e){return JSON.stringify({error:`GitHub App cannot access /user endpoint. Use github_list_repos instead. (${e.message})`})}case"github_list_orgs":try{let r=(await j("/installation/repositories?per_page=100")).repositories||[],n=new Map;for(let o of r)o.owner.type==="Organization"&&(n.has(o.owner.login)||n.set(o.owner.login,{login:o.owner.login,description:null,url:o.owner.url}));let i=Array.from(n.values());return JSON.stringify({count:i.length,orgs:i,message:"Extracted from accessible repositories (GitHub Apps cannot access /user/orgs directly)"})}catch(e){return JSON.stringify({error:`GitHub App cannot list orgs via /user/orgs. Error: ${e.message}`})}case"github_clone":{let h=function(y){let k=y.replace(/^~(?=$|\/|\\)/,m);return c(k)},{owner:e,repo:r,destination:n}=t;if(!e||!r)return JSON.stringify({error:"owner and repo are required"});let{execSync:i}=await import("child_process"),{join:o,resolve:c}=await import("path"),{existsSync:a,mkdirSync:u}=await import("fs"),{homedir:l,platform:p}=await import("os"),{token:d}=await
|
|
105
|
+
- Use github_get_file to read individual files via API`,resolve(){let s={};for(let t of this.envKeys)process.env[t]&&(s[t]=process.env[t]);return{command:"npx",args:["-y","@modelcontextprotocol/server-github@latest"],env:s}},async handleToolCall(s,t){try{switch(s){case"github_search_issues":{let e=t.query;if(!e)return JSON.stringify({error:"query is required"});let r=await j(`/search/issues?q=${encodeURIComponent(e)}&per_page=${t.limit||20}`),n=(r.items||[]).map(i=>({number:i.number,title:i.title,state:i.state,repo:i.repository_url?.split("/").slice(-2).join("/"),url:i.html_url,user:i.user?.login,isPR:!!i.pull_request,labels:(i.labels||[]).map(o=>o.name),createdAt:i.created_at}));return JSON.stringify({total:r.total_count,items:n})}case"github_search_code":{let e=t.query;if(!e)return JSON.stringify({error:"query is required"});let r=t.repo?`+repo:${t.repo}`:"",n=t.language?`+language:${t.language}`:"",i=await j(`/search/code?q=${encodeURIComponent(e)}${r}${n}&per_page=${t.limit||15}`),o=(i.items||[]).map(c=>({name:c.name,path:c.path,repo:c.repository?.full_name,url:c.html_url,score:c.score}));return JSON.stringify({total:i.total_count,items:o})}case"github_get_pr":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}`);return JSON.stringify({number:i.number,title:i.title,state:i.state,merged:i.merged,body:i.body?.slice(0,5e3),user:i.user?.login,branch:i.head?.ref,base:i.base?.ref,changedFiles:i.changed_files,additions:i.additions,deletions:i.deletions,createdAt:i.created_at,mergedAt:i.merged_at,url:i.html_url,labels:(i.labels||[]).map(o=>o.name)})}case"github_get_pr_diff":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}`,{accept:"application/vnd.github.v3.diff",raw:!0}),o=i.length>15e3;return JSON.stringify({number:n,diff:o?i.slice(0,15e3):i,truncated:o,totalLength:i.length})}case"github_list_pr_files":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}/files?per_page=100`);return JSON.stringify({total:i.length,files:i.map(o=>({filename:o.filename,status:o.status,additions:o.additions,deletions:o.deletions,patch:o.patch?.slice(0,3e3)}))})}case"github_list_pr_comments":{let{owner:e,repo:r,number:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and number are required"});let i=await j(`/repos/${e}/${r}/pulls/${n}/comments?per_page=50`),o=await j(`/repos/${e}/${r}/issues/${n}/comments?per_page=50`),c=[...i.map(a=>({type:"review",user:a.user?.login,body:a.body?.slice(0,1e3),path:a.path,line:a.line,createdAt:a.created_at})),...o.map(a=>({type:"issue",user:a.user?.login,body:a.body?.slice(0,1e3),createdAt:a.created_at}))].sort((a,u)=>new Date(a.createdAt)-new Date(u.createdAt));return JSON.stringify({total:c.length,comments:c})}case"github_list_commits":{let{owner:e,repo:r,branch:n,path:i,limit:o}=t;if(!e||!r)return JSON.stringify({error:"owner and repo are required"});let c=`/repos/${e}/${r}/commits?per_page=${o||20}`;n&&(c+=`&sha=${encodeURIComponent(n)}`),i&&(c+=`&path=${encodeURIComponent(i)}`);let a=await j(c);return JSON.stringify({total:a.length,commits:a.map(u=>({sha:u.sha?.slice(0,8),fullSha:u.sha,message:u.commit?.message?.slice(0,300),author:u.commit?.author?.name,date:u.commit?.author?.date,url:u.html_url}))})}case"github_get_commit":{let{owner:e,repo:r,sha:n}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and sha are required"});let i=await j(`/repos/${e}/${r}/commits/${n}`);return JSON.stringify({sha:i.sha?.slice(0,8),message:i.commit?.message,author:i.commit?.author?.name,date:i.commit?.author?.date,stats:i.stats,files:(i.files||[]).map(o=>({filename:o.filename,status:o.status,additions:o.additions,deletions:o.deletions,patch:o.patch?.slice(0,3e3)}))})}case"github_get_file":{let{owner:e,repo:r,path:n,ref:i}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and path are required"});let o=`/repos/${e}/${r}/contents/${encodeURIComponent(n)}`;i&&(o+=`?ref=${encodeURIComponent(i)}`);let c=await j(o);if(c.type!=="file")return Array.isArray(c)?JSON.stringify({type:"directory",path:n,entries:c.map(l=>({name:l.name,type:l.type,size:l.size,path:l.path}))}):JSON.stringify({error:`Not a file: ${c.type}`});let a=Buffer.from(c.content||"","base64").toString("utf-8"),u=a.length>2e4;return JSON.stringify({path:c.path,size:c.size,sha:c.sha?.slice(0,8),content:u?a.slice(0,2e4):a,truncated:u})}case"github_get_user":try{let e=await j("/installation/repositories?per_page=1");if(e.repositories&&e.repositories.length>0){let r=e.repositories[0],n=r.owner.login,i=r.owner.type,o=i==="Organization"?`/orgs/${n}`:`/users/${n}`,c=await j(o);return JSON.stringify({login:c.login,name:c.name||c.login,avatar:c.avatar_url,bio:c.bio||c.description,type:i,isOrg:i==="Organization",publicRepos:c.public_repos,message:"Showing GitHub App installation owner (GitHub Apps cannot access /user endpoint)"})}return JSON.stringify({error:"No repositories accessible to this GitHub App installation"})}catch(e){return JSON.stringify({error:`GitHub App cannot access /user endpoint. Use github_list_repos instead. (${e.message})`})}case"github_list_orgs":try{let r=(await j("/installation/repositories?per_page=100")).repositories||[],n=new Map;for(let o of r)o.owner.type==="Organization"&&(n.has(o.owner.login)||n.set(o.owner.login,{login:o.owner.login,description:null,url:o.owner.url}));let i=Array.from(n.values());return JSON.stringify({count:i.length,orgs:i,message:"Extracted from accessible repositories (GitHub Apps cannot access /user/orgs directly)"})}catch(e){return JSON.stringify({error:`GitHub App cannot list orgs via /user/orgs. Error: ${e.message}`})}case"github_clone":{let h=function(y){let k=y.replace(/^~(?=$|\/|\\)/,m);return c(k)},{owner:e,repo:r,destination:n}=t;if(!e||!r)return JSON.stringify({error:"owner and repo are required"});let{execSync:i}=await import("child_process"),{join:o,resolve:c}=await import("path"),{existsSync:a,mkdirSync:u}=await import("fs"),{homedir:l,platform:p}=await import("os"),{token:d}=await Ve("github"),m=l(),_=n?h(n):o(m,"zibby-repos"),g=o(_,r);if(u(_,{recursive:!0}),a(g))return JSON.stringify({error:`Directory ${g} already exists. Remove it first or use a different destination.`,existingPath:g});try{let y=`https://x-access-token:${d}@github.com/${e}/${r}.git`;i(`git clone ${y} "${g}"`,{stdio:"pipe"});let k=p()==="win32",f;return k?f=i(`dir "${g}"`,{encoding:"utf-8",shell:"cmd.exe"}):f=i(`ls -la "${g}"`,{encoding:"utf-8"}),JSON.stringify({success:!0,path:g,message:`Cloned ${e}/${r} to ${g}`,contents:f.split(`
|
|
106
106
|
`).slice(0,30).join(`
|
|
107
|
-
`),instructions:"IMPORTANT: Show the contents field to the user - it contains the directory listing."})}catch(y){return JSON.stringify({error:`Clone failed: ${y.message}`})}}case"github_search_repos":{let{query:e,limit:r}=t;if(!e)return JSON.stringify({error:"query is required"});let n=await this.handleToolCall("github_list_repos",{limit:200},{}),i=JSON.parse(n);if(i.error)return JSON.stringify(i);let o=e.toLowerCase(),c=i.repos.filter(a=>a.name.toLowerCase().includes(o)||a.fullName.toLowerCase().includes(o)||a.description&&a.description.toLowerCase().includes(o));return JSON.stringify({query:e,count:c.length,repos:c.slice(0,r||20)})}case"github_list_repos":{let{owner:e,type:r,sort:n,direction:i,limit:o}=t,c=100,a=o||200,u=[];if(!e){let h=1,_=!0;for(;_&&u.length<a;){let f=`/installation/repositories?per_page=${c}&page=${h}`,x=(await j(f)).repositories||[];if(x.length===0)break;u=u.concat(x),_=x.length===c,h++}let g=u.slice(0,a).map(f=>({name:f.name,fullName:f.full_name,private:f.private,description:f.description,language:f.language,defaultBranch:f.default_branch,updatedAt:f.updated_at,stars:f.stargazers_count,url:f.html_url})),y=g.filter(f=>f.private).length,k=g.filter(f=>!f.private).length;return JSON.stringify({count:g.length,repos:g,privateCount:y,publicCount:k,message:`Found ${y} private and ${k} public repos`})}let l=await j(`/orgs/${e}`).then(()=>!0).catch(()=>!1),p=1,d=!0;for(;d&&u.length<a;){let h;l?h=`/orgs/${e}/repos?per_page=${c}&page=${p}&type=${r||"all"}&sort=${n||"updated"}&direction=${i||"desc"}`:h=`/users/${e}/repos?per_page=${c}&page=${p}&type=${r||"all"}&sort=${n||"updated"}&direction=${i||"desc"}`;let _=await j(h),g=Array.isArray(_)?_:[];if(g.length===0)break;u=u.concat(g),d=g.length===c,p++}let m=u.slice(0,a).map(h=>({name:h.name,fullName:h.full_name,private:h.private,description:h.description,language:h.language,defaultBranch:h.default_branch,updatedAt:h.updated_at,stars:h.stargazers_count,url:h.html_url}));return JSON.stringify({count:m.length,repos:m})}case"github_create_issue":{let{owner:e,repo:r,title:n,body:i}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and title are required"});let o=await j(`/repos/${e}/${r}/issues`,{method:"POST",body:{title:n,body:i||""}});return JSON.stringify({number:o.number,url:o.html_url,title:o.title})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},tools:[{name:"github_get_user",description:"Get the authenticated GitHub user profile and their organizations",input_schema:{type:"object",properties:{}}},{name:"github_list_orgs",description:"List GitHub organizations the authenticated user belongs to",input_schema:{type:"object",properties:{}}},{name:"github_list_repos",description:"List repositories for a user or org. If no owner given, lists the authenticated user's repos.",input_schema:{type:"object",properties:{owner:{type:"string",description:"Org or user login. Omit to list your own repos."},type:{type:"string",enum:["all","public","private","forks","sources","member"],description:"Filter by type (default: all)"},sort:{type:"string",enum:["created","updated","pushed","full_name"],description:"Sort field (default: updated)"},direction:{type:"string",enum:["asc","desc"],description:"Sort direction (default: desc)"},limit:{type:"number",description:"Max repos to return (default: 30)"}}}},{name:"github_clone",description:'Clone a GitHub repository to the local filesystem. Use when user says "check out" or "clone" a repo.',input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner (user or org name)"},repo:{type:"string",description:"Repository name"},destination:{type:"string",description:"Destination directory. Accepts absolute paths, ~-prefixed paths, or relative names. Defaults to ~/zibby-repos/<repo>."}},required:["owner","repo"]}},{name:"github_search_repos",description:"Search accessible repositories by name or description. Use this when the user asks to find a specific repo.",input_schema:{type:"object",properties:{query:{type:"string",description:'Search term to match against repo name or description (e.g., "electron", "my-app")'},limit:{type:"number",description:"Max results (default: 20)"}},required:["query"]}},{name:"github_search_issues",description:"Search GitHub issues and pull requests",input_schema:{type:"object",properties:{query:{type:"string",description:'GitHub search query (e.g. "SCRUM-123", "login bug repo:org/app")'},limit:{type:"number",description:"Max results (default: 20)"}},required:["query"]}},{name:"github_search_code",description:"Search code across GitHub repositories by keyword",input_schema:{type:"object",properties:{query:{type:"string",description:'Code search query (e.g. "handleLogin", "class AuthService")'},repo:{type:"string",description:'Scope to a specific repo (e.g. "org/app"). Optional.'},language:{type:"string",description:'Filter by language (e.g. "javascript", "python"). Optional.'},limit:{type:"number",description:"Max results (default: 15)"}},required:["query"]}},{name:"github_get_pr",description:"Get details of a pull request \u2014 title, description, branch, stats",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_get_pr_diff",description:"Get the unified diff of a pull request \u2014 the actual code changes",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_list_pr_files",description:"List files changed in a PR with per-file patches",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_list_pr_comments",description:"Get all review and issue comments on a PR",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_list_commits",description:"List recent commits on a branch, optionally filtered by file path",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},branch:{type:"string",description:"Branch name (default: repo default branch)"},path:{type:"string",description:"Filter commits touching this file path"},limit:{type:"number",description:"Max commits (default: 20)"}},required:["owner","repo"]}},{name:"github_get_commit",description:"Get details of a specific commit \u2014 message, stats, file diffs",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},sha:{type:"string",description:"Commit SHA (full or short)"}},required:["owner","repo","sha"]}},{name:"github_get_file",description:"Read a file (or list a directory) from a GitHub repo. Works on any branch/ref.",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},path:{type:"string",description:'File or directory path (e.g. "src/auth/login.ts")'},ref:{type:"string",description:"Branch, tag, or commit SHA (default: repo default branch)"}},required:["owner","repo","path"]}},{name:"github_create_issue",description:"Create a GitHub issue",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},title:{type:"string",description:"Issue title"},body:{type:"string",description:"Issue body (markdown)"}},required:["owner","repo","title"]}}]};import{existsSync as Dr}from"fs";import{fileURLToPath as Ur}from"url";import{dirname as Pr,resolve as Mr}from"path";import{resolveIntegrationToken as qr}from"@zibby/core/backend-client.js";function Kr(){if(process.env.MCP_SLACK_PATH)return process.env.MCP_SLACK_PATH;let s=Pr(Ur(import.meta.url)),t=Mr(s,"..","bin","mcp-slack.mjs");return Dr(t)?t:null}async function J(s,t={}){let{token:e}=await qr("slack"),r=["conversations.list","users.list","users.profile.get","users.lookupByEmail","usergroups.list","usergroups.users.list","conversations.history","conversations.replies"].includes(s),n=`https://slack.com/api/${s}`,i={Authorization:`Bearer ${e}`},o;if(r){let u=new URLSearchParams(t).toString();u&&(n+=`?${u}`)}else i["Content-Type"]="application/json; charset=utf-8",o=JSON.stringify(t);let a=await(await fetch(n,{method:r?"GET":"POST",headers:i,body:o})).json();if(!a.ok)throw new Error(`Slack API error: ${a.error}`);return a}var
|
|
107
|
+
`),instructions:"IMPORTANT: Show the contents field to the user - it contains the directory listing."})}catch(y){return JSON.stringify({error:`Clone failed: ${y.message}`})}}case"github_search_repos":{let{query:e,limit:r}=t;if(!e)return JSON.stringify({error:"query is required"});let n=await this.handleToolCall("github_list_repos",{limit:200},{}),i=JSON.parse(n);if(i.error)return JSON.stringify(i);let o=e.toLowerCase(),c=i.repos.filter(a=>a.name.toLowerCase().includes(o)||a.fullName.toLowerCase().includes(o)||a.description&&a.description.toLowerCase().includes(o));return JSON.stringify({query:e,count:c.length,repos:c.slice(0,r||20)})}case"github_list_repos":{let{owner:e,type:r,sort:n,direction:i,limit:o}=t,c=100,a=o||200,u=[];if(!e){let h=1,_=!0;for(;_&&u.length<a;){let f=`/installation/repositories?per_page=${c}&page=${h}`,x=(await j(f)).repositories||[];if(x.length===0)break;u=u.concat(x),_=x.length===c,h++}let g=u.slice(0,a).map(f=>({name:f.name,fullName:f.full_name,private:f.private,description:f.description,language:f.language,defaultBranch:f.default_branch,updatedAt:f.updated_at,stars:f.stargazers_count,url:f.html_url})),y=g.filter(f=>f.private).length,k=g.filter(f=>!f.private).length;return JSON.stringify({count:g.length,repos:g,privateCount:y,publicCount:k,message:`Found ${y} private and ${k} public repos`})}let l=await j(`/orgs/${e}`).then(()=>!0).catch(()=>!1),p=1,d=!0;for(;d&&u.length<a;){let h;l?h=`/orgs/${e}/repos?per_page=${c}&page=${p}&type=${r||"all"}&sort=${n||"updated"}&direction=${i||"desc"}`:h=`/users/${e}/repos?per_page=${c}&page=${p}&type=${r||"all"}&sort=${n||"updated"}&direction=${i||"desc"}`;let _=await j(h),g=Array.isArray(_)?_:[];if(g.length===0)break;u=u.concat(g),d=g.length===c,p++}let m=u.slice(0,a).map(h=>({name:h.name,fullName:h.full_name,private:h.private,description:h.description,language:h.language,defaultBranch:h.default_branch,updatedAt:h.updated_at,stars:h.stargazers_count,url:h.html_url}));return JSON.stringify({count:m.length,repos:m})}case"github_create_issue":{let{owner:e,repo:r,title:n,body:i}=t;if(!e||!r||!n)return JSON.stringify({error:"owner, repo, and title are required"});let o=await j(`/repos/${e}/${r}/issues`,{method:"POST",body:{title:n,body:i||""}});return JSON.stringify({number:o.number,url:o.html_url,title:o.title})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},tools:[{name:"github_get_user",description:"Get the authenticated GitHub user profile and their organizations",input_schema:{type:"object",properties:{}}},{name:"github_list_orgs",description:"List GitHub organizations the authenticated user belongs to",input_schema:{type:"object",properties:{}}},{name:"github_list_repos",description:"List repositories for a user or org. If no owner given, lists the authenticated user's repos.",input_schema:{type:"object",properties:{owner:{type:"string",description:"Org or user login. Omit to list your own repos."},type:{type:"string",enum:["all","public","private","forks","sources","member"],description:"Filter by type (default: all)"},sort:{type:"string",enum:["created","updated","pushed","full_name"],description:"Sort field (default: updated)"},direction:{type:"string",enum:["asc","desc"],description:"Sort direction (default: desc)"},limit:{type:"number",description:"Max repos to return (default: 30)"}}}},{name:"github_clone",description:'Clone a GitHub repository to the local filesystem. Use when user says "check out" or "clone" a repo.',input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner (user or org name)"},repo:{type:"string",description:"Repository name"},destination:{type:"string",description:"Destination directory. Accepts absolute paths, ~-prefixed paths, or relative names. Defaults to ~/zibby-repos/<repo>."}},required:["owner","repo"]}},{name:"github_search_repos",description:"Search accessible repositories by name or description. Use this when the user asks to find a specific repo.",input_schema:{type:"object",properties:{query:{type:"string",description:'Search term to match against repo name or description (e.g., "electron", "my-app")'},limit:{type:"number",description:"Max results (default: 20)"}},required:["query"]}},{name:"github_search_issues",description:"Search GitHub issues and pull requests",input_schema:{type:"object",properties:{query:{type:"string",description:'GitHub search query (e.g. "SCRUM-123", "login bug repo:org/app")'},limit:{type:"number",description:"Max results (default: 20)"}},required:["query"]}},{name:"github_search_code",description:"Search code across GitHub repositories by keyword",input_schema:{type:"object",properties:{query:{type:"string",description:'Code search query (e.g. "handleLogin", "class AuthService")'},repo:{type:"string",description:'Scope to a specific repo (e.g. "org/app"). Optional.'},language:{type:"string",description:'Filter by language (e.g. "javascript", "python"). Optional.'},limit:{type:"number",description:"Max results (default: 15)"}},required:["query"]}},{name:"github_get_pr",description:"Get details of a pull request \u2014 title, description, branch, stats",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_get_pr_diff",description:"Get the unified diff of a pull request \u2014 the actual code changes",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_list_pr_files",description:"List files changed in a PR with per-file patches",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_list_pr_comments",description:"Get all review and issue comments on a PR",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},number:{type:"number",description:"PR number"}},required:["owner","repo","number"]}},{name:"github_list_commits",description:"List recent commits on a branch, optionally filtered by file path",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},branch:{type:"string",description:"Branch name (default: repo default branch)"},path:{type:"string",description:"Filter commits touching this file path"},limit:{type:"number",description:"Max commits (default: 20)"}},required:["owner","repo"]}},{name:"github_get_commit",description:"Get details of a specific commit \u2014 message, stats, file diffs",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},sha:{type:"string",description:"Commit SHA (full or short)"}},required:["owner","repo","sha"]}},{name:"github_get_file",description:"Read a file (or list a directory) from a GitHub repo. Works on any branch/ref.",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},path:{type:"string",description:'File or directory path (e.g. "src/auth/login.ts")'},ref:{type:"string",description:"Branch, tag, or commit SHA (default: repo default branch)"}},required:["owner","repo","path"]}},{name:"github_create_issue",description:"Create a GitHub issue",input_schema:{type:"object",properties:{owner:{type:"string",description:"Repository owner"},repo:{type:"string",description:"Repository name"},title:{type:"string",description:"Issue title"},body:{type:"string",description:"Issue body (markdown)"}},required:["owner","repo","title"]}}]};import{existsSync as Dr}from"fs";import{fileURLToPath as Ur}from"url";import{dirname as Pr,resolve as Mr}from"path";import{resolveIntegrationToken as qr}from"@zibby/core/backend-client.js";function Kr(){if(process.env.MCP_SLACK_PATH)return process.env.MCP_SLACK_PATH;let s=Pr(Ur(import.meta.url)),t=Mr(s,"..","bin","mcp-slack.mjs");return Dr(t)?t:null}async function J(s,t={}){let{token:e}=await qr("slack"),r=["conversations.list","users.list","users.profile.get","users.lookupByEmail","usergroups.list","usergroups.users.list","conversations.history","conversations.replies"].includes(s),n=`https://slack.com/api/${s}`,i={Authorization:`Bearer ${e}`},o;if(r){let u=new URLSearchParams(t).toString();u&&(n+=`?${u}`)}else i["Content-Type"]="application/json; charset=utf-8",o=JSON.stringify(t);let a=await(await fetch(n,{method:r?"GET":"POST",headers:i,body:o})).json();if(!a.ok)throw new Error(`Slack API error: ${a.error}`);return a}var D={id:"slack",serverName:"slack",allowedTools:["mcp__slack__*"],requiresIntegration:A.SLACK,envKeys:["SLACK_BOT_TOKEN","SLACK_TEAM_ID"],description:"Slack MCP Server",promptFragment:`## Slack (connected)
|
|
108
108
|
You have access to the user's Slack workspace. Use these tools:
|
|
109
109
|
- slack_list_channels, slack_post_message, slack_reply_to_thread
|
|
110
110
|
- slack_add_reaction, slack_get_channel_history, slack_get_thread_replies
|
|
111
111
|
- slack_get_users, slack_get_user_profile
|
|
112
112
|
- slack_lookup_user_by_email (precise email\u2192user_id, prefer this over scanning slack_get_users)
|
|
113
|
-
- slack_list_usergroups, slack_get_usergroup_members (workspace-defined teams like @oncall, @platform)`,resolve(){let s=Kr();if(!s)return null;let t={};for(let e of["PROJECT_API_TOKEN","PROGRESS_API_URL","EXECUTION_ID","PROJECT_ID","STAGE"])process.env[e]&&(t[e]=process.env[e]);for(let e of this.envKeys)process.env[e]&&(t[e]=process.env[e]);return{type:"stdio",command:"node",args:[s],env:t,alwaysLoad:!0}},async handleToolCall(s,t){try{switch(s){case"slack_list_channels":{let e=await J("conversations.list",{types:"public_channel",limit:100});return JSON.stringify({channels:(e.channels||[]).map(r=>({id:r.id,name:r.name,topic:r.topic?.value}))})}case"slack_post_message":{if(!t.channel||!t.text)return JSON.stringify({error:"channel and text are required"});let e=await J("chat.postMessage",{channel:t.channel,text:t.text});return JSON.stringify({ok:!0,ts:e.ts,channel:e.channel})}case"slack_reply_to_thread":{if(!t.channel||!t.thread_ts||!t.text)return JSON.stringify({error:"channel, thread_ts, and text are required"});let e=await J("chat.postMessage",{channel:t.channel,thread_ts:t.thread_ts,text:t.text});return JSON.stringify({ok:!0,ts:e.ts})}case"slack_add_reaction":return!t.channel||!t.timestamp||!t.reaction?JSON.stringify({error:"channel, timestamp, and reaction are required"}):(await J("reactions.add",{channel:t.channel,timestamp:t.timestamp,name:t.reaction}),JSON.stringify({ok:!0}));case"slack_get_channel_history":{if(!t.channel)return JSON.stringify({error:"channel is required"});let e=await J("conversations.history",{channel:t.channel,limit:t.limit||20});return JSON.stringify({messages:(e.messages||[]).map(r=>({user:r.user,text:r.text,ts:r.ts}))})}case"slack_get_thread_replies":{if(!t.channel||!t.thread_ts)return JSON.stringify({error:"channel and thread_ts are required"});let e=await J("conversations.replies",{channel:t.channel,ts:t.thread_ts});return JSON.stringify({messages:(e.messages||[]).map(r=>({user:r.user,text:r.text,ts:r.ts}))})}case"slack_get_users":{let e=await J("users.list",{limit:100});return JSON.stringify({users:(e.members||[]).filter(r=>!r.is_bot&&!r.deleted).map(r=>({id:r.id,name:r.real_name||r.name}))})}case"slack_get_user_profile":{if(!t.user_id)return JSON.stringify({error:"user_id is required"});let e=await J("users.profile.get",{user:t.user_id});return JSON.stringify({profile:e.profile})}case"slack_lookup_user_by_email":{if(!t.email)return JSON.stringify({error:"email is required"});try{let e=await J("users.lookupByEmail",{email:t.email});return JSON.stringify({ok:!0,user:{id:e.user?.id,name:e.user?.real_name||e.user?.name,email:e.user?.profile?.email||t.email}})}catch(e){if(/users_not_found/.test(e.message))return JSON.stringify({ok:!1,reason:"users_not_found"});throw e}}case"slack_list_usergroups":{let e=await J("usergroups.list",{});return JSON.stringify({usergroups:(e.usergroups||[]).map(r=>({id:r.id,handle:r.handle,name:r.name,description:r.description||"",user_count:Number(r.user_count||0)}))})}case"slack_get_usergroup_members":{if(!t.usergroup)return JSON.stringify({error:"usergroup id is required"});let e=await J("usergroups.users.list",{usergroup:t.usergroup});return JSON.stringify({users:e.users||[]})}case"slack_search_users":{if(!t.query||typeof t.query!="string")return JSON.stringify({error:"query is required"});let e=t.query.trim().toLowerCase();if(!e)return JSON.stringify({ok:!0,matches:[]});let r=Math.max(1,Math.min(Number(t.limit)||5,25)),n=[],i,o=5;for(let a=0;a<o;a+=1){let u={limit:200};i&&(u.cursor=i);let l=await J("users.list",u);for(let p of l.members||[])p.deleted||p.is_bot||n.push(p);if(i=l.response_metadata?.next_cursor,!i)break}let c=[];for(let a of n){let u=(a.real_name||"").toLowerCase(),l=(a.profile?.display_name||"").toLowerCase(),p=(a.name||"").toLowerCase(),d=0;u.includes(e)&&(d+=100-Math.abs(u.length-e.length)),l.includes(e)&&(d+=60-Math.abs(l.length-e.length)),p.includes(e)&&(d+=30-Math.abs(p.length-e.length)),(u===e||l===e)&&(d+=200),d>0&&c.push({id:a.id,name:a.real_name||a.profile?.display_name||a.name,email:a.profile?.email||void 0,_score:d})}return c.sort((a,u)=>u._score-a._score),JSON.stringify({ok:!0,matches:c.slice(0,r).map(({_score:a,...u})=>u),scanned:n.length})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},tools:[{name:"slack_list_channels",description:"List public channels in the workspace",input_schema:{type:"object",properties:{}}},{name:"slack_post_message",description:"Post a message to a Slack channel or DM",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID or name"},text:{type:"string",description:"Message text"}},required:["channel","text"]}},{name:"slack_reply_to_thread",description:"Reply to a specific message thread",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},thread_ts:{type:"string",description:"Thread timestamp"},text:{type:"string",description:"Reply text"}},required:["channel","thread_ts","text"]}},{name:"slack_add_reaction",description:"Add an emoji reaction to a message",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},timestamp:{type:"string",description:"Message timestamp"},reaction:{type:"string",description:"Emoji name without colons"}},required:["channel","timestamp","reaction"]}},{name:"slack_get_channel_history",description:"Get recent messages from a channel",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},limit:{type:"number",description:"Number of messages"}},required:["channel"]}},{name:"slack_get_thread_replies",description:"Get all replies in a message thread",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},thread_ts:{type:"string",description:"Thread timestamp"}},required:["channel","thread_ts"]}},{name:"slack_get_users",description:"List workspace users with basic profiles",input_schema:{type:"object",properties:{}}},{name:"slack_get_user_profile",description:"Get detailed profile for a specific user",input_schema:{type:"object",properties:{user_id:{type:"string",description:"Slack user ID"}},required:["user_id"]}},{name:"slack_lookup_user_by_email",description:"Find a Slack user by email. Returns { ok:true, user:{id,name,email} } on hit, { ok:false } when no user has that email. Prefer this over slack_get_users for email-based routing \u2014 single API call, exact match.",input_schema:{type:"object",properties:{email:{type:"string",description:"Email address to look up"}},required:["email"]}},{name:"slack_list_usergroups",description:"List workspace-defined user groups (e.g. @oncall, @platform). Each item has { id, handle, name, description, user_count }. Use the id with slack_get_usergroup_members to expand the membership.",input_schema:{type:"object",properties:{}}},{name:"slack_get_usergroup_members",description:"List user IDs that belong to a Slack usergroup. Pair with slack_post_message to DM each member, or use the group id directly in a channel message as <!subteam^ID> to @-mention.",input_schema:{type:"object",properties:{usergroup:{type:"string",description:"Usergroup id, e.g. S012ABC"}},required:["usergroup"]}},{name:"slack_search_users",description:'Fuzzy-search workspace users by display name or real name. Use when the user said something like "send to Sam" without an email. Returns up to `limit` ranked matches { id, name, email }. Slack has no native name-search API \u2014 this scans paginated users.list + does substring scoring (real_name > display_name > name). For large workspaces consider higher limit + ask the user to confirm if multiple hit.',input_schema:{type:"object",properties:{query:{type:"string",description:"Substring to match against names (case-insensitive)"},limit:{type:"number",description:"Max matches to return (default 5, max 25)"}},required:["query"]}}]};import{existsSync as Br}from"fs";import{fileURLToPath as Fr}from"url";import{dirname as zr,resolve as Gr}from"path";import{resolveIntegrationToken as Wr}from"@zibby/core/backend-client.js";function Hr(){if(process.env.MCP_LARK_PATH)return process.env.MCP_LARK_PATH;let s=zr(Fr(import.meta.url)),t=Gr(s,"..","bin","mcp-lark.mjs");return Br(t)?t:null}var Yr=6e3*1e3,re=null;async function
|
|
113
|
+
- slack_list_usergroups, slack_get_usergroup_members (workspace-defined teams like @oncall, @platform)`,resolve(){let s=Kr();if(!s)return null;let t={};for(let e of["PROJECT_API_TOKEN","PROGRESS_API_URL","EXECUTION_ID","PROJECT_ID","STAGE"])process.env[e]&&(t[e]=process.env[e]);for(let e of this.envKeys)process.env[e]&&(t[e]=process.env[e]);return{type:"stdio",command:"node",args:[s],env:t,alwaysLoad:!0}},async handleToolCall(s,t){try{switch(s){case"slack_list_channels":{let e=await J("conversations.list",{types:"public_channel",limit:100});return JSON.stringify({channels:(e.channels||[]).map(r=>({id:r.id,name:r.name,topic:r.topic?.value}))})}case"slack_post_message":{if(!t.channel||!t.text)return JSON.stringify({error:"channel and text are required"});let e=await J("chat.postMessage",{channel:t.channel,text:t.text});return JSON.stringify({ok:!0,ts:e.ts,channel:e.channel})}case"slack_reply_to_thread":{if(!t.channel||!t.thread_ts||!t.text)return JSON.stringify({error:"channel, thread_ts, and text are required"});let e=await J("chat.postMessage",{channel:t.channel,thread_ts:t.thread_ts,text:t.text});return JSON.stringify({ok:!0,ts:e.ts})}case"slack_add_reaction":return!t.channel||!t.timestamp||!t.reaction?JSON.stringify({error:"channel, timestamp, and reaction are required"}):(await J("reactions.add",{channel:t.channel,timestamp:t.timestamp,name:t.reaction}),JSON.stringify({ok:!0}));case"slack_get_channel_history":{if(!t.channel)return JSON.stringify({error:"channel is required"});let e=await J("conversations.history",{channel:t.channel,limit:t.limit||20});return JSON.stringify({messages:(e.messages||[]).map(r=>({user:r.user,text:r.text,ts:r.ts}))})}case"slack_get_thread_replies":{if(!t.channel||!t.thread_ts)return JSON.stringify({error:"channel and thread_ts are required"});let e=await J("conversations.replies",{channel:t.channel,ts:t.thread_ts});return JSON.stringify({messages:(e.messages||[]).map(r=>({user:r.user,text:r.text,ts:r.ts}))})}case"slack_get_users":{let e=await J("users.list",{limit:100});return JSON.stringify({users:(e.members||[]).filter(r=>!r.is_bot&&!r.deleted).map(r=>({id:r.id,name:r.real_name||r.name}))})}case"slack_get_user_profile":{if(!t.user_id)return JSON.stringify({error:"user_id is required"});let e=await J("users.profile.get",{user:t.user_id});return JSON.stringify({profile:e.profile})}case"slack_lookup_user_by_email":{if(!t.email)return JSON.stringify({error:"email is required"});try{let e=await J("users.lookupByEmail",{email:t.email});return JSON.stringify({ok:!0,user:{id:e.user?.id,name:e.user?.real_name||e.user?.name,email:e.user?.profile?.email||t.email}})}catch(e){if(/users_not_found/.test(e.message))return JSON.stringify({ok:!1,reason:"users_not_found"});throw e}}case"slack_list_usergroups":{let e=await J("usergroups.list",{});return JSON.stringify({usergroups:(e.usergroups||[]).map(r=>({id:r.id,handle:r.handle,name:r.name,description:r.description||"",user_count:Number(r.user_count||0)}))})}case"slack_get_usergroup_members":{if(!t.usergroup)return JSON.stringify({error:"usergroup id is required"});let e=await J("usergroups.users.list",{usergroup:t.usergroup});return JSON.stringify({users:e.users||[]})}case"slack_search_users":{if(!t.query||typeof t.query!="string")return JSON.stringify({error:"query is required"});let e=t.query.trim().toLowerCase();if(!e)return JSON.stringify({ok:!0,matches:[]});let r=Math.max(1,Math.min(Number(t.limit)||5,25)),n=[],i,o=5;for(let a=0;a<o;a+=1){let u={limit:200};i&&(u.cursor=i);let l=await J("users.list",u);for(let p of l.members||[])p.deleted||p.is_bot||n.push(p);if(i=l.response_metadata?.next_cursor,!i)break}let c=[];for(let a of n){let u=(a.real_name||"").toLowerCase(),l=(a.profile?.display_name||"").toLowerCase(),p=(a.name||"").toLowerCase(),d=0;u.includes(e)&&(d+=100-Math.abs(u.length-e.length)),l.includes(e)&&(d+=60-Math.abs(l.length-e.length)),p.includes(e)&&(d+=30-Math.abs(p.length-e.length)),(u===e||l===e)&&(d+=200),d>0&&c.push({id:a.id,name:a.real_name||a.profile?.display_name||a.name,email:a.profile?.email||void 0,_score:d})}return c.sort((a,u)=>u._score-a._score),JSON.stringify({ok:!0,matches:c.slice(0,r).map(({_score:a,...u})=>u),scanned:n.length})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},tools:[{name:"slack_list_channels",description:"List public channels in the workspace",input_schema:{type:"object",properties:{}}},{name:"slack_post_message",description:"Post a message to a Slack channel or DM",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID or name"},text:{type:"string",description:"Message text"}},required:["channel","text"]}},{name:"slack_reply_to_thread",description:"Reply to a specific message thread",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},thread_ts:{type:"string",description:"Thread timestamp"},text:{type:"string",description:"Reply text"}},required:["channel","thread_ts","text"]}},{name:"slack_add_reaction",description:"Add an emoji reaction to a message",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},timestamp:{type:"string",description:"Message timestamp"},reaction:{type:"string",description:"Emoji name without colons"}},required:["channel","timestamp","reaction"]}},{name:"slack_get_channel_history",description:"Get recent messages from a channel",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},limit:{type:"number",description:"Number of messages"}},required:["channel"]}},{name:"slack_get_thread_replies",description:"Get all replies in a message thread",input_schema:{type:"object",properties:{channel:{type:"string",description:"Channel ID"},thread_ts:{type:"string",description:"Thread timestamp"}},required:["channel","thread_ts"]}},{name:"slack_get_users",description:"List workspace users with basic profiles",input_schema:{type:"object",properties:{}}},{name:"slack_get_user_profile",description:"Get detailed profile for a specific user",input_schema:{type:"object",properties:{user_id:{type:"string",description:"Slack user ID"}},required:["user_id"]}},{name:"slack_lookup_user_by_email",description:"Find a Slack user by email. Returns { ok:true, user:{id,name,email} } on hit, { ok:false } when no user has that email. Prefer this over slack_get_users for email-based routing \u2014 single API call, exact match.",input_schema:{type:"object",properties:{email:{type:"string",description:"Email address to look up"}},required:["email"]}},{name:"slack_list_usergroups",description:"List workspace-defined user groups (e.g. @oncall, @platform). Each item has { id, handle, name, description, user_count }. Use the id with slack_get_usergroup_members to expand the membership.",input_schema:{type:"object",properties:{}}},{name:"slack_get_usergroup_members",description:"List user IDs that belong to a Slack usergroup. Pair with slack_post_message to DM each member, or use the group id directly in a channel message as <!subteam^ID> to @-mention.",input_schema:{type:"object",properties:{usergroup:{type:"string",description:"Usergroup id, e.g. S012ABC"}},required:["usergroup"]}},{name:"slack_search_users",description:'Fuzzy-search workspace users by display name or real name. Use when the user said something like "send to Sam" without an email. Returns up to `limit` ranked matches { id, name, email }. Slack has no native name-search API \u2014 this scans paginated users.list + does substring scoring (real_name > display_name > name). For large workspaces consider higher limit + ask the user to confirm if multiple hit.',input_schema:{type:"object",properties:{query:{type:"string",description:"Substring to match against names (case-insensitive)"},limit:{type:"number",description:"Max matches to return (default 5, max 25)"}},required:["query"]}}]};import{existsSync as Br}from"fs";import{fileURLToPath as Fr}from"url";import{dirname as zr,resolve as Gr}from"path";import{resolveIntegrationToken as Wr}from"@zibby/core/backend-client.js";function Hr(){if(process.env.MCP_LARK_PATH)return process.env.MCP_LARK_PATH;let s=zr(Fr(import.meta.url)),t=Gr(s,"..","bin","mcp-lark.mjs");return Br(t)?t:null}var Yr=6e3*1e3,re=null;async function Vr(){let{appId:s,appSecret:t,host:e}=await Wr("lark");if(re&&re.appId===s&&re.expiresAt>Date.now())return{token:re.token,host:e};let n=await(await fetch(`${e}/open-apis/auth/v3/tenant_access_token/internal`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({app_id:s,app_secret:t})})).json();if(n.code!==0)throw new Error(`Lark tenant_access_token failed: ${n.msg||n.code}`);return re={token:n.tenant_access_token,expiresAt:Date.now()+Yr,appId:s},{token:n.tenant_access_token,host:e}}async function H(s,t,e={}){let{token:r,host:n}=await Vr(),i=`${n}${t}`,o={method:s,headers:{Authorization:`Bearer ${r}`,"Content-Type":"application/json; charset=utf-8"}};s!=="GET"&&(o.body=JSON.stringify(e));let a=await(await fetch(i,o)).json();if(a.code!==0)throw new Error(`Lark API ${t} error: ${a.msg||a.code}`);return a.data||{}}function Qe(s){return JSON.stringify({text:s})}function Zr(s){return!s||typeof s!="string"||s.startsWith("oc_")?"chat_id":s.startsWith("ou_")?"open_id":s.startsWith("on_")?"union_id":s.startsWith("cli_")?"app_id":s.includes("@")?"email":"chat_id"}var P={id:"lark",serverName:"lark",allowedTools:["mcp__lark__*"],requiresIntegration:A.LARK,description:"Lark / Feishu messaging \u2014 send messages and reply in threads.",envKeys:[],promptFragment:`## Lark (connected)
|
|
114
114
|
You can send messages and replies on Lark. Use:
|
|
115
115
|
- lark_send_message: post a message to a chat, user, or DM
|
|
116
116
|
- lark_reply: reply to an existing message (threaded)
|
|
117
117
|
- lark_list_chats: list chats the bot is a member of
|
|
118
118
|
- lark_get_chat_history: fetch recent messages in a chat
|
|
119
119
|
- lark_lookup_user_by_email: resolve an email \u2192 open_id for direct DM (prefer this over emailing through lark_send_message when the agent has a user_id already)
|
|
120
|
-
When responding to an incoming event, prefer lark_reply with the source message_id so the response threads cleanly.`,resolve(){let s=Hr();if(!s)return null;let t={};for(let e of["PROJECT_API_TOKEN","PROGRESS_API_URL","EXECUTION_ID","PROJECT_ID","STAGE"])process.env[e]&&(t[e]=process.env[e]);return{type:"stdio",command:"node",args:[s],env:t,alwaysLoad:!0}},tools:[{name:"lark_send_message",description:"Send a text message to a Lark chat, user, or DM. receive_id can be a chat_id (oc_*), open_id (ou_*), union_id (on_*), or email.",input_schema:{type:"object",properties:{receive_id:{type:"string",description:"Target id: chat_id (oc_*), open_id (ou_*), union_id (on_*), or email"},text:{type:"string",description:"Message text"}},required:["receive_id","text"]}},{name:"lark_reply",description:"Reply to an existing Lark message (creates a thread). Use the message_id from the inbound event.",input_schema:{type:"object",properties:{message_id:{type:"string",description:"Lark message id (om_*) to reply to"},text:{type:"string",description:"Reply text"}},required:["message_id","text"]}},{name:"lark_list_chats",description:"List chats (groups + DMs) the bot is a member of.",input_schema:{type:"object",properties:{page_size:{type:"number",description:"Max results (default 50)"}}}},{name:"lark_get_chat_history",description:"Fetch recent messages in a chat.",input_schema:{type:"object",properties:{chat_id:{type:"string",description:"Chat id (oc_*)"},page_size:{type:"number",description:"Max messages (default 20)"}},required:["chat_id"]}},{name:"lark_lookup_user_by_email",description:"Resolve an email address to a Lark user id (open_id). Returns { ok:true, user:{open_id,email,name} } on hit, { ok:false } if no Lark user has that email. Use the open_id as `receive_id` in lark_send_message to DM.",input_schema:{type:"object",properties:{email:{type:"string",description:"Email address to look up"}},required:["email"]}},{name:"lark_search_users",description:'Fuzzy-search users by name across chats the bot is a member of. Lark has no public org-wide user search API for bots \u2014 this walks the bot\'s chat memberships and matches names client-side. Best for "send to Sam" style routing where you have a name but no email. Returns up to `limit` ranked matches { open_id, name }.',input_schema:{type:"object",properties:{query:{type:"string",description:"Substring to match against user names (case-insensitive)"},limit:{type:"number",description:"Max matches to return (default 5, max 25)"}},required:["query"]}}],async handleToolCall(s,t){try{switch(s){case"lark_send_message":{if(!t.receive_id||!t.text)return JSON.stringify({error:"receive_id and text are required"});let e=
|
|
120
|
+
When responding to an incoming event, prefer lark_reply with the source message_id so the response threads cleanly.`,resolve(){let s=Hr();if(!s)return null;let t={};for(let e of["PROJECT_API_TOKEN","PROGRESS_API_URL","EXECUTION_ID","PROJECT_ID","STAGE"])process.env[e]&&(t[e]=process.env[e]);return{type:"stdio",command:"node",args:[s],env:t,alwaysLoad:!0}},tools:[{name:"lark_send_message",description:"Send a text message to a Lark chat, user, or DM. receive_id can be a chat_id (oc_*), open_id (ou_*), union_id (on_*), or email.",input_schema:{type:"object",properties:{receive_id:{type:"string",description:"Target id: chat_id (oc_*), open_id (ou_*), union_id (on_*), or email"},text:{type:"string",description:"Message text"}},required:["receive_id","text"]}},{name:"lark_reply",description:"Reply to an existing Lark message (creates a thread). Use the message_id from the inbound event.",input_schema:{type:"object",properties:{message_id:{type:"string",description:"Lark message id (om_*) to reply to"},text:{type:"string",description:"Reply text"}},required:["message_id","text"]}},{name:"lark_list_chats",description:"List chats (groups + DMs) the bot is a member of.",input_schema:{type:"object",properties:{page_size:{type:"number",description:"Max results (default 50)"}}}},{name:"lark_get_chat_history",description:"Fetch recent messages in a chat.",input_schema:{type:"object",properties:{chat_id:{type:"string",description:"Chat id (oc_*)"},page_size:{type:"number",description:"Max messages (default 20)"}},required:["chat_id"]}},{name:"lark_lookup_user_by_email",description:"Resolve an email address to a Lark user id (open_id). Returns { ok:true, user:{open_id,email,name} } on hit, { ok:false } if no Lark user has that email. Use the open_id as `receive_id` in lark_send_message to DM.",input_schema:{type:"object",properties:{email:{type:"string",description:"Email address to look up"}},required:["email"]}},{name:"lark_search_users",description:'Fuzzy-search users by name across chats the bot is a member of. Lark has no public org-wide user search API for bots \u2014 this walks the bot\'s chat memberships and matches names client-side. Best for "send to Sam" style routing where you have a name but no email. Returns up to `limit` ranked matches { open_id, name }.',input_schema:{type:"object",properties:{query:{type:"string",description:"Substring to match against user names (case-insensitive)"},limit:{type:"number",description:"Max matches to return (default 5, max 25)"}},required:["query"]}}],async handleToolCall(s,t){try{switch(s){case"lark_send_message":{if(!t.receive_id||!t.text)return JSON.stringify({error:"receive_id and text are required"});let e=Zr(t.receive_id),r=await H("POST",`/open-apis/im/v1/messages?receive_id_type=${e}`,{receive_id:t.receive_id,msg_type:"text",content:Qe(t.text)});return JSON.stringify({ok:!0,message_id:r.message_id})}case"lark_reply":{if(!t.message_id||!t.text)return JSON.stringify({error:"message_id and text are required"});let e=await H("POST",`/open-apis/im/v1/messages/${encodeURIComponent(t.message_id)}/reply`,{msg_type:"text",content:Qe(t.text)});return JSON.stringify({ok:!0,message_id:e.message_id})}case"lark_list_chats":{let e=t.page_size||50,n=((await H("GET",`/open-apis/im/v1/chats?page_size=${e}`)).items||[]).map(i=>({chat_id:i.chat_id,name:i.name,description:i.description,owner_id:i.owner_id,chat_mode:i.chat_mode}));return JSON.stringify({chats:n})}case"lark_get_chat_history":{if(!t.chat_id)return JSON.stringify({error:"chat_id is required"});let e=t.page_size||20,n=((await H("GET",`/open-apis/im/v1/messages?container_id_type=chat&container_id=${encodeURIComponent(t.chat_id)}&page_size=${e}&sort_type=ByCreateTimeDesc`)).items||[]).map(i=>({message_id:i.message_id,sender_id:i.sender?.id,sender_type:i.sender?.sender_type,msg_type:i.msg_type,content:i.body?.content,create_time:i.create_time}));return JSON.stringify({messages:n})}case"lark_lookup_user_by_email":{if(!t.email)return JSON.stringify({error:"email is required"});let r=((await H("POST","/open-apis/contact/v3/users/batch_get_id?user_id_type=open_id",{emails:[t.email]})).user_list||[]).find(n=>n.email===t.email&&n.user_id);return JSON.stringify(r?{ok:!0,user:{open_id:r.user_id,email:r.email,name:r.name||void 0}}:{ok:!1,reason:"no_lark_user_for_email"})}case"lark_search_users":{if(!t.query||typeof t.query!="string")return JSON.stringify({error:"query is required"});let e=t.query.trim().toLowerCase();if(!e)return JSON.stringify({ok:!0,matches:[]});let r=Math.max(1,Math.min(Number(t.limit)||5,25)),n=200,o=((await H("GET","/open-apis/im/v1/chats?page_size=100")).items||[]).map(l=>l.chat_id),c=new Set,a=[];for(let l of o){if(a.length>=n)break;try{let p=await H("GET",`/open-apis/im/v1/chats/${encodeURIComponent(l)}/members?member_id_type=open_id&page_size=100`);for(let d of p.items||[])if(!(!d.member_id||c.has(d.member_id))&&(c.add(d.member_id),a.push({open_id:d.member_id,name:d.name||""}),a.length>=n))break}catch(p){console.warn(`[lark] member scan failed for ${l}: ${p.message}`)}}let u=[];for(let l of a){let p=(l.name||"").toLowerCase();if(!p)continue;let d=0;p.includes(e)&&(d+=100-Math.abs(p.length-e.length)),p===e&&(d+=200),d>0&&u.push({open_id:l.open_id,name:l.name,_score:d})}return u.sort((l,p)=>p._score-l._score),JSON.stringify({ok:!0,matches:u.slice(0,r).map(({_score:l,...p})=>p),scanned:a.length})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}}};var Xe={id:"chat_notify",description:"Chat notification meta-skill \u2014 routes to whichever messaging integration (Slack OR Lark) the user has configured for this project.",envKeys:[...D.envKeys||[],...P.envKeys||[]],get serverName(){if(process.env.SLACK_CHANNEL)return D.serverName;if(process.env.LARK_RECEIVE_ID)return P.serverName},get allowedTools(){return process.env.SLACK_CHANNEL?D.allowedTools||[]:process.env.LARK_RECEIVE_ID?P.allowedTools||[]:[]},promptFragment:`## Chat notifications (Slack OR Lark \u2014 at least one connected)
|
|
121
121
|
You can post chat messages via either:
|
|
122
122
|
- slack_post_message (channel, text) \u2014 Slack, when SLACK_CHANNEL is set
|
|
123
123
|
- lark_send_message (receive_id, text) \u2014 Lark, when LARK_RECEIVE_ID is set
|
|
124
|
-
Use whichever the user has configured.`,resolve(s){return process.env.SLACK_CHANNEL&&typeof
|
|
124
|
+
Use whichever the user has configured.`,resolve(s){return process.env.SLACK_CHANNEL&&typeof D.resolve=="function"?D.resolve(s):process.env.LARK_RECEIVE_ID&&typeof P.resolve=="function"?P.resolve(s):null},async handleToolCall(s,t,e){return typeof s=="string"&&s.startsWith("slack_")?D.handleToolCall(s,t,e):typeof s=="string"&&s.startsWith("lark_")?P.handleToolCall(s,t,e):JSON.stringify({error:`chat_notify: unknown tool "${s}". Expected slack_* or lark_*.`})},get tools(){return[...D.tools||[],...P.tools||[]]}};import{createRequire as Qr}from"module";import{execFileSync as Xr}from"child_process";import{join as et}from"path";import{existsSync as es}from"fs";var ts=Qr(import.meta.url);function rs(){if(process.env.MCP_MEMORY_PATH)return process.env.MCP_MEMORY_PATH;try{return ts.resolve("@zibby/ui-memory/mcp-server")}catch{return null}}var tt={id:"memory",serverName:"memory",allowedTools:["mcp__memory__*"],envKeys:[],description:"Zibby Memory MCP Server (test history, selectors, page model)",async middleware(){try{let{createMemoryMiddleware:s}=await import("@zibby/ui-memory");return s()}catch{return null}},promptFragment:`BEFORE executing browser actions:
|
|
125
125
|
- Review any test memory/history above. Prefer selectors proven to work.
|
|
126
126
|
- If a previous run failed, avoid the same approach.
|
|
127
127
|
- After setup/login completes, navigate directly to the target page instead of clicking through menus.
|
|
@@ -162,7 +162,7 @@ AFTER completing the test, you MUST call memory_save_insight at least once:
|
|
|
162
162
|
You have access to the user's Sentry. Use these tools:
|
|
163
163
|
- sentry_list_projects: List projects in the organization
|
|
164
164
|
- sentry_list_issues: List errors/issues (supports Sentry search query, project filter, sort)
|
|
165
|
-
- sentry_get_issue: Get detailed info about a specific issue (requires issueId)`,resolve(){let s=xs();if(!s)return null;let t={};for(let e of["PROJECT_API_TOKEN","PROGRESS_API_URL","EXECUTION_ID","PROJECT_ID","STAGE"])process.env[e]&&(t[e]=process.env[e]);return{type:"stdio",command:"node",args:[s],env:t,alwaysLoad:!0}},async handleToolCall(s,t={}){try{switch(s){case"sentry_list_projects":{let e=await As();return JSON.stringify({projects:e.map(r=>({slug:r.slug,name:r.name,platform:r.platform}))})}case"sentry_list_issues":{let e=await Ts({query:t.query,sort:t.sort,project:t.project,limit:t.limit});return JSON.stringify({issues:e.map(r=>({id:r.id,title:r.title,culprit:r.culprit,count:r.count,firstSeen:r.firstSeen,lastSeen:r.lastSeen,level:r.level,status:r.status}))})}case"sentry_get_issue":{let e=await Es(t.issueId);return JSON.stringify({id:e.id,title:e.title,culprit:e.culprit,metadata:e.metadata,count:e.count,userCount:e.userCount,firstSeen:e.firstSeen,lastSeen:e.lastSeen,level:e.level,status:e.status,project:{slug:e.project?.slug,name:e.project?.name}})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},toolsForAssistant:[{name:"sentry_list_projects",description:"List Sentry projects",input_schema:{type:"object",properties:{}}},{name:"sentry_list_issues",description:"List Sentry issues (errors)",input_schema:{type:"object",properties:{project:{type:"string",description:"Project slug (optional)"},query:{type:"string",description:"Sentry search query (default: is:unresolved)"},sort:{type:"string",description:"Sort order: date, new, priority, freq, user (default: date)"},limit:{type:"number",description:"Max issues to return (default 25)"}}}},{name:"sentry_get_issue",description:"Get details of a specific Sentry issue",input_schema:{type:"object",properties:{issueId:{type:"string",description:"Sentry issue ID"}},required:["issueId"]}}]};pe.tools=pe.toolsForAssistant;import{spawn as bt}from"child_process";import{writeFileSync as Cs,mkdirSync as pt,existsSync as
|
|
165
|
+
- sentry_get_issue: Get detailed info about a specific issue (requires issueId)`,resolve(){let s=xs();if(!s)return null;let t={};for(let e of["PROJECT_API_TOKEN","PROGRESS_API_URL","EXECUTION_ID","PROJECT_ID","STAGE"])process.env[e]&&(t[e]=process.env[e]);return{type:"stdio",command:"node",args:[s],env:t,alwaysLoad:!0}},async handleToolCall(s,t={}){try{switch(s){case"sentry_list_projects":{let e=await As();return JSON.stringify({projects:e.map(r=>({slug:r.slug,name:r.name,platform:r.platform}))})}case"sentry_list_issues":{let e=await Ts({query:t.query,sort:t.sort,project:t.project,limit:t.limit});return JSON.stringify({issues:e.map(r=>({id:r.id,title:r.title,culprit:r.culprit,count:r.count,firstSeen:r.firstSeen,lastSeen:r.lastSeen,level:r.level,status:r.status}))})}case"sentry_get_issue":{let e=await Es(t.issueId);return JSON.stringify({id:e.id,title:e.title,culprit:e.culprit,metadata:e.metadata,count:e.count,userCount:e.userCount,firstSeen:e.firstSeen,lastSeen:e.lastSeen,level:e.level,status:e.status,project:{slug:e.project?.slug,name:e.project?.name}})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(e){return JSON.stringify({error:e.message})}},toolsForAssistant:[{name:"sentry_list_projects",description:"List Sentry projects",input_schema:{type:"object",properties:{}}},{name:"sentry_list_issues",description:"List Sentry issues (errors)",input_schema:{type:"object",properties:{project:{type:"string",description:"Project slug (optional)"},query:{type:"string",description:"Sentry search query (default: is:unresolved)"},sort:{type:"string",description:"Sort order: date, new, priority, freq, user (default: date)"},limit:{type:"number",description:"Max issues to return (default 25)"}}}},{name:"sentry_get_issue",description:"Get details of a specific Sentry issue",input_schema:{type:"object",properties:{issueId:{type:"string",description:"Sentry issue ID"}},required:["issueId"]}}]};pe.tools=pe.toolsForAssistant;import{spawn as bt}from"child_process";import{writeFileSync as Cs,mkdirSync as pt,existsSync as M,readdirSync as he,readFileSync as fe,unlinkSync as Ls,createWriteStream as Js,statSync as Ds}from"fs";import{resolve as V,join as U}from"path";import{resolveMaxParallelRuns as wt}from"@zibby/core/utils/parallel-config.js";import{zibbyScratchSpecsDir as Us}from"@zibby/core/constants/zibby-scratch.js";var Re="sessions",$e=".zibby/output",de=process.env.ZIBBY_RUNNER_NODE_PROGRESS==="1",Ps=process.env.ZIBBY_RUNNER_STATUS_STREAM==="1",St=process.env.ZIBBY_RUNNER_SPAWN_LOGS==="1",E=new Map,z=[],Ms=0,Oe=0,dt=3e3;function vt(){return`run_${++Ms}_${Date.now().toString(36)}`}function mt(s){let t=Math.floor(s/1e3);return t<60?`${t}s`:`${Math.floor(t/60)}m ${t%60}s`}function Nt(s){return s.replace(/\x1b\[[0-9;]*[a-zA-Z]/g,"")}function T(s,t,e){if(!Ps)return;let r=`
|
|
166
166
|
${t} [${s}] ${e}
|
|
167
167
|
`;try{process.stderr.write(r)}catch{}}function je(){z.length=0;for(let[,s]of E)if(s.status==="queued"&&(s.status="cancelled"),s.status==="running"&&s._child)try{s._child.kill("SIGTERM")}catch{}}process.on("exit",je);process.on("SIGINT",()=>{je(),process.exit(0)});process.on("SIGTERM",()=>{je(),process.exit(0)});var Ot={id:"runner",description:"Run zibby test workflows from chat (parallel supported)",envKeys:[],promptFragment:`## Test Runner
|
|
168
168
|
You can run zibby test workflows directly from chat:
|
|
@@ -302,7 +302,7 @@ Each run generates:
|
|
|
302
302
|
- raw_stream_output.txt: Agent log
|
|
303
303
|
|
|
304
304
|
Use run_artifacts({ runId, type }) and run_diagnose({ runId }) to inspect and explain failures.`,resolve(){return null},async handleToolCall(s,t,e){let r=e?.options?.workspace||process.cwd();try{switch(s){case"run_generate":return await qs(t,r);case"run_test":return await Ws(t,r,e);case"run_status":return Hs(t);case"run_cancel":return Ys(t);case"run_artifacts":return Qs(t,r);case"run_diagnose":return Xs(t,r);case"list_specs":return en(t,r);default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(n){return JSON.stringify({error:n.message})}},tools:[{name:"run_generate",description:"Generate specs from codebase. CRITICAL: DO NOT USE if ticket has test steps in comments. Only use when: (1) NO steps in ticket AND (2) local codebase exists (not external URLs). For tickets with steps, use run_test with inline format.",input_schema:{type:"object",properties:{ticket:{type:"string",description:"Jira ticket key (e.g. SCRUM-123). Auto-fetches ticket details."},description:{type:"string",description:"Ticket description text (use if no Jira key available)"},input:{type:"string",description:"Path to a file containing ticket/requirements text"},repo:{type:"string",description:"Path to the codebase (default: current directory)"},agent:{type:"string",description:"Optional agent override (cursor, gemini, claude, codex, assistant). Omit to use configured agent."},output:{type:"string",description:"Output directory for spec files (default: test-specs)"}}}},{name:"run_test",description:"Start a test (async, returns runId). spec = file path, or inline:+steps, or a Jira-shaped issue key (e.g. PROJ-123): when Jira is connected, the runner loads that issue's description+comments into an inline spec. After starting, tell the user and let them ask for progress via run_status.",input_schema:{type:"object",properties:{spec:{type:"string",description:"Workspace file path; or inline:+steps; or Jira issue key (KEY-123) to auto-fetch from Jira when the jira skill is active."},ticketKey:{type:"string",description:"Optional label (e.g. SCRUM-123). If spec is an issue key, this defaults to that key."},agent:{type:"string",description:"Optional agent override (cursor, gemini, claude, codex, assistant). Omit to use configured agent."},headless:{type:"boolean",description:"Run browser headless (default false)"},workflow:{type:"string",description:"Workflow override (e.g. quick-smoke)"}},required:["spec"]}},{name:"run_status",description:'Instant progress check \u2014 returns immediately. Use this whenever user asks about test progress. ALWAYS use runId="all".',input_schema:{type:"object",properties:{runId:{type:"string",description:'Use "all" to see all runs in this session (recommended). Or a specific run ID if known.'}},required:["runId"]}},{name:"run_cancel",description:'Cancel/kill a running test. ONLY use when the USER explicitly asks to cancel or stop a run. NEVER auto-cancel \u2014 tests take 1-5 minutes and "running" is normal. Use runId="all" to cancel all active runs.',input_schema:{type:"object",properties:{runId:{type:"string",description:'Run ID to cancel, or "all" to cancel all active runs'}},required:["runId"]}},{name:"run_artifacts",description:"Read artifacts from a test run session. Can list files, read results/events/logs, or search across all sessions.",input_schema:{type:"object",properties:{runId:{type:"string",description:"Run ID from run_test. Omit to search across all sessions."},type:{type:"string",enum:["list","result","events","log","search"],description:'What to retrieve: "list" = all files in session, "result" = result.json, "events" = events.json, "log" = raw output tail, "search" = search text across sessions'},node:{type:"string",description:'Node name to read from (e.g. "execute_live", "generate_script"). Default: "execute_live"'},query:{type:"string",description:'Search text (only for type="search"). Searches across all session logs/events.'},tail:{type:"number",description:"Number of characters from end of log to return (default: 3000)"}},required:["type"]}},{name:"run_diagnose",description:"Diagnose one or all runs, especially failed ones. Uses run logs + known error patterns and returns likely root cause with suggested next action.",input_schema:{type:"object",properties:{runId:{type:"string",description:'Run ID from run_test, or "all" (default) to diagnose all known runs'},tail:{type:"number",description:"Characters of run log tail to inspect (default: 2000)"}}}},{name:"list_specs",description:"List available test spec files in the project",input_schema:{type:"object",properties:{directory:{type:"string",description:'Directory to scan (default: "test-specs")'}}}}]};function Ie(){let s=0;for(let[,t]of E)t.status==="running"&&s++;return s}function ft(){for(;z.length>0;){let s=wt(z[0]?.context?.options?.config);if(Ie()>=s)break;let{args:t,cwd:e,context:r}=z.shift();It(t,e,r)}}async function qs(s,t){let{ticket:e,description:r,input:n,repo:i,agent:o,output:c}=s,a=["generate"];e&&a.push("--ticket",e),r&&a.push("--description",r),n&&a.push("--input",n),i&&a.push("--repo",i),c&&a.push("--output",c);let u=["assistant","cursor","claude","codex","gemini"],l=o||process.env.AGENT_TYPE,p=l&&u.includes(l)?l:null;p&&a.push("--agent",p);let d=e||"generate";return T(d,"\u{1F9EA}","Starting test spec generation (real agent with codebase access)..."),new Promise(m=>{St&&console.error(`[zibby:spawn] skill=run_generate parentPid=${process.pid} \u2192 child zibby ${a.map(y=>/\s/.test(y)?JSON.stringify(y):y).join(" ")} cwd=${t}`);let h=bt("zibby",a,{cwd:t,env:{...process.env},stdio:["ignore","pipe","pipe"],detached:!1}),_="",g="";h.stdout.on("data",y=>{let k=y.toString();_+=k;for(let f of k.split(`
|
|
305
|
-
`)){let S=Nt(f).trim();S.startsWith("\u2705")?T(d,"\u2705",S.slice(2).trim()):S.startsWith("\u2713")&&T(d,"\u2714",S.slice(2).trim())}}),h.stderr.on("data",y=>{g+=y.toString()}),h.on("close",y=>{if(y!==0){T(d,"\u274C",`Generation failed (exit ${y})`),m(JSON.stringify({error:`zibby generate failed with exit code ${y}`,stderr:g.slice(-1e3)}));return}let k=
|
|
305
|
+
`)){let S=Nt(f).trim();S.startsWith("\u2705")?T(d,"\u2705",S.slice(2).trim()):S.startsWith("\u2713")&&T(d,"\u2714",S.slice(2).trim())}}),h.stderr.on("data",y=>{g+=y.toString()}),h.on("close",y=>{if(y!==0){T(d,"\u274C",`Generation failed (exit ${y})`),m(JSON.stringify({error:`zibby generate failed with exit code ${y}`,stderr:g.slice(-1e3)}));return}let k=V(t,c||"test-specs"),f=[];try{let S=e?e.toLowerCase().replace(/[^a-z0-9]+/g,"-"):"";f=he(k).filter(x=>x.endsWith(".txt")&&(!S||x.startsWith(S))).map(x=>U(k,x))}catch{}T(d,"\u2705",`Generated ${f.length} test spec files`),m(JSON.stringify({success:!0,ticketKey:e||null,specFiles:f.map(S=>S.replace(`${t}/`,"")),total:f.length,message:`Generated ${f.length} specs. Now call run_test for each file.`}))}),h.on("error",y=>{T(d,"\u274C",`Spawn error: ${y.message}`),m(JSON.stringify({error:y.message}))})})}var ht=1e5,yt=/^[A-Z][A-Z0-9]+-\d+$/,Ks=new Set(["paragraph","heading","bulletList","orderedList","listItem","blockquote","codeBlock","rule","table","tableRow","tableCell","tableHeader","mediaSingle","panel"]);function Bs(s,t){if(!t||!t.length)return s;let e=s;for(let r of t)r.type==="strong"?e=`**${e}**`:r.type==="em"?e=`_${e}_`:r.type==="code"?e=`\`${e}\``:r.type==="strike"?e=`~~${e}~~`:r.type==="link"&&r.attrs?.href&&(e=`[${e}](${r.attrs.href})`);return e}function me(s,t=0){if(!Array.isArray(s))return"";let e=[];for(let r of s){if(r.type==="text"){e.push(Bs(r.text||"",r.marks));continue}if(r.type==="hardBreak"){e.push(`
|
|
306
306
|
`);continue}if(r.type==="rule"){e.push(`
|
|
307
307
|
---
|
|
308
308
|
`);continue}let n=r.content?me(r.content,t+1):"";if(r.type==="listItem")e.push(n);else if(r.type==="bulletList"){let i=(r.content||[]).map(o=>`- ${me(o.content||[],t+1).trim()}`);e.push(`
|
|
@@ -326,10 +326,10 @@ ${n}
|
|
|
326
326
|
|
|
327
327
|
`).trim();return l?(l.length>ht&&(l=`${l.slice(0,ht)}
|
|
328
328
|
|
|
329
|
-
...[truncated]`),{inlineSpec:`inline:${l}`,issueKey:s}):null}catch{return null}}function Gs(s,t){try{let e=JSON.parse(s);return JSON.stringify({...e,...t})}catch{return s}}async function Ws(s,t,e){let r={...s},n=String(r.spec??"").trim();if(!n)return JSON.stringify({error:"spec is required"});let i=null;if(yt.test(n)&&!n.startsWith("inline:")){let l=await zs(n);l&&(n=l.inlineSpec,r.spec=n,String(r.ticketKey||"").trim()||(r.ticketKey=l.issueKey),i=l.issueKey)}let o=String(r.ticketKey||"").trim();if(o){for(let[l,p]of E.entries())if(p?.ticketKey===o&&!(p?.status!=="running"&&p?.status!=="queued"))return JSON.stringify({runId:l,ticketKey:o,status:p.status,reused:!0,message:`A run for ${o} is already ${p.status}. Reusing existing run instead of starting a duplicate.`})}if(!n.startsWith("inline:")){let l=
|
|
329
|
+
...[truncated]`),{inlineSpec:`inline:${l}`,issueKey:s}):null}catch{return null}}function Gs(s,t){try{let e=JSON.parse(s);return JSON.stringify({...e,...t})}catch{return s}}async function Ws(s,t,e){let r={...s},n=String(r.spec??"").trim();if(!n)return JSON.stringify({error:"spec is required"});let i=null;if(yt.test(n)&&!n.startsWith("inline:")){let l=await zs(n);l&&(n=l.inlineSpec,r.spec=n,String(r.ticketKey||"").trim()||(r.ticketKey=l.issueKey),i=l.issueKey)}let o=String(r.ticketKey||"").trim();if(o){for(let[l,p]of E.entries())if(p?.ticketKey===o&&!(p?.status!=="running"&&p?.status!=="queued"))return JSON.stringify({runId:l,ticketKey:o,status:p.status,reused:!0,message:`A run for ${o} is already ${p.status}. Reusing existing run instead of starting a duplicate.`})}if(!n.startsWith("inline:")){let l=V(t,n);if(!M(l))return yt.test(n)?JSON.stringify({error:`Invalid run_test spec: "${n}" is an issue id, not a spec.`,reason:"Jira auto-load was attempted but did not return usable text, or Jira is not configured.",doNext:["Confirm the jira skill is active and authenticated.",'Or call tracker tools yourself, then run_test with spec: "inline:" + steps.'],validExample:{spec:"inline:1. Open https://example.com \u2026 2. Verify \u2026",ticketKey:n},invalidExample:{spec:n,ticketKey:n}}):JSON.stringify({error:`Test spec not found: ${n}`,hint:'If this should be issue steps, load the issue with your tracker tools first, then run_test with spec: "inline:" + steps. Otherwise use a real file path.'})}let c=wt(e?.options?.config);if(Ie()>=c){let l=vt(),p=r.ticketKey||l,d={runId:l,spec:r.ticketKey?`${r.ticketKey}: ${r.spec}`:r.spec,ticketKey:r.ticketKey||null,status:"queued",startTime:Date.now(),exitCode:null,output:"",error:""};E.set(l,d),z.push({args:{...r,_queuedRunId:l},cwd:t,context:e}),T(p,"\u23F3",`Queued (${Ie()}/${c} running, ${z.length} queued)`);let m={runId:l,spec:d.spec,ticketKey:d.ticketKey,status:"queued",message:`Queued \u2014 will start when a slot opens (max ${c} concurrent).`};return i&&(m.resolvedFromJiraIssue=i,m.message+=` (spec built from Jira ${i})`),JSON.stringify(m)}let a=Date.now()-Oe;a<dt&&Oe>0&&await new Promise(l=>setTimeout(l,dt-a)),Oe=Date.now();let u=It(r,t,e);return i?Gs(u,{resolvedFromJiraIssue:i,message:`Spec was loaded from Jira issue ${i} (description + comments).`}):u}function It(s,t,e){let{spec:r,ticketKey:n,agent:i,headless:o,workflow:c,_queuedRunId:a}=s,u=a||vt(),l=r,p=!1;if(r.startsWith("inline:")){p=!0;let I=Us(t);pt(I,{recursive:!0}),l=U(I,`${u}.txt`),Cs(l,r.slice(7).trim(),"utf-8")}let d=V(t,".zibby","output","runs");pt(d,{recursive:!0});let m=U(d,`${u}.log`),h=Js(m,{flags:"a"}),g=i&&["assistant","cursor","claude","codex","gemini"].includes(i)?i:null,y=["test",l];g&&y.push("--agent",g),o&&y.push("--headless"),c&&y.push("--workflow",c),St&&console.error(`[zibby:spawn] skill=run_test parentPid=${process.pid} \u2192 child zibby ${y.map(I=>/\s/.test(I)?JSON.stringify(I):I).join(" ")} cwd=${t}`);let k=bt("zibby",y,{cwd:t,env:{...process.env,ZIBBY_WORKFLOW_GRAPH_LOG_MARKERS:"1"},stdio:["ignore","pipe","pipe"],detached:!1}),f={runId:u,spec:n?`${n}: ${r}`:r,ticketKey:n||null,specPath:l,logPath:m,isInline:p,pid:k.pid,status:"running",output:"",error:"",startTime:Date.now(),exitCode:null,currentNode:null,completedNodes:[]},S=n||u,x="";function ze(I){let N=Nt(I).trim();if(!N)return;if(N.startsWith("__WORKFLOW_GRAPH_LOG__")){try{let $=JSON.parse(N.slice(22));$.phase==="node_begin"?f.currentNode=$.node:$.phase==="node_end"&&($.node&&!f.completedNodes.includes($.node)&&f.completedNodes.push($.node),f.currentNode===$.node&&(f.currentNode=null))}catch{}return}let te=N.match(/Session\s+(\S+)/);if(te&&!f.sessionId&&(f.sessionId=te[1],f.sessionPath=V(t,$e,Re,f.sessionId)),N.startsWith("\u250C ")||N.startsWith("\u250C ")){let $=N.slice(2).trim();f.currentNode=$,de&&T(S,"\u25B6",`${$}`)}else if(N.startsWith("\u2514 ")||N.startsWith("\u2514 ")){let $=N.slice(2).trim();$.startsWith("done")?(f.currentNode&&!f.completedNodes.includes(f.currentNode)&&f.completedNodes.push(f.currentNode),de&&T(S,"\u2714",`${f.currentNode||"node"} done ${$.replace("done","").trim()}`),f.currentNode=null):$.startsWith("failed")&&(de&&T(S,"\u2718",`${f.currentNode||"node"} failed ${$.replace("failed","").trim()}`),f.currentNode=null)}else N.includes("Workflow completed")&&(f.currentNode=null,de&&T(S,"\u2714",`Workflow completed (${mt(Date.now()-f.startTime)})`))}function yr(I){let N=I.toString();f.output+=N,h.write(N),f.output.length>5e4&&(f.output=f.output.slice(-3e4)),x+=N;let te=x.split(`
|
|
330
330
|
`);x=te.pop();for(let $ of te)ze($)}return k.stdout.on("data",yr),k.stderr.on("data",I=>{let N=I.toString();f.error+=N,h.write(N),f.error.length>2e4&&(f.error=f.error.slice(-1e4))}),k.on("close",I=>{f.status=I===0?"passed":"failed",f.exitCode=I,f.endTime=Date.now(),x&&ze(x),h.end();let N=mt(Date.now()-f.startTime);if(I===0?T(S,"\u2705",`Passed (${N})`):T(S,"\u274C",`Failed (${N})`),f.isInline)try{Ls(f.specPath)}catch{}ft()}),k.on("error",I=>{f.status="error",f.error+=`
|
|
331
|
-
Spawn error: ${I.message}`,T(S,"\u274C",`Spawn error: ${I.message}`),h.end(),ft()}),f._child=k,E.set(u,f),JSON.stringify({runId:u,spec:f.spec,ticketKey:f.ticketKey,status:"running",pid:k.pid,logFile:m})}function gt(s){let t=Math.round(((s.endTime||Date.now())-s.startTime)/1e3),e=s.completedNodes||[],r=s.currentNode||null;if(s.status!=="running")return{elapsed:t,stage:s.status,completedNodes:e,currentNode:null};let n;return r?(n=`Actively executing node "${r}"`,e.length&&(n+=` (completed: ${e.join(", ")})`)):e.length?n=`Between nodes (completed: ${e.join(", ")})`:n="Starting up (initializing workflow)",n+=`. Elapsed: ${t}s. This is normal progress \u2014 do not cancel.`,{elapsed:t,stage:"running",currentNode:r,completedNodes:e,progress:n}}function Hs(s){let{runId:t}=s;if(!t)return JSON.stringify({error:"runId is required"});if(t==="all"){let i=[...E.entries()].map(([l,p])=>{let d=gt(p),m={runId:l,spec:p.spec,ticketKey:p.ticketKey,status:p.status,elapsed:d.elapsed,exitCode:p.exitCode,sessionId:p.sessionId||null};return p.status==="running"?(m.currentNode=d.currentNode,m.completedNodes=d.completedNodes,m.progress=d.progress):m.outputTail=p.output.slice(-500),m}),o=i.filter(l=>l.status==="running").length,c=i.filter(l=>l.status==="passed").length,a=i.filter(l=>l.status==="failed").length,u={total:i.length,running:o,passed:c,failed:a,runs:i};return o>0&&(u._hint="All running tests are progressing normally through their workflow nodes. Do NOT cancel, diagnose, or interpret as stuck. Just tell the user they are still running."),JSON.stringify(u)}let e=E.get(t);if(!e)return JSON.stringify({error:`Run not found: ${t}`});let r=gt(e),n={runId:t,spec:e.spec,ticketKey:e.ticketKey,status:e.status,elapsed:r.elapsed,exitCode:e.exitCode,sessionId:e.sessionId||null};return e.status==="running"?(n.currentNode=r.currentNode,n.completedNodes=r.completedNodes,n.progress=r.progress):(n.outputTail=e.output.slice(-1e3),n.errorTail=e.error.slice(-500)),e.status==="running"&&(n._hint="This run is actively progressing. Do NOT cancel, diagnose, or assume stuck. Just tell the user it is still running."),JSON.stringify(n)}function _t(s,t){if(t.status==="queued"){let e=z.findIndex(r=>r.args._queuedRunId===s);return e>=0&&z.splice(e,1),t.status="cancelled",t.endTime=Date.now(),{ok:!0,runId:s,status:"cancelled"}}if(t.status!=="running")return{ok:!1,runId:s,error:`Run is not active (status: ${t.status})`};try{return t._child.kill("SIGTERM"),t.status="cancelled",t.endTime=Date.now(),{ok:!0,runId:s,status:"cancelled"}}catch(e){return{ok:!1,runId:s,error:`Failed to cancel: ${e.message}`}}}function Ys(s){let{runId:t}=s;if(!t)return JSON.stringify({error:"runId is required"});if(t==="all"){let r=[];for(let[n,i]of E.entries())(i.status==="running"||i.status==="queued")&&r.push(_t(n,i));return r.length===0?JSON.stringify({ok:!0,message:"No active runs to cancel"}):JSON.stringify({ok:!0,cancelled:r.length,results:r})}let e=E.get(t);return JSON.stringify(e?_t(t,e):{error:`Run not found: ${t}`})}function
|
|
332
|
-
${e||""}`.toLowerCase(),i={runId:s?.runId||null,status:s?.status||null,exitCode:s?.exitCode??null,likelyCause:"Unknown failure",confidence:"low",nextStep:'Call run_artifacts({ runId, type: "log" }) with larger tail and inspect full logs.'};return s?.status==="running"||s?.status==="queued"?{...i,likelyCause:"Run is still active; no terminal failure to diagnose yet.",confidence:"high",nextStep:'Call run_status({ runId: "all" }) to check progress.'}:n.includes("test spec not found")?{...i,likelyCause:"Invalid spec input: run_test received a non-existent spec path.",confidence:"high",nextStep:"Use spec as inline:... or a real file path from list_specs. For ticket keys, fetch steps first via Jira then build inline spec."}:n.includes("unknown command")&&n.includes("'run'")?{...i,likelyCause:"CLI command mismatch (`zibby run` unsupported in current CLI).",confidence:"high",nextStep:"Use `zibby test ...` spawn path (runner should already do this)."}:n.includes("missing openai_api_key")||n.includes("didn't provide an api key")||n.includes("401")?{...i,likelyCause:"Provider authentication/config issue (API key/proxy auth missing or rejected).",confidence:"medium",nextStep:"Verify proxy/token env and auth mode, then retry once configuration is valid."}:n.includes("spawn error")||n.includes("enoent")?{...i,likelyCause:"Failed to spawn CLI process (binary/path/environment issue).",confidence:"medium",nextStep:"Confirm `zibby` is installed and available in PATH for the chat process."}:n.includes("security command failed")||n.includes("security process exited with code: 45")||n.includes("password not found for account")?{...i,likelyCause:"Cursor agent keychain/auth failed during preflight (often transient, more common under parallel starts).",confidence:"high",nextStep:'Retry failed ticket sequentially (not parallel), or run with a different agent via run_test({ ..., agent: "codex" }).'}:i}function Qs(s,t){let{runId:e,type:r,node:n="execute_live",query:i,tail:o=3e3}=s;if(r==="search"){if(!i)return JSON.stringify({error:'query is required for type="search"'});let a=
|
|
331
|
+
Spawn error: ${I.message}`,T(S,"\u274C",`Spawn error: ${I.message}`),h.end(),ft()}),f._child=k,E.set(u,f),JSON.stringify({runId:u,spec:f.spec,ticketKey:f.ticketKey,status:"running",pid:k.pid,logFile:m})}function gt(s){let t=Math.round(((s.endTime||Date.now())-s.startTime)/1e3),e=s.completedNodes||[],r=s.currentNode||null;if(s.status!=="running")return{elapsed:t,stage:s.status,completedNodes:e,currentNode:null};let n;return r?(n=`Actively executing node "${r}"`,e.length&&(n+=` (completed: ${e.join(", ")})`)):e.length?n=`Between nodes (completed: ${e.join(", ")})`:n="Starting up (initializing workflow)",n+=`. Elapsed: ${t}s. This is normal progress \u2014 do not cancel.`,{elapsed:t,stage:"running",currentNode:r,completedNodes:e,progress:n}}function Hs(s){let{runId:t}=s;if(!t)return JSON.stringify({error:"runId is required"});if(t==="all"){let i=[...E.entries()].map(([l,p])=>{let d=gt(p),m={runId:l,spec:p.spec,ticketKey:p.ticketKey,status:p.status,elapsed:d.elapsed,exitCode:p.exitCode,sessionId:p.sessionId||null};return p.status==="running"?(m.currentNode=d.currentNode,m.completedNodes=d.completedNodes,m.progress=d.progress):m.outputTail=p.output.slice(-500),m}),o=i.filter(l=>l.status==="running").length,c=i.filter(l=>l.status==="passed").length,a=i.filter(l=>l.status==="failed").length,u={total:i.length,running:o,passed:c,failed:a,runs:i};return o>0&&(u._hint="All running tests are progressing normally through their workflow nodes. Do NOT cancel, diagnose, or interpret as stuck. Just tell the user they are still running."),JSON.stringify(u)}let e=E.get(t);if(!e)return JSON.stringify({error:`Run not found: ${t}`});let r=gt(e),n={runId:t,spec:e.spec,ticketKey:e.ticketKey,status:e.status,elapsed:r.elapsed,exitCode:e.exitCode,sessionId:e.sessionId||null};return e.status==="running"?(n.currentNode=r.currentNode,n.completedNodes=r.completedNodes,n.progress=r.progress):(n.outputTail=e.output.slice(-1e3),n.errorTail=e.error.slice(-500)),e.status==="running"&&(n._hint="This run is actively progressing. Do NOT cancel, diagnose, or assume stuck. Just tell the user it is still running."),JSON.stringify(n)}function _t(s,t){if(t.status==="queued"){let e=z.findIndex(r=>r.args._queuedRunId===s);return e>=0&&z.splice(e,1),t.status="cancelled",t.endTime=Date.now(),{ok:!0,runId:s,status:"cancelled"}}if(t.status!=="running")return{ok:!1,runId:s,error:`Run is not active (status: ${t.status})`};try{return t._child.kill("SIGTERM"),t.status="cancelled",t.endTime=Date.now(),{ok:!0,runId:s,status:"cancelled"}}catch(e){return{ok:!1,runId:s,error:`Failed to cancel: ${e.message}`}}}function Ys(s){let{runId:t}=s;if(!t)return JSON.stringify({error:"runId is required"});if(t==="all"){let r=[];for(let[n,i]of E.entries())(i.status==="running"||i.status==="queued")&&r.push(_t(n,i));return r.length===0?JSON.stringify({ok:!0,message:"No active runs to cancel"}):JSON.stringify({ok:!0,cancelled:r.length,results:r})}let e=E.get(t);return JSON.stringify(e?_t(t,e):{error:`Run not found: ${t}`})}function Vs(s,t){let e=E.get(s);if(e?.sessionPath&&M(e.sessionPath))return e.sessionPath;if(e?.sessionId){let r=V(t,$e,Re,e.sessionId);if(M(r))return r}return null}function Rt(s,t=""){let e=[];if(!M(s))return e;for(let r of he(s,{withFileTypes:!0})){let n=t?`${t}/${r.name}`:r.name;if(r.isDirectory())e.push(...Rt(U(s,r.name),n));else{let i=Ds(U(s,r.name));e.push({path:n,size:i.size})}}return e}function kt(s){if(!M(s))return null;try{return JSON.parse(fe(s,"utf-8"))}catch{return null}}function $t(s,t=2e3){if(!s||!M(s))return"";try{return fe(s,"utf-8").slice(-Math.max(200,Number(t)||2e3))}catch{return""}}function Zs({run:s,logTail:t,errorTail:e}){let n=`${t||""}
|
|
332
|
+
${e||""}`.toLowerCase(),i={runId:s?.runId||null,status:s?.status||null,exitCode:s?.exitCode??null,likelyCause:"Unknown failure",confidence:"low",nextStep:'Call run_artifacts({ runId, type: "log" }) with larger tail and inspect full logs.'};return s?.status==="running"||s?.status==="queued"?{...i,likelyCause:"Run is still active; no terminal failure to diagnose yet.",confidence:"high",nextStep:'Call run_status({ runId: "all" }) to check progress.'}:n.includes("test spec not found")?{...i,likelyCause:"Invalid spec input: run_test received a non-existent spec path.",confidence:"high",nextStep:"Use spec as inline:... or a real file path from list_specs. For ticket keys, fetch steps first via Jira then build inline spec."}:n.includes("unknown command")&&n.includes("'run'")?{...i,likelyCause:"CLI command mismatch (`zibby run` unsupported in current CLI).",confidence:"high",nextStep:"Use `zibby test ...` spawn path (runner should already do this)."}:n.includes("missing openai_api_key")||n.includes("didn't provide an api key")||n.includes("401")?{...i,likelyCause:"Provider authentication/config issue (API key/proxy auth missing or rejected).",confidence:"medium",nextStep:"Verify proxy/token env and auth mode, then retry once configuration is valid."}:n.includes("spawn error")||n.includes("enoent")?{...i,likelyCause:"Failed to spawn CLI process (binary/path/environment issue).",confidence:"medium",nextStep:"Confirm `zibby` is installed and available in PATH for the chat process."}:n.includes("security command failed")||n.includes("security process exited with code: 45")||n.includes("password not found for account")?{...i,likelyCause:"Cursor agent keychain/auth failed during preflight (often transient, more common under parallel starts).",confidence:"high",nextStep:'Retry failed ticket sequentially (not parallel), or run with a different agent via run_test({ ..., agent: "codex" }).'}:i}function Qs(s,t){let{runId:e,type:r,node:n="execute_live",query:i,tail:o=3e3}=s;if(r==="search"){if(!i)return JSON.stringify({error:'query is required for type="search"'});let a=V(t,$e,Re);if(!M(a))return JSON.stringify({matches:[],message:"No sessions found"});let u=[],l=i.toLowerCase();for(let p of he(a,{withFileTypes:!0})){if(!p.isDirectory())continue;let d=U(a,p.name),m=[{file:"execute_live/result.json",label:"result"},{file:"execute_live/events.json",label:"events"},{file:"execute_live/raw_stream_output.txt",label:"log"},{file:"generate_script/raw_stream_output.txt",label:"script_log"},{file:"title.txt",label:"title"}];for(let{file:h,label:_}of m){let g=U(d,h);if(M(g))try{let y=fe(g,"utf-8");if(y.toLowerCase().includes(l)){let k=y.toLowerCase().indexOf(l),f=Math.max(0,k-100),S=Math.min(y.length,k+i.length+100);u.push({sessionId:p.name,artifact:_,snippet:y.slice(f,S)})}}catch{}}if(u.length>=20)break}return JSON.stringify({query:i,matches:u,total:u.length})}if(!e)return JSON.stringify({error:"runId is required for this type"});if(r==="log"){let a=E.get(e),u=$t(a?.logPath,o);if(u)return JSON.stringify({runId:e,source:"run-log",totalLength:u.length,tail:u})}let c=Vs(e,t);if(!c)return JSON.stringify({error:`No session found for run ${e}. The run may still be starting.`});switch(r){case"list":{let a=Rt(c);return JSON.stringify({sessionId:c.split("/").pop(),files:a,total:a.length})}case"result":{let a=kt(U(c,n,"result.json"));return JSON.stringify(a?{sessionId:c.split("/").pop(),node:n,result:a}:{error:`No result.json found in ${n}`})}case"events":{let a=kt(U(c,n,"events.json"));if(!a)return JSON.stringify({error:`No events.json found in ${n}`});let u=Array.isArray(a)?a:a.events||[];return JSON.stringify({sessionId:c.split("/").pop(),node:n,totalEvents:u.length,events:u.slice(-50)})}case"log":{let a=U(c,n,"raw_stream_output.txt");if(!M(a))return JSON.stringify({error:`No log found in ${n}`});let u=fe(a,"utf-8");return JSON.stringify({sessionId:c.split("/").pop(),node:n,totalLength:u.length,tail:u.slice(-o)})}default:return JSON.stringify({error:`Unknown artifact type: ${r}. Use: list, result, events, log, search`})}}function Xs(s,t){let e=String(s?.runId||"all"),r=Number(s?.tail||2e3),n=e==="all"?[...E.keys()]:[e];if(n.length===0)return JSON.stringify({error:"No runs available to diagnose. Call run_test first."});let i=n.map(a=>{let u=E.get(a);if(!u)return{runId:a,error:`Run not found: ${a}`};let l=$t(u.logPath,r),p=String(u.error||"").slice(-Math.max(200,r));return{...Zs({run:u,logTail:l,errorTail:p}),ticketKey:u.ticketKey||null,spec:u.spec,logTail:l,errorTail:p}}),o=i.filter(a=>a.status==="failed"||a.status==="error"),c=i.filter(a=>a.status==="running"||a.status==="queued");return JSON.stringify({total:i.length,failed:o.length,active:c.length,diagnoses:i})}function en(s,t){let e=s?.directory||"test-specs",r=V(t,e);if(!M(r))return JSON.stringify({specs:[],directory:e,message:`Directory not found: ${e}`});try{let i=function(o,c){for(let a of he(o,{withFileTypes:!0})){let u=c?`${c}/${a.name}`:a.name;a.isDirectory()?i(U(o,a.name),u):(a.name.endsWith(".txt")||a.name.endsWith(".md"))&&n.push(u)}},n=[];return i(r,""),JSON.stringify({specs:n.map(o=>`${e}/${o}`),total:n.length,directory:e})}catch(n){return JSON.stringify({error:n.message})}}import{spawn as tn}from"child_process";import{existsSync as q,mkdirSync as rn,readdirSync as jt,statSync as sn,readFileSync as nn}from"fs";import{resolve as xe,join as K,basename as on}from"path";var Ae=".zibby/repos";function ye(s,t,e={}){return new Promise((r,n)=>{let i=tn(s,{cwd:t,shell:!0,env:{...process.env,GIT_TERMINAL_PROMPT:"0",...e}}),o="",c="";i.stdout.on("data",a=>{o+=a.toString()}),i.stderr.on("data",a=>{c+=a.toString()}),i.on("close",a=>{a!==0?n(new Error(`Exit ${a}: ${c.trim()||o.trim()}`)):r(o.trim())}),i.on("error",a=>n(a))})}var xt={id:"git",description:"Clone and manage git repositories for codebase analysis",envKeys:["GITHUB_TOKEN","GITLAB_TOKEN"],promptFragment:`## Git Repositories
|
|
333
333
|
You can clone and explore git repositories locally for codebase analysis:
|
|
334
334
|
- git_checkout: Clone a repo (or pull if already cloned). Supports GitHub and GitLab with auto-auth.
|
|
335
335
|
- git_list_repos: List locally cloned repos
|
|
@@ -340,7 +340,7 @@ When a test ticket lacks context, use this workflow:
|
|
|
340
340
|
2. Use git_explore to understand the project structure
|
|
341
341
|
3. Use shell commands (grep, cat) to read specific files for deeper understanding
|
|
342
342
|
4. Use GitHub/GitLab skills to read related PRs and commits
|
|
343
|
-
5. Build well-informed test specs and save them to files before running tests`,resolve(){return null},async handleToolCall(s,t,e){let r=e?.options?.workspace||process.cwd();try{switch(s){case"git_checkout":return await an(t,r);case"git_list_repos":return cn(t,r);case"git_explore":return ln(t,r);default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(n){return JSON.stringify({error:n.message})}},tools:[{name:"git_checkout",description:"Clone a git repository locally (or pull latest if already cloned). Auto-authenticates with GitHub/GitLab tokens if available.",input_schema:{type:"object",properties:{url:{type:"string",description:'Repository URL (e.g. "https://github.com/org/repo" or "org/repo" shorthand for GitHub)'},branch:{type:"string",description:"Branch to checkout (default: repo default branch)"},shallow:{type:"boolean",description:"Shallow clone with depth 1 (default: true, faster)"},name:{type:"string",description:"Local directory name override (default: repo name from URL)"}},required:["url"]}},{name:"git_list_repos",description:"List locally cloned repositories",input_schema:{type:"object",properties:{}}},{name:"git_explore",description:"Quick structural overview of a cloned repo: key files, package.json info, directory tree (top 2 levels), detected framework/language",input_schema:{type:"object",properties:{repo:{type:"string",description:"Repo name (as listed by git_list_repos)"},depth:{type:"number",description:"Directory tree depth (default: 2)"}},required:["repo"]}}]};async function an(s,t){let{url:e,branch:r,shallow:n=!0,name:i}=s;!e.includes("://")&&!e.startsWith("git@")&&(e=`https://github.com/${e}`);let o=i||on(e.replace(/\.git$/,"")),c=xe(t,Ae);rn(c,{recursive:!0});let a=
|
|
343
|
+
5. Build well-informed test specs and save them to files before running tests`,resolve(){return null},async handleToolCall(s,t,e){let r=e?.options?.workspace||process.cwd();try{switch(s){case"git_checkout":return await an(t,r);case"git_list_repos":return cn(t,r);case"git_explore":return ln(t,r);default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(n){return JSON.stringify({error:n.message})}},tools:[{name:"git_checkout",description:"Clone a git repository locally (or pull latest if already cloned). Auto-authenticates with GitHub/GitLab tokens if available.",input_schema:{type:"object",properties:{url:{type:"string",description:'Repository URL (e.g. "https://github.com/org/repo" or "org/repo" shorthand for GitHub)'},branch:{type:"string",description:"Branch to checkout (default: repo default branch)"},shallow:{type:"boolean",description:"Shallow clone with depth 1 (default: true, faster)"},name:{type:"string",description:"Local directory name override (default: repo name from URL)"}},required:["url"]}},{name:"git_list_repos",description:"List locally cloned repositories",input_schema:{type:"object",properties:{}}},{name:"git_explore",description:"Quick structural overview of a cloned repo: key files, package.json info, directory tree (top 2 levels), detected framework/language",input_schema:{type:"object",properties:{repo:{type:"string",description:"Repo name (as listed by git_list_repos)"},depth:{type:"number",description:"Directory tree depth (default: 2)"}},required:["repo"]}}]};async function an(s,t){let{url:e,branch:r,shallow:n=!0,name:i}=s;!e.includes("://")&&!e.startsWith("git@")&&(e=`https://github.com/${e}`);let o=i||on(e.replace(/\.git$/,"")),c=xe(t,Ae);rn(c,{recursive:!0});let a=K(c,o),u=e,l=process.env.GITHUB_TOKEN,p=process.env.GITLAB_TOKEN,d=process.env.GITLAB_URL;if(e.includes("github.com")&&l)u=e.replace("https://github.com",`https://x-access-token:${l}@github.com`);else if(p&&d)try{let _=new URL(d).host;e.includes(_)&&(u=e.replace(`https://${_}`,`https://oauth2:${p}@${_}`))}catch{}if(q(K(a,".git"))){let _=r?`git -C "${a}" fetch origin ${r} && git -C "${a}" checkout ${r} && git -C "${a}" pull origin ${r}`:`git -C "${a}" pull`;await ye(_,t);let g=await ye(`git -C "${a}" log -1 --format="%h %s"`,t);return JSON.stringify({action:"updated",repo:o,path:a,branch:r||"default",head:g})}let m=["git","clone"];n&&m.push("--depth","1"),r&&m.push("--branch",r),m.push(`"${u}"`,`"${a}"`),await ye(m.join(" "),t);let h=await ye(`git -C "${a}" log -1 --format="%h %s"`,t);return JSON.stringify({action:"cloned",repo:o,path:a,branch:r||"default",shallow:n,head:h})}function cn(s,t){let e=xe(t,Ae);if(!q(e))return JSON.stringify({repos:[],message:"No repos cloned yet"});let r=[];for(let n of jt(e,{withFileTypes:!0})){if(!n.isDirectory())continue;let i=K(e,n.name);if(!q(K(i,".git")))continue;let o=sn(i);r.push({name:n.name,path:i,lastModified:o.mtime.toISOString()})}return JSON.stringify({repos:r,total:r.length,directory:e})}function ln(s,t){let{repo:e,depth:r=2}=s,n=xe(t,Ae,e);if(!q(n))return JSON.stringify({error:`Repo not found: ${e}. Run git_checkout first.`});let i={repo:e,path:n},o=K(n,"package.json");if(q(o))try{let m=JSON.parse(nn(o,"utf-8"));i.packageJson={name:m.name,version:m.version,scripts:m.scripts?Object.keys(m.scripts):[],dependencies:m.dependencies?Object.keys(m.dependencies).slice(0,30):[],devDependencies:m.devDependencies?Object.keys(m.devDependencies).slice(0,20):[]},m.dependencies?.react?i.framework="React":m.dependencies?.next?i.framework="Next.js":m.dependencies?.vue?i.framework="Vue":m.dependencies?.angular?i.framework="Angular":m.dependencies?.express?i.framework="Express":m.dependencies?.fastify&&(i.framework="Fastify")}catch{}let c=K(n,"pyproject.toml");q(c)&&(i.language="Python");let a=K(n,"go.mod");q(a)&&(i.language="Go");let u=K(n,"Cargo.toml");q(u)&&(i.language="Rust"),q(o)&&(i.language=i.language||"JavaScript/TypeScript");let l=[];function p(m,h,_){if(_>r)return;let g;try{g=jt(m,{withFileTypes:!0})}catch{return}let y=g.filter(k=>!k.name.startsWith(".")&&k.name!=="node_modules"&&k.name!=="__pycache__"&&k.name!=="dist"&&k.name!=="build"&&k.name!==".git").sort((k,f)=>k.isDirectory()!==f.isDirectory()?k.isDirectory()?-1:1:k.name.localeCompare(f.name));for(let k of y){let f=k.isDirectory();l.push(`${h}${f?"\u{1F4C1}":"\u{1F4C4}"} ${k.name}`),f&&_<r&&p(K(m,k.name),`${h} `,_+1)}}p(n,"",1),i.tree=l.slice(0,80),l.length>80&&(i.treeTruncated=!0);let d=["README.md","README.rst","src/App.tsx","src/App.jsx","src/App.js","src/routes.tsx","src/routes.js","app/routes.tsx","app/routes.js","src/index.tsx","src/index.ts","src/main.tsx","src/main.ts","pages/_app.tsx","pages/_app.js","app/layout.tsx","docker-compose.yml","Dockerfile",".env.example"];return i.keyFilesFound=d.filter(m=>q(K(n,m))),JSON.stringify(i)}import{execFileSync as Ut}from"child_process";import{existsSync as Pt,mkdirSync as un}from"fs";import{join as se,basename as pn}from"path";import{randomBytes as dn}from"crypto";import{pathToFileURL as ke}from"url";import{createRequire as Mt}from"module";var At=".zibby/memory",qt="dolt",Kt={encoding:"utf-8",stdio:["pipe","pipe","pipe"],timeout:15e3},Ce="dolt",Te=new Map,ge=new Map,mn=Mt(import.meta.url),we=()=>dn(8).toString("hex"),Z=()=>new Date().toISOString(),fn=[`CREATE TABLE IF NOT EXISTS chat_memory (
|
|
344
344
|
id VARCHAR(64) PRIMARY KEY,
|
|
345
345
|
memory_key VARCHAR(160),
|
|
346
346
|
category VARCHAR(32) NOT NULL,
|
|
@@ -374,7 +374,7 @@ When a test ticket lacks context, use this workflow:
|
|
|
374
374
|
)`],Tt=new Set;function C(s,t){return Ut(qt,t,{...Kt,cwd:s})}function G(s,t){try{let e=C(s,["sql","-q",t,"-r","json"]);return JSON.parse(e.trim()).rows||[]}catch{return[]}}function L(s,t){C(s,["sql","-q",t])}function Et(s){if(Tt.has(s))return!0;if(!Pt(se(s,".dolt"))){if(!hn())return!1;un(s,{recursive:!0}),C(s,["init","--name","Zibby Chat Memory","--email","chat@zibby.app"])}let t=`${fn.join(`;
|
|
375
375
|
`)};`;L(s,t);try{L(s,"ALTER TABLE chat_memory ADD COLUMN tier VARCHAR(16) DEFAULT 'mid'")}catch{}try{L(s,"ALTER TABLE chat_memory ADD COLUMN memory_key VARCHAR(160)")}catch{}return Tt.add(s),!0}function hn(){try{return Ut(qt,["version"],{...Kt,timeout:5e3}),!0}catch{return!1}}function w(s){return s==null?"NULL":`'${String(s).replace(/'/g,"''")}'`}function Le(s){return String(s||"").toLowerCase().replace(/[“”]/g,'"').replace(/[‘’]/g,"'").replace(/[\s_-]+/g," ").replace(/[^\w\s"']/g,"").replace(/\s+/g," ").trim()}function X(s){return s==="long"?3:s==="mid"?2:s==="short"?1:0}function Je(s,t){let e=["short","mid","long"].includes(s)?s:"mid";return new Set(["fact","decision","preference","credential","url","workaround"]).has(String(t||"").toLowerCase())&&e==="short"?"mid":e}function Bt(s){let t=new Map;for(let e of s||[]){let r=Le(e.content),n=e.memory_key?`key:${e.memory_key}`:r?`norm:${r}`:"";if(!n)continue;let i=t.get(n);if(!i){t.set(n,e);continue}let o=X(i.tier),c=X(e.tier);if(c>o){t.set(n,e);continue}c===o&&Number(e.relevance||0)>Number(i.relevance||0)&&t.set(n,e)}return[...t.values()]}function be(s,t){let e=String(s??"");return e.length<=t?e:t<=1?e.slice(0,t):`${e.slice(0,t-1)}\u2026`}function _e(s,t){let e={recentSessions:Array.isArray(s?.recentSessions)?s.recentSessions:[],topMemories:Array.isArray(s?.topMemories)?s.topMemories:[],taskStats:Array.isArray(s?.taskStats)?s.taskStats:[],ticketFilter:s?.ticketFilter||null,backend:t||String(s?.backend||Ce),error:s?.error||null};return e.backend==="mem0"?{...e,recentSessions:[],taskStats:[]}:e}function yn(s){let t=[];if(s.recentSessions?.length>0){t.push("Recent sessions:");for(let e of s.recentSessions.slice(0,3))e?.summary?.trim()&&t.push(`- ${be(e.summary,150)}${e.tickets?` [${e.tickets}]`:""}`)}if(s.topMemories?.length>0){t.push("Known facts:");for(let e of s.topMemories.slice(0,10)){let r=e.tier==="long"?"\u2605":"\xB7";t.push(`${r} [${e.category}] ${be(e.content,120)}`)}}return t.length===0?"":`## Memory Context
|
|
376
376
|
${t.join(`
|
|
377
|
-
`)}`}function Ee(s){return{backend:s.backend,recentSessions:s.recentSessions.slice(0,3).map(t=>({summary:be(String(t?.summary||""),160),tickets:t?.tickets||null,created_at:t?.created_at||null})),topMemories:s.topMemories.slice(0,8).map(t=>({category:t?.category||null,tier:t?.tier||null,content:be(String(t?.content||""),140),source:t?.source||null})),taskStats:s.taskStats,error:s.error||null}}async function Ct(s,t){let e=String(process.env.ZIBBY_MEMORY_BACKEND||"").trim().toLowerCase();if(e==="mem0"||e==="dolt")return e;let r=String(t?.options?.memoryBackend||t?.options?.config?.memory?.backend||"").trim().toLowerCase();if(r==="mem0"||r==="dolt")return r;if(ge.has(s))return ge.get(s);try{let n=se(s,".zibby.config.mjs");if(Pt(n)){let i=await import(ke(n).href),o=String(i?.default?.memory?.backend||"").trim().toLowerCase();if(o==="mem0"||o==="dolt")return ge.set(s,o),o}}catch{}return ge.set(s,Ce),Ce}function Ft(s){let t=String(process.env.ZIBBY_MEMORY_USER_ID||"").trim();return t||`workspace:${pn(s||process.cwd())}`}function gn(){let s=String(process.env.ZIBBY_MEM0_OPENAI_BASE_URL||"").trim();if(!s)return null;let t=String(process.env.ZIBBY_MEM0_API_KEY||process.env.ZIBBY_USER_TOKEN||process.env.OPENAI_API_KEY||"").trim(),e=String(process.env.ZIBBY_MEM0_LLM_MODEL||"gpt-4.1-mini").trim(),r=String(process.env.ZIBBY_MEM0_EMBEDDER_MODEL||"text-embedding-3-small").trim(),n=Number(process.env.ZIBBY_MEM0_EMBEDDING_DIMS||1536);return{llm:{provider:"openai",config:{model:e,baseURL:s,...t?{apiKey:t}:{}}},embedder:{provider:"openai",config:{model:r,embeddingDims:n,baseURL:s,...t?{apiKey:t}:{}}},vectorStore:{provider:"memory",config:{dimension:n}}}}async function zt(s){let t=s||process.cwd();if(Te.has(t))return Te.get(t);let e;try{let c=Mt(ke(se(t,"package.json")).href).resolve("mem0ai/oss");e=await import(ke(c).href)}catch{try{let o=mn.resolve("mem0ai/oss");e=await import(ke(o).href)}catch(o){throw new Error(`Cannot find package 'mem0ai' for workspace "${t}". Install in that project: npm install mem0ai. (${o.message})`,{cause:o})}}let r=e?.Memory;if(!r)throw new Error("mem0ai/oss does not export Memory");let n=gn(),i=n?new r(n):new r;return Te.set(t,i),i}function Lt(s,t="mid"){return(Array.isArray(s)?s:Array.isArray(s?.results)?s.results:[]).map(r=>({id:r?.id||we(),memory_key:r?.metadata?.memoryKey||r?.metadata?.memory_key||null,category:r?.metadata?.category||"fact",content:r?.memory||r?.content||"",source:r?.metadata?.source||"mem0",ticket_key:r?.metadata?.ticketKey||r?.metadata?.ticket_key||null,tier:Je(r?.metadata?.tier||t,r?.metadata?.category||"fact"),relevance:Number(r?.score??r?.metadata?.relevance??.8),created_at:r?.created_at||r?.metadata?.created_at||
|
|
377
|
+
`)}`}function Ee(s){return{backend:s.backend,recentSessions:s.recentSessions.slice(0,3).map(t=>({summary:be(String(t?.summary||""),160),tickets:t?.tickets||null,created_at:t?.created_at||null})),topMemories:s.topMemories.slice(0,8).map(t=>({category:t?.category||null,tier:t?.tier||null,content:be(String(t?.content||""),140),source:t?.source||null})),taskStats:s.taskStats,error:s.error||null}}async function Ct(s,t){let e=String(process.env.ZIBBY_MEMORY_BACKEND||"").trim().toLowerCase();if(e==="mem0"||e==="dolt")return e;let r=String(t?.options?.memoryBackend||t?.options?.config?.memory?.backend||"").trim().toLowerCase();if(r==="mem0"||r==="dolt")return r;if(ge.has(s))return ge.get(s);try{let n=se(s,".zibby.config.mjs");if(Pt(n)){let i=await import(ke(n).href),o=String(i?.default?.memory?.backend||"").trim().toLowerCase();if(o==="mem0"||o==="dolt")return ge.set(s,o),o}}catch{}return ge.set(s,Ce),Ce}function Ft(s){let t=String(process.env.ZIBBY_MEMORY_USER_ID||"").trim();return t||`workspace:${pn(s||process.cwd())}`}function gn(){let s=String(process.env.ZIBBY_MEM0_OPENAI_BASE_URL||"").trim();if(!s)return null;let t=String(process.env.ZIBBY_MEM0_API_KEY||process.env.ZIBBY_USER_TOKEN||process.env.OPENAI_API_KEY||"").trim(),e=String(process.env.ZIBBY_MEM0_LLM_MODEL||"gpt-4.1-mini").trim(),r=String(process.env.ZIBBY_MEM0_EMBEDDER_MODEL||"text-embedding-3-small").trim(),n=Number(process.env.ZIBBY_MEM0_EMBEDDING_DIMS||1536);return{llm:{provider:"openai",config:{model:e,baseURL:s,...t?{apiKey:t}:{}}},embedder:{provider:"openai",config:{model:r,embeddingDims:n,baseURL:s,...t?{apiKey:t}:{}}},vectorStore:{provider:"memory",config:{dimension:n}}}}async function zt(s){let t=s||process.cwd();if(Te.has(t))return Te.get(t);let e;try{let c=Mt(ke(se(t,"package.json")).href).resolve("mem0ai/oss");e=await import(ke(c).href)}catch{try{let o=mn.resolve("mem0ai/oss");e=await import(ke(o).href)}catch(o){throw new Error(`Cannot find package 'mem0ai' for workspace "${t}". Install in that project: npm install mem0ai. (${o.message})`,{cause:o})}}let r=e?.Memory;if(!r)throw new Error("mem0ai/oss does not export Memory");let n=gn(),i=n?new r(n):new r;return Te.set(t,i),i}function Lt(s,t="mid"){return(Array.isArray(s)?s:Array.isArray(s?.results)?s.results:[]).map(r=>({id:r?.id||we(),memory_key:r?.metadata?.memoryKey||r?.metadata?.memory_key||null,category:r?.metadata?.category||"fact",content:r?.memory||r?.content||"",source:r?.metadata?.source||"mem0",ticket_key:r?.metadata?.ticketKey||r?.metadata?.ticket_key||null,tier:Je(r?.metadata?.tier||t,r?.metadata?.category||"fact"),relevance:Number(r?.score??r?.metadata?.relevance??.8),created_at:r?.created_at||r?.metadata?.created_at||Z()})).filter(r=>String(r.content||"").trim().length>0)}var Gt={id:"chat-memory",description:"Persistent chat memory and task history (Dolt-backed)",envKeys:[],promptFragment:`## Chat Memory (persistent)
|
|
378
378
|
You have persistent memory across sessions. Use it to avoid losing context:
|
|
379
379
|
- **memory_store**: Save important facts, decisions, or context. Anything worth remembering.
|
|
380
380
|
- **memory_recall**: Search your memory by keyword or category. Use this at the START of conversations to recall relevant context.
|
|
@@ -401,7 +401,7 @@ fact, decision, context, insight, credential, url, error, workaround`,resolve(){
|
|
|
401
401
|
ticket_key = ${w(i)},
|
|
402
402
|
tier = ${w(S)},
|
|
403
403
|
relevance = ${x},
|
|
404
|
-
created_at = ${w(
|
|
404
|
+
created_at = ${w(Z())}
|
|
405
405
|
WHERE id = ${w(y.id)}`);try{C(t,["add","."]),C(t,["commit","-m",`memory upsert: ${r} \u2014 ${String(e).slice(0,60)}`])}catch{}return JSON.stringify({ok:!0,id:y.id,category:r,tier:S,memoryKey:p,upserted:!0})}}let m=G(t,`SELECT id, content, tier, relevance
|
|
406
406
|
FROM chat_memory
|
|
407
407
|
WHERE category = ${w(r)}
|
|
@@ -410,7 +410,7 @@ fact, decision, context, insight, credential, url, error, workaround`,resolve(){
|
|
|
410
410
|
SET tier = ${w(k?u:g)},
|
|
411
411
|
relevance = ${Math.max(l,y)}
|
|
412
412
|
WHERE id = ${w(m.id)}`);try{C(t,["add","."]),C(t,["commit","-m",`memory promote: ${r} \u2014 ${String(e).slice(0,60)}`])}catch{}return JSON.stringify({ok:!0,id:m.id,category:r,tier:k?u:g,deduped:!0,promoted:!0})}return JSON.stringify({ok:!0,id:m.id,category:r,tier:g,deduped:!0,promoted:!1})}let h=we(),_=process.env.ZIBBY_CHAT_SESSION_ID||null;L(t,`INSERT INTO chat_memory (id, memory_key, category, content, source, ticket_key, session_id, tier, relevance, created_at)
|
|
413
|
-
VALUES (${w(h)}, ${w(p||null)}, ${w(r)}, ${w(e)}, ${w(n)}, ${w(i)}, ${w(_)}, ${w(u)}, ${l}, ${w(
|
|
413
|
+
VALUES (${w(h)}, ${w(p||null)}, ${w(r)}, ${w(e)}, ${w(n)}, ${w(i)}, ${w(_)}, ${w(u)}, ${l}, ${w(Z())})`);try{C(t,["add","."]),C(t,["commit","-m",`memory: ${r} \u2014 ${e.slice(0,60)}`])}catch{}return JSON.stringify({ok:!0,id:h,category:r,tier:u,memoryKey:p||null,stored:e.slice(0,100)})}async function _n(s,t,e){let{content:r,category:n,source:i,ticketKey:o,tier:c,memoryKey:a}=s;if(!r||!n)return JSON.stringify({error:"content and category are required"});try{let u=await zt(e),l=Ft(e),p=Je(c,n);return await u.add([{role:"user",content:String(r)}],{userId:l,metadata:{memoryKey:a||null,category:n,tier:p,source:i||"zibby-chat",ticketKey:o||null,created_at:Z()}}),JSON.stringify({ok:!0,backend:"mem0",userId:l,category:n,tier:p,memoryKey:a||null,stored:String(r).slice(0,100)})}catch(u){throw new Error(`mem0 store failed: ${u.message}. If mem0 is not installed, run: npm install mem0ai`,{cause:u})}}function kn(s,t){let{query:e,category:r,ticketKey:n,tier:i,limit:o=20}=s,c=[];e&&c.push(`content LIKE ${w(`%${e}%`)}`),r&&c.push(`category = ${w(r)}`),n&&c.push(`ticket_key = ${w(n)}`),i&&c.push(`tier = ${w(i)}`);let u=`SELECT id, memory_key, category, content, source, ticket_key, tier, relevance, created_at
|
|
414
414
|
FROM chat_memory ${c.length>0?`WHERE ${c.join(" AND ")}`:""}
|
|
415
415
|
ORDER BY relevance DESC, created_at DESC
|
|
416
416
|
LIMIT ${o}`,l=G(t,u);return JSON.stringify({total:l.length,memories:l})}async function Ht(s,t,e){let{query:r,category:n,ticketKey:i,tier:o,limit:c=20}=s;try{let a=await zt(e),u=Ft(e),l=[];if(r&&String(r).trim()){let p=await a.search(String(r),{userId:u,limit:c});l=Lt(p)}else{let p=await a.getAll({userId:u,limit:Math.max(c,50)});l=Lt(p)}return n&&(l=l.filter(p=>p.category===n)),i&&(l=l.filter(p=>p.ticket_key===i)),o&&(l=l.filter(p=>p.tier===o)),l=l.slice(0,c),JSON.stringify({total:l.length,memories:l,backend:"mem0"})}catch(a){throw new Error(`mem0 recall failed: ${a.message}. If mem0 is not installed, run: npm install mem0ai`,{cause:a})}}function Jt(s,t){let{ticketKey:e}=s;Nn(t);let n=G(t,`SELECT session_id, summary, tickets, tasks_run, tasks_passed, tasks_failed, created_at
|
|
@@ -418,10 +418,10 @@ fact, decision, context, insight, credential, url, error, workaround`,resolve(){
|
|
|
418
418
|
WHERE tier = 'long' ${i} ORDER BY relevance DESC, created_at DESC LIMIT 10`),c=G(t,`SELECT memory_key, category, content, source, tier, relevance, created_at FROM chat_memory
|
|
419
419
|
WHERE tier = 'mid' ${i} ORDER BY relevance DESC, created_at DESC LIMIT 8`),u=G(t,`SELECT type, status, COUNT(*) as cnt FROM chat_tasks
|
|
420
420
|
GROUP BY type, status ORDER BY cnt DESC LIMIT 10`),l=Bt([...o,...c]);return JSON.stringify({recentSessions:n,topMemories:l,taskStats:u,ticketFilter:e||null})}async function Dt(s,t,e){let{ticketKey:r}=s,n=await Ht({limit:80},t,e),i=JSON.parse(n||"{}"),o=Array.isArray(i.memories)?i.memories:[];r&&(o=o.filter(d=>d.ticket_key===r));let c=d=>{let m=Date.parse(String(d?.created_at||""))||0;return Number(d?.relevance||0)*1e12+m},a=(d,m)=>c(m)-c(d),u=o.filter(d=>d.tier==="long").sort(a).slice(0,10),l=o.filter(d=>d.tier==="mid").sort(a).slice(0,8),p=Bt([...u,...l]);return JSON.stringify({recentSessions:[],topMemories:p,taskStats:[],ticketFilter:r||null,backend:"mem0"})}function bn(s,t){let{summary:e,tickets:r,tasksRun:n=0,tasksPassed:i=0,tasksFailed:o=0,keyFacts:c}=s;if(!e)return JSON.stringify({error:"summary is required"});let a=process.env.ZIBBY_CHAT_SESSION_ID||`session_${we()}`;if(L(t,`INSERT INTO chat_sessions (session_id, summary, tickets, tasks_run, tasks_passed, tasks_failed, key_facts, created_at)
|
|
421
|
-
VALUES (${w(a)}, ${w(e)}, ${w(r)}, ${n}, ${i}, ${o}, ${w(c)}, ${w(
|
|
422
|
-
VALUES (${w(a)}, ${w(i)}, ${w(r)}, ${w(e)}, ${w(n)}, ${w(o)}, ${w(u)}, ${w(c)}, ${w(
|
|
421
|
+
VALUES (${w(a)}, ${w(e)}, ${w(r)}, ${n}, ${i}, ${o}, ${w(c)}, ${w(Z())})`),c)for(let u of c.split(";").map(l=>l.trim()).filter(Boolean))Wt({content:u,category:"fact",source:"session_summary",tier:"mid"},t);vn(t);try{C(t,["add","."]),C(t,["commit","-m",`session end: ${e.slice(0,60)}`])}catch{}return JSON.stringify({ok:!0,sessionId:a,summary:e.slice(0,200)})}function wn(s,t){let{title:e,type:r,status:n,ticketKey:i,specPath:o,resultSummary:c}=s;if(!e||!r||!n)return JSON.stringify({error:"title, type, and status are required"});let a=we(),u=process.env.ZIBBY_CHAT_SESSION_ID||null;L(t,`INSERT INTO chat_tasks (id, ticket_key, type, title, status, spec_path, session_id, result_summary, created_at, finished_at)
|
|
422
|
+
VALUES (${w(a)}, ${w(i)}, ${w(r)}, ${w(e)}, ${w(n)}, ${w(o)}, ${w(u)}, ${w(c)}, ${w(Z())}, ${w(Z())})`);try{C(t,["add","."]),C(t,["commit","-m",`task: ${n} \u2014 ${e.slice(0,60)}`])}catch{}return JSON.stringify({ok:!0,id:a,title:e,type:r,status:n})}function Sn(s,t){let{ticketKey:e,type:r,status:n,limit:i=20}=s,o=[];e&&o.push(`ticket_key = ${w(e)}`),r&&o.push(`type = ${w(r)}`),n&&o.push(`status = ${w(n)}`);let a=`SELECT id, ticket_key, type, title, status, spec_path, result_summary, created_at, finished_at
|
|
423
423
|
FROM chat_tasks ${o.length>0?`WHERE ${o.join(" AND ")}`:""}
|
|
424
|
-
ORDER BY created_at DESC LIMIT ${i}`,u=G(t,a);return JSON.stringify({total:u.length,tasks:u})}function vn(s){try{L(s,"UPDATE chat_memory SET relevance = relevance * 0.98 WHERE tier = 'long' AND relevance > 0.5"),L(s,"UPDATE chat_memory SET relevance = relevance * 0.90 WHERE tier = 'mid' AND relevance > 0.1"),L(s,"UPDATE chat_memory SET relevance = relevance * 0.70 WHERE tier = 'short' AND relevance > 0.05"),L(s,"DELETE FROM chat_memory WHERE relevance < 0.05")}catch{}}function Nn(s){try{let t=new Date(Date.now()-864e5).toISOString();L(s,`DELETE FROM chat_memory WHERE tier = 'short' AND created_at < ${w(t)}`)}catch{}}import{existsSync as
|
|
424
|
+
ORDER BY created_at DESC LIMIT ${i}`,u=G(t,a);return JSON.stringify({total:u.length,tasks:u})}function vn(s){try{L(s,"UPDATE chat_memory SET relevance = relevance * 0.98 WHERE tier = 'long' AND relevance > 0.5"),L(s,"UPDATE chat_memory SET relevance = relevance * 0.90 WHERE tier = 'mid' AND relevance > 0.1"),L(s,"UPDATE chat_memory SET relevance = relevance * 0.70 WHERE tier = 'short' AND relevance > 0.05"),L(s,"DELETE FROM chat_memory WHERE relevance < 0.05")}catch{}}function Nn(s){try{let t=new Date(Date.now()-864e5).toISOString();L(s,`DELETE FROM chat_memory WHERE tier = 'short' AND created_at < ${w(t)}`)}catch{}}import{existsSync as B,readFileSync as ie,readdirSync as De,mkdirSync as On,writeFileSync as ee,statSync as Zt}from"fs";import{join as v,resolve as Ue,relative as Qt,dirname as Xt}from"path";import{fileURLToPath as In}from"url";import{createRequire as Rn}from"module";var $n=Rn(import.meta.url),jn=`## Workflow Builder
|
|
425
425
|
|
|
426
426
|
You can help users build custom AI workflows using the Zibby workflow framework.
|
|
427
427
|
|
|
@@ -531,7 +531,7 @@ Call with no arguments to see all available topics.
|
|
|
531
531
|
- Workflow names must be kebab-case (e.g., ticket-triage, pr-review).
|
|
532
532
|
- State flows through: each node's validated output is stored under its name in state (e.g., state.classify_ticket).
|
|
533
533
|
- Downstream nodes reference upstream outputs in their prompt function (e.g., \\\`\\\${JSON.stringify(state.classify_ticket, null, 2)}\\\`).
|
|
534
|
-
- Nodes can declare skills to get MCP tool access \u2014 the framework handles server lifecycle automatically.`,er=/^[a-z][a-z0-9-]{0,62}[a-z0-9]$/;function tr(s){return`${s.split("-").map(t=>t.charAt(0).toUpperCase()+t.slice(1)).join("")}Workflow`}function ne(s){return`${s.replace(/_([a-z])/g,(t,e)=>e.toUpperCase())}Node`}function xn(s){let t=s?.agent;return t?t.provider?t.provider:t.gemini?"gemini":t.codex?"codex":t.claude?"claude":t.cursor?"cursor":process.env.AGENT_TYPE||"cursor":process.env.AGENT_TYPE||"cursor"}async function An(s){let t=Ue(s,".zibby.config.mjs");if(!
|
|
534
|
+
- Nodes can declare skills to get MCP tool access \u2014 the framework handles server lifecycle automatically.`,er=/^[a-z][a-z0-9-]{0,62}[a-z0-9]$/;function tr(s){return`${s.split("-").map(t=>t.charAt(0).toUpperCase()+t.slice(1)).join("")}Workflow`}function ne(s){return`${s.replace(/_([a-z])/g,(t,e)=>e.toUpperCase())}Node`}function xn(s){let t=s?.agent;return t?t.provider?t.provider:t.gemini?"gemini":t.codex?"codex":t.claude?"claude":t.cursor?"cursor":process.env.AGENT_TYPE||"cursor":process.env.AGENT_TYPE||"cursor"}async function An(s){let t=Ue(s,".zibby.config.mjs");if(!B(t))return{};try{return(await import(t)).default||{}}catch{return{}}}function Tn(){try{let s=Xt($n.resolve("@zibby/core/package.json")),t=v(s,"templates","browser-test-automation"),e=ie(v(t,"nodes","preflight.mjs"),"utf-8"),r=ie(v(t,"graph.mjs"),"utf-8");return{preflight:e,graph:r}}catch{return null}}var Yt=Xt(In(import.meta.url));function rr(){let s=Ue(Yt,"..","..","..","docsite","docs");if(B(s))return s;let t=Ue(Yt,"..","docs");return B(t)?t:null}function Vt(){let s=rr();if(!s)return[];try{let t=(e,r="")=>{let n=[];for(let i of De(e)){let o=v(e,i);try{if(Zt(o).isDirectory())n=n.concat(t(o,`${r}${i}/`));else if(i.endsWith(".md")){let c=`${r}${i.replace(/\.md$/,"")}`;n.push(c)}}catch{}}return n};return t(s)}catch{return[]}}function sr(s){let t=rr();if(!t)return null;let e=v(t,`${s}.md`);if(!B(e))return null;try{return ie(e,"utf-8")}catch{return null}}function En(s){let t=s.nodes.map(o=>{let c=o.inputFields?.length?`Input fields: ${o.inputFields.join(", ")}`:"Input: receives full state",a=o.outputFields?.length?`Output fields: ${o.outputFields.join(", ")}`:"Output: determined by task",u=o.skills?.length?`Skills: ${o.skills.join(", ")}`:"";return`- ${o.name}: ${o.description}. ${c}. ${a}.${u?` ${u}`:""}`}).join(`
|
|
535
535
|
`),e=s.edges.map(o=>o.condition?`- ${o.from} \u2192 ${o.to} (conditional: ${o.condition})`:`- ${o.from} \u2192 ${o.to}`).join(`
|
|
536
536
|
`),r=Tn(),n=sr("custom-workflows"),i="";return r&&(i+=`
|
|
537
537
|
## Real working examples from the Zibby framework
|
|
@@ -658,8 +658,8 @@ ${m}
|
|
|
658
658
|
}
|
|
659
659
|
}
|
|
660
660
|
`;ee(v(o,"graph.mjs"),h,"utf-8");let _={name:n,description:e.description||`${i} workflow`,entryClass:i,triggers:{api:!0}};ee(v(o,"workflow.json"),`${JSON.stringify(_,null,2)}
|
|
661
|
-
`,"utf-8");let g=["graph.mjs","workflow.json","nodes/index.mjs",...a.map(y=>`nodes/${y.replace(/_/g,"-")}.mjs`)];return{workflowDir:Qt(s,o),files:g,className:i,slug:n}}async function Jn(s){let{name:t,description:e,nodes:r,edges:n}=s;if(!t||!er.test(t.toLowerCase()))return JSON.stringify({error:`Invalid workflow name "${t}". Must be kebab-case, 2-64 chars, lowercase letters/numbers/hyphens.`});if(!r||r.length===0)return JSON.stringify({error:"At least one node is required."});let i={name:t.toLowerCase(),description:e||`${tr(t.toLowerCase())} workflow`,nodes:r.map(o=>({name:o.name.replace(/-/g,"_"),description:o.description||`Process ${o.name}`,inputFields:o.inputFields||[],outputFields:o.outputFields||[]})),edges:n||[]};if(i.edges.length===0&&i.nodes.length>0){for(let o=0;o<i.nodes.length-1;o++)i.edges.push({from:i.nodes[o].name,to:i.nodes[o+1].name});i.edges.push({from:i.nodes[i.nodes.length-1].name,to:"END"})}return JSON.stringify({ok:!0,spec:i,message:`Workflow "${i.name}" designed with ${i.nodes.length} node(s). Call build_workflow to generate the code.`,preview:{nodes:i.nodes.map(o=>o.name),flow:i.edges.map(o=>o.condition?`${o.from} \u2192(if ${o.condition})\u2192 ${o.to}`:`${o.from} \u2192 ${o.to}`)}})}async function Dn(s,t){let{name:e,spec:r}=s,n=(e||r?.name||"").toLowerCase();if(!n||!er.test(n))return JSON.stringify({error:`Invalid workflow name "${n}".`});if(!r||!r.nodes||r.nodes.length===0)return JSON.stringify({error:"spec with nodes is required. Call design_workflow first."});let i=v(t,".zibby","workflows",n);if(
|
|
662
|
-
`,k=K(_)?ie(_,"utf-8"):"";return k.includes(g)||ee(_,k+y,"utf-8"),JSON.stringify({ok:!0,file:`nodes/${h}`,exportName:g,message:`Node "${a}" added. Update graph.mjs to wire it into the graph.`})}async function Pn(s,t){let{name:e,projectId:r}=s,n=(e||"").toLowerCase();if(!n)return JSON.stringify({error:"Workflow name is required."});if(!r)return JSON.stringify({error:"projectId is required."});let i=v(t,".zibby","workflows",n);if(!K(i))return JSON.stringify({error:`Workflow "${n}" not found at .zibby/workflows/${n}/`});try{let{execSync:o}=await import("child_process"),c=o(`node "${v(t,"packages/cli/bin/zibby.js")}" deploy ${n} --project ${r}`,{cwd:t,encoding:"utf-8",timeout:3e4,stdio:["pipe","pipe","pipe"]});return JSON.stringify({ok:!0,output:c.trim()})}catch{try{let{execSync:c}=await import("child_process"),a=c(`npx zibby deploy ${n} --project ${r}`,{cwd:t,encoding:"utf-8",timeout:3e4,stdio:["pipe","pipe","pipe"]});return JSON.stringify({ok:!0,output:a.trim()})}catch(c){return JSON.stringify({error:`Deploy failed: ${c.message}`})}}}function Mn(s){let t=v(s,".zibby","workflows");if(!K(t))return JSON.stringify({workflows:[],message:"No workflows found. Use build_workflow to create one."});let r=De(t).filter(n=>{try{return Vt(v(t,n)).isDirectory()}catch{return!1}}).map(n=>{let i=v(t,n,"workflow.json"),o={};try{o=JSON.parse(ie(i,"utf-8"))}catch{}let c=v(t,n,"nodes"),a=0;try{a=De(c).filter(u=>u.endsWith(".mjs")&&u!=="index.mjs").length}catch{}return{name:n,description:o.description||"",nodeCount:a,path:Qt(s,v(t,n))}});return JSON.stringify({workflows:r})}var ir={id:"workflow-builder",description:"Build, scaffold, and deploy custom AI workflows via conversation",envKeys:[],promptFragment:jn,tools:[{name:"design_workflow",description:"Design a workflow spec (nodes, edges, descriptions) for the user to review before building. Call this after understanding requirements.",input_schema:{type:"object",properties:{name:{type:"string",description:"Workflow name in kebab-case (e.g., ticket-triage)"},description:{type:"string",description:"What the workflow does"},nodes:{type:"array",items:{type:"object",properties:{name:{type:"string",description:"Node name in snake_case (e.g., classify_ticket)"},description:{type:"string",description:"What this node does \u2014 be specific about input/output"},inputFields:{type:"array",items:{type:"string"},description:"Key fields this node reads from state"},outputFields:{type:"array",items:{type:"string"},description:"Key fields this node produces"}},required:["name","description"]},description:"Workflow nodes (processing steps)"},edges:{type:"array",items:{type:"object",properties:{from:{type:"string",description:"Source node name"},to:{type:"string",description:'Target node name (or "END")'},condition:{type:"string",description:"JS expression for conditional routing (optional)"}},required:["from","to"]},description:"Edges connecting nodes. If omitted, nodes are wired linearly."}},required:["name","description","nodes"]}},{name:"build_workflow",description:"Generate real workflow code on disk from a design spec. Uses the configured AI agent for high-quality code generation.",input_schema:{type:"object",properties:{name:{type:"string",description:"Workflow name (from design_workflow)"},spec:{type:"object",description:"The full spec object returned by design_workflow",properties:{name:{type:"string"},description:{type:"string"},nodes:{type:"array",items:{type:"object"}},edges:{type:"array",items:{type:"object"}}}}},required:["name","spec"]}},{name:"add_node",description:"Add a new node to an existing workflow. Generates the node file and updates the barrel export.",input_schema:{type:"object",properties:{workflowName:{type:"string",description:"Existing workflow name (kebab-case)"},nodeName:{type:"string",description:"New node name (snake_case)"},description:{type:"string",description:"What this node does"},inputFields:{type:"array",items:{type:"string"},description:"Fields read from state"},outputFields:{type:"array",items:{type:"string"},description:"Fields produced"}},required:["workflowName","nodeName","description"]}},{name:"deploy_workflow",description:"Deploy a workflow to Zibby Cloud. Returns the trigger URL.",input_schema:{type:"object",properties:{name:{type:"string",description:"Workflow name to deploy"},projectId:{type:"string",description:"Target project ID"}},required:["name","projectId"]}},{name:"list_workflows",description:"List all local workflows in .zibby/workflows/.",input_schema:{type:"object",properties:{}}},{name:"explore_framework_docs",description:"Read Zibby framework documentation on demand. Call this before building complex workflows or when you need details on advanced patterns (middleware, conditional routing, skills, deployment, CLI commands).",input_schema:{type:"object",properties:{topic:{type:"string",description:'Doc topic to read (e.g., "workflow", "custom-workflows", "cli-reference", "packages/core", "packages/skills", "integrations/jira"). Call with no topic to list all available docs.'}}}}],async handleToolCall(s,t,e){let r=e?.options?.workspace||process.cwd();try{switch(s){case"design_workflow":return await Jn(t);case"build_workflow":return await Dn(t,r);case"add_node":return await Un(t,r);case"deploy_workflow":return await Pn(t,r);case"list_workflows":return Mn(r);case"explore_framework_docs":{let n=(t.topic||"").trim();if(!n){let o=Zt();return JSON.stringify({available:o,hint:"Call again with a topic to read its content."})}let i=sr(n);if(!i){let o=Zt();return JSON.stringify({error:`Doc "${n}" not found.`,available:o})}return JSON.stringify({topic:n,content:i})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(n){return JSON.stringify({error:n.message})}},resolve(){return null}};import{resolveIntegrationToken as qn}from"@zibby/core/backend-client.js";var Me=Object.freeze({id:"openai_billing",requiresIntegration:A.OPENAI_BILLING,description:"OpenAI organization billing/usage admin API (paste sk-admin-... key)"}),qe=Object.freeze({id:"anthropic_billing",requiresIntegration:A.ANTHROPIC_BILLING,description:"Anthropic organization cost/usage admin API (paste sk-ant-admin-... key)"}),Ke=Object.freeze({id:"cursor_admin",requiresIntegration:A.CURSOR_ADMIN,description:"Cursor Team/Enterprise admin API (paste admin key)"});function or(s){return Math.floor(s/1e3)}function Pe(s){return new Date(s).toISOString().slice(0,10)}function ar(s){return new Date(s).toISOString()}async function oe(s){let t=await qn(s);if(!t?.token)throw new Error(`${s} token resolver returned no token`);return t.token}async function cr({startMs:s,endMs:t,groupBy:e=["project_id","line_item"]}){let r=await oe("openai_billing"),n=[],i=0,o=e.map(a=>`group_by[]=${encodeURIComponent(a)}`).join("&"),c=null;for(let a=0;a<50;a++){let l=`https://api.openai.com/v1/organization/costs?${[`start_time=${or(s)}`,`end_time=${or(t)}`,"bucket_width=1d","limit=180",o,c?`page=${encodeURIComponent(c)}`:""].filter(Boolean).join("&")}`,p=await fetch(l,{headers:{Authorization:`Bearer ${r}`}});if(!p.ok){let m=await p.text().catch(()=>"");throw new Error(`OpenAI costs API ${p.status}: ${m.slice(0,200)}`)}let d=await p.json();for(let m of d.data||[]){i+=1;let h=Pe((m.start_time||0)*1e3);for(let _ of m.results||[])n.push({provider:"openai",day:h,costUsd:Number(_.amount?.value??0),projectId:_.project_id||void 0,apiKeyId:_.api_key_id||void 0,model:_.line_item||void 0})}if(!d.has_more||!d.next_page)break;c=d.next_page}return{ok:!0,items:n,rawBuckets:i}}async function Kn(){let s=await oe("openai_billing"),e=await fetch("https://api.openai.com/v1/organization/projects?limit=100",{headers:{Authorization:`Bearer ${s}`}});if(!e.ok){let i=await e.text().catch(()=>"");throw new Error(`OpenAI projects API ${e.status}: ${i.slice(0,200)}`)}let r=await e.json(),n=new Map;for(let i of r.data||[])n.set(i.id,i.name);return n}async function lr({startMs:s,endMs:t,groupBy:e=["workspace_id"]}){let r=await oe("anthropic_billing"),n=[],i=0,o=e.map(a=>`group_by[]=${encodeURIComponent(a)}`).join("&"),c=null;for(let a=0;a<50;a++){let l=`https://api.anthropic.com/v1/organizations/cost_report?${[`starting_at=${encodeURIComponent(ar(s))}`,`ending_at=${encodeURIComponent(ar(t))}`,"bucket=1d","limit=100",o,c?`page=${encodeURIComponent(c)}`:""].filter(Boolean).join("&")}`,p=await fetch(l,{headers:{"x-api-key":r,"anthropic-version":"2023-06-01"}});if(!p.ok){let m=await p.text().catch(()=>"");throw new Error(`Anthropic cost_report ${p.status}: ${m.slice(0,200)}`)}let d=await p.json();for(let m of d.data||[]){i+=1;let h=(m.starting_at||"").slice(0,10);for(let _ of m.results||[])n.push({provider:"anthropic",day:h,costUsd:Number(_.amount??_.cost??0),workspaceId:_.workspace_id||void 0,apiKeyId:_.api_key_id||void 0,model:_.model||void 0,tokensIn:_.uncached_input_tokens!=null?Number(_.uncached_input_tokens):void 0,tokensOut:_.output_tokens!=null?Number(_.output_tokens):void 0,cachedTokens:_.cached_input_tokens!=null?Number(_.cached_input_tokens):void 0})}if(!d.has_more||!d.next_page)break;c=d.next_page}return{ok:!0,items:n,rawBuckets:i}}async function Bn(){let s=await oe("anthropic_billing"),e=await fetch("https://api.anthropic.com/v1/organizations/workspaces?limit=100",{headers:{"x-api-key":s,"anthropic-version":"2023-06-01"}});if(!e.ok){let i=await e.text().catch(()=>"");throw new Error(`Anthropic workspaces ${e.status}: ${i.slice(0,200)}`)}let r=await e.json(),n=new Map;for(let i of r.data||[])n.set(i.id,i.name);return n}async function ur({startMs:s,endMs:t}){let e=await oe("cursor_admin"),r=Pe(s),n=Pe(t),i=`https://api.cursor.com/teams/daily-usage-data?startDate=${r}&endDate=${n}`,o=await fetch(i,{headers:{Authorization:`Bearer ${e}`}});if(!o.ok){let l=await o.text().catch(()=>"");throw new Error(`Cursor daily-usage ${o.status}: ${l.slice(0,200)}`)}let c=await o.json(),a=[],u=0;for(let l of c.data||[]){u+=1;let p=l.date;for(let d of l.userMetrics||[]){for(let m of d.modelUsage||[]){let h=Number(m.acceptedLines??0),_=Number(m.suggestedLines??0);a.push({provider:"cursor",day:p,costUsd:Number(m.totalCents??0)/100,userEmail:d.email,model:m.model,requestCount:Number(m.requestCount??0),acceptanceRate:_>0?h/_:void 0})}(!d.modelUsage||d.modelUsage.length===0)&&a.push({provider:"cursor",day:p,costUsd:Number(d.totalCents??0)/100,userEmail:d.email})}}return{ok:!0,items:a,rawBuckets:u}}async function Fn({startMs:s,endMs:t}){let[e,r,n]=await Promise.allSettled([cr({startMs:s,endMs:t}),lr({startMs:s,endMs:t}),ur({startMs:s,endMs:t})]),i=l=>l.status==="fulfilled"?l.value:{ok:!1,error:l.reason?.message||String(l.reason),items:[]},o=i(e),c=i(r),a=i(n),u=[{provider:"openai",totalUsd:o.items.reduce((l,p)=>l+(p.costUsd||0),0)},{provider:"anthropic",totalUsd:c.items.reduce((l,p)=>l+(p.costUsd||0),0)},{provider:"cursor",totalUsd:a.items.reduce((l,p)=>l+(p.costUsd||0),0)}];return{openai:o,anthropic:c,cursor:a,totals:u}}function zn(s,t){let e=new Map;for(let r of s){let n=t(r);if(!n)continue;let i=e.get(n)||{key:n,totalUsd:0,count:0};i.totalUsd+=r.costUsd||0,i.count+=1,e.set(n,i)}return[...e.values()].sort((r,n)=>n.totalUsd-r.totalUsd)}function Gn(s){if(!s.length)return{mean:0,stddev:0};let t=s.reduce((r,n)=>r+n,0)/s.length,e=s.reduce((r,n)=>r+(n-t)**2,0)/s.length;return{mean:t,stddev:Math.sqrt(e)}}import{z as b}from"zod";var ae=["ok","info","warn","critical"],Wn=b.object({primary:b.string().min(1).max(200).describe('Headline number or phrase (e.g. "$8,240"). Rendered in large/bold.'),delta:b.object({value:b.string().max(40).describe('Delta vs baseline (e.g. "+12% wow"). Free-form string.'),direction:b.enum(["up","down","flat"]).optional(),severity:b.enum(ae).optional().describe("Color severity for the delta (warn/critical highlights regressions).")}).optional().describe("Optional comparison vs baseline. Renders inline next to primary."),summary:b.string().max(800).optional().describe('One-sentence narrative ("why this number"). Plain prose.')}),Hn=b.object({kind:b.literal("trend"),title:b.string().max(120).optional(),labels:b.array(b.string().max(60)).min(2).max(20).describe('Bucket labels (e.g. ["Week-3", "Week-2", "Week-1", "This wk"]).'),values:b.array(b.number()).min(2).max(20).describe("Numeric values, one per label. Must match labels.length."),highlight:b.enum(["last","max","min","none"]).default("last").optional().describe("Which bucket to visually highlight in the rendered card."),severity:b.enum(ae).optional()}).refine(s=>s.labels.length===s.values.length,{message:"labels.length must equal values.length"}),Yn=b.object({kind:b.literal("table"),title:b.string().max(120).optional(),headers:b.array(b.string().max(40)).min(1).max(8),rows:b.array(b.array(b.union([b.string().max(200),b.number()])).min(1).max(8)).max(40).describe("2D matrix. Each inner array must have headers.length entries.")}).refine(s=>s.rows.every(t=>t.length===s.headers.length),{message:"every row must have headers.length entries"}),Zn=b.object({kind:b.literal("callouts"),title:b.string().max(120).optional(),tone:b.enum(ae).default("info").optional(),items:b.array(b.string().min(1).max(600)).min(1).max(10).describe("Each item renders as a bullet with a severity emoji.")}),Vn=b.object({kind:b.literal("breakdown"),title:b.string().max(120).optional(),rows:b.array(b.object({label:b.string().min(1).max(80),value:b.string().min(1).max(80),sub:b.string().max(120).optional(),severity:b.enum(ae).optional()})).min(1).max(20)}),Qn=b.object({kind:b.literal("paragraph"),title:b.string().max(120).optional(),text:b.string().min(1).max(3e3)}),Xn=b.discriminatedUnion("kind",[Hn,Yn,Zn,Vn,Qn]),Be=b.object({title:b.string().min(1).max(200).describe('Card title (e.g. "Weekly AI Spend Report").'),subtitle:b.string().max(200).optional().describe('Date range or smaller header (e.g. "May 13 \u2014 May 20").'),headline:Wn,sections:b.array(Xn).max(20).default([]),footer:b.object({viewUrl:b.string().url().optional().describe('Optional "View in Zibby" button URL.'),rerunUrl:b.string().url().optional().describe('Optional "Run again" button URL.')}).optional()}),W=Object.freeze({ok:"\u{1F7E2}",info:"\u{1F535}",warn:"\u{1F7E0}",critical:"\u{1F534}"}),pr=Object.freeze({up:"\u2191",down:"\u2193",flat:"\u2192"}),ei=Object.freeze({ok:"green",info:"blue",warn:"orange",critical:"red"});function dr(s,t,e=12){if(!Number.isFinite(s)||!Number.isFinite(t)||t<=0)return"";let r=Math.max(0,Math.min(1,s/t)),n=Math.round(r*e);return"\u2593".repeat(n)+"\u2591".repeat(e-n)}function Fe(s,t){let e=String(s);return e.length>=t?e:e+" ".repeat(t-e.length)}function mr(s,t){let e=String(s);return e.length>=t?e:" ".repeat(t-e.length)+e}function fr({headers:s,rows:t}){let e=s.map((o,c)=>{let a=Math.max(String(o).length,...t.map(u=>String(u[c]??"").length));return Math.min(a,32)}),r=o=>o.map((c,a)=>Fe(c,e[a])).join(" "),n=e.map(o=>"\u2500".repeat(o)).join(" ");return"```\n"+[r(s),n,...t.map(o=>r(o))].join(`
|
|
661
|
+
`,"utf-8");let g=["graph.mjs","workflow.json","nodes/index.mjs",...a.map(y=>`nodes/${y.replace(/_/g,"-")}.mjs`)];return{workflowDir:Qt(s,o),files:g,className:i,slug:n}}async function Jn(s){let{name:t,description:e,nodes:r,edges:n}=s;if(!t||!er.test(t.toLowerCase()))return JSON.stringify({error:`Invalid workflow name "${t}". Must be kebab-case, 2-64 chars, lowercase letters/numbers/hyphens.`});if(!r||r.length===0)return JSON.stringify({error:"At least one node is required."});let i={name:t.toLowerCase(),description:e||`${tr(t.toLowerCase())} workflow`,nodes:r.map(o=>({name:o.name.replace(/-/g,"_"),description:o.description||`Process ${o.name}`,inputFields:o.inputFields||[],outputFields:o.outputFields||[]})),edges:n||[]};if(i.edges.length===0&&i.nodes.length>0){for(let o=0;o<i.nodes.length-1;o++)i.edges.push({from:i.nodes[o].name,to:i.nodes[o+1].name});i.edges.push({from:i.nodes[i.nodes.length-1].name,to:"END"})}return JSON.stringify({ok:!0,spec:i,message:`Workflow "${i.name}" designed with ${i.nodes.length} node(s). Call build_workflow to generate the code.`,preview:{nodes:i.nodes.map(o=>o.name),flow:i.edges.map(o=>o.condition?`${o.from} \u2192(if ${o.condition})\u2192 ${o.to}`:`${o.from} \u2192 ${o.to}`)}})}async function Dn(s,t){let{name:e,spec:r}=s,n=(e||r?.name||"").toLowerCase();if(!n||!er.test(n))return JSON.stringify({error:`Invalid workflow name "${n}".`});if(!r||!r.nodes||r.nodes.length===0)return JSON.stringify({error:"spec with nodes is required. Call design_workflow first."});let i=v(t,".zibby","workflows",n);if(B(i))return JSON.stringify({error:`Workflow "${n}" already exists at .zibby/workflows/${n}/. Delete it first or choose a different name.`});let o=await nr(r,t),c=Ln(t,n,r,o);return JSON.stringify({ok:!0,...c,message:`Workflow "${n}" created at ${c.workflowDir}/`,nextSteps:[`Test locally: zibby start ${n}`,`Deploy to cloud: zibby deploy ${n} --project <project-id>`,`Tail logs: zibby logs --workflow ${n} --project <project-id>`]})}async function Un(s,t){let{workflowName:e,nodeName:r,description:n,inputFields:i,outputFields:o}=s,c=(e||"").toLowerCase(),a=(r||"").replace(/-/g,"_"),u=v(t,".zibby","workflows",c);if(!B(u))return JSON.stringify({error:`Workflow "${c}" not found. Create it first with build_workflow.`});let l={name:c,description:"",nodes:[{name:a,description:n||`Process ${a}`,inputFields:i||[],outputFields:o||[]}],edges:[]},d=(await nr(l,t)).nodes?.[a]?.code;if(!d)return JSON.stringify({error:"Failed to generate node code."});let m=v(u,"nodes"),h=`${a.replace(/_/g,"-")}.mjs`;ee(v(m,h),d,"utf-8");let _=v(m,"index.mjs"),g=ne(a),y=`export { ${g} } from './${a.replace(/_/g,"-")}.mjs';
|
|
662
|
+
`,k=B(_)?ie(_,"utf-8"):"";return k.includes(g)||ee(_,k+y,"utf-8"),JSON.stringify({ok:!0,file:`nodes/${h}`,exportName:g,message:`Node "${a}" added. Update graph.mjs to wire it into the graph.`})}async function Pn(s,t){let{name:e,projectId:r}=s,n=(e||"").toLowerCase();if(!n)return JSON.stringify({error:"Workflow name is required."});if(!r)return JSON.stringify({error:"projectId is required."});let i=v(t,".zibby","workflows",n);if(!B(i))return JSON.stringify({error:`Workflow "${n}" not found at .zibby/workflows/${n}/`});try{let{execSync:o}=await import("child_process"),c=o(`node "${v(t,"packages/cli/bin/zibby.js")}" deploy ${n} --project ${r}`,{cwd:t,encoding:"utf-8",timeout:3e4,stdio:["pipe","pipe","pipe"]});return JSON.stringify({ok:!0,output:c.trim()})}catch{try{let{execSync:c}=await import("child_process"),a=c(`npx zibby deploy ${n} --project ${r}`,{cwd:t,encoding:"utf-8",timeout:3e4,stdio:["pipe","pipe","pipe"]});return JSON.stringify({ok:!0,output:a.trim()})}catch(c){return JSON.stringify({error:`Deploy failed: ${c.message}`})}}}function Mn(s){let t=v(s,".zibby","workflows");if(!B(t))return JSON.stringify({workflows:[],message:"No workflows found. Use build_workflow to create one."});let r=De(t).filter(n=>{try{return Zt(v(t,n)).isDirectory()}catch{return!1}}).map(n=>{let i=v(t,n,"workflow.json"),o={};try{o=JSON.parse(ie(i,"utf-8"))}catch{}let c=v(t,n,"nodes"),a=0;try{a=De(c).filter(u=>u.endsWith(".mjs")&&u!=="index.mjs").length}catch{}return{name:n,description:o.description||"",nodeCount:a,path:Qt(s,v(t,n))}});return JSON.stringify({workflows:r})}var ir={id:"workflow-builder",description:"Build, scaffold, and deploy custom AI workflows via conversation",envKeys:[],promptFragment:jn,tools:[{name:"design_workflow",description:"Design a workflow spec (nodes, edges, descriptions) for the user to review before building. Call this after understanding requirements.",input_schema:{type:"object",properties:{name:{type:"string",description:"Workflow name in kebab-case (e.g., ticket-triage)"},description:{type:"string",description:"What the workflow does"},nodes:{type:"array",items:{type:"object",properties:{name:{type:"string",description:"Node name in snake_case (e.g., classify_ticket)"},description:{type:"string",description:"What this node does \u2014 be specific about input/output"},inputFields:{type:"array",items:{type:"string"},description:"Key fields this node reads from state"},outputFields:{type:"array",items:{type:"string"},description:"Key fields this node produces"}},required:["name","description"]},description:"Workflow nodes (processing steps)"},edges:{type:"array",items:{type:"object",properties:{from:{type:"string",description:"Source node name"},to:{type:"string",description:'Target node name (or "END")'},condition:{type:"string",description:"JS expression for conditional routing (optional)"}},required:["from","to"]},description:"Edges connecting nodes. If omitted, nodes are wired linearly."}},required:["name","description","nodes"]}},{name:"build_workflow",description:"Generate real workflow code on disk from a design spec. Uses the configured AI agent for high-quality code generation.",input_schema:{type:"object",properties:{name:{type:"string",description:"Workflow name (from design_workflow)"},spec:{type:"object",description:"The full spec object returned by design_workflow",properties:{name:{type:"string"},description:{type:"string"},nodes:{type:"array",items:{type:"object"}},edges:{type:"array",items:{type:"object"}}}}},required:["name","spec"]}},{name:"add_node",description:"Add a new node to an existing workflow. Generates the node file and updates the barrel export.",input_schema:{type:"object",properties:{workflowName:{type:"string",description:"Existing workflow name (kebab-case)"},nodeName:{type:"string",description:"New node name (snake_case)"},description:{type:"string",description:"What this node does"},inputFields:{type:"array",items:{type:"string"},description:"Fields read from state"},outputFields:{type:"array",items:{type:"string"},description:"Fields produced"}},required:["workflowName","nodeName","description"]}},{name:"deploy_workflow",description:"Deploy a workflow to Zibby Cloud. Returns the trigger URL.",input_schema:{type:"object",properties:{name:{type:"string",description:"Workflow name to deploy"},projectId:{type:"string",description:"Target project ID"}},required:["name","projectId"]}},{name:"list_workflows",description:"List all local workflows in .zibby/workflows/.",input_schema:{type:"object",properties:{}}},{name:"explore_framework_docs",description:"Read Zibby framework documentation on demand. Call this before building complex workflows or when you need details on advanced patterns (middleware, conditional routing, skills, deployment, CLI commands).",input_schema:{type:"object",properties:{topic:{type:"string",description:'Doc topic to read (e.g., "workflow", "custom-workflows", "cli-reference", "packages/core", "packages/skills", "integrations/jira"). Call with no topic to list all available docs.'}}}}],async handleToolCall(s,t,e){let r=e?.options?.workspace||process.cwd();try{switch(s){case"design_workflow":return await Jn(t);case"build_workflow":return await Dn(t,r);case"add_node":return await Un(t,r);case"deploy_workflow":return await Pn(t,r);case"list_workflows":return Mn(r);case"explore_framework_docs":{let n=(t.topic||"").trim();if(!n){let o=Vt();return JSON.stringify({available:o,hint:"Call again with a topic to read its content."})}let i=sr(n);if(!i){let o=Vt();return JSON.stringify({error:`Doc "${n}" not found.`,available:o})}return JSON.stringify({topic:n,content:i})}default:return JSON.stringify({error:`Unknown tool: ${s}`})}}catch(n){return JSON.stringify({error:n.message})}},resolve(){return null}};import{resolveIntegrationToken as qn}from"@zibby/core/backend-client.js";var Me=Object.freeze({id:"openai_billing",requiresIntegration:A.OPENAI_BILLING,description:"OpenAI organization billing/usage admin API (paste sk-admin-... key)"}),qe=Object.freeze({id:"anthropic_billing",requiresIntegration:A.ANTHROPIC_BILLING,description:"Anthropic organization cost/usage admin API (paste sk-ant-admin-... key)"}),Ke=Object.freeze({id:"cursor_admin",requiresIntegration:A.CURSOR_ADMIN,description:"Cursor Team/Enterprise admin API (paste admin key)"});function or(s){return Math.floor(s/1e3)}function Pe(s){return new Date(s).toISOString().slice(0,10)}function ar(s){return new Date(s).toISOString()}async function oe(s){let t=await qn(s);if(!t?.token)throw new Error(`${s} token resolver returned no token`);return t.token}async function cr({startMs:s,endMs:t,groupBy:e=["project_id","line_item"]}){let r=await oe("openai_billing"),n=[],i=0,o=e.map(a=>`group_by[]=${encodeURIComponent(a)}`).join("&"),c=null;for(let a=0;a<50;a++){let l=`https://api.openai.com/v1/organization/costs?${[`start_time=${or(s)}`,`end_time=${or(t)}`,"bucket_width=1d","limit=180",o,c?`page=${encodeURIComponent(c)}`:""].filter(Boolean).join("&")}`,p=await fetch(l,{headers:{Authorization:`Bearer ${r}`}});if(!p.ok){let m=await p.text().catch(()=>"");throw new Error(`OpenAI costs API ${p.status}: ${m.slice(0,200)}`)}let d=await p.json();for(let m of d.data||[]){i+=1;let h=Pe((m.start_time||0)*1e3);for(let _ of m.results||[])n.push({provider:"openai",day:h,costUsd:Number(_.amount?.value??0),projectId:_.project_id||void 0,apiKeyId:_.api_key_id||void 0,model:_.line_item||void 0})}if(!d.has_more||!d.next_page)break;c=d.next_page}return{ok:!0,items:n,rawBuckets:i}}async function Kn(){let s=await oe("openai_billing"),e=await fetch("https://api.openai.com/v1/organization/projects?limit=100",{headers:{Authorization:`Bearer ${s}`}});if(!e.ok){let i=await e.text().catch(()=>"");throw new Error(`OpenAI projects API ${e.status}: ${i.slice(0,200)}`)}let r=await e.json(),n=new Map;for(let i of r.data||[])n.set(i.id,i.name);return n}async function lr({startMs:s,endMs:t,groupBy:e=["workspace_id"]}){let r=await oe("anthropic_billing"),n=[],i=0,o=e.map(a=>`group_by[]=${encodeURIComponent(a)}`).join("&"),c=null;for(let a=0;a<50;a++){let l=`https://api.anthropic.com/v1/organizations/cost_report?${[`starting_at=${encodeURIComponent(ar(s))}`,`ending_at=${encodeURIComponent(ar(t))}`,"bucket=1d","limit=100",o,c?`page=${encodeURIComponent(c)}`:""].filter(Boolean).join("&")}`,p=await fetch(l,{headers:{"x-api-key":r,"anthropic-version":"2023-06-01"}});if(!p.ok){let m=await p.text().catch(()=>"");throw new Error(`Anthropic cost_report ${p.status}: ${m.slice(0,200)}`)}let d=await p.json();for(let m of d.data||[]){i+=1;let h=(m.starting_at||"").slice(0,10);for(let _ of m.results||[])n.push({provider:"anthropic",day:h,costUsd:Number(_.amount??_.cost??0),workspaceId:_.workspace_id||void 0,apiKeyId:_.api_key_id||void 0,model:_.model||void 0,tokensIn:_.uncached_input_tokens!=null?Number(_.uncached_input_tokens):void 0,tokensOut:_.output_tokens!=null?Number(_.output_tokens):void 0,cachedTokens:_.cached_input_tokens!=null?Number(_.cached_input_tokens):void 0})}if(!d.has_more||!d.next_page)break;c=d.next_page}return{ok:!0,items:n,rawBuckets:i}}async function Bn(){let s=await oe("anthropic_billing"),e=await fetch("https://api.anthropic.com/v1/organizations/workspaces?limit=100",{headers:{"x-api-key":s,"anthropic-version":"2023-06-01"}});if(!e.ok){let i=await e.text().catch(()=>"");throw new Error(`Anthropic workspaces ${e.status}: ${i.slice(0,200)}`)}let r=await e.json(),n=new Map;for(let i of r.data||[])n.set(i.id,i.name);return n}async function ur({startMs:s,endMs:t}){let e=await oe("cursor_admin"),r=Pe(s),n=Pe(t),i=`https://api.cursor.com/teams/daily-usage-data?startDate=${r}&endDate=${n}`,o=await fetch(i,{headers:{Authorization:`Bearer ${e}`}});if(!o.ok){let l=await o.text().catch(()=>"");throw new Error(`Cursor daily-usage ${o.status}: ${l.slice(0,200)}`)}let c=await o.json(),a=[],u=0;for(let l of c.data||[]){u+=1;let p=l.date;for(let d of l.userMetrics||[]){for(let m of d.modelUsage||[]){let h=Number(m.acceptedLines??0),_=Number(m.suggestedLines??0);a.push({provider:"cursor",day:p,costUsd:Number(m.totalCents??0)/100,userEmail:d.email,model:m.model,requestCount:Number(m.requestCount??0),acceptanceRate:_>0?h/_:void 0})}(!d.modelUsage||d.modelUsage.length===0)&&a.push({provider:"cursor",day:p,costUsd:Number(d.totalCents??0)/100,userEmail:d.email})}}return{ok:!0,items:a,rawBuckets:u}}async function Fn({startMs:s,endMs:t}){let[e,r,n]=await Promise.allSettled([cr({startMs:s,endMs:t}),lr({startMs:s,endMs:t}),ur({startMs:s,endMs:t})]),i=l=>l.status==="fulfilled"?l.value:{ok:!1,error:l.reason?.message||String(l.reason),items:[]},o=i(e),c=i(r),a=i(n),u=[{provider:"openai",totalUsd:o.items.reduce((l,p)=>l+(p.costUsd||0),0)},{provider:"anthropic",totalUsd:c.items.reduce((l,p)=>l+(p.costUsd||0),0)},{provider:"cursor",totalUsd:a.items.reduce((l,p)=>l+(p.costUsd||0),0)}];return{openai:o,anthropic:c,cursor:a,totals:u}}function zn(s,t){let e=new Map;for(let r of s){let n=t(r);if(!n)continue;let i=e.get(n)||{key:n,totalUsd:0,count:0};i.totalUsd+=r.costUsd||0,i.count+=1,e.set(n,i)}return[...e.values()].sort((r,n)=>n.totalUsd-r.totalUsd)}function Gn(s){if(!s.length)return{mean:0,stddev:0};let t=s.reduce((r,n)=>r+n,0)/s.length,e=s.reduce((r,n)=>r+(n-t)**2,0)/s.length;return{mean:t,stddev:Math.sqrt(e)}}import{z as b}from"zod";var ae=["ok","info","warn","critical"],Wn=b.object({primary:b.string().min(1).max(200).describe('Headline number or phrase (e.g. "$8,240"). Rendered in large/bold.'),delta:b.object({value:b.string().max(40).describe('Delta vs baseline (e.g. "+12% wow"). Free-form string.'),direction:b.enum(["up","down","flat"]).optional(),severity:b.enum(ae).optional().describe("Color severity for the delta (warn/critical highlights regressions).")}).optional().describe("Optional comparison vs baseline. Renders inline next to primary."),summary:b.string().max(800).optional().describe('One-sentence narrative ("why this number"). Plain prose.')}),Hn=b.object({kind:b.literal("trend"),title:b.string().max(120).optional(),labels:b.array(b.string().max(60)).min(2).max(20).describe('Bucket labels (e.g. ["Week-3", "Week-2", "Week-1", "This wk"]).'),values:b.array(b.number()).min(2).max(20).describe("Numeric values, one per label. Must match labels.length."),highlight:b.enum(["last","max","min","none"]).default("last").optional().describe("Which bucket to visually highlight in the rendered card."),severity:b.enum(ae).optional()}).refine(s=>s.labels.length===s.values.length,{message:"labels.length must equal values.length"}),Yn=b.object({kind:b.literal("table"),title:b.string().max(120).optional(),headers:b.array(b.string().max(40)).min(1).max(8),rows:b.array(b.array(b.union([b.string().max(200),b.number()])).min(1).max(8)).max(40).describe("2D matrix. Each inner array must have headers.length entries.")}).refine(s=>s.rows.every(t=>t.length===s.headers.length),{message:"every row must have headers.length entries"}),Vn=b.object({kind:b.literal("callouts"),title:b.string().max(120).optional(),tone:b.enum(ae).default("info").optional(),items:b.array(b.string().min(1).max(600)).min(1).max(10).describe("Each item renders as a bullet with a severity emoji.")}),Zn=b.object({kind:b.literal("breakdown"),title:b.string().max(120).optional(),rows:b.array(b.object({label:b.string().min(1).max(80),value:b.string().min(1).max(80),sub:b.string().max(120).optional(),severity:b.enum(ae).optional()})).min(1).max(20)}),Qn=b.object({kind:b.literal("paragraph"),title:b.string().max(120).optional(),text:b.string().min(1).max(3e3)}),Xn=b.discriminatedUnion("kind",[Hn,Yn,Vn,Zn,Qn]),Be=b.object({title:b.string().min(1).max(200).describe('Card title (e.g. "Weekly AI Spend Report").'),subtitle:b.string().max(200).optional().describe('Date range or smaller header (e.g. "May 13 \u2014 May 20").'),headline:Wn,sections:b.array(Xn).max(20).default([]),footer:b.object({viewUrl:b.string().url().optional().describe('Optional "View in Zibby" button URL.'),rerunUrl:b.string().url().optional().describe('Optional "Run again" button URL.')}).optional()}),W=Object.freeze({ok:"\u{1F7E2}",info:"\u{1F535}",warn:"\u{1F7E0}",critical:"\u{1F534}"}),pr=Object.freeze({up:"\u2191",down:"\u2193",flat:"\u2192"}),ei=Object.freeze({ok:"green",info:"blue",warn:"orange",critical:"red"});function dr(s,t,e=12){if(!Number.isFinite(s)||!Number.isFinite(t)||t<=0)return"";let r=Math.max(0,Math.min(1,s/t)),n=Math.round(r*e);return"\u2593".repeat(n)+"\u2591".repeat(e-n)}function Fe(s,t){let e=String(s);return e.length>=t?e:e+" ".repeat(t-e.length)}function mr(s,t){let e=String(s);return e.length>=t?e:" ".repeat(t-e.length)+e}function fr({headers:s,rows:t}){let e=s.map((o,c)=>{let a=Math.max(String(o).length,...t.map(u=>String(u[c]??"").length));return Math.min(a,32)}),r=o=>o.map((c,a)=>Fe(c,e[a])).join(" "),n=e.map(o=>"\u2500".repeat(o)).join(" ");return"```\n"+[r(s),n,...t.map(o=>r(o))].join(`
|
|
663
663
|
`)+"\n```"}function ti(s){let t=Be.parse(s),e=[];e.push({type:"header",text:{type:"plain_text",text:t.title.slice(0,150),emoji:!0}}),t.subtitle&&e.push({type:"context",elements:[{type:"mrkdwn",text:t.subtitle}]});let r=[`*${t.headline.primary}*`];if(t.headline.delta){let i=pr[t.headline.delta.direction]||"",o=t.headline.delta.severity?W[t.headline.delta.severity]:"";r.push(`${i} ${t.headline.delta.value} ${o}`.trim())}let n=r.join(" ");t.headline.summary&&(n+=`
|
|
664
664
|
`+t.headline.summary),e.push({type:"section",text:{type:"mrkdwn",text:n}});for(let i of t.sections)switch(e.push({type:"divider"}),i.title&&e.push({type:"section",text:{type:"mrkdwn",text:`*${i.title}*`}}),i.kind){case"trend":{let o=Math.max(...i.values),c=i.labels.map((a,u)=>{let l=i.values[u],p=dr(l,o),m=(i.highlight==="last"&&u===i.labels.length-1||i.highlight==="max"&&l===o||i.highlight==="min"&&l===Math.min(...i.values))&&i.severity?` ${W[i.severity]}`:"";return`${Fe(a,10)} ${mr(l.toLocaleString(),8)} ${p}${m}`});e.push({type:"section",text:{type:"mrkdwn",text:"```\n"+c.join(`
|
|
665
665
|
`)+"\n```"}});break}case"table":{e.push({type:"section",text:{type:"mrkdwn",text:fr(i)}});break}case"callouts":{let o=W[i.tone||"info"];e.push({type:"section",text:{type:"mrkdwn",text:i.items.map(c=>`${o} ${c}`).join(`
|
|
@@ -669,4 +669,4 @@ _${c.sub}_`:""}${c.severity?` ${W[c.severity]}`:""}`}));for(let c=0;c<o.length;c
|
|
|
669
669
|
`+t.headline.summary),e.push({tag:"div",text:{tag:"lark_md",content:n}});for(let o of t.sections)switch(e.push({tag:"hr"}),o.title&&e.push({tag:"div",text:{tag:"lark_md",content:`**${o.title}**`}}),o.kind){case"trend":{let c=Math.max(...o.values),a=o.labels.map((u,l)=>{let p=o.values[l],d=dr(p,c),h=(o.highlight==="last"&&l===o.labels.length-1||o.highlight==="max"&&p===c||o.highlight==="min"&&p===Math.min(...o.values))&&o.severity?` ${W[o.severity]}`:"";return`${Fe(u,10)} ${mr(p.toLocaleString(),8)} ${d}${h}`});e.push({tag:"div",text:{tag:"lark_md",content:"```\n"+a.join(`
|
|
670
670
|
`)+"\n```"}});break}case"table":{e.push({tag:"div",text:{tag:"lark_md",content:fr(o)}});break}case"callouts":{let c=W[o.tone||"info"];e.push({tag:"div",text:{tag:"lark_md",content:o.items.map(a=>`${c} ${a}`).join(`
|
|
671
671
|
`)}});break}case"breakdown":{let c=o.rows.map(a=>{let u=a.severity?` ${W[a.severity]}`:"",l=a.sub?` *${a.sub}*`:"";return`**${a.label}** ${a.value}${l}${u}`});e.push({tag:"div",text:{tag:"lark_md",content:c.join(`
|
|
672
|
-
`)}});break}case"paragraph":{e.push({tag:"div",text:{tag:"lark_md",content:o.text}});break}}if(t.footer&&(t.footer.viewUrl||t.footer.rerunUrl)){let o=[];t.footer.viewUrl&&o.push({tag:"button",text:{tag:"plain_text",content:"View in Zibby"},url:t.footer.viewUrl,type:"primary"}),t.footer.rerunUrl&&o.push({tag:"button",text:{tag:"plain_text",content:"Run again"},url:t.footer.rerunUrl,type:"default"}),e.push({tag:"hr"}),e.push({tag:"action",actions:o})}let i="blue";return t.headline.delta?.severity&&(i=ei[t.headline.delta.severity]||"blue"),{config:{wide_screen_mode:!0},header:{title:{tag:"plain_text",content:t.title.slice(0,200)},subtitle:t.subtitle?{tag:"plain_text",content:t.subtitle.slice(0,200)}:void 0,template:i},elements:e}}var Co=Object.freeze({ok:"green_background",info:"blue_background",warn:"orange_background",critical:"red_background"}),Lo=Object.freeze({ok:"\u{1F7E2}",info:"\u2139\uFE0F",warn:"\u26A0\uFE0F",critical:"\u{1F6A8}"});import{createRequire as si}from"module";import{fileURLToPath as ni}from"url";import{registerHandlers as ii}from"@zibby/core/function-skill-registry.js";import{registerSkill as oi}from"@zibby/agent-workflow";var ai=si(import.meta.url);function ci(){try{return ai.resolve("@zibby/core/function-bridge.js")}catch{return null}}var li=import.meta.url;function ui(){let s=Error.prepareStackTrace;try{Error.prepareStackTrace=(r,n)=>n;let e=new Error().stack;for(let r=2;r<e.length;r++){let n=e[r].getFileName();if(n&&n!==li&&!n.startsWith("node:"))return n.startsWith("file://")?ni(n):n}return null}finally{Error.prepareStackTrace=s}}function pi(s){if(!s||typeof s!="object")return{type:"object",properties:{},required:[]};let t={},e=[];for(let[r,n]of Object.entries(s))if(typeof n=="string")t[r]={type:n},e.push(r);else{let{required:i,...o}=n;t[r]=o,i!==!1&&e.push(r)}return{type:"object",properties:t,required:e}}function di(s,t,e){if(typeof e.handler!="function")throw new Error(`Skill "${s}" must have a handler function`);let r={[s]:e.handler},n=[{name:s,description:e.description||"",input_schema:pi(e.input)}];return ii(s,r,n),{id:s,type:"function",serverName:s,allowedTools:[`mcp__${s}__*`],description:e.description||`Function skill: ${s}`,envKeys:[],tools:n,resolve(){let i=ci();return i?{command:"node",args:[i,t,s]}:null}}}function mi(s,t){return{id:s,type:"mcp",serverName:t.serverName||s,allowedTools:t.allowedTools||[`mcp__${t.serverName||s}__*`],description:t.description||`MCP skill: ${s}`,envKeys:t.envKeys||[],tools:t.tools||[],resolve:t.resolve,...t.cursorKey&&{cursorKey:t.cursorKey},...t.sessionEnvKey&&{sessionEnvKey:t.sessionEnvKey}}}function hr(s,t){let e;if("handler"in t){if(typeof t.handler!="function")throw new Error(`Skill "${s}" must have a handler function`);let r=ui();if(!r)throw new Error(`Could not resolve caller file for skill "${s}".`);e=di(s,r,t)}else if(typeof t.resolve=="function")e=mi(s,t);else throw new Error(`Skill "${s}" must have either a handler (function skill) or resolve (MCP skill).`);return oi(e),e}var fi=hr;import{registerSkill as ua,getSkill as pa,hasSkill as da,getAllSkills as ma,listSkillIds as fa}from"@zibby/agent-workflow";R(Ge);R(Ye);R(
|
|
672
|
+
`)}});break}case"paragraph":{e.push({tag:"div",text:{tag:"lark_md",content:o.text}});break}}if(t.footer&&(t.footer.viewUrl||t.footer.rerunUrl)){let o=[];t.footer.viewUrl&&o.push({tag:"button",text:{tag:"plain_text",content:"View in Zibby"},url:t.footer.viewUrl,type:"primary"}),t.footer.rerunUrl&&o.push({tag:"button",text:{tag:"plain_text",content:"Run again"},url:t.footer.rerunUrl,type:"default"}),e.push({tag:"hr"}),e.push({tag:"action",actions:o})}let i="blue";return t.headline.delta?.severity&&(i=ei[t.headline.delta.severity]||"blue"),{config:{wide_screen_mode:!0},header:{title:{tag:"plain_text",content:t.title.slice(0,200)},subtitle:t.subtitle?{tag:"plain_text",content:t.subtitle.slice(0,200)}:void 0,template:i},elements:e}}var Co=Object.freeze({ok:"green_background",info:"blue_background",warn:"orange_background",critical:"red_background"}),Lo=Object.freeze({ok:"\u{1F7E2}",info:"\u2139\uFE0F",warn:"\u26A0\uFE0F",critical:"\u{1F6A8}"});import{createRequire as si}from"module";import{fileURLToPath as ni}from"url";import{registerHandlers as ii}from"@zibby/core/function-skill-registry.js";import{registerSkill as oi}from"@zibby/agent-workflow";var ai=si(import.meta.url);function ci(){try{return ai.resolve("@zibby/core/function-bridge.js")}catch{return null}}var li=import.meta.url;function ui(){let s=Error.prepareStackTrace;try{Error.prepareStackTrace=(r,n)=>n;let e=new Error().stack;for(let r=2;r<e.length;r++){let n=e[r].getFileName();if(n&&n!==li&&!n.startsWith("node:"))return n.startsWith("file://")?ni(n):n}return null}finally{Error.prepareStackTrace=s}}function pi(s){if(!s||typeof s!="object")return{type:"object",properties:{},required:[]};let t={},e=[];for(let[r,n]of Object.entries(s))if(typeof n=="string")t[r]={type:n},e.push(r);else{let{required:i,...o}=n;t[r]=o,i!==!1&&e.push(r)}return{type:"object",properties:t,required:e}}function di(s,t,e){if(typeof e.handler!="function")throw new Error(`Skill "${s}" must have a handler function`);let r={[s]:e.handler},n=[{name:s,description:e.description||"",input_schema:pi(e.input)}];return ii(s,r,n),{id:s,type:"function",serverName:s,allowedTools:[`mcp__${s}__*`],description:e.description||`Function skill: ${s}`,envKeys:[],tools:n,resolve(){let i=ci();return i?{command:"node",args:[i,t,s]}:null}}}function mi(s,t){return{id:s,type:"mcp",serverName:t.serverName||s,allowedTools:t.allowedTools||[`mcp__${t.serverName||s}__*`],description:t.description||`MCP skill: ${s}`,envKeys:t.envKeys||[],tools:t.tools||[],resolve:t.resolve,...t.cursorKey&&{cursorKey:t.cursorKey},...t.sessionEnvKey&&{sessionEnvKey:t.sessionEnvKey}}}function hr(s,t){let e;if("handler"in t){if(typeof t.handler!="function")throw new Error(`Skill "${s}" must have a handler function`);let r=ui();if(!r)throw new Error(`Could not resolve caller file for skill "${s}".`);e=di(s,r,t)}else if(typeof t.resolve=="function")e=mi(s,t);else throw new Error(`Skill "${s}" must have either a handler (function skill) or resolve (MCP skill).`);return oi(e),e}var fi=hr;import{registerSkill as ua,getSkill as pa,hasSkill as da,getAllSkills as ma,listSkillIds as fa}from"@zibby/agent-workflow";R(Ge);R(Ye);R(Ze);R(D);R(P);R(Xe);R(pe);R(tt);R(Ot);R(xt);R(st);R(ct);R(Gt);R(ir);R(Me);R(qe);R(Ke);R({...D,id:"slack_notify"});var na={BROWSER:"browser",JIRA:"jira",GITHUB:"github",GIT:"git",SLACK:"slack",LARK:"lark",CHAT_NOTIFY:"chat_notify",SENTRY:"sentry",MEMORY:"memory",RUNNER:"runner",SKILL_INSTALLER:"skill-installer",CORE_TOOLS:"core-tools",CHAT_MEMORY:"chat-memory",WORKFLOW_BUILDER:"workflow-builder",OPENAI_BILLING:"openai_billing",ANTHROPIC_BILLING:"anthropic_billing",CURSOR_ADMIN:"cursor_admin"};export{A as INTEGRATIONS,Or as INTEGRATION_REGISTRY,ae as REPORT_SEVERITIES,na as SKILLS,qe as anthropicBillingSkill,Ge as browserSkill,Gt as chatMemorySkill,Xe as chatNotifySkill,ct as coreToolsSkill,Ke as cursorAdminSkill,Fn as fetchAllProviders,lr as fetchAnthropicCosts,Bn as fetchAnthropicWorkspaces,ur as fetchCursorSpend,cr as fetchOpenAICosts,Kn as fetchOpenAIProjects,fi as functionSkill,ma as getAllSkills,pa as getSkill,xt as gitSkill,Ze as githubSkill,zn as groupByKey,da as hasSkill,Ye as jiraSkill,P as larkSkill,fa as listSkillIds,Gn as meanStddev,tt as memorySkill,Me as openaiBillingSkill,ua as registerSkill,Be as reportObjectSchema,ti as reportToBlockKit,ri as reportToLarkCard,Ot as runnerSkill,pe as sentrySkill,hr as skill,st as skillInstallerSkill,D as slackSkill,Ot as testRunnerSkill,ir as workflowBuilderSkill};
|