ci(fuzz): Fix issue where we'd call abort_file
on an empty ZIP
This commit is contained in:
parent
0b5fe20e10
commit
9cd005e37f
1 changed files with 103 additions and 9 deletions
|
@ -1,13 +1,14 @@
|
||||||
#![no_main]
|
#![no_main]
|
||||||
|
|
||||||
use arbitrary::Arbitrary;
|
use arbitrary::Arbitrary;
|
||||||
|
use core::fmt::{Debug, Formatter};
|
||||||
use libfuzzer_sys::fuzz_target;
|
use libfuzzer_sys::fuzz_target;
|
||||||
use replace_with::replace_with_or_abort;
|
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;
|
||||||
use zip::result::ZipError;
|
use zip::result::ZipError;
|
||||||
|
|
||||||
#[derive(Arbitrary, Clone, Debug)]
|
#[derive(Arbitrary, Clone)]
|
||||||
pub enum BasicFileOperation<'k> {
|
pub enum BasicFileOperation<'k> {
|
||||||
WriteNormalFile {
|
WriteNormalFile {
|
||||||
contents: Box<[Box<[u8]>]>,
|
contents: Box<[Box<[u8]>]>,
|
||||||
|
@ -32,7 +33,7 @@ pub enum ReopenOption {
|
||||||
ViaFinishIntoReadable
|
ViaFinishIntoReadable
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Arbitrary, Clone, Debug)]
|
#[derive(Arbitrary, Clone)]
|
||||||
pub struct FileOperation<'k> {
|
pub struct FileOperation<'k> {
|
||||||
basic: BasicFileOperation<'k>,
|
basic: BasicFileOperation<'k>,
|
||||||
path: PathBuf,
|
path: PathBuf,
|
||||||
|
@ -40,18 +41,104 @@ pub struct FileOperation<'k> {
|
||||||
// 'abort' flag is separate, to prevent trying to copy an aborted file
|
// 'abort' flag is separate, to prevent trying to copy an aborted file
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Arbitrary, Clone, Debug)]
|
impl <'k> Debug for FileOperation<'k> {
|
||||||
|
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||||
|
match &self.basic {
|
||||||
|
BasicFileOperation::WriteNormalFile {contents, options} => {
|
||||||
|
f.write_fmt(format_args!("let options = {:?};\n\
|
||||||
|
writer.start_file_from_path({:?}, options)?;\n\
|
||||||
|
writer.write_all(&({:?}[..] as [u8]))?;\n\
|
||||||
|
drop(options);\n",
|
||||||
|
options, self.path, contents))
|
||||||
|
},
|
||||||
|
BasicFileOperation::WriteDirectory(options) => {
|
||||||
|
f.write_fmt(format_args!("let options = {:?};\n\
|
||||||
|
writer.add_directory_from_path({:?}, options)?;\n\
|
||||||
|
drop(options);\n",
|
||||||
|
options, self.path))
|
||||||
|
},
|
||||||
|
BasicFileOperation::WriteSymlinkWithTarget {target, options} => {
|
||||||
|
f.write_fmt(format_args!("let options = {:?};\n\
|
||||||
|
writer.add_symlink_from_path({:?}, {:?}, options)?;\n",
|
||||||
|
options, self.path, target.to_owned()))
|
||||||
|
},
|
||||||
|
BasicFileOperation::ShallowCopy(base) => {
|
||||||
|
f.write_fmt(format_args!("let path = {:?};\n\
|
||||||
|
{{\n\
|
||||||
|
{:?}
|
||||||
|
}}\n\
|
||||||
|
writer.shallow_copy_file_from_path(path, {:?})?;\n", base.path, base, self.path))
|
||||||
|
},
|
||||||
|
BasicFileOperation::DeepCopy(base) => {
|
||||||
|
f.write_fmt(format_args!("let path = {:?};\n\
|
||||||
|
{{\n\
|
||||||
|
{:?}
|
||||||
|
}}\n\
|
||||||
|
writer.shallow_copy_file_from_path(path, {:?})?;\n\
|
||||||
|
drop(path);\n", base.path, base, self.path))
|
||||||
|
},
|
||||||
|
BasicFileOperation::MergeWithOtherFile {operations} => {
|
||||||
|
f.write_str("let sub_writer = {\n\
|
||||||
|
let mut writer = ZipWriter::new(Cursor::new(Vec::new()));\n")?;
|
||||||
|
operations.iter().map(|op| {
|
||||||
|
f.write_fmt(format_args!("{:?}", op.0))?;
|
||||||
|
if op.1 {
|
||||||
|
f.write_str("writer.abort_file()?;")
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}).collect::<Result<(), _>>()?;
|
||||||
|
f.write_str("writer\n\
|
||||||
|
};\n\
|
||||||
|
writer = sub_writer;\n")
|
||||||
|
},
|
||||||
|
}?;
|
||||||
|
match &self.reopen {
|
||||||
|
ReopenOption::DoNotReopen => Ok(()),
|
||||||
|
ReopenOption::ViaFinish => {
|
||||||
|
f.write_str("writer = ZipWriter::new_append(writer.finish()?)?;\n")
|
||||||
|
},
|
||||||
|
ReopenOption::ViaFinishIntoReadable => {
|
||||||
|
f.write_str("writer = ZipWriter::new_append(writer.finish_into_readable()?.into_inner())?;\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Arbitrary, Clone)]
|
||||||
pub struct FuzzTestCase<'k> {
|
pub struct FuzzTestCase<'k> {
|
||||||
comment: Box<[u8]>,
|
comment: Box<[u8]>,
|
||||||
operations: Box<[(FileOperation<'k>, bool)]>,
|
operations: Box<[(FileOperation<'k>, bool)]>,
|
||||||
flush_on_finish_file: bool,
|
flush_on_finish_file: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl <'k> Debug for FuzzTestCase<'k> {
|
||||||
|
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_fmt(format_args!(
|
||||||
|
"let mut writer = ZipWriter::new(Cursor::new(Vec::new()));\n\
|
||||||
|
writer.set_flush_on_finish_file({:?});\n", self.flush_on_finish_file))?;
|
||||||
|
self.operations.iter().map(|op| {
|
||||||
|
f.write_fmt(format_args!("{:?}", op.0))?;
|
||||||
|
if op.1 {
|
||||||
|
f.write_str("writer.abort_file()?;")
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Result<(), _>>()?;
|
||||||
|
if self.comment.len() > 0 {
|
||||||
|
f.write_fmt(format_args!("writer.set_raw_comment(Box::<[u8]>::from({:?}));\n", self.comment))?;
|
||||||
|
}
|
||||||
|
f.write_str("writer\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn do_operation<'k, T>(
|
fn do_operation<'k, T>(
|
||||||
writer: &mut zip::ZipWriter<T>,
|
writer: &mut zip::ZipWriter<T>,
|
||||||
operation: &FileOperation<'k>,
|
operation: &FileOperation<'k>,
|
||||||
abort: bool,
|
abort: bool,
|
||||||
flush_on_finish_file: bool,
|
flush_on_finish_file: bool,
|
||||||
|
files_added: &mut usize
|
||||||
) -> Result<(), Box<dyn std::error::Error>>
|
) -> Result<(), Box<dyn std::error::Error>>
|
||||||
where
|
where
|
||||||
T: Read + Write + Seek,
|
T: Read + Write + Seek,
|
||||||
|
@ -73,40 +160,45 @@ where
|
||||||
for chunk in contents.iter() {
|
for chunk in contents.iter() {
|
||||||
writer.write_all(&chunk)?;
|
writer.write_all(&chunk)?;
|
||||||
}
|
}
|
||||||
|
files_added += 1;
|
||||||
}
|
}
|
||||||
BasicFileOperation::WriteDirectory(options) => {
|
BasicFileOperation::WriteDirectory(options) => {
|
||||||
writer.add_directory_from_path(path, options.to_owned())?;
|
writer.add_directory_from_path(path, options.to_owned())?;
|
||||||
|
files_added += 1;
|
||||||
}
|
}
|
||||||
BasicFileOperation::WriteSymlinkWithTarget { target, options } => {
|
BasicFileOperation::WriteSymlinkWithTarget { target, options } => {
|
||||||
writer.add_symlink_from_path(&path, target, options.to_owned())?;
|
writer.add_symlink_from_path(&path, target, options.to_owned())?;
|
||||||
|
files_added += 1;
|
||||||
}
|
}
|
||||||
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.shallow_copy_file_from_path(&base.path, &path)?;
|
writer.shallow_copy_file_from_path(&base.path, &path)?;
|
||||||
|
files_added += 1;
|
||||||
}
|
}
|
||||||
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.deep_copy_file_from_path(&base.path, &path)?;
|
writer.deep_copy_file_from_path(&base.path, &path)?;
|
||||||
|
files_added += 1;
|
||||||
}
|
}
|
||||||
BasicFileOperation::MergeWithOtherFile { operations } => {
|
BasicFileOperation::MergeWithOtherFile { operations } => {
|
||||||
let mut other_writer = zip::ZipWriter::new(Cursor::new(Vec::new()));
|
let mut other_writer = zip::ZipWriter::new(Cursor::new(Vec::new()));
|
||||||
|
let mut inner_files_added = 0;
|
||||||
operations.iter().for_each(|(operation, abort)| {
|
operations.iter().for_each(|(operation, abort)| {
|
||||||
let _ = do_operation(
|
let _ = do_operation(
|
||||||
&mut other_writer,
|
&mut other_writer,
|
||||||
&operation,
|
&operation,
|
||||||
*abort,
|
*abort,
|
||||||
false,
|
false,
|
||||||
|
&mut inner_files_added
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
writer.merge_archive(other_writer.finish_into_readable()?)?;
|
writer.merge_archive(other_writer.finish_into_readable()?)?;
|
||||||
|
files_added += inner_files_added;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if abort {
|
if abort && files_added != 0 {
|
||||||
match writer.abort_file() {
|
writer.abort_file()?;
|
||||||
Ok(()) => {},
|
files_added -= 1;
|
||||||
Err(ZipError::FileNotFound) => {},
|
|
||||||
Err(e) => return Err(Box::new(e))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
let old_comment = writer.get_raw_comment().to_owned();
|
let old_comment = writer.get_raw_comment().to_owned();
|
||||||
match operation.reopen {
|
match operation.reopen {
|
||||||
|
@ -123,6 +215,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
fuzz_target!(|test_case: FuzzTestCase| {
|
fuzz_target!(|test_case: FuzzTestCase| {
|
||||||
|
let mut file_added = false;
|
||||||
let mut writer = zip::ZipWriter::new(Cursor::new(Vec::new()));
|
let mut writer = zip::ZipWriter::new(Cursor::new(Vec::new()));
|
||||||
writer.set_raw_comment(test_case.comment);
|
writer.set_raw_comment(test_case.comment);
|
||||||
let mut final_reopen = false;
|
let mut final_reopen = false;
|
||||||
|
@ -139,6 +232,7 @@ fuzz_target!(|test_case: FuzzTestCase| {
|
||||||
&operation,
|
&operation,
|
||||||
*abort,
|
*abort,
|
||||||
test_case.flush_on_finish_file,
|
test_case.flush_on_finish_file,
|
||||||
|
&mut file_added
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
if final_reopen {
|
if final_reopen {
|
||||||
|
|
Loading…
Add table
Reference in a new issue