Merge remote-tracking branch 'mikaku/master' into amdenergy

This commit is contained in:
Andreas Bachlechner 2022-02-07 09:34:45 +01:00
commit d86820370c
13 changed files with 1228 additions and 60 deletions

13
Changes
View File

@ -1,3 +1,16 @@
N.NN.N - DD-MMM-YYYY
====================
- Added the 'additional_graph_name' configuration option. [#388]
- Fixed an error in 'pgsql.pm' if a non-privileged user is used to read the
statistics. [#384]
- Fixed the amdgpu section in 'monitorix.conf'. [#385]
- Fixed a change in the quotation marks introduced in 3.14 that prevented from
sending reports. [#391]
(thanks to Michael Bronzini, mb20 AT bronzeware.se for pointing this out)
- Fixed wrong index for image directory in 'amdgpu.pm', 'nvidiagpu.pm' and
'nvme.pm'. [#387]
3.14.0 - 18-Jan-2022 3.14.0 - 18-Jan-2022
==================== ====================
- Added a complete graph to support NVMe device statistics ('nvme.pm'). [#215] - Added a complete graph to support NVMe device statistics ('nvme.pm'). [#215]

View File

@ -86,6 +86,7 @@ install-bin:
$(INSTALL_DATA) lib/HTTPServer.pm "$(DESTDIR)$(LIBDIR)/HTTPServer.pm" $(INSTALL_DATA) lib/HTTPServer.pm "$(DESTDIR)$(LIBDIR)/HTTPServer.pm"
$(INSTALL_DATA) lib/icecast.pm "$(DESTDIR)$(LIBDIR)/icecast.pm" $(INSTALL_DATA) lib/icecast.pm "$(DESTDIR)$(LIBDIR)/icecast.pm"
$(INSTALL_DATA) lib/int.pm "$(DESTDIR)$(LIBDIR)/int.pm" $(INSTALL_DATA) lib/int.pm "$(DESTDIR)$(LIBDIR)/int.pm"
$(INSTALL_DATA) lib/intelrapl.pm "$(DESTDIR)$(LIBDIR)/intelrapl.pm"
$(INSTALL_DATA) lib/ipmi.pm "$(DESTDIR)$(LIBDIR)/ipmi.pm" $(INSTALL_DATA) lib/ipmi.pm "$(DESTDIR)$(LIBDIR)/ipmi.pm"
$(INSTALL_DATA) lib/kern.pm "$(DESTDIR)$(LIBDIR)/kern.pm" $(INSTALL_DATA) lib/kern.pm "$(DESTDIR)$(LIBDIR)/kern.pm"
$(INSTALL_DATA) lib/libvirt.pm "$(DESTDIR)$(LIBDIR)/libvirt.pm" $(INSTALL_DATA) lib/libvirt.pm "$(DESTDIR)$(LIBDIR)/libvirt.pm"

View File

@ -679,7 +679,7 @@ sub amdgpu_cgi {
$cdef_sensor_allvalues .= ",0,GT,1,UNKN,IF"; $cdef_sensor_allvalues .= ",0,GT,1,UNKN,IF";
} }
my $plot_title = $config->{graphs}->{'_amdgpu' . ($n_plot + 1)}; my $plot_title = $config->{graphs}->{'_amdgpu' . ($n_plot + 1)};
$pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$e * 3 + $n_plot]", $pic = $rrd{$version}->("$IMG_DIR" . $IMG[$e * $number_of_sensor_values_in_use + $n_plot],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})", "--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc", "--imgformat=$imgfmt_uc",
@ -697,10 +697,10 @@ sub amdgpu_cgi {
@CDEF, @CDEF,
@tmp); @tmp);
$err = RRDs::error; $err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . "$IMG[$e * 3 + $n_plot]: $err\n") if $err; push(@output, "ERROR: while graphing $IMG_DIR" . $IMG[$e * $number_of_sensor_values_in_use + $n_plot]. ": $err\n") if $err;
if(lc($config->{enable_zoom}) eq "y") { if(lc($config->{enable_zoom}) eq "y") {
($width, $height) = split('x', $config->{graph_size}->{zoom}); ($width, $height) = split('x', $config->{graph_size}->{zoom});
$picz = $rrd{$version}->("$IMG_DIR" . "$IMGz[$e * 3 + $n_plot]", $picz = $rrd{$version}->("$IMG_DIR" . $IMGz[$e * $number_of_sensor_values_in_use + $n_plot],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})", "--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc", "--imgformat=$imgfmt_uc",
@ -719,13 +719,13 @@ sub amdgpu_cgi {
@CDEF, @CDEF,
@tmpz); @tmpz);
$err = RRDs::error; $err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . "$IMGz[$e * 3 + $n_plot]: $err\n") if $err; push(@output, "ERROR: while graphing $IMG_DIR" . $IMGz[$e * $number_of_sensor_values_in_use + $n_plot]. ": $err\n") if $err;
} }
$e2 = $e + $n_plot + 1; $e2 = $e + $n_plot + 1;
if($title || ($silent =~ /imagetag/ && $graph =~ /amdgpu$e2/)) { if($title || ($silent =~ /imagetag/ && $graph =~ /amdgpu$e2/)) {
if(lc($config->{enable_zoom}) eq "y") { if(lc($config->{enable_zoom}) eq "y") {
if(lc($config->{disable_javascript_void}) eq "y") { if(lc($config->{disable_javascript_void}) eq "y") {
push(@output, " " . picz_a_element(config => $config, IMGz => $IMGz[$e * 3 + $n_plot], IMG => $IMG[$e * 3 + $n_plot]) . "\n"); push(@output, " " . picz_a_element(config => $config, IMGz => $IMGz[$e * $number_of_sensor_values_in_use + $n_plot], IMG => $IMG[$e * $number_of_sensor_values_in_use + $n_plot]) . "\n");
} else { } else {
if($version eq "new") { if($version eq "new") {
$picz_width = $picz->{image_width} * $config->{global_zoom}; $picz_width = $picz->{image_width} * $config->{global_zoom};
@ -734,10 +734,10 @@ sub amdgpu_cgi {
$picz_width = $width + 115; $picz_width = $width + 115;
$picz_height = $height + 100; $picz_height = $height + 100;
} }
push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * 3 + $n_plot], IMG => $IMG[$e * 3 + $n_plot]) . "\n"); push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * $number_of_sensor_values_in_use + $n_plot], IMG => $IMG[$e * $number_of_sensor_values_in_use + $n_plot]) . "\n");
} }
} else { } else {
push(@output, " " . img_element(config => $config, IMG => $IMG[$e * 3 + $n_plot]) . "\n"); push(@output, " " . img_element(config => $config, IMG => $IMG[$e * $number_of_sensor_values_in_use + $n_plot]) . "\n");
} }
} }
} }

View File

