#![no_main] use libfuzzer_sys::fuzz_target; use arbitrary::{Arbitrary, Unstructured}; use arbitrary::size_hint::and_all; use std::fmt::{Debug, Formatter}; use std::io::{Cursor, Read, Seek, Write}; #[derive(Arbitrary,Debug)] pub struct File { pub name: String, pub contents: Vec> } const LARGE_FILE_BUF_SIZE: usize = u32::MAX as usize + 1; pub struct LargeFile { pub name: String, pub large_contents: Vec, pub extra_contents: Vec> } impl Arbitrary<'_> for LargeFile { fn arbitrary(u: &mut Unstructured) -> arbitrary::Result { Ok(LargeFile { name: String::arbitrary(u)?, large_contents: u.bytes(LARGE_FILE_BUF_SIZE)?.to_vec(), extra_contents: Vec::arbitrary(u)? }) } fn size_hint(depth: usize) -> (usize, Option) { and_all(&[::size_hint(depth), > as Arbitrary>::size_hint(depth), (LARGE_FILE_BUF_SIZE, Some(LARGE_FILE_BUF_SIZE))]) } } impl Debug for LargeFile { fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { f.debug_struct("LargeFile") .field("name", &self.name) .field("extra_contents", &self.extra_contents) .finish() } } #[derive(Arbitrary,Debug)] pub enum FileOperation { Write { file: File, options: zip_next::write::FileOptions }, WriteLarge { file: LargeFile, options: zip_next::write::FileOptions }, ShallowCopy { base: Box, new_name: String }, DeepCopy { base: Box, new_name: String } } impl FileOperation { pub fn get_name(&self) -> String { match self { FileOperation::Write {file, ..} => &file.name, FileOperation::WriteLarge {file, ..} => &file.name, FileOperation::ShallowCopy {new_name, ..} => new_name, FileOperation::DeepCopy {new_name, ..} => new_name }.to_owned() } } fn do_operation(writer: &mut zip_next::ZipWriter, operation: &FileOperation) -> Result<(), Box> where T: Read + Write + Seek { match operation { FileOperation::Write {file, mut options} => { if file.contents.iter().map(Vec::len).sum::() >= u32::MAX as usize { options = options.large_file(true); } writer.start_file(file.name.to_owned(), options)?; for chunk in &file.contents { writer.write_all(chunk.as_slice())?; } } FileOperation::WriteLarge {file, mut options} => { options = options.large_file(true); writer.start_file(file.name.to_owned(), options)?; writer.write_all(&*file.large_contents)?; for chunk in &file.extra_contents { writer.write_all(chunk.as_slice())?; } } FileOperation::ShallowCopy {base, new_name} => { do_operation(writer, base)?; writer.shallow_copy_file(&base.get_name(), new_name)?; } FileOperation::DeepCopy {base, new_name} => { do_operation(writer, base)?; writer.deep_copy_file(&base.get_name(), new_name)?; } } Ok(()) } fuzz_target!(|data: Vec| { let mut writer = zip_next::ZipWriter::new(Cursor::new(Vec::new())); for operation in data { let _ = do_operation(&mut writer, &operation); } let _ = zip_next::ZipArchive::new(writer.finish().unwrap()); });