use crate::{codec::Encode, util::PartialBuffer};
use std::io;
use flate2::{Compress, Compression, FlushCompress, Status};
#[derive(Debug)]
pub struct FlateEncoder {
compress: Compress,
flushed: bool,
}
impl FlateEncoder {
pub(crate) fn new(level: Compression, zlib_header: bool) -> Self {
Self {
compress: Compress::new(level, zlib_header),
flushed: true,
}
}
pub(crate) fn get_ref(&self) -> &Compress {
&self.compress
}
fn encode(
&mut self,
input: &mut PartialBuffer<impl AsRef<[u8]>>,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
flush: FlushCompress,
) -> io::Result<Status> {
let prior_in = self.compress.total_in();
let prior_out = self.compress.total_out();
let status = self
.compress
.compress(input.unwritten(), output.unwritten_mut(), flush)?;
input.advance((self.compress.total_in() - prior_in) as usize);
output.advance((self.compress.total_out() - prior_out) as usize);
Ok(status)
}
}
impl Encode for FlateEncoder {
fn encode(
&mut self,
input: &mut PartialBuffer<impl AsRef<[u8]>>,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<()> {
self.flushed = false;
match self.encode(input, output, FlushCompress::None)? {
Status::Ok => Ok(()),
Status::StreamEnd => unreachable!(),
Status::BufError => Err(io::Error::new(io::ErrorKind::Other, "unexpected BufError")),
}
}
fn flush(
&mut self,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<bool> {
if self.flushed {
return Ok(true);
}
self.encode(
&mut PartialBuffer::new(&[][..]),
output,
FlushCompress::Sync,
)?;
loop {
let old_len = output.written().len();
self.encode(
&mut PartialBuffer::new(&[][..]),
output,
FlushCompress::None,
)?;
if output.written().len() == old_len {
break;
}
}
self.flushed = true;
Ok(!output.unwritten().is_empty())
}
fn finish(
&mut self,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<bool> {
self.flushed = false;
match self.encode(
&mut PartialBuffer::new(&[][..]),
output,
FlushCompress::Finish,
)? {
Status::Ok => Ok(false),
Status::StreamEnd => Ok(true),
Status::BufError => Err(io::Error::new(io::ErrorKind::Other, "unexpected BufError")),
}
}
}