@ebowwa/pkg-ops 0.1.20 → 0.1.22
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/package.json +1 -2
- package/rust/src/lib.rs +3 -1
- package/src/bridge.ts +0 -506
- package/src/config.ts +0 -364
- package/src/health-server.ts +0 -280
- package/src/index.ts +0 -1187
- package/src/service-manager.ts +0 -216
- package/src/types.ts +0 -240
package/src/index.ts
DELETED
|
@@ -1,1187 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env bun
|
|
2
|
-
/**
|
|
3
|
-
* PkgOps CLI
|
|
4
|
-
*
|
|
5
|
-
* Package operations CLI that installs @ebowwa/* npm packages
|
|
6
|
-
* and manages systemd services.
|
|
7
|
-
*
|
|
8
|
-
* @example
|
|
9
|
-
* ```bash
|
|
10
|
-
* # Install a package
|
|
11
|
-
* pkg-ops install @ebowwa/stack@0.7.12
|
|
12
|
-
*
|
|
13
|
-
* # Service management
|
|
14
|
-
* pkg-ops service start stack
|
|
15
|
-
* pkg-ops service status stack
|
|
16
|
-
*
|
|
17
|
-
* # Health check
|
|
18
|
-
* pkg-ops health
|
|
19
|
-
*
|
|
20
|
-
* # List packages
|
|
21
|
-
* pkg-ops list
|
|
22
|
-
* ```
|
|
23
|
-
*/
|
|
24
|
-
|
|
25
|
-
import { readFileSync, existsSync } from "node:fs";
|
|
26
|
-
import {
|
|
27
|
-
loadConfig,
|
|
28
|
-
parsePackageSpec,
|
|
29
|
-
isValidPackageName,
|
|
30
|
-
getPackageConfig,
|
|
31
|
-
updatePackageConfig,
|
|
32
|
-
removePackageConfig,
|
|
33
|
-
listManagedPackages,
|
|
34
|
-
} from "./config.js";
|
|
35
|
-
import { ServiceManager, getServiceManager } from "./service-manager.js";
|
|
36
|
-
import { RustBridge, startBridge, stopBridge } from "./bridge.js";
|
|
37
|
-
import { startHealthServer, stopHealthServer, getHealthServer } from "./health-server.js";
|
|
38
|
-
import type { VerifyResult, AuditResult, BundleSize, InstalledPackageInfo } from "./bridge.js";
|
|
39
|
-
|
|
40
|
-
// ---------------------------------------------------------------------------
|
|
41
|
-
// Types
|
|
42
|
-
// ---------------------------------------------------------------------------
|
|
43
|
-
|
|
44
|
-
interface Command {
|
|
45
|
-
name: string;
|
|
46
|
-
description: string;
|
|
47
|
-
usage: string;
|
|
48
|
-
handler: (args: string[]) => Promise<void> | void;
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
interface Subcommand {
|
|
52
|
-
name: string;
|
|
53
|
-
description: string;
|
|
54
|
-
usage: string;
|
|
55
|
-
handler: (args: string[]) => Promise<void> | void;
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// ---------------------------------------------------------------------------
|
|
59
|
-
// Package Management Commands
|
|
60
|
-
// ---------------------------------------------------------------------------
|
|
61
|
-
|
|
62
|
-
async function handleInstall(args: string[]): Promise<void> {
|
|
63
|
-
if (args.length === 0) {
|
|
64
|
-
console.error("Usage: pkg-ops install <package>[@version]");
|
|
65
|
-
process.exit(1);
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
const spec = args[0];
|
|
69
|
-
const { name, version } = parsePackageSpec(spec);
|
|
70
|
-
|
|
71
|
-
if (!isValidPackageName(name)) {
|
|
72
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
73
|
-
process.exit(1);
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
console.log(`Installing ${name}@${version}...`);
|
|
77
|
-
|
|
78
|
-
try {
|
|
79
|
-
const bridge = await startBridge();
|
|
80
|
-
const result = await bridge.install(name, version);
|
|
81
|
-
|
|
82
|
-
if (result.success) {
|
|
83
|
-
console.log(`Successfully installed ${name}@${result.version}`);
|
|
84
|
-
if (result.previousVersion) {
|
|
85
|
-
console.log(` Previous version: ${result.previousVersion}`);
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
// Update config (add version to versions map)
|
|
89
|
-
const existingConfig = getPackageConfig(name);
|
|
90
|
-
updatePackageConfig(name, {
|
|
91
|
-
version: result.version,
|
|
92
|
-
versions: {
|
|
93
|
-
...existingConfig?.versions,
|
|
94
|
-
[result.version]: {
|
|
95
|
-
installedAt: new Date().toISOString(),
|
|
96
|
-
distSizeBytes: null,
|
|
97
|
-
fileCount: null,
|
|
98
|
-
},
|
|
99
|
-
},
|
|
100
|
-
service: name.replace("@ebowwa/", ""),
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
// Start service if configured
|
|
104
|
-
const config = getPackageConfig(name);
|
|
105
|
-
if (config?.service && config.autoStart !== false) {
|
|
106
|
-
const sm = getServiceManager();
|
|
107
|
-
const startResult = await sm.start(config.service);
|
|
108
|
-
if (startResult.success) {
|
|
109
|
-
console.log(` Service ${config.service} started`);
|
|
110
|
-
} else {
|
|
111
|
-
console.warn(` Failed to start service: ${startResult.message}`);
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
} else {
|
|
115
|
-
console.error(`Failed to install ${name}: ${result.message}`);
|
|
116
|
-
process.exit(1);
|
|
117
|
-
}
|
|
118
|
-
} catch (error) {
|
|
119
|
-
console.error(`Install failed:`, error);
|
|
120
|
-
process.exit(1);
|
|
121
|
-
} finally {
|
|
122
|
-
await stopBridge();
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
async function handleUpdate(args: string[]): Promise<void> {
|
|
127
|
-
if (args.length === 0) {
|
|
128
|
-
console.error("Usage: pkg-ops update <package>");
|
|
129
|
-
process.exit(1);
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
const packageName = args[0];
|
|
133
|
-
|
|
134
|
-
if (!isValidPackageName(packageName)) {
|
|
135
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
136
|
-
process.exit(1);
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
console.log(`Updating ${packageName}...`);
|
|
140
|
-
|
|
141
|
-
try {
|
|
142
|
-
const bridge = await startBridge();
|
|
143
|
-
const result = await bridge.update(packageName);
|
|
144
|
-
|
|
145
|
-
if (result.success) {
|
|
146
|
-
console.log(`Successfully updated ${packageName} to ${result.version}`);
|
|
147
|
-
if (result.previousVersion) {
|
|
148
|
-
console.log(` Previous version: ${result.previousVersion}`);
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
// Update config (add version to versions map)
|
|
152
|
-
const existingConfig = getPackageConfig(packageName);
|
|
153
|
-
updatePackageConfig(packageName, {
|
|
154
|
-
version: result.version,
|
|
155
|
-
versions: {
|
|
156
|
-
...existingConfig?.versions,
|
|
157
|
-
[result.version]: {
|
|
158
|
-
installedAt: new Date().toISOString(),
|
|
159
|
-
distSizeBytes: null,
|
|
160
|
-
fileCount: null,
|
|
161
|
-
},
|
|
162
|
-
},
|
|
163
|
-
service: packageName.replace("@ebowwa/", ""),
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
// Restart service if running
|
|
167
|
-
const sm = getServiceManager();
|
|
168
|
-
const config = getPackageConfig(packageName);
|
|
169
|
-
if (config?.service) {
|
|
170
|
-
const status = await sm.status(config.service);
|
|
171
|
-
if (status.active) {
|
|
172
|
-
await sm.restart(config.service);
|
|
173
|
-
console.log(` Service ${config.service} restarted`);
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
} else {
|
|
177
|
-
console.error(`Failed to update ${packageName}: ${result.message}`);
|
|
178
|
-
process.exit(1);
|
|
179
|
-
}
|
|
180
|
-
} catch (error) {
|
|
181
|
-
console.error(`Update failed:`, error);
|
|
182
|
-
process.exit(1);
|
|
183
|
-
} finally {
|
|
184
|
-
await stopBridge();
|
|
185
|
-
}
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
async function handleUpdateAll(_args: string[]): Promise<void> {
|
|
189
|
-
console.log("Updating all managed packages...");
|
|
190
|
-
|
|
191
|
-
try {
|
|
192
|
-
const bridge = await startBridge();
|
|
193
|
-
const results = await bridge.updateAll();
|
|
194
|
-
|
|
195
|
-
for (const result of results) {
|
|
196
|
-
if (result.success) {
|
|
197
|
-
console.log(`Updated ${result.version}`);
|
|
198
|
-
} else {
|
|
199
|
-
console.error(`Failed to update: ${result.message}`);
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
} catch (error) {
|
|
203
|
-
console.error("Update all failed:", error);
|
|
204
|
-
process.exit(1);
|
|
205
|
-
} finally {
|
|
206
|
-
await stopBridge();
|
|
207
|
-
}
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
async function handleList(_args: string[]): Promise<void> {
|
|
211
|
-
console.log("Installed packages:\n");
|
|
212
|
-
|
|
213
|
-
try {
|
|
214
|
-
const bridge = await startBridge();
|
|
215
|
-
const packages = await bridge.list();
|
|
216
|
-
|
|
217
|
-
if (packages.length === 0) {
|
|
218
|
-
console.log(" No packages installed");
|
|
219
|
-
return;
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
for (const pkg of packages) {
|
|
223
|
-
const status = pkg.installed ? "installed" : "not installed";
|
|
224
|
-
const service = pkg.service ? ` (service: ${pkg.service})` : "";
|
|
225
|
-
console.log(` ${pkg.name}@${pkg.version} [${status}]${service}`);
|
|
226
|
-
}
|
|
227
|
-
} catch (error) {
|
|
228
|
-
console.error("Failed to list packages:", error);
|
|
229
|
-
process.exit(1);
|
|
230
|
-
} finally {
|
|
231
|
-
await stopBridge();
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
async function handleRollback(args: string[]): Promise<void> {
|
|
236
|
-
if (args.length === 0) {
|
|
237
|
-
console.error("Usage: pkg-ops rollback <package>");
|
|
238
|
-
process.exit(1);
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
const packageName = args[0];
|
|
242
|
-
|
|
243
|
-
console.log(`Rolling back ${packageName}...`);
|
|
244
|
-
|
|
245
|
-
try {
|
|
246
|
-
const bridge = await startBridge();
|
|
247
|
-
const result = await bridge.rollback(packageName);
|
|
248
|
-
|
|
249
|
-
if (result.success) {
|
|
250
|
-
console.log(`Rolled back ${packageName} from ${result.currentVersion} to ${result.previousVersion}`);
|
|
251
|
-
|
|
252
|
-
// Restart service if running
|
|
253
|
-
const sm = getServiceManager();
|
|
254
|
-
const config = getPackageConfig(packageName);
|
|
255
|
-
if (config?.service) {
|
|
256
|
-
const status = await sm.status(config.service);
|
|
257
|
-
if (status.active) {
|
|
258
|
-
await sm.restart(config.service);
|
|
259
|
-
console.log(` Service ${config.service} restarted`);
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
} else {
|
|
263
|
-
console.error(`Rollback failed: ${result.message}`);
|
|
264
|
-
process.exit(1);
|
|
265
|
-
}
|
|
266
|
-
} catch (error) {
|
|
267
|
-
console.error("Rollback failed:", error);
|
|
268
|
-
process.exit(1);
|
|
269
|
-
} finally {
|
|
270
|
-
await stopBridge();
|
|
271
|
-
}
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
// ---------------------------------------------------------------------------
|
|
275
|
-
// Service Management Commands
|
|
276
|
-
// ---------------------------------------------------------------------------
|
|
277
|
-
|
|
278
|
-
async function handleServiceStart(args: string[]): Promise<void> {
|
|
279
|
-
if (args.length === 0) {
|
|
280
|
-
console.error("Usage: pkg-ops service start <name>");
|
|
281
|
-
process.exit(1);
|
|
282
|
-
}
|
|
283
|
-
|
|
284
|
-
const serviceName = args[0];
|
|
285
|
-
const sm = getServiceManager();
|
|
286
|
-
|
|
287
|
-
console.log(`Starting ${serviceName}...`);
|
|
288
|
-
const result = await sm.start(serviceName);
|
|
289
|
-
|
|
290
|
-
if (result.success) {
|
|
291
|
-
console.log(`Service ${serviceName} started`);
|
|
292
|
-
} else {
|
|
293
|
-
console.error(`Failed to start ${serviceName}: ${result.message}`);
|
|
294
|
-
process.exit(1);
|
|
295
|
-
}
|
|
296
|
-
}
|
|
297
|
-
|
|
298
|
-
async function handleServiceStop(args: string[]): Promise<void> {
|
|
299
|
-
if (args.length === 0) {
|
|
300
|
-
console.error("Usage: pkg-ops service stop <name>");
|
|
301
|
-
process.exit(1);
|
|
302
|
-
}
|
|
303
|
-
|
|
304
|
-
const serviceName = args[0];
|
|
305
|
-
const sm = getServiceManager();
|
|
306
|
-
|
|
307
|
-
console.log(`Stopping ${serviceName}...`);
|
|
308
|
-
const result = await sm.stop(serviceName);
|
|
309
|
-
|
|
310
|
-
if (result.success) {
|
|
311
|
-
console.log(`Service ${serviceName} stopped`);
|
|
312
|
-
} else {
|
|
313
|
-
console.error(`Failed to stop ${serviceName}: ${result.message}`);
|
|
314
|
-
process.exit(1);
|
|
315
|
-
}
|
|
316
|
-
}
|
|
317
|
-
|
|
318
|
-
async function handleServiceRestart(args: string[]): Promise<void> {
|
|
319
|
-
if (args.length === 0) {
|
|
320
|
-
console.error("Usage: pkg-ops service restart <name>");
|
|
321
|
-
process.exit(1);
|
|
322
|
-
}
|
|
323
|
-
|
|
324
|
-
const serviceName = args[0];
|
|
325
|
-
const sm = getServiceManager();
|
|
326
|
-
|
|
327
|
-
console.log(`Restarting ${serviceName}...`);
|
|
328
|
-
const result = await sm.restart(serviceName);
|
|
329
|
-
|
|
330
|
-
if (result.success) {
|
|
331
|
-
console.log(`Service ${serviceName} restarted`);
|
|
332
|
-
} else {
|
|
333
|
-
console.error(`Failed to restart ${serviceName}: ${result.message}`);
|
|
334
|
-
process.exit(1);
|
|
335
|
-
}
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
async function handleServiceStatus(args: string[]): Promise<void> {
|
|
339
|
-
if (args.length === 0) {
|
|
340
|
-
console.error("Usage: pkg-ops service status <name>");
|
|
341
|
-
process.exit(1);
|
|
342
|
-
}
|
|
343
|
-
|
|
344
|
-
const serviceName = args[0];
|
|
345
|
-
const sm = getServiceManager();
|
|
346
|
-
|
|
347
|
-
const info = await sm.info(serviceName);
|
|
348
|
-
|
|
349
|
-
if (!info.exists) {
|
|
350
|
-
console.log(`Service ${serviceName} not found`);
|
|
351
|
-
return;
|
|
352
|
-
}
|
|
353
|
-
|
|
354
|
-
console.log(`Service: ${serviceName}`);
|
|
355
|
-
console.log(` Loaded: ${info.status.loaded}`);
|
|
356
|
-
console.log(` Active: ${info.status.active}`);
|
|
357
|
-
console.log(` State: ${info.status.subState}`);
|
|
358
|
-
console.log(` PID: ${info.status.mainPid || "N/A"}`);
|
|
359
|
-
console.log(` Description: ${info.status.description || "N/A"}`);
|
|
360
|
-
}
|
|
361
|
-
|
|
362
|
-
async function handleServiceLogs(args: string[]): Promise<void> {
|
|
363
|
-
if (args.length === 0) {
|
|
364
|
-
console.error("Usage: pkg-ops service logs <name> [--lines N]");
|
|
365
|
-
process.exit(1);
|
|
366
|
-
}
|
|
367
|
-
|
|
368
|
-
const serviceName = args[0];
|
|
369
|
-
const linesIndex = args.indexOf("--lines");
|
|
370
|
-
const lines = linesIndex >= 0 && args[linesIndex + 1]
|
|
371
|
-
? parseInt(args[linesIndex + 1], 10)
|
|
372
|
-
: 100;
|
|
373
|
-
|
|
374
|
-
const sm = getServiceManager();
|
|
375
|
-
const logs = await sm.logs(serviceName, { lines });
|
|
376
|
-
|
|
377
|
-
console.log(logs);
|
|
378
|
-
}
|
|
379
|
-
|
|
380
|
-
async function handleServiceEnable(args: string[]): Promise<void> {
|
|
381
|
-
if (args.length === 0) {
|
|
382
|
-
console.error("Usage: pkg-ops service enable <name>");
|
|
383
|
-
process.exit(1);
|
|
384
|
-
}
|
|
385
|
-
|
|
386
|
-
const serviceName = args[0];
|
|
387
|
-
const sm = getServiceManager();
|
|
388
|
-
|
|
389
|
-
console.log(`Enabling ${serviceName}...`);
|
|
390
|
-
const result = await sm.enable(serviceName);
|
|
391
|
-
|
|
392
|
-
if (result.success) {
|
|
393
|
-
console.log(`Service ${serviceName} enabled on boot`);
|
|
394
|
-
} else {
|
|
395
|
-
console.error(`Failed to enable ${serviceName}: ${result.message}`);
|
|
396
|
-
process.exit(1);
|
|
397
|
-
}
|
|
398
|
-
}
|
|
399
|
-
|
|
400
|
-
async function handleServiceDisable(args: string[]): Promise<void> {
|
|
401
|
-
if (args.length === 0) {
|
|
402
|
-
console.error("Usage: pkg-ops service disable <name>");
|
|
403
|
-
process.exit(1);
|
|
404
|
-
}
|
|
405
|
-
|
|
406
|
-
const serviceName = args[0];
|
|
407
|
-
const sm = getServiceManager();
|
|
408
|
-
|
|
409
|
-
console.log(`Disabling ${serviceName}...`);
|
|
410
|
-
const result = await sm.disable(serviceName);
|
|
411
|
-
|
|
412
|
-
if (result.success) {
|
|
413
|
-
console.log(`Service ${serviceName} disabled from boot`);
|
|
414
|
-
} else {
|
|
415
|
-
console.error(`Failed to disable ${serviceName}: ${result.message}`);
|
|
416
|
-
process.exit(1);
|
|
417
|
-
}
|
|
418
|
-
}
|
|
419
|
-
|
|
420
|
-
// ---------------------------------------------------------------------------
|
|
421
|
-
// Health Commands
|
|
422
|
-
// ---------------------------------------------------------------------------
|
|
423
|
-
|
|
424
|
-
async function handleHealth(args: string[]): Promise<void> {
|
|
425
|
-
const serviceName = args[0];
|
|
426
|
-
|
|
427
|
-
const healthServer = getHealthServer();
|
|
428
|
-
if (!healthServer) {
|
|
429
|
-
console.error("Health server not initialized");
|
|
430
|
-
process.exit(1);
|
|
431
|
-
}
|
|
432
|
-
|
|
433
|
-
const health = await healthServer.checkHealth(serviceName);
|
|
434
|
-
|
|
435
|
-
if (!health) {
|
|
436
|
-
console.log("Service not found");
|
|
437
|
-
return;
|
|
438
|
-
}
|
|
439
|
-
|
|
440
|
-
if ("services" in health) {
|
|
441
|
-
// SystemHealth
|
|
442
|
-
if (health.healthy) {
|
|
443
|
-
console.log("All services healthy");
|
|
444
|
-
} else {
|
|
445
|
-
console.log("Some services unhealthy");
|
|
446
|
-
}
|
|
447
|
-
|
|
448
|
-
for (const service of health.services) {
|
|
449
|
-
const icon = service.healthy ? "OK" : "FAIL";
|
|
450
|
-
console.log(` [${icon}] ${service.name}: ${service.status.subState}`);
|
|
451
|
-
}
|
|
452
|
-
} else {
|
|
453
|
-
// ServiceHealth
|
|
454
|
-
const icon = health.healthy ? "OK" : "FAIL";
|
|
455
|
-
console.log(`[${icon}] ${health.name}: ${health.status.subState}`);
|
|
456
|
-
}
|
|
457
|
-
}
|
|
458
|
-
|
|
459
|
-
// ---------------------------------------------------------------------------
|
|
460
|
-
// Config Commands
|
|
461
|
-
// ---------------------------------------------------------------------------
|
|
462
|
-
|
|
463
|
-
async function handleConfigShow(_args: string[]): Promise<void> {
|
|
464
|
-
const config = loadConfig();
|
|
465
|
-
|
|
466
|
-
console.log("PkgOps Configuration");
|
|
467
|
-
console.log(`Config path: /etc/pkg-ops/config.json`);
|
|
468
|
-
console.log(`Health port: ${config.healthPort}`);
|
|
469
|
-
console.log(`Work dir: ${config.workDir}`);
|
|
470
|
-
console.log(`Log level: ${config.logLevel}`);
|
|
471
|
-
console.log("\nPackages:");
|
|
472
|
-
|
|
473
|
-
const packages = listManagedPackages();
|
|
474
|
-
if (packages.length === 0) {
|
|
475
|
-
console.log(" No packages configured");
|
|
476
|
-
return;
|
|
477
|
-
}
|
|
478
|
-
|
|
479
|
-
for (const { name, config: pkgConfig } of packages) {
|
|
480
|
-
console.log(` ${name}:`);
|
|
481
|
-
console.log(` Version: ${pkgConfig.version}`);
|
|
482
|
-
if (pkgConfig.service) {
|
|
483
|
-
console.log(` Service: ${pkgConfig.service}`);
|
|
484
|
-
}
|
|
485
|
-
if (pkgConfig.autoStart !== undefined) {
|
|
486
|
-
console.log(` Auto-start: ${pkgConfig.autoStart}`);
|
|
487
|
-
}
|
|
488
|
-
}
|
|
489
|
-
}
|
|
490
|
-
|
|
491
|
-
async function handleConfigSet(args: string[]): Promise<void> {
|
|
492
|
-
if (args.length < 3) {
|
|
493
|
-
console.error("Usage: pkg-ops config set <package> <key> <value>");
|
|
494
|
-
process.exit(1);
|
|
495
|
-
}
|
|
496
|
-
|
|
497
|
-
const packageName = args[0];
|
|
498
|
-
const key = args[1];
|
|
499
|
-
const value = args[2];
|
|
500
|
-
|
|
501
|
-
if (!isValidPackageName(packageName)) {
|
|
502
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
503
|
-
process.exit(1);
|
|
504
|
-
}
|
|
505
|
-
|
|
506
|
-
const config = getPackageConfig(packageName) ?? {
|
|
507
|
-
version: "latest",
|
|
508
|
-
versions: {},
|
|
509
|
-
};
|
|
510
|
-
|
|
511
|
-
// Handle different value types
|
|
512
|
-
switch (key) {
|
|
513
|
-
case "version":
|
|
514
|
-
config.version = value;
|
|
515
|
-
break;
|
|
516
|
-
case "service":
|
|
517
|
-
config.service = value;
|
|
518
|
-
break;
|
|
519
|
-
case "autoStart":
|
|
520
|
-
config.autoStart = value === "true";
|
|
521
|
-
break;
|
|
522
|
-
default:
|
|
523
|
-
console.error(`Unknown config key: ${key}`);
|
|
524
|
-
console.error("Valid keys: version, service, autoStart");
|
|
525
|
-
process.exit(1);
|
|
526
|
-
}
|
|
527
|
-
|
|
528
|
-
updatePackageConfig(packageName, config);
|
|
529
|
-
console.log(`Updated ${packageName}.${key} = ${value}`);
|
|
530
|
-
}
|
|
531
|
-
|
|
532
|
-
// ---------------------------------------------------------------------------
|
|
533
|
-
// Verification & Audit Commands
|
|
534
|
-
// ---------------------------------------------------------------------------
|
|
535
|
-
|
|
536
|
-
async function handleVerify(args: string[]): Promise<void> {
|
|
537
|
-
if (args.length === 0) {
|
|
538
|
-
console.error("Usage: pkg-ops verify <package>");
|
|
539
|
-
process.exit(1);
|
|
540
|
-
}
|
|
541
|
-
|
|
542
|
-
const packageName = args[0];
|
|
543
|
-
|
|
544
|
-
if (!isValidPackageName(packageName)) {
|
|
545
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
546
|
-
process.exit(1);
|
|
547
|
-
}
|
|
548
|
-
|
|
549
|
-
console.log(`Verifying ${packageName}...`);
|
|
550
|
-
|
|
551
|
-
try {
|
|
552
|
-
const bridge = await startBridge();
|
|
553
|
-
const result = await bridge.verify(packageName);
|
|
554
|
-
|
|
555
|
-
console.log(`\nPackage: ${result.packageName}@${result.version}`);
|
|
556
|
-
console.log(` Status: ${result.success ? "VALID" : "INVALID"}`);
|
|
557
|
-
console.log(` Dist exists: ${result.distExists ? "Yes" : "No"}`);
|
|
558
|
-
if (result.checksum) {
|
|
559
|
-
console.log(` Checksum: ${result.checksum}`);
|
|
560
|
-
}
|
|
561
|
-
console.log(` Message: ${result.message}`);
|
|
562
|
-
} catch (error) {
|
|
563
|
-
console.error("Verification failed:", error);
|
|
564
|
-
process.exit(1);
|
|
565
|
-
} finally {
|
|
566
|
-
await stopBridge();
|
|
567
|
-
}
|
|
568
|
-
}
|
|
569
|
-
|
|
570
|
-
async function handleSyncStatus(args: string[]): Promise<void> {
|
|
571
|
-
const localIndex = args.indexOf("--local");
|
|
572
|
-
const localPathIndex = args.indexOf("--local-path");
|
|
573
|
-
|
|
574
|
-
let localPath = "./package.json";
|
|
575
|
-
if (localPathIndex >= 0 && args[localPathIndex + 1]) {
|
|
576
|
-
localPath = args[localPathIndex + 1];
|
|
577
|
-
}
|
|
578
|
-
|
|
579
|
-
const showLocal = localIndex >= 0 || localPathIndex >= 0;
|
|
580
|
-
|
|
581
|
-
try {
|
|
582
|
-
const bridge = await startBridge();
|
|
583
|
-
const vpsPackages = await bridge.getInstalledInfo();
|
|
584
|
-
|
|
585
|
-
console.log("Sync Status: Local vs VPS\n");
|
|
586
|
-
|
|
587
|
-
if (showLocal && existsSync(localPath)) {
|
|
588
|
-
const localContent = readFileSync(localPath, "utf-8");
|
|
589
|
-
const localPkg = JSON.parse(localContent);
|
|
590
|
-
const deps = { ...localPkg.dependencies, ...localPkg.devDependencies };
|
|
591
|
-
|
|
592
|
-
// Build a map of VPS packages
|
|
593
|
-
const vpsMap = new Map<string, string>();
|
|
594
|
-
for (const pkg of vpsPackages) {
|
|
595
|
-
vpsMap.set(pkg.packageName, pkg.version);
|
|
596
|
-
}
|
|
597
|
-
|
|
598
|
-
// Compare local vs VPS
|
|
599
|
-
for (const [name, localVersion] of Object.entries(deps)) {
|
|
600
|
-
if (!name.startsWith("@ebowwa/")) continue;
|
|
601
|
-
|
|
602
|
-
const vpsVersion = vpsMap.get(name);
|
|
603
|
-
const cleanLocal = (localVersion as string).replace(/^[\^~]/, "");
|
|
604
|
-
|
|
605
|
-
if (vpsVersion) {
|
|
606
|
-
if (cleanLocal === vpsVersion) {
|
|
607
|
-
console.log(` ${name}: local=${cleanLocal}, vps=${vpsVersion} (in sync)`);
|
|
608
|
-
} else if (cleanLocal > vpsVersion) {
|
|
609
|
-
console.log(` ${name}: local=${cleanLocal}, vps=${vpsVersion} (VPS behind)`);
|
|
610
|
-
} else {
|
|
611
|
-
console.log(` ${name}: local=${cleanLocal}, vps=${vpsVersion} (local behind)`);
|
|
612
|
-
}
|
|
613
|
-
vpsMap.delete(name);
|
|
614
|
-
} else {
|
|
615
|
-
console.log(` ${name}: local=${cleanLocal}, vps=not installed`);
|
|
616
|
-
}
|
|
617
|
-
}
|
|
618
|
-
|
|
619
|
-
// Show VPS-only packages
|
|
620
|
-
for (const [name, version] of vpsMap) {
|
|
621
|
-
console.log(` ${name}: local=not in package.json, vps=${version}`);
|
|
622
|
-
}
|
|
623
|
-
} else {
|
|
624
|
-
// Just show VPS installed versions
|
|
625
|
-
if (vpsPackages.length === 0) {
|
|
626
|
-
console.log(" No packages installed on VPS");
|
|
627
|
-
} else {
|
|
628
|
-
for (const pkg of vpsPackages) {
|
|
629
|
-
const sizeInfo = pkg.distSizeBytes
|
|
630
|
-
? ` (${(pkg.distSizeBytes / 1024).toFixed(1)} KB)`
|
|
631
|
-
: "";
|
|
632
|
-
console.log(` ${pkg.packageName}@${pkg.version}${sizeInfo}`);
|
|
633
|
-
}
|
|
634
|
-
}
|
|
635
|
-
}
|
|
636
|
-
} catch (error) {
|
|
637
|
-
console.error("Failed to get sync status:", error);
|
|
638
|
-
process.exit(1);
|
|
639
|
-
} finally {
|
|
640
|
-
await stopBridge();
|
|
641
|
-
}
|
|
642
|
-
}
|
|
643
|
-
|
|
644
|
-
async function handleAudit(_args: string[]): Promise<void> {
|
|
645
|
-
console.log("Running vulnerability scan...\n");
|
|
646
|
-
|
|
647
|
-
try {
|
|
648
|
-
const bridge = await startBridge();
|
|
649
|
-
const results = await bridge.audit();
|
|
650
|
-
|
|
651
|
-
if (results.length === 0) {
|
|
652
|
-
console.log("No vulnerabilities found.");
|
|
653
|
-
return;
|
|
654
|
-
}
|
|
655
|
-
|
|
656
|
-
console.log(`Found ${results.length} vulnerability(es):\n`);
|
|
657
|
-
|
|
658
|
-
for (const vuln of results) {
|
|
659
|
-
console.log(`[${vuln.severity.toUpperCase()}] ${vuln.packageName}`);
|
|
660
|
-
console.log(` Vulnerability: ${vuln.vulnerability}`);
|
|
661
|
-
console.log(` Description: ${vuln.description}`);
|
|
662
|
-
console.log("");
|
|
663
|
-
}
|
|
664
|
-
} catch (error) {
|
|
665
|
-
console.error("Audit failed:", error);
|
|
666
|
-
process.exit(1);
|
|
667
|
-
} finally {
|
|
668
|
-
await stopBridge();
|
|
669
|
-
}
|
|
670
|
-
}
|
|
671
|
-
|
|
672
|
-
async function handleScan(args: string[]): Promise<void> {
|
|
673
|
-
// Alias for audit
|
|
674
|
-
await handleAudit(args);
|
|
675
|
-
}
|
|
676
|
-
|
|
677
|
-
async function handleSizes(_args: string[]): Promise<void> {
|
|
678
|
-
console.log("Bundle sizes:\n");
|
|
679
|
-
|
|
680
|
-
try {
|
|
681
|
-
const bridge = await startBridge();
|
|
682
|
-
const sizes = await bridge.getBundleSizes();
|
|
683
|
-
|
|
684
|
-
if (sizes.length === 0) {
|
|
685
|
-
console.log(" No packages installed");
|
|
686
|
-
return;
|
|
687
|
-
}
|
|
688
|
-
|
|
689
|
-
// Print table header
|
|
690
|
-
console.log(" Package | Version | Dist Size");
|
|
691
|
-
console.log(" --------------------------------|----------|-----------");
|
|
692
|
-
|
|
693
|
-
for (const pkg of sizes) {
|
|
694
|
-
const name = pkg.packageName.padEnd(32);
|
|
695
|
-
const version = pkg.version.padEnd(9);
|
|
696
|
-
const sizeKB = (pkg.distSizeBytes / 1024).toFixed(1) + " KB";
|
|
697
|
-
const files = pkg.fileCount ? ` (${pkg.fileCount} files)` : "";
|
|
698
|
-
console.log(` ${name} | ${version} | ${sizeKB}${files}`);
|
|
699
|
-
}
|
|
700
|
-
} catch (error) {
|
|
701
|
-
console.error("Failed to get bundle sizes:", error);
|
|
702
|
-
process.exit(1);
|
|
703
|
-
} finally {
|
|
704
|
-
await stopBridge();
|
|
705
|
-
}
|
|
706
|
-
}
|
|
707
|
-
|
|
708
|
-
// ---------------------------------------------------------------------------
|
|
709
|
-
// Multi-Version Commands
|
|
710
|
-
// ---------------------------------------------------------------------------
|
|
711
|
-
|
|
712
|
-
async function handleVersions(args: string[]): Promise<void> {
|
|
713
|
-
if (args.length === 0) {
|
|
714
|
-
console.error("Usage: pkg-ops versions <package>");
|
|
715
|
-
process.exit(1);
|
|
716
|
-
}
|
|
717
|
-
|
|
718
|
-
const packageName = args[0];
|
|
719
|
-
|
|
720
|
-
if (!isValidPackageName(packageName)) {
|
|
721
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
722
|
-
process.exit(1);
|
|
723
|
-
}
|
|
724
|
-
|
|
725
|
-
try {
|
|
726
|
-
const bridge = await startBridge();
|
|
727
|
-
const versions = await bridge.listVersions(packageName);
|
|
728
|
-
await stopBridge();
|
|
729
|
-
|
|
730
|
-
if (versions.length === 0) {
|
|
731
|
-
console.log(`No versions of ${packageName} installed.`);
|
|
732
|
-
return;
|
|
733
|
-
}
|
|
734
|
-
|
|
735
|
-
console.log(`Installed versions for ${packageName}:\n`);
|
|
736
|
-
|
|
737
|
-
// Sort versions descending (newest first)
|
|
738
|
-
const sorted = [...versions].sort((a, b) => b.installedAt.localeCompare(a.installedAt));
|
|
739
|
-
|
|
740
|
-
for (const v of sorted) {
|
|
741
|
-
const active = v.active ? " [ACTIVE]" : "";
|
|
742
|
-
const size = v.distSizeBytes ? ` (${(v.distSizeBytes / 1024).toFixed(1)} KB)` : "";
|
|
743
|
-
const date = new Date(v.installedAt).toLocaleString();
|
|
744
|
-
console.log(` ${v.version}${active}${size}`);
|
|
745
|
-
console.log(` Installed: ${date}`);
|
|
746
|
-
if (v.fileCount) {
|
|
747
|
-
console.log(` Files: ${v.fileCount}`);
|
|
748
|
-
}
|
|
749
|
-
}
|
|
750
|
-
|
|
751
|
-
console.log(`\nTotal: ${versions.length} version(s)`);
|
|
752
|
-
} catch (error) {
|
|
753
|
-
console.error("Failed to list versions:", error);
|
|
754
|
-
process.exit(1);
|
|
755
|
-
} finally {
|
|
756
|
-
await stopBridge();
|
|
757
|
-
}
|
|
758
|
-
}
|
|
759
|
-
|
|
760
|
-
async function handleSwitch(args: string[]): Promise<void> {
|
|
761
|
-
if (args.length === 0) {
|
|
762
|
-
console.error("Usage: pkg-ops switch <package>@<version>");
|
|
763
|
-
process.exit(1);
|
|
764
|
-
}
|
|
765
|
-
|
|
766
|
-
const spec = args[0];
|
|
767
|
-
const { name, version } = parsePackageSpec(spec);
|
|
768
|
-
|
|
769
|
-
if (!isValidPackageName(name)) {
|
|
770
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
771
|
-
process.exit(1);
|
|
772
|
-
}
|
|
773
|
-
|
|
774
|
-
if (!version || version === "latest") {
|
|
775
|
-
console.error("Error: Must specify a version to switch to (e.g., @ebowwa/stack@0.7.12)");
|
|
776
|
-
process.exit(1);
|
|
777
|
-
}
|
|
778
|
-
|
|
779
|
-
console.log(`Switching ${name} to version ${version}...`);
|
|
780
|
-
|
|
781
|
-
try {
|
|
782
|
-
const bridge = await startBridge();
|
|
783
|
-
const result = await bridge.switchVersion(name, version);
|
|
784
|
-
|
|
785
|
-
if (result.success) {
|
|
786
|
-
console.log(`Switched ${name} from ${result.fromVersion} to ${result.toVersion}`);
|
|
787
|
-
|
|
788
|
-
// Restart service if running
|
|
789
|
-
const sm = getServiceManager();
|
|
790
|
-
const config = getPackageConfig(name);
|
|
791
|
-
if (config?.service) {
|
|
792
|
-
const status = await sm.status(config.service);
|
|
793
|
-
if (status.active) {
|
|
794
|
-
await sm.restart(config.service);
|
|
795
|
-
console.log(` Service ${config.service} restarted`);
|
|
796
|
-
}
|
|
797
|
-
}
|
|
798
|
-
} else {
|
|
799
|
-
console.error(`Switch failed: ${result.message}`);
|
|
800
|
-
process.exit(1);
|
|
801
|
-
}
|
|
802
|
-
} catch (error) {
|
|
803
|
-
console.error("Switch failed:", error);
|
|
804
|
-
process.exit(1);
|
|
805
|
-
} finally {
|
|
806
|
-
await stopBridge();
|
|
807
|
-
}
|
|
808
|
-
}
|
|
809
|
-
|
|
810
|
-
async function handlePrune(args: string[]): Promise<void> {
|
|
811
|
-
if (args.length === 0) {
|
|
812
|
-
console.error("Usage: pkg-ops prune <package> [--keep N]");
|
|
813
|
-
process.exit(1);
|
|
814
|
-
}
|
|
815
|
-
|
|
816
|
-
const packageName = args[0];
|
|
817
|
-
const keepIndex = args.indexOf("--keep");
|
|
818
|
-
const keepCount = keepIndex >= 0 && args[keepIndex + 1]
|
|
819
|
-
? parseInt(args[keepIndex + 1], 10)
|
|
820
|
-
: 2; // Default: keep 2 versions
|
|
821
|
-
|
|
822
|
-
if (!isValidPackageName(packageName)) {
|
|
823
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
824
|
-
process.exit(1);
|
|
825
|
-
}
|
|
826
|
-
|
|
827
|
-
console.log(`Pruning ${packageName}, keeping ${keepCount} most recent version(s)...`);
|
|
828
|
-
|
|
829
|
-
try {
|
|
830
|
-
const bridge = await startBridge();
|
|
831
|
-
const result = await bridge.pruneVersions(packageName, keepCount);
|
|
832
|
-
await stopBridge();
|
|
833
|
-
|
|
834
|
-
if (result.success) {
|
|
835
|
-
if (result.removedVersions.length === 0) {
|
|
836
|
-
console.log(`No versions to remove (only ${result.keptVersions.length} installed).`);
|
|
837
|
-
} else {
|
|
838
|
-
console.log(`Removed ${result.removedVersions.length} version(s):`);
|
|
839
|
-
for (const v of result.removedVersions) {
|
|
840
|
-
console.log(` - ${v}`);
|
|
841
|
-
}
|
|
842
|
-
console.log(`Freed: ${(result.freedBytes / 1024).toFixed(1)} KB`);
|
|
843
|
-
console.log(`Kept: ${result.keptVersions.join(", ")}`);
|
|
844
|
-
}
|
|
845
|
-
} else {
|
|
846
|
-
console.error(`Prune failed: ${result.message}`);
|
|
847
|
-
process.exit(1);
|
|
848
|
-
}
|
|
849
|
-
} catch (error) {
|
|
850
|
-
console.error("Prune failed:", error);
|
|
851
|
-
process.exit(1);
|
|
852
|
-
} finally {
|
|
853
|
-
await stopBridge();
|
|
854
|
-
}
|
|
855
|
-
}
|
|
856
|
-
|
|
857
|
-
async function handleRemoveVersion(args: string[]): Promise<void> {
|
|
858
|
-
if (args.length === 0) {
|
|
859
|
-
console.error("Usage: pkg-ops remove-version <package>@<version>");
|
|
860
|
-
process.exit(1);
|
|
861
|
-
}
|
|
862
|
-
|
|
863
|
-
const spec = args[0];
|
|
864
|
-
const { name, version } = parsePackageSpec(spec);
|
|
865
|
-
|
|
866
|
-
if (!isValidPackageName(name)) {
|
|
867
|
-
console.error(`Invalid package name. Only @ebowwa/* packages are supported.`);
|
|
868
|
-
process.exit(1);
|
|
869
|
-
}
|
|
870
|
-
|
|
871
|
-
if (!version || version === "latest") {
|
|
872
|
-
console.error("Error: Must specify a version to remove (e.g., @ebowwa/stack@0.7.12)");
|
|
873
|
-
process.exit(1);
|
|
874
|
-
}
|
|
875
|
-
|
|
876
|
-
// Check if trying to remove active version
|
|
877
|
-
const config = getPackageConfig(name);
|
|
878
|
-
if (config?.version === version) {
|
|
879
|
-
console.error(`Error: Cannot remove active version ${version}. Switch to another version first.`);
|
|
880
|
-
process.exit(1);
|
|
881
|
-
}
|
|
882
|
-
|
|
883
|
-
console.log(`Removing ${name}@${version}...`);
|
|
884
|
-
|
|
885
|
-
try {
|
|
886
|
-
const bridge = await startBridge();
|
|
887
|
-
const result = await bridge.removeVersion(name, version);
|
|
888
|
-
await stopBridge();
|
|
889
|
-
|
|
890
|
-
if (result.success) {
|
|
891
|
-
console.log(`Removed ${name}@${version}`);
|
|
892
|
-
} else {
|
|
893
|
-
console.error(`Remove failed: ${result.message}`);
|
|
894
|
-
process.exit(1);
|
|
895
|
-
}
|
|
896
|
-
} catch (error) {
|
|
897
|
-
console.error("Remove failed:", error);
|
|
898
|
-
process.exit(1);
|
|
899
|
-
} finally {
|
|
900
|
-
await stopBridge();
|
|
901
|
-
}
|
|
902
|
-
}
|
|
903
|
-
|
|
904
|
-
async function handleMultiVersionList(_args: string[]): Promise<void> {
|
|
905
|
-
try {
|
|
906
|
-
const bridge = await startBridge();
|
|
907
|
-
const packages = await bridge.getMultiVersionPackages();
|
|
908
|
-
await stopBridge();
|
|
909
|
-
|
|
910
|
-
if (packages.length === 0) {
|
|
911
|
-
console.log("No packages with multiple versions installed.");
|
|
912
|
-
return;
|
|
913
|
-
}
|
|
914
|
-
|
|
915
|
-
console.log("Packages with multiple versions:\n");
|
|
916
|
-
|
|
917
|
-
for (const pkg of packages) {
|
|
918
|
-
console.log(` ${pkg.packageName}:`);
|
|
919
|
-
console.log(` Active: ${pkg.activeVersion}`);
|
|
920
|
-
console.log(` Total: ${pkg.totalVersions} version(s)`);
|
|
921
|
-
console.log(` Versions: ${pkg.versions.join(", ")}`);
|
|
922
|
-
console.log("");
|
|
923
|
-
}
|
|
924
|
-
} catch (error) {
|
|
925
|
-
console.error("Failed to list multi-version packages:", error);
|
|
926
|
-
process.exit(1);
|
|
927
|
-
} finally {
|
|
928
|
-
await stopBridge();
|
|
929
|
-
}
|
|
930
|
-
}
|
|
931
|
-
|
|
932
|
-
// ---------------------------------------------------------------------------
|
|
933
|
-
// CLI Entry Point
|
|
934
|
-
// ---------------------------------------------------------------------------
|
|
935
|
-
|
|
936
|
-
const commands: Command[] = [
|
|
937
|
-
// Package management
|
|
938
|
-
{ name: "install", description: "Install an @ebowwa/* package", usage: "pkg-ops install <package>[@version]", handler: handleInstall },
|
|
939
|
-
{ name: "update", description: "Update a package to latest version", usage: "pkg-ops update <package>", handler: handleUpdate },
|
|
940
|
-
{ name: "update-all", description: "Update all managed packages", usage: "pkg-ops update-all", handler: handleUpdateAll },
|
|
941
|
-
{ name: "list", description: "List installed packages", usage: "pkg-ops list", handler: handleList },
|
|
942
|
-
{ name: "rollback", description: "Rollback a package to previous version", usage: "pkg-ops rollback <package>", handler: handleRollback },
|
|
943
|
-
// Multi-version management
|
|
944
|
-
{ name: "versions", description: "List all installed versions of a package", usage: "pkg-ops versions <package>", handler: handleVersions },
|
|
945
|
-
{ name: "switch", description: "Switch to a specific installed version", usage: "pkg-ops switch <package>@<version>", handler: handleSwitch },
|
|
946
|
-
{ name: "prune", description: "Remove old versions, keeping N most recent", usage: "pkg-ops prune <package> [--keep N]", handler: handlePrune },
|
|
947
|
-
{ name: "remove-version", description: "Remove a specific version", usage: "pkg-ops remove-version <package>@<version>", handler: handleRemoveVersion },
|
|
948
|
-
{ name: "multi", description: "List packages with multiple versions", usage: "pkg-ops multi", handler: handleMultiVersionList },
|
|
949
|
-
// Service management
|
|
950
|
-
{ name: "start", description: "Start a systemd service", usage: "pkg-ops service start <name>", handler: handleServiceStart },
|
|
951
|
-
{ name: "stop", description: "Stop a systemd service", usage: "pkg-ops service stop <name>", handler: handleServiceStop },
|
|
952
|
-
{ name: "restart", description: "Restart a systemd service", usage: "pkg-ops service restart <name>", handler: handleServiceRestart },
|
|
953
|
-
{ name: "status", description: "Get service status", usage: "pkg-ops service status <name>", handler: handleServiceStatus },
|
|
954
|
-
{ name: "logs", description: "View service logs", usage: "pkg-ops service logs <name> [--lines N]", handler: handleServiceLogs },
|
|
955
|
-
{ name: "enable", description: "Enable service on boot", usage: "pkg-ops service enable <name>", handler: handleServiceEnable },
|
|
956
|
-
{ name: "disable", description: "Disable service from boot", usage: "pkg-ops service disable <name>", handler: handleServiceDisable },
|
|
957
|
-
// Health
|
|
958
|
-
{ name: "health", description: "Check health of all services", usage: "pkg-ops health [service]", handler: handleHealth },
|
|
959
|
-
// Config
|
|
960
|
-
{ name: "config", description: "Show configuration", usage: "pkg-ops config show", handler: handleConfigShow },
|
|
961
|
-
{ name: "set-config", description: "Set a config value", usage: "pkg-ops config set <package> <key> <value>", handler: handleConfigSet },
|
|
962
|
-
// Verification & Audit
|
|
963
|
-
{ name: "verify", description: "Verify package integrity", usage: "pkg-ops verify <package>", handler: handleVerify },
|
|
964
|
-
{ name: "sync-status", description: "Show local vs VPS version sync", usage: "pkg-ops sync-status [--local-path path]", handler: handleSyncStatus },
|
|
965
|
-
{ name: "audit", description: "Check for vulnerabilities", usage: "pkg-ops audit", handler: handleAudit },
|
|
966
|
-
{ name: "scan", description: "Alias for audit", usage: "pkg-ops scan", handler: handleScan },
|
|
967
|
-
{ name: "sizes", description: "Show bundle sizes", usage: "pkg-ops sizes", handler: handleSizes },
|
|
968
|
-
];
|
|
969
|
-
|
|
970
|
-
const serviceCommands: Subcommand[] = [
|
|
971
|
-
{ name: "start", description: "Start a systemd service", usage: "pkg-ops service start <name>", handler: handleServiceStart },
|
|
972
|
-
{ name: "stop", description: "Stop a systemd service", usage: "pkg-ops service stop <name>", handler: handleServiceStop },
|
|
973
|
-
{ name: "restart", description: "Restart a systemd service", usage: "pkg-ops service restart <name>", handler: handleServiceRestart },
|
|
974
|
-
{ name: "status", description: "Get service status", usage: "pkg-ops service status <name>", handler: handleServiceStatus },
|
|
975
|
-
{ name: "logs", description: "View service logs", usage: "pkg-ops service logs <name> [--lines N]", handler: handleServiceLogs },
|
|
976
|
-
{ name: "enable", description: "Enable service on boot", usage: "pkg-ops service enable <name>", handler: handleServiceEnable },
|
|
977
|
-
{ name: "disable", description: "Disable service from boot", usage: "pkg-ops service disable <name>", handler: handleServiceDisable },
|
|
978
|
-
];
|
|
979
|
-
|
|
980
|
-
const configCommands: Subcommand[] = [
|
|
981
|
-
{ name: "show", description: "Show configuration", usage: "pkg-ops config show", handler: handleConfigShow },
|
|
982
|
-
{ name: "set", description: "Set a config value", usage: "pkg-ops config set <package> <key> <value>", handler: handleConfigSet },
|
|
983
|
-
];
|
|
984
|
-
|
|
985
|
-
function printHelp(): void {
|
|
986
|
-
console.log(`
|
|
987
|
-
PkgOps - Package operations CLI for VPS
|
|
988
|
-
|
|
989
|
-
Usage:
|
|
990
|
-
pkg-ops <command> [arguments]
|
|
991
|
-
|
|
992
|
-
Package Management:
|
|
993
|
-
install <package>[@version] Install an @ebowwa/* package
|
|
994
|
-
update <package> Update a package to latest version
|
|
995
|
-
update-all Update all managed packages
|
|
996
|
-
list List installed packages
|
|
997
|
-
rollback <package> Rollback a package to previous version
|
|
998
|
-
|
|
999
|
-
Multi-Version Management:
|
|
1000
|
-
versions <package> List all installed versions
|
|
1001
|
-
switch <package>@<version> Switch to a specific version
|
|
1002
|
-
prune <package> [--keep N] Remove old versions (default: keep 2)
|
|
1003
|
-
remove-version <pkg>@<ver> Remove a specific version
|
|
1004
|
-
multi List packages with multiple versions
|
|
1005
|
-
|
|
1006
|
-
Service Management:
|
|
1007
|
-
service start <name> Start a systemd service
|
|
1008
|
-
service stop <name> Stop a systemd service
|
|
1009
|
-
service restart <name> Restart a systemd service
|
|
1010
|
-
service status <name> Get service status
|
|
1011
|
-
service logs <name> [--lines N] View service logs
|
|
1012
|
-
service enable <name> Enable service on boot
|
|
1013
|
-
service disable <name> Disable service from boot
|
|
1014
|
-
|
|
1015
|
-
Health:
|
|
1016
|
-
health [service] Check health of all services or specific one
|
|
1017
|
-
|
|
1018
|
-
Verification & Audit:
|
|
1019
|
-
verify <package> Verify package integrity
|
|
1020
|
-
sync-status [--local-path p] Show local vs VPS version sync
|
|
1021
|
-
audit Check for vulnerabilities
|
|
1022
|
-
scan Alias for audit
|
|
1023
|
-
sizes Show bundle sizes
|
|
1024
|
-
|
|
1025
|
-
Config:
|
|
1026
|
-
config show Show configuration
|
|
1027
|
-
config set <pkg> <key> <val> Set a config value
|
|
1028
|
-
|
|
1029
|
-
Options:
|
|
1030
|
-
--help, -h Show this help message
|
|
1031
|
-
--version, -v Show version
|
|
1032
|
-
|
|
1033
|
-
Examples:
|
|
1034
|
-
pkg-ops install @ebowwa/stack@0.7.12
|
|
1035
|
-
pkg-ops service start stack
|
|
1036
|
-
pkg-ops health
|
|
1037
|
-
pkg-ops sync-status --local-path ./package.json
|
|
1038
|
-
pkg-ops verify @ebowwa/stack
|
|
1039
|
-
pkg-ops audit
|
|
1040
|
-
`);
|
|
1041
|
-
}
|
|
1042
|
-
|
|
1043
|
-
async function main(): Promise<void> {
|
|
1044
|
-
const args = process.argv.slice(2);
|
|
1045
|
-
|
|
1046
|
-
if (args.length === 0 || args[0] === "--help" || args[0] === "-h") {
|
|
1047
|
-
printHelp();
|
|
1048
|
-
process.exit(0);
|
|
1049
|
-
}
|
|
1050
|
-
|
|
1051
|
-
if (args[0] === "--version" || args[0] === "-v") {
|
|
1052
|
-
const config = loadConfig();
|
|
1053
|
-
console.log(`pkg-ops ${config.packages?.["@ebowwa/pkg-ops"]?.version ?? "0.1.0"}`);
|
|
1054
|
-
process.exit(0);
|
|
1055
|
-
}
|
|
1056
|
-
|
|
1057
|
-
const commandName = args[0];
|
|
1058
|
-
const commandArgs = args.slice(1);
|
|
1059
|
-
|
|
1060
|
-
// Handle service subcommands
|
|
1061
|
-
if (commandName === "service") {
|
|
1062
|
-
if (commandArgs.length === 0) {
|
|
1063
|
-
console.error("Missing service subcommand");
|
|
1064
|
-
console.error("Usage: pkg-ops service <start|stop|restart|status|logs|enable|disable> <name>");
|
|
1065
|
-
process.exit(1);
|
|
1066
|
-
}
|
|
1067
|
-
|
|
1068
|
-
const subcommandName = commandArgs[0];
|
|
1069
|
-
const subcommand = serviceCommands.find((c) => c.name === subcommandName);
|
|
1070
|
-
|
|
1071
|
-
if (!subcommand) {
|
|
1072
|
-
console.error(`Unknown service subcommand: ${subcommandName}`);
|
|
1073
|
-
process.exit(1);
|
|
1074
|
-
}
|
|
1075
|
-
|
|
1076
|
-
await subcommand.handler(commandArgs.slice(1));
|
|
1077
|
-
return;
|
|
1078
|
-
}
|
|
1079
|
-
|
|
1080
|
-
// Handle config subcommands
|
|
1081
|
-
if (commandName === "config") {
|
|
1082
|
-
if (commandArgs.length === 0) {
|
|
1083
|
-
console.error("Missing config subcommand");
|
|
1084
|
-
console.error("Usage: pkg-ops config <show|set> [args...]");
|
|
1085
|
-
process.exit(1);
|
|
1086
|
-
}
|
|
1087
|
-
|
|
1088
|
-
const subcommandName = commandArgs[0];
|
|
1089
|
-
const subcommand = configCommands.find((c) => c.name === subcommandName);
|
|
1090
|
-
|
|
1091
|
-
if (!subcommand) {
|
|
1092
|
-
console.error(`Unknown config subcommand: ${subcommandName}`);
|
|
1093
|
-
process.exit(1);
|
|
1094
|
-
}
|
|
1095
|
-
|
|
1096
|
-
await subcommand.handler(commandArgs.slice(1));
|
|
1097
|
-
return;
|
|
1098
|
-
}
|
|
1099
|
-
|
|
1100
|
-
// Handle regular commands
|
|
1101
|
-
const command = commands.find((c) => c.name === commandName);
|
|
1102
|
-
|
|
1103
|
-
if (!command) {
|
|
1104
|
-
console.error(`Unknown command: ${commandName}`);
|
|
1105
|
-
console.error("Run 'pkg-ops --help' for usage");
|
|
1106
|
-
process.exit(1);
|
|
1107
|
-
}
|
|
1108
|
-
|
|
1109
|
-
await command.handler(commandArgs);
|
|
1110
|
-
}
|
|
1111
|
-
|
|
1112
|
-
// Run CLI only when executed directly via bin (not when imported as library)
|
|
1113
|
-
// This check works after bundling where import.meta.main gets transformed
|
|
1114
|
-
const isCliInvocation = process.argv[1]?.includes("pkg-ops");
|
|
1115
|
-
if (isCliInvocation) {
|
|
1116
|
-
main().catch((error) => {
|
|
1117
|
-
console.error("Fatal error:", error);
|
|
1118
|
-
process.exit(1);
|
|
1119
|
-
});
|
|
1120
|
-
}
|
|
1121
|
-
|
|
1122
|
-
// ---------------------------------------------------------------------------
|
|
1123
|
-
// Re-exports for library usage
|
|
1124
|
-
// ---------------------------------------------------------------------------
|
|
1125
|
-
|
|
1126
|
-
export {
|
|
1127
|
-
loadConfig,
|
|
1128
|
-
saveConfig,
|
|
1129
|
-
parsePackageSpec,
|
|
1130
|
-
isValidPackageName,
|
|
1131
|
-
getPackageConfig,
|
|
1132
|
-
updatePackageConfig,
|
|
1133
|
-
removePackageConfig,
|
|
1134
|
-
listManagedPackages,
|
|
1135
|
-
getConfigPath,
|
|
1136
|
-
ensureConfigDir,
|
|
1137
|
-
type PackageConfig,
|
|
1138
|
-
type PkgOpsConfig,
|
|
1139
|
-
} from "./config.js";
|
|
1140
|
-
|
|
1141
|
-
export {
|
|
1142
|
-
ServiceManager,
|
|
1143
|
-
getServiceManager,
|
|
1144
|
-
type ServiceInfo,
|
|
1145
|
-
type ServiceLogsOptions,
|
|
1146
|
-
} from "./service-manager.js";
|
|
1147
|
-
|
|
1148
|
-
export {
|
|
1149
|
-
RustBridge,
|
|
1150
|
-
startBridge,
|
|
1151
|
-
stopBridge,
|
|
1152
|
-
getBridge,
|
|
1153
|
-
type InstallResult,
|
|
1154
|
-
type PackageInfo,
|
|
1155
|
-
type RollbackResult,
|
|
1156
|
-
type VerifyResult,
|
|
1157
|
-
type AuditResult,
|
|
1158
|
-
type BundleSize,
|
|
1159
|
-
type InstalledPackageInfo,
|
|
1160
|
-
type VersionInfo,
|
|
1161
|
-
type SwitchResult,
|
|
1162
|
-
type PruneResult,
|
|
1163
|
-
} from "./bridge.js";
|
|
1164
|
-
|
|
1165
|
-
// Multi-version config helpers
|
|
1166
|
-
export {
|
|
1167
|
-
getInstalledVersions,
|
|
1168
|
-
isVersionInstalled,
|
|
1169
|
-
getActiveVersion,
|
|
1170
|
-
addPackageVersion,
|
|
1171
|
-
removePackageVersion,
|
|
1172
|
-
setActiveVersion,
|
|
1173
|
-
getVersionCount,
|
|
1174
|
-
getPackagesWithMultipleVersions,
|
|
1175
|
-
type VersionMetadata,
|
|
1176
|
-
} from "./config.js";
|
|
1177
|
-
|
|
1178
|
-
export {
|
|
1179
|
-
startHealthServer,
|
|
1180
|
-
stopHealthServer,
|
|
1181
|
-
getHealthServer,
|
|
1182
|
-
HealthServer,
|
|
1183
|
-
type HealthCheckResult,
|
|
1184
|
-
type ServiceHealth,
|
|
1185
|
-
type SystemHealth,
|
|
1186
|
-
} from "./health-server.js";
|
|
1187
|
-
|