use std::io::Read; use std::path::PathBuf; use email::MimeMessage; use libimagref::hasher::Hasher; use libimagref::hasher::DefaultHasher; use libimagref::error::RefErrorKind as REK; use libimagref::error::MapErrInto; use libimagref::result::Result as RResult; use libimagerror::into::IntoError; use error::MailErrorKind as MEK; pub struct MailHasher { defaulthasher: DefaultHasher, } impl MailHasher { pub fn new() -> MailHasher { MailHasher { defaulthasher: DefaultHasher::new() } } } impl Hasher for MailHasher { fn hash_name(&self) -> &'static str { "default_mail_hasher" } fn create_hash(&mut self, pb: &PathBuf, c: &mut R) -> RResult { use filters::filter::Filter; use email::Header; let mut s = String::new(); try!(c.read_to_string(&mut s).map_err_into(REK::UTF8Error).map_err_into(REK::IOError)); MimeMessage::parse(&s) .map_err(Box::new) .map_err(|e| MEK::MailParsingError.into_error_with_cause(e)) .map_err_into(REK::RefHashingError) .and_then(|mail| { let has_key = |hdr: &Header, exp: &str| hdr.name == exp; let subject_filter = |hdr: &Header| has_key(hdr, "Subject"); let from_filter = |hdr: &Header| has_key(hdr, "From"); let to_filter = |hdr: &Header| has_key(hdr, "To"); let filter = subject_filter.or(from_filter).or(to_filter); let mut v : Vec = vec![]; for hdr in mail.headers.iter().filter(|item| filter.filter(item)) { let s = try!(hdr .get_value() .map_err(Box::new) .map_err(|e| REK::RefHashingError.into_error_with_cause(e))); v.push(s); } let s : String = v.join(""); self.defaulthasher.create_hash(pb, &mut s.as_bytes()) }) } }