2013-08-27 04:34:17 +08:00
# rubyzip
2015-02-03 05:23:04 +08:00
[](http://badge.fury.io/rb/rubyzip)
2014-12-05 18:29:32 +08:00
[](http://travis-ci.org/rubyzip/rubyzip)
[](https://codeclimate.com/github/rubyzip/rubyzip)
[](https://coveralls.io/r/rubyzip/rubyzip?branch=master)
2002-01-03 01:48:31 +08:00
2015-04-20 21:08:06 +08:00
Rubyzip is a ruby library for reading and writing zip files.
2002-01-03 01:48:31 +08:00
2013-06-03 16:05:08 +08:00
## Important note
2015-04-20 21:08:06 +08:00
The Rubyzip interface has changed!!! No need to do `require "zip/zip"` and `Zip` prefix in class names removed.
2013-06-03 16:05:08 +08:00
2015-01-09 04:02:31 +08:00
If you have issues with any third-party gems that require an old version of rubyzip, you can use this workaround:
2013-08-31 04:37:50 +08:00
```ruby
2013-12-06 23:22:16 +08:00
gem 'rubyzip', '>= 1.0.0' # will load new rubyzip version
gem 'zip-zip' # will load compatibility for old rubyzip API.
2013-08-31 04:37:50 +08:00
```
2013-10-30 05:34:17 +08:00
## Requirements
* Ruby 1.9.2 or greater
2012-05-22 04:09:31 +08:00
## Installation
2015-04-20 21:08:06 +08:00
Rubyzip is available on RubyGems:
2002-03-22 05:12:19 +08:00
2012-04-08 05:21:02 +08:00
```
gem install rubyzip
```
2002-03-22 05:12:19 +08:00
2012-04-08 05:21:02 +08:00
Or in your Gemfile:
```ruby
gem 'rubyzip'
```
2002-01-06 06:09:48 +08:00
2012-05-22 04:09:31 +08:00
## Usage
2005-02-18 05:49:16 +08:00
2012-05-22 04:09:31 +08:00
### Basic zip archive creation
2012-03-13 07:31:55 +08:00
2012-05-22 04:09:31 +08:00
```ruby
require 'rubygems'
2013-06-03 18:42:16 +08:00
require 'zip'
2012-08-30 09:50:41 +08:00
2012-05-22 04:09:31 +08:00
folder = "Users/me/Desktop/stuff_to_zip"
input_filenames = ['image.jpg', 'description.txt', 'stats.csv']
2012-08-30 09:50:41 +08:00
2012-05-22 04:09:31 +08:00
zipfile_name = "/Users/me/Desktop/archive.zip"
2012-08-30 09:50:41 +08:00
2013-06-03 18:42:16 +08:00
Zip::File.open(zipfile_name, Zip::File::CREATE) do |zipfile|
2012-05-22 04:09:31 +08:00
input_filenames.each do |filename|
# Two arguments:
# - The name of the file as it will appear in the archive
# - The original file, including the path to find it
2017-04-04 03:15:31 +08:00
zipfile.add(filename, File.join(folder, filename))
2012-05-22 04:09:31 +08:00
end
2013-12-06 19:02:12 +08:00
zipfile.get_output_stream("myFile") { |os| os.write "myFile contains just this" }
2012-05-22 04:09:31 +08:00
end
2012-03-13 07:31:55 +08:00
```
2013-04-15 07:02:05 +08:00
### Zipping a directory recursively
2014-11-06 11:02:38 +08:00
Copy from [here ](https://github.com/rubyzip/rubyzip/blob/05916bf89181e1955118fd3ea059f18acac28cc8/samples/example_recursive.rb )
2013-04-15 07:02:05 +08:00
```ruby
2013-06-03 18:42:16 +08:00
require 'zip'
2016-03-22 18:25:11 +08:00
2014-11-06 10:38:10 +08:00
# This is a simple example which uses rubyzip to
# recursively generate a zip file from the contents of
# a specified directory. The directory itself is not
# included in the archive, rather just its contents.
#
# Usage:
2016-03-22 18:25:11 +08:00
# directory_to_zip = "/tmp/input"
# output_file = "/tmp/out.zip"
# zf = ZipFileGenerator.new(directory_to_zip, output_file)
# zf.write()
2014-11-06 10:38:10 +08:00
class ZipFileGenerator
# Initialize with the directory to zip and the location of the output archive.
2015-06-01 13:25:19 +08:00
def initialize(input_dir, output_file)
@input_dir = input_dir
@output_file = output_file
2014-11-06 10:38:10 +08:00
end
2015-06-01 13:25:19 +08:00
2014-11-06 10:38:10 +08:00
# Zip the input directory.
2015-06-01 13:25:19 +08:00
def write
entries = Dir.entries(@input_dir) - %w(. ..)
::Zip::File.open(@output_file, ::Zip::File::CREATE) do |io|
write_entries entries, '', io
end
2014-11-06 10:38:10 +08:00
end
2015-06-01 13:25:19 +08:00
2014-11-06 10:38:10 +08:00
private
2015-06-01 13:25:19 +08:00
# A helper method to make the recursion work.
def write_entries(entries, path, io)
entries.each do |e|
zip_file_path = path == '' ? e : File.join(path, e)
disk_file_path = File.join(@input_dir, zip_file_path)
puts "Deflating #{disk_file_path}"
if File.directory? disk_file_path
recursively_deflate_directory(disk_file_path, io, zip_file_path)
2014-11-06 10:38:10 +08:00
else
2015-06-01 13:25:19 +08:00
put_into_archive(disk_file_path, io, zip_file_path)
2014-08-18 18:40:31 +08:00
end
2015-06-01 13:25:19 +08:00
end
end
def recursively_deflate_directory(disk_file_path, io, zip_file_path)
io.mkdir zip_file_path
subdir = Dir.entries(disk_file_path) - %w(. ..)
write_entries subdir, zip_file_path, io
end
def put_into_archive(disk_file_path, io, zip_file_path)
io.get_output_stream(zip_file_path) do |f|
2016-03-22 18:25:11 +08:00
f.write(File.open(disk_file_path, 'rb').read)
2015-06-01 13:25:19 +08:00
end
2014-11-06 11:03:21 +08:00
end
2014-11-06 10:38:10 +08:00
end
2013-04-15 07:02:05 +08:00
```
2013-10-21 04:09:40 +08:00
### Save zip archive entries in sorted by name state
2015-04-20 21:08:06 +08:00
To save zip archives in sorted order like below, you need to set `::Zip.sort_entries` to `true`
2013-10-21 04:09:40 +08:00
```
Vegetable/
Vegetable/bean
Vegetable/carrot
Vegetable/celery
fruit/
fruit/apple
fruit/kiwi
fruit/mango
fruit/orange
```
2015-04-20 21:08:06 +08:00
After this, entries in the zip archive will be saved in ordered state.
2013-10-21 04:09:40 +08:00
2015-06-25 05:58:54 +08:00
### Default permissions of zip archives
On Posix file systems the default file permissions applied to a new archive
are (0666 - umask), which mimics the behavior of standard tools such as `touch` .
On Windows the default file permissions are set to 0644 as suggested by the
[Ruby File documentation ](http://ruby-doc.org/core-2.2.2/File.html ).
When modifying a zip archive the file permissions of the archive are preserved.
2014-03-04 23:05:56 +08:00
### Reading a Zip file
```ruby
Zip::File.open('foo.zip') do |zip_file|
# Handle entries one by one
zip_file.each do |entry|
# Extract to file/directory/symlink
puts "Extracting #{entry.name}"
entry.extract(dest_file)
# Read into memory
content = entry.get_input_stream.read
end
# Find specific entry
entry = zip_file.glob('*.csv').first
puts entry.get_input_stream.read
end
```
2015-02-17 03:51:44 +08:00
#### Notice about ::Zip::InputStream
`::Zip::InputStream` usable for fast reading zip file content because it not read Central directory.
2015-04-20 21:08:06 +08:00
But there is one exception when it is not working - General Purpose Flag Bit 3.
2015-02-17 03:51:44 +08:00
```
If bit 3 (0x08) of the general-purpose flags field is set, then the CRC-32 and file sizes are not known when the header is written. The fields in the local header are filled with zero, and the CRC-32 and size are appended in a 12-byte structure (optionally preceded by a 4-byte signature) immediately after the compressed data
```
2015-04-20 21:08:06 +08:00
If `::Zip::InputStream` finds such entry in the zip archive it will raise an exception.
2015-02-17 03:51:44 +08:00
2015-01-17 18:38:13 +08:00
### Password Protection (Experimental)
2015-04-20 21:08:06 +08:00
Rubyzip supports reading/writing zip files with traditional zip encryption (a.k.a. "ZipCrypto"). AES encryption is not yet supported. It can be used with buffer streams, e.g.:
2015-01-17 18:38:13 +08:00
```ruby
Zip::OutputStream.write_buffer(::StringIO.new(''), Zip::TraditionalEncrypter.new('password')) do |out|
out.put_next_entry("my_file.txt")
out.write my_data
end.string
```
This is an experimental feature and the interface for encryption may change in future versions.
2013-06-01 03:55:43 +08:00
## Known issues
### Modify docx file with rubyzip
Use `write_buffer` instead `open` . Thanks to @jondruse
```ruby
2013-06-03 18:42:16 +08:00
buffer = Zip::OutputStream.write_buffer do |out|
2013-06-01 03:55:43 +08:00
@zip_file .entries.each do |e|
unless [DOCUMENT_FILE_PATH, RELS_FILE_PATH].include?(e.name)
out.put_next_entry(e.name)
out.write e.get_input_stream.read
end
end
2013-08-15 06:00:27 +08:00
2013-06-01 03:55:43 +08:00
out.put_next_entry(DOCUMENT_FILE_PATH)
out.write xml_doc.to_xml(:indent => 0).gsub("\n","")
2013-08-15 06:00:27 +08:00
2013-06-01 03:55:43 +08:00
out.put_next_entry(RELS_FILE_PATH)
out.write rels.to_xml(:indent => 0).gsub("\n","")
end
2016-03-28 09:41:50 +08:00
File.open(new_path, "wb") {|f| f.write(buffer.string) }
2013-06-01 03:55:43 +08:00
```
2012-05-22 04:09:31 +08:00
## Configuration
2002-01-06 06:09:48 +08:00
2012-05-22 04:09:31 +08:00
By default, rubyzip will not overwrite files if they already exist inside of the extracted path. To change this behavior, you may specify a configuration option like so:
2003-08-18 00:28:18 +08:00
2013-08-15 06:00:27 +08:00
```ruby
Zip.on_exists_proc = true
2012-05-22 04:09:31 +08:00
```
If you're using rubyzip with rails, consider placing this snippet of code in an initializer file such as `config/initializers/rubyzip.rb`
Additionally, if you want to configure rubyzip to overwrite existing files while creating a .zip file, you can do so with the following:
2013-08-15 06:00:27 +08:00
```ruby
Zip.continue_on_exists_proc = true
2012-05-22 04:09:31 +08:00
```
2013-08-15 06:00:27 +08:00
2015-04-20 21:08:06 +08:00
If you want to store non-english names and want to open them on Windows(pre 7) you need to set this option:
2013-08-15 06:00:27 +08:00
```ruby
Zip.unicode_names = true
```
2015-04-20 21:08:06 +08:00
Some zip files might have an invalid date format, which will raise a warning. You can hide this warning with the following setting:
2014-01-20 22:23:35 +08:00
```ruby
2014-12-07 21:49:45 +08:00
Zip.warn_invalid_date = false
2014-01-20 22:23:35 +08:00
```
2014-12-07 21:49:45 +08:00
You can set the default compression level like so:
2014-12-02 15:55:30 +08:00
```ruby
2014-12-07 21:49:45 +08:00
Zip.default_compression = Zlib::DEFAULT_COMPRESSION
2014-12-02 15:55:30 +08:00
```
2014-01-20 22:23:35 +08:00
It defaults to `Zlib::DEFAULT_COMPRESSION` . Possible values are `Zlib::BEST_COMPRESSION` , `Zlib::DEFAULT_COMPRESSION` and `Zlib::NO_COMPRESSION`
2015-04-20 21:08:06 +08:00
You can set multiple settings at the same time by using a block:
2013-08-15 06:00:27 +08:00
```ruby
Zip.setup do |c|
c.on_exists_proc = true
c.continue_on_exists_proc = true
c.unicode_names = true
2014-01-20 22:23:35 +08:00
c.default_compression = Zlib::BEST_COMPRESSION
2013-08-15 06:00:27 +08:00
end
2012-05-22 04:09:31 +08:00
```
2015-04-20 21:08:06 +08:00
By default, Zip64 support is disabled for writing. To enable it do this:
2014-01-24 17:37:38 +08:00
```ruby
Zip.write_zip64_support = true
```
_NOTE_: If you will enable Zip64 writing then you will need zip extractor with Zip64 support to extract archive.
2012-05-22 04:09:31 +08:00
## Developing
2015-04-20 21:08:06 +08:00
To run the test you need to do this:
2012-05-22 04:09:31 +08:00
```
bundle install
rake
```
## Website and Project Home
2002-03-30 08:52:13 +08:00
2013-07-02 06:26:52 +08:00
http://github.com/rubyzip/rubyzip
2002-03-30 08:52:13 +08:00
2013-07-02 06:26:52 +08:00
http://rdoc.info/github/rubyzip/rubyzip/master/frames
2005-02-18 06:21:59 +08:00
2012-05-22 04:09:31 +08:00
## Authors
2002-01-05 08:37:45 +08:00
2012-04-08 05:21:02 +08:00
Alexander Simonov ( alex at simonov.me)
2010-09-24 21:22:25 +08:00
Alan Harper ( alan at aussiegeek.net)
2005-02-18 04:27:02 +08:00
Thomas Sondergaard (thomas at sondergaard.cc)
2004-01-30 23:07:56 +08:00
2006-02-26 16:57:08 +08:00
Technorama Ltd. (oss-ruby-zip at technorama.net)
2012-03-13 07:31:55 +08:00
extra-field support contributed by Tatsuki Sugiura (sugi at nemui.org)
2012-05-22 04:09:31 +08:00
## License
2015-04-20 21:08:06 +08:00
Rubyzip is distributed under the same license as ruby. See
2012-05-22 04:09:31 +08:00
http://www.ruby-lang.org/en/LICENSE.txt