-
Notifications
You must be signed in to change notification settings - Fork 10
/
amahi-download
executable file
·173 lines (140 loc) · 4.73 KB
/
amahi-download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
#! /usr/bin/ruby
# Download files from Amahi mirrors, with integrity checking
#
#
# FIXME: add support for mirrors in the future
require 'optparse'
require 'digest'
require 'pp'
require 'fileutils'
require 'uri'
require 'net/http'
require 'digest/sha1'
# NOTE: this location should be kept in sync at all times with the platform
# at [platform]/lib/downloader.rb in production)
AMAHI_DOWNLOAD_CACHE = "/var/hda/tmp/amahi-download-cache"
HDA_TMP_DIR = '/var/hda/tmp'
DEFAULT_URL = "http://dl.amahi.org"
# utilities to manage a temporary cache of downloaded or generated files
class TempCache
class << self
# expire files that have not been accessed in a while
def expire_unused_files
Dir.glob(File.join(HDA_TMP_DIR, "**/**")) do |f|
begin
if File.exists?(f) && File.atime(f) < 3.months.ago
FileUtils.rm_rf(f)
end
rescue
# ignore errors, because it's theoretically possible that
# there might be files in-flight and the exists? is true
# yet the atime fails, or even the rm
end
end
end
# return a unique name for creating a file
def unique_filename(base)
expire_unused_files
File.join(HDA_TMP_DIR, "#{base}-%d.%d" % [$$, rand(9999)])
end
end
end
class Downloader
HDA_DOWNLOAD_CACHE = File.join(HDA_TMP_DIR, "amahi-download-cache")
AMAHI_DOWNLOAD_MIRROR_SITE = 'http://mirror.amahi.org'
class SHA1VerificationFailed < Exception; end
class TooManyRedirects < Exception; end
# download a file, but check in the cache first, also
# check the sha1
def self.download_and_check_sha1(url, sha1)
raise SHA1VerificationFailed, "#{url}, sha1sum provided is empty!" unless sha1
TempCache.expire_unused_files
FileUtils.mkdir_p(HDA_DOWNLOAD_CACHE)
cached_filename = File.join(HDA_DOWNLOAD_CACHE, sha1)
if File.exists?(cached_filename)
file = IO.binread(cached_filename)
new_sha1 = Digest::SHA1.hexdigest(file)
if new_sha1 == sha1
puts "file #{cached_filename} picked up from cache."
FileUtils.touch cached_filename
# return the file name, not the data
return cached_filename
else
puts "WARNING: file #{cached_filename} in cache was found to be corrupted! Discarding it."
end
end
# download if the above fails, i.e. no cached file OR the sha1sum failed!
download(url, cached_filename, sha1)
# return the file name, not the data
cached_filename
end
private
def self.download_direct(url)
redirect_limit = 5
while redirect_limit > 0
u = URI.parse(url)
http = Net::HTTP.new(u.host, u.port)
http.use_ssl = (u.scheme == 'https')
request = Net::HTTP::Get.new(u.path, { 'accept-encoding' => '' })
response = http.start { |http| http.request(request) }
return response.body unless response.kind_of?(Net::HTTPRedirection)
# it's a redirection
location = response['location']
old_url = url
url = location.nil? ? (response.body.match(/<a href=\"([^>]+)\">/i)[1]) : location
puts "NOTICE: redirected '#{old_url}' --> '#{url}' ..."
redirect_limit -= 1
end
# reached end of redirect limit]!
raise TooManyRedirects, "#{url}"
end
# try to download from the original download site. if that fails,
# then try to download from amahi's mirror
def self.download(url, filename, sha1)
new_sha1 = "badbadsha1"
begin
file = download_direct(url)
open(filename, "wb") {|f| f.write file }
puts "file #{filename} written in cache"
new_sha1 = Digest::SHA1.hexdigest(file)
rescue => e
puts "WARNING: primary downloaded of #{url} barfed with exception \"#{e.inspect}\""
end
if new_sha1 != sha1
puts "WARNING: primary downloaded file #{filename} did not pass signature check - got #{new_sha1}, expected #{sha1}"
new_url = File.join(AMAHI_DOWNLOAD_MIRROR_SITE, sha1)
file = download_direct(new_url)
open(filename, "wb") { |f| f.write file }
puts "new file #{filename} from Amahi's mirror written in the cache"
new_sha1 = Digest::SHA1.hexdigest(file)
raise SHA1VerificationFailed, "#{new_url} (from original #{url}), '#{new_sha1}' vs. '#{sha1}' " if new_sha1 != sha1
end
end
end
def parse
options = {}
ARGV.clone.options do |opts|
script_name = File.basename($0)
opts.banner = "Usage: #{$0} [options] <filename|url> <sha1sum>"
#opts.separator ""
opts.on("-h", "--help",
"Show this help message.") { $stderr.puts opts; exit -1 }
opts.parse!
if ARGV.size < 2
$stderr.puts opts.banner
exit -1
end
end
ARGV
end
def main
(filename, sha1) = parse
url = filename
url = File.join(DEFAULT_URL, filename) unless filename =~ /^(http:|https:|ftp:)/
cached_filename = Downloader.download_and_check_sha1(url, sha1)
local_filename = File.basename((URI::split url)[5])
# remove the link first, in case it's a stale link
FileUtils.rm_rf(local_filename)
File.symlink(cached_filename, local_filename)
end
main