use crate::{
progress::{ProgressArgs, copy_with_progress},
utils::*,
};
use anyhow::bail;
use clap::Args;
use std::{
fs::File,
io::{self, BufReader, BufWriter, Read, Write},
};
use xz2::read::XzDecoder;
use xz2::write::XzEncoder;
#[derive(Args, Debug)]
pub struct XzArgs {
#[clap(flatten)]
pub common_args: CommonArgs,
#[clap(flatten)]
progress_args: ProgressArgs,
#[clap(flatten)]
pub level_args: LevelArgs,
}
pub struct Xz {
pub level: i32,
pub progress_args: ProgressArgs,
}
impl Default for Xz {
fn default() -> Self {
let validator = DefaultCompressionValidator;
Xz {
level: validator.default_level(),
progress_args: ProgressArgs::default(),
}
}
}
impl Xz {
pub fn new(args: &XzArgs) -> Xz {
let validator = DefaultCompressionValidator;
let level = validator.validate_and_clamp_level(args.level_args.level.level);
Xz {
level,
progress_args: args.progress_args,
}
}
}
impl Compressor for Xz {
fn extension(&self) -> &str {
"xz"
}
fn name(&self) -> &str {
"xz"
}
fn compress(&self, input: CmprssInput, output: CmprssOutput) -> Result {
let mut file_size = None;
let mut input_stream = match input {
CmprssInput::Path(paths) => {
if paths.len() > 1 {
bail!("Multiple input files not supported for xz");
}
let path = &paths[0];
file_size = Some(std::fs::metadata(path)?.len());
Box::new(BufReader::new(File::open(path)?)) as Box<dyn Read + Send>
}
CmprssInput::Pipe(pipe) => Box::new(pipe) as Box<dyn Read + Send>,
CmprssInput::Reader(reader) => reader.0,
};
if let CmprssOutput::Writer(writer) = output {
let mut encoder = XzEncoder::new(writer, self.level as u32);
io::copy(&mut input_stream, &mut encoder)?;
encoder.finish()?;
} else {
let output_stream: Box<dyn Write + Send> = match &output {
CmprssOutput::Path(path) => Box::new(BufWriter::new(File::create(path)?)),
CmprssOutput::Pipe(pipe) => Box::new(pipe) as Box<dyn Write + Send>,
CmprssOutput::Writer(_) => unreachable!(),
};
let mut encoder = XzEncoder::new(output_stream, self.level as u32);
copy_with_progress(
&mut input_stream,
&mut encoder,
self.progress_args.chunk_size.size_in_bytes,
file_size,
self.progress_args.progress,
&output,
)?;
}
Ok(())
}
fn extract(&self, input: CmprssInput, output: CmprssOutput) -> Result {
let mut file_size = None;
let input_stream: Box<dyn Read + Send> = match input {
CmprssInput::Path(paths) => {
if paths.len() > 1 {
bail!("Multiple input files not supported for xz extraction");
}
let path = &paths[0];
file_size = Some(std::fs::metadata(path)?.len());
Box::new(BufReader::new(File::open(path)?)) as Box<dyn Read + Send>
}
CmprssInput::Pipe(pipe) => Box::new(pipe) as Box<dyn Read + Send>,
CmprssInput::Reader(reader) => reader.0,
};
let mut decoder = XzDecoder::new(input_stream);
if let CmprssOutput::Writer(mut writer) = output {
io::copy(&mut decoder, &mut writer)?;
} else {
let mut output_stream: Box<dyn Write + Send> = match &output {
CmprssOutput::Path(path) => Box::new(BufWriter::new(File::create(path)?)),
CmprssOutput::Pipe(pipe) => Box::new(pipe) as Box<dyn Write + Send>,
CmprssOutput::Writer(_) => unreachable!(),
};
copy_with_progress(
&mut decoder,
&mut *output_stream,
self.progress_args.chunk_size.size_in_bytes,
file_size,
self.progress_args.progress,
&output,
)?;
}
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::test_utils::*;
#[test]
fn test_xz_interface() {
let compressor = Xz::default();
test_compressor_interface(&compressor, "xz", Some("xz"));
}
#[test]
fn test_xz_default_compression() -> Result {
let compressor = Xz::default();
test_compression(&compressor)
}
#[test]
fn test_xz_fast_compression() -> Result {
let fast_compressor = Xz {
level: 1,
progress_args: ProgressArgs::default(),
};
test_compression(&fast_compressor)
}
#[test]
fn test_xz_best_compression() -> Result {
let best_compressor = Xz {
level: 9,
progress_args: ProgressArgs::default(),
};
test_compression(&best_compressor)
}
}