Added zstd method, compiling & tests running
This commit is contained in:
parent
3fd44ffd5d
commit
6ea3d553bf
6 changed files with 54 additions and 3 deletions
|
@ -17,6 +17,7 @@ byteorder = "1.3"
|
||||||
bzip2 = { version = "0.4", optional = true }
|
bzip2 = { version = "0.4", optional = true }
|
||||||
crc32fast = "1.0"
|
crc32fast = "1.0"
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
|
zstd = { version = "0.8", optional = true }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
bencher = "0.1"
|
bencher = "0.1"
|
||||||
|
@ -28,7 +29,7 @@ deflate = ["flate2/rust_backend"]
|
||||||
deflate-miniz = ["flate2/default"]
|
deflate-miniz = ["flate2/default"]
|
||||||
deflate-zlib = ["flate2/zlib"]
|
deflate-zlib = ["flate2/zlib"]
|
||||||
unreserved = []
|
unreserved = []
|
||||||
default = ["bzip2", "deflate", "time"]
|
default = ["bzip2", "deflate", "time", "zstd"]
|
||||||
|
|
||||||
[[bench]]
|
[[bench]]
|
||||||
name = "read_entry"
|
name = "read_entry"
|
||||||
|
|
|
@ -17,6 +17,7 @@ Supported compression formats:
|
||||||
* stored (i.e. none)
|
* stored (i.e. none)
|
||||||
* deflate
|
* deflate
|
||||||
* bzip2
|
* bzip2
|
||||||
|
* zstd (in progress...)
|
||||||
|
|
||||||
Currently unsupported zip extensions:
|
Currently unsupported zip extensions:
|
||||||
|
|
||||||
|
@ -42,9 +43,10 @@ zip = { version = "0.5", default-features = false }
|
||||||
|
|
||||||
The features available are:
|
The features available are:
|
||||||
|
|
||||||
* `deflate`: Enables the deflate compression algorithm, which is the default for zipfiles
|
* `deflate`: Enables the deflate compression algorithm, which is the default for zipfiles.
|
||||||
* `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.
|
||||||
|
* `zstd`: Enables the Zstandard compression algorithm.
|
||||||
|
|
||||||
All of these are enabled by default.
|
All of these are enabled by default.
|
||||||
|
|
||||||
|
|
|
@ -32,6 +32,11 @@ const METHOD_BZIP2: Option<zip::CompressionMethod> = Some(zip::CompressionMethod
|
||||||
#[cfg(not(feature = "bzip2"))]
|
#[cfg(not(feature = "bzip2"))]
|
||||||
const METHOD_BZIP2: Option<zip::CompressionMethod> = None;
|
const METHOD_BZIP2: Option<zip::CompressionMethod> = None;
|
||||||
|
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
const METHOD_ZSTD: Option<zip::CompressionMethod> = Some(zip::CompressionMethod::Zstd);
|
||||||
|
#[cfg(not(feature = "zstd"))]
|
||||||
|
const METHOD_ZSTD: Option<zip::CompressionMethod> = None;
|
||||||
|
|
||||||
fn real_main() -> i32 {
|
fn real_main() -> i32 {
|
||||||
let args: Vec<_> = std::env::args().collect();
|
let args: Vec<_> = std::env::args().collect();
|
||||||
if args.len() < 3 {
|
if args.len() < 3 {
|
||||||
|
@ -44,7 +49,7 @@ fn real_main() -> i32 {
|
||||||
|
|
||||||
let src_dir = &*args[1];
|
let src_dir = &*args[1];
|
||||||
let dst_file = &*args[2];
|
let dst_file = &*args[2];
|
||||||
for &method in [METHOD_STORED, METHOD_DEFLATED, METHOD_BZIP2].iter() {
|
for &method in [METHOD_STORED, METHOD_DEFLATED, METHOD_BZIP2, METHOD_ZSTD].iter() {
|
||||||
if method.is_none() {
|
if method.is_none() {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,9 @@ pub enum CompressionMethod {
|
||||||
/// Compress the file using BZIP2
|
/// Compress the file using BZIP2
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
Bzip2,
|
Bzip2,
|
||||||
|
/// Compress the file using ZStandard
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
Zstd,
|
||||||
/// Unsupported compression method
|
/// Unsupported compression method
|
||||||
#[deprecated(since = "0.5.7", note = "use the constants instead")]
|
#[deprecated(since = "0.5.7", note = "use the constants instead")]
|
||||||
Unsupported(u16),
|
Unsupported(u16),
|
||||||
|
@ -60,6 +63,9 @@ impl CompressionMethod {
|
||||||
pub const IBM_ZOS_CMPSC: Self = CompressionMethod::Unsupported(16);
|
pub const IBM_ZOS_CMPSC: Self = CompressionMethod::Unsupported(16);
|
||||||
pub const IBM_TERSE: Self = CompressionMethod::Unsupported(18);
|
pub const IBM_TERSE: Self = CompressionMethod::Unsupported(18);
|
||||||
pub const ZSTD_DEPRECATED: Self = CompressionMethod::Unsupported(20);
|
pub const ZSTD_DEPRECATED: Self = CompressionMethod::Unsupported(20);
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
pub const ZSTD: Self = CompressionMethod::Zstd;
|
||||||
|
#[cfg(not(feature = "zstd"))]
|
||||||
pub const ZSTD: Self = CompressionMethod::Unsupported(93);
|
pub const ZSTD: Self = CompressionMethod::Unsupported(93);
|
||||||
pub const MP3: Self = CompressionMethod::Unsupported(94);
|
pub const MP3: Self = CompressionMethod::Unsupported(94);
|
||||||
pub const XZ: Self = CompressionMethod::Unsupported(95);
|
pub const XZ: Self = CompressionMethod::Unsupported(95);
|
||||||
|
@ -85,6 +91,8 @@ impl CompressionMethod {
|
||||||
8 => CompressionMethod::Deflated,
|
8 => CompressionMethod::Deflated,
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
12 => CompressionMethod::Bzip2,
|
12 => CompressionMethod::Bzip2,
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
93 => CompressionMethod::Zstd,
|
||||||
|
|
||||||
v => CompressionMethod::Unsupported(v),
|
v => CompressionMethod::Unsupported(v),
|
||||||
}
|
}
|
||||||
|
@ -107,6 +115,9 @@ impl CompressionMethod {
|
||||||
CompressionMethod::Deflated => 8,
|
CompressionMethod::Deflated => 8,
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
CompressionMethod::Bzip2 => 12,
|
CompressionMethod::Bzip2 => 12,
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
CompressionMethod::Zstd => 93,
|
||||||
|
|
||||||
CompressionMethod::Unsupported(v) => v,
|
CompressionMethod::Unsupported(v) => v,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -145,6 +156,9 @@ mod test {
|
||||||
methods.push(CompressionMethod::Deflated);
|
methods.push(CompressionMethod::Deflated);
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
methods.push(CompressionMethod::Bzip2);
|
methods.push(CompressionMethod::Bzip2);
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
methods.push(CompressionMethod::Zstd);
|
||||||
|
|
||||||
methods
|
methods
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
14
src/read.rs
14
src/read.rs
|
@ -24,6 +24,9 @@ use flate2::read::DeflateDecoder;
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
use bzip2::read::BzDecoder;
|
use bzip2::read::BzDecoder;
|
||||||
|
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
use zstd::stream::read::Decoder as ZstdDecoder;
|
||||||
|
|
||||||
mod ffi {
|
mod ffi {
|
||||||
pub const S_IFDIR: u32 = 0o0040000;
|
pub const S_IFDIR: u32 = 0o0040000;
|
||||||
pub const S_IFREG: u32 = 0o0100000;
|
pub const S_IFREG: u32 = 0o0100000;
|
||||||
|
@ -90,6 +93,8 @@ enum ZipFileReader<'a> {
|
||||||
Deflated(Crc32Reader<flate2::read::DeflateDecoder<CryptoReader<'a>>>),
|
Deflated(Crc32Reader<flate2::read::DeflateDecoder<CryptoReader<'a>>>),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
Bzip2(Crc32Reader<BzDecoder<CryptoReader<'a>>>),
|
Bzip2(Crc32Reader<BzDecoder<CryptoReader<'a>>>),
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
Zstd(Crc32Reader<ZstdDecoder<'a, io::BufReader<CryptoReader<'a>>>>),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Read for ZipFileReader<'a> {
|
impl<'a> Read for ZipFileReader<'a> {
|
||||||
|
@ -106,6 +111,8 @@ impl<'a> Read for ZipFileReader<'a> {
|
||||||
ZipFileReader::Deflated(r) => r.read(buf),
|
ZipFileReader::Deflated(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")]
|
||||||
|
ZipFileReader::Zstd(r) => r.read(buf),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -125,6 +132,8 @@ impl<'a> ZipFileReader<'a> {
|
||||||
ZipFileReader::Deflated(r) => r.into_inner().into_inner().into_inner(),
|
ZipFileReader::Deflated(r) => r.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")]
|
||||||
|
ZipFileReader::Zstd(r) => r.into_inner().finish().into_inner().into_inner(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -210,6 +219,11 @@ fn make_reader<'a>(
|
||||||
let bzip2_reader = BzDecoder::new(reader);
|
let bzip2_reader = BzDecoder::new(reader);
|
||||||
ZipFileReader::Bzip2(Crc32Reader::new(bzip2_reader, crc32))
|
ZipFileReader::Bzip2(Crc32Reader::new(bzip2_reader, crc32))
|
||||||
}
|
}
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
CompressionMethod::Zstd => {
|
||||||
|
let zstd_reader = ZstdDecoder::new(reader).unwrap();
|
||||||
|
ZipFileReader::Zstd(Crc32Reader::new(zstd_reader, crc32))
|
||||||
|
}
|
||||||
_ => panic!("Compression method not supported"),
|
_ => panic!("Compression method not supported"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
15
src/write.rs
15
src/write.rs
|
@ -22,6 +22,9 @@ use flate2::write::DeflateEncoder;
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
use bzip2::write::BzEncoder;
|
use bzip2::write::BzEncoder;
|
||||||
|
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
use zstd::stream::write::Encoder as ZstdEncoder;
|
||||||
|
|
||||||
enum GenericZipWriter<W: Write + io::Seek> {
|
enum GenericZipWriter<W: Write + io::Seek> {
|
||||||
Closed,
|
Closed,
|
||||||
Storer(W),
|
Storer(W),
|
||||||
|
@ -33,6 +36,8 @@ enum GenericZipWriter<W: Write + io::Seek> {
|
||||||
Deflater(DeflateEncoder<W>),
|
Deflater(DeflateEncoder<W>),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
Bzip2(BzEncoder<W>),
|
Bzip2(BzEncoder<W>),
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
Zstd(ZstdEncoder<'static, W>),
|
||||||
}
|
}
|
||||||
|
|
||||||
/// ZIP archive generator
|
/// ZIP archive generator
|
||||||
|
@ -804,6 +809,8 @@ impl<W: Write + io::Seek> GenericZipWriter<W> {
|
||||||
GenericZipWriter::Deflater(w) => w.finish()?,
|
GenericZipWriter::Deflater(w) => w.finish()?,
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
GenericZipWriter::Bzip2(w) => w.finish()?,
|
GenericZipWriter::Bzip2(w) => w.finish()?,
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
GenericZipWriter::Zstd(w) => w.finish()?,
|
||||||
GenericZipWriter::Closed => {
|
GenericZipWriter::Closed => {
|
||||||
return Err(io::Error::new(
|
return Err(io::Error::new(
|
||||||
io::ErrorKind::BrokenPipe,
|
io::ErrorKind::BrokenPipe,
|
||||||
|
@ -830,6 +837,10 @@ impl<W: Write + io::Seek> GenericZipWriter<W> {
|
||||||
CompressionMethod::Bzip2 => {
|
CompressionMethod::Bzip2 => {
|
||||||
GenericZipWriter::Bzip2(BzEncoder::new(bare, bzip2::Compression::default()))
|
GenericZipWriter::Bzip2(BzEncoder::new(bare, bzip2::Compression::default()))
|
||||||
}
|
}
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
CompressionMethod::Zstd => {
|
||||||
|
GenericZipWriter::Zstd(ZstdEncoder::new(bare, 0).unwrap())
|
||||||
|
}
|
||||||
CompressionMethod::Unsupported(..) => {
|
CompressionMethod::Unsupported(..) => {
|
||||||
return Err(ZipError::UnsupportedArchive("Unsupported compression"))
|
return Err(ZipError::UnsupportedArchive("Unsupported compression"))
|
||||||
}
|
}
|
||||||
|
@ -850,6 +861,8 @@ impl<W: Write + io::Seek> GenericZipWriter<W> {
|
||||||
GenericZipWriter::Deflater(ref mut w) => Some(w as &mut dyn Write),
|
GenericZipWriter::Deflater(ref mut w) => Some(w as &mut dyn Write),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
GenericZipWriter::Bzip2(ref mut w) => Some(w as &mut dyn Write),
|
GenericZipWriter::Bzip2(ref mut w) => Some(w as &mut dyn Write),
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
GenericZipWriter::Zstd(ref mut w) => Some(w as &mut dyn Write),
|
||||||
GenericZipWriter::Closed => None,
|
GenericZipWriter::Closed => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -879,6 +892,8 @@ impl<W: Write + io::Seek> GenericZipWriter<W> {
|
||||||
GenericZipWriter::Deflater(..) => Some(CompressionMethod::Deflated),
|
GenericZipWriter::Deflater(..) => Some(CompressionMethod::Deflated),
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
GenericZipWriter::Bzip2(..) => Some(CompressionMethod::Bzip2),
|
GenericZipWriter::Bzip2(..) => Some(CompressionMethod::Bzip2),
|
||||||
|
#[cfg(feature = "zstd")]
|
||||||
|
GenericZipWriter::Zstd(..) => Some(CompressionMethod::Zstd),
|
||||||
GenericZipWriter::Closed => None,
|
GenericZipWriter::Closed => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue