package Test::Pod; use strict; =head1 NAME Test::Pod - check for POD errors in files =head1 VERSION Version 1.26 =cut use vars qw( $VERSION ); $VERSION = '1.26'; =head1 SYNOPSIS C lets you check the validity of a POD file, and report its results in standard C fashion. use Test::Pod tests => $num_tests; pod_file_ok( $file, "Valid POD file" ); Module authors can include the following in a F file and have C automatically find and check all POD files in a module distribution: use Test::More; eval "use Test::Pod 1.00"; plan skip_all => "Test::Pod 1.00 required for testing POD" if $@; all_pod_files_ok(); You can also specify a list of files to check, using the C function supplied: use strict; use Test::More; eval "use Test::Pod 1.00"; plan skip_all => "Test::Pod 1.00 required for testing POD" if $@; my @poddirs = qw( blib script ); all_pod_files_ok( all_pod_files( @poddirs ) ); Or even (if you're running under L): use strict; use Test::More; eval "use Test::Pod 1.00"; plan skip_all => "Test::Pod 1.00 required for testing POD" if $@; my @poddirs = qw( blib script ); use File::Spec::Functions qw( catdir updir ); all_pod_files_ok( all_pod_files( map { catdir updir, $_ } @poddirs ) ); =head1 DESCRIPTION Check POD files for errors or warnings in a test file, using C to do the heavy lifting. =cut use 5.004; use Pod::Simple; use Test::Builder; use File::Spec; my $Test = Test::Builder->new; sub import { my $self = shift; my $caller = caller; for my $func ( qw( pod_file_ok all_pod_files all_pod_files_ok ) ) { no strict 'refs'; *{$caller."::".$func} = \&$func; } $Test->exported_to($caller); $Test->plan(@_); } =head1 FUNCTIONS =head2 pod_file_ok( FILENAME[, TESTNAME ] ) C will okay the test if the POD parses correctly. Certain conditions are not reported yet, such as a file with no pod in it at all. When it fails, C will show any pod checking errors as diagnostics. The optional second argument TESTNAME is the name of the test. If it is omitted, C chooses a default test name "POD test for FILENAME". =cut sub pod_file_ok { my $file = shift; my $name = @_ ? shift : "POD test for $file"; if ( !-f $file ) { $Test->ok( 0, $name ); $Test->diag( "$file does not exist" ); return; } my $checker = Pod::Simple->new; $checker->output_string( \my $trash ); # Ignore any output $checker->parse_file( $file ); my $ok = !$checker->any_errata_seen; $Test->ok( $ok, $name ); if ( !$ok ) { my $lines = $checker->{errata}; for my $line ( sort { $a<=>$b } keys %$lines ) { my $errors = $lines->{$line}; $Test->diag( "$file ($line): $_" ) for @$errors; } } return $ok; } # pod_file_ok =head2 all_pod_files_ok( [@files/@directories] ) Checks all the files in C<@files> for valid POD. It runs L on each file/directory, and calls the C function for you (one test for each function), so you can't have already called C. If C<@files> is empty or not passed, the function finds all POD files in the F directory if it exists, or the F directory if not. A POD file is one that ends with F<.pod>, F<.pl> and F<.pm>, or any file where the first line looks like a shebang line. If you're testing a module, just make a F: use Test::More; eval "use Test::Pod 1.00"; plan skip_all => "Test::Pod 1.00 required for testing POD" if $@; all_pod_files_ok(); Returns true if all pod files are ok, or false if any fail. =cut sub all_pod_files_ok { my @files = @_ ? @_ : all_pod_files(); $Test->plan( tests => scalar @files ); my $ok = 1; foreach my $file ( @files ) { pod_file_ok( $file, $file ) or undef $ok; } return $ok; } =head2 all_pod_files( [@dirs] ) Returns a list of all the Perl files in I<$dir> and in directories below. If no directories are passed, it defaults to F if F exists, or else F if not. Skips any files in CVS or .svn directories. A Perl file is: =over 4 =item * Any file that ends in F<.PL>, F<.pl>, F<.pm>, F<.pod> or F<.t>. =item * Any file that has a first line with a shebang and "perl" on it. =back The order of the files returned is machine-dependent. If you want them sorted, you'll have to sort them yourself. =cut sub all_pod_files { my @queue = @_ ? @_ : _starting_points(); my @pod = (); while ( @queue ) { my $file = shift @queue; if ( -d $file ) { local *DH; opendir DH, $file or next; my @newfiles = readdir DH; closedir DH; @newfiles = File::Spec->no_upwards( @newfiles ); @newfiles = grep { $_ ne "CVS" && $_ ne ".svn" } @newfiles; foreach my $newfile (@newfiles) { my $filename = File::Spec->catfile( $file, $newfile ); if ( -f $filename ) { push @queue, $filename; } else { push @queue, File::Spec->catdir( $file, $newfile ); } } } if ( -f $file ) { push @pod, $file if _is_perl( $file ); } } # while return @pod; } sub _starting_points { return 'blib' if -e 'blib'; return 'lib'; } sub _is_perl { my $file = shift; return 1 if $file =~ /\.PL$/; return 1 if $file =~ /\.p(l|m|od)$/; return 1 if $file =~ /\.t$/; local *FH; open FH, $file or return; my $first = ; close FH; return 1 if defined $first && ($first =~ /^#!.*perl/); return; } =head1 TODO STUFF TO DO Note the changes that are being made. Note that you no longer can test for "no pod". =head1 AUTHOR Currently maintained by Andy Lester, C<< >>. Originally by brian d foy. =head1 ACKNOWLEDGEMENTS Thanks to David Wheeler and Peter Edwards for contributions and to C for the original code. =head1 COPYRIGHT Copyright 2006, Andy Lester, All Rights Reserved. You may use, modify, and distribute this package under the same terms as Perl itself. =cut 1;