repak/src/entry.rs

101 lines
3.3 KiB
Rust
Raw Normal View History

2023-01-13 15:23:51 +00:00
use super::{Compression, ReadExt, Version};
use byteorder::{ReadBytesExt, LE};
2023-01-11 13:52:40 +00:00
use std::io;
2023-01-11 13:52:40 +00:00
#[derive(Debug)]
pub struct Block {
pub offset: u64,
/// size of the compressed block
pub size: u64,
}
impl Block {
pub fn new<R: io::Read>(reader: &mut R) -> Result<Self, super::Error> {
Ok(Self {
offset: reader.read_u64::<LE>()?,
size: reader.read_u64::<LE>()?,
})
}
}
#[derive(Debug)]
pub struct Entry {
pub offset: u64,
pub compressed: u64,
pub uncompressed: u64,
pub compression: Compression,
pub timestamp: Option<u64>,
pub hash: [u8; 20],
pub compression_blocks: Option<Vec<Block>>,
2023-01-10 22:29:20 +00:00
pub encrypted: bool,
pub block_uncompressed: Option<u32>,
}
impl Entry {
2023-01-11 13:52:40 +00:00
pub fn new<R: io::Read>(reader: &mut R, version: super::Version) -> Result<Self, super::Error> {
2023-01-13 12:38:30 +00:00
// since i need the compression flags, i have to store these as variables which is mildly annoying
let offset = reader.read_u64::<LE>()?;
let compressed = reader.read_u64::<LE>()?;
let uncompressed = reader.read_u64::<LE>()?;
let compression = match reader.read_u32::<LE>()? {
2023-01-10 22:29:20 +00:00
0x01 | 0x10 | 0x20 => Compression::Zlib,
_ => Compression::None,
};
Ok(Self {
offset,
compressed,
uncompressed,
compression,
2023-01-10 22:29:20 +00:00
timestamp: match version == Version::Initial {
true => Some(reader.read_u64::<LE>()?),
false => None,
},
hash: reader.read_guid()?,
2023-01-10 22:29:20 +00:00
compression_blocks: match version >= Version::CompressionEncryption
&& compression != Compression::None
2023-01-10 22:29:20 +00:00
{
true => Some(reader.read_array(Block::new)?),
false => None,
},
encrypted: version >= Version::CompressionEncryption && reader.read_bool()?,
block_uncompressed: match version >= Version::CompressionEncryption {
true => Some(reader.read_u32::<LE>()?),
false => None,
},
})
}
2023-01-11 13:52:40 +00:00
pub fn read<R: io::Read + io::Seek>(
2023-01-13 12:38:30 +00:00
&self,
2023-01-11 13:52:40 +00:00
reader: &mut R,
version: super::Version,
2023-01-13 12:38:30 +00:00
key: Option<&aes::Aes256Dec>,
2023-01-11 13:52:40 +00:00
) -> Result<Vec<u8>, super::Error> {
2023-01-13 15:23:51 +00:00
let mut buf = io::BufWriter::new(Vec::with_capacity(self.uncompressed as usize));
reader.seek(io::SeekFrom::Start(self.offset))?;
let mut data = reader.read_len(match self.encrypted {
// add alignment (aes block size: 16) then zero out alignment bits
true => (self.compressed + 15) & !17,
false => self.compressed,
} as usize)?;
if self.encrypted {
if let Some(key) = key {
use aes::cipher::BlockDecrypt;
for block in data.chunks_mut(16) {
key.decrypt_block(aes::Block::from_mut_slice(block))
}
data.truncate(self.compressed as usize);
}
}
use io::Write;
match self.compression {
Compression::None => buf.write_all(&data)?,
Compression::Zlib => todo!(),
Compression::Gzip => todo!(),
Compression::Oodle => todo!(),
}
buf.flush()?;
2023-01-11 13:52:40 +00:00
Ok(buf.into_inner()?)
}
}