2017-05-23 13:31:20 -04:00
|
|
|
require "final_destination"
|
2017-06-13 07:27:05 -04:00
|
|
|
require "mini_mime"
|
|
|
|
require "open-uri"
|
2014-04-22 11:11:06 -04:00
|
|
|
|
2014-04-14 16:55:57 -04:00
|
|
|
class FileHelper
|
|
|
|
|
2017-09-27 19:00:13 -04:00
|
|
|
def self.log(log_level, message)
|
2017-09-27 19:07:43 -04:00
|
|
|
Rails.logger.public_send(
|
|
|
|
log_level,
|
|
|
|
"#{RailsMultisite::ConnectionManagement.current_db}: #{message}"
|
|
|
|
)
|
|
|
|
end
|
2017-09-27 19:00:13 -04:00
|
|
|
|
2014-04-14 16:55:57 -04:00
|
|
|
def self.is_image?(filename)
|
|
|
|
filename =~ images_regexp
|
|
|
|
end
|
|
|
|
|
2017-09-28 02:35:27 -04:00
|
|
|
class FakeIO
|
|
|
|
attr_accessor :status
|
|
|
|
end
|
|
|
|
|
2017-05-24 13:46:57 -04:00
|
|
|
def self.download(url,
|
|
|
|
max_file_size:,
|
|
|
|
tmp_file_name:,
|
|
|
|
follow_redirect: false,
|
|
|
|
read_timeout: 5,
|
2017-09-27 21:32:26 -04:00
|
|
|
skip_rate_limit: false,
|
2018-02-22 12:15:42 -05:00
|
|
|
verbose: nil)
|
|
|
|
|
|
|
|
# verbose logging is default while debugging onebox
|
|
|
|
verbose = verbose.nil? ? true : verbose
|
2017-05-24 13:46:57 -04:00
|
|
|
|
2017-05-15 15:32:55 -04:00
|
|
|
url = "https:" + url if url.start_with?("//")
|
2014-05-12 10:57:52 -04:00
|
|
|
raise Discourse::InvalidParameters.new(:url) unless url =~ /^https?:\/\//
|
2014-04-14 16:55:57 -04:00
|
|
|
|
2018-02-22 12:15:42 -05:00
|
|
|
dest = FinalDestination.new(
|
2017-05-24 13:46:57 -04:00
|
|
|
url,
|
|
|
|
max_redirects: follow_redirect ? 5 : 1,
|
2017-10-31 12:03:03 -04:00
|
|
|
skip_rate_limit: skip_rate_limit,
|
|
|
|
verbose: verbose
|
2017-09-28 02:35:27 -04:00
|
|
|
)
|
2018-02-22 12:15:42 -05:00
|
|
|
uri = dest.resolve
|
2017-06-13 07:27:05 -04:00
|
|
|
|
2018-02-22 12:15:42 -05:00
|
|
|
if !uri && dest.status_code.to_i >= 400
|
|
|
|
# attempt error API compatability
|
|
|
|
io = FakeIO.new
|
|
|
|
io.status = [dest.status_code.to_s, ""]
|
|
|
|
|
|
|
|
# TODO perhaps translate and add Discourse::DownloadError
|
|
|
|
raise OpenURI::HTTPError.new("#{dest.status_code} Error", io)
|
|
|
|
end
|
|
|
|
|
|
|
|
unless uri
|
|
|
|
log(:error, "FinalDestination did not work for: #{url}") if verbose
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
downloaded = uri.open("rb", read_timeout: read_timeout)
|
|
|
|
|
|
|
|
extension = File.extname(uri.path)
|
2017-06-13 07:27:05 -04:00
|
|
|
|
2018-02-22 12:15:42 -05:00
|
|
|
if extension.blank? && downloaded.content_type.present?
|
|
|
|
ext = MiniMime.lookup_by_content_type(downloaded.content_type)&.extension
|
|
|
|
ext = "jpg" if ext == "jpe"
|
|
|
|
extension = "." + ext if ext.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
tmp = Tempfile.new([tmp_file_name, extension])
|
2014-04-14 16:55:57 -04:00
|
|
|
|
2018-02-22 12:15:42 -05:00
|
|
|
File.open(tmp.path, "wb") do |f|
|
|
|
|
while f.size <= max_file_size && data = downloaded.read(512.kilobytes)
|
|
|
|
f.write(data)
|
|
|
|
end
|
2014-04-14 16:55:57 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
tmp
|
2018-02-22 12:15:42 -05:00
|
|
|
ensure
|
|
|
|
downloaded&.close
|
2014-04-14 16:55:57 -04:00
|
|
|
end
|
|
|
|
|
2017-07-25 05:48:39 -04:00
|
|
|
def self.optimize_image!(filename)
|
|
|
|
ImageOptim.new(
|
|
|
|
# GLOBAL
|
|
|
|
timeout: 15,
|
|
|
|
skip_missing_workers: true,
|
|
|
|
# PNG
|
|
|
|
optipng: { level: 2, strip: SiteSetting.strip_image_metadata },
|
|
|
|
advpng: false,
|
|
|
|
pngcrush: false,
|
|
|
|
pngout: false,
|
|
|
|
pngquant: false,
|
|
|
|
# JPG
|
|
|
|
jpegoptim: { strip: SiteSetting.strip_image_metadata ? "all" : "none" },
|
|
|
|
jpegtran: false,
|
|
|
|
jpegrecompress: false,
|
|
|
|
).optimize_image!(filename)
|
|
|
|
end
|
|
|
|
|
2014-04-14 16:55:57 -04:00
|
|
|
private
|
|
|
|
|
2017-07-25 05:48:39 -04:00
|
|
|
def self.images
|
|
|
|
@@images ||= Set.new %w{jpg jpeg png gif tif tiff bmp svg webp ico}
|
|
|
|
end
|
2014-04-14 16:55:57 -04:00
|
|
|
|
2017-07-25 05:48:39 -04:00
|
|
|
def self.images_regexp
|
|
|
|
@@images_regexp ||= /\.(#{images.to_a.join("|")})$/i
|
|
|
|
end
|
2014-04-14 16:55:57 -04:00
|
|
|
|
|
|
|
end
|