use crate::entry::Entry; use std::io::{BufWriter, Write}; pub const RETURN_SEP: &[u8] = b"|"; pub const RETURN_START: &[u8] = b"filter-dataline|"; #[derive(Debug)] pub struct Message { session_id: String, token: String, content: Vec, nb_lines: usize, } impl Message { pub fn from_entry(entry: &Entry) -> Self { let mut ret = Self { session_id: entry.get_session_id().to_string(), token: entry.get_token().to_string(), content: Vec::with_capacity(crate::DEFAULT_MSG_SIZE), nb_lines: 0, }; if !entry.is_end_of_message() { ret.append_line(entry.get_data()); } ret } pub fn append_line(&mut self, line: &[u8]) { self.nb_lines += 1; if self.content.len() + line.len() > self.content.capacity() { self.content.reserve(crate::DEFAULT_MSG_SIZE); } self.content.extend_from_slice(line); match line.last() { Some(&c) => { if c != b'\r' { self.content.push(b'\r'); } } None => { self.content.push(b'\r'); } } self.content.push(b'\n'); } pub fn nb_lines(&self) -> usize { self.nb_lines } pub fn sign_and_return(&self) { // TODO: sign the message using DKIM let i = self.content.len() - 1; for line in self.content[0..i].split(|&b| b == b'\n') { self.print_line(line); } self.print_line(b"."); } fn print_line(&self, line: &[u8]) { let mut stdout = BufWriter::new(std::io::stdout()); stdout.write_all(RETURN_START).unwrap(); stdout.write_all(self.session_id.as_bytes()).unwrap(); stdout.write_all(RETURN_SEP).unwrap(); stdout.write_all(self.token.as_bytes()).unwrap(); stdout.write_all(RETURN_SEP).unwrap(); stdout.write_all(line).unwrap(); stdout.write_all(b"\n").unwrap(); } }