{
$Dist::Zilla::Plugin::AutoPrereqs::VERSION = '4.300039';
}
use Moose;
with(
'Dist::Zilla::Role::PrereqSource',
'Dist::Zilla::Role::FileFinderUser' => {
default_finders => [ ':InstallModules', ':ExecFiles' ],
},
'Dist::Zilla::Role::FileFinderUser' => {
method => 'found_test_files',
finder_arg_names => [ 'test_finder' ],
default_finders => [ ':TestFiles' ],
},
'Dist::Zilla::Role::FileFinderUser' => {
method => 'found_configure_files',
finder_arg_names => [ 'configure_finder' ],
default_finders => [],
},
);
# ABSTRACT: automatically extract prereqs from your modules
use List::AllUtils 'uniq';
use Perl::PrereqScanner 1.016; # don't skip "lib"
use PPI;
use version;
sub mvp_multivalue_args { qw(extra_scanners scanners skips) }
sub mvp_aliases { return { extra_scanner => 'extra_scanners',
scanner => 'scanners', skip => 'skips' } }
has extra_scanners => (
is => 'ro',
isa => 'ArrayRef[Str]',
default => sub { [] },
);
has scanners => (
is => 'ro',
isa => 'ArrayRef[Str]',
predicate => 'has_scanners',
);
has skips => (
is => 'ro',
isa => 'ArrayRef[Str]',
);
sub register_prereqs {
my $self = shift;
my @modules;
my $scanner = Perl::PrereqScanner->new(
($self->has_scanners ? (scanners => $self->scanners) : ()),
extra_scanners => $self->extra_scanners,
);
my @sets = (
[ configure => 'found_configure_files' ], # must come before runtime
[ runtime => 'found_files' ],
[ test => 'found_test_files' ],
);
my %runtime_final;
for my $fileset (@sets) {
my ($phase, $method) = @$fileset;
my $req = CPAN::Meta::Requirements->new;
my $files = $self->$method;
foreach my $file (@$files) {
# parse only perl files
next unless $file->name =~ /\.(?:pm|pl|t|psgi)$/i
|| $file->content =~ /^#!(?:.*)perl(?:$|\s)/;
# RT#76305 skip extra tests produced by ExtraTests plugin
next if $file->name =~ m{^t/(?:author|release)-[^/]*\.t$};
# store module name, to trim it from require list later on
my @this_thing = $file->name;
if ($this_thing[0] =~ /^t/) {
push @this_thing, ($this_thing[0]) x 2;
$this_thing[1] =~ s{^t/}{};
$this_thing[2] =~ s{^t/lib/}{};
} else {
$this_thing[0] =~ s{^lib/}{};
}
@this_thing = uniq @this_thing;
s{\.pm$}{} for @this_thing;
s{/}{::}g for @this_thing;
push @modules, @this_thing;
# parse a file, and merge with existing prereqs
my $file_req = $scanner->scan_string($file->content);
$req->add_requirements($file_req);
}
# remove prereqs shipped with current dist
$req->clear_requirement($_) for @modules;
$req->clear_requirement($_) for qw(Config Errno); # never indexed
# remove prereqs from skiplist
for my $skip (($self->skips || [])->flatten) {
my $re = qr/$skip/;
foreach my $k ($req->required_modules) {
$req->clear_requirement($k) if $k =~ $re;
}
}
# we're done, return what we've found
my %got = %{ $req->as_string_hash };
if ($phase eq 'runtime') {
%runtime_final = %got;
} else {
delete $got{$_} for
grep { exists $got{$_} and $runtime_final{$_} ge $got{$_} }
keys %runtime_final;
}
$self->zilla->register_prereqs({ phase => $phase }, %got);
}
}
__PACKAGE__->meta->make_immutable;
1;
__END__
=pod
=head1 NAME
Dist::Zilla::Plugin::AutoPrereqs - automatically extract prereqs from your modules
=head1 VERSION
version 4.300039
=head1 SYNOPSIS
In your F<dist.ini>:
[AutoPrereqs]
skip = ^Foo|Bar$
skip = ^Other::Dist
=head1 DESCRIPTION
This plugin will extract loosely your distribution prerequisites from
your files using L<Perl::PrereqScanner>.
If some prereqs are not found, you can still add them manually with the
L<Prereqs|Dist::Zilla::Plugin::Prereqs> plugin.
This plugin will skip the modules shipped within your dist.
=head1 ATTRIBUTES
=head2 finder
This is the name of a L<FileFinder|Dist::Zilla::Role::FileFinder>
whose files will be scanned to determine runtime prerequisites. It
may be specified multiple times. The default value is
C<:InstallModules> and C<:ExecFiles>.
=head2 test_finder
Just like C<finder>, but for test-phase prerequisites. The default
value is C<:TestFiles>.
=head2 configure_finder
Just like C<finder>, but for configure-phase prerequisites. There is
no default value; AutoPrereqs will not determine configure-phase
prerequisites unless you set configure_finder.
=head2 extra_scanners
This is an arrayref of scanner names (as expected by Perl::PrereqScanner).
It will be passed as the C<extra_scanners> parameter to Perl::PrereqScanner.
=head2 scanners
This is an arrayref of scanner names (as expected by Perl::PrereqScanner).
If present, it will be passed as the C<scanners> parameter to
Perl::PrereqScanner, which means that it will replace the default list
of scanners.
=head2 skips
This is an arrayref of regular expressions, derived from all the 'skip' lines
in the configuration. Any module names matching any of these regexes will not
be registered as prerequisites.
=head1 SEE ALSO
L<Prereqs|Dist::Zilla::Plugin::Prereqs>, L<Perl::PrereqScanner>.
=head1 CREDITS
This plugin was originally contributed by Jerome Quelin.
=head1 AUTHOR
Ricardo SIGNES <rjbs@cpan.org>
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2013 by Ricardo SIGNES.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=cut