use std::io::{Cursor, Result};
use arcode::{
bitbit::{BitReader, BitWriter, MSB},
ArithmeticDecoder, ArithmeticEncoder, EOFKind, Model,
};
use sherlock::SHERLOCK;
mod sherlock;
fn encode(data: &[u8]) -> Result<Vec<u8>> {
let mut model = Model::builder().num_bits(8).eof(EOFKind::EndAddOne).build();
let compressed = Cursor::new(vec![]);
let mut compressed_writer = BitWriter::new(compressed);
let mut encoder = ArithmeticEncoder::new(48);
for &sym in data {
encoder.encode(sym.into(), &model, &mut compressed_writer)?;
model.update_symbol(sym.into());
}
encoder.encode(model.eof(), &model, &mut compressed_writer)?;
encoder.finish_encode(&mut compressed_writer)?;
compressed_writer.pad_to_byte()?;
Ok(compressed_writer.get_ref().get_ref().clone())
}
fn decode(data: &[u8]) -> Result<Vec<u8>> {
let mut model = Model::builder().num_bits(8).eof(EOFKind::EndAddOne).build();
let mut input_reader = BitReader::<_, MSB>::new(data);
let mut decoder = ArithmeticDecoder::new(48);
let mut decompressed_data = vec![];
while !decoder.finished() {
let sym = decoder.decode(&model, &mut input_reader)?;
model.update_symbol(sym);
decompressed_data.push(sym as u8);
}
decompressed_data.pop();
Ok(decompressed_data)
}
#[test]
fn sherlock_e2e() {
let sherlock_bytes = SHERLOCK.bytes().into_iter().collect::<Vec<u8>>();
let compressed = encode(&sherlock_bytes).unwrap();
let decompressed = decode(&compressed).unwrap();
assert_eq!(sherlock_bytes.len(), decompressed.len());
sherlock_bytes
.iter()
.zip(decompressed.iter())
.enumerate()
.for_each(|(idx, (a, b))| {
assert_eq!(a, b, "Found mismatch {} != {} at index {}", a, b, idx);
});
}