use anyhow::Context; use std::io::prelude::*; use zip_next::{result::ZipError, write::SimpleFileOptions}; use std::fs::File; use std::path::Path; use walkdir::{DirEntry, WalkDir}; fn main() { std::process::exit(real_main()); } const METHOD_STORED: Option = Some(zip_next::CompressionMethod::Stored); #[cfg(any( feature = "deflate", feature = "deflate-miniz", feature = "deflate-zlib", feature = "deflate-zlib-ng" ))] const METHOD_DEFLATED: Option = Some(zip_next::CompressionMethod::Deflated); #[cfg(not(any( feature = "deflate", feature = "deflate-miniz", feature = "deflate-zlib", feature = "deflate-zlib-ng", feature = "deflate-zopfli" )))] const METHOD_DEFLATED: Option = None; #[cfg(feature = "bzip2")] const METHOD_BZIP2: Option = Some(zip_next::CompressionMethod::Bzip2); #[cfg(not(feature = "bzip2"))] const METHOD_BZIP2: Option = None; #[cfg(feature = "zstd")] const METHOD_ZSTD: Option = Some(zip_next::CompressionMethod::Zstd); #[cfg(not(feature = "zstd"))] const METHOD_ZSTD: Option = None; fn real_main() -> i32 { let args: Vec<_> = std::env::args().collect(); if args.len() < 3 { println!( "Usage: {} ", args[0] ); return 1; } let src_dir = &*args[1]; let dst_file = &*args[2]; for &method in [METHOD_STORED, METHOD_DEFLATED, METHOD_BZIP2, METHOD_ZSTD].iter() { if method.is_none() { continue; } match doit(src_dir, dst_file, method.unwrap()) { Ok(_) => println!("done: {src_dir} written to {dst_file}"), Err(e) => eprintln!("Error: {e:?}"), } } 0 } fn zip_dir( it: &mut dyn Iterator, prefix: &str, writer: T, method: zip_next::CompressionMethod, ) -> anyhow::Result<()> where T: Write + Seek, { let mut zip = zip_next::ZipWriter::new(writer); let options = SimpleFileOptions::default() .compression_method(method) .unix_permissions(0o755); let prefix = Path::new(prefix); let mut buffer = Vec::new(); for entry in it { let path = entry.path(); let name = path.strip_prefix(prefix).unwrap(); let path_as_string = name .to_str() .map(str::to_owned) .with_context(|| format!("{name:?} Is a Non UTF-8 Path"))?; // Write file or directory explicitly // Some unzip tools unzip files with directory paths correctly, some do not! if path.is_file() { println!("adding file {path:?} as {name:?} ..."); zip.start_file(path_as_string, options)?; let mut f = File::open(path)?; f.read_to_end(&mut buffer)?; zip.write_all(&buffer)?; buffer.clear(); } else if !name.as_os_str().is_empty() { // Only if not root! Avoids path spec / warning // and mapname conversion failed error on unzip println!("adding dir {path_as_string:?} as {name:?} ..."); zip.add_directory(path_as_string, options)?; } } zip.finish()?; Ok(()) } fn doit(src_dir: &str, dst_file: &str, method: zip_next::CompressionMethod) -> anyhow::Result<()> { if !Path::new(src_dir).is_dir() { return Err(ZipError::FileNotFound.into()); } let path = Path::new(dst_file); let file = File::create(path).unwrap(); let walkdir = WalkDir::new(src_dir); let it = walkdir.into_iter(); zip_dir(&mut it.filter_map(|e| e.ok()), src_dir, file, method)?; Ok(()) }