use crate::header_value_parser::{create_header, EmailHeader};
use std::collections::HashMap;
#[derive(Debug)]
pub struct Email {
pub headers: HashMap<String, EmailHeader>,
pub body: EmailBody,
pub children: Vec<Email>,
}
type EmailBody = String;
impl Email {
pub fn from_str(s: String) -> Email {
let mut allheaders = HashMap::new();
let val: Vec<&str> = s.split("\n\n").collect();
println!("{:?}", val);
let mut headers = Some(val[0]);
while let Some(_) = headers {
match Email::get_one_header(headers.unwrap()) {
(Some(headerval), rest) => {
let key_val: Vec<&str> = headerval.rsplitn(2, ':').collect();
allheaders.insert(
String::from(key_val[1].to_lowercase()),
create_header(key_val[1], key_val[0].trim_start()),
);
headers = rest;
}
_ => break,
}
}
if let Some(EmailHeader::ContentType {
maintype,
subtype,
value,
}) = allheaders.get("content-type")
{
if maintype == "multipart" {
println!("Found a multipart email.")
}
}
Email {
headers: allheaders,
body: val[1].to_string(),
children: vec![],
}
}
fn get_one_header(_s: &str) -> (Option<String>, Option<&str>) {
let mut header_line = String::new();
let mut last = 0;
let bytes = _s.as_bytes();
for (i, &x) in bytes.iter().enumerate() {
last = i;
if x == b'\n' {
if bytes[i + 1] == b' ' {
continue;
} else {
break;
}
} else {
header_line.push(x as char);
}
}
let mut rest = Some(&_s[last + 1..]);
if last + 1 == _s.len() {
rest = None;
}
(Some(header_line), rest)
}
fn from_bytes() -> Email {
unimplemented!();
}
fn from_file() -> Email {
unimplemented!();
}
fn new() -> Email {
unimplemented!();
}
}
#[test]
fn test_get_one_simple_header() {
let headers = "From: Someone
To: Person
Date: Today";
assert_eq!(
Email::get_one_header(headers),
(
Some("From: Someone".to_string()),
Some("To: Person\nDate: Today")
)
)
}
#[test]
fn test_get_one_multiline_header() {
let headers = "From: acomplexheader
Subject: This is a complex header which goes to
2nd line identified by whitespace at the
start of each next line of header.";
let (header, rest) = Email::get_one_header(headers);
assert_eq!(header, Some("From: acomplexheader".to_string()));
assert_eq!(rest.is_some(), true);
let (header, rest) = Email::get_one_header(rest.unwrap());
assert_eq!(
header,
Some(
"Subject: This is a complex \
header which goes to 2nd line identified by whitespace at \
the start of each next line of header."
.to_string()
)
);
assert_eq!(rest, None);
}