@ -53,6 +53,8 @@ sub du_init {
$heartbeat = 2 * $refresh_interval; $heartbeat = 2 * $refresh_interval;
} }
my @disk_list = split(',', $du->{list});
if(-e $rrd) { if(-e $rrd) {
$info = RRDs::info($rrd); $info = RRDs::info($rrd);
for my $key (keys %$info) { for my $key (keys %$info) {
@ -76,8 +78,8 @@ sub du_init {
} }
} }
} }
if(scalar(@ds) / 9 != scalar(my @fl = split(',', $du->{list}))) { if(scalar(@ds) / 9 != scalar(@disk_list)) {
logger("$myself: Detected size mismatch between 'list' (" . scalar(my @fl = split(',', $du->{list})) . ") and $rrd (" . scalar(@ds) / 9 . "). Resizing it accordingly. All historical data will be lost. Backup file created."); logger("$myself: Detected size mismatch between 'list' (" . scalar(@disk_list) . ") and $rrd (" . scalar(@ds) / 9 . "). Resizing it accordingly. All historical data will be lost. Backup file created.");
rename($rrd, "$rrd.bak"); rename($rrd, "$rrd.bak");
} }
if(scalar(@rra) < 12 + (4 * $config->{max_historic_years})) { if(scalar(@rra) < 12 + (4 * $config->{max_historic_years})) {
@ -106,7 +108,7 @@ sub du_init {
push(@max, "RRA:MAX:0.5:1440:" . (365 * $n)); push(@max, "RRA:MAX:0.5:1440:" . (365 * $n));
push(@last, "RRA:LAST:0.5:1440:" . (365 * $n)); push(@last, "RRA:LAST:0.5:1440:" . (365 * $n));
} }
for($n = 0; $n < scalar(my @fl = split(',', $du->{list})); $n++) { for($n = 0; $n < scalar(@disk_list); $n++) {
push(@tmp, "DS:du" . $n . "_d1:GAUGE:" . $heartbeat . ":0:U"); push(@tmp, "DS:du" . $n . "_d1:GAUGE:" . $heartbeat . ":0:U");
push(@tmp, "DS:du" . $n . "_d2:GAUGE:" . $heartbeat . ":0:U"); push(@tmp, "DS:du" . $n . "_d2:GAUGE:" . $heartbeat . ":0:U");
push(@tmp, "DS:du" . $n . "_d3:GAUGE:" . $heartbeat . ":0:U"); push(@tmp, "DS:du" . $n . "_d3:GAUGE:" . $heartbeat . ":0:U");
@ -177,8 +179,10 @@ sub du_update {
return if(($min + 60 * $hour) % int($refresh_interval / 60)); return if(($min + 60 * $hour) % int($refresh_interval / 60));
} }
my @disk_list = split(',', $du->{list});
my $e = 0; my $e = 0;
while($e < scalar(my @dl = split(',', $du->{list}))) { while($e < scalar(@disk_list)) {
my $type; my $type;
my $e2 = 0; my $e2 = 0;
@ -215,7 +219,7 @@ sub du_update {
} }
$e = 0; $e = 0;
while($e < scalar(my @dl = split(',', $du->{list}))) { while($e < scalar(@disk_list)) {
for($n = 0; $n < 9; $n++) { for($n = 0; $n < 9; $n++) {
$dirs[$e][$n] = 0 unless defined $dirs[$e][$n]; $dirs[$e][$n] = 0 unless defined $dirs[$e][$n];
$rrdata .= ":" . $dirs[$e][$n]; $rrdata .= ":" . $dirs[$e][$n];
@ -291,7 +295,7 @@ sub du_cgi {
} }
$title = !$silent ? $title : ""; $title = !$silent ? $title : "";
my @disk_list = split(',', $du->{list});
# text mode # text mode
# #
@ -312,7 +316,7 @@ sub du_cgi {
my $line3; my $line3;
push(@output, " <pre style='font-size: 12px; color: $colors->{fg_color}';>\n"); push(@output, " <pre style='font-size: 12px; color: $colors->{fg_color}';>\n");
push(@output, " "); push(@output, " ");
for($n = 0; $n < scalar(my @dl = split(',', $du->{list})); $n++) { for($n = 0; $n < scalar(@disk_list); $n++) {
$line1 = ""; $line1 = "";
foreach my $i (split(',', $du->{desc}->{$n})) { foreach my $i (split(',', $du->{desc}->{$n})) {
$i = trim($i); $i = trim($i);
@ -324,7 +328,7 @@ sub du_cgi {
} }
if($line1) { if($line1) {
my $i = length($line1); my $i = length($line1);
push(@output, sprintf(sprintf("%${i}s", sprintf("%s", trim($dl[$n]))))); push(@output, sprintf(sprintf("%${i}s", sprintf("%s", trim($disk_list[$n])))));
} }
} }
push(@output, "\n"); push(@output, "\n");
@ -341,7 +345,7 @@ sub du_cgi {
$line = @$data[$n]; $line = @$data[$n];
$time = $time - (1 / $tf->{ts}); $time = $time - (1 / $tf->{ts});
push(@output, sprintf(" %2d$tf->{tc} ", $time)); push(@output, sprintf(" %2d$tf->{tc} ", $time));
for($n2 = 0; $n2 < scalar(my @dl = split(',', $du->{list})); $n2++) { for($n2 = 0; $n2 < scalar(@disk_list); $n2++) {
$n3 = 0; $n3 = 0;
foreach my $i (split(',', $du->{desc}->{$n2})) { foreach my $i (split(',', $du->{desc}->{$n2})) {
$from = $n2 * 9 + $n3++; $from = $n2 * 9 + $n3++;
@ -375,7 +379,7 @@ sub du_cgi {
$u = ""; $u = "";
} }
for($n = 0; $n < scalar(my @dl = split(',', $du->{list})); $n++) { for($n = 0; $n < scalar(@disk_list); $n++) {
$str = $u . $package . $n . "." . $tf->{when} . ".$imgfmt_lc"; $str = $u . $package . $n . "." . $tf->{when} . ".$imgfmt_lc";
push(@IMG, $str); push(@IMG, $str);
unlink("$IMG_DIR" . $str); unlink("$IMG_DIR" . $str);
@ -386,21 +390,41 @@ sub du_cgi {
} }
} }
my $graphs_per_row = $du->{graphs_per_row};
my @linpad =(0) x scalar(@disk_list);
if ($graphs_per_row > 1) {
for(my $n = 0; $n < scalar(@disk_list); $n++) {
my @ls = split(',', $du->{desc}->{$n});
$linpad[$n] = scalar(@ls);
}
for(my $n = 0; $n < scalar(@linpad); $n++) {
if ($n % $graphs_per_row == 0) {
my $max_number_of_lines = 0;
for (my $sub_n = $n; $sub_n < min($n + $graphs_per_row, scalar(@linpad)); $sub_n++) {
$max_number_of_lines = max($max_number_of_lines, $linpad[$sub_n]);
}
for (my $sub_n = $n; $sub_n < min($n + $graphs_per_row, scalar(@linpad)); $sub_n++) {
$linpad[$sub_n] = $max_number_of_lines;
}
}
}
}
@riglim = @{setup_riglim($rigid[0], $limit[0])}; @riglim = @{setup_riglim($rigid[0], $limit[0])};
$n = 0; $n = 0;
while($n < scalar(my @dl = split(',', $du->{list}))) { while($n < scalar(@disk_list)) {
if($title) { if($title) {
if($n == 0) { if($n == 0) {
push(@output, main::graph_header($title, $du->{graphs_per_row})); push(@output, main::graph_header($title, $graphs_per_row));
} }
push(@output, " <tr>\n"); push(@output, " <tr>\n");
} }
for($n2 = 0; $n2 < $du->{graphs_per_row}; $n2++) { for($n2 = 0; $n2 < $graphs_per_row; $n2++) {
my $type; my $type;
my @DEF0; my @DEF0;
my @CDEF0; my @CDEF0;
last unless $n < scalar(my @dl = split(',', $du->{list})); last unless $n < scalar(@disk_list);
if($title) { if($title) {
push(@output, " <td>\n"); push(@output, " <td>\n");
} }
@ -460,7 +484,7 @@ sub du_cgi {
push(@tmpz, "LINE2:d" . ($e + 1) . $LC[$e] . ":$str"); push(@tmpz, "LINE2:d" . ($e + 1) . $LC[$e] . ":$str");
$e++; $e++;
} }
while($e < 9) { while($e < $linpad[$n]) {
push(@tmp, "COMMENT: \\n"); push(@tmp, "COMMENT: \\n");
$e++; $e++;
} }
@ -470,7 +494,7 @@ sub du_cgi {
push(@CDEF, "CDEF:wrongdata=allvalues,UN,INF,UNKN,IF"); push(@CDEF, "CDEF:wrongdata=allvalues,UN,INF,UNKN,IF");
} }
($width, $height) = split('x', $config->{graph_size}->{medium}); ($width, $height) = split('x', $config->{graph_size}->{medium});
$str = substr(trim($dl[$n]), 0, 25); $str = substr(trim($disk_list[$n]), 0, 25);
$pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$n]", $pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$n]",
"--title=$str ($tf->{nwhen}$tf->{twhen})", "--title=$str ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",

View File

@ -1,7 +1,7 @@
# #
# Monitorix - A lightweight system monitoring tool. # Monitorix - A lightweight system monitoring tool.
# #
# Copyright (C) 2005-2017 by Jordi Sanfeliu <jordi@fibranet.cat> # Copyright (C) 2005-2022 by Jordi Sanfeliu <jordi@fibranet.cat>
# #
# This program is free software; you can redistribute it and/or modify # This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by # it under the terms of the GNU General Public License as published by
@ -83,7 +83,7 @@ EOF
<a href='https://www.monitorix.org'><img src='cid:image_logo' border='0'></a> <a href='https://www.monitorix.org'><img src='cid:image_logo' border='0'></a>
<br> <br>
<font face='Verdana, sans-serif' color='000000' size='-2'> <font face='Verdana, sans-serif' color='000000' size='-2'>
Copyright &copy; 2005-2017 Jordi Sanfeliu Copyright &copy; 2005-2022 Jordi Sanfeliu
</font> </font>
</body> </body>
</html> </html>
@ -135,7 +135,7 @@ EOF
foreach (split('\n', $graph)) { foreach (split('\n', $graph)) {
if(/<img src=/) { if(/<img src=/) {
push(@tmp, "<img src='cid:image_$g$e$n.$imgfmt_lc' border='0'>"); push(@tmp, "<img src='cid:image_$g$e$n.$imgfmt_lc' border='0'>");
($url) = $_ =~ m/<img src='(.*?)' /; ($url) = $_ =~ m/<img src=['"](.*?)['"] /;
my $uri = URI->new($url); my $uri = URI->new($url);
my $path = $uri->path || ""; my $path = $uri->path || "";
$response = $ua->request(HTTP::Request->new('GET', "$prefix$path")); $response = $ua->request(HTTP::Request->new('GET', "$prefix$path"));

798
lib/intelrapl.pm Normal file
View File

@ -0,0 +1,798 @@
#
# Monitorix - A lightweight system monitoring tool.
#
# Copyright (C) 2005-2022 by Jordi Sanfeliu <jordi@fibranet.cat>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
#
package intelrapl;
use strict;
use warnings;
use Monitorix;
use RRDs;
use Time::HiRes;
use Cwd 'abs_path';
use File::Basename;
use Exporter 'import';
our @EXPORT = qw(intelrapl_init intelrapl_update intelrapl_cgi);
my $epoc_identifier = "last_epoc";
my $val_identifier = "last_val";
my $list_delimiter = ",";
sub get_max_number_of_values_per_group {
my ($intelrapl) = @_;
my $default_max_number_of_values_per_group = 10; # Can be overwritten via config file but changes will break history.
if(defined($intelrapl->{max_number_of_values_per_group})) {
return $intelrapl->{max_number_of_values_per_group};
}
return $default_max_number_of_values_per_group;
}
sub hue_to_rgb {
my ($p, $q, $t) = @_;
if($t < 0) {
$t += 1;
}
if($t > 1) {
$t -= 1;
}
if($t < 1/6) {
return $p + ($q - $p) * 6 * $t;
}
if($t < 1/2) {
return $q;
}
if($t < 2/3) {
return $p + ($q - $p) * (2/3 - $t) * 6;
}
return $p;
}
sub hsl_to_rgb {
my ($H, $S, $L) = @_;
my $h = $H/360;
my $s = $S/100;
my $l = $L/100;
my ($r, $g, $b);
if($s == 0) {
$r = $g = $b = $l;
} else {
my $q = $l < 0.5 ? $l * (1 + $s) : $l + $s - $l * $s;
my $p = 2 * $l - $q;
$r = hue_to_rgb($p, $q, $h + 1/3);
$g = hue_to_rgb($p, $q, $h);
$b = hue_to_rgb($p, $q, $h - 1/3);
}
return (round($r * 255), round($g * 255), round($b * 255));
}
sub line_color {
my ($n) = @_;
my @LC = (
"#44EEEE",
"#EE44EE",
"#44EE44",
"#4444EE",
"#ff9100",
"#a600ff",
"#EEEE00",
"#448844",
"#EE4444",
"#EE44EE",
);
if ($n < scalar(@LC)) {
return $LC[$n];
}
my $h_step = 31;
my $h_min = ($n-1) * $h_step;
my $h_max = $h_min + $h_step;
my ($r,$g,$b) = hsl_to_rgb($n*($h_max-$h_min)+$h_min, 100, 50);
return sprintf("#%02x%02x%02x",$r,$g,$b);
}
sub measure {
my ($myself, $config, $intelrapl) = @_;
my $use_nan_for_missing_data = lc($intelrapl->{use_nan_for_missing_data} || "") eq "y" ? 1 : 0;
my @sensors_all;
my $rrdata = "N";
my $max_number_of_values_per_group = get_max_number_of_values_per_group($intelrapl);
foreach my $k (sort keys %{$intelrapl->{list}}) {
my $package_sensor;
if(defined($intelrapl->{package_sensors}) && defined($intelrapl->{package_sensors}->{$k})) {
$package_sensor = trim($intelrapl->{package_sensors}->{$k});
}
my $package_index;
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$k});
my @sensors = ($use_nan_for_missing_data ? (0+"nan") : 0) x $max_number_of_values_per_group;
for(my $n = 0; $n < min(scalar(@sensor_group), $max_number_of_values_per_group); $n++) {
my $str = trim($sensor_group[$n] || "");
my $sensor_path = trim($intelrapl->{sensors}->{$str} || "");
chomp($sensor_path);
my $last_epoc = ($config->{intelrapl_hist}->{$k}->{$n}->{$epoc_identifier} || 0);
my $epoc = Time::HiRes::time();
$config->{intelrapl_hist}->{$k}->{$n}->{$epoc_identifier} = $epoc;
if ($sensor_path ne "") {
my $sensor_file = $sensor_path;
if(open(IN, $sensor_file)) {
my $val = <IN>;
close(IN);
$val = trim($val);
chomp($val);
my $last_sensor_val = ($config->{intelrapl_hist}->{$k}->{$n}->{$val_identifier} || 0);
my $sensor_val = $val;
$config->{intelrapl_hist}->{$k}->{$n}->{$val_identifier} = $sensor_val;
if ($last_epoc ne 0 && $sensor_val >= $last_sensor_val) {
$sensors[$n] = ($sensor_val - $last_sensor_val) / ($epoc - $last_epoc); # Conversion from muJoule to muWatt during the time interval.
if (defined($package_sensor) && $str eq $package_sensor) {
$package_index = $n;
}
}
} else {
logger("$myself: ERROR: unable to open '$sensor_file'.");
}
}
}
push(@sensors_all, @sensors);
# intelrapl alert
if(defined($intelrapl->{alerts}) && lc($intelrapl->{alerts}->{packagepower_enabled}) eq "y") {
my $sensor_index = $package_index;
if (defined($sensor_index)) {
$config->{intelrapl_hist_alert1}->{$k} = 0 if(!$config->{intelrapl_hist_alert1}->{$k});
if($sensors[$sensor_index] >= $intelrapl->{alerts}->{packagepower_threshold} && $config->{intelrapl_hist_alert1}->{$k} < $sensors[$sensor_index]) {
if(-x $intelrapl->{alerts}->{packagepower_script}) {
logger("$myself: ALERT: executing script '$intelrapl->{alerts}->{packagepower_script}'.");
system($intelrapl->{alerts}->{packagepower_script} . " " .$intelrapl->{alerts}->{packagepower_timeintvl} . " " . $intelrapl->{alerts}->{packagepower_threshold} . " " . $sensors[$sensor_index]);
} else {
logger("$myself: ERROR: script '$intelrapl->{alerts}->{packagepower_script}' doesn't exist or don't has execution permissions.");
}
$config->{intelrapl_hist_alert1}->{$k} = $sensors[$sensor_index];
}
} else {
logger("$myself: ERROR: could not find $package_sensor in sensors. Alarms will not work!");
}
}
}
foreach(@sensors_all) {
$rrdata .= ":$_";
}
return $rrdata;
}
sub intelrapl_init {
my $myself = (caller(0))[3];
my ($package, $config, $debug) = @_;
my $rrd = $config->{base_lib} . $package . ".rrd";
my $intelrapl = $config->{intelrapl};
my $info;
my @ds;
my @rra;
my @tmp;
my $n;
my @average;
my @min;
my @max;
my @last;
my $max_number_of_values_per_group = get_max_number_of_values_per_group($intelrapl);
foreach my $k (sort keys %{$intelrapl->{list}}) {
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$k});
for(my $n = 0; $n < min(scalar(@sensor_group), $max_number_of_values_per_group); $n++) {
my $str = trim($sensor_group[$n] || "");
my $sensor_path = trim($intelrapl->{sensors}->{$str} || "");
chomp($sensor_path);
if ($sensor_path ne "") {
my $sensor_file = $sensor_path;
unless(-e $sensor_file) {
logger("$myself: ERROR: invalid or inexistent device name '$sensor_file'.");
if(lc($intelrapl->{accept_invalid} || "") ne "y") {
logger("$myself: 'accept_invalid' option is not set.");
logger("$myself: WARNING: initialization aborted.");
return;
}
}
}
}
}
if(-e $rrd) {
my $rrd_n_groups = 0;
my $rrd_n_groups_times_n_values = 0;
$info = RRDs::info($rrd);
for my $key (keys %$info) {
if(index($key, 'ds[') == 0) {
if(index($key, '.type') != -1) {
push(@ds, substr($key, 3, index($key, ']') - 3));
}
if(index($key, '_val0].index') != -1) {
$rrd_n_groups += 1;
}
if(index($key, '.index') != -1) {
$rrd_n_groups_times_n_values += 1;
}
}
if(index($key, 'rra[') == 0) {
if(index($key, '.rows') != -1) {
push(@rra, substr($key, 4, index($key, ']') - 4));
}
}
}
my $total_number_of_groups = 0;
foreach my $k (sort keys %{$intelrapl->{list}}) {
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$k});
$total_number_of_groups += scalar(@sensor_group);
}
if(scalar(@ds) / ($rrd_n_groups_times_n_values / $rrd_n_groups) != keys(%{$intelrapl->{list}})) {
logger("$myself: Detected size mismatch between <list>...</list> (" . keys(%{$intelrapl->{list}}) . ") and $rrd (" . scalar(@ds) / ($rrd_n_groups_times_n_values / $rrd_n_groups) . "). Resizing it accordingly. All historical data will be lost. Backup file created.");
rename($rrd, "$rrd.bak");
}
if($rrd_n_groups_times_n_values / $rrd_n_groups < $max_number_of_values_per_group) {
logger("$myself: Detected size mismatch between max_number_of_values_per_group (" . $max_number_of_values_per_group . ") and $rrd (" . ($rrd_n_groups_times_n_values / $rrd_n_groups) . "). Resizing it accordingly. All historical data will be lost. Backup file created.");
rename($rrd, "$rrd.bak");
}
if(scalar(@rra) < 12 + (4 * $config->{max_historic_years})) {
logger("$myself: Detected size mismatch between 'max_historic_years' (" . $config->{max_historic_years} . ") and $rrd (" . ((scalar(@rra) -12) / 4) . "). Resizing it accordingly. All historical data will be lost. Backup file created.");
rename($rrd, "$rrd.bak");
}
}
if(!(-e $rrd)) {
logger("Creating '$rrd' file.");
for($n = 1; $n <= $config->{max_historic_years}; $n++) {
push(@average, "RRA:AVERAGE:0.5:1440:" . (365 * $n));
push(@min, "RRA:MIN:0.5:1440:" . (365 * $n));
push(@max, "RRA:MAX:0.5:1440:" . (365 * $n));
push(@last, "RRA:LAST:0.5:1440:" . (365 * $n));
}
for(my $k = 0; $k < keys(%{$intelrapl->{list}}); $k++) {
for($n = 0; $n < $max_number_of_values_per_group; $n++) {
push(@tmp, "DS:rapl" . $k . "_val" . $n . ":GAUGE:120:0:U");
}
}
eval {
RRDs::create($rrd,
"--step=60",
@tmp,
"RRA:AVERAGE:0.5:1:1440",
"RRA:AVERAGE:0.5:30:336",
"RRA:AVERAGE:0.5:60:744",
@average,
"RRA:MIN:0.5:1:1440",
"RRA:MIN:0.5:30:336",
"RRA:MIN:0.5:60:744",
@min,
"RRA:MAX:0.5:1:1440",
"RRA:MAX:0.5:30:336",
"RRA:MAX:0.5:60:744",
@max,
"RRA:LAST:0.5:1:1440",
"RRA:LAST:0.5:30:336",
"RRA:LAST:0.5:60:744",
@last,
);
};
my $err = RRDs::error;
if($@ || $err) {
logger("$@") unless !$@;
if($err) {
logger("ERROR: while creating $rrd: $err");
if($err eq "RRDs::error") {
logger("... is the RRDtool Perl package installed?");
}
}
return;
}
}
# check dependencies
if(defined($intelrapl->{alerts}) && lc($intelrapl->{alerts}->{packagepower_enabled} || "") eq "y") {
if(! -x $intelrapl->{alerts}->{packagepower_script}) {
logger("$myself: ERROR: script '$intelrapl->{alerts}->{packagepower_script}' doesn't exist or don't has execution permissions.");
}
}
$config->{intelrapl_hist_alert1} = ();
$config->{intelrapl_hist} = ();
push(@{$config->{func_update}}, $package);
measure($myself, $config, $intelrapl);
logger("$myself: Ok") if $debug;
}
sub intelrapl_update {
my $myself = (caller(0))[3];
my ($package, $config, $debug) = @_;
my $rrd = $config->{base_lib} . $package . ".rrd";
my $intelrapl = $config->{intelrapl};
my $rrdata = measure($myself, $config, $intelrapl);
RRDs::update($rrd, $rrdata);
logger("$myself: $rrdata") if $debug;
my $err = RRDs::error;
logger("ERROR: while updating $rrd: $err") if $err;
}
sub round {
my ($float) = @_;
return int($float + $float/abs($float*2 || 1));
}
sub intelrapl_cgi {
my ($package, $config, $cgi) = @_;
my @output;
my $intelrapl = $config->{intelrapl};
my @rigid = split(',', ($intelrapl->{rigid} || ""));
my @limit = split(',', ($intelrapl->{limit} || ""));
my $tf = $cgi->{tf};
my $colors = $cgi->{colors};
my $graph = $cgi->{graph};
my $silent = $cgi->{silent};
my $zoom = "--zoom=" . $config->{global_zoom};
my %rrd = (
'new' => \&RRDs::graphv,
'old' => \&RRDs::graph,
);
my $version = "new";
my @full_size_mode;
my $pic;
my $picz;
my $picz_width;
my $picz_height;
my $u = "";
my $width;
my $height;
my @extra;
my @riglim;
my @IMG;
my @IMGz;
my @tmp;
my @tmpz;
my @CDEF;
my $n;
my $n2;
my $e;
my $e2;
my $str;
my $err;
$version = "old" if $RRDs::VERSION < 1.3;
push(@full_size_mode, "--full-size-mode") if $RRDs::VERSION > 1.3;
my $rrd = $config->{base_lib} . $package . ".rrd";
my $title = $config->{graph_title}->{$package};
my $IMG_DIR = $config->{base_dir} . "/" . $config->{imgs_dir};
my $imgfmt_uc = uc($config->{image_format});
my $imgfmt_lc = lc($config->{image_format});
foreach my $i (split(',', $config->{rrdtool_extra_options} || "")) {
push(@extra, trim($i)) if trim($i);
}
$title = !$silent ? $title : "";
my $gap_on_all_nan = lc($intelrapl->{gap_on_all_nan} || "") eq "y" ? 1 : 0;
my $max_number_of_values_per_group = get_max_number_of_values_per_group($intelrapl);
# text mode
#
if(lc($config->{iface_mode}) eq "text") {
if($title) {
push(@output, main::graph_header($title, 2));
push(@output, " <tr>\n");
push(@output, " <td>\n");
}
my (undef, undef, undef, $data) = RRDs::fetch("$rrd",
"--resolution=$tf->{res}",
"--start=-$tf->{nwhen}$tf->{twhen}",
"AVERAGE");
$err = RRDs::error;
push(@output, "ERROR: while fetching $rrd: $err\n") if $err;
my $line1;
my $line2;
my $line3;
push(@output, " <pre style='font-size: 12px; color: $colors->{fg_color}';>\n");
foreach my $k (sort keys %{$intelrapl->{list}}) {
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$k});
for($n = 0; $n < min(scalar(@sensor_group), $max_number_of_values_per_group); $n++) {
$str = sprintf(" RAPL power %d ", $n + 1);
$line1 .= $str;
$str = sprintf(" Sensor values ");
$line2 .= $str;
$line3 .= "----------------------";
}
}
push(@output, " $line1\n");
push(@output, "Time $line2\n");
push(@output, "-----$line3\n");
my $line;
my @row;
my $time;
my $from;
my $to;
for($n = 0, $time = $tf->{tb}; $n < ($tf->{tb} * $tf->{ts}); $n++) {
$line = @$data[$n];
$time = $time - (1 / $tf->{ts});
push(@output, sprintf(" %2d$tf->{tc} ", $time));
$e = 0;
foreach my $k (sort keys %{$intelrapl->{list}}) {
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$k});
for($n2 = 0; $n2 < min(scalar(@sensor_group), $max_number_of_values_per_group); $n2++) {
my $str = trim($sensor_group[$n] || "");
$from = ($e * scalar(@sensor_group) + $n2);
$to = $from + 3;
my @sensor_values = @$line[$from..$to];
@row = (celsius_to($config, $sensor_values[0]), @sensor_values[1, -1]);
my $format_string = "%7.0f" x scalar(@row);
push(@output, sprintf(" " . $format_string. " ", @row));
}
$e++;
}
push(@output, "\n");
}
push(@output, " </pre>\n");
if($title) {
push(@output, " </td>\n");
push(@output, " </tr>\n");
push(@output, main::graph_footer());
}
push(@output, " <br>\n");
return @output;
}
# graph mode
#
if($silent eq "yes" || $silent eq "imagetag") {
$colors->{fg_color} = "#000000"; # visible color for text mode
$u = "_";
}
if($silent eq "imagetagbig") {
$colors->{fg_color} = "#000000"; # visible color for text mode
$u = "";
}
my $plots_per_list_item = 1;
for($n = 0; $n < keys(%{$intelrapl->{list}}); $n++) {
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$n});
for($n2 = 0; $n2 < $plots_per_list_item; $n2++) {
$str = $u . $package . $n . $n2 . "." . $tf->{when} . ".$imgfmt_lc";
push(@IMG, $str);
unlink("$IMG_DIR" . $str);
if(lc($config->{enable_zoom}) eq "y") {
$str = $u . $package . $n . $n2 . "z." . $tf->{when} . ".$imgfmt_lc";
push(@IMGz, $str);
unlink("$IMG_DIR" . $str);
}
}
}
$e = 0;
foreach my $k (sort keys %{$intelrapl->{list}}) {
my @sensor_group = split($list_delimiter, $intelrapl->{list}->{$k});
if($title && $e == 0) {
push(@output, main::graph_header($title, 1));
push(@output, " <tr>\n");
push(@output, " <td>\n");
}
my $n_plot = $k;
@riglim = @{setup_riglim($rigid[$n_plot], $limit[$n_plot])};
undef(@tmp);
undef(@tmpz);
undef(@CDEF);
my $dstr = $k;
if(defined($intelrapl->{list_item_names}->{$k})) {
$dstr = $intelrapl->{list_item_names}->{$k};
}
my $core_string = $dstr;
$str = $dstr;
my $legend_label_format = "%7.2lf";
my $value_transformation = ",1e-6,*"; # muWatt to Watts
my $legend_size = 16;
my $cpu_label_size = 50;
my $cpu_label = sprintf("%-" . $cpu_label_size . "s", $str);
my $cpu_label_empty = sprintf("%-" . $cpu_label_size . "s", "");
my $sum_name = "sum";
my $cdef_sum = "CDEF:" . $sum_name . "=";
my @sum_group = ();
if(defined($intelrapl->{sum}) && defined($intelrapl->{sum}->{$k})) {
@sum_group = split($list_delimiter, $intelrapl->{sum}->{$k});
}
my $package_sensor = "package";
my $package_index;
if(defined($intelrapl->{package_sensors}) && defined($intelrapl->{package_sensors}->{$k})) {
$package_sensor = trim($intelrapl->{package_sensors}->{$k});
}
my $noncore_name = "noncore";
my $cdef_noncore = "CDEF:" . $noncore_name . "=";
my @package_content_group = ();
if(defined($intelrapl->{package_content}) && defined($intelrapl->{package_content}->{$k})) {
@package_content_group = split($list_delimiter, $intelrapl->{package_content}->{$k});
}
my $sum_counter = 0;
my $package_content_counter = 0;
for($n = 0; $n < min(scalar(@sensor_group), $max_number_of_values_per_group); $n += 1) {
my $value_name = "val" . $n;
my $sensor_name = trim($sensor_group[$n]);
my $value_label = $sensor_name;
if(defined($intelrapl->{sensor_names}->{$value_label})) {
$value_label = $intelrapl->{sensor_names}->{$value_label};
}
$value_label = sprintf("%-".$legend_size."s", $value_label);
push(@CDEF, "CDEF:trans_" . $value_name . "=" . $value_name . $value_transformation);
if(scalar(@sum_group) != 0) {
if(grep {$_ eq $sensor_name} @sum_group) {
if ($sum_counter != 0) {
$cdef_sum .= ",";
}
$cdef_sum .= $value_name;
$sum_counter += 1;
}
}
if(scalar(@package_content_group) != 0) {
if(grep {$_ eq $sensor_name} @package_content_group) {
if ($package_content_counter != 0) {
$cdef_noncore .= ",";
}
$cdef_noncore .= $value_name;
$package_content_counter += 1;
}
}
if ($package_sensor eq $sensor_name) {
$package_index = $n;
}
my $legend_label = $value_label;
if (defined($package_index) && scalar(@package_content_group) != 0 && $sensor_name eq $package_sensor) {
my $noncore_label_full = "Non-Core";
if(defined($intelrapl->{noncore_names}) && defined($intelrapl->{noncore_names}->{$k})) {
$noncore_label_full = trim($intelrapl->{noncore_names}->{$k});
}
my $noncore_label = sprintf("%-".$legend_size."s", $noncore_label_full);
my $noncore_color = line_color(scalar(@sensor_group));
my $package_sensor_name = $package_sensor;
if(defined($intelrapl->{sensor_names}->{$package_sensor})) {
$package_sensor_name = $intelrapl->{sensor_names}->{$package_sensor};
}
my $package_info = "(".$noncore_label_full . " = " . $package_sensor_name;
for(my $i = 0; $i < scalar(@package_content_group); $i += 1) {
my $package_item = trim($package_content_group[$i]);
my $package_item_name = $package_item;
if(defined($intelrapl->{sensor_names}->{$package_item})) {
$package_item_name = $intelrapl->{sensor_names}->{$package_item};
}
$package_info .= " - ";
$package_info .= $package_item_name;
}
$package_info .= ")";
$package_info = sprintf("%-" . ($cpu_label_size) . "s" , substr($package_info, 0, ($cpu_label_size)));
if(defined($intelrapl->{show_noncore_info}) && lc(trim($intelrapl->{show_noncore_info})) eq "y") {
push(@tmp, "COMMENT:" . $package_info);
} else {
push(@tmp, "COMMENT:" . $cpu_label_empty);
}
push(@tmp, "LINE1:trans_" . $noncore_name . $noncore_color . ":" . $noncore_label);
push(@tmpz, "LINE1:trans_" . $noncore_name . $noncore_color . ":" . $noncore_label);
push(@tmp, "GPRINT:trans_" . $noncore_name . ":LAST:Current\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $noncore_name . ":AVERAGE:Average\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $noncore_name . ":MIN:Min\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $noncore_name . ":MAX:Max\\:" . $legend_label_format . "\\n");
}
if ($n == 0) {
push(@tmp, "COMMENT:" . $cpu_label);
} else {
push(@tmp, "COMMENT:" . $cpu_label_empty);
}
my $hex_color_n = line_color($n);
if ($sensor_name eq "core") {
my $hex_transparency = "E6";
push(@tmp, "AREA:trans_" . $value_name . $hex_color_n .$hex_transparency. ":" . $legend_label);
push(@tmpz, "AREA:trans_" . $value_name . $hex_color_n .$hex_transparency. ":" . $legend_label);
} else {
push(@tmp, "LINE1:trans_" . $value_name . $hex_color_n . ":" . $legend_label);
push(@tmpz, "LINE1:trans_" . $value_name . $hex_color_n . ":" . $legend_label);
}
push(@tmp, "GPRINT:trans_" . $value_name . ":LAST:Current\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $value_name . ":AVERAGE:Average\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $value_name . ":MIN:Min\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $value_name . ":MAX:Max\\:" . $legend_label_format . "\\n");
}
if ($sum_counter != 0) {
$cdef_sum .= ",+" x ($sum_counter-1);
push(@CDEF, $cdef_sum);
push(@CDEF, "CDEF:trans_" . $sum_name . "=" . $sum_name . $value_transformation);
my $sum_label = "Sum";
if(defined($intelrapl->{sum_names}->{$k})) {
$sum_label = $intelrapl->{sum_names}->{$k};
}
$sum_label = sprintf("%-".$legend_size."s", $sum_label);
my $socket_color = line_color(scalar(@sensor_group)+1);
push(@tmp, "COMMENT:". $cpu_label_empty);
push(@tmp, "LINE1:trans_" . $sum_name . $socket_color . ":" . $sum_label);
push(@tmpz, "LINE1:trans_" . $sum_name . $socket_color . ":" . $sum_label);
push(@tmp, "GPRINT:trans_" . $sum_name . ":LAST:Current\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $sum_name . ":AVERAGE:Average\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $sum_name . ":MIN:Min\\:" . $legend_label_format);
push(@tmp, "GPRINT:trans_" . $sum_name . ":MAX:Max\\:" . $legend_label_format . "\\n");
}
if (defined($package_index) && $package_content_counter != 0) {
$cdef_noncore .= ",+" x ($package_content_counter-1);
$cdef_noncore .= ",val" . $package_index . ",-,-1,*";
push(@CDEF, $cdef_noncore);
push(@CDEF, "CDEF:trans_" . $noncore_name . "=" . $noncore_name . $value_transformation);
}
if(lc($config->{show_gaps}) eq "y") {
push(@tmp, "AREA:wrongdata#$colors->{gap}:");
push(@tmpz, "AREA:wrongdata#$colors->{gap}:");
push(@CDEF, "CDEF:wrongdata=allvalues,UN,INF,UNKN,IF");
}
my $graph_size_name = "large";
($width, $height) = split('x', $config->{graph_size}->{$graph_size_name});
if($silent =~ /imagetag/) {
($width, $height) = split('x', $config->{graph_size}->{remote}) if $silent eq "imagetag";
($width, $height) = split('x', $config->{graph_size}->{$graph_size_name}) if $silent eq "imagetagbig";
@tmp = @tmpz;
push(@tmp, "COMMENT: \\n");
push(@tmp, "COMMENT: \\n");
push(@tmp, "COMMENT: \\n");
}
my @def_sensor_average;
my $cdef_sensor_allvalues = "CDEF:allvalues=";
my $sum_of_cores = 0;
for(my $n_group = 0; $n_group < min(scalar(@sensor_group), $max_number_of_values_per_group); $n_group++) {
my $dstr = trim($sensor_group[$n_group]);
$sum_of_cores += 1;
my $value_name = "val" . $n_group;
push(@def_sensor_average, "DEF:" . $value_name . "=$rrd:rapl" . $e . "_" . $value_name . ":AVERAGE");
if($n_group != 0) {
$cdef_sensor_allvalues .= ",";
}
if ($gap_on_all_nan) {
$cdef_sensor_allvalues .= $value_name . ",UN,0,1,IF";
} else {
$cdef_sensor_allvalues .= $value_name;
}
}
$cdef_sensor_allvalues .= ",+" x ($sum_of_cores-1);
if ($gap_on_all_nan) {
$cdef_sensor_allvalues .= ",0,GT,1,UNKN,IF";
}
my $y_axis_title = "Watt";
my $large_plot = 1;
my $plot_title = $config->{graphs}->{'_intelrapl1'};
if(defined($intelrapl->{desc}) && defined($intelrapl->{desc}->{$k})) {
$plot_title = $intelrapl->{desc}->{$k};
}
$pic = $rrd{$version}->("$IMG_DIR" . $IMG[$e * $plots_per_list_item + $n_plot],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc",
"--vertical-label=" . $y_axis_title,
"--width=$width",
"--height=$height",
@extra,
@riglim,
$zoom,
@{$cgi->{version12}},
$large_plot ? () : @{$cgi->{version12_small}},
@{$colors->{graph_colors}},
@def_sensor_average,
$cdef_sensor_allvalues,
@CDEF,
@tmp);
$err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . $IMG[$e * $plots_per_list_item + $n_plot] . ": $err\n") if $err;
if(lc($config->{enable_zoom}) eq "y") {
($width, $height) = split('x', $config->{graph_size}->{zoom});
$picz = $rrd{$version}->("$IMG_DIR" . $IMGz[$e * $plots_per_list_item + $n_plot],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc",
"--vertical-label=" . $y_axis_title,
"--width=$width",
"--height=$height",
@full_size_mode,
@extra,
@riglim,
$zoom,
@{$cgi->{version12}},
$large_plot ? () : @{$cgi->{version12_small}},
@{$colors->{graph_colors}},
@def_sensor_average,
$cdef_sensor_allvalues,
@CDEF,
@tmpz);
$err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . $IMGz[$e * $plots_per_list_item + $n_plot] . ": $err\n") if $err;
}
$e2 = $e + $n_plot + 1;
if($title || ($silent =~ /imagetag/ && $graph =~ /intelrapl$e2/)) {
if(lc($config->{enable_zoom}) eq "y") {
if(lc($config->{disable_javascript_void}) eq "y") {
push(@output, " <a href=\"" . $config->{url} . "/" . $config->{imgs_dir} . $IMGz[$e * $plots_per_list_item + $n_plot] . "\"><img src='" . $config->{url} . "/" . $config->{imgs_dir} . $IMG[$e * $plots_per_list_item + $n_plot] . "' border='0'></a>\n");
} else {
if($version eq "new") {
$picz_width = $picz->{image_width} * $config->{global_zoom};
$picz_height = $picz->{image_height} * $config->{global_zoom};
} else {
$picz_width = $width + 115;
$picz_height = $height + 100;
}
push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * $plots_per_list_item + $n_plot], IMG => $IMG[$e * $plots_per_list_item + $n_plot]) . "\n");
}
} else {
push(@output, " " . img_element(config => $config, IMG => $IMG[$e * $plots_per_list_item + $n_plot]) . "\n");
}
}
$e++;
}
if($title) {
push(@output, " </td>\n");
push(@output, " </tr>\n");
push(@output, main::graph_footer());
}
push(@output, " <br>\n");
return @output;
}
1;

View File

@ -77,7 +77,7 @@ sub ipmi_init {
push(@max, "RRA:MAX:0.5:1440:" . (365 * $n)); push(@max, "RRA:MAX:0.5:1440:" . (365 * $n));
push(@last, "RRA:LAST:0.5:1440:" . (365 * $n)); push(@last, "RRA:LAST:0.5:1440:" . (365 * $n));
} }
for($n = 0; $n < scalar(my @sl = split(',', $ipmi->{list})); $n++) { for($n = 0; $n < scalar(my @sensor_list = split(',', $ipmi->{list})); $n++) {
push(@tmp, "DS:ipmi" . $n . "_s1:GAUGE:120:U:U"); push(@tmp, "DS:ipmi" . $n . "_s1:GAUGE:120:U:U");
push(@tmp, "DS:ipmi" . $n . "_s2:GAUGE:120:U:U"); push(@tmp, "DS:ipmi" . $n . "_s2:GAUGE:120:U:U");
push(@tmp, "DS:ipmi" . $n . "_s3:GAUGE:120:U:U"); push(@tmp, "DS:ipmi" . $n . "_s3:GAUGE:120:U:U");
@ -149,8 +149,10 @@ sub ipmi_update {
my @data = <IN>; my @data = <IN>;
close(IN); close(IN);
my @sensor_list = split(',', $ipmi->{list});
my $e = 0; my $e = 0;
while($e < scalar(my @sl = split(',', $ipmi->{list}))) { while($e < scalar(@sensor_list)) {
my $e2 = 0; my $e2 = 0;
foreach my $i (split(',', $ipmi->{desc}->{$e})) { foreach my $i (split(',', $ipmi->{desc}->{$e})) {
my $unit; my $unit;
@ -195,7 +197,7 @@ sub ipmi_update {
} }
$e = 0; $e = 0;
while($e < scalar(my @sl = split(',', $ipmi->{list}))) { while($e < scalar(@sensor_list)) {
for($n = 0; $n < 9; $n++) { for($n = 0; $n < 9; $n++) {
$sens[$e][$n] = 0 unless defined $sens[$e][$n]; $sens[$e][$n] = 0 unless defined $sens[$e][$n];
$rrdata .= ":" . $sens[$e][$n]; $rrdata .= ":" . $sens[$e][$n];
@ -271,7 +273,7 @@ sub ipmi_cgi {
} }
$title = !$silent ? $title : ""; $title = !$silent ? $title : "";
my @sensor_list = split(',', $ipmi->{list});
# text mode # text mode
# #
@ -292,7 +294,7 @@ sub ipmi_cgi {
my $line3; my $line3;
push(@output, " <pre style='font-size: 12px; color: $colors->{fg_color}';>\n"); push(@output, " <pre style='font-size: 12px; color: $colors->{fg_color}';>\n");
push(@output, " "); push(@output, " ");
for($n = 0; $n < scalar(my @sl = split(',', $ipmi->{list})); $n++) { for($n = 0; $n < scalar(@sensor_list); $n++) {
$line1 = ""; $line1 = "";
foreach my $i (split(',', $ipmi->{desc}->{$n})) { foreach my $i (split(',', $ipmi->{desc}->{$n})) {
$i = trim($i); $i = trim($i);
@ -304,7 +306,7 @@ sub ipmi_cgi {
} }
if($line1) { if($line1) {
my $i = length($line1); my $i = length($line1);
push(@output, sprintf(sprintf("%${i}s", sprintf("%s", trim($sl[$n]))))); push(@output, sprintf(sprintf("%${i}s", sprintf("%s", trim($sensor_list[$n])))));
} }
} }
push(@output, "\n"); push(@output, "\n");
@ -320,7 +322,7 @@ sub ipmi_cgi {
$line = @$data[$n]; $line = @$data[$n];
$time = $time - (1 / $tf->{ts}); $time = $time - (1 / $tf->{ts});
push(@output, sprintf(" %2d$tf->{tc} ", $time)); push(@output, sprintf(" %2d$tf->{tc} ", $time));
for($n2 = 0; $n2 < scalar(my @sl = split(',', $ipmi->{list})); $n2++) { for($n2 = 0; $n2 < scalar(@sensor_list); $n2++) {
$n3 = $n2 * 9; $n3 = $n2 * 9;
foreach my $i (split(',', $ipmi->{desc}->{$n2})) { foreach my $i (split(',', $ipmi->{desc}->{$n2})) {
$from = $n3++; $from = $n3++;
@ -353,7 +355,7 @@ sub ipmi_cgi {
$u = ""; $u = "";
} }
for($n = 0; $n < scalar(my @sl = split(',', $ipmi->{list})); $n++) { for($n = 0; $n < scalar(@sensor_list); $n++) {
$str = $u . $package . $n . "." . $tf->{when} . ".$imgfmt_lc"; $str = $u . $package . $n . "." . $tf->{when} . ".$imgfmt_lc";
push(@IMG, $str); push(@IMG, $str);
unlink("$IMG_DIR" . $str); unlink("$IMG_DIR" . $str);
@ -364,16 +366,38 @@ sub ipmi_cgi {
} }
} }
my $graphs_per_row = $ipmi->{graphs_per_row};
my @linpad =(0) x scalar(@sensor_list);
if ($graphs_per_row > 1) {
for(my $n = 0; $n < scalar(@sensor_list); $n++) {
my @ls = split(',', $ipmi->{desc}->{$n});
$linpad[$n] = scalar(@ls);
}
for(my $n = 0; $n < scalar(@linpad); $n++) {
if ($n % $graphs_per_row == 0) {
my $max_number_of_lines = 0;
for (my $sub_n = $n; $sub_n < min($n + $graphs_per_row, scalar(@linpad)); $sub_n++) {
$max_number_of_lines = max($max_number_of_lines, $linpad[$sub_n]);
}
for (my $sub_n = $n; $sub_n < min($n + $graphs_per_row, scalar(@linpad)); $sub_n++) {
$linpad[$sub_n] = $max_number_of_lines;
}
}
}
}
my $whitespace_key_support = lc($ipmi->{whitespace_key_support} || "") eq "y" ? 1 : 0;
$n = 0; $n = 0;
while($n < scalar(my @sl = split(',', $ipmi->{list}))) { while($n < scalar(@sensor_list)) {
if($title) { if($title) {
if($n == 0) { if($n == 0) {
push(@output, main::graph_header($title, $ipmi->{graphs_per_row})); push(@output, main::graph_header($title, $graphs_per_row));
} }
push(@output, " <tr>\n"); push(@output, " <tr>\n");
} }
for($n2 = 0; $n2 < $ipmi->{graphs_per_row}; $n2++) { for($n2 = 0; $n2 < $graphs_per_row; $n2++) {
last unless $n < scalar(my @sl = split(',', $ipmi->{list})); last unless $n < scalar(@sensor_list);
if($title) { if($title) {
push(@output, " <td>\n"); push(@output, " <td>\n");
} }
@ -385,6 +409,9 @@ sub ipmi_cgi {
my $unit = $ipmi->{units}->{$n}; my $unit = $ipmi->{units}->{$n};
foreach my $i (split(',', $ipmi->{desc}->{$n})) { foreach my $i (split(',', $ipmi->{desc}->{$n})) {
$i = trim($i); $i = trim($i);
if ($whitespace_key_support) {
$i=~s/ /_/g;
}
$str = $ipmi->{map}->{$i} || $i; $str = $ipmi->{map}->{$i} || $i;
$str = sprintf("%-40s", substr($str, 0, 40)); $str = sprintf("%-40s", substr($str, 0, 40));
push(@tmp, "LINE2:s" . ($e + 1) . $LC[$e] . ":$str"); push(@tmp, "LINE2:s" . ($e + 1) . $LC[$e] . ":$str");
@ -392,7 +419,7 @@ sub ipmi_cgi {
push(@tmpz, "LINE2:s" . ($e + 1) . $LC[$e] . ":$str"); push(@tmpz, "LINE2:s" . ($e + 1) . $LC[$e] . ":$str");
$e++; $e++;
} }
while($e < 9) { while($e < $linpad[$n]) {
push(@tmp, "COMMENT: \\n"); push(@tmp, "COMMENT: \\n");
$e++; $e++;
} }
@ -402,7 +429,7 @@ sub ipmi_cgi {
push(@CDEF, "CDEF:wrongdata=allvalues,UN,INF,UNKN,IF"); push(@CDEF, "CDEF:wrongdata=allvalues,UN,INF,UNKN,IF");
} }
($width, $height) = split('x', $config->{graph_size}->{medium}); ($width, $height) = split('x', $config->{graph_size}->{medium});
$str = substr(trim($sl[$n]), 0, 25); $str = substr(trim($sensor_list[$n]), 0, 25);
my $cdef_allvalues = $gap_on_all_nan ? "CDEF:allvalues=s1,UN,0,1,IF,s2,UN,0,1,IF,s3,UN,0,1,IF,s4,UN,0,1,IF,s5,UN,0,1,IF,s6,UN,0,1,IF,s7,UN,0,1,IF,s8,UN,0,1,IF,s9,UN,0,1,IF,+,+,+,+,+,+,+,+,0,GT,1,UNKN,IF" : "CDEF:allvalues=s1,s2,s3,s4,s5,s6,s7,s8,s9,+,+,+,+,+,+,+,+"; my $cdef_allvalues = $gap_on_all_nan ? "CDEF:allvalues=s1,UN,0,1,IF,s2,UN,0,1,IF,s3,UN,0,1,IF,s4,UN,0,1,IF,s5,UN,0,1,IF,s6,UN,0,1,IF,s7,UN,0,1,IF,s8,UN,0,1,IF,s9,UN,0,1,IF,+,+,+,+,+,+,+,+,0,GT,1,UNKN,IF" : "CDEF:allvalues=s1,s2,s3,s4,s5,s6,s7,s8,s9,+,+,+,+,+,+,+,+";
$pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$n]", $pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$n]",
"--title=$str ($tf->{nwhen}$tf->{twhen})", "--title=$str ($tf->{nwhen}$tf->{twhen})",

View File

@ -632,7 +632,7 @@ sub nvidiagpu_cgi {
$cdef_sensor_allvalues .= ",0,GT,1,UNKN,IF"; $cdef_sensor_allvalues .= ",0,GT,1,UNKN,IF";
} }
my $plot_title = $config->{graphs}->{'_nvidiagpu' . ($n_plot + 1)}; my $plot_title = $config->{graphs}->{'_nvidiagpu' . ($n_plot + 1)};
$pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$e * 3 + $n_plot]", $pic = $rrd{$version}->("$IMG_DIR" . $IMG[$e * $number_of_sensor_values_in_use + $n_plot],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})", "--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc", "--imgformat=$imgfmt_uc",
@ -650,10 +650,10 @@ sub nvidiagpu_cgi {
@CDEF, @CDEF,
@tmp); @tmp);
$err = RRDs::error; $err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . "$IMG[$e * 3 + $n_plot]: $err\n") if $err; push(@output, "ERROR: while graphing $IMG_DIR" . $IMG[$e * $number_of_sensor_values_in_use + $n_plot] . ": $err\n") if $err;
if(lc($config->{enable_zoom}) eq "y") { if(lc($config->{enable_zoom}) eq "y") {
($width, $height) = split('x', $config->{graph_size}->{zoom}); ($width, $height) = split('x', $config->{graph_size}->{zoom});
$picz = $rrd{$version}->("$IMG_DIR" . "$IMGz[$e * 3 + $n_plot]", $picz = $rrd{$version}->("$IMG_DIR" . $IMGz[$e * $number_of_sensor_values_in_use + $n_plot],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})", "--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc", "--imgformat=$imgfmt_uc",
@ -672,13 +672,13 @@ sub nvidiagpu_cgi {
@CDEF, @CDEF,
@tmpz); @tmpz);
$err = RRDs::error; $err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . "$IMGz[$e * 3 + $n_plot]: $err\n") if $err; push(@output, "ERROR: while graphing $IMG_DIR" . $IMGz[$e * $number_of_sensor_values_in_use + $n_plot]. ": $err\n") if $err;
} }
$e2 = $e + $n_plot + 1; $e2 = $e + $n_plot + 1;
if($title || ($silent =~ /imagetag/ && $graph =~ /nvidiagpu$e2/)) { if($title || ($silent =~ /imagetag/ && $graph =~ /nvidiagpu$e2/)) {
if(lc($config->{enable_zoom}) eq "y") { if(lc($config->{enable_zoom}) eq "y") {
if(lc($config->{disable_javascript_void}) eq "y") { if(lc($config->{disable_javascript_void}) eq "y") {
push(@output, " " . picz_a_element(config => $config, IMGz => $IMGz[$e * 3 + $n_plot], IMG => $IMG[$e * 3 + $n_plot]) . "\n"); push(@output, " " . picz_a_element(config => $config, IMGz => $IMGz[$e * $number_of_sensor_values_in_use + $n_plot], IMG => $IMG[$e * $number_of_sensor_values_in_use + $n_plot]) . "\n");
} else { } else {
if($version eq "new") { if($version eq "new") {
$picz_width = $picz->{image_width} * $config->{global_zoom}; $picz_width = $picz->{image_width} * $config->{global_zoom};
@ -687,10 +687,10 @@ sub nvidiagpu_cgi {
$picz_width = $width + 115; $picz_width = $width + 115;
$picz_height = $height + 100; $picz_height = $height + 100;
} }
push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * 3 + $n_plot], IMG => $IMG[$e * 3 + $n_plot]) . "\n"); push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * $number_of_sensor_values_in_use + $n_plot], IMG => $IMG[$e * $number_of_sensor_values_in_use + $n_plot]) . "\n");
} }
} else { } else {
push(@output, " " . img_element(config => $config, IMG => $IMG[$e * 3 + $n_plot]) . "\n"); push(@output, " " . img_element(config => $config, IMG => $IMG[$e * $number_of_sensor_values_in_use + $n_plot]) . "\n");
} }
} }
} }

View File

@ -631,7 +631,7 @@ sub nvme_cgi {
push(@scaling_options, "--alt-y-grid"); push(@scaling_options, "--alt-y-grid");
} }
my $plot_title = $config->{graphs}->{'_nvme' . ($n_smart + 1)}; my $plot_title = $config->{graphs}->{'_nvme' . ($n_smart + 1)};
$pic = $rrd{$version}->("$IMG_DIR" . "$IMG[$e * 3 + $n_smart]", $pic = $rrd{$version}->("$IMG_DIR" . $IMG[$e * $number_of_smart_values_in_use + $n_smart],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})", "--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc", "--imgformat=$imgfmt_uc",
@ -650,10 +650,10 @@ sub nvme_cgi {
@CDEF, @CDEF,
@tmp); @tmp);
$err = RRDs::error; $err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . "$IMG[$e * 3 + $n_smart]: $err\n") if $err; push(@output, "ERROR: while graphing $IMG_DIR" . $IMG[$e * $number_of_smart_values_in_use + $n_smart]. ": $err\n") if $err;
if(lc($config->{enable_zoom}) eq "y") { if(lc($config->{enable_zoom}) eq "y") {
($width, $height) = split('x', $config->{graph_size}->{zoom}); ($width, $height) = split('x', $config->{graph_size}->{zoom});
$picz = $rrd{$version}->("$IMG_DIR" . "$IMGz[$e * 3 + $n_smart]", $picz = $rrd{$version}->("$IMG_DIR" . $IMGz[$e * $number_of_smart_values_in_use + $n_smart],
"--title=$plot_title ($tf->{nwhen}$tf->{twhen})", "--title=$plot_title ($tf->{nwhen}$tf->{twhen})",
"--start=-$tf->{nwhen}$tf->{twhen}", "--start=-$tf->{nwhen}$tf->{twhen}",
"--imgformat=$imgfmt_uc", "--imgformat=$imgfmt_uc",
@ -673,13 +673,13 @@ sub nvme_cgi {
@CDEF, @CDEF,
@tmpz); @tmpz);
$err = RRDs::error; $err = RRDs::error;
push(@output, "ERROR: while graphing $IMG_DIR" . "$IMGz[$e * 3 + $n_smart]: $err\n") if $err; push(@output, "ERROR: while graphing $IMG_DIR" . $IMGz[$e * $number_of_smart_values_in_use + $n_smart]. ": $err\n") if $err;
} }
$e2 = $e + $n_smart + 1; $e2 = $e + $n_smart + 1;
if($title || ($silent =~ /imagetag/ && $graph =~ /nvme$e2/)) { if($title || ($silent =~ /imagetag/ && $graph =~ /nvme$e2/)) {
if(lc($config->{enable_zoom}) eq "y") { if(lc($config->{enable_zoom}) eq "y") {
if(lc($config->{disable_javascript_void}) eq "y") { if(lc($config->{disable_javascript_void}) eq "y") {
push(@output, " " . picz_a_element(config => $config, IMGz => $IMGz[$e * 3 + $n_smart], IMG => $IMG[$e * 3 + $n_smart]) . "\n"); push(@output, " " . picz_a_element(config => $config, IMGz => $IMGz[$e * $number_of_smart_values_in_use + $n_smart], IMG => $IMG[$e * $number_of_smart_values_in_use + $n_smart]) . "\n");
} else { } else {
if($version eq "new") { if($version eq "new") {
$picz_width = $picz->{image_width} * $config->{global_zoom}; $picz_width = $picz->{image_width} * $config->{global_zoom};
@ -688,10 +688,10 @@ sub nvme_cgi {
$picz_width = $width + 115; $picz_width = $width + 115;
$picz_height = $height + 100; $picz_height = $height + 100;
} }
push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * 3 + $n_smart], IMG => $IMG[$e * 3 + $n_smart]) . "\n"); push(@output, " " . picz_js_a_element(width => $picz_width, height => $picz_height, config => $config, IMGz => $IMGz[$e * $number_of_smart_values_in_use + $n_smart], IMG => $IMG[$e * $number_of_smart_values_in_use + $n_smart]) . "\n");
} }
} else { } else {
push(@output, " " . img_element(config => $config, IMG => $IMG[$e * 3 + $n_smart]) . "\n"); push(@output, " " . img_element(config => $config, IMG => $IMG[$e * $number_of_smart_values_in_use + $n_smart]) . "\n");
} }
} }
} }

