2019-05-02 17:17:27 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-06-30 09:55:01 -05:00
|
|
|
require "uri"
|
2017-06-13 06:27:05 -05:00
|
|
|
require "mini_mime"
|
2015-06-01 04:13:56 -05:00
|
|
|
require_dependency "file_store/base_store"
|
2014-09-24 15:52:09 -05:00
|
|
|
require_dependency "s3_helper"
|
2014-04-15 06:04:14 -05:00
|
|
|
require_dependency "file_helper"
|
2013-07-31 16:26:34 -05:00
|
|
|
|
2013-11-05 12:04:47 -06:00
|
|
|
module FileStore
|
2013-07-31 16:26:34 -05:00
|
|
|
|
2013-11-05 12:04:47 -06:00
|
|
|
class S3Store < BaseStore
|
2015-05-25 10:59:00 -05:00
|
|
|
TOMBSTONE_PREFIX ||= "tombstone/"
|
|
|
|
|
2018-09-10 02:14:30 -05:00
|
|
|
attr_reader :s3_helper
|
|
|
|
|
2017-07-27 20:20:09 -05:00
|
|
|
def initialize(s3_helper = nil)
|
2018-12-18 23:32:32 -06:00
|
|
|
@s3_helper = s3_helper || S3Helper.new(s3_bucket,
|
|
|
|
Rails.configuration.multisite ? multisite_tombstone_prefix : TOMBSTONE_PREFIX
|
|
|
|
)
|
2014-09-24 15:52:09 -05:00
|
|
|
end
|
2013-07-31 16:26:34 -05:00
|
|
|
|
2015-05-29 11:39:47 -05:00
|
|
|
def store_upload(file, upload, content_type = nil)
|
2016-08-15 03:06:29 -05:00
|
|
|
path = get_path_for_upload(upload)
|
2020-01-15 21:50:27 -06:00
|
|
|
url, upload.etag = store_file(
|
|
|
|
file,
|
|
|
|
path,
|
|
|
|
filename: upload.original_filename,
|
|
|
|
content_type: content_type,
|
|
|
|
cache_locally: true,
|
|
|
|
private_acl: upload.secure?
|
|
|
|
)
|
2019-01-04 00:16:22 -06:00
|
|
|
url
|
2013-11-05 12:04:47 -06:00
|
|
|
end
|
2013-07-31 16:26:34 -05:00
|
|
|
|
2019-11-17 19:25:42 -06:00
|
|
|
def store_optimized_image(file, optimized_image, content_type = nil, secure: false)
|
2016-08-15 03:06:29 -05:00
|
|
|
path = get_path_for_optimized_image(optimized_image)
|
2019-11-17 19:25:42 -06:00
|
|
|
url, optimized_image.etag = store_file(file, path, content_type: content_type, private_acl: secure)
|
2019-01-04 00:16:22 -06:00
|
|
|
url
|
2016-08-12 04:18:19 -05:00
|
|
|
end
|
|
|
|
|
2015-05-29 11:39:47 -05:00
|
|
|
# options
|
|
|
|
# - filename
|
|
|
|
# - content_type
|
|
|
|
# - cache_locally
|
2017-07-27 20:20:09 -05:00
|
|
|
def store_file(file, path, opts = {})
|
2019-05-02 17:17:27 -05:00
|
|
|
path = path.dup
|
|
|
|
|
2016-10-17 12:16:29 -05:00
|
|
|
filename = opts[:filename].presence || File.basename(path)
|
2015-05-29 11:39:47 -05:00
|
|
|
# cache file locally when needed
|
|
|
|
cache_file(file, File.basename(path)) if opts[:cache_locally]
|
2016-10-17 12:16:29 -05:00
|
|
|
options = {
|
2019-11-17 19:25:42 -06:00
|
|
|
acl: opts[:private_acl] ? "private" : "public-read",
|
2019-08-06 12:55:17 -05:00
|
|
|
cache_control: 'max-age=31556952, public, immutable',
|
2017-06-13 06:27:05 -05:00
|
|
|
content_type: opts[:content_type].presence || MiniMime.lookup_by_filename(filename)&.content_type
|
2016-10-17 12:16:29 -05:00
|
|
|
}
|
2020-06-16 20:16:37 -05:00
|
|
|
|
|
|
|
# add a "content disposition: attachment" header with the original filename
|
|
|
|
# for everything but images. audio and video will still stream correctly in
|
|
|
|
# HTML players, and when a direct link is provided to any file but an image
|
|
|
|
# it will download correctly in the browser.
|
|
|
|
if !FileHelper.is_supported_image?(filename)
|
2020-06-23 02:10:56 -05:00
|
|
|
options[:content_disposition] = ActionDispatch::Http::ContentDisposition.format(
|
|
|
|
disposition: "attachment", filename: filename
|
|
|
|
)
|
2020-06-16 20:16:37 -05:00
|
|
|
end
|
2018-11-28 22:11:48 -06:00
|
|
|
|
2018-12-02 22:04:14 -06:00
|
|
|
path.prepend(File.join(upload_path, "/")) if Rails.configuration.multisite
|
2018-11-28 22:11:48 -06:00
|
|
|
|
2019-01-04 00:16:22 -06:00
|
|
|
# if this fails, it will throw an exception
|
|
|
|
path, etag = @s3_helper.upload(file, path, options)
|
|
|
|
|
|
|
|
# return the upload url and etag
|
2019-11-14 14:10:51 -06:00
|
|
|
[File.join(absolute_base_url, path), etag]
|
2013-11-05 12:04:47 -06:00
|
|
|
end
|
2013-07-31 16:26:34 -05:00
|
|
|
|
2016-08-14 22:21:24 -05:00
|
|
|
def remove_file(url, path)
|
2015-05-29 11:39:47 -05:00
|
|
|
return unless has_been_uploaded?(url)
|
|
|
|
# copy the removed file to tombstone
|
2016-08-15 03:06:29 -05:00
|
|
|
@s3_helper.remove(path, true)
|
2013-11-05 12:04:47 -06:00
|
|
|
end
|
2013-08-13 15:08:29 -05:00
|
|
|
|
2018-08-07 22:26:05 -05:00
|
|
|
def copy_file(url, source, destination)
|
|
|
|
return unless has_been_uploaded?(url)
|
|
|
|
@s3_helper.copy(source, destination)
|
|
|
|
end
|
|
|
|
|
2013-11-05 12:04:47 -06:00
|
|
|
def has_been_uploaded?(url)
|
2015-05-26 04:47:33 -05:00
|
|
|
return false if url.blank?
|
2016-06-30 09:55:01 -05:00
|
|
|
|
2020-05-22 23:56:13 -05:00
|
|
|
begin
|
2020-06-17 02:47:05 -05:00
|
|
|
parsed_url = URI.parse(UrlHelper.encode(url))
|
2020-06-25 00:00:15 -05:00
|
|
|
rescue
|
|
|
|
# There are many exceptions possible here including Addressable::URI:: excpetions
|
|
|
|
# and URI:: exceptions, catch all may seem wide, but it makes no sense to raise ever
|
|
|
|
# on an invalid url here
|
2020-05-22 23:56:13 -05:00
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2016-06-30 09:55:01 -05:00
|
|
|
base_hostname = URI.parse(absolute_base_url).hostname
|
2020-05-22 23:56:13 -05:00
|
|
|
if url[base_hostname]
|
|
|
|
# if the hostnames match it means the upload is in the same
|
|
|
|
# bucket on s3. however, the bucket folder path may differ in
|
|
|
|
# some cases, and we do not want to assume the url is uploaded
|
|
|
|
# here. e.g. the path of the current site could be /prod and the
|
|
|
|
# other site could be /staging
|
|
|
|
if s3_bucket_folder_path.present?
|
|
|
|
return parsed_url.path.starts_with?("/#{s3_bucket_folder_path}")
|
|
|
|
else
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
2016-06-30 09:55:01 -05:00
|
|
|
|
2017-10-06 00:20:01 -05:00
|
|
|
return false if SiteSetting.Upload.s3_cdn_url.blank?
|
|
|
|
cdn_hostname = URI.parse(SiteSetting.Upload.s3_cdn_url || "").hostname
|
2020-05-22 23:56:13 -05:00
|
|
|
return true if cdn_hostname.presence && url[cdn_hostname]
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
def s3_bucket_folder_path
|
|
|
|
@s3_helper.s3_bucket_folder_path
|
2013-11-05 12:04:47 -06:00
|
|
|
end
|
2013-08-13 15:08:29 -05:00
|
|
|
|
2017-10-06 00:20:01 -05:00
|
|
|
def s3_bucket_name
|
|
|
|
@s3_helper.s3_bucket_name
|
|
|
|
end
|
|
|
|
|
2013-11-05 12:04:47 -06:00
|
|
|
def absolute_base_url
|
2017-10-06 00:20:01 -05:00
|
|
|
@absolute_base_url ||= SiteSetting.Upload.absolute_base_url
|
2013-11-05 12:04:47 -06:00
|
|
|
end
|
2013-08-13 15:08:29 -05:00
|
|
|
|
2019-11-17 19:25:42 -06:00
|
|
|
def s3_upload_host
|
|
|
|
SiteSetting.Upload.s3_cdn_url.present? ? SiteSetting.Upload.s3_cdn_url : "https:#{absolute_base_url}"
|
|
|
|
end
|
|
|
|
|
2013-11-05 12:04:47 -06:00
|
|
|
def external?
|
|
|
|
true
|
|
|
|
end
|
2013-08-13 15:08:29 -05:00
|
|
|
|
2013-11-27 15:01:41 -06:00
|
|
|
def purge_tombstone(grace_period)
|
2014-09-24 15:52:09 -05:00
|
|
|
@s3_helper.update_tombstone_lifecycle(grace_period)
|
2013-11-27 15:01:41 -06:00
|
|
|
end
|
|
|
|
|
2018-12-18 23:32:32 -06:00
|
|
|
def multisite_tombstone_prefix
|
|
|
|
File.join("uploads", "tombstone", RailsMultisite::ConnectionManagement.current_db, "/")
|
|
|
|
end
|
|
|
|
|
2019-07-04 10:32:51 -05:00
|
|
|
def download_url(upload)
|
|
|
|
return unless upload
|
|
|
|
"#{upload.short_path}?dl=1"
|
|
|
|
end
|
|
|
|
|
2015-05-25 22:08:31 -05:00
|
|
|
def path_for(upload)
|
2019-05-28 20:00:25 -05:00
|
|
|
url = upload&.url
|
2019-05-29 05:36:09 -05:00
|
|
|
FileStore::LocalStore.new.path_for(upload) if url && url[/^\/[^\/]/]
|
2015-05-25 22:08:31 -05:00
|
|
|
end
|
|
|
|
|
2019-07-04 10:32:51 -05:00
|
|
|
def url_for(upload, force_download: false)
|
2019-11-17 19:25:42 -06:00
|
|
|
upload.secure? || force_download ?
|
|
|
|
presigned_url(get_upload_key(upload), force_download: force_download, filename: upload.original_filename) :
|
|
|
|
upload.url
|
2019-06-05 22:27:24 -05:00
|
|
|
end
|
|
|
|
|
2015-05-26 21:02:57 -05:00
|
|
|
def cdn_url(url)
|
2017-10-06 00:20:01 -05:00
|
|
|
return url if SiteSetting.Upload.s3_cdn_url.blank?
|
2016-06-30 09:55:01 -05:00
|
|
|
schema = url[/^(https?:)?\/\//, 1]
|
2018-07-06 17:15:28 -05:00
|
|
|
folder = @s3_helper.s3_bucket_folder_path.nil? ? "" : "#{@s3_helper.s3_bucket_folder_path}/"
|
2018-11-28 22:11:48 -06:00
|
|
|
url.sub(File.join("#{schema}#{absolute_base_url}", folder), File.join(SiteSetting.Upload.s3_cdn_url, "/"))
|
2015-05-26 21:02:57 -05:00
|
|
|
end
|
|
|
|
|
2019-11-17 19:25:42 -06:00
|
|
|
def signed_url_for_path(path)
|
|
|
|
key = path.sub(absolute_base_url + "/", "")
|
|
|
|
presigned_url(key)
|
|
|
|
end
|
|
|
|
|
2015-05-29 11:39:47 -05:00
|
|
|
def cache_avatar(avatar, user_id)
|
|
|
|
source = avatar.url.sub(absolute_base_url + "/", "")
|
|
|
|
destination = avatar_template(avatar, user_id).sub(absolute_base_url + "/", "")
|
|
|
|
@s3_helper.copy(source, destination)
|
|
|
|
end
|
2013-11-27 15:01:41 -06:00
|
|
|
|
2015-05-29 11:39:47 -05:00
|
|
|
def avatar_template(avatar, user_id)
|
|
|
|
UserAvatar.external_avatar_url(user_id, avatar.upload_id, avatar.width)
|
|
|
|
end
|
2013-07-31 16:26:34 -05:00
|
|
|
|
2016-08-19 01:08:04 -05:00
|
|
|
def s3_bucket
|
2017-10-06 00:20:01 -05:00
|
|
|
raise Discourse::SiteSettingMissing.new("s3_upload_bucket") if SiteSetting.Upload.s3_upload_bucket.blank?
|
|
|
|
SiteSetting.Upload.s3_upload_bucket.downcase
|
2015-05-29 11:39:47 -05:00
|
|
|
end
|
2018-11-26 13:24:51 -06:00
|
|
|
|
2019-02-14 13:04:35 -06:00
|
|
|
def list_missing_uploads(skip_optimized: false)
|
2019-01-31 22:40:48 -06:00
|
|
|
if SiteSetting.enable_s3_inventory
|
|
|
|
require 's3_inventory'
|
2019-02-19 10:24:35 -06:00
|
|
|
S3Inventory.new(s3_helper, :upload).backfill_etags_and_list_missing
|
|
|
|
S3Inventory.new(s3_helper, :optimized).backfill_etags_and_list_missing unless skip_optimized
|
2019-01-31 22:40:48 -06:00
|
|
|
else
|
2019-03-13 04:39:07 -05:00
|
|
|
list_missing(Upload.by_users, "original/")
|
2019-01-31 22:40:48 -06:00
|
|
|
list_missing(OptimizedImage, "optimized/") unless skip_optimized
|
|
|
|
end
|
2018-11-26 13:24:51 -06:00
|
|
|
end
|
|
|
|
|
2019-06-05 22:27:24 -05:00
|
|
|
def update_upload_ACL(upload)
|
|
|
|
key = get_upload_key(upload)
|
2019-11-17 19:25:42 -06:00
|
|
|
update_ACL(key, upload.secure?)
|
2019-06-05 22:27:24 -05:00
|
|
|
|
2019-11-17 19:25:42 -06:00
|
|
|
upload.optimized_images.find_each do |optimized_image|
|
|
|
|
optimized_image_key = get_path_for_optimized_image(optimized_image)
|
|
|
|
update_ACL(optimized_image_key, upload.secure?)
|
2019-06-05 22:27:24 -05:00
|
|
|
end
|
2019-11-17 19:25:42 -06:00
|
|
|
|
|
|
|
true
|
2019-06-05 22:27:24 -05:00
|
|
|
end
|
|
|
|
|
2019-07-01 13:38:36 -05:00
|
|
|
def download_file(upload, destination_path)
|
|
|
|
@s3_helper.download_file(get_upload_key(upload), destination_path)
|
|
|
|
end
|
|
|
|
|
2020-01-12 17:12:27 -06:00
|
|
|
def copy_from(source_path)
|
|
|
|
local_store = FileStore::LocalStore.new
|
|
|
|
public_upload_path = File.join(local_store.public_dir, local_store.upload_path)
|
|
|
|
|
|
|
|
# The migration to S3 and lots of other code expects files to exist in public/uploads,
|
|
|
|
# so lets move them there before executing the migration.
|
|
|
|
if public_upload_path != source_path
|
|
|
|
if Dir.exist?(public_upload_path)
|
|
|
|
old_upload_path = "#{public_upload_path}_#{SecureRandom.hex}"
|
|
|
|
FileUtils.mv(public_upload_path, old_upload_path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
FileUtils.mkdir_p(File.expand_path("..", public_upload_path))
|
|
|
|
FileUtils.symlink(source_path, public_upload_path)
|
|
|
|
|
|
|
|
FileStore::ToS3Migration.new(
|
2020-04-19 13:24:27 -05:00
|
|
|
s3_options: FileStore::ToS3Migration.s3_options_from_site_settings,
|
2020-01-12 17:12:27 -06:00
|
|
|
migrate_to_multisite: Rails.configuration.multisite,
|
|
|
|
).migrate
|
|
|
|
|
|
|
|
ensure
|
|
|
|
FileUtils.rm(public_upload_path) if File.symlink?(public_upload_path)
|
|
|
|
FileUtils.mv(old_upload_path, public_upload_path) if old_upload_path
|
|
|
|
end
|
|
|
|
|
2018-11-26 13:24:51 -06:00
|
|
|
private
|
|
|
|
|
2019-11-17 19:25:42 -06:00
|
|
|
def presigned_url(url, force_download: false, filename: false)
|
|
|
|
opts = { expires_in: S3Helper::DOWNLOAD_URL_EXPIRES_AFTER_SECONDS }
|
|
|
|
if force_download && filename
|
|
|
|
opts[:response_content_disposition] = ActionDispatch::Http::ContentDisposition.format(
|
|
|
|
disposition: "attachment", filename: filename
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
obj = @s3_helper.object(url)
|
|
|
|
obj.presigned_url(:get, opts)
|
|
|
|
end
|
|
|
|
|
2019-06-05 22:27:24 -05:00
|
|
|
def get_upload_key(upload)
|
|
|
|
if Rails.configuration.multisite
|
|
|
|
File.join(upload_path, "/", get_path_for_upload(upload))
|
|
|
|
else
|
|
|
|
get_path_for_upload(upload)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-11-17 19:25:42 -06:00
|
|
|
def update_ACL(key, secure)
|
|
|
|
begin
|
|
|
|
@s3_helper.object(key).acl.put(acl: secure ? "private" : "public-read")
|
|
|
|
rescue Aws::S3::Errors::NoSuchKey
|
|
|
|
Rails.logger.warn("Could not update ACL on upload with key: '#{key}'. Upload is missing.")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-26 13:24:51 -06:00
|
|
|
def list_missing(model, prefix)
|
|
|
|
connection = ActiveRecord::Base.connection.raw_connection
|
|
|
|
connection.exec('CREATE TEMP TABLE verified_ids(val integer PRIMARY KEY)')
|
|
|
|
marker = nil
|
|
|
|
files = @s3_helper.list(prefix, marker)
|
|
|
|
|
|
|
|
while files.count > 0 do
|
|
|
|
verified_ids = []
|
|
|
|
|
|
|
|
files.each do |f|
|
2019-10-21 05:32:27 -05:00
|
|
|
id = model.where("url LIKE '%#{f.key}' AND etag = '#{f.etag}'").pluck_first(:id)
|
2018-11-26 13:24:51 -06:00
|
|
|
verified_ids << id if id.present?
|
|
|
|
marker = f.key
|
|
|
|
end
|
|
|
|
|
|
|
|
verified_id_clause = verified_ids.map { |id| "('#{PG::Connection.escape_string(id.to_s)}')" }.join(",")
|
|
|
|
connection.exec("INSERT INTO verified_ids VALUES #{verified_id_clause}")
|
|
|
|
files = @s3_helper.list(prefix, marker)
|
|
|
|
end
|
|
|
|
|
2019-01-31 22:40:48 -06:00
|
|
|
missing_uploads = model.joins('LEFT JOIN verified_ids ON verified_ids.val = id').where("verified_ids.val IS NULL")
|
2018-11-26 13:24:51 -06:00
|
|
|
missing_count = missing_uploads.count
|
|
|
|
|
|
|
|
if missing_count > 0
|
|
|
|
missing_uploads.find_each do |upload|
|
|
|
|
puts upload.url
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "#{missing_count} of #{model.count} #{model.name.underscore.pluralize} are missing"
|
|
|
|
end
|
|
|
|
ensure
|
2018-11-26 13:45:29 -06:00
|
|
|
connection.exec('DROP TABLE verified_ids') unless connection.nil?
|
2018-11-26 13:24:51 -06:00
|
|
|
end
|
2013-07-31 16:26:34 -05:00
|
|
|
end
|
|
|
|
end
|