use crate::{codec::Decode, util::PartialBuffer};
use std::{fmt, io};
use bzip2::{Decompress, Status};
pub struct BzDecoder {
decompress: Decompress,
}
impl fmt::Debug for BzDecoder {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"BzDecoder {{total_in: {}, total_out: {}}}",
self.decompress.total_in(),
self.decompress.total_out()
)
}
}
impl BzDecoder {
pub(crate) fn new() -> Self {
Self {
decompress: Decompress::new(false),
}
}
fn decode(
&mut self,
input: &mut PartialBuffer<impl AsRef<[u8]>>,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<Status> {
let prior_in = self.decompress.total_in();
let prior_out = self.decompress.total_out();
let status = self
.decompress
.decompress(input.unwritten(), output.unwritten_mut())
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
input.advance((self.decompress.total_in() - prior_in) as usize);
output.advance((self.decompress.total_out() - prior_out) as usize);
Ok(status)
}
}
impl Decode for BzDecoder {
fn reinit(&mut self) -> io::Result<()> {
self.decompress = Decompress::new(false);
Ok(())
}
fn decode(
&mut self,
input: &mut PartialBuffer<impl AsRef<[u8]>>,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<bool> {
match self.decode(input, output)? {
Status::Ok => Ok(false),
Status::FlushOk => unreachable!(),
Status::RunOk => unreachable!(),
Status::FinishOk => unreachable!(),
Status::StreamEnd => Ok(true),
Status::MemNeeded => Err(io::Error::new(io::ErrorKind::Other, "out of memory")),
}
}
fn flush(
&mut self,
output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<bool> {
self.decode(&mut PartialBuffer::new(&[][..]), output)?;
loop {
let old_len = output.written().len();
self.decode(&mut PartialBuffer::new(&[][..]), output)?;
if output.written().len() == old_len {
break;
}
}
Ok(!output.unwritten().is_empty())
}
fn finish(
&mut self,
_output: &mut PartialBuffer<impl AsRef<[u8]> + AsMut<[u8]>>,
) -> io::Result<bool> {
Ok(true)
}
}