Compare commits

..

22 Commits

Author SHA1 Message Date
f860f39e59 2.6.14 2025-03-26 18:15:17 +00:00
fa4516de3b fix(systemd): Shorten closing log divider in systemd service installation output for consistent formatting. 2025-03-26 18:15:17 +00:00
539547beb8 2.6.13 2025-03-26 18:13:12 +00:00
6eb92959ec fix(cli): Fix CLI update output box formatting 2025-03-26 18:13:12 +00:00
4af9af0845 2.6.12 2025-03-26 18:10:49 +00:00
f7e12cdcbb fix(systemd): Adjust logging border in systemd service installation output 2025-03-26 18:10:49 +00:00
002498b91b 2.6.11 2025-03-26 18:08:43 +00:00
459911fe5f fix(cli, systemd): Adjust log formatting for consistent output in CLI and systemd commands 2025-03-26 18:08:43 +00:00
9859a02ea2 2.6.10 2025-03-26 18:04:12 +00:00
65444b6d25 fix(daemon): Adjust console log box formatting for consistent output in daemon status messages 2025-03-26 18:04:12 +00:00
d049e8741f 2.6.9 2025-03-26 18:00:55 +00:00
1123a99aea fix(cli): Improve console output formatting for status banners and logging messages 2025-03-26 18:00:54 +00:00
d01e878310 2.6.8 2025-03-26 17:49:50 +00:00
588aeabf4b fix(cli): Improve CLI formatting, refine debug option filtering, and remove unused dgram import in SNMP manager 2025-03-26 17:49:50 +00:00
87005e72f1 2.6.7 2025-03-26 15:56:31 +00:00
f799c2ee66 fix(setup.sh): Clarify net-snmp dependency installation message in setup.sh 2025-03-26 15:56:31 +00:00
1a029ba493 2.6.6 2025-03-26 15:53:38 +00:00
5b756dd223 fix(setup.sh): Improve setup script to detect and execute npm-cli.js directly using the Node.js binary 2025-03-26 15:53:38 +00:00
4cac599a58 2.6.5 2025-03-26 15:49:54 +00:00
be6a7314c3 fix(daemon, setup): Improve shutdown command detection and fallback logic; update setup script to use absolute Node/npm paths 2025-03-26 15:49:54 +00:00
83ba9c2611 2.6.4 2025-03-26 14:09:01 +00:00
22ab472e58 fix(setup): Improve installation process in setup script by cleaning up package files and ensuring a minimal net-snmp dependency installation. 2025-03-26 14:09:01 +00:00
10 changed files with 587 additions and 255 deletions

View File

@@ -1,5 +1,76 @@
# 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) ## 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. Update setup script to install only net-snmp dependency and create a minimal package-lock.json for better dependency control.

View File

@@ -1,6 +1,6 @@
{ {
"name": "@serve.zone/nupst", "name": "@serve.zone/nupst",
"version": "2.6.3", "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
View File

@@ -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':

104
setup.sh
View File

@@ -239,57 +239,89 @@ 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"
# Add our Node.js bin directory to the PATH temporarily # Set up Node.js binary path
NODE_BIN_DIR="$SCRIPT_DIR/vendor/$NODE_DIR/bin" NODE_BIN_DIR="$SCRIPT_DIR/vendor/$NODE_DIR/bin"
OLD_PATH="$PATH" NODE_BIN="$NODE_BIN_DIR/node"
export PATH="$NODE_BIN_DIR:$PATH" NPM_CLI_JS="$NODE_BIN_DIR/../lib/node_modules/npm/bin/npm-cli.js"
# Remove existing node_modules directory # Ensure we have executable permissions
echo "Removing existing node_modules directory..." chmod +x "$NODE_BIN"
# Make sure the npm-cli.js exists
if [ ! -f "$NPM_CLI_JS" ]; then
# Try to find npm-cli.js
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
echo "Found npm-cli.js at: $NPM_CLI_JS"
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 -rf "$SCRIPT_DIR/node_modules"
rm -f "$SCRIPT_DIR/package-lock.json"
# Install ONLY the net-snmp production dependency # Back up existing package.json if it exists
echo "Installing ONLY net-snmp production dependency..." if [ -f "$SCRIPT_DIR/package.json" ]; then
echo "Using Node.js binary from: $NODE_BIN_DIR" echo "Backing up existing package.json..."
echo "Node version: $(node --version)" cp "$SCRIPT_DIR/package.json" "$SCRIPT_DIR/package.json.bak"
echo "NPM version: $(npm --version)" fi
# Install just net-snmp directly, don't rely on package.json # Create a clean minimal package.json with ONLY net-snmp dependency
npm --prefix "$SCRIPT_DIR" install --no-save net-snmp@3.20.0 --no-audit --no-fund 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"
# Verify only net-snmp is installed # Install ONLY net-snmp
echo "Verifying only net-snmp is installed in node_modules..." echo "Installing ONLY net-snmp dependency (+ 2 subdependencies)..."
find "$SCRIPT_DIR/node_modules" -maxdepth 1 -type d | sort 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=$? INSTALL_STATUS=$?
if [ $INSTALL_STATUS -ne 0 ]; then if [ $INSTALL_STATUS -ne 0 ]; then
echo "Error: Failed to install net-snmp dependency. NUPST may not function correctly." echo "Error: Failed to install net-snmp dependency. NUPST may not function correctly."
echo "You can try to install dependencies manually by running:" echo "Restoring original package.json..."
echo "cd $SCRIPT_DIR && npm install net-snmp@3.20.0" mv "$SCRIPT_DIR/package.json.bak" "$SCRIPT_DIR/package.json"
exit 1
else else
echo "net-snmp dependency installed successfully." 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
# Create minimal package-lock.json if it doesn't exist # Remove backup if successful
if [ ! -f "$SCRIPT_DIR/package-lock.json" ]; then rm -f "$SCRIPT_DIR/package.json.bak"
echo "Creating minimal package-lock.json..."
echo '{
"name": "@serve.zone/nupst",
"version": "2.6.2",
"lockfileVersion": 2,
"requires": true,
"packages": {
"": {
"dependencies": {
"net-snmp": "3.20.0"
}
}
}
}' > "$SCRIPT_DIR/package-lock.json"
fi
fi fi
# Restore the original PATH # No temporary files to clean up
export PATH="$OLD_PATH"
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"

View File

@@ -3,6 +3,6 @@
*/ */
export const commitinfo = { export const commitinfo = {
name: '@serve.zone/nupst', name: '@serve.zone/nupst',
version: '2.6.3', 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'
} }

214
ts/cli.ts
View File

@@ -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,7 +422,9 @@ 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...');
@@ -426,11 +438,19 @@ Options:
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);

View File

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

View File

@@ -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) {

View File

@@ -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';

View File

@@ -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('└──────────────────────────────────────────┘');
} }