1#!perl
2
3use strict;
4use warnings;
5
6# This test was generated by Dist::Zilla::Plugin::Test::ReportPrereqs 0.027
7
8use Test::More tests => 1;
9
10use ExtUtils::MakeMaker;
11use File::Spec;
12
13# from $version::LAX
14my $lax_version_re =
15    qr/(?: undef | (?: (?:[0-9]+) (?: \. | (?:\.[0-9]+) (?:_[0-9]+)? )?
16            |
17            (?:\.[0-9]+) (?:_[0-9]+)?
18        ) | (?:
19            v (?:[0-9]+) (?: (?:\.[0-9]+)+ (?:_[0-9]+)? )?
20            |
21            (?:[0-9]+)? (?:\.[0-9]+){2,} (?:_[0-9]+)?
22        )
23    )/x;
24
25# hide optional CPAN::Meta modules from prereq scanner
26# and check if they are available
27my $cpan_meta = "CPAN::Meta";
28my $cpan_meta_pre = "CPAN::Meta::Prereqs";
29my $HAS_CPAN_META = eval "require $cpan_meta; $cpan_meta->VERSION('2.120900')" && eval "require $cpan_meta_pre"; ## no critic
30
31# Verify requirements?
32my $DO_VERIFY_PREREQS = 1;
33
34sub _max {
35    my $max = shift;
36    $max = ( $_ > $max ) ? $_ : $max for @_;
37    return $max;
38}
39
40sub _merge_prereqs {
41    my ($collector, $prereqs) = @_;
42
43    # CPAN::Meta::Prereqs object
44    if (ref $collector eq $cpan_meta_pre) {
45        return $collector->with_merged_prereqs(
46            CPAN::Meta::Prereqs->new( $prereqs )
47        );
48    }
49
50    # Raw hashrefs
51    for my $phase ( keys %$prereqs ) {
52        for my $type ( keys %{ $prereqs->{$phase} } ) {
53            for my $module ( keys %{ $prereqs->{$phase}{$type} } ) {
54                $collector->{$phase}{$type}{$module} = $prereqs->{$phase}{$type}{$module};
55            }
56        }
57    }
58
59    return $collector;
60}
61
62my @include = qw(
63
64);
65
66my @exclude = qw(
67
68);
69
70# Add static prereqs to the included modules list
71my $static_prereqs = do './t/00-report-prereqs.dd';
72
73# Merge all prereqs (either with ::Prereqs or a hashref)
74my $full_prereqs = _merge_prereqs(
75    ( $HAS_CPAN_META ? $cpan_meta_pre->new : {} ),
76    $static_prereqs
77);
78
79# Add dynamic prereqs to the included modules list (if we can)
80my ($source) = grep { -f } 'MYMETA.json', 'MYMETA.yml';
81my $cpan_meta_error;
82if ( $source && $HAS_CPAN_META
83    && (my $meta = eval { CPAN::Meta->load_file($source) } )
84) {
85    $full_prereqs = _merge_prereqs($full_prereqs, $meta->prereqs);
86}
87else {
88    $cpan_meta_error = $@;    # capture error from CPAN::Meta->load_file($source)
89    $source = 'static metadata';
90}
91
92my @full_reports;
93my @dep_errors;
94my $req_hash = $HAS_CPAN_META ? $full_prereqs->as_string_hash : $full_prereqs;
95
96# Add static includes into a fake section
97for my $mod (@include) {
98    $req_hash->{other}{modules}{$mod} = 0;
99}
100
101for my $phase ( qw(configure build test runtime develop other) ) {
102    next unless $req_hash->{$phase};
103    next if ($phase eq 'develop' and not $ENV{AUTHOR_TESTING});
104
105    for my $type ( qw(requires recommends suggests conflicts modules) ) {
106        next unless $req_hash->{$phase}{$type};
107
108        my $title = ucfirst($phase).' '.ucfirst($type);
109        my @reports = [qw/Module Want Have/];
110
111        for my $mod ( sort keys %{ $req_hash->{$phase}{$type} } ) {
112            next if $mod eq 'perl';
113            next if grep { $_ eq $mod } @exclude;
114
115            my $file = $mod;
116            $file =~ s{::}{/}g;
117            $file .= ".pm";
118            my ($prefix) = grep { -e File::Spec->catfile($_, $file) } @INC;
119
120            my $want = $req_hash->{$phase}{$type}{$mod};
121            $want = "undef" unless defined $want;
122            $want = "any" if !$want && $want == 0;
123
124            my $req_string = $want eq 'any' ? 'any version required' : "version '$want' required";
125
126            if ($prefix) {
127                my $have = MM->parse_version( File::Spec->catfile($prefix, $file) );
128                $have = "undef" unless defined $have;
129                push @reports, [$mod, $want, $have];
130
131                if ( $DO_VERIFY_PREREQS && $HAS_CPAN_META && $type eq 'requires' ) {
132                    if ( $have !~ /\A$lax_version_re\z/ ) {
133                        push @dep_errors, "$mod version '$have' cannot be parsed ($req_string)";
134                    }
135                    elsif ( ! $full_prereqs->requirements_for( $phase, $type )->accepts_module( $mod => $have ) ) {
136                        push @dep_errors, "$mod version '$have' is not in required range '$want'";
137                    }
138                }
139            }
140            else {
141                push @reports, [$mod, $want, "missing"];
142
143                if ( $DO_VERIFY_PREREQS && $type eq 'requires' ) {
144                    push @dep_errors, "$mod is not installed ($req_string)";
145                }
146            }
147        }
148
149        if ( @reports ) {
150            push @full_reports, "=== $title ===\n\n";
151
152            my $ml = _max( map { length $_->[0] } @reports );
153            my $wl = _max( map { length $_->[1] } @reports );
154            my $hl = _max( map { length $_->[2] } @reports );
155
156            if ($type eq 'modules') {
157                splice @reports, 1, 0, ["-" x $ml, "", "-" x $hl];
158                push @full_reports, map { sprintf("    %*s %*s\n", -$ml, $_->[0], $hl, $_->[2]) } @reports;
159            }
160            else {
161                splice @reports, 1, 0, ["-" x $ml, "-" x $wl, "-" x $hl];
162                push @full_reports, map { sprintf("    %*s %*s %*s\n", -$ml, $_->[0], $wl, $_->[1], $hl, $_->[2]) } @reports;
163            }
164
165            push @full_reports, "\n";
166        }
167    }
168}
169
170if ( @full_reports ) {
171    diag "\nVersions for all modules listed in $source (including optional ones):\n\n", @full_reports;
172}
173
174if ( $cpan_meta_error || @dep_errors ) {
175    diag "\n*** WARNING WARNING WARNING WARNING WARNING WARNING WARNING WARNING ***\n";
176}
177
178if ( $cpan_meta_error ) {
179    my ($orig_source) = grep { -f } 'MYMETA.json', 'MYMETA.yml';
180    diag "\nCPAN::Meta->load_file('$orig_source') failed with: $cpan_meta_error\n";
181}
182
183if ( @dep_errors ) {
184    diag join("\n",
185        "\nThe following REQUIRED prerequisites were not satisfied:\n",
186        @dep_errors,
187        "\n"
188    );
189}
190
191pass;
192
193# vim: ts=4 sts=4 sw=4 et:
194