From 17ddf22e73a536dba3e22c6f7533a14992d0b47a Mon Sep 17 00:00:00 2001 From: Truman Kilen Date: Fri, 10 Feb 2023 23:37:45 -0600 Subject: [PATCH] Allow converting PakReader into PakWriter to append files to an existing pak --- repak/src/pak.rs | 15 ++++++++++++++ repak/tests/test.rs | 48 ++++++++++++++++++++++++++++++++++++++++++--- 2 files changed, 60 insertions(+), 3 deletions(-) diff --git a/repak/src/pak.rs b/repak/src/pak.rs index c8d4ca2..8c20db6 100644 --- a/repak/src/pak.rs +++ b/repak/src/pak.rs @@ -22,6 +22,7 @@ pub struct PakWriter { pub struct Pak { version: Version, mount_point: String, + index_offset: Option, index: Index, compression: Vec, } @@ -31,6 +32,7 @@ impl Pak { Pak { version, mount_point, + index_offset: None, index: Index::new(path_hash_seed), compression: vec![], } @@ -121,6 +123,18 @@ impl PakReader { pub fn files(&self) -> Vec { self.pak.index.entries().keys().cloned().collect() } + + pub fn into_pakwriter( + self, + mut writer: W, + ) -> Result, super::Error> { + writer.seek(io::SeekFrom::Start(self.pak.index_offset.unwrap()))?; + Ok(PakWriter { + pak: self.pak, + key: self.key, + writer, + }) + } } impl PakWriter { @@ -311,6 +325,7 @@ impl Pak { Ok(Pak { version, mount_point, + index_offset: Some(footer.index_offset), index, compression: footer.compression, }) diff --git a/repak/tests/test.rs b/repak/tests/test.rs index 5335c19..5cddc43 100644 --- a/repak/tests/test.rs +++ b/repak/tests/test.rs @@ -82,7 +82,7 @@ mod test { static AES_KEY: &str = "lNJbw660IOC+kU7cnVQ1oeqrXyhk4J6UAZrCBbcnp94="; -fn test_read(version: repak::Version, bytes: &[u8]) { +fn test_read(version: repak::Version, _file_name: &str, bytes: &[u8]) { use aes::cipher::KeyInit; use base64::{engine::general_purpose, Engine as _}; let key = general_purpose::STANDARD @@ -148,7 +148,7 @@ fn test_read(version: repak::Version, bytes: &[u8]) { } } -fn test_write(_version: repak::Version, bytes: &[u8]) { +fn test_write(_version: repak::Version, _file_name: &str, bytes: &[u8]) { use aes::cipher::KeyInit; use base64::{engine::general_purpose, Engine as _}; let key = general_purpose::STANDARD @@ -183,6 +183,29 @@ fn test_write(_version: repak::Version, bytes: &[u8]) { ); } +fn test_rewrite_index(_version: repak::Version, _file_name: &str, bytes: &[u8]) { + use aes::cipher::KeyInit; + use base64::{engine::general_purpose, Engine as _}; + let key = general_purpose::STANDARD + .decode(AES_KEY) + .as_ref() + .map_err(|_| repak::Error::Base64) + .and_then(|bytes| aes::Aes256::new_from_slice(bytes).map_err(|_| repak::Error::Aes)) + .unwrap(); + + let mut buf = std::io::Cursor::new(bytes.to_vec()); + let pak_reader = repak::PakReader::new_any(&mut buf, Some(key)).unwrap(); + + let rewrite = pak_reader + .into_pakwriter(buf) + .unwrap() + .write_index() + .unwrap() + .into_inner(); + + assert_eq!(bytes, rewrite); +} + macro_rules! matrix_test { ( $name:literal, ($($version:literal $exp_version:expr),* $(,)?), $compress:tt, $encrypt:tt, $encryptindex:tt, $body:tt ) => { $( matrix_test_compress!($name, $version, $exp_version, $compress, $encrypt, $encryptindex, $body); )* @@ -212,7 +235,10 @@ macro_rules! matrix_test_body { paste! { #[test] fn [< test_ $name _version_ $version $compress $encrypt $encryptindex >]() { - $body($exp_version, include_bytes!(concat!("packs/pack_", $version, $compress, $encrypt, $encryptindex, ".pak"))); + $body( + $exp_version, + concat!("pack_", $version, $compress, $encrypt, $encryptindex, ".pak"), + include_bytes!(concat!("packs/pack_", $version, $compress, $encrypt, $encryptindex, ".pak"))); } } }; @@ -249,3 +275,19 @@ matrix_test!( ("", /*"_encryptindex"*/), test_write ); + +matrix_test!( + "rewrite_index", + ( + "v5" repak::Version::V5, + "v7" repak::Version::V7, + "v8a" repak::Version::V8A, + "v8b" repak::Version::V8B, + "v9" repak::Version::V9, + "v11" repak::Version::V11, + ), + ("", "_compress"), + ("", "_encrypt"), + ("", /*"_encryptindex"*/), + test_rewrite_index +);