Bug fix for abort_file when deleting an entry that isn't the last

This commit is contained in:
Chris Hennick 2023-05-21 11:26:33 -07:00
parent 4753b6ecb9
commit fa045ad4c5
No known key found for this signature in database
GPG key ID: 25653935CC8B6C74
3 changed files with 42 additions and 3 deletions

View file

@ -165,3 +165,9 @@
### Merged from upstream
- Uses the `aes::cipher::KeyInit` trait from `aes` 0.8.2 where appropriate.
### Fixed
- Calling `abort_file()` no longer corrupts the archive if called on a
shallow copy of a remaining file, or on an archive whose CDR entries are out
of sequence. However, it may leave an unused entry in the archive.

View file

@ -25,6 +25,7 @@ pub enum BasicFileOperation {
pub struct FileOperation {
basic: BasicFileOperation,
name: String,
abort: bool,
reopen: bool,
}
@ -70,6 +71,9 @@ fn do_operation<T>(writer: &mut RefCell<zip_next::ZipWriter<T>>,
writer.borrow_mut().deep_copy_file(&base_name, &name)?;
}
}
if operation.abort {
writer.abort_file().unwrap();
}
if operation.reopen {
let new_writer = zip_next::ZipWriter::new_append(writer.borrow_mut().finish().unwrap()).unwrap();
*writer = new_writer.into();

View file

@ -713,9 +713,19 @@ impl<W: Write + Seek> ZipWriter<W> {
.inner
.prepare_next_writer(CompressionMethod::Stored, None)?;
self.inner.switch_to(make_plain_writer)?;
self.inner
.get_plain()
.seek(SeekFrom::Start(last_file.header_start))?;
// Make sure this is the last file, and that no shallow copies of it remain; otherwise we'd
// overwrite a valid file and corrupt the archive
if !self.writing_to_file
&& self
.files
.iter()
.all(|file| file.data_start.load() < last_file.data_start.load())
{
self.inner
.get_plain()
.seek(SeekFrom::Start(last_file.header_start))?;
}
self.writing_to_file = false;
Ok(())
}
@ -1774,6 +1784,25 @@ mod test {
let _ = ZipArchive::new(zip).unwrap();
Ok(())
}
#[test]
fn remove_shallow_copy_keeps_original() -> ZipResult<()> {
let mut writer = ZipWriter::new(io::Cursor::new(Vec::new()));
writer
.start_file("original", FileOptions::default())
.unwrap();
writer.write_all(RT_TEST_TEXT.as_bytes()).unwrap();
writer
.shallow_copy_file("original", "shallow_copy")
.unwrap();
writer.abort_file().unwrap();
let mut zip = ZipArchive::new(writer.finish().unwrap()).unwrap();
let mut file = zip.by_name("original").unwrap();
let mut contents = Vec::new();
file.read_to_end(&mut contents).unwrap();
assert_eq!(RT_TEST_TEXT.as_bytes(), contents);
Ok(())
}
}
#[cfg(not(feature = "unreserved"))]