use std::io::{self, Read};
use byteorder::WriteBytesExt;
use flate2::read::{DeflateDecoder, ZlibDecoder};
use crate::errors::Result;
use crate::packet::PacketTrait;
use crate::ser::Serialize;
use crate::types::{CompressionAlgorithm, Tag, Version};
#[derive(Clone, PartialEq, Eq, derive_more::Debug)]
pub struct CompressedData {
packet_version: Version,
compression_algorithm: CompressionAlgorithm,
#[debug("{}", hex::encode(compressed_data))]
compressed_data: Vec<u8>,
}
pub enum Decompressor<R> {
Uncompressed(R),
Zip(DeflateDecoder<R>),
Zlib(ZlibDecoder<R>),
Bzip2,
}
impl<'a> Read for Decompressor<&'a [u8]> {
fn read(&mut self, into: &mut [u8]) -> io::Result<usize> {
match self {
Decompressor::Uncompressed(ref mut c) => c.read(into),
Decompressor::Zip(ref mut c) => c.read(into),
Decompressor::Zlib(ref mut c) => c.read(into),
Decompressor::Bzip2 => unimplemented!("bzip2"),
}
}
}
impl CompressedData {
pub fn from_slice(packet_version: Version, input: &[u8]) -> Result<Self> {
ensure!(input.len() > 1, "input too short");
let alg = CompressionAlgorithm::from(input[0]);
Ok(CompressedData {
packet_version,
compression_algorithm: alg,
compressed_data: input[1..].to_vec(),
})
}
pub fn from_compressed(alg: CompressionAlgorithm, data: Vec<u8>) -> Self {
CompressedData {
packet_version: Default::default(),
compression_algorithm: alg,
compressed_data: data,
}
}
pub fn decompress(&self) -> Result<Decompressor<&[u8]>> {
match self.compression_algorithm {
CompressionAlgorithm::Uncompressed => {
Ok(Decompressor::Uncompressed(&self.compressed_data[..]))
}
CompressionAlgorithm::ZIP => Ok(Decompressor::Zip(DeflateDecoder::new(
&self.compressed_data[..],
))),
CompressionAlgorithm::ZLIB => Ok(Decompressor::Zlib(ZlibDecoder::new(
&self.compressed_data[..],
))),
CompressionAlgorithm::BZip2 => unimplemented_err!("BZip2"),
CompressionAlgorithm::Private10 | CompressionAlgorithm::Other(_) => unsupported_err!(
"CompressionAlgorithm {} is unsupported",
u8::from(self.compression_algorithm)
),
}
}
pub fn compressed_data(&self) -> &[u8] {
&self.compressed_data
}
}
impl Serialize for CompressedData {
fn to_writer<W: io::Write>(&self, writer: &mut W) -> Result<()> {
writer.write_u8(self.compression_algorithm.into())?;
writer.write_all(&self.compressed_data)?;
Ok(())
}
}
impl PacketTrait for CompressedData {
fn packet_version(&self) -> Version {
self.packet_version
}
fn tag(&self) -> Tag {
Tag::CompressedData
}
}