1use std::io::Write;
2
3use crate::{
4 internal::ppmd7::{PPMd7, RangeEncoder},
5 Error, PPMD7_MAX_MEM_SIZE, PPMD7_MAX_ORDER, PPMD7_MIN_MEM_SIZE, PPMD7_MIN_ORDER, SYM_END,
6};
7
8pub struct Ppmd7Encoder<W: Write> {
10 ppmd: PPMd7<RangeEncoder<W>>,
11}
12
13unsafe impl<W: Write> Send for Ppmd7Encoder<W> {}
14unsafe impl<W: Write> Sync for Ppmd7Encoder<W> {}
15
16impl<W: Write> Ppmd7Encoder<W> {
17 pub fn new(writer: W, order: u32, mem_size: u32) -> crate::Result<Self> {
22 if !(PPMD7_MIN_ORDER..=PPMD7_MAX_ORDER).contains(&order)
23 || !(PPMD7_MIN_MEM_SIZE..=PPMD7_MAX_MEM_SIZE).contains(&mem_size)
24 {
25 return Err(Error::InvalidParameter);
26 }
27
28 let ppmd = PPMd7::new_encoder(writer, order, mem_size)?;
29
30 Ok(Self { ppmd })
31 }
32
33 pub fn get_ref(&self) -> &W {
35 self.ppmd.get_ref()
36 }
37
38 pub fn get_mut(&mut self) -> &mut W {
43 self.ppmd.get_mut()
44 }
45
46 pub fn into_inner(self) -> W {
48 self.ppmd.into_inner()
49 }
50
51 pub fn finish(mut self, with_end_marker: bool) -> std::io::Result<W> {
55 if with_end_marker {
56 self.ppmd.encode_symbol(SYM_END)?;
57 }
58 self.flush()?;
59 Ok(self.into_inner())
60 }
61}
62
63impl<W: Write> Write for Ppmd7Encoder<W> {
64 fn write(&mut self, buf: &[u8]) -> std::io::Result<usize> {
65 if buf.is_empty() {
66 return Ok(0);
67 }
68
69 for &byte in buf.iter() {
70 self.ppmd.encode_symbol(byte as i32)?;
71 }
72
73 Ok(buf.len())
74 }
75
76 fn flush(&mut self) -> std::io::Result<()> {
77 self.ppmd.flush_range_encoder()
78 }
79}
80
81#[cfg(test)]
82mod test {
83 use std::io::{Read, Write};
84
85 use super::{super::decoder_7::Ppmd7Decoder, Ppmd7Encoder};
86
87 const ORDER: u32 = 8;
88 const MEM_SIZE: u32 = 262144;
89
90 #[test]
91 fn ppmd7encoder_without_end_marker() {
92 let test_data = include_str!("../tests/fixtures/text/apache2.txt");
93
94 let mut data = Vec::new();
95 {
96 let mut encoder = Ppmd7Encoder::new(&mut data, ORDER, MEM_SIZE).unwrap();
97 encoder.write_all(test_data.as_bytes()).unwrap();
98 encoder.finish(false).unwrap();
99 }
100
101 let mut decoder = Ppmd7Decoder::new(data.as_slice(), ORDER, MEM_SIZE).unwrap();
102
103 let mut decoded = vec![0; test_data.len()];
104 decoder.read_exact(&mut decoded).unwrap();
105
106 let decoded_data = String::from_utf8(decoded).unwrap();
107 assert_eq!(decoded_data, test_data);
108 }
109
110 #[test]
111 fn ppmd7encoder_with_end_marker() {
112 let test_data = include_str!("../tests/fixtures/text/apache2.txt");
113
114 let mut data = Vec::new();
115 {
116 let mut encoder = Ppmd7Encoder::new(&mut data, ORDER, MEM_SIZE).unwrap();
117 encoder.write_all(test_data.as_bytes()).unwrap();
118 encoder.finish(true).unwrap();
119 }
120
121 let mut decoder = Ppmd7Decoder::new(data.as_slice(), ORDER, MEM_SIZE).unwrap();
122
123 let mut decoded = Vec::new();
124 decoder.read_to_end(&mut decoded).unwrap();
125
126 let decoded_data = String::from_utf8(decoded).unwrap();
127 assert_eq!(decoded_data, test_data);
128 }
129}