@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/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
-