remove a lot of boilerplate for Block impls
This commit is contained in:
parent
08385d52e1
commit
7eb5907622
2 changed files with 92 additions and 120 deletions
154
src/spec.rs
154
src/spec.rs
|
@ -60,8 +60,20 @@ 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;
|
||||||
|
|
||||||
pub trait Block: Sized + Copy {
|
pub trait Block: Sized + Copy {
|
||||||
|
const MAGIC: Magic;
|
||||||
|
|
||||||
|
fn magic(self) -> Magic;
|
||||||
|
|
||||||
|
const ERROR: ZipError;
|
||||||
|
|
||||||
/* TODO: use smallvec? */
|
/* TODO: use smallvec? */
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self>;
|
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
||||||
|
let block = Self::deserialize(&bytes).from_le();
|
||||||
|
if block.magic() != Self::MAGIC {
|
||||||
|
return Err(Self::ERROR);
|
||||||
|
}
|
||||||
|
Ok(block)
|
||||||
|
}
|
||||||
|
|
||||||
fn deserialize(block: &[u8]) -> Self {
|
fn deserialize(block: &[u8]) -> Self {
|
||||||
assert_eq!(block.len(), mem::size_of::<Self>());
|
assert_eq!(block.len(), mem::size_of::<Self>());
|
||||||
|
@ -69,18 +81,27 @@ pub trait Block: Sized + Copy {
|
||||||
unsafe { block_ptr.read() }
|
unsafe { block_ptr.read() }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::wrong_self_convention)]
|
||||||
|
fn from_le(self) -> Self;
|
||||||
|
|
||||||
fn parse<T: Read>(reader: &mut T) -> ZipResult<Self> {
|
fn parse<T: Read>(reader: &mut T) -> ZipResult<Self> {
|
||||||
let mut block = vec![0u8; mem::size_of::<Self>()];
|
let mut block = vec![0u8; mem::size_of::<Self>()];
|
||||||
reader.read_exact(&mut block)?;
|
reader.read_exact(&mut block)?;
|
||||||
Self::interpret(block.into_boxed_slice())
|
Self::interpret(block.into_boxed_slice())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn encode(self) -> Box<[u8]>;
|
fn encode(self) -> Box<[u8]> {
|
||||||
|
self.to_le().serialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn to_le(self) -> Self;
|
||||||
|
|
||||||
|
/* TODO: use Box<[u8; mem::size_of::<Self>()]> when generic_const_exprs are stabilized! */
|
||||||
fn serialize(self) -> Box<[u8]> {
|
fn serialize(self) -> Box<[u8]> {
|
||||||
|
/* TODO: use Box::new_zeroed() when stabilized! */
|
||||||
|
/* TODO: also consider using smallvec! */
|
||||||
let mut out_block = vec![0u8; mem::size_of::<Self>()];
|
let mut out_block = vec![0u8; mem::size_of::<Self>()];
|
||||||
let out_view: &mut [u8] = out_block.as_mut();
|
let out_ptr: *mut Self = out_block.as_mut_ptr().cast();
|
||||||
let out_ptr: *mut Self = out_view.as_mut_ptr().cast();
|
|
||||||
unsafe {
|
unsafe {
|
||||||
out_ptr.write(self);
|
out_ptr.write(self);
|
||||||
}
|
}
|
||||||
|
@ -135,10 +156,18 @@ pub struct Zip32CDEBlock {
|
||||||
pub zip_file_comment_length: u16,
|
pub zip_file_comment_length: u16,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Zip32CDEBlock {
|
impl Block for Zip32CDEBlock {
|
||||||
#[allow(clippy::wrong_self_convention)]
|
const MAGIC: Magic = CENTRAL_DIRECTORY_END_SIGNATURE;
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn from_le(mut self) -> Self {
|
fn magic(self) -> Magic {
|
||||||
|
self.magic
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERROR: ZipError = ZipError::InvalidArchive("Invalid digital signature header");
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
fn from_le(mut self) -> Self {
|
||||||
from_le![
|
from_le![
|
||||||
self,
|
self,
|
||||||
[
|
[
|
||||||
|
@ -156,7 +185,7 @@ impl Zip32CDEBlock {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn to_le(mut self) -> Self {
|
fn to_le(mut self) -> Self {
|
||||||
to_le![
|
to_le![
|
||||||
self,
|
self,
|
||||||
[
|
[
|
||||||
|
@ -174,23 +203,6 @@ impl Zip32CDEBlock {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Block for Zip32CDEBlock {
|
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
|
||||||
let block = Self::deserialize(&bytes).from_le();
|
|
||||||
|
|
||||||
let magic = block.magic;
|
|
||||||
if magic != CENTRAL_DIRECTORY_END_SIGNATURE {
|
|
||||||
return Err(ZipError::InvalidArchive("Invalid digital signature header"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(block)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn encode(self) -> Box<[u8]> {
|
|
||||||
self.to_le().serialize()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Zip32CentralDirectoryEnd {
|
pub struct Zip32CentralDirectoryEnd {
|
||||||
pub disk_number: u16,
|
pub disk_number: u16,
|
||||||
|
@ -344,10 +356,19 @@ pub struct Zip64CDELocatorBlock {
|
||||||
pub number_of_disks: u32,
|
pub number_of_disks: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Zip64CDELocatorBlock {
|
impl Block for Zip64CDELocatorBlock {
|
||||||
#[allow(clippy::wrong_self_convention)]
|
const MAGIC: Magic = ZIP64_CENTRAL_DIRECTORY_END_LOCATOR_SIGNATURE;
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn from_le(mut self) -> Self {
|
fn magic(self) -> Magic {
|
||||||
|
self.magic
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERROR: ZipError =
|
||||||
|
ZipError::InvalidArchive("Invalid zip64 locator digital signature header");
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
fn from_le(mut self) -> Self {
|
||||||
from_le![
|
from_le![
|
||||||
self,
|
self,
|
||||||
[
|
[
|
||||||
|
@ -361,7 +382,7 @@ impl Zip64CDELocatorBlock {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn to_le(mut self) -> Self {
|
fn to_le(mut self) -> Self {
|
||||||
to_le![
|
to_le![
|
||||||
self,
|
self,
|
||||||
[
|
[
|
||||||
|
@ -375,25 +396,6 @@ impl Zip64CDELocatorBlock {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Block for Zip64CDELocatorBlock {
|
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
|
||||||
let block = Self::deserialize(&bytes).from_le();
|
|
||||||
|
|
||||||
let magic = block.magic;
|
|
||||||
if magic != ZIP64_CENTRAL_DIRECTORY_END_LOCATOR_SIGNATURE {
|
|
||||||
return Err(ZipError::InvalidArchive(
|
|
||||||
"Invalid zip64 locator digital signature header",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(block)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn encode(self) -> Box<[u8]> {
|
|
||||||
self.to_le().serialize()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Zip64CentralDirectoryEndLocator {
|
pub struct Zip64CentralDirectoryEndLocator {
|
||||||
pub disk_with_central_directory: u32,
|
pub disk_with_central_directory: u32,
|
||||||
pub end_of_central_directory_offset: u64,
|
pub end_of_central_directory_offset: u64,
|
||||||
|
@ -451,10 +453,17 @@ pub struct Zip64CDEBlock {
|
||||||
pub central_directory_offset: u64,
|
pub central_directory_offset: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Zip64CDEBlock {
|
impl Block for Zip64CDEBlock {
|
||||||
#[allow(clippy::wrong_self_convention)]
|
const MAGIC: Magic = ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE;
|
||||||
|
|
||||||
|
fn magic(self) -> Magic {
|
||||||
|
self.magic
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERROR: ZipError = ZipError::InvalidArchive("Invalid digital signature header");
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn from_le(mut self) -> Self {
|
fn from_le(mut self) -> Self {
|
||||||
from_le![
|
from_le![
|
||||||
self,
|
self,
|
||||||
[
|
[
|
||||||
|
@ -474,7 +483,7 @@ impl Zip64CDEBlock {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub(crate) fn to_le(mut self) -> Self {
|
fn to_le(mut self) -> Self {
|
||||||
to_le![
|
to_le![
|
||||||
self,
|
self,
|
||||||
[
|
[
|
||||||
|
@ -494,23 +503,6 @@ impl Zip64CDEBlock {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Block for Zip64CDEBlock {
|
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
|
||||||
let block = Self::deserialize(&bytes).from_le();
|
|
||||||
|
|
||||||
let magic = block.magic;
|
|
||||||
if magic != ZIP64_CENTRAL_DIRECTORY_END_SIGNATURE {
|
|
||||||
return Err(ZipError::InvalidArchive("Invalid digital signature header"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(block)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn encode(self) -> Box<[u8]> {
|
|
||||||
self.to_le().serialize()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Zip64CentralDirectoryEnd {
|
pub struct Zip64CentralDirectoryEnd {
|
||||||
pub version_made_by: u16,
|
pub version_made_by: u16,
|
||||||
pub version_needed_to_extract: u16,
|
pub version_needed_to_extract: u16,
|
||||||
|
@ -729,27 +721,25 @@ mod test {
|
||||||
pub file_name_length: u16,
|
pub file_name_length: u16,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TestBlock {
|
impl Block for TestBlock {
|
||||||
#[allow(clippy::wrong_self_convention)]
|
const MAGIC: Magic = Magic::literal(0x01111);
|
||||||
pub(crate) fn from_le(mut self) -> Self {
|
|
||||||
|
fn magic(self) -> Magic {
|
||||||
|
self.magic
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERROR: ZipError = ZipError::InvalidArchive("unreachable");
|
||||||
|
|
||||||
|
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
|
||||||
}
|
}
|
||||||
pub(crate) fn to_le(mut self) -> Self {
|
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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Block for TestBlock {
|
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
|
||||||
Ok(Self::deserialize(&bytes).from_le())
|
|
||||||
}
|
|
||||||
fn encode(self) -> Box<[u8]> {
|
|
||||||
self.to_le().serialize()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Demonstrate that a block object can be safely written to memory and deserialized back out.
|
/// Demonstrate that a block object can be safely written to memory and deserialized back out.
|
||||||
#[test]
|
#[test]
|
||||||
fn block_serde() {
|
fn block_serde() {
|
||||||
|
|
58
src/types.rs
58
src/types.rs
|
@ -810,8 +810,16 @@ pub(crate) struct ZipEntryBlock {
|
||||||
pub offset: u32,
|
pub offset: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ZipEntryBlock {
|
impl Block for ZipEntryBlock {
|
||||||
#[allow(clippy::wrong_self_convention)]
|
const MAGIC: spec::Magic = spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE;
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
fn magic(self) -> spec::Magic {
|
||||||
|
self.magic
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERROR: ZipError = ZipError::InvalidArchive("Invalid Central Directory header");
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
fn from_le(mut self) -> Self {
|
fn from_le(mut self) -> Self {
|
||||||
from_le![
|
from_le![
|
||||||
|
@ -867,23 +875,6 @@ impl ZipEntryBlock {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Block for ZipEntryBlock {
|
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
|
||||||
let block = Self::deserialize(&bytes).from_le();
|
|
||||||
|
|
||||||
let magic = block.magic;
|
|
||||||
if magic != spec::CENTRAL_DIRECTORY_HEADER_SIGNATURE {
|
|
||||||
return Err(ZipError::InvalidArchive("Invalid Central Directory header"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(block)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn encode(self) -> Box<[u8]> {
|
|
||||||
self.to_le().serialize()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug)]
|
#[derive(Copy, Clone, Debug)]
|
||||||
#[repr(packed)]
|
#[repr(packed)]
|
||||||
pub(crate) struct ZipLocalEntryBlock {
|
pub(crate) struct ZipLocalEntryBlock {
|
||||||
|
@ -900,8 +891,16 @@ pub(crate) struct ZipLocalEntryBlock {
|
||||||
pub extra_field_length: u16,
|
pub extra_field_length: u16,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ZipLocalEntryBlock {
|
impl Block for ZipLocalEntryBlock {
|
||||||
#[allow(clippy::wrong_self_convention)]
|
const MAGIC: spec::Magic = spec::LOCAL_FILE_HEADER_SIGNATURE;
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
fn magic(self) -> spec::Magic {
|
||||||
|
self.magic
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERROR: ZipError = ZipError::InvalidArchive("Invalid local file header");
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
fn from_le(mut self) -> Self {
|
fn from_le(mut self) -> Self {
|
||||||
from_le![
|
from_le![
|
||||||
|
@ -945,23 +944,6 @@ impl ZipLocalEntryBlock {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Block for ZipLocalEntryBlock {
|
|
||||||
fn interpret(bytes: Box<[u8]>) -> ZipResult<Self> {
|
|
||||||
let block = Self::deserialize(&bytes).from_le();
|
|
||||||
|
|
||||||
let magic = block.magic;
|
|
||||||
if magic != spec::LOCAL_FILE_HEADER_SIGNATURE {
|
|
||||||
return Err(ZipError::InvalidArchive("Invalid local file header"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(block)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn encode(self) -> Box<[u8]> {
|
|
||||||
self.to_le().serialize()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// The encryption specification used to encrypt a file with AES.
|
/// The encryption specification used to encrypt a file with AES.
|
||||||
///
|
///
|
||||||
/// According to the [specification](https://www.winzip.com/win/en/aes_info.html#winzip11) AE-2
|
/// According to the [specification](https://www.winzip.com/win/en/aes_info.html#winzip11) AE-2
|
||||||
|
|
Loading…
Add table
Reference in a new issue