2019-05-02 18:17:27 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2013-06-16 04:39:48 -04:00
|
|
|
class OptimizedImage < ActiveRecord::Base
|
2019-04-08 16:37:35 -04:00
|
|
|
include HasUrl
|
2013-06-16 04:39:48 -04:00
|
|
|
belongs_to :upload
|
|
|
|
|
2015-05-27 19:03:24 -04:00
|
|
|
# BUMP UP if optimized image algorithm changes
|
2019-01-03 01:07:30 -05:00
|
|
|
VERSION = 2
|
2019-04-08 16:37:35 -04:00
|
|
|
URL_REGEX ||= /(\/optimized\/\dX[\/\.\w]*\/([a-zA-Z0-9]+)[\.\w]*)/
|
2015-05-27 19:03:24 -04:00
|
|
|
|
2018-01-23 16:22:08 -05:00
|
|
|
def self.lock(upload_id, width, height)
|
2020-02-17 23:11:30 -05:00
|
|
|
@hostname ||= Discourse.os_hostname
|
2019-01-04 02:43:53 -05:00
|
|
|
# note, the extra lock here ensures we only optimize one image per machine on webs
|
|
|
|
# this can very easily lead to runaway CPU so slowing it down is beneficial and it is hijacked
|
|
|
|
#
|
|
|
|
# we can not afford this blocking in Sidekiq cause it can lead to starvation
|
|
|
|
if Sidekiq.server?
|
2018-01-23 16:22:08 -05:00
|
|
|
DistributedMutex.synchronize("optimized_image_#{upload_id}_#{width}_#{height}") do
|
|
|
|
yield
|
|
|
|
end
|
2019-01-04 02:43:53 -05:00
|
|
|
else
|
|
|
|
DistributedMutex.synchronize("optimized_image_host_#{@hostname}") do
|
|
|
|
DistributedMutex.synchronize("optimized_image_#{upload_id}_#{width}_#{height}") do
|
|
|
|
yield
|
|
|
|
end
|
|
|
|
end
|
2018-01-23 16:22:08 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-05-22 03:34:33 -04:00
|
|
|
def self.create_for(upload, width, height, opts = {})
|
2013-11-05 13:04:47 -05:00
|
|
|
return unless width > 0 && height > 0
|
2015-06-10 12:15:10 -04:00
|
|
|
return if upload.try(:sha1).blank?
|
2013-07-07 19:39:08 -04:00
|
|
|
|
2018-08-17 00:00:27 -04:00
|
|
|
# no extension so try to guess it
|
|
|
|
if (!upload.extension)
|
|
|
|
upload.fix_image_extension
|
|
|
|
end
|
|
|
|
|
2018-11-06 23:29:14 -05:00
|
|
|
if !upload.extension.match?(IM_DECODERS) && upload.extension != "svg"
|
2018-08-17 00:00:27 -04:00
|
|
|
if !opts[:raise_on_error]
|
|
|
|
# nothing to do ... bad extension, not an image
|
|
|
|
return
|
|
|
|
else
|
|
|
|
raise InvalidAccess
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-09-19 01:49:18 -04:00
|
|
|
# prefer to look up the thumbnail without grabbing any locks
|
|
|
|
thumbnail = find_by(upload_id: upload.id, width: width, height: height)
|
|
|
|
|
|
|
|
# correct bad thumbnail if needed
|
2019-01-03 01:07:30 -05:00
|
|
|
if thumbnail && (thumbnail.url.blank? || thumbnail.version != VERSION)
|
2018-09-19 02:07:29 -04:00
|
|
|
thumbnail.destroy!
|
2018-09-19 01:49:18 -04:00
|
|
|
thumbnail = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
return thumbnail if thumbnail
|
|
|
|
|
2020-05-14 22:44:17 -04:00
|
|
|
# create the thumbnail otherwise
|
|
|
|
original_path = Discourse.store.path_for(upload)
|
|
|
|
|
|
|
|
if original_path.blank?
|
|
|
|
# download is protected with a DistributedMutex
|
|
|
|
external_copy = Discourse.store.download(upload) rescue nil
|
|
|
|
original_path = external_copy.try(:path)
|
|
|
|
end
|
|
|
|
|
2018-01-23 16:22:08 -05:00
|
|
|
lock(upload.id, width, height) do
|
2018-09-19 01:49:18 -04:00
|
|
|
# may have been generated since we got the lock
|
2015-05-12 10:45:33 -04:00
|
|
|
thumbnail = find_by(upload_id: upload.id, width: width, height: height)
|
2013-06-16 20:46:42 -04:00
|
|
|
|
2015-05-12 10:45:33 -04:00
|
|
|
# return the previous thumbnail if any
|
2018-09-19 01:49:18 -04:00
|
|
|
return thumbnail if thumbnail
|
2013-06-16 20:46:42 -04:00
|
|
|
|
2015-05-12 10:45:33 -04:00
|
|
|
if original_path.blank?
|
|
|
|
Rails.logger.error("Could not find file in the store located at url: #{upload.url}")
|
|
|
|
else
|
|
|
|
# create a temp file with the same extension as the original
|
2018-12-14 17:44:38 -05:00
|
|
|
extension = ".#{opts[:format] || upload.extension}"
|
2018-08-16 02:32:36 -04:00
|
|
|
|
|
|
|
if extension.length == 1
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
2015-05-12 10:45:33 -04:00
|
|
|
temp_file = Tempfile.new(["discourse-thumbnail", extension])
|
|
|
|
temp_path = temp_file.path
|
|
|
|
|
2020-10-23 12:38:28 -04:00
|
|
|
target_quality = upload.target_image_quality(original_path, SiteSetting.image_preview_jpg_quality)
|
|
|
|
opts = opts.merge(quality: target_quality) if target_quality
|
|
|
|
|
2018-11-06 23:29:14 -05:00
|
|
|
if upload.extension == "svg"
|
2015-05-12 10:45:33 -04:00
|
|
|
FileUtils.cp(original_path, temp_path)
|
|
|
|
resized = true
|
2016-05-23 10:18:30 -04:00
|
|
|
elsif opts[:crop]
|
|
|
|
resized = crop(original_path, temp_path, width, height, opts)
|
2013-11-05 13:04:47 -05:00
|
|
|
else
|
2015-05-12 10:45:33 -04:00
|
|
|
resized = resize(original_path, temp_path, width, height, opts)
|
2013-11-05 13:04:47 -05:00
|
|
|
end
|
2015-05-12 10:45:33 -04:00
|
|
|
|
|
|
|
if resized
|
|
|
|
thumbnail = OptimizedImage.create!(
|
|
|
|
upload_id: upload.id,
|
2016-09-02 02:50:13 -04:00
|
|
|
sha1: Upload.generate_digest(temp_path),
|
2015-05-12 10:45:33 -04:00
|
|
|
extension: extension,
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
url: "",
|
2019-01-03 01:07:30 -05:00
|
|
|
filesize: File.size(temp_path),
|
|
|
|
version: VERSION
|
2015-05-12 10:45:33 -04:00
|
|
|
)
|
2018-12-14 17:44:38 -05:00
|
|
|
|
2015-05-12 10:45:33 -04:00
|
|
|
# store the optimized image and update its url
|
2015-05-29 07:02:05 -04:00
|
|
|
File.open(temp_path) do |file|
|
2020-01-15 22:50:27 -05:00
|
|
|
url = Discourse.store.store_optimized_image(file, thumbnail, nil, secure: upload.secure?)
|
2015-05-29 07:02:05 -04:00
|
|
|
if url.present?
|
|
|
|
thumbnail.url = url
|
|
|
|
thumbnail.save
|
2020-02-03 12:28:45 -05:00
|
|
|
else
|
|
|
|
Rails.logger.error("Failed to store optimized image of size #{width}x#{height} from url: #{upload.url}\nTemp image path: #{temp_path}")
|
2015-05-29 07:02:05 -04:00
|
|
|
end
|
2015-05-12 10:45:33 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# close && remove temp file
|
|
|
|
temp_file.close!
|
2013-11-05 13:04:47 -05:00
|
|
|
end
|
|
|
|
|
2015-05-12 10:45:33 -04:00
|
|
|
# make sure we remove the cached copy from external stores
|
|
|
|
if Discourse.store.external?
|
2018-03-28 04:20:08 -04:00
|
|
|
external_copy&.close
|
2015-05-12 10:45:33 -04:00
|
|
|
end
|
2013-06-16 20:46:42 -04:00
|
|
|
|
2015-05-12 10:45:33 -04:00
|
|
|
thumbnail
|
2015-02-09 11:00:58 -05:00
|
|
|
end
|
2013-06-16 19:00:25 -04:00
|
|
|
end
|
|
|
|
|
2013-06-21 03:34:02 -04:00
|
|
|
def destroy
|
|
|
|
OptimizedImage.transaction do
|
2019-03-22 04:46:13 -04:00
|
|
|
Discourse.store.remove_optimized_image(self) if self.upload
|
2013-06-21 03:34:02 -04:00
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-05-25 22:32:52 -04:00
|
|
|
def local?
|
|
|
|
!(url =~ /^(https?:)?\/\//)
|
|
|
|
end
|
|
|
|
|
2018-08-27 22:48:43 -04:00
|
|
|
def calculate_filesize
|
|
|
|
path =
|
|
|
|
if local?
|
|
|
|
Discourse.store.path_for(self)
|
|
|
|
else
|
|
|
|
Discourse.store.download(self).path
|
|
|
|
end
|
|
|
|
File.size(path)
|
|
|
|
end
|
|
|
|
|
|
|
|
def filesize
|
|
|
|
if size = read_attribute(:filesize)
|
|
|
|
size
|
|
|
|
else
|
2020-07-06 11:01:29 -04:00
|
|
|
size = calculate_filesize
|
2018-08-27 22:48:43 -04:00
|
|
|
|
|
|
|
write_attribute(:filesize, size)
|
|
|
|
if !new_record?
|
|
|
|
update_columns(filesize: size)
|
|
|
|
end
|
|
|
|
size
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-18 18:16:18 -05:00
|
|
|
def self.safe_path?(path)
|
|
|
|
# this matches instructions which call #to_s
|
|
|
|
path = path.to_s
|
|
|
|
return false if path != File.expand_path(path)
|
2017-01-02 10:28:14 -05:00
|
|
|
return false if path !~ /\A[\w\-\.\/]+\z/m
|
2016-12-18 18:16:18 -05:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.ensure_safe_paths!(*paths)
|
|
|
|
paths.each do |path|
|
|
|
|
raise Discourse::InvalidAccess unless safe_path?(path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-12-21 18:09:00 -05:00
|
|
|
IM_DECODERS ||= /\A(jpe?g|png|ico|gif|webp)\z/i
|
2018-07-25 16:00:04 -04:00
|
|
|
|
2018-08-19 22:18:49 -04:00
|
|
|
def self.prepend_decoder!(path, ext_path = nil, opts = nil)
|
2018-12-18 14:55:09 -05:00
|
|
|
opts ||= {}
|
|
|
|
|
|
|
|
# This logic is a little messy but the result of using mocks for most
|
|
|
|
# of the image tests. The idea here is you shouldn't trust the "original"
|
|
|
|
# path of a file to figure out its extension. However, in certain cases
|
|
|
|
# such as generating the loading upload thumbnail, we force the format,
|
|
|
|
# and this allows us to use the forced format in that case.
|
|
|
|
extension = nil
|
|
|
|
if (opts[:format] && path != ext_path)
|
|
|
|
extension = File.extname(path)[1..-1]
|
|
|
|
else
|
|
|
|
extension = File.extname(opts[:filename] || ext_path || path)[1..-1]
|
|
|
|
end
|
|
|
|
|
2018-08-17 00:00:27 -04:00
|
|
|
raise Discourse::InvalidAccess if !extension || !extension.match?(IM_DECODERS)
|
2018-07-25 17:55:06 -04:00
|
|
|
"#{extension}:#{path}"
|
2018-07-25 16:00:04 -04:00
|
|
|
end
|
|
|
|
|
2017-07-25 05:48:39 -04:00
|
|
|
def self.thumbnail_or_resize
|
|
|
|
SiteSetting.strip_image_metadata ? "thumbnail" : "resize"
|
|
|
|
end
|
2016-12-18 18:16:18 -05:00
|
|
|
|
2015-02-25 09:08:33 -05:00
|
|
|
def self.resize_instructions(from, to, dimensions, opts = {})
|
2016-12-18 18:16:18 -05:00
|
|
|
ensure_safe_paths!(from, to)
|
|
|
|
|
2018-08-16 02:32:36 -04:00
|
|
|
# note FROM my not be named correctly
|
2018-08-19 22:18:49 -04:00
|
|
|
from = prepend_decoder!(from, to, opts)
|
|
|
|
to = prepend_decoder!(to, to, opts)
|
2018-07-25 16:00:04 -04:00
|
|
|
|
2018-12-14 17:44:38 -05:00
|
|
|
instructions = ['convert', "#{from}[0]"]
|
|
|
|
|
|
|
|
if opts[:colors]
|
|
|
|
instructions << "-colors" << opts[:colors].to_s
|
|
|
|
end
|
|
|
|
|
2020-10-23 12:38:28 -04:00
|
|
|
if opts[:quality]
|
|
|
|
instructions << "-quality" << opts[:quality].to_s
|
|
|
|
end
|
|
|
|
|
2014-06-11 10:01:01 -04:00
|
|
|
# NOTE: ORDER is important!
|
2018-12-14 17:44:38 -05:00
|
|
|
instructions.concat(%W{
|
2017-06-22 10:53:49 -04:00
|
|
|
-auto-orient
|
2015-02-25 09:08:33 -05:00
|
|
|
-gravity center
|
|
|
|
-background transparent
|
2017-07-25 05:48:39 -04:00
|
|
|
-#{thumbnail_or_resize} #{dimensions}^
|
2015-02-25 09:08:33 -05:00
|
|
|
-extent #{dimensions}
|
2018-07-17 03:48:59 -04:00
|
|
|
-interpolate catrom
|
2015-02-25 09:08:33 -05:00
|
|
|
-unsharp 2x0.5+0.7+0
|
2017-06-26 17:19:48 -04:00
|
|
|
-interlace none
|
2016-03-07 17:26:28 -05:00
|
|
|
-profile #{File.join(Rails.root, 'vendor', 'data', 'RT_sRGB.icm')}
|
2015-02-25 09:08:33 -05:00
|
|
|
#{to}
|
2018-12-14 17:44:38 -05:00
|
|
|
})
|
2015-02-20 11:24:37 -05:00
|
|
|
end
|
2014-05-22 03:34:33 -04:00
|
|
|
|
2016-05-23 10:18:30 -04:00
|
|
|
def self.crop_instructions(from, to, dimensions, opts = {})
|
2016-12-18 18:16:18 -05:00
|
|
|
ensure_safe_paths!(from, to)
|
|
|
|
|
2018-08-19 22:18:49 -04:00
|
|
|
from = prepend_decoder!(from, to, opts)
|
|
|
|
to = prepend_decoder!(to, to, opts)
|
2018-07-25 16:00:04 -04:00
|
|
|
|
2020-10-23 12:38:28 -04:00
|
|
|
instructions = %W{
|
2016-05-23 10:18:30 -04:00
|
|
|
convert
|
|
|
|
#{from}[0]
|
2017-06-22 10:53:49 -04:00
|
|
|
-auto-orient
|
2016-05-23 10:18:30 -04:00
|
|
|
-gravity north
|
|
|
|
-background transparent
|
2017-07-25 05:48:39 -04:00
|
|
|
-#{thumbnail_or_resize} #{opts[:width]}
|
2016-05-23 10:18:30 -04:00
|
|
|
-crop #{dimensions}+0+0
|
|
|
|
-unsharp 2x0.5+0.7+0
|
2017-06-26 17:19:48 -04:00
|
|
|
-interlace none
|
2016-05-23 10:18:30 -04:00
|
|
|
-profile #{File.join(Rails.root, 'vendor', 'data', 'RT_sRGB.icm')}
|
|
|
|
}
|
2020-10-23 12:38:28 -04:00
|
|
|
|
|
|
|
if opts[:quality]
|
|
|
|
instructions << "-quality" << opts[:quality].to_s
|
|
|
|
end
|
|
|
|
|
|
|
|
instructions << to
|
2016-05-23 10:18:30 -04:00
|
|
|
end
|
|
|
|
|
2015-02-25 09:08:33 -05:00
|
|
|
def self.downsize_instructions(from, to, dimensions, opts = {})
|
2016-12-18 18:16:18 -05:00
|
|
|
ensure_safe_paths!(from, to)
|
|
|
|
|
2018-08-19 22:18:49 -04:00
|
|
|
from = prepend_decoder!(from, to, opts)
|
|
|
|
to = prepend_decoder!(to, to, opts)
|
2018-07-25 16:00:04 -04:00
|
|
|
|
2015-02-25 09:08:33 -05:00
|
|
|
%W{
|
2015-07-22 11:10:42 -04:00
|
|
|
convert
|
2015-02-25 09:08:33 -05:00
|
|
|
#{from}[0]
|
2017-06-22 10:53:49 -04:00
|
|
|
-auto-orient
|
2015-02-25 09:08:33 -05:00
|
|
|
-gravity center
|
|
|
|
-background transparent
|
2017-06-26 17:19:48 -04:00
|
|
|
-interlace none
|
2017-05-10 18:16:57 -04:00
|
|
|
-resize #{dimensions}
|
2016-03-07 17:26:28 -05:00
|
|
|
-profile #{File.join(Rails.root, 'vendor', 'data', 'RT_sRGB.icm')}
|
2015-02-25 09:08:33 -05:00
|
|
|
#{to}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2015-02-21 12:37:37 -05:00
|
|
|
def self.resize(from, to, width, height, opts = {})
|
2015-08-12 12:33:13 -04:00
|
|
|
optimize("resize", from, to, "#{width}x#{height}", opts)
|
2015-02-20 11:24:37 -05:00
|
|
|
end
|
|
|
|
|
2016-05-23 10:18:30 -04:00
|
|
|
def self.crop(from, to, width, height, opts = {})
|
|
|
|
opts[:width] = width
|
|
|
|
optimize("crop", from, to, "#{width}x#{height}", opts)
|
|
|
|
end
|
|
|
|
|
2015-08-12 12:33:13 -04:00
|
|
|
def self.downsize(from, to, dimensions, opts = {})
|
|
|
|
optimize("downsize", from, to, dimensions, opts)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.optimize(operation, from, to, dimensions, opts = {})
|
2015-02-25 09:08:33 -05:00
|
|
|
method_name = "#{operation}_instructions"
|
2019-05-06 21:27:05 -04:00
|
|
|
|
|
|
|
instructions = self.public_send(method_name.to_sym, from, to, dimensions, opts)
|
2018-07-25 22:17:38 -04:00
|
|
|
convert_with(instructions, to, opts)
|
2015-02-25 09:08:33 -05:00
|
|
|
end
|
|
|
|
|
2019-01-02 01:19:52 -05:00
|
|
|
MAX_PNGQUANT_SIZE = 500_000
|
2021-04-11 23:55:54 -04:00
|
|
|
MAX_CONVERT_SECONDS = 20
|
2019-01-02 01:19:52 -05:00
|
|
|
|
2018-07-25 22:17:38 -04:00
|
|
|
def self.convert_with(instructions, to, opts = {})
|
2021-04-11 23:55:54 -04:00
|
|
|
Discourse::Utils.execute_command("nice", "-n", "10", *instructions, timeout: MAX_CONVERT_SECONDS)
|
2019-01-02 01:19:52 -05:00
|
|
|
|
|
|
|
allow_pngquant = to.downcase.ends_with?(".png") && File.size(to) < MAX_PNGQUANT_SIZE
|
|
|
|
FileHelper.optimize_image!(to, allow_pngquant: allow_pngquant)
|
2015-02-20 11:24:37 -05:00
|
|
|
true
|
2018-07-18 02:11:23 -04:00
|
|
|
rescue => e
|
2018-07-25 22:17:38 -04:00
|
|
|
if opts[:raise_on_error]
|
2018-07-25 21:16:14 -04:00
|
|
|
raise e
|
|
|
|
else
|
2018-08-14 23:27:24 -04:00
|
|
|
error = +"Failed to optimize image:"
|
|
|
|
|
|
|
|
if e.message =~ /^convert:([^`]+)/
|
|
|
|
error << $1
|
|
|
|
else
|
|
|
|
error << " unknown reason"
|
|
|
|
end
|
|
|
|
|
2021-06-04 08:13:58 -04:00
|
|
|
Discourse.warn(error, location: to, error_message: e.message, instructions: instructions)
|
2018-07-25 21:16:14 -04:00
|
|
|
false
|
|
|
|
end
|
2015-02-20 11:24:37 -05:00
|
|
|
end
|
2013-06-16 04:39:48 -04:00
|
|
|
end
|
2013-06-16 20:48:58 -04:00
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: optimized_images
|
|
|
|
#
|
2020-07-14 06:50:33 -04:00
|
|
|
# id :integer not null, primary key
|
|
|
|
# sha1 :string(40) not null
|
|
|
|
# extension :string(10) not null
|
|
|
|
# width :integer not null
|
|
|
|
# height :integer not null
|
|
|
|
# upload_id :integer not null
|
|
|
|
# url :string not null
|
|
|
|
# filesize :integer
|
|
|
|
# etag :string
|
|
|
|
# version :integer
|
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
2013-06-16 20:48:58 -04:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
2019-01-11 12:19:23 -05:00
|
|
|
# index_optimized_images_on_etag (etag)
|
2013-06-16 20:48:58 -04:00
|
|
|
# index_optimized_images_on_upload_id (upload_id)
|
|
|
|
# index_optimized_images_on_upload_id_and_width_and_height (upload_id,width,height) UNIQUE
|
|
|
|
#
|