make Magic into a wrapper struct

This commit is contained in:
Danny McClanahan 2024-05-18 02:38:57 -04:00
parent 46c42c7f82
commit 3fa0d84554
No known key found for this signature in database
GPG key ID: 6105C10F1A199CC7
3 changed files with 68 additions and 26 deletions

View file

@ -221,7 +221,8 @@ pub(crate) fn find_content<'a>(
) -> ZipResult<io::Take<&'a mut dyn Read>> { ) -> ZipResult<io::Take<&'a mut dyn Read>> {
// Parse local header // Parse local header
reader.seek(io::SeekFrom::Start(data.header_start))?; reader.seek(io::SeekFrom::Start(data.header_start))?;
let signature = reader.read_u32_le()?; /* FIXME: read this in blocks too! ::literal is not for general use! */
let signature = spec::Magic::literal(reader.read_u32_le()?);
if signature != spec::LOCAL_FILE_HEADER_SIGNATURE { if signature != spec::LOCAL_FILE_HEADER_SIGNATURE {
return Err(ZipError::InvalidArchive("Invalid local file header")); return Err(ZipError::InvalidArchive("Invalid local file header"));
} }
@ -1430,11 +1431,7 @@ pub fn read_zipfile_from_stream<'a, R: Read>(reader: &'a mut R) -> ZipResult<Opt
reader.read_exact(&mut block)?; reader.read_exact(&mut block)?;
let block: Box<[u8]> = block.into(); let block: Box<[u8]> = block.into();
let signature = spec::Magic::from_le_bytes( let signature = spec::Magic::from_first_le_bytes(&block);
block[..mem::size_of_val(&spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE)]
.try_into()
.unwrap(),
);
match signature { match signature {
spec::LOCAL_FILE_HEADER_SIGNATURE => (), spec::LOCAL_FILE_HEADER_SIGNATURE => (),

View file

@ -8,13 +8,53 @@ use std::io::prelude::*;
use std::mem; use std::mem;
use std::path::{Component, Path, MAIN_SEPARATOR}; use std::path::{Component, Path, MAIN_SEPARATOR};
pub type Magic = u32; /// "Magic" header values used in the zip spec to locate metadata records.
///
/// These values currently always take up a fixed four bytes, so we can parse and wrap them in this
/// struct to enforce some small amount of type safety.
#[derive(Copy, Clone, Debug, PartialOrd, Ord, PartialEq, Eq, Hash)]
#[repr(transparent)]
pub struct Magic(u32);
pub const LOCAL_FILE_HEADER_SIGNATURE: Magic = 0x04034b50; impl Magic {
pub const CENTRAL_DIRECTORY_HEADER_SIGNATURE: Magic = 0x02014b50; pub(crate) const fn literal(x: u32) -> Self {
pub(crate) const CENTRAL_DIRECTORY_END_SIGNATURE: Magic = 0x06054b50; Self(x)
pub const ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE: Magic = 0x06064b50; }
pub(crate) const ZIP64_CENTRAL_DIRECTORY_END_LOCATOR_SIGNATURE: Magic = 0x07064b50;
#[inline(always)]
pub(crate) const fn from_le_bytes(bytes: [u8; 4]) -> Self {
Self(u32::from_le_bytes(bytes))
}
#[inline(always)]
pub(crate) fn from_first_le_bytes(data: &[u8]) -> Self {
let first_bytes: [u8; 4] = data[..mem::size_of::<Self>()].try_into().unwrap();
Self::from_le_bytes(first_bytes)
}
#[inline(always)]
pub(crate) const fn to_le_bytes(self) -> [u8; 4] {
self.0.to_le_bytes()
}
#[allow(clippy::wrong_self_convention)]
#[inline(always)]
pub(crate) fn from_le(self) -> Self {
Self(u32::from_le(self.0))
}
#[allow(clippy::wrong_self_convention)]
#[inline(always)]
pub(crate) fn to_le(self) -> Self {
Self(u32::to_le(self.0))
}
}
pub const LOCAL_FILE_HEADER_SIGNATURE: Magic = Magic::literal(0x04034b50);
pub const CENTRAL_DIRECTORY_HEADER_SIGNATURE: Magic = Magic::literal(0x02014b50);
pub(crate) const CENTRAL_DIRECTORY_END_SIGNATURE: Magic = Magic::literal(0x06054b50);
pub const ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE: Magic = Magic::literal(0x06064b50);
pub(crate) const ZIP64_CENTRAL_DIRECTORY_END_LOCATOR_SIGNATURE: Magic = Magic::literal(0x07064b50);
pub const ZIP64_BYTES_THR: u64 = u32::MAX as u64; pub const ZIP64_BYTES_THR: u64 = u32::MAX as u64;
pub const ZIP64_ENTRY_THR: usize = u16::MAX as usize; pub const ZIP64_ENTRY_THR: usize = u16::MAX as usize;
@ -98,7 +138,7 @@ pub struct Zip32CDEBlock {
impl Zip32CDEBlock { impl Zip32CDEBlock {
#[allow(clippy::wrong_self_convention)] #[allow(clippy::wrong_self_convention)]
#[inline(always)] #[inline(always)]
fn from_le(mut self) -> Self { pub(crate) fn from_le(mut self) -> Self {
from_le![ from_le![
self, self,
[ [
@ -116,7 +156,7 @@ impl Zip32CDEBlock {
} }
#[inline(always)] #[inline(always)]
fn to_le(mut self) -> Self { pub(crate) fn to_le(mut self) -> Self {
to_le![ to_le![
self, self,
[ [
@ -138,7 +178,8 @@ impl Block for Zip32CDEBlock {
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> { fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
let block = Self::deserialize(&bytes).from_le(); let block = Self::deserialize(&bytes).from_le();
if block.magic != CENTRAL_DIRECTORY_END_SIGNATURE { let magic = block.magic;
if magic != CENTRAL_DIRECTORY_END_SIGNATURE {
return Err(ZipError::InvalidArchive("Invalid digital signature header")); return Err(ZipError::InvalidArchive("Invalid digital signature header"));
} }
@ -306,7 +347,7 @@ pub struct Zip64CDELocatorBlock {
impl Zip64CDELocatorBlock { impl Zip64CDELocatorBlock {
#[allow(clippy::wrong_self_convention)] #[allow(clippy::wrong_self_convention)]
#[inline(always)] #[inline(always)]
fn from_le(mut self) -> Self { pub(crate) fn from_le(mut self) -> Self {
from_le![ from_le![
self, self,
[ [
@ -320,7 +361,7 @@ impl Zip64CDELocatorBlock {
} }
#[inline(always)] #[inline(always)]
fn to_le(mut self) -> Self { pub(crate) fn to_le(mut self) -> Self {
to_le![ to_le![
self, self,
[ [
@ -338,7 +379,8 @@ impl Block for Zip64CDELocatorBlock {
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> { fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
let block = Self::deserialize(&bytes).from_le(); let block = Self::deserialize(&bytes).from_le();
if block.magic != ZIP64_CENTRAL_DIRECTORY_END_LOCATOR_SIGNATURE { let magic = block.magic;
if magic != ZIP64_CENTRAL_DIRECTORY_END_LOCATOR_SIGNATURE {
return Err(ZipError::InvalidArchive( return Err(ZipError::InvalidArchive(
"Invalid zip64 locator digital signature header", "Invalid zip64 locator digital signature header",
)); ));
@ -412,7 +454,7 @@ pub struct Zip64CDEBlock {
impl Zip64CDEBlock { impl Zip64CDEBlock {
#[allow(clippy::wrong_self_convention)] #[allow(clippy::wrong_self_convention)]
#[inline(always)] #[inline(always)]
fn from_le(mut self) -> Self { pub(crate) fn from_le(mut self) -> Self {
from_le![ from_le![
self, self,
[ [
@ -432,7 +474,7 @@ impl Zip64CDEBlock {
} }
#[inline(always)] #[inline(always)]
fn to_le(mut self) -> Self { pub(crate) fn to_le(mut self) -> Self {
to_le![ to_le![
self, self,
[ [
@ -456,7 +498,8 @@ impl Block for Zip64CDEBlock {
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> { fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
let block = Self::deserialize(&bytes).from_le(); let block = Self::deserialize(&bytes).from_le();
if block.magic != ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE { let magic = block.magic;
if magic != ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE {
return Err(ZipError::InvalidArchive("Invalid digital signature header")); return Err(ZipError::InvalidArchive("Invalid digital signature header"));
} }
@ -688,11 +731,11 @@ mod test {
impl TestBlock { impl TestBlock {
#[allow(clippy::wrong_self_convention)] #[allow(clippy::wrong_self_convention)]
fn from_le(mut self) -> Self { pub(crate) fn from_le(mut self) -> Self {
from_le![self, [(magic, Magic), (file_name_length, u16)]]; from_le![self, [(magic, Magic), (file_name_length, u16)]];
self self
} }
fn to_le(mut self) -> Self { pub(crate) fn to_le(mut self) -> Self {
to_le![self, [(magic, Magic), (file_name_length, u16)]]; to_le![self, [(magic, Magic), (file_name_length, u16)]];
self self
} }
@ -711,7 +754,7 @@ mod test {
#[test] #[test]
fn block_serde() { fn block_serde() {
let block = TestBlock { let block = TestBlock {
magic: 0x01111, magic: Magic::literal(0x01111),
file_name_length: 3, file_name_length: 3,
}; };
let mut c = Cursor::new(Vec::new()); let mut c = Cursor::new(Vec::new());

View file

@ -871,7 +871,8 @@ impl Block for ZipEntryBlock {
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> { fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
let block = Self::deserialize(&bytes).from_le(); let block = Self::deserialize(&bytes).from_le();
if block.magic != spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE { let magic = block.magic;
if magic != spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE {
return Err(ZipError::InvalidArchive("Invalid Central Directory header")); return Err(ZipError::InvalidArchive("Invalid Central Directory header"));
} }
@ -948,7 +949,8 @@ impl Block for ZipLocalEntryBlock {
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> { fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
let block = Self::deserialize(&bytes).from_le(); let block = Self::deserialize(&bytes).from_le();
if block.magic != spec::LOCAL_FILE_HEADER_SIGNATURE { let magic = block.magic;
if magic != spec::LOCAL_FILE_HEADER_SIGNATURE {
return Err(ZipError::InvalidArchive("Invalid local file header")); return Err(ZipError::InvalidArchive("Invalid local file header"));
} }