Merge pull request #18 from anatawa12/deflate64-next

Deflate64
This commit is contained in:
Chris Hennick 2024-02-28 15:18:49 -08:00 committed by GitHub
commit 7e6c5c38d9
Signed by: DevComp
GPG key ID: B5690EEEBB952194
10 changed files with 72 additions and 0 deletions

View file

@ -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"

View file

@ -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`.

View file

@ -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")]

View file

@ -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);

View file

@ -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
View 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

Binary file not shown.

Binary file not shown.

21
tests/deflate64.rs Normal file
View 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[..]);
}

View file

@ -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());