Merge pull request #98 from zip-rs/finish_owned

refactor: Make `ZipWriter::finish()` consume the `ZipWriter`
This commit is contained in:
Chris Hennick 2024-05-06 06:17:13 +00:00 committed by GitHub
commit d629b364e8
Signed by: DevComp
GPG key ID: B5690EEEBB952194
5 changed files with 28 additions and 31 deletions

View file

@ -59,7 +59,7 @@ fn merge_archive_stored(bench: &mut Bencher) {
bench.iter(|| { bench.iter(|| {
let buf = Cursor::new(Vec::new()); let buf = Cursor::new(Vec::new());
let zip = ZipWriter::new(buf); let zip = ZipWriter::new(buf);
let mut zip = perform_merge(src.clone(), zip).unwrap(); let zip = perform_merge(src.clone(), zip).unwrap();
let buf = zip.finish().unwrap().into_inner(); let buf = zip.finish().unwrap().into_inner();
assert_eq!(buf.len(), len); assert_eq!(buf.len(), len);
}); });
@ -75,7 +75,7 @@ fn merge_archive_compressed(bench: &mut Bencher) {
bench.iter(|| { bench.iter(|| {
let buf = Cursor::new(Vec::new()); let buf = Cursor::new(Vec::new());
let zip = ZipWriter::new(buf); let zip = ZipWriter::new(buf);
let mut zip = perform_merge(src.clone(), zip).unwrap(); let zip = perform_merge(src.clone(), zip).unwrap();
let buf = zip.finish().unwrap().into_inner(); let buf = zip.finish().unwrap().into_inner();
assert_eq!(buf.len(), len); assert_eq!(buf.len(), len);
}); });
@ -90,7 +90,7 @@ fn merge_archive_raw_copy_file_stored(bench: &mut Bencher) {
bench.iter(|| { bench.iter(|| {
let buf = Cursor::new(Vec::new()); let buf = Cursor::new(Vec::new());
let zip = ZipWriter::new(buf); let zip = ZipWriter::new(buf);
let mut zip = perform_raw_copy_file(src.clone(), zip).unwrap(); let zip = perform_raw_copy_file(src.clone(), zip).unwrap();
let buf = zip.finish().unwrap().into_inner(); let buf = zip.finish().unwrap().into_inner();
assert_eq!(buf.len(), len); assert_eq!(buf.len(), len);
}); });
@ -106,7 +106,7 @@ fn merge_archive_raw_copy_file_compressed(bench: &mut Bencher) {
bench.iter(|| { bench.iter(|| {
let buf = Cursor::new(Vec::new()); let buf = Cursor::new(Vec::new());
let zip = ZipWriter::new(buf); let zip = ZipWriter::new(buf);
let mut zip = perform_raw_copy_file(src.clone(), zip).unwrap(); let zip = perform_raw_copy_file(src.clone(), zip).unwrap();
let buf = zip.finish().unwrap().into_inner(); let buf = zip.finish().unwrap().into_inner();
assert_eq!(buf.len(), len); assert_eq!(buf.len(), len);
}); });

View file

@ -11,6 +11,7 @@ cargo-fuzz = true
[dependencies] [dependencies]
libfuzzer-sys = "0.4" libfuzzer-sys = "0.4"
arbitrary = { version = "1.3.0", features = ["derive"] } arbitrary = { version = "1.3.0", features = ["derive"] }
replace_with = "0.1.7"
[dependencies.zip] [dependencies.zip]
path = ".." path = ".."

View file

@ -2,7 +2,7 @@
use arbitrary::Arbitrary; use arbitrary::Arbitrary;
use libfuzzer_sys::fuzz_target; use libfuzzer_sys::fuzz_target;
use std::cell::RefCell; use replace_with::replace_with_or_abort;
use std::io::{Cursor, Read, Seek, Write}; use std::io::{Cursor, Read, Seek, Write};
use std::path::PathBuf; use std::path::PathBuf;
@ -37,7 +37,7 @@ pub struct FuzzTestCase {
} }
fn do_operation<T>( fn do_operation<T>(
writer: &mut RefCell<zip::ZipWriter<T>>, writer: &mut zip::ZipWriter<T>,
operation: &FileOperation, operation: &FileOperation,
abort: bool, abort: bool,
flush_on_finish_file: bool, flush_on_finish_file: bool,
@ -45,9 +45,7 @@ fn do_operation<T>(
where where
T: Read + Write + Seek, T: Read + Write + Seek,
{ {
writer writer.set_flush_on_finish_file(flush_on_finish_file);
.borrow_mut()
.set_flush_on_finish_file(flush_on_finish_file);
let path = &operation.path; let path = &operation.path;
match &operation.basic { match &operation.basic {
BasicFileOperation::WriteNormalFile { BasicFileOperation::WriteNormalFile {
@ -60,44 +58,44 @@ where
if uncompressed_size >= u32::MAX as usize { if uncompressed_size >= u32::MAX as usize {
options = options.large_file(true); options = options.large_file(true);
} }
writer.borrow_mut().start_file_from_path(path, options)?; writer.start_file_from_path(path, options)?;
for chunk in contents { for chunk in contents {
writer.borrow_mut().write_all(chunk.as_slice())?; writer.write_all(chunk.as_slice())?;
} }
} }
BasicFileOperation::WriteDirectory(options) => { BasicFileOperation::WriteDirectory(options) => {
writer.borrow_mut().add_directory_from_path(path, options.to_owned())?; writer.add_directory_from_path(path, options.to_owned())?;
} }
BasicFileOperation::WriteSymlinkWithTarget { target, options } => { BasicFileOperation::WriteSymlinkWithTarget { target, options } => {
writer writer.add_symlink_from_path(&path, target, options.to_owned())?;
.borrow_mut()
.add_symlink_from_path(&path, target, options.to_owned())?;
} }
BasicFileOperation::ShallowCopy(base) => { BasicFileOperation::ShallowCopy(base) => {
do_operation(writer, &base, false, flush_on_finish_file)?; do_operation(writer, &base, false, flush_on_finish_file)?;
writer.borrow_mut().shallow_copy_file_from_path(&base.path, &path)?; writer.shallow_copy_file_from_path(&base.path, &path)?;
} }
BasicFileOperation::DeepCopy(base) => { BasicFileOperation::DeepCopy(base) => {
do_operation(writer, &base, false, flush_on_finish_file)?; do_operation(writer, &base, false, flush_on_finish_file)?;
writer.borrow_mut().deep_copy_file_from_path(&base.path, &path)?; writer.deep_copy_file_from_path(&base.path, &path)?;
} }
} }
if abort { if abort {
writer.borrow_mut().abort_file().unwrap(); writer.abort_file().unwrap();
} }
if operation.reopen { if operation.reopen {
let old_comment = writer.borrow().get_raw_comment().to_owned(); let old_comment = writer.get_raw_comment().to_owned();
let new_writer = replace_with_or_abort(writer, |old_writer: zip::ZipWriter<T>| {
zip::ZipWriter::new_append(writer.borrow_mut().finish().unwrap()).unwrap(); let new_writer =
assert_eq!(&old_comment, new_writer.get_raw_comment()); zip::ZipWriter::new_append(old_writer.finish().unwrap()).unwrap();
*writer = new_writer.into(); assert_eq!(&old_comment, new_writer.get_raw_comment());
new_writer
});
} }
Ok(()) Ok(())
} }
fuzz_target!(|test_case: FuzzTestCase| { fuzz_target!(|test_case: FuzzTestCase| {
let mut writer = RefCell::new(zip::ZipWriter::new(Cursor::new(Vec::new()))); let mut writer = zip::ZipWriter::new(Cursor::new(Vec::new()));
writer.borrow_mut().set_raw_comment(test_case.comment); writer.set_raw_comment(test_case.comment);
for (operation, abort) in test_case.operations { for (operation, abort) in test_case.operations {
let _ = do_operation( let _ = do_operation(
&mut writer, &mut writer,
@ -106,5 +104,5 @@ fuzz_target!(|test_case: FuzzTestCase| {
test_case.flush_on_finish_file, test_case.flush_on_finish_file,
); );
} }
let _ = zip::ZipArchive::new(writer.borrow_mut().finish().unwrap()); let _ = zip::ZipArchive::new(writer.finish().unwrap());
}); });

View file

@ -1188,7 +1188,7 @@ impl<W: Write + Seek> ZipWriter<W> {
/// ///
/// This will return the writer, but one should normally not append any data to the end of the file. /// This will return the writer, but one should normally not append any data to the end of the file.
/// Note that the zipfile will also be finished on drop. /// Note that the zipfile will also be finished on drop.
pub fn finish(&mut self) -> ZipResult<W> { pub fn finish(mut self) -> ZipResult<W> {
let _central_start = self.finalize()?; let _central_start = self.finalize()?;
let inner = mem::replace(&mut self.inner, Closed); let inner = mem::replace(&mut self.inner, Closed);
Ok(inner.unwrap()) Ok(inner.unwrap())
@ -1440,14 +1440,13 @@ impl<W: Write + Seek> GenericZipWriter<W> {
feature = "deflate-zlib-ng", feature = "deflate-zlib-ng",
))] ))]
{ {
return Ok(Box::new(move |bare| { Ok(Box::new(move |bare| {
GenericZipWriter::Deflater(DeflateEncoder::new( GenericZipWriter::Deflater(DeflateEncoder::new(
bare, bare,
Compression::new(level), Compression::new(level),
)) ))
})); }))
} }
unreachable!()
} }
#[cfg(feature = "deflate64")] #[cfg(feature = "deflate64")]
CompressionMethod::Deflate64 => Err(ZipError::UnsupportedArchive( CompressionMethod::Deflate64 => Err(ZipError::UnsupportedArchive(

View file

@ -49,7 +49,6 @@ fn encrypting_file() {
.unwrap(); .unwrap();
archive.write_all(b"test").unwrap(); archive.write_all(b"test").unwrap();
archive.finish().unwrap(); archive.finish().unwrap();
drop(archive);
let mut archive = zip::ZipArchive::new(Cursor::new(&mut buf)).unwrap(); let mut archive = zip::ZipArchive::new(Cursor::new(&mut buf)).unwrap();
let mut file = archive.by_index_decrypt(0, b"password").unwrap(); let mut file = archive.by_index_decrypt(0, b"password").unwrap();
let mut buf = Vec::new(); let mut buf = Vec::new();