* First figure out what symlinks need to be created, then create
them. This prevents unnecessary unlink operations to resolve collisions between directories. svn path=/nixpkgs/trunk/; revision=22526
This commit is contained in:
parent
eb2cb9afd4
commit
a6fcf45726
@ -9,10 +9,6 @@ use File::Basename;
|
|||||||
STDOUT->autoflush(1);
|
STDOUT->autoflush(1);
|
||||||
|
|
||||||
my $out = $ENV{"out"};
|
my $out = $ENV{"out"};
|
||||||
mkdir "$out", 0755 || die "error creating $out";
|
|
||||||
|
|
||||||
|
|
||||||
my $symlinks = 0;
|
|
||||||
|
|
||||||
|
|
||||||
my @pathsToLink = split ' ', $ENV{"pathsToLink"};
|
my @pathsToLink = split ' ', $ENV{"pathsToLink"};
|
||||||
@ -27,95 +23,58 @@ sub isInPathsToLink {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
sub symLinkMkdir {
|
# For each activated package, determine what symlinks to create.
|
||||||
my $src = shift;
|
|
||||||
my $dst = shift;
|
my %symlinks;
|
||||||
my $dir = dirname $dst;
|
$symlinks{""} = ""; # create root directory
|
||||||
mkpath $dir;
|
|
||||||
symlink($src, $dst) ||
|
sub findFiles;
|
||||||
die "error creating link `$dst': $!";
|
|
||||||
$symlinks++;
|
sub findFilesInDir {
|
||||||
|
my ($relName, $target, $ignoreCollisions) = @_;
|
||||||
|
|
||||||
|
opendir DIR, "$target" or die "cannot open `$target': $!";
|
||||||
|
my @names = readdir DIR or die;
|
||||||
|
closedir DIR;
|
||||||
|
|
||||||
|
foreach my $name (@names) {
|
||||||
|
next if $name eq "." || $name eq "..";
|
||||||
|
findFiles("$relName/$name", "$target/$name", $name, $ignoreCollisions);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub findFiles {
|
||||||
|
my ($relName, $target, $baseName, $ignoreCollisions) = @_;
|
||||||
|
|
||||||
|
# Urgh, hacky...
|
||||||
|
return if
|
||||||
|
$relName eq "/propagated-build-inputs" ||
|
||||||
|
$relName eq "/nix-support" ||
|
||||||
|
$relName =~ /info\/dir/ ||
|
||||||
|
( $relName =~ /^\/share\/mime\// && !( $relName =~ /^\/share\/mime\/packages/ ) ) ||
|
||||||
|
$baseName eq "perllocal.pod" ||
|
||||||
|
$baseName eq "log";
|
||||||
|
|
||||||
# For each activated package, create symlinks.
|
my $oldTarget = $symlinks{$relName};
|
||||||
|
|
||||||
sub createLinks {
|
if (!defined $oldTarget) {
|
||||||
my $relName = shift;
|
$symlinks{$relName} = $target;
|
||||||
my $srcDir = shift;
|
return;
|
||||||
my $dstDir = shift;
|
}
|
||||||
my $ignoreCollisions = shift;
|
|
||||||
|
|
||||||
my @srcFiles = glob("$srcDir/*");
|
unless (-d $target && ($oldTarget eq "" || -d $oldTarget)) {
|
||||||
|
if ($ignoreCollisions) {
|
||||||
foreach my $srcFile (@srcFiles) {
|
warn "collision between `$target' and `$oldTarget'";
|
||||||
my $baseName = $srcFile;
|
return;
|
||||||
$baseName =~ s/^.*\///g; # strip directory
|
} else {
|
||||||
my $dstFile = "$dstDir/$baseName";
|
die "collision between `$target' and `$oldTarget'";
|
||||||
my $relName2 = "$relName/$baseName";
|
|
||||||
|
|
||||||
# Urgh, hacky...
|
|
||||||
if ($srcFile =~ /\/propagated-build-inputs$/ ||
|
|
||||||
$srcFile =~ /\/nix-support$/ ||
|
|
||||||
$srcFile =~ /\/perllocal.pod$/ ||
|
|
||||||
$srcFile =~ /\/info\/dir$/ ||
|
|
||||||
( $relName2 =~ /^\/share\/mime\// && !( $relName2 =~ /^\/share\/mime\/packages/ ) ) ||
|
|
||||||
$srcFile =~ /\/log$/)
|
|
||||||
{
|
|
||||||
# Do nothing.
|
|
||||||
}
|
|
||||||
|
|
||||||
elsif (-d $srcFile) {
|
|
||||||
|
|
||||||
if (!isInPathsToLink($relName2)) {
|
|
||||||
# This path is not in the list of paths to link, but
|
|
||||||
# some of its children may be.
|
|
||||||
createLinks($relName2, $srcFile, $dstFile, $ignoreCollisions);
|
|
||||||
next;
|
|
||||||
}
|
|
||||||
|
|
||||||
lstat $dstFile;
|
|
||||||
|
|
||||||
if (-d _) {
|
|
||||||
createLinks($relName2, $srcFile, $dstFile, $ignoreCollisions);
|
|
||||||
}
|
|
||||||
|
|
||||||
elsif (-l _) {
|
|
||||||
my $target = readlink $dstFile or die;
|
|
||||||
if (!-d $target) {
|
|
||||||
die "collission between directory `$srcFile' and non-directory `$target'";
|
|
||||||
}
|
|
||||||
unlink $dstFile or die "error unlinking `$dstFile': $!";
|
|
||||||
mkpath $dstFile;
|
|
||||||
createLinks($relName2, $target, $dstFile, $ignoreCollisions);
|
|
||||||
createLinks($relName2, $srcFile, $dstFile, $ignoreCollisions);
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
symLinkMkdir $srcFile, $dstFile;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
elsif (-l $dstFile) {
|
|
||||||
my $oldTarget = readlink $dstFile;
|
|
||||||
my $oldTargetReal = abs_path $oldTarget;
|
|
||||||
my $newTarget = $srcFile;
|
|
||||||
my $newTargetReal = abs_path $newTarget;
|
|
||||||
unless ($newTargetReal eq $oldTargetReal) {
|
|
||||||
if ($ignoreCollisions) {
|
|
||||||
warn "collision between `$newTarget' and `$oldTarget'\n";
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
die "collision between `$newTarget' and `$oldTarget'";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
next unless isInPathsToLink($relName2);
|
|
||||||
symLinkMkdir $srcFile, $dstFile;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
findFilesInDir($relName, $oldTarget, $ignoreCollisions) unless $oldTarget eq "";
|
||||||
|
findFilesInDir($relName, $target, $ignoreCollisions);
|
||||||
|
|
||||||
|
$symlinks{$relName} = ""; # denotes directory
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -130,8 +89,7 @@ sub addPkg($;$) {
|
|||||||
return if (defined $done{$pkgDir});
|
return if (defined $done{$pkgDir});
|
||||||
$done{$pkgDir} = 1;
|
$done{$pkgDir} = 1;
|
||||||
|
|
||||||
# print "symlinking $pkgDir\n";
|
findFiles("", "$pkgDir", "", $ignoreCollisions);
|
||||||
createLinks("", "$pkgDir", "$out", $ignoreCollisions);
|
|
||||||
|
|
||||||
my $propagatedFN = "$pkgDir/nix-support/propagated-user-env-packages";
|
my $propagatedFN = "$pkgDir/nix-support/propagated-user-env-packages";
|
||||||
if (-e $propagatedFN) {
|
if (-e $propagatedFN) {
|
||||||
@ -140,7 +98,6 @@ sub addPkg($;$) {
|
|||||||
close PROP;
|
close PROP;
|
||||||
my @propagated = split ' ', $propagated;
|
my @propagated = split ' ', $propagated;
|
||||||
foreach my $p (@propagated) {
|
foreach my $p (@propagated) {
|
||||||
print "$pkgDir propagates $p\n";
|
|
||||||
$postponed{$p} = 1 unless defined $done{$p};
|
$postponed{$p} = 1 unless defined $done{$p};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -167,17 +124,36 @@ while (scalar(keys %postponed) > 0) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# Create the symlinks.
|
||||||
|
my $nrLinks = 0;
|
||||||
|
foreach my $relName (sort keys %symlinks) {
|
||||||
|
my $target = $symlinks{$relName};
|
||||||
|
my $abs = "$out/$relName";
|
||||||
|
next unless isInPathsToLink $relName;
|
||||||
|
if ($target eq "") {
|
||||||
|
#print "creating directory $relName\n";
|
||||||
|
mkpath $abs or die "cannot create directory `$abs': $!";
|
||||||
|
} else {
|
||||||
|
#print "creating symlink $relName to $target\n";
|
||||||
|
symlink $target, $abs ||
|
||||||
|
die "error creating link `$abs': $!";
|
||||||
|
$nrLinks++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if (-x "$out/bin/update-mime-database" && -d "$out/share/mime/packages") {
|
if (-x "$out/bin/update-mime-database" && -d "$out/share/mime/packages") {
|
||||||
system("$out/bin/update-mime-database -V $out/share/mime") == 0
|
system("$out/bin/update-mime-database -V $out/share/mime") == 0
|
||||||
or die "Can't update mime-database";
|
or die "Can't update mime-database";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
print STDERR "created $symlinks symlinks in user environment\n";
|
print STDERR "created $nrLinks symlinks in user environment\n";
|
||||||
|
|
||||||
|
|
||||||
my $manifest = $ENV{"manifest"};
|
my $manifest = $ENV{"manifest"};
|
||||||
if ($manifest ne "") {
|
if ($manifest) {
|
||||||
symlink($manifest, "$out/manifest") or die "cannot create manifest";
|
symlink($manifest, "$out/manifest") or die "cannot create manifest";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user