2013-11-06 02:04:47 +08:00
|
|
|
require "digest/sha1"
|
2014-04-15 04:55:57 +08:00
|
|
|
require_dependency "file_helper"
|
2015-06-12 18:02:36 +08:00
|
|
|
require_dependency "url_helper"
|
|
|
|
require_dependency "db_helper"
|
2014-04-15 04:55:57 +08:00
|
|
|
require_dependency "validators/upload_validator"
|
2015-06-12 18:02:36 +08:00
|
|
|
require_dependency "file_store/local_store"
|
2017-08-22 23:46:15 +08:00
|
|
|
require_dependency "base62"
|
2013-02-06 03:16:51 +08:00
|
|
|
|
|
|
|
class Upload < ActiveRecord::Base
|
|
|
|
belongs_to :user
|
|
|
|
|
2013-11-06 02:04:47 +08:00
|
|
|
has_many :post_uploads, dependent: :destroy
|
2013-06-14 05:44:24 +08:00
|
|
|
has_many :posts, through: :post_uploads
|
2013-06-13 07:43:50 +08:00
|
|
|
|
2013-06-21 15:34:02 +08:00
|
|
|
has_many :optimized_images, dependent: :destroy
|
2013-06-16 16:39:48 +08:00
|
|
|
|
2017-06-13 04:41:29 +08:00
|
|
|
attr_accessor :for_group_message
|
2017-05-10 05:20:28 +08:00
|
|
|
attr_accessor :for_theme
|
2017-06-13 04:41:29 +08:00
|
|
|
attr_accessor :for_private_message
|
2018-04-19 19:30:31 +08:00
|
|
|
attr_accessor :for_export
|
2016-03-01 05:39:24 +08:00
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
validates_presence_of :filesize
|
|
|
|
validates_presence_of :original_filename
|
|
|
|
|
2014-04-15 04:55:57 +08:00
|
|
|
validates_with ::Validators::UploadValidator
|
|
|
|
|
2013-11-06 02:04:47 +08:00
|
|
|
def thumbnail(width = self.width, height = self.height)
|
2014-05-06 21:41:59 +08:00
|
|
|
optimized_images.find_by(width: width, height: height)
|
2013-06-17 07:00:25 +08:00
|
|
|
end
|
|
|
|
|
2013-11-06 02:04:47 +08:00
|
|
|
def has_thumbnail?(width, height)
|
2013-09-27 16:55:50 +08:00
|
|
|
thumbnail(width, height).present?
|
2013-06-17 07:00:25 +08:00
|
|
|
end
|
|
|
|
|
2017-07-28 09:20:09 +08:00
|
|
|
def create_thumbnail!(width, height, crop = false)
|
2013-06-17 07:00:25 +08:00
|
|
|
return unless SiteSetting.create_thumbnails?
|
2015-09-21 04:01:03 +08:00
|
|
|
|
2016-05-23 22:18:30 +08:00
|
|
|
opts = {
|
|
|
|
allow_animation: SiteSetting.allow_animated_thumbnails,
|
2016-05-23 22:42:19 +08:00
|
|
|
crop: crop
|
2016-05-23 22:18:30 +08:00
|
|
|
}
|
2015-09-21 04:01:03 +08:00
|
|
|
|
2018-08-17 12:00:27 +08:00
|
|
|
if get_optimized_image(width, height, opts)
|
|
|
|
# TODO: this code is not right, we may have multiple
|
|
|
|
# thumbs
|
2013-09-27 16:55:50 +08:00
|
|
|
self.width = width
|
|
|
|
self.height = height
|
2015-06-27 07:26:16 +08:00
|
|
|
save(validate: false)
|
2013-09-27 16:55:50 +08:00
|
|
|
end
|
2013-06-17 07:00:25 +08:00
|
|
|
end
|
|
|
|
|
2018-08-17 12:00:27 +08: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)
|
|
|
|
rescue
|
|
|
|
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 15:34:02 +08:00
|
|
|
def destroy
|
2013-06-20 03:51:41 +08:00
|
|
|
Upload.transaction do
|
2013-08-14 04:08:29 +08:00
|
|
|
Discourse.store.remove_upload(self)
|
2013-06-20 03:51:41 +08:00
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-22 23:46:15 +08:00
|
|
|
def short_url
|
|
|
|
"upload://#{Base62.encode(sha1.hex)}.#{extension}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.sha1_from_short_url(url)
|
|
|
|
if url =~ /(upload:\/\/)?([a-zA-Z0-9]+)(\..*)?/
|
|
|
|
sha1 = Base62.decode($2).to_s(16)
|
2017-08-23 23:08:18 +08:00
|
|
|
|
|
|
|
if sha1.length > 40
|
|
|
|
nil
|
|
|
|
else
|
|
|
|
sha1.rjust(40, '0')
|
|
|
|
end
|
2017-08-22 23:46:15 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-09-02 14:50:13 +08:00
|
|
|
def self.generate_digest(path)
|
|
|
|
Digest::SHA1.file(path).hexdigest
|
|
|
|
end
|
|
|
|
|
2013-07-08 07:39:08 +08:00
|
|
|
def self.get_from_url(url)
|
2014-07-18 23:54:18 +08:00
|
|
|
return if url.blank?
|
2016-10-18 15:58:45 +08:00
|
|
|
|
2018-03-28 16:20:08 +08:00
|
|
|
uri = begin
|
2018-05-17 15:13:30 +08:00
|
|
|
URI(URI.unescape(url))
|
2018-08-14 18:23:32 +08:00
|
|
|
rescue URI::Error
|
2018-03-28 16:20:08 +08:00
|
|
|
end
|
|
|
|
|
2018-08-04 07:29:32 +08:00
|
|
|
return if uri&.path.blank?
|
|
|
|
|
|
|
|
path = uri.path[/(\/original\/\dX\/[\/\.\w]+)/, 1]
|
2016-10-18 15:58:45 +08:00
|
|
|
|
2018-08-04 07:56:26 +08:00
|
|
|
return if path.blank?
|
|
|
|
|
2018-08-04 07:29:32 +08:00
|
|
|
Upload.find_by("url LIKE ?", "%#{path}")
|
2013-07-08 07:39:08 +08:00
|
|
|
end
|
|
|
|
|
2017-07-28 09:20:09 +08:00
|
|
|
def self.migrate_to_new_scheme(limit = nil)
|
2015-06-12 18:02:36 +08:00
|
|
|
problems = []
|
|
|
|
|
|
|
|
if SiteSetting.migrate_to_new_scheme
|
|
|
|
max_file_size_kb = [SiteSetting.max_image_size_kb, SiteSetting.max_attachment_size_kb].max.kilobytes
|
|
|
|
local_store = FileStore::LocalStore.new
|
|
|
|
|
2016-09-02 10:55:11 +08:00
|
|
|
scope = Upload.where("url NOT LIKE '%/original/_X/%'").order(id: :desc)
|
2018-03-22 10:56:06 +08:00
|
|
|
scope = scope.limit(limit) if limit
|
2016-09-02 10:55:11 +08:00
|
|
|
|
|
|
|
scope.each do |upload|
|
2015-06-12 18:02:36 +08:00
|
|
|
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
|
2017-05-25 01:42:52 +08:00
|
|
|
file = FileHelper.download(
|
|
|
|
url,
|
|
|
|
max_file_size: max_file_size_kb,
|
|
|
|
tmp_file_name: "discourse",
|
|
|
|
follow_redirect: true
|
|
|
|
) rescue nil
|
2015-06-12 18:02:36 +08:00
|
|
|
path = file.path
|
|
|
|
else
|
|
|
|
path = local_store.path_for(upload)
|
|
|
|
end
|
|
|
|
# compute SHA if missing
|
|
|
|
if upload.sha1.blank?
|
2016-09-02 14:50:13 +08:00
|
|
|
upload.sha1 = Upload.generate_digest(path)
|
2015-06-12 18:02:36 +08:00
|
|
|
end
|
|
|
|
# optimize if image
|
2017-07-25 17:48:39 +08:00
|
|
|
FileHelper.optimize_image!(path) if FileHelper.is_image?(File.basename(path))
|
2015-06-12 18:02:36 +08:00
|
|
|
# store to new location & update the filesize
|
|
|
|
File.open(path) do |f|
|
2016-09-02 11:58:56 +08:00
|
|
|
upload.url = Discourse.store.store_upload(f, upload)
|
|
|
|
upload.filesize = f.size
|
2016-09-02 11:59:03 +08:00
|
|
|
upload.save!
|
2015-06-12 18:02:36 +08:00
|
|
|
end
|
|
|
|
# remap the URLs
|
|
|
|
DbHelper.remap(UrlHelper.absolute(previous_url), upload.url) unless external
|
|
|
|
DbHelper.remap(previous_url, upload.url)
|
|
|
|
# remove the old file (when local)
|
|
|
|
unless external
|
2018-03-28 16:20:08 +08:00
|
|
|
FileUtils.rm(path, force: true)
|
2015-06-12 18:02:36 +08:00
|
|
|
end
|
|
|
|
rescue => e
|
|
|
|
problems << { upload: upload, ex: e }
|
|
|
|
ensure
|
2018-03-28 16:20:08 +08:00
|
|
|
file&.unlink
|
|
|
|
file&.close
|
2015-06-12 18:02:36 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
problems
|
|
|
|
end
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
2013-05-24 10:48:32 +08:00
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: uploads
|
|
|
|
#
|
|
|
|
# id :integer not null, primary key
|
|
|
|
# user_id :integer not null
|
2018-02-20 14:28:58 +08:00
|
|
|
# original_filename :string not null
|
2013-05-24 10:48:32 +08:00
|
|
|
# filesize :integer not null
|
|
|
|
# width :integer
|
|
|
|
# height :integer
|
2018-02-20 14:28:58 +08:00
|
|
|
# url :string not null
|
2014-08-27 13:19:25 +08:00
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
2013-06-18 04:16:14 +08:00
|
|
|
# sha1 :string(40)
|
2013-12-05 14:40:35 +08:00
|
|
|
# origin :string(1000)
|
2014-11-20 11:53:15 +08:00
|
|
|
# retain_hours :integer
|
2017-08-16 22:38:11 +08:00
|
|
|
# extension :string(10)
|
2013-05-24 10:48:32 +08:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
2017-10-06 11:13:01 +08:00
|
|
|
# index_uploads_on_extension (lower((extension)::text))
|
2013-10-04 11:28:49 +08: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-24 10:48:32 +08:00
|
|
|
#
|