Как распаковать zip-папку с помощью rubyzip - PullRequest
6 голосов
/ 16 мая 2011

Я знаю, как извлечь содержимое обычного zip-файла с помощью rubyzip. Но у меня возникли проблемы с распаковыванием содержимого архива, и я надеюсь, что любой из вас, ребята, сможет мне помочь.

это код, который я использую для распаковки:

Zip::ZipFile::open(@file_location) do |zip|
 zip.each do |entry|
  next if entry.name =~ /__MACOSX/ or entry.name =~ /\.DS_Store/ or !entry.file?
  logger.debug "#{entry.name}"
  @data = File.new("#{Rails.root.to_s}/tmp/#{entry.name}")
 end
end

entry.name дает мне имя файла внутри zip-файла. Это прекрасно работает с обычным zipfile. Но когда zip-файл создается из папки, тогда имена записей выглядят примерно так: test-folder / test.pdf. Когда я тогда пытаюсь создать файл, он говорит мне, что файл не может быть найден. Вероятно, это связано с тем, что она находится внутри папки «test», которая находится внутри zip.

Если я проверяю запись как папку, папка не может быть найдена. Поэтому я решил, что решением будет прочитать запись в виде потока, а затем сохранить ее в виде файла. Получить входной поток легко, но как мне сохранить его в виде файла? Это то, что я получил до сих пор.

Zip::ZipFile::open(@file_location) do |zip|
 zip.each do |entry|
  next if entry.name =~ /__MACOSX/ or entry.name =~ /\.DS_Store/ or !entry.file?
  logger.debug "#{entry.name}"
  @data = entry.get_input_stream.read
  # How do i create a file from a stream?
 end
end

В основном мой вопрос: как я могу создать файл из потока? Или есть более простой подход к этому, чем мой?

=== РЕДАКТИРОВАТЬ === Я использую скрепку для хранения файлов.

Ответы [ 3 ]

3 голосов
/ 15 апреля 2015

Я обнаружил, что более простой подход, основанный на jhwist, работает нормально:

Zip::File.open(@file_location) do |zipfile|
  zipfile.each do |entry|
    # The 'next if...' code can go here, though I didn't use it
    unless File.exist?(entry.name)
      FileUtils::mkdir_p(File.dirname(entry.name))
      zipfile.extract(entry, entry.name) 
    end
  end
end

Условное условие явно необязательно, но без него код выдаст ошибку, если попытается перезаписать существующий файл.*

0 голосов
/ 17 мая 2011

Я решил это с помощью потока и создания StringIO. Вот код

Zip::ZipFile::open(@file_location) do |zip|
 zip.each do |entry|
  next if entry.name =~ /__MACOSX/ or entry.name =~ /\.DS_Store/ or !entry.file?

  begin
   # the normal unzip-code
  rescue Errno::ENOENT
   # when the entry can not be found
   @data = entry.get_input_stream.read
   @file = StringIO.new(@data)
   @file.class.class_eval { attr_accessor :original_filename, :content_type }
   @file.original_filename = entry.name
   @file.content_type = MIME::Types.type_for(entry.name)

   # save it / whatever
  end
 end
end
0 голосов
/ 16 мая 2011

Я думаю, что ваша проблема не в том, нужно ли вам записывать файл из потока или нет.По сути, если вы позвоните File.new, создаст новый поток ввода-вывода (File - это подкласс IO).Поэтому все, что вы хотите сделать с потоком из zip-файла, также должно работать с обычным файлом.

Когда вы говорите

Когда я тогда пытаюсь создать файл, он говорит мнефайл не может быть найден

Я думаю, что происходит то, что родительский каталог для файла, который вы хотите создать, не существует (в вашем случае test-folder).Что вы хотите сделать, это что-то вроде этого (не проверено):

Zip::ZipFile::open(@file_location) do |zip|
 zip.each do |entry|
   next if entry.name =~ /__MACOSX/ or entry.name =~ /\.DS_Store/ or !entry.file?
   logger.debug "#{entry.name}"
   FileUtils::mkdir_p(File.dirname(entry.name)) # might want to check if it already exists    
   @data = File.new("#{Rails.root.to_s}/tmp/#{entry.name}")
 end
end
...