Downloads::File#http_get_streaming: clean up retry logic.

Replace handrolled retry logic with retriable gem (already pulled in by another gem).
This commit is contained in:
evazion
2018-09-18 02:10:20 -05:00
parent 96e89cecfb
commit d3c135ec72
4 changed files with 29 additions and 43 deletions

View File

@@ -50,6 +50,7 @@ gem 'activemodel-serializers-xml'
gem 'ptools' gem 'ptools'
gem 'jquery-rails' gem 'jquery-rails'
gem 'webpacker', '>= 4.0.x' gem 'webpacker', '>= 4.0.x'
gem 'retriable'
# needed for looser jpeg header compat # needed for looser jpeg header compat
gem 'ruby-imagespec', :require => "image_spec", :git => "https://github.com/r888888888/ruby-imagespec.git", :branch => "exif-fixes" gem 'ruby-imagespec', :require => "image_spec", :git => "https://github.com/r888888888/ruby-imagespec.git", :branch => "exif-fixes"

View File

@@ -489,6 +489,7 @@ DEPENDENCIES
rakismet rakismet
recaptcha recaptcha
responders responders
retriable
ruby-imagespec! ruby-imagespec!
ruby-prof ruby-prof
ruby-vips ruby-vips

View File

@@ -2,6 +2,8 @@ module Downloads
class File class File
class Error < Exception ; end class Error < Exception ; end
RETRIABLE_ERRORS = [Errno::ECONNRESET, Errno::ETIMEDOUT, Errno::EIO, Errno::EHOSTUNREACH, Errno::ECONNREFUSED, Timeout::Error, IOError]
attr_reader :data, :options attr_reader :data, :options
attr_accessor :source, :referer attr_accessor :source, :referer
@@ -52,16 +54,15 @@ module Downloads
end end
end end
def download! def download!(tries: 3)
strategy = Sources::Strategies.find(source, referer) strategy = Sources::Strategies.find(source, referer)
output_file = Tempfile.new(binmode: true) output_file = Tempfile.new(binmode: true)
url = self.class.uncached_url(strategy.file_url, strategy.headers)
@data = strategy.data @data = strategy.data
http_get_streaming( Retriable.retriable(on: RETRIABLE_ERRORS, tries: tries, base_interval: 0) do
self.class.uncached_url(strategy.file_url, strategy.headers), http_get_streaming(url, output_file, strategy.headers)
output_file, end
strategy.headers
)
[output_file, strategy] [output_file, strategy]
end end
@@ -74,42 +75,30 @@ module Downloads
end end
def http_get_streaming(src, file, headers = {}, max_size: Danbooru.config.max_file_size) def http_get_streaming(src, file, headers = {}, max_size: Danbooru.config.max_file_size)
tries = 0
url = URI.parse(src) url = URI.parse(src)
while true unless url.is_a?(URI::HTTP) || url.is_a?(URI::HTTPS)
unless url.is_a?(URI::HTTP) || url.is_a?(URI::HTTPS) raise Error.new("URL must be HTTP or HTTPS")
raise Error.new("URL must be HTTP or HTTPS") end
end
validate_local_hosts(url) validate_local_hosts(url)
begin size = 0
size = 0 options = { stream_body: true, timeout: 10, headers: headers }
options = { stream_body: true, timeout: 10, headers: headers }
res = HTTParty.get(url, options.deep_merge(Danbooru.config.httparty_options)) do |chunk| res = HTTParty.get(url, options.deep_merge(Danbooru.config.httparty_options)) do |chunk|
size += chunk.size size += chunk.size
raise Error.new("File is too large (max size: #{max_size})") if size > max_size && max_size > 0 raise Error.new("File is too large (max size: #{max_size})") if size > max_size && max_size > 0
file.write(chunk) file.write(chunk)
end end
if res.success? if res.success?
file.rewind file.rewind
return file return file
else else
raise Error.new("HTTP error code: #{res.code} #{res.message}") raise Error.new("HTTP error code: #{res.code} #{res.message}")
end end
rescue Errno::ECONNRESET, Errno::ETIMEDOUT, Errno::EIO, Errno::EHOSTUNREACH, Errno::ECONNREFUSED, Timeout::Error, IOError => x
tries += 1
if tries < 3
retry
else
raise
end
end
end # while
end # def end # def
end end
end end

View File

@@ -10,14 +10,9 @@ module Downloads
end end
context "that fails" do context "that fails" do
setup do should "retry three times before giving up" do
HTTParty.stubs(:get).raises(Errno::ETIMEDOUT) HTTParty.expects(:get).times(3).raises(Errno::ETIMEDOUT)
end assert_raises(Errno::ETIMEDOUT) { @download.download! }
should "retry three times" do
assert_raises(Errno::ETIMEDOUT) do
@download.http_get_streaming(@source, @tempfile)
end
end end
end end