* Integrate binary cache generation into the mirror-channel script.
git-svn-id: https://nixos.org/repos/nix/release/trunk/channels@34658 70bd8c7a-acb8-0310-9f0d-9cc1c95dcdbb
This commit is contained in:
parent
8cc83aaa27
commit
cbfc30c73b
4 changed files with 61 additions and 134 deletions
|
@ -1,94 +0,0 @@
|
||||||
#! /var/run/current-system/sw/bin/perl -w
|
|
||||||
|
|
||||||
use strict;
|
|
||||||
use Nix::Manifest;
|
|
||||||
use File::Basename;
|
|
||||||
|
|
||||||
my $cacheDir = "/data/releases/binary-cache";
|
|
||||||
|
|
||||||
if (! -e "$cacheDir/nix-cache-info") {
|
|
||||||
open FILE, ">$cacheDir/nix-cache-info" or die;
|
|
||||||
print FILE "StoreDir: /nix/store\n";
|
|
||||||
print FILE "WantMassQuery: 1\n";
|
|
||||||
close FILE;
|
|
||||||
}
|
|
||||||
|
|
||||||
my @manifests = split " ", `find /data/releases/{nixos,nixpkgs,nix,patchelf} -name MANIFEST | grep -v '.tmp' | sort`;
|
|
||||||
die if $? != 0;
|
|
||||||
#my @manifests = ("/data/releases/nixpkgs/nixpkgs-0.5/MANIFEST");
|
|
||||||
#my @manifests = ("/data/releases/nixpkgs/nixpkgs-1.0pre19955_f823b62/MANIFEST");
|
|
||||||
|
|
||||||
foreach my $manifest (@manifests) {
|
|
||||||
my %narFiles;
|
|
||||||
my %patches;
|
|
||||||
if (readManifest($manifest, \%narFiles, \%patches, 1) < 3) {
|
|
||||||
print STDERR "warning: skipping old manifest $manifest\n";
|
|
||||||
next;
|
|
||||||
}
|
|
||||||
print STDERR "processing $manifest (", scalar(keys(%narFiles)), " paths)...\n";
|
|
||||||
|
|
||||||
while (my ($storePath, $nars) = each %narFiles) {
|
|
||||||
my $pathHash = substr(basename($storePath), 0, 32);
|
|
||||||
my $dst = "$cacheDir/$pathHash.narinfo";
|
|
||||||
|
|
||||||
foreach my $nar (@{$nars}) {
|
|
||||||
if (! -e $dst) {
|
|
||||||
print STDERR " $storePath\n";
|
|
||||||
#print STDERR " $nar->{url} -> $dst\n";
|
|
||||||
|
|
||||||
my $fileName = "/data/releases/nars/" . basename $nar->{url};
|
|
||||||
if (! -e $fileName) {
|
|
||||||
warn "NAR not found: $fileName\n";
|
|
||||||
next;
|
|
||||||
}
|
|
||||||
|
|
||||||
my $narHash = $nar->{narHash};
|
|
||||||
die unless defined $narHash;
|
|
||||||
if (length($narHash) != 59) {
|
|
||||||
$narHash =~ s/.*://;
|
|
||||||
my $s = `nix-hash --type sha256 --to-base32 $narHash`;
|
|
||||||
die if $? != 0;
|
|
||||||
chomp $s;
|
|
||||||
$narHash = "sha256:$s";
|
|
||||||
}
|
|
||||||
die unless defined $nar->{size};
|
|
||||||
die unless length($nar->{hash} || "") == 59;
|
|
||||||
|
|
||||||
$nar->{hash} =~ /^sha256:(.*)$/ or die;
|
|
||||||
my $fileHash = $1;
|
|
||||||
|
|
||||||
my $link = "$cacheDir/nar/$fileHash.nar.bz2";
|
|
||||||
if (! -e $link) {
|
|
||||||
link $fileName, $link
|
|
||||||
or die "creating link: $!";
|
|
||||||
}
|
|
||||||
|
|
||||||
my $narSize = $nar->{narSize};
|
|
||||||
unless (defined $narSize) {
|
|
||||||
$narSize = `bzip2 -d < $link | wc -c`;
|
|
||||||
chomp $narSize;
|
|
||||||
die unless $narSize =~ /^[0-9]+$/;
|
|
||||||
}
|
|
||||||
|
|
||||||
my $info;
|
|
||||||
$info .= "StorePath: $storePath\n";
|
|
||||||
$info .= "URL: nar/$fileHash.nar.bz2\n";
|
|
||||||
$info .= "Compression: bzip2\n";
|
|
||||||
$info .= "FileHash: $nar->{hash}\n";
|
|
||||||
$info .= "FileSize: $nar->{size}\n";
|
|
||||||
$info .= "NarHash: $narHash\n";
|
|
||||||
$info .= "NarSize: $narSize\n";
|
|
||||||
$info .= "References: " . join(" ", map { basename $_ } (split " ", $nar->{references})) . "\n";
|
|
||||||
$info .= "Deriver: " . basename $nar->{deriver} . "\n" if $nar->{deriver} ne "";
|
|
||||||
$info .= "System: $nar->{system}\n" if defined $nar->{system};
|
|
||||||
|
|
||||||
my $tmp = "$cacheDir/.tmp.$$.$pathHash.narinfo";
|
|
||||||
open INFO, ">$tmp" or die;
|
|
||||||
print INFO "$info" or die;
|
|
||||||
close INFO or die;
|
|
||||||
rename($tmp, $dst) or die "cannot rename $tmp to $dst: $!\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -5,6 +5,7 @@
|
||||||
use strict;
|
use strict;
|
||||||
use Nix::Manifest;
|
use Nix::Manifest;
|
||||||
use Nix::GeneratePatches;
|
use Nix::GeneratePatches;
|
||||||
|
use Nix::Utils;
|
||||||
use File::Basename;
|
use File::Basename;
|
||||||
use File::stat;
|
use File::stat;
|
||||||
|
|
||||||
|
@ -14,19 +15,20 @@ if (scalar @ARGV < 4 || scalar @ARGV > 6) {
|
||||||
exit 1;
|
exit 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
my $curl = "curl --location --no-progress-bar --show-error --fail";
|
my $curl = "curl --location --silent --show-error --fail";
|
||||||
|
|
||||||
my $srcChannelURL = $ARGV[0];
|
my $srcChannelURL = $ARGV[0];
|
||||||
my $dstChannelPath = $ARGV[1];
|
my $dstChannelPath = $ARGV[1];
|
||||||
my $narPath = $ARGV[2];
|
my $cacheDir = $ARGV[2];
|
||||||
my $narURL = $ARGV[3];
|
my $cacheURL = $ARGV[3];
|
||||||
my $allPatchesManifest = $ARGV[4] || "";
|
my $allPatchesManifest = $ARGV[4] || "";
|
||||||
my $nixexprsURL = $ARGV[5] || "$srcChannelURL/nixexprs.tar.bz2";
|
my $nixexprsURL = $ARGV[5] || "$srcChannelURL/nixexprs.tar.bz2";
|
||||||
|
|
||||||
die "$dstChannelPath doesn't exist\n" unless -d $dstChannelPath;
|
die "$dstChannelPath doesn't exist\n" unless -d $dstChannelPath;
|
||||||
die "$narPath doesn't exist\n" unless -d $narPath;
|
die "$cacheDir doesn't exist\n" unless -d $cacheDir;
|
||||||
|
|
||||||
my $manifestPath = "$dstChannelPath/MANIFEST";
|
my $manifestPath = "$dstChannelPath/MANIFEST";
|
||||||
|
my $narDir = "$cacheDir/nar";
|
||||||
|
|
||||||
|
|
||||||
# Fetch the manifest.
|
# Fetch the manifest.
|
||||||
|
@ -50,7 +52,7 @@ readManifest("$dstChannelPath/MANIFEST", \%narFiles, \%patches);
|
||||||
%patches = (); # not supported yet
|
%patches = (); # not supported yet
|
||||||
|
|
||||||
my $size = scalar (keys %narFiles);
|
my $size = scalar (keys %narFiles);
|
||||||
print "$size store paths in manifest\n";
|
print STDERR "$size store paths in manifest\n";
|
||||||
|
|
||||||
|
|
||||||
# Protect against Hydra problems that leave the channel empty.
|
# Protect against Hydra problems that leave the channel empty.
|
||||||
|
@ -61,46 +63,65 @@ die "cowardly refusing to mirror an empty channel" if $size == 0;
|
||||||
# to point to the mirror. Also fill in the size and hash fields in
|
# to point to the mirror. Also fill in the size and hash fields in
|
||||||
# the manifest in order to be compatible with Nix < 0.13.
|
# the manifest in order to be compatible with Nix < 0.13.
|
||||||
|
|
||||||
while (my ($storePath, $files) = each %narFiles) {
|
while (my ($storePath, $nars) = each %narFiles) {
|
||||||
foreach my $file (@{$files}) {
|
|
||||||
my $narHash = $file->{narHash};
|
|
||||||
my $srcURL = $file->{url};
|
|
||||||
my $dstName = $narHash;
|
|
||||||
$dstName =~ s/:/_/; # `:' in filenames might cause problems
|
|
||||||
my $dstFile = "$narPath/$dstName";
|
|
||||||
my $dstURL = "$narURL/$dstName";
|
|
||||||
|
|
||||||
$file->{url} = $dstURL;
|
my $pathHash = substr(basename($storePath), 0, 32);
|
||||||
if (! -e $dstFile) {
|
my $narInfoFile = "$cacheDir/$pathHash.narinfo";
|
||||||
print "downloading $srcURL\n";
|
|
||||||
my $dstFileTmp = "$narPath/.tmp.$$.nar.$dstName";
|
foreach my $nar (@{$nars}) {
|
||||||
system("$curl '$srcURL' > $dstFileTmp") == 0 or die "failed to download `$srcURL'";
|
if (! -e $narInfoFile) {
|
||||||
|
my $dstFileTmp = "$narDir/.tmp.$$.nar.$nar->{narHash}";
|
||||||
|
print STDERR "downloading $nar->{url}\n";
|
||||||
|
system("$curl '$nar->{url}' > $dstFileTmp") == 0 or die "failed to download `$nar->{url}'";
|
||||||
|
|
||||||
# Verify whether the downloaded file is a bzipped NAR file
|
# Verify whether the downloaded file is a bzipped NAR file
|
||||||
# that matches the NAR hash given in the manifest.
|
# that matches the NAR hash given in the manifest.
|
||||||
my $hash = `bunzip2 < $dstFileTmp | nix-hash --type sha256 --flat /dev/stdin` or die;
|
my $narHash = `bunzip2 < $dstFileTmp | nix-hash --type sha256 --flat /dev/stdin` or die;
|
||||||
chomp $hash;
|
chomp $narHash;
|
||||||
die "hash mismatch in downloaded file `$srcURL'" if "sha256:$hash" ne $file->{narHash};
|
die "hash mismatch in downloaded file `$nar->{url}'" if "sha256:$narHash" ne $nar->{narHash};
|
||||||
|
|
||||||
rename($dstFileTmp, $dstFile) or die "cannot rename $dstFileTmp";
|
# Compute the hash of the compressed NAR (Hydra doesn't provide one).
|
||||||
}
|
my $fileHash = `nix-hash --flat --type sha256 --base32 '$dstFileTmp'` or die;
|
||||||
|
chomp $fileHash;
|
||||||
|
$nar->{hash} = "sha256:$fileHash";
|
||||||
|
|
||||||
$file->{size} = stat($dstFile)->size or die "cannot get size of $dstFile";
|
my $dstFile = "$narDir/$fileHash.nar.bz2";
|
||||||
|
if (-e $dstFile) {
|
||||||
my $hashFile = "$narPath/.hash.$dstName";
|
unlink($dstFileTmp) or die;
|
||||||
my $hash;
|
|
||||||
if (-e $hashFile) {
|
|
||||||
open HASH, "<$hashFile" or die;
|
|
||||||
$hash = <HASH>;
|
|
||||||
close HASH;
|
|
||||||
} else {
|
} else {
|
||||||
$hash = `nix-hash --flat --type sha256 --base32 '$dstFile'` or die;
|
rename($dstFileTmp, $dstFile) or die "cannot rename $dstFileTmp to $dstFile";
|
||||||
chomp $hash;
|
|
||||||
open HASH, ">$hashFile" or die;
|
|
||||||
print HASH $hash;
|
|
||||||
close HASH;
|
|
||||||
}
|
}
|
||||||
$file->{hash} = "sha256:$hash";
|
|
||||||
|
$nar->{size} = stat($dstFile)->size;
|
||||||
|
|
||||||
|
# Write the .narinfo.
|
||||||
|
my $info;
|
||||||
|
$info .= "StorePath: $storePath\n";
|
||||||
|
$info .= "URL: nar/$fileHash.nar.bz2\n";
|
||||||
|
$info .= "Compression: bzip2\n";
|
||||||
|
$info .= "FileHash: $nar->{hash}\n";
|
||||||
|
$info .= "FileSize: $nar->{size}\n";
|
||||||
|
$info .= "NarHash: $nar->{narHash}\n";
|
||||||
|
$info .= "NarSize: $nar->{narSize}\n";
|
||||||
|
$info .= "References: " . join(" ", map { basename $_ } (split " ", $nar->{references})) . "\n";
|
||||||
|
$info .= "Deriver: " . basename $nar->{deriver} . "\n" if $nar->{deriver} ne "";
|
||||||
|
$info .= "System: $nar->{system}\n" if defined $nar->{system};
|
||||||
|
|
||||||
|
my $tmp = "$cacheDir/.tmp.$$.$pathHash.narinfo";
|
||||||
|
open INFO, ">$tmp" or die;
|
||||||
|
print INFO "$info" or die;
|
||||||
|
close INFO or die;
|
||||||
|
rename($tmp, $narInfoFile) or die "cannot rename $tmp to $narInfoFile: $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
my $narInfo = parseNARInfo($storePath, readFile($narInfoFile));
|
||||||
|
$nar->{hash} = $narInfo->{fileHash};
|
||||||
|
$nar->{size} = $narInfo->{fileSize};
|
||||||
|
$nar->{narHash} = $narInfo->{narHash};
|
||||||
|
$nar->{narSize} = $narInfo->{narSize};
|
||||||
|
$nar->{url} = "$cacheURL/$narInfo->{url}";
|
||||||
|
|
||||||
|
warn "archive `$cacheDir/$narInfo->{url}' has gone missing!\n" unless -f "$cacheDir/$narInfo->{url}";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -34,7 +34,7 @@ else
|
||||||
$wget --directory=$tmpDir $url/nixos.iso_graphical-x86_64-linux/download
|
$wget --directory=$tmpDir $url/nixos.iso_graphical-x86_64-linux/download
|
||||||
|
|
||||||
perl -w ./mirror-channel.pl "$url/eval/channel" "$tmpDir" \
|
perl -w ./mirror-channel.pl "$url/eval/channel" "$tmpDir" \
|
||||||
/data/releases/nars http://nixos.org/releases/nars \
|
/data/releases/binary-cache http://nixos.org/binary-cache \
|
||||||
/data/releases/patches/all-patches "$url/nixos.channel/download/1"
|
/data/releases/patches/all-patches "$url/nixos.channel/download/1"
|
||||||
|
|
||||||
mv $tmpDir $releaseDir
|
mv $tmpDir $releaseDir
|
||||||
|
|
|
@ -28,7 +28,7 @@ else
|
||||||
echo $url > $tmpDir/src-url
|
echo $url > $tmpDir/src-url
|
||||||
|
|
||||||
perl -w ./mirror-channel.pl "$url/eval/channel" "$tmpDir" \
|
perl -w ./mirror-channel.pl "$url/eval/channel" "$tmpDir" \
|
||||||
/data/releases/nars http://nixos.org/releases/nars \
|
/data/releases/binary-cache http://nixos.org/binary-cache \
|
||||||
/data/releases/patches/all-patches "$url/tarball/download/4"
|
/data/releases/patches/all-patches "$url/tarball/download/4"
|
||||||
|
|
||||||
mv $tmpDir $releaseDir
|
mv $tmpDir $releaseDir
|
||||||
|
|
Loading…
Reference in a new issue