| 1 | #!/usr/bin/perl | 
|---|
| 2 |  | 
|---|
| 3 | # File:    $Id: show.cgi,v 1.22 2006/04/12 09:42:16 sauber Exp $ | 
|---|
| 4 | # Author:  (c) Soren Dossing, 2005 | 
|---|
| 5 | # License: OSI Artistic License | 
|---|
| 6 | #          http://www.opensource.org/licenses/artistic-license.php | 
|---|
| 7 |  | 
|---|
| 8 | use strict; | 
|---|
| 9 | use RRDs; | 
|---|
| 10 | use CGI qw/:standard/; | 
|---|
| 11 |  | 
|---|
| 12 | # Configuration | 
|---|
| 13 | my $configfile = '/home/nagios/ng/etc/nagiosgraph.conf'; | 
|---|
| 14 |  | 
|---|
| 15 | # Main program - change nothing below | 
|---|
| 16 |  | 
|---|
| 17 | my %Config; | 
|---|
| 18 |  | 
|---|
| 19 | # Read in configuration data | 
|---|
| 20 | # | 
|---|
| 21 | sub readconfig { | 
|---|
| 22 | die "config file not found" unless -r $configfile; | 
|---|
| 23 |  | 
|---|
| 24 | # Read configuration data | 
|---|
| 25 | open FH, $configfile; | 
|---|
| 26 | while (<FH>) { | 
|---|
| 27 | s/\s*#.*//;    # Strip comments | 
|---|
| 28 | /^(\w+)\s*=\s*(.*?)\s*$/ and do { | 
|---|
| 29 | $Config{$1} = $2; | 
|---|
| 30 | debug(5, "CGI Config $1:$2"); | 
|---|
| 31 | }; | 
|---|
| 32 | } | 
|---|
| 33 | close FH; | 
|---|
| 34 |  | 
|---|
| 35 | # Make sure log file can be written to | 
|---|
| 36 | unless ( -w $Config{logfile} ) { | 
|---|
| 37 | my $msg = "Log file $Config{logfile} not writable"; | 
|---|
| 38 | print header(-type => "text/html", -expires => 0); | 
|---|
| 39 | print p($msg); | 
|---|
| 40 | debug (2, "CGI Config $msg"); | 
|---|
| 41 | return undef; | 
|---|
| 42 | } | 
|---|
| 43 |  | 
|---|
| 44 | # Make sure rrddir is readable | 
|---|
| 45 | unless ( -r $Config{rrddir} ) { | 
|---|
| 46 | my $msg = "rrd dir $Config{rrddir} not readable"; | 
|---|
| 47 | print header(-type => "text/html", -expires => 0); | 
|---|
| 48 | print p($msg); | 
|---|
| 49 | debug (2, "CGI Config $msg"); | 
|---|
| 50 | return undef; | 
|---|
| 51 | } | 
|---|
| 52 |  | 
|---|
| 53 | return 1; | 
|---|
| 54 | } | 
|---|
| 55 |  | 
|---|
| 56 | # Write debug information to log file | 
|---|
| 57 | # | 
|---|
| 58 | sub debug { | 
|---|
| 59 | my($l, $text) = @_; | 
|---|
| 60 | if ( $l <= $Config{debug} ) { | 
|---|
| 61 | $l = qw(none critical error warn info debug)[$l]; | 
|---|
| 62 | $text =~ s/(\w+)/$1 $l:/; | 
|---|
| 63 | open LOG, ">>$Config{logfile}"; | 
|---|
| 64 | print LOG scalar localtime; | 
|---|
| 65 | print LOG " $text\n"; | 
|---|
| 66 | close LOG; | 
|---|
| 67 | } | 
|---|
| 68 | } | 
|---|
| 69 |  | 
|---|
| 70 | # URL encode a string | 
|---|
| 71 | # | 
|---|
| 72 | sub urlencode { | 
|---|
| 73 | $_[0] =~ s/([\W])/"%" . uc(sprintf("%2.2x",ord($1)))/eg; | 
|---|
| 74 | return $_[0]; | 
|---|
| 75 | } | 
|---|
| 76 |  | 
|---|
| 77 | # Get list of matching rrd files | 
|---|
| 78 | # | 
|---|
| 79 | sub dbfilelist { | 
|---|
| 80 | my($host,$service) = @_; | 
|---|
| 81 | my $hs = urlencode "${host}_${service}"; | 
|---|
| 82 | my @rrd; | 
|---|
| 83 | opendir DH, $Config{rrddir}; | 
|---|
| 84 | @rrd = grep s/^${hs}_(.+)\.rrd$/$1/, readdir DH; | 
|---|
| 85 | closedir DH; | 
|---|
| 86 | return @rrd; | 
|---|
| 87 | } | 
|---|
| 88 |  | 
|---|
| 89 | # Find graphs and values | 
|---|
| 90 | # | 
|---|
| 91 | sub graphinfo { | 
|---|
| 92 | my($host,$service,@db) = @_; | 
|---|
| 93 | my(@rrd,$ds,$f,$dsout,@values,$hs,%H,%R); | 
|---|
| 94 |  | 
|---|
| 95 | $hs = urlencode "${host}_${service}"; | 
|---|
| 96 |  | 
|---|
| 97 | debug(5, 'CGI @db=' . join '&', @db); | 
|---|
| 98 |  | 
|---|
| 99 | # Determine which files to read lines from | 
|---|
| 100 | if ( @db ) { | 
|---|
| 101 | my $n = 0; | 
|---|
| 102 | for my $d ( @db ) { | 
|---|
| 103 | my($db,@lines) = split ',', $d; | 
|---|
| 104 | $rrd[$n]{file} = $hs . urlencode("_$db") . '.rrd'; | 
|---|
| 105 | for my $l ( @lines ) { | 
|---|
| 106 | my($line,$unit) = split '~', $l; | 
|---|
| 107 | if ( $unit ) { | 
|---|
| 108 | $rrd[$n]{line}{$line}{unit} = $unit if $unit; | 
|---|
| 109 | } else { | 
|---|
| 110 | $rrd[$n]{line}{$line} = 1; | 
|---|
| 111 | } | 
|---|
| 112 | } | 
|---|
| 113 | $n++; | 
|---|
| 114 | } | 
|---|
| 115 | debug(4, "CGI Specified $hs db files in $Config{rrddir}: " | 
|---|
| 116 | . join ', ', map { $_->{file} } @rrd); | 
|---|
| 117 | } else { | 
|---|
| 118 | @rrd = map {{ file=>$_ }} | 
|---|
| 119 | map { "${hs}_${_}.rrd" } | 
|---|
| 120 | dbfilelist($host,$service); | 
|---|
| 121 | debug(4, "CGI Listing $hs db files in $Config{rrddir}: " | 
|---|
| 122 | . join ', ', map { $_->{file} } @rrd); | 
|---|
| 123 | } | 
|---|
| 124 |  | 
|---|
| 125 | for $f ( @rrd ) { | 
|---|
| 126 | unless ( $f->{line} ) { | 
|---|
| 127 | $ds = RRDs::info "$Config{rrddir}/$f->{file}"; | 
|---|
| 128 | debug(2, "CGI RRDs::info ERR " . RRDs::error) if RRDs::error; | 
|---|
| 129 | map { $f->{line}{$_} = 1} | 
|---|
| 130 | grep {!$H{$_}++} | 
|---|
| 131 | map { /ds\[(.*)\]/; $1 } | 
|---|
| 132 | grep /ds\[(.*)\]/, | 
|---|
| 133 | keys %$ds; | 
|---|
| 134 | } | 
|---|
| 135 | debug(5, "CGI DS $f->{file} lines: " | 
|---|
| 136 | . join ', ', keys %{ $f->{line} } ); | 
|---|
| 137 | } | 
|---|
| 138 | return \@rrd; | 
|---|
| 139 | } | 
|---|
| 140 |  | 
|---|
| 141 | # Choose a color for service | 
|---|
| 142 | # | 
|---|
| 143 | sub hashcolor { | 
|---|
| 144 | my$c=$Config{colorscheme}; | 
|---|
| 145 | map{ | 
|---|
| 146 | $c=(51*$c+ord)%(216) | 
|---|
| 147 | } split//,"$_[0]x"; | 
|---|
| 148 | my($i,$n,$m,@h); | 
|---|
| 149 | @h=(51*int$c/36, | 
|---|
| 150 | 51*int$c/6%6, | 
|---|
| 151 | 51*($c%6)); | 
|---|
| 152 | #debug(2, "hashcolor $_[0], $c, $h[0]"); | 
|---|
| 153 | for$i(0..2){ | 
|---|
| 154 | $m=$i if$h[$i]<$h[$m]; | 
|---|
| 155 | $n=$i if$h[$i]>$h[$n] | 
|---|
| 156 | } | 
|---|
| 157 | $h[$m]=102 if$h[$m]>102; | 
|---|
| 158 | $h[$n]=153 if$h[$n]<153; | 
|---|
| 159 | #debug(2, "hashcolor $_[0]\t$c\t$h[0]\t$h[1]\t$h[2]"); | 
|---|
| 160 | #$c=sprintf"%06X",$h[2]+$h[1]*256+$h[0]*16**4; | 
|---|
| 161 | $n = $h[2]+$h[1]*256+$h[0]*16**4; | 
|---|
| 162 | $c=sprintf"%06X",$n; | 
|---|
| 163 | #debug(2, "hashcolor $_[0]\t$n\t$c"); | 
|---|
| 164 | return $c; | 
|---|
| 165 | } | 
|---|
| 166 |  | 
|---|
| 167 | # Generate all the parameters for rrd to produce a graph | 
|---|
| 168 | # | 
|---|
| 169 | sub rrdline { | 
|---|
| 170 | my($host,$service,$geom,$rrdopts,$G,$time) = @_; | 
|---|
| 171 | my($g,$f,$v,$c,@ds); | 
|---|
| 172 |  | 
|---|
| 173 | @ds = ('-', '-a', 'PNG', '--start', "-$time"); | 
|---|
| 174 | # Identify where to pull data from and what to call it | 
|---|
| 175 | for $g ( @$G ) { | 
|---|
| 176 | $f = $g->{file}; | 
|---|
| 177 | debug(5, "CGI file=$f"); | 
|---|
| 178 | for $v ( sort keys %{ $g->{line} } ) { | 
|---|
| 179 | $c = hashcolor($v); | 
|---|
| 180 | debug(5, "CGI file=$f line=$v color=$c"); | 
|---|
| 181 | my $sv = "$v"; | 
|---|
| 182 | push @ds , "DEF:$sv=$Config{rrddir}/$f:$v:AVERAGE" | 
|---|
| 183 | , "LINE2:${sv}#$c:$sv" | 
|---|
| 184 | , "GPRINT:$sv:MAX:Max\\: %6.2lf%s" | 
|---|
| 185 | , "GPRINT:$sv:AVERAGE:Avg\\: %6.2lf%s" | 
|---|
| 186 | , "GPRINT:$sv:MIN:Min\\: %6.2lf%s" | 
|---|
| 187 | , "GPRINT:$sv:LAST:Cur\\: %6.2lf%s\\n"; | 
|---|
| 188 | } | 
|---|
| 189 | } | 
|---|
| 190 |  | 
|---|
| 191 | # Dimensions of graph if geom is specified | 
|---|
| 192 | if ( $geom ) { | 
|---|
| 193 | my($w,$h) = split 'x', $geom; | 
|---|
| 194 | push @ds, '-w', $w, '-h', $h; | 
|---|
| 195 | } | 
|---|
| 196 | # Additional parameters to rrd graph, if specified | 
|---|
| 197 | if ( $rrdopts ) { | 
|---|
| 198 | push @ds, split /\s+/, $rrdopts; | 
|---|
| 199 | } | 
|---|
| 200 | return @ds; | 
|---|
| 201 | } | 
|---|
| 202 |  | 
|---|
| 203 | # Write a pretty page with various graphs | 
|---|
| 204 | # | 
|---|
| 205 | sub page { | 
|---|
| 206 | my($h,$s,$d,$o,@db) = @_; | 
|---|
| 207 |  | 
|---|
| 208 | # Reencode rrdopts | 
|---|
| 209 | $o = urlencode $o; | 
|---|
| 210 |  | 
|---|
| 211 | # Detect available db files | 
|---|
| 212 | @db = dbfilelist($h,$s) unless @db; | 
|---|
| 213 | debug(5, "CGI dbfilelist @db"); | 
|---|
| 214 |  | 
|---|
| 215 | # Define graph sizes | 
|---|
| 216 | #   Daily   =  33h =   118800s | 
|---|
| 217 | #   Weekly  =   9d =   777600s | 
|---|
| 218 | #   Monthly =   5w =  3024000s | 
|---|
| 219 | #   Yearly  = 400d = 34560000s | 
|---|
| 220 | my @T=(['dai',118800], ['week',777600], ['month',3024000], ['year',34560000]); | 
|---|
| 221 | print h1("Nagiosgraph"); | 
|---|
| 222 | print p("Performance data for ".strong("Host: ").tt($h).' · '.strong("Service: ").tt($s)); | 
|---|
| 223 | for my $l ( @T ) { | 
|---|
| 224 | my($p,$t) = ($l->[0],$l->[1]); | 
|---|
| 225 | print h2(ucfirst $p . "ly"); | 
|---|
| 226 | if ( @db ) { | 
|---|
| 227 | for my $g ( @db ) { | 
|---|
| 228 | my $arg = join '&', "host=$h", "service=$s", "db=$g", "graph=$t", | 
|---|
| 229 | "geom=$d", "rrdopts=$o"; | 
|---|
| 230 | my @gl = split ',', $g; | 
|---|
| 231 | my $ds = shift @gl; | 
|---|
| 232 | print div({-class => "graphs"}, img( {-src => "?$arg", -alt => "Graph"} ) ); | 
|---|
| 233 | print div({-class => "graph_description"}, cite(strong($ds).br().small(join(", ", @gl)))); | 
|---|
| 234 | } | 
|---|
| 235 | } else { | 
|---|
| 236 | my $arg = join '&', "host=$h", "service=$s", "graph=$t", | 
|---|
| 237 | "geom=$d", "rrdopts=$o"; | 
|---|
| 238 | print div({-class => "graphs"}, img( {-src => "?$arg", -alt => "Graph"} ) ); | 
|---|
| 239 | } | 
|---|
| 240 | } | 
|---|
| 241 | } | 
|---|
| 242 |  | 
|---|
| 243 | exit unless readconfig(); | 
|---|
| 244 |  | 
|---|
| 245 | # Expect host, service and db input | 
|---|
| 246 | my $host = param('host') if param('host'); | 
|---|
| 247 | my $service = param('service') if param('service'); | 
|---|
| 248 | my @db = param('db') if param('db'); | 
|---|
| 249 | my $graph = param('graph') if param('graph'); | 
|---|
| 250 | my $geom = param('geom') if param('geom'); | 
|---|
| 251 | my $rrdopts = param('rrdopts') if param('rrdopts'); | 
|---|
| 252 |  | 
|---|
| 253 | # Draw a graph or a page | 
|---|
| 254 | if ( $graph ) { | 
|---|
| 255 | $| = 1; # Make sure headers arrive before image data | 
|---|
| 256 | print header(-type => "image/png"); | 
|---|
| 257 | # Figure out db files and line labels | 
|---|
| 258 | my $G = graphinfo($host,$service,@db); | 
|---|
| 259 | my @ds = rrdline($host,$service,$geom,$rrdopts,$G,$graph); | 
|---|
| 260 | debug(4, "CGI RRDs::graph ". join ' ', @ds); | 
|---|
| 261 | RRDs::graph(@ds); | 
|---|
| 262 | debug(2, "CGI RRDs::graph ERR " . RRDs::error) if RRDs::error; | 
|---|
| 263 | exit; | 
|---|
| 264 | } else { | 
|---|
| 265 | my @style; | 
|---|
| 266 | if ($Config{stylesheet}) { | 
|---|
| 267 | @style = ( -style => {-src => "$Config{stylesheet}"} ); | 
|---|
| 268 | } | 
|---|
| 269 | print header, start_html(-id=>"nagiosgraph", -title => "nagiosgraph: $host-$service", | 
|---|
| 270 | -meta => { -http_equiv => "Refresh", -content => "300" }, | 
|---|
| 271 | @style | 
|---|
| 272 | ); | 
|---|
| 273 | page($host,$service,$geom,$rrdopts,@db); | 
|---|
| 274 | print div({-id => "footer"}, hr(), small( "Created by ". a( {-href=>"http://nagiosgraph.sf.net/"}, "nagiosgraph"). "." )); | 
|---|
| 275 | print end_html(); | 
|---|
| 276 | } | 
|---|