2019-10-03 09:19:35 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module Compression
|
|
|
|
class Strategy
|
|
|
|
ExtractFailed = Class.new(StandardError)
|
|
|
|
DestinationFileExistsError = Class.new(StandardError)
|
|
|
|
|
|
|
|
def can_handle?(file_name)
|
|
|
|
file_name.include?(extension)
|
|
|
|
end
|
|
|
|
|
2019-10-11 13:38:10 -04:00
|
|
|
def decompress(dest_path, compressed_file_path, max_size, allow_non_root_folder: false)
|
2019-10-03 09:19:35 -04:00
|
|
|
sanitized_compressed_file_path = sanitize_path(compressed_file_path)
|
|
|
|
|
|
|
|
get_compressed_file_stream(sanitized_compressed_file_path) do |compressed_file|
|
2019-10-11 13:38:10 -04:00
|
|
|
available_size = calculate_available_size(max_size)
|
2019-10-03 09:19:35 -04:00
|
|
|
|
|
|
|
entries_of(compressed_file).each do |entry|
|
|
|
|
entry_path = build_entry_path(
|
|
|
|
compressed_file, sanitize_path(dest_path),
|
|
|
|
sanitized_compressed_file_path, entry,
|
|
|
|
allow_non_root_folder
|
|
|
|
)
|
|
|
|
|
|
|
|
if is_file?(entry)
|
|
|
|
remaining_size = extract_file(entry, entry_path, available_size)
|
|
|
|
available_size = remaining_size
|
|
|
|
else
|
|
|
|
extract_folder(entry, entry_path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-10-09 10:41:16 -04:00
|
|
|
def strip_directory(from, to, relative: false)
|
2019-10-10 10:21:36 -04:00
|
|
|
sanitized_from = sanitize_path(from) rescue nil
|
|
|
|
sanitized_to = sanitize_path(to) rescue nil
|
|
|
|
return unless sanitized_from && sanitized_to
|
|
|
|
|
2019-10-09 10:41:16 -04:00
|
|
|
glob_path = relative ? "#{sanitized_from}/*/*" : "#{sanitized_from}/**"
|
|
|
|
FileUtils.mv(Dir.glob(glob_path), sanitized_to) if File.directory?(sanitized_from)
|
|
|
|
end
|
|
|
|
|
2019-10-03 09:19:35 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def sanitize_path(filename)
|
|
|
|
Pathname.new(filename).realpath.to_s
|
|
|
|
end
|
|
|
|
|
|
|
|
# https://guides.rubyonrails.org/security.html#file-uploads
|
|
|
|
def sanitize_filename(filename)
|
|
|
|
filename.strip.tap do |name|
|
|
|
|
# NOTE: File.basename doesn't work right with Windows paths on Unix
|
|
|
|
# get only the filename, not the whole path
|
|
|
|
name.sub! /\A.*(\\|\/)/, ''
|
|
|
|
# Finally, replace all non alphanumeric, underscore
|
|
|
|
# or periods with underscore
|
|
|
|
name.gsub! /[^\w\.\-]/, '_'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-10-11 13:38:10 -04:00
|
|
|
def calculate_available_size(max_size)
|
|
|
|
1024**2 * (max_size / 1.049) # Mb to Mib
|
2019-10-03 09:19:35 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def entries_of(compressed_file)
|
|
|
|
compressed_file
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_file?(entry)
|
|
|
|
entry.file?
|
|
|
|
end
|
|
|
|
|
|
|
|
def chunk_size
|
|
|
|
@chunk_size ||= 1024**2 * 2 # 2MiB
|
|
|
|
end
|
|
|
|
|
|
|
|
def extract_file(entry, entry_path, available_size)
|
|
|
|
remaining_size = available_size
|
|
|
|
|
|
|
|
if ::File.exist?(entry_path)
|
|
|
|
raise DestinationFileExistsError, "Destination '#{entry_path}' already exists"
|
|
|
|
end
|
|
|
|
|
|
|
|
::File.open(entry_path, 'wb') do |os|
|
|
|
|
buf = ''.dup
|
|
|
|
while (buf = entry.read(chunk_size))
|
|
|
|
remaining_size -= chunk_size
|
|
|
|
raise ExtractFailed if remaining_size.negative?
|
|
|
|
os << buf
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
remaining_size
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|