ci(fuzz): Fix issue where we'd call abort_file on an empty ZIP

This commit is contained in:
Chris Hennick 2024-05-28 08:47:14 -07:00
parent 0b5fe20e10
commit 9cd005e37f
No known key found for this signature in database
GPG key ID: DA47AABA4961C509

View file

@ -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 {