2013-11-05 13:04:47 -05:00
|
|
|
require 'file_store/base_store'
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
module FileStore
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
class S3Store < BaseStore
|
|
|
|
@fog_loaded ||= require 'fog'
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def store_upload(file, upload)
|
|
|
|
path = get_path_for_upload(file, upload)
|
|
|
|
store_file(file, path, upload.original_filename, file.content_type)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def store_optimized_image(file, optimized_image)
|
|
|
|
path = get_path_for_optimized_image(file, optimized_image)
|
|
|
|
store_file(file, path)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def store_avatar(file, avatar, size)
|
|
|
|
path = get_path_for_avatar(file, avatar, size)
|
|
|
|
store_file(file, path)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def remove_upload(upload)
|
|
|
|
remove_file(upload.url)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def remove_optimized_image(optimized_image)
|
|
|
|
remove_file(optimized_image.url)
|
|
|
|
end
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def has_been_uploaded?(url)
|
|
|
|
url.start_with?(absolute_base_url)
|
|
|
|
end
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def absolute_base_url
|
|
|
|
"//#{s3_bucket}.s3.amazonaws.com"
|
|
|
|
end
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def external?
|
|
|
|
true
|
|
|
|
end
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def internal?
|
|
|
|
!external?
|
|
|
|
end
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def download(upload)
|
|
|
|
@open_uri_loaded ||= require 'open-uri'
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
extension = File.extname(upload.original_filename)
|
|
|
|
temp_file = Tempfile.new(["discourse-s3", extension])
|
2013-12-16 05:44:59 -05:00
|
|
|
url = SiteSetting.scheme + ":" + upload.url
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
File.open(temp_file.path, "wb") do |f|
|
|
|
|
f.write(open(url, "rb", read_timeout: 5).read)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
temp_file
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2014-01-07 11:45:06 -05:00
|
|
|
def avatar_template(avatar)
|
|
|
|
template = relative_avatar_template(avatar)
|
|
|
|
"#{absolute_base_url}/#{template}"
|
2013-11-05 13:04:47 -05:00
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-27 16:01:41 -05:00
|
|
|
def purge_tombstone(grace_period)
|
|
|
|
update_tombstone_lifecycle(grace_period)
|
|
|
|
end
|
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
private
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def get_path_for_upload(file, upload)
|
|
|
|
"#{upload.id}#{upload.sha1}#{upload.extension}"
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def get_path_for_optimized_image(file, optimized_image)
|
|
|
|
"#{optimized_image.id}#{optimized_image.sha1}_#{optimized_image.width}x#{optimized_image.height}#{optimized_image.extension}"
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def get_path_for_avatar(file, avatar, size)
|
2014-01-07 11:45:06 -05:00
|
|
|
relative_avatar_template(avatar).gsub("{size}", size.to_s)
|
|
|
|
end
|
|
|
|
|
|
|
|
def relative_avatar_template(avatar)
|
|
|
|
"avatars/#{avatar.sha1}/{size}#{avatar.extension}"
|
2013-11-05 13:04:47 -05:00
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def store_file(file, path, filename = nil, content_type = nil)
|
|
|
|
# if this fails, it will throw an exception
|
|
|
|
upload(file, path, filename, content_type)
|
|
|
|
# url
|
|
|
|
"#{absolute_base_url}/#{path}"
|
2013-07-31 17:26:34 -04:00
|
|
|
end
|
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def remove_file(url)
|
|
|
|
return unless has_been_uploaded?(url)
|
|
|
|
filename = File.basename(url)
|
|
|
|
remove(filename)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def s3_bucket
|
|
|
|
SiteSetting.s3_upload_bucket.downcase
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def check_missing_site_settings
|
|
|
|
raise Discourse::SiteSettingMissing.new("s3_upload_bucket") if SiteSetting.s3_upload_bucket.blank?
|
|
|
|
raise Discourse::SiteSettingMissing.new("s3_access_key_id") if SiteSetting.s3_access_key_id.blank?
|
|
|
|
raise Discourse::SiteSettingMissing.new("s3_secret_access_key") if SiteSetting.s3_secret_access_key.blank?
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def s3_options
|
|
|
|
options = {
|
|
|
|
provider: 'AWS',
|
|
|
|
aws_access_key_id: SiteSetting.s3_access_key_id,
|
|
|
|
aws_secret_access_key: SiteSetting.s3_secret_access_key,
|
2013-12-16 05:44:59 -05:00
|
|
|
scheme: SiteSetting.scheme,
|
|
|
|
# cf. https://github.com/fog/fog/issues/2381
|
|
|
|
path_style: dns_compatible?(s3_bucket, SiteSetting.use_ssl?),
|
2013-11-05 13:04:47 -05:00
|
|
|
}
|
|
|
|
options[:region] = SiteSetting.s3_region unless SiteSetting.s3_region.empty?
|
|
|
|
options
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-27 16:01:41 -05:00
|
|
|
def fog_with_options
|
|
|
|
check_missing_site_settings
|
|
|
|
Fog::Storage.new(s3_options)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_or_create_directory(bucket)
|
|
|
|
fog = fog_with_options
|
|
|
|
directory = fog.directories.get(bucket)
|
|
|
|
directory = fog.directories.create(key: bucket) unless directory
|
|
|
|
directory
|
|
|
|
end
|
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def upload(file, unique_filename, filename=nil, content_type=nil)
|
|
|
|
args = {
|
|
|
|
key: unique_filename,
|
|
|
|
public: true,
|
|
|
|
body: file
|
|
|
|
}
|
|
|
|
args[:content_disposition] = "attachment; filename=\"#{filename}\"" if filename
|
|
|
|
args[:content_type] = content_type if content_type
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
get_or_create_directory(s3_bucket).files.create(args)
|
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-11-05 13:04:47 -05:00
|
|
|
def remove(unique_filename)
|
2013-11-27 16:01:41 -05:00
|
|
|
fog = fog_with_options
|
|
|
|
# copy the file in tombstone
|
|
|
|
fog.copy_object(unique_filename, s3_bucket, tombstone_prefix + unique_filename, s3_bucket)
|
|
|
|
# delete the file
|
|
|
|
fog.delete_object(s3_bucket, unique_filename)
|
|
|
|
end
|
2013-11-05 13:04:47 -05:00
|
|
|
|
2013-11-27 16:01:41 -05:00
|
|
|
def update_tombstone_lifecycle(grace_period)
|
|
|
|
# cf. http://docs.aws.amazon.com/AmazonS3/latest/dev/object-lifecycle-mgmt.html
|
|
|
|
fog_with_options.put_bucket_lifecycle(s3_bucket, lifecycle(grace_period))
|
|
|
|
end
|
2013-11-05 13:04:47 -05:00
|
|
|
|
2013-11-27 16:01:41 -05:00
|
|
|
def lifecycle(grace_period)
|
|
|
|
{
|
|
|
|
"Rules" => [{
|
|
|
|
"Prefix" => tombstone_prefix,
|
|
|
|
"Enabled" => true,
|
|
|
|
"Expiration" => { "Days" => grace_period }
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def tombstone_prefix
|
|
|
|
"tombstone/"
|
2013-11-05 13:04:47 -05:00
|
|
|
end
|
2013-07-31 17:26:34 -04:00
|
|
|
|
2013-12-16 05:44:59 -05:00
|
|
|
# cf. https://github.com/aws/aws-sdk-core-ruby/blob/master/lib/aws/plugins/s3_bucket_dns.rb#L56-L78
|
|
|
|
def dns_compatible?(bucket_name, ssl)
|
|
|
|
if valid_subdomain?(bucket_name)
|
|
|
|
bucket_name.match(/\./) && ssl ? false : true
|
|
|
|
else
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def valid_subdomain?(bucket_name)
|
|
|
|
bucket_name.size < 64 &&
|
|
|
|
bucket_name =~ /^[a-z0-9][a-z0-9.-]+[a-z0-9]$/ &&
|
|
|
|
bucket_name !~ /(\d+\.){3}\d+/ &&
|
|
|
|
bucket_name !~ /[.-]{2}/
|
|
|
|
end
|
|
|
|
|
2013-07-31 17:26:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|