123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372 |
- #!/usr/bin/env perl
- #
- # Copyright (C) 2006 OpenWrt.org
- # Copyright (C) 2016 LEDE project
- #
- # This is free software, licensed under the GNU General Public License v2.
- # See /LICENSE for more information.
- #
- use strict;
- use warnings;
- use File::Basename;
- use File::Copy;
- use Text::ParseWords;
- @ARGV > 2 or die "Syntax: $0 <target dir> <filename> <hash> <url filename> [<mirror> ...]\n";
- my $url_filename;
- my $target = glob(shift @ARGV);
- my $filename = shift @ARGV;
- my $file_hash = shift @ARGV;
- $url_filename = shift @ARGV unless $ARGV[0] =~ /:\/\//;
- my $scriptdir = dirname($0);
- my @mirrors;
- my $ok;
- my $check_certificate = $ENV{DOWNLOAD_CHECK_CERTIFICATE} eq "y";
- my $custom_tool = $ENV{DOWNLOAD_TOOL_CUSTOM};
- my $download_tool;
- $url_filename or $url_filename = $filename;
- sub localmirrors {
- my @mlist;
- open LM, "$scriptdir/localmirrors" and do {
- while (<LM>) {
- chomp $_;
- push @mlist, $_ if $_;
- }
- close LM;
- };
- open CONFIG, "<".$ENV{'TOPDIR'}."/.config" and do {
- while (<CONFIG>) {
- /^CONFIG_LOCALMIRROR="(.+)"/ and do {
- chomp;
- my @local_mirrors = split(/;/, $1);
- push @mlist, @local_mirrors;
- };
- }
- close CONFIG;
- };
- my $mirror = $ENV{'DOWNLOAD_MIRROR'};
- $mirror and push @mlist, split(/;/, $mirror);
- return @mlist;
- }
- sub which($) {
- my $prog = shift;
- my $res = `command -v $prog`;
- $res or return undef;
- return $res;
- }
- sub hash_cmd() {
- my $len = length($file_hash);
- my $cmd;
- $len == 64 and return "$ENV{'MKHASH'} sha256";
- $len == 32 and return "$ENV{'MKHASH'} md5";
- return undef;
- }
- sub tool_present {
- my $tool_name = shift;
- my $compare_line = shift;
- my $present = 0;
- if (open TOOL, "$tool_name --version 2>/dev/null |") {
- if (defined(my $line = readline TOOL)) {
- $present = 1 if $line =~ /^$compare_line /;
- }
- close TOOL;
- }
- return $present
- }
- sub select_tool {
- $custom_tool =~ tr/"//d;
- if ($custom_tool) {
- return $custom_tool;
- }
- # Try to use curl if available
- if (tool_present("curl", "curl")) {
- return "curl";
- }
- # No tool found, fallback to wget
- return "wget";
- }
- sub download_cmd {
- my $url = shift;
- my $filename = shift;
- if ($download_tool eq "curl") {
- return (qw(curl -f --connect-timeout 20 --retry 5 --location),
- $check_certificate ? () : '--insecure',
- shellwords($ENV{CURL_OPTIONS} || ''),
- $url);
- } elsif ($download_tool eq "wget") {
- return (qw(wget --tries=5 --timeout=20 --output-document=-),
- $check_certificate ? () : '--no-check-certificate',
- shellwords($ENV{WGET_OPTIONS} || ''),
- $url);
- } elsif ($download_tool eq "aria2c") {
- my $additional_mirrors = join(" ", map "$_/$filename", @_);
- my @chArray = ('a'..'z', 'A'..'Z', 0..9);
- my $rfn = join '', "${filename}_", map{ $chArray[int rand @chArray] } 0..9;
- @mirrors=();
- return join(" ", "[ -d $ENV{'TMPDIR'}/aria2c ] || mkdir $ENV{'TMPDIR'}/aria2c;",
- "touch $ENV{'TMPDIR'}/aria2c/${rfn}_spp;",
- qw(aria2c --stderr -c -x2 -s10 -j10 -k1M), $url, $additional_mirrors,
- $check_certificate ? () : '--check-certificate=false',
- "--server-stat-of=$ENV{'TMPDIR'}/aria2c/${rfn}_spp",
- "--server-stat-if=$ENV{'TMPDIR'}/aria2c/${rfn}_spp",
- "--daemon=false --no-conf", shellwords($ENV{ARIA2C_OPTIONS} || ''),
- "-d $ENV{'TMPDIR'}/aria2c -o $rfn;",
- "cat $ENV{'TMPDIR'}/aria2c/$rfn;",
- "rm $ENV{'TMPDIR'}/aria2c/$rfn $ENV{'TMPDIR'}/aria2c/${rfn}_spp");
- } else {
- return join(" ", $download_tool, $url);
- }
- }
- my $hash_cmd = hash_cmd();
- $hash_cmd or ($file_hash eq "skip") or die "Cannot find appropriate hash command, ensure the provided hash is either a MD5 or SHA256 checksum.\n";
- sub download
- {
- my $mirror = shift;
- my $download_filename = shift;
- my @additional_mirrors = @_;
- $mirror =~ s!/$!!;
- if ($mirror =~ s!^file://!!) {
- if (! -d "$mirror") {
- print STDERR "Wrong local cache directory -$mirror-.\n";
- cleanup();
- return;
- }
- if (! -d "$target") {
- system("mkdir", "-p", "$target/");
- }
- if (! open TMPDLS, "find $mirror -follow -name $filename 2>/dev/null |") {
- print("Failed to search for $filename in $mirror\n");
- return;
- }
- my $link;
- while (defined(my $line = readline TMPDLS)) {
- chomp ($link = $line);
- if ($. > 1) {
- print("$. or more instances of $filename in $mirror found . Only one instance allowed.\n");
- return;
- }
- }
- close TMPDLS;
- if (! $link) {
- print("No instances of $filename found in $mirror.\n");
- return;
- }
- print("Copying $filename from $link\n");
- copy($link, "$target/$filename.dl");
- $hash_cmd and do {
- if (system("cat '$target/$filename.dl' | $hash_cmd > '$target/$filename.hash'")) {
- print("Failed to generate hash for $filename\n");
- return;
- }
- };
- } else {
- my @cmd = download_cmd("$mirror/$download_filename", $download_filename, @additional_mirrors);
- print STDERR "+ ".join(" ",@cmd)."\n";
- open(FETCH_FD, '-|', @cmd) or die "Cannot launch aria2c, curl or wget.\n";
- $hash_cmd and do {
- open MD5SUM, "| $hash_cmd > '$target/$filename.hash'" or die "Cannot launch $hash_cmd.\n";
- };
- open OUTPUT, "> $target/$filename.dl" or die "Cannot create file $target/$filename.dl: $!\n";
- my $buffer;
- while (read FETCH_FD, $buffer, 1048576) {
- $hash_cmd and print MD5SUM $buffer;
- print OUTPUT $buffer;
- }
- $hash_cmd and close MD5SUM;
- close FETCH_FD;
- close OUTPUT;
- if ($? >> 8) {
- print STDERR "Download failed.\n";
- cleanup();
- return;
- }
- }
- $hash_cmd and do {
- my $sum = `cat "$target/$filename.hash"`;
- $sum =~ /^(\w+)\s*/ or die "Could not generate file hash\n";
- $sum = $1;
- if ($sum ne $file_hash) {
- print STDERR "Hash of the downloaded file does not match (file: $sum, requested: $file_hash) - deleting download.\n";
- cleanup();
- return;
- }
- };
- unlink "$target/$filename";
- system("mv", "$target/$filename.dl", "$target/$filename");
- cleanup();
- }
- sub cleanup
- {
- unlink "$target/$filename.dl";
- unlink "$target/$filename.hash";
- }
- @mirrors = localmirrors();
- foreach my $mirror (@ARGV) {
- if ($mirror =~ /^\@SF\/(.+)$/) {
- # give sourceforge a few more tries, because it redirects to different mirrors
- for (1 .. 5) {
- push @mirrors, "https://downloads.sourceforge.net/$1";
- }
- } elsif ($mirror =~ /^\@LIBRECMC$/) {
- # use libreCMC source server directly
- } elsif ($mirror =~ /^\@DEBIAN\/(.+)$/) {
- push @mirrors, "https://ftp.debian.org/debian/$1";
- push @mirrors, "https://mirror.leaseweb.com/debian/$1";
- push @mirrors, "https://mirror.netcologne.de/debian/$1";
- push @mirrors, "https://mirrors.tuna.tsinghua.edu.cn/debian/$1";
- push @mirrors, "https://mirrors.ustc.edu.cn/debian/$1"
- } elsif ($mirror =~ /^\@APACHE\/(.+)$/) {
- push @mirrors, "https://dlcdn.apache.org/$1";
- push @mirrors, "https://mirror.netcologne.de/apache.org/$1";
- push @mirrors, "https://mirror.aarnet.edu.au/pub/apache/$1";
- push @mirrors, "https://mirror.csclub.uwaterloo.ca/apache/$1";
- push @mirrors, "https://archive.apache.org/dist/$1";
- push @mirrors, "http://mirror.cogentco.com/pub/apache/$1";
- push @mirrors, "http://mirror.navercorp.com/apache/$1";
- push @mirrors, "http://ftp.jaist.ac.jp/pub/apache/$1";
- push @mirrors, "ftp://apache.cs.utah.edu/apache.org/$1";
- push @mirrors, "ftp://apache.mirrors.ovh.net/ftp.apache.org/dist/$1";
- push @mirrors, "https://mirrors.tuna.tsinghua.edu.cn/apache/$1";
- push @mirrors, "https://mirrors.ustc.edu.cn/apache/$1";
- } elsif ($mirror =~ /^\@GITHUB\/(.+)$/) {
- # give github a few more tries (different mirrors)
- for (1 .. 5) {
- push @mirrors, "https://raw.githubusercontent.com/$1";
- }
- } elsif ($mirror =~ /^\@GNU\/(.+)$/) {
- push @mirrors, "https://mirror.csclub.uwaterloo.ca/gnu/$1";
- push @mirrors, "https://mirror.netcologne.de/gnu/$1";
- push @mirrors, "http://ftp.kddilabs.jp/GNU/gnu/$1";
- push @mirrors, "http://www.nic.funet.fi/pub/gnu/gnu/$1";
- push @mirrors, "http://mirror.internode.on.net/pub/gnu/$1";
- push @mirrors, "http://mirror.navercorp.com/gnu/$1";
- push @mirrors, "ftp://mirrors.rit.edu/gnu/$1";
- push @mirrors, "ftp://download.xs4all.nl/pub/gnu/$1";
- push @mirrors, "https://ftp.gnu.org/gnu/$1";
- push @mirrors, "https://mirrors.tuna.tsinghua.edu.cn/gnu/$1";
- push @mirrors, "https://mirrors.ustc.edu.cn/gnu/$1";
- } elsif ($mirror =~ /^\@SAVANNAH\/(.+)$/) {
- push @mirrors, "https://mirror.netcologne.de/savannah/$1";
- push @mirrors, "https://mirror.csclub.uwaterloo.ca/nongnu/$1";
- push @mirrors, "http://ftp.acc.umu.se/mirror/gnu.org/savannah/$1";
- push @mirrors, "http://nongnu.uib.no/$1";
- push @mirrors, "http://ftp.igh.cnrs.fr/pub/nongnu/$1";
- push @mirrors, "ftp://cdimage.debian.org/mirror/gnu.org/savannah/$1";
- push @mirrors, "ftp://ftp.acc.umu.se/mirror/gnu.org/savannah/$1";
- } elsif ($mirror =~ /^\@KERNEL\/(.+)$/) {
- my @extra = ( $1 );
- if ($filename =~ /linux-\d+\.\d+(?:\.\d+)?-rc/) {
- push @extra, "$extra[0]/testing";
- } elsif ($filename =~ /linux-(\d+\.\d+(?:\.\d+)?)/) {
- push @extra, "$extra[0]/longterm/v$1";
- }
- foreach my $dir (@extra) {
- push @mirrors, "https://cdn.kernel.org/pub/$dir";
- push @mirrors, "https://download.xs4all.nl/ftp.kernel.org/pub/$dir";
- push @mirrors, "https://mirrors.mit.edu/kernel/$dir";
- push @mirrors, "http://ftp.nara.wide.ad.jp/pub/kernel.org/$dir";
- push @mirrors, "http://www.ring.gr.jp/archives/linux/kernel.org/$dir";
- push @mirrors, "ftp://ftp.riken.jp/Linux/kernel.org/$dir";
- push @mirrors, "ftp://www.mirrorservice.org/sites/ftp.kernel.org/pub/$dir";
- push @mirrors, "https://mirrors.tuna.tsinghua.edu.cn/kernel/$dir";
- push @mirrors, "https://mirrors.ustc.edu.cn/kernel.org/$dir";
- }
- } elsif ($mirror =~ /^\@KERNEL_LIBRE\/(.+)$/) {
- my @extra = ( $1 );
- if ($filename =~ /linux-libre-\d+\.\d+(?:\.\d+)?-rc-gnu/) {
- push @extra, "$extra[0]/testing";
- } elsif ($filename =~ /linux-libre-(\d+\.\d+(?:\.\d+)?)-gnu/) {
- push @extra, "$extra[0]/v$1";
- }
- foreach my $dir (@extra) {
- push @mirrors, "https://linux-libre.fsfla.org/pub/linux-libre/releases/$dir";
- push @mirrors, "https://librecmc.org/pub/linux-libre/releases/$dir";
- }
- } elsif ($mirror =~ /^\@GNOME\/(.+)$/) {
- push @mirrors, "https://download.gnome.org/sources/$1";
- push @mirrors, "https://mirror.csclub.uwaterloo.ca/gnome/sources/$1";
- push @mirrors, "http://ftp.acc.umu.se/pub/GNOME/sources/$1";
- push @mirrors, "http://ftp.kaist.ac.kr/gnome/sources/$1";
- push @mirrors, "http://www.mirrorservice.org/sites/ftp.gnome.org/pub/GNOME/sources/$1";
- push @mirrors, "http://mirror.internode.on.net/pub/gnome/sources/$1";
- push @mirrors, "http://ftp.belnet.be/ftp.gnome.org/sources/$1";
- push @mirrors, "ftp://ftp.cse.buffalo.edu/pub/Gnome/sources/$1";
- push @mirrors, "ftp://ftp.nara.wide.ad.jp/pub/X11/GNOME/sources/$1";
- push @mirrors, "https://mirrors.ustc.edu.cn/gnome/sources/$1";
- } else {
- push @mirrors, $mirror;
- }
- }
- push @mirrors, 'https://librecmc.org/librecmc/downloads/sources/v6.0';
- if (-f "$target/$filename") {
- $hash_cmd and do {
- if (system("cat '$target/$filename' | $hash_cmd > '$target/$filename.hash'")) {
- die "Failed to generate hash for $filename\n";
- }
- my $sum = `cat "$target/$filename.hash"`;
- $sum =~ /^(\w+)\s*/ or die "Could not generate file hash\n";
- $sum = $1;
- cleanup();
- exit 0 if $sum eq $file_hash;
- die "Hash of the local file $filename does not match (file: $sum, requested: $file_hash) - deleting download.\n";
- unlink "$target/$filename";
- };
- }
- $download_tool = select_tool();
- while (!-f "$target/$filename") {
- my $mirror = shift @mirrors;
- $mirror or die "No more mirrors to try - giving up.\n";
- download($mirror, $url_filename, @mirrors);
- if (!-f "$target/$filename" && $url_filename ne $filename) {
- download($mirror, $filename, @mirrors);
- }
- }
- $SIG{INT} = \&cleanup;
|