discourse/app/models/upload.rb

227 lines
5.9 KiB
Ruby
Raw Normal View History

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