commit
7e6c5c38d9
10 changed files with 72 additions and 0 deletions
|
@ -26,6 +26,7 @@ sha1 = {version = "0.10.5", optional = true }
|
||||||
time = { version = "0.3.22", optional = true, default-features = false, features = ["std"] }
|
time = { version = "0.3.22", optional = true, default-features = false, features = ["std"] }
|
||||||
zstd = { version = "0.12.3", optional = true, default-features = false }
|
zstd = { version = "0.12.3", optional = true, default-features = false }
|
||||||
zopfli = { version = "0.7.4", optional = true }
|
zopfli = { version = "0.7.4", optional = true }
|
||||||
|
deflate64 = { version = "0.1.5", optional = true }
|
||||||
|
|
||||||
[target.'cfg(any(all(target_arch = "arm", target_pointer_width = "32"), target_arch = "mips", target_arch = "powerpc"))'.dependencies]
|
[target.'cfg(any(all(target_arch = "arm", target_pointer_width = "32"), target_arch = "mips", target_arch = "powerpc"))'.dependencies]
|
||||||
crossbeam-utils = "0.8.16"
|
crossbeam-utils = "0.8.16"
|
||||||
|
|
|
@ -17,6 +17,7 @@ Supported compression formats:
|
||||||
|
|
||||||
* stored (i.e. none)
|
* stored (i.e. none)
|
||||||
* deflate
|
* deflate
|
||||||
|
* deflate64 (decompression only)
|
||||||
* bzip2
|
* bzip2
|
||||||
* zstd
|
* zstd
|
||||||
|
|
||||||
|
@ -52,6 +53,7 @@ The features available are:
|
||||||
This is the fastest `deflate` implementation available.
|
This is the fastest `deflate` implementation available.
|
||||||
* `deflate-zopfli`: Enables deflating files with the `zopfli` library (used when compression quality is 10..=264). This
|
* `deflate-zopfli`: Enables deflating files with the `zopfli` library (used when compression quality is 10..=264). This
|
||||||
is the most effective `deflate` implementation available.
|
is the most effective `deflate` implementation available.
|
||||||
|
* `deflate64`: Enables the deflate64 compression algorithm. Decompression is only supported.
|
||||||
* `bzip2`: Enables the BZip2 compression algorithm.
|
* `bzip2`: Enables the BZip2 compression algorithm.
|
||||||
* `time`: Enables features using the [time](https://github.com/rust-lang-deprecated/time) crate.
|
* `time`: Enables features using the [time](https://github.com/rust-lang-deprecated/time) crate.
|
||||||
* `chrono`: Enables converting last-modified `zip_next::DateTime` to and from `chrono::NaiveDateTime`.
|
* `chrono`: Enables converting last-modified `zip_next::DateTime` to and from `chrono::NaiveDateTime`.
|
||||||
|
|
|
@ -25,6 +25,10 @@ pub enum CompressionMethod {
|
||||||
feature = "deflate-zopfli"
|
feature = "deflate-zopfli"
|
||||||
))]
|
))]
|
||||||
Deflated,
|
Deflated,
|
||||||
|
/// Compress the file using Deflate64.
|
||||||
|
/// Decoding deflate64 is supported but encoding deflate64 is not supported.
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
Deflate64,
|
||||||
/// Compress the file using BZIP2
|
/// Compress the file using BZIP2
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
Bzip2,
|
Bzip2,
|
||||||
|
@ -70,6 +74,9 @@ impl CompressionMethod {
|
||||||
feature = "deflate-zopfli"
|
feature = "deflate-zopfli"
|
||||||
)))]
|
)))]
|
||||||
pub const DEFLATE: Self = CompressionMethod::Unsupported(8);
|
pub const DEFLATE: Self = CompressionMethod::Unsupported(8);
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
pub const DEFLATE64: Self = CompressionMethod::Deflate64;
|
||||||
|
#[cfg(not(feature = "deflate64"))]
|
||||||
pub const DEFLATE64: Self = CompressionMethod::Unsupported(9);
|
pub const DEFLATE64: Self = CompressionMethod::Unsupported(9);
|
||||||
pub const PKWARE_IMPLODE: Self = CompressionMethod::Unsupported(10);
|
pub const PKWARE_IMPLODE: Self = CompressionMethod::Unsupported(10);
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
|
@ -112,6 +119,8 @@ impl CompressionMethod {
|
||||||
feature = "deflate-zopfli"
|
feature = "deflate-zopfli"
|
||||||
))]
|
))]
|
||||||
8 => CompressionMethod::Deflated,
|
8 => CompressionMethod::Deflated,
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
9 => CompressionMethod::Deflate64,
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
12 => CompressionMethod::Bzip2,
|
12 => CompressionMethod::Bzip2,
|
||||||
#[cfg(feature = "zstd")]
|
#[cfg(feature = "zstd")]
|
||||||
|
@ -140,6 +149,8 @@ impl CompressionMethod {
|
||||||
feature = "deflate-zopfli"
|
feature = "deflate-zopfli"
|
||||||
))]
|
))]
|
||||||
CompressionMethod::Deflated => 8,
|
CompressionMethod::Deflated => 8,
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
CompressionMethod::Deflate64 => 9,
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
CompressionMethod::Bzip2 => 12,
|
CompressionMethod::Bzip2 => 12,
|
||||||
#[cfg(feature = "aes-crypto")]
|
#[cfg(feature = "aes-crypto")]
|
||||||
|
@ -219,6 +230,8 @@ pub const SUPPORTED_COMPRESSION_METHODS: &[CompressionMethod] = &[
|
||||||
feature = "deflate-zopfli"
|
feature = "deflate-zopfli"
|
||||||
))]
|
))]
|
||||||
CompressionMethod::Deflated,
|
CompressionMethod::Deflated,
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
CompressionMethod::Deflate64,
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
CompressionMethod::Bzip2,
|
CompressionMethod::Bzip2,
|
||||||
#[cfg(feature = "zstd")]
|
#[cfg(feature = "zstd")]
|
||||||
|
|
14
src/read.rs
14
src/read.rs
|
@ -24,6 +24,9 @@ use std::sync::Arc;
|
||||||
))]
|
))]
|
||||||
use flate2::read::DeflateDecoder;
|
use flate2::read::DeflateDecoder;
|
||||||
|
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
use deflate64::Deflate64Decoder;
|
||||||
|
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
use bzip2::read::BzDecoder;
|
use bzip2::read::BzDecoder;
|
||||||
|
|
||||||
|
@ -132,6 +135,8 @@ pub(crate) enum ZipFileReader<'a> {
|
||||||
feature = "deflate-zlib-ng"
|
feature = "deflate-zlib-ng"
|
||||||
))]
|
))]
|
||||||
Deflated(Crc32Reader<DeflateDecoder<CryptoReader<'a>>>),
|
Deflated(Crc32Reader<DeflateDecoder<CryptoReader<'a>>>),
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
Deflate64(Crc32Reader<Deflate64Decoder<io::BufReader<CryptoReader<'a>>>>),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
Bzip2(Crc32Reader<BzDecoder<CryptoReader<'a>>>),
|
Bzip2(Crc32Reader<BzDecoder<CryptoReader<'a>>>),
|
||||||
#[cfg(feature = "zstd")]
|
#[cfg(feature = "zstd")]
|
||||||
|
@ -151,6 +156,8 @@ impl<'a> Read for ZipFileReader<'a> {
|
||||||
feature = "deflate-zlib-ng"
|
feature = "deflate-zlib-ng"
|
||||||
))]
|
))]
|
||||||
ZipFileReader::Deflated(r) => r.read(buf),
|
ZipFileReader::Deflated(r) => r.read(buf),
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
ZipFileReader::Deflate64(r) => r.read(buf),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
ZipFileReader::Bzip2(r) => r.read(buf),
|
ZipFileReader::Bzip2(r) => r.read(buf),
|
||||||
#[cfg(feature = "zstd")]
|
#[cfg(feature = "zstd")]
|
||||||
|
@ -173,6 +180,8 @@ impl<'a> ZipFileReader<'a> {
|
||||||
feature = "deflate-zlib-ng"
|
feature = "deflate-zlib-ng"
|
||||||
))]
|
))]
|
||||||
ZipFileReader::Deflated(r) => r.into_inner().into_inner().into_inner(),
|
ZipFileReader::Deflated(r) => r.into_inner().into_inner().into_inner(),
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
ZipFileReader::Deflate64(r) => r.into_inner().into_inner().into_inner().into_inner(),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
ZipFileReader::Bzip2(r) => r.into_inner().into_inner().into_inner(),
|
ZipFileReader::Bzip2(r) => r.into_inner().into_inner().into_inner(),
|
||||||
#[cfg(feature = "zstd")]
|
#[cfg(feature = "zstd")]
|
||||||
|
@ -283,6 +292,11 @@ pub(crate) fn make_reader(
|
||||||
let deflate_reader = DeflateDecoder::new(reader);
|
let deflate_reader = DeflateDecoder::new(reader);
|
||||||
ZipFileReader::Deflated(Crc32Reader::new(deflate_reader, crc32, ae2_encrypted))
|
ZipFileReader::Deflated(Crc32Reader::new(deflate_reader, crc32, ae2_encrypted))
|
||||||
}
|
}
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
CompressionMethod::Deflate64 => {
|
||||||
|
let deflate64_reader = Deflate64Decoder::new(reader);
|
||||||
|
ZipFileReader::Deflate64(Crc32Reader::new(deflate64_reader, crc32, ae2_encrypted))
|
||||||
|
}
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
CompressionMethod::Bzip2 => {
|
CompressionMethod::Bzip2 => {
|
||||||
let bzip2_reader = BzDecoder::new(reader);
|
let bzip2_reader = BzDecoder::new(reader);
|
||||||
|
|
|
@ -1273,6 +1273,12 @@ impl<W: Write + Seek> GenericZipWriter<W> {
|
||||||
}
|
}
|
||||||
unreachable!()
|
unreachable!()
|
||||||
}
|
}
|
||||||
|
#[cfg(feature = "deflate64")]
|
||||||
|
CompressionMethod::Deflate64 => {
|
||||||
|
Err(ZipError::UnsupportedArchive(
|
||||||
|
"Compressing Deflate64 is not supported",
|
||||||
|
))
|
||||||
|
}
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
CompressionMethod::Bzip2 => {
|
CompressionMethod::Bzip2 => {
|
||||||
let level = clamp_opt(
|
let level = clamp_opt(
|
||||||
|
|
6
tests/data/README.md
Normal file
6
tests/data/README.md
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
A few assets in directory is copied from [dotnet-assets].
|
||||||
|
|
||||||
|
- [`deflate64.zip`](./deflate64.zip) is originally at https://github.com/dotnet/runtime-assets/blob/95277f38e68b66f1b48600d90d456c32c9ae0fa2/src/System.IO.Compression.TestData/ZipTestData/compat/deflate64.zip
|
||||||
|
- [`folder/binary.wmv`](./folder/binary.wmv) is originally at https://github.com/dotnet/runtime-assets/tree/95277f38e68b66f1b48600d90d456c32c9ae0fa2/src/System.IO.Compression.TestData/ZipTestData/refzipfolders/normal/binary.wmv
|
||||||
|
|
||||||
|
[dotnet-assets]: https://github.com/dotnet/runtime-assets
|
BIN
tests/data/deflate64.zip
Normal file
BIN
tests/data/deflate64.zip
Normal file
Binary file not shown.
BIN
tests/data/folder/binary.wmv
Normal file
BIN
tests/data/folder/binary.wmv
Normal file
Binary file not shown.
21
tests/deflate64.rs
Normal file
21
tests/deflate64.rs
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
#![cfg(feature = "deflate64")]
|
||||||
|
|
||||||
|
use std::io::{self, Read};
|
||||||
|
use zip_next::ZipArchive;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn decompress_deflate64() {
|
||||||
|
let mut v = Vec::new();
|
||||||
|
v.extend_from_slice(include_bytes!("data/deflate64.zip"));
|
||||||
|
let mut archive = ZipArchive::new(io::Cursor::new(v)).expect("couldn't open test zip file");
|
||||||
|
|
||||||
|
let mut file = archive
|
||||||
|
.by_name("binary.wmv")
|
||||||
|
.expect("couldn't find file in archive");
|
||||||
|
assert_eq!("binary.wmv", file.name());
|
||||||
|
|
||||||
|
let mut content = Vec::new();
|
||||||
|
file.read_to_end(&mut content)
|
||||||
|
.expect("couldn't read encrypted and compressed file");
|
||||||
|
assert_eq!(include_bytes!("data/folder/binary.wmv"), &content[..]);
|
||||||
|
}
|
|
@ -12,6 +12,9 @@ use zip_next::{CompressionMethod, ZipWriter, SUPPORTED_COMPRESSION_METHODS};
|
||||||
#[test]
|
#[test]
|
||||||
fn end_to_end() {
|
fn end_to_end() {
|
||||||
for &method in SUPPORTED_COMPRESSION_METHODS {
|
for &method in SUPPORTED_COMPRESSION_METHODS {
|
||||||
|
if method == CompressionMethod::DEFLATE64 {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
let file = &mut Cursor::new(Vec::new());
|
let file = &mut Cursor::new(Vec::new());
|
||||||
|
|
||||||
println!("Writing file with {method} compression");
|
println!("Writing file with {method} compression");
|
||||||
|
@ -28,6 +31,9 @@ fn end_to_end() {
|
||||||
#[test]
|
#[test]
|
||||||
fn copy() {
|
fn copy() {
|
||||||
for &method in SUPPORTED_COMPRESSION_METHODS {
|
for &method in SUPPORTED_COMPRESSION_METHODS {
|
||||||
|
if method == CompressionMethod::DEFLATE64 {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
let src_file = &mut Cursor::new(Vec::new());
|
let src_file = &mut Cursor::new(Vec::new());
|
||||||
write_test_archive(src_file, method, false);
|
write_test_archive(src_file, method, false);
|
||||||
|
|
||||||
|
@ -67,6 +73,9 @@ fn copy() {
|
||||||
#[test]
|
#[test]
|
||||||
fn append() {
|
fn append() {
|
||||||
for &method in SUPPORTED_COMPRESSION_METHODS {
|
for &method in SUPPORTED_COMPRESSION_METHODS {
|
||||||
|
if method == CompressionMethod::DEFLATE64 {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
for shallow_copy in &[false, true] {
|
for shallow_copy in &[false, true] {
|
||||||
println!("Writing file with {method} compression, shallow_copy {shallow_copy}");
|
println!("Writing file with {method} compression, shallow_copy {shallow_copy}");
|
||||||
let mut file = &mut Cursor::new(Vec::new());
|
let mut file = &mut Cursor::new(Vec::new());
|
||||||
|
|
Loading…
Add table
Reference in a new issue