First cut of a rust delta implementation

This commit is contained in:
Caelan Sayler
2025-05-13 15:57:03 +01:00
committed by Caelan
parent a1a740b10a
commit d71acd5a20
11 changed files with 494 additions and 88 deletions

213
Cargo.lock generated
View File

@@ -8,6 +8,17 @@ version = "2.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627" checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627"
[[package]]
name = "aes"
version = "0.8.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0"
dependencies = [
"cfg-if 1.0.0",
"cipher",
"cpufeatures",
]
[[package]] [[package]]
name = "aho-corasick" name = "aho-corasick"
version = "1.1.3" version = "1.1.3"
@@ -310,6 +321,25 @@ version = "1.10.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f61dac84819c6588b558454b194026eb1f09c293b9036ae9b159e74e73ab6cf9" checksum = "f61dac84819c6588b558454b194026eb1f09c293b9036ae9b159e74e73ab6cf9"
[[package]]
name = "bzip2"
version = "0.5.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "49ecfb22d906f800d4fe833b6282cf4dc1c298f5057ca0b5445e5c209735ca47"
dependencies = [
"bzip2-sys",
]
[[package]]
name = "bzip2-sys"
version = "0.1.13+1.0.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "225bff33b2141874fe80d71e07d6eec4f85c5c216453dd96388240f96e1acc14"
dependencies = [
"cc",
"pkg-config",
]
[[package]] [[package]]
name = "cbindgen" name = "cbindgen"
version = "0.28.0" version = "0.28.0"
@@ -366,6 +396,16 @@ dependencies = [
"windows-link", "windows-link",
] ]
[[package]]
name = "cipher"
version = "0.4.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad"
dependencies = [
"crypto-common",
"inout",
]
[[package]] [[package]]
name = "clap" name = "clap"
version = "4.5.38" version = "4.5.38"
@@ -427,6 +467,12 @@ dependencies = [
"crossbeam-utils", "crossbeam-utils",
] ]
[[package]]
name = "constant_time_eq"
version = "0.3.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6"
[[package]] [[package]]
name = "core-foundation" name = "core-foundation"
version = "0.10.0" version = "0.10.0"
@@ -452,6 +498,21 @@ dependencies = [
"libc", "libc",
] ]
[[package]]
name = "crc"
version = "3.3.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9710d3b3739c2e349eb44fe848ad0b7c8cb1e42bd87ee49371df2f7acaf3e675"
dependencies = [
"crc-catalog",
]
[[package]]
name = "crc-catalog"
version = "2.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "19d374276b40fb8bbdee95aef7c7fa6b5316ec764510eb64b8dd0e2ed0d7e7f5"
[[package]] [[package]]
name = "crc32fast" name = "crc32fast"
version = "1.4.2" version = "1.4.2"
@@ -521,6 +582,12 @@ dependencies = [
"syn 1.0.109", "syn 1.0.109",
] ]
[[package]]
name = "deflate64"
version = "0.1.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "da692b8d1080ea3045efaab14434d40468c3d8657e42abddfffca87b428f4c1b"
[[package]] [[package]]
name = "deranged" name = "deranged"
version = "0.4.1" version = "0.4.1"
@@ -576,6 +643,7 @@ checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
dependencies = [ dependencies = [
"block-buffer", "block-buffer",
"crypto-common", "crypto-common",
"subtle",
] ]
[[package]] [[package]]
@@ -634,7 +702,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "33d852cb9b869c2a9b3df2f71a3074817f01e1844f839a144f5fcef059a4eb5d" checksum = "33d852cb9b869c2a9b3df2f71a3074817f01e1844f839a144f5fcef059a4eb5d"
dependencies = [ dependencies = [
"libc", "libc",
"windows-sys 0.52.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@@ -796,8 +864,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8" checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8"
dependencies = [ dependencies = [
"cfg-if 1.0.0", "cfg-if 1.0.0",
"js-sys",
"libc", "libc",
"wasi 0.13.3+wasi-0.2.2", "wasi 0.13.3+wasi-0.2.2",
"wasm-bindgen",
"windows-targets 0.52.6", "windows-targets 0.52.6",
] ]
@@ -853,6 +923,15 @@ version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fbf6a919d6cf397374f7dfeeea91d974c7c0a7221d0d0f4f20d859d329e53fcc" checksum = "fbf6a919d6cf397374f7dfeeea91d974c7c0a7221d0d0f4f20d859d329e53fcc"
[[package]]
name = "hmac"
version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e"
dependencies = [
"digest",
]
[[package]] [[package]]
name = "http" name = "http"
version = "1.2.0" version = "1.2.0"
@@ -1064,6 +1143,15 @@ dependencies = [
"hashbrown", "hashbrown",
] ]
[[package]]
name = "inout"
version = "0.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "879f10e63c20629ecabbb64a8010319738c66a5cd0c29b02d63d272b03751d01"
dependencies = [
"generic-array",
]
[[package]] [[package]]
name = "is_terminal_polyfill" name = "is_terminal_polyfill"
version = "1.70.1" version = "1.70.1"
@@ -1133,7 +1221,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6a793df0d7afeac54f95b471d3af7f0d4fb975699f972341a4b76988d49cdf0c" checksum = "6a793df0d7afeac54f95b471d3af7f0d4fb975699f972341a4b76988d49cdf0c"
dependencies = [ dependencies = [
"cfg-if 1.0.0", "cfg-if 1.0.0",
"windows-targets 0.48.5", "windows-targets 0.52.6",
] ]
[[package]] [[package]]
@@ -1200,6 +1288,27 @@ dependencies = [
"log", "log",
] ]
[[package]]
name = "lzma-rs"
version = "0.3.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "297e814c836ae64db86b36cf2a557ba54368d03f6afcd7d947c266692f71115e"
dependencies = [
"byteorder",
"crc",
]
[[package]]
name = "lzma-sys"
version = "0.1.20"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5fda04ab3764e6cde78b9974eec4f779acaba7c4e84b36eca3cf77c581b85d27"
dependencies = [
"cc",
"libc",
"pkg-config",
]
[[package]] [[package]]
name = "memchr" name = "memchr"
version = "2.7.4" version = "2.7.4"
@@ -1351,6 +1460,16 @@ version = "2.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba" checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba"
[[package]]
name = "pbkdf2"
version = "0.12.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f8ed6a7761f76e3b9f92dfb0a60a6a6477c61024b775147ff0973a02653abaf2"
dependencies = [
"digest",
"hmac",
]
[[package]] [[package]]
name = "percent-encoding" name = "percent-encoding"
version = "2.3.1" version = "2.3.1"
@@ -1382,9 +1501,9 @@ dependencies = [
[[package]] [[package]]
name = "pkg-config" name = "pkg-config"
version = "0.3.31" version = "0.3.32"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2" checksum = "7edddbd0b52d732b21ad9a5fab5c704c14cd949e5e9a1ec5929a24fded1b904c"
[[package]] [[package]]
name = "png" name = "png"
@@ -1595,7 +1714,7 @@ dependencies = [
"errno", "errno",
"libc", "libc",
"linux-raw-sys 0.4.15", "linux-raw-sys 0.4.15",
"windows-sys 0.52.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@@ -1608,7 +1727,7 @@ dependencies = [
"errno", "errno",
"libc", "libc",
"linux-raw-sys 0.9.2", "linux-raw-sys 0.9.2",
"windows-sys 0.52.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@@ -1841,6 +1960,15 @@ dependencies = [
"syn 2.0.98", "syn 2.0.98",
] ]
[[package]]
name = "substring"
version = "1.4.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "42ee6433ecef213b2e72f587ef64a2f5943e7cd16fbd82dbe8bc07486c534c86"
dependencies = [
"autocfg",
]
[[package]] [[package]]
name = "subtle" name = "subtle"
version = "2.6.1" version = "2.6.1"
@@ -1901,7 +2029,7 @@ dependencies = [
"getrandom 0.3.1", "getrandom 0.3.1",
"once_cell", "once_cell",
"rustix 1.0.1", "rustix 1.0.1",
"windows-sys 0.52.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@@ -2232,7 +2360,6 @@ dependencies = [
"windows", "windows",
"xml", "xml",
"zip", "zip",
"zstd",
] ]
[[package]] [[package]]
@@ -2273,15 +2400,20 @@ dependencies = [
"simplelog", "simplelog",
"strsim 0.11.1", "strsim 0.11.1",
"strum", "strum",
"substring",
"tempfile", "tempfile",
"time 0.3.41", "time 0.3.41",
"velopack", "velopack",
"wait-timeout", "wait-timeout",
"waitpid-any", "waitpid-any",
"walkdir",
"webview2-com-sys", "webview2-com-sys",
"windows", "windows",
"winres", "winres",
"winsafe", "winsafe",
"zip",
"zip-extensions",
"zstd",
] ]
[[package]] [[package]]
@@ -2338,6 +2470,16 @@ dependencies = [
"windows-sys 0.59.0", "windows-sys 0.59.0",
] ]
[[package]]
name = "walkdir"
version = "2.5.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b"
dependencies = [
"same-file",
"winapi-util",
]
[[package]] [[package]]
name = "wasi" name = "wasi"
version = "0.10.0+wasi-snapshot-preview1" version = "0.10.0+wasi-snapshot-preview1"
@@ -2500,7 +2642,7 @@ version = "0.1.9"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
dependencies = [ dependencies = [
"windows-sys 0.48.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@@ -2827,6 +2969,15 @@ version = "0.8.25"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c5b940ebc25896e71dd073bad2dbaa2abfe97b0a391415e22ad1326d9c54e3c4" checksum = "c5b940ebc25896e71dd073bad2dbaa2abfe97b0a391415e22ad1326d9c54e3c4"
[[package]]
name = "xz2"
version = "0.1.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "388c44dc09d76f1536602ead6d325eb532f5c122f17782bd57fb47baeeb767e2"
dependencies = [
"lzma-sys",
]
[[package]] [[package]]
name = "yansi" name = "yansi"
version = "1.0.1" version = "1.0.1"
@@ -2924,6 +3075,20 @@ name = "zeroize"
version = "1.8.1" version = "1.8.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde"
dependencies = [
"zeroize_derive",
]
[[package]]
name = "zeroize_derive"
version = "1.4.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69"
dependencies = [
"proc-macro2",
"quote",
"syn 2.0.98",
]
[[package]] [[package]]
name = "zerovec" name = "zerovec"
@@ -2953,13 +3118,35 @@ version = "2.6.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1dcb24d0152526ae49b9b96c1dcf71850ca1e0b882e4e28ed898a93c41334744" checksum = "1dcb24d0152526ae49b9b96c1dcf71850ca1e0b882e4e28ed898a93c41334744"
dependencies = [ dependencies = [
"aes",
"arbitrary", "arbitrary",
"bzip2",
"constant_time_eq",
"crc32fast", "crc32fast",
"crossbeam-utils", "crossbeam-utils",
"deflate64",
"flate2", "flate2",
"getrandom 0.3.1",
"hmac",
"indexmap", "indexmap",
"lzma-rs",
"memchr", "memchr",
"pbkdf2",
"sha1",
"time 0.3.41",
"xz2",
"zeroize",
"zopfli", "zopfli",
"zstd",
]
[[package]]
name = "zip-extensions"
version = "0.8.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "79cdbf826e5a6eec81fc5a0d33cd7c09c31fd8f9918f15434f74c42d39ef337a"
dependencies = [
"zip",
] ]
[[package]] [[package]]
@@ -2987,18 +3174,18 @@ dependencies = [
[[package]] [[package]]
name = "zstd-safe" name = "zstd-safe"
version = "7.2.3" version = "7.2.4"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f3051792fbdc2e1e143244dc28c60f73d8470e93f3f9cbd0ead44da5ed802722" checksum = "8f49c4d5f0abb602a93fb8736af2a4f4dd9512e36f7f570d66e65ff867ed3b9d"
dependencies = [ dependencies = [
"zstd-sys", "zstd-sys",
] ]
[[package]] [[package]]
name = "zstd-sys" name = "zstd-sys"
version = "2.0.14+zstd.1.5.7" version = "2.0.15+zstd.1.5.7"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8fb060d4926e4ac3a3ad15d864e99ceb5f343c6b34f5bd6d81ae6ed417311be5" checksum = "eb81183ddd97d0c74cedf1d50d85c8d08c1b8b68ee863bdee9e706eedba1a237"
dependencies = [ dependencies = [
"cc", "cc",
"pkg-config", "pkg-config",

View File

@@ -34,7 +34,8 @@ derivative = "2.2"
glob = "0.3" glob = "0.3"
serde = { version = "1.0", features = ["derive"] } serde = { version = "1.0", features = ["derive"] }
serde_json = { version = "1.0" } serde_json = { version = "1.0" }
zip = { version = "2.2", default-features = false, features = ["deflate"] } zip = { version = "2.6", default-features = false, features = ["deflate"] }
zip-extensions = "0.8.2"
thiserror = "2.0" thiserror = "2.0"
lazy_static = "1.5" lazy_static = "1.5"
regex = "1.10" regex = "1.10"
@@ -83,6 +84,8 @@ log-panics = "2.1.0"
core-foundation = "0.10" core-foundation = "0.10"
core-foundation-sys = "0.8" core-foundation-sys = "0.8"
uuid = { version = "1.13.1", features = ["v4", "fast-rng", "macro-diagnostics"] } uuid = { version = "1.13.1", features = ["v4", "fast-rng", "macro-diagnostics"] }
walkdir = "2.5"
substring = " 1.4"
# default to small, optimized workspace release binaries # default to small, optimized workspace release binaries
[profile.release] [profile.release]

View File

@@ -63,6 +63,12 @@ wait-timeout.workspace = true
pretty-bytes-rust.workspace = true pretty-bytes-rust.workspace = true
enum-flags.workspace = true enum-flags.workspace = true
log-panics.workspace = true log-panics.workspace = true
zstd.workspace = true
zip.workspace = true
zip-extensions.workspace = true
walkdir.workspace = true
sha1_smol.workspace = true
substring.workspace = true
[target.'cfg(target_os="linux")'.dependencies] [target.'cfg(target_os="linux")'.dependencies]
waitpid-any.workspace = true waitpid-any.workspace = true
@@ -101,6 +107,7 @@ windows = { workspace = true, features = [
"Win32_UI_Shell_Common", "Win32_UI_Shell_Common",
"Win32_UI_Shell_PropertiesSystem", "Win32_UI_Shell_PropertiesSystem",
"Win32_UI_WindowsAndMessaging", "Win32_UI_WindowsAndMessaging",
"Win32_System_ApplicationInstallationAndServicing",
"Win32_System_Kernel", "Win32_System_Kernel",
"Wdk", "Wdk",
"Wdk_System", "Wdk_System",
@@ -116,7 +123,6 @@ same-file.workspace = true
tempfile.workspace = true tempfile.workspace = true
ntest.workspace = true ntest.workspace = true
pretty_assertions.workspace = true pretty_assertions.workspace = true
sha1_smol.workspace = true
[build-dependencies] [build-dependencies]
semver.workspace = true semver.workspace = true

View File

@@ -4,6 +4,9 @@ pub use apply::*;
mod start; mod start;
pub use start::*; pub use start::*;
mod patch;
pub use patch::*;
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]
mod apply_linux_impl; mod apply_linux_impl;
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]

View File

@@ -0,0 +1,206 @@
use anyhow::{anyhow, bail, Result};
use std::os::windows::fs::MetadataExt;
use std::{
collections::HashSet,
fs, io,
path::{Path, PathBuf},
};
use walkdir::WalkDir;
use zip::{write::SimpleFileOptions, CompressionMethod};
use zip_extensions::{zip_create_from_directory_with_options, zip_extract};
pub fn zstd_patch_single<P1: AsRef<Path>, P2: AsRef<Path>, P3: AsRef<Path>>(old_file: P1, patch_file: P2, output_file: P3) -> Result<()> {
let old_file = old_file.as_ref();
let patch_file = patch_file.as_ref();
let output_file = output_file.as_ref();
if !old_file.exists() {
bail!("Old file does not exist: {}", old_file.to_string_lossy());
}
if !patch_file.exists() {
bail!("Patch file does not exist: {}", patch_file.to_string_lossy());
}
let dict = fs::read(old_file)?;
// info!("Loading Dictionary (Size: {})", dict.len());
let patch = fs::OpenOptions::new().read(true).open(patch_file)?;
let patch_reader = io::BufReader::new(patch);
let mut decoder = zstd::Decoder::with_dictionary(patch_reader, &dict)?;
let window_log = fio_highbit64(dict.len() as u64) + 1;
if window_log >= 27 {
info!("Large File detected. Overriding windowLog to {}", window_log);
decoder.window_log_max(window_log)?;
}
// info!("Decoder loaded. Beginning patch...");
let mut output = fs::OpenOptions::new().write(true).create(true).truncate(true).open(output_file)?;
io::copy(&mut decoder, &mut output)?;
// info!("Patch applied successfully.");
Ok(())
}
fn fio_highbit64(v: u64) -> u32 {
let mut count: u32 = 0;
let mut v = v;
v >>= 1;
while v > 0 {
v >>= 1;
count += 1;
}
return count;
}
pub fn delta<P1: AsRef<Path>, P2: AsRef<Path>, P3: AsRef<Path>>(
old_file: P1,
delta_files: Vec<&PathBuf>,
temp_dir: P2,
output_file: P3,
) -> Result<()> {
let old_file = old_file.as_ref().to_path_buf();
let temp_dir = temp_dir.as_ref().to_path_buf();
let output_file = output_file.as_ref().to_path_buf();
if !old_file.exists() {
bail!("Old file does not exist: {}", old_file.to_string_lossy());
}
if delta_files.is_empty() {
bail!("No delta files provided.");
}
for delta_file in &delta_files {
if !delta_file.exists() {
bail!("Delta file does not exist: {}", delta_file.to_string_lossy());
}
}
let time = simple_stopwatch::Stopwatch::start_new();
info!("Extracting base package for delta patching: {}", temp_dir.to_string_lossy());
let work_dir = temp_dir.join("_work");
fs::create_dir_all(&work_dir)?;
zip_extract(&old_file, &work_dir)?;
info!("Base package extracted. {} delta packages to apply.", delta_files.len());
for (i, delta_file) in delta_files.iter().enumerate() {
info!("{}: extracting apply delta patch: {}", i, delta_file.to_string_lossy());
let delta_dir = temp_dir.join(format!("delta_{}", i));
fs::create_dir_all(&delta_dir)?;
zip_extract(delta_file, &delta_dir)?;
let delta_relative_paths: Vec<PathBuf> = WalkDir::new(&delta_dir)
.follow_links(false)
.into_iter()
.filter_map(|entry| entry.ok())
.filter(|entry| entry.file_type().is_file())
.map(|entry| entry.path().strip_prefix(&delta_dir).map(|p| p.to_path_buf()))
.filter_map(|entry| entry.ok())
.collect();
let mut visited_paths = HashSet::new();
// apply all the zsdiff patches for files which exist in both the delta and the base package
for relative_path in &delta_relative_paths {
if relative_path.starts_with("lib") {
let file_name = relative_path.file_name().ok_or(anyhow!("Failed to get file name"))?;
let file_name_str = file_name.to_string_lossy();
if file_name_str.ends_with(".zsdiff") || file_name_str.ends_with(".diff") || file_name_str.ends_with(".bsdiff") {
// this is a zsdiff patch, we need to apply it to the old file
let file_without_extension = relative_path.with_extension("");
// let shasum_path = delta_dir.join(relative_path).with_extension("shasum");
let old_file_path = work_dir.join(&file_without_extension);
let patch_file_path = delta_dir.join(&relative_path);
let output_file_path = delta_dir.join(&file_without_extension);
visited_paths.insert(file_without_extension);
if fs::metadata(&patch_file_path)?.file_size() == 0 {
// file has not changed, so we can continue.
continue;
}
if file_name_str.ends_with(".zsdiff") {
info!("{}: applying zsdiff patch: {:?}", i, relative_path);
zstd_patch_single(&old_file_path, &patch_file_path, &output_file_path)?;
} else {
bail!("Unsupported patch format: {:?}", relative_path);
}
fs::rename(&output_file_path, &old_file_path)?;
} else if file_name_str.ends_with(".shasum") {
// skip shasum files
} else {
// if this file is inside the lib folder without a known extension, it is a new file
let file_path = delta_dir.join(relative_path);
let dest_path = work_dir.join(relative_path);
info!("{}: new file: {:?}", i, relative_path);
fs::copy(&file_path, &dest_path)?;
visited_paths.insert(relative_path.clone());
}
} else {
// if this file is not inside the lib folder, we always copy it over
let file_path = delta_dir.join(relative_path);
let dest_path = work_dir.join(relative_path);
info!("{}: copying metadata file: {:?}", i, relative_path);
fs::copy(&file_path, &dest_path)?;
visited_paths.insert(relative_path.clone());
}
}
// anything in the work dir which was not visited is an old / deleted file and should be removed
let workdir_relative_paths: Vec<PathBuf> = WalkDir::new(&work_dir)
.follow_links(false)
.into_iter()
.filter_map(|entry| entry.ok())
.filter(|entry| entry.file_type().is_file())
.map(|entry| entry.path().strip_prefix(&work_dir).map(|p| p.to_path_buf()))
.filter_map(|entry| entry.ok())
.collect();
for relative_path in &workdir_relative_paths {
if !visited_paths.contains(relative_path) {
let file_to_delete = work_dir.join(relative_path);
info!("{}: deleting old/removed file: {:?}", i, relative_path);
let _ = fs::remove_file(file_to_delete); // soft error
}
}
}
info!("All delta patches applied. Asembling output package at: {}", output_file.to_string_lossy());
// NOTE: zstd is not supported by older versions of Squirrel/Velopack, but
// it's assumed if someone is using this code, they are using a recent enough version
let options = SimpleFileOptions::default().compression_method(CompressionMethod::Zstd);
zip_create_from_directory_with_options(&output_file, &work_dir, |_| options)?;
info!("Successfully applied {} delta patches in {}s.", delta_files.len(), time.s());
Ok(())
}
// NOTE: this is some code to do checksum verification, but it is not being used
// by the current implementation because zstd patching already has checksum verification
//
// let actual_checksum = get_sha1(&output_file_path);
// let expected_checksum = load_release_entry_shasum(&shasum_path)?;
//
// if !actual_checksum.eq_ignore_ascii_case(&expected_checksum) {
// bail!("Checksum mismatch for: {:?}. Expected: {}, Actual: {}", relative_path, expected_checksum, actual_checksum);
// }
// fn load_release_entry_shasum(file: &PathBuf) -> Result<String> {
// let raw_text = fs::read_to_string(file)?.trim().to_string();
// let first_word = raw_text.splitn(2, ' ').next().unwrap();
// let cleaned = first_word.trim().trim_matches(|c: char| !c.is_ascii_hexdigit());
// Ok(cleaned.to_string())
// }
//
// fn get_sha1(file: &PathBuf) -> String {
// let file_bytes = fs::read(file).unwrap();
// let mut sha1 = sha1_smol::Sha1::new();
// sha1.update(&file_bytes);
// sha1.digest().to_string()
// }

View File

@@ -5,7 +5,7 @@
extern crate log; extern crate log;
use anyhow::{anyhow, bail, Result}; use anyhow::{anyhow, bail, Result};
use clap::{arg, value_parser, ArgMatches, Command}; use clap::{arg, value_parser, ArgAction, ArgMatches, Command};
use std::{env, path::PathBuf}; use std::{env, path::PathBuf};
use velopack::locator::{auto_locate_app_manifest, LocationContext}; use velopack::locator::{auto_locate_app_manifest, LocationContext};
use velopack::logging::*; use velopack::logging::*;
@@ -34,9 +34,9 @@ fn root_command() -> Command {
.long_flag_aliases(vec!["processStart", "processStartAndWait"]) .long_flag_aliases(vec!["processStart", "processStartAndWait"])
) )
.subcommand(Command::new("patch") .subcommand(Command::new("patch")
.about("Applies a Zstd patch file") .about("Applies a series of delta bundles to a base file")
.arg(arg!(--old <FILE> "Base / old file to apply the patch to").required(true).value_parser(value_parser!(PathBuf))) .arg(arg!(--old <FILE> "Base / old file to apply the patch to").required(true).value_parser(value_parser!(PathBuf)))
.arg(arg!(--patch <FILE> "The Zstd patch to apply to the old file").required(true).value_parser(value_parser!(PathBuf))) .arg(arg!(--delta <FILE> "The delta bundle to apply to the base package").required(true).action(ArgAction::Append).value_parser(value_parser!(PathBuf)))
.arg(arg!(--output <FILE> "The file to create with the patch applied").required(true).value_parser(value_parser!(PathBuf))) .arg(arg!(--output <FILE> "The file to create with the patch applied").required(true).value_parser(value_parser!(PathBuf)))
) )
.arg(arg!(--verbose "Print debug messages to console / log").global(true)) .arg(arg!(--verbose "Print debug messages to console / log").global(true))
@@ -180,19 +180,34 @@ fn main() -> Result<()> {
fn patch(matches: &ArgMatches) -> Result<()> { fn patch(matches: &ArgMatches) -> Result<()> {
let old_file = matches.get_one::<PathBuf>("old"); let old_file = matches.get_one::<PathBuf>("old");
let patch_file = matches.get_one::<PathBuf>("patch"); let deltas: Vec<&PathBuf> = matches.get_many::<PathBuf>("delta").unwrap_or_default().collect();
let output_file = matches.get_one::<PathBuf>("output"); let output_file = matches.get_one::<PathBuf>("output");
info!("Command: Patch"); info!("Command: Patch");
info!(" Old File: {:?}", old_file); info!(" Old File: {:?}", old_file);
info!(" Patch File: {:?}", patch_file); info!(" Delta Files: {:?}", deltas);
info!(" Output File: {:?}", output_file); info!(" Output File: {:?}", output_file);
if old_file.is_none() || patch_file.is_none() || output_file.is_none() { if old_file.is_none() || deltas.is_empty() || output_file.is_none() {
bail!("Missing required arguments. Please provide --old, --patch, and --output."); bail!("Missing required arguments. Please provide --old, --delta, and --output.");
} }
velopack::delta::zstd_patch_single(old_file.unwrap(), patch_file.unwrap(), output_file.unwrap())?; let temp_dir = match auto_locate_app_manifest(LocationContext::IAmUpdateExe) {
Ok(locator) => locator.get_temp_dir_rand16(),
Err(_) => {
let mut temp_dir = std::env::temp_dir();
let rand = shared::random_string(16);
temp_dir.push("velopack_".to_owned() + &rand);
temp_dir
}
};
let result = commands::delta(old_file.unwrap(), deltas, &temp_dir, output_file.unwrap());
let _ = remove_dir_all::remove_dir_all(temp_dir);
if let Err(e) = result {
bail!("Delta error: {}", e);
}
Ok(()) Ok(())
} }

View File

@@ -3,13 +3,14 @@
mod common; mod common;
use common::*; use common::*;
use std::{fs, path::Path, path::PathBuf}; use std::{fs, path::Path, path::PathBuf};
use std::hint::assert_unchecked;
use tempfile::tempdir; use tempfile::tempdir;
use velopack_bins::*; use velopack_bins::*;
#[cfg(target_os = "windows")]
use winsafe::{self as w, co};
use velopack::bundle::load_bundle_from_file; use velopack::bundle::load_bundle_from_file;
use velopack::locator::{auto_locate_app_manifest, LocationContext}; use velopack::locator::{auto_locate_app_manifest, LocationContext};
#[cfg(target_os = "windows")]
use winsafe::{self as w, co};
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
#[test] #[test]
@@ -119,13 +120,42 @@ pub fn test_patch_apply() {
let expected_sha1 = get_sha1(&new_file); let expected_sha1 = get_sha1(&new_file);
let tmp_file = Path::new("temp.patch").to_path_buf(); let tmp_file = Path::new("temp.patch").to_path_buf();
velopack::delta::zstd_patch_single(&old_file, &p1, &tmp_file).unwrap(); velopack_bins::commands::zstd_patch_single(&old_file, &p1, &tmp_file).unwrap();
let tmp_sha1 = get_sha1(&tmp_file); let tmp_sha1 = get_sha1(&tmp_file);
fs::remove_file(&tmp_file).unwrap(); fs::remove_file(&tmp_file).unwrap();
assert_eq!(expected_sha1, tmp_sha1); assert_eq!(expected_sha1, tmp_sha1);
velopack::delta::zstd_patch_single(&old_file, &p2, &tmp_file).unwrap(); velopack_bins::commands::zstd_patch_single(&old_file, &p2, &tmp_file).unwrap();
let tmp_sha1 = get_sha1(&tmp_file); let tmp_sha1 = get_sha1(&tmp_file);
fs::remove_file(&tmp_file).unwrap(); fs::remove_file(&tmp_file).unwrap();
assert_eq!(expected_sha1, tmp_sha1); assert_eq!(expected_sha1, tmp_sha1);
} }
#[test]
pub fn test_delta_apply_legacy() {
dialogs::set_silent(true);
let fixtures = find_fixtures();
let base = fixtures.join("Clowd-3.4.287-full.nupkg");
let d1 = fixtures.join("Clowd-3.4.288-delta-zstd.nupkg");
let d2 = fixtures.join("Clowd-3.4.291-delta-zstd.nupkg");
let d3 = fixtures.join("Clowd-3.4.292-delta-zstd.nupkg");
let d4 = fixtures.join("Clowd-3.4.293-delta-zstd.nupkg");
let deltas = vec![&d1, &d2, &d3, &d4];
let tmp_dir = tempdir().unwrap();
let temp_output = tmp_dir.path().join("Clowd-3.4.293-full.nupkg");
commands::delta(&base, deltas, tmp_dir.path(), &temp_output).unwrap();
let mut bundle = load_bundle_from_file(temp_output).unwrap();
let manifest = bundle.read_manifest().unwrap();
assert_eq!(manifest.id, "Clowd");
assert_eq!(manifest.version, semver::Version::parse("3.4.293").unwrap());
let extract_dir = tmp_dir.path().join("_extracted");
bundle.extract_lib_contents_to_path(&extract_dir, |_| {}).unwrap();
let extracted = extract_dir.join("Clowd.dll");
assert!(extracted.exists());
}

View File

@@ -14,14 +14,13 @@ edition.workspace = true
rust-version.workspace = true rust-version.workspace = true
[features] [features]
default = ["zstd"] default = []
delta = ["zstd"]
async = ["async-std"] async = ["async-std"]
typescript = ["ts-rs"] typescript = ["ts-rs"]
file-logging = ["log-panics", "simplelog", "file-rotate", "time"] file-logging = ["log-panics", "simplelog", "file-rotate", "time"]
[package.metadata.docs.rs] [package.metadata.docs.rs]
features = ["async", "delta"] features = ["async"]
[lib] [lib]
name = "velopack" name = "velopack"
@@ -54,9 +53,6 @@ uuid.workspace = true
# typescript # typescript
ts-rs = { workspace = true, optional = true } ts-rs = { workspace = true, optional = true }
# delta packages
zstd = { workspace = true, optional = true }
# async # async
async-std = { workspace = true, optional = true } async-std = { workspace = true, optional = true }

View File

@@ -1,48 +0,0 @@
use std::{fs, io, path::Path};
use crate::Error;
/// Applies a zstd patch to a single file by loading the patch as a dictionary.
pub fn zstd_patch_single<P1: AsRef<Path>, P2: AsRef<Path>, P3: AsRef<Path>>(old_file: P1, patch_file: P2, output_file: P3) -> Result<(), Error> {
let old_file = old_file.as_ref();
let patch_file = patch_file.as_ref();
let output_file = output_file.as_ref();
if !old_file.exists() {
return Err(Error::FileNotFound(old_file.to_string_lossy().to_string()));
}
if !patch_file.exists() {
return Err(Error::FileNotFound(patch_file.to_string_lossy().to_string()));
}
let dict = fs::read(old_file)?;
info!("Loading Dictionary (Size: {})", dict.len());
let patch = fs::OpenOptions::new().read(true).open(patch_file)?;
let patch_reader = io::BufReader::new(patch);
let mut decoder = zstd::Decoder::with_dictionary(patch_reader, &dict)?;
let window_log = fio_highbit64(dict.len() as u64) + 1;
if window_log >= 27 {
info!("Large File detected. Overriding windowLog to {}", window_log);
decoder.window_log_max(window_log)?;
}
info!("Decoder loaded. Beginning patch...");
let mut output = fs::OpenOptions::new().write(true).create(true).truncate(true).open(output_file)?;
io::copy(&mut decoder, &mut output)?;
info!("Patch applied successfully.");
Ok(())
}
fn fio_highbit64(v: u64) -> u32 {
let mut count: u32 = 0;
let mut v = v;
v >>= 1;
while v > 0 {
v >>= 1;
count += 1;
}
return count;
}

View File

@@ -97,9 +97,6 @@ pub mod locator;
/// Sources contains abstractions for custom update sources (eg. url, local file, github releases, etc). /// Sources contains abstractions for custom update sources (eg. url, local file, github releases, etc).
pub mod sources; pub mod sources;
/// Functions to patch files and reconstruct Velopack delta packages.
pub mod delta;
/// Acquire and manage file-system based lock files. /// Acquire and manage file-system based lock files.
pub mod lockfile; pub mod lockfile;

View File

@@ -76,7 +76,13 @@ pub fn default_logfile_path<L: TryInto<VelopackLocator>>(locator: L) -> PathBuf
/// It can only be called once per process, and should be called early in the process lifecycle. /// It can only be called once per process, and should be called early in the process lifecycle.
/// Future calls to this function will fail. /// Future calls to this function will fail.
#[cfg(feature = "file-logging")] #[cfg(feature = "file-logging")]
pub fn init_logging(process_name: &str, file: Option<&PathBuf>, console: bool, verbose: bool, custom_log_cb: Option<Box<dyn SharedLogger>>) { pub fn init_logging(
process_name: &str,
file: Option<&PathBuf>,
console: bool,
verbose: bool,
custom_log_cb: Option<Box<dyn SharedLogger>>,
) {
let mut loggers: Vec<Box<dyn SharedLogger>> = Vec::new(); let mut loggers: Vec<Box<dyn SharedLogger>> = Vec::new();
if let Some(cb) = custom_log_cb { if let Some(cb) = custom_log_cb {
loggers.push(cb); loggers.push(cb);
@@ -105,6 +111,11 @@ pub fn init_logging(process_name: &str, file: Option<&PathBuf>, console: bool, v
} }
} }
/// Initialize a Trace / Console logger for the current process.
pub fn trace_logger() {
TermLogger::init(LevelFilter::Trace, get_config(None), TerminalMode::Mixed, ColorChoice::Never).unwrap();
}
#[cfg(feature = "file-logging")] #[cfg(feature = "file-logging")]
fn get_config(process_name: Option<&str>) -> Config { fn get_config(process_name: Option<&str>) -> Config {
let mut c = ConfigBuilder::default(); let mut c = ConfigBuilder::default();