totebag 0.8.14

An API for extracting/archiving files and directories in multiple formats.
Documentation
#[cfg(target_os = "windows")]
use crate::archiver::os;

#[cfg(any(target_os = "linux", target_os = "macos"))]
use crate::archiver::os;

use std::fs::File;
use std::io::BufReader;
use std::path::{Path, PathBuf};
use zip::ZipWriter;

use crate::archiver::{ArchiveEntry, ToteArchiver};
use crate::{Result, Error};

/// ZIP format archiver implementation.
///
/// This archiver creates ZIP archive files.
pub(super) struct Archiver {}

impl Archiver {
    pub fn new() -> Self {
        Self {}
    }
    fn process_file(
        &self,
        zw: &mut ZipWriter<File>,
        target: &Path,
        dest_path: PathBuf,
        level: u8,
    ) -> Result<()> {
        let opts = os::create_file_opts(target, level as i64);
        let name = dest_path.to_str().unwrap();
        if let Err(e) = zw.start_file(name, opts) {
            Err(Error::Fatal(Box::new(e)))
        } else {
            let mut file = BufReader::new(File::open(target).unwrap());
            match std::io::copy(&mut file, zw) {
                Ok(_) => Ok(()),
                Err(e) => Err(Error::IO(e)),
            }
        }
    }
}

impl ToteArchiver for Archiver {
    fn perform(
        &self,
        file: File,
        targets: &[PathBuf],
        config: &crate::ArchiveConfig,
    ) -> Result<Vec<ArchiveEntry>> {
        let mut errs = vec![];
        let mut zw = zip::ZipWriter::new(file);
        let mut entries = vec![];
        for tp in targets.iter() {
            for entry in config.iter(tp) {
                let path = entry.path().to_path_buf();
                entries.push(ArchiveEntry::from(&path));
                if path.is_file() {
                    if let Err(e) = self.process_file(
                        &mut zw,
                        &path,
                        config.path_in_archive(&path),
                        config.level,
                    ) {
                        errs.push(e);
                    }
                }
            }
        }
        match zw.finish() {
            Ok(_) => Ok(entries),
            Err(e) => {
                errs.push(Error::Archiver(e.to_string()));
                Err(Error::Array(errs))
            }
        }
    }

    fn enable(&self) -> bool {
        true
    }
}

#[cfg(test)]
mod tests {
    use core::panic;

    use super::*;
    fn run_test<F>(f: F)
    where
        F: FnOnce(),
    {
        // setup(); // 予めやりたい処理
        let result = std::panic::catch_unwind(std::panic::AssertUnwindSafe(f));
        teardown(); // 後片付け処理

        if let Err(err) = result {
            std::panic::resume_unwind(err);
        }
    }

    #[test]
    fn test_zip() {
        run_test(|| {
            let config = crate::ArchiveConfig::builder()
                .dest("results/test.zip")
                .overwrite(true)
                .build();
            let v = vec!["lib", "cli", "Cargo.toml"]
                .into_iter()
                .map(|s| PathBuf::from(s))
                .collect::<Vec<_>>();
            if let Err(e) = crate::archive(&v, &config) {
                panic!("{e:?}")
            }
        });
    }

    fn teardown() {
        let _ = std::fs::remove_file("results/test.zip");
    }
}