2019-05-03 06:17:27 +08:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-05-24 01:31:20 +08:00
|
|
|
require "final_destination"
|
2017-06-13 19:27:05 +08:00
|
|
|
require "mini_mime"
|
|
|
|
require "open-uri"
|
2014-04-22 23:11:06 +08:00
|
|
|
|
2014-04-15 04:55:57 +08:00
|
|
|
class FileHelper
|
2017-09-28 07:00:13 +08:00
|
|
|
def self.log(log_level, message)
|
2017-09-28 07:07:43 +08:00
|
|
|
Rails.logger.public_send(
|
|
|
|
log_level,
|
|
|
|
"#{RailsMultisite::ConnectionManagement.current_db}: #{message}",
|
|
|
|
)
|
|
|
|
end
|
2017-09-28 07:00:13 +08:00
|
|
|
|
2018-09-10 10:22:45 +08:00
|
|
|
def self.is_supported_image?(filename)
|
2020-12-30 21:08:02 +08:00
|
|
|
filename.match?(supported_images_regexp)
|
2014-04-15 04:55:57 +08:00
|
|
|
end
|
|
|
|
|
2022-02-14 13:48:27 +08:00
|
|
|
def self.is_supported_video?(filename)
|
|
|
|
filename.match?(supported_video_regexp)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.is_supported_audio?(filename)
|
|
|
|
filename.match?(supported_audio_regexp)
|
|
|
|
end
|
|
|
|
|
2020-07-09 11:31:48 +08:00
|
|
|
def self.is_inline_image?(filename)
|
2020-12-30 21:08:02 +08:00
|
|
|
filename.match?(inline_images_regexp)
|
2020-07-09 11:31:48 +08:00
|
|
|
end
|
|
|
|
|
2025-01-07 10:32:32 +08:00
|
|
|
def self.is_svg?(filename)
|
|
|
|
filename.match?(/\.svg\z/i)
|
|
|
|
end
|
|
|
|
|
2019-11-18 09:25:42 +08:00
|
|
|
def self.is_supported_media?(filename)
|
2020-12-30 21:08:02 +08:00
|
|
|
filename.match?(supported_media_regexp)
|
2019-11-18 09:25:42 +08:00
|
|
|
end
|
|
|
|
|
2022-02-14 13:48:27 +08:00
|
|
|
def self.is_supported_playable_media?(filename)
|
|
|
|
filename.match?(supported_playable_media_regexp)
|
|
|
|
end
|
|
|
|
|
2017-09-28 14:35:27 +08:00
|
|
|
class FakeIO
|
|
|
|
attr_accessor :status
|
|
|
|
end
|
|
|
|
|
2017-05-25 01:46:57 +08:00
|
|
|
def self.download(
|
|
|
|
url,
|
|
|
|
max_file_size:,
|
|
|
|
tmp_file_name:,
|
|
|
|
follow_redirect: false,
|
|
|
|
read_timeout: 5,
|
2017-09-28 09:32:26 +08:00
|
|
|
skip_rate_limit: false,
|
2018-08-17 16:52:55 +08:00
|
|
|
verbose: false,
|
2019-05-28 08:28:57 +08:00
|
|
|
validate_uri: true,
|
2023-08-23 09:18:33 +08:00
|
|
|
retain_on_max_file_size_exceeded: false,
|
2024-05-09 15:11:56 +08:00
|
|
|
include_port_in_host_header: false,
|
|
|
|
extra_headers: {}
|
2018-08-17 16:52:55 +08:00
|
|
|
)
|
2017-05-16 03:32:55 +08:00
|
|
|
url = "https:" + url if url.start_with?("//")
|
2023-01-21 02:52:49 +08:00
|
|
|
raise Discourse::InvalidParameters.new(:url) unless url =~ %r{\Ahttps?://}
|
2014-04-15 04:55:57 +08:00
|
|
|
|
2018-02-24 19:35:57 +08:00
|
|
|
tmp = nil
|
|
|
|
|
|
|
|
fd =
|
|
|
|
FinalDestination.new(
|
2017-05-25 01:46:57 +08:00
|
|
|
url,
|
2019-05-28 08:28:57 +08:00
|
|
|
max_redirects: follow_redirect ? 5 : 0,
|
2017-11-01 00:03:03 +08:00
|
|
|
skip_rate_limit: skip_rate_limit,
|
2019-05-28 08:28:57 +08:00
|
|
|
verbose: verbose,
|
2021-05-03 15:21:11 +08:00
|
|
|
validate_uri: validate_uri,
|
|
|
|
timeout: read_timeout,
|
2023-08-23 09:18:33 +08:00
|
|
|
include_port_in_host_header: include_port_in_host_header,
|
2024-05-09 15:11:56 +08:00
|
|
|
headers: extra_headers,
|
2017-09-28 14:35:27 +08:00
|
|
|
)
|
2017-06-13 19:27:05 +08:00
|
|
|
|
2018-02-24 19:35:57 +08:00
|
|
|
fd.get do |response, chunk, uri|
|
|
|
|
if tmp.nil?
|
|
|
|
# error handling
|
|
|
|
if uri.blank?
|
|
|
|
if response.code.to_i >= 400
|
|
|
|
# attempt error API compatibility
|
|
|
|
io = FakeIO.new
|
|
|
|
io.status = [response.code, ""]
|
2022-06-17 18:50:30 +08:00
|
|
|
raise OpenURI::HTTPError.new("#{response.code} Error", io)
|
2018-02-24 19:35:57 +08:00
|
|
|
else
|
|
|
|
log(:error, "FinalDestination did not work for: #{url}") if verbose
|
|
|
|
throw :done
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-07-30 10:48:44 +08:00
|
|
|
if response.content_type.present?
|
2018-02-24 19:35:57 +08:00
|
|
|
ext = MiniMime.lookup_by_content_type(response.content_type)&.extension
|
|
|
|
ext = "jpg" if ext == "jpe"
|
|
|
|
tmp_file_ext = "." + ext if ext.present?
|
|
|
|
end
|
|
|
|
|
2018-07-30 10:48:44 +08:00
|
|
|
tmp_file_ext ||= File.extname(uri.path)
|
2018-02-24 19:35:57 +08:00
|
|
|
tmp = Tempfile.new([tmp_file_name, tmp_file_ext])
|
|
|
|
tmp.binmode
|
|
|
|
end
|
2018-02-23 01:15:42 +08:00
|
|
|
|
2018-02-24 19:35:57 +08:00
|
|
|
tmp.write(chunk)
|
2014-04-15 04:55:57 +08:00
|
|
|
|
2018-08-17 16:17:58 +08:00
|
|
|
if tmp.size > max_file_size
|
2018-08-17 16:52:55 +08:00
|
|
|
unless retain_on_max_file_size_exceeded
|
|
|
|
tmp.close
|
|
|
|
tmp = nil
|
|
|
|
end
|
|
|
|
|
2018-08-17 16:17:58 +08:00
|
|
|
throw :done
|
|
|
|
end
|
2014-04-15 04:55:57 +08:00
|
|
|
end
|
|
|
|
|
2018-02-24 19:35:57 +08:00
|
|
|
tmp&.rewind
|
2014-04-15 04:55:57 +08:00
|
|
|
tmp
|
|
|
|
end
|
|
|
|
|
2019-01-02 14:19:52 +08:00
|
|
|
def self.optimize_image!(filename, allow_pngquant: false)
|
2019-01-09 09:28:06 +08:00
|
|
|
image_optim(
|
|
|
|
allow_pngquant: allow_pngquant,
|
|
|
|
strip_image_metadata: SiteSetting.strip_image_metadata,
|
|
|
|
).optimize_image!(filename)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.image_optim(allow_pngquant: false, strip_image_metadata: true)
|
|
|
|
# memoization is critical, initializing an ImageOptim object is very expensive
|
|
|
|
# sometimes up to 200ms searching for binaries and looking at versions
|
|
|
|
memoize("image_optim", allow_pngquant, strip_image_metadata) do
|
|
|
|
pngquant_options = false
|
|
|
|
pngquant_options = { allow_lossy: true } if allow_pngquant
|
|
|
|
|
|
|
|
ImageOptim.new(
|
|
|
|
# GLOBAL
|
|
|
|
timeout: 15,
|
|
|
|
skip_missing_workers: true,
|
|
|
|
# PNG
|
2021-11-23 01:16:35 +08:00
|
|
|
oxipng: {
|
|
|
|
level: 3,
|
|
|
|
strip: strip_image_metadata,
|
|
|
|
},
|
|
|
|
optipng: false,
|
2019-01-09 09:28:06 +08:00
|
|
|
advpng: false,
|
|
|
|
pngcrush: false,
|
|
|
|
pngout: false,
|
|
|
|
pngquant: pngquant_options,
|
|
|
|
# JPG
|
|
|
|
jpegoptim: {
|
|
|
|
strip: strip_image_metadata ? "all" : "none",
|
|
|
|
},
|
|
|
|
jpegtran: false,
|
|
|
|
jpegrecompress: false,
|
2021-04-09 20:52:06 +08:00
|
|
|
# Skip looking for gifsicle, svgo binaries
|
|
|
|
gifsicle: false,
|
2021-10-15 05:17:47 +08:00
|
|
|
svgo: false,
|
2019-01-09 09:28:06 +08:00
|
|
|
)
|
2019-01-02 14:19:52 +08:00
|
|
|
end
|
2019-01-09 09:28:06 +08:00
|
|
|
end
|
2019-01-02 14:19:52 +08:00
|
|
|
|
2019-01-09 09:28:06 +08:00
|
|
|
def self.memoize(*args)
|
|
|
|
(@memoized ||= {})[args] ||= yield
|
2017-07-25 17:48:39 +08:00
|
|
|
end
|
|
|
|
|
2019-07-31 11:16:03 +08:00
|
|
|
def self.supported_gravatar_extensions
|
|
|
|
@@supported_gravatar_images ||= Set.new(%w[jpg jpeg png gif])
|
|
|
|
end
|
|
|
|
|
2018-09-10 10:03:44 +08:00
|
|
|
def self.supported_images
|
2023-05-25 03:13:36 +08:00
|
|
|
@@supported_images ||= Set.new %w[jpg jpeg png gif svg ico webp avif]
|
2017-07-25 17:48:39 +08:00
|
|
|
end
|
2014-04-15 04:55:57 +08:00
|
|
|
|
2020-07-09 11:31:48 +08:00
|
|
|
def self.inline_images
|
|
|
|
# SVG cannot safely be shown as a document
|
|
|
|
@@inline_images ||= supported_images - %w[svg]
|
|
|
|
end
|
|
|
|
|
2019-11-18 09:25:42 +08:00
|
|
|
def self.supported_audio
|
2022-02-14 13:48:27 +08:00
|
|
|
@@supported_audio ||= Set.new %w[mp3 ogg oga opus wav m4a m4b m4p m4r aac flac]
|
2019-11-18 09:25:42 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.supported_video
|
2022-02-14 13:48:27 +08:00
|
|
|
@@supported_video ||= Set.new %w[mov mp4 webm ogv m4v 3gp avi mpeg]
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.supported_video_regexp
|
2023-01-21 02:52:49 +08:00
|
|
|
@@supported_video_regexp ||= /\.(#{supported_video.to_a.join("|")})\z/i
|
2022-02-14 13:48:27 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.supported_audio_regexp
|
2023-01-21 02:52:49 +08:00
|
|
|
@@supported_audio_regexp ||= /\.(#{supported_audio.to_a.join("|")})\z/i
|
2019-11-18 09:25:42 +08:00
|
|
|
end
|
|
|
|
|
2018-09-10 10:22:45 +08:00
|
|
|
def self.supported_images_regexp
|
2023-01-21 02:52:49 +08:00
|
|
|
@@supported_images_regexp ||= /\.(#{supported_images.to_a.join("|")})\z/i
|
2017-07-25 17:48:39 +08:00
|
|
|
end
|
2014-04-15 04:55:57 +08:00
|
|
|
|
2020-07-09 11:31:48 +08:00
|
|
|
def self.inline_images_regexp
|
2023-01-21 02:52:49 +08:00
|
|
|
@@inline_images_regexp ||= /\.(#{inline_images.to_a.join("|")})\z/i
|
2020-07-09 11:31:48 +08:00
|
|
|
end
|
|
|
|
|
2019-11-18 09:25:42 +08:00
|
|
|
def self.supported_media_regexp
|
2020-12-30 21:08:02 +08:00
|
|
|
@@supported_media_regexp ||=
|
|
|
|
begin
|
|
|
|
media = supported_images | supported_audio | supported_video
|
2023-01-21 02:52:49 +08:00
|
|
|
/\.(#{media.to_a.join("|")})\z/i
|
2020-12-30 21:08:02 +08:00
|
|
|
end
|
2019-11-18 09:25:42 +08:00
|
|
|
end
|
2022-02-14 13:48:27 +08:00
|
|
|
|
|
|
|
def self.supported_playable_media_regexp
|
|
|
|
@@supported_playable_media_regexp ||=
|
|
|
|
begin
|
|
|
|
media = supported_audio | supported_video
|
2023-01-21 02:52:49 +08:00
|
|
|
/\.(#{media.to_a.join("|")})\z/i
|
2022-02-14 13:48:27 +08:00
|
|
|
end
|
|
|
|
end
|
2014-04-15 04:55:57 +08:00
|
|
|
end
|