Compare commits
	
		
			30 Commits
		
	
	
		
			v2.5.2
			...
			f860f39e59
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f860f39e59 | |||
| fa4516de3b | |||
| 539547beb8 | |||
| 6eb92959ec | |||
| 4af9af0845 | |||
| f7e12cdcbb | |||
| 002498b91b | |||
| 459911fe5f | |||
| 9859a02ea2 | |||
| 65444b6d25 | |||
| d049e8741f | |||
| 1123a99aea | |||
| d01e878310 | |||
| 588aeabf4b | |||
| 87005e72f1 | |||
| f799c2ee66 | |||
| 1a029ba493 | |||
| 5b756dd223 | |||
| 4cac599a58 | |||
| be6a7314c3 | |||
| 83ba9c2611 | |||
| 22ab472e58 | |||
| 9a77030377 | |||
| ceff285ff5 | |||
| d8bfbf0be3 | |||
| 3e6b883b38 | |||
| 47ef918128 | |||
| 5951638967 | |||
| b06e2b2273 | |||
| cc1cfe894c | 
							
								
								
									
										99
									
								
								changelog.md
									
									
									
									
									
								
							
							
						
						
									
										99
									
								
								changelog.md
									
									
									
									
									
								
							| @@ -1,5 +1,104 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.14 - fix(systemd) | ||||||
