ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/i-scream/experimental/reports/graphing/graph.pl
Revision: 1.6
Committed: Mon Mar 11 17:06:45 2002 UTC (22 years, 8 months ago) by tdb
Content type: text/plain
Branch: MAIN
Changes since 1.5: +21 -9 lines
Log Message:
Fixed the memory/swap issue. Now displays "in use" rather than free. Also
tweaked the algorithm used to make the consolidated graphs - it now does
averaging rather than just taking the maximum value over the period. This
was done because some of the graphs gave unbelievable results - cpu graphs
with a total of well over 100% looks silly ;)
Finally, ruled out being able to set the maximum for disk/mem/swap graphs
from a dynamic value. The only possibly solution would be for watch.pl to
dump out the current maximums to a file, which this could then read and
make use of (external to the RRD stuff - which would have been best). This
would, in my opinion, be silly as it would generate an awful lot more I/O
operations than is actually required.

File Contents

# User Rev Content
1 tdb 1.1 #!/usr/bin/perl -w
2    
3     # -----------------------------------------------------------
4     # i-scream graph generation script
5     # http://www.i-scream.org.uk
6     #
7     # Generates graphs from rrd databases for i-scream data.
8     #
9 tdb 1.4 # $Author: tdb $
10 tdb 1.6 # $Id: graph.pl,v 1.5 2002/03/11 00:25:35 tdb Exp $
11 tdb 1.1 #------------------------------------------------------------
12    
13     ## TODO
14     # possibly make more configurable?
15     # -- allow configurable periods of graphs
16     # -- comments, types, etc
17     # -- move all to external config file
18    
19     $| = 1;
20     use strict;
21     use RRDs;
22    
23     # Base directory for images
24     # (a directory will be constructed for each host under this)
25     my($imgdir) = "/home/tdb/public_html/rrd";
26    
27     # Location of RRD databases
28     my($rrddir) = "/u1/i-scream/rrd";
29    
30     # / converted to a decimal then hex'd
31     my($hex_slash) = "_2f";
32     # _ converted to a decimal then hex'd
33     my($hex_underscore) = "_5f";
34    
35     # Read the contents of the base directory
36     # and pull out the list of subdirectories (except . and .. :)
37     opendir(DIR, $rrddir);
38     my(@rrddirlist) = grep { -d "$rrddir/$_" && !/^\.$/ && !/^\.\.$/ } readdir(DIR);
39     closedir DIR;
40    
41     # look through each directoty, as they might
42     # contain rrds for a particular machine
43     foreach my $machine (@rrddirlist) {
44     # Read the contents of the directory
45     opendir(DIR, "$rrddir/$machine");
46     my(@rrdlist) = grep { /\.rrd$/ && -f "$rrddir/$machine/$_" } readdir(DIR);
47     closedir DIR;
48    
49     # See what rrd we have, and generate the graphs accordingly
50     foreach my $rrd (@rrdlist) {
51     chomp $rrd;
52     if($rrd =~ /^(cpu)\.rrd$/) {
53     my(@data);
54 tdb 1.5 my(@rawdata);
55 tdb 1.3 push @data, "LINE2:$1:idle:idle#00FF00:idle cpu";
56     push @data, "LINE2:$1:user:user#0000FF:user cpu";
57     push @data, "LINE2:$1:kernel:kernel#00FFFF:kernel cpu";
58     push @data, "LINE2:$1:swap:swap#FF00FF:swap cpu";
59     push @data, "LINE2:$1:iowait:iowait#FF0000:iowait cpu";
60 tdb 1.5 push @rawdata, "--upper-limit=100";
61     &makegraph($machine, $1, "CPU Usage for $machine", \@data, \@rawdata);
62 tdb 1.1 }
63     if($rrd =~ /^(mem)\.rrd$/) {
64     my(@data);
65 tdb 1.5 my(@rawdata);
66 tdb 1.6 # we don't actually want to display free memory,
67     # although we need it to do inuse...
68     push @data, "NONE:$1:free:free#CCCCFF:free memory";
69 tdb 1.3 push @data, "LINE2:$1:total:total#0000FF:total memory";
70 tdb 1.6 # calculate inuse
71     push @rawdata, "CDEF:inuse=total,free,-";
72     # and add it to the graph
73     push @rawdata, "AREA:inuse#CCCCFF:memory in use";
74 tdb 1.5 push @rawdata, "--base=1024";
75     &makegraph($machine, $1, "Memory Usage for $machine", \@data, \@rawdata);
76 tdb 1.1 }
77     if($rrd =~ /^(load)\.rrd$/) {
78     my(@data);
79 tdb 1.5 push @data, "LINE2:$1:load1:load1#CCCCFF:1 minute load average";
80     push @data, "LINE2:$1:load5:load5#7777FF:5 minute load average";
81     push @data, "LINE2:$1:load15:load15#0000FF:15 minute load average";
82 tdb 1.1 &makegraph($machine, $1, "Loads for $machine", \@data);
83     }
84     if($rrd =~ /^(proc)\.rrd$/) {
85     my(@data);
86 tdb 1.3 push @data, "LINE2:$1:cpu:cpu#00FF00:cpu processes";
87     push @data, "LINE2:$1:sleeping:sleeping#0000FF:sleeping processes";
88     push @data, "LINE2:$1:stopped:stopped#00FFFF:stopped processes";
89     push @data, "LINE2:$1:total:total#FF00FF:total processes";
90     push @data, "LINE2:$1:zombie:zombie#FF0000:zombie processes";
91 tdb 1.1 &makegraph($machine, $1, "Processes on $machine", \@data);
92     }
93     if($rrd =~ /^(swap)\.rrd$/) {
94     my(@data);
95 tdb 1.5 my(@rawdata);
96 tdb 1.6 # we don't actually want to display free swap,
97     # although we need it to do inuse...
98     push @data, "NONE:$1:free:free#CCCCFF:free swap";
99 tdb 1.3 push @data, "LINE2:$1:total:total#0000FF:total swap";
100 tdb 1.6 # calculate inuse
101     push @rawdata, "CDEF:inuse=total,free,-";
102     # and add it to the graph
103     push @rawdata, "AREA:inuse#CCCCFF:swap in use";
104 tdb 1.5 push @rawdata, "--base=1024";
105     &makegraph($machine, $1, "Swap Usage for $machine", \@data, \@rawdata);
106 tdb 1.1 }
107     if($rrd =~ /^(users)\.rrd$/) {
108     my(@data);
109 tdb 1.5 push @data, "AREA:$1:count:count#CCCCFF:user count";
110 tdb 1.1 &makegraph($machine, $1, "User Count for $machine", \@data);
111     }
112     if($rrd =~ /^(disk)-(\S+).rrd$/) {
113     my(@data);
114 tdb 1.5 my(@rawdata);
115 tdb 1.3 push @data, "LINE2:$1-$2:kbytes:kbytes#0000FF:total size";
116 tdb 1.5 push @data, "AREA:$1-$2:used:used#CCCCFF:used";
117     push @rawdata, "--base=1024";
118 tdb 1.1 my($type) = $1;
119     my($name) = $2;
120     my($nicename) = $2;
121     $nicename =~ s/$hex_slash/\//g;
122     $nicename =~ s/$hex_underscore/_/g;
123 tdb 1.5 &makegraph($machine, "$type-$name", "Disk Usage for $machine on $nicename", \@data, \@rawdata);
124 tdb 1.1 }
125     # probably a queue with a name like this :)
126     if($rrd =~ /^(\d+)_0\.rrd$/) {
127     my(@data);
128     my(@rawdata);
129     my($baserrd) = $1;
130     my($i) = 0;
131     while( -f "$rrddir/$machine/$baserrd\_$i.rrd" ) {
132 tdb 1.3 push @data, "LINE2:$baserrd\_$i:size:size$i" . &get_colour($i) . ":queue$i size ";
133 tdb 1.1 ++$i;
134     }
135 tdb 1.3 push @data, "LINE2:$baserrd\_0:total:total#FF0000:packets/sec - currently";
136 tdb 1.2 push @rawdata, "GPRINT:total:LAST:%lf %spackets/sec";
137 tdb 1.1 my($comment);
138     if(-f "$rrddir/$machine/$baserrd.def") {
139     open(DEF, "$rrddir/$machine/$baserrd.def");
140     $comment = <DEF>;
141     chomp $comment if defined $comment;
142     }
143     $comment = "unknown queue" if not defined $comment;
144     &makegraph($machine, $baserrd, $comment, \@data, \@rawdata);
145     }
146     }
147     }
148    
149     #
150     # subroutine to make some graphs
151     #
152     # $machine = name of the machine
153     # (eg. kernow.ukc.ac.uk)
154     # $type = the type of graph for the machine
155     # (eg. cpu)
156     # $title = the title for the graph
157     # (eg. kernow CPU usage)
158     # $dataref = a reference to an array containing information for the graph
159 tdb 1.3 # elements of format: "gtype:rrdname:dsname:name#colour:comment with spaces"
160 tdb 1.6 # (if gtype is "NONE" only a DEF of 'name' will be defined, no line will be plotted)
161 tdb 1.1 # $rawcmdref = a reference to an array containing raw rrd commands
162     # elements a single command each, no spaces
163     #
164 tdb 1.2
165 tdb 1.1 sub makegraph() {
166     my($machine, $type, $title, $dataref, $rawcmdref) = @_;
167     # pass in these arrays by reference
168     my(@data) = @$dataref if defined $dataref;
169     my(@rawcmd) = @$rawcmdref if defined $rawcmdref;
170     # check if directory exists for images
171     if(! -d "$imgdir/$machine") {
172     # not sure on this umask, but it seems to work?
173     mkdir "$imgdir/$machine", 0777;
174     }
175     my(@rrdcmd);
176     foreach my $dataitem (@data) {
177 tdb 1.3 # dataitem should be: "gtype:rrdname:dsname:name#colour:comment with spaces"
178 tdb 1.6 # (if gtype is "NONE" only a DEF of 'name' will be defined, no line will be plotted)
179 tdb 1.3 if($dataitem =~ /^(\S+):(\S+):(\S+):(\S+)#(.{6}):(.*)$/) {
180 tdb 1.6 push @rrdcmd, "DEF:$4=$rrddir/$machine/$2.rrd:$3:AVERAGE";
181     if($1 ne "NONE") {
182     push @rrdcmd, "$1:$4#$5:$6";
183     }
184 tdb 1.1 }
185     }
186     push @rrdcmd, "--title=$title";
187     push @rrdcmd, "--imgformat=PNG";
188     push @rrdcmd, "--lower-limit=0";
189 tdb 1.5 # not entirely convinced this is good...
190     push @rrdcmd, "--alt-autoscale-max";
191 tdb 1.1 # add any further raw commands
192     push @rrdcmd, @rawcmd;
193     RRDs::graph ("$imgdir/$machine/$type-3h.png", "--start=-10800", @rrdcmd);
194     my($err_3h) = RRDs::error;
195 tdb 1.2 print STDERR "Error generating 3h graph for $machine/$type: $err_3h\n" if $err_3h;
196 tdb 1.1 RRDs::graph ("$imgdir/$machine/$type-1d.png", "--start=-86400", @rrdcmd);
197     my($err_1d) = RRDs::error;
198 tdb 1.2 print STDERR "Error generating 1d graph for $machine/$type: $err_1d\n" if $err_1d;
199 tdb 1.1 RRDs::graph ("$imgdir/$machine/$type-1w.png", "--start=-604800", @rrdcmd);
200     my($err_1w) = RRDs::error;
201 tdb 1.2 print STDERR "Error generating 1w graph for $machine/$type: $err_1w\n" if $err_1w;
202 tdb 1.1 RRDs::graph ("$imgdir/$machine/$type-1m.png", "--start=-2678400", @rrdcmd);
203     my($err_1m) = RRDs::error;
204 tdb 1.2 print STDERR "Error generating 1m graph for $machine/$type: $err_1m\n" if $err_1m;
205 tdb 1.1 return;
206     }
207    
208     # hacky subroutine to return a colour
209     # could be done much better somehow :/
210     sub get_colour {
211     my($col) = @_;
212     if($col == 0) {
213     return "#0000FF";
214     }
215     elsif($col == 1) {
216     return "#00FF00";
217     }
218     elsif($col == 2) {
219     return "#FF00FF";
220     }
221     elsif($col == 3) {
222     return "#FFFF00";
223     }
224     elsif($col == 4) {
225     return "#00FFFF";
226     }
227     else {
228     return "#000066";
229     }
230     }