feat: expose the location of the central header

This commit is contained in:
Marli Frost 2020-09-10 09:34:24 +01:00
parent 63a3e89ef8
commit f863497e9b
No known key found for this signature in database
GPG key ID: CB0BEA7CF9BD1245
3 changed files with 16 additions and 2 deletions

View file

@ -433,6 +433,7 @@ fn central_header_to_zip_file<R: Read + io::Seek>(
reader: &mut R, reader: &mut R,
archive_offset: u64, archive_offset: u64,
) -> ZipResult<ZipFileData> { ) -> ZipResult<ZipFileData> {
let central_header_start = reader.seek(io::SeekFrom::Current(0))?;
// Parse central header // Parse central header
let signature = reader.read_u32::<LittleEndian>()?; let signature = reader.read_u32::<LittleEndian>()?;
if signature != spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE { if signature != spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE {
@ -490,6 +491,7 @@ fn central_header_to_zip_file<R: Read + io::Seek>(
file_name_raw, file_name_raw,
file_comment, file_comment,
header_start: offset, header_start: offset,
central_header_start,
data_start: 0, data_start: 0,
external_attributes: external_file_attributes, external_attributes: external_file_attributes,
}; };
@ -646,6 +648,10 @@ impl<'a> ZipFile<'a> {
pub fn header_start(&self) -> u64 { pub fn header_start(&self) -> u64 {
self.data.header_start self.data.header_start
} }
/// Get the starting offset of the zip header in the central directory for this file
pub fn central_header_start(&self) -> u64 {
self.data.central_header_start
}
} }
impl<'a> Read for ZipFile<'a> { impl<'a> Read for ZipFile<'a> {
@ -747,6 +753,7 @@ pub fn read_zipfile_from_stream<'a, R: io::Read>(
// not available. // not available.
header_start: 0, header_start: 0,
data_start: 0, data_start: 0,
central_header_start: 0,
// The external_attributes field is only available in the central directory. // The external_attributes field is only available in the central directory.
// We set this to zero, which should be valid as the docs state 'If input came // We set this to zero, which should be valid as the docs state 'If input came
// from standard input, this field is set to zero.' // from standard input, this field is set to zero.'
@ -811,14 +818,15 @@ mod test {
} }
#[test] #[test]
fn zip_comment() { fn zip_contents() {
use super::ZipArchive; use super::ZipArchive;
use std::io; use std::io;
let mut v = Vec::new(); let mut v = Vec::new();
v.extend_from_slice(include_bytes!("../tests/data/mimetype.zip")); v.extend_from_slice(include_bytes!("../tests/data/mimetype.zip"));
let reader = ZipArchive::new(io::Cursor::new(v)).unwrap(); let mut reader = ZipArchive::new(io::Cursor::new(v)).unwrap();
assert!(reader.comment() == b""); assert!(reader.comment() == b"");
assert_eq!(reader.by_index(0).unwrap().central_header_start(), 77);
} }
#[test] #[test]

View file

@ -234,6 +234,10 @@ pub struct ZipFileData {
pub file_comment: String, pub file_comment: String,
/// Specifies where the local header of the file starts /// Specifies where the local header of the file starts
pub header_start: u64, pub header_start: u64,
/// Specifies where the central header of the file starts
///
/// Note that when this is not known, it is set to 0
pub central_header_start: u64,
/// Specifies where the compressed data of the file starts /// Specifies where the compressed data of the file starts
pub data_start: u64, pub data_start: u64,
/// External file attributes /// External file attributes
@ -309,6 +313,7 @@ mod test {
file_comment: String::new(), file_comment: String::new(),
header_start: 0, header_start: 0,
data_start: 0, data_start: 0,
central_header_start: 0,
external_attributes: 0, external_attributes: 0,
}; };
assert_eq!( assert_eq!(

View file

@ -236,6 +236,7 @@ impl<W: Write + io::Seek> ZipWriter<W> {
file_comment: String::new(), file_comment: String::new(),
header_start, header_start,
data_start: 0, data_start: 0,
central_header_start: 0,
external_attributes: permissions << 16, external_attributes: permissions << 16,
}; };
write_local_file_header(writer, &file)?; write_local_file_header(writer, &file)?;