# # Very dirty and inefficient In Memory File System, mainly for tests. # module Vfs module Storages class HashFs < Hash def initialize super self['/'] = {dir: true} end def open_fs &block block.call self end # # Attributes # def attributes path base, name = split_path path # if path == '/' # return {dir: true, file: false} # end # stat = cd(base)[name] attrs = {} attrs[:file] = !!stat[:file] attrs[:dir] = !!stat[:dir] attrs rescue Exception {} end def set_attributes path, attrs raise 'not supported' end # # File # def read_file path, &block base, name = split_path path assert cd(base)[name], :include?, :file block.call cd(base)[name][:content] end def write_file path, append, &block base, name = split_path path os = if append file = cd(base)[name] file ? file[:content] : '' else assert_not cd(base), :include?, name '' end writer = -> buff {os << buff} block.call writer cd(base)[name] = {file: true, content: os} end def delete_file path base, name = split_path path assert cd(base)[name], :include?, :file cd(base).delete name end # def move_file path # raise 'not supported' # end # # Dir # def create_dir path base, name = split_path path assert_not cd(base), :include?, name cd(base)[name] = {dir: true} end def delete_dir path base, name = split_path path assert cd(base)[name], :include?, :dir # empty = true # cd(base)[name].each do |key, value| # empty = false if key.is_a? String # end # raise 'you are trying to delete not empty dir!' unless empty cd(base).delete name end # def move_dir path # raise 'not supported' # end def each_entry path, &block base, name = split_path path assert cd(base)[name], :include?, :dir cd(base)[name].each do |relative_name, content| next if relative_name.is_a? Symbol if content[:dir] block.call relative_name, :dir else block.call relative_name, :file end end end def efficient_dir_copy from, to, override from.storage.open_fs do |from_fs| to.storage.open_fs do |to_fs| if from_fs == to_fs for_spec_helper_effective_copy_used _efficient_dir_copy from.path, to.path, override true else false end end end end def _efficient_dir_copy from_path, to_path, override from_base, from_name = split_path from_path assert cd(from_base)[from_name], :include?, :dir to_base, to_name = split_path to_path # assert_not cd(to_base), :include?, to_name if cd(to_base).include? to_name if cd(to_base)[to_name][:dir] each_entry from_path do |name, type| if type == :dir _efficient_dir_copy "#{from_path}/#{name}", "#{to_path}/#{name}", override else raise "file #{to_path}/#{name} already exist!" if cd(to_base)[to_name].include?(name) and !override cd(to_base)[to_name][name] = cd(from_base)[from_name][name].clone end end else raise "can't copy dir #{from_path} to file #{to_path}!" end else cd(to_base)[to_name] = {dir: true} _efficient_dir_copy from_path, to_path, override end end protected :_efficient_dir_copy def for_spec_helper_effective_copy_used; end # def upload_directory from_local_path, to_remote_path # FileUtils.cp_r from_local_path, to_remote_path # end # # def download_directory from_remote_path, to_local_path # FileUtils.cp_r from_remote_path, to_local_path # end # # Other # def local?; true end def to_s; 'hash_fs' end def tmp &block tmp_dir = "/tmp_#{rand(10**6)}" create_dir tmp_dir if block begin block.call tmp_dir ensure delete_dir tmp_dir end else tmp_dir end end protected def assert obj, method, arg raise "#{obj} should #{method} #{arg}" unless obj.send method, arg end def assert_not obj, method, arg raise "#{obj} should not #{method} #{arg}" if obj.send method, arg end def split_path path parts = path[1..-1].split('/') parts.unshift '/' name = parts.pop return parts, name end def cd parts current = self iterator = parts.clone while iterator.first current = current[iterator.first] iterator.shift end current end end end end