View File

@ -180,6 +180,13 @@ All the configuration files in there will be loaded in alphabetic order, so the
Default value: \fI/etc/monitorix/conf.d\fP Default value: \fI/etc/monitorix/conf.d\fP
.RE .RE
.P .P
.BI additional_graph_name
.RS
This is an optional configuration option for configuration files in \fIinclude_dir\fP. It takes a comma separated list of additional modules that is appended to the \fIgraph_name\fP option. This enables you to add custom modules without changing the \fImonitorix.conf\fP or copy all names from \fIgraph_name\fP to your config file.
.P
Default value: \fINone\fP
.RE
.P
.BI ip_default_table .BI ip_default_table
.RS .RS
This option will define in which table Monitorix will put all \fIiptables\fP rules for network traffic accounting monitoring. It only works on Linux. This option will define in which table Monitorix will put all \fIiptables\fP rules for network traffic accounting monitoring. It only works on Linux.
@ -644,6 +651,8 @@ This option sets the size of all processors graphs.
.P .P
The possible values are: The possible values are:
.RS .RS
\fIlarge\fP for 750x180 graphs.
.br
\fImain\fP for 450x150 graphs. \fImain\fP for 450x150 graphs.
.br .br
\fImedium\fP for 325x150 graphs. \fImedium\fP for 325x150 graphs.
@ -866,6 +875,235 @@ This option, when enabled via \fIy\fP, combined with the \fIshow_gaps\fP option
.P .P
Default value: \fIn\fP Default value: \fIn\fP
.RE .RE
.SS Intel RAPL power consumption (intelrapl.pm)
This graph is able to monitor the power draw of an unlimited number of Intel RAPL supporting devices like Intel CPU packages. For example it can (if the device RAPL data is available) show power draw of the cores, integrated GPU, package, DRAM and the non-core part of the power draw. The non-core power draw composition can be specified by setting which parts will be subtracted from the package power draw.
.P
.BI list
.RS
This is a list of groups of CPU sensors you want to monitor with an arbitrary name. Each group will become a plot and there may be an unlimited number of groups. You can define sensor names like \fcore\fP and \fpackage\fP.
.P
WARNING: Every time the number of groups or the size of the groups in this option changes, Monitorix will resize the \fIintelrapl.rrd\fP file accordingly, removing all historical data.
.P
To collect the CPU power usage the energy counters from the \fIintel-rapl\fP framework are used and a power consumption in the monitored time interval calculated.
.P
It is recommended that you first check if the \fIintel-rapl\fP class tree and collect your desired energy counters from the devices that you plan to monitor. For example check \fItree /sys/class/powercap/intel-rapl\fP for available \fIenergy_uj\fP energy counters. The corresponding \fIname\fP files helps identifying the sensors.
You can add it to the group 0 like this:
.P
.RS
<list>
.br
0 = core, uncore, package, dram
.br
1 = dram
.br
</list>
.RE
.P
.RE
.P
.BI list_item_names
.RS
This list complements the \fBlist\fP option. It allows you to change the group name that will appear in the graph, hiding the sensor name. If no association is defined, then Monitorix will display the name of the device as it is specified in the config file.
.P
.RS
<list_item_names>
.br
0 = CPU 1
.br
1 = DRAM modules
.br
</list_item_names>
.RE
.RE
.P
.BI sensors
.RS
This list sets the energy counter files for the devices specified in the \fBlist\fP option.
.P
.RS
<sensors>
.br
core = /sys/class/powercap/intel-rapl:0:0/energy_uj
.br
uncore = /sys/class/powercap/intel-rapl:0:1/energy_uj
.br
package = /sys/class/powercap/intel-rapl:0/energy_uj
.br
dram = /sys/class/powercap/intel-rapl:0:2/energy_uj
.br
</sensors>
.RE
.RE
.P
.BI sensor_names
.RS
This list complements the \fBsensors\fP option. It allows you to change the device name that will appear in the graph, hiding the device name. If no association is defined, then Monitorix will display the name of the device as it is specified in the config file.
.P
.RS
<sensor_names>
.br
package = Package
.br
core = Cores
.br
uncore = IGP
.br
dram = DRAM
.br
</sensor_names>
.RE
.RE
.P
.BI package_sensors
.RS
This list allows you to set a device name for the package device. This specified device is used as total power draw to calculate the non-core power draw by subtracting the power draw of the devices specified for the same group in \fBackage_sensors\fP. Non-core power draw will not be shown if no package sensor is specified for a certain group.
.P
.RS
<package_sensors>
.br
0 = package
.br
</package_sensors>
.RE
.RE
.P
.BI package_content
.RS
This list complements the \fBpackage_sensors\fP option. It allows specify the devices that are contained in the total package power draw. This information is needed if you want to visualize non-core power draw. Non-core power draw will be calculated by subtracting the sum power draw of a package content group from the package power draw.
.P
.RS
<package_content>
.br
0 = core, uncore
.br
</package_content>
.RE
.RE
.P
.BI noncore_names
.RS
This list allows you to change the non-core name for a certain group that will appear in the graph legend.
.P
.RS
<noncore_names>
.br
0 = Non-Core
.br
</noncore_names>
.RE
.RE
.P
.BI desc
.RS
This list complements the \fBlist\fP option. It allows you to include a title for every group of devicess. The title will appear in the title above the plot.
.P
.RS
<desc>
.br
0 = CPU power distribution
.br
1 = Memory power distribution
.br
</desc>
.RE
.RE
.P
.BI sum
.RS
This list allows you to add a specific sum of devices to the plot of a certain group. No sum will be shown if not specified for the group.
.P
.RS
<sum>
.br
0 = package, dram
.br
</sum>
.RE
.RE
.P
.BI sum_names
.RS
This list complements the \fBseum\fP option. It allows specify the name of the sum if desired.
.P
.RS
<sum_names>
.br
0 = Package + DRAM
.br
</sum_names>
.RE
.RE
.P
.BI packagepower_enabled
.RS
This section enables or disables the alert capabilities for this graph; the alert for the package power draw. It works as follows:
.P
If the package power draw specified by \fBpackage_sensors\fP of any of the specified device reaches or succeeds the \fBpackagepower_threshold\fP (the interval of time is not used here), Monitorix will execute the external alert script defined in \fBpackagepower_script\fP.
.P
The default Monitorix installation includes an example of a shell-script alert called \fBmonitorix-alert.sh\fP which you can use as a base for your own script.
.P
Default value: \fIn\fP
.RE
.P
.BI packagepower_timeintvl
.RS
Not used in this alert.
.P
Default value: \fINone\fP
.RE
.P
.BI packagepower_threshold
.RS
This is the value that needs to be reached or succeeded to trigger the mechanism for a particular action, which in this case is the execution of an external alert script.
.P
Default value: \fINone\fP
.RE
.P
.BI packagepower_script
.RS
This is the full path name of the script that will be executed by this alert.
.P
It will receive the following three parameters:
.P
1st - the value currently defined in \fBpackagepower_timeintvl\fP.
.br
2nd - the value currently defined in \fBpackagepower_threshold\fP.
.br
3rd - the current package power draw.
.P
Default value: \fI/path/to/script.sh\fP
.RE
.P
.BI use_nan_for_missing_data
.RS
This option, when enabled via \fIy\fP, shows \fnan\fP values for missing data instead of \f0\fP. This is useful when \f0\fP could be mistaken for valid data.
.P
Default value: \fIn\fP
.RE
.P
.BI gap_on_all_nan
.RS
This option, when enabled via \fIy\fP, combined with the \fIshow_gaps\fP option shows gaps only if all data points are \fInan\fP instead of requiring only one to be \fInan\fP for a gap. This can be useful if not all sensor data are required for normal operation.
.P
Default value: \fIn\fP
.RE
.P
.BI show_noncore_info
.RS
This option, when enabled via \fIy\fP, enable showing description string in the legend how the non-core part is calculated.
.P
Default value: \fIn\fP
.RE
.P
.BI max_number_of_values_per_group
.RS
This option sets the maximum number of devices per group if more values are required.
.P
WARNING: Every time the \fImax_number_of_values_per_group\fP value changes, Monitorix will resize the \fIintelrapl.rrd\fP file accordingly, removing all historical data.
.P
Default value: \fI10\fP
.RE
.SS HP ProLiant System Health (hptemp.pm) .SS HP ProLiant System Health (hptemp.pm)
.BI list .BI list
.RS .RS
@ -1373,7 +1611,7 @@ This is the type of sensor in each group. It's important to not mix different ty
.P .P
.BI map .BI map
.RS .RS
This list complements the \fBdesc\fP option. It basically allows you to change the name that will appear in the graph, hiding the real name of the sensor. If no association is defined, then Monitorix will display the name specified in the \fBdesc\fP option. Note, this only works in names that don't include whitespaces. This list complements the \fBdesc\fP option. It basically allows you to change the name that will appear in the graph, hiding the real name of the sensor. If no association is defined, then Monitorix will display the name specified in the \fBdesc\fP option. Note, this only works in names that don't include whitespaces unless you use the \fBwhitespace_key_support\fP option and replace whitespaces with underscores in the key.
.RE .RE
.P .P
.BI alerts .BI alerts
@ -1415,6 +1653,12 @@ The external script will receive the following arguments:
.P .P
.RE .RE
.RE .RE
.BI whitespace_key_support
.RS
This list complements the \fBmap\fP option. It changes the handling of the key names. Without this option enabled you can't rename keys with whitespaces using the \fBmap\fP option. When enabled, whitespaces in the map key will be replaced by underscores(_) and enables you to specify the key names this way.
.P
Default value: \fIn\fP
.RE
.BI graphs_per_row .BI graphs_per_row
.RS .RS
This is the number of graphs that will be put in a row. This is the number of graphs that will be put in a row.
@ -3044,7 +3288,7 @@ Default value: \fI3\fP
.P .P
.BI size .BI size
.RS .RS
This is the size of the graphs of the ports. It currently accepts up to 8 different size names: \fImain\fP, \fImedium\fP, \fImedium2\fP, \fIsmall\fP, \fImini\fP, \fItiny\fP, \fIzoom\fP and \fIremote\fP. You can check the resolution of each size in the configuration file, inside the list named \fBgraph_size\fP. This is the size of the graphs of the ports. It currently accepts up to 8 different size names: \fIlarge\fP, \fImain\fP, \fImedium\fP, \fImedium2\fP, \fIsmall\fP, \fImini\fP, \fItiny\fP, \fIzoom\fP and \fIremote\fP. You can check the resolution of each size in the configuration file, inside the list named \fBgraph_size\fP.
.P .P
Default value: \fImini\fP Default value: \fImini\fP
.RE .RE

View File

@ -607,7 +607,11 @@ if($config{include_dir} && opendir(DIR, $config{include_dir})) {
# delete $config_inc{$key}; # delete $config_inc{$key};
# next; # next;
#} #}
if($key eq "additional_graph_name") {
$config{graph_name} .= ", $val";
delete $config_inc{$key};
next;
}
# one level options # one level options
$config{$key} = $val; $config{$key} = $val;
delete $config_inc{$key}; delete $config_inc{$key};

View File

@ -295,7 +295,11 @@ if($config{include_dir} && opendir(DIR, $config{include_dir})) {
# delete $config_inc{graph_name}; # delete $config_inc{graph_name};
# next; # next;
#} #}
if($key eq "additional_graph_name") {
$config{graph_name} .= ", $val";
delete $config_inc{$key};
next;
}
# one level options # one level options
$config{$key} = $val; $config{$key} = $val;
delete $config_inc{$key}; delete $config_inc{$key};

View File

@ -80,6 +80,7 @@ secure_log_date_format = %b %e
kern = y kern = y
proc = y proc = y
amdenergy = n amdenergy = n
intelrapl = n
hptemp = n hptemp = n
lmsens = n lmsens = n
gensens = n gensens = n
@ -199,6 +200,55 @@ secure_log_date_format = %b %e
</amdenergy> </amdenergy>
# intelrapl graph
# -----------------------------------------------------------------------------
<intelrapl>
<list>
0 = core, uncore, package, dram
1 = dram
</list>
<list_item_names>
0 = CPU 1
1 = DRAM modules
</list_item_names>
<sensors>
core = /sys/class/powercap/intel-rapl:0:0/energy_uj
uncore = /sys/class/powercap/intel-rapl:0:1/energy_uj
package = /sys/class/powercap/intel-rapl:0/energy_uj
dram = /sys/class/powercap/intel-rapl:0:2/energy_uj
</sensors>
<sensor_names>
package = Package
core = Cores
uncore = IGP
dram = DRAM
</sensor_names>
<package_sensors>
0 = package
</package_sensors>
<package_content>
0 = core, uncore
</package_content>
<noncore_names>
0 = Non-Core
</noncore_names>
<desc>
0 = CPU power distribution
1 = Memory power distribution
</desc>
<sum>
</sum>
<sum_names>
</sum_names>
rigid = 0
limit = 100
use_nan_for_missing_data = n
gap_on_all_nan = n
show_noncore_info = n
max_number_of_values_per_group = 10
</intelrapl>
# HPTEMP graph # HPTEMP graph
# ----------------------------------------------------------------------------- # -----------------------------------------------------------------------------
<hptemp> <hptemp>
@ -1084,6 +1134,7 @@ logo_bottom = logo_bot.png
</theme> </theme>
<graph_size> <graph_size>
large = 750x180
main = 450x150 main = 450x150
medium = 325x150 medium = 325x150
medium2 = 325x70 medium2 = 325x70
@ -1094,13 +1145,14 @@ logo_bottom = logo_bot.png
remote = 300x100 remote = 300x100
</graph_size> </graph_size>
graph_name = system, kern, proc, amdenergy, hptemp, lmsens, gensens, ipmi, ambsens, amdgpu, nvidiagpu, nvidia, disk, nvme, fs, zfs, du, net, netstat, tinyproxy, tc, libvirt, process, serv, mail, port, user, ftp, apache, nginx, lighttpd, mysql, pgsql, mongodb, varnish, pagespeed, squid, nfss, nfsc, bind, unbound, ntp, chrony, fail2ban, icecast, raspberrypi, phpapc, memcached, redis, phpfpm, apcupsd, nut, wowza, int, verlihub graph_name = system, kern, proc, amdenergy, intelrapl, hptemp, lmsens, gensens, ipmi, ambsens, amdgpu, nvidiagpu, nvidia, disk, nvme, fs, zfs, du, net, netstat, tinyproxy, tc, libvirt, process, serv, mail, port, user, ftp, apache, nginx, lighttpd, mysql, pgsql, mongodb, varnish, pagespeed, squid, nfss, nfsc, bind, unbound, ntp, chrony, fail2ban, icecast, raspberrypi, phpapc, memcached, redis, phpfpm, apcupsd, nut, wowza, int, verlihub
<graph_title> <graph_title>
system = System load average and usage system = System load average and usage
kern = Global kernel usage kern = Global kernel usage
proc = Kernel usage per processor proc = Kernel usage per processor
amdenergy = AMD power usage amdenergy = AMD power usage
intelrapl = RAPL power usage
hptemp = HP ProLiant System Health hptemp = HP ProLiant System Health
lmsens = LM-Sensors and GPU temperatures lmsens = LM-Sensors and GPU temperatures
gensens = Generic sensor statistics gensens = Generic sensor statistics
@ -1165,6 +1217,7 @@ graph_name = system, kern, proc, amdenergy, hptemp, lmsens, gensens, ipmi, ambse
_kern3 = VFS usage _kern3 = VFS usage
_proc = Processor _proc = Processor
_amdenergy1 = Socket power distribution _amdenergy1 = Socket power distribution
_intelrapl1 = Power distribution
_hptemp1 = Temperatures 1 _hptemp1 = Temperatures 1
_hptemp2 = Temperatures 2 _hptemp2 = Temperatures 2
_hptemp3 = Temperatures 3 _hptemp3 = Temperatures 3