Bug fixes: file can't be ZIP64 if CDR start is after CDR end
This commit is contained in:
parent
61502b22a7
commit
8c6816fb33
3 changed files with 66 additions and 6 deletions
|
@ -351,6 +351,9 @@ impl<R: Read + Seek> ZipArchive<R> {
|
|||
.ok_or(ZipError::InvalidArchive(
|
||||
"Invalid central directory size or offset",
|
||||
))?;
|
||||
if directory_start > search_upper_bound {
|
||||
return Err(ZipError::InvalidArchive("Invalid central directory size or offset"));
|
||||
}
|
||||
|
||||
if footer64.disk_number != footer64.disk_with_central_directory {
|
||||
return unsupported_zip_error("Support for multi-disk files is not implemented");
|
||||
|
|
12
src/spec.rs
12
src/spec.rs
|
@ -61,14 +61,14 @@ impl CentralDirectoryEnd {
|
|||
})
|
||||
}
|
||||
|
||||
pub fn find_and_parse<T: Read + io::Seek>(
|
||||
pub fn find_and_parse<T: Read + Seek>(
|
||||
reader: &mut T,
|
||||
) -> ZipResult<(CentralDirectoryEnd, u64)> {
|
||||
const HEADER_SIZE: u64 = 22;
|
||||
const BYTES_BETWEEN_MAGIC_AND_COMMENT_SIZE: u64 = HEADER_SIZE - 6;
|
||||
let file_length = reader.seek(io::SeekFrom::End(0))?;
|
||||
|
||||
let search_upper_bound = file_length.saturating_sub(HEADER_SIZE + ::std::u16::MAX as u64);
|
||||
let search_upper_bound = file_length.saturating_sub(HEADER_SIZE + u16::MAX as u64);
|
||||
|
||||
if file_length < HEADER_SIZE {
|
||||
return Err(ZipError::InvalidArchive("Invalid zip header"));
|
||||
|
@ -155,14 +155,14 @@ pub struct Zip64CentralDirectoryEnd {
|
|||
}
|
||||
|
||||
impl Zip64CentralDirectoryEnd {
|
||||
pub fn find_and_parse<T: Read + io::Seek>(
|
||||
pub fn find_and_parse<T: Read + Seek>(
|
||||
reader: &mut T,
|
||||
nominal_offset: u64,
|
||||
search_upper_bound: u64,
|
||||
) -> ZipResult<(Zip64CentralDirectoryEnd, u64)> {
|
||||
let mut pos = nominal_offset;
|
||||
let mut pos = search_upper_bound;
|
||||
|
||||
while pos <= search_upper_bound {
|
||||
while pos >= nominal_offset {
|
||||
reader.seek(io::SeekFrom::Start(pos))?;
|
||||
|
||||
if reader.read_u32::<LittleEndian>()? == ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE {
|
||||
|
@ -195,7 +195,7 @@ impl Zip64CentralDirectoryEnd {
|
|||
));
|
||||
}
|
||||
|
||||
pos += 1;
|
||||
pos -= 1;
|
||||
}
|
||||
|
||||
Err(ZipError::InvalidArchive(
|
||||
|
|
57
src/write.rs
57
src/write.rs
|
@ -1481,6 +1481,7 @@ mod test {
|
|||
use crate::ZipArchive;
|
||||
use std::io;
|
||||
use std::io::{Read, Write};
|
||||
use zstd::zstd_safe::WriteBuf;
|
||||
|
||||
#[test]
|
||||
fn write_empty_zip() {
|
||||
|
@ -1732,6 +1733,62 @@ mod test {
|
|||
.expect_err("Expected duplicate filename not to be allowed");
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_filename_looks_like_zip64_locator() {
|
||||
let mut writer = ZipWriter::new(io::Cursor::new(Vec::new()));
|
||||
writer
|
||||
.start_file(
|
||||
"PK\u{6}\u{7}\0\0\0\u{11}\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||
FileOptions::default(),
|
||||
)
|
||||
.unwrap();
|
||||
let zip = writer.finish().unwrap();
|
||||
let _ = ZipArchive::new(zip).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_filename_looks_like_zip64_locator_2() {
|
||||
let mut writer = ZipWriter::new(io::Cursor::new(Vec::new()));
|
||||
writer
|
||||
.start_file(
|
||||
"PK\u{6}\u{6}\0\0\0\0\0\0\0\0\0\0PK\u{6}\u{7}\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||
FileOptions::default(),
|
||||
)
|
||||
.unwrap();
|
||||
let zip = writer.finish().unwrap();
|
||||
println!("{:02x?}", zip.get_ref());
|
||||
let _ = ZipArchive::new(zip).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_filename_looks_like_zip64_locator_2a() {
|
||||
let mut writer = ZipWriter::new(io::Cursor::new(Vec::new()));
|
||||
writer
|
||||
.start_file(
|
||||
"PK\u{6}\u{6}PK\u{6}\u{7}\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||
FileOptions::default(),
|
||||
)
|
||||
.unwrap();
|
||||
let zip = writer.finish().unwrap();
|
||||
println!("{:02x?}", zip.get_ref());
|
||||
let _ = ZipArchive::new(zip).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_filename_looks_like_zip64_locator_3() {
|
||||
let mut writer = ZipWriter::new(io::Cursor::new(Vec::new()));
|
||||
writer.start_file("\0PK\u{6}\u{6}", FileOptions::default()).unwrap();
|
||||
writer
|
||||
.start_file(
|
||||
"\0\u{4}\0\0PK\u{6}\u{7}\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\u{3}",
|
||||
FileOptions::default(),
|
||||
)
|
||||
.unwrap();
|
||||
let zip = writer.finish().unwrap();
|
||||
println!("{:02x?}", zip.get_ref());
|
||||
let _ = ZipArchive::new(zip).unwrap();
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn path_to_string() {
|
||||
let mut path = std::path::PathBuf::new();
|
||||
|
|
Loading…
Add table
Reference in a new issue