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
 |