2019-05-02 18:17:27 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
require "digest/sha1"
|
2013-02-05 14:16:51 -05:00
|
|
|
|
|
|
|
class Upload < ActiveRecord::Base
|
2019-02-20 21:13:37 -05:00
|
|
|
include ActionView::Helpers::NumberHelper
|
2019-04-08 16:37:35 -04:00
|
|
|
include HasUrl
|
2019-02-20 21:13:37 -05:00
|
|
|
|
2018-09-09 22:10:39 -04:00
|
|
|
SHA1_LENGTH = 40
|
2019-01-02 02:29:17 -05:00
|
|
|
SEEDED_ID_THRESHOLD = 0
|
2019-04-08 16:37:35 -04:00
|
|
|
URL_REGEX ||= /(\/original\/\dX[\/\.\w]*\/([a-zA-Z0-9]+)[\.\w]*)/
|
2018-09-09 22:10:39 -04:00
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
belongs_to :user
|
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
has_many :post_uploads, dependent: :destroy
|
2013-06-13 17:44:24 -04:00
|
|
|
has_many :posts, through: :post_uploads
|
2013-06-12 19:43:50 -04:00
|
|
|
|
2013-06-21 03:34:02 -04:00
|
|
|
has_many :optimized_images, dependent: :destroy
|
2018-09-20 01:33:10 -04:00
|
|
|
has_many :user_uploads, dependent: :destroy
|
2013-06-16 04:39:48 -04:00
|
|
|
|
2017-06-12 16:41:29 -04:00
|
|
|
attr_accessor :for_group_message
|
2017-05-09 17:20:28 -04:00
|
|
|
attr_accessor :for_theme
|
2017-06-12 16:41:29 -04:00
|
|
|
attr_accessor :for_private_message
|
2018-04-19 07:30:31 -04:00
|
|
|
attr_accessor :for_export
|
2018-11-14 02:03:02 -05:00
|
|
|
attr_accessor :for_site_setting
|
2019-07-30 23:16:03 -04:00
|
|
|
attr_accessor :for_gravatar
|
2016-02-29 16:39:24 -05:00
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
validates_presence_of :filesize
|
|
|
|
validates_presence_of :original_filename
|
|
|
|
|
2019-10-02 00:01:53 -04:00
|
|
|
validates_with UploadValidator
|
2014-04-14 16:55:57 -04:00
|
|
|
|
2018-08-31 00:46:22 -04:00
|
|
|
after_destroy do
|
|
|
|
User.where(uploaded_avatar_id: self.id).update_all(uploaded_avatar_id: nil)
|
|
|
|
UserAvatar.where(gravatar_upload_id: self.id).update_all(gravatar_upload_id: nil)
|
|
|
|
UserAvatar.where(custom_upload_id: self.id).update_all(custom_upload_id: nil)
|
|
|
|
end
|
|
|
|
|
2019-01-02 02:29:17 -05:00
|
|
|
scope :by_users, -> { where("uploads.id > ?", SEEDED_ID_THRESHOLD) }
|
|
|
|
|
2018-11-14 02:03:02 -05:00
|
|
|
def to_s
|
|
|
|
self.url
|
|
|
|
end
|
|
|
|
|
2018-08-27 22:48:43 -04:00
|
|
|
def thumbnail(width = self.thumbnail_width, height = self.thumbnail_height)
|
2014-05-06 09:41:59 -04:00
|
|
|
optimized_images.find_by(width: width, height: height)
|
2013-06-16 19:00:25 -04:00
|
|
|
end
|
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def has_thumbnail?(width, height)
|
2013-09-27 04:55:50 -04:00
|
|
|
thumbnail(width, height).present?
|
2013-06-16 19:00:25 -04:00
|
|
|
end
|
|
|
|
|
2018-12-14 16:50:28 -05:00
|
|
|
def create_thumbnail!(width, height, opts = nil)
|
2013-06-16 19:00:25 -04:00
|
|
|
return unless SiteSetting.create_thumbnails?
|
2018-12-14 16:50:28 -05:00
|
|
|
opts ||= {}
|
|
|
|
opts[:allow_animation] = SiteSetting.allow_animated_thumbnails
|
2015-09-20 16:01:03 -04:00
|
|
|
|
2018-08-17 00:00:27 -04:00
|
|
|
if get_optimized_image(width, height, opts)
|
2015-06-26 19:26:16 -04:00
|
|
|
save(validate: false)
|
2013-09-27 04:55:50 -04:00
|
|
|
end
|
2013-06-16 19:00:25 -04:00
|
|
|
end
|
|
|
|
|
2018-08-17 00:00:27 -04:00
|
|
|
# this method attempts to correct old incorrect extensions
|
|
|
|
def get_optimized_image(width, height, opts)
|
|
|
|
if (!extension || extension.length == 0)
|
|
|
|
fix_image_extension
|
|
|
|
end
|
|
|
|
|
|
|
|
opts = opts.merge(raise_on_error: true)
|
|
|
|
begin
|
|
|
|
OptimizedImage.create_for(self, width, height, opts)
|
2018-12-14 17:44:38 -05:00
|
|
|
rescue => ex
|
|
|
|
Rails.logger.info ex if Rails.env.development?
|
2018-08-17 00:00:27 -04:00
|
|
|
opts = opts.merge(raise_on_error: false)
|
|
|
|
if fix_image_extension
|
|
|
|
OptimizedImage.create_for(self, width, height, opts)
|
|
|
|
else
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def fix_image_extension
|
|
|
|
return false if extension == "unknown"
|
|
|
|
|
|
|
|
begin
|
|
|
|
# this is relatively cheap once cached
|
|
|
|
original_path = Discourse.store.path_for(self)
|
|
|
|
if original_path.blank?
|
|
|
|
external_copy = Discourse.store.download(self) rescue nil
|
|
|
|
original_path = external_copy.try(:path)
|
|
|
|
end
|
|
|
|
|
|
|
|
image_info = FastImage.new(original_path) rescue nil
|
|
|
|
new_extension = image_info&.type&.to_s || "unknown"
|
|
|
|
|
|
|
|
if new_extension != self.extension
|
|
|
|
self.update_columns(extension: new_extension)
|
|
|
|
true
|
|
|
|
end
|
|
|
|
rescue
|
|
|
|
self.update_columns(extension: "unknown")
|
|
|
|
true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-06-21 03:34:02 -04:00
|
|
|
def destroy
|
2013-06-19 15:51:41 -04:00
|
|
|
Upload.transaction do
|
2013-08-13 16:08:29 -04:00
|
|
|
Discourse.store.remove_upload(self)
|
2013-06-19 15:51:41 -04:00
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-22 11:46:15 -04:00
|
|
|
def short_url
|
2019-05-28 21:00:25 -04:00
|
|
|
"upload://#{short_url_basename}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def short_path
|
|
|
|
self.class.short_path(sha1: self.sha1, extension: self.extension)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.short_path(sha1:, extension:)
|
|
|
|
@url_helpers ||= Rails.application.routes.url_helpers
|
|
|
|
|
|
|
|
@url_helpers.upload_short_path(
|
|
|
|
base62: self.base62_sha1(sha1),
|
|
|
|
extension: extension
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.base62_sha1(sha1)
|
|
|
|
Base62.encode(sha1.hex)
|
|
|
|
end
|
|
|
|
|
|
|
|
def base62_sha1
|
2019-06-04 02:10:46 -04:00
|
|
|
Upload.base62_sha1(self.sha1)
|
2017-08-22 11:46:15 -04:00
|
|
|
end
|
|
|
|
|
2018-08-27 22:48:43 -04:00
|
|
|
def local?
|
|
|
|
!(url =~ /^(https?:)?\/\//)
|
|
|
|
end
|
|
|
|
|
|
|
|
def fix_dimensions!
|
2018-09-09 22:22:45 -04:00
|
|
|
return if !FileHelper.is_supported_image?("image.#{extension}")
|
2018-08-27 22:48:43 -04:00
|
|
|
|
|
|
|
path =
|
|
|
|
if local?
|
|
|
|
Discourse.store.path_for(self)
|
|
|
|
else
|
|
|
|
Discourse.store.download(self).path
|
|
|
|
end
|
|
|
|
|
2018-12-03 10:19:49 -05:00
|
|
|
begin
|
2018-12-26 10:17:08 -05:00
|
|
|
w, h = FastImage.new(path, raise_on_failure: true).size
|
|
|
|
|
|
|
|
self.width = w || 0
|
|
|
|
self.height = h || 0
|
|
|
|
|
|
|
|
self.thumbnail_width, self.thumbnail_height = ImageSizer.resize(w, h)
|
|
|
|
|
|
|
|
self.update_columns(
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
thumbnail_width: thumbnail_width,
|
|
|
|
thumbnail_height: thumbnail_height
|
|
|
|
)
|
2018-12-03 10:19:49 -05:00
|
|
|
rescue => e
|
|
|
|
Discourse.warn_exception(e, message: "Error getting image dimensions")
|
|
|
|
end
|
2018-08-27 22:48:43 -04:00
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
|
|
|
# on demand image size calculation, this allows us to null out image sizes
|
|
|
|
# and still handle as needed
|
|
|
|
def get_dimension(key)
|
|
|
|
if v = read_attribute(key)
|
|
|
|
return v
|
|
|
|
end
|
|
|
|
fix_dimensions!
|
|
|
|
read_attribute(key)
|
|
|
|
end
|
|
|
|
|
|
|
|
def width
|
|
|
|
get_dimension(:width)
|
|
|
|
end
|
|
|
|
|
|
|
|
def height
|
|
|
|
get_dimension(:height)
|
|
|
|
end
|
|
|
|
|
|
|
|
def thumbnail_width
|
|
|
|
get_dimension(:thumbnail_width)
|
|
|
|
end
|
|
|
|
|
|
|
|
def thumbnail_height
|
|
|
|
get_dimension(:thumbnail_height)
|
|
|
|
end
|
|
|
|
|
2019-05-28 21:00:25 -04:00
|
|
|
def self.sha1_from_short_path(path)
|
|
|
|
if path =~ /(\/uploads\/short-url\/)([a-zA-Z0-9]+)(\..*)?/
|
|
|
|
self.sha1_from_base62_encoded($2)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-22 11:46:15 -04:00
|
|
|
def self.sha1_from_short_url(url)
|
|
|
|
if url =~ /(upload:\/\/)?([a-zA-Z0-9]+)(\..*)?/
|
2019-05-28 21:00:25 -04:00
|
|
|
self.sha1_from_base62_encoded($2)
|
|
|
|
end
|
|
|
|
end
|
2019-05-28 11:18:21 -04:00
|
|
|
|
2019-05-28 21:00:25 -04:00
|
|
|
def self.sha1_from_base62_encoded(encoded_sha1)
|
|
|
|
sha1 = Base62.decode(encoded_sha1).to_s(16)
|
|
|
|
|
|
|
|
if sha1.length > SHA1_LENGTH
|
|
|
|
nil
|
|
|
|
else
|
|
|
|
sha1.rjust(SHA1_LENGTH, '0')
|
2017-08-22 11:46:15 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
def self.generate_digest(path)
|
|
|
|
Digest::SHA1.file(path).hexdigest
|
|
|
|
end
|
|
|
|
|
2019-02-20 21:13:37 -05:00
|
|
|
def human_filesize
|
|
|
|
number_to_human_size(self.filesize)
|
|
|
|
end
|
|
|
|
|
2019-04-03 00:37:50 -04:00
|
|
|
def rebake_posts_on_old_scheme
|
|
|
|
self.posts.where("cooked LIKE '%/_optimized/%'").find_each(&:rebake!)
|
|
|
|
end
|
|
|
|
|
2019-11-17 20:25:42 -05:00
|
|
|
def update_secure_status
|
|
|
|
return false if self.for_theme || self.for_site_setting
|
|
|
|
mark_secure = should_be_secure?
|
|
|
|
|
|
|
|
self.update_column("secure", mark_secure)
|
|
|
|
Discourse.store.update_upload_ACL(self) if Discourse.store.external?
|
|
|
|
end
|
|
|
|
|
|
|
|
def should_be_secure?
|
|
|
|
mark_secure = false
|
|
|
|
if FileHelper.is_supported_media?(self.original_filename)
|
|
|
|
if SiteSetting.secure_media?
|
|
|
|
mark_secure = true if SiteSetting.login_required?
|
|
|
|
unless SiteSetting.login_required?
|
|
|
|
# first post associated with upload determines secure status
|
|
|
|
# i.e. an already public upload will stay public even if added to a new PM
|
|
|
|
first_post_with_upload = self.posts.order(sort_order: :asc).first
|
|
|
|
mark_secure = first_post_with_upload ? first_post_with_upload.with_secure_media? : false
|
|
|
|
end
|
|
|
|
else
|
|
|
|
mark_secure = false
|
|
|
|
end
|
|
|
|
else
|
|
|
|
mark_secure = SiteSetting.prevent_anons_from_downloading_files?
|
|
|
|
end
|
|
|
|
mark_secure
|
|
|
|
end
|
|
|
|
|
2019-04-23 23:56:48 -04:00
|
|
|
def self.migrate_to_new_scheme(limit: nil)
|
2015-06-12 06:02:36 -04:00
|
|
|
problems = []
|
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
DistributedMutex.synchronize("migrate_upload_to_new_scheme") do
|
|
|
|
if SiteSetting.migrate_to_new_scheme
|
|
|
|
max_file_size_kb = [
|
|
|
|
SiteSetting.max_image_size_kb,
|
|
|
|
SiteSetting.max_attachment_size_kb
|
|
|
|
].max.kilobytes
|
2015-06-12 06:02:36 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
local_store = FileStore::LocalStore.new
|
|
|
|
db = RailsMultisite::ConnectionManagement.current_db
|
2019-03-14 00:38:16 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
scope = Upload.by_users
|
|
|
|
.where("url NOT LIKE '%/original/_X/%' AND url LIKE '%/uploads/#{db}%'")
|
|
|
|
.order(id: :desc)
|
2019-04-24 01:59:23 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
scope = scope.limit(limit) if limit
|
2019-04-24 01:59:23 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
if scope.count == 0
|
|
|
|
SiteSetting.migrate_to_new_scheme = false
|
|
|
|
return problems
|
|
|
|
end
|
2019-03-28 03:58:42 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
remap_scope = nil
|
|
|
|
|
|
|
|
scope.each do |upload|
|
|
|
|
begin
|
|
|
|
# keep track of the url
|
|
|
|
previous_url = upload.url.dup
|
|
|
|
# where is the file currently stored?
|
|
|
|
external = previous_url =~ /^\/\//
|
|
|
|
# download if external
|
|
|
|
if external
|
|
|
|
url = SiteSetting.scheme + ":" + previous_url
|
|
|
|
|
|
|
|
begin
|
|
|
|
retries ||= 0
|
|
|
|
|
|
|
|
file = FileHelper.download(
|
|
|
|
url,
|
|
|
|
max_file_size: max_file_size_kb,
|
|
|
|
tmp_file_name: "discourse",
|
|
|
|
follow_redirect: true
|
|
|
|
)
|
|
|
|
rescue OpenURI::HTTPError
|
|
|
|
retry if (retires += 1) < 1
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
|
|
|
path = file.path
|
|
|
|
else
|
|
|
|
path = local_store.path_for(upload)
|
|
|
|
end
|
|
|
|
# compute SHA if missing
|
|
|
|
if upload.sha1.blank?
|
|
|
|
upload.sha1 = Upload.generate_digest(path)
|
|
|
|
end
|
2019-03-26 03:07:50 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
# store to new location & update the filesize
|
|
|
|
File.open(path) do |f|
|
|
|
|
upload.url = Discourse.store.store_upload(f, upload)
|
|
|
|
upload.filesize = f.size
|
|
|
|
upload.save!(validate: false)
|
|
|
|
end
|
|
|
|
# remap the URLs
|
|
|
|
DbHelper.remap(UrlHelper.absolute(previous_url), upload.url) unless external
|
|
|
|
|
|
|
|
DbHelper.remap(
|
|
|
|
previous_url,
|
|
|
|
upload.url,
|
|
|
|
excluded_tables: %w{
|
|
|
|
posts
|
|
|
|
post_search_data
|
2019-04-24 23:55:48 -04:00
|
|
|
incoming_emails
|
|
|
|
notifications
|
|
|
|
single_sign_on_records
|
|
|
|
stylesheet_cache
|
|
|
|
topic_search_data
|
|
|
|
users
|
|
|
|
user_emails
|
|
|
|
draft_sequences
|
|
|
|
optimized_images
|
2019-04-24 05:07:10 -04:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
remap_scope ||= begin
|
|
|
|
Post.with_deleted
|
2019-04-24 06:19:25 -04:00
|
|
|
.where("raw ~ '/uploads/#{db}/\\d+/' OR raw ~ '/uploads/#{db}/original/(\\d|[a-z])/'")
|
2019-04-25 02:26:40 -04:00
|
|
|
.select(:id, :raw, :cooked)
|
2019-04-24 05:07:10 -04:00
|
|
|
.all
|
|
|
|
end
|
2019-04-23 23:56:48 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
remap_scope.each do |post|
|
|
|
|
post.raw.gsub!(previous_url, upload.url)
|
|
|
|
post.cooked.gsub!(previous_url, upload.url)
|
2019-04-25 02:26:40 -04:00
|
|
|
Post.with_deleted.where(id: post.id).update_all(raw: post.raw, cooked: post.cooked) if post.changed?
|
2019-04-24 05:07:10 -04:00
|
|
|
end
|
2019-04-02 19:38:57 -04:00
|
|
|
|
2019-04-24 05:07:10 -04:00
|
|
|
upload.optimized_images.find_each(&:destroy!)
|
|
|
|
upload.rebake_posts_on_old_scheme
|
|
|
|
# remove the old file (when local)
|
|
|
|
unless external
|
|
|
|
FileUtils.rm(path, force: true)
|
|
|
|
end
|
|
|
|
rescue => e
|
|
|
|
problems << { upload: upload, ex: e }
|
|
|
|
ensure
|
|
|
|
file&.unlink
|
|
|
|
file&.close
|
2015-06-12 06:02:36 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
problems
|
|
|
|
end
|
|
|
|
|
2019-08-21 04:23:20 -04:00
|
|
|
def self.reset_unknown_extensions!
|
|
|
|
Upload.where(extension: "unknown").update_all(extension: nil)
|
|
|
|
end
|
|
|
|
|
2019-05-28 21:00:25 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def short_url_basename
|
2019-06-18 21:10:50 -04:00
|
|
|
"#{Upload.base62_sha1(sha1)}#{extension.present? ? ".#{extension}" : ""}"
|
2019-05-28 21:00:25 -04:00
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-05-23 22:48:32 -04:00
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: uploads
|
|
|
|
#
|
|
|
|
# id :integer not null, primary key
|
|
|
|
# user_id :integer not null
|
2019-01-11 14:29:56 -05:00
|
|
|
# original_filename :string not null
|
2013-05-23 22:48:32 -04:00
|
|
|
# filesize :integer not null
|
|
|
|
# width :integer
|
|
|
|
# height :integer
|
2019-01-11 14:29:56 -05:00
|
|
|
# url :string not null
|
2014-08-27 01:19:25 -04:00
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
2013-06-17 16:16:14 -04:00
|
|
|
# sha1 :string(40)
|
2013-12-05 01:40:35 -05:00
|
|
|
# origin :string(1000)
|
2014-11-19 22:53:15 -05:00
|
|
|
# retain_hours :integer
|
2017-08-16 10:38:11 -04:00
|
|
|
# extension :string(10)
|
2018-09-19 22:40:51 -04:00
|
|
|
# thumbnail_width :integer
|
|
|
|
# thumbnail_height :integer
|
2019-01-04 01:16:22 -05:00
|
|
|
# etag :string
|
2019-11-17 20:25:42 -05:00
|
|
|
# secure :boolean default(FALSE), not null
|
2013-05-23 22:48:32 -04:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
2019-01-11 12:19:23 -05:00
|
|
|
# index_uploads_on_etag (etag)
|
2017-10-05 23:13:01 -04:00
|
|
|
# index_uploads_on_extension (lower((extension)::text))
|
2013-10-03 23:28:49 -04:00
|
|
|
# index_uploads_on_id_and_url (id,url)
|
|
|
|
# index_uploads_on_sha1 (sha1) UNIQUE
|
|
|
|
# index_uploads_on_url (url)
|
|
|
|
# index_uploads_on_user_id (user_id)
|
2013-05-23 22:48:32 -04:00
|
|
|
#
|