419 lines
12 KiB
Perl
419 lines
12 KiB
Perl
#line 1
|
|
package Module::Install::Makefile;
|
|
|
|
use strict 'vars';
|
|
use ExtUtils::MakeMaker ();
|
|
use Module::Install::Base ();
|
|
use Fcntl qw/:flock :seek/;
|
|
|
|
use vars qw{$VERSION @ISA $ISCORE};
|
|
BEGIN {
|
|
$VERSION = '1.06';
|
|
@ISA = 'Module::Install::Base';
|
|
$ISCORE = 1;
|
|
}
|
|
|
|
sub Makefile { $_[0] }
|
|
|
|
my %seen = ();
|
|
|
|
sub prompt {
|
|
shift;
|
|
|
|
# Infinite loop protection
|
|
my @c = caller();
|
|
if ( ++$seen{"$c[1]|$c[2]|$_[0]"} > 3 ) {
|
|
die "Caught an potential prompt infinite loop ($c[1]|$c[2]|$_[0])";
|
|
}
|
|
|
|
# In automated testing or non-interactive session, always use defaults
|
|
if ( ($ENV{AUTOMATED_TESTING} or -! -t STDIN) and ! $ENV{PERL_MM_USE_DEFAULT} ) {
|
|
local $ENV{PERL_MM_USE_DEFAULT} = 1;
|
|
goto &ExtUtils::MakeMaker::prompt;
|
|
} else {
|
|
goto &ExtUtils::MakeMaker::prompt;
|
|
}
|
|
}
|
|
|
|
# Store a cleaned up version of the MakeMaker version,
|
|
# since we need to behave differently in a variety of
|
|
# ways based on the MM version.
|
|
my $makemaker = eval $ExtUtils::MakeMaker::VERSION;
|
|
|
|
# If we are passed a param, do a "newer than" comparison.
|
|
# Otherwise, just return the MakeMaker version.
|
|
sub makemaker {
|
|
( @_ < 2 or $makemaker >= eval($_[1]) ) ? $makemaker : 0
|
|
}
|
|
|
|
# Ripped from ExtUtils::MakeMaker 6.56, and slightly modified
|
|
# as we only need to know here whether the attribute is an array
|
|
# or a hash or something else (which may or may not be appendable).
|
|
my %makemaker_argtype = (
|
|
C => 'ARRAY',
|
|
CONFIG => 'ARRAY',
|
|
# CONFIGURE => 'CODE', # ignore
|
|
DIR => 'ARRAY',
|
|
DL_FUNCS => 'HASH',
|
|
DL_VARS => 'ARRAY',
|
|
EXCLUDE_EXT => 'ARRAY',
|
|
EXE_FILES => 'ARRAY',
|
|
FUNCLIST => 'ARRAY',
|
|
H => 'ARRAY',
|
|
IMPORTS => 'HASH',
|
|
INCLUDE_EXT => 'ARRAY',
|
|
LIBS => 'ARRAY', # ignore ''
|
|
MAN1PODS => 'HASH',
|
|
MAN3PODS => 'HASH',
|
|
META_ADD => 'HASH',
|
|
META_MERGE => 'HASH',
|
|
PL_FILES => 'HASH',
|
|
PM => 'HASH',
|
|
PMLIBDIRS => 'ARRAY',
|
|
PMLIBPARENTDIRS => 'ARRAY',
|
|
PREREQ_PM => 'HASH',
|
|
CONFIGURE_REQUIRES => 'HASH',
|
|
SKIP => 'ARRAY',
|
|
TYPEMAPS => 'ARRAY',
|
|
XS => 'HASH',
|
|
# VERSION => ['version',''], # ignore
|
|
# _KEEP_AFTER_FLUSH => '',
|
|
|
|
clean => 'HASH',
|
|
depend => 'HASH',
|
|
dist => 'HASH',
|
|
dynamic_lib=> 'HASH',
|
|
linkext => 'HASH',
|
|
macro => 'HASH',
|
|
postamble => 'HASH',
|
|
realclean => 'HASH',
|
|
test => 'HASH',
|
|
tool_autosplit => 'HASH',
|
|
|
|
# special cases where you can use makemaker_append
|
|
CCFLAGS => 'APPENDABLE',
|
|
DEFINE => 'APPENDABLE',
|
|
INC => 'APPENDABLE',
|
|
LDDLFLAGS => 'APPENDABLE',
|
|
LDFROM => 'APPENDABLE',
|
|
);
|
|
|
|
sub makemaker_args {
|
|
my ($self, %new_args) = @_;
|
|
my $args = ( $self->{makemaker_args} ||= {} );
|
|
foreach my $key (keys %new_args) {
|
|
if ($makemaker_argtype{$key}) {
|
|
if ($makemaker_argtype{$key} eq 'ARRAY') {
|
|
$args->{$key} = [] unless defined $args->{$key};
|
|
unless (ref $args->{$key} eq 'ARRAY') {
|
|
$args->{$key} = [$args->{$key}]
|
|
}
|
|
push @{$args->{$key}},
|
|
ref $new_args{$key} eq 'ARRAY'
|
|
? @{$new_args{$key}}
|
|
: $new_args{$key};
|
|
}
|
|
elsif ($makemaker_argtype{$key} eq 'HASH') {
|
|
$args->{$key} = {} unless defined $args->{$key};
|
|
foreach my $skey (keys %{ $new_args{$key} }) {
|
|
$args->{$key}{$skey} = $new_args{$key}{$skey};
|
|
}
|
|
}
|
|
elsif ($makemaker_argtype{$key} eq 'APPENDABLE') {
|
|
$self->makemaker_append($key => $new_args{$key});
|
|
}
|
|
}
|
|
else {
|
|
if (defined $args->{$key}) {
|
|
warn qq{MakeMaker attribute "$key" is overriden; use "makemaker_append" to append values\n};
|
|
}
|
|
$args->{$key} = $new_args{$key};
|
|
}
|
|
}
|
|
return $args;
|
|
}
|
|
|
|
# For mm args that take multiple space-seperated args,
|
|
# append an argument to the current list.
|
|
sub makemaker_append {
|
|
my $self = shift;
|
|
my $name = shift;
|
|
my $args = $self->makemaker_args;
|
|
$args->{$name} = defined $args->{$name}
|
|
? join( ' ', $args->{$name}, @_ )
|
|
: join( ' ', @_ );
|
|
}
|
|
|
|
sub build_subdirs {
|
|
my $self = shift;
|
|
my $subdirs = $self->makemaker_args->{DIR} ||= [];
|
|
for my $subdir (@_) {
|
|
push @$subdirs, $subdir;
|
|
}
|
|
}
|
|
|
|
sub clean_files {
|
|
my $self = shift;
|
|
my $clean = $self->makemaker_args->{clean} ||= {};
|
|
%$clean = (
|
|
%$clean,
|
|
FILES => join ' ', grep { length $_ } ($clean->{FILES} || (), @_),
|
|
);
|
|
}
|
|
|
|
sub realclean_files {
|
|
my $self = shift;
|
|
my $realclean = $self->makemaker_args->{realclean} ||= {};
|
|
%$realclean = (
|
|
%$realclean,
|
|
FILES => join ' ', grep { length $_ } ($realclean->{FILES} || (), @_),
|
|
);
|
|
}
|
|
|
|
sub libs {
|
|
my $self = shift;
|
|
my $libs = ref $_[0] ? shift : [ shift ];
|
|
$self->makemaker_args( LIBS => $libs );
|
|
}
|
|
|
|
sub inc {
|
|
my $self = shift;
|
|
$self->makemaker_args( INC => shift );
|
|
}
|
|
|
|
sub _wanted_t {
|
|
}
|
|
|
|
sub tests_recursive {
|
|
my $self = shift;
|
|
my $dir = shift || 't';
|
|
unless ( -d $dir ) {
|
|
die "tests_recursive dir '$dir' does not exist";
|
|
}
|
|
my %tests = map { $_ => 1 } split / /, ($self->tests || '');
|
|
require File::Find;
|
|
File::Find::find(
|
|
sub { /\.t$/ and -f $_ and $tests{"$File::Find::dir/*.t"} = 1 },
|
|
$dir
|
|
);
|
|
$self->tests( join ' ', sort keys %tests );
|
|
}
|
|
|
|
sub write {
|
|
my $self = shift;
|
|
die "&Makefile->write() takes no arguments\n" if @_;
|
|
|
|
# Check the current Perl version
|
|
my $perl_version = $self->perl_version;
|
|
if ( $perl_version ) {
|
|
eval "use $perl_version; 1"
|
|
or die "ERROR: perl: Version $] is installed, "
|
|
. "but we need version >= $perl_version";
|
|
}
|
|
|
|
# Make sure we have a new enough MakeMaker
|
|
require ExtUtils::MakeMaker;
|
|
|
|
if ( $perl_version and $self->_cmp($perl_version, '5.006') >= 0 ) {
|
|
# This previous attempted to inherit the version of
|
|
# ExtUtils::MakeMaker in use by the module author, but this
|
|
# was found to be untenable as some authors build releases
|
|
# using future dev versions of EU:MM that nobody else has.
|
|
# Instead, #toolchain suggests we use 6.59 which is the most
|
|
# stable version on CPAN at time of writing and is, to quote
|
|
# ribasushi, "not terminally fucked, > and tested enough".
|
|
# TODO: We will now need to maintain this over time to push
|
|
# the version up as new versions are released.
|
|
$self->build_requires( 'ExtUtils::MakeMaker' => 6.59 );
|
|
$self->configure_requires( 'ExtUtils::MakeMaker' => 6.59 );
|
|
} else {
|
|
# Allow legacy-compatibility with 5.005 by depending on the
|
|
# most recent EU:MM that supported 5.005.
|
|
$self->build_requires( 'ExtUtils::MakeMaker' => 6.36 );
|
|
$self->configure_requires( 'ExtUtils::MakeMaker' => 6.36 );
|
|
}
|
|
|
|
# Generate the MakeMaker params
|
|
my $args = $self->makemaker_args;
|
|
$args->{DISTNAME} = $self->name;
|
|
$args->{NAME} = $self->module_name || $self->name;
|
|
$args->{NAME} =~ s/-/::/g;
|
|
$args->{VERSION} = $self->version or die <<'EOT';
|
|
ERROR: Can't determine distribution version. Please specify it
|
|
explicitly via 'version' in Makefile.PL, or set a valid $VERSION
|
|
in a module, and provide its file path via 'version_from' (or
|
|
'all_from' if you prefer) in Makefile.PL.
|
|
EOT
|
|
|
|
if ( $self->tests ) {
|
|
my @tests = split ' ', $self->tests;
|
|
my %seen;
|
|
$args->{test} = {
|
|
TESTS => (join ' ', grep {!$seen{$_}++} @tests),
|
|
};
|
|
} elsif ( $Module::Install::ExtraTests::use_extratests ) {
|
|
# Module::Install::ExtraTests doesn't set $self->tests and does its own tests via harness.
|
|
# So, just ignore our xt tests here.
|
|
} elsif ( -d 'xt' and ($Module::Install::AUTHOR or $ENV{RELEASE_TESTING}) ) {
|
|
$args->{test} = {
|
|
TESTS => join( ' ', map { "$_/*.t" } grep { -d $_ } qw{ t xt } ),
|
|
};
|
|
}
|
|
if ( $] >= 5.005 ) {
|
|
$args->{ABSTRACT} = $self->abstract;
|
|
$args->{AUTHOR} = join ', ', @{$self->author || []};
|
|
}
|
|
if ( $self->makemaker(6.10) ) {
|
|
$args->{NO_META} = 1;
|
|
#$args->{NO_MYMETA} = 1;
|
|
}
|
|
if ( $self->makemaker(6.17) and $self->sign ) {
|
|
$args->{SIGN} = 1;
|
|
}
|
|
unless ( $self->is_admin ) {
|
|
delete $args->{SIGN};
|
|
}
|
|
if ( $self->makemaker(6.31) and $self->license ) {
|
|
$args->{LICENSE} = $self->license;
|
|
}
|
|
|
|
my $prereq = ($args->{PREREQ_PM} ||= {});
|
|
%$prereq = ( %$prereq,
|
|
map { @$_ } # flatten [module => version]
|
|
map { @$_ }
|
|
grep $_,
|
|
($self->requires)
|
|
);
|
|
|
|
# Remove any reference to perl, PREREQ_PM doesn't support it
|
|
delete $args->{PREREQ_PM}->{perl};
|
|
|
|
# Merge both kinds of requires into BUILD_REQUIRES
|
|
my $build_prereq = ($args->{BUILD_REQUIRES} ||= {});
|
|
%$build_prereq = ( %$build_prereq,
|
|
map { @$_ } # flatten [module => version]
|
|
map { @$_ }
|
|
grep $_,
|
|
($self->configure_requires, $self->build_requires)
|
|
);
|
|
|
|
# Remove any reference to perl, BUILD_REQUIRES doesn't support it
|
|
delete $args->{BUILD_REQUIRES}->{perl};
|
|
|
|
# Delete bundled dists from prereq_pm, add it to Makefile DIR
|
|
my $subdirs = ($args->{DIR} || []);
|
|
if ($self->bundles) {
|
|
my %processed;
|
|
foreach my $bundle (@{ $self->bundles }) {
|
|
my ($mod_name, $dist_dir) = @$bundle;
|
|
delete $prereq->{$mod_name};
|
|
$dist_dir = File::Basename::basename($dist_dir); # dir for building this module
|
|
if (not exists $processed{$dist_dir}) {
|
|
if (-d $dist_dir) {
|
|
# List as sub-directory to be processed by make
|
|
push @$subdirs, $dist_dir;
|
|
}
|
|
# Else do nothing: the module is already present on the system
|
|
$processed{$dist_dir} = undef;
|
|
}
|
|
}
|
|
}
|
|
|
|
unless ( $self->makemaker('6.55_03') ) {
|
|
%$prereq = (%$prereq,%$build_prereq);
|
|
delete $args->{BUILD_REQUIRES};
|
|
}
|
|
|
|
if ( my $perl_version = $self->perl_version ) {
|
|
eval "use $perl_version; 1"
|
|
or die "ERROR: perl: Version $] is installed, "
|
|
. "but we need version >= $perl_version";
|
|
|
|
if ( $self->makemaker(6.48) ) {
|
|
$args->{MIN_PERL_VERSION} = $perl_version;
|
|
}
|
|
}
|
|
|
|
if ($self->installdirs) {
|
|
warn qq{old INSTALLDIRS (probably set by makemaker_args) is overriden by installdirs\n} if $args->{INSTALLDIRS};
|
|
$args->{INSTALLDIRS} = $self->installdirs;
|
|
}
|
|
|
|
my %args = map {
|
|
( $_ => $args->{$_} ) } grep {defined($args->{$_} )
|
|
} keys %$args;
|
|
|
|
my $user_preop = delete $args{dist}->{PREOP};
|
|
if ( my $preop = $self->admin->preop($user_preop) ) {
|
|
foreach my $key ( keys %$preop ) {
|
|
$args{dist}->{$key} = $preop->{$key};
|
|
}
|
|
}
|
|
|
|
my $mm = ExtUtils::MakeMaker::WriteMakefile(%args);
|
|
$self->fix_up_makefile($mm->{FIRST_MAKEFILE} || 'Makefile');
|
|
}
|
|
|
|
sub fix_up_makefile {
|
|
my $self = shift;
|
|
my $makefile_name = shift;
|
|
my $top_class = ref($self->_top) || '';
|
|
my $top_version = $self->_top->VERSION || '';
|
|
|
|
my $preamble = $self->preamble
|
|
? "# Preamble by $top_class $top_version\n"
|
|
. $self->preamble
|
|
: '';
|
|
my $postamble = "# Postamble by $top_class $top_version\n"
|
|
. ($self->postamble || '');
|
|
|
|
local *MAKEFILE;
|
|
open MAKEFILE, "+< $makefile_name" or die "fix_up_makefile: Couldn't open $makefile_name: $!";
|
|
eval { flock MAKEFILE, LOCK_EX };
|
|
my $makefile = do { local $/; <MAKEFILE> };
|
|
|
|
$makefile =~ s/\b(test_harness\(\$\(TEST_VERBOSE\), )/$1'inc', /;
|
|
$makefile =~ s/( -I\$\(INST_ARCHLIB\))/ -Iinc$1/g;
|
|
$makefile =~ s/( "-I\$\(INST_LIB\)")/ "-Iinc"$1/g;
|
|
$makefile =~ s/^(FULLPERL = .*)/$1 "-Iinc"/m;
|
|
$makefile =~ s/^(PERL = .*)/$1 "-Iinc"/m;
|
|
|
|
# Module::Install will never be used to build the Core Perl
|
|
# Sometimes PERL_LIB and PERL_ARCHLIB get written anyway, which breaks
|
|
# PREFIX/PERL5LIB, and thus, install_share. Blank them if they exist
|
|
$makefile =~ s/^PERL_LIB = .+/PERL_LIB =/m;
|
|
#$makefile =~ s/^PERL_ARCHLIB = .+/PERL_ARCHLIB =/m;
|
|
|
|
# Perl 5.005 mentions PERL_LIB explicitly, so we have to remove that as well.
|
|
$makefile =~ s/(\"?)-I\$\(PERL_LIB\)\1//g;
|
|
|
|
# XXX - This is currently unused; not sure if it breaks other MM-users
|
|
# $makefile =~ s/^pm_to_blib\s+:\s+/pm_to_blib :: /mg;
|
|
|
|
seek MAKEFILE, 0, SEEK_SET;
|
|
truncate MAKEFILE, 0;
|
|
print MAKEFILE "$preamble$makefile$postamble" or die $!;
|
|
close MAKEFILE or die $!;
|
|
|
|
1;
|
|
}
|
|
|
|
sub preamble {
|
|
my ($self, $text) = @_;
|
|
$self->{preamble} = $text . $self->{preamble} if defined $text;
|
|
$self->{preamble};
|
|
}
|
|
|
|
sub postamble {
|
|
my ($self, $text) = @_;
|
|
$self->{postamble} ||= $self->admin->postamble;
|
|
$self->{postamble} .= $text if defined $text;
|
|
$self->{postamble}
|
|
}
|
|
|
|
1;
|
|
|
|
__END__
|
|
|
|
#line 544
|