Group
Extension

UI-Various/builder/dod-check.pl

#!/usr/bin/perl
#
# Author, Copyright and License: see end of file

=head1 NAME

dod-check.pl - check Definition of Done

=head1 SYNOPSIS

    dod-check.pl
    dod-check.pl 1 2 3 4 5 8 9  # only run checks 1 to 9
    dod-check.pl -11 -12        # run all checks except 11 and 12
    dod-check.pl 0		# list all steps without running them

=head1 ABSTRACT

This helper script checks if the current snapshot of C<UI::Various>
satisfies the Definition of Done and therefore is ready for distribution.

With one or more numbers given on the command-line it only runs the
specified tests.

=head1 DESCRIPTION

The script checks the following Definition of Done:

=over

=item The build must run without any error.

=item C<L<update-language.pl> --check> must produce an empty report.

=item All sources may only use Perl features of the version required.

=item All tests must run without any error.

=item All tests must have real plans (no C<done_testing>).

=item All regular expressions in the tests must match fully.

=item Test coverage must be 100% except for the list of uncoverable items in
C<confess-uncoverable.lst> and those marked as C<>.

=item All POD tests must run without any error.

=item All POD coverage tests must run without any error.

=item All cross-reference links in the generated HTML pages must be correct.

=item to be continued (TODO)

=back

That the script can be run from anywhere, it knows the relative path to the
C<UI::Various>'s root directory.

=cut

#########################################################################

##################
# load packages: #
##################

use v5.22;
use strictures;
no indirect 'fatal';
no multidimensional;
use warnings 'once';

use Cwd 'abs_path';
use File::Find;
use File::Remove;

#########################
# predefined constants: #
#########################

# path to package and some of its files / paths:
use constant DIR0	=> abs_path(substr($0, 0, rindex($0, '/')));
use constant ROOT_PATH	=> abs_path(substr(DIR0, 0, rindex(DIR0, '/')));
use constant MY_DIR	=> substr(DIR0, rindex(DIR0, '/') + 1);
use constant ID		=> substr(ROOT_PATH, rindex(ROOT_PATH, '/') + 1);
use constant ID_DIR	=> eval{ $_ = ID; tr|-|/|; $_ };
use constant ID_MOD	=> eval{ $_ = ID; s|-|::|g; $_ };
use constant CODE_DIRS => (glob("builder*"),
			   glob("examples*"),
			   glob("script*"),
			   glob("lib*"),
			   glob("t*"));
use constant GREP_CLEAN_CODE =>
    "grep --recursive --extended-regexp --line-number '[ \t]+\$' " .
    join(' ', CODE_DIRS);
use constant TESTS_RE => ROOT_PATH . '/' . MY_DIR . '/tests.re';
use constant UNCOVERABLE => ROOT_PATH . '/' . MY_DIR . '/confess-uncoverable.lst';
use constant GREP_UNCOVERABLE =>
    ('grep', '--recursive', '--fixed-strings', '--after=1', '--include=*.pm',
     'uncoverable', 'lib');
use constant HTML_ROOT => ROOT_PATH . '/blib/libhtml/site/lib';

use constant BOLD_RED	=> "\e[1;31m";
use constant RESET	=> "\e[0m";

my $id = Cwd::getcwd();
$id =~ s|^.*/||;
my $id_dir = $id;
$id_dir =~ tr|-|/|;
my $id_mod = $id;
$id_mod =~ s/-/::/g;

########################
# function prototypes: #
########################

sub _error(@);
sub _warn(@);
sub _info(@);
sub run_and_check($$$@);
sub check_fixmes_todos();
sub check_own_links();
sub check_uncoverable();
sub check_unit_tests();

###############
# run checks: #
###############

my $tests;
BEGIN  {  $tests = 15 + scalar(CODE_DIRS);  }
use Test::More tests => $tests;

