1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
use std::collections::HashMap;
#[derive(Debug)]
pub struct Email {
pub headers: HashMap<String, String>,
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.rsplitn(2, "\n\n").collect();
let mut headers = Some(val[1]);
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]),
String::from(key_val[0].trim_start()));
headers = rest;
},
_ => break,
}
}
Email{
headers: allheaders,
body: val[0].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);
}