App-MechaCPAN
view release on metacpan or search on metacpan
lib/App/MechaCPAN/Perl.pm view on Meta::CPAN
package App::MechaCPAN::Perl;
use v5.14;
use autodie;
use Config;
use FindBin;
use File::Spec;
use App::MechaCPAN qw/:go/;
our @args = (
'threads!',
'jobs=i',
'skip-tests!',
'skip-local!',
'skip-lib!',
'smart-tests!',
'devel!',
'shared-lib!',
'source-only!',
);
my $perl5_ver_re = qr/v? 5 [.] (\d{1,2}) (?: [.] (\d{1,2}) )?/xms;
my $perl5_re = qr/^ $perl5_ver_re $/xms;
our $JOBS = 2; # The number of jobs to run with make
sub go
{
my $class = shift;
my $opts = shift;
my $src = shift;
my @argv = shift;
if ( $^O eq 'MSWin32' )
{
info 'Cannot build perl on Win32';
return 0;
}
my $orig_dir = &get_project_dir;
my $dest_dir = &dest_dir;
my @dest_dir = File::Spec->splitdir("$dest_dir");
my $dest_len = $#dest_dir;
my $perl_dir = "$dest_dir/perl";
my $pv_ver; # Version in .perl-version file
# source-only currently defaults to true, downloading binaries currently
# will only happen if no-source-only is given
$opts->{'source-only'} //= 0;
# Attempt to find the perl version if none was given
if ( -f '.perl-version' )
{
open my $pvFH, '<', '.perl-version';
$pv_ver = do { local $/; <$pvFH> };
$pv_ver =~ s/\s+//xmsg;
if ( $pv_ver !~ $perl5_re )
{
info "$pv_ver in .perl-version doesn't look like a perl5 version";
undef $pv_ver;
}
}
my ( $src_tz, $version ) = _get_targz( $src // $pv_ver, $opts );
my $bin_tz;
if ( ref $src_tz eq 'ARRAY' )
{
( $bin_tz, $src_tz ) = @$src_tz;
}
# If _get_targz couldn't find a version, guess based on the file
lib/App/MechaCPAN/Perl.pm view on Meta::CPAN
_build_configure( $perl_dir, $opts ),
q[-Accflags=-DAPPLLIB_EXP=\"] . join( ":", @otherlib ) . q[\"],
qq[-A'eval:scriptdir=$perl_dir/bin'],
);
local %ENV = %ENV;
delete @ENV{qw(PERL5LIB PERL5OPT)};
# Make sure no tomfoolery is happening with perl, like plenv shims
$ENV{PATH} = $Config{binexp} . ":$ENV{PATH}";
eval {
require Devel::PatchPerl;
info $verstr, "Patching $verstr";
Devel::PatchPerl->patch_source();
};
info $verstr, "Configuring $verstr";
_run_configure(@config);
info $verstr, "Building $verstr";
_run_make();
my $skip_tests = $opts->{'skip-tests'};
if ( !$skip_tests && $opts->{'smart-tests'} )
{
$skip_tests = $pv_ver eq $version;
}
if ( !$skip_tests )
{
info $verstr, "Testing $verstr";
_run_make('test_harness');
}
info $verstr, "Installing $verstr";
_run_make('install');
success "Installed $verstr";
chdir $orig_dir;
&restart_script();
return 0;
}
# These are split out mostly so we can control testing
sub _build_configure
{
my $perl_dir = shift;
my $opts = shift;
my @config = (
q[-des],
qq[-Dprefix=$perl_dir],
);
if ( $opts->{threads} )
{
push @config, '-Dusethreads';
}
if ( $opts->{'shared-lib'} )
{
push @config, '-Duseshrplib';
}
if ( $opts->{devel} )
{
push @config, '-Dusedevel';
}
return @config;
}
sub _run_configure
{
my @config = @_;
run qw[sh Configure], @config;
}
sub _run_make
{
my @cmd = @_;
state $make = $Config{make};
state $can_jobs;
if ( !defined $can_jobs )
{
$can_jobs = '';
my $make_help
= eval { run( $make, '-h' ) } // eval { run( $make, '--help' ) } // '';
if ( $make_help =~ m/^\s*-j\s+/xms )
{
$can_jobs = '-j';
}
elsif ( $make_help =~ m/^\s*--jobs\s+/xms )
{
$can_jobs = '--jobs';
}
}
my @jobs_cmd;
if ( $JOBS > 1 && $can_jobs )
{
@jobs_cmd = ( $can_jobs, $JOBS );
}
# Give perl more time to be silent during the make process than normal
local $App::MechaCPAN::TIMEOUT = $App::MechaCPAN::TIMEOUT * 10;
run $make, @jobs_cmd, @cmd;
}
sub slugline
{
my $perl = shift || File::Spec->canonpath($^X);
my $version = shift || '';
my $use_threads = shift;
my $script = <<'EOD';
use strict;
use Config;
use File::Basename qw/basename/;
use ExtUtils::Liblist;
my $version = $ARGV[0] || $^V;
my $usethreads = defined $ARGV[1] ? $ARGV[1] : 0;
my $libcname = 'unknown';
my $libcver = 'ukn';
my $archname = ( split '-', $Config{archname} )[0];
my $osname = $Config{osname};
my $threads = $usethreads ? 'threads-' : '';
if ( $Config{gnulibc_version} )
{
$libcname = 'glibc';
$libcver = $Config{gnulibc_version};
}
else
{
my $libc_re = qr/libc (\W|$)/xms;
my ($libc_basename) = grep {m/$libc_re/} split( / /, $Config{libsfiles} );
my ($libc_path) = grep {m/$libc_basename/} split / /, $Config{libsfound};
my $libc_so = $libc_path;
$libc_so =~ s/[.]a([\d.]*)$/.so$1/;
if ( -x $libc_so )
{
my $help = `$libc_so 2>&1`;
if ( $help =~ m/^ musl \s libc .* Version \s* ([0-9.]+)/xms )
{
$libcname = 'musl';
$libcver = $1;
}
}
}
# Add the name of each lib found with Liblist
my @short_libs;
my @libs;
my %rename = (
# c, m and dl are assumed to be part of libc, which we already handle
'-lc' => '', '-lm' => '',
'-ldl' => '', '-lld' => '',
'-lpthread' => '',
# Commonly found libraries can be shortened
'-lsocket' => 's', '-linet' => 'i',
'-lnsl' => 'n', '-lcrypt' => 'y',
'-lutil' => 'u', '-lposix' => 'p',
'-lgdbm' => 'gd', '-ldbm' => 'd',
);
foreach my $libs ( sort split ' ', $Config{libs} )
{
my @ext = ExtUtils::Liblist->ext( "$libs", 0, 1 );
my $n = $libs;
next
unless $ext[4]->[0];
my $ver = basename $ext[4]->[0];
# Static libraries (.a) don't matter
next
if $ver =~ m/[.]a$/xms;
$n =~ s/^-l/lib/;
$ver =~ s/^.*$n([.]so[.]?)?//;
$ver =~ s/^(\d+([.]\d+)?)([.]\d+)?$/$1/;
$ver ||= '.';
if ( exists $rename{$libs} )
{
if ( $rename{$libs} )
{
push @short_libs, "$rename{$libs}$ver";
}
next;
}
push @libs, "$n$ver";
}
my $libsver = join('', @short_libs, @libs);
if ($libsver)
{
$libsver = "-$libsver";
}
print "perl-$version-$archname-$osname-$threads$libcname-$libcver$libsver";
EOD
my $script_file = humane_tmpfile;
$script_file->print($script);
$script_file->close;
my $slugline = run(
$perl,
"$script_file",
$version,
( defined $use_threads ? ($use_threads) : () )
);
chomp $slugline;
return $slugline;
}
sub _check_perl_binary
{
my $perl_bin = shift;
# We include POSIX, that's a good litmus that libc is not completely broken
# and we use crypt to test that the crypt lib is loadable. This is simply
# a bare minimum check and it may change in the future
no warnings 'qw';
my @check = qw/-MPOSIX -e crypt('00','test')/;
run "$perl_bin", @check;
return 1;
}
sub build_reusable
{
my $version = shift;
my $perl_dir = shift;
my $src_tz = shift;
my $opts = shift;
# Determine what to compress it with
my $compress
= eval { run(qw/xz --version/); 'xz' }
// eval { run(qw/bzip2 --version/); 'bzip2' }
// eval { run(qw/gzip --version/); 'gzip' }
// die 'Cannot find anything to compress with';
# Make sure we can call tar before we get too far
die 'Cannot find tar to create an archive'
if !( eval { run(qw/tar --version/) } );
$perl_dir = humane_tmpdir("perl-$version");
my $verstr = "perl $version";
info $verstr, "Fetching $verstr";
my $src_dir = inflate_archive($src_tz);
my @src_dirs = File::Spec->splitdir("$src_dir");
chdir $src_dir;
if ( !-e 'Configure' )
{
my @files = glob('*');
if ( @files != 1 )
{
die qq{Could not find perl to configure.}
. qq{Inflated to "$src_dir" extracted from $src_tz};
}
chdir $files[0];
}
my $local_dir = File::Spec->catdir(qw/... .. .. lib perl5/);
my $lib_dir = File::Spec->catdir(qw/... .. .. .. lib/);
my @otherlib = (
!$opts->{'skip-local'} ? $local_dir : (),
!$opts->{'skip-lib'} ? $lib_dir : (),
);
my @config = (
_build_configure( $perl_dir, $opts ),
q[-Accflags=-DAPPLLIB_EXP=\"] . join( ":", @otherlib ) . q[\"],
q{-Dstartperl='#!/usr/bin/env\ perl'},
q{-Dperlpath='/usr/bin/env\ perl'},
qq{-Dinstallprefix=/v$version},
qq{-Dprefix=/v$version},
q{-Dman1dir=.../../man/man1},
q{-Dman3dir=.../../man/man3},
q{-Duserelocatableinc},
);
if ( $opts->{threads} )
{
push @config, '-Dusethreads';
}
local %ENV = %ENV;
delete @ENV{qw(PERL5LIB PERL5OPT)};
$ENV{DESTDIR} = $perl_dir;
# Make sure no tomfoolery is happening with perl, like plenv shims
$ENV{PATH} = $Config{binexp} . ":$ENV{PATH}";
eval {
require Devel::PatchPerl;
info $verstr, "Patching $verstr";
Devel::PatchPerl->patch_source();
};
info $verstr, "Configuring $verstr";
_run_configure(@config);
info $verstr, "Building $verstr";
_run_make();
my $skip_tests = $opts->{'skip-tests'} // $opts->{'smart-tests'};
if ( !$skip_tests )
{
info $verstr, "Testing $verstr";
_run_make('test_harness');
}
info $verstr, "Installing $verstr";
_run_make('install');
# Verify that the relocatable bits worked
local $@;
eval { _check_perl_binary("$perl_dir/v$version/bin/perl") };
my $error = $@;
if ($error)
{
die "The built relocatable binary appears broken: $error\n";
}
my $slugline = slugline("$perl_dir/v$version/bin/perl", undef, $opts->{threads});
my $orig_dir = &get_project_dir;
my $output = "$slugline.tar.$compress";
chdir $perl_dir;
run("tar cf - v$version/ | $compress > $orig_dir/$output");
success $verstr, "Created $verstr: $output";
return 0;
}
sub _handle_bin_tz
{
my $bin_tz = shift;
my $version = shift;
info "Binary URL: $bin_tz";
local $@;
my $src_dir = eval { inflate_archive($bin_tz) };
if ( !$src_dir )
{
info "Could not find binary $version: $@";
return;
}
my @src_dirs = File::Spec->splitdir("$src_dir");
chdir $src_dir;
if ( -e -x File::Spec->catdir( @src_dirs, qw/bin perl/ ) )
{
local $@;
my $success
= eval { _install_binary( File::Spec->catdir(@src_dirs), $version ) };
my $error = $@;
if ($error)
{
info "Binary in $bin_tz does not appear to be usable: $error";
}
return $success;
}
logmsg "$bin_tz did not have a perl binary";
return 1;
}
sub _install_binary
{
my $src_dir = shift;
my $version = shift;
my @src_dirs = File::Spec->splitdir("$src_dir");
my $dest_dir = &dest_dir;
my $perl_dir = File::Spec->catdir( $dest_dir, 'perl' );
info $version, "Installing $version";
use File::Copy qw/copy move/;
use File::Path qw/make_path/;
use Fatal qw/copy move/;
chdir $dest_dir;
my $output = eval { run "$src_dir/bin/perl", '-e', 'print $^V' };
chomp $output;
if ( $output ne "v$version" )
{
die qq{Binary versions mismatch expectations: }
. qq{"$output" (found) ne "$version" (expected)};
}
# Attempt to run something more rigorous
local $@;
eval { _check_perl_binary("$src_dir/bin/perl") };
my $error = $@;
if ($error)
{
die "Binary does not appear to be usable: $error";
}
make_path($perl_dir);
move( $src_dir, $perl_dir );
success "Installed binary $version";
return 1;
}
our $source_mirror = 'https://www.cpan.org/src/5.0';
our $binary_mirror = 'https://dnld.mechacpan.us/dist';
sub _dnld_url
{
my $version = shift;
my $minor = shift;
return "$source_mirror/perl-5.$version.$minor.tar.gz";
}
sub _bin_url
{
my $version = shift;
my $minor = shift;
my $opts = shift;
my $fullver = "v5.$version.$minor";
my $slugline = slugline( undef, $fullver, $opts->{threads} );
return "$binary_mirror/$slugline.tar.xz";
}
sub _get_targz
{
my $src = shift;
my $opts = shift;
# If there's no src, find the newest version.
if ( !defined $src )
{
# Do a terrible job of guessing what the current version is
use Time::localtime;
my $year = localtime->year() + 1900;
# 5.12 was released in 2010, and approximatly every May, a new even
# version was released
my $major = ( $year - 2010 ) * 2 + ( localtime->mon < 4 ? 10 : 12 );
# Verify our guess
{
my $dnld = _dnld_url( $major, 0 ) . ".md5.txt";
my $contents = '';
my $where = eval { fetch_file( $dnld => \$contents ) };
if ( !defined $where && $major > 12 )
{
$major -= 2;
redo;
}
}
$src = "5.$major";
}
# file
if ( -e $src )
{
return ( rel_start_to_abs($src), '' );
}
my $url;
# URL
if ( $src =~ url_re )
{
return ( $src, '' );
}
# CPAN
if ( $src =~ $perl5_re )
{
my $version = $1;
my $minor = $2;
# They probably want the latest if minor wasn't given
if ( !defined $minor )
{
# 11 is the highest minor version seen as of this writing
my @possible = ( 0 .. 15 );
lib/App/MechaCPAN/Perl.pm view on Meta::CPAN
],
"5.$version.$minor"
);
}
die "Cannot find $src\n";
}
1;
__END__
=encoding utf-8
=head1 NAME
App::MechaCPAN::Perl - Mechanize the installation of Perl.
=head1 SYNOPSIS
# Install 5.24 into local/
user@host:~$ mechacpan perl 5.24
=head1 DESCRIPTION
The C<perl> command is used to install L<perl> into C<local/>. This removes the package's dependency on the operating system perl. It will do this by either downloading a binary archive or by building from a L<perl> source archive.
=head2 Methods
=head3 go( \%opts, $version )
There is only a single public function that should be called. This will install the version of perl given in C<$version> using the options in C<\%opts>. The options available are listed in the L<arguments|/Arguments> section below.
C<$version> is either 0 or 1 parameter:
=over
=item * If 0 parameters are given and there is a .perl-version file, it will try and use that as the version to install.
=item * Otherwise, if 0 parameters are given, it will attempt to find and install the newest, stable version of perl.
=item * If the parameter is a file, it will try to use that file as an archive to install perl.
=item * If the parameter looks like a URL, it will fetch that URL and try to use it as an archive to install perl.
=item * If the parameter is a major version (5.XX), it will attempt to download and install the newest minor version of that major version.
=item * If the parameter is a minor version (5.XX.X), it will attempt to download and install that exact version.
=back
In the cases where a version is given, and the C<--no-source-only> option is given, C<App::MechaCPAN::Perl> will attempt to download a binary archive prebuilt for the operating system. This guess is made by looking at how the currently executing L<pe...
After an archive is retrieved, it will be checked to see if it is a binary or source package. This is accomplished by checking for an executable C<bin/perl> file in the archive. Basic tests are ran to make sure the binary is usable, notably by runnin...
=head2 Arguments
=head3 source-only
By default a source archive is attempted to be retreived and installed. If you want it to attempt to also retrieve a binary archive, you can use C<--no-source-only>. If you do not want C<App::MechaCPAN::Perl> to even attempt to use a binary archive, ...
=head3 threads
By default, perl is compiled without threads. If you'd like to enable threads, use this argument.
=head3 shared-lib
By default, perl will not generate a libperl.a file. If you need libperl.so, then use this argument.
=head3 jobs
How many make jobs to use when running make. The code will guess if C<make> supports running multiple jobs, and as such, it may not work for all versions of make. Defaults to 2.
=head3 skip-tests
Test for perl are ran by default. If you are sure that the tests will pass and you want to save some time, you can skip the testing phase with this option.
=head3 smart-tests
As an alternative to telling C<App::MechaCPAN::Perl> to use tests or not, C<App::MechaCPAN::Perl> can try to be clever and guess if it needs to run tests. If there is a C<.perl-version> file and it is the same version that is being installed, then te...
C<smart-tests> are off by default, but are enabled by L<App::MechaCPAN::Deploy> when there is a C<cpanfile.snapshot> file. See L<App::MechaCPAN::Install/smart-tests>.
=head3 skip-local
Since perl and modules will be installed by L<App::MechaCPAN> into C<local/>, by default C<local/> will be added to C<@INC>. This means that if you use the C<local/> installed perl you do not need to use L<local::lib> or other C<@INC> tricks. If you ...
=head3 skip-lib
If a C<lib/> directory exists in the same directory as the C<local/> directory, then C<lib/> will also bee added to C<@INC>. This is helpful if you're installing to run an application that includes a C<lib/> directory. If you do not want this to be a...
=head3 devel
By default, perl will not compile a development version without -Dusedevel passed to configure. This adds that flag to the configure step so that perl will install unstable development versions. This is B<NOT> recommended except for testing.
=head1 WIN32 LIMITATION
Building perl from scratch on Win32 is nothing like building it on other platforms. At this point, the C<perl> command does not work on Win32.
=head1 AUTHOR
Jon Gentle E<lt>cpan@atrodo.orgE<gt>
=head1 COPYRIGHT
Copyright 2017- Jon Gentle
=head1 LICENSE
This is free software. You may redistribute copies of it under the terms of the Artistic License 2 as published by The Perl Foundation.
=head1 SEE ALSO
=over
=item L<plenv|https://github.com/tokuhirom/plenv>
=item L<App::perlbrew>
=back
=cut
( run in 0.712 second using v1.01-cache-2.11-cpan-39bf76dae61 )