repak/src/footer.rs
2023-01-13 12:38:30 +00:00

67 lines
2.3 KiB
Rust

use std::str::FromStr;
use byteorder::{ReadBytesExt, LE};
use super::{Compression, ReadExt, Version};
#[derive(Debug)]
pub struct Footer {
pub encryption_uuid: Option<u128>,
pub encrypted: bool,
pub magic: u32,
pub version: Version,
pub index_offset: u64,
pub index_size: u64,
pub hash: [u8; 20],
pub frozen: bool,
pub compression: Vec<Compression>,
}
impl Footer {
pub fn new<R: std::io::Read>(reader: &mut R, version: Version) -> Result<Self, super::Error> {
let footer = Self {
encryption_uuid: match version >= Version::EncryptionKeyGuid {
true => Some(reader.read_u128::<LE>()?),
false => None,
},
encrypted: version >= Version::IndexEncryption && reader.read_bool()?,
magic: reader.read_u32::<LE>()?,
version: Version::from_repr(reader.read_u32::<LE>()?).unwrap_or(version),
index_offset: reader.read_u64::<LE>()?,
index_size: reader.read_u64::<LE>()?,
hash: reader.read_guid()?,
frozen: version == Version::FrozenIndex && reader.read_bool()?,
compression: {
let mut compression = Vec::with_capacity(match version {
ver if ver < Version::FNameBasedCompression => 0,
Version::FNameBasedCompression => 4,
_ => 5,
});
for _ in 0..compression.capacity() {
compression.push(
Compression::from_str(
&reader
.read_len(32)?
.iter()
// filter out whitespace and convert to char
.filter_map(|&ch| (ch != 0).then_some(ch as char))
.collect::<String>(),
)
.unwrap_or_default(),
)
}
compression
},
};
if super::MAGIC != footer.magic {
return Err(super::Error::Magic(footer.magic));
}
if version != footer.version {
return Err(super::Error::Version {
used: version,
version: footer.version,
});
}
Ok(footer)
}
}