forked from Icinga/icinga-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathp1.pl.in
777 lines (544 loc) · 31.3 KB
/
p1.pl.in
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
package Embed::Persistent;
# p1.pl for Icinga
use strict ;
use Text::ParseWords qw(shellwords) ;
use constant LEAVE_MSG => 1 ;
use constant CACHE_DUMP => 2 ;
use constant PLUGIN_DUMP => 4 ;
use constant DEBUG_LEVEL => 0 ;
# use constant DEBUG_LEVEL => CACHE_DUMP ;
# use constant DEBUG_LEVEL => LEAVE_MSG ;
# use constant DEBUG_LEVEL => LEAVE_MSG | CACHE_DUMP ;
# use constant DEBUG_LEVEL => LEAVE_MSG | CACHE_DUMP | PLUGIN_DUMP ;
use constant DEBUG_LOG_PATH => '@LOGDIR@' ;
# use constant DEBUG_LOG_PATH => './' ;
use constant LEAVE_MSG_STREAM => DEBUG_LOG_PATH . 'epn_leave-msgs.log' ;
use constant CACHE_DUMP_STREAM => DEBUG_LOG_PATH . 'epn_cache-dump.log' ;
use constant PLUGIN_DUMP_STREAM => DEBUG_LOG_PATH . 'epn_plugin-dump.log' ;
use constant NUMBER_OF_PERL_PLUGINS => 60 ;
use constant Cache_Dump_Interval => 20 ;
# Cache will be dumped every Cache_Dump_Interval plugin compilations
(DEBUG_LEVEL & LEAVE_MSG) && do {
open LH, '>> ' . LEAVE_MSG_STREAM
or die "Can't open " . LEAVE_MSG_STREAM . ": $!" ;
# Unbuffer LH since this will be written by child processes.
select( (select(LH), $| = 1)[0] ) ;
} ;
(DEBUG_LEVEL & CACHE_DUMP) && do {
(open CH, '>> ' . CACHE_DUMP_STREAM
or die "Can't open " . CACHE_DUMP_STREAM . ": $!") ;
select( (select(CH), $| = 1)[0] ) ;
} ;
(DEBUG_LEVEL & PLUGIN_DUMP) && (open PH, '>> ' . PLUGIN_DUMP_STREAM
or die "Can't open " . PLUGIN_DUMP_STREAM . ": $!") ;
require Data::Dumper
if DEBUG_LEVEL & CACHE_DUMP ;
my (%Cache, $Current_Run) ;
keys %Cache = NUMBER_OF_PERL_PLUGINS ;
# Offsets in %Cache{$filename}
use constant MTIME => 0 ;
use constant PLUGIN_ARGS => 1 ;
use constant PLUGIN_ERROR => 2 ;
use constant PLUGIN_HNDLR => 3 ;
package main;
use subs 'CORE::GLOBAL::exit';
sub CORE::GLOBAL::exit { die "ExitTrap: $_[0] (Redefine exit to trap plugin exit with eval BLOCK)" }
package OutputTrap;
# Methods for use by tied STDOUT in embedded PERL module.
# Simply ties STDOUT to a scalar and caches values written to it.
# NB No more than 4KB characters per line are kept.
sub TIEHANDLE {
my ($class) = @_;
my $me = '';
bless \$me, $class;
}
sub PRINT {
my $self = shift;
# $$self = substr(join('',@_), 0, 256) ;
$$self .= substr(join('',@_), 0, 4096) ;
}
sub PRINTF {
my $self = shift;
my $fmt = shift;
# $$self = substr(sprintf($fmt,@_), 0, 256) ;
$$self .= substr(sprintf($fmt,@_), 0, 4096) ;
}
sub READLINE {
my $self = shift;
# CHANGED 12/26/07 EG Following two statements didn't allow for multi-line output or output > 256 chars
# Omit all lines after the first, per the icinga plugin guidelines
# $$self = (split /\n/, $$self)[0];
# Perl code other than plugins may print nothing; in this case return "(No output!)\n".
# return $$self ? substr($$self, 0, 256) : "(No output!)\n" ;
return $$self;
}
sub CLOSE {
my $self = shift;
undef $self ;
}
sub DESTROY {
my $self = shift;
undef $self;
}
package Embed::Persistent;
sub valid_package_name {
local $_ = shift ;
s|([^A-Za-z0-9\/])|sprintf("_%2x",unpack("C",$1))|eg;
# second pass only for words starting with a digit
s|/(\d)|sprintf("/_%2x",unpack("C",$1))|eg;
# Dress it up as a real package name
s|/|::|g;
return /^::/ ? "Embed$_" : "Embed::$_" ;
}
# Perl 5.005_03 only traps warnings for errors classed by perldiag
# as Fatal (eg 'Global symbol """"%s"""" requires explicit package name').
# Therefore treat all warnings as fatal.
sub throw_exception {
die shift ;
}
sub eval_file {
my ($filename, $delete, undef, $plugin_args) = @_ ;
my $mtime = -M $filename ;
my $ts = localtime(time())
if DEBUG_LEVEL ;
if (
exists($Cache{$filename}) &&
$Cache{$filename}[MTIME] &&
$Cache{$filename}[MTIME] <= $mtime
) {
# We have compiled this plugin before and
# it has not changed on disk; nothing to do except
# 1 parse the plugin arguments and cache them (to save
# repeated parsing of the same args) - the same
# plugin could be called with different args.
# 2 return the error from a former compilation
# if there was one.
$Cache{$filename}[PLUGIN_ARGS]{$plugin_args} ||= [ shellwords($plugin_args) ]
if $plugin_args ;
if ( $Cache{$filename}[PLUGIN_ERROR] ) {
print LH qq($ts eval_file: $filename failed compilation formerly and file has not changed; skipping compilation.\n)
if DEBUG_LEVEL & LEAVE_MSG ;
die qq(**ePN failed to compile $filename: "$Cache{$filename}[PLUGIN_ERROR]") ;
} else {
print LH qq($ts eval_file: $filename already successfully compiled and file has not changed; skipping compilation.\n)
if DEBUG_LEVEL & LEAVE_MSG ;
return $Cache{$filename}[PLUGIN_HNDLR] ;
}
}
my $package = valid_package_name($filename) ;
$Cache{$filename}[PLUGIN_ARGS]{$plugin_args} ||= [ shellwords($plugin_args) ]
if $plugin_args ;
local *FH;
# die will be trapped by caller (checking ERRSV)
open FH, $filename
or die qq(**ePN failed to open "$filename": "$!") ;
my $sub ;
sysread FH, $sub, -s FH ;
close FH;
# Cater for scripts that have embedded EOF symbols (__END__)
# XXXX
# Doesn't make sense to me.
# $sub =~ s/__END__/\;}\n__END__/;
# Wrap the code into a subroutine inside our unique package
# (using $_ here [to save a lexical] is not a good idea since
# the definition of the package is visible to any other Perl
# code that uses [non localised] $_).
my $hndlr = <<EOSUB ;
package $package;
sub hndlr {
\@ARGV = \@_ ;
local \$^W = 1 ;
\$ENV{NAGIOS_PLUGIN} = '$filename' ;
# <<< START of PLUGIN (first line of plugin is line 8 in the text) >>>
$sub
# <<< END of PLUGIN >>>
}
EOSUB
$Cache{$filename}[MTIME] = $mtime
unless $delete ;
# Suppress warning display.
local $SIG{__WARN__} = \&throw_exception ;
# Following 3 lines added 10/18/07 by Larry Low to fix problem where
# modified Perl plugins didn't get recached by the epn
no strict 'refs';
undef %{$package.'::'};
use strict 'refs';
# Compile &$package::hndlr. Since non executable code is being eval'd
# there is no need to protect lexicals in this scope.
eval $hndlr;
# $@ is set for any warning and error.
# This guarantees that the plugin will not be run.
if ($@) {
# Report error line number wrt to original plugin text (7 lines added by eval_file).
# Error text looks like
# 'Use of uninitialized ..' at (eval 23) line 186, <DATA> line 218
# The error line number is 'line 186'
chomp($@) ;
$@ =~ s/line (\d+)[\.,]/'line ' . ($1 - 7) . ','/e ;
print LH qq($ts eval_file: syntax error in $filename: "$@".\n)
if DEBUG_LEVEL & LEAVE_MSG ;
if ( DEBUG_LEVEL & PLUGIN_DUMP ) {
my $i = 1 ;
$_ = $hndlr ;
s/^/sprintf('%10d ', $i++)/meg ;
# Will only get here once (when a faulty plugin is compiled).
# Therefore only _faulty_ plugins are dumped once each time the text changes.
print PH qq($ts eval_file: transformed plugin "$filename" to ==>\n$_\n) ;
}
$@ = substr($@, 0, 4096)
if length($@) > 4096 ;
$Cache{$filename}[PLUGIN_ERROR] = $@ ;
# If the compilation fails, leave nothing behind that may affect subsequent
# compilations. This will be trapped by caller (checking ERRSV).
die qq(**ePN failed to compile $filename: "$@") ;
} else {
$Cache{$filename}[PLUGIN_ERROR] = '' ;
}
print LH qq($ts eval_file: successfully compiled "$filename $plugin_args".\n)
if DEBUG_LEVEL & LEAVE_MSG ;
print CH qq($ts eval_file: after $Current_Run compilations \%Cache =>\n), Data::Dumper->Dump([\%Cache], [qw(*Cache)]), "\n"
if ( (DEBUG_LEVEL & CACHE_DUMP) && (++$Current_Run % Cache_Dump_Interval == 0) ) ;
no strict 'refs' ;
return $Cache{$filename}[PLUGIN_HNDLR] = *{ $package . '::hndlr' }{CODE} ;
}
sub run_package {
my ($filename, undef, $plugin_hndlr_cr, $plugin_args) = @_;
# Second parm (after $filename) _may_ be used to wallop stashes.
my $has_exit = 0 ;
my $res = 3 ;
my $ts = localtime( time() ) if DEBUG_LEVEL ;
local $SIG{__WARN__} = \&throw_exception ;
my $stdout = tie (*STDOUT, 'OutputTrap');
my @plugin_args = $plugin_args ? @{ $Cache{$filename}[PLUGIN_ARGS]{$plugin_args} } : () ;
# If the plugin has args, they have been cached by eval_file.
# ( cannot cache @plugin_args here because run_package() is
# called by child processes so cannot update %Cache.)
eval { $plugin_hndlr_cr->(@plugin_args) } ;
if ($@) {
# Error => normal plugin termination (exit) || run time error.
$_ = $@ ;
/^ExitTrap: (-?\d+)/ ? do { $has_exit = 1; $res = $1 } :
# For normal plugin exit, $@ will always match /^ExitTrap: (-?\d+)/
/^ExitTrap: / ? do { $has_exit = 1; $res = 0 } : do {
# Run time error/abnormal plugin termination.
chomp ;
# Report error line number wrt to original plugin text (7 lines added by eval_file).
s/line (\d+)[\.,]/'line ' . ($1 - 7) . ','/e ;
print STDOUT qq(**ePN $filename: "$_".\n) ;
} ;
($@, $_) = ('', '') ;
}
# ! Error => Perl code is not a plugin (fell off the end; no exit)
# !! (read any output from the tied file handle.)
my $plugin_output = <STDOUT> ;
undef $stdout ;
untie *STDOUT;
$plugin_output = "**ePN $filename: plugin did not call exit()\n".$plugin_output if $has_exit == 0;
print LH qq($ts run_package: "$filename $plugin_args" returning ($res, "$plugin_output").\n)
if DEBUG_LEVEL & LEAVE_MSG ;
return ($res, $plugin_output) ;
}
1;
=head1 NAME
p1.pl - Perl program to provide Perl code persistence for the Icinga project (http://www.Icinga.Org).
This program provides an API for calling Icinga Perl plugins from Icinga when it is built with embedded Perl support. The
intent is to tradeoff memory usage (see BUGS) against repeated context switches to recompile Perl plugins.
=head1 SYNOPSIS
B<From C>
/* 1 Initialise Perl - see perlembed - maintaining a persistent interpreter> */
/* 2 Push the arguments (char *args[]) of call_pv() onto the Perl stack */
/* 3 Compile the plugin - char *args[] is an array of pointers to strings required by p1.pl */
call_pv("Embed::Persistent::eval_file", G_SCALAR | G_EVAL)
/* 4 if (SvTrue(ERRSV)) */
goto outahere ;
/* 5 Pop the code reference to the Perl sub (corresp to the plugin) returned by Perl */
/* 6 Push the arguments (char *args[]) of call_pv() onto the Perl stack */
/* 7 Run the plugin */
call_pv("Embed::Persistent::run_package", G_ARRAY)
/* 8 example arguments for call_ functions */
args = { "check_rcp", /* pointer to plugin file name */
"1", /* 1 to recompile plugins each time */
"", /* temporary file name - no longer used */
"-H sundev -C nfs" /* pointer to plugin argument string */
}
B<From Perl>
my ($plugin_file, $plugin_args) = split(/\s+/, $_, 2) ;
my $plugin_hndlr_cr ;
eval {
# 'eval {}' simulates the G_EVAL flag to perl_call_argv('code', 'flags')
# Otherwise, die in 'eval_file' will end the caller also.
$plugin_hndlr_cr = Embed::Persistent::eval_file($plugin_file, 0, '', $plugin_args) ;
} ;
if ( $@) {
print "plugin compilation failed.\n" ;
} else {
my ($rc, $output) = Embed::Persistent::run_package($plugin_file, 0, $plugin_hndlr_cr, $plugin_args) ;
printf "embedded perl plugin return code and output was: %d & %s\n", $rc, $output) ;
In the p1.pl text, 'use constant' statements set the log path and the log level.
The log level flags determine the amount and type of messages logged in the log path.
The default log level results in similar behaviour to former versions of p1.pl -
log files will B<not> be opened.
If you want to enable logging
=over 4
=item 1 Choose log options (see below)
=item 2 Choose a log path
=item 3 Edit p1.pl
=back
Set the values of (the 'use constant' statements) the log path, B<DEBUG_LOG_PATH>, and set the B<DEBUG_LEVEL> constant to
one or more of the log options (B<LEAVE_MSG> and friends ) or'ed together.
The default is to log nothing and to use S<<path_to_Icinga>/var/epn_stderr.log> as the log path.
=head1 DESCRIPTION
Icinga is a program to monitor service availability by scheduling 'plugins' - discrete programs
that check a service (by for example simulating a users interaction with a web server using WWW::Mechanize) and output a line of
text (the summary of the service state) for those responsible for the service, and exit with a coded value to relay the same information to Icinga.
Each plugin is run in a new child process forked by Icinga.
Plugins, like CGIs, can be coded in Perl. The persistence framework embeds a Perl interpreter in Icinga to
=over 4
=item * reduce the time taken for the Perl compile and execute cycle.
=item * eliminate the need for Icinga to fork a process (with popen) to run the Perl code.
=item * eliminate reloading of frequently used modules.
=back
and all the good things mentioned in the B<perlembed> man page under 'Maintaining a persistent interpreter'.
Plugin run-time and syntax errors, are returned to Icinga as the 'plugin output'. These messages
appear in the Icinga log like S<**ePN 'check_test' Global symbol "$status" requires explicit package name at (eval 54) line 15.>
Extra logging is given by setting DEBUG_LEVEL to include
B<LEAVE_MSG>
B<1> opens an extra output stream in the path given by the value of DEBUG_LOG_PATH
B<2> logs messages describing the success or otherwise of the plugin compilation and the result of the plugin run.
An example of such messages are
Fri Apr 22 11:54:21 2005 eval_file: successfully compiled "/usr/local/icinga/libexec/check_bass ".
Fri Apr 22 11:54:21 2005 run_package: "/usr/local/icinga/libexec/check_bass " returning ("0", "BASS Transaction completed Ok.
").
Fri Apr 22 11:55:02 2005 eval_file: successfully compiled "/usr/local/icinga/libexec/check_ad -D production.prod -S".
Fri Apr 22 11:55:02 2005 run_package: "/usr/local/icinga/libexec/check_ad -D foo.dom -S" returning ("0", "Ok. Expected 2 domain controllers [foo1 foo2] for "foo.dom.prod" domain from "1.1.2.3" DNS, found 8 [foo1 foo2 ..]
").
Fri Apr 22 11:55:19 2005 eval_file: successfully compiled "/usr/local/icinga/libexec/check_ldap adonis".
Fri Apr 22 11:55:19 2005 run_package: "/usr/local/icinga/libexec/check_ldap adonis" returning ("0", "Ok. Schema query response DN: dc=ipaustralia,dc=gov,dc=au aci: (target="ldap:///dc=ipaustralia,dc=gov,dc=au")(targetattr!="userPassword")(targetfi
").
Fri Apr 22 11:55:29 2005 eval_file: successfully compiled "/usr/local/icinga/libexec/check_scheduler -H aphrodite -p 7003".
Fri Apr 22 11:55:30 2005 eval_file: successfully compiled "/usr/local/icinga/libexec/check_pams -H aphrodite -p 7003 -R".
Fri Apr 22 11:55:29 2005 run_package: "/usr/local/icinga/libexec/check_scheduler -H aphrodite -p 7003" returning ("0", "Ok. COMSQ last ran 31 seconds ago. System: 0.02s Number of jobs waiting 0 "Detail" system sch_V2_6 14/01/2005 12:22:53 aimali Jobs: COMSQ/PollerManager Fri Apr 22 11:55:00, adhoc pause Fri Apr 22 09:00:00, PAMS/SchedExamDocCheck Thu Apr 21 23:00:00, CFX Cl"
).
Fri Apr 22 11:55:30 2005 run_package: "/usr/local/icinga/libexec/check_pams -H aphrodite -p 7003 -R" returning ("0", "OK PAMS Worst: Test Time 2.61 Failure Ratio 0 [0:5] Statii: BASE OK Oracle (direct) OK COMS Processor OK CCS Name Search (direct) OK Correspondence Manager OK PAMS Tier OK CASEWORK OK Objective (direct) OK Customer Manager OK
").
Fri Apr 22 11:55:45 2005 eval_file: successfully compiled "/usr/local/icinga/libexec/check_coms ".
Fri Apr 22 11:55:45 2005 run_package: "/usr/local/icinga/libexec/check_coms " returning ("0", "COMS Ok. 11 successes 20 minutes ago. 55 minute deltas: (0 0 0 11 0 1 3 4 0 6) or <a href='http://tsitc/cgi-bin/coms_graph_deltas?INT=-2h'>graph</a>
)
.. after all the plugins are compiled, the 'successfully compiled mesages' are replaced by 'skipping compilation'
Fri Apr 22 12:05:10 2005 eval_file: /usr/local/icinga/libexec/check_adds already successfully compiled and file has not changed; skipping compilation.
Fri Apr 22 12:05:11 2005 eval_file: /usr/local/icinga/libexec/check_aub already successfully compiled and file has not changed; skipping compilation
.
Fri Apr 22 12:05:10 2005 run_package: "/usr/local/icinga/libexec/check_adds " returning ("0", "ADDS Transaction completed Ok.
").
Fri Apr 22 12:05:13 2005 eval_file: /usr/local/icinga/libexec/check_eForm already successfully compiled and file has not changed; skipping compilation.
Fri Apr 22 12:05:13 2005 run_package: "/usr/local/icinga/libexec/check_eForm " returning ("0", "eForm Transaction completed Ok.
").
Fri Apr 22 12:05:15 2005 eval_file: /usr/local/icinga/libexec/check_cfx_log already successfully compiled and file has not changed; skipping compilation.
Fri Apr 22 12:05:15 2005 run_package: "/usr/local/icinga/libexec/check_cfx_log -H faxgw1" returning ("0", "Ok. Last write of "//faxgw1/Faxloader$/cfxFaxLoaderClient.log" 0.0 minutes ago. File info (create, access, modify, write times): "Wed Mar 26 17:19:42 2003 Fri Apr 22 12:05:13 2005 Fri Apr 22 12:05:13 2005 Fri Apr 22 12:05:13 2005".
").
Fri Apr 22 12:05:16 2005 eval_file: /usr/local/icinga/libexec/check_cfx_log already successfully compiled and file has not changed; skipping compilation.
Fri Apr 22 12:05:16 2005 run_package: "/usr/local/icinga/libexec/check_cfx_log -H faxgw2" returning ("0", "Ok. Last write of "//faxgw2/Faxloader$/cfxFaxLoaderClient.log" 0.3 minutes ago. File info (create, access, modify, write times): "Wed Mar 26 17:27:24 2003 Fri Apr 22 12:04:55 2005 Fri Apr 22 12:04:55 2005 Fri Apr 22 12:04:55 2005".
").
Fri Apr 22 12:05:17 2005 eval_file: /usr/local/icinga/libexec/check_apps_asearch already successfully compiled and file has not changed; skipping compilation.
Fri Apr 22 12:05:18 2005 eval_file: /usr/local/icinga/libexec/check_aurioness already successfully compiled and file has not changed; skipping compi lation.
Fri Apr 22 12:05:11 2005 run_package: "/usr/local/icinga/libexec/check_aub " returning ("0", "AU-B Transaction completed Ok.
").
If you are lucky enough to have plugins with errors in them,
Fri Apr 22 12:16:01 2005 run_package: "//usr/local/icinga/libexec/eventhandlers/restart_coldfusion OK SOFT" returning ("3", "**ePN "//usr/local/icinga/libexec/eventhandlers/restart_coldfusion": "Can't use string ("") as a subroutine ref while "strict refs" in use at /usr/local/icinga/bin/p1.pl line 291, <DATA> line 218".
B<PLUGIN_DUMP>
B<1> opens an extra output stream in the path given by the value of DEBUG_LOG_PATH.
B<2> logs a listing of the text of any B<faulty> plugin - as transformed by the persistence framework. Note that plugins that compile
are B<never> dumped. This option is only useful for investigating WTF a plugin that runs from the CLI does not run under Icinga with embedded Perl.
Sat Apr 23 19:25:32 2005 eval_file: transformed plugin "check_dummy_plugin" to ==>
1 package Embed::check_5fdummy_5fplugin;
2
3 sub hndlr {
4 @ARGV = @_ ;
5 local $^W = 1 ;
6
7 # <<< START of PLUGIN (first line of plugin is line 8 in the text) >>>
8 #!/usr/bin/perl -w
9
10 use strict
11 # use strict ;
12
13 my @texts = split(/\n/, <<EOTEXTS) ;
14 The Lord is near to the brokenhearted, and saves those who are crushed in spirit. Psalms 34:18 ,
15 Let the words of my mouth, and the meditation of my heart, be acceptable in thy sight, O LORD, my strength, and my redeemer. Psalms 19:14,
16 The LORD is my shepherd; I shall not want. He restoreth my soul: He leadeth me in the paths of righteousness for His name's sake. Psalms 23:1-2,
17 But as many as received him, to them gave he power to become the sons of God, even to them that believe on his name John 1:12,
18 And he said to them all, 'If any man will come after me, let him deny himself, and take up his cross daily, and follow me.' Luke 9:23,
19 EOTEXTS
20 #EOTEXTS
21
22 print $texts[ int(rand($#texts) + 0.5) ], "\n" ;
23 my $rc = int(rand(1) + 0.5) > 0 ? 2 : 0 ;
24 exit $rc ;
25
26
27 # <<< END of PLUGIN >>>
28 }
This listing is logged each time the plugin source file modification time stamp is changed (when the file is
compiled for the first time and then either by correcting a syntax error or touching the file).
Note that the plugin text (lines 8 to 27 inclusive) has been transformed by the persistence framework as described below.
B<CACHE_DUMP>
B<1> opens an extra output stream in the path given by the value of DEBUG_LOG_PATH.
B<2> A dump of the %Cache data structure (showing the plugin file modification time, a hash keyed by the plugin argument string of arrays of parsed
arguments (if non null), the last compilation error, and a code ref to the Perl subroutine corresponding the plugin (this is undef if the plugin failed to compile).
Sat Apr 23 19:24:59 2005 eval_file: after 5 compilations %Cache =>
%Cache = (
'/usr/local/icinga/libexec/check_adds' => [
'100.230810185185',
undef,
'',
sub { "DUMMY" }
],
'check_adds' => [
'3.96288194444444',
undef,
'',
sub { "DUMMY" }
],
'check_atmoss' => [
'3.96288194444444',
undef,
'',
sub { "DUMMY" }
],
'/usr/local/icinga/libexec/check_pams' => [
'1.90859953703704',
{
'-R -I -H asterix -p 7003' => [
'-R',
'-I',
'-H',
'asterix',
'-p',
'7003'
]
},
sub { "DUMMY" }
],
'check_dummy_plugin' => [
'3.96288194444444',
undef,
'',
sub { "DUMMY" }
]
);
..
This dump is produced periodically: each B<$Cache_Dump_Interval> plugin compilations the %Cache data structure is dumped.
=head1 SUBROUTINES
Unless otherwise stated, all subroutines take two (4) arguments :-
=over 4
=item 1 plugin_filename - char * (called from C) or scalar (called from Perl): the path to the plugin.
=item 2 DO_CLEAN - boolean: set if plugin is not to be cached. Defaults to 0.
Setting this flag means that the plugin is compiled each time it is executed. Icinga B<never> sets this flag when the
Icinga is compiled with the configure setting --with-perlcache.
=item 3 (SV *) code ref to the Perl subroutine corresponding to the plugin
This argument is only used by run_package(); it is returned by eval_file().
=item 4 plugin arguments - char ** (called from C) or scalar (called from Perl); the plugin options and arguments
=back
=over 4
=item Embed::Persistent::eval_file( plugin_filename, DO_CLEAN, "", plugin_arguments )
E<10>
Returns B<either> a Perl code reference (an SV containing a hard reference to a subroutine) to the subroutine that
has been produced and compiled by eval_file, B<or> raises an exception (by calling die) and setting the value of B<ERRSV> or
B<$@> (if called from Perl).
eval_file() transforms the plugin to a subroutine in a package, by compiling the string containing the
transformed plugin, and caches the plugin file modification time (to avoid recompiling it),
the parsed plugin arguments. and either the error trapped when the plugin is compiled or a code reference to the
compiled subroutine representing the plugin.
eval_file() caches these values in the cache named B<%Cache>. The plugin file name is the key to an array containing
the values illustrated above.
If the plugin file has not been modified, eval_file returns the cached plugin error B<or> the code ref to the plugin subroutine.
Otherwise, the plugin is compiled into a subroutine in a new package by
=over 4
=item 1 creating a package name from the plugin file name (C<Embed::Persistent::valid_package_name>)
=item 2 turning off subroutine redefinition warnings (C<use subs 'CORE::GLOBAL::exit'>)
=item 3 overriding CORE::GLOBAL::exit from within package main (C<sub CORE::GLOBAL::exit { die "ExitTrap: \$_[0] (yada)"; }>)
This allows the plugin to both call exit without taking down the persistence framework, and to return the exit code to the
Icinga.
=item 4 prepending the plugin text with code to let the plugin function as a subroutine.
The plugin command line arguments are expected to be in @ARGV, so @ARGV is set to the subroutine arguments (@_).
The new subroutine also sets the warning level to trap all warnings that may have been caused by
by the transformation (the -w option to Perl in the text is no longer significant (because the shebang line is not fed to exec()).
=item 5 writing the plugin as the subroutine named B<hndlr> in the new package.
=item 6 returning either a code reference to the subroutine named hndlr in the package named in item 1, OR the
compilation error trapped by eval 'string'. It is the callers responsibility to check either ERRSV (from C) or $@ (from Perl)
and skip run_package() if these values are true.
=back
=item Embed::Persistent::run_package( plugin_filename, DO_CLEAN, (SV *) plugin_hndlr_cr, plugin_argument_string )
E<10>
Returns (plugin_return_code, plugin_output)
run_plugin() actually runs the plugins with the arguments contained in the (space separated string) 4th argument.
=back
=head1 DIFFERENCES FROM PERSISTENT.PL
The example B<persistent.pl> takes no account of
=over 4
=item * Capturing output from the Perl program being run
This framework ties STDOUT to a scalar that stores the result of PRINT or PRINTF.
=item * Running Perl programs in child processes
This is the largest single difference between this framework and the example program persistent.pl. The example uses one
subroutine (eval_file()) to compile and run the program. This is unsuitable for a process like Icinga that
fork a new process to run a plugin. (It is unsuitable because were the child process
to call eval_file() and then the update its copy of %Cache, other child processes would not get the updated %Cache,
and would therefore recompile the plugin).
Instead, eval_file() is split into two: eval_file() and run_package().
Eval_file is called by the Icinga parent process to compile the plugin
and update %Cache. Child processes forked in base/checks.c have the same copy of %Cache and call run_plugin() to check the
last compilation error (from %Cache) and run the plugin if the plugin was error free.
=item * Dealing with Perl programs that call exit
This framework redefines exit() to die emitting a string containing the plugin return code (the first argument of exit).
Since the plugin is run by eval(), B<$@> contains this string from which the return code is extracted.
=item * Providing command line arguments to the Perl program
This framework sets @ARGV in the B<Hndlr> subroutine to the remaining subroutine arguments.
All of these clever ideas came from, AFAIK, Stephen Davies.
=back
=head1 BUGS
=item * MEMORY LEAK
This framework does nothing to prevent the memory leaks mentioned in B<perlembed>, relying on operator intervention.
Probably the best way of doing so is by periodically scheduling
=over 4
=item 1 A check of the memory used by the Icinga process (by running for example the standard Icinga plugin check_procs)
=item 2 Restarting Icinga with the (supplied with Icinga) startup script (restart command).
=back
If you do periodically restart Icinga, make sure that
=over 4
=item 1 plugins all set the PATH environment variable if they need other system binaries (otherwise, if the
init script is excec'd by cron, the PATH will be reset and the plugins will fail - but only when reatsrted by cron).
=item 2 that the group owning the Icinga command pipe is the same as the Icinga group (otherwise, the restarted
Icinga will not be able to read from the command pipe).
=back
Icinga installations using the persistence framework B<must> monitor the memory use of the Icinga process and stop/start it when
the usage is exorbidant (eg, for a site with 400 services on 200 hosts and custom Perl plugins used for about 10% of the
service checks, the Icinga process uses ~80 MB after 20-30 days runningi with Perl 5.005 [Memory usage is
B<much> greater with recent Perls]. It is usually stopped and started at this point).
Note that a HUP signal is B<not> sufficient to deallocate the Perl memory; the Icinga process must be stopped and started. In fact, since HUP
causes Icinga to re-run the Perl interpreter initialisation code, memory use increases significantly. B<Don't use HUP>; use the 'restart' argument
of the Icinga supplied startup script.
There are all sorts of suprising gotchas about the debug logging including
=over 4
=item * No dump of good plugins.
Only plugins that fail to compile (after transformation) are dumped.
=item * Cache dump is periodic
The Cache is only dumped after the user specified number of plugin B<compilations>. If plugins are not compiled, you get
no dump of the cache.
=item * Debug level set at compile time
Icinga must be restarted to change the debug level (use the examples if you have a troublesome plugin;l you may need a debug copy
of Icinga)
=item * Not all Cached fields visible
Always compile one more plugin to ensure that all the fields in the cache are set.
=back
=head1 SEE ALSO
=over 4
=item * perlembed (section on maintaining a persistent interpreter)
=item * examples in the examples/ directory including both C and Perl drivers that run Icinga plugins using p1.pl.
=back
=head1 AUTHOR
Originally by Stephen Davies.
Now maintained by Stanley Hopcroft <[email protected]> who retains responsibility for the 'bad bits'.
=head1 COPYRIGHT
Copyright (c) 2004 Stanley Hopcroft. All rights reserved.
This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
=cut