|  | Shorten closing log divider in systemd service installation output for consistent formatting. | ||||||
|  |  | ||||||
|  | - Replaced the overly long footer with a shorter one in ts/systemd.ts. | ||||||
|  | - This change improves log readability without affecting functionality. | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.13 - fix(cli) | ||||||
|  | Fix CLI update output box formatting | ||||||
|  |  | ||||||
|  | - Adjusted the closing box line in the update process log messages for consistent visual formatting | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.12 - fix(systemd) | ||||||
|  | Adjust logging border in systemd service installation output | ||||||
|  |  | ||||||
|  | - Updated the closing border line for consistent output formatting in ts/systemd.ts | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.11 - fix(cli, systemd) | ||||||
|  | Adjust log formatting for consistent output in CLI and systemd commands | ||||||
|  |  | ||||||
|  | - Fixed spacing issues in service installation and status log messages in the systemd module. | ||||||
|  | - Revised output formatting in the CLI to improve message clarity. | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.10 - fix(daemon) | ||||||
|  | Adjust console log box formatting for consistent output in daemon status messages | ||||||
|  |  | ||||||
|  | - Updated closing box borders to align properly in configuration error, periodic updates, and UPS status logs | ||||||
|  | - Improved visual consistency in log messages | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.9 - fix(cli) | ||||||
|  | Improve console output formatting for status banners and logging messages | ||||||
|  |  | ||||||
|  | - Standardize banner messages in daemon status updates | ||||||
|  | - Refine version information banner in nupst logging | ||||||
|  | - Update UPS connection and status banners in systemd | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.8 - fix(cli) | ||||||
|  | Improve CLI formatting, refine debug option filtering, and remove unused dgram import in SNMP manager | ||||||
|  |  | ||||||
|  | - Standardize whitespace and formatting in ts/cli.ts for consistency | ||||||
|  | - Refine argument filtering for debug mode and prompt messages | ||||||
|  | - Remove unused 'dgram' import from ts/snmp/manager.ts | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.7 - fix(setup.sh) | ||||||
|  | Clarify net-snmp dependency installation message in setup.sh | ||||||
|  |  | ||||||
|  | - Updated echo statement to indicate installation of net-snmp along with 2 subdependencies | ||||||
|  | - Improves clarity on dependency installation during setup | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.6 - fix(setup.sh) | ||||||
|  | Improve setup script to detect and execute npm-cli.js directly using the Node.js binary | ||||||
|  |  | ||||||
|  | - Replace use of the npm binary with direct execution of npm-cli.js | ||||||
|  | - Add fallback logic to locate npm-cli.js when not found at the expected path | ||||||
|  | - Simplify cleanup by removing unnecessary PATH modifications | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.5 - fix(daemon, setup) | ||||||
|  | Improve shutdown command detection and fallback logic; update setup script to use absolute Node/npm paths | ||||||
|  |  | ||||||
|  | - Use execFileAsync to execute shutdown commands reliably | ||||||
|  | - Add multiple fallback alternatives for shutdown and emergency shutdown handling | ||||||
|  | - Update setup.sh to log the Node and NPM versions using absolute paths without modifying PATH | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.4 - fix(setup) | ||||||
|  | Improve installation process in setup script by cleaning up package files and ensuring a minimal net-snmp dependency installation. | ||||||
|  |  | ||||||
|  | - Remove existing package-lock.json along with node_modules to prevent stale artifacts. | ||||||
|  | - Back up the original package.json before modifying it. | ||||||
|  | - Create a minimal package.json with only the net-snmp dependency based on the backed-up version. | ||||||
|  | - Use a clean install to guarantee that only net-snmp is installed. | ||||||
|  | - Restore the original package.json if the installation fails. | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.3 - fix(setup) | ||||||
|  | Update setup script to install only net-snmp dependency and create a minimal package-lock.json for better dependency control. | ||||||
|  |  | ||||||
|  | - Removed full production dependency install in favor of installing only net-snmp@3.20.0 | ||||||
|  | - Added verification step to confirm net-snmp installation | ||||||
|  | - Generate a minimal package-lock.json if one does not exist | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.2 - fix(setup/readme) | ||||||
|  | Improve force update instructions and dependency installation process in setup.sh and readme.md | ||||||
|  |  | ||||||
|  | - Clarify force update commands with explicit paths in readme.md | ||||||
|  | - Remove existing node_modules before installing dependencies in setup.sh | ||||||
|  | - Switch from 'npm ci --only=production' to 'npm install --omit=dev' with updated error instructions | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.1 - fix(setup) | ||||||
|  | Update setup.sh to temporarily add vendor Node.js binary to PATH for dependency installation, log Node and npm versions, and restore the original PATH afterwards. | ||||||
|  |  | ||||||
|  | - Temporarily prepend vendor Node.js binary directory to PATH to ensure proper npm execution. | ||||||
|  | - Log Node.js and npm versions for debugging purposes. | ||||||
|  | - Restore the original PATH after installing dependencies. | ||||||
|  |  | ||||||
|  | ## 2025-03-26 - 2.6.0 - feat(setup) | ||||||
|  | Add --force update flag to setup script and update installation instructions | ||||||
|  |  | ||||||
|  | - Implemented --force option in setup.sh to force-update Node.js binary and dependencies | ||||||
|  | - Updated readme.md to document the --force flag and revised update steps | ||||||
|  | - Modified ts/cli.ts update command to pass the --force flag to setup.sh | ||||||
|  |  | ||||||
| ## 2025-03-26 - 2.5.2 - fix(installer) | ## 2025-03-26 - 2.5.2 - fix(installer) | ||||||
| Improve Node.js binary detection, dependency management, and SNMPv3 fallback logic | Improve Node.js binary detection, dependency management, and SNMPv3 fallback logic | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@serve.zone/nupst", |   "name": "@serve.zone/nupst", | ||||||
|   "version": "2.5.2", |   "version": "2.6.14", | ||||||
|   "description": "Node.js UPS Shutdown Tool for SNMP-enabled UPS devices", |   "description": "Node.js UPS Shutdown Tool for SNMP-enabled UPS devices", | ||||||
|   "main": "dist/index.js", |   "main": "dist/index.js", | ||||||
|   "bin": { |   "bin": { | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							| @@ -9,7 +9,7 @@ importers: | |||||||
|   .: |   .: | ||||||
|     dependencies: |     dependencies: | ||||||
|       net-snmp: |       net-snmp: | ||||||
|         specifier: ^3.20.0 |         specifier: 3.20.0 | ||||||
|         version: 3.20.0 |         version: 3.20.0 | ||||||
|     devDependencies: |     devDependencies: | ||||||
|       '@git.zone/tsbuild': |       '@git.zone/tsbuild': | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								readme.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								readme.md
									
									
									
									
									
								
							| @@ -227,8 +227,19 @@ sudo nupst update | |||||||
| This will: | This will: | ||||||
| 1. Pull the latest changes from the git repository | 1. Pull the latest changes from the git repository | ||||||
| 2. Run the installation scripts | 2. Run the installation scripts | ||||||
| 3. Refresh the systemd service configuration | 3. Force-update Node.js and all dependencies, even if they already exist | ||||||
| 4. Restart the service if it was running | 4. Refresh the systemd service configuration | ||||||
|  | 5. Restart the service if it was running | ||||||
|  |  | ||||||
|  | You can also manually run the setup script with the force flag to update Node.js and dependencies without updating the application code: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # If you're in the nupst directory: | ||||||
|  | bash ./setup.sh --force | ||||||
|  |  | ||||||
|  | # If you're in another directory, specify the full path: | ||||||
|  | bash /opt/nupst/setup.sh --force | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## Security | ## Security | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										108
									
								
								setup.sh
									
									
									
									
									
								
							
							
						
						
									
										108
									
								
								setup.sh
									
									
									
									
									
								
							| @@ -2,6 +2,22 @@ | |||||||
|  |  | ||||||
| # NUPST Setup Script | # NUPST Setup Script | ||||||
| # Downloads the appropriate Node.js binary for the current platform | # Downloads the appropriate Node.js binary for the current platform | ||||||
|  | # and installs production dependencies | ||||||
|  |  | ||||||
|  | # Parse command line arguments | ||||||
|  | FORCE_UPDATE=0 | ||||||
|  |  | ||||||
|  | for arg in "$@"; do | ||||||
|  |   case $arg in | ||||||
|  |     --force|-f) | ||||||
|  |       FORCE_UPDATE=1 | ||||||
|  |       shift | ||||||
|  |       ;; | ||||||
|  |     *) | ||||||
|  |       # Unknown option | ||||||
|  |       ;; | ||||||
|  |   esac | ||||||
|  | done | ||||||
|  |  | ||||||
| # Find the directory where this script is located | # Find the directory where this script is located | ||||||
| SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )" | SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )" | ||||||
| @@ -74,8 +90,9 @@ case "$OS" in | |||||||
| esac | esac | ||||||
|  |  | ||||||
| # Check if we already have the Node.js binary | # Check if we already have the Node.js binary | ||||||
| if [ -f "$SCRIPT_DIR/vendor/$NODE_DIR/bin/node" ]; then | if [ -f "$SCRIPT_DIR/vendor/$NODE_DIR/bin/node" ] && [ $FORCE_UPDATE -eq 0 ]; then | ||||||
|   echo "Node.js binary already exists for $OS-$ARCH. Skipping download." |   echo "Node.js binary already exists for $OS-$ARCH. Skipping download." | ||||||
|  |   echo "Use --force or -f to force update Node.js." | ||||||
| else | else | ||||||
|   echo "Downloading Node.js v$NODE_VERSION for $OS-$ARCH..." |   echo "Downloading Node.js v$NODE_VERSION for $OS-$ARCH..." | ||||||
|    |    | ||||||
| @@ -222,25 +239,90 @@ echo "dist_ts directory successfully downloaded from npm registry." | |||||||
| # Make launcher script executable | # Make launcher script executable | ||||||
| chmod +x "$SCRIPT_DIR/bin/nupst" | chmod +x "$SCRIPT_DIR/bin/nupst" | ||||||
|  |  | ||||||
| # Install production dependencies | # Set up Node.js binary path | ||||||
| echo "Installing production dependencies..." | NODE_BIN_DIR="$SCRIPT_DIR/vendor/$NODE_DIR/bin" | ||||||
| "$SCRIPT_DIR/vendor/$NODE_DIR/bin/npm" --prefix "$SCRIPT_DIR" ci --only=production --no-audit --no-fund | NODE_BIN="$NODE_BIN_DIR/node" | ||||||
|  | NPM_CLI_JS="$NODE_BIN_DIR/../lib/node_modules/npm/bin/npm-cli.js" | ||||||
|  |  | ||||||
| if [ $? -ne 0 ]; then | # Ensure we have executable permissions | ||||||
|   echo "Warning: Failed to install dependencies with 'npm ci'. Trying 'npm install'..." | chmod +x "$NODE_BIN" | ||||||
|   "$SCRIPT_DIR/vendor/$NODE_DIR/bin/npm" --prefix "$SCRIPT_DIR" install --only=production --no-audit --no-fund |  | ||||||
|  |  | ||||||
|   if [ $? -ne 0 ]; then | # Make sure the npm-cli.js exists | ||||||
|     echo "Error: Failed to install dependencies. NUPST may not function correctly." | if [ ! -f "$NPM_CLI_JS" ]; then | ||||||
|     echo "You can try to install dependencies manually by running:" |   # Try to find npm-cli.js | ||||||
|     echo "cd $SCRIPT_DIR && npm install --only=production" |   NPM_CLI_JS=$(find "$NODE_BIN_DIR/.." -name "npm-cli.js" | head -1) | ||||||
|  |    | ||||||
|  |   if [ -z "$NPM_CLI_JS" ]; then | ||||||
|  |     echo "Warning: Could not find npm-cli.js, npm commands may fail" | ||||||
|  |     # Set to a fallback value so code can continue | ||||||
|  |     NPM_CLI_JS="$NODE_BIN_DIR/npm" | ||||||
|   else |   else | ||||||
|     echo "Dependencies installed successfully with 'npm install'." |     echo "Found npm-cli.js at: $NPM_CLI_JS" | ||||||
|   fi |   fi | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | # Display which binaries we're using | ||||||
|  | echo "Using Node binary: $NODE_BIN" | ||||||
|  | echo "Using NPM CLI JS: $NPM_CLI_JS" | ||||||
|  |  | ||||||
|  | # Remove existing node_modules directory and package files | ||||||
|  | echo "Cleaning up existing installation..." | ||||||
|  | rm -rf "$SCRIPT_DIR/node_modules" | ||||||
|  | rm -f "$SCRIPT_DIR/package-lock.json" | ||||||
|  |  | ||||||
|  | # Back up existing package.json if it exists | ||||||
|  | if [ -f "$SCRIPT_DIR/package.json" ]; then | ||||||
|  |   echo "Backing up existing package.json..." | ||||||
|  |   cp "$SCRIPT_DIR/package.json" "$SCRIPT_DIR/package.json.bak" | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | # Create a clean minimal package.json with ONLY net-snmp dependency | ||||||
|  | echo "Creating minimal package.json with only net-snmp dependency..." | ||||||
|  | VERSION=$(grep -o '"version": "[^"]*"' "$SCRIPT_DIR/package.json.bak" | head -1 | cut -d'"' -f4 || echo "2.6.3") | ||||||
|  | echo '{ | ||||||
|  |   "name": "@serve.zone/nupst", | ||||||
|  |   "version": "'$VERSION'", | ||||||
|  |   "description": "Node.js UPS Shutdown Tool for SNMP-enabled UPS devices", | ||||||
|  |   "main": "dist_ts/index.js", | ||||||
|  |   "type": "module", | ||||||
|  |   "bin": { | ||||||
|  |     "nupst": "bin/nupst" | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "net-snmp": "3.20.0" | ||||||
|  |   }, | ||||||
|  |   "engines": { | ||||||
|  |     "node": ">=16.0.0" | ||||||
|  |   }, | ||||||
|  |   "private": true | ||||||
|  | }' > "$SCRIPT_DIR/package.json" | ||||||
|  |  | ||||||
|  | # Install ONLY net-snmp | ||||||
|  | echo "Installing ONLY net-snmp dependency (+ 2 subdependencies)..." | ||||||
|  | echo "Node version: $("$NODE_BIN" --version)" | ||||||
|  | echo "Executing NPM directly with Node.js" | ||||||
|  |  | ||||||
|  | # Execute npm-cli.js directly with our Node.js binary | ||||||
|  | "$NODE_BIN" "$NPM_CLI_JS" --prefix "$SCRIPT_DIR" install --no-audit --no-fund | ||||||
|  |  | ||||||
|  | INSTALL_STATUS=$? | ||||||
|  | if [ $INSTALL_STATUS -ne 0 ]; then | ||||||
|  |   echo "Error: Failed to install net-snmp dependency. NUPST may not function correctly." | ||||||
|  |   echo "Restoring original package.json..." | ||||||
|  |   mv "$SCRIPT_DIR/package.json.bak" "$SCRIPT_DIR/package.json" | ||||||
|  |   exit 1 | ||||||
| else | else | ||||||
|   echo "Dependencies installed successfully with 'npm ci'." |   echo "net-snmp dependency installed successfully." | ||||||
|  |   # Show what's actually installed | ||||||
|  |   echo "Installed modules:" | ||||||
|  |   find "$SCRIPT_DIR/node_modules" -maxdepth 1 -type d | grep -v "^$SCRIPT_DIR/node_modules$" | sort | ||||||
|  |    | ||||||
|  |   # Remove backup if successful | ||||||
|  |   rm -f "$SCRIPT_DIR/package.json.bak" | ||||||
| fi | fi | ||||||
|  |  | ||||||
|  | # No temporary files to clean up | ||||||
|  |  | ||||||
| echo "NUPST setup completed successfully." | echo "NUPST setup completed successfully." | ||||||
| echo "You can now run NUPST using: $SCRIPT_DIR/bin/nupst" | echo "You can now run NUPST using: $SCRIPT_DIR/bin/nupst" | ||||||
| echo "To install NUPST globally, run: sudo ln -s $SCRIPT_DIR/bin/nupst /usr/local/bin/nupst" | echo "To install NUPST globally, run: sudo ln -s $SCRIPT_DIR/bin/nupst /usr/local/bin/nupst" | ||||||
|   | |||||||
| @@ -3,6 +3,6 @@ | |||||||
|  */ |  */ | ||||||
| export const commitinfo = { | export const commitinfo = { | ||||||
|   name: '@serve.zone/nupst', |   name: '@serve.zone/nupst', | ||||||
|   version: '2.5.2', |   version: '2.6.14', | ||||||
|   description: 'Node.js UPS Shutdown Tool for SNMP-enabled UPS devices' |   description: 'Node.js UPS Shutdown Tool for SNMP-enabled UPS devices' | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										220
									
								
								ts/cli.ts
									
									
									
									
									
								
							
							
						
						
									
										220
									
								
								ts/cli.ts
									
									
									
									
									
								
							| @@ -46,7 +46,7 @@ export class NupstCli { | |||||||
|   private extractDebugOptions(args: string[]): { debugMode: boolean; cleanedArgs: string[] } { |   private extractDebugOptions(args: string[]): { debugMode: boolean; cleanedArgs: string[] } { | ||||||
|     const debugMode = args.includes('--debug') || args.includes('-d'); |     const debugMode = args.includes('--debug') || args.includes('-d'); | ||||||
|     // Remove debug flags from args |     // Remove debug flags from args | ||||||
|     const cleanedArgs = args.filter(arg => arg !== '--debug' && arg !== '-d'); |     const cleanedArgs = args.filter((arg) => arg !== '--debug' && arg !== '-d'); | ||||||
|  |  | ||||||
|     return { debugMode, cleanedArgs }; |     return { debugMode, cleanedArgs }; | ||||||
|   } |   } | ||||||
| @@ -151,7 +151,7 @@ export class NupstCli { | |||||||
|       console.log('Tailing nupst service logs (Ctrl+C to exit)...\n'); |       console.log('Tailing nupst service logs (Ctrl+C to exit)...\n'); | ||||||
|  |  | ||||||
|       const journalctl = spawn('journalctl', ['-u', 'nupst.service', '-n', '50', '-f'], { |       const journalctl = spawn('journalctl', ['-u', 'nupst.service', '-n', '50', '-f'], { | ||||||
|         stdio: ['ignore', 'inherit', 'inherit'] |         stdio: ['ignore', 'inherit', 'inherit'], | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       // Forward signals to child process |       // Forward signals to child process | ||||||
| @@ -236,7 +236,7 @@ export class NupstCli { | |||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         console.error('┌─ Configuration Error ─────────────────────┐'); |         console.error('┌─ Configuration Error ─────────────────────┐'); | ||||||
|         console.error('│ No configuration found.'); |         console.error('│ No configuration found.'); | ||||||
|         console.error('│ Please run \'nupst setup\' first to create a configuration.'); |         console.error("│ Please run 'nupst setup' first to create a configuration."); | ||||||
|         console.error('└──────────────────────────────────────────┘'); |         console.error('└──────────────────────────────────────────┘'); | ||||||
|         return; |         return; | ||||||
|       } |       } | ||||||
| @@ -306,7 +306,7 @@ export class NupstCli { | |||||||
|       // Create a test config with a short timeout |       // Create a test config with a short timeout | ||||||
|       const testConfig = { |       const testConfig = { | ||||||
|         ...config.snmp, |         ...config.snmp, | ||||||
|         timeout: Math.min(config.snmp.timeout, 10000) // Use at most 10 seconds for testing |         timeout: Math.min(config.snmp.timeout, 10000), // Use at most 10 seconds for testing | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       const status = await this.nupst.getSnmp().getUpsStatus(testConfig); |       const status = await this.nupst.getSnmp().getUpsStatus(testConfig); | ||||||
| @@ -326,7 +326,7 @@ export class NupstCli { | |||||||
|       console.error('┌─ Connection Failed! ───────────────────────┐'); |       console.error('┌─ Connection Failed! ───────────────────────┐'); | ||||||
|       console.error(`│ Error: ${error.message}`); |       console.error(`│ Error: ${error.message}`); | ||||||
|       console.error('└──────────────────────────────────────────┘'); |       console.error('└──────────────────────────────────────────┘'); | ||||||
|       console.log('\nPlease check your settings and run \'nupst setup\' to reconfigure.'); |       console.log("\nPlease check your settings and run 'nupst setup' to reconfigure."); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -340,20 +340,28 @@ export class NupstCli { | |||||||
|  |  | ||||||
|     if (status.batteryCapacity < config.thresholds.battery) { |     if (status.batteryCapacity < config.thresholds.battery) { | ||||||
|       console.log('│ ⚠️ WARNING: Battery capacity below threshold'); |       console.log('│ ⚠️ WARNING: Battery capacity below threshold'); | ||||||
|       console.log(`│   Current: ${status.batteryCapacity}% | Threshold: ${config.thresholds.battery}%`); |       console.log( | ||||||
|  |         `│   Current: ${status.batteryCapacity}% | Threshold: ${config.thresholds.battery}%` | ||||||
|  |       ); | ||||||
|       console.log('│   System would initiate shutdown'); |       console.log('│   System would initiate shutdown'); | ||||||
|     } else { |     } else { | ||||||
|       console.log('│ ✓ Battery capacity above threshold'); |       console.log('│ ✓ Battery capacity above threshold'); | ||||||
|       console.log(`│   Current: ${status.batteryCapacity}% | Threshold: ${config.thresholds.battery}%`); |       console.log( | ||||||
|  |         `│   Current: ${status.batteryCapacity}% | Threshold: ${config.thresholds.battery}%` | ||||||
|  |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (status.batteryRuntime < config.thresholds.runtime) { |     if (status.batteryRuntime < config.thresholds.runtime) { | ||||||
|       console.log('│ ⚠️ WARNING: Runtime below threshold'); |       console.log('│ ⚠️ WARNING: Runtime below threshold'); | ||||||
|       console.log(`│   Current: ${status.batteryRuntime} min | Threshold: ${config.thresholds.runtime} min`); |       console.log( | ||||||
|  |         `│   Current: ${status.batteryRuntime} min | Threshold: ${config.thresholds.runtime} min` | ||||||
|  |       ); | ||||||
|       console.log('│   System would initiate shutdown'); |       console.log('│   System would initiate shutdown'); | ||||||
|     } else { |     } else { | ||||||
|       console.log('│ ✓ Runtime above threshold'); |       console.log('│ ✓ Runtime above threshold'); | ||||||
|       console.log(`│   Current: ${status.batteryRuntime} min | Threshold: ${config.thresholds.runtime} min`); |       console.log( | ||||||
|  |         `│   Current: ${status.batteryRuntime} min | Threshold: ${config.thresholds.runtime} min` | ||||||
|  |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     console.log('└──────────────────────────────────────────┘'); |     console.log('└──────────────────────────────────────────┘'); | ||||||
| @@ -393,7 +401,9 @@ Options: | |||||||
|   private async update(): Promise<void> { |   private async update(): Promise<void> { | ||||||
|     try { |     try { | ||||||
|       // Check if running as root |       // Check if running as root | ||||||
|       this.checkRootAccess('This command must be run as root to update NUPST and refresh the systemd service.'); |       this.checkRootAccess( | ||||||
|  |         'This command must be run as root to update NUPST and refresh the systemd service.' | ||||||
|  |       ); | ||||||
|  |  | ||||||
|       console.log('┌─ NUPST Update Process ──────────────────┐'); |       console.log('┌─ NUPST Update Process ──────────────────┐'); | ||||||
|       console.log('│ Updating NUPST from repository...'); |       console.log('│ Updating NUPST from repository...'); | ||||||
| @@ -412,25 +422,35 @@ Options: | |||||||
|       try { |       try { | ||||||
|         // 1. Update the repository |         // 1. Update the repository | ||||||
|         console.log('│ Pulling latest changes from git repository...'); |         console.log('│ Pulling latest changes from git repository...'); | ||||||
|         execSync(`cd ${installDir} && git fetch origin && git reset --hard origin/main`, { stdio: 'pipe' }); |         execSync(`cd ${installDir} && git fetch origin && git reset --hard origin/main`, { | ||||||
|  |           stdio: 'pipe', | ||||||
|  |         }); | ||||||
|  |  | ||||||
|         // 2. Run the install.sh script |         // 2. Run the install.sh script | ||||||
|         console.log('│ Running install.sh to update NUPST...'); |         console.log('│ Running install.sh to update NUPST...'); | ||||||
|         execSync(`cd ${installDir} && bash ./install.sh`, { stdio: 'pipe' }); |         execSync(`cd ${installDir} && bash ./install.sh`, { stdio: 'pipe' }); | ||||||
|  |  | ||||||
|         // 3. Run the setup.sh script  |         // 3. Run the setup.sh script with force flag to update Node.js and dependencies | ||||||
|         console.log('│ Running setup.sh to update dependencies...'); |         console.log('│ Running setup.sh to update Node.js and dependencies...'); | ||||||
|         execSync(`cd ${installDir} && bash ./setup.sh`, { stdio: 'pipe' }); |         execSync(`cd ${installDir} && bash ./setup.sh --force`, { stdio: 'pipe' }); | ||||||
|  |  | ||||||
|         // 4. Refresh the systemd service |         // 4. Refresh the systemd service | ||||||
|         console.log('│ Refreshing systemd service...'); |         console.log('│ Refreshing systemd service...'); | ||||||
|  |  | ||||||
|         // First check if service exists |         // First check if service exists | ||||||
|         const serviceExists = execSync('systemctl list-unit-files | grep nupst.service').toString().includes('nupst.service'); |         let serviceExists = false; | ||||||
|  |         try { | ||||||
|  |           const output = execSync('systemctl list-unit-files | grep nupst.service').toString(); | ||||||
|  |           serviceExists = output.includes('nupst.service'); | ||||||
|  |         } catch (error) { | ||||||
|  |           // If grep fails (service not found), serviceExists remains false | ||||||
|  |           serviceExists = false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (serviceExists) { |         if (serviceExists) { | ||||||
|           // Stop the service if it's running |           // Stop the service if it's running | ||||||
|           const isRunning = execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; |           const isRunning = | ||||||
|  |             execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; | ||||||
|           if (isRunning) { |           if (isRunning) { | ||||||
|             console.log('│ Stopping nupst service...'); |             console.log('│ Stopping nupst service...'); | ||||||
|             execSync('systemctl stop nupst.service'); |             execSync('systemctl stop nupst.service'); | ||||||
| @@ -451,11 +471,11 @@ Options: | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         console.log('│ Update completed successfully!'); |         console.log('│ Update completed successfully!'); | ||||||
|         console.log('└──────────────────────────────────────────┘'); |         console.log('└─────────────────────────────────────────────┘'); | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         console.error('│ Error during update process:'); |         console.error('│ Error during update process:'); | ||||||
|         console.error(`│ ${error.message}`); |         console.error(`│ ${error.message}`); | ||||||
|         console.error('└──────────────────────────────────────────┘'); |         console.error('└─────────────────────────────────────────────┘'); | ||||||
|         process.exit(1); |         process.exit(1); | ||||||
|       } |       } | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
| @@ -474,7 +494,7 @@ Options: | |||||||
|  |  | ||||||
|       const rl = readline.createInterface({ |       const rl = readline.createInterface({ | ||||||
|         input: process.stdin, |         input: process.stdin, | ||||||
|         output: process.stdout |         output: process.stdout, | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       // Helper function to prompt for input |       // Helper function to prompt for input | ||||||
| @@ -546,7 +566,10 @@ Options: | |||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    * @returns Updated configuration |    * @returns Updated configuration | ||||||
|    */ |    */ | ||||||
|   private async gatherSnmpSettings(config: any, prompt: (question: string) => Promise<string>): Promise<any> { |   private async gatherSnmpSettings( | ||||||
|  |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<any> { | ||||||
|     // SNMP IP Address |     // SNMP IP Address | ||||||
|     const defaultHost = config.snmp.host; |     const defaultHost = config.snmp.host; | ||||||
|     const host = await prompt(`UPS IP Address [${defaultHost}]: `); |     const host = await prompt(`UPS IP Address [${defaultHost}]: `); | ||||||
| @@ -556,7 +579,7 @@ Options: | |||||||
|     const defaultPort = config.snmp.port; |     const defaultPort = config.snmp.port; | ||||||
|     const portInput = await prompt(`SNMP Port [${defaultPort}]: `); |     const portInput = await prompt(`SNMP Port [${defaultPort}]: `); | ||||||
|     const port = parseInt(portInput, 10); |     const port = parseInt(portInput, 10); | ||||||
|     config.snmp.port = (portInput.trim() && !isNaN(port)) ? port : defaultPort; |     config.snmp.port = portInput.trim() && !isNaN(port) ? port : defaultPort; | ||||||
|  |  | ||||||
|     // SNMP Version |     // SNMP Version | ||||||
|     const defaultVersion = config.snmp.version; |     const defaultVersion = config.snmp.version; | ||||||
| @@ -566,7 +589,10 @@ Options: | |||||||
|     console.log('  3) SNMPv3 (with security features)'); |     console.log('  3) SNMPv3 (with security features)'); | ||||||
|     const versionInput = await prompt(`Select SNMP version [${defaultVersion}]: `); |     const versionInput = await prompt(`Select SNMP version [${defaultVersion}]: `); | ||||||
|     const version = parseInt(versionInput, 10); |     const version = parseInt(versionInput, 10); | ||||||
|     config.snmp.version = (versionInput.trim() && (version === 1 || version === 2 || version === 3)) ? version : defaultVersion; |     config.snmp.version = | ||||||
|  |       versionInput.trim() && (version === 1 || version === 2 || version === 3) | ||||||
|  |         ? version | ||||||
|  |         : defaultVersion; | ||||||
|  |  | ||||||
|     if (config.snmp.version === 1 || config.snmp.version === 2) { |     if (config.snmp.version === 1 || config.snmp.version === 2) { | ||||||
|       // SNMP Community String (for v1/v2c) |       // SNMP Community String (for v1/v2c) | ||||||
| @@ -587,7 +613,10 @@ Options: | |||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    * @returns Updated configuration |    * @returns Updated configuration | ||||||
|    */ |    */ | ||||||
|   private async gatherSnmpV3Settings(config: any, prompt: (question: string) => Promise<string>): Promise<any> { |   private async gatherSnmpV3Settings( | ||||||
|  |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<any> { | ||||||
|     console.log('\nSNMPv3 Security Settings:'); |     console.log('\nSNMPv3 Security Settings:'); | ||||||
|  |  | ||||||
|     // Security Level |     // Security Level | ||||||
| @@ -595,9 +624,13 @@ Options: | |||||||
|     console.log('  1) noAuthNoPriv (No Authentication, No Privacy)'); |     console.log('  1) noAuthNoPriv (No Authentication, No Privacy)'); | ||||||
|     console.log('  2) authNoPriv (Authentication, No Privacy)'); |     console.log('  2) authNoPriv (Authentication, No Privacy)'); | ||||||
|     console.log('  3) authPriv (Authentication and Privacy)'); |     console.log('  3) authPriv (Authentication and Privacy)'); | ||||||
|     const defaultSecLevel = config.snmp.securityLevel ?  |     const defaultSecLevel = config.snmp.securityLevel | ||||||
|       (config.snmp.securityLevel === 'noAuthNoPriv' ? 1 :  |       ? config.snmp.securityLevel === 'noAuthNoPriv' | ||||||
|        config.snmp.securityLevel === 'authNoPriv' ? 2 : 3) : 3; |         ? 1 | ||||||
|  |         : config.snmp.securityLevel === 'authNoPriv' | ||||||
|  |         ? 2 | ||||||
|  |         : 3 | ||||||
|  |       : 3; | ||||||
|     const secLevelInput = await prompt(`Select Security Level [${defaultSecLevel}]: `); |     const secLevelInput = await prompt(`Select Security Level [${defaultSecLevel}]: `); | ||||||
|     const secLevel = parseInt(secLevelInput, 10) || defaultSecLevel; |     const secLevel = parseInt(secLevelInput, 10) || defaultSecLevel; | ||||||
|  |  | ||||||
| @@ -639,7 +672,9 @@ Options: | |||||||
|  |  | ||||||
|       // Allow customizing the timeout value |       // Allow customizing the timeout value | ||||||
|       const defaultTimeout = config.snmp.timeout / 1000; // Convert from ms to seconds for display |       const defaultTimeout = config.snmp.timeout / 1000; // Convert from ms to seconds for display | ||||||
|       console.log('\nSNMPv3 operations with authentication and privacy may require longer timeouts.'); |       console.log( | ||||||
|  |         '\nSNMPv3 operations with authentication and privacy may require longer timeouts.' | ||||||
|  |       ); | ||||||
|       const timeoutInput = await prompt(`SNMP Timeout in seconds [${defaultTimeout}]: `); |       const timeoutInput = await prompt(`SNMP Timeout in seconds [${defaultTimeout}]: `); | ||||||
|       const timeout = parseInt(timeoutInput, 10); |       const timeout = parseInt(timeoutInput, 10); | ||||||
|       if (timeoutInput.trim() && !isNaN(timeout)) { |       if (timeoutInput.trim() && !isNaN(timeout)) { | ||||||
| @@ -656,13 +691,18 @@ Options: | |||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    * @returns Updated configuration |    * @returns Updated configuration | ||||||
|    */ |    */ | ||||||
|   private async gatherAuthenticationSettings(config: any, prompt: (question: string) => Promise<string>): Promise<any> { |   private async gatherAuthenticationSettings( | ||||||
|  |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<any> { | ||||||
|     // Authentication protocol |     // Authentication protocol | ||||||
|     console.log('\nAuthentication Protocol:'); |     console.log('\nAuthentication Protocol:'); | ||||||
|     console.log('  1) MD5'); |     console.log('  1) MD5'); | ||||||
|     console.log('  2) SHA'); |     console.log('  2) SHA'); | ||||||
|     const defaultAuthProtocol = config.snmp.authProtocol === 'SHA' ? 2 : 1; |     const defaultAuthProtocol = config.snmp.authProtocol === 'SHA' ? 2 : 1; | ||||||
|     const authProtocolInput = await prompt(`Select Authentication Protocol [${defaultAuthProtocol}]: `); |     const authProtocolInput = await prompt( | ||||||
|  |       `Select Authentication Protocol [${defaultAuthProtocol}]: ` | ||||||
|  |     ); | ||||||
|     const authProtocol = parseInt(authProtocolInput, 10) || defaultAuthProtocol; |     const authProtocol = parseInt(authProtocolInput, 10) || defaultAuthProtocol; | ||||||
|     config.snmp.authProtocol = authProtocol === 2 ? 'SHA' : 'MD5'; |     config.snmp.authProtocol = authProtocol === 2 ? 'SHA' : 'MD5'; | ||||||
|  |  | ||||||
| @@ -680,7 +720,10 @@ Options: | |||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    * @returns Updated configuration |    * @returns Updated configuration | ||||||
|    */ |    */ | ||||||
|   private async gatherPrivacySettings(config: any, prompt: (question: string) => Promise<string>): Promise<any> { |   private async gatherPrivacySettings( | ||||||
|  |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<any> { | ||||||
|     // Privacy protocol |     // Privacy protocol | ||||||
|     console.log('\nPrivacy Protocol:'); |     console.log('\nPrivacy Protocol:'); | ||||||
|     console.log('  1) DES'); |     console.log('  1) DES'); | ||||||
| @@ -704,22 +747,31 @@ Options: | |||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    * @returns Updated configuration |    * @returns Updated configuration | ||||||
|    */ |    */ | ||||||
|   private async gatherThresholdSettings(config: any, prompt: (question: string) => Promise<string>): Promise<any> { |   private async gatherThresholdSettings( | ||||||
|  |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<any> { | ||||||
|     console.log('\nShutdown Thresholds:'); |     console.log('\nShutdown Thresholds:'); | ||||||
|  |  | ||||||
|     // Battery threshold |     // Battery threshold | ||||||
|     const defaultBatteryThreshold = config.thresholds.battery; |     const defaultBatteryThreshold = config.thresholds.battery; | ||||||
|     const batteryThresholdInput = await prompt(`Battery percentage threshold [${defaultBatteryThreshold}%]: `); |     const batteryThresholdInput = await prompt( | ||||||
|  |       `Battery percentage threshold [${defaultBatteryThreshold}%]: ` | ||||||
|  |     ); | ||||||
|     const batteryThreshold = parseInt(batteryThresholdInput, 10); |     const batteryThreshold = parseInt(batteryThresholdInput, 10); | ||||||
|     config.thresholds.battery = (batteryThresholdInput.trim() && !isNaN(batteryThreshold))  |     config.thresholds.battery = | ||||||
|  |       batteryThresholdInput.trim() && !isNaN(batteryThreshold) | ||||||
|         ? batteryThreshold |         ? batteryThreshold | ||||||
|         : defaultBatteryThreshold; |         : defaultBatteryThreshold; | ||||||
|  |  | ||||||
|     // Runtime threshold |     // Runtime threshold | ||||||
|     const defaultRuntimeThreshold = config.thresholds.runtime; |     const defaultRuntimeThreshold = config.thresholds.runtime; | ||||||
|     const runtimeThresholdInput = await prompt(`Runtime minutes threshold [${defaultRuntimeThreshold} minutes]: `); |     const runtimeThresholdInput = await prompt( | ||||||
|  |       `Runtime minutes threshold [${defaultRuntimeThreshold} minutes]: ` | ||||||
|  |     ); | ||||||
|     const runtimeThreshold = parseInt(runtimeThresholdInput, 10); |     const runtimeThreshold = parseInt(runtimeThresholdInput, 10); | ||||||
|     config.thresholds.runtime = (runtimeThresholdInput.trim() && !isNaN(runtimeThreshold))  |     config.thresholds.runtime = | ||||||
|  |       runtimeThresholdInput.trim() && !isNaN(runtimeThreshold) | ||||||
|         ? runtimeThreshold |         ? runtimeThreshold | ||||||
|         : defaultRuntimeThreshold; |         : defaultRuntimeThreshold; | ||||||
|  |  | ||||||
| @@ -727,7 +779,8 @@ Options: | |||||||
|     const defaultInterval = config.checkInterval / 1000; // Convert from ms to seconds for display |     const defaultInterval = config.checkInterval / 1000; // Convert from ms to seconds for display | ||||||
|     const intervalInput = await prompt(`Check interval in seconds [${defaultInterval}]: `); |     const intervalInput = await prompt(`Check interval in seconds [${defaultInterval}]: `); | ||||||
|     const interval = parseInt(intervalInput, 10); |     const interval = parseInt(intervalInput, 10); | ||||||
|     config.checkInterval = (intervalInput.trim() && !isNaN(interval))  |     config.checkInterval = | ||||||
|  |       intervalInput.trim() && !isNaN(interval) | ||||||
|         ? interval * 1000 // Convert to ms |         ? interval * 1000 // Convert to ms | ||||||
|         : defaultInterval * 1000; |         : defaultInterval * 1000; | ||||||
|  |  | ||||||
| @@ -740,7 +793,10 @@ Options: | |||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    * @returns Updated configuration |    * @returns Updated configuration | ||||||
|    */ |    */ | ||||||
|   private async gatherUpsModelSettings(config: any, prompt: (question: string) => Promise<string>): Promise<any> { |   private async gatherUpsModelSettings( | ||||||
|  |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<any> { | ||||||
|     console.log('\nUPS Model Selection:'); |     console.log('\nUPS Model Selection:'); | ||||||
|     console.log('  1) CyberPower'); |     console.log('  1) CyberPower'); | ||||||
|     console.log('  2) APC'); |     console.log('  2) APC'); | ||||||
| @@ -749,12 +805,20 @@ Options: | |||||||
|     console.log('  5) Liebert/Vertiv'); |     console.log('  5) Liebert/Vertiv'); | ||||||
|     console.log('  6) Custom (Advanced)'); |     console.log('  6) Custom (Advanced)'); | ||||||
|  |  | ||||||
|     const defaultModelValue = config.snmp.upsModel === 'cyberpower' ? 1 : |     const defaultModelValue = | ||||||
|                            config.snmp.upsModel === 'apc' ? 2 : |       config.snmp.upsModel === 'cyberpower' | ||||||
|                            config.snmp.upsModel === 'eaton' ? 3 : |         ? 1 | ||||||
|                            config.snmp.upsModel === 'tripplite' ? 4 : |         : config.snmp.upsModel === 'apc' | ||||||
|                            config.snmp.upsModel === 'liebert' ? 5 :  |         ? 2 | ||||||
|                            config.snmp.upsModel === 'custom' ? 6 : 1; |         : config.snmp.upsModel === 'eaton' | ||||||
|  |         ? 3 | ||||||
|  |         : config.snmp.upsModel === 'tripplite' | ||||||
|  |         ? 4 | ||||||
|  |         : config.snmp.upsModel === 'liebert' | ||||||
|  |         ? 5 | ||||||
|  |         : config.snmp.upsModel === 'custom' | ||||||
|  |         ? 6 | ||||||
|  |         : 1; | ||||||
|  |  | ||||||
|     const modelInput = await prompt(`Select UPS model [${defaultModelValue}]: `); |     const modelInput = await prompt(`Select UPS model [${defaultModelValue}]: `); | ||||||
|     const modelValue = parseInt(modelInput, 10) || defaultModelValue; |     const modelValue = parseInt(modelInput, 10) || defaultModelValue; | ||||||
| @@ -783,7 +847,7 @@ Options: | |||||||
|       config.snmp.customOIDs = { |       config.snmp.customOIDs = { | ||||||
|         POWER_STATUS: powerStatusOID.trim(), |         POWER_STATUS: powerStatusOID.trim(), | ||||||
|         BATTERY_CAPACITY: batteryCapacityOID.trim(), |         BATTERY_CAPACITY: batteryCapacityOID.trim(), | ||||||
|         BATTERY_RUNTIME: batteryRuntimeOID.trim() |         BATTERY_RUNTIME: batteryRuntimeOID.trim(), | ||||||
|       }; |       }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -799,7 +863,9 @@ Options: | |||||||
|     console.log(`│ SNMP Host: ${config.snmp.host}:${config.snmp.port}`); |     console.log(`│ SNMP Host: ${config.snmp.host}:${config.snmp.port}`); | ||||||
|     console.log(`│ SNMP Version: ${config.snmp.version}`); |     console.log(`│ SNMP Version: ${config.snmp.version}`); | ||||||
|     console.log(`│ UPS Model: ${config.snmp.upsModel}`); |     console.log(`│ UPS Model: ${config.snmp.upsModel}`); | ||||||
|     console.log(`│ Thresholds: ${config.thresholds.battery}% battery, ${config.thresholds.runtime} min runtime`); |     console.log( | ||||||
|  |       `│ Thresholds: ${config.thresholds.battery}% battery, ${config.thresholds.runtime} min runtime` | ||||||
|  |     ); | ||||||
|     console.log(`│ Check Interval: ${config.checkInterval / 1000} seconds`); |     console.log(`│ Check Interval: ${config.checkInterval / 1000} seconds`); | ||||||
|     console.log('└──────────────────────────────────────────┘\n'); |     console.log('└──────────────────────────────────────────┘\n'); | ||||||
|   } |   } | ||||||
| @@ -809,15 +875,20 @@ Options: | |||||||
|    * @param config Current configuration |    * @param config Current configuration | ||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    */ |    */ | ||||||
|   private async optionallyTestConnection(config: any, prompt: (question: string) => Promise<string>): Promise<void> { |   private async optionallyTestConnection( | ||||||
|     const testConnection = await prompt('Would you like to test the connection to your UPS? (y/N): '); |     config: any, | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<void> { | ||||||
|  |     const testConnection = await prompt( | ||||||
|  |       'Would you like to test the connection to your UPS? (y/N): ' | ||||||
|  |     ); | ||||||
|     if (testConnection.toLowerCase() === 'y') { |     if (testConnection.toLowerCase() === 'y') { | ||||||
|       console.log('\nTesting connection to UPS...'); |       console.log('\nTesting connection to UPS...'); | ||||||
|       try { |       try { | ||||||
|         // Create a test config with a short timeout |         // Create a test config with a short timeout | ||||||
|         const testConfig = { |         const testConfig = { | ||||||
|           ...config.snmp, |           ...config.snmp, | ||||||
|           timeout: Math.min(config.snmp.timeout, 10000) // Use at most 10 seconds for testing |           timeout: Math.min(config.snmp.timeout, 10000), // Use at most 10 seconds for testing | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         const status = await this.nupst.getSnmp().getUpsStatus(testConfig); |         const status = await this.nupst.getSnmp().getUpsStatus(testConfig); | ||||||
| @@ -843,11 +914,12 @@ Options: | |||||||
|   private async restartServiceIfRunning(): Promise<void> { |   private async restartServiceIfRunning(): Promise<void> { | ||||||
|     try { |     try { | ||||||
|       // Check if the service is active |       // Check if the service is active | ||||||
|       const isActive = execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; |       const isActive = | ||||||
|  |         execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; | ||||||
|  |  | ||||||
|       if (isActive) { |       if (isActive) { | ||||||
|         // Service is running, restart it |         // Service is running, restart it | ||||||
|         console.log('┌─ Service Update ─────────────────────────┐'); |         console.log('┌─ Service Update ──────────────────────────┐'); | ||||||
|         console.log('│ Configuration has changed.'); |         console.log('│ Configuration has changed.'); | ||||||
|         console.log('│ Restarting NUPST service to apply changes...'); |         console.log('│ Restarting NUPST service to apply changes...'); | ||||||
|  |  | ||||||
| @@ -867,7 +939,7 @@ Options: | |||||||
|           console.log('│   sudo systemctl restart nupst.service'); |           console.log('│   sudo systemctl restart nupst.service'); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         console.log('└──────────────────────────────────────────┘'); |         console.log('└───────────────────────────────────────────┘'); | ||||||
|       } |       } | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       // Ignore errors checking service status |       // Ignore errors checking service status | ||||||
| @@ -878,18 +950,24 @@ Options: | |||||||
|    * Optionally enable and start systemd service |    * Optionally enable and start systemd service | ||||||
|    * @param prompt Function to prompt for user input |    * @param prompt Function to prompt for user input | ||||||
|    */ |    */ | ||||||
|   private async optionallyEnableService(prompt: (question: string) => Promise<string>): Promise<void> { |   private async optionallyEnableService( | ||||||
|  |     prompt: (question: string) => Promise<string> | ||||||
|  |   ): Promise<void> { | ||||||
|     if (process.getuid && process.getuid() !== 0) { |     if (process.getuid && process.getuid() !== 0) { | ||||||
|       console.log('\nNote: Run "sudo nupst enable" to set up NUPST as a system service.'); |       console.log('\nNote: Run "sudo nupst enable" to set up NUPST as a system service.'); | ||||||
|     } else { |     } else { | ||||||
|       const setupService = await prompt('Would you like to enable NUPST as a system service? (y/N): '); |       const setupService = await prompt( | ||||||
|  |         'Would you like to enable NUPST as a system service? (y/N): ' | ||||||
|  |       ); | ||||||
|       if (setupService.toLowerCase() === 'y') { |       if (setupService.toLowerCase() === 'y') { | ||||||
|         try { |         try { | ||||||
|           await this.nupst.getSystemd().install(); |           await this.nupst.getSystemd().install(); | ||||||
|           console.log('Service installed and enabled to start on boot.'); |           console.log('Service installed and enabled to start on boot.'); | ||||||
|  |  | ||||||
|           // Ask if the user wants to start the service now |           // Ask if the user wants to start the service now | ||||||
|           const startService = await prompt('Would you like to start the NUPST service now? (Y/n): '); |           const startService = await prompt( | ||||||
|  |             'Would you like to start the NUPST service now? (Y/n): ' | ||||||
|  |           ); | ||||||
|           if (startService.toLowerCase() !== 'n') { |           if (startService.toLowerCase() !== 'n') { | ||||||
|             await this.nupst.getSystemd().start(); |             await this.nupst.getSystemd().start(); | ||||||
|             console.log('NUPST service started successfully.'); |             console.log('NUPST service started successfully.'); | ||||||
| @@ -914,7 +992,7 @@ Options: | |||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         console.error('┌─ Configuration Error ─────────────────────┐'); |         console.error('┌─ Configuration Error ─────────────────────┐'); | ||||||
|         console.error('│ No configuration found.'); |         console.error('│ No configuration found.'); | ||||||
|         console.error('│ Please run \'nupst setup\' first to create a configuration.'); |         console.error("│ Please run 'nupst setup' first to create a configuration."); | ||||||
|         console.error('└──────────────────────────────────────────┘'); |         console.error('└──────────────────────────────────────────┘'); | ||||||
|         return; |         return; | ||||||
|       } |       } | ||||||
| @@ -938,7 +1016,10 @@ Options: | |||||||
|         console.log(`│   Username: ${config.snmp.username}`); |         console.log(`│   Username: ${config.snmp.username}`); | ||||||
|  |  | ||||||
|         // Show auth and privacy details based on security level |         // Show auth and privacy details based on security level | ||||||
|         if (config.snmp.securityLevel === 'authNoPriv' || config.snmp.securityLevel === 'authPriv') { |         if ( | ||||||
|  |           config.snmp.securityLevel === 'authNoPriv' || | ||||||
|  |           config.snmp.securityLevel === 'authPriv' | ||||||
|  |         ) { | ||||||
|           console.log(`│   Auth Protocol: ${config.snmp.authProtocol || 'None'}`); |           console.log(`│   Auth Protocol: ${config.snmp.authProtocol || 'None'}`); | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -954,7 +1035,9 @@ Options: | |||||||
|       if (config.snmp.upsModel === 'custom' && config.snmp.customOIDs) { |       if (config.snmp.upsModel === 'custom' && config.snmp.customOIDs) { | ||||||
|         console.log('│ Custom OIDs:'); |         console.log('│ Custom OIDs:'); | ||||||
|         console.log(`│   Power Status: ${config.snmp.customOIDs.POWER_STATUS || 'Not set'}`); |         console.log(`│   Power Status: ${config.snmp.customOIDs.POWER_STATUS || 'Not set'}`); | ||||||
|         console.log(`│   Battery Capacity: ${config.snmp.customOIDs.BATTERY_CAPACITY || 'Not set'}`); |         console.log( | ||||||
|  |           `│   Battery Capacity: ${config.snmp.customOIDs.BATTERY_CAPACITY || 'Not set'}` | ||||||
|  |         ); | ||||||
|         console.log(`│   Battery Runtime: ${config.snmp.customOIDs.BATTERY_RUNTIME || 'Not set'}`); |         console.log(`│   Battery Runtime: ${config.snmp.customOIDs.BATTERY_RUNTIME || 'Not set'}`); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -973,8 +1056,10 @@ Options: | |||||||
|  |  | ||||||
|       // Show service status |       // Show service status | ||||||
|       try { |       try { | ||||||
|         const isActive = execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; |         const isActive = | ||||||
|         const isEnabled = execSync('systemctl is-enabled nupst.service || true').toString().trim() === 'enabled'; |           execSync('systemctl is-active nupst.service || true').toString().trim() === 'active'; | ||||||
|  |         const isEnabled = | ||||||
|  |           execSync('systemctl is-enabled nupst.service || true').toString().trim() === 'enabled'; | ||||||
|  |  | ||||||
|         console.log('┌─ Service Status ─────────────────────────┐'); |         console.log('┌─ Service Status ─────────────────────────┐'); | ||||||
|         console.log(`│ Service Active: ${isActive ? 'Yes' : 'No'}`); |         console.log(`│ Service Active: ${isActive ? 'Yes' : 'No'}`); | ||||||
| @@ -983,7 +1068,6 @@ Options: | |||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         // Ignore errors checking service status |         // Ignore errors checking service status | ||||||
|       } |       } | ||||||
|        |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       console.error(`Failed to display configuration: ${error.message}`); |       console.error(`Failed to display configuration: ${error.message}`); | ||||||
|     } |     } | ||||||
| @@ -1002,7 +1086,7 @@ Options: | |||||||
|  |  | ||||||
|       const rl = readline.createInterface({ |       const rl = readline.createInterface({ | ||||||
|         input: process.stdin, |         input: process.stdin, | ||||||
|         output: process.stdout |         output: process.stdout, | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       // Helper function to prompt for input |       // Helper function to prompt for input | ||||||
| @@ -1019,7 +1103,9 @@ Options: | |||||||
|       console.log('This will completely remove NUPST from your system.\n'); |       console.log('This will completely remove NUPST from your system.\n'); | ||||||
|  |  | ||||||
|       // Ask about removing configuration |       // Ask about removing configuration | ||||||
|       const removeConfig = await prompt('Do you want to remove the NUPST configuration files? (y/N): '); |       const removeConfig = await prompt( | ||||||
|  |         'Do you want to remove the NUPST configuration files? (y/N): ' | ||||||
|  |       ); | ||||||
|  |  | ||||||
|       // Find the uninstall.sh script location |       // Find the uninstall.sh script location | ||||||
|       let uninstallScriptPath: string; |       let uninstallScriptPath: string; | ||||||
| @@ -1036,10 +1122,7 @@ Options: | |||||||
|         await fs.access(uninstallScriptPath); |         await fs.access(uninstallScriptPath); | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         // If we can't find it in the expected location, try common installation paths |         // If we can't find it in the expected location, try common installation paths | ||||||
|         const commonPaths = [ |         const commonPaths = ['/opt/nupst/uninstall.sh', join(process.cwd(), 'uninstall.sh')]; | ||||||
|           '/opt/nupst/uninstall.sh', |  | ||||||
|           join(process.cwd(), 'uninstall.sh') |  | ||||||
|         ]; |  | ||||||
|  |  | ||||||
|         for (const path of commonPaths) { |         for (const path of commonPaths) { | ||||||
|           try { |           try { | ||||||
| @@ -1069,15 +1152,14 @@ Options: | |||||||
|         ...process.env, |         ...process.env, | ||||||
|         REMOVE_CONFIG: removeConfig.toLowerCase() === 'y' ? 'yes' : 'no', |         REMOVE_CONFIG: removeConfig.toLowerCase() === 'y' ? 'yes' : 'no', | ||||||
|         REMOVE_REPO: 'yes', // Always remove repo as requested |         REMOVE_REPO: 'yes', // Always remove repo as requested | ||||||
|         NUPST_CLI_CALL: 'true'  // Flag to indicate this is being called from CLI |         NUPST_CLI_CALL: 'true', // Flag to indicate this is being called from CLI | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       // Run the uninstall script with sudo |       // Run the uninstall script with sudo | ||||||
|       execSync(`sudo bash ${uninstallScriptPath}`, { |       execSync(`sudo bash ${uninstallScriptPath}`, { | ||||||
|         env, |         env, | ||||||
|         stdio: 'inherit'  // Show output in the terminal |         stdio: 'inherit', // Show output in the terminal | ||||||
|       }); |       }); | ||||||
|        |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       console.error(`Uninstall failed: ${error.message}`); |       console.error(`Uninstall failed: ${error.message}`); | ||||||
|       process.exit(1); |       process.exit(1); | ||||||
|   | |||||||
							
								
								
									
										188
									
								
								ts/daemon.ts
									
									
									
									
									
								
							
							
						
						
									
										188
									
								
								ts/daemon.ts
									
									
									
									
									
								
							| @@ -1,11 +1,12 @@ | |||||||
| import * as fs from 'fs'; | import * as fs from 'fs'; | ||||||
| import * as path from 'path'; | import * as path from 'path'; | ||||||
| import { exec } from 'child_process'; | import { exec, execFile } from 'child_process'; | ||||||
| import { promisify } from 'util'; | import { promisify } from 'util'; | ||||||
| import { NupstSnmp } from './snmp/manager.js'; | import { NupstSnmp } from './snmp/manager.js'; | ||||||
| import type { ISnmpConfig } from './snmp/types.js'; | import type { ISnmpConfig } from './snmp/types.js'; | ||||||
|  |  | ||||||
| const execAsync = promisify(exec); | const execAsync = promisify(exec); | ||||||
|  | const execFileAsync = promisify(execFile); | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Configuration interface for the daemon |  * Configuration interface for the daemon | ||||||
| @@ -124,7 +125,7 @@ export class NupstDaemon { | |||||||
|     console.error('┌─ Configuration Error ─────────────────────┐'); |     console.error('┌─ Configuration Error ─────────────────────┐'); | ||||||
|     console.error(`│ ${message}`); |     console.error(`│ ${message}`); | ||||||
|     console.error('│ Please run \'nupst setup\' first to create a configuration.'); |     console.error('│ Please run \'nupst setup\' first to create a configuration.'); | ||||||
|     console.error('└──────────────────────────────────────────┘'); |     console.error('└───────────────────────────────────────────┘'); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -195,7 +196,7 @@ export class NupstDaemon { | |||||||
|     console.log(`│   Battery: ${this.config.thresholds.battery}%`); |     console.log(`│   Battery: ${this.config.thresholds.battery}%`); | ||||||
|     console.log(`│   Runtime: ${this.config.thresholds.runtime} minutes`); |     console.log(`│   Runtime: ${this.config.thresholds.runtime} minutes`); | ||||||
|     console.log(`│ Check Interval: ${this.config.checkInterval / 1000} seconds`); |     console.log(`│ Check Interval: ${this.config.checkInterval / 1000} seconds`); | ||||||
|     console.log('└──────────────────────────────────────────┘'); |     console.log('└────────────────────────────────────────────┘'); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
| @@ -225,20 +226,20 @@ export class NupstDaemon { | |||||||
|          |          | ||||||
|         // Log status changes |         // Log status changes | ||||||
|         if (status.powerStatus !== lastStatus) { |         if (status.powerStatus !== lastStatus) { | ||||||
|           console.log('┌──────────────────────────────────────────┐'); |           console.log('┌─ Power Status Change ─────────────────────┐'); | ||||||
|           console.log(`│ Power status changed: ${lastStatus} → ${status.powerStatus}`); |           console.log(`│ Status changed: ${lastStatus} → ${status.powerStatus}`); | ||||||
|           console.log('└──────────────────────────────────────────┘'); |           console.log('└───────────────────────────────────────────┘'); | ||||||
|           lastStatus = status.powerStatus; |           lastStatus = status.powerStatus; | ||||||
|           lastLogTime = currentTime; // Reset log timer when status changes |           lastLogTime = currentTime; // Reset log timer when status changes | ||||||
|         } |         } | ||||||
|         // Log status periodically (at least every 5 minutes) |         // Log status periodically (at least every 5 minutes) | ||||||
|         else if (shouldLogStatus) { |         else if (shouldLogStatus) { | ||||||
|           const timestamp = new Date().toISOString(); |           const timestamp = new Date().toISOString(); | ||||||
|           console.log('┌──────────────────────────────────────────┐'); |           console.log('┌─ Periodic Status Update ──────────────────┐'); | ||||||
|           console.log(`│ [${timestamp}] Periodic Status Update`); |           console.log(`│ Timestamp: ${timestamp}`); | ||||||
|           console.log(`│ Power Status: ${status.powerStatus}`); |           console.log(`│ Power Status: ${status.powerStatus}`); | ||||||
|           console.log(`│ Battery: ${status.batteryCapacity}% | Runtime: ${status.batteryRuntime} min`); |           console.log(`│ Battery: ${status.batteryCapacity}% | Runtime: ${status.batteryRuntime} min`); | ||||||
|           console.log('└──────────────────────────────────────────┘'); |           console.log('└───────────────────────────────────────────┘'); | ||||||
|           lastLogTime = currentTime; |           lastLogTime = currentTime; | ||||||
|         } |         } | ||||||
|          |          | ||||||
| @@ -266,8 +267,8 @@ export class NupstDaemon { | |||||||
|     batteryCapacity: number, |     batteryCapacity: number, | ||||||
|     batteryRuntime: number |     batteryRuntime: number | ||||||
|   }): Promise<void> { |   }): Promise<void> { | ||||||
|     console.log('┌─ UPS Status ───────────────────────────────┐'); |     console.log('┌─ UPS Status ─────────────────────────────┐'); | ||||||
|     console.log(`│ Battery: ${status.batteryCapacity}% | Runtime: ${status.batteryRuntime} min │`); |     console.log(`│ Battery: ${status.batteryCapacity}% | Runtime: ${status.batteryRuntime} min`); | ||||||
|     console.log('└──────────────────────────────────────────┘'); |     console.log('└──────────────────────────────────────────┘'); | ||||||
|      |      | ||||||
|     // Check battery threshold |     // Check battery threshold | ||||||
| @@ -298,24 +299,102 @@ export class NupstDaemon { | |||||||
|     const shutdownDelayMinutes = 5; |     const shutdownDelayMinutes = 5; | ||||||
|      |      | ||||||
|     try { |     try { | ||||||
|  |       // Find shutdown command in common system paths | ||||||
|  |       const shutdownPaths = [ | ||||||
|  |         '/sbin/shutdown', | ||||||
|  |         '/usr/sbin/shutdown', | ||||||
|  |         '/bin/shutdown', | ||||||
|  |         '/usr/bin/shutdown' | ||||||
|  |       ]; | ||||||
|  |        | ||||||
|  |       let shutdownCmd = ''; | ||||||
|  |       for (const path of shutdownPaths) { | ||||||
|  |         try { | ||||||
|  |           if (fs.existsSync(path)) { | ||||||
|  |             shutdownCmd = path; | ||||||
|  |             console.log(`Found shutdown command at: ${shutdownCmd}`); | ||||||
|  |             break; | ||||||
|  |           } | ||||||
|  |         } catch (e) { | ||||||
|  |           // Continue checking other paths | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       if (shutdownCmd) { | ||||||
|         // Execute shutdown command with delay to allow for VM graceful shutdown |         // Execute shutdown command with delay to allow for VM graceful shutdown | ||||||
|       const { stdout } = await execAsync(`shutdown -h +${shutdownDelayMinutes} "UPS battery critical, shutting down in ${shutdownDelayMinutes} minutes"`); |         console.log(`Executing: ${shutdownCmd} -h +${shutdownDelayMinutes} "UPS battery critical..."`); | ||||||
|  |         const { stdout } = await execFileAsync(shutdownCmd, [ | ||||||
|  |           '-h',  | ||||||
|  |           `+${shutdownDelayMinutes}`,  | ||||||
|  |           `UPS battery critical, shutting down in ${shutdownDelayMinutes} minutes` | ||||||
|  |         ]); | ||||||
|         console.log('Shutdown initiated:', stdout); |         console.log('Shutdown initiated:', stdout); | ||||||
|         console.log(`Allowing ${shutdownDelayMinutes} minutes for VMs to shut down safely`); |         console.log(`Allowing ${shutdownDelayMinutes} minutes for VMs to shut down safely`); | ||||||
|  |       } else { | ||||||
|  |         // Try using the PATH to find shutdown | ||||||
|  |         try { | ||||||
|  |           console.log('Shutdown command not found in common paths, trying via PATH...'); | ||||||
|  |           const { stdout } = await execAsync(`shutdown -h +${shutdownDelayMinutes} "UPS battery critical, shutting down in ${shutdownDelayMinutes} minutes"`, { | ||||||
|  |             env: process.env // Pass the current environment | ||||||
|  |           }); | ||||||
|  |           console.log('Shutdown initiated:', stdout); | ||||||
|  |         } catch (e) { | ||||||
|  |           throw new Error(`Shutdown command not found: ${e.message}`); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|        |        | ||||||
|       // Monitor UPS during shutdown and force immediate shutdown if battery gets too low |       // Monitor UPS during shutdown and force immediate shutdown if battery gets too low | ||||||
|       console.log('Monitoring UPS during shutdown process...'); |       console.log('Monitoring UPS during shutdown process...'); | ||||||
|       await this.monitorDuringShutdown(); |       await this.monitorDuringShutdown(); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       console.error('Failed to initiate shutdown:', error); |       console.error('Failed to initiate shutdown:', error); | ||||||
|       // Try a different method if first one fails |        | ||||||
|  |       // Try alternative shutdown methods | ||||||
|  |       const alternatives = [ | ||||||
|  |         { cmd: 'poweroff', args: ['--force'] }, | ||||||
|  |         { cmd: 'halt', args: ['-p'] }, | ||||||
|  |         { cmd: 'systemctl', args: ['poweroff'] }, | ||||||
|  |         { cmd: 'reboot', args: ['-p'] } // Some systems allow reboot -p for power off | ||||||
|  |       ]; | ||||||
|  |        | ||||||
|  |       for (const alt of alternatives) { | ||||||
|         try { |         try { | ||||||
|         console.log('Trying alternative shutdown method...'); |           // First check if command exists in common system paths | ||||||
|         await execAsync('poweroff --force'); |           const paths = [ | ||||||
|       } catch (innerError) { |             `/sbin/${alt.cmd}`, | ||||||
|         console.error('All shutdown methods failed:', innerError); |             `/usr/sbin/${alt.cmd}`, | ||||||
|  |             `/bin/${alt.cmd}`, | ||||||
|  |             `/usr/bin/${alt.cmd}` | ||||||
|  |           ]; | ||||||
|  |            | ||||||
|  |           let cmdPath = ''; | ||||||
|  |           for (const path of paths) { | ||||||
|  |             if (fs.existsSync(path)) { | ||||||
|  |               cmdPath = path; | ||||||
|  |               break; | ||||||
|             } |             } | ||||||
|           } |           } | ||||||
|  |            | ||||||
|  |           if (cmdPath) { | ||||||
|  |             console.log(`Trying alternative shutdown method: ${cmdPath} ${alt.args.join(' ')}`); | ||||||
|  |             await execFileAsync(cmdPath, alt.args); | ||||||
|  |             return; // Exit if successful | ||||||
|  |           } else { | ||||||
|  |             // Try using PATH environment | ||||||
|  |             console.log(`Trying alternative via PATH: ${alt.cmd} ${alt.args.join(' ')}`); | ||||||
|  |             await execAsync(`${alt.cmd} ${alt.args.join(' ')}`, { | ||||||
|  |               env: process.env // Pass the current environment | ||||||
|  |             }); | ||||||
|  |             return; // Exit if successful | ||||||
|  |           } | ||||||
|  |         } catch (altError) { | ||||||
|  |           console.error(`Alternative method ${alt.cmd} failed:`, altError); | ||||||
|  |           // Continue to next method | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       console.error('All shutdown methods failed'); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   /** |   /** | ||||||
| @@ -346,10 +425,79 @@ export class NupstDaemon { | |||||||
|           console.log('└──────────────────────────────────────────┘'); |           console.log('└──────────────────────────────────────────┘'); | ||||||
|            |            | ||||||
|           try { |           try { | ||||||
|             await execAsync('shutdown -h now "EMERGENCY: UPS battery critically low, shutting down NOW"'); |             // Find shutdown command in common system paths | ||||||
|  |             const shutdownPaths = [ | ||||||
|  |               '/sbin/shutdown', | ||||||
|  |               '/usr/sbin/shutdown', | ||||||
|  |               '/bin/shutdown', | ||||||
|  |               '/usr/bin/shutdown' | ||||||
|  |             ]; | ||||||
|  |              | ||||||
|  |             let shutdownCmd = ''; | ||||||
|  |             for (const path of shutdownPaths) { | ||||||
|  |               if (fs.existsSync(path)) { | ||||||
|  |                 shutdownCmd = path; | ||||||
|  |                 console.log(`Found shutdown command at: ${shutdownCmd}`); | ||||||
|  |                 break; | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             if (shutdownCmd) { | ||||||
|  |               console.log(`Executing emergency shutdown: ${shutdownCmd} -h now`); | ||||||
|  |               await execFileAsync(shutdownCmd, ['-h', 'now', 'EMERGENCY: UPS battery critically low, shutting down NOW']); | ||||||
|  |             } else { | ||||||
|  |               // Try using the PATH to find shutdown | ||||||
|  |               console.log('Shutdown command not found in common paths, trying via PATH...'); | ||||||
|  |               await execAsync('shutdown -h now "EMERGENCY: UPS battery critically low, shutting down NOW"', { | ||||||
|  |                 env: process.env // Pass the current environment | ||||||
|  |               }); | ||||||
|  |             } | ||||||
|           } catch (error) { |           } catch (error) { | ||||||
|             console.error('Emergency shutdown failed, trying alternative method...'); |             console.error('Emergency shutdown failed, trying alternative methods...'); | ||||||
|             await execAsync('poweroff --force'); |              | ||||||
|  |             // Try alternative shutdown methods in sequence | ||||||
|  |             const alternatives = [ | ||||||
|  |               { cmd: 'poweroff', args: ['--force'] }, | ||||||
|  |               { cmd: 'halt', args: ['-p'] }, | ||||||
|  |               { cmd: 'systemctl', args: ['poweroff'] } | ||||||
|  |             ]; | ||||||
|  |              | ||||||
|  |             for (const alt of alternatives) { | ||||||
|  |               try { | ||||||
|  |                 // Check common paths | ||||||
|  |                 const paths = [ | ||||||
|  |                   `/sbin/${alt.cmd}`, | ||||||
|  |                   `/usr/sbin/${alt.cmd}`, | ||||||
|  |                   `/bin/${alt.cmd}`, | ||||||
|  |                   `/usr/bin/${alt.cmd}` | ||||||
|  |                 ]; | ||||||
|  |                  | ||||||
|  |                 let cmdPath = ''; | ||||||
|  |                 for (const path of paths) { | ||||||
|  |                   if (fs.existsSync(path)) { | ||||||
|  |                     cmdPath = path; | ||||||
|  |                     break; | ||||||
|  |                   } | ||||||
|  |                 } | ||||||
|  |                  | ||||||
|  |                 if (cmdPath) { | ||||||
|  |                   console.log(`Emergency: using ${cmdPath} ${alt.args.join(' ')}`); | ||||||
|  |                   await execFileAsync(cmdPath, alt.args); | ||||||
|  |                   return; // Exit if successful | ||||||
|  |                 } else { | ||||||
|  |                   // Try using PATH | ||||||
|  |                   console.log(`Emergency: trying ${alt.cmd} via PATH`); | ||||||
|  |                   await execAsync(`${alt.cmd} ${alt.args.join(' ')}`, { | ||||||
|  |                     env: process.env | ||||||
|  |                   }); | ||||||
|  |                   return; // Exit if successful | ||||||
|  |                 } | ||||||
|  |               } catch (altError) { | ||||||
|  |                 // Continue to next method | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             console.error('All emergency shutdown methods failed'); | ||||||
|           } |           } | ||||||
|            |            | ||||||
|           // Stop monitoring after initiating emergency shutdown |           // Stop monitoring after initiating emergency shutdown | ||||||
|   | |||||||
| @@ -162,7 +162,7 @@ export class Nupst { | |||||||
|    */ |    */ | ||||||
|   public logVersionInfo(checkForUpdates: boolean = true): void { |   public logVersionInfo(checkForUpdates: boolean = true): void { | ||||||
|     const version = this.getVersion(); |     const version = this.getVersion(); | ||||||
|     console.log('┌─ NUPST Version ────────────────────────┐'); |     console.log('┌─ NUPST Version ────────────────────────────┐'); | ||||||
|     console.log(`│ Current Version: ${version}`); |     console.log(`│ Current Version: ${version}`); | ||||||
|      |      | ||||||
|     if (this.updateAvailable && this.latestVersion) { |     if (this.updateAvailable && this.latestVersion) { | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import * as dgram from 'dgram'; |  | ||||||
| import * as snmp from 'net-snmp'; | import * as snmp from 'net-snmp'; | ||||||
| import type { IOidSet, ISnmpConfig, TUpsModel, IUpsStatus } from './types.js'; | import type { IOidSet, ISnmpConfig, TUpsModel, IUpsStatus } from './types.js'; | ||||||
| import { UpsOidSets } from './oid-sets.js'; | import { UpsOidSets } from './oid-sets.js'; | ||||||
|   | |||||||
| @@ -66,7 +66,7 @@ WantedBy=multi-user.target | |||||||
|        |        | ||||||
|       // Write the service file |       // Write the service file | ||||||
|       await fs.writeFile(this.serviceFilePath, this.serviceTemplate); |       await fs.writeFile(this.serviceFilePath, this.serviceTemplate); | ||||||
|       console.log('┌─ Service Installation ─────────────────────┐'); |       console.log('┌─ Service Installation ──────────────────────┐'); | ||||||
|       console.log(`│ Service file created at ${this.serviceFilePath}`); |       console.log(`│ Service file created at ${this.serviceFilePath}`); | ||||||
|  |  | ||||||
|       // Reload systemd daemon |       // Reload systemd daemon | ||||||
| @@ -76,7 +76,7 @@ WantedBy=multi-user.target | |||||||
|       // Enable the service |       // Enable the service | ||||||
|       execSync('systemctl enable nupst.service'); |       execSync('systemctl enable nupst.service'); | ||||||
|       console.log('│ Service enabled to start on boot'); |       console.log('│ Service enabled to start on boot'); | ||||||
|       console.log('└──────────────────────────────────────────┘'); |       console.log('└─────────────────────────────────────────────┘'); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       if (error.message === 'Configuration not found') { |       if (error.message === 'Configuration not found') { | ||||||
|         // Just rethrow the error as the message has already been displayed |         // Just rethrow the error as the message has already been displayed | ||||||
| @@ -97,9 +97,9 @@ WantedBy=multi-user.target | |||||||
|       await this.checkConfigExists(); |       await this.checkConfigExists(); | ||||||
|        |        | ||||||
|       execSync('systemctl start nupst.service'); |       execSync('systemctl start nupst.service'); | ||||||
|       console.log('┌─ Service Status ─────────────────────────┐'); |       console.log('┌─ Service Status ───────────────────────────┐'); | ||||||
|       console.log('│ NUPST service started successfully'); |       console.log('│ NUPST service started successfully'); | ||||||
|       console.log('└──────────────────────────────────────────┘'); |       console.log('└────────────────────────────────────────────┘'); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       if (error.message === 'Configuration not found') { |       if (error.message === 'Configuration not found') { | ||||||
|         // Exit with error code since configuration is required |         // Exit with error code since configuration is required | ||||||
| @@ -190,20 +190,20 @@ WantedBy=multi-user.target | |||||||
|         timeout: Math.min(config.snmp.timeout, 10000) // Use at most 10 seconds for status check |         timeout: Math.min(config.snmp.timeout, 10000) // Use at most 10 seconds for status check | ||||||
|       }; |       }; | ||||||
|        |        | ||||||
|       console.log('┌─ Connecting to UPS... ────────────────────┐'); |       console.log('┌─ Connecting to UPS... ─────────────────────┐'); | ||||||
|       console.log(`│ Host: ${config.snmp.host}:${config.snmp.port}`); |       console.log(`│ Host: ${config.snmp.host}:${config.snmp.port}`); | ||||||
|       console.log(`│ UPS Model: ${config.snmp.upsModel || 'cyberpower'}`); |       console.log(`│ UPS Model: ${config.snmp.upsModel || 'cyberpower'}`); | ||||||
|       console.log('└──────────────────────────────────────────┘'); |       console.log('└────────────────────────────────────────────┘'); | ||||||
|        |        | ||||||
|       const status = await snmp.getUpsStatus(snmpConfig); |       const status = await snmp.getUpsStatus(snmpConfig); | ||||||
|        |        | ||||||
|       console.log('┌─ UPS Status ───────────────────────────────┐'); |       console.log('┌─ UPS Status ─────────────────────────────┐'); | ||||||
|       console.log(`│ Power Status: ${status.powerStatus}`); |       console.log(`│ Power Status: ${status.powerStatus}`); | ||||||
|       console.log(`│ Battery Capacity: ${status.batteryCapacity}%`); |       console.log(`│ Battery Capacity: ${status.batteryCapacity}%`); | ||||||
|       console.log(`│ Runtime Remaining: ${status.batteryRuntime} minutes`); |       console.log(`│ Runtime Remaining: ${status.batteryRuntime} minutes`); | ||||||
|       console.log('└──────────────────────────────────────────┘'); |       console.log('└──────────────────────────────────────────┘'); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       console.error('┌─ UPS Status ───────────────────────────────┐'); |       console.error('┌─ UPS Status ─────────────────────────────┐'); | ||||||
|       console.error(`│ Failed to retrieve UPS status: ${error.message}`); |       console.error(`│ Failed to retrieve UPS status: ${error.message}`); | ||||||
|       console.error('└──────────────────────────────────────────┘'); |       console.error('└──────────────────────────────────────────┘'); | ||||||
|     } |     } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user