|
package ExtUtils::Manifest; |
|
|
|
require Exporter; |
|
use Config; |
|
use File::Basename; |
|
use File::Copy 'copy'; |
|
use File::Find; |
|
use File::Spec 0.8; |
|
use Carp; |
|
use strict; |
|
use warnings; |
|
|
|
our $VERSION = '1.73'; |
|
our @ISA = ('Exporter'); |
|
our @EXPORT_OK = qw(mkmanifest |
|
manicheck filecheck fullcheck skipcheck |
|
manifind maniread manicopy maniadd |
|
maniskip |
|
); |
|
|
|
our $Is_MacOS = $^O eq 'MacOS'; |
|
our $Is_VMS = $^O eq 'VMS'; |
|
our $Is_VMS_mode = 0; |
|
our $Is_VMS_lc = 0; |
|
our $Is_VMS_nodot = 0; |
|
|
|
if ($Is_VMS) { |
|
require VMS::Filespec if $Is_VMS; |
|
my $vms_unix_rpt; |
|
my $vms_efs; |
|
my $vms_case; |
|
|
|
$Is_VMS_mode = 1; |
|
$Is_VMS_lc = 1; |
|
$Is_VMS_nodot = 1; |
|
if (eval { local $SIG{__DIE__}; require VMS::Feature; }) { |
|
$vms_unix_rpt = VMS::Feature::current("filename_unix_report"); |
|
$vms_efs = VMS::Feature::current("efs_charset"); |
|
$vms_case = VMS::Feature::current("efs_case_preserve"); |
|
} else { |
|
my $unix_rpt = $ENV{'DECC$FILENAME_UNIX_REPORT'} || ''; |
|
my $efs_charset = $ENV{'DECC$EFS_CHARSET'} || ''; |
|
my $efs_case = $ENV{'DECC$EFS_CASE_PRESERVE'} || ''; |
|
$vms_unix_rpt = $unix_rpt =~ /^[ET1]/i; |
|
$vms_efs = $efs_charset =~ /^[ET1]/i; |
|
$vms_case = $efs_case =~ /^[ET1]/i; |
|
} |
|
$Is_VMS_lc = 0 if ($vms_case); |
|
$Is_VMS_mode = 0 if ($vms_unix_rpt); |
|
$Is_VMS_nodot = 0 if ($vms_efs); |
|
} |
|
|
|
our $Debug = $ENV{PERL_MM_MANIFEST_DEBUG} || 0; |
|
our $Verbose = defined $ENV{PERL_MM_MANIFEST_VERBOSE} ? |
|
$ENV{PERL_MM_MANIFEST_VERBOSE} : 1; |
|
our $Quiet = 0; |
|
our $MANIFEST = 'MANIFEST'; |
|
|
|
our $DEFAULT_MSKIP = File::Spec->rel2abs(File::Spec->catfile( dirname(__FILE__), "$MANIFEST.SKIP" )); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub _sort { |
|
return sort { lc $a cmp lc $b } @_; |
|
} |
|
|
|
sub mkmanifest { |
|
my $manimiss = 0; |
|
my $read = (-r 'MANIFEST' && maniread()) or $manimiss++; |
|
$read = {} if $manimiss; |
|
local *M; |
|
my $bakbase = $MANIFEST; |
|
$bakbase =~ s/\./_/g if $Is_VMS_nodot; |
|
rename $MANIFEST, "$bakbase.bak" unless $manimiss; |
|
open M, "> $MANIFEST" or die "Could not open $MANIFEST: $!"; |
|
binmode M, ':raw'; |
|
my $skip = maniskip(); |
|
my $found = manifind(); |
|
my($key,$val,$file,%all); |
|
%all = (%$found, %$read); |
|
$all{$MANIFEST} = ($Is_VMS_mode ? "$MANIFEST\t\t" : '') . |
|
'This list of files' |
|
if $manimiss; |
|
foreach $file (_sort keys %all) { |
|
if ($skip->($file)) { |
|
|
|
|
|
warn "Removed from $MANIFEST: $file\n" if $Verbose and exists $read->{$file}; |
|
next; |
|
} |
|
if ($Verbose){ |
|
warn "Added to $MANIFEST: $file\n" unless exists $read->{$file}; |
|
} |
|
my $text = $all{$file}; |
|
$file = _unmacify($file); |
|
my $tabs = (5 - (length($file)+1)/8); |
|
$tabs = 1 if $tabs < 1; |
|
$tabs = 0 unless $text; |
|
if ($file =~ /\s/) { |
|
$file =~ s/([\\'])/\\$1/g; |
|
$file = "'$file'"; |
|
} |
|
print M $file, "\t" x $tabs, $text, "\n"; |
|
} |
|
close M; |
|
} |
|
|
|
|
|
|
|
sub clean_up_filename { |
|
my $filename = shift; |
|
$filename =~ s|^\./||; |
|
$filename =~ s/^:([^:]+)$/$1/ if $Is_MacOS; |
|
if ( $Is_VMS ) { |
|
$filename =~ s/\.$//; |
|
$filename = VMS::Filespec::unixify($filename); |
|
if( $Is_VMS_lc ) { |
|
$filename = lc($filename); |
|
$filename = uc($filename) if $filename =~ /^MANIFEST(\.SKIP)?$/i; |
|
} |
|
} |
|
return $filename; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub manifind { |
|
my $p = shift || {}; |
|
my $found = {}; |
|
|
|
my $wanted = sub { |
|
my $name = clean_up_filename($File::Find::name); |
|
warn "Debug: diskfile $name\n" if $Debug; |
|
return if -d $_; |
|
$found->{$name} = ""; |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
find({wanted => $wanted, follow_fast => 1}, |
|
$Is_MacOS ? ":" : "."); |
|
|
|
return $found; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub manicheck { |
|
return _check_files(); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub filecheck { |
|
return _check_manifest(); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub fullcheck { |
|
return [_check_files()], [_check_manifest()]; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub skipcheck { |
|
my($p) = @_; |
|
my $found = manifind(); |
|
my $matches = maniskip(); |
|
|
|
my @skipped = (); |
|
foreach my $file (_sort keys %$found){ |
|
if (&$matches($file)){ |
|
warn "Skipping $file\n" unless $Quiet; |
|
push @skipped, $file; |
|
next; |
|
} |
|
} |
|
|
|
return @skipped; |
|
} |
|
|
|
|
|
sub _check_files { |
|
my $p = shift; |
|
my $dosnames=(defined(&Dos::UseLFN) && Dos::UseLFN()==0); |
|
my $read = maniread() || {}; |
|
my $found = manifind($p); |
|
|
|
my(@missfile) = (); |
|
foreach my $file (_sort keys %$read){ |
|
warn "Debug: manicheck checking from $MANIFEST $file\n" if $Debug; |
|
if ($dosnames){ |
|
$file = lc $file; |
|
$file =~ s=(\.(\w|-)+)=substr ($1,0,4)=ge; |
|
$file =~ s=((\w|-)+)=substr ($1,0,8)=ge; |
|
} |
|
unless ( exists $found->{$file} ) { |
|
warn "No such file: $file\n" unless $Quiet; |
|
push @missfile, $file; |
|
} |
|
} |
|
|
|
return @missfile; |
|
} |
|
|
|
|
|
sub _check_manifest { |
|
my($p) = @_; |
|
my $read = maniread() || {}; |
|
my $found = manifind($p); |
|
my $skip = maniskip(); |
|
|
|
my @missentry = (); |
|
foreach my $file (_sort keys %$found){ |
|
next if $skip->($file); |
|
warn "Debug: manicheck checking from disk $file\n" if $Debug; |
|
unless ( exists $read->{$file} ) { |
|
my $canon = $Is_MacOS ? "\t" . _unmacify($file) : ''; |
|
warn "Not in $MANIFEST: $file$canon\n" unless $Quiet; |
|
push @missentry, $file; |
|
} |
|
} |
|
|
|
return @missentry; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub maniread { |
|
my ($mfile) = @_; |
|
$mfile ||= $MANIFEST; |
|
my $read = {}; |
|
local *M; |
|
unless (open M, "< $mfile"){ |
|
warn "Problem opening $mfile: $!"; |
|
return $read; |
|
} |
|
local $_; |
|
while (<M>){ |
|
chomp; |
|
next if /^\s* |
|
|
|
my($file, $comment); |
|
|
|
|
|
|
|
if (($file, $comment) = /^'((?:\\[\\']|.+)+)'\s*(.*)/) { |
|
$file =~ s/\\([\\'])/$1/g; |
|
} |
|
else { |
|
($file, $comment) = /^(\S+)\s*(.*)/; |
|
} |
|
next unless $file; |
|
|
|
if ($Is_MacOS) { |
|
$file = _macify($file); |
|
$file =~ s/\\([0-3][0-7][0-7])/sprintf("%c", oct($1))/ge; |
|
} |
|
elsif ($Is_VMS_mode) { |
|
require File::Basename; |
|
my($base,$dir) = File::Basename::fileparse($file); |
|
|
|
if ($Is_VMS_nodot) { |
|
$dir =~ tr/./_/; |
|
my(@pieces) = split(/\./,$base); |
|
if (@pieces > 2) |
|
{ $base = shift(@pieces) . '.' . join('_',@pieces); } |
|
my $okfile = "$dir$base"; |
|
warn "Debug: Illegal name $file changed to $okfile\n" if $Debug; |
|
$file = $okfile; |
|
} |
|
if( $Is_VMS_lc ) { |
|
$file = lc($file); |
|
$file = uc($file) if $file =~ /^MANIFEST(\.SKIP)?$/i; |
|
} |
|
} |
|
|
|
$read->{$file} = $comment; |
|
} |
|
close M; |
|
$read; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub maniskip { |
|
my @skip ; |
|
my $mfile = shift || "$MANIFEST.SKIP"; |
|
_check_mskip_directives($mfile) if -f $mfile; |
|
local(*M, $_); |
|
open M, "< $mfile" or open M, "< $DEFAULT_MSKIP" or return sub {0}; |
|
while (<M>){ |
|
chomp; |
|
s/\r//; |
|
$_ =~ qr{^\s*(?:(?:'([^\\']*(?:\\.[^\\']*)*)')|([^#\s]\S*))?(?:(?:\s*)|(?:\s+(.*?)\s*))$}; |
|
|
|
my $filename = $2; |
|
if ( defined($1) ) { |
|
$filename = $1; |
|
$filename =~ s/\\(['\\])/$1/g; |
|
} |
|
next if (not defined($filename) or not $filename); |
|
push @skip, _macify($filename); |
|
} |
|
close M; |
|
return sub {0} unless (scalar @skip > 0); |
|
|
|
my $opts = $Is_VMS_mode ? '(?i)' : ''; |
|
|
|
|
|
|
|
my $regex = join '|', map "(?:$_)", @skip; |
|
|
|
return sub { $_[0] =~ qr{$opts$regex} }; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub _check_mskip_directives { |
|
my $mfile = shift; |
|
local (*M, $_); |
|
my @lines = (); |
|
my $flag = 0; |
|
unless (open M, "< $mfile") { |
|
warn "Problem opening $mfile: $!"; |
|
return; |
|
} |
|
while (<M>) { |
|
if (/^#!include_default\s*$/) { |
|
if (my @default = _include_mskip_file()) { |
|
push @lines, @default; |
|
warn "Debug: Including default MANIFEST.SKIP\n" if $Debug; |
|
$flag++; |
|
} |
|
next; |
|
} |
|
if (/^#!include\s+(.*)\s*$/) { |
|
my $external_file = $1; |
|
if (my @external = _include_mskip_file($external_file)) { |
|
push @lines, @external; |
|
warn "Debug: Including external $external_file\n" if $Debug; |
|
$flag++; |
|
} |
|
next; |
|
} |
|
push @lines, $_; |
|
} |
|
close M; |
|
return unless $flag; |
|
my $bakbase = $mfile; |
|
$bakbase =~ s/\./_/g if $Is_VMS_nodot; |
|
rename $mfile, "$bakbase.bak"; |
|
warn "Debug: Saving original $mfile as $bakbase.bak\n" if $Debug; |
|
unless (open M, "> $mfile") { |
|
warn "Problem opening $mfile: $!"; |
|
return; |
|
} |
|
binmode M, ':raw'; |
|
print M $_ for (@lines); |
|
close M; |
|
return; |
|
} |
|
|
|
|
|
|
|
sub _include_mskip_file { |
|
my $mskip = shift || $DEFAULT_MSKIP; |
|
unless (-f $mskip) { |
|
warn qq{Included file "$mskip" not found - skipping}; |
|
return; |
|
} |
|
local (*M, $_); |
|
unless (open M, "< $mskip") { |
|
warn "Problem opening $mskip: $!"; |
|
return; |
|
} |
|
my @lines = (); |
|
push @lines, "\n#!start included $mskip\n"; |
|
push @lines, $_ while <M>; |
|
close M; |
|
push @lines, "#!end included $mskip\n\n"; |
|
return @lines; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub manicopy { |
|
my($read,$target,$how)=@_; |
|
croak "manicopy() called without target argument" unless defined $target; |
|
$how ||= 'cp'; |
|
require File::Path; |
|
require File::Basename; |
|
|
|
$target = VMS::Filespec::unixify($target) if $Is_VMS_mode; |
|
File::Path::mkpath([ $target ],! $Quiet,$Is_VMS ? undef : 0755); |
|
foreach my $file (keys %$read){ |
|
if ($Is_MacOS) { |
|
if ($file =~ m!:!) { |
|
my $dir = _maccat($target, $file); |
|
$dir =~ s/[^:]+$//; |
|
File::Path::mkpath($dir,1,0755); |
|
} |
|
cp_if_diff($file, _maccat($target, $file), $how); |
|
} else { |
|
$file = VMS::Filespec::unixify($file) if $Is_VMS_mode; |
|
if ($file =~ m!/!) { # Ilya, that hurts, I fear, or maybe not? |
|
my $dir = File::Basename::dirname($file); |
|
$dir = VMS::Filespec::unixify($dir) if $Is_VMS_mode; |
|
File::Path::mkpath(["$target/$dir"],! $Quiet,$Is_VMS ? undef : 0755); |
|
} |
|
cp_if_diff($file, "$target/$file", $how); |
|
} |
|
} |
|
} |
|
|
|
sub cp_if_diff { |
|
my($from, $to, $how)=@_; |
|
if (! -f $from) { |
|
carp "$from not found"; |
|
return; |
|
} |
|
my($diff) = 0; |
|
local(*F,*T); |
|
open(F,"< $from\0") or die "Can't read $from: $!\n"; |
|
if (open(T,"< $to\0")) { |
|
local $_; |
|
while (<F>) { $diff++,last if $_ ne <T>; } |
|
$diff++ unless eof(T); |
|
close T; |
|
} |
|
else { $diff++; } |
|
close F; |
|
if ($diff) { |
|
if (-e $to) { |
|
unlink($to) or confess "unlink $to: $!"; |
|
} |
|
STRICT_SWITCH: { |
|
best($from,$to), last STRICT_SWITCH if $how eq 'best'; |
|
cp($from,$to), last STRICT_SWITCH if $how eq 'cp'; |
|
ln($from,$to), last STRICT_SWITCH if $how eq 'ln'; |
|
croak("ExtUtils::Manifest::cp_if_diff " . |
|
"called with illegal how argument [$how]. " . |
|
"Legal values are 'best', 'cp', and 'ln'."); |
|
} |
|
} |
|
} |
|
|
|
sub cp { |
|
my ($srcFile, $dstFile) = @_; |
|
my ($access,$mod) = (stat $srcFile)[8,9]; |
|
|
|
copy($srcFile,$dstFile); |
|
utime $access, $mod + ($Is_VMS ? 1 : 0), $dstFile; |
|
_manicopy_chmod($srcFile, $dstFile); |
|
} |
|
|
|
|
|
sub ln { |
|
my ($srcFile, $dstFile) = @_; |
|
# Fix-me - VMS can support links. |
|
return &cp if $Is_VMS or ($^O eq 'MSWin32' and Win32::IsWin95()); |
|
link($srcFile, $dstFile); |
|
|
|
unless( _manicopy_chmod($srcFile, $dstFile) ) { |
|
unlink $dstFile; |
|
return; |
|
} |
|
1; |
|
} |
|
|
|
# 1) Strip off all group and world permissions. |
|
# 2) Let everyone read it. |
|
# 3) If the owner can execute it, everyone can. |
|
sub _manicopy_chmod { |
|
my($srcFile, $dstFile) = @_; |
|
|
|
my $perm = 0444 | (stat $srcFile)[2] & 0700; |
|
chmod( $perm | ( $perm & 0100 ? 0111 : 0 ), $dstFile ); |
|
} |
|
|
|
# Files that are often modified in the distdir. Don't hard link them. |
|
my @Exceptions = qw(MANIFEST META.yml SIGNATURE); |
|
sub best { |
|
my ($srcFile, $dstFile) = @_; |
|
|
|
my $is_exception = grep $srcFile =~ /$_/, @Exceptions; |
|
if ($is_exception or !$Config{d_link} or -l $srcFile) { |
|
cp($srcFile, $dstFile); |
|
} else { |
|
ln($srcFile, $dstFile) or cp($srcFile, $dstFile); |
|
} |
|
} |
|
|
|
sub _macify { |
|
my($file) = @_; |
|
|
|
return $file unless $Is_MacOS; |
|
|
|
$file =~ s|^\./||; |
|
if ($file =~ m|/|) { |
|
$file =~ s|/+|:|g; |
|
$file = ":$file"; |
|
} |
|
|
|
$file; |
|
} |
|
|
|
sub _maccat { |
|
my($f1, $f2) = @_; |
|
|
|
return "$f1/$f2" unless $Is_MacOS; |
|
|
|
$f1 .= ":$f2"; |
|
$f1 =~ s/([^:]:):/$1/g; |
|
return $f1; |
|
} |
|
|
|
sub _unmacify { |
|
my($file) = @_; |
|
|
|
return $file unless $Is_MacOS; |
|
|
|
$file =~ s|^:||; |
|
$file =~ s|([/ \n])|sprintf("\\%03o", unpack("c", $1))|ge; |
|
$file =~ y|:|/|; |
|
|
|
$file; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub maniadd { |
|
my($additions) = shift; |
|
|
|
_normalize($additions); |
|
_fix_manifest($MANIFEST); |
|
|
|
my $manifest = maniread(); |
|
my @needed = grep !exists $manifest->{$_}, keys %$additions; |
|
return 1 unless @needed; |
|
|
|
open(MANIFEST, ">>$MANIFEST") or |
|
die "maniadd() could not open $MANIFEST: $!"; |
|
binmode MANIFEST, ':raw'; |
|
|
|
foreach my $file (_sort @needed) { |
|
my $comment = $additions->{$file} || ''; |
|
if ($file =~ /\s/) { |
|
$file =~ s/([\\'])/\\$1/g; |
|
$file = "'$file'"; |
|
} |
|
printf MANIFEST "%-40s %s\n", $file, $comment; |
|
} |
|
close MANIFEST or die "Error closing $MANIFEST: $!"; |
|
|
|
return 1; |
|
} |
|
|
|
|
|
|
|
|
|
sub _fix_manifest { |
|
my $manifest_file = shift; |
|
|
|
open MANIFEST, $MANIFEST or die "Could not open $MANIFEST: $!"; |
|
local $/; |
|
my @manifest = split /(\015\012|\012|\015)/, <MANIFEST>, -1; |
|
close MANIFEST; |
|
my $must_rewrite = ""; |
|
if ($manifest[-1] eq ""){ |
|
|
|
pop @manifest; |
|
for (my $i=1; $i<=$#manifest; $i+=2) { |
|
unless ($manifest[$i] eq "\n") { |
|
$must_rewrite = "not a newline at pos $i"; |
|
last; |
|
} |
|
} |
|
} else { |
|
$must_rewrite = "last line without newline"; |
|
} |
|
|
|
if ( $must_rewrite ) { |
|
1 while unlink $MANIFEST; |
|
open MANIFEST, ">", $MANIFEST or die "(must_rewrite=$must_rewrite) Could not open >$MANIFEST: $!"; |
|
binmode MANIFEST, ':raw'; |
|
for (my $i=0; $i<=$#manifest; $i+=2) { |
|
print MANIFEST "$manifest[$i]\n"; |
|
} |
|
close MANIFEST or die "could not write $MANIFEST: $!"; |
|
} |
|
} |
|
|
|
|
|
|
|
sub _normalize { |
|
return; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1; |
|
|