Implement shallow copy from within the file being written
This commit is contained in:
parent
e32db515a2
commit
cde5d5ed11
22 changed files with 148 additions and 85 deletions
|
@ -1,7 +1,8 @@
|
||||||
[package]
|
[package]
|
||||||
name = "zip"
|
name = "zip_next"
|
||||||
version = "0.6.4"
|
version = "0.6.5"
|
||||||
authors = ["Mathijs van de Nes <git@mathijs.vd-nes.nl>", "Marli Frost <marli@frost.red>", "Ryan Levick <ryan.levick@gmail.com>"]
|
authors = ["Mathijs van de Nes <git@mathijs.vd-nes.nl>", "Marli Frost <marli@frost.red>", "Ryan Levick <ryan.levick@gmail.com>",
|
||||||
|
"Chris Hennick <hennickc@amazon.com>"]
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
repository = "https://github.com/zip-rs/zip.git"
|
repository = "https://github.com/zip-rs/zip.git"
|
||||||
keywords = ["zip", "archive"]
|
keywords = ["zip", "archive"]
|
||||||
|
|
|
@ -32,14 +32,14 @@ With all default features:
|
||||||
|
|
||||||
```toml
|
```toml
|
||||||
[dependencies]
|
[dependencies]
|
||||||
zip = "0.6.4"
|
zip-next = "0.6.5"
|
||||||
```
|
```
|
||||||
|
|
||||||
Without the default features:
|
Without the default features:
|
||||||
|
|
||||||
```toml
|
```toml
|
||||||
[dependencies]
|
[dependencies]
|
||||||
zip = { version = "0.6.4", default-features = false }
|
zip-next = { version = "0.6.5", default-features = false }
|
||||||
```
|
```
|
||||||
|
|
||||||
The features available are:
|
The features available are:
|
||||||
|
|
|
@ -4,13 +4,13 @@ use std::io::{Cursor, Read, Write};
|
||||||
|
|
||||||
use bencher::Bencher;
|
use bencher::Bencher;
|
||||||
use getrandom::getrandom;
|
use getrandom::getrandom;
|
||||||
use zip::{ZipArchive, ZipWriter};
|
use zip_next::{ZipArchive, ZipWriter};
|
||||||
|
|
||||||
fn generate_random_archive(size: usize) -> Vec<u8> {
|
fn generate_random_archive(size: usize) -> Vec<u8> {
|
||||||
let data = Vec::new();
|
let data = Vec::new();
|
||||||
let mut writer = ZipWriter::new(Cursor::new(data));
|
let mut writer = ZipWriter::new(Cursor::new(data));
|
||||||
let options =
|
let options =
|
||||||
zip::write::FileOptions::default().compression_method(zip::CompressionMethod::Stored);
|
zip_next::write::FileOptions::default().compression_method(zip_next::CompressionMethod::Stored);
|
||||||
|
|
||||||
writer.start_file("random.dat", options).unwrap();
|
writer.start_file("random.dat", options).unwrap();
|
||||||
let mut bytes = vec![0u8; size];
|
let mut bytes = vec![0u8; size];
|
||||||
|
|
|
@ -3,7 +3,7 @@ use bencher::{benchmark_group, benchmark_main};
|
||||||
use std::io::{Cursor, Write};
|
use std::io::{Cursor, Write};
|
||||||
|
|
||||||
use bencher::Bencher;
|
use bencher::Bencher;
|
||||||
use zip::{ZipArchive, ZipWriter};
|
use zip_next::{ZipArchive, ZipWriter};
|
||||||
|
|
||||||
const FILE_COUNT: usize = 15_000;
|
const FILE_COUNT: usize = 15_000;
|
||||||
const FILE_SIZE: usize = 1024;
|
const FILE_SIZE: usize = 1024;
|
||||||
|
@ -12,7 +12,7 @@ fn generate_random_archive(count_files: usize, file_size: usize) -> Vec<u8> {
|
||||||
let data = Vec::new();
|
let data = Vec::new();
|
||||||
let mut writer = ZipWriter::new(Cursor::new(data));
|
let mut writer = ZipWriter::new(Cursor::new(data));
|
||||||
let options =
|
let options =
|
||||||
zip::write::FileOptions::default().compression_method(zip::CompressionMethod::Stored);
|
zip_next::write::FileOptions::default().compression_method(zip_next::CompressionMethod::Stored);
|
||||||
|
|
||||||
let bytes = vec![0u8; file_size];
|
let bytes = vec![0u8; file_size];
|
||||||
|
|
||||||
|
|
|
@ -14,7 +14,7 @@ fn real_main() -> i32 {
|
||||||
let fname = std::path::Path::new(&*args[1]);
|
let fname = std::path::Path::new(&*args[1]);
|
||||||
let file = fs::File::open(fname).unwrap();
|
let file = fs::File::open(fname).unwrap();
|
||||||
|
|
||||||
let mut archive = zip::ZipArchive::new(file).unwrap();
|
let mut archive = zip_next::ZipArchive::new(file).unwrap();
|
||||||
|
|
||||||
for i in 0..archive.len() {
|
for i in 0..archive.len() {
|
||||||
let mut file = archive.by_index(i).unwrap();
|
let mut file = archive.by_index(i).unwrap();
|
||||||
|
|
|
@ -13,7 +13,7 @@ fn real_main() -> i32 {
|
||||||
let fname = std::path::Path::new(&*args[1]);
|
let fname = std::path::Path::new(&*args[1]);
|
||||||
let zipfile = std::fs::File::open(fname).unwrap();
|
let zipfile = std::fs::File::open(fname).unwrap();
|
||||||
|
|
||||||
let mut archive = zip::ZipArchive::new(zipfile).unwrap();
|
let mut archive = zip_next::ZipArchive::new(zipfile).unwrap();
|
||||||
|
|
||||||
let mut file = match archive.by_name("test/lorem_ipsum.txt") {
|
let mut file = match archive.by_name("test/lorem_ipsum.txt") {
|
||||||
Ok(file) => file,
|
Ok(file) => file,
|
||||||
|
|
|
@ -15,7 +15,7 @@ fn real_main() -> i32 {
|
||||||
let file = fs::File::open(fname).unwrap();
|
let file = fs::File::open(fname).unwrap();
|
||||||
let reader = BufReader::new(file);
|
let reader = BufReader::new(file);
|
||||||
|
|
||||||
let mut archive = zip::ZipArchive::new(reader).unwrap();
|
let mut archive = zip_next::ZipArchive::new(reader).unwrap();
|
||||||
|
|
||||||
for i in 0..archive.len() {
|
for i in 0..archive.len() {
|
||||||
let file = archive.by_index(i).unwrap();
|
let file = archive.by_index(i).unwrap();
|
||||||
|
|
|
@ -10,7 +10,7 @@ fn real_main() -> i32 {
|
||||||
let mut buf = [0u8; 16];
|
let mut buf = [0u8; 16];
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match zip::read::read_zipfile_from_stream(&mut stdin_handle) {
|
match zip_next::read::read_zipfile_from_stream(&mut stdin_handle) {
|
||||||
Ok(Some(mut file)) => {
|
Ok(Some(mut file)) => {
|
||||||
println!(
|
println!(
|
||||||
"{}: {} bytes ({} bytes packed)",
|
"{}: {} bytes ({} bytes packed)",
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
use std::io::prelude::*;
|
use std::io::prelude::*;
|
||||||
use std::io::{Seek, Write};
|
use std::io::{Seek, Write};
|
||||||
use std::iter::Iterator;
|
use std::iter::Iterator;
|
||||||
use zip::result::ZipError;
|
use zip_next::result::ZipError;
|
||||||
use zip::write::FileOptions;
|
use zip_next::write::FileOptions;
|
||||||
|
|
||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
@ -12,30 +12,30 @@ fn main() {
|
||||||
std::process::exit(real_main());
|
std::process::exit(real_main());
|
||||||
}
|
}
|
||||||
|
|
||||||
const METHOD_STORED: Option<zip::CompressionMethod> = Some(zip::CompressionMethod::Stored);
|
const METHOD_STORED: Option<zip_next::CompressionMethod> = Some(zip_next::CompressionMethod::Stored);
|
||||||
|
|
||||||
#[cfg(any(
|
#[cfg(any(
|
||||||
feature = "deflate",
|
feature = "deflate",
|
||||||
feature = "deflate-miniz",
|
feature = "deflate-miniz",
|
||||||
feature = "deflate-zlib"
|
feature = "deflate-zlib"
|
||||||
))]
|
))]
|
||||||
const METHOD_DEFLATED: Option<zip::CompressionMethod> = Some(zip::CompressionMethod::Deflated);
|
const METHOD_DEFLATED: Option<zip_next::CompressionMethod> = Some(zip_next::CompressionMethod::Deflated);
|
||||||
#[cfg(not(any(
|
#[cfg(not(any(
|
||||||
feature = "deflate",
|
feature = "deflate",
|
||||||
feature = "deflate-miniz",
|
feature = "deflate-miniz",
|
||||||
feature = "deflate-zlib"
|
feature = "deflate-zlib"
|
||||||
)))]
|
)))]
|
||||||
const METHOD_DEFLATED: Option<zip::CompressionMethod> = None;
|
const METHOD_DEFLATED: Option<zip_next::CompressionMethod> = None;
|
||||||
|
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
const METHOD_BZIP2: Option<zip::CompressionMethod> = Some(zip::CompressionMethod::Bzip2);
|
const METHOD_BZIP2: Option<zip_next::CompressionMethod> = Some(zip_next::CompressionMethod::Bzip2);
|
||||||
#[cfg(not(feature = "bzip2"))]
|
#[cfg(not(feature = "bzip2"))]
|
||||||
const METHOD_BZIP2: Option<zip::CompressionMethod> = None;
|
const METHOD_BZIP2: Option<zip_next::CompressionMethod> = None;
|
||||||
|
|
||||||
#[cfg(feature = "zstd")]
|
#[cfg(feature = "zstd")]
|
||||||
const METHOD_ZSTD: Option<zip::CompressionMethod> = Some(zip::CompressionMethod::Zstd);
|
const METHOD_ZSTD: Option<zip_next::CompressionMethod> = Some(zip_next::CompressionMethod::Zstd);
|
||||||
#[cfg(not(feature = "zstd"))]
|
#[cfg(not(feature = "zstd"))]
|
||||||
const METHOD_ZSTD: Option<zip::CompressionMethod> = None;
|
const METHOD_ZSTD: Option<zip_next::CompressionMethod> = None;
|
||||||
|
|
||||||
fn real_main() -> i32 {
|
fn real_main() -> i32 {
|
||||||
let args: Vec<_> = std::env::args().collect();
|
let args: Vec<_> = std::env::args().collect();
|
||||||
|
@ -66,12 +66,12 @@ fn zip_dir<T>(
|
||||||
it: &mut dyn Iterator<Item = DirEntry>,
|
it: &mut dyn Iterator<Item = DirEntry>,
|
||||||
prefix: &str,
|
prefix: &str,
|
||||||
writer: T,
|
writer: T,
|
||||||
method: zip::CompressionMethod,
|
method: zip_next::CompressionMethod,
|
||||||
) -> zip::result::ZipResult<()>
|
) -> zip_next::result::ZipResult<()>
|
||||||
where
|
where
|
||||||
T: Write + Seek,
|
T: Write + Seek,
|
||||||
{
|
{
|
||||||
let mut zip = zip::ZipWriter::new(writer);
|
let mut zip = zip_next::ZipWriter::new(writer);
|
||||||
let options = FileOptions::default()
|
let options = FileOptions::default()
|
||||||
.compression_method(method)
|
.compression_method(method)
|
||||||
.unix_permissions(0o755);
|
.unix_permissions(0o755);
|
||||||
|
@ -107,8 +107,8 @@ where
|
||||||
fn doit(
|
fn doit(
|
||||||
src_dir: &str,
|
src_dir: &str,
|
||||||
dst_file: &str,
|
dst_file: &str,
|
||||||
method: zip::CompressionMethod,
|
method: zip_next::CompressionMethod,
|
||||||
) -> zip::result::ZipResult<()> {
|
) -> zip_next::result::ZipResult<()> {
|
||||||
if !Path::new(src_dir).is_dir() {
|
if !Path::new(src_dir).is_dir() {
|
||||||
return Err(ZipError::FileNotFound);
|
return Err(ZipError::FileNotFound);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use std::io::prelude::*;
|
use std::io::prelude::*;
|
||||||
use zip::write::FileOptions;
|
use zip_next::write::FileOptions;
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
std::process::exit(real_main());
|
std::process::exit(real_main());
|
||||||
|
@ -21,16 +21,16 @@ fn real_main() -> i32 {
|
||||||
0
|
0
|
||||||
}
|
}
|
||||||
|
|
||||||
fn doit(filename: &str) -> zip::result::ZipResult<()> {
|
fn doit(filename: &str) -> zip_next::result::ZipResult<()> {
|
||||||
let path = std::path::Path::new(filename);
|
let path = std::path::Path::new(filename);
|
||||||
let file = std::fs::File::create(path).unwrap();
|
let file = std::fs::File::create(path).unwrap();
|
||||||
|
|
||||||
let mut zip = zip::ZipWriter::new(file);
|
let mut zip = zip_next::ZipWriter::new(file);
|
||||||
|
|
||||||
zip.add_directory("test/", Default::default())?;
|
zip.add_directory("test/", Default::default())?;
|
||||||
|
|
||||||
let options = FileOptions::default()
|
let options = FileOptions::default()
|
||||||
.compression_method(zip::CompressionMethod::Stored)
|
.compression_method(zip_next::CompressionMethod::Stored)
|
||||||
.unix_permissions(0o755);
|
.unix_permissions(0o755);
|
||||||
zip.start_file("test/☃.txt", options)?;
|
zip.start_file("test/☃.txt", options)?;
|
||||||
zip.write_all(b"Hello, World!\n")?;
|
zip.write_all(b"Hello, World!\n")?;
|
||||||
|
|
|
@ -3,7 +3,7 @@ use libfuzzer_sys::fuzz_target;
|
||||||
|
|
||||||
fn decompress_all(data: &[u8]) -> Result<(), Box<dyn std::error::Error>> {
|
fn decompress_all(data: &[u8]) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
let reader = std::io::Cursor::new(data);
|
let reader = std::io::Cursor::new(data);
|
||||||
let mut zip = zip::ZipArchive::new(reader)?;
|
let mut zip = zip_next::ZipArchive::new(reader)?;
|
||||||
|
|
||||||
for i in 0..zip.len() {
|
for i in 0..zip.len() {
|
||||||
let mut file = zip.by_index(i)?;
|
let mut file = zip.by_index(i)?;
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
//!
|
//!
|
||||||
//!
|
//!
|
||||||
//!
|
//!
|
||||||
|
#![feature(read_buf)]
|
||||||
#![warn(missing_docs)]
|
#![warn(missing_docs)]
|
||||||
|
|
||||||
pub use crate::compression::{CompressionMethod, SUPPORTED_COMPRESSION_METHODS};
|
pub use crate::compression::{CompressionMethod, SUPPORTED_COMPRESSION_METHODS};
|
||||||
|
|
22
src/read.rs
22
src/read.rs
|
@ -51,8 +51,8 @@ pub(crate) mod zip_archive {
|
||||||
///
|
///
|
||||||
/// ```no_run
|
/// ```no_run
|
||||||
/// use std::io::prelude::*;
|
/// use std::io::prelude::*;
|
||||||
/// fn list_zip_contents(reader: impl Read + Seek) -> zip::result::ZipResult<()> {
|
/// fn list_zip_contents(reader: impl Read + Seek) -> zip_next::result::ZipResult<()> {
|
||||||
/// let mut zip = zip::ZipArchive::new(reader)?;
|
/// let mut zip = zip_next::ZipArchive::new(reader)?;
|
||||||
///
|
///
|
||||||
/// for i in 0..zip.len() {
|
/// for i in 0..zip.len() {
|
||||||
/// let mut file = zip.by_index(i)?;
|
/// let mut file = zip.by_index(i)?;
|
||||||
|
@ -72,7 +72,7 @@ pub(crate) mod zip_archive {
|
||||||
|
|
||||||
pub use zip_archive::ZipArchive;
|
pub use zip_archive::ZipArchive;
|
||||||
#[allow(clippy::large_enum_variant)]
|
#[allow(clippy::large_enum_variant)]
|
||||||
enum CryptoReader<'a> {
|
pub(crate) enum CryptoReader<'a> {
|
||||||
Plaintext(io::Take<&'a mut dyn Read>),
|
Plaintext(io::Take<&'a mut dyn Read>),
|
||||||
ZipCrypto(ZipCryptoReaderValid<io::Take<&'a mut dyn Read>>),
|
ZipCrypto(ZipCryptoReaderValid<io::Take<&'a mut dyn Read>>),
|
||||||
#[cfg(feature = "aes-crypto")]
|
#[cfg(feature = "aes-crypto")]
|
||||||
|
@ -119,7 +119,7 @@ impl<'a> CryptoReader<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
enum ZipFileReader<'a> {
|
pub(crate) enum ZipFileReader<'a> {
|
||||||
NoReader,
|
NoReader,
|
||||||
Raw(io::Take<&'a mut dyn io::Read>),
|
Raw(io::Take<&'a mut dyn io::Read>),
|
||||||
Stored(Crc32Reader<CryptoReader<'a>>),
|
Stored(Crc32Reader<CryptoReader<'a>>),
|
||||||
|
@ -178,12 +178,12 @@ impl<'a> ZipFileReader<'a> {
|
||||||
|
|
||||||
/// A struct for reading a zip file
|
/// A struct for reading a zip file
|
||||||
pub struct ZipFile<'a> {
|
pub struct ZipFile<'a> {
|
||||||
data: Cow<'a, ZipFileData>,
|
pub(crate) data: Cow<'a, ZipFileData>,
|
||||||
crypto_reader: Option<CryptoReader<'a>>,
|
pub(crate) crypto_reader: Option<CryptoReader<'a>>,
|
||||||
reader: ZipFileReader<'a>,
|
pub(crate) reader: ZipFileReader<'a>,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_content<'a>(
|
pub(crate) fn find_content<'a>(
|
||||||
data: &ZipFileData,
|
data: &ZipFileData,
|
||||||
reader: &'a mut (impl Read + Seek),
|
reader: &'a mut (impl Read + Seek),
|
||||||
) -> ZipResult<io::Take<&'a mut dyn Read>> {
|
) -> ZipResult<io::Take<&'a mut dyn Read>> {
|
||||||
|
@ -206,7 +206,7 @@ fn find_content<'a>(
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn make_crypto_reader<'a>(
|
pub(crate) fn make_crypto_reader<'a>(
|
||||||
compression_method: crate::compression::CompressionMethod,
|
compression_method: crate::compression::CompressionMethod,
|
||||||
crc32: u32,
|
crc32: u32,
|
||||||
last_modified_time: DateTime,
|
last_modified_time: DateTime,
|
||||||
|
@ -257,7 +257,7 @@ fn make_crypto_reader<'a>(
|
||||||
Ok(Ok(reader))
|
Ok(Ok(reader))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn make_reader(
|
pub(crate) fn make_reader(
|
||||||
compression_method: CompressionMethod,
|
compression_method: CompressionMethod,
|
||||||
crc32: u32,
|
crc32: u32,
|
||||||
reader: CryptoReader,
|
reader: CryptoReader,
|
||||||
|
@ -991,7 +991,7 @@ impl<'a> Drop for ZipFile<'a> {
|
||||||
// Get the inner `Take` reader so all decryption, decompression and CRC calculation is skipped.
|
// Get the inner `Take` reader so all decryption, decompression and CRC calculation is skipped.
|
||||||
let mut reader: std::io::Take<&mut dyn std::io::Read> = match &mut self.reader {
|
let mut reader: std::io::Take<&mut dyn std::io::Read> = match &mut self.reader {
|
||||||
ZipFileReader::NoReader => {
|
ZipFileReader::NoReader => {
|
||||||
let innerreader = ::std::mem::replace(&mut self.crypto_reader, None);
|
let innerreader = self.crypto_reader.take();
|
||||||
innerreader.expect("Invalid reader state").into_inner()
|
innerreader.expect("Invalid reader state").into_inner()
|
||||||
}
|
}
|
||||||
reader => {
|
reader => {
|
||||||
|
|
|
@ -65,8 +65,8 @@ impl ZipError {
|
||||||
/// The text used as an error when a password is required and not supplied
|
/// The text used as an error when a password is required and not supplied
|
||||||
///
|
///
|
||||||
/// ```rust,no_run
|
/// ```rust,no_run
|
||||||
/// # use zip::result::ZipError;
|
/// # use zip_next::result::ZipError;
|
||||||
/// # let mut archive = zip::ZipArchive::new(std::io::Cursor::new(&[])).unwrap();
|
/// # let mut archive = zip_next::ZipArchive::new(std::io::Cursor::new(&[])).unwrap();
|
||||||
/// match archive.by_index(1) {
|
/// match archive.by_index(1) {
|
||||||
/// Err(ZipError::UnsupportedArchive(ZipError::PASSWORD_REQUIRED)) => eprintln!("a password is needed to unzip this file"),
|
/// Err(ZipError::UnsupportedArchive(ZipError::PASSWORD_REQUIRED)) => eprintln!("a password is needed to unzip this file"),
|
||||||
/// _ => (),
|
/// _ => (),
|
||||||
|
|
92
src/write.rs
92
src/write.rs
|
@ -52,17 +52,17 @@ pub(crate) mod zip_writer {
|
||||||
/// API to edit its contents.
|
/// API to edit its contents.
|
||||||
///
|
///
|
||||||
/// ```
|
/// ```
|
||||||
/// # fn doit() -> zip::result::ZipResult<()>
|
/// # fn doit() -> zip_next::result::ZipResult<()>
|
||||||
/// # {
|
/// # {
|
||||||
/// # use zip::ZipWriter;
|
/// # use zip_next::ZipWriter;
|
||||||
/// use std::io::Write;
|
/// use std::io::Write;
|
||||||
/// use zip::write::FileOptions;
|
/// use zip_next::write::FileOptions;
|
||||||
///
|
///
|
||||||
/// // We use a buffer here, though you'd normally use a `File`
|
/// // We use a buffer here, though you'd normally use a `File`
|
||||||
/// let mut buf = [0; 65536];
|
/// let mut buf = [0; 65536];
|
||||||
/// let mut zip = zip::ZipWriter::new(std::io::Cursor::new(&mut buf[..]));
|
/// let mut zip = zip_next::ZipWriter::new(std::io::Cursor::new(&mut buf[..]));
|
||||||
///
|
///
|
||||||
/// let options = zip::write::FileOptions::default().compression_method(zip::CompressionMethod::Stored);
|
/// let options = zip_next::write::FileOptions::default().compression_method(zip_next::CompressionMethod::Stored);
|
||||||
/// zip.start_file("hello_world.txt", options)?;
|
/// zip.start_file("hello_world.txt", options)?;
|
||||||
/// zip.write(b"Hello, World!")?;
|
/// zip.write(b"Hello, World!")?;
|
||||||
///
|
///
|
||||||
|
@ -82,7 +82,7 @@ pub(crate) mod zip_writer {
|
||||||
pub(super) writing_to_extra_field: bool,
|
pub(super) writing_to_extra_field: bool,
|
||||||
pub(super) writing_to_central_extra_field_only: bool,
|
pub(super) writing_to_central_extra_field_only: bool,
|
||||||
pub(super) writing_raw: bool,
|
pub(super) writing_raw: bool,
|
||||||
pub(super) comment: Vec<u8>,
|
pub(super) comment: Vec<u8>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub use zip_writer::ZipWriter;
|
pub use zip_writer::ZipWriter;
|
||||||
|
@ -291,7 +291,7 @@ impl<A: Read + Write + io::Seek> ZipWriter<A> {
|
||||||
writing_to_extra_field: false,
|
writing_to_extra_field: false,
|
||||||
writing_to_central_extra_field_only: false,
|
writing_to_central_extra_field_only: false,
|
||||||
comment: footer.zip_file_comment,
|
comment: footer.zip_file_comment,
|
||||||
writing_raw: true, // avoid recomputing the last file's header
|
writing_raw: true // avoid recomputing the last file's header
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -309,7 +309,7 @@ impl<W: Write + io::Seek> ZipWriter<W> {
|
||||||
writing_to_extra_field: false,
|
writing_to_extra_field: false,
|
||||||
writing_to_central_extra_field_only: false,
|
writing_to_central_extra_field_only: false,
|
||||||
writing_raw: false,
|
writing_raw: false,
|
||||||
comment: Vec::new(),
|
comment: Vec::new()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -493,8 +493,8 @@ impl<W: Write + io::Seek> ZipWriter<W> {
|
||||||
///
|
///
|
||||||
/// ```
|
/// ```
|
||||||
/// use byteorder::{LittleEndian, WriteBytesExt};
|
/// use byteorder::{LittleEndian, WriteBytesExt};
|
||||||
/// use zip::{ZipArchive, ZipWriter, result::ZipResult};
|
/// use zip_next::{ZipArchive, ZipWriter, result::ZipResult};
|
||||||
/// use zip::{write::FileOptions, CompressionMethod};
|
/// use zip_next::{write::FileOptions, CompressionMethod};
|
||||||
/// use std::io::{Write, Cursor};
|
/// use std::io::{Write, Cursor};
|
||||||
///
|
///
|
||||||
/// # fn main() -> ZipResult<()> {
|
/// # fn main() -> ZipResult<()> {
|
||||||
|
@ -623,12 +623,12 @@ impl<W: Write + io::Seek> ZipWriter<W> {
|
||||||
/// ```no_run
|
/// ```no_run
|
||||||
/// use std::fs::File;
|
/// use std::fs::File;
|
||||||
/// use std::io::{Read, Seek, Write};
|
/// use std::io::{Read, Seek, Write};
|
||||||
/// use zip::{ZipArchive, ZipWriter};
|
/// use zip_next::{ZipArchive, ZipWriter};
|
||||||
///
|
///
|
||||||
/// fn copy_rename<R, W>(
|
/// fn copy_rename<R, W>(
|
||||||
/// src: &mut ZipArchive<R>,
|
/// src: &mut ZipArchive<R>,
|
||||||
/// dst: &mut ZipWriter<W>,
|
/// dst: &mut ZipWriter<W>,
|
||||||
/// ) -> zip::result::ZipResult<()>
|
/// ) -> zip_next::result::ZipResult<()>
|
||||||
/// where
|
/// where
|
||||||
/// R: Read + Seek,
|
/// R: Read + Seek,
|
||||||
/// W: Write + Seek,
|
/// W: Write + Seek,
|
||||||
|
@ -676,9 +676,9 @@ impl<W: Write + io::Seek> ZipWriter<W> {
|
||||||
/// ```no_run
|
/// ```no_run
|
||||||
/// use std::fs::File;
|
/// use std::fs::File;
|
||||||
/// use std::io::{Read, Seek, Write};
|
/// use std::io::{Read, Seek, Write};
|
||||||
/// use zip::{ZipArchive, ZipWriter};
|
/// use zip_next::{ZipArchive, ZipWriter};
|
||||||
///
|
///
|
||||||
/// fn copy<R, W>(src: &mut ZipArchive<R>, dst: &mut ZipWriter<W>) -> zip::result::ZipResult<()>
|
/// fn copy<R, W>(src: &mut ZipArchive<R>, dst: &mut ZipWriter<W>) -> zip_next::result::ZipResult<()>
|
||||||
/// where
|
/// where
|
||||||
/// R: Read + Seek,
|
/// R: Read + Seek,
|
||||||
/// W: Write + Seek,
|
/// W: Write + Seek,
|
||||||
|
@ -841,6 +841,32 @@ impl<W: Write + io::Seek> ZipWriter<W> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl <RW: Read + Write + io::Seek> ZipWriter<RW> {
|
||||||
|
fn data_by_name(&mut self, name: &str) -> ZipResult<&ZipFileData> {
|
||||||
|
self.finish_file()?;
|
||||||
|
for file in self.files.iter() {
|
||||||
|
if file.file_name == name {
|
||||||
|
return Ok(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(ZipError::FileNotFound)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Adds another entry to the central directory referring to the same content as an existing
|
||||||
|
/// entry. The file's local-file header will still refer to it by its original name, so
|
||||||
|
/// unzipping the file will technically be unspecified behavior. However, both [ZipArchive] and
|
||||||
|
/// OpenJDK ignore the filename in the local-file header and treat the central directory as
|
||||||
|
/// authoritative.
|
||||||
|
pub fn shallow_copy_file(&mut self, src_name: &str, dest_name: &str) -> ZipResult<()> {
|
||||||
|
self.finish_file()?;
|
||||||
|
let src_data = self.data_by_name(src_name)?;
|
||||||
|
let mut dest_data = src_data.to_owned();
|
||||||
|
dest_data.file_name = dest_name.into();
|
||||||
|
self.files.push(dest_data);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<W: Write + io::Seek> Drop for ZipWriter<W> {
|
impl<W: Write + io::Seek> Drop for ZipWriter<W> {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
if !self.inner.is_closed() {
|
if !self.inner.is_closed() {
|
||||||
|
@ -1309,7 +1335,8 @@ mod test {
|
||||||
use crate::compression::CompressionMethod;
|
use crate::compression::CompressionMethod;
|
||||||
use crate::types::DateTime;
|
use crate::types::DateTime;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::io::Write;
|
use std::io::{Read, Write};
|
||||||
|
use crate::ZipArchive;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn write_empty_zip() {
|
fn write_empty_zip() {
|
||||||
|
@ -1441,6 +1468,41 @@ mod test {
|
||||||
assert_eq!(result.get_ref(), &v);
|
assert_eq!(result.get_ref(), &v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
const RT_TEST_TEXT: &str = "And I can't stop thinking about the moments that I lost to you\
|
||||||
|
And I can't stop thinking of things I used to do\
|
||||||
|
And I can't stop making bad decisions\
|
||||||
|
And I can't stop eating stuff you make me chew\
|
||||||
|
I put on a smile like you wanna see\
|
||||||
|
Another day goes by that I long to be like you";
|
||||||
|
#[cfg(test)] const RT_TEST_FILENAME: &str = "subfolder/sub-subfolder/can't_stop.txt";
|
||||||
|
#[cfg(test)] const SECOND_FILENAME: &str = "different_name.xyz";
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_shallow_copy() {
|
||||||
|
let mut writer = ZipWriter::new(io::Cursor::new(Vec::new()));
|
||||||
|
let options = FileOptions {
|
||||||
|
compression_method: CompressionMethod::Deflated,
|
||||||
|
compression_level: Some(9),
|
||||||
|
last_modified_time: DateTime::default(),
|
||||||
|
permissions: Some(33188),
|
||||||
|
large_file: false,
|
||||||
|
};
|
||||||
|
writer.start_file(RT_TEST_FILENAME, options).unwrap();
|
||||||
|
writer.write(RT_TEST_TEXT.as_ref()).unwrap();
|
||||||
|
writer.shallow_copy_file(RT_TEST_FILENAME, SECOND_FILENAME).unwrap();
|
||||||
|
let zip = writer.finish().unwrap();
|
||||||
|
let mut reader = ZipArchive::new(zip).unwrap();
|
||||||
|
let file_names: Vec<&str> = reader.file_names().collect();
|
||||||
|
assert_eq!(file_names, vec![RT_TEST_FILENAME, SECOND_FILENAME]);
|
||||||
|
let mut first_file_content = String::new();
|
||||||
|
reader.by_name(RT_TEST_FILENAME).unwrap().read_to_string(&mut first_file_content).unwrap();
|
||||||
|
assert_eq!(first_file_content, RT_TEST_TEXT);
|
||||||
|
let mut second_file_content = String::new();
|
||||||
|
reader.by_name(SECOND_FILENAME).unwrap().read_to_string(&mut second_file_content).unwrap();
|
||||||
|
assert_eq!(second_file_content, RT_TEST_TEXT);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn path_to_string() {
|
fn path_to_string() {
|
||||||
let mut path = std::path::PathBuf::new();
|
let mut path = std::path::PathBuf::new();
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
#![cfg(feature = "aes-crypto")]
|
#![cfg(feature = "aes-crypto")]
|
||||||
|
|
||||||
use std::io::{self, Read};
|
use std::io::{self, Read};
|
||||||
use zip::ZipArchive;
|
use zip_next::ZipArchive;
|
||||||
|
|
||||||
const SECRET_CONTENT: &str = "Lorem ipsum dolor sit amet";
|
const SECRET_CONTENT: &str = "Lorem ipsum dolor sit amet";
|
||||||
|
|
||||||
|
|
|
@ -3,8 +3,8 @@ use std::collections::HashSet;
|
||||||
use std::io::prelude::*;
|
use std::io::prelude::*;
|
||||||
use std::io::{Cursor, Seek};
|
use std::io::{Cursor, Seek};
|
||||||
use std::iter::FromIterator;
|
use std::iter::FromIterator;
|
||||||
use zip::write::FileOptions;
|
use zip_next::write::FileOptions;
|
||||||
use zip::{CompressionMethod, SUPPORTED_COMPRESSION_METHODS};
|
use zip_next::{CompressionMethod, SUPPORTED_COMPRESSION_METHODS};
|
||||||
|
|
||||||
// This test asserts that after creating a zip file, then reading its contents back out,
|
// This test asserts that after creating a zip file, then reading its contents back out,
|
||||||
// the extracted data will *always* be exactly the same as the original data.
|
// the extracted data will *always* be exactly the same as the original data.
|
||||||
|
@ -32,8 +32,8 @@ fn copy() {
|
||||||
let mut tgt_file = &mut Cursor::new(Vec::new());
|
let mut tgt_file = &mut Cursor::new(Vec::new());
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut src_archive = zip::ZipArchive::new(src_file).unwrap();
|
let mut src_archive = zip_next::ZipArchive::new(src_file).unwrap();
|
||||||
let mut zip = zip::ZipWriter::new(&mut tgt_file);
|
let mut zip = zip_next::ZipWriter::new(&mut tgt_file);
|
||||||
|
|
||||||
{
|
{
|
||||||
let file = src_archive
|
let file = src_archive
|
||||||
|
@ -53,7 +53,7 @@ fn copy() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut tgt_archive = zip::ZipArchive::new(tgt_file).unwrap();
|
let mut tgt_archive = zip_next::ZipArchive::new(tgt_file).unwrap();
|
||||||
|
|
||||||
check_archive_file_contents(&mut tgt_archive, ENTRY_NAME, LOREM_IPSUM);
|
check_archive_file_contents(&mut tgt_archive, ENTRY_NAME, LOREM_IPSUM);
|
||||||
check_archive_file_contents(&mut tgt_archive, COPY_ENTRY_NAME, LOREM_IPSUM);
|
check_archive_file_contents(&mut tgt_archive, COPY_ENTRY_NAME, LOREM_IPSUM);
|
||||||
|
@ -69,7 +69,7 @@ fn append() {
|
||||||
write_test_archive(file, method).expect("Couldn't write to test file");
|
write_test_archive(file, method).expect("Couldn't write to test file");
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut zip = zip::ZipWriter::new_append(&mut file).unwrap();
|
let mut zip = zip_next::ZipWriter::new_append(&mut file).unwrap();
|
||||||
zip.start_file(
|
zip.start_file(
|
||||||
COPY_ENTRY_NAME,
|
COPY_ENTRY_NAME,
|
||||||
FileOptions::default().compression_method(method),
|
FileOptions::default().compression_method(method),
|
||||||
|
@ -79,7 +79,7 @@ fn append() {
|
||||||
zip.finish().unwrap();
|
zip.finish().unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut zip = zip::ZipArchive::new(&mut file).unwrap();
|
let mut zip = zip_next::ZipArchive::new(&mut file).unwrap();
|
||||||
check_archive_file_contents(&mut zip, ENTRY_NAME, LOREM_IPSUM);
|
check_archive_file_contents(&mut zip, ENTRY_NAME, LOREM_IPSUM);
|
||||||
check_archive_file_contents(&mut zip, COPY_ENTRY_NAME, LOREM_IPSUM);
|
check_archive_file_contents(&mut zip, COPY_ENTRY_NAME, LOREM_IPSUM);
|
||||||
}
|
}
|
||||||
|
@ -89,8 +89,8 @@ fn append() {
|
||||||
fn write_test_archive(
|
fn write_test_archive(
|
||||||
file: &mut Cursor<Vec<u8>>,
|
file: &mut Cursor<Vec<u8>>,
|
||||||
method: CompressionMethod,
|
method: CompressionMethod,
|
||||||
) -> zip::result::ZipResult<()> {
|
) -> zip_next::result::ZipResult<()> {
|
||||||
let mut zip = zip::ZipWriter::new(file);
|
let mut zip = zip_next::ZipWriter::new(file);
|
||||||
|
|
||||||
zip.add_directory("test/", Default::default())?;
|
zip.add_directory("test/", Default::default())?;
|
||||||
|
|
||||||
|
@ -116,8 +116,8 @@ fn write_test_archive(
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load an archive from buffer and check for test data.
|
// Load an archive from buffer and check for test data.
|
||||||
fn check_test_archive<R: Read + Seek>(zip_file: R) -> zip::result::ZipResult<zip::ZipArchive<R>> {
|
fn check_test_archive<R: Read + Seek>(zip_file: R) -> zip_next::result::ZipResult<zip_next::ZipArchive<R>> {
|
||||||
let mut archive = zip::ZipArchive::new(zip_file).unwrap();
|
let mut archive = zip_next::ZipArchive::new(zip_file).unwrap();
|
||||||
|
|
||||||
// Check archive contains expected file names.
|
// Check archive contains expected file names.
|
||||||
{
|
{
|
||||||
|
@ -147,9 +147,9 @@ fn check_test_archive<R: Read + Seek>(zip_file: R) -> zip::result::ZipResult<zip
|
||||||
|
|
||||||
// Read a file in the archive as a string.
|
// Read a file in the archive as a string.
|
||||||
fn read_archive_file<R: Read + Seek>(
|
fn read_archive_file<R: Read + Seek>(
|
||||||
archive: &mut zip::ZipArchive<R>,
|
archive: &mut zip_next::ZipArchive<R>,
|
||||||
name: &str,
|
name: &str,
|
||||||
) -> zip::result::ZipResult<String> {
|
) -> zip_next::result::ZipResult<String> {
|
||||||
let mut file = archive.by_name(name)?;
|
let mut file = archive.by_name(name)?;
|
||||||
|
|
||||||
let mut contents = String::new();
|
let mut contents = String::new();
|
||||||
|
@ -183,7 +183,7 @@ fn check_archive_file(
|
||||||
|
|
||||||
// Check a file in the archive contains the given data.
|
// Check a file in the archive contains the given data.
|
||||||
fn check_archive_file_contents<R: Read + Seek>(
|
fn check_archive_file_contents<R: Read + Seek>(
|
||||||
archive: &mut zip::ZipArchive<R>,
|
archive: &mut zip_next::ZipArchive<R>,
|
||||||
name: &str,
|
name: &str,
|
||||||
expected: &[u8],
|
expected: &[u8],
|
||||||
) {
|
) {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use std::io::Cursor;
|
use std::io::Cursor;
|
||||||
use zip::read::ZipArchive;
|
use zip_next::read::ZipArchive;
|
||||||
|
|
||||||
const BUF: &[u8] = &[
|
const BUF: &[u8] = &[
|
||||||
0x50, 0x4b, 0x03, 0x04, 0x0a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
0x50, 0x4b, 0x03, 0x04, 0x0a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use zip::result::ZipError;
|
use zip_next::result::ZipError;
|
||||||
|
|
||||||
const BUF: &[u8] = &[
|
const BUF: &[u8] = &[
|
||||||
0, 80, 75, 1, 2, 127, 120, 0, 3, 3, 75, 80, 232, 3, 0, 0, 0, 0, 0, 0, 3, 0, 1, 0, 7, 0, 0, 0,
|
0, 80, 75, 1, 2, 127, 120, 0, 3, 3, 75, 80, 232, 3, 0, 0, 0, 0, 0, 0, 3, 0, 1, 0, 7, 0, 0, 0,
|
||||||
|
@ -23,7 +23,7 @@ const BUF: &[u8] = &[
|
||||||
#[test]
|
#[test]
|
||||||
fn invalid_header() {
|
fn invalid_header() {
|
||||||
let reader = std::io::Cursor::new(&BUF);
|
let reader = std::io::Cursor::new(&BUF);
|
||||||
let archive = zip::ZipArchive::new(reader);
|
let archive = zip_next::ZipArchive::new(reader);
|
||||||
match archive {
|
match archive {
|
||||||
Err(ZipError::InvalidArchive(_)) => {}
|
Err(ZipError::InvalidArchive(_)) => {}
|
||||||
value => panic!("Unexpected value: {value:?}"),
|
value => panic!("Unexpected value: {value:?}"),
|
||||||
|
|
|
@ -190,7 +190,7 @@ impl Read for Zip64File {
|
||||||
#[test]
|
#[test]
|
||||||
fn zip64_large() {
|
fn zip64_large() {
|
||||||
let zipfile = Zip64File::new();
|
let zipfile = Zip64File::new();
|
||||||
let mut archive = zip::ZipArchive::new(zipfile).unwrap();
|
let mut archive = zip_next::ZipArchive::new(zipfile).unwrap();
|
||||||
let mut buf = [0u8; 32];
|
let mut buf = [0u8; 32];
|
||||||
|
|
||||||
for i in 0..archive.len() {
|
for i in 0..archive.len() {
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
// 0000002e
|
// 0000002e
|
||||||
|
|
||||||
use std::io;
|
use std::io;
|
||||||
use zip::ZipArchive;
|
use zip_next::ZipArchive;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn correctly_handle_zip_with_garbage_after_comment() {
|
fn correctly_handle_zip_with_garbage_after_comment() {
|
||||||
|
|
|
@ -39,7 +39,7 @@ fn encrypted_file() {
|
||||||
0x00, 0x00,
|
0x00, 0x00,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
let mut archive = zip::ZipArchive::new(zip_file_bytes).unwrap();
|
let mut archive = zip_next::ZipArchive::new(zip_file_bytes).unwrap();
|
||||||
|
|
||||||
assert_eq!(archive.len(), 1); //Only one file inside archive: `test.txt`
|
assert_eq!(archive.len(), 1); //Only one file inside archive: `test.txt`
|
||||||
|
|
||||||
|
@ -47,8 +47,8 @@ fn encrypted_file() {
|
||||||
// No password
|
// No password
|
||||||
let file = archive.by_index(0);
|
let file = archive.by_index(0);
|
||||||
match file {
|
match file {
|
||||||
Err(zip::result::ZipError::UnsupportedArchive(
|
Err(zip_next::result::ZipError::UnsupportedArchive(
|
||||||
zip::result::ZipError::PASSWORD_REQUIRED,
|
zip_next::result::ZipError::PASSWORD_REQUIRED,
|
||||||
)) => (),
|
)) => (),
|
||||||
Err(_) => panic!(
|
Err(_) => panic!(
|
||||||
"Expected PasswordRequired error when opening encrypted file without password"
|
"Expected PasswordRequired error when opening encrypted file without password"
|
||||||
|
@ -61,7 +61,7 @@ fn encrypted_file() {
|
||||||
// Wrong password
|
// Wrong password
|
||||||
let file = archive.by_index_decrypt(0, b"wrong password");
|
let file = archive.by_index_decrypt(0, b"wrong password");
|
||||||
match file {
|
match file {
|
||||||
Ok(Err(zip::result::InvalidPassword)) => (),
|
Ok(Err(zip_next::result::InvalidPassword)) => (),
|
||||||
Err(_) => panic!(
|
Err(_) => panic!(
|
||||||
"Expected InvalidPassword error when opening encrypted file with wrong password"
|
"Expected InvalidPassword error when opening encrypted file with wrong password"
|
||||||
),
|
),
|
||||||
|
|
Loading…
Add table
Reference in a new issue