165 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Text
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Text
		
	
	
		
			Executable file
		
	
	
	
	
| #! @shell@ -e
 | |
| 
 | |
| url=$1
 | |
| expHash=$2
 | |
| 
 | |
| binDir=@bindir@
 | |
| if [ -n "$NIX_BIN_DIR" ]; then binDir="$NIX_BIN_DIR"; fi
 | |
| 
 | |
| # needed to make it work on NixOS
 | |
| export PATH=$PATH:@coreutils@
 | |
| 
 | |
| hashType=$NIX_HASH_ALGO
 | |
| if test -z "$hashType"; then
 | |
|     hashType=sha256
 | |
| fi
 | |
| 
 | |
| hashFormat=
 | |
| if test "$hashType" != "md5"; then
 | |
|     hashFormat=--base32
 | |
| fi
 | |
| 
 | |
| if test -z "$url"; then
 | |
|     echo "syntax: nix-prefetch-url URL [EXPECTED-HASH]" >&2
 | |
|     exit 1
 | |
| fi
 | |
| 
 | |
| # Handle escaped characters in the URI.  `+', `=' and `?' are the only
 | |
| # characters that are valid in Nix store path names but have a special
 | |
| # meaning in URIs.
 | |
| name=$(basename "$url" | @sed@ -e 's/%2b/+/g' -e 's/%3d/=/g' -e 's/%3f/\?/g')
 | |
| if test -z "$name"; then echo "invalid url"; exit 1; fi
 | |
| 
 | |
| 
 | |
| # If the hash was given, a file with that hash may already be in the
 | |
| # store.
 | |
| if test -n "$expHash"; then
 | |
|     finalPath=$($binDir/nix-store --print-fixed-path "$hashType" "$expHash" "$name")
 | |
|     if ! $bindir/nix-store --check-validity "$finalPath" 2> /dev/null; then
 | |
|         finalPath=
 | |
|     fi
 | |
|     hash=$expHash
 | |
| fi
 | |
| 
 | |
| 
 | |
| mkTempDir() {
 | |
|     if test -n "$tmpPath"; then return; fi
 | |
|     local i=0
 | |
|     while true; do
 | |
|         if test -z "$TMPDIR"; then TMPDIR=/tmp; fi
 | |
|         tmpPath=$TMPDIR/nix-prefetch-url-$$-$i
 | |
|         if mkdir "$tmpPath"; then break; fi
 | |
|         # !!! to bad we can't check for ENOENT in mkdir, so this check
 | |
|         # is slightly racy (it bombs out if somebody just removed
 | |
|         # $tmpPath...).
 | |
|         if ! test -e "$tmpPath"; then exit 1; fi
 | |
|         i=$((i + 1))
 | |
|     done
 | |
|     trap removeTempDir EXIT SIGINT SIGQUIT
 | |
| }
 | |
| 
 | |
| removeTempDir() {
 | |
|     if test -n "$tmpPath"; then
 | |
|         rm -rf "$tmpPath" || true
 | |
|     fi
 | |
| }
 | |
| 
 | |
| 
 | |
| doDownload() {
 | |
|     @curl@ $cacheFlags --fail --location --max-redirs 20 --disable-epsv \
 | |
|         --cookie-jar $tmpPath/cookies "$url" -o $tmpFile
 | |
| }
 | |
| 
 | |
| 
 | |
| # Hack to support the mirror:// scheme from Nixpkgs.
 | |
| if test "${url:0:9}" = "mirror://"; then
 | |
|     if test -z "$NIXPKGS_ALL"; then
 | |
|         echo "Resolving mirror:// URLs requires Nixpkgs.  Please point \$NIXPKGS_ALL at a Nixpkgs tree." >&2
 | |
|         exit 1
 | |
|     fi
 | |
| 
 | |
|     mkTempDir
 | |
|     nix-build "$NIXPKGS_ALL" -A resolveMirrorURLs --argstr url "$url" -o $tmpPath/urls > /dev/null
 | |
| 
 | |
|     expanded=($(cat $tmpPath/urls))
 | |
|     if test "${#expanded[*]}" = 0; then
 | |
|         echo "$0: cannot resolve $url." >&2
 | |
|         exit 1
 | |
|     fi
 | |
| 
 | |
|     echo "$url expands to ${expanded[*]} (using ${expanded[0]})" >&2
 | |
|     url="${expanded[0]}"
 | |
| fi
 | |
| 
 | |
| 
 | |
| # If we don't know the hash or a file with that hash doesn't exist,
 | |
| # download the file and add it to the store.
 | |
| if test -z "$finalPath"; then
 | |
| 
 | |
|     mkTempDir
 | |
|     tmpFile=$tmpPath/$name
 | |
| 
 | |
|     # Optionally do timestamp-based caching of the download.
 | |
|     # Actually, the only thing that we cache in $NIX_DOWNLOAD_CACHE is
 | |
|     # the hash and the timestamp of the file at $url.  The caching of
 | |
|     # the file *contents* is done in Nix store, where it can be
 | |
|     # garbage-collected independently.
 | |
|     if test -n "$NIX_DOWNLOAD_CACHE"; then
 | |
|         echo -n "$url" > $tmpPath/url
 | |
|         urlHash=$($binDir/nix-hash --type sha256 --base32 --flat $tmpPath/url)
 | |
|         echo "$url" > "$NIX_DOWNLOAD_CACHE/$urlHash.url"
 | |
|         cachedHashFN="$NIX_DOWNLOAD_CACHE/$urlHash.$hashType"
 | |
|         cachedTimestampFN="$NIX_DOWNLOAD_CACHE/$urlHash.stamp"
 | |
|         cacheFlags="--remote-time"
 | |
|         if test -e "$cachedTimestampFN" -a -e "$cachedHashFN"; then
 | |
|             # Only download the file if it is newer than the cached version.
 | |
|             cacheFlags="$cacheFlags --time-cond $cachedTimestampFN"
 | |
|         fi
 | |
|     fi
 | |
| 
 | |
|     # Perform the download.
 | |
|     doDownload
 | |
| 
 | |
|     if test -n "$NIX_DOWNLOAD_CACHE" -a ! -e $tmpFile; then
 | |
|         # Curl didn't create $tmpFile, so apparently there's no newer
 | |
|         # file on the server.
 | |
|         hash=$(cat $cachedHashFN)
 | |
|         finalPath=$($binDir/nix-store --print-fixed-path "$hashType" "$hash" "$name") 
 | |
|         if ! $binDir/nix-store --check-validity "$finalPath" 2> /dev/null; then
 | |
|             echo "cached contents of \`$url' disappeared, redownloading..." >&2
 | |
|             finalPath=
 | |
|             cacheFlags="--remote-time"
 | |
|             doDownload
 | |
|         fi
 | |
|     fi
 | |
| 
 | |
|     if test -z "$finalPath"; then
 | |
| 
 | |
|         # Compute the hash.
 | |
|         hash=$($binDir/nix-hash --type "$hashType" $hashFormat --flat $tmpFile)
 | |
|         if ! test -n "$QUIET"; then echo "hash is $hash" >&2; fi
 | |
| 
 | |
|         if test -n "$NIX_DOWNLOAD_CACHE"; then
 | |
|             echo $hash > $cachedHashFN
 | |
|             touch -r $tmpFile $cachedTimestampFN
 | |
|         fi
 | |
| 
 | |
|         # Add the downloaded file to the Nix store.
 | |
|         finalPath=$($binDir/nix-store --add-fixed "$hashType" $tmpFile)
 | |
| 
 | |
|         if test -n "$expHash" -a "$expHash" != "$hash"; then
 | |
|             echo "hash mismatch for URL \`$url'" >&2
 | |
|             exit 1
 | |
|         fi
 | |
|         
 | |
|     fi
 | |
| fi
 | |
| 
 | |
| 
 | |
| if ! test -n "$QUIET"; then echo "path is $finalPath" >&2; fi
 | |
| 
 | |
| echo $hash
 | |
| 
 | |
| if test -n "$PRINT_PATH"; then
 | |
|     echo $finalPath
 | |
| fi
 |