Netgate Discussion Forum
    • Categories
    • Recent
    • Tags
    • Popular
    • Users
    • Search
    • Register
    • Login

    NUT Dashboard Widget (2.1.4)

    Scheduled Pinned Locked Moved webGUI
    40 Posts 12 Posters 10.7k Views
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • S
      SunStroke
      last edited by

      Hi, all.

      I`m added ajax autoupdates to widget.

      Step 1:
      in widget`s code replace strings (to get unique id):

      
      replace to
      
      

      Step 2:
      in file /usr/local/www/includes/functions.inc.php add:

      
      in function get_stats() add one string:
      ...
      	$stats['statepercent'] = get_pfstate(true);
      	$stats['nut'] = get_nutstats(true);   // Add this string
      	$stats = join("|", $stats);
      ...
      
      after function get_stats() add two functions:
      
      function secs2hms_($secs) {
      	if ($secs<0) return false;
      	$m = (int)($secs / 60); $s = $secs % 60;
      	$h = (int)($m / 60); $m = $m % 60;
      	return array($h, "h ", $m, "m ", $s, "s");
      }
      
      function get_nutstats() {
      	global $config;
      	$data = "";
      	$cmd = "";
      	$nut_config = $config['installedpackages']['nut']['config'][0];
      	// "Monitoring" field and get command to fetch ups data - [0] return array element
      	if($nut_config['monitor'] == 'local') {
      		$data = "Local UPS";
      		$cmd  = "upsc {$nut_config['name']}@localhost";
      	} elseif($nut_config['monitor'] == 'remote') {
      		$data = "Remote UPS";
      		$cmd  = "upsc {$nut_config['remotename']}@{$nut_config['remoteaddr']}";
      	} elseif($nut_config['monitor'] == 'snmp') {
      		$data = "SNMP UPS";
      		$cmd  = "upsc {$nut_config['snmpname']}@localhost";
      	}
      	// Find upsmon process
      	$running = ((int)exec('pgrep upsmon | wc -l') > 0) ? true : false;
      	if($running) {
      		// Fetch ups data
      		$handle = popen($cmd, 'r');
      	} elseif(isset($nut_config)) {
      		// No service running
      		if($nut_config['monitor'] == 'snmp') {
      			$condition = "NUT enabled but service not running!\nSNMP UPS may be unreachable.";
      		} else {
      			$condition = "NUT enabled but service not running!";
      		}
      	} else {
      		// No NUT package installed
      		$condition = "No NUT installed!";	
      	}
      	// Parse ups data
      	if($handle) {
      		$read = fread($handle, 4096);
      		pclose($handle);
      		$lines = explode("\n", $read);
      		$ups = array();
      		foreach($lines as $line) {
      			$line = explode(':', $line);
      			$ups[$line[0]] = trim($line[1]);
      		}
      		if(count($lines) == 1) 
      			$condition = "ERROR:Data stale!";
      	}
      	if(isset($condition)) {
      		// Drop all data and return "" if error (todo: return human readable result)
      		return "";
      	}
      	// "Model" field - [1] return array element
      	$data .= ":" . $ups['ups.model'];
      	// "Status" field - [2] return array element
      	$status = explode(' ', $ups['ups.status']);
      	foreach($status as $condition) {
      		if($disp_status) $disp_status .= ', ';
      		switch ($condition) {
      			case 'WAIT':
      				$disp_status .= 'Waiting';
      				break;
      			case 'OFF':
      				$disp_status .= 'Off Line';
      				break;
      			case 'OL':
      				$disp_status .= 'On Line';
      				break;
      			case 'OB':
      				$disp_status .= 'On Battery';
      				break;
      			case 'TRIM':
      				$disp_status .= 'SmartTrim';
      				break;
      			case 'BOOST':
      				$disp_status .= 'SmartBoost';
      				break;
      			case 'OVER':
      				$disp_status .= 'Overload';
      				break;
      			case 'LB':
      				$disp_status .= 'Battery Low';
      				break;
      			case 'RB':
      				$disp_status .= 'Replace Battery';
      				break;
      			case 'CAL':
      				$disp_status .= 'Calibration';
      				break;
      			case 'CHRG':
      				$disp_status .= 'Charging';
      				break;
      			default:
      				$disp_status .= $condition;
      				break;
      		}
      	}
      	$data .= ":" . $disp_status;
      	// "Battery Charge" bars and field - [3],[4],[5] return array elements
      	$data .= ":" . round($ups['battery.charge']);
      	$data .= ":" . (100 - round($ups['battery.charge']));
      	$data .= ":" . $ups['battery.charge'] . '%';
      	// "Time Remaning" field - [6] return array element
      	$data .= ":" . implode(secs2hms_($ups['battery.runtime']));
      	// "Battery Voltage or Battery Temp" field - [7] return array element
      	if($ups['battery.voltage'] > 0) {
      		$data .= ":" . $ups['battery.voltage'] . " V";
      	} elseif ($ups['ups.temperature'] > 0) {
      		$data .= ":" . $ups['ups.temperature'] . "&#176;C";
      	} else {
      		$data .= ":" . "";
      	}
      	// "Load" bars and field - [8],[9],[10] return array elements
      	$data .= ":" . round($ups['ups.load']);
      	$data .= ":" . (100 - round($ups['ups.load']));
      	$data .= ":" . $ups['ups.load'] . '%';
      	// "Input Voltage" field - [11] return array element
      	$data .= ":" . $ups['input.voltage'] . " V";
      	// "Output Voltage" field - [12] return array element
      	$data .= ":" . $ups['output.voltage'] . " V";
      
      	return $data;
      }
      
      

      Step 3:
      in file /usr/local/www/javascript/index/ajax.js add:

      
      in function stats(x) add one string:
      ...
              updateStateMeter(values[13]);
              updateNutStats(values[14]);   // Add this string
      }
      ...
      
      after function stats(x) add one function:
      
      function updateNutStats(x) {
      //console.log(x);
      	if (widgetActive("ups")){
      		upsdata_split = x.split(":");
      		if(upsdata_split.length) {
      			if(jQuery('#monitoring')) jQuery("#monitoring").html(upsdata_split[0]);
      			if(jQuery('#model')) jQuery("#model").html(upsdata_split[1]);
      			if(jQuery('#status')) jQuery("#status").html(upsdata_split[2]);
      			if(jQuery('#batwidtha')) jQuery("#batwidtha").css('width', upsdata_split[3] + 'px');
      			if(jQuery('#batwidthb')) jQuery("#batwidthb").css('width', upsdata_split[4] + 'px');
      			if(jQuery('#batmeter')) jQuery("#batmeter").html(upsdata_split[5]);
      			if(jQuery('#runtime')) jQuery("#runtime").html(upsdata_split[6]);
      			if(jQuery('#bvoltage')) jQuery("#bvoltage").html(upsdata_split[7]);
      			if(jQuery('#loadwidtha')) jQuery("#loadwidtha").css('width', upsdata_split[8] + 'px');
      			if(jQuery('#loadwidthb')) jQuery("#loadwidthb").css('width', upsdata_split[9] + 'px');
      			if(jQuery('#loadmeter')) jQuery("#loadmeter").html(upsdata_split[10]);
      			if(jQuery('#InputV')) jQuery("#InputV").html(upsdata_split[11]);
      			if(jQuery('#OutputV')) jQuery("#OutputV").html(upsdata_split[12]);
      		}
      	}
      }
      
      

      In my case works good. Do not know is this chages remain after reboot.
      And my code maybe need some rework to reflect all ups and NUT package status. Do it in the future.

      PS. Sorry for my bad english.

      Reviews and patches are welcomes.

      1 Reply Last reply Reply Quote 0
      • P
        phil.davis
        last edited by

        Since this seems useful for some people, one of you should go to https://github.com/pfsense/pfsense/tree/master/usr/local/www/widgets/widgets hit the plus sign, add these files and submit a pull request for review. If it is put in the official system then everyone can have the benefit without having to go through these manual install steps…

        As the Greek philosopher Isosceles used to say, "There are 3 sides to every triangle."
        If I helped you, then help someone else - buy someone a gift from the INF catalog http://secure.inf.org/gifts/usd/

        1 Reply Last reply Reply Quote 0
        • S
          SunStroke
          last edited by

          @phil.davis:

          … hit the plus sign, add these files and submit a pull request for review...

          Can i use various git tools (clone and then pull commits), or should i use on github "+" & "pull" if i nearly newbie to git. And should i put all modified files to "widget" tree, although not all these files from that path?

          1 Reply Last reply Reply Quote 0
          • P
            phil.davis
            last edited by

            Put the new files in the places they need to go in the final install - /usr/local/www/widgets/widgets /usr/local/www/widgets/include
            Edit other files in the place they are already.
            For this you can just use GitHub online, click the pencil to edit, the plus to add a new file. It will make a repository for you. It will do a commit every time you do something - so you end up with a few commits then make a pull request. It looks a bit messier having individual commits, but for small changes it is easy to do all online - no need to learn about git clone and so on.
            If you mess it all up you cam always delete your repository and start again apply the changes until you have a clean set of commits, then do the pull request.

            As the Greek philosopher Isosceles used to say, "There are 3 sides to every triangle."
            If I helped you, then help someone else - buy someone a gift from the INF catalog http://secure.inf.org/gifts/usd/

            1 Reply Last reply Reply Quote 0
            • K
              kejianshi
              last edited by

              I recently installed 2000W Enterprise UPS with 648 Watt/Hours of battery at each site where I have pfsense running for my personal use to keep the modems, switches and pfsense up during outages. (Very very long outages).  I hope these GUI changes become available as part of a normal package in pfsense.  It would be very convenient.

              1 Reply Last reply Reply Quote 0
              • S
                SunStroke
                last edited by

                Sent request to merge changes to main project repository.

                1 Reply Last reply Reply Quote 0
                • T
                  tweek
                  last edited by

                  Is there any progress on SunStroke's merge request?  I'm currently unaware of where I can track it & check on the status.  I'm eager to see this widget in a release.

                  1 Reply Last reply Reply Quote 0
                  • S
                    SunStroke
                    last edited by

                    @tweek:

                    Is there any progress on SunStroke's merge request?  I'm currently unaware of where I can track it & check on the status.  I'm eager to see this widget in a release.

                    No, my commits was not merged till now. You can install "System Patches" package and try to apply my patches individually. I done it for myself.

                    NUT widget (add main file) - http://github.com/pfsense/pfsense/commit/d9fcbc2048041285690cc28159753acdb0bbb7a3.patch
                    NUT widget (add 'inc' file) - http://github.com/pfsense/pfsense/commit/4dda40b25ee72d899add9fb18e41310dfcc53a17.patch
                    NUT widget ajax callback - http://github.com/pfsense/pfsense/commit/5c5e2a846b4c98cb67fe7828ddb131054d71a644.patch
                    NUT widget update func. - http://github.com/pfsense/pfsense/commit/225ea5746d83b6e79b96904e3f3dfb7c7b7defd4.patch

                    1 Reply Last reply Reply Quote 0
                    • D
                      doktornotor Banned
                      last edited by

                      @SunStroke:

                      No, my commits was not merged till now. You can install "System Patches" package and try to apply my patches individually. I done it for myself.

                      NUT widget (add main file) - http://github.com/pfsense/pfsense/commit/d9fcbc2048041285690cc28159753acdb0bbb7a3.patch
                      NUT widget (add 'inc' file) - http://github.com/pfsense/pfsense/commit/4dda40b25ee72d899add9fb18e41310dfcc53a17.patch
                      NUT widget ajax callback - http://github.com/pfsense/pfsense/commit/5c5e2a846b4c98cb67fe7828ddb131054d71a644.patch
                      NUT widget update func. - http://github.com/pfsense/pfsense/commit/225ea5746d83b6e79b96904e3f3dfb7c7b7defd4.patch

                      Thank's god, finally. Will test for sure as time allows.

                      General note: People, when you are doing some patches, please post diffs. Do not describe what to edit where. Total waste of time and error prone like hell.

                      Edit: Seems to work pretty well…

                      1 Reply Last reply Reply Quote 0
                      • GertjanG
                        Gertjan
                        last edited by

                        Same thing here.

                        Well done !!

                        No "help me" PM's please. Use the forum, the community will thank you.
                        Edit : and where are the logs ??

                        1 Reply Last reply Reply Quote 0
                        • R
                          r00f
                          last edited by

                          @SunStroke:

                          No, my commits was not merged till now. You can install "System Patches" package and try to apply my patches individually. I done it for myself.

                          NUT widget (add main file) - http://github.com/pfsense/pfsense/commit/d9fcbc2048041285690cc28159753acdb0bbb7a3.patch
                          NUT widget (add 'inc' file) - http://github.com/pfsense/pfsense/commit/4dda40b25ee72d899add9fb18e41310dfcc53a17.patch
                          NUT widget ajax callback - http://github.com/pfsense/pfsense/commit/5c5e2a846b4c98cb67fe7828ddb131054d71a644.patch
                          NUT widget update func. - http://github.com/pfsense/pfsense/commit/225ea5746d83b6e79b96904e3f3dfb7c7b7defd4.patch

                          When patching will be done, you need to remove "/usr/local/www/javascript/index/ajax.js.orig". Cos "/usr/local/www/index.php" load all scripts from "/usr/local/www/javascript/index/" directory and updates are not working.

                          1 Reply Last reply Reply Quote 0
                          • arrmoA
                            arrmo
                            last edited by

                            Sorry, perhaps a dumb question, but … is the only way to install (add 4 patches)?

                            Just trying to understand - this looks very good!

                            Thanks.

                            1 Reply Last reply Reply Quote 0
                            • S
                              SunStroke
                              last edited by

                              @arrmo:

                              Sorry, perhaps a dumb question, but … is the only way to install (add 4 patches)?

                              Just trying to understand - this looks very good!

                              Thanks.

                              Yes. At this moment add 4 pathes is the only way to install widget. If anyone (maybe i) restructures code to make standalone package, it will be included in main package repository.

                              1 Reply Last reply Reply Quote 0
                              • arrmoA
                                arrmo
                                last edited by

                                Works well, thanks! Just a couple minor comments,

                                • it would be good to have the title "UPS Status" to link to the appropriate page (like other widgets do)
                                • my UPS shows up fine, all works … but Status shows as "ERROR"?

                                Thanks again!

                                1 Reply Last reply Reply Quote 0
                                • S
                                  SunStroke
                                  last edited by

                                  @arrmo:

                                  Works well, thanks! Just a couple minor comments,

                                  • it would be good to have the title "UPS Status" to link to the appropriate page (like other widgets do)
                                  • my UPS shows up fine, all works … but Status shows as "ERROR"?

                                  Thanks again!

                                  • about link - it is already done in my repository, patch links slightly outdated
                                  • about "ERROR" - what error description is in "Model" column (screenshot, maybe?)?
                                  1 Reply Last reply Reply Quote 0
                                  • arrmoA
                                    arrmo
                                    last edited by

                                    Sounds good on the link, thanks!

                                    As for model / details, here you go (copy / paste, sorry) …

                                    Monitoring Model Status
                                    Local UPS UPS CP600 On Line
                                    Battery Charge Time Remain Battery Voltage
                                    left barred bargray barright bar
                                    100% 0h 52m 0s 13.3 V
                                    Load Input Voltage Output Voltage
                                    left barred bargray barright bar
                                    20% 121.0 V 121.0 V
                                    Monitoring Model Status
                                    Local UPS OL ERROR

                                    Thanks again - this is great!

                                    1 Reply Last reply Reply Quote 0
                                    • GertjanG
                                      Gertjan
                                      last edited by

                                      @arrmo:

                                      …..
                                      Monitoring Model Status
                                      Local UPS UPS CP600 On Line
                                      Battery Charge Time Remain Battery Voltage
                                      left barred bargray barright bar
                                      100% 0h 52m 0s 13.3 V
                                      Load Input Voltage Output Voltage
                                      left barred bargray barright bar
                                      20% 121.0 V 121.0 V
                                      Monitoring Model Status
                                      Local UPS OL ERROR

                                      If being seen this also.

                                      But then I discovered something strange with
                                      /usr/local/www/widgets/widgets/ups.widget.php
                                      It contained twice all the instructions !!
                                      It start with

                                      /*
                                      	$Id$
                                      	Copyright 2008 Seth Mos
                                      	Part of pfSense widgets (www.pfsense.com)
                                      ....
                                      
                                      

                                      If you find the same sequence half way the file, you have everything doubled … and the widget tries to show something that doesn't exists.

                                      I removed this second part - and the error disappeared.

                                      No "help me" PM's please. Use the forum, the community will thank you.
                                      Edit : and where are the logs ??

                                      1 Reply Last reply Reply Quote 0
                                      • arrmoA
                                        arrmo
                                        last edited by

                                        Yep, that fixes it - thanks!

                                        Sounds like some feedback for the person generating the patches - minor mod needed.

                                        Awesome!

                                        1 Reply Last reply Reply Quote 0
                                        • D
                                          doktornotor Banned
                                          last edited by

                                          The patches are just fine. You applied the same patch twice.

                                          1 Reply Last reply Reply Quote 0
                                          • arrmoA
                                            arrmo
                                            last edited by

                                            Hmmm … I did test and then apply, but I could have double clicked. That is possible, won't say it's not ... :).

                                            But a patch should fail if applied twice, no?

                                            Thanks!

                                            1 Reply Last reply Reply Quote 0
                                            • First post
                                              Last post
                                            Copyright 2025 Rubicon Communications LLC (Netgate). All rights reserved.