103 lines
No EOL
3.6 KiB
Rust
103 lines
No EOL
3.6 KiB
Rust
#![no_main]
|
|
|
|
use std::cell::RefCell;
|
|
use libfuzzer_sys::fuzz_target;
|
|
use arbitrary::Arbitrary;
|
|
use std::io::{Cursor, Read, Seek, Write};
|
|
use std::path::{PathBuf};
|
|
|
|
#[derive(Arbitrary,Clone,Debug)]
|
|
pub enum BasicFileOperation {
|
|
WriteNormalFile {
|
|
contents: Vec<Vec<u8>>,
|
|
options: zip_next::write::FileOptions,
|
|
},
|
|
WriteDirectory(zip_next::write::FileOptions),
|
|
WriteSymlinkWithTarget {
|
|
target: Box<PathBuf>,
|
|
options: zip_next::write::FileOptions,
|
|
},
|
|
ShallowCopy(Box<FileOperation>),
|
|
DeepCopy(Box<FileOperation>),
|
|
}
|
|
|
|
#[derive(Arbitrary,Clone,Debug)]
|
|
pub struct FileOperation {
|
|
basic: BasicFileOperation,
|
|
name: String,
|
|
reopen: bool,
|
|
// 'abort' flag is separate, to prevent trying to copy an aborted file
|
|
}
|
|
|
|
#[derive(Arbitrary,Clone,Debug)]
|
|
pub struct FuzzTestCase {
|
|
comment: Vec<u8>,
|
|
operations: Vec<(FileOperation, bool)>,
|
|
flush_on_finish_file: bool,
|
|
}
|
|
|
|
impl FileOperation {
|
|
fn referenceable_name(&self) -> String {
|
|
if let BasicFileOperation::WriteDirectory(_) = self.basic {
|
|
if !self.name.ends_with('\\') && !self.name.ends_with('/') {
|
|
return self.name.to_owned() + "/";
|
|
}
|
|
}
|
|
self.name.to_owned()
|
|
}
|
|
}
|
|
|
|
fn do_operation<T>(writer: &mut RefCell<zip_next::ZipWriter<T>>,
|
|
operation: FileOperation,
|
|
abort: bool, flush_on_finish_file: bool) -> Result<(), Box<dyn std::error::Error>>
|
|
where T: Read + Write + Seek {
|
|
let name = operation.name;
|
|
match operation.basic {
|
|
BasicFileOperation::WriteNormalFile {contents, mut options, ..} => {
|
|
if contents.iter().map(Vec::len).sum::<usize>() >= u32::MAX as usize {
|
|
options = options.large_file(true);
|
|
}
|
|
writer.borrow_mut().start_file(name, options)?;
|
|
for chunk in contents {
|
|
writer.borrow_mut().write_all(chunk.as_slice())?;
|
|
}
|
|
}
|
|
BasicFileOperation::WriteDirectory(options) => {
|
|
writer.borrow_mut().add_directory(name, options)?;
|
|
}
|
|
BasicFileOperation::WriteSymlinkWithTarget {target, options} => {
|
|
writer.borrow_mut().add_symlink(name, target.to_string_lossy(), options)?;
|
|
}
|
|
BasicFileOperation::ShallowCopy(base) => {
|
|
let base_name = base.referenceable_name();
|
|
do_operation(writer, *base, false, flush_on_finish_file)?;
|
|
writer.borrow_mut().shallow_copy_file(&base_name, &name)?;
|
|
}
|
|
BasicFileOperation::DeepCopy(base) => {
|
|
let base_name = base.referenceable_name();
|
|
do_operation(writer, *base, false, flush_on_finish_file)?;
|
|
writer.borrow_mut().deep_copy_file(&base_name, &name)?;
|
|
}
|
|
}
|
|
if abort {
|
|
writer.borrow_mut().abort_file().unwrap();
|
|
}
|
|
if operation.reopen {
|
|
let old_comment = writer.borrow().get_raw_comment().to_owned();
|
|
let new_writer = zip_next::ZipWriter::new_append(
|
|
writer.borrow_mut().finish().unwrap(), flush_on_finish_file).unwrap();
|
|
assert_eq!(&old_comment, new_writer.get_raw_comment());
|
|
*writer = new_writer.into();
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fuzz_target!(|test_case: FuzzTestCase| {
|
|
let mut writer = RefCell::new(zip_next::ZipWriter::new(Cursor::new(Vec::new()),
|
|
test_case.flush_on_finish_file));
|
|
writer.borrow_mut().set_raw_comment(test_case.comment);
|
|
for (operation, abort) in test_case.operations {
|
|
let _ = do_operation(&mut writer, operation, abort, test_case.flush_on_finish_file);
|
|
}
|
|
let _ = zip_next::ZipArchive::new(writer.borrow_mut().finish().unwrap());
|
|
}); |