2010-12-10 16:30:45 +01:00
|
|
|
#!/usr/bin/perl -w
|
|
|
|
# -*- perl -*-
|
|
|
|
# vim: sts=8 sw=8 ts=8
|
|
|
|
|
|
|
|
#
|
2017-04-17 22:43:38 +02:00
|
|
|
# Run "perldoc thisfilename" to get a well-formatted set of documentation
|
2010-12-10 16:30:45 +01:00
|
|
|
# for this munin plugin.
|
|
|
|
#
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
openvzcpu - Munin plugin to monitor the amount of CPU used by each OpenVZ
|
|
|
|
container running on this machine.
|
|
|
|
|
|
|
|
=head2 SYNOPSIS
|
|
|
|
|
|
|
|
Draws a stacked graph showing system/user/nice CPU usage for each container
|
|
|
|
running on a OpenVZ hardware node. Must be run from outside of any container
|
|
|
|
in order to gain access to the values generated by the OpenVZ kernel code in
|
|
|
|
C</proc/vz/vestat>
|
|
|
|
|
|
|
|
=head1 CONFIGURATION
|
|
|
|
|
|
|
|
The following perl libraries are used:
|
|
|
|
|
|
|
|
Graphics::ColorNames
|
|
|
|
Graphics::ColorObject
|
|
|
|
|
|
|
|
... on Debian based systems (i.e. Ubuntu etc.) you may just:
|
|
|
|
|
|
|
|
aptitude install libcolor-calc-perl libgraphics-colorobject-perl
|
|
|
|
|
|
|
|
Must be run as root in order to read C</proc/vz/vestat>
|
|
|
|
|
|
|
|
Place the following in a file such as C</etc/munin/plugin-conf.d/openvzcpu>
|
|
|
|
B<and then restart munin-node>.
|
|
|
|
|
|
|
|
[openvzcpu]
|
|
|
|
user root
|
|
|
|
|
|
|
|
=head2 OPTIONS
|
|
|
|
|
|
|
|
The following may be added to the file above, in order to enable the graphing
|
|
|
|
of idle and iowait times:
|
|
|
|
|
|
|
|
env.drawidle 1
|
|
|
|
|
|
|
|
For kernels which have other than 100 jiffies per second (sic) n.b. this is
|
|
|
|
unlikely to be necessary - you may add the followin to the plugin-specific
|
|
|
|
configuration:
|
|
|
|
|
|
|
|
env.HZ 1000
|
|
|
|
|
|
|
|
|
|
|
|
If you have a high number of containers running on the machine, you may be
|
|
|
|
able to gain extra clarity by asking munin to create a "taller" graph by
|
|
|
|
adding the following to /etc/munin.conf, or in a file under
|
|
|
|
C</etc/munin/munin-conf.d/> if you are using Munin v1.4 or above:
|
|
|
|
|
|
|
|
openvzcpu.graph_height 700
|
|
|
|
|
|
|
|
=head1 SEE ALSO
|
|
|
|
|
|
|
|
http://wiki.openvz.org/Vestat
|
|
|
|
|
|
|
|
=head1 TODO
|
|
|
|
|
|
|
|
. Add Munin 1.4 multigraph support
|
|
|
|
|
|
|
|
. Sort graphing order so that the smallest container is rendered at the
|
|
|
|
bottom of the graph.
|
|
|
|
|
|
|
|
. Make the colour list into a configuration item.
|
|
|
|
|
|
|
|
. Fix graphing of non-container CPU usage.
|
|
|
|
|
|
|
|
=head1 VERSION
|
|
|
|
|
|
|
|
0.6 Initial Public Release
|
|
|
|
|
|
|
|
=head1 AUTHOR
|
|
|
|
|
|
|
|
Tim Small <tim@seoss.co.uk>
|
|
|
|
|
2010-12-13 15:46:28 +01:00
|
|
|
Copyright 2010 South East Open Source Solutions Ltd.
|
|
|
|
|
|
|
|
The creation of this plugin was funded by Latitude Hosting
|
|
|
|
|
|
|
|
http://www.latitudehosting.net/
|
|
|
|
|
2010-12-10 16:30:45 +01:00
|
|
|
Patches welcome.
|
|
|
|
|
|
|
|
=head1 LICENSE
|
|
|
|
|
|
|
|
GPLv2
|
|
|
|
|
2010-12-13 15:46:28 +01:00
|
|
|
=begin comment
|
|
|
|
|
|
|
|
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; version 2 dated June, 1991.
|
|
|
|
|
|
|
|
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., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
|
|
|
|
USA.
|
2010-12-10 16:30:45 +01:00
|
|
|
|
2010-12-13 15:46:28 +01:00
|
|
|
=end comment
|
2010-12-10 16:30:45 +01:00
|
|
|
|
|
|
|
=head1 MAGIC MARKERS
|
|
|
|
|
|
|
|
#%# family=auto
|
|
|
|
#%# capabilities=autoconf
|
|
|
|
|
|
|
|
=cut
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use Carp;
|
|
|
|
use Graphics::ColorNames 2.10;
|
|
|
|
use Graphics::ColorObject;
|
|
|
|
|
|
|
|
my $vestat = '/proc/vz/vestat';
|
|
|
|
my $stat = '/proc/stat';
|
|
|
|
|
|
|
|
my @vestatitems = ('user', 'system', 'nice');
|
|
|
|
|
2010-12-13 14:02:10 +01:00
|
|
|
my @colorlist = ('purple', 'green', 'red', 'pink2', 'green', 'brown', 'cyan', 'orange', 'blue', 'grey');
|
2010-12-10 16:30:45 +01:00
|
|
|
|
|
|
|
############
|
|
|
|
# autoconf #
|
|
|
|
############
|
|
|
|
|
|
|
|
if ( defined $ARGV[0] && $ARGV[0] eq 'autoconf' ) {
|
|
|
|
if ( -r '/proc/stat' && -r '/proc/vz/vestat' ) {
|
|
|
|
print "yes\n";
|
|
|
|
} else {
|
|
|
|
print "no\n";
|
|
|
|
}
|
|
|
|
exit 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $hz = 100;
|
|
|
|
|
|
|
|
|
|
|
|
#use Data::Dumper; croak Dumper(\%ENV);
|
|
|
|
|
|
|
|
|
|
|
|
$hz = $ENV{'HZ'} if ( defined $ENV{'HZ'} );
|
|
|
|
|
|
|
|
my $drawidle = 0;
|
|
|
|
|
|
|
|
$drawidle = $ENV{'drawidle'} if ( defined $ENV{'drawidle'} );
|
|
|
|
|
|
|
|
open VESTAT, "<$vestat" or croak "Failed to open $vestat\n";
|
|
|
|
|
|
|
|
my %vejif;
|
|
|
|
|
|
|
|
while (<VESTAT>) {
|
|
|
|
my ($veid, $user, $nice, $system);
|
|
|
|
($veid, $user, $nice, $system) = m,^\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+),;
|
|
|
|
if (defined $veid) {
|
|
|
|
$vejif{$veid}->{'user'} = $user;
|
|
|
|
$vejif{$veid}->{'nice'} = $nice;
|
|
|
|
$vejif{$veid}->{'system'} = $system;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
close VESTAT or croak "Failed to close $vestat\n";
|
|
|
|
|
|
|
|
open STAT, "<$stat" or croak "Failed to open $stat\n";
|
|
|
|
|
|
|
|
my $line = 0;
|
|
|
|
my $cores = 0;
|
|
|
|
|
|
|
|
while (<STAT>) {
|
|
|
|
my ($user, $nice, $system, $idle, $iowait, $irq, $softirq);
|
|
|
|
($user, $nice, $system, $idle, $iowait, $irq, $softirq) = m,^cpu\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+)\W+(\d+),;
|
|
|
|
if (defined $user) {
|
|
|
|
$vejif{'global'}->{'user'} = $user;
|
|
|
|
$vejif{'global'}->{'nice'} = $nice;
|
|
|
|
$vejif{'global'}->{'system'} = $system;
|
|
|
|
$vejif{'global'}->{'idle'} = $idle;
|
|
|
|
$vejif{'global'}->{'iowait'} = $iowait;
|
|
|
|
$vejif{'global'}->{'irq'} = $irq;
|
|
|
|
$vejif{'global'}->{'softirq'} = $softirq;
|
|
|
|
}
|
|
|
|
# Count number of CPU cores on this system while we are here
|
|
|
|
$cores++ if m,cpu\d,;
|
|
|
|
}
|
|
|
|
|
|
|
|
$cores = 1 if $cores == 0;
|
|
|
|
|
|
|
|
foreach my $thing (@vestatitems) {
|
|
|
|
##FIXME this doesn't appear to work on 2.6.32. OpenVZ bug?
|
|
|
|
## #Subtract total container usage from overall usage to get non-openvz
|
|
|
|
## # usage (AKA VEID 0)
|
|
|
|
## $vejif{'0'}->{$thing} = $vejif{'global'}->{$thing};
|
|
|
|
## foreach my $veid (keys %vejif) {
|
|
|
|
## next if $veid eq 'global';
|
|
|
|
## $vejif{'0'}->{$thing} = $vejif{'0'}->{$thing} - $vejif{$veid}->{$thing};
|
|
|
|
## }
|
|
|
|
delete $vejif{'global'}->{$thing};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
##########
|
|
|
|
# config #
|
|
|
|
##########
|
|
|
|
|
|
|
|
if ( defined $ARGV[0] && $ARGV[0] eq 'config' ) {
|
|
|
|
my $uplimit = $cores;
|
|
|
|
my $first = 1;
|
|
|
|
my $veprocessed = 0;
|
|
|
|
#print 'graph_args --base 1000 -r --lower-limit 0 --units-length 4 --y-grid 0.01:10';
|
|
|
|
print 'graph_args --base 1000 -r --lower-limit 0 -Y';
|
|
|
|
print " --upper-limit $uplimit" if $drawidle;
|
|
|
|
print <<STDCONFIG;
|
|
|
|
|
|
|
|
graph_title OpenVZ container CPU usage
|
|
|
|
graph_vlabel CPU cores used ($cores available)
|
|
|
|
graph_scale no
|
|
|
|
graph_info This graph shows how CPU time is spent.
|
2017-02-23 15:31:40 +01:00
|
|
|
graph_category virtualization
|
2010-12-10 16:30:45 +01:00
|
|
|
graph_period second
|
|
|
|
STDCONFIG
|
|
|
|
if (0) {
|
|
|
|
print "graph_printf %6.3lf\n";
|
|
|
|
}
|
|
|
|
foreach my $veid (keys %vejif) {
|
|
|
|
next if $veid eq 'global';
|
|
|
|
my $color = $colorlist[$veprocessed++ % @colorlist];
|
|
|
|
my $vename = `vzlist -H -o name $veid`;
|
|
|
|
($vename) = ($vename =~ m,(\w*),);
|
|
|
|
|
|
|
|
my $cnameobj = new Graphics::ColorNames(qw( X ));
|
|
|
|
|
|
|
|
# Use LCHab colour space in order to be able to change the
|
|
|
|
# Apparent brightness of the color (by scaling the "L"
|
|
|
|
# component). Convert back to RGB for munin.
|
|
|
|
my @lchab = @{
|
|
|
|
(Graphics::ColorObject->new_RGBhex(
|
|
|
|
$cnameobj->hex($color)
|
|
|
|
))->as_LCHab()
|
|
|
|
};
|
|
|
|
|
|
|
|
foreach my $item ('system', 'user', 'nice') {
|
|
|
|
my @itemc;
|
|
|
|
@itemc = @lchab;
|
|
|
|
$itemc[0] = 1.3 * $itemc[0] if ($item eq 'nice');
|
|
|
|
$itemc[0] = 0.7 * $itemc[0] if ($item eq 'system');
|
|
|
|
my $itemcolor = Graphics::ColorObject->new_LCHab(\@itemc);
|
|
|
|
print "v${veid}${item}.label $vename $item\n";
|
|
|
|
if($first) {
|
|
|
|
print "v${veid}${item}.draw AREA\n";
|
|
|
|
$first = 0;
|
|
|
|
} else {
|
|
|
|
print "v${veid}${item}.draw STACK\n";
|
|
|
|
}
|
|
|
|
print "v${veid}${item}.colour " . $itemcolor->as_RGBhex() . "\n";
|
|
|
|
print "v${veid}${item}.min 0\n";
|
|
|
|
print "v${veid}${item}.type DERIVE\n";
|
|
|
|
print "v${veid}${item}.cdef v${veid}${item},100,/\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
print <<STDCONFIG;
|
|
|
|
vglobalirq.label irq
|
|
|
|
vglobalirq.draw STACK
|
|
|
|
vglobalirq.min 0
|
|
|
|
vglobalirq.colour EEEE00
|
|
|
|
vglobalirq.type DERIVE
|
|
|
|
vglobalirq.cdef vglobalirq,100,/
|
|
|
|
vglobalirq.info CPU time spent handling interrupts
|
|
|
|
vglobalsoftirq.label softirq
|
|
|
|
vglobalsoftirq.draw STACK
|
|
|
|
vglobalsoftirq.min 0
|
|
|
|
vglobalsoftirq.colour CCCC00
|
|
|
|
vglobalsoftirq.type DERIVE
|
|
|
|
vglobalsoftirq.cdef vglobalsoftirq,100,/
|
|
|
|
vglobalsoftirq.info CPU time spent handling "batched" interrupts
|
|
|
|
STDCONFIG
|
|
|
|
if ($drawidle) {
|
|
|
|
print <<STDCONFIG
|
|
|
|
vglobalidle.label idle
|
|
|
|
vglobalidle.draw STACK
|
|
|
|
vglobalidle.colour EEEEEE
|
|
|
|
vglobalidle.min 0
|
|
|
|
vglobalidle.type DERIVE
|
|
|
|
vglobalidle.cdef vglobalidle,100,/
|
|
|
|
vglobalidle.info Idle CPU time
|
|
|
|
vglobaliowait.label iowait
|
|
|
|
vglobaliowait.draw STACK
|
|
|
|
vglobaliowait.colour DDDDDD
|
|
|
|
vglobaliowait.min 0
|
|
|
|
vglobaliowait.type DERIVE
|
|
|
|
vglobaliowait.cdef vglobaliowait,100,/
|
|
|
|
vglobaliowait.info CPU time spent waiting for I/O operations to finish when there is nothing else to do.
|
|
|
|
STDCONFIG
|
|
|
|
}
|
|
|
|
exit 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
foreach my $veid (keys %vejif) {
|
|
|
|
foreach my $datapoint (keys %{$vejif{$veid}}) {
|
|
|
|
printf "v${veid}${datapoint}.value %.0f\n", $vejif{$veid}->{$datapoint} / $hz * 100;
|
|
|
|
}
|
|
|
|
}
|