d84650917a
it will also automatically prune empty directories for `FileUploader`-based uploaders.
164 lines
3.7 KiB
Ruby
164 lines
3.7 KiB
Ruby
# This class breaks the actual CarrierWave concept.
|
|
# Every uploader should use a base_dir that is model agnostic so we can build
|
|
# back URLs from base_dir-relative paths saved in the `Upload` model.
|
|
#
|
|
# As the `.base_dir` is model dependent and **not** saved in the upload model (see #upload_path)
|
|
# there is no way to build back the correct file path without the model, which defies
|
|
# CarrierWave way of storing files.
|
|
#
|
|
class FileUploader < GitlabUploader
|
|
include UploaderHelper
|
|
include RecordsUploads::Concern
|
|
|
|
MARKDOWN_PATTERN = %r{\!?\[.*?\]\(/uploads/(?<secret>[0-9a-f]{32})/(?<file>.*?)\)}
|
|
DYNAMIC_PATH_PATTERN = %r{(?<secret>\h{32})/(?<identifier>.*)}
|
|
|
|
storage :file
|
|
|
|
after :remove, :prune_store_dir
|
|
|
|
def self.root
|
|
File.join(options.storage_path, 'uploads')
|
|
end
|
|
|
|
def self.absolute_path(upload)
|
|
File.join(
|
|
absolute_base_dir(upload.model),
|
|
upload.path # already contain the dynamic_segment, see #upload_path
|
|
)
|
|
end
|
|
|
|
def self.base_dir(model)
|
|
model_path_segment(model)
|
|
end
|
|
|
|
# used in migrations and import/exports
|
|
def self.absolute_base_dir(model)
|
|
File.join(root, base_dir(model))
|
|
end
|
|
|
|
# Returns the part of `store_dir` that can change based on the model's current
|
|
# path
|
|
#
|
|
# This is used to build Upload paths dynamically based on the model's current
|
|
# namespace and path, allowing us to ignore renames or transfers.
|
|
#
|
|
# model - Object that responds to `full_path` and `disk_path`
|
|
#
|
|
# Returns a String without a trailing slash
|
|
def self.model_path_segment(model)
|
|
if model.hashed_storage?(:attachments)
|
|
model.disk_path
|
|
else
|
|
model.full_path
|
|
end
|
|
end
|
|
|
|
def self.upload_path(secret, identifier)
|
|
File.join(secret, identifier)
|
|
end
|
|
|
|
def self.generate_secret
|
|
SecureRandom.hex
|
|
end
|
|
|
|
attr_accessor :model
|
|
|
|
def initialize(model, mounted_as = nil, **uploader_context)
|
|
super(model, nil, **uploader_context)
|
|
|
|
@model = model
|
|
apply_context!(uploader_context)
|
|
end
|
|
|
|
def base_dir
|
|
self.class.base_dir(@model)
|
|
end
|
|
|
|
# we don't need to know the actual path, an uploader instance should be
|
|
# able to yield the file content on demand, so we should build the digest
|
|
def absolute_path
|
|
self.class.absolute_path(@upload)
|
|
end
|
|
|
|
def upload_path
|
|
self.class.upload_path(dynamic_segment, identifier)
|
|
end
|
|
|
|
def model_path_segment
|
|
self.class.model_path_segment(@model)
|
|
end
|
|
|
|
def store_dir
|
|
File.join(base_dir, dynamic_segment)
|
|
end
|
|
|
|
def markdown_link
|
|
markdown = "[#{markdown_name}](#{secure_url})"
|
|
markdown.prepend("!") if image_or_video? || dangerous?
|
|
markdown
|
|
end
|
|
|
|
def to_h
|
|
{
|
|
alt: markdown_name,
|
|
url: secure_url,
|
|
markdown: markdown_link
|
|
}
|
|
end
|
|
|
|
def filename
|
|
self.file.filename
|
|
end
|
|
|
|
def upload=(value)
|
|
super
|
|
|
|
return unless value
|
|
return if apply_context!(value.uploader_context)
|
|
|
|
# fallback to the regex based extraction
|
|
if matches = DYNAMIC_PATH_PATTERN.match(value.path)
|
|
@secret = matches[:secret]
|
|
@identifier = matches[:identifier]
|
|
end
|
|
end
|
|
|
|
def secret
|
|
@secret ||= self.class.generate_secret
|
|
end
|
|
|
|
private
|
|
|
|
def apply_context!(uploader_context)
|
|
@secret, @identifier = uploader_context.values_at(:secret, :identifier)
|
|
|
|
!!(@secret && @identifier)
|
|
end
|
|
|
|
def build_upload
|
|
super.tap do |upload|
|
|
upload.secret = secret
|
|
end
|
|
end
|
|
|
|
def prune_store_dir
|
|
storage.delete_dir!(store_dir) # only remove when empty
|
|
end
|
|
|
|
def markdown_name
|
|
(image_or_video? ? File.basename(filename, File.extname(filename)) : filename).gsub("]", "\\]")
|
|
end
|
|
|
|
def identifier
|
|
@identifier ||= filename
|
|
end
|
|
|
|
def dynamic_segment
|
|
secret
|
|
end
|
|
|
|
def secure_url
|
|
File.join('/uploads', @secret, file.filename)
|
|
end
|
|
end
|