UNPKG

grix-connector

Version:

Connect local AI coding agents (Claude, Codex, Gemini, Qwen, DeepSeek, Cursor, OpenCode, Pi, OpenHuman, Reasonix) to the Grix scheduling platform. Also serves as an OpenClaw plugin for Grix channel transport.

47 lines (44 loc) 10.2 kB
import{mkdir as P,readdir as b,readFile as k,rm as g,stat as E,writeFile as S}from"node:fs/promises";import p from"node:os";import d from"node:path";import{runCommand as u,spawnDetached as W,killProcessesByCommandLine as L,isWindowsElevated as D}from"./process-control.js";import{getServicePrefix as x,parseConfigDirFromPlistXML as R,parseConfigDirFromSystemdUnit as j,resolveLinuxUserUnitPath as M,resolveMacOSLaunchAgentPath as O}from"./service-paths.js";function m(t){return String(t??"").replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;")}function B(t){return`'${String(t??"").replace(/'/g,"'\\''")}'`}function T(t){return[t.nodePath,t.cliPath,...t.configDir?["--config-dir",t.configDir]:[]]}function U(t){const r=T(t);return`<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> <plist version="1.0"> <dict> <key>Label</key> <string>${m(t.serviceID)}</string> <key>ProgramArguments</key> <array> ${r.map(e=>` <string>${m(e)}</string>`).join(` `)} </array> <key>RunAtLoad</key> <true/> <key>KeepAlive</key> <true/> <key>WorkingDirectory</key> <string>${m(d.dirname(t.cliPath))}</string> ${t.environmentPath?` <key>EnvironmentVariables</key> <dict> <key>PATH</key> <string>${m(t.environmentPath)}</string> </dict> `:""} <key>StandardOutPath</key> <string>${m(t.stdoutPath)}</string> <key>StandardErrorPath</key> <string>${m(t.stderrPath)}</string> </dict> </plist> `}function V(t){const r=T(t).map(e=>B(e)).join(" ");return`[Unit] Description=grix-connector daemon (${t.serviceID}) After=network.target [Service] Type=simple ExecStart=${r} WorkingDirectory=${d.dirname(t.cliPath)} Restart=always RestartSec=2 StandardOutput=append:${t.stdoutPath} StandardError=append:${t.stderrPath} [Install] WantedBy=default.target `}function w(t,r){const e=t.replace(/[^a-zA-Z0-9._-]/g,"_");return d.join(r,".grix",`${e}-wrapper.vbs`)}function A(t,r){const e=t.replace(/[^a-zA-Z0-9._-]/g,"_"),a=d.join(r,"AppData","Roaming","Microsoft","Windows","Start Menu","Programs","Startup");return d.join(a,`${e}.lnk`)}function f(t){return String(t??"").replace(/"/g,'""')}function Q(t,r){const e=r.replace(/\.lnk$/,"-create.vbs"),a=["Option Explicit","Dim shell, shortcut",'Set shell = CreateObject("WScript.Shell")',`Set shortcut = shell.CreateShortcut("${f(r)}")`,'shortcut.TargetPath = "wscript.exe"',`shortcut.Arguments = "//B //NoLogo \\"${f(t)}\\""`,"shortcut.WindowStyle = 7","shortcut.Save",""].join(`\r `);return{path:e,content:a}}function _(t){const r=[`"${f(t.nodePath)}"`,`"${f(t.cliPath)}"`,...t.configDir?["--config-dir",`"${f(t.configDir)}"`]:[]].join(" ");return["Option Explicit","Dim shell, command, delayMs, rapidCount",`command = "${f(r)}"`,'Set shell = CreateObject("WScript.Shell")',"delayMs = 5000","rapidCount = 0","Do"," shell.Run command, 0, True"," rapidCount = rapidCount + 1"," If rapidCount >= 10 Then"," WScript.Sleep 300000"," rapidCount = 0"," Else"," WScript.Sleep delayMs"," End If","Loop",""].join(`\r `)}function y(t){return`gui/${t??0}`}function v(t){const r=String(t?.stdout??"").trim();return String(t?.stderr??"").trim()||r||`exit=${Number(t?.exitCode??-1)}`}function F(t,r){if(Number(t?.exitCode??0)!==0)throw new Error(`${r}: ${v(t)}`)}function z(){return{platform:"darwin",kind:"launchd",async install({serviceID:t,nodePath:r,cliPath:e,configDir:a,stdoutPath:n,stderrPath:i,environmentPath:c="",homeDir:o=p.homedir()}){const s=O(t,o);return await P(d.dirname(s),{recursive:!0}),await S(s,U({serviceID:t,nodePath:r,cliPath:e,configDir:a,stdoutPath:n,stderrPath:i,environmentPath:c}),{encoding:"utf8",mode:384}),{definitionPath:s}},async start({serviceID:t,definitionPath:r,runCommand:e=u,uid:a=process.getuid?.()??0}){const n=y(a);let i=await e("launchctl",["bootstrap",n,r],{allowFailure:!0});Number(i?.exitCode??0)!==0&&(await e("launchctl",["bootout",`${n}/${t}`],{allowFailure:!0}),i=await e("launchctl",["bootstrap",n,r],{allowFailure:!0}),F(i,`launchctl bootstrap ${n}`));const c=await e("launchctl",["kickstart","-k",`${n}/${t}`],{allowFailure:!0});if(Number(c?.exitCode??0)!==0){const o=Number(i?.exitCode??0)===0?"":`, bootstrap=${v(i)}`;throw new Error(`launchctl start failed for ${n}/${t}: ${v(c)}${o}`)}},async stop({serviceID:t,runCommand:r=u,uid:e=process.getuid?.()??0}){const a=y(e);await r("launchctl",["bootout",`${a}/${t}`],{allowFailure:!0})},async restart({serviceID:t,definitionPath:r,runCommand:e=u,uid:a=process.getuid?.()??0}){const n=y(a);await e("launchctl",["bootout",`${n}/${t}`],{allowFailure:!0});for(let o=0;o<20&&(await e("launchctl",["print",`${n}/${t}`],{allowFailure:!0})).exitCode===0;o+=1)await new Promise(l=>setTimeout(l,250));try{await E(r)}catch{throw new Error(`launchd plist missing: ${r}`)}const i=await e("launchctl",["bootstrap",n,r],{allowFailure:!0});F(i,`launchctl bootstrap ${n}`);const c=await e("launchctl",["kickstart","-k",`${n}/${t}`],{allowFailure:!0});F(c,`launchctl kickstart ${n}/${t}`)},async uninstall({serviceID:t,definitionPath:r,runCommand:e=u,uid:a=process.getuid?.()??0}){const n=y(a);await e("launchctl",["bootout",`${n}/${t}`],{allowFailure:!0}),await g(r,{force:!0})},async discoverServices({homeDir:t=p.homedir()}={}){const r=d.join(t,"Library","LaunchAgents"),e=await b(r).catch(()=>[]),a=x("darwin"),n=[];for(const i of e){if(!i.startsWith(a)||!i.endsWith(".plist"))continue;const c=i.slice(0,-6),o=d.join(r,i);let s=null;try{const l=await k(o,"utf8");s=R(l)}catch{}n.push({serviceID:c,definitionPath:o,configDir:s})}return n},async isServiceLoaded({serviceID:t,runCommand:r=u,uid:e=process.getuid?.()??0}){const a=y(e),n=await r("launchctl",["print",`${a}/${t}`],{allowFailure:!0});return Number(n?.exitCode??1)===0}}}function H(){return{platform:"win32",kind:"task-scheduler",async install({serviceID:t,nodePath:r,cliPath:e,configDir:a,runCommand:n=u,homeDir:i=p.homedir()}){const c=w(t,i);await P(d.dirname(c),{recursive:!0}),await S(c,_({nodePath:r,cliPath:e,configDir:a}),"utf8");let o=!1;if(D())try{await n("schtasks",["/Create","/TN",t,"/SC","ONCE","/ST","00:00","/SD","2099/12/31","/RL","LIMITED","/F","/TR",`wscript.exe //B //NoLogo "${c}"`]),o=!0}catch{}const s=A(t,i),l=Q(c,s);return await S(l.path,l.content,"utf8"),await n("wscript.exe",[l.path,"//B","//NoLogo"]).catch(()=>{}),await g(l.path,{force:!0}),{definitionPath:o?`task:${t}`:`startup:${t}`}},async start({serviceID:t,definitionPath:r,runCommand:e=u,homeDir:a=p.homedir()}){const n=w(t,a);if(r.startsWith("task:"))try{await e("schtasks",["/Run","/TN",t]);return}catch{}W("wscript.exe",["//B","//NoLogo",n])},async stop({serviceID:t,runCommand:r=u,homeDir:e=p.homedir()}){await r("schtasks",["/End","/TN",t],{allowFailure:!0});const a=`${t}-wrapper.vbs`;await L(a,{platform:"win32"})},async restart({serviceID:t,definitionPath:r,runCommand:e=u,homeDir:a=p.homedir()}){await e("schtasks",["/End","/TN",t],{allowFailure:!0});const n=`${t}-wrapper.vbs`;await L(n,{platform:"win32"});const i=w(t,a);if(r?.startsWith("task:"))try{await e("schtasks",["/Run","/TN",t]);return}catch{}W("wscript.exe",["//B","//NoLogo",i])},async uninstall({serviceID:t,runCommand:r=u,homeDir:e=p.homedir()}){await r("schtasks",["/Delete","/TN",t,"/F"],{allowFailure:!0});const a=w(t,e);await g(a,{force:!0});const n=A(t,e);await g(n,{force:!0})},async discoverServices({homeDir:t=p.homedir(),runCommand:r=u}={}){const e=x("win32"),a=await r("schtasks",["/Query","/FO","CSV","/NH"],{allowFailure:!0}),n=[];if(Number(a?.exitCode??-1)===0){const o=String(a.stdout??"").split(/\r?\n/);for(const s of o){const l=s.match(/^"([^"]+)"/);if(!l)continue;const h=l[1];if(!h.startsWith(e))continue;let C=null;try{const $=w(h,t),N=(await k($,"utf8")).match(/--config-dir\s+""([^""]+)""/);N&&(C=N[1])}catch{}n.push({serviceID:h,definitionPath:`task:${h}`,configDir:C})}}const i=d.join(t,"AppData","Roaming","Microsoft","Windows","Start Menu","Programs","Startup"),c=await b(i).catch(()=>[]);for(const o of c){if(!o.startsWith(e)||!o.endsWith(".lnk"))continue;const s=o.slice(0,-4);if(n.some(h=>h.serviceID===s))continue;let l=null;try{const h=w(s,t),$=(await k(h,"utf8")).match(/--config-dir\s+""([^""]+)""/);$&&(l=$[1])}catch{}n.push({serviceID:s,definitionPath:`startup:${s}`,configDir:l})}return n},async isServiceLoaded({serviceID:t,runCommand:r=u}){const e=await r("schtasks",["/Query","/TN",t,"/NH"],{allowFailure:!0});return Number(e?.exitCode??1)===0}}}function Z(){return{platform:"linux",kind:"systemd-user",async install({serviceID:t,nodePath:r,cliPath:e,configDir:a,stdoutPath:n,stderrPath:i,homeDir:c=p.homedir(),runCommand:o=u}){const s=M(t,c);return await P(d.dirname(s),{recursive:!0}),await S(s,V({serviceID:t,nodePath:r,cliPath:e,configDir:a,stdoutPath:n,stderrPath:i}),{encoding:"utf8",mode:384}),await o("systemctl",["--user","daemon-reload"]),await o("systemctl",["--user","enable",`${t}.service`]),{definitionPath:s}},async start({serviceID:t,runCommand:r=u}){await r("systemctl",["--user","start",`${t}.service`])},async stop({serviceID:t,runCommand:r=u}){await r("systemctl",["--user","stop",`${t}.service`],{allowFailure:!0})},async restart({serviceID:t,runCommand:r=u}){await r("systemctl",["--user","restart",`${t}.service`])},async uninstall({serviceID:t,definitionPath:r,runCommand:e=u}){await e("systemctl",["--user","stop",`${t}.service`],{allowFailure:!0}),await e("systemctl",["--user","disable",`${t}.service`],{allowFailure:!0}),await g(r,{force:!0}),await e("systemctl",["--user","daemon-reload"])},async discoverServices({homeDir:t=p.homedir()}={}){const r=d.join(t,".config","systemd","user"),e=await b(r).catch(()=>[]),a=x("linux"),n=[];for(const i of e){if(!i.startsWith(a)||!i.endsWith(".service"))continue;const c=i.slice(0,-8),o=d.join(r,i);let s=null;try{const l=await k(o,"utf8");s=j(l)}catch{}n.push({serviceID:c,definitionPath:o,configDir:s})}return n},async isServiceLoaded({serviceID:t,runCommand:r=u}){const e=await r("systemctl",["--user","is-active",`${t}.service`],{allowFailure:!0});return String(e?.stdout??"").trim()==="active"}}}function I(t=process.platform){return t==="darwin"?z():t==="win32"?H():Z()}export{I as getPlatformServiceAdapter};