@vibecompany/247-cli 0.1.0

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.
Files changed (74) hide show
  1. package/README.md +110 -0
  2. package/dist/commands/doctor.d.ts +3 -0
  3. package/dist/commands/doctor.d.ts.map +1 -0
  4. package/dist/commands/doctor.js +277 -0
  5. package/dist/commands/doctor.js.map +1 -0
  6. package/dist/commands/hooks.d.ts +3 -0
  7. package/dist/commands/hooks.d.ts.map +1 -0
  8. package/dist/commands/hooks.js +127 -0
  9. package/dist/commands/hooks.js.map +1 -0
  10. package/dist/commands/init.d.ts +3 -0
  11. package/dist/commands/init.d.ts.map +1 -0
  12. package/dist/commands/init.js +116 -0
  13. package/dist/commands/init.js.map +1 -0
  14. package/dist/commands/logs.d.ts +3 -0
  15. package/dist/commands/logs.d.ts.map +1 -0
  16. package/dist/commands/logs.js +38 -0
  17. package/dist/commands/logs.js.map +1 -0
  18. package/dist/commands/service.d.ts +3 -0
  19. package/dist/commands/service.d.ts.map +1 -0
  20. package/dist/commands/service.js +235 -0
  21. package/dist/commands/service.js.map +1 -0
  22. package/dist/commands/start.d.ts +3 -0
  23. package/dist/commands/start.d.ts.map +1 -0
  24. package/dist/commands/start.js +109 -0
  25. package/dist/commands/start.js.map +1 -0
  26. package/dist/commands/status.d.ts +3 -0
  27. package/dist/commands/status.d.ts.map +1 -0
  28. package/dist/commands/status.js +62 -0
  29. package/dist/commands/status.js.map +1 -0
  30. package/dist/commands/stop.d.ts +3 -0
  31. package/dist/commands/stop.d.ts.map +1 -0
  32. package/dist/commands/stop.js +23 -0
  33. package/dist/commands/stop.js.map +1 -0
  34. package/dist/commands/update.d.ts +3 -0
  35. package/dist/commands/update.d.ts.map +1 -0
  36. package/dist/commands/update.js +121 -0
  37. package/dist/commands/update.js.map +1 -0
  38. package/dist/hooks/installer.d.ts +36 -0
  39. package/dist/hooks/installer.d.ts.map +1 -0
  40. package/dist/hooks/installer.js +175 -0
  41. package/dist/hooks/installer.js.map +1 -0
  42. package/dist/index.d.ts +3 -0
  43. package/dist/index.d.ts.map +1 -0
  44. package/dist/index.js +38 -0
  45. package/dist/index.js.map +1 -0
  46. package/dist/lib/config.d.ts +46 -0
  47. package/dist/lib/config.d.ts.map +1 -0
  48. package/dist/lib/config.js +88 -0
  49. package/dist/lib/config.js.map +1 -0
  50. package/dist/lib/paths.d.ts +34 -0
  51. package/dist/lib/paths.d.ts.map +1 -0
  52. package/dist/lib/paths.js +76 -0
  53. package/dist/lib/paths.js.map +1 -0
  54. package/dist/lib/prerequisites.d.ts +36 -0
  55. package/dist/lib/prerequisites.d.ts.map +1 -0
  56. package/dist/lib/prerequisites.js +181 -0
  57. package/dist/lib/prerequisites.js.map +1 -0
  58. package/dist/lib/process.d.ts +39 -0
  59. package/dist/lib/process.d.ts.map +1 -0
  60. package/dist/lib/process.js +188 -0
  61. package/dist/lib/process.js.map +1 -0
  62. package/dist/service/index.d.ts +44 -0
  63. package/dist/service/index.d.ts.map +1 -0
  64. package/dist/service/index.js +18 -0
  65. package/dist/service/index.js.map +1 -0
  66. package/dist/service/launchd.d.ts +18 -0
  67. package/dist/service/launchd.d.ts.map +1 -0
  68. package/dist/service/launchd.js +208 -0
  69. package/dist/service/launchd.js.map +1 -0
  70. package/dist/service/systemd.d.ts +18 -0
  71. package/dist/service/systemd.d.ts.map +1 -0
  72. package/dist/service/systemd.js +196 -0
  73. package/dist/service/systemd.js.map +1 -0
  74. package/package.json +57 -0
@@ -0,0 +1,208 @@
1
+ import { existsSync, writeFileSync, unlinkSync, mkdirSync } from 'fs';
2
+ import { join } from 'path';
3
+ import { homedir } from 'os';
4
+ import { exec } from 'child_process';
5
+ import { promisify } from 'util';
6
+ import { getAgentPaths } from '../lib/paths.js';
7
+ import { checkTmux } from '../lib/prerequisites.js';
8
+ const execAsync = promisify(exec);
9
+ const SERVICE_LABEL = 'co.thevibecompany.247';
10
+ export class LaunchdService {
11
+ platform = 'macos';
12
+ serviceName = SERVICE_LABEL;
13
+ get plistPath() {
14
+ return join(homedir(), 'Library', 'LaunchAgents', `${SERVICE_LABEL}.plist`);
15
+ }
16
+ async status() {
17
+ const installed = existsSync(this.plistPath);
18
+ let running = false;
19
+ let pid;
20
+ if (installed) {
21
+ try {
22
+ const { stdout } = await execAsync(`launchctl list | grep ${SERVICE_LABEL}`);
23
+ const parts = stdout.trim().split(/\s+/);
24
+ if (parts[0] && parts[0] !== '-') {
25
+ pid = parseInt(parts[0], 10);
26
+ running = !isNaN(pid);
27
+ }
28
+ }
29
+ catch {
30
+ // Service not running or not found in list
31
+ }
32
+ }
33
+ return {
34
+ installed,
35
+ running,
36
+ enabled: installed,
37
+ pid,
38
+ configPath: installed ? this.plistPath : undefined,
39
+ };
40
+ }
41
+ async install(options = {}) {
42
+ const paths = getAgentPaths();
43
+ // Verify tmux is installed
44
+ const tmuxCheck = checkTmux();
45
+ if (tmuxCheck.status === 'error') {
46
+ return {
47
+ success: false,
48
+ error: 'tmux is not installed. Please install it first: brew install tmux',
49
+ };
50
+ }
51
+ // Create LaunchAgents directory if needed
52
+ const launchAgentsDir = join(homedir(), 'Library', 'LaunchAgents');
53
+ if (!existsSync(launchAgentsDir)) {
54
+ mkdirSync(launchAgentsDir, { recursive: true });
55
+ }
56
+ // Create log directory
57
+ const logDir = join(homedir(), 'Library', 'Logs', '247-agent');
58
+ if (!existsSync(logDir)) {
59
+ mkdirSync(logDir, { recursive: true });
60
+ }
61
+ const logs = this.getLogPaths();
62
+ // Determine entry point
63
+ const entryPoint = paths.isDev
64
+ ? join(paths.agentRoot, 'src', 'index.ts')
65
+ : join(paths.agentRoot, 'dist', 'index.js');
66
+ // Generate plist content
67
+ const plistContent = this.generatePlist({
68
+ label: SERVICE_LABEL,
69
+ nodePath: paths.nodePath,
70
+ agentScript: entryPoint,
71
+ workingDirectory: paths.agentRoot,
72
+ stdoutPath: logs.stdout,
73
+ stderrPath: logs.stderr,
74
+ runAtLoad: options.enableAtBoot ?? true,
75
+ keepAlive: true,
76
+ isDev: paths.isDev,
77
+ configPath: paths.configPath,
78
+ dataDir: paths.dataDir,
79
+ });
80
+ writeFileSync(this.plistPath, plistContent, 'utf-8');
81
+ // Load the service if requested
82
+ if (options.startNow) {
83
+ const startResult = await this.start();
84
+ if (!startResult.success) {
85
+ return { ...startResult, configPath: this.plistPath };
86
+ }
87
+ }
88
+ return { success: true, configPath: this.plistPath };
89
+ }
90
+ async uninstall() {
91
+ const status = await this.status();
92
+ // Stop service first if running
93
+ if (status.running) {
94
+ await this.stop();
95
+ }
96
+ // Remove plist file
97
+ if (existsSync(this.plistPath)) {
98
+ try {
99
+ unlinkSync(this.plistPath);
100
+ }
101
+ catch (err) {
102
+ return { success: false, error: `Failed to remove plist: ${err.message}` };
103
+ }
104
+ }
105
+ return { success: true };
106
+ }
107
+ async start() {
108
+ try {
109
+ await execAsync(`launchctl load "${this.plistPath}"`);
110
+ return { success: true };
111
+ }
112
+ catch (err) {
113
+ return { success: false, error: err.message };
114
+ }
115
+ }
116
+ async stop() {
117
+ try {
118
+ await execAsync(`launchctl unload "${this.plistPath}"`);
119
+ return { success: true };
120
+ }
121
+ catch (err) {
122
+ return { success: false, error: err.message };
123
+ }
124
+ }
125
+ async restart() {
126
+ await this.stop();
127
+ await new Promise(resolve => setTimeout(resolve, 500));
128
+ return this.start();
129
+ }
130
+ getLogPaths() {
131
+ const logDir = join(homedir(), 'Library', 'Logs', '247-agent');
132
+ return {
133
+ stdout: join(logDir, 'agent.log'),
134
+ stderr: join(logDir, 'agent.error.log'),
135
+ };
136
+ }
137
+ generatePlist(options) {
138
+ let programArgs;
139
+ if (options.isDev) {
140
+ programArgs = ` <string>/usr/bin/env</string>
141
+ <string>npx</string>
142
+ <string>tsx</string>
143
+ <string>${escapeXml(options.agentScript)}</string>`;
144
+ }
145
+ else {
146
+ programArgs = ` <string>${escapeXml(options.nodePath)}</string>
147
+ <string>${escapeXml(options.agentScript)}</string>`;
148
+ }
149
+ return `<?xml version="1.0" encoding="UTF-8"?>
150
+ <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
151
+ <plist version="1.0">
152
+ <dict>
153
+ <key>Label</key>
154
+ <string>${escapeXml(options.label)}</string>
155
+
156
+ <key>ProgramArguments</key>
157
+ <array>
158
+ ${programArgs}
159
+ </array>
160
+
161
+ <key>WorkingDirectory</key>
162
+ <string>${escapeXml(options.workingDirectory)}</string>
163
+
164
+ <key>RunAtLoad</key>
165
+ <${options.runAtLoad}/>
166
+
167
+ <key>KeepAlive</key>
168
+ <dict>
169
+ <key>SuccessfulExit</key>
170
+ <false/>
171
+ <key>Crashed</key>
172
+ <true/>
173
+ </dict>
174
+
175
+ <key>StandardOutPath</key>
176
+ <string>${escapeXml(options.stdoutPath)}</string>
177
+
178
+ <key>StandardErrorPath</key>
179
+ <string>${escapeXml(options.stderrPath)}</string>
180
+
181
+ <key>EnvironmentVariables</key>
182
+ <dict>
183
+ <key>AGENT_247_CONFIG</key>
184
+ <string>${escapeXml(options.configPath)}</string>
185
+ <key>AGENT_247_DATA</key>
186
+ <string>${escapeXml(options.dataDir)}</string>
187
+ <key>PATH</key>
188
+ <string>/opt/homebrew/bin:/usr/local/bin:/usr/bin:/bin</string>
189
+ </dict>
190
+
191
+ <key>ProcessType</key>
192
+ <string>Interactive</string>
193
+
194
+ <key>ThrottleInterval</key>
195
+ <integer>5</integer>
196
+ </dict>
197
+ </plist>`;
198
+ }
199
+ }
200
+ function escapeXml(str) {
201
+ return str
202
+ .replace(/&/g, '&amp;')
203
+ .replace(/</g, '&lt;')
204
+ .replace(/>/g, '&gt;')
205
+ .replace(/"/g, '&quot;')
206
+ .replace(/'/g, '&apos;');
207
+ }
208
+ //# sourceMappingURL=launchd.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"launchd.js","sourceRoot":"","sources":["../../src/service/launchd.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import type { ServiceManager, ServiceStatus, ServiceInstallOptions, ServiceResult } from './index.js';
2
+ export declare class SystemdService implements ServiceManager {
3
+ platform: "linux";
4
+ serviceName: string;
5
+ private get unitPath();
6
+ status(): Promise<ServiceStatus>;
7
+ install(options?: ServiceInstallOptions): Promise<ServiceResult>;
8
+ uninstall(): Promise<ServiceResult>;
9
+ start(): Promise<ServiceResult>;
10
+ stop(): Promise<ServiceResult>;
11
+ restart(): Promise<ServiceResult>;
12
+ getLogPaths(): {
13
+ stdout: string;
14
+ stderr: string;
15
+ };
16
+ private generateUnit;
17
+ }
18
+ //# sourceMappingURL=systemd.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"systemd.d.ts","sourceRoot":"","sources":["../../src/service/systemd.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,cAAc,EAAE,aAAa,EAAE,qBAAqB,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AAQtG,qBAAa,cAAe,YAAW,cAAc;IACnD,QAAQ,EAAG,OAAO,CAAU;IAC5B,WAAW,SAAgB;IAE3B,OAAO,KAAK,QAAQ,GAEnB;IAEK,MAAM,IAAI,OAAO,CAAC,aAAa,CAAC;IAyChC,OAAO,CAAC,OAAO,GAAE,qBAA0B,GAAG,OAAO,CAAC,aAAa,CAAC;IAqEpE,SAAS,IAAI,OAAO,CAAC,aAAa,CAAC;IA4BnC,KAAK,IAAI,OAAO,CAAC,aAAa,CAAC;IAS/B,IAAI,IAAI,OAAO,CAAC,aAAa,CAAC;IAS9B,OAAO,IAAI,OAAO,CAAC,aAAa,CAAC;IASvC,WAAW,IAAI;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE;IAOjD,OAAO,CAAC,YAAY;CAqCrB"}
@@ -0,0 +1,196 @@
1
+ import { existsSync, writeFileSync, unlinkSync, mkdirSync } from 'fs';
2
+ import { join } from 'path';
3
+ import { homedir } from 'os';
4
+ import { exec } from 'child_process';
5
+ import { promisify } from 'util';
6
+ import { getAgentPaths } from '../lib/paths.js';
7
+ import { checkTmux } from '../lib/prerequisites.js';
8
+ const execAsync = promisify(exec);
9
+ const SERVICE_NAME = '247-agent';
10
+ export class SystemdService {
11
+ platform = 'linux';
12
+ serviceName = SERVICE_NAME;
13
+ get unitPath() {
14
+ return join(homedir(), '.config', 'systemd', 'user', `${SERVICE_NAME}.service`);
15
+ }
16
+ async status() {
17
+ const installed = existsSync(this.unitPath);
18
+ let running = false;
19
+ let enabled = false;
20
+ let pid;
21
+ if (installed) {
22
+ try {
23
+ const { stdout: statusOutput } = await execAsync(`systemctl --user is-active ${SERVICE_NAME} 2>/dev/null || true`);
24
+ running = statusOutput.trim() === 'active';
25
+ const { stdout: enabledOutput } = await execAsync(`systemctl --user is-enabled ${SERVICE_NAME} 2>/dev/null || true`);
26
+ enabled = enabledOutput.trim() === 'enabled';
27
+ if (running) {
28
+ const { stdout: pidOutput } = await execAsync(`systemctl --user show ${SERVICE_NAME} --property=MainPID --value 2>/dev/null || true`);
29
+ const parsedPid = parseInt(pidOutput.trim(), 10);
30
+ if (!isNaN(parsedPid) && parsedPid > 0) {
31
+ pid = parsedPid;
32
+ }
33
+ }
34
+ }
35
+ catch {
36
+ // Service not available
37
+ }
38
+ }
39
+ return {
40
+ installed,
41
+ running,
42
+ enabled,
43
+ pid,
44
+ configPath: installed ? this.unitPath : undefined,
45
+ };
46
+ }
47
+ async install(options = {}) {
48
+ const paths = getAgentPaths();
49
+ // Verify tmux is installed
50
+ const tmuxCheck = checkTmux();
51
+ if (tmuxCheck.status === 'error') {
52
+ return {
53
+ success: false,
54
+ error: 'tmux is not installed. Please install it first: sudo apt install tmux',
55
+ };
56
+ }
57
+ // Create systemd user directory
58
+ const systemdUserDir = join(homedir(), '.config', 'systemd', 'user');
59
+ if (!existsSync(systemdUserDir)) {
60
+ mkdirSync(systemdUserDir, { recursive: true });
61
+ }
62
+ // Create log directory
63
+ const logDir = join(homedir(), '.local', 'log', '247-agent');
64
+ if (!existsSync(logDir)) {
65
+ mkdirSync(logDir, { recursive: true });
66
+ }
67
+ // Determine entry point
68
+ const entryPoint = paths.isDev
69
+ ? join(paths.agentRoot, 'src', 'index.ts')
70
+ : join(paths.agentRoot, 'dist', 'index.js');
71
+ // Generate unit content
72
+ const unitContent = this.generateUnit({
73
+ description: '247 Agent - The Vibe Company',
74
+ nodePath: paths.nodePath,
75
+ agentScript: entryPoint,
76
+ workingDirectory: paths.agentRoot,
77
+ isDev: paths.isDev,
78
+ configPath: paths.configPath,
79
+ dataDir: paths.dataDir,
80
+ });
81
+ writeFileSync(this.unitPath, unitContent, 'utf-8');
82
+ // Reload systemd
83
+ try {
84
+ await execAsync('systemctl --user daemon-reload');
85
+ }
86
+ catch (err) {
87
+ return { success: false, error: `Failed to reload systemd: ${err.message}` };
88
+ }
89
+ // Enable at boot if requested
90
+ if (options.enableAtBoot ?? true) {
91
+ try {
92
+ await execAsync(`systemctl --user enable ${SERVICE_NAME}`);
93
+ }
94
+ catch (err) {
95
+ return { success: false, error: `Failed to enable service: ${err.message}` };
96
+ }
97
+ }
98
+ // Start now if requested
99
+ if (options.startNow) {
100
+ const startResult = await this.start();
101
+ if (!startResult.success) {
102
+ return { ...startResult, configPath: this.unitPath };
103
+ }
104
+ }
105
+ return { success: true, configPath: this.unitPath };
106
+ }
107
+ async uninstall() {
108
+ const status = await this.status();
109
+ // Stop and disable first
110
+ if (status.running) {
111
+ await this.stop();
112
+ }
113
+ if (status.enabled) {
114
+ try {
115
+ await execAsync(`systemctl --user disable ${SERVICE_NAME}`);
116
+ }
117
+ catch {
118
+ // Ignore disable errors
119
+ }
120
+ }
121
+ // Remove unit file
122
+ if (existsSync(this.unitPath)) {
123
+ try {
124
+ unlinkSync(this.unitPath);
125
+ await execAsync('systemctl --user daemon-reload');
126
+ }
127
+ catch (err) {
128
+ return { success: false, error: `Failed to remove unit file: ${err.message}` };
129
+ }
130
+ }
131
+ return { success: true };
132
+ }
133
+ async start() {
134
+ try {
135
+ await execAsync(`systemctl --user start ${SERVICE_NAME}`);
136
+ return { success: true };
137
+ }
138
+ catch (err) {
139
+ return { success: false, error: err.message };
140
+ }
141
+ }
142
+ async stop() {
143
+ try {
144
+ await execAsync(`systemctl --user stop ${SERVICE_NAME}`);
145
+ return { success: true };
146
+ }
147
+ catch (err) {
148
+ return { success: false, error: err.message };
149
+ }
150
+ }
151
+ async restart() {
152
+ try {
153
+ await execAsync(`systemctl --user restart ${SERVICE_NAME}`);
154
+ return { success: true };
155
+ }
156
+ catch (err) {
157
+ return { success: false, error: err.message };
158
+ }
159
+ }
160
+ getLogPaths() {
161
+ return {
162
+ stdout: `journalctl --user -u ${SERVICE_NAME} -o cat`,
163
+ stderr: `journalctl --user -u ${SERVICE_NAME} -p err -o cat`,
164
+ };
165
+ }
166
+ generateUnit(options) {
167
+ let execStart;
168
+ if (options.isDev) {
169
+ execStart = `/usr/bin/env npx tsx ${options.agentScript}`;
170
+ }
171
+ else {
172
+ execStart = `${options.nodePath} ${options.agentScript}`;
173
+ }
174
+ return `[Unit]
175
+ Description=${options.description}
176
+ After=network.target
177
+
178
+ [Service]
179
+ Type=simple
180
+ ExecStart=${execStart}
181
+ WorkingDirectory=${options.workingDirectory}
182
+ Restart=on-failure
183
+ RestartSec=5
184
+ StandardOutput=journal
185
+ StandardError=journal
186
+
187
+ Environment="AGENT_247_CONFIG=${options.configPath}"
188
+ Environment="AGENT_247_DATA=${options.dataDir}"
189
+ Environment="PATH=/usr/local/bin:/usr/bin:/bin"
190
+
191
+ [Install]
192
+ WantedBy=default.target
193
+ `;
194
+ }
195
+ }
196
+ //# sourceMappingURL=systemd.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"systemd.js","sourceRoot":"","sources":["../../src/service/systemd.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,57 @@
1
+ {
2
+ "name": "@vibecompany/247-cli",
3
+ "version": "0.1.0",
4
+ "description": "247 - Access Claude Code from anywhere 24/7",
5
+ "keywords": ["claude", "terminal", "remote", "ai", "cli", "247", "vibecompany"],
6
+ "license": "MIT",
7
+ "author": "The Vibe Company",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "https://github.com/The-Vibe-Company/247",
11
+ "directory": "packages/cli"
12
+ },
13
+ "homepage": "https://247.thevibecompany.co",
14
+ "bugs": {
15
+ "url": "https://github.com/The-Vibe-Company/247/issues"
16
+ },
17
+ "type": "module",
18
+ "engines": {
19
+ "node": ">=22.0.0"
20
+ },
21
+ "os": ["darwin", "linux"],
22
+ "cpu": ["x64", "arm64"],
23
+ "bin": {
24
+ "247": "./dist/index.js"
25
+ },
26
+ "files": [
27
+ "dist",
28
+ "README.md"
29
+ ],
30
+ "scripts": {
31
+ "dev": "tsx src/index.ts",
32
+ "build": "tsc",
33
+ "typecheck": "tsc --noEmit",
34
+ "prepublishOnly": "npm run build"
35
+ },
36
+ "dependencies": {
37
+ "commander": "^12.0.0",
38
+ "chalk": "^5.3.0",
39
+ "ora": "^8.0.0",
40
+ "enquirer": "^2.4.0",
41
+ "fs-extra": "^11.2.0",
42
+ "semver": "^7.6.0",
43
+ "@homebridge/node-pty-prebuilt-multiarch": "^0.13.1",
44
+ "better-sqlite3": "^12.5.0",
45
+ "express": "^4.21.0",
46
+ "ws": "^8.18.0",
47
+ "cors": "^2.8.5",
48
+ "http-proxy": "^1.18.1"
49
+ },
50
+ "devDependencies": {
51
+ "@types/better-sqlite3": "^7.6.13",
52
+ "@types/fs-extra": "^11.0.4",
53
+ "@types/node": "^22.10.5",
54
+ "tsx": "^4.19.2",
55
+ "typescript": "^5.7.2"
56
+ }
57
+ }