badgerclaw 0.2.16 → 0.2.17
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/index.js +3 -3
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
"use strict";var wo=Object.create;var pe=Object.defineProperty;var _o=Object.getOwnPropertyDescriptor;var bo=Object.getOwnPropertyNames;var $o=Object.getPrototypeOf,So=Object.prototype.hasOwnProperty;var Co=(e,t)=>()=>(e&&(t=e(e=0)),t);var vo=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),Io=(e,t)=>{for(var o in t)pe(e,o,{get:t[o],enumerable:!0})},dt=(e,t,o,n)=>{if(t&&typeof t=="object"||typeof t=="function")for(let s of bo(t))!So.call(e,s)&&s!==o&&pe(e,s,{get:()=>t[s],enumerable:!(n=_o(t,s))||n.enumerable});return e};var c=(e,t,o)=>(o=e!=null?wo($o(e)):{},dt(t||!e||!e.__esModule?pe(o,"default",{value:e,enumerable:!0}):o,e)),mt=e=>dt(pe({},"__esModule",{value:!0}),e);var F=vo((Qs,Uo)=>{Uo.exports={name:"badgerclaw",version:"0.2.
|
|
2
|
+
"use strict";var wo=Object.create;var pe=Object.defineProperty;var _o=Object.getOwnPropertyDescriptor;var bo=Object.getOwnPropertyNames;var $o=Object.getPrototypeOf,So=Object.prototype.hasOwnProperty;var Co=(e,t)=>()=>(e&&(t=e(e=0)),t);var vo=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),Io=(e,t)=>{for(var o in t)pe(e,o,{get:t[o],enumerable:!0})},dt=(e,t,o,n)=>{if(t&&typeof t=="object"||typeof t=="function")for(let s of bo(t))!So.call(e,s)&&s!==o&&pe(e,s,{get:()=>t[s],enumerable:!(n=_o(t,s))||n.enumerable});return e};var c=(e,t,o)=>(o=e!=null?wo($o(e)):{},dt(t||!e||!e.__esModule?pe(o,"default",{value:e,enumerable:!0}):o,e)),mt=e=>dt(pe({},"__esModule",{value:!0}),e);var F=vo((Qs,Uo)=>{Uo.exports={name:"badgerclaw",version:"0.2.17",description:"BadgerClaw CLI \u2014 one-click bot provisioning",main:"dist/index.js",bin:{badgerclaw:"./dist/index.js"},files:["dist/","README.md"],scripts:{build:"node build.mjs",test:'echo "Error: no test specified" && exit 1'},keywords:[],author:"",license:"ISC",dependencies:{"@modelcontextprotocol/sdk":"^1.29.0",axios:"^1.6.0",chalk:"^4.1.2",commander:"^12.0.0",eventsource:"^1.1.2",open:"^8.4.2",ora:"^5.4.1"},devDependencies:{"@types/eventsource":"^1.1.15","@types/node":"^20.0.0",esbuild:"^0.28.0",typescript:"^5.3.0"}}});var rt={};Io(rt,{getActiveSessions:()=>nt,launchClaudeCode:()=>tt,launchMCPServer:()=>et,recordSession:()=>ot,stopSession:()=>st});function Ke(){try{if(I.default.existsSync(Ye))return JSON.parse(I.default.readFileSync(Ye,"utf-8"))}catch{}return[]}function Xe(e){I.default.mkdirSync(H,{recursive:!0}),I.default.writeFileSync(Ye,JSON.stringify(e,null,2))}function Ae(e){try{return process.kill(e,0),!0}catch{return!1}}function zo(){if(process.env.TMUX)return"tmux";try{if((0,U.execSync)(`osascript -e 'tell application "System Events" to (name of processes) contains "iTerm2"'`,{stdio:"pipe",timeout:3e3}),I.default.existsSync("/Applications/iTerm.app"))return"iterm2"}catch{}return process.platform==="darwin"?"terminal":"direct"}function Qe(e,t){let o=D.default.join(H,`claude-launcher-${t}.exp`),n=D.default.join(H,`pending-input-${t}.txt`);I.default.writeFileSync(n,"","utf-8");let s=`#!/usr/bin/expect -f
|
|
3
3
|
set timeout -1
|
|
4
4
|
set msgfile "${n}"
|
|
5
5
|
|
|
@@ -112,7 +112,7 @@ Reply using the reply tool with botId="..." and roomId="..."
|
|
|
112
112
|
ALWAYS use the \`reply\` MCP tool to send your response back to the chat room.
|
|
113
113
|
Use the exact botId and roomId provided in the message \u2014 do not modify them.
|
|
114
114
|
Keep responses concise and helpful.
|
|
115
|
-
`),e}var Ee=b.default.join(me.default.homedir(),".badgerclaw","claude-code.lock");function ns(){try{u.default.mkdirSync(b.default.dirname(Ee),{recursive:!0});try{let e=u.default.statSync(Ee);if(Date.now()-e.mtimeMs<6e4)return!1;u.default.unlinkSync(Ee)}catch{}return u.default.writeFileSync(Ee,String(process.pid),{flag:"wx"}),!0}catch{return!1}}async function rs(e){if(!e?.bot_id)return{success:!1,message:"Missing bot_id in payload"};if(!ns())return{success:!0,message:"Claude Code start already in progress, skipping duplicate"};try{let t=e.bot_user_id?.startsWith("@")?e.bot_user_id:await ue(e.bot_id),{launchClaudeCode:o,recordSession:n,getActiveSessions:s,stopSession:r}=(Re(),mt(rt)),a=new Set;try{let E=(await k.default.get("http://localhost:7331/claude-code/status",{timeout:5e3})).data?.bots||[];for(let _ of E)if(_.botId!==t&&_.enabled!==!1){let S=_.port;typeof S=="number"&&S>=7332&&a.add(S)}}catch{}for(let f=7332;f<=7340;f++)try{(0,C.execSync)(`lsof -ti :${f} >/dev/null 2>&1`,{stdio:"pipe",timeout:2e3}),a.add(f)}catch{}let i=7332;for(;a.has(i);)i++;let l=s();for(let f of l)f.port===i&&(console.log(`[command-executor] Stopping existing session ${f.sessionId} on port ${i}`),r(f.sessionId));try{(0,C.execSync)(`lsof -ti :${i} | xargs kill -9 2>/dev/null`,{stdio:"pipe",timeout:3e3})}catch{}let m=e.session_id||`session-${Date.now()}`;await k.default.post("http://localhost:7331/claude-code/toggle",{botId:t,enabled:!0,sessionId:m,port:i},{timeout:1e4});let w=ss();return o({sessionId:m,port:i,projectDir:w}),n(m,0,0,i,w),{success:!0,message:`Claude Code started globally for bot ${t} (session: ${m}, port: ${i})`}}catch(t){return{success:!1,message:`Failed to start Claude Code: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ue(e){if(e.startsWith("@"))return e;try{let o=(await k.default.get("http://localhost:7331/health",{timeout:1e4})).data?.bots||[],n=b.default.join(me.default.homedir(),".badgerclaw","bot-mapping.json");if(u.default.existsSync(n)){let s=JSON.parse(u.default.readFileSync(n,"utf-8"));if(s[e])return s[e]}return o.length===1?o[0].botId:(console.warn(`[command-executor] Could not resolve bot UUID ${e} to Matrix ID, trying as-is`),e)}catch{return e}}async function as(e){if(!e?.bot_id)return{success:!1,message:"Missing bot_id in payload"};try{let t=e.bot_user_id?.startsWith("@")?e.bot_user_id:await ue(e.bot_id),o=e.session_id;if(!o)try{let n=await k.default.get("http://localhost:7331/claude-code/status",{timeout:5e3}),r=(n.data?.bots||n.data?.rooms||[]).find(a=>a.botId===t);r&&(o=r.sessionId)}catch{}if(await k.default.post("http://localhost:7331/claude-code/toggle",{botId:t,enabled:!1},{timeout:1e4}),o){let{stopSession:n}=(Re(),mt(rt));n(o)}return{success:!0,message:"Claude Code stopped"}}catch(t){return{success:!1,message:`Failed to stop Claude Code: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function is(e){try{let t=e?.bot_user_id||e?.bot_id;if(!t)return{success:!1,message:"Missing bot_id in payload"};let o=t;o.startsWith("@")||(o=await ue(t));let n=e?.session_id||`cc-${Date.now()}`;return await k.default.post("http://localhost:7331/claude-code/toggle",{botId:o,enabled:!0,sessionId:n,mode:"primary",tools:e?.tools},{timeout:1e4}),console.log(`[claude-control] Primary mode enabled for ${o} (session: ${n})`),{success:!0,message:`Claude Control (primary) started for ${o}`}}catch(t){return{success:!1,message:`Failed to start Claude Control: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function cs(e){try{let t=e?.bot_user_id||e?.bot_id;if(!t)return{success:!1,message:"Missing bot_id in payload"};let o=t;return o.startsWith("@")||(o=await ue(t)),await k.default.post("http://localhost:7331/claude-code/toggle",{botId:o,enabled:!1},{timeout:1e4}),console.log(`[claude-control] Primary mode disabled for ${o}`),{success:!0,message:`Claude Control stopped for ${o}`}}catch(t){return{success:!1,message:`Failed to stop Claude Control: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ls(e){try{let t=e?.bot_user_id||e?.bot_id;if(!t||!e?.tools)return{success:!1,message:"Missing bot_id or tools in payload"};let o=t;return o.startsWith("@")||(o=await ue(t)),await k.default.post("http://localhost:7331/claude-code/tools",{botId:o,tools:e.tools},{timeout:1e4}),{success:!0,message:`Tools updated for ${o}`}}catch(t){return{success:!1,message:`Failed to update tools: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ds(e){if(!e?.bot_user_id||!e?.shared_with_user_id)return{success:!1,message:"Missing bot_user_id or shared_with_user_id"};try{return await k.default.post("http://localhost:7331/bot-share/notify",{botId:e.bot_user_id,sharedWithUserId:e.shared_with_user_id,ownerDisplayName:e.owner_display_name||void 0,botDisplayName:e.bot_name||void 0,expiresAt:e.expires_at||void 0},{timeout:1e4}),console.log(`[bot-share] Notify: ${e.shared_with_user_id} can now access ${e.bot_user_id}`),{success:!0,message:`Share notification sent for ${e.shared_with_user_id}`}}catch(t){return{success:!1,message:`Failed to notify share: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ms(e){if(!e?.bot_user_id||!e?.shared_with_user_id)return{success:!1,message:"Missing bot_user_id or shared_with_user_id"};try{return await k.default.post("http://localhost:7331/bot-share/revoked",{botId:e.bot_user_id,sharedWithUserId:e.shared_with_user_id,botDisplayName:e.bot_name||void 0},{timeout:1e4}),console.log(`[bot-share] Revoked: ${e.shared_with_user_id} lost access to ${e.bot_user_id}`),{success:!0,message:`Share revocation sent for ${e.shared_with_user_id}`}}catch(t){return{success:!1,message:`Failed to revoke share: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function us(e){if(!e?.bot_user_id||!e?.shared_with_user_id)return{success:!1,message:"Missing bot_user_id or shared_with_user_id"};try{return await k.default.post("http://localhost:7331/bot-share/expired",{botId:e.bot_user_id,sharedWithUserId:e.shared_with_user_id,botDisplayName:e.bot_name||void 0},{timeout:1e4}),console.log(`[bot-share] Expired: ${e.shared_with_user_id} access to ${e.bot_user_id}`),{success:!0,message:`Share expiry sent for ${e.shared_with_user_id}`}}catch(t){return{success:!1,message:`Failed to expire share: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function gs(e){if(!e?.bot_id||!e?.room_id)return{success:!1,message:"Missing bot_id or room_id in payload"};try{return{success:!0,message:(await k.default.post("http://localhost:7331/leave-room",{botId:e.bot_id,roomId:e.room_id},{timeout:15e3})).data?.message||"Left room"}}catch(t){return{success:!1,message:`Failed to leave room: ${t.response?.data?.message||t.message||"Unknown error"}`}}}var Yt=3e4,ps=15e4,Zt="/api/v1/dashboard/heartbeat",fs="/api/v1/
|
|
115
|
+
`),e}var Ee=b.default.join(me.default.homedir(),".badgerclaw","claude-code.lock");function ns(){try{u.default.mkdirSync(b.default.dirname(Ee),{recursive:!0});try{let e=u.default.statSync(Ee);if(Date.now()-e.mtimeMs<6e4)return!1;u.default.unlinkSync(Ee)}catch{}return u.default.writeFileSync(Ee,String(process.pid),{flag:"wx"}),!0}catch{return!1}}async function rs(e){if(!e?.bot_id)return{success:!1,message:"Missing bot_id in payload"};if(!ns())return{success:!0,message:"Claude Code start already in progress, skipping duplicate"};try{let t=e.bot_user_id?.startsWith("@")?e.bot_user_id:await ue(e.bot_id),{launchClaudeCode:o,recordSession:n,getActiveSessions:s,stopSession:r}=(Re(),mt(rt)),a=new Set;try{let E=(await k.default.get("http://localhost:7331/claude-code/status",{timeout:5e3})).data?.bots||[];for(let _ of E)if(_.botId!==t&&_.enabled!==!1){let S=_.port;typeof S=="number"&&S>=7332&&a.add(S)}}catch{}for(let f=7332;f<=7340;f++)try{(0,C.execSync)(`lsof -ti :${f} >/dev/null 2>&1`,{stdio:"pipe",timeout:2e3}),a.add(f)}catch{}let i=7332;for(;a.has(i);)i++;let l=s();for(let f of l)f.port===i&&(console.log(`[command-executor] Stopping existing session ${f.sessionId} on port ${i}`),r(f.sessionId));try{(0,C.execSync)(`lsof -ti :${i} | xargs kill -9 2>/dev/null`,{stdio:"pipe",timeout:3e3})}catch{}let m=e.session_id||`session-${Date.now()}`;await k.default.post("http://localhost:7331/claude-code/toggle",{botId:t,enabled:!0,sessionId:m,port:i},{timeout:1e4});let w=ss();return o({sessionId:m,port:i,projectDir:w}),n(m,0,0,i,w),{success:!0,message:`Claude Code started globally for bot ${t} (session: ${m}, port: ${i})`}}catch(t){return{success:!1,message:`Failed to start Claude Code: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ue(e){if(e.startsWith("@"))return e;try{let o=(await k.default.get("http://localhost:7331/health",{timeout:1e4})).data?.bots||[],n=b.default.join(me.default.homedir(),".badgerclaw","bot-mapping.json");if(u.default.existsSync(n)){let s=JSON.parse(u.default.readFileSync(n,"utf-8"));if(s[e])return s[e]}return o.length===1?o[0].botId:(console.warn(`[command-executor] Could not resolve bot UUID ${e} to Matrix ID, trying as-is`),e)}catch{return e}}async function as(e){if(!e?.bot_id)return{success:!1,message:"Missing bot_id in payload"};try{let t=e.bot_user_id?.startsWith("@")?e.bot_user_id:await ue(e.bot_id),o=e.session_id;if(!o)try{let n=await k.default.get("http://localhost:7331/claude-code/status",{timeout:5e3}),r=(n.data?.bots||n.data?.rooms||[]).find(a=>a.botId===t);r&&(o=r.sessionId)}catch{}if(await k.default.post("http://localhost:7331/claude-code/toggle",{botId:t,enabled:!1},{timeout:1e4}),o){let{stopSession:n}=(Re(),mt(rt));n(o)}return{success:!0,message:"Claude Code stopped"}}catch(t){return{success:!1,message:`Failed to stop Claude Code: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function is(e){try{let t=e?.bot_user_id||e?.bot_id;if(!t)return{success:!1,message:"Missing bot_id in payload"};let o=t;o.startsWith("@")||(o=await ue(t));let n=e?.session_id||`cc-${Date.now()}`;return await k.default.post("http://localhost:7331/claude-code/toggle",{botId:o,enabled:!0,sessionId:n,mode:"primary",tools:e?.tools},{timeout:1e4}),console.log(`[claude-control] Primary mode enabled for ${o} (session: ${n})`),{success:!0,message:`Claude Control (primary) started for ${o}`}}catch(t){return{success:!1,message:`Failed to start Claude Control: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function cs(e){try{let t=e?.bot_user_id||e?.bot_id;if(!t)return{success:!1,message:"Missing bot_id in payload"};let o=t;return o.startsWith("@")||(o=await ue(t)),await k.default.post("http://localhost:7331/claude-code/toggle",{botId:o,enabled:!1},{timeout:1e4}),console.log(`[claude-control] Primary mode disabled for ${o}`),{success:!0,message:`Claude Control stopped for ${o}`}}catch(t){return{success:!1,message:`Failed to stop Claude Control: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ls(e){try{let t=e?.bot_user_id||e?.bot_id;if(!t||!e?.tools)return{success:!1,message:"Missing bot_id or tools in payload"};let o=t;return o.startsWith("@")||(o=await ue(t)),await k.default.post("http://localhost:7331/claude-code/tools",{botId:o,tools:e.tools},{timeout:1e4}),{success:!0,message:`Tools updated for ${o}`}}catch(t){return{success:!1,message:`Failed to update tools: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ds(e){if(!e?.bot_user_id||!e?.shared_with_user_id)return{success:!1,message:"Missing bot_user_id or shared_with_user_id"};try{return await k.default.post("http://localhost:7331/bot-share/notify",{botId:e.bot_user_id,sharedWithUserId:e.shared_with_user_id,ownerDisplayName:e.owner_display_name||void 0,botDisplayName:e.bot_name||void 0,expiresAt:e.expires_at||void 0},{timeout:1e4}),console.log(`[bot-share] Notify: ${e.shared_with_user_id} can now access ${e.bot_user_id}`),{success:!0,message:`Share notification sent for ${e.shared_with_user_id}`}}catch(t){return{success:!1,message:`Failed to notify share: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function ms(e){if(!e?.bot_user_id||!e?.shared_with_user_id)return{success:!1,message:"Missing bot_user_id or shared_with_user_id"};try{return await k.default.post("http://localhost:7331/bot-share/revoked",{botId:e.bot_user_id,sharedWithUserId:e.shared_with_user_id,botDisplayName:e.bot_name||void 0},{timeout:1e4}),console.log(`[bot-share] Revoked: ${e.shared_with_user_id} lost access to ${e.bot_user_id}`),{success:!0,message:`Share revocation sent for ${e.shared_with_user_id}`}}catch(t){return{success:!1,message:`Failed to revoke share: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function us(e){if(!e?.bot_user_id||!e?.shared_with_user_id)return{success:!1,message:"Missing bot_user_id or shared_with_user_id"};try{return await k.default.post("http://localhost:7331/bot-share/expired",{botId:e.bot_user_id,sharedWithUserId:e.shared_with_user_id,botDisplayName:e.bot_name||void 0},{timeout:1e4}),console.log(`[bot-share] Expired: ${e.shared_with_user_id} access to ${e.bot_user_id}`),{success:!0,message:`Share expiry sent for ${e.shared_with_user_id}`}}catch(t){return{success:!1,message:`Failed to expire share: ${t.response?.data?.message||t.message||"Unknown error"}`}}}async function gs(e){if(!e?.bot_id||!e?.room_id)return{success:!1,message:"Missing bot_id or room_id in payload"};try{return{success:!0,message:(await k.default.post("http://localhost:7331/leave-room",{botId:e.bot_id,roomId:e.room_id},{timeout:15e3})).data?.message||"Left room"}}catch(t){return{success:!1,message:`Failed to leave room: ${t.response?.data?.message||t.message||"Unknown error"}`}}}var Yt=3e4,ps=15e4,Zt="/api/v1/dashboard/heartbeat",fs="/api/v1/me/active-shares",De=new Map,hs=3e4;function Ne(e,t){if(e!=="start_claude_code"&&e!=="stop_claude_code")return!1;let o=`${e}:${t||"unknown"}`,n=Date.now();for(let[s,r]of De)n-r>hs&&De.delete(s);return De.has(o)?!0:(De.set(o,n),!1)}function eo(e,t){let o=xe();return{instance_id:de(),timestamp:new Date().toISOString(),cli_version:t,plugin_version:e.pluginVersion||"unknown",hostname:o.hostname,os:o.os,arch:o.arch,uptime_seconds:o.uptimeSeconds,mem_free_mb:o.memFreeMb,gateway_status:e.status,gateway_pid:e.pid,last_gateway_restart:e.lastRestart,bots:e.bots.map(n=>({bot_id:n.botId,bot_username:n.botUsername,status:n.status,activity_state:n.activityState||"idle",active_task:n.activeTask||null,last_activity_at:n.lastActivity,messages_received:n.messagesReceived,messages_sent:n.messagesSent,chunked_messages:n.chunkedMessages,total_chunks_sent:n.totalChunksSent,errors:n.errors,last_error:n.lastError,last_error_at:n.lastErrorAt,uptime_seconds:n.uptimeSeconds,rooms_active:n.roomsActive,room_details:(n.roomDetails||[]).map(s=>({roomId:s.roomId,roomName:s.roomName,workspaceName:s.workspaceName||null,messagesInRoom:s.messagesInRoom,lastActivityInRoom:s.lastActivityInRoom}))}))}}async function x(){let{version:e}=F(),t=await le(),o=eo(t,e);await v().post(Zt,o)}function ys(e,t){if(!e||e.status!==t.status)return!0;let o=new Map(e.bots.map(n=>[n.botId,n]));for(let n of t.bots){let s=o.get(n.botId);if(!s||s.status!==n.status||s.errors!==n.errors)return!0}return e.bots.length!==t.bots.length}async function Kt(){try{let o=(await v().get(fs)).data?.shares||[],n={shares:o.map(s=>({botUserId:s.bot_user_id,sharedWithUserId:s.shared_with_user_id,expiresAt:s.expires_at||null,shareId:s.share_id}))};await Qt.default.post("http://localhost:7331/bot-share/sync",n,{timeout:1e4}),console.log(d.default.dim(` [${y()}] Share sync: ${o.length} active share(s) pushed to plugin`))}catch(e){console.log(d.default.dim(` [${y()}] Share sync failed (non-fatal): ${e.message}`))}}var to=new Xt.Command("heartbeat").description("Run heartbeat daemon \u2014 reports telemetry to the BadgerClaw dashboard every 30s").action(async()=>{let e=h();e||(console.log(d.default.yellow("Not logged in. Run `badgerclaw login` first.")),process.exit(1));let{version:t}=F();console.log(d.default.green(`Heartbeat daemon started (v${t})`)),console.log(d.default.dim(` Instance: ${de()}`)),console.log(d.default.dim(` Interval: ${Yt/1e3}s`)),console.log(d.default.dim(` Press Ctrl+C to stop.
|
|
116
116
|
`));let o=null,n=async()=>{try{await oo();let s=await le(),r=eo(s,t);ys(o,s)&&o!==null&&console.log(d.default.cyan(` [${new Date().toISOString()}] State change detected \u2014 pushing immediately`));let i=v(),l=await i.post(Zt,r),m=s.bots.length,w=s.bots.filter(_=>_.status==="running").length;console.log(d.default.dim(` [${new Date().toISOString()}] gateway=${s.status} bots=${w}/${m} running mem=${r.mem_free_mb}MB free`)),o=s;let f=l.data?.pending_commands||[],E=!1;for(let _ of f)try{if(console.log(d.default.cyan(` [${new Date().toISOString()}] Received command: ${_.command_type} (${_.id})`)),await i.post(`/api/v1/dashboard/commands/${_.id}/ack`),Ne(_.command_type,_.payload?.bot_id)){console.log(d.default.dim(` [${new Date().toISOString()}] Skipping duplicate ${_.command_type}`));continue}let S=await M(_);console.log(S.success?d.default.green(` [${new Date().toISOString()}] ${S.message}`):d.default.red(` [${new Date().toISOString()}] ${S.message}`)),await i.post(`/api/v1/dashboard/commands/${_.id}/result`,{status:S.success?"success":"failed",result:S.message,new_version:S.newVersion||null}),_.command_type==="update_cli"&&S.success&&(E=!0)}catch(S){console.log(d.default.dim(` [${new Date().toISOString()}] Command ${_.id} error: ${S.message}`))}if(E){console.log(d.default.cyan(` [${new Date().toISOString()}] CLI updated \u2014 restarting in 2s...`));try{await x()}catch{}setTimeout(()=>process.exit(0),2e3);return}}catch(s){console.log(d.default.dim(` [${new Date().toISOString()}] Heartbeat failed: ${s.message}`))}};await n(),await Kt(),setInterval(n,Yt),setInterval(Kt,ps),_s(e.access_token,t),await new Promise(()=>{})});async function oo(){let e=h();if(!e?.expires_at)return;let t=new Date(e.expires_at).getTime(),o=Date.now(),n=300*1e3;if(t-o<n){console.log(d.default.dim(` [${y()}] Token expires soon \u2014 refreshing proactively...`));let s=await Be();console.log(s?d.default.green(` [${y()}] Token refreshed successfully`):d.default.yellow(` [${y()}] Token refresh failed \u2014 no refresh_token available`))}}var ws=2700*1e3;function _s(e,t){let o=3e3,n=3e4,s=null;async function r(){try{await oo();let i=h()?.access_token||e,l=require("eventsource"),m=new l(`${B}/api/v1/openclaw/events`,{headers:{Authorization:`Bearer ${i}`}});m.onopen=()=>{o=3e3,console.log(d.default.dim(` [${y()}] SSE connected \u2014 listening for real-time events`)),s&&clearTimeout(s),s=setTimeout(()=>{console.log(d.default.dim(` [${y()}] SSE proactive reconnect \u2014 refreshing token before expiry`)),m.close(),r()},ws)},m.onmessage=async w=>{try{let f=JSON.parse(w.data);await bs(f)}catch{}},m.onerror=async()=>{m.close(),s&&(clearTimeout(s),s=null),console.log(d.default.dim(` [${y()}] SSE disconnected \u2014 reconnecting in ${o/1e3}s...`)),setTimeout(r,o),o=Math.min(o*1.5,n)}}catch(a){console.log(d.default.dim(` [${y()}] SSE connection failed: ${a.message}`)),setTimeout(r,o),o=Math.min(o*1.5,n)}}r()}function y(){return new Date().toISOString()}async function bs(e){if(e.type==="pair"){console.log(d.default.cyan(` [${y()}] Pair event: ${e.bot_name}`)),await te(e.pair_code,e.bot_name,e.bot_user_id,!0);try{let{execSync:t}=await import("child_process");t("openclaw gateway restart",{stdio:"ignore"}),console.log(d.default.green(` [${y()}] Gateway restarted \u2014 ${e.bot_name} is live`))}catch{}try{await x()}catch{}}else if(e.type==="bot.delete"&&e.bot_user_id){let t=e.bot_user_id,o=t.split(":")[0].replace("@","").replace(/_bot$/,"");console.log(d.default.yellow(` [${y()}] Bot deleted: ${t}`));try{let n=await import("fs"),s=await import("path"),r=await import("os"),a=s.join(r.homedir(),".openclaw","openclaw.json"),i=JSON.parse(n.readFileSync(a,"utf-8")),l=!1;if(i.channels?.badgerclaw?.accounts?.[o]&&(delete i.channels.badgerclaw.accounts[o],l=!0),i.agents?.list){let m=i.agents.list.length;i.agents.list=i.agents.list.filter(w=>w.id!==o),i.agents.list.length!==m&&(l=!0)}if(l){n.writeFileSync(a,JSON.stringify(i,null,2)),console.log(d.default.green(` [${y()}] Removed "${o}" from openclaw.json`));let{execSync:m}=await import("child_process");try{m("openclaw gateway restart",{stdio:"ignore"}),console.log(d.default.green(` [${y()}] Gateway restarted`))}catch{}}}catch(n){console.log(d.default.red(` [${y()}] Failed to clean up bot: ${n}`))}try{await x()}catch{}}else if(e.type==="gateway-restart"){console.log(d.default.cyan(` [${y()}] Remote gateway-restart received`));let t=await K();console.log(t.success?d.default.green(` [${y()}] Gateway restarted`):d.default.red(` [${y()}] Gateway restart failed: ${t.message}`));try{await x()}catch{}}else if(e.type==="claude_code.start"&&e.bot_id){if(console.log(d.default.cyan(` [${y()}] Claude Code START: bot=${e.bot_id}`)),Ne("start_claude_code",e.bot_id)){console.log(d.default.dim(` [${y()}] Skipping duplicate start_claude_code (already processed)`));return}try{let t=await M({id:e.command_id||`cc-start-${Date.now()}`,command_type:"start_claude_code",payload:{bot_id:e.bot_id,bot_user_id:e.bot_user_id,room_id:e.room_id,session_id:e.session_id}});e.command_id&&await v().post(`/api/v1/dashboard/commands/${e.command_id}/result`,{status:t.success?"success":"failed",result:t.message}).catch(()=>{})}catch{}try{await x()}catch{}}else if(e.type==="claude_code.stop"&&e.bot_id){if(console.log(d.default.cyan(` [${y()}] Claude Code STOP: bot=${e.bot_id}`)),Ne("stop_claude_code",e.bot_id)){console.log(d.default.dim(` [${y()}] Skipping duplicate stop_claude_code (already processed)`));return}try{let t=await M({id:e.command_id||`cc-stop-${Date.now()}`,command_type:"stop_claude_code",payload:{bot_id:e.bot_id,bot_user_id:e.bot_user_id,room_id:e.room_id,session_id:e.session_id}});e.command_id&&await v().post(`/api/v1/dashboard/commands/${e.command_id}/result`,{status:t.success?"success":"failed",result:t.message}).catch(()=>{})}catch{}try{await x()}catch{}}else if(e.type==="command.execute"&&e.command_id){if(console.log(d.default.cyan(` [${y()}] Command: ${e.command_type} (${e.command_id})`)),Ne(e.command_type,e.payload?.bot_id)){console.log(d.default.dim(` [${y()}] Skipping duplicate ${e.command_type} (already processed)`));return}try{let t=v();await t.post(`/api/v1/dashboard/commands/${e.command_id}/ack`);let o=await M({id:e.command_id,command_type:e.command_type,payload:e.payload});await t.post(`/api/v1/dashboard/commands/${e.command_id}/result`,{status:o.success?"success":"failed",result:o.message,new_version:o.newVersion||null}),e.command_type==="update_cli"&&o.success&&(console.log(d.default.cyan(` [${y()}] CLI updated \u2014 restarting in 2s...`)),setTimeout(()=>process.exit(0),2e3))}catch{}try{await x()}catch{}}}var no=new so.Command("watch").description("Watch for bot pair events in real-time (no polling)").action(async()=>{let e=h();e||(console.log(g.default.yellow("Not logged in.")),process.exit(1)),await oe(!1),console.log(g.default.green("\u{1F534} Listening for pair events... (Ctrl+C to stop)"));let t=require("eventsource"),o=new t(`${B}/api/v1/openclaw/events`,{headers:{Authorization:`Bearer ${e.access_token}`}});o.onmessage=async n=>{try{let s=JSON.parse(n.data);if(s.type==="pair"){console.log(g.default.cyan(`
|
|
117
117
|
\u{1F4F1} Pair event received: ${s.bot_name}`)),await te(s.pair_code,s.bot_name,s.bot_user_id,!1);try{let{execSync:r}=await import("child_process");r("openclaw gateway restart",{stdio:"ignore"}),console.log(g.default.green(" \u2705 Gateway restarted \u2014 bot is live!"))}catch{console.log(g.default.dim(" Gateway restart failed \u2014 run: openclaw gateway restart"))}}else if(s.type==="gateway-restart"){console.log(g.default.cyan(`
|
|
118
118
|
\u{1F504} Remote gateway-restart command received`));let r=await K();r.success?console.log(g.default.green(` \u2705 Gateway restarted: ${r.message}`)):console.log(g.default.red(` \u274C Gateway restart failed: ${r.message}`));try{await x(),console.log(g.default.dim(" Heartbeat pushed with updated status."))}catch{console.log(g.default.dim(" Could not push heartbeat."))}}else if(s.type==="bot.delete"&&s.bot_user_id){let r=s.bot_user_id,a=r.split(":")[0].replace("@","").replace(/_bot$/,"");console.log(g.default.yellow(`
|
|
@@ -128,4 +128,4 @@ Keep responses concise and helpful.
|
|
|
128
128
|
BadgerClaw Dashboard
|
|
129
129
|
`)),console.log($.default.bold(" Instance")),console.log(` ID: ${de()}`),console.log(` CLI Version: ${t}`),console.log(` Plugin: ${n.pluginVersion}`),console.log(),console.log($.default.bold(" Machine")),console.log(` Hostname: ${o.hostname}`),console.log(` OS: ${o.os} / ${o.arch}`),console.log(` Uptime: ${lo(o.uptimeSeconds)}`),console.log(` Free Memory: ${o.memFreeMb} MB`),console.log();let s=n.status==="running"?$.default.green:n.status==="error"?$.default.red:$.default.yellow;if(console.log($.default.bold(" Gateway")),console.log(` Status: ${s(n.status)}`),console.log(` PID: ${n.pid??"N/A"}`),console.log(` Last Restart: ${n.lastRestart??"N/A"}`),console.log(),n.bots.length===0)console.log($.default.bold(" Bots")),console.log($.default.dim(" No bots detected. Pair bots at https://badgerclaw.ai")),console.log();else{console.log($.default.bold(` Bots (${n.bots.length})`)),console.log();for(let r of n.bots){let a=r.status==="running"?$.default.green:r.status==="error"?$.default.red:$.default.yellow;if(console.log(` ${$.default.bold(r.botUsername)} ${a(`[${r.status}]`)}`),console.log(` ID: ${r.botId}`),console.log(` Uptime: ${lo(r.uptimeSeconds)}`),console.log(` Messages: ${r.messagesReceived} in / ${r.messagesSent} out`),r.chunkedMessages>0&&console.log(` Chunked: ${r.chunkedMessages} messages, ${r.totalChunksSent} chunks`),console.log(` Rooms: ${r.roomsActive} active`),console.log(` Errors: ${r.errors}`),r.lastError&&(console.log(` Last Error: ${$.default.red(r.lastError)}`),console.log(` Error At: ${r.lastErrorAt}`)),console.log(` Last Active: ${r.lastActivity??"N/A"}`),r.roomDetails&&r.roomDetails.length>0){console.log($.default.dim(" Rooms:"));for(let i of r.roomDetails)console.log($.default.dim(` ${i.roomName} \u2014 ${i.messagesInRoom} msgs, last: ${i.lastActivityInRoom??"N/A"}`))}console.log()}}console.log($.default.dim(` Last checked: ${new Date().toISOString()}`)),console.log()});function lo(e){if(e<60)return`${e}s`;if(e<3600)return`${Math.floor(e/60)}m ${e%60}s`;let t=Math.floor(e/3600),o=Math.floor(e%3600/60);return t<24?`${t}h ${o}m`:`${Math.floor(t/24)}d ${t%24}h ${o}m`}var ct=require("commander"),Q=c(require("chalk")),go=c(require("ora"));var Cs=new ct.Command("restart").description("Restart the OpenClaw gateway via the plugin probe endpoint").action(async()=>{h()||(console.log(Q.default.yellow("Not logged in. Run `badgerclaw login` first.")),process.exit(1));let t=(0,go.default)("Restarting gateway...").start(),o=await K();if(o.success){t.succeed(Q.default.green(`Gateway restarted: ${o.message}`));try{await x(),console.log(Q.default.dim(" Heartbeat pushed with updated status."))}catch{console.log(Q.default.dim(" Could not push heartbeat \u2014 daemon may not be running."))}}else t.fail(Q.default.red(`Gateway restart failed: ${o.message}`)),process.exit(1)}),po=new ct.Command("gateway").description("Manage the OpenClaw gateway").addCommand(Cs);var W=require("commander"),p=c(require("chalk")),J=c(require("ora")),G=c(require("axios"));Re();var q="http://localhost:7331",vs=new W.Command("start").description("Start a Claude Code session for a bot+room").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").option("--session-id <id>","Session ID (default: auto-generated)").option("--port <port>","MCP server port (default: 7332)","7332").option("--project <dir>","Project directory",process.cwd()).action(async e=>{h()||(console.log(p.default.yellow("Not logged in. Run `badgerclaw login` first.")),process.exit(1));let o=e.sessionId||`session-${Date.now()}`,n=parseInt(e.port,10),s=e.project.replace(/^~/,process.env.HOME||""),r=(0,J.default)("Starting Claude Code session...").start();try{await G.default.post(`${q}/claude-code/toggle`,{botId:e.bot,roomId:e.room,enabled:!0,sessionId:o},{timeout:1e4}),r.text="Gateway toggled \u2014 launching MCP server...";let i=et({sessionId:o,port:n,projectDir:s}).pid||0;await new Promise(l=>setTimeout(l,1500)),r.text="MCP server running \u2014 launching Claude Code...",tt({sessionId:o,port:n,projectDir:s}),ot(o,0,i,n,s),r.succeed(p.default.green("Claude Code session started")+p.default.dim(` (session: ${o}, port: ${n})`)),console.log(p.default.dim(` Bot: ${e.bot}`)),console.log(p.default.dim(` Room: ${e.room}`)),console.log(p.default.dim(` Project: ${s}`)),console.log(p.default.dim(" Claude Code should open in a new terminal window."))}catch(a){let i=a.response?.data?.message||a.message||"Unknown error";r.fail(p.default.red(`Failed to start Claude Code: ${i}`)),process.exit(1)}}),Is=new W.Command("stop").description("Stop a Claude Code session").option("--bot <botId>","Bot ID or username").option("--room <roomId>","Matrix room ID").option("--session-id <id>","Session ID to stop").action(async e=>{h()||(console.log(p.default.yellow("Not logged in. Run `badgerclaw login` first.")),process.exit(1));let o=(0,J.default)("Stopping Claude Code session...").start();try{e.bot&&e.room&&await G.default.post(`${q}/claude-code/toggle`,{botId:e.bot,roomId:e.room,enabled:!1},{timeout:1e4});let n=e.sessionId;if(!n)try{let a=((await G.default.get(`${q}/claude-code/status`,{timeout:5e3})).data?.rooms||[]).find(i=>i.botId===e.bot&&i.roomId===e.room);a&&(n=a.sessionId)}catch{}n?(st(n),o.succeed(p.default.green(`Claude Code session stopped (${n})`))):(o.succeed(p.default.green("Claude Code toggled off at gateway")),console.log(p.default.dim(" No local session found to kill \u2014 may need to close Claude Code manually.")))}catch(n){let s=n.response?.data?.message||n.message||"Unknown error";o.fail(p.default.red(`Failed to stop Claude Code: ${s}`)),process.exit(1)}}),ks=new W.Command("status").description("Show active Claude Code sessions").action(async()=>{h()||(console.log(p.default.yellow("Not logged in. Run `badgerclaw login` first.")),process.exit(1));let t=(0,J.default)("Fetching Claude Code status...").start();try{let o=[];try{o=(await G.default.get(`${q}/claude-code/status`,{timeout:5e3})).data?.rooms||[]}catch{}let n=nt();if(t.stop(),o.length===0&&n.length===0){console.log(p.default.dim("No active Claude Code sessions."));return}if(o.length>0){console.log(p.default.bold(`
|
|
130
130
|
Gateway Claude Code Rooms:`)),console.log(p.default.dim(" Bot".padEnd(30)+"Room".padEnd(30)+"Session".padEnd(20)+"Enabled")),console.log(p.default.dim(" "+"-".repeat(88)));for(let s of o){let r=s.enabled?p.default.green("ON"):p.default.dim("OFF");console.log(` ${(s.botId||"").padEnd(30)}${(s.roomName||s.roomId||"").padEnd(30)}${(s.sessionId||"-").padEnd(20)}${r}`)}}if(n.length>0){console.log(p.default.bold(`
|
|
131
|
-
Local Sessions:`)),console.log(p.default.dim(" Session".padEnd(25)+"Port".padEnd(8)+"MCP PID".padEnd(10)+"Project".padEnd(40)+"Started")),console.log(p.default.dim(" "+"-".repeat(90)));for(let s of n)console.log(` ${s.sessionId.padEnd(25)}${String(s.port).padEnd(8)}${String(s.mcpPid).padEnd(10)}${s.projectDir.padEnd(40)}${s.startedAt}`)}console.log("")}catch(o){t.fail(p.default.red(`Failed to get status: ${o.message}`)),process.exit(1)}}),xs=new W.Command("group").description("Group bots/rooms into a shared Claude Code session").requiredOption("--session <id>","Session ID to group under").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").action(async e=>{let t=(0,J.default)("Grouping bot+room into session...").start();try{await G.default.post(`${q}/claude-code/toggle`,{botId:e.bot,roomId:e.room,enabled:!0,sessionId:e.session},{timeout:1e4}),t.succeed(p.default.green(`Grouped ${e.bot} + ${e.room} into session "${e.session}"`))}catch(o){let n=o.response?.data?.message||o.message||"Unknown error";t.fail(p.default.red(`Failed to group: ${n}`)),process.exit(1)}}),As=new W.Command("allow").description("Add a user to the Claude Code allowlist for a bot+room").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").requiredOption("--user <userId>","Matrix user ID to allow (e.g. @alice:server)").action(async e=>{let t=(0,J.default)("Adding user to allowlist...").start();try{await G.default.post(`${q}/claude-code/allow`,{botId:e.bot,roomId:e.room,userId:e.user},{timeout:1e4}),t.succeed(p.default.green(`Added ${e.user} to allowlist for ${e.bot} in ${e.room}`))}catch(o){let n=o.response?.data?.message||o.message||"Unknown error";t.fail(p.default.red(`Failed to add user: ${n}`)),process.exit(1)}}),Ps=new W.Command("revoke").description("Remove a user from the Claude Code allowlist for a bot+room").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").requiredOption("--user <userId>","Matrix user ID to revoke").action(async e=>{let t=(0,J.default)("Removing user from allowlist...").start();try{await G.default.post(`${q}/claude-code/revoke`,{botId:e.bot,roomId:e.room,userId:e.user},{timeout:1e4}),t.succeed(p.default.green(`Removed ${e.user} from allowlist for ${e.bot} in ${e.room}`))}catch(o){let n=o.response?.data?.message||o.message||"Unknown error";t.fail(p.default.red(`Failed to revoke user: ${n}`)),process.exit(1)}}),fo=new W.Command("claude-code").description("Manage Claude Code sessions (relay Matrix messages to local Claude Code)").addCommand(vs).addCommand(Is).addCommand(ks).addCommand(xs).addCommand(As).addCommand(Ps);var R=new ho.Command;R.name("badgerclaw").description("BadgerClaw CLI \u2014 one-click bot provisioning").version("0.2.
|
|
131
|
+
Local Sessions:`)),console.log(p.default.dim(" Session".padEnd(25)+"Port".padEnd(8)+"MCP PID".padEnd(10)+"Project".padEnd(40)+"Started")),console.log(p.default.dim(" "+"-".repeat(90)));for(let s of n)console.log(` ${s.sessionId.padEnd(25)}${String(s.port).padEnd(8)}${String(s.mcpPid).padEnd(10)}${s.projectDir.padEnd(40)}${s.startedAt}`)}console.log("")}catch(o){t.fail(p.default.red(`Failed to get status: ${o.message}`)),process.exit(1)}}),xs=new W.Command("group").description("Group bots/rooms into a shared Claude Code session").requiredOption("--session <id>","Session ID to group under").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").action(async e=>{let t=(0,J.default)("Grouping bot+room into session...").start();try{await G.default.post(`${q}/claude-code/toggle`,{botId:e.bot,roomId:e.room,enabled:!0,sessionId:e.session},{timeout:1e4}),t.succeed(p.default.green(`Grouped ${e.bot} + ${e.room} into session "${e.session}"`))}catch(o){let n=o.response?.data?.message||o.message||"Unknown error";t.fail(p.default.red(`Failed to group: ${n}`)),process.exit(1)}}),As=new W.Command("allow").description("Add a user to the Claude Code allowlist for a bot+room").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").requiredOption("--user <userId>","Matrix user ID to allow (e.g. @alice:server)").action(async e=>{let t=(0,J.default)("Adding user to allowlist...").start();try{await G.default.post(`${q}/claude-code/allow`,{botId:e.bot,roomId:e.room,userId:e.user},{timeout:1e4}),t.succeed(p.default.green(`Added ${e.user} to allowlist for ${e.bot} in ${e.room}`))}catch(o){let n=o.response?.data?.message||o.message||"Unknown error";t.fail(p.default.red(`Failed to add user: ${n}`)),process.exit(1)}}),Ps=new W.Command("revoke").description("Remove a user from the Claude Code allowlist for a bot+room").requiredOption("--bot <botId>","Bot ID or username").requiredOption("--room <roomId>","Matrix room ID").requiredOption("--user <userId>","Matrix user ID to revoke").action(async e=>{let t=(0,J.default)("Removing user from allowlist...").start();try{await G.default.post(`${q}/claude-code/revoke`,{botId:e.bot,roomId:e.room,userId:e.user},{timeout:1e4}),t.succeed(p.default.green(`Removed ${e.user} from allowlist for ${e.bot} in ${e.room}`))}catch(o){let n=o.response?.data?.message||o.message||"Unknown error";t.fail(p.default.red(`Failed to revoke user: ${n}`)),process.exit(1)}}),fo=new W.Command("claude-code").description("Manage Claude Code sessions (relay Matrix messages to local Claude Code)").addCommand(vs).addCommand(Is).addCommand(ks).addCommand(xs).addCommand(As).addCommand(Ps);var R=new ho.Command;R.name("badgerclaw").description("BadgerClaw CLI \u2014 one-click bot provisioning").version("0.2.17");R.addCommand(Ut);R.addCommand(Ft);R.addCommand(Wt);R.addCommand(Vt);R.addCommand(vt);R.addCommand(no);R.addCommand(co);R.addCommand(to);R.addCommand(uo);R.addCommand(po);R.addCommand(fo);R.parse(process.argv);
|