5 ##################################################
7 ##################################################
16 use HTTP
::Cookies
::Netscape
;
17 use File
::Temp qw
/ tempfile /;
18 use File
::Copy qw
/ cp /;
23 plan
( skip_all
=> "creating testsites requires root permission" );
25 our $omd_symlink_created = 0;
27 ##################################################
28 # HTML::Lint installed?
29 my $use_html_lint = 0;
35 ##################################################
36 # dont test over a proxy
37 delete $ENV{'HTTP_PROXY'};
38 delete $ENV{'HTTPS_PROXY'};
39 delete $ENV{'FTP_PROXY'};
40 delete $ENV{'http_proxy'};
41 delete $ENV{'https_proxy'};
42 delete $ENV{'ftp_proxy'};
45 ##################################################
49 returns path to omd binary
55 return $omd_bin if defined $omd_bin;
57 $omd_bin = $ENV{'OMD_BIN'} || '/usr/bin/omd';
61 if($omd_bin eq '/usr/bin/omd') {
62 BAIL_OUT
('Broken installation, got /usr/bin/omd but no /omd')
63 } elsif($omd_bin eq 'destdir/opt/omd/versions/default/bin/omd') {
64 symlink(getcwd
()."/destdir/omd", '/omd');
65 $omd_symlink_created = 1;
67 BAIL_OUT
('did not find a valid /omd, please make sure it exists')
72 my $target = readlink('/omd');
73 if($omd_bin eq '/usr/bin/omd') {
74 if($target ne "/opt/omd" && $target ne "opt/omd") {
75 BAIL_OUT
('symlink for /omd already exists but is wrong: should be: /opt/omd but got: '.$target);
78 elsif($omd_bin eq 'destdir/opt/omd/versions/default/bin/omd') {
79 if($target ne getcwd
()."/destdir/omd") {
80 BAIL_OUT
('symlink for /omd already exists but is wrong: should be: '.getcwd
().'/destdir/omd but got: '.$target);
84 BAIL_OUT
('cannot run tests, /omd has to be a symlink to '.getcwd
().'/destdir/omd (or /opt/omd for testing packages) in order to run tests for the source version');
88 -x
$omd_bin or BAIL_OUT
($omd_bin." is required for further tests: $!");
93 ##################################################
97 execute a test command
101 cmd => command line to execute
102 exit => expected exit code
103 like => (list of) regular expressions which have to match stdout
104 errlike => (list of) regular expressions which have to match stderr, default: empty
105 sleep => time to wait after executing the command
111 my($rc, $stderr) = ( -1, '') ;
115 isnt
($test->{'cmd'}, undef, "running cmd: ".$test->{'cmd'}) or $return = 0;
117 my($prg,$arg) = split(/\s+/, $test->{'cmd'}, 2);
118 my $t = Test
::Cmd
->new(prog
=> $prg, workdir
=> '') or die($!);
121 local $SIG{ALRM
} = sub { die "timeout on cmd: ".$test->{'cmd'}."\n" };
122 $t->run(args
=> $arg, stdin
=> $test->{'stdin'});
128 $stderr = $t->stderr;
129 $stderr = TestUtils
::_clean_stderr
($stderr);
134 $test->{'exit'} = 0 unless exists $test->{'exit'};
135 if(defined $test->{'exit'} and $test->{'exit'} != -1) {
136 ok
($rc == $test->{'exit'}, "exit code: ".$rc." == ".$test->{'exit'}) or do { _diag_cmd
($test, $t); $return = 0 };
140 if(defined $test->{'like'}) {
141 for my $expr (ref $test->{'like'} eq 'ARRAY' ? @
{$test->{'like'}} : $test->{'like'} ) {
142 like
($t->stdout, $expr, "stdout like ".$expr) or do { diag
("\ncmd: '".$test->{'cmd'}."' failed\n"); $return = 0 };
147 $test->{'errlike'} = '/^\s*$/' unless exists $test->{'errlike'};
148 if(defined $test->{'errlike'}) {
149 for my $expr (ref $test->{'errlike'} eq 'ARRAY' ? @
{$test->{'errlike'}} : $test->{'errlike'} ) {
150 like
($stderr, $expr, "stderr like ".$expr) or do { diag
("\ncmd: '".$test->{'cmd'}."' failed"); $return = 0 };
154 # sleep after the command?
155 if(defined $test->{'sleep'}) {
156 ok
(sleep($test->{'sleep'}), "slept $test->{'sleep'} seconds") or do { $return = 0 };
160 $test->{'stdout'} = $t->stdout;
161 $test->{'stderr'} = $t->stderr;
162 $test->{'exit'} = $rc;
168 ##################################################
172 verify contents of a file
176 file => file to check
177 like => (list of) regular expressions which have to match
178 unlike => (list of) regular expressions which must not match stderr
188 if(defined $test->{'like'}) {
189 @like = ref $test->{'like'} eq 'ARRAY' ? @
{$test->{'like'}} : $test->{'like'};
192 if(defined $test->{'unlike'}) {
193 @unlike = ref $test->{'unlike'} eq 'ARRAY' ? @
{$test->{'unlike'}} : $test->{'unlike'};
196 ok
(-r
$test->{'file'}, $test->{'file'}." does exist");
199 skip
'file missing', (scalar @like + scalar @unlike) unless -r
$test->{'file'};
202 open my $fh, $test->{'file'} or die "Couldn't open file ".$test->{'file'}.": $!";
207 if(defined $test->{'like'}) {
208 for my $expr (@like) {
209 like
($content, $expr, "content like ".$expr) or $failed++;
214 if(defined $test->{'unlike'}) {
215 for my $expr (@unlike) {
216 unlike
($content, $expr, "output unlike ".$expr) or $failed++;
221 return 1 unless $failed;
226 ##################################################
228 =head2 create_test_site
230 creates a test site and returns the name
233 sub create_test_site
{
234 my $site = "testsite";
235 if(test_command
({ cmd
=> TestUtils
::get_omd_bin
()." create $site" })) {
242 ##################################################
244 =head2 remove_test_site
249 sub remove_test_site
{
251 test_command
({ cmd
=> TestUtils
::get_omd_bin
()." rm $site", stdin
=> "yes\n" });
256 ##################################################
264 url => url to request
265 auth => authentication (realm:user:pass)
266 code => expected response code
267 like => (list of) regular expressions which have to match content
268 unlike => (list of) regular expressions which must not match content
269 skip_html_lint => flag to disable the html lint checking
270 skip_link_check => (list of) regular expressions to skip the link checks for
271 waitfor => wait till regex occurs (max 60sec)
279 my $page = _request
($test);
281 # wait for something?
282 if(defined $test->{'waitfor'}) {
284 my $waitfor = $test->{'waitfor'};
286 while($now < $start + 60) {
287 if($page->{'content'} =~ m/$waitfor/mx) {
288 ok
(1, "content ".$waitfor." found after ".($now - $start)."seconds");
294 $page = _request
($test);
296 fail
("content did not occur within 60 seconds") unless $found;
301 $test->{'code'} = 200 unless exists $test->{'code'};
302 if(defined $test->{'code'}) {
303 is
($page->{'code'}, $test->{'code'}, "response code for ".$test->{'url'}." is: ".$test->{'code'}) or _diag_request
($test, $page);
307 if(defined $test->{'content_type'}) {
308 is
($page->{'content_type'}, $test->{'content_type'}, 'Content-Type is: '.$test->{'content_type'});
312 if(defined $test->{'like'}) {
313 for my $expr (ref $test->{'like'} eq 'ARRAY' ? @
{$test->{'like'}} : $test->{'like'} ) {
314 like
($page->{'content'}, $expr, "content like ".$expr);
318 # not matching output
319 if(defined $test->{'unlike'}) {
320 for my $expr (ref $test->{'unlike'} eq 'ARRAY' ? @
{$test->{'unlike'}} : $test->{'unlike'} ) {
321 unlike
($page->{'content'}, $expr, "content unlike ".$expr) or _diag_request
($test, $page);
327 if($page->{'content_type'} =~ 'text\/html') {
328 unless(defined $test->{'skip_html_lint'} && $test->{'skip_html_lint'} == 1) {
329 if($use_html_lint == 0) {
330 skip
"no HTML::Lint installed", 2;
332 if($page->{'content'} =~ /^\[.*\]$/mx) {
333 skip
"no lint check for json data", 2;
335 my $lint = new HTML
::Lint
;
336 isa_ok
( $lint, "HTML::Lint" );
338 $lint->parse($page->{'content'});
339 my @errors = $lint->errors;
340 @errors = _diag_lint_errors_and_remove_some_exceptions
($lint);
341 is
( scalar @errors, 0, "No errors found in HTML" );
342 $lint->clear_errors();
347 # check for missing images / css or js
348 if($page->{'content_type'} =~ 'text\/html'
349 and (!defined $test->{'skip_html_links'} or $test->{'skip_html_links'} == 0)
351 my $content = $page->{'content'};
352 $content =~ s/<\!\-\-.*?\-\->//gsmx;
353 my @matches = $content =~ m/(src|href)=['|"](.+?)['|"]/gi;
356 for my $match (@matches) {
359 next if $match =~ m/^http/;
360 next if $match =~ m/^mailto:/;
361 next if $match =~ m/^#/;
362 next if $match =~ m/^javascript:/;
363 next if $match =~ m/internal&srv=runtime/;
364 if(defined $test->{'skip_link_check'}) {
366 for my $expr (ref $test->{'skip_link_check'} eq 'ARRAY' ? @
{$test->{'skip_link_check'}} : $test->{'skip_link_check'} ) {
367 if($skip == 0 and $match =~ m/$expr/) {
373 $links_to_check->{$match} = 1;
376 for my $test_url (keys %{$links_to_check}) {
377 $test_url = _get_url
($test->{'url'}, $test_url);
378 our $already_checked;
379 $already_checked = {} unless defined $already_checked;
380 next if defined $already_checked->{$test_url};
381 #diag("checking link: ".$test_url);
382 my $req = _request
({url
=> $test_url, auth
=> $test->{'auth'}});
383 if($req->{'response'}->is_redirect) {
385 while(my $location = $req->{'response'}->{'_headers'}->{'location'}) {
386 if($location !~ m/^(http|\/)/gmx
) { $location = _relative_url
($location, $req->{'response'}->base()->as_string()); }
387 $req= _request
($location);
389 last if $redirects > 10;
392 if($req->{'code'} == 200) {
393 $already_checked->{$test_url} = 1;
396 diag
("got status ".$req->{'code'}." for url: '$test_url'");
398 my $tmp_test = { 'url' => $test_url };
399 _diag_request
($tmp_test, $req);
400 TestUtils
::bail_out_clean
("error in url '$test_url' linked from '".$test->{'url'}."'");
403 is
( $errors, 0, 'All stylesheets, images and javascript exist' );
408 ##################################################
418 return $config->{$key} if defined $config;
420 my $conf_file = "/omd/versions/default/share/omd/distro.info";
421 $config = read_config
($conf_file);
423 return $config->{$key};
427 ##################################################
431 return config from file
435 my $conf_file = shift;
438 open(my $fh, '<', $conf_file) or carp
("cannot open $conf_file: $!");
442 next if $line =~ m/^\s*(#|$)/;
443 $line =~ s/\s*#.*$//;
445 my($key,$value) = split/\s+\+=\s*/,$line,2;
449 ($key,$value) = split/\s+=\s*/,$line,2;
452 $value =~ s/\s+$// if defined $value;
454 $config->{$key} .= " ".$value;
456 $config->{$key} = $value;
465 ##################################################
474 my $timeout = shift || 120;
478 pass
("file: $file does already exist");
481 while($x < $timeout) {
483 pass
("file: $file appeared after $x seconds");
489 fail
("file: $file did not appear within $x seconds");
494 ##################################################
496 =head2 wait_for_content
498 waits for web page content until timeout
502 url => url to request
503 auth => authentication (realm:user:pass)
504 code => expected response code
505 like => (list of) regular expressions which have to match content
509 sub wait_for_content
{
511 my $timeout = shift || 120;
515 while ($x < $timeout) {
516 $req = _request
($test);
517 if($req->{'code'} == 200) {
518 #diag("code:$req->{code} url:$test->{url} auth:$test->{auth}");
520 for my $pattern (@
{$test->{'like'}}) {
521 if ($req->{'content'}!~/$pattern/) {
522 #diag("errors:$errors pattern:$pattern");
527 pass
(sprintf "content: [ %s ] appeared after $x seconds", join(',',@
{$test->{'like'}}));
531 diag
("Error searching for web content:\ncode:$req->{code}\nurl:$test->{url}\nauth:$test->{auth}\ncontent:$req->{content}");
536 fail
(sprintf "content: [ %s ] did not appear within $x seconds", join(',',@
{$test->{'like'}}));
541 ##################################################
543 =head2 prepare_obj_config
545 prepare test object config
548 sub prepare_obj_config
{
553 my $files = join(" ", (ref $src eq 'ARRAY' ? @
{$src} : $src));
554 for my $file (`find $files -type f`) {
557 if(-d
$dst) { $dstfile = $dst.'/'.basename
($file); }
558 cp
($file, $dstfile) or die("copy $file $dstfile failed: $!");
559 test_command
({ cmd
=> "/usr/bin/env sed -i $dstfile -e 's/###SITE###/".$site."/' -e 's|###ROOT###|/omd/sites/".$site."|'" }) if defined $site;
566 ##################################################
568 =head2 bail_out_clean
570 bail out from testing but some minor cleanup before
576 carp
("cleaning up before bailout");
578 my $omd_bin = get_omd_bin
();
579 for my $site (qw
/testsite testsite2 testsite3/) {
580 test_command
({ cmd
=> $omd_bin." rm $site", stdin
=> "yes\n", 'exit' => undef, errlike
=> undef });
587 ##################################################
589 =head2 _diag_lint_errors_and_remove_some_exceptions
591 removes some lint errors we want to ignore
594 sub _diag_lint_errors_and_remove_some_exceptions
{
597 LINT_ERROR
: for my $error ( $lint->errors ) {
598 my $err_str = $error->as_string;
599 for my $exclude_pattern (
600 "<IMG SRC=[^>]*>\ tag\ has\ no\ HEIGHT\ and\ WIDTH\ attributes",
601 "<IMG SRC=[^>]*>\ does\ not\ have\ ALT\ text\ defined",
602 "<input>\ is\ not\ a\ container\ \-\-\ <\/input>\ is\ not\ allowed",
603 "Unknown attribute \"start\" for tag <div>",
604 "Unknown attribute \"end\" for tag <div>",
606 "Unknown\ attribute\ \"placeholder\"\ for\ tag\ <input>",
608 next LINT_ERROR
if($err_str =~ m/$exclude_pattern/i);
610 diag
($error->as_string."\n");
611 push @return, $error;
617 ##################################################
625 url => url to request
626 auth => authentication (realm:user:pass)
636 if(!defined $cookie_jar) {
637 my($fh, $cookie_file) = tempfile
(undef, UNLINK
=> 1);
638 unlink ($cookie_file);
639 $cookie_jar = HTTP
::Cookies
::Netscape
->new(
640 file
=> $cookie_file,
645 my $ua = LWP
::UserAgent
->new;
648 $ua->cookie_jar($cookie_jar);
650 if(defined $data->{'auth'}) {
651 $data->{'url'} =~ m/(http|https):\/\
/(.*?)(\/|:\d
+)/;
654 if(defined $port and $port =~ m/^:(\d+)/) { $port = $1; } else { $port = 80; }
655 my($realm,$user,$pass) = split(/:/, $data->{'auth'}, 3);
656 $ua->credentials($netloc.":".$port, $realm, $user, $pass);
660 if(defined $data->{'post'}) {
661 $response = $ua->post($data->{'url'}, $data->{'post'});
663 $response = $ua->get($data->{'url'});
666 $return->{'response'} = $response;
667 $return->{'code'} = $response->code;
668 $return->{'content'} = $response->decoded_content;
669 $return->{'content_type'} = $response->header('Content-Type');
675 ##################################################
679 returns a absolute url
691 # split original url in host, path and file
692 if($url =~ m/^(http|https):\/\
/([^\/]*)(|\
/|:\d+)(.*?)$/) {
693 my $host = $1."://".$2.$3;
694 $host =~ s/\/$//; # remove last /
695 my $fullpath = $4 || '';
696 $fullpath =~ s/\?.*$//;
697 $fullpath =~ s/^\///;
698 my($path,$file) = ('', '');
699 if($fullpath =~ m/^(.+)\/(.*)$/) {
706 $path =~ s/^\///; # remove first /
708 if($link =~ m/^(http|https):\/\
//) {
711 elsif($link =~ m/^\//) { # absolute link
715 return $host."/".$link;
717 return $host."/".$path."/".$link;
721 TestUtils
::bail_out_clean
("unknown url scheme in _get_url: '".$url."'");
728 ##################################################
732 remove some know errors from stderr
736 my $text = shift || '';
737 $text =~ s/[\w\-]+: Could not reliably determine the server's fully qualified domain name, using .*? for ServerName//g;
738 $text =~ s/\[warn\] module \w+ is already loaded, skipping//g;
739 $text =~ s/Syntax OK//g;
740 $text =~ s/no crontab for \w+//g;
744 ##################################################
748 print diagnostic output for failed commands
754 my $stdout = $cmd->stdout || '';
755 my $stderr = $cmd->stderr || '';
756 diag
("\ncmd: '".$test->{'cmd'}."' failed\n");
757 diag
("stdout: ".$stdout."\n");
758 diag
("stderr: ".$stderr."\n");
760 # check logfiles on apache errors
761 if( $stdout =~ m/Starting dedicated Apache for site (\w+)[\.\ ]*ERROR/
762 or $stdout =~ m/500 Internal Server Error/) {
764 _tail
("apache logs:", "/omd/sites/$site/var/log/apache/error_log") if defined $site;
767 if( $stderr =~ m/User '(\w+)' still logged in or running processes/ ) {
769 diag
("ps: ".`ps -fu $site`) if $site;
774 ##################################################
778 print diagnostic output for failed requests
785 $test->{'url'} =~ m/localhost\/(\w
+)\
//;
787 return unless defined $site;
789 # check logfiles on apache errors
790 if( $page->{'code'} == 500
791 or $page->{'content'} =~ m/Internal Server Error/) {
792 _tail
("apache logs:", "/omd/sites/$site/var/log/apache/error_log");
794 _tail
("thruk logs:", "/omd/sites/$site/var/log/thruk.log") if $test->{'url'} =~ m
/\
/thruk\//;
796 diag
(Dumper
($page->{'response'}));
800 ##################################################
810 return unless defined $file;
813 diag
(`tail -n100 $file`);
815 diag
("cannot read $file: $!");
821 ##################################################
823 =head2 _tail_apache_logs
825 print tail of all apache logs
828 sub _tail_apache_logs
{
829 _tail
("global apache logs:", glob('/var/log/apache*/error*log'));
830 _tail
("global apache logs:", glob('/var/log/httpd*/error*log'));
834 ##################################################
837 if(defined $omd_symlink_created and $omd_symlink_created == 1) {