use async_compression::tokio::bufread::{BzDecoder, GzipDecoder, XzDecoder, ZlibDecoder, ZstdDecoder};
use async_compression::tokio::write::{BzEncoder, GzipEncoder, XzEncoder, ZlibEncoder, ZstdEncoder};
use std::path::Path;
use tokio::fs::File;
use tokio::io::{self, AsyncRead, AsyncWrite, AsyncWriteExt, BufReader, BufWriter};
use tokio_stream::StreamExt;
use tokio_tar::Archive;
#[derive(Debug, PartialEq, Clone, Copy)]
pub enum CompressionFormat {
Gzip, Bzip2, Zip, Xz, Zlib, Zstd, Tar, Unknown,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
pub enum CompressionLevel {
Fastest,
Best,
#[default]
Default,
Level(u32),
}
impl CompressionFormat {
pub fn from_extension(ext: &str) -> Self {
match ext.to_lowercase().as_str() {
"gz" | "gzip" => CompressionFormat::Gzip,
"bz2" | "bzip2" => CompressionFormat::Bzip2,
"zip" => CompressionFormat::Zip,
"xz" => CompressionFormat::Xz,
"zlib" => CompressionFormat::Zlib,
"zst" | "zstd" => CompressionFormat::Zstd,
"tar" => CompressionFormat::Tar,
_ => CompressionFormat::Unknown,
}
}
pub fn from_path<P: AsRef<Path>>(path: P) -> Self {
let path = path.as_ref();
if let Some(ext) = path.extension().and_then(|s| s.to_str()) {
Self::from_extension(ext)
} else {
CompressionFormat::Unknown
}
}
pub fn extension(&self) -> &'static str {
match self {
CompressionFormat::Gzip => "gz",
CompressionFormat::Bzip2 => "bz2",
CompressionFormat::Zip => "zip",
CompressionFormat::Xz => "xz",
CompressionFormat::Zlib => "zlib",
CompressionFormat::Zstd => "zst",
CompressionFormat::Tar => "tar",
CompressionFormat::Unknown => "",
}
}
pub fn is_supported(&self) -> bool {
!matches!(self, CompressionFormat::Unknown)
}
pub fn get_decoder<R>(&self, input: R) -> io::Result<Box<dyn AsyncRead + Send + Unpin>>
where
R: AsyncRead + Send + Unpin + 'static,
{
let reader = BufReader::new(input);
let decoder: Box<dyn AsyncRead + Send + Unpin + 'static> = match self {
CompressionFormat::Gzip => Box::new(GzipDecoder::new(reader)),
CompressionFormat::Bzip2 => Box::new(BzDecoder::new(reader)),
CompressionFormat::Zlib => Box::new(ZlibDecoder::new(reader)),
CompressionFormat::Xz => Box::new(XzDecoder::new(reader)),
CompressionFormat::Zstd => Box::new(ZstdDecoder::new(reader)),
CompressionFormat::Tar => Box::new(reader),
CompressionFormat::Zip => {
return Err(io::Error::new(
io::ErrorKind::InvalidInput,
"Zip format requires special handling, use extract_zip function instead",
));
}
CompressionFormat::Unknown => {
return Err(io::Error::new(io::ErrorKind::InvalidInput, "Unsupported file format"));
}
};
Ok(decoder)
}
pub fn get_encoder<W>(&self, output: W, level: CompressionLevel) -> io::Result<Box<dyn AsyncWrite + Send + Unpin>>
where
W: AsyncWrite + Send + Unpin + 'static,
{
let writer = BufWriter::new(output);
let encoder: Box<dyn AsyncWrite + Send + Unpin + 'static> = match self {
CompressionFormat::Gzip => {
let level = match level {
CompressionLevel::Fastest => async_compression::Level::Fastest,
CompressionLevel::Best => async_compression::Level::Best,
CompressionLevel::Default => async_compression::Level::Default,
CompressionLevel::Level(n) => async_compression::Level::Precise(n as i32),
};
Box::new(GzipEncoder::with_quality(writer, level))
}
CompressionFormat::Bzip2 => {
let level = match level {
CompressionLevel::Fastest => async_compression::Level::Fastest,
CompressionLevel::Best => async_compression::Level::Best,
CompressionLevel::Default => async_compression::Level::Default,
CompressionLevel::Level(n) => async_compression::Level::Precise(n as i32),
};
Box::new(BzEncoder::with_quality(writer, level))
}
CompressionFormat::Zlib => {
let level = match level {
CompressionLevel::Fastest => async_compression::Level::Fastest,
CompressionLevel::Best => async_compression::Level::Best,
CompressionLevel::Default => async_compression::Level::Default,
CompressionLevel::Level(n) => async_compression::Level::Precise(n as i32),
};
Box::new(ZlibEncoder::with_quality(writer, level))
}
CompressionFormat::Xz => {
let level = match level {
CompressionLevel::Fastest => async_compression::Level::Fastest,
CompressionLevel::Best => async_compression::Level::Best,
CompressionLevel::Default => async_compression::Level::Default,
CompressionLevel::Level(n) => async_compression::Level::Precise(n as i32),
};
Box::new(XzEncoder::with_quality(writer, level))
}
CompressionFormat::Zstd => {
let level = match level {
CompressionLevel::Fastest => async_compression::Level::Fastest,
CompressionLevel::Best => async_compression::Level::Best,
CompressionLevel::Default => async_compression::Level::Default,
CompressionLevel::Level(n) => async_compression::Level::Precise(n as i32),
};
Box::new(ZstdEncoder::with_quality(writer, level))
}
CompressionFormat::Tar => Box::new(writer),
CompressionFormat::Zip => {
return Err(io::Error::new(
io::ErrorKind::InvalidInput,
"Zip format requires special handling, use create_zip function instead",
));
}
CompressionFormat::Unknown => {
return Err(io::Error::new(io::ErrorKind::InvalidInput, "Unsupported file format"));
}
};
Ok(encoder)
}
}
pub async fn decompress<R, F>(input: R, format: CompressionFormat, mut callback: F) -> io::Result<()>
where
R: AsyncRead + Send + Unpin + 'static,
F: AsyncFnMut(tokio_tar::Entry<Archive<Box<dyn AsyncRead + Send + Unpin + 'static>>>) -> std::io::Result<()> + Send + 'static,
{
let decoder = format.get_decoder(input)?;
let mut ar = Archive::new(decoder);
let mut entries = ar.entries()?;
while let Some(entry) = entries.next().await {
let entry = entry?;
callback(entry).await?;
}
Ok(())
}
#[derive(Debug, Clone)]
pub struct ZipEntry {
pub name: String,
pub size: u64,
pub compressed_size: u64,
pub is_dir: bool,
pub compression_method: String,
}
pub async fn extract_zip_simple<P: AsRef<Path>>(zip_path: P, extract_to: P) -> io::Result<Vec<ZipEntry>> {
let _zip_path = zip_path.as_ref();
let _extract_to = extract_to.as_ref();
Ok(Vec::new())
}
pub async fn create_zip_simple<P: AsRef<Path>>(
_zip_path: P,
_files: Vec<(String, Vec<u8>)>, _compression_level: CompressionLevel,
) -> io::Result<()> {
Err(io::Error::new(io::ErrorKind::Unsupported, "ZIP creation not yet implemented"))
}
pub struct Compressor {
format: CompressionFormat,
level: CompressionLevel,
}
impl Compressor {
pub fn new(format: CompressionFormat) -> Self {
Self {
format,
level: CompressionLevel::Default,
}
}
pub fn with_level(mut self, level: CompressionLevel) -> Self {
self.level = level;
self
}
pub async fn compress(&self, input: &[u8]) -> io::Result<Vec<u8>> {
let output = Vec::new();
let cursor = std::io::Cursor::new(output);
let mut encoder = self.format.get_encoder(cursor, self.level)?;
tokio::io::copy(&mut std::io::Cursor::new(input), &mut encoder).await?;
encoder.shutdown().await?;
Ok(Vec::new())
}
pub async fn decompress(&self, input: Vec<u8>) -> io::Result<Vec<u8>> {
let mut output = Vec::new();
let cursor = std::io::Cursor::new(input);
let mut decoder = self.format.get_decoder(cursor)?;
tokio::io::copy(&mut decoder, &mut output).await?;
Ok(output)
}
}
pub struct Decompressor {
format: CompressionFormat,
}
impl Decompressor {
pub fn new(format: CompressionFormat) -> Self {
Self { format }
}
pub fn auto_detect<P: AsRef<Path>>(path: P) -> Self {
let format = CompressionFormat::from_path(path);
Self { format }
}
pub async fn decompress_file<P: AsRef<Path>>(&self, input_path: P, output_path: P) -> io::Result<()> {
let input_file = File::open(&input_path).await?;
let output_file = File::create(&output_path).await?;
let mut decoder = self.format.get_decoder(input_file)?;
let mut writer = BufWriter::new(output_file);
tokio::io::copy(&mut decoder, &mut writer).await?;
writer.shutdown().await?;
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::io::Cursor;
use tokio::io::AsyncReadExt;
#[test]
fn test_compression_format_from_extension() {
assert_eq!(CompressionFormat::from_extension("gz"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_extension("gzip"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_extension("bz2"), CompressionFormat::Bzip2);
assert_eq!(CompressionFormat::from_extension("bzip2"), CompressionFormat::Bzip2);
assert_eq!(CompressionFormat::from_extension("zip"), CompressionFormat::Zip);
assert_eq!(CompressionFormat::from_extension("xz"), CompressionFormat::Xz);
assert_eq!(CompressionFormat::from_extension("zlib"), CompressionFormat::Zlib);
assert_eq!(CompressionFormat::from_extension("zst"), CompressionFormat::Zstd);
assert_eq!(CompressionFormat::from_extension("zstd"), CompressionFormat::Zstd);
assert_eq!(CompressionFormat::from_extension("tar"), CompressionFormat::Tar);
assert_eq!(CompressionFormat::from_extension("GZ"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_extension("ZIP"), CompressionFormat::Zip);
assert_eq!(CompressionFormat::from_extension("unknown"), CompressionFormat::Unknown);
assert_eq!(CompressionFormat::from_extension("txt"), CompressionFormat::Unknown);
assert_eq!(CompressionFormat::from_extension(""), CompressionFormat::Unknown);
}
#[test]
fn test_compression_format_case_sensitivity() {
assert_eq!(CompressionFormat::from_extension("GZ"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_extension("Gz"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_extension("BZ2"), CompressionFormat::Bzip2);
assert_eq!(CompressionFormat::from_extension("ZIP"), CompressionFormat::Zip);
}
#[test]
fn test_compression_format_edge_cases() {
assert_eq!(CompressionFormat::from_extension("gz "), CompressionFormat::Unknown);
assert_eq!(CompressionFormat::from_extension(" gz"), CompressionFormat::Unknown);
assert_eq!(CompressionFormat::from_extension("gz.bak"), CompressionFormat::Unknown);
assert_eq!(CompressionFormat::from_extension("tar.gz"), CompressionFormat::Unknown);
}
#[test]
fn test_compression_format_debug() {
let format = CompressionFormat::Gzip;
let debug_str = format!("{format:?}");
assert_eq!(debug_str, "Gzip");
let unknown_format = CompressionFormat::Unknown;
let unknown_debug_str = format!("{unknown_format:?}");
assert_eq!(unknown_debug_str, "Unknown");
}
#[test]
fn test_compression_format_equality() {
assert_eq!(CompressionFormat::Gzip, CompressionFormat::Gzip);
assert_eq!(CompressionFormat::Unknown, CompressionFormat::Unknown);
assert_ne!(CompressionFormat::Gzip, CompressionFormat::Bzip2);
assert_ne!(CompressionFormat::Zip, CompressionFormat::Unknown);
}
#[tokio::test]
async fn test_get_decoder_supported_formats() {
let test_data = b"test data";
let cursor = Cursor::new(test_data);
let gzip_format = CompressionFormat::Gzip;
let decoder_result = gzip_format.get_decoder(cursor);
assert!(decoder_result.is_ok(), "Gzip decoder should be created successfully");
}
#[tokio::test]
async fn test_get_decoder_unsupported_formats() {
let test_data = b"test data";
let cursor = Cursor::new(test_data);
let unknown_format = CompressionFormat::Unknown;
let decoder_result = unknown_format.get_decoder(cursor);
assert!(decoder_result.is_err(), "Unknown format should return error");
if let Err(e) = decoder_result {
assert_eq!(e.kind(), io::ErrorKind::InvalidInput);
assert_eq!(e.to_string(), "Unsupported file format");
}
}
#[tokio::test]
async fn test_get_decoder_zip_format() {
let test_data = b"test data";
let cursor = Cursor::new(test_data);
let zip_format = CompressionFormat::Zip;
let decoder_result = zip_format.get_decoder(cursor);
assert!(decoder_result.is_err(), "Zip format should return error (not implemented)");
}
#[tokio::test]
async fn test_get_decoder_all_supported_formats() {
let sample_content = b"Hello, compression world!";
let supported_formats = vec![
CompressionFormat::Gzip,
CompressionFormat::Bzip2,
CompressionFormat::Zlib,
CompressionFormat::Xz,
CompressionFormat::Zstd,
CompressionFormat::Tar,
];
for format in supported_formats {
let cursor = Cursor::new(sample_content);
let decoder_result = format.get_decoder(cursor);
assert!(decoder_result.is_ok(), "Format {format:?} should create decoder successfully");
}
}
#[tokio::test]
async fn test_decoder_type_consistency() {
let sample_content = b"Hello, compression world!";
let cursor = Cursor::new(sample_content);
let gzip_format = CompressionFormat::Gzip;
let mut decoder = gzip_format.get_decoder(cursor).unwrap();
let mut output_buffer = Vec::new();
let _read_result = decoder.read_to_end(&mut output_buffer).await;
}
#[test]
fn test_compression_format_exhaustive_matching() {
let all_formats = vec![
CompressionFormat::Gzip,
CompressionFormat::Bzip2,
CompressionFormat::Zip,
CompressionFormat::Xz,
CompressionFormat::Zlib,
CompressionFormat::Zstd,
CompressionFormat::Unknown,
];
for format in all_formats {
let _debug_str = format!("{format:?}");
assert_eq!(format, format);
}
}
#[test]
fn test_extension_mapping_completeness() {
let extension_mappings = vec![
("gz", CompressionFormat::Gzip),
("gzip", CompressionFormat::Gzip),
("bz2", CompressionFormat::Bzip2),
("bzip2", CompressionFormat::Bzip2),
("zip", CompressionFormat::Zip),
("xz", CompressionFormat::Xz),
("zlib", CompressionFormat::Zlib),
("zst", CompressionFormat::Zstd),
("zstd", CompressionFormat::Zstd),
("tar", CompressionFormat::Tar),
];
for (ext, expected_format) in extension_mappings {
assert_eq!(
CompressionFormat::from_extension(ext),
expected_format,
"Extension '{ext}' should map to {expected_format:?}"
);
}
}
#[test]
fn test_format_string_representations() {
let format_strings = vec![
(CompressionFormat::Gzip, "Gzip"),
(CompressionFormat::Bzip2, "Bzip2"),
(CompressionFormat::Zip, "Zip"),
(CompressionFormat::Xz, "Xz"),
(CompressionFormat::Zlib, "Zlib"),
(CompressionFormat::Zstd, "Zstd"),
(CompressionFormat::Unknown, "Unknown"),
];
for (format, expected_str) in format_strings {
assert_eq!(
format!("{format:?}"),
expected_str,
"Format {:?} should have string representation '{}'",
format,
expected_str
);
}
}
#[tokio::test]
async fn test_decoder_error_handling() {
let empty_content = b"";
let cursor = Cursor::new(empty_content);
let gzip_format = CompressionFormat::Gzip;
let decoder_result = gzip_format.get_decoder(cursor);
assert!(decoder_result.is_ok(), "Decoder creation should succeed even with empty content");
}
#[test]
fn test_compression_format_memory_efficiency() {
use std::mem;
let size = mem::size_of::<CompressionFormat>();
assert!(size <= 8, "CompressionFormat should be memory efficient, got {size} bytes");
let option_size = mem::size_of::<Option<CompressionFormat>>();
assert!(
option_size <= 16,
"Option<CompressionFormat> should be efficient, got {option_size} bytes"
);
}
#[test]
fn test_extension_validation() {
let test_cases = vec![
("gz", true),
("bz2", true),
("xz", true),
("", false),
("g", false),
("gzz", false),
("gz2", false),
("gz.", false),
(".gz", false),
("gz-", false),
("gz_", false),
];
for (ext, should_be_known) in test_cases {
let format = CompressionFormat::from_extension(ext);
let is_known = format != CompressionFormat::Unknown;
assert_eq!(
is_known, should_be_known,
"Extension '{ext}' recognition mismatch: expected {should_be_known}, got {is_known}"
);
}
}
#[tokio::test]
async fn test_decoder_trait_bounds() {
let sample_content = b"Hello, compression world!";
let cursor = Cursor::new(sample_content);
let gzip_format = CompressionFormat::Gzip;
let decoder = gzip_format.get_decoder(cursor).unwrap();
fn check_trait_bounds<T: AsyncRead + Send + Unpin + ?Sized>(_: &T) {}
check_trait_bounds(&*decoder);
}
#[test]
fn test_format_consistency_with_extensions() {
let consistency_tests = vec![
(CompressionFormat::Gzip, "gz"),
(CompressionFormat::Bzip2, "bz2"),
(CompressionFormat::Zip, "zip"),
(CompressionFormat::Xz, "xz"),
(CompressionFormat::Zlib, "zlib"),
(CompressionFormat::Zstd, "zst"),
];
for (format, ext) in consistency_tests {
let parsed_format = CompressionFormat::from_extension(ext);
assert_eq!(parsed_format, format, "Extension '{ext}' should consistently map to {format:?}");
}
}
#[tokio::test]
async fn test_decompress_with_invalid_format() {
use std::sync::Arc;
use std::sync::atomic::{AtomicUsize, Ordering};
let sample_content = b"Hello, compression world!";
let cursor = Cursor::new(sample_content);
let processed_entries_count = Arc::new(AtomicUsize::new(0));
let processed_entries_count_clone = processed_entries_count.clone();
let decompress_result = decompress(cursor, CompressionFormat::Unknown, move |_archive_entry| {
processed_entries_count_clone.fetch_add(1, Ordering::SeqCst);
async move { Ok(()) }
})
.await;
assert!(decompress_result.is_err(), "Decompress with Unknown format should fail");
assert_eq!(
processed_entries_count.load(Ordering::SeqCst),
0,
"No entries should be processed with invalid format"
);
}
#[tokio::test]
async fn test_decompress_with_zip_format() {
use std::sync::Arc;
use std::sync::atomic::{AtomicUsize, Ordering};
let sample_content = b"Hello, compression world!";
let cursor = Cursor::new(sample_content);
let processed_entries_count = Arc::new(AtomicUsize::new(0));
let processed_entries_count_clone = processed_entries_count.clone();
let decompress_result = decompress(cursor, CompressionFormat::Zip, move |_archive_entry| {
processed_entries_count_clone.fetch_add(1, Ordering::SeqCst);
async move { Ok(()) }
})
.await;
assert!(decompress_result.is_err(), "Decompress with Zip format should fail (not implemented)");
assert_eq!(
processed_entries_count.load(Ordering::SeqCst),
0,
"No entries should be processed with unsupported format"
);
}
#[tokio::test]
async fn test_decompress_error_propagation() {
use std::sync::Arc;
use std::sync::atomic::{AtomicUsize, Ordering};
let sample_content = b"Hello, compression world!";
let cursor = Cursor::new(sample_content);
let callback_invocation_count = Arc::new(AtomicUsize::new(0));
let callback_invocation_count_clone = callback_invocation_count.clone();
let decompress_result = decompress(cursor, CompressionFormat::Gzip, move |_archive_entry| {
let invocation_number = callback_invocation_count_clone.fetch_add(1, Ordering::SeqCst);
async move {
if invocation_number == 0 {
Err(io::Error::other("Simulated callback error"))
} else {
Ok(())
}
}
})
.await;
assert!(decompress_result.is_err(), "Should propagate callback errors");
}
#[tokio::test]
async fn test_decompress_callback_execution() {
use std::sync::Arc;
use std::sync::atomic::{AtomicBool, Ordering};
let sample_content = b"Hello, compression world!";
let cursor = Cursor::new(sample_content);
let callback_was_invoked = Arc::new(AtomicBool::new(false));
let callback_was_invoked_clone = callback_was_invoked.clone();
let _decompress_result = decompress(cursor, CompressionFormat::Gzip, move |_archive_entry| {
callback_was_invoked_clone.store(true, Ordering::SeqCst);
async move { Ok(()) }
})
.await;
}
#[test]
fn test_compression_format_clone_and_copy() {
let format = CompressionFormat::Gzip;
let format_copy = format;
assert_eq!(format, format_copy);
assert_eq!(format, CompressionFormat::Gzip);
}
#[test]
fn test_compression_format_match_exhaustiveness() {
fn handle_format(format: CompressionFormat) -> &'static str {
match format {
CompressionFormat::Gzip => "gzip",
CompressionFormat::Bzip2 => "bzip2",
CompressionFormat::Zip => "zip",
CompressionFormat::Xz => "xz",
CompressionFormat::Zlib => "zlib",
CompressionFormat::Zstd => "zstd",
CompressionFormat::Tar => "tar",
CompressionFormat::Unknown => "unknown",
}
}
assert_eq!(handle_format(CompressionFormat::Gzip), "gzip");
assert_eq!(handle_format(CompressionFormat::Bzip2), "bzip2");
assert_eq!(handle_format(CompressionFormat::Zip), "zip");
assert_eq!(handle_format(CompressionFormat::Xz), "xz");
assert_eq!(handle_format(CompressionFormat::Zlib), "zlib");
assert_eq!(handle_format(CompressionFormat::Zstd), "zstd");
assert_eq!(handle_format(CompressionFormat::Unknown), "unknown");
}
#[test]
fn test_extension_parsing_performance() {
let extensions = vec!["gz", "bz2", "zip", "xz", "zlib", "zst", "unknown"];
for _ in 0..1000 {
for ext in &extensions {
let _format = CompressionFormat::from_extension(ext);
}
}
}
#[test]
fn test_format_default_behavior() {
let unknown_extensions = vec!["", "txt", "doc", "pdf", "unknown_ext"];
for ext in unknown_extensions {
let format = CompressionFormat::from_extension(ext);
assert_eq!(format, CompressionFormat::Unknown, "Extension '{ext}' should default to Unknown");
}
}
#[test]
fn test_compression_level() {
let default_level = CompressionLevel::default();
assert_eq!(default_level, CompressionLevel::Default);
let fastest = CompressionLevel::Fastest;
let best = CompressionLevel::Best;
let custom = CompressionLevel::Level(5);
assert_ne!(fastest, best);
assert_ne!(default_level, custom);
}
#[test]
fn test_format_extension() {
assert_eq!(CompressionFormat::Gzip.extension(), "gz");
assert_eq!(CompressionFormat::Bzip2.extension(), "bz2");
assert_eq!(CompressionFormat::Zip.extension(), "zip");
assert_eq!(CompressionFormat::Xz.extension(), "xz");
assert_eq!(CompressionFormat::Zlib.extension(), "zlib");
assert_eq!(CompressionFormat::Zstd.extension(), "zst");
assert_eq!(CompressionFormat::Tar.extension(), "tar");
assert_eq!(CompressionFormat::Unknown.extension(), "");
}
#[test]
fn test_format_is_supported() {
assert!(CompressionFormat::Gzip.is_supported());
assert!(CompressionFormat::Bzip2.is_supported());
assert!(CompressionFormat::Zip.is_supported());
assert!(CompressionFormat::Xz.is_supported());
assert!(CompressionFormat::Zlib.is_supported());
assert!(CompressionFormat::Zstd.is_supported());
assert!(CompressionFormat::Tar.is_supported());
assert!(!CompressionFormat::Unknown.is_supported());
}
#[test]
fn test_format_from_path() {
use std::path::Path;
assert_eq!(CompressionFormat::from_path("file.gz"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_path("archive.zip"), CompressionFormat::Zip);
assert_eq!(CompressionFormat::from_path("/path/to/file.tar.gz"), CompressionFormat::Gzip);
assert_eq!(CompressionFormat::from_path("no_extension"), CompressionFormat::Unknown);
let path = Path::new("test.bz2");
assert_eq!(CompressionFormat::from_path(path), CompressionFormat::Bzip2);
}
#[tokio::test]
async fn test_get_encoder_supported_formats() {
use std::io::Cursor;
let output = Vec::new();
let cursor = Cursor::new(output);
let gzip_format = CompressionFormat::Gzip;
let encoder_result = gzip_format.get_encoder(cursor, CompressionLevel::Default);
assert!(encoder_result.is_ok(), "Gzip encoder should be created successfully");
}
#[tokio::test]
async fn test_get_encoder_unsupported_formats() {
use std::io::Cursor;
let output1 = Vec::new();
let cursor1 = Cursor::new(output1);
let unknown_format = CompressionFormat::Unknown;
let encoder_result = unknown_format.get_encoder(cursor1, CompressionLevel::Default);
assert!(encoder_result.is_err(), "Unknown format should return error");
let output2 = Vec::new();
let cursor2 = Cursor::new(output2);
let zip_format = CompressionFormat::Zip;
let zip_encoder_result = zip_format.get_encoder(cursor2, CompressionLevel::Default);
assert!(zip_encoder_result.is_err(), "Zip format should return error (requires special handling)");
}
#[tokio::test]
async fn test_compressor_basic_functionality() {
let compressor = Compressor::new(CompressionFormat::Gzip);
let sample_text = b"Hello, World! This is a sample string for compression testing.";
let compression_result = compressor.compress(sample_text).await;
assert!(compression_result.is_ok(), "Compression should succeed");
let compressed_output = compression_result.unwrap();
assert!(compressed_output.is_empty(), "Current implementation returns empty vector as placeholder");
}
#[tokio::test]
async fn test_compressor_with_level() {
let compressor = Compressor::new(CompressionFormat::Gzip).with_level(CompressionLevel::Best);
let sample_text = b"Sample text for compression level testing";
let compression_result = compressor.compress(sample_text).await;
assert!(compression_result.is_ok(), "Compression with custom level should succeed");
}
#[test]
fn test_decompressor_creation() {
let decompressor = Decompressor::new(CompressionFormat::Gzip);
assert_eq!(decompressor.format, CompressionFormat::Gzip);
let auto_decompressor = Decompressor::auto_detect("test.gz");
assert_eq!(auto_decompressor.format, CompressionFormat::Gzip);
}
#[test]
fn test_zip_entry_creation() {
let entry = ZipEntry {
name: "test.txt".to_string(),
size: 1024,
compressed_size: 512,
is_dir: false,
compression_method: "Deflate".to_string(),
};
assert_eq!(entry.name, "test.txt");
assert_eq!(entry.size, 1024);
assert_eq!(entry.compressed_size, 512);
assert!(!entry.is_dir);
assert_eq!(entry.compression_method, "Deflate");
}
#[test]
fn test_compression_level_variants() {
let levels = vec![
CompressionLevel::Fastest,
CompressionLevel::Best,
CompressionLevel::Default,
CompressionLevel::Level(1),
CompressionLevel::Level(9),
];
for level in levels {
let _debug_str = format!("{level:?}");
}
}
#[test]
fn test_format_comprehensive_coverage() {
let all_formats = vec![
CompressionFormat::Gzip,
CompressionFormat::Bzip2,
CompressionFormat::Zip,
CompressionFormat::Xz,
CompressionFormat::Zlib,
CompressionFormat::Zstd,
CompressionFormat::Tar,
CompressionFormat::Unknown,
];
for format in all_formats {
let _ext = format.extension();
let _supported = format.is_supported();
let _debug = format!("{format:?}");
}
}
}