my $test_default = 0 < @ARGV && $ARGV[0] =~ m/^\d+$/ ? 0 : 1;
my %test = map { ($_ => $test_default) } 1..$tests;
foreach (@ARGV)
{
    m/^-?\d+$/
	or  die "bad non-numeric argument '$_', use 0 for list of steps\n";
    if ($_ > 0)
    {   $test{$_} = 1;   }
    elsif ($_ < 0)
    {   $test{-$_} = 0;   }
}
my $n = 0;

chdir ROOT_PATH  or  die "couldn't chdir to ", ROOT_PATH, ': ', $!, "\n";

File::Remove::remove(\1, 'blib', 'cover_db', 'pod2htmd.tmp');

# function to skip or run code and - in case of errors - skip rest / exit:
sub skip_or_run($$$)
{
    my ($test, $resolve, $action) = @_;
    my $exit = 0;
 SKIP: {
	if ($test{++$n})
	{
	    unless (&$action)
	    {
		if ($resolve =~ s/^-//)
		{   _warn $resolve;   }
		else
		{   $exit = 1;   skip $resolve, $tests - $n;   }
	    }
	}
	else
	{   skip $test, 1;   }
    }
    0 == $exit  or  exit $exit;
}

skip_or_run('check clean code', '1st check code for cleanliness',
	    sub {
		# return code 1 == no match:
		run_and_check('check clean code', GREP_CLEAN_CODE, 1 << 8);
	    });

skip_or_run('Build.PL', 'check this weird error',
	    sub {
		run_and_check
		    ('Build.PL', 'perl Build.PL', 0,
		     'Created MYMETA.yml and MYMETA.json',
		     "Creating new 'Build' script for '" . ID . "' version .*");
	    });

skip_or_run('build', 'repair build 1st',
	    sub {
		run_and_check('build', './Build', 0, 'Building ' . ID);
	    });

skip_or_run('check EN messages', 'fix EN language source 1st',
	    sub {
		run_and_check('check EN messages',
			      ROOT_PATH . '/' . MY_DIR .
			      '/update-language.pl --check',
			      0);
	    });

my %re_vers = (builder => '(v5\.14\.0 +\| v5\.6\.0|v5\.22\.0 +\| v5\.21\.8)',
	       examples => '((v5\.14\.0|~) +\| (v5\.6\.0|v5.4\.5|~))',
	       lib => '(v5\.14\.0 +\| v5\.[68]\.0)',
	       script => '(v5\.14\.0 +\| v5\.[68]\.0)',
	       t => '(v5\.14\.0 +\| v5\.6\.0|~ +\| (~|v5\.[68]\.0))');
foreach my $dir (CODE_DIRS)
{
    defined $re_vers{$dir}
	or  die "regular expression missing for version check of $dir";
    skip_or_run('check Perl versions in ' . $dir,
		'fix unsupported syntax / features in ' . $dir,
		sub {
		    run_and_check
			('check Perl version in ' . $dir, 'perlver ' . $dir, 0,
			 '^$+',
			 '^\s+-{50,}\s+$',
			 '^.* file +\| explicit +\| syntax +\| external.*',
			 '^\s+\|\s+-{50,}\s+\|$',
			 '^\s+\|\s+.* +\| ' . $re_vers{$dir} . ' +\| +n/a +\|$+',
			 '^\s+\|\s+-{50,}\s+\|$',
			 '^\s+\|\s+Minimum [a-z ]+: v5\.\d+\.\d+\s+\|$+',
			 '^\s+-{50,}\s+$',
			 '^\s*$');
		});
}

open T, '<', TESTS_RE  or  die "can't open " . TESTS_RE . ": $!\n";
my @re_tests = (<T>);
close T  or  die "can't close " . TESTS_RE . ": $!\n";
skip_or_run('tests', 'repair tests 1st',
	    sub {
		run_and_check('tests', './Build test', 0,
			      '# Testing ' . ID_MOD . ' .* Perl v5\..*',
			      @re_tests,
			      'All tests successful\.',
			      'Files=\d+, Tests=\d+, .*',
			      'Result: PASS');
	    });

skip_or_run('check unit tests', 'finish tests 1st',
	    sub {   check_unit_tests();   });

skip_or_run('check Minilla prerequisites', 'add untracked files 1st',
	    sub {
		run_and_check('check Minilla prerequisites',
			      "git status | grep 'Untracked files:'",
			      1 << 8);
	    });

skip_or_run('update Minilla', 'repair Minilla configuration & prerequisites',
	    sub {
		run_and_check
		    ('update Minilla', 'minil test', 0,
		     '[^C].*$+',
		     'Creating working directory: .*',
		     '[^C].*$+',
		     'Created MYMETA.yml and MYMETA.json',
		     "Creating new 'Build' script for '" . ID . "' version .*",
		     '[^B].*$+',
		     'Building ' . ID,
		     '[^t][^/].*$+',
		     @re_tests,
		     'All tests successful\.',
		     'Files=\d+, Tests=\d+, .*',
		     'Result: PASS',
		     'Removing .*');
	    });

skip_or_run('Build.PL again', 'check this even weirder error',
	    sub {
		run_and_check
		    ('Build.PL again', 'perl Build.PL', 0,
		     'Created MYMETA.yml and MYMETA.json',
		     "Creating new 'Build' script for '" . ID . "' version .*");
	    });

skip_or_run('cross-check uncoverable',
	    "don't cheat test coverage (or update " . UNCOVERABLE . ')',
	    sub { check_uncoverable(); });

skip_or_run('test coverage', 'improve test coverage',
	    sub {
		run_and_check
		    ('test coverage', './Build testcover', 0,
		     '# Testing ' . ID_MOD . ' .* Perl v5\..*',
		     @re_tests,
		     'All tests successful.',
		     'Files=\d+, Tests=\d+, .*',
		     'Result: PASS',
		     'Reading database from .*',
		     '^$+',
		     '^-----.*',
		     '^File .*',
		     '^-----.*',
		     '^.* 100\.0$+',
		     '^-----.*',
		     '^$+',
		     '^HTML output written to .*',
		     'done\.');
	    });

skip_or_run('POD tests', 'fix documentation',
	    sub {
		run_and_check
		    ('POD tests', './Build testpod', 0,
		     '1\.\.\d+',
		     'ok \d+ - POD test for blib/lib/' . ID_DIR . '.*$+',
		     'ok \d+ - POD test for blib/script/.*$+');
	    });

skip_or_run('POD coverage', 'improve documentation',
	    sub {
		run_and_check('POD coverage', './Build testpodcoverage', 0,
			      '1\.\.\d+',
			      'ok \d+ - Pod coverage on ' . ID_MOD . '.*$+');
	    });

# Module::Build::Base (0.4231) sets an incomplete (missing vendor / arch)
# and partly wrong (there is an additional "lib" under "libhtml/site")
# podpath in htmlify_pods.  So we ignore all errors here and check our own
# links later.  (Note that this only works correctly with a patched
# Module/Build/Base.pm!)
skip_or_run('build HTML', 'repair generation of HTML pages',
	    sub {
		run_and_check('build HTML', './Build html', 0,
			      'Cannot find .* in podpath: .*$+');
	    });

skip_or_run('check HTML', 'fix broken links',
	    sub {   check_own_links();   });

check_fixmes_todos();


#########################################################################
#########################################################################
########		internal functions following		#########
#########################################################################
#########################################################################

=head1 INTERNAL FUNCTIONS

=cut

#########################################################################

=head2 _error / _warn / _info - print error or warning

    _error(@text);
    _warn(@text);

=head3 parameters:

    @text       main text of error / warning / information

=head3 description:

This function prints the given error, warning or information on the standard
error output using markup specific for each message type.

=cut

# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
sub _error(@)
{   print STDERR BOLD_RED, "*******\t", @_, ' *******', RESET, "\n";   }
sub _warn(@)
{   print STDERR BOLD_RED, @_, RESET, "!\n";   }
sub _info(@)
{   print STDERR map { "\t".$_ } @_;   }

#########################################################################

=head2 run_and_check - run command and check return code / output

    run_and_check($description, $command, $return_code, @re_expected_output);

=head3 example:

    run_and_check('building', './Build', 0, '^Building UI-Various$');

=head3 parameters:

    $description        a (unique!) short text describing the test
    $command            the command that is run (usually C<./Build ...>>)
    $return_code        the expected return code (usually C<0>>)
    @re_expected_output the expected output (combined STDOUT/STDERR),
                        must match whole line

=head3 description:

This function runs the command and checks both return code and its output
against the expected output.  The tests are run as sub-tests of one major
test.

Note that lines in the expected output (C<@re_expected_output>) can be
marked as optional by ending the regular expression with C<$?>.  In addition
they can be marked as repeatedly by ending the regular expression with
C<$+>.

=head3 returns:

1 if no problem could been found, 0 otherwise

=cut

# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
sub run_and_check($$$@)
{
    my ($description, $command, $return_code, @re_expected_output) = @_;
    my $return = 0;
    local $_;

    sub upcoming_match($@)
    {
	my ($line, @re_expected_rest) = @_;
	local $_;
	my $i = 0;
	while (defined $re_expected_rest[$i])
	{
	    $_ = $re_expected_rest[$i];
	    if (s/\$([?+])$//  and  $line !~ m/$_/n)
	    {
		$i++;
		next;
	    }
	    return $line =~ m/$_/n ? $i : undef;
	}
	return undef;
    }

    0 == @re_expected_output  and  @re_expected_output = ('^$');
    subtest $description => sub{
	my @output =
	    # Devel::Cover causes additional warnings for ReadLine/Gnu/XS.pm:
	    grep {!m|^Devel::Cover: .*blib/lib/Term/ReadLine/Gnu/XS.pm |}
	    # TODO: filter out temporary diagnostics of some tests:
	    grep {!m/^# terminal size is \d+x\d+$/}
	    grep {!m/^# UI::Various::Tk has been initialised$/}
	    `$command 2>&1`;
	is($?, $return_code, '"' . $command . '" runs without error');
	unless ($? == $return_code)
	{
	    _error(($? & 0x7f) ? 'SIGNAL ' . $? & 0x7f . ' ' : '',
		   ($? & 0x80) ? 'COREDUMP ' : '',
		   'RC ', ($? >> 8 == 255 ? -1 : $? >> 8));
	    _info @output;
	    _error 'failing command: ', $command;
	    return;
	}
	my $errors = 0;
	my $ie = 0;			# index for expected output
	foreach my $io (0..$#output)	# index for real output
	{
	    $_ = $re_expected_output[$ie];
	    unless (defined $_)
	    {
		fail('running out of expected output of ' . $command);
		$errors++;
		last;
	    }
	    s/\$([?+])$//;
	    my $offset = $1;
	    s/^\^//;
	    $_ = '^(?:' . $_ . ')$';
	    my $message = 'line ' . ($io + 1) . ' OK';
	    my $line = $output[$io];
	    if ($line =~ m/$_/n)
	    {
		like($line, qr/$_/n, $message);
		$ie++ unless defined $offset  and  $offset eq '+';
	    }
	    elsif (defined $offset  and  $offset eq '+')
	    {
		my $next =
		    upcoming_match
		    ($line,
		     @re_expected_output[ $ie + 1  ..  $#re_expected_output ]);
		if (defined $next)
		{
		    $ie += $next + 1;
		    redo;
		}
		like($line, qr/$_/n, $message);
		$line =~ m/$_/n  or  $errors++;
	    }
	    elsif (defined $offset  and  $offset eq '?')
	    {
		$ie++;
		redo;
	    }
	    else
	    {
		like($line, qr/$_/n, $message);
		$line =~ m/$_/n  or  $errors++;
	    }
	}
	$errors == 0  and  $return = 1;
    };
    $return  or  _error 'command failed: ', $command;
    return $return;
}

#########################################################################

=head2 check_fixmes_todos - check (count) FIXMEs and TODOs

    check_fixmes_todos();

=head3 description:

This function checks the Perl sources for FIXMEs and TODOs (uppercase and
whole words!) and counts and reports them.

=head3 returns:

1 if none could be found, 0 otherwise

=cut

# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
sub check_fixmes_todos()
{
    my ($fixmes, $todos) = (0, 0);
    my $my_dir = MY_DIR;
    local $_;

    find(sub {
	     return unless m/\.(p[lm]|t)$/n;
	     return if $File::Find::dir =~ m!(?:^|/)$my_dir!o;
	     open SRC, '<', $File::Find::name  or  die "can't open $_: $!\n";
	     while (<SRC>)
	     {
		 $fixmes++ while s/\bFIXME\b//;
		 $todos++ while s/\bTODO\b//;
	     }
	     close SRC  or  die "can't close $File::Find::name: $!\n";
	 },
	 map { $_ =  ROOT_PATH . '/' . $_ } CODE_DIRS
	);

    if ($fixmes > 0  or  $todos > 0)
    {
	print(STDERR
	      "\n", 'In addition there are ', $fixmes, ' FIXMEs and ', $todos,
	      ' TODOs left, check them in ', ROOT_PATH, " with:\n",
	      'grep --recursive --extended-regexp --include=*.p[lm] ',
	      "--include=*.t '\\<(FIXME|TODO)\\>' ",
	      join(' ', CODE_DIRS), "\n\n");
	return 0;
    }
    return 1;
}

#########################################################################

=head2 check_own_links - check links between own HTML pages

    check_own_links();

=head3 description:

This function checks the HTML pages generated by L<pod2html> (using the
command C<./Build html>).  All links within or between the pages of the
package itself are checked for (approximate) correctness.  (The file path is
still wrong, but we ignore that for now.)

=head3 returns:

1 if no clear error could be found, 0 otherwise

=cut

# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
sub check_own_links()
{
    my $return = 1;
    local $_;

    my $html_root = HTML_ROOT;
    my %name = ();
    my %href = ();
    # step 1: find all links and possible targets:
    find(sub {
	     return unless m/\.html$/;
	     open H, '<', $File::Find::name  or  die "can't open $_: $!\n";
	     (my $name = $File::Find::name) =~ s|^$html_root/||o;
	     while (<H>)
	     {
		 $name{"$name#$1"} = $.
		     while s/<(?:body|dt|h[1-4]) id="([^"]+)">//;
		 while (s/<a href="([^"]+)"//)
		 {
		     my $url = $1;
		     if ($url =~ m/^#/)
			 {
			     $url = "$name$url";
			 }
		     elsif (not $url =~ s|^.*$html_root/||)
			 {
			     next;
			 }
		     defined $href{$url}  or  $href{$url} = [];
		     push @{$href{$url}}, "$name:$.";
		 }
	     }
	     close H  or  die "can't close $File::Find::name: $!\n";
	 },
	 $html_root);
    # step 2: find links not having a corresponding target:
    my ($urls, $errors) = (0, 0);
    foreach my $url (sort keys %href)
    {
	$urls += @{$href{$url}};
	next if defined $name{$url};
	$errors++;
	_error 'bad link';
	_info @{$href{$url}}, '=> ' . $url;
	$return = 0;
    }
    subtest 'check HTML' => sub{
	ok($urls > 150, 'found >> 150 URLs');
	is($errors, 0, 'no errors in ' . $urls . ' URLs');
    };
    return $errors == 0;
}

#########################################################################

=head2 check_uncoverable - check that all uncoverable items are confessed

    check_uncoverable();

=head3 description:

This function checks that all C<uncoverable> code markers in the Perl
sources are correctly confessed in the list of uncoverable code.

=head3 returns:

1 if all tests are planned, 0 otherwise

=cut

# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
sub check_uncoverable()
{
    local $_;
    my %uncoverable = ();
    open UC, '<', UNCOVERABLE
	or  die "can't open ", UNCOVERABLE, ': ', $!, "\n";
    my $location = '';
    while (<UC>)
    {
	s/\r//;
	$location .= $_  unless  m/^--$/;
	if (m/^--$/  or  eof UC)
	{
	    $uncoverable{$location} = 0;   $location = '';
	}
    }
    close UC  or  die "can't close ", UNCOVERABLE, ': ', $!, "\n";
    $location eq ''  or  die 'internal error';

    my $errors = 0;
    subtest 'cross-check uncoverable' => sub{
	open UC, '-|', GREP_UNCOVERABLE
	    or  die "can't run ", GREP_UNCOVERABLE, ': ', $!, "\n";
	while (<UC>)
	{
	    s/\r//;
	    $location .= $_  unless  m/^--$/;
	    if (m/^--$/  or  eof UC)
	    {
		if ($location =~ m/ uncoverable .* # TODO/)
		{}
		elsif ($location =~ m|/confess-uncoverable\.lst|)
		{}
		elsif (defined $uncoverable{$location})
		{
		    ok(1, "found expected location");
		    $uncoverable{$location}++;
		}
		else
		{
		    fail("found unexpected uncoverable location:\n" . $location);
		    $errors++;
		}
		$location = '';
	    }
	}
	foreach (sort keys %uncoverable)
	{
	    1 == $uncoverable{$_}  and  next;
	    if (0 == $uncoverable{$_})
	    {
		fail("didn't found expected uncoverable location:\n" . $_);
		$errors++;
	    }
	    else
	    {
		fail('found expected uncoverable location ' . $uncoverable{$_} .
		     " times:\n" . $_);
		$errors++;
	    }
	}
	close UC  or  die "can't close ", GREP_UNCOVERABLE, ': ', $!, "\n";
    };
    return $errors == 0;
}

#########################################################################

=head2 check_unit_tests - check that all tests are correctly planned

    check_unit_tests();

=head3 description:

This function checks the Perl test sources for missing test plans and
reports them.

=head3 returns:

1 if all tests are planned, 0 otherwise

=cut

# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
sub check_unit_tests()
{
    local $_;
    my $errors = 0;

    subtest 'check unit tests' => sub{
	my @tests = glob(ROOT_PATH . '/t/*.t');
	foreach my $test (@tests)
	{
	    my ($test_plan, $done_testing) = (0, 0);
	    open SRC, '<', $test  or  die "can't open ", $test, ': ', $!, "\n";
	     while (<SRC>)
	     {
		 $test_plan++
		     if  m/^\s*use\s+Test::More\s+tests\s*=>\s*\d+\d*;/
		     or  m/^\s*plan\s+tests\s*=>\s*\d+\d*;/;
		 $done_testing++  if  m/^\s*done_testing\b/;
		 if (m|\bqr/|)
		 {
		     my $tl = $test . ', line ' . $.;
		     unless (m|^\s*my \$re_msg_tail|)
		     {
			 unlike($_, qr|\bqr/[^^]|,
				"regular expression starts with '^' $tl");
		     }
		     unless (m|\$re_msg_tail(_\w+)?/|n)
		     {
			 unlike($_, qr|[^\$]\/[;,]$|,
				"regular expression ends with '\$' $tl");
		     }
		 }
	     }
	    close SRC  or  die "can't close ", $test, ': ', $!, "\n";
	    is($test_plan, 1, 'tests are planned (exactly once) in ' . $test);
	    is($done_testing, 0, 'no "done_testing" in ' . $test);
	    $errors++  unless  $test_plan == 1  and  $done_testing == 0;
	}
    };
    return $errors == 0;
}

#########################################################################
#########################################################################

=head1 SEE ALSO

C<L<UI::Various>>

=head1 LICENSE

Copyright (C) Thomas Dorner.

This library is free software; you can redistribute it and/or modify it
under the same terms as Perl itself.  See LICENSE file for more details.

=head1 AUTHOR

Thomas Dorner E<lt>dorner (at) cpan (dot) orgE<gt>

=cut


Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.