9e975458b4
need any info on substitutable paths, we just call the substituters (such as download-using-manifests.pl) directly. This means that it's no longer necessary for nix-pull to register substitutes or for nix-channel to clear them, which makes those operations much faster (NIX-95). Also, we don't have to worry about keeping nix-pull manifests (in /nix/var/nix/manifests) and the database in sync with each other. The downside is that there is some overhead in calling an external program to get the substitutes info. For instance, "nix-env -qas" takes a bit longer. Abolishing the substitutes table also makes the logic in local-store.cc simpler, as we don't need to store info for invalid paths. On the downside, you cannot do things like "nix-store -qR" on a substitutable but invalid path (but nobody did that anyway). * Never catch interrupts (the Interrupted exception).
189 lines
4.6 KiB
Plaintext
189 lines
4.6 KiB
Plaintext
#! @perl@ -w
|
|
|
|
use strict;
|
|
|
|
my $rootsDir = "@localstatedir@/nix/gcroots/channels";
|
|
|
|
my $stateDir = $ENV{"NIX_STATE_DIR"};
|
|
$stateDir = "@localstatedir@/nix" unless defined $stateDir;
|
|
|
|
|
|
# Turn on caching in nix-prefetch-url.
|
|
my $channelCache = "$stateDir/channel-cache";
|
|
$ENV{'NIX_DOWNLOAD_CACHE'} = $channelCache;
|
|
mkdir $channelCache, 0755 unless -e $channelCache;
|
|
|
|
|
|
# Figure out the name of the `.nix-channels' file to use.
|
|
my $home = $ENV{"HOME"};
|
|
die '$HOME not set' unless defined $home;
|
|
my $channelsList = "$home/.nix-channels";
|
|
|
|
|
|
my @channels;
|
|
|
|
|
|
# Reads the list of channels from the file $channelsList;
|
|
sub readChannels {
|
|
return if (!-f $channelsList);
|
|
open CHANNELS, "<$channelsList" or die "cannot open `$channelsList': $!";
|
|
while (<CHANNELS>) {
|
|
chomp;
|
|
next if /^\s*\#/;
|
|
push @channels, $_;
|
|
}
|
|
close CHANNELS;
|
|
}
|
|
|
|
|
|
# Writes the list of channels to the file $channelsList;
|
|
sub writeChannels {
|
|
open CHANNELS, ">$channelsList" or die "cannot open `$channelsList': $!";
|
|
foreach my $url (@channels) {
|
|
print CHANNELS "$url\n";
|
|
}
|
|
close CHANNELS;
|
|
}
|
|
|
|
|
|
# Adds a channel to the file $channelsList;
|
|
sub addChannel {
|
|
my $url = shift;
|
|
readChannels;
|
|
foreach my $url2 (@channels) {
|
|
return if $url eq $url2;
|
|
}
|
|
push @channels, $url;
|
|
writeChannels;
|
|
}
|
|
|
|
|
|
# Remove a channel from the file $channelsList;
|
|
sub removeChannel {
|
|
my $url = shift;
|
|
my @left = ();
|
|
readChannels;
|
|
foreach my $url2 (@channels) {
|
|
push @left, $url2 if $url ne $url2;
|
|
}
|
|
@channels = @left;
|
|
writeChannels;
|
|
}
|
|
|
|
|
|
# Fetch Nix expressions and pull cache manifests from the subscribed
|
|
# channels.
|
|
sub update {
|
|
readChannels;
|
|
|
|
# Remove all the old manifests.
|
|
for my $manifest (glob "$stateDir/manifests/*.nixmanifest") {
|
|
unlink $manifest or die "cannot remove `$manifest': $!";
|
|
}
|
|
|
|
# Pull cache manifests.
|
|
foreach my $url (@channels) {
|
|
#print "pulling cache manifest from `$url'\n";
|
|
system("@bindir@/nix-pull", "--skip-wrong-store", "$url/MANIFEST") == 0
|
|
or die "cannot pull cache manifest from `$url'";
|
|
}
|
|
|
|
# Create a Nix expression that fetches and unpacks the channel Nix
|
|
# expressions.
|
|
|
|
my $inputs = "[";
|
|
foreach my $url (@channels) {
|
|
$url =~ /\/([^\/]+)\/?$/;
|
|
my $channelName = $1;
|
|
$channelName = "unnamed" unless defined $channelName;
|
|
|
|
my $fullURL = "$url/nixexprs.tar.bz2";
|
|
print "downloading Nix expressions from `$fullURL'...\n";
|
|
$ENV{"PRINT_PATH"} = 1;
|
|
$ENV{"QUIET"} = 1;
|
|
my ($hash, $path) = `@bindir@/nix-prefetch-url '$fullURL'`;
|
|
die "cannot fetch `$fullURL'" if $? != 0;
|
|
chomp $path;
|
|
$inputs .= '"' . $channelName . '"' . " " . $path . " ";
|
|
}
|
|
$inputs .= "]";
|
|
|
|
# Figure out a name for the GC root.
|
|
my $userName = getpwuid($<);
|
|
die "who ARE you? go away" unless defined $userName;
|
|
|
|
my $rootFile = "$rootsDir/$userName";
|
|
|
|
# Instantiate the Nix expression.
|
|
print "unpacking channel Nix expressions...\n";
|
|
my $storeExpr = `@bindir@/nix-instantiate --add-root '$rootFile'.tmp @datadir@/nix/corepkgs/channels/unpack.nix --argstr system @system@ --arg inputs '$inputs'`
|
|
or die "cannot instantiate Nix expression";
|
|
chomp $storeExpr;
|
|
|
|
# Build the resulting derivation.
|
|
my $outPath = `@bindir@/nix-store --add-root '$rootFile' -r '$storeExpr'`
|
|
or die "cannot realise store expression";
|
|
chomp $outPath;
|
|
|
|
unlink "$rootFile.tmp";
|
|
|
|
# Make it the default Nix expression for `nix-env'.
|
|
system("@bindir@/nix-env", "--import", "$outPath") == 0
|
|
or die "cannot pull set default Nix expression to `$outPath'";
|
|
}
|
|
|
|
|
|
sub usageError {
|
|
print STDERR <<EOF;
|
|
Usage:
|
|
nix-channel --add URL
|
|
nix-channel --remove URL
|
|
nix-channel --list
|
|
nix-channel --update
|
|
EOF
|
|
exit 1;
|
|
}
|
|
|
|
|
|
usageError if scalar @ARGV == 0;
|
|
|
|
|
|
while (scalar @ARGV) {
|
|
my $arg = shift @ARGV;
|
|
|
|
if ($arg eq "--add") {
|
|
usageError if scalar @ARGV != 1;
|
|
addChannel (shift @ARGV);
|
|
last;
|
|
}
|
|
|
|
if ($arg eq "--remove") {
|
|
usageError if scalar @ARGV != 1;
|
|
removeChannel (shift @ARGV);
|
|
last;
|
|
}
|
|
|
|
if ($arg eq "--list") {
|
|
usageError if scalar @ARGV != 0;
|
|
readChannels;
|
|
foreach my $url (@channels) {
|
|
print "$url\n";
|
|
}
|
|
last;
|
|
}
|
|
|
|
elsif ($arg eq "--update") {
|
|
usageError if scalar @ARGV != 0;
|
|
update;
|
|
last;
|
|
}
|
|
|
|
elsif ($arg eq "--help") {
|
|
usageError;
|
|
}
|
|
|
|
else {
|
|
die "unknown argument `$arg'; try `--help'";
|
|
}
|
|
}
|