Compare commits
	
		
			33 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6481572981 | |||
| 0dc14a6ea1 | |||
| dea344e6ba | |||
| f81f5957ab | |||
| 281d3fbbeb | |||
| c1cb136a7d | |||
| b80275a594 | |||
| b64a515c94 | |||
| 68c4eb6480 | |||
| 6c8f6ac33f | |||
| ffa491c7a1 | |||
| 777d48d82e | |||
| b7a0bbcf6d | |||
| fbe1cd64cb | |||
| 9ba50da73c | |||
| 684319983d | |||
| 18bd9f6cda | |||
| f03c683d02 | |||
| f750299780 | |||
| ca1039408d | |||
| df3e0b9424 | |||
| c8e5960abd | |||
| 7304a62357 | |||
| a5a88e53ba | |||
| 73bc271c59 | |||
| 1e98181e71 | |||
| eb5a8185ae | |||
| ef3d3f3fa3 | |||
| 34e6e850ad | |||
| 992a776fd2 | |||
| 3e15a2d52f | |||
| d1a3576d31 | |||
| 1ca05e879b | 
							
								
								
									
										31
									
								
								.gitea/release-template.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								.gitea/release-template.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | ## NUPST {{VERSION}} | ||||||
|  |  | ||||||
|  | Pre-compiled binaries for multiple platforms. | ||||||
|  |  | ||||||
|  | ### Installation | ||||||
|  |  | ||||||
|  | #### Option 1: Via npm (recommended) | ||||||
|  | ```bash | ||||||
|  | npm install -g @serve.zone/nupst | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | #### Option 2: Via installer script | ||||||
|  | ```bash | ||||||
|  | curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | #### Option 3: Direct binary download | ||||||
|  | Download the appropriate binary for your platform from the assets below and make it executable. | ||||||
|  |  | ||||||
|  | ### Supported Platforms | ||||||
|  | - Linux x86_64 (x64) | ||||||
|  | - Linux ARM64 (aarch64) | ||||||
|  | - macOS x86_64 (Intel) | ||||||
|  | - macOS ARM64 (Apple Silicon) | ||||||
|  | - Windows x86_64 | ||||||
|  |  | ||||||
|  | ### Checksums | ||||||
|  | SHA256 checksums are provided in `SHA256SUMS.txt` for binary verification. | ||||||
|  |  | ||||||
|  | ### npm Package | ||||||
|  | The npm package includes automatic binary detection and installation for your platform. | ||||||
| @@ -21,7 +21,7 @@ jobs: | |||||||
|       - name: Set up Deno |       - name: Set up Deno | ||||||
|         uses: denoland/setup-deno@v1 |         uses: denoland/setup-deno@v1 | ||||||
|         with: |         with: | ||||||
|           deno-version: v1.x |           deno-version: v2.x | ||||||
|  |  | ||||||
|       - name: Check TypeScript types |       - name: Check TypeScript types | ||||||
|         run: deno check mod.ts |         run: deno check mod.ts | ||||||
| @@ -45,7 +45,7 @@ jobs: | |||||||
|       - name: Set up Deno |       - name: Set up Deno | ||||||
|         uses: denoland/setup-deno@v1 |         uses: denoland/setup-deno@v1 | ||||||
|         with: |         with: | ||||||
|           deno-version: v1.x |           deno-version: v2.x | ||||||
|  |  | ||||||
|       - name: Compile for current platform |       - name: Compile for current platform | ||||||
|         run: | |         run: | | ||||||
| @@ -71,7 +71,7 @@ jobs: | |||||||
|       - name: Set up Deno |       - name: Set up Deno | ||||||
|         uses: denoland/setup-deno@v1 |         uses: denoland/setup-deno@v1 | ||||||
|         with: |         with: | ||||||
|           deno-version: v1.x |           deno-version: v2.x | ||||||
|  |  | ||||||
|       - name: Compile all platform binaries |       - name: Compile all platform binaries | ||||||
|         run: bash scripts/compile-all.sh |         run: bash scripts/compile-all.sh | ||||||
|   | |||||||
							
								
								
									
										129
									
								
								.gitea/workflows/npm-publish.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								.gitea/workflows/npm-publish.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | |||||||
|  | name: Publish to npm | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     tags: | ||||||
|  |       - 'v*' | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   npm-publish: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |       - name: Set up Deno | ||||||
|  |         uses: denoland/setup-deno@v1 | ||||||
|  |         with: | ||||||
|  |           deno-version: v2.x | ||||||
|  |  | ||||||
|  |       - name: Setup Node.js for npm publishing | ||||||
|  |         uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version: '18.x' | ||||||
|  |           registry-url: 'https://registry.npmjs.org/' | ||||||
|  |  | ||||||
|  |       - name: Get version from tag | ||||||
|  |         id: version | ||||||
|  |         run: | | ||||||
|  |           VERSION=${GITHUB_REF#refs/tags/} | ||||||
|  |           echo "version=$VERSION" >> $GITHUB_OUTPUT | ||||||
|  |           echo "version_number=${VERSION#v}" >> $GITHUB_OUTPUT | ||||||
|  |           echo "Publishing version: $VERSION" | ||||||
|  |  | ||||||
|  |       - name: Verify deno.json version matches tag | ||||||
|  |         run: | | ||||||
|  |           DENO_VERSION=$(grep -o '"version": "[^"]*"' deno.json | cut -d'"' -f4) | ||||||
|  |           TAG_VERSION="${{ steps.version.outputs.version_number }}" | ||||||
|  |           echo "deno.json version: $DENO_VERSION" | ||||||
|  |           echo "Tag version: $TAG_VERSION" | ||||||
|  |           if [ "$DENO_VERSION" != "$TAG_VERSION" ]; then | ||||||
|  |             echo "ERROR: Version mismatch!" | ||||||
|  |             echo "deno.json has version $DENO_VERSION but tag is $TAG_VERSION" | ||||||
|  |             exit 1 | ||||||
|  |           fi | ||||||
|  |  | ||||||
|  |       - name: Compile binaries for npm package | ||||||
|  |         run: | | ||||||
|  |           echo "Compiling binaries for npm package..." | ||||||
|  |           deno task compile | ||||||
|  |           echo "" | ||||||
|  |           echo "Binary sizes:" | ||||||
|  |           ls -lh dist/binaries/ | ||||||
|  |  | ||||||
|  |       - name: Generate SHA256 checksums | ||||||
|  |         run: | | ||||||
|  |           cd dist/binaries | ||||||
|  |           sha256sum * > SHA256SUMS | ||||||
|  |           cat SHA256SUMS | ||||||
|  |           cd ../.. | ||||||
|  |  | ||||||
|  |       - name: Sync package.json version | ||||||
|  |         run: | | ||||||
|  |           VERSION="${{ steps.version.outputs.version_number }}" | ||||||
|  |           echo "Syncing package.json to version ${VERSION}..." | ||||||
|  |           npm version ${VERSION} --no-git-tag-version --allow-same-version | ||||||
|  |           echo "package.json version: $(grep '"version"' package.json | head -1)" | ||||||
|  |  | ||||||
|  |       - name: Create npm package | ||||||
|  |         run: | | ||||||
|  |           echo "Creating npm package..." | ||||||
|  |           npm pack | ||||||
|  |           echo "" | ||||||
|  |           echo "Package created:" | ||||||
|  |           ls -lh *.tgz | ||||||
|  |  | ||||||
|  |       - name: Test local installation | ||||||
|  |         run: | | ||||||
|  |           echo "Testing local package installation..." | ||||||
|  |           PACKAGE_FILE=$(ls *.tgz) | ||||||
|  |           npm install -g ${PACKAGE_FILE} | ||||||
|  |           echo "" | ||||||
|  |           echo "Testing nupst command:" | ||||||
|  |           nupst --version || echo "Note: Binary execution may fail in CI environment" | ||||||
|  |           echo "" | ||||||
|  |           echo "Checking installed files:" | ||||||
|  |           npm ls -g @serve.zone/nupst || true | ||||||
|  |  | ||||||
|  |       - name: Publish to npm | ||||||
|  |         env: | ||||||
|  |           NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }} | ||||||
|  |         run: | | ||||||
|  |           echo "Publishing to npm registry..." | ||||||
|  |           npm publish --access public | ||||||
|  |           echo "" | ||||||
|  |           echo "✅ Successfully published @serve.zone/nupst to npm!" | ||||||
|  |           echo "" | ||||||
|  |           echo "Package info:" | ||||||
|  |           npm view @serve.zone/nupst | ||||||
|  |  | ||||||
|  |       - name: Verify npm package | ||||||
|  |         run: | | ||||||
|  |           echo "Waiting for npm propagation..." | ||||||
|  |           sleep 30 | ||||||
|  |           echo "" | ||||||
|  |           echo "Verifying published package..." | ||||||
|  |           npm view @serve.zone/nupst | ||||||
|  |           echo "" | ||||||
|  |           echo "Testing installation from npm:" | ||||||
|  |           npm install -g @serve.zone/nupst | ||||||
|  |           echo "" | ||||||
|  |           echo "Package installed successfully!" | ||||||
|  |           which nupst || echo "Binary location check skipped" | ||||||
|  |  | ||||||
|  |       - name: Publish Summary | ||||||
|  |         run: | | ||||||
|  |           echo "================================================" | ||||||
|  |           echo "  npm Publish Complete!" | ||||||
|  |           echo "================================================" | ||||||
|  |           echo "" | ||||||
|  |           echo "✅ Package: @serve.zone/nupst" | ||||||
|  |           echo "✅ Version: ${{ steps.version.outputs.version }}" | ||||||
|  |           echo "" | ||||||
|  |           echo "Installation:" | ||||||
|  |           echo "  npm install -g @serve.zone/nupst" | ||||||
|  |           echo "" | ||||||
|  |           echo "Registry:" | ||||||
|  |           echo "  https://www.npmjs.com/package/@serve.zone/nupst" | ||||||
|  |           echo "" | ||||||
| @@ -18,7 +18,7 @@ jobs: | |||||||
|       - name: Set up Deno |       - name: Set up Deno | ||||||
|         uses: denoland/setup-deno@v1 |         uses: denoland/setup-deno@v1 | ||||||
|         with: |         with: | ||||||
|           deno-version: v1.x |           deno-version: v2.x | ||||||
|  |  | ||||||
|       - name: Get version from tag |       - name: Get version from tag | ||||||
|         id: version |         id: version | ||||||
|   | |||||||
							
								
								
									
										54
									
								
								.npmignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								.npmignore
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | |||||||
|  | # Source code (not needed for binary distribution) | ||||||
|  | /ts/ | ||||||
|  | /test/ | ||||||
|  | mod.ts | ||||||
|  | *.ts | ||||||
|  |  | ||||||
|  | # Development files | ||||||
|  | .git/ | ||||||
|  | .gitea/ | ||||||
|  | .claude/ | ||||||
|  | .serena/ | ||||||
|  | .nogit/ | ||||||
|  | .github/ | ||||||
|  | deno.json | ||||||
|  | deno.lock | ||||||
|  | tsconfig.json | ||||||
|  |  | ||||||
|  | # Scripts not needed for npm | ||||||
|  | /scripts/compile-all.sh | ||||||
|  | install.sh | ||||||
|  | uninstall.sh | ||||||
|  | example-action.sh | ||||||
|  |  | ||||||
|  | # Documentation files not needed for npm package | ||||||
|  | readme.plan.md | ||||||
|  | readme.hints.md | ||||||
|  | npm-publish-instructions.md | ||||||
|  | docs/ | ||||||
|  |  | ||||||
|  | # IDE and editor files | ||||||
|  | .vscode/ | ||||||
|  | .idea/ | ||||||
|  | *.swp | ||||||
|  | *.swo | ||||||
|  | *~ | ||||||
|  | .DS_Store | ||||||
|  |  | ||||||
|  | # Keep only the install-binary.js in scripts/ | ||||||
|  | /scripts/* | ||||||
|  | !/scripts/install-binary.js | ||||||
|  |  | ||||||
|  | # Exclude all dist directory (binaries will be downloaded during install) | ||||||
|  | /dist/ | ||||||
|  |  | ||||||
|  | # Logs and temporary files | ||||||
|  | *.log | ||||||
|  | npm-debug.log* | ||||||
|  | yarn-debug.log* | ||||||
|  | yarn-error.log* | ||||||
|  |  | ||||||
|  | # Other | ||||||
|  | node_modules/ | ||||||
|  | .env | ||||||
|  | .env.* | ||||||
							
								
								
									
										108
									
								
								bin/nupst-wrapper.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								bin/nupst-wrapper.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | |||||||
|  | #!/usr/bin/env node | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * NUPST npm wrapper | ||||||
|  |  * This script executes the appropriate pre-compiled binary based on the current platform | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | import { spawn } from 'child_process'; | ||||||
|  | import { fileURLToPath } from 'url'; | ||||||
|  | import { dirname, join } from 'path'; | ||||||
|  | import { existsSync } from 'fs'; | ||||||
|  | import { platform, arch } from 'os'; | ||||||
|  |  | ||||||
|  | const __filename = fileURLToPath(import.meta.url); | ||||||
|  | const __dirname = dirname(__filename); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Get the binary name for the current platform | ||||||
|  |  */ | ||||||
|  | function getBinaryName() { | ||||||
|  |   const plat = platform(); | ||||||
|  |   const architecture = arch(); | ||||||
|  |  | ||||||
|  |   // Map Node's platform/arch to our binary naming | ||||||
|  |   const platformMap = { | ||||||
|  |     'darwin': 'macos', | ||||||
|  |     'linux': 'linux', | ||||||
|  |     'win32': 'windows' | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const archMap = { | ||||||
|  |     'x64': 'x64', | ||||||
|  |     'arm64': 'arm64' | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const mappedPlatform = platformMap[plat]; | ||||||
|  |   const mappedArch = archMap[architecture]; | ||||||
|  |  | ||||||
|  |   if (!mappedPlatform || !mappedArch) { | ||||||
|  |     console.error(`Error: Unsupported platform/architecture: ${plat}/${architecture}`); | ||||||
|  |     console.error('Supported platforms: Linux, macOS, Windows'); | ||||||
|  |     console.error('Supported architectures: x64, arm64'); | ||||||
|  |     process.exit(1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Construct binary name | ||||||
|  |   let binaryName = `nupst-${mappedPlatform}-${mappedArch}`; | ||||||
|  |   if (plat === 'win32') { | ||||||
|  |     binaryName += '.exe'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return binaryName; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Execute the binary | ||||||
|  |  */ | ||||||
|  | function executeBinary() { | ||||||
|  |   const binaryName = getBinaryName(); | ||||||
|  |   const binaryPath = join(__dirname, '..', 'dist', 'binaries', binaryName); | ||||||
|  |  | ||||||
|  |   // Check if binary exists | ||||||
|  |   if (!existsSync(binaryPath)) { | ||||||
|  |     console.error(`Error: Binary not found at ${binaryPath}`); | ||||||
|  |     console.error('This might happen if:'); | ||||||
|  |     console.error('1. The postinstall script failed to run'); | ||||||
|  |     console.error('2. The platform is not supported'); | ||||||
|  |     console.error('3. The package was not installed correctly'); | ||||||
|  |     console.error(''); | ||||||
|  |     console.error('Try reinstalling the package:'); | ||||||
|  |     console.error('  npm uninstall -g @serve.zone/nupst'); | ||||||
|  |     console.error('  npm install -g @serve.zone/nupst'); | ||||||
|  |     process.exit(1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Spawn the binary with all arguments passed through | ||||||
|  |   const child = spawn(binaryPath, process.argv.slice(2), { | ||||||
|  |     stdio: 'inherit', | ||||||
|  |     shell: false | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Handle child process events | ||||||
|  |   child.on('error', (err) => { | ||||||
|  |     console.error(`Error executing nupst: ${err.message}`); | ||||||
|  |     process.exit(1); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   child.on('exit', (code, signal) => { | ||||||
|  |     if (signal) { | ||||||
|  |       process.kill(process.pid, signal); | ||||||
|  |     } else { | ||||||
|  |       process.exit(code || 0); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Forward signals to child process | ||||||
|  |   const signals = ['SIGINT', 'SIGTERM', 'SIGHUP']; | ||||||
|  |   signals.forEach(signal => { | ||||||
|  |     process.on(signal, () => { | ||||||
|  |       if (!child.killed) { | ||||||
|  |         child.kill(signal); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Execute | ||||||
|  | executeBinary(); | ||||||
							
								
								
									
										25
									
								
								changelog.md
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								changelog.md
									
									
									
									
									
								
							| @@ -1,5 +1,30 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## 2025-10-23 - 5.1.2 - fix(scripts) | ||||||
|  | Add build script to package.json and include local dev tool settings | ||||||
|  |  | ||||||
|  | - Add a 'build' script to package.json (no-op placeholder) to provide an explicit build step | ||||||
|  | - Minor scripts section formatting tidy in package.json | ||||||
|  | - Add a hidden local settings file for development tooling permissions to the repository (local-only configuration) | ||||||
|  |  | ||||||
|  | ## 2025-10-23 - 5.1.1 - fix(tooling) | ||||||
|  | Add .claude/settings.local.json with local automation permissions | ||||||
|  |  | ||||||
|  | - Add .claude/settings.local.json to specify allowed permissions for local automated tasks | ||||||
|  | - Grants permissions for various developer/CI actions (deno check/lint/fmt, npm/npm pack, selective Bash commands, WebFetch to docs.deno.com and code.foss.global, and file/read/replace helpers) | ||||||
|  | - This is a developer/local tooling config only and does not change runtime code or package behavior | ||||||
|  |  | ||||||
|  | ## 2025-10-22 - 5.1.0 - feat(packaging) | ||||||
|  | Add npm packaging and installer: wrapper, postinstall downloader, publish workflow, and packaging files | ||||||
|  |  | ||||||
|  | - Add package.json (v5.0.5) and npm packaging metadata to publish @serve.zone/nupst | ||||||
|  | - Include a small Node.js wrapper (bin/nupst-wrapper.js) to execute platform-specific precompiled binaries | ||||||
|  | - Add postinstall script (scripts/install-binary.js) that downloads the correct binary for the current platform and sets executable permissions | ||||||
|  | - Add GitHub Actions workflow (.github/workflows/npm-publish.yml) to build binaries, pack and publish to npm, and create releases | ||||||
|  | - Add .npmignore to keep source, tests and dev files out of npm package; keep only runtime installer and wrapper | ||||||
|  | - Move example action script into docs (docs/example-action.sh) and remove the top-level example-action.sh | ||||||
|  | - Include generated npm package artifact (serve.zone-nupst-5.0.5.tgz) and npmextra.json | ||||||
|  |  | ||||||
| ## 2025-10-18 - 4.0.0 - BREAKING CHANGE(core): Complete migration to Deno runtime | ## 2025-10-18 - 4.0.0 - BREAKING CHANGE(core): Complete migration to Deno runtime | ||||||
|  |  | ||||||
| **MAJOR RELEASE: NUPST v4.0 is a complete rewrite powered by Deno** | **MAJOR RELEASE: NUPST v4.0 is a complete rewrite powered by Deno** | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| { | { | ||||||
|   "name": "@serve.zone/nupst", |   "name": "@serve.zone/nupst", | ||||||
|   "version": "4.3.0", |   "version": "5.1.8", | ||||||
|   "exports": "./mod.ts", |   "exports": "./mod.ts", | ||||||
|  |   "nodeModulesDir": "auto", | ||||||
|   "tasks": { |   "tasks": { | ||||||
|     "dev": "deno run --allow-all mod.ts", |     "dev": "deno run --allow-all mod.ts", | ||||||
|     "compile": "deno task compile:all", |     "compile": "deno task compile:all", | ||||||
|   | |||||||
							
								
								
									
										104
									
								
								install.sh
									
									
									
									
									
								
							
							
						
						
									
										104
									
								
								install.sh
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| #!/bin/bash | #!/bin/bash | ||||||
|  |  | ||||||
| # NUPST Installer Script (v4.0+) | # NUPST Installer Script (v5.0+) | ||||||
| # Downloads and installs pre-compiled NUPST binary from Gitea releases | # Downloads and installs pre-compiled NUPST binary from Gitea releases | ||||||
| # | # | ||||||
| # Usage: | # Usage: | ||||||
| @@ -8,7 +8,7 @@ | |||||||
| #     curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash | #     curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash | ||||||
| # | # | ||||||
| #   With version specification: | #   With version specification: | ||||||
| #     curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash -s -- --version v4.0.0 | #     curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash -s -- --version v5.0.0 | ||||||
| # | # | ||||||
| # Options: | # Options: | ||||||
| #   -h, --help             Show this help message | #   -h, --help             Show this help message | ||||||
| @@ -48,14 +48,14 @@ while [[ $# -gt 0 ]]; do | |||||||
| done | done | ||||||
|  |  | ||||||
| if [ $SHOW_HELP -eq 1 ]; then | if [ $SHOW_HELP -eq 1 ]; then | ||||||
|   echo "NUPST Installer Script (v4.0+)" |   echo "NUPST Installer Script (v5.0+)" | ||||||
|   echo "Downloads and installs pre-compiled NUPST binary" |   echo "Downloads and installs pre-compiled NUPST binary" | ||||||
|   echo "" |   echo "" | ||||||
|   echo "Usage: $0 [options]" |   echo "Usage: $0 [options]" | ||||||
|   echo "" |   echo "" | ||||||
|   echo "Options:" |   echo "Options:" | ||||||
|   echo "  -h, --help             Show this help message" |   echo "  -h, --help             Show this help message" | ||||||
|   echo "  --version VERSION      Install specific version (e.g., v4.0.0)" |   echo "  --version VERSION      Install specific version (e.g., v5.0.0)" | ||||||
|   echo "  --install-dir DIR      Installation directory (default: /opt/nupst)" |   echo "  --install-dir DIR      Installation directory (default: /opt/nupst)" | ||||||
|   echo "" |   echo "" | ||||||
|   echo "Examples:" |   echo "Examples:" | ||||||
| @@ -63,7 +63,7 @@ if [ $SHOW_HELP -eq 1 ]; then | |||||||
|   echo "  curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash" |   echo "  curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash" | ||||||
|   echo "" |   echo "" | ||||||
|   echo "  # Install specific version" |   echo "  # Install specific version" | ||||||
|   echo "  curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash -s -- --version v4.0.0" |   echo "  curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash -s -- --version v5.0.0" | ||||||
|   exit 0 |   exit 0 | ||||||
| fi | fi | ||||||
|  |  | ||||||
| @@ -145,7 +145,7 @@ get_latest_version() { | |||||||
|  |  | ||||||
| # Main installation process | # Main installation process | ||||||
| echo "================================================" | echo "================================================" | ||||||
| echo "  NUPST Installation Script (v4.0+)" | echo "  NUPST Installation Script (v5.0+)" | ||||||
| echo "================================================" | echo "================================================" | ||||||
| echo "" | echo "" | ||||||
|  |  | ||||||
| @@ -169,51 +169,26 @@ DOWNLOAD_URL="${GITEA_BASE_URL}/${GITEA_REPO}/releases/download/${VERSION}/${BIN | |||||||
| echo "Download URL: $DOWNLOAD_URL" | echo "Download URL: $DOWNLOAD_URL" | ||||||
| echo "" | echo "" | ||||||
|  |  | ||||||
| # Check if installation directory exists | # Check if service is running and stop it | ||||||
| SERVICE_WAS_RUNNING=0 | SERVICE_WAS_RUNNING=0 | ||||||
| OLD_NODE_INSTALL=0 | if systemctl is-enabled --quiet nupst 2>/dev/null || systemctl is-active --quiet nupst 2>/dev/null; then | ||||||
|  |   SERVICE_WAS_RUNNING=1 | ||||||
| if [ -d "$INSTALL_DIR" ]; then |   if systemctl is-active --quiet nupst 2>/dev/null; then | ||||||
|   # Check if this is an old Node.js-based installation |     echo "Stopping NUPST service..." | ||||||
|   if [ -f "$INSTALL_DIR/package.json" ] || [ -d "$INSTALL_DIR/node_modules" ]; then |     systemctl stop nupst | ||||||
|     OLD_NODE_INSTALL=1 |  | ||||||
|     echo "Detected old Node.js-based NUPST installation (v3.x or earlier)" |  | ||||||
|     echo "This installer will migrate to the new Deno-based binary version (v4.0+)" |  | ||||||
|     echo "" |  | ||||||
|   fi |   fi | ||||||
|  |  | ||||||
|   echo "Updating existing installation at $INSTALL_DIR..." |  | ||||||
|  |  | ||||||
|   # Check if service exists (enabled or running) and stop it if active |  | ||||||
|   if systemctl is-enabled --quiet nupst 2>/dev/null || systemctl is-active --quiet nupst 2>/dev/null; then |  | ||||||
|     SERVICE_WAS_RUNNING=1 |  | ||||||
|     if systemctl is-active --quiet nupst 2>/dev/null; then |  | ||||||
|       echo "Stopping NUPST service..." |  | ||||||
|       systemctl stop nupst |  | ||||||
|     else |  | ||||||
|       echo "Service is installed but not currently running (will be updated)..." |  | ||||||
|     fi |  | ||||||
|   fi |  | ||||||
|  |  | ||||||
|   # Clean up old Node.js installation files |  | ||||||
|   if [ $OLD_NODE_INSTALL -eq 1 ]; then |  | ||||||
|     echo "Cleaning up old Node.js installation files..." |  | ||||||
|     rm -rf "$INSTALL_DIR/node_modules" 2>/dev/null || true |  | ||||||
|     rm -rf "$INSTALL_DIR/vendor" 2>/dev/null || true |  | ||||||
|     rm -rf "$INSTALL_DIR/dist_ts" 2>/dev/null || true |  | ||||||
|     rm -f "$INSTALL_DIR/package.json" 2>/dev/null || true |  | ||||||
|     rm -f "$INSTALL_DIR/package-lock.json" 2>/dev/null || true |  | ||||||
|     rm -f "$INSTALL_DIR/pnpm-lock.yaml" 2>/dev/null || true |  | ||||||
|     rm -f "$INSTALL_DIR/tsconfig.json" 2>/dev/null || true |  | ||||||
|     rm -f "$INSTALL_DIR/setup.sh" 2>/dev/null || true |  | ||||||
|     rm -rf "$INSTALL_DIR/bin" 2>/dev/null || true |  | ||||||
|     echo "Old installation files removed." |  | ||||||
|   fi |  | ||||||
| else |  | ||||||
|   echo "Creating installation directory: $INSTALL_DIR" |  | ||||||
|   mkdir -p "$INSTALL_DIR" |  | ||||||
| fi | fi | ||||||
|  |  | ||||||
|  | # Clean installation directory - ensure only binary exists | ||||||
|  | if [ -d "$INSTALL_DIR" ]; then | ||||||
|  |   echo "Cleaning installation directory: $INSTALL_DIR" | ||||||
|  |   rm -rf "$INSTALL_DIR" | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | # Create fresh installation directory | ||||||
|  | echo "Creating installation directory: $INSTALL_DIR" | ||||||
|  | mkdir -p "$INSTALL_DIR" | ||||||
|  |  | ||||||
| # Download binary | # Download binary | ||||||
| echo "Downloading NUPST binary..." | echo "Downloading NUPST binary..." | ||||||
| TEMP_FILE="$INSTALL_DIR/nupst.download" | TEMP_FILE="$INSTALL_DIR/nupst.download" | ||||||
| @@ -241,9 +216,20 @@ fi | |||||||
| BINARY_PATH="$INSTALL_DIR/nupst" | BINARY_PATH="$INSTALL_DIR/nupst" | ||||||
| mv "$TEMP_FILE" "$BINARY_PATH" | mv "$TEMP_FILE" "$BINARY_PATH" | ||||||
|  |  | ||||||
|  | if [ $? -ne 0 ] || [ ! -f "$BINARY_PATH" ]; then | ||||||
|  |   echo "Error: Failed to move binary to $BINARY_PATH" | ||||||
|  |   rm -f "$TEMP_FILE" 2>/dev/null | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
| # Make executable | # Make executable | ||||||
| chmod +x "$BINARY_PATH" | chmod +x "$BINARY_PATH" | ||||||
|  |  | ||||||
|  | if [ $? -ne 0 ]; then | ||||||
|  |   echo "Error: Failed to make binary executable" | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
| echo "Binary installed successfully to: $BINARY_PATH" | echo "Binary installed successfully to: $BINARY_PATH" | ||||||
| echo "" | echo "" | ||||||
|  |  | ||||||
| @@ -260,18 +246,10 @@ echo "Symlink created: $BIN_DIR/nupst -> $BINARY_PATH" | |||||||
|  |  | ||||||
| echo "" | echo "" | ||||||
|  |  | ||||||
| # Update systemd service file if migrating from v3 |  | ||||||
| if [ $SERVICE_WAS_RUNNING -eq 1 ] && [ $OLD_NODE_INSTALL -eq 1 ]; then |  | ||||||
|   echo "Updating systemd service file for v4..." |  | ||||||
|   $BINARY_PATH service enable > /dev/null 2>&1 |  | ||||||
|   echo "Service file updated." |  | ||||||
|   echo "" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Restart service if it was running before update | # Restart service if it was running before update | ||||||
| if [ $SERVICE_WAS_RUNNING -eq 1 ]; then | if [ $SERVICE_WAS_RUNNING -eq 1 ]; then | ||||||
|   echo "Restarting NUPST service..." |   echo "Restarting NUPST service..." | ||||||
|   systemctl start nupst |   systemctl restart nupst | ||||||
|   echo "Service restarted successfully." |   echo "Service restarted successfully." | ||||||
|   echo "" |   echo "" | ||||||
| fi | fi | ||||||
| @@ -280,20 +258,6 @@ echo "================================================" | |||||||
| echo "  NUPST Installation Complete!" | echo "  NUPST Installation Complete!" | ||||||
| echo "================================================" | echo "================================================" | ||||||
| echo "" | echo "" | ||||||
|  |  | ||||||
| if [ $OLD_NODE_INSTALL -eq 1 ]; then |  | ||||||
|   echo "Migration from v3.x to v4.0 successful!" |  | ||||||
|   echo "" |  | ||||||
|   echo "What changed:" |  | ||||||
|   echo "  • Node.js runtime removed (now a self-contained binary)" |  | ||||||
|   echo "  • Faster startup and lower memory usage" |  | ||||||
|   echo "  • CLI commands now use subcommand structure" |  | ||||||
|   echo "    (old commands still work with deprecation warnings)" |  | ||||||
|   echo "" |  | ||||||
|   echo "See readme for migration details: https://code.foss.global/serve.zone/nupst#migration-from-v3x" |  | ||||||
|   echo "" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| echo "Installation details:" | echo "Installation details:" | ||||||
| echo "  Binary location: $BINARY_PATH" | echo "  Binary location: $BINARY_PATH" | ||||||
| echo "  Symlink location: $BIN_DIR/nupst" | echo "  Symlink location: $BIN_DIR/nupst" | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								npmextra.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								npmextra.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | {} | ||||||
							
								
								
									
										64
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | |||||||
|  | { | ||||||
|  |   "name": "@serve.zone/nupst", | ||||||
|  |   "version": "5.1.8", | ||||||
|  |   "description": "Network UPS Shutdown Tool - Monitor SNMP-enabled UPS devices and orchestrate graceful system shutdowns during power emergencies", | ||||||
|  |   "keywords": [ | ||||||
|  |     "ups", | ||||||
|  |     "snmp", | ||||||
|  |     "power", | ||||||
|  |     "shutdown", | ||||||
|  |     "monitoring", | ||||||
|  |     "cyberpower", | ||||||
|  |     "apc", | ||||||
|  |     "eaton", | ||||||
|  |     "tripplite", | ||||||
|  |     "liebert", | ||||||
|  |     "vertiv", | ||||||
|  |     "battery", | ||||||
|  |     "backup" | ||||||
|  |   ], | ||||||
|  |   "homepage": "https://code.foss.global/serve.zone/nupst", | ||||||
|  |   "bugs": { | ||||||
|  |     "url": "https://code.foss.global/serve.zone/nupst/issues" | ||||||
|  |   }, | ||||||
|  |   "repository": { | ||||||
|  |     "type": "git", | ||||||
|  |     "url": "git+https://code.foss.global/serve.zone/nupst.git" | ||||||
|  |   }, | ||||||
|  |   "author": "Serve Zone", | ||||||
|  |   "license": "MIT", | ||||||
|  |   "type": "module", | ||||||
|  |   "bin": { | ||||||
|  |     "nupst": "./bin/nupst-wrapper.js" | ||||||
|  |   }, | ||||||
|  |   "scripts": { | ||||||
|  |     "postinstall": "node scripts/install-binary.js", | ||||||
|  |     "prepublishOnly": "echo 'Publishing NUPST binaries to npm...'", | ||||||
|  |     "test": "echo 'Tests are run with Deno: deno task test'", | ||||||
|  |     "build": "echo 'no build needed'" | ||||||
|  |   }, | ||||||
|  |   "files": [ | ||||||
|  |     "bin/", | ||||||
|  |     "scripts/install-binary.js", | ||||||
|  |     "readme.md", | ||||||
|  |     "license", | ||||||
|  |     "changelog.md" | ||||||
|  |   ], | ||||||
|  |   "engines": { | ||||||
|  |     "node": ">=14.0.0" | ||||||
|  |   }, | ||||||
|  |   "os": [ | ||||||
|  |     "darwin", | ||||||
|  |     "linux", | ||||||
|  |     "win32" | ||||||
|  |   ], | ||||||
|  |   "cpu": [ | ||||||
|  |     "x64", | ||||||
|  |     "arm64" | ||||||
|  |   ], | ||||||
|  |   "publishConfig": { | ||||||
|  |     "access": "public", | ||||||
|  |     "registry": "https://registry.npmjs.org/" | ||||||
|  |   }, | ||||||
|  |   "packageManager": "pnpm@10.18.1+sha512.77a884a165cbba2d8d1c19e3b4880eee6d2fcabd0d879121e282196b80042351d5eb3ca0935fa599da1dc51265cc68816ad2bddd2a2de5ea9fdf92adbec7cd34" | ||||||
|  | } | ||||||
							
								
								
									
										231
									
								
								scripts/install-binary.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										231
									
								
								scripts/install-binary.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,231 @@ | |||||||
|  | #!/usr/bin/env node | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * NUPST npm postinstall script | ||||||
|  |  * Downloads the appropriate binary for the current platform from GitHub releases | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | import { platform, arch } from 'os'; | ||||||
|  | import { existsSync, mkdirSync, writeFileSync, chmodSync, unlinkSync } from 'fs'; | ||||||
|  | import { join, dirname } from 'path'; | ||||||
|  | import { fileURLToPath } from 'url'; | ||||||
|  | import https from 'https'; | ||||||
|  | import { pipeline } from 'stream'; | ||||||
|  | import { promisify } from 'util'; | ||||||
|  | import { createWriteStream } from 'fs'; | ||||||
|  |  | ||||||
|  | const __filename = fileURLToPath(import.meta.url); | ||||||
|  | const __dirname = dirname(__filename); | ||||||
|  | const streamPipeline = promisify(pipeline); | ||||||
|  |  | ||||||
|  | // Configuration | ||||||
|  | const REPO_BASE = 'https://code.foss.global/serve.zone/nupst'; | ||||||
|  | const VERSION = process.env.npm_package_version || '5.0.5'; | ||||||
|  |  | ||||||
|  | function getBinaryInfo() { | ||||||
|  |   const plat = platform(); | ||||||
|  |   const architecture = arch(); | ||||||
|  |  | ||||||
|  |   const platformMap = { | ||||||
|  |     'darwin': 'macos', | ||||||
|  |     'linux': 'linux', | ||||||
|  |     'win32': 'windows' | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const archMap = { | ||||||
|  |     'x64': 'x64', | ||||||
|  |     'arm64': 'arm64' | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const mappedPlatform = platformMap[plat]; | ||||||
|  |   const mappedArch = archMap[architecture]; | ||||||
|  |  | ||||||
|  |   if (!mappedPlatform || !mappedArch) { | ||||||
|  |     return { supported: false, platform: plat, arch: architecture }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   let binaryName = `nupst-${mappedPlatform}-${mappedArch}`; | ||||||
|  |   if (plat === 'win32') { | ||||||
|  |     binaryName += '.exe'; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     supported: true, | ||||||
|  |     platform: mappedPlatform, | ||||||
|  |     arch: mappedArch, | ||||||
|  |     binaryName, | ||||||
|  |     originalPlatform: plat | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function downloadFile(url, destination) { | ||||||
|  |   return new Promise((resolve, reject) => { | ||||||
|  |     console.log(`Downloading from: ${url}`); | ||||||
|  |  | ||||||
|  |     // Follow redirects | ||||||
|  |     const download = (url, redirectCount = 0) => { | ||||||
|  |       if (redirectCount > 5) { | ||||||
|  |         reject(new Error('Too many redirects')); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       https.get(url, (response) => { | ||||||
|  |         if (response.statusCode === 301 || response.statusCode === 302) { | ||||||
|  |           console.log(`Following redirect to: ${response.headers.location}`); | ||||||
|  |           download(response.headers.location, redirectCount + 1); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (response.statusCode !== 200) { | ||||||
|  |           reject(new Error(`Failed to download: ${response.statusCode} ${response.statusMessage}`)); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const totalSize = parseInt(response.headers['content-length'], 10); | ||||||
|  |         let downloadedSize = 0; | ||||||
|  |         let lastProgress = 0; | ||||||
|  |  | ||||||
|  |         response.on('data', (chunk) => { | ||||||
|  |           downloadedSize += chunk.length; | ||||||
|  |           const progress = Math.round((downloadedSize / totalSize) * 100); | ||||||
|  |  | ||||||
|  |           // Only log every 10% to reduce noise | ||||||
|  |           if (progress >= lastProgress + 10) { | ||||||
|  |             console.log(`Download progress: ${progress}%`); | ||||||
|  |             lastProgress = progress; | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         const file = createWriteStream(destination); | ||||||
|  |  | ||||||
|  |         pipeline(response, file, (err) => { | ||||||
|  |           if (err) { | ||||||
|  |             reject(err); | ||||||
|  |           } else { | ||||||
|  |             console.log('Download complete!'); | ||||||
|  |             resolve(); | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |       }).on('error', reject); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     download(url); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function main() { | ||||||
|  |   console.log('==========================================='); | ||||||
|  |   console.log('  NUPST - Binary Installation'); | ||||||
|  |   console.log('==========================================='); | ||||||
|  |   console.log(''); | ||||||
|  |  | ||||||
|  |   const binaryInfo = getBinaryInfo(); | ||||||
|  |  | ||||||
|  |   if (!binaryInfo.supported) { | ||||||
|  |     console.error(`❌ Error: Unsupported platform/architecture: ${binaryInfo.platform}/${binaryInfo.arch}`); | ||||||
|  |     console.error(''); | ||||||
|  |     console.error('Supported platforms:'); | ||||||
|  |     console.error('  • Linux (x64, arm64)'); | ||||||
|  |     console.error('  • macOS (x64, arm64)'); | ||||||
|  |     console.error('  • Windows (x64)'); | ||||||
|  |     console.error(''); | ||||||
|  |     console.error('If you believe your platform should be supported, please file an issue:'); | ||||||
|  |     console.error('  https://code.foss.global/serve.zone/nupst/issues'); | ||||||
|  |     process.exit(1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   console.log(`Platform: ${binaryInfo.platform} (${binaryInfo.originalPlatform})`); | ||||||
|  |   console.log(`Architecture: ${binaryInfo.arch}`); | ||||||
|  |   console.log(`Binary: ${binaryInfo.binaryName}`); | ||||||
|  |   console.log(`Version: ${VERSION}`); | ||||||
|  |   console.log(''); | ||||||
|  |  | ||||||
|  |   // Create dist/binaries directory if it doesn't exist | ||||||
|  |   const binariesDir = join(__dirname, '..', 'dist', 'binaries'); | ||||||
|  |   if (!existsSync(binariesDir)) { | ||||||
|  |     console.log('Creating binaries directory...'); | ||||||
|  |     mkdirSync(binariesDir, { recursive: true }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const binaryPath = join(binariesDir, binaryInfo.binaryName); | ||||||
|  |  | ||||||
|  |   // Check if binary already exists and skip download | ||||||
|  |   if (existsSync(binaryPath)) { | ||||||
|  |     console.log('✓ Binary already exists, skipping download'); | ||||||
|  |   } else { | ||||||
|  |     // Construct download URL | ||||||
|  |     // Try release URL first, fall back to raw branch if needed | ||||||
|  |     const releaseUrl = `${REPO_BASE}/releases/download/v${VERSION}/${binaryInfo.binaryName}`; | ||||||
|  |     const fallbackUrl = `${REPO_BASE}/raw/branch/main/dist/binaries/${binaryInfo.binaryName}`; | ||||||
|  |  | ||||||
|  |     console.log('Downloading platform-specific binary...'); | ||||||
|  |     console.log('This may take a moment depending on your connection speed.'); | ||||||
|  |     console.log(''); | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       // Try downloading from release | ||||||
|  |       await downloadFile(releaseUrl, binaryPath); | ||||||
|  |     } catch (err) { | ||||||
|  |       console.log(`Release download failed: ${err.message}`); | ||||||
|  |       console.log('Trying fallback URL...'); | ||||||
|  |  | ||||||
|  |       try { | ||||||
|  |         // Try fallback URL | ||||||
|  |         await downloadFile(fallbackUrl, binaryPath); | ||||||
|  |       } catch (fallbackErr) { | ||||||
|  |         console.error(`❌ Error: Failed to download binary`); | ||||||
|  |         console.error(`  Primary URL: ${releaseUrl}`); | ||||||
|  |         console.error(`  Fallback URL: ${fallbackUrl}`); | ||||||
|  |         console.error(''); | ||||||
|  |         console.error('This might be because:'); | ||||||
|  |         console.error('1. The release has not been created yet'); | ||||||
|  |         console.error('2. Network connectivity issues'); | ||||||
|  |         console.error('3. The version specified does not exist'); | ||||||
|  |         console.error(''); | ||||||
|  |         console.error('You can try:'); | ||||||
|  |         console.error('1. Installing from source: https://code.foss.global/serve.zone/nupst'); | ||||||
|  |         console.error('2. Downloading the binary manually from the releases page'); | ||||||
|  |         console.error('3. Using the install script: curl -sSL https://code.foss.global/serve.zone/nupst/raw/branch/main/install.sh | sudo bash'); | ||||||
|  |  | ||||||
|  |         // Clean up partial download | ||||||
|  |         if (existsSync(binaryPath)) { | ||||||
|  |           unlinkSync(binaryPath); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         process.exit(1); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     console.log(`✓ Binary downloaded successfully`); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // On Unix-like systems, ensure the binary is executable | ||||||
|  |   if (binaryInfo.originalPlatform !== 'win32') { | ||||||
|  |     try { | ||||||
|  |       console.log('Setting executable permissions...'); | ||||||
|  |       chmodSync(binaryPath, 0o755); | ||||||
|  |       console.log('✓ Binary permissions updated'); | ||||||
|  |     } catch (err) { | ||||||
|  |       console.error(`⚠️  Warning: Could not set executable permissions: ${err.message}`); | ||||||
|  |       console.error('   You may need to manually run:'); | ||||||
|  |       console.error(`   chmod +x ${binaryPath}`); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   console.log(''); | ||||||
|  |   console.log('✅ NUPST installation completed successfully!'); | ||||||
|  |   console.log(''); | ||||||
|  |   console.log('You can now use NUPST by running:'); | ||||||
|  |   console.log('  nupst --help'); | ||||||
|  |   console.log(''); | ||||||
|  |   console.log('For initial setup, run:'); | ||||||
|  |   console.log('  sudo nupst ups add'); | ||||||
|  |   console.log(''); | ||||||
|  |   console.log('==========================================='); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Run the installation | ||||||
|  | main().catch(err => { | ||||||
|  |   console.error(`❌ Installation failed: ${err.message}`); | ||||||
|  |   process.exit(1); | ||||||
|  | }); | ||||||
							
								
								
									
										
											BIN
										
									
								
								serve.zone-nupst-5.0.5.tgz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								serve.zone-nupst-5.0.5.tgz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,10 +0,0 @@ | |||||||
| /** |  | ||||||
|  * commitinfo - reads version from deno.json |  | ||||||
|  */ |  | ||||||
| import denoConfig from '../deno.json' with { type: 'json' }; |  | ||||||
|  |  | ||||||
| export const commitinfo = { |  | ||||||
|   name: denoConfig.name, |  | ||||||
|   version: denoConfig.version, |  | ||||||
|   description: 'Network UPS Shutdown Tool (https://nupst.serve.zone)', |  | ||||||
| }; |  | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| import * as path from 'node:path'; | import * as path from 'node:path'; | ||||||
| import * as fs from 'node:fs'; | import * as fs from 'node:fs'; | ||||||
|  | import process from 'node:process'; | ||||||
| import { exec } from 'node:child_process'; | import { exec } from 'node:child_process'; | ||||||
| import { promisify } from 'node:util'; | import { promisify } from 'node:util'; | ||||||
| import { Action, type IActionConfig, type IActionContext } from './base-action.ts'; | import { Action, type IActionConfig, type IActionContext } from './base-action.ts'; | ||||||
|   | |||||||
							
								
								
									
										162
									
								
								ts/cli.ts
									
									
									
									
									
								
							
							
						
						
									
										162
									
								
								ts/cli.ts
									
									
									
									
									
								
							| @@ -127,8 +127,7 @@ export class NupstCli { | |||||||
|           break; |           break; | ||||||
|         } |         } | ||||||
|         case 'remove': |         case 'remove': | ||||||
|         case 'rm': // Alias |         case 'rm': { | ||||||
|         case 'delete': { // Backward compatibility |  | ||||||
|           const upsIdToRemove = subcommandArgs[0]; |           const upsIdToRemove = subcommandArgs[0]; | ||||||
|           if (!upsIdToRemove) { |           if (!upsIdToRemove) { | ||||||
|             logger.error('UPS ID is required for remove command'); |             logger.error('UPS ID is required for remove command'); | ||||||
| @@ -172,8 +171,7 @@ export class NupstCli { | |||||||
|           break; |           break; | ||||||
|         } |         } | ||||||
|         case 'remove': |         case 'remove': | ||||||
|         case 'rm': // Alias |         case 'rm': { | ||||||
|         case 'delete': { // Backward compatibility |  | ||||||
|           const groupIdToRemove = subcommandArgs[0]; |           const groupIdToRemove = subcommandArgs[0]; | ||||||
|           if (!groupIdToRemove) { |           if (!groupIdToRemove) { | ||||||
|             logger.error('Group ID is required for remove command'); |             logger.error('Group ID is required for remove command'); | ||||||
| @@ -206,8 +204,7 @@ export class NupstCli { | |||||||
|           break; |           break; | ||||||
|         } |         } | ||||||
|         case 'remove': |         case 'remove': | ||||||
|         case 'rm': // Alias |         case 'rm': { | ||||||
|         case 'delete': { // Backward compatibility |  | ||||||
|           const upsId = subcommandArgs[0]; |           const upsId = subcommandArgs[0]; | ||||||
|           const actionIndex = subcommandArgs[1]; |           const actionIndex = subcommandArgs[1]; | ||||||
|           await actionHandler.remove(upsId, actionIndex); |           await actionHandler.remove(upsId, actionIndex); | ||||||
| @@ -226,6 +223,24 @@ export class NupstCli { | |||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // Handle feature subcommands | ||||||
|  |     if (command === 'feature') { | ||||||
|  |       const subcommand = commandArgs[0]; | ||||||
|  |       const featureHandler = this.nupst.getFeatureHandler(); | ||||||
|  |  | ||||||
|  |       switch (subcommand) { | ||||||
|  |         case 'httpServer': | ||||||
|  |         case 'http-server': | ||||||
|  |         case 'http': | ||||||
|  |           await featureHandler.configureHttpServer(); | ||||||
|  |           break; | ||||||
|  |         default: | ||||||
|  |           this.showFeatureHelp(); | ||||||
|  |           break; | ||||||
|  |       } | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     // Handle config subcommand |     // Handle config subcommand | ||||||
|     if (command === 'config') { |     if (command === 'config') { | ||||||
|       const subcommand = commandArgs[0] || 'show'; |       const subcommand = commandArgs[0] || 'show'; | ||||||
| @@ -242,72 +257,8 @@ export class NupstCli { | |||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Handle top-level commands and backward compatibility |     // Handle top-level commands | ||||||
|     switch (command) { |     switch (command) { | ||||||
|       // Backward compatibility - old UPS commands |  | ||||||
|       case 'add': |  | ||||||
|         logger.log("Note: 'nupst add' is deprecated. Use 'nupst ups add' instead."); |  | ||||||
|         await upsHandler.add(); |  | ||||||
|         break; |  | ||||||
|       case 'edit': |  | ||||||
|         logger.log("Note: 'nupst edit' is deprecated. Use 'nupst ups edit' instead."); |  | ||||||
|         await upsHandler.edit(commandArgs[0]); |  | ||||||
|         break; |  | ||||||
|       case 'delete': |  | ||||||
|         logger.log("Note: 'nupst delete' is deprecated. Use 'nupst ups remove' instead."); |  | ||||||
|         if (!commandArgs[0]) { |  | ||||||
|           logger.error('UPS ID is required for delete command'); |  | ||||||
|           this.showHelp(); |  | ||||||
|           return; |  | ||||||
|         } |  | ||||||
|         await upsHandler.remove(commandArgs[0]); |  | ||||||
|         break; |  | ||||||
|       case 'list': |  | ||||||
|         logger.log("Note: 'nupst list' is deprecated. Use 'nupst ups list' instead."); |  | ||||||
|         await upsHandler.list(); |  | ||||||
|         break; |  | ||||||
|       case 'test': |  | ||||||
|         logger.log("Note: 'nupst test' is deprecated. Use 'nupst ups test' instead."); |  | ||||||
|         await upsHandler.test(debugMode); |  | ||||||
|         break; |  | ||||||
|       case 'setup': |  | ||||||
|         logger.log("Note: 'nupst setup' is deprecated. Use 'nupst ups edit' instead."); |  | ||||||
|         await upsHandler.edit(undefined); |  | ||||||
|         break; |  | ||||||
|  |  | ||||||
|       // Backward compatibility - old service commands |  | ||||||
|       case 'enable': |  | ||||||
|         logger.log("Note: 'nupst enable' is deprecated. Use 'nupst service enable' instead."); |  | ||||||
|         await serviceHandler.enable(); |  | ||||||
|         break; |  | ||||||
|       case 'disable': |  | ||||||
|         logger.log("Note: 'nupst disable' is deprecated. Use 'nupst service disable' instead."); |  | ||||||
|         await serviceHandler.disable(); |  | ||||||
|         break; |  | ||||||
|       case 'start': |  | ||||||
|         logger.log("Note: 'nupst start' is deprecated. Use 'nupst service start' instead."); |  | ||||||
|         await serviceHandler.start(); |  | ||||||
|         break; |  | ||||||
|       case 'stop': |  | ||||||
|         logger.log("Note: 'nupst stop' is deprecated. Use 'nupst service stop' instead."); |  | ||||||
|         await serviceHandler.stop(); |  | ||||||
|         break; |  | ||||||
|       case 'status': |  | ||||||
|         logger.log("Note: 'nupst status' is deprecated. Use 'nupst service status' instead."); |  | ||||||
|         await serviceHandler.status(); |  | ||||||
|         break; |  | ||||||
|       case 'logs': |  | ||||||
|         logger.log("Note: 'nupst logs' is deprecated. Use 'nupst service logs' instead."); |  | ||||||
|         await serviceHandler.logs(); |  | ||||||
|         break; |  | ||||||
|       case 'daemon-start': |  | ||||||
|         logger.log( |  | ||||||
|           "Note: 'nupst daemon-start' is deprecated. Use 'nupst service start-daemon' instead.", |  | ||||||
|         ); |  | ||||||
|         await serviceHandler.daemonStart(debugMode); |  | ||||||
|         break; |  | ||||||
|  |  | ||||||
|       // Top-level commands (no changes) |  | ||||||
|       case 'update': |       case 'update': | ||||||
|         await serviceHandler.update(); |         await serviceHandler.update(); | ||||||
|         break; |         break; | ||||||
| @@ -361,6 +312,26 @@ export class NupstCli { | |||||||
|           `  ${theme.path('/etc/nupst/config.json')}`, |           `  ${theme.path('/etc/nupst/config.json')}`, | ||||||
|         ], 60, 'info'); |         ], 60, 'info'); | ||||||
|  |  | ||||||
|  |         // HTTP Server Status (if configured) | ||||||
|  |         if (config.httpServer) { | ||||||
|  |           const serverStatus = config.httpServer.enabled | ||||||
|  |             ? theme.success('Enabled') | ||||||
|  |             : theme.dim('Disabled'); | ||||||
|  |  | ||||||
|  |           logger.log(''); | ||||||
|  |           logger.logBox('HTTP Server', [ | ||||||
|  |             `Status: ${serverStatus}`, | ||||||
|  |             ...(config.httpServer.enabled ? [ | ||||||
|  |               `Port: ${theme.highlight(String(config.httpServer.port))}`, | ||||||
|  |               `Path: ${theme.highlight(config.httpServer.path)}`, | ||||||
|  |               `Auth Token: ${theme.dim('***' + config.httpServer.authToken.slice(-4))}`, | ||||||
|  |               '', | ||||||
|  |               theme.dim('Usage:'), | ||||||
|  |               `  curl -H "Authorization: Bearer TOKEN" http://localhost:${config.httpServer.port}${config.httpServer.path}`, | ||||||
|  |             ] : []), | ||||||
|  |           ], 70, config.httpServer.enabled ? 'success' : 'default'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         // UPS Devices Table |         // UPS Devices Table | ||||||
|         if (config.upsDevices.length > 0) { |         if (config.upsDevices.length > 0) { | ||||||
|           const upsRows = config.upsDevices.map((ups) => ({ |           const upsRows = config.upsDevices.map((ups) => ({ | ||||||
| @@ -533,6 +504,7 @@ export class NupstCli { | |||||||
|     this.printCommand('ups <subcommand>', 'Manage UPS devices'); |     this.printCommand('ups <subcommand>', 'Manage UPS devices'); | ||||||
|     this.printCommand('group <subcommand>', 'Manage UPS groups'); |     this.printCommand('group <subcommand>', 'Manage UPS groups'); | ||||||
|     this.printCommand('action <subcommand>', 'Manage UPS actions'); |     this.printCommand('action <subcommand>', 'Manage UPS actions'); | ||||||
|  |     this.printCommand('feature <subcommand>', 'Manage optional features'); | ||||||
|     this.printCommand('config [show]', 'Display current configuration'); |     this.printCommand('config [show]', 'Display current configuration'); | ||||||
|     this.printCommand('update', 'Update NUPST from repository', theme.dim('(requires root)')); |     this.printCommand('update', 'Update NUPST from repository', theme.dim('(requires root)')); | ||||||
|     this.printCommand('uninstall', 'Completely remove NUPST', theme.dim('(requires root)')); |     this.printCommand('uninstall', 'Completely remove NUPST', theme.dim('(requires root)')); | ||||||
| @@ -571,9 +543,14 @@ export class NupstCli { | |||||||
|  |  | ||||||
|     // Action subcommands |     // Action subcommands | ||||||
|     logger.log(theme.info('Action Subcommands:')); |     logger.log(theme.info('Action Subcommands:')); | ||||||
|     this.printCommand('nupst action add <ups-id>', 'Add a new action to a UPS'); |     this.printCommand('nupst action add <target-id>', 'Add a new action to a UPS or group'); | ||||||
|     this.printCommand('nupst action remove <ups-id> <index>', 'Remove an action by index'); |     this.printCommand('nupst action remove <target-id> <index>', 'Remove an action by index'); | ||||||
|     this.printCommand('nupst action list [ups-id]', 'List all actions (optionally for specific UPS)'); |     this.printCommand('nupst action list [target-id]', 'List all actions (optionally for specific target)'); | ||||||
|  |     console.log(''); | ||||||
|  |  | ||||||
|  |     // Feature subcommands | ||||||
|  |     logger.log(theme.info('Feature Subcommands:')); | ||||||
|  |     this.printCommand('nupst feature httpServer', 'Configure HTTP server for JSON status export'); | ||||||
|     console.log(''); |     console.log(''); | ||||||
|  |  | ||||||
|     // Options |     // Options | ||||||
| @@ -589,11 +566,6 @@ export class NupstCli { | |||||||
|     logger.dim('  nupst group list         # Show all configured groups'); |     logger.dim('  nupst group list         # Show all configured groups'); | ||||||
|     logger.dim('  nupst config             # Display current configuration'); |     logger.dim('  nupst config             # Display current configuration'); | ||||||
|     console.log(''); |     console.log(''); | ||||||
|  |  | ||||||
|     // Note about deprecated commands |  | ||||||
|     logger.warn('Note: Old command format (e.g., \'nupst add\') still works but is deprecated.'); |  | ||||||
|     logger.dim('      Use the new format (e.g., \'nupst ups add\') going forward.'); |  | ||||||
|     console.log(''); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -691,18 +663,34 @@ Usage: | |||||||
|   nupst action <subcommand> [arguments] |   nupst action <subcommand> [arguments] | ||||||
|  |  | ||||||
| Subcommands: | Subcommands: | ||||||
|   add <ups-id>                    - Add a new action to a UPS interactively |   add <ups-id|group-id>                   - Add a new action to a UPS or group interactively | ||||||
|   remove <ups-id> <index>         - Remove an action by index (alias: rm, delete) |   remove <ups-id|group-id> <index>        - Remove an action by index (alias: rm) | ||||||
|   list [ups-id]                   - List all actions (optionally for specific UPS) (alias: ls) |   list [ups-id|group-id]                  - List all actions (optionally for specific target) (alias: ls) | ||||||
|  |  | ||||||
| Options: | Options: | ||||||
|   --debug, -d                     - Enable debug mode for detailed logging |   --debug, -d                             - Enable debug mode for detailed logging | ||||||
|  |  | ||||||
| Examples: | Examples: | ||||||
|   nupst action list               - List actions for all UPS devices |   nupst action list                       - List actions for all UPS devices and groups | ||||||
|   nupst action list default       - List actions for UPS with ID 'default' |   nupst action list default               - List actions for UPS or group with ID 'default' | ||||||
|   nupst action add default        - Add a new action to UPS 'default' |   nupst action add default                - Add a new action to UPS or group 'default' | ||||||
|   nupst action remove default 0   - Remove action at index 0 from UPS 'default' |   nupst action remove default 0           - Remove action at index 0 from UPS or group 'default' | ||||||
|  |   nupst action add dc-rack-1              - Add a new action to group 'dc-rack-1' | ||||||
|  | `); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private showFeatureHelp(): void { | ||||||
|  |     logger.log(` | ||||||
|  | NUPST - Feature Management Commands | ||||||
|  |  | ||||||
|  | Usage: | ||||||
|  |   nupst feature <subcommand> | ||||||
|  |  | ||||||
|  | Subcommands: | ||||||
|  |   httpServer              - Configure HTTP server for JSON status export | ||||||
|  |  | ||||||
|  | Examples: | ||||||
|  |   nupst feature httpServer    - Enable/disable HTTP server with interactive setup | ||||||
| `); | `); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import { Nupst } from '../nupst.ts'; | |||||||
| import { logger, type ITableColumn } from '../logger.ts'; | import { logger, type ITableColumn } from '../logger.ts'; | ||||||
| import { theme, symbols } from '../colors.ts'; | import { theme, symbols } from '../colors.ts'; | ||||||
| import type { IActionConfig } from '../actions/base-action.ts'; | import type { IActionConfig } from '../actions/base-action.ts'; | ||||||
| import type { IUpsConfig } from '../daemon.ts'; | import type { IUpsConfig, IGroupConfig } from '../daemon.ts'; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Class for handling action-related CLI commands |  * Class for handling action-related CLI commands | ||||||
| @@ -21,30 +21,42 @@ export class ActionHandler { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Add a new action to a UPS |    * Add a new action to a UPS or group | ||||||
|    */ |    */ | ||||||
|   public async add(upsId?: string): Promise<void> { |   public async add(targetId?: string): Promise<void> { | ||||||
|     try { |     try { | ||||||
|       if (!upsId) { |       if (!targetId) { | ||||||
|         logger.error('UPS ID is required'); |         logger.error('Target ID is required'); | ||||||
|         logger.log(`  ${theme.dim('Usage:')} ${theme.command('nupst action add <ups-id>')}`); |         logger.log( | ||||||
|  |           `  ${theme.dim('Usage:')} ${theme.command('nupst action add <ups-id|group-id>')}`, | ||||||
|  |         ); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.log(`  ${theme.dim('List UPS devices:')} ${theme.command('nupst ups list')}`); |         logger.log(`  ${theme.dim('List UPS devices:')} ${theme.command('nupst ups list')}`); | ||||||
|  |         logger.log(`  ${theme.dim('List groups:')} ${theme.command('nupst group list')}`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         process.exit(1); |         process.exit(1); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       const config = await this.nupst.getDaemon().loadConfig(); |       const config = await this.nupst.getDaemon().loadConfig(); | ||||||
|       const ups = config.upsDevices.find((u) => u.id === upsId); |  | ||||||
|  |  | ||||||
|       if (!ups) { |       // Check if it's a UPS | ||||||
|         logger.error(`UPS with ID '${upsId}' not found`); |       const ups = config.upsDevices.find((u) => u.id === targetId); | ||||||
|  |       // Check if it's a group | ||||||
|  |       const group = config.groups?.find((g) => g.id === targetId); | ||||||
|  |  | ||||||
|  |       if (!ups && !group) { | ||||||
|  |         logger.error(`UPS or Group with ID '${targetId}' not found`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.log(`  ${theme.dim('List available UPS devices:')} ${theme.command('nupst ups list')}`); |         logger.log(`  ${theme.dim('List available UPS devices:')} ${theme.command('nupst ups list')}`); | ||||||
|  |         logger.log(`  ${theme.dim('List available groups:')} ${theme.command('nupst group list')}`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         process.exit(1); |         process.exit(1); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       const target = ups || group; | ||||||
|  |       const targetType = ups ? 'UPS' : 'Group'; | ||||||
|  |       const targetName = ups ? ups.name : group!.name; | ||||||
|  |  | ||||||
|       const readline = await import('node:readline'); |       const readline = await import('node:readline'); | ||||||
|       const rl = readline.createInterface({ |       const rl = readline.createInterface({ | ||||||
|         input: process.stdin, |         input: process.stdin, | ||||||
| @@ -61,7 +73,7 @@ export class ActionHandler { | |||||||
|  |  | ||||||
|       try { |       try { | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.info(`Add Action to ${theme.highlight(ups.name)}`); |         logger.info(`Add Action to ${targetType} ${theme.highlight(targetName)}`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|  |  | ||||||
|         // Action type (currently only shutdown is supported) |         // Action type (currently only shutdown is supported) | ||||||
| @@ -130,37 +142,38 @@ export class ActionHandler { | |||||||
|           shutdownDelay, |           shutdownDelay, | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         // Add to UPS |         // Add to target (UPS or group) | ||||||
|         if (!ups.actions) { |         if (!target!.actions) { | ||||||
|           ups.actions = []; |           target!.actions = []; | ||||||
|         } |         } | ||||||
|         ups.actions.push(newAction); |         target!.actions.push(newAction); | ||||||
|  |  | ||||||
|         await this.nupst.getDaemon().saveConfig(config); |         await this.nupst.getDaemon().saveConfig(config); | ||||||
|  |  | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.success(`Action added to ${ups.name}`); |         logger.success(`Action added to ${targetType} ${targetName}`); | ||||||
|         logger.log(''); |         logger.log(`  ${theme.dim('Changes saved and will be applied automatically')}`); | ||||||
|         logger.log(`  ${theme.dim('Restart service to apply changes:')} ${theme.command('nupst service restart')}`); |  | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|       } finally { |       } finally { | ||||||
|         rl.close(); |         rl.close(); | ||||||
|       } |       } | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       logger.error(`Failed to add action: ${error instanceof Error ? error.message : String(error)}`); |       logger.error( | ||||||
|  |         `Failed to add action: ${error instanceof Error ? error.message : String(error)}`, | ||||||
|  |       ); | ||||||
|       process.exit(1); |       process.exit(1); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Remove an action from a UPS |    * Remove an action from a UPS or group | ||||||
|    */ |    */ | ||||||
|   public async remove(upsId?: string, actionIndexStr?: string): Promise<void> { |   public async remove(targetId?: string, actionIndexStr?: string): Promise<void> { | ||||||
|     try { |     try { | ||||||
|       if (!upsId || !actionIndexStr) { |       if (!targetId || !actionIndexStr) { | ||||||
|         logger.error('UPS ID and action index are required'); |         logger.error('Target ID and action index are required'); | ||||||
|         logger.log( |         logger.log( | ||||||
|           `  ${theme.dim('Usage:')} ${theme.command('nupst action remove <ups-id> <action-index>')}`, |           `  ${theme.dim('Usage:')} ${theme.command('nupst action remove <ups-id|group-id> <action-index>')}`, | ||||||
|         ); |         ); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.log(`  ${theme.dim('List actions:')} ${theme.command('nupst action list')}`); |         logger.log(`  ${theme.dim('List actions:')} ${theme.command('nupst action list')}`); | ||||||
| @@ -175,47 +188,57 @@ export class ActionHandler { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       const config = await this.nupst.getDaemon().loadConfig(); |       const config = await this.nupst.getDaemon().loadConfig(); | ||||||
|       const ups = config.upsDevices.find((u) => u.id === upsId); |  | ||||||
|  |  | ||||||
|       if (!ups) { |       // Check if it's a UPS | ||||||
|         logger.error(`UPS with ID '${upsId}' not found`); |       const ups = config.upsDevices.find((u) => u.id === targetId); | ||||||
|  |       // Check if it's a group | ||||||
|  |       const group = config.groups?.find((g) => g.id === targetId); | ||||||
|  |  | ||||||
|  |       if (!ups && !group) { | ||||||
|  |         logger.error(`UPS or Group with ID '${targetId}' not found`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.log(`  ${theme.dim('List available UPS devices:')} ${theme.command('nupst ups list')}`); |         logger.log(`  ${theme.dim('List available UPS devices:')} ${theme.command('nupst ups list')}`); | ||||||
|  |         logger.log(`  ${theme.dim('List available groups:')} ${theme.command('nupst group list')}`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         process.exit(1); |         process.exit(1); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (!ups.actions || ups.actions.length === 0) { |       const target = ups || group; | ||||||
|         logger.error(`No actions configured for UPS '${ups.name}'`); |       const targetType = ups ? 'UPS' : 'Group'; | ||||||
|  |       const targetName = ups ? ups.name : group!.name; | ||||||
|  |  | ||||||
|  |       if (!target!.actions || target!.actions.length === 0) { | ||||||
|  |         logger.error(`No actions configured for ${targetType} '${targetName}'`); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         process.exit(1); |         process.exit(1); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (actionIndex >= ups.actions.length) { |       if (actionIndex >= target!.actions.length) { | ||||||
|         logger.error( |         logger.error( | ||||||
|           `Invalid action index. UPS '${ups.name}' has ${ups.actions.length} action(s) (index 0-${ups.actions.length - 1})`, |           `Invalid action index. ${targetType} '${targetName}' has ${target!.actions.length} action(s) (index 0-${target!.actions.length - 1})`, | ||||||
|         ); |         ); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.log(`  ${theme.dim('List actions:')} ${theme.command(`nupst action list ${upsId}`)}`); |         logger.log( | ||||||
|  |           `  ${theme.dim('List actions:')} ${theme.command(`nupst action list ${targetId}`)}`, | ||||||
|  |         ); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         process.exit(1); |         process.exit(1); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       const removedAction = ups.actions[actionIndex]; |       const removedAction = target!.actions[actionIndex]; | ||||||
|       ups.actions.splice(actionIndex, 1); |       target!.actions.splice(actionIndex, 1); | ||||||
|  |  | ||||||
|       await this.nupst.getDaemon().saveConfig(config); |       await this.nupst.getDaemon().saveConfig(config); | ||||||
|  |  | ||||||
|       logger.log(''); |       logger.log(''); | ||||||
|       logger.success(`Action removed from ${ups.name}`); |       logger.success(`Action removed from ${targetType} ${targetName}`); | ||||||
|       logger.log(`  ${theme.dim('Type:')} ${removedAction.type}`); |       logger.log(`  ${theme.dim('Type:')} ${removedAction.type}`); | ||||||
|       if (removedAction.thresholds) { |       if (removedAction.thresholds) { | ||||||
|         logger.log( |         logger.log( | ||||||
|           `  ${theme.dim('Thresholds:')} Battery: ${removedAction.thresholds.battery}%, Runtime: ${removedAction.thresholds.runtime}min`, |           `  ${theme.dim('Thresholds:')} Battery: ${removedAction.thresholds.battery}%, Runtime: ${removedAction.thresholds.runtime}min`, | ||||||
|         ); |         ); | ||||||
|       } |       } | ||||||
|       logger.log(''); |       logger.log(`  ${theme.dim('Changes saved and will be applied automatically')}`); | ||||||
|       logger.log(`  ${theme.dim('Restart service to apply changes:')} ${theme.command('nupst service restart')}`); |  | ||||||
|       logger.log(''); |       logger.log(''); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       logger.error( |       logger.error( | ||||||
| @@ -226,43 +249,61 @@ export class ActionHandler { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * List all actions for a specific UPS or all UPS devices |    * List all actions for a specific UPS/group or all devices | ||||||
|    */ |    */ | ||||||
|   public async list(upsId?: string): Promise<void> { |   public async list(targetId?: string): Promise<void> { | ||||||
|     try { |     try { | ||||||
|       const config = await this.nupst.getDaemon().loadConfig(); |       const config = await this.nupst.getDaemon().loadConfig(); | ||||||
|  |  | ||||||
|       if (upsId) { |       if (targetId) { | ||||||
|         // List actions for specific UPS |         // List actions for specific UPS or group | ||||||
|         const ups = config.upsDevices.find((u) => u.id === upsId); |         const ups = config.upsDevices.find((u) => u.id === targetId); | ||||||
|  |         const group = config.groups?.find((g) => g.id === targetId); | ||||||
|  |  | ||||||
|         if (!ups) { |         if (!ups && !group) { | ||||||
|           logger.error(`UPS with ID '${upsId}' not found`); |           logger.error(`UPS or Group with ID '${targetId}' not found`); | ||||||
|           logger.log(''); |           logger.log(''); | ||||||
|           logger.log(`  ${theme.dim('List available UPS devices:')} ${theme.command('nupst ups list')}`); |           logger.log(`  ${theme.dim('List available UPS devices:')} ${theme.command('nupst ups list')}`); | ||||||
|  |           logger.log(`  ${theme.dim('List available groups:')} ${theme.command('nupst group list')}`); | ||||||
|           logger.log(''); |           logger.log(''); | ||||||
|           process.exit(1); |           process.exit(1); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         this.displayUpsActions(ups); |         if (ups) { | ||||||
|  |           this.displayTargetActions(ups, 'UPS'); | ||||||
|  |         } else { | ||||||
|  |           this.displayTargetActions(group!, 'Group'); | ||||||
|  |         } | ||||||
|       } else { |       } else { | ||||||
|         // List actions for all UPS devices |         // List actions for all UPS devices and groups | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|         logger.info('Actions for All UPS Devices'); |         logger.info('Actions for All UPS Devices and Groups'); | ||||||
|         logger.log(''); |         logger.log(''); | ||||||
|  |  | ||||||
|         let hasAnyActions = false; |         let hasAnyActions = false; | ||||||
|  |  | ||||||
|  |         // Display UPS actions | ||||||
|         for (const ups of config.upsDevices) { |         for (const ups of config.upsDevices) { | ||||||
|           if (ups.actions && ups.actions.length > 0) { |           if (ups.actions && ups.actions.length > 0) { | ||||||
|             hasAnyActions = true; |             hasAnyActions = true; | ||||||
|             this.displayUpsActions(ups); |             this.displayTargetActions(ups, 'UPS'); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Display Group actions | ||||||
|  |         for (const group of config.groups || []) { | ||||||
|  |           if (group.actions && group.actions.length > 0) { | ||||||
|  |             hasAnyActions = true; | ||||||
|  |             this.displayTargetActions(group, 'Group'); | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (!hasAnyActions) { |         if (!hasAnyActions) { | ||||||
|           logger.log(`  ${theme.dim('No actions configured')}`); |           logger.log(`  ${theme.dim('No actions configured')}`); | ||||||
|           logger.log(''); |           logger.log(''); | ||||||
|           logger.log(`  ${theme.dim('Add an action:')} ${theme.command('nupst action add <ups-id>')}`); |           logger.log( | ||||||
|  |             `  ${theme.dim('Add an action:')} ${theme.command('nupst action add <ups-id|group-id>')}`, | ||||||
|  |           ); | ||||||
|           logger.log(''); |           logger.log(''); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
| @@ -275,13 +316,18 @@ export class ActionHandler { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Display actions for a single UPS |    * Display actions for a single UPS or Group | ||||||
|    */ |    */ | ||||||
|   private displayUpsActions(ups: IUpsConfig): void { |   private displayTargetActions( | ||||||
|     logger.log(`${symbols.info} ${theme.highlight(ups.name)} ${theme.dim(`(${ups.id})`)}`); |     target: IUpsConfig | IGroupConfig, | ||||||
|  |     targetType: 'UPS' | 'Group', | ||||||
|  |   ): void { | ||||||
|  |     logger.log( | ||||||
|  |       `${symbols.info} ${targetType} ${theme.highlight(target.name)} ${theme.dim(`(${target.id})`)}`, | ||||||
|  |     ); | ||||||
|     logger.log(''); |     logger.log(''); | ||||||
|  |  | ||||||
|     if (!ups.actions || ups.actions.length === 0) { |     if (!target.actions || target.actions.length === 0) { | ||||||
|       logger.log(`  ${theme.dim('No actions configured')}`); |       logger.log(`  ${theme.dim('No actions configured')}`); | ||||||
|       logger.log(''); |       logger.log(''); | ||||||
|       return; |       return; | ||||||
| @@ -296,7 +342,7 @@ export class ActionHandler { | |||||||
|       { header: 'Delay', key: 'delay', align: 'right' }, |       { header: 'Delay', key: 'delay', align: 'right' }, | ||||||
|     ]; |     ]; | ||||||
|  |  | ||||||
|     const rows = ups.actions.map((action, index) => ({ |     const rows = target.actions.map((action, index) => ({ | ||||||
|       index: theme.dim(index.toString()), |       index: theme.dim(index.toString()), | ||||||
|       type: theme.highlight(action.type), |       type: theme.highlight(action.type), | ||||||
|       battery: action.thresholds ? `${action.thresholds.battery}%` : theme.dim('N/A'), |       battery: action.thresholds ? `${action.thresholds.battery}%` : theme.dim('N/A'), | ||||||
|   | |||||||
							
								
								
									
										213
									
								
								ts/cli/feature-handler.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								ts/cli/feature-handler.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,213 @@ | |||||||
|  | import process from 'node:process'; | ||||||
|  | import { execSync } from 'node:child_process'; | ||||||
|  | import { Nupst } from '../nupst.ts'; | ||||||
|  | import { logger } from '../logger.ts'; | ||||||
|  | import { theme } from '../colors.ts'; | ||||||
|  | import * as helpers from '../helpers/index.ts'; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Class for handling feature-related CLI commands | ||||||
|  |  * Provides interface for managing optional features like HTTP server | ||||||
|  |  */ | ||||||
|  | export class FeatureHandler { | ||||||
|  |   private readonly nupst: Nupst; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Create a new feature handler | ||||||
|  |    * @param nupst Reference to the main Nupst instance | ||||||
|  |    */ | ||||||
|  |   constructor(nupst: Nupst) { | ||||||
|  |     this.nupst = nupst; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Configure HTTP server feature | ||||||
|  |    */ | ||||||
|  |   public async configureHttpServer(): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       const readline = await import('node:readline'); | ||||||
|  |       const rl = readline.createInterface({ | ||||||
|  |         input: process.stdin, | ||||||
|  |         output: process.stdout, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       const prompt = (question: string): Promise<string> => { | ||||||
|  |         return new Promise((resolve) => { | ||||||
|  |           rl.question(question, (answer: string) => { | ||||||
|  |             resolve(answer); | ||||||
|  |           }); | ||||||
|  |         }); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       try { | ||||||
|  |         await this.runHttpServerConfig(prompt); | ||||||
|  |       } finally { | ||||||
|  |         rl.close(); | ||||||
|  |         process.stdin.destroy(); | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       logger.error(`HTTP Server config error: ${error instanceof Error ? error.message : String(error)}`); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Run the interactive HTTP server configuration process | ||||||
|  |    * @param prompt Function to prompt for user input | ||||||
|  |    */ | ||||||
|  |   private async runHttpServerConfig(prompt: (question: string) => Promise<string>): Promise<void> { | ||||||
|  |     logger.log(''); | ||||||
|  |     logger.logBoxTitle('HTTP Server Feature Configuration', 60); | ||||||
|  |     logger.logBoxLine('Configure the HTTP server to expose UPS status as JSON'); | ||||||
|  |     logger.logBoxEnd(); | ||||||
|  |     logger.log(''); | ||||||
|  |  | ||||||
|  |     // Load config | ||||||
|  |     let config; | ||||||
|  |     try { | ||||||
|  |       await this.nupst.getDaemon().loadConfig(); | ||||||
|  |       config = this.nupst.getDaemon().getConfig(); | ||||||
|  |     } catch (error) { | ||||||
|  |       logger.error('No configuration found. Please run "nupst ups add" first.'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Show current status | ||||||
|  |     if (config.httpServer?.enabled) { | ||||||
|  |       logger.info('HTTP Server is currently: ' + theme.success('ENABLED')); | ||||||
|  |       logger.log(`  Port: ${theme.highlight(String(config.httpServer.port))}`); | ||||||
|  |       logger.log(`  Path: ${theme.highlight(config.httpServer.path)}`); | ||||||
|  |       logger.log(`  Auth Token: ${theme.dim('***' + config.httpServer.authToken.slice(-4))}`); | ||||||
|  |       logger.log(''); | ||||||
|  |     } else { | ||||||
|  |       logger.info('HTTP Server is currently: ' + theme.dim('DISABLED')); | ||||||
|  |       logger.log(''); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Ask enable/disable | ||||||
|  |     const action = await prompt('Enable or disable HTTP server? (enable/disable/cancel): '); | ||||||
|  |  | ||||||
|  |     if (action.toLowerCase() === 'cancel' || action.toLowerCase() === 'c') { | ||||||
|  |       logger.log('Cancelled.'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (action.toLowerCase() === 'disable' || action.toLowerCase() === 'd') { | ||||||
|  |       // Disable HTTP server | ||||||
|  |       config.httpServer = { | ||||||
|  |         enabled: false, | ||||||
|  |         port: config.httpServer?.port || 8080, | ||||||
|  |         path: config.httpServer?.path || '/ups-status', | ||||||
|  |         authToken: config.httpServer?.authToken || '', | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       this.nupst.getDaemon().saveConfig(config); | ||||||
|  |  | ||||||
|  |       logger.log(''); | ||||||
|  |       logger.success('HTTP Server disabled'); | ||||||
|  |       logger.log(''); | ||||||
|  |  | ||||||
|  |       await this.restartServiceIfRunning(); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (action.toLowerCase() !== 'enable' && action.toLowerCase() !== 'e') { | ||||||
|  |       logger.error('Invalid option. Please enter "enable", "disable", or "cancel".'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Enable - gather configuration | ||||||
|  |     logger.log(''); | ||||||
|  |  | ||||||
|  |     const portInput = await prompt(`HTTP Server Port [${config.httpServer?.port || 8080}]: `); | ||||||
|  |     const port = portInput ? parseInt(portInput, 10) : (config.httpServer?.port || 8080); | ||||||
|  |  | ||||||
|  |     if (isNaN(port) || port < 1 || port > 65535) { | ||||||
|  |       logger.error('Invalid port number. Must be between 1 and 65535.'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const pathInput = await prompt(`URL Path [${config.httpServer?.path || '/ups-status'}]: `); | ||||||
|  |     const path = pathInput || config.httpServer?.path || '/ups-status'; | ||||||
|  |  | ||||||
|  |     // Ensure path starts with / | ||||||
|  |     const finalPath = path.startsWith('/') ? path : `/${path}`; | ||||||
|  |  | ||||||
|  |     // Generate or reuse auth token | ||||||
|  |     let authToken = config.httpServer?.authToken; | ||||||
|  |     if (!authToken) { | ||||||
|  |       // Generate new random token | ||||||
|  |       authToken = helpers.shortId() + helpers.shortId() + helpers.shortId(); | ||||||
|  |       logger.log(''); | ||||||
|  |       logger.info('Generated new authentication token'); | ||||||
|  |     } else { | ||||||
|  |       const regenerate = await prompt('Regenerate authentication token? (y/N): '); | ||||||
|  |       if (regenerate.toLowerCase() === 'y' || regenerate.toLowerCase() === 'yes') { | ||||||
|  |         authToken = helpers.shortId() + helpers.shortId() + helpers.shortId(); | ||||||
|  |         logger.info('Generated new authentication token'); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Save configuration | ||||||
|  |     config.httpServer = { | ||||||
|  |       enabled: true, | ||||||
|  |       port, | ||||||
|  |       path: finalPath, | ||||||
|  |       authToken, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     this.nupst.getDaemon().saveConfig(config); | ||||||
|  |  | ||||||
|  |     // Display summary | ||||||
|  |     logger.log(''); | ||||||
|  |     logger.logBoxTitle('HTTP Server Configuration', 70, 'success'); | ||||||
|  |     logger.logBoxLine(`Status: ${theme.success('ENABLED')}`); | ||||||
|  |     logger.logBoxLine(`Port: ${theme.highlight(String(port))}`); | ||||||
|  |     logger.logBoxLine(`Path: ${theme.highlight(finalPath)}`); | ||||||
|  |     logger.logBoxLine(`Auth Token: ${theme.warning(authToken)}`); | ||||||
|  |     logger.logBoxLine(''); | ||||||
|  |     logger.logBoxLine(theme.dim('Usage examples:')); | ||||||
|  |     logger.logBoxLine(`  curl -H "Authorization: Bearer ${authToken}" http://localhost:${port}${finalPath}`); | ||||||
|  |     logger.logBoxLine(`  curl "http://localhost:${port}${finalPath}?token=${authToken}"`); | ||||||
|  |     logger.logBoxEnd(); | ||||||
|  |     logger.log(''); | ||||||
|  |  | ||||||
|  |     logger.warn('IMPORTANT: Save the authentication token securely!'); | ||||||
|  |     logger.log(''); | ||||||
|  |  | ||||||
|  |     await this.restartServiceIfRunning(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Restart the service if it's currently running | ||||||
|  |    */ | ||||||
|  |   private async restartServiceIfRunning(): Promise<void> { | ||||||
|  |     try { | ||||||
|  |       const isActive = execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; | ||||||
|  |  | ||||||
|  |       if (isActive) { | ||||||
|  |         logger.log(''); | ||||||
|  |         const readline = await import('node:readline'); | ||||||
|  |         const rl = readline.createInterface({ | ||||||
|  |           input: process.stdin, | ||||||
|  |           output: process.stdout, | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         const answer = await new Promise<string>((resolve) => { | ||||||
|  |           rl.question('Service is running. Restart to apply changes? (Y/n): ', resolve); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         rl.close(); | ||||||
|  |  | ||||||
|  |         if (!answer || answer.toLowerCase() === 'y' || answer.toLowerCase() === 'yes') { | ||||||
|  |           logger.info('Restarting service...'); | ||||||
|  |           execSync('sudo systemctl restart nupst.service'); | ||||||
|  |           logger.success('Service restarted successfully'); | ||||||
|  |         } else { | ||||||
|  |           logger.warn('Changes will take effect on next service restart'); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } catch (error) { | ||||||
|  |       // Ignore errors - service might not be installed | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								ts/daemon.ts
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								ts/daemon.ts
									
									
									
									
									
								
							| @@ -10,6 +10,7 @@ import { MigrationRunner } from './migrations/index.ts'; | |||||||
| import { theme, symbols, getBatteryColor, getRuntimeColor, formatPowerStatus } from './colors.ts'; | import { theme, symbols, getBatteryColor, getRuntimeColor, formatPowerStatus } from './colors.ts'; | ||||||
| import type { IActionConfig } from './actions/base-action.ts'; | import type { IActionConfig } from './actions/base-action.ts'; | ||||||
| import { ActionManager, type IActionContext, type TPowerStatus } from './actions/index.ts'; | import { ActionManager, type IActionContext, type TPowerStatus } from './actions/index.ts'; | ||||||
|  | import { NupstHttpServer } from './http-server.ts'; | ||||||
|  |  | ||||||
| const execAsync = promisify(exec); | const execAsync = promisify(exec); | ||||||
| const execFileAsync = promisify(execFile); | const execFileAsync = promisify(execFile); | ||||||
| @@ -46,6 +47,20 @@ export interface IGroupConfig { | |||||||
|   actions?: IActionConfig[]; |   actions?: IActionConfig[]; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * HTTP Server configuration interface | ||||||
|  |  */ | ||||||
|  | export interface IHttpServerConfig { | ||||||
|  |   /** Whether HTTP server is enabled */ | ||||||
|  |   enabled: boolean; | ||||||
|  |   /** Port to listen on */ | ||||||
|  |   port: number; | ||||||
|  |   /** URL path for the endpoint */ | ||||||
|  |   path: string; | ||||||
|  |   /** Authentication token */ | ||||||
|  |   authToken: string; | ||||||
|  | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Configuration interface for the daemon |  * Configuration interface for the daemon | ||||||
|  */ |  */ | ||||||
| @@ -58,6 +73,8 @@ export interface INupstConfig { | |||||||
|   groups: IGroupConfig[]; |   groups: IGroupConfig[]; | ||||||
|   /** Check interval in milliseconds */ |   /** Check interval in milliseconds */ | ||||||
|   checkInterval: number; |   checkInterval: number; | ||||||
|  |   /** HTTP Server configuration */ | ||||||
|  |   httpServer?: IHttpServerConfig; | ||||||
|  |  | ||||||
|   // Legacy fields for backward compatibility (will be migrated away) |   // Legacy fields for backward compatibility (will be migrated away) | ||||||
|   /** UPS list (v3 format - legacy) */ |   /** UPS list (v3 format - legacy) */ | ||||||
| @@ -82,6 +99,10 @@ export interface IUpsStatus { | |||||||
|   powerStatus: 'online' | 'onBattery' | 'unknown'; |   powerStatus: 'online' | 'onBattery' | 'unknown'; | ||||||
|   batteryCapacity: number; |   batteryCapacity: number; | ||||||
|   batteryRuntime: number; |   batteryRuntime: number; | ||||||
|  |   outputLoad: number;        // Load percentage (0-100%) | ||||||
|  |   outputPower: number;        // Power in watts | ||||||
|  |   outputVoltage: number;      // Voltage in volts | ||||||
|  |   outputCurrent: number;      // Current in amps | ||||||
|   lastStatusChange: number; |   lastStatusChange: number; | ||||||
|   lastCheckTime: number; |   lastCheckTime: number; | ||||||
| } | } | ||||||
| @@ -139,6 +160,7 @@ export class NupstDaemon { | |||||||
|   private snmp: NupstSnmp; |   private snmp: NupstSnmp; | ||||||
|   private isRunning: boolean = false; |   private isRunning: boolean = false; | ||||||
|   private upsStatus: Map<string, IUpsStatus> = new Map(); |   private upsStatus: Map<string, IUpsStatus> = new Map(); | ||||||
|  |   private httpServer?: NupstHttpServer; | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Create a new daemon instance with the given SNMP manager |    * Create a new daemon instance with the given SNMP manager | ||||||
| @@ -278,6 +300,21 @@ export class NupstDaemon { | |||||||
|       // Initialize UPS status tracking |       // Initialize UPS status tracking | ||||||
|       this.initializeUpsStatus(); |       this.initializeUpsStatus(); | ||||||
|  |  | ||||||
|  |       // Start HTTP server if configured | ||||||
|  |       if (this.config.httpServer?.enabled && this.config.httpServer.authToken) { | ||||||
|  |         try { | ||||||
|  |           this.httpServer = new NupstHttpServer( | ||||||
|  |             this.config.httpServer.port, | ||||||
|  |             this.config.httpServer.path, | ||||||
|  |             this.config.httpServer.authToken, | ||||||
|  |             () => this.upsStatus | ||||||
|  |           ); | ||||||
|  |           this.httpServer.start(); | ||||||
|  |         } catch (error) { | ||||||
|  |           logger.error(`Failed to start HTTP server: ${error instanceof Error ? error.message : String(error)}`); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|       // Start UPS monitoring |       // Start UPS monitoring | ||||||
|       this.isRunning = true; |       this.isRunning = true; | ||||||
|       await this.monitor(); |       await this.monitor(); | ||||||
| @@ -304,6 +341,10 @@ export class NupstDaemon { | |||||||
|           powerStatus: 'unknown', |           powerStatus: 'unknown', | ||||||
|           batteryCapacity: 100, |           batteryCapacity: 100, | ||||||
|           batteryRuntime: 999, // High value as default |           batteryRuntime: 999, // High value as default | ||||||
|  |           outputLoad: 0, | ||||||
|  |           outputPower: 0, | ||||||
|  |           outputVoltage: 0, | ||||||
|  |           outputCurrent: 0, | ||||||
|           lastStatusChange: Date.now(), |           lastStatusChange: Date.now(), | ||||||
|           lastCheckTime: 0, |           lastCheckTime: 0, | ||||||
|         }); |         }); | ||||||
| @@ -377,6 +418,12 @@ export class NupstDaemon { | |||||||
|    */ |    */ | ||||||
|   public stop(): void { |   public stop(): void { | ||||||
|     logger.log('Stopping NUPST daemon...'); |     logger.log('Stopping NUPST daemon...'); | ||||||
|  |  | ||||||
|  |     // Stop HTTP server if running | ||||||
|  |     if (this.httpServer) { | ||||||
|  |       this.httpServer.stop(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     this.isRunning = false; |     this.isRunning = false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -437,6 +484,10 @@ export class NupstDaemon { | |||||||
|             powerStatus: 'unknown', |             powerStatus: 'unknown', | ||||||
|             batteryCapacity: 100, |             batteryCapacity: 100, | ||||||
|             batteryRuntime: 999, |             batteryRuntime: 999, | ||||||
|  |             outputLoad: 0, | ||||||
|  |             outputPower: 0, | ||||||
|  |             outputVoltage: 0, | ||||||
|  |             outputCurrent: 0, | ||||||
|             lastStatusChange: Date.now(), |             lastStatusChange: Date.now(), | ||||||
|             lastCheckTime: 0, |             lastCheckTime: 0, | ||||||
|           }); |           }); | ||||||
| @@ -456,6 +507,10 @@ export class NupstDaemon { | |||||||
|           powerStatus: status.powerStatus, |           powerStatus: status.powerStatus, | ||||||
|           batteryCapacity: status.batteryCapacity, |           batteryCapacity: status.batteryCapacity, | ||||||
|           batteryRuntime: status.batteryRuntime, |           batteryRuntime: status.batteryRuntime, | ||||||
|  |           outputLoad: status.outputLoad, | ||||||
|  |           outputPower: status.outputPower, | ||||||
|  |           outputVoltage: status.outputVoltage, | ||||||
|  |           outputCurrent: status.outputCurrent, | ||||||
|           lastCheckTime: currentTime, |           lastCheckTime: currentTime, | ||||||
|           lastStatusChange: currentStatus?.lastStatusChange || currentTime, |           lastStatusChange: currentStatus?.lastStatusChange || currentTime, | ||||||
|         }; |         }; | ||||||
|   | |||||||
							
								
								
									
										113
									
								
								ts/http-server.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								ts/http-server.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | |||||||
|  | import * as http from 'node:http'; | ||||||
|  | import { URL } from 'node:url'; | ||||||
|  | import { logger } from './logger.ts'; | ||||||
|  | import type { IUpsStatus } from './daemon.ts'; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * HTTP Server for exposing UPS status as JSON | ||||||
|  |  * Serves cached data from the daemon's monitoring loop | ||||||
|  |  */ | ||||||
|  | export class NupstHttpServer { | ||||||
|  |   private server?: http.Server; | ||||||
|  |   private port: number; | ||||||
|  |   private path: string; | ||||||
|  |   private authToken: string; | ||||||
|  |   private getUpsStatus: () => Map<string, IUpsStatus>; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Create a new HTTP server instance | ||||||
|  |    * @param port Port to listen on | ||||||
|  |    * @param path URL path for the endpoint | ||||||
|  |    * @param authToken Authentication token required for access | ||||||
|  |    * @param getUpsStatus Function to retrieve cached UPS status | ||||||
|  |    */ | ||||||
|  |   constructor( | ||||||
|  |     port: number, | ||||||
|  |     path: string, | ||||||
|  |     authToken: string, | ||||||
|  |     getUpsStatus: () => Map<string, IUpsStatus> | ||||||
|  |   ) { | ||||||
|  |     this.port = port; | ||||||
|  |     this.path = path; | ||||||
|  |     this.authToken = authToken; | ||||||
|  |     this.getUpsStatus = getUpsStatus; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Verify authentication token from request | ||||||
|  |    * Supports both Bearer token in Authorization header and token query parameter | ||||||
|  |    * @param req HTTP request | ||||||
|  |    * @returns True if authenticated, false otherwise | ||||||
|  |    */ | ||||||
|  |   private isAuthenticated(req: http.IncomingMessage): boolean { | ||||||
|  |     // Check Authorization header (Bearer token) | ||||||
|  |     const authHeader = req.headers.authorization; | ||||||
|  |     if (authHeader?.startsWith('Bearer ')) { | ||||||
|  |       const token = authHeader.substring(7); | ||||||
|  |       return token === this.authToken; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Check token query parameter | ||||||
|  |     if (req.url) { | ||||||
|  |       const url = new URL(req.url, `http://localhost:${this.port}`); | ||||||
|  |       const tokenParam = url.searchParams.get('token'); | ||||||
|  |       return tokenParam === this.authToken; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Start the HTTP server | ||||||
|  |    */ | ||||||
|  |   public start(): void { | ||||||
|  |     this.server = http.createServer((req, res) => { | ||||||
|  |       // Parse URL | ||||||
|  |       const reqUrl = new URL(req.url || '/', `http://localhost:${this.port}`); | ||||||
|  |  | ||||||
|  |       if (reqUrl.pathname === this.path && req.method === 'GET') { | ||||||
|  |         // Check authentication | ||||||
|  |         if (!this.isAuthenticated(req)) { | ||||||
|  |           res.writeHead(401, { | ||||||
|  |             'Content-Type': 'application/json', | ||||||
|  |             'WWW-Authenticate': 'Bearer' | ||||||
|  |           }); | ||||||
|  |           res.end(JSON.stringify({ error: 'Unauthorized' })); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Get cached status (no refresh) | ||||||
|  |         const statusMap = this.getUpsStatus(); | ||||||
|  |         const statusArray = Array.from(statusMap.values()); | ||||||
|  |  | ||||||
|  |         res.writeHead(200, { | ||||||
|  |           'Content-Type': 'application/json', | ||||||
|  |           'Cache-Control': 'no-cache' | ||||||
|  |         }); | ||||||
|  |         res.end(JSON.stringify(statusArray, null, 2)); | ||||||
|  |       } else { | ||||||
|  |         res.writeHead(404, { 'Content-Type': 'application/json' }); | ||||||
|  |         res.end(JSON.stringify({ error: 'Not Found' })); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     this.server.listen(this.port, () => { | ||||||
|  |       logger.success(`HTTP server started on port ${this.port} at ${this.path}`); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     this.server.on('error', (error: any) => { | ||||||
|  |       logger.error(`HTTP server error: ${error.message}`); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Stop the HTTP server | ||||||
|  |    */ | ||||||
|  |   public stop(): void { | ||||||
|  |     if (this.server) { | ||||||
|  |       this.server.close(() => { | ||||||
|  |         logger.log('HTTP server stopped'); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								ts/nupst.ts
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								ts/nupst.ts
									
									
									
									
									
								
							| @@ -1,12 +1,13 @@ | |||||||
| import { NupstSnmp } from './snmp/manager.ts'; | import { NupstSnmp } from './snmp/manager.ts'; | ||||||
| import { NupstDaemon } from './daemon.ts'; | import { NupstDaemon } from './daemon.ts'; | ||||||
| import { NupstSystemd } from './systemd.ts'; | import { NupstSystemd } from './systemd.ts'; | ||||||
| import { commitinfo } from './00_commitinfo_data.ts'; | import denoConfig from '../deno.json' with { type: 'json' }; | ||||||
| import { logger } from './logger.ts'; | import { logger } from './logger.ts'; | ||||||
| import { UpsHandler } from './cli/ups-handler.ts'; | import { UpsHandler } from './cli/ups-handler.ts'; | ||||||
| import { GroupHandler } from './cli/group-handler.ts'; | import { GroupHandler } from './cli/group-handler.ts'; | ||||||
| import { ServiceHandler } from './cli/service-handler.ts'; | import { ServiceHandler } from './cli/service-handler.ts'; | ||||||
| import { ActionHandler } from './cli/action-handler.ts'; | import { ActionHandler } from './cli/action-handler.ts'; | ||||||
|  | import { FeatureHandler } from './cli/feature-handler.ts'; | ||||||
| import * as https from 'node:https'; | import * as https from 'node:https'; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -21,6 +22,7 @@ export class Nupst { | |||||||
|   private readonly groupHandler: GroupHandler; |   private readonly groupHandler: GroupHandler; | ||||||
|   private readonly serviceHandler: ServiceHandler; |   private readonly serviceHandler: ServiceHandler; | ||||||
|   private readonly actionHandler: ActionHandler; |   private readonly actionHandler: ActionHandler; | ||||||
|  |   private readonly featureHandler: FeatureHandler; | ||||||
|   private updateAvailable: boolean = false; |   private updateAvailable: boolean = false; | ||||||
|   private latestVersion: string = ''; |   private latestVersion: string = ''; | ||||||
|  |  | ||||||
| @@ -39,6 +41,7 @@ export class Nupst { | |||||||
|     this.groupHandler = new GroupHandler(this); |     this.groupHandler = new GroupHandler(this); | ||||||
|     this.serviceHandler = new ServiceHandler(this); |     this.serviceHandler = new ServiceHandler(this); | ||||||
|     this.actionHandler = new ActionHandler(this); |     this.actionHandler = new ActionHandler(this); | ||||||
|  |     this.featureHandler = new FeatureHandler(this); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -90,12 +93,19 @@ export class Nupst { | |||||||
|     return this.actionHandler; |     return this.actionHandler; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Get the Feature handler for feature management | ||||||
|  |    */ | ||||||
|  |   public getFeatureHandler(): FeatureHandler { | ||||||
|  |     return this.featureHandler; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Get the current version of NUPST |    * Get the current version of NUPST | ||||||
|    * @returns The current version string |    * @returns The current version string | ||||||
|    */ |    */ | ||||||
|   public getVersion(): string { |   public getVersion(): string { | ||||||
|     return commitinfo.version; |     return denoConfig.version; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -143,8 +153,8 @@ export class Nupst { | |||||||
|   private getLatestVersion(): Promise<string> { |   private getLatestVersion(): Promise<string> { | ||||||
|     return new Promise<string>((resolve, reject) => { |     return new Promise<string>((resolve, reject) => { | ||||||
|       const options = { |       const options = { | ||||||
|         hostname: 'registry.npmjs.org', |         hostname: 'code.foss.global', | ||||||
|         path: '/@serve.zone/nupst', |         path: '/api/v1/repos/serve.zone/nupst/releases/latest', | ||||||
|         method: 'GET', |         method: 'GET', | ||||||
|         headers: { |         headers: { | ||||||
|           'Accept': 'application/json', |           'Accept': 'application/json', | ||||||
| @@ -162,10 +172,14 @@ export class Nupst { | |||||||
|         res.on('end', () => { |         res.on('end', () => { | ||||||
|           try { |           try { | ||||||
|             const response = JSON.parse(data); |             const response = JSON.parse(data); | ||||||
|             if (response['dist-tags'] && response['dist-tags'].latest) { |             if (response.tag_name) { | ||||||
|               resolve(response['dist-tags'].latest); |               // Strip 'v' prefix from tag name (e.g., "v5.1.7" -> "5.1.7") | ||||||
|  |               const version = response.tag_name.startsWith('v')  | ||||||
|  |                 ? response.tag_name.substring(1)  | ||||||
|  |                 : response.tag_name; | ||||||
|  |               resolve(version); | ||||||
|             } else { |             } else { | ||||||
|               reject(new Error('Failed to parse version from npm registry response')); |               reject(new Error('Failed to parse version from Gitea API response')); | ||||||
|             } |             } | ||||||
|           } catch (error) { |           } catch (error) { | ||||||
|             reject(error); |             reject(error); | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import * as snmp from 'npm:net-snmp@3.20.0'; | import * as snmp from 'npm:net-snmp@3.26.0'; | ||||||
| import { Buffer } from 'node:buffer'; | import { Buffer } from 'node:buffer'; | ||||||
| import type { IOidSet, ISnmpConfig, IUpsStatus, TUpsModel } from './types.ts'; | import type { IOidSet, ISnmpConfig, IUpsStatus, TUpsModel } from './types.ts'; | ||||||
| import { UpsOidSets } from './oid-sets.ts'; | import { UpsOidSets } from './oid-sets.ts'; | ||||||
| @@ -304,6 +304,10 @@ export class NupstSnmp { | |||||||
|         console.log('  Power Status:', this.activeOIDs.POWER_STATUS); |         console.log('  Power Status:', this.activeOIDs.POWER_STATUS); | ||||||
|         console.log('  Battery Capacity:', this.activeOIDs.BATTERY_CAPACITY); |         console.log('  Battery Capacity:', this.activeOIDs.BATTERY_CAPACITY); | ||||||
|         console.log('  Battery Runtime:', this.activeOIDs.BATTERY_RUNTIME); |         console.log('  Battery Runtime:', this.activeOIDs.BATTERY_RUNTIME); | ||||||
|  |         console.log('  Output Load:', this.activeOIDs.OUTPUT_LOAD); | ||||||
|  |         console.log('  Output Power:', this.activeOIDs.OUTPUT_POWER); | ||||||
|  |         console.log('  Output Voltage:', this.activeOIDs.OUTPUT_VOLTAGE); | ||||||
|  |         console.log('  Output Current:', this.activeOIDs.OUTPUT_CURRENT); | ||||||
|         console.log('---------------------------------------'); |         console.log('---------------------------------------'); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -324,20 +328,65 @@ export class NupstSnmp { | |||||||
|         config, |         config, | ||||||
|       ) || 0; |       ) || 0; | ||||||
|  |  | ||||||
|  |       // Get power draw metrics | ||||||
|  |       const outputLoad = await this.getSNMPValueWithRetry( | ||||||
|  |         this.activeOIDs.OUTPUT_LOAD, | ||||||
|  |         'output load', | ||||||
|  |         config, | ||||||
|  |       ) || 0; | ||||||
|  |       const outputPower = await this.getSNMPValueWithRetry( | ||||||
|  |         this.activeOIDs.OUTPUT_POWER, | ||||||
|  |         'output power', | ||||||
|  |         config, | ||||||
|  |       ) || 0; | ||||||
|  |       const outputVoltage = await this.getSNMPValueWithRetry( | ||||||
|  |         this.activeOIDs.OUTPUT_VOLTAGE, | ||||||
|  |         'output voltage', | ||||||
|  |         config, | ||||||
|  |       ) || 0; | ||||||
|  |       const outputCurrent = await this.getSNMPValueWithRetry( | ||||||
|  |         this.activeOIDs.OUTPUT_CURRENT, | ||||||
|  |         'output current', | ||||||
|  |         config, | ||||||
|  |       ) || 0; | ||||||
|  |  | ||||||
|       // Determine power status - handle different values for different UPS models |       // Determine power status - handle different values for different UPS models | ||||||
|       const powerStatus = this.determinePowerStatus(config.upsModel, powerStatusValue); |       const powerStatus = this.determinePowerStatus(config.upsModel, powerStatusValue); | ||||||
|  |  | ||||||
|       // Convert to minutes for UPS models with different time units |       // Convert to minutes for UPS models with different time units | ||||||
|       const processedRuntime = this.processRuntimeValue(config.upsModel, batteryRuntime); |       const processedRuntime = this.processRuntimeValue(config.upsModel, batteryRuntime); | ||||||
|  |  | ||||||
|  |       // Process power metrics with vendor-specific scaling | ||||||
|  |       const processedVoltage = this.processVoltageValue(config.upsModel, outputVoltage); | ||||||
|  |       const processedCurrent = this.processCurrentValue(config.upsModel, outputCurrent); | ||||||
|  |  | ||||||
|  |       // Calculate power from voltage × current if not provided by UPS | ||||||
|  |       let processedPower = outputPower; | ||||||
|  |       if (outputPower === 0 && processedVoltage > 0 && processedCurrent > 0) { | ||||||
|  |         processedPower = Math.round(processedVoltage * processedCurrent); | ||||||
|  |         if (this.debug) { | ||||||
|  |           console.log( | ||||||
|  |             `Calculated power from V×I: ${processedVoltage}V × ${processedCurrent}A = ${processedPower}W`, | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|       const result = { |       const result = { | ||||||
|         powerStatus, |         powerStatus, | ||||||
|         batteryCapacity, |         batteryCapacity, | ||||||
|         batteryRuntime: processedRuntime, |         batteryRuntime: processedRuntime, | ||||||
|  |         outputLoad, | ||||||
|  |         outputPower: processedPower, | ||||||
|  |         outputVoltage: processedVoltage, | ||||||
|  |         outputCurrent: processedCurrent, | ||||||
|         raw: { |         raw: { | ||||||
|           powerStatus: powerStatusValue, |           powerStatus: powerStatusValue, | ||||||
|           batteryCapacity, |           batteryCapacity, | ||||||
|           batteryRuntime, |           batteryRuntime, | ||||||
|  |           outputLoad, | ||||||
|  |           outputPower, | ||||||
|  |           outputVoltage, | ||||||
|  |           outputCurrent, | ||||||
|         }, |         }, | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
| @@ -347,6 +396,10 @@ export class NupstSnmp { | |||||||
|         console.log('  Power Status:', result.powerStatus); |         console.log('  Power Status:', result.powerStatus); | ||||||
|         console.log('  Battery Capacity:', result.batteryCapacity + '%'); |         console.log('  Battery Capacity:', result.batteryCapacity + '%'); | ||||||
|         console.log('  Battery Runtime:', result.batteryRuntime, 'minutes'); |         console.log('  Battery Runtime:', result.batteryRuntime, 'minutes'); | ||||||
|  |         console.log('  Output Load:', result.outputLoad + '%'); | ||||||
|  |         console.log('  Output Power:', result.outputPower, 'watts'); | ||||||
|  |         console.log('  Output Voltage:', result.outputVoltage, 'volts'); | ||||||
|  |         console.log('  Output Current:', result.outputCurrent, 'amps'); | ||||||
|         console.log('---------------------------------------'); |         console.log('---------------------------------------'); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -602,4 +655,74 @@ export class NupstSnmp { | |||||||
|  |  | ||||||
|     return batteryRuntime; |     return batteryRuntime; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Process voltage value based on UPS model | ||||||
|  |    * @param upsModel UPS model | ||||||
|  |    * @param outputVoltage Raw output voltage value | ||||||
|  |    * @returns Processed voltage in volts | ||||||
|  |    */ | ||||||
|  |   private processVoltageValue( | ||||||
|  |     upsModel: TUpsModel | undefined, | ||||||
|  |     outputVoltage: number, | ||||||
|  |   ): number { | ||||||
|  |     if (this.debug) { | ||||||
|  |       console.log('Raw voltage value:', outputVoltage); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (upsModel === 'cyberpower' && outputVoltage > 0) { | ||||||
|  |       // CyberPower: Voltage is in 0.1V, convert to volts | ||||||
|  |       const volts = outputVoltage / 10; | ||||||
|  |       if (this.debug) { | ||||||
|  |         console.log( | ||||||
|  |           `Converting CyberPower voltage from ${outputVoltage} (0.1V) to ${volts} volts`, | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |       return volts; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return outputVoltage; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Process current value based on UPS model | ||||||
|  |    * @param upsModel UPS model | ||||||
|  |    * @param outputCurrent Raw output current value | ||||||
|  |    * @returns Processed current in amps | ||||||
|  |    */ | ||||||
|  |   private processCurrentValue( | ||||||
|  |     upsModel: TUpsModel | undefined, | ||||||
|  |     outputCurrent: number, | ||||||
|  |   ): number { | ||||||
|  |     if (this.debug) { | ||||||
|  |       console.log('Raw current value:', outputCurrent); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (upsModel === 'cyberpower' && outputCurrent > 0) { | ||||||
|  |       // CyberPower: Current is in 0.1A, convert to amps | ||||||
|  |       const amps = outputCurrent / 10; | ||||||
|  |       if (this.debug) { | ||||||
|  |         console.log( | ||||||
|  |           `Converting CyberPower current from ${outputCurrent} (0.1A) to ${amps} amps`, | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |       return amps; | ||||||
|  |     } else if ((upsModel === 'tripplite' || upsModel === 'liebert') && outputCurrent > 0) { | ||||||
|  |       // RFC 1628 standard: Current is in 0.1A, convert to amps | ||||||
|  |       const amps = outputCurrent / 10; | ||||||
|  |       if (this.debug) { | ||||||
|  |         console.log( | ||||||
|  |           `Converting RFC 1628 current from ${outputCurrent} (0.1A) to ${amps} amps`, | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |       return amps; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Eaton XUPS-MIB and APC PowerNet report current directly in RMS Amps (no scaling needed) | ||||||
|  |     if ((upsModel === 'eaton' || upsModel === 'apc') && this.debug && outputCurrent > 0) { | ||||||
|  |       console.log(`${upsModel.toUpperCase()} current already in RMS Amps: ${outputCurrent}A`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return outputCurrent; | ||||||
|  |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -14,28 +14,40 @@ export class UpsOidSets { | |||||||
|       POWER_STATUS: '1.3.6.1.4.1.3808.1.1.1.4.1.1.0', // upsBaseOutputStatus |       POWER_STATUS: '1.3.6.1.4.1.3808.1.1.1.4.1.1.0', // upsBaseOutputStatus | ||||||
|       BATTERY_CAPACITY: '1.3.6.1.4.1.3808.1.1.1.2.2.1.0', // upsAdvanceBatteryCapacity (percentage) |       BATTERY_CAPACITY: '1.3.6.1.4.1.3808.1.1.1.2.2.1.0', // upsAdvanceBatteryCapacity (percentage) | ||||||
|       BATTERY_RUNTIME: '1.3.6.1.4.1.3808.1.1.1.2.2.4.0', // upsAdvanceBatteryRunTimeRemaining (TimeTicks) |       BATTERY_RUNTIME: '1.3.6.1.4.1.3808.1.1.1.2.2.4.0', // upsAdvanceBatteryRunTimeRemaining (TimeTicks) | ||||||
|  |       OUTPUT_LOAD: '1.3.6.1.4.1.3808.1.1.1.4.2.3.0', // upsAdvanceOutputLoad (percentage) | ||||||
|  |       OUTPUT_POWER: '1.3.6.1.4.1.3808.1.1.1.4.2.5.0', // upsAdvanceOutputPower (watts) | ||||||
|  |       OUTPUT_VOLTAGE: '1.3.6.1.4.1.3808.1.1.1.4.2.1.0', // upsAdvanceOutputVoltage (0.1V scale) | ||||||
|  |       OUTPUT_CURRENT: '1.3.6.1.4.1.3808.1.1.1.4.2.4.0', // upsAdvanceOutputCurrent (0.1A scale) | ||||||
|       POWER_STATUS_VALUES: { |       POWER_STATUS_VALUES: { | ||||||
|         online: 2, // upsBaseOutputStatus: 2=onLine |         online: 2, // upsBaseOutputStatus: 2=onLine | ||||||
|         onBattery: 3, // upsBaseOutputStatus: 3=onBattery |         onBattery: 3, // upsBaseOutputStatus: 3=onBattery | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|     // APC OIDs |     // APC OIDs (PowerNet MIB) | ||||||
|     apc: { |     apc: { | ||||||
|       POWER_STATUS: '1.3.6.1.4.1.318.1.1.1.4.1.1.0', // upsBasicOutputStatus |       POWER_STATUS: '1.3.6.1.4.1.318.1.1.1.4.1.1.0', // upsBasicOutputStatus | ||||||
|       BATTERY_CAPACITY: '1.3.6.1.4.1.318.1.1.1.2.2.1.0', // Battery capacity in percentage |       BATTERY_CAPACITY: '1.3.6.1.4.1.318.1.1.1.2.2.1.0', // Battery capacity in percentage | ||||||
|       BATTERY_RUNTIME: '1.3.6.1.4.1.318.1.1.1.2.2.3.0', // Remaining runtime in minutes |       BATTERY_RUNTIME: '1.3.6.1.4.1.318.1.1.1.2.2.3.0', // Remaining runtime in minutes | ||||||
|  |       OUTPUT_LOAD: '1.3.6.1.4.1.318.1.1.1.4.2.3.0', // upsAdvOutputLoad (percentage) | ||||||
|  |       OUTPUT_POWER: '1.3.6.1.4.1.318.1.1.1.4.2.8.0', // upsAdvOutputActivePower (watts) | ||||||
|  |       OUTPUT_VOLTAGE: '1.3.6.1.4.1.318.1.1.1.4.2.1.0', // upsAdvOutputVoltage | ||||||
|  |       OUTPUT_CURRENT: '1.3.6.1.4.1.318.1.1.1.4.2.4.0', // upsAdvOutputCurrent | ||||||
|       POWER_STATUS_VALUES: { |       POWER_STATUS_VALUES: { | ||||||
|         online: 2, // upsBasicOutputStatus: 2=onLine |         online: 2, // upsBasicOutputStatus: 2=onLine | ||||||
|         onBattery: 3, // upsBasicOutputStatus: 3=onBattery |         onBattery: 3, // upsBasicOutputStatus: 3=onBattery | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|     // Eaton OIDs |     // Eaton OIDs (XUPS-MIB) | ||||||
|     eaton: { |     eaton: { | ||||||
|       POWER_STATUS: '1.3.6.1.4.1.534.1.4.4.0', // xupsOutputSource |       POWER_STATUS: '1.3.6.1.4.1.534.1.4.4.0', // xupsOutputSource | ||||||
|       BATTERY_CAPACITY: '1.3.6.1.4.1.534.1.2.4.0', // xupsBatCapacity (percentage) |       BATTERY_CAPACITY: '1.3.6.1.4.1.534.1.2.4.0', // xupsBatCapacity (percentage) | ||||||
|       BATTERY_RUNTIME: '1.3.6.1.4.1.534.1.2.1.0', // xupsBatTimeRemaining (seconds) |       BATTERY_RUNTIME: '1.3.6.1.4.1.534.1.2.1.0', // xupsBatTimeRemaining (seconds) | ||||||
|  |       OUTPUT_LOAD: '1.3.6.1.4.1.534.1.4.4.1.8.1', // xupsOutputPercentLoad (phase 1) | ||||||
|  |       OUTPUT_POWER: '1.3.6.1.4.1.534.1.4.4.1.4.1', // xupsOutputWatts (phase 1) | ||||||
|  |       OUTPUT_VOLTAGE: '1.3.6.1.4.1.534.1.4.4.1.2.1', // xupsOutputVoltage (phase 1) | ||||||
|  |       OUTPUT_CURRENT: '1.3.6.1.4.1.534.1.4.4.1.3.1', // xupsOutputCurrent (phase 1) | ||||||
|       POWER_STATUS_VALUES: { |       POWER_STATUS_VALUES: { | ||||||
|         online: 3, // xupsOutputSource: 3=normal (mains power) |         online: 3, // xupsOutputSource: 3=normal (mains power) | ||||||
|         onBattery: 5, // xupsOutputSource: 5=battery |         onBattery: 5, // xupsOutputSource: 5=battery | ||||||
| @@ -47,6 +59,10 @@ export class UpsOidSets { | |||||||
|       POWER_STATUS: '1.3.6.1.4.1.850.1.1.3.1.1.1.0', // tlUpsOutputSource |       POWER_STATUS: '1.3.6.1.4.1.850.1.1.3.1.1.1.0', // tlUpsOutputSource | ||||||
|       BATTERY_CAPACITY: '1.3.6.1.4.1.850.1.1.3.2.4.1.0', // Battery capacity in percentage |       BATTERY_CAPACITY: '1.3.6.1.4.1.850.1.1.3.2.4.1.0', // Battery capacity in percentage | ||||||
|       BATTERY_RUNTIME: '1.3.6.1.4.1.850.1.1.3.2.2.1.0', // Remaining runtime in minutes |       BATTERY_RUNTIME: '1.3.6.1.4.1.850.1.1.3.2.2.1.0', // Remaining runtime in minutes | ||||||
|  |       OUTPUT_LOAD: '1.3.6.1.2.1.33.1.4.4.1.5.1', // RFC 1628: upsOutputPercentLoad | ||||||
|  |       OUTPUT_POWER: '1.3.6.1.2.1.33.1.4.4.1.4.1', // RFC 1628: upsOutputPower (watts) | ||||||
|  |       OUTPUT_VOLTAGE: '1.3.6.1.2.1.33.1.4.4.1.2.1', // RFC 1628: upsOutputVoltage | ||||||
|  |       OUTPUT_CURRENT: '1.3.6.1.2.1.33.1.4.4.1.3.1', // RFC 1628: upsOutputCurrent (0.1A scale) | ||||||
|       POWER_STATUS_VALUES: { |       POWER_STATUS_VALUES: { | ||||||
|         online: 2, // tlUpsOutputSource: 2=normal (mains power) |         online: 2, // tlUpsOutputSource: 2=normal (mains power) | ||||||
|         onBattery: 3, // tlUpsOutputSource: 3=onBattery |         onBattery: 3, // tlUpsOutputSource: 3=onBattery | ||||||
| @@ -58,6 +74,10 @@ export class UpsOidSets { | |||||||
|       POWER_STATUS: '1.3.6.1.4.1.476.1.42.3.9.20.1.20.1.2.1.2.1', // lgpPwrOutputSource |       POWER_STATUS: '1.3.6.1.4.1.476.1.42.3.9.20.1.20.1.2.1.2.1', // lgpPwrOutputSource | ||||||
|       BATTERY_CAPACITY: '1.3.6.1.4.1.476.1.42.3.9.20.1.20.1.2.1.4.1', // Battery capacity in percentage |       BATTERY_CAPACITY: '1.3.6.1.4.1.476.1.42.3.9.20.1.20.1.2.1.4.1', // Battery capacity in percentage | ||||||
|       BATTERY_RUNTIME: '1.3.6.1.4.1.476.1.42.3.9.20.1.20.1.2.1.5.1', // Remaining runtime in minutes |       BATTERY_RUNTIME: '1.3.6.1.4.1.476.1.42.3.9.20.1.20.1.2.1.5.1', // Remaining runtime in minutes | ||||||
|  |       OUTPUT_LOAD: '1.3.6.1.2.1.33.1.4.4.1.5.1', // RFC 1628: upsOutputPercentLoad | ||||||
|  |       OUTPUT_POWER: '1.3.6.1.2.1.33.1.4.4.1.4.1', // RFC 1628: upsOutputPower (watts) | ||||||
|  |       OUTPUT_VOLTAGE: '1.3.6.1.2.1.33.1.4.4.1.2.1', // RFC 1628: upsOutputVoltage | ||||||
|  |       OUTPUT_CURRENT: '1.3.6.1.2.1.33.1.4.4.1.3.1', // RFC 1628: upsOutputCurrent (0.1A scale) | ||||||
|       POWER_STATUS_VALUES: { |       POWER_STATUS_VALUES: { | ||||||
|         online: 2, // lgpPwrOutputSource: 2=normal (mains power) |         online: 2, // lgpPwrOutputSource: 2=normal (mains power) | ||||||
|         onBattery: 3, // lgpPwrOutputSource: 3=onBattery |         onBattery: 3, // lgpPwrOutputSource: 3=onBattery | ||||||
| @@ -69,6 +89,10 @@ export class UpsOidSets { | |||||||
|       POWER_STATUS: '', |       POWER_STATUS: '', | ||||||
|       BATTERY_CAPACITY: '', |       BATTERY_CAPACITY: '', | ||||||
|       BATTERY_RUNTIME: '', |       BATTERY_RUNTIME: '', | ||||||
|  |       OUTPUT_LOAD: '', | ||||||
|  |       OUTPUT_POWER: '', | ||||||
|  |       OUTPUT_VOLTAGE: '', | ||||||
|  |       OUTPUT_CURRENT: '', | ||||||
|     }, |     }, | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
| @@ -90,6 +114,10 @@ export class UpsOidSets { | |||||||
|       'power status': '1.3.6.1.2.1.33.1.4.1.0', // upsOutputSource |       'power status': '1.3.6.1.2.1.33.1.4.1.0', // upsOutputSource | ||||||
|       'battery capacity': '1.3.6.1.2.1.33.1.2.4.0', // upsEstimatedChargeRemaining |       'battery capacity': '1.3.6.1.2.1.33.1.2.4.0', // upsEstimatedChargeRemaining | ||||||
|       'battery runtime': '1.3.6.1.2.1.33.1.2.3.0', // upsEstimatedMinutesRemaining |       'battery runtime': '1.3.6.1.2.1.33.1.2.3.0', // upsEstimatedMinutesRemaining | ||||||
|  |       'output load': '1.3.6.1.2.1.33.1.4.4.1.5.1', // upsOutputPercentLoad (indexed by line) | ||||||
|  |       'output power': '1.3.6.1.2.1.33.1.4.4.1.4.1', // upsOutputPower in watts (indexed by line) | ||||||
|  |       'output voltage': '1.3.6.1.2.1.33.1.4.4.1.2.1', // upsOutputVoltage (indexed by line) | ||||||
|  |       'output current': '1.3.6.1.2.1.33.1.4.4.1.3.1', // upsOutputCurrent in 0.1A (indexed by line) | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -14,6 +14,14 @@ export interface IUpsStatus { | |||||||
|   batteryCapacity: number; |   batteryCapacity: number; | ||||||
|   /** Remaining runtime in minutes */ |   /** Remaining runtime in minutes */ | ||||||
|   batteryRuntime: number; |   batteryRuntime: number; | ||||||
|  |   /** Output load percentage (0-100) */ | ||||||
|  |   outputLoad: number; | ||||||
|  |   /** Output power in watts */ | ||||||
|  |   outputPower: number; | ||||||
|  |   /** Output voltage in volts */ | ||||||
|  |   outputVoltage: number; | ||||||
|  |   /** Output current in amps */ | ||||||
|  |   outputCurrent: number; | ||||||
|   /** Raw values from SNMP responses */ |   /** Raw values from SNMP responses */ | ||||||
|   raw: Record<string, any>; |   raw: Record<string, any>; | ||||||
| } | } | ||||||
| @@ -28,6 +36,14 @@ export interface IOidSet { | |||||||
|   BATTERY_CAPACITY: string; |   BATTERY_CAPACITY: string; | ||||||
|   /** OID for battery runtime */ |   /** OID for battery runtime */ | ||||||
|   BATTERY_RUNTIME: string; |   BATTERY_RUNTIME: string; | ||||||
|  |   /** OID for output load percentage */ | ||||||
|  |   OUTPUT_LOAD: string; | ||||||
|  |   /** OID for output power in watts */ | ||||||
|  |   OUTPUT_POWER: string; | ||||||
|  |   /** OID for output voltage */ | ||||||
|  |   OUTPUT_VOLTAGE: string; | ||||||
|  |   /** OID for output current */ | ||||||
|  |   OUTPUT_CURRENT: string; | ||||||
|   /** Power status value mappings */ |   /** Power status value mappings */ | ||||||
|   POWER_STATUS_VALUES?: { |   POWER_STATUS_VALUES?: { | ||||||
|     /** SNMP value that indicates UPS is online (on AC power) */ |     /** SNMP value that indicates UPS is online (on AC power) */ | ||||||
|   | |||||||
| @@ -277,6 +277,9 @@ WantedBy=multi-user.target | |||||||
|         for (const ups of config.upsDevices) { |         for (const ups of config.upsDevices) { | ||||||
|           await this.displaySingleUpsStatus(ups, snmp); |           await this.displaySingleUpsStatus(ups, snmp); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         // Display groups after UPS devices | ||||||
|  |         this.displayGroupsStatus(); | ||||||
|       } else if (config.snmp) { |       } else if (config.snmp) { | ||||||
|         // Legacy single UPS configuration (v1/v2 format) |         // Legacy single UPS configuration (v1/v2 format) | ||||||
|         logger.info('UPS Devices (1):'); |         logger.info('UPS Devices (1):'); | ||||||
| @@ -352,6 +355,9 @@ WantedBy=multi-user.target | |||||||
|  |  | ||||||
|       logger.log(`    Battery: ${batteryColor(status.batteryCapacity + '%')} ${batterySymbol}  Runtime: ${getRuntimeColor(status.batteryRuntime)(status.batteryRuntime + ' min')}`); |       logger.log(`    Battery: ${batteryColor(status.batteryCapacity + '%')} ${batterySymbol}  Runtime: ${getRuntimeColor(status.batteryRuntime)(status.batteryRuntime + ' min')}`); | ||||||
|  |  | ||||||
|  |       // Display power metrics | ||||||
|  |       logger.log(`    Load: ${theme.highlight(status.outputLoad + '%')}  Power: ${theme.highlight(status.outputPower + 'W')}  Voltage: ${theme.highlight(status.outputVoltage + 'V')}  Current: ${theme.highlight(status.outputCurrent + 'A')}`); | ||||||
|  |  | ||||||
|       // Display host info |       // Display host info | ||||||
|       logger.log(`    ${theme.dim(`Host: ${ups.snmp.host}:${ups.snmp.port}`)}`); |       logger.log(`    ${theme.dim(`Host: ${ups.snmp.host}:${ups.snmp.port}`)}`); | ||||||
|  |  | ||||||
| @@ -365,6 +371,27 @@ WantedBy=multi-user.target | |||||||
|         logger.log(`    ${theme.dim(`Groups: ${groupNames.join(', ')}`)}`); |         logger.log(`    ${theme.dim(`Groups: ${groupNames.join(', ')}`)}`); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       // Display actions if any | ||||||
|  |       if (ups.actions && ups.actions.length > 0) { | ||||||
|  |         for (const action of ups.actions) { | ||||||
|  |           let actionDesc = `${action.type}`; | ||||||
|  |           if (action.thresholds) { | ||||||
|  |             actionDesc += ` (${action.triggerMode || 'onlyThresholds'}: battery<${action.thresholds.battery}%, runtime<${action.thresholds.runtime}min`; | ||||||
|  |             if (action.shutdownDelay) { | ||||||
|  |               actionDesc += `, delay=${action.shutdownDelay}s`; | ||||||
|  |             } | ||||||
|  |             actionDesc += ')'; | ||||||
|  |           } else { | ||||||
|  |             actionDesc += ` (${action.triggerMode || 'onlyPowerChanges'}`; | ||||||
|  |             if (action.shutdownDelay) { | ||||||
|  |               actionDesc += `, delay=${action.shutdownDelay}s`; | ||||||
|  |             } | ||||||
|  |             actionDesc += ')'; | ||||||
|  |           } | ||||||
|  |           logger.log(`    ${theme.dim('Action:')} ${theme.info(actionDesc)}`); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|       logger.log(''); |       logger.log(''); | ||||||
|  |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
| @@ -376,6 +403,69 @@ WantedBy=multi-user.target | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Display status of all groups | ||||||
|  |    * @private | ||||||
|  |    */ | ||||||
|  |   private displayGroupsStatus(): void { | ||||||
|  |     const config = this.daemon.getConfig(); | ||||||
|  |  | ||||||
|  |     if (!config.groups || config.groups.length === 0) { | ||||||
|  |       return; // No groups to display | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     logger.log(''); | ||||||
|  |     logger.info(`Groups (${config.groups.length}):`); | ||||||
|  |  | ||||||
|  |     for (const group of config.groups) { | ||||||
|  |       // Display group name and mode | ||||||
|  |       const modeColor = group.mode === 'redundant' ? theme.success : theme.warning; | ||||||
|  |       logger.log( | ||||||
|  |         `  ${symbols.info} ${theme.highlight(group.name)} ${theme.dim(`(${modeColor(group.mode)})`)}`, | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       // Display description if present | ||||||
|  |       if (group.description) { | ||||||
|  |         logger.log(`    ${theme.dim(group.description)}`); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Display UPS devices in this group | ||||||
|  |       const upsInGroup = config.upsDevices.filter((ups) => | ||||||
|  |         ups.groups && ups.groups.includes(group.id) | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       if (upsInGroup.length > 0) { | ||||||
|  |         const upsNames = upsInGroup.map((ups) => ups.name).join(', '); | ||||||
|  |         logger.log(`    ${theme.dim(`UPS Devices (${upsInGroup.length}):`)} ${upsNames}`); | ||||||
|  |       } else { | ||||||
|  |         logger.log(`    ${theme.dim('UPS Devices: None')}`); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Display actions if any | ||||||
|  |       if (group.actions && group.actions.length > 0) { | ||||||
|  |         for (const action of group.actions) { | ||||||
|  |           let actionDesc = `${action.type}`; | ||||||
|  |           if (action.thresholds) { | ||||||
|  |             actionDesc += ` (${action.triggerMode || 'onlyThresholds'}: battery<${action.thresholds.battery}%, runtime<${action.thresholds.runtime}min`; | ||||||
|  |             if (action.shutdownDelay) { | ||||||
|  |               actionDesc += `, delay=${action.shutdownDelay}s`; | ||||||
|  |             } | ||||||
|  |             actionDesc += ')'; | ||||||
|  |           } else { | ||||||
|  |             actionDesc += ` (${action.triggerMode || 'onlyPowerChanges'}`; | ||||||
|  |             if (action.shutdownDelay) { | ||||||
|  |               actionDesc += `, delay=${action.shutdownDelay}s`; | ||||||
|  |             } | ||||||
|  |             actionDesc += ')'; | ||||||
|  |           } | ||||||
|  |           logger.log(`    ${theme.dim('Action:')} ${theme.info(actionDesc)}`); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       logger.log(''); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Disable and uninstall the systemd service |    * Disable and uninstall the systemd service | ||||||
|    * @throws Error if disabling fails |    * @throws Error if disabling fails | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user