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
use crate::errors::Error;
#[inline]
pub fn escape_html(input: &str) -> String {
let mut output = String::with_capacity(input.len() * 2);
for c in input.chars() {
match c {
'&' => output.push_str("&"),
'<' => output.push_str("<"),
'>' => output.push_str(">"),
'"' => output.push_str("""),
'\'' => output.push_str("'"),
'/' => output.push_str("/"),
_ => output.push(c),
}
}
output
}
pub(crate) fn render_to_string<C, F, E>(context: C, render: F) -> Result<String, Error>
where
C: FnOnce() -> String,
F: FnOnce(&mut Vec<u8>) -> Result<(), E>,
Error: From<E>,
{
let mut buffer = Vec::new();
render(&mut buffer).map_err(Error::from)?;
buffer_to_string(context, buffer)
}
pub(crate) fn buffer_to_string<F>(context: F, buffer: Vec<u8>) -> Result<String, Error>
where
F: FnOnce() -> String,
{
String::from_utf8(buffer).map_err(|error| Error::utf8_conversion_error(error, context()))
}
#[cfg(test)]
mod tests {
use super::escape_html;
use super::render_to_string;
#[test]
fn test_escape_html() {
let tests = vec![
(r"", ""),
(r"a&b", "a&b"),
(r"<a", "<a"),
(r">a", ">a"),
(r#"""#, """),
(r#"'"#, "'"),
(r#"大阪"#, "大阪"),
];
for (input, expected) in tests {
assert_eq!(escape_html(input), expected);
}
let empty = String::new();
assert_eq!(escape_html(&empty), empty);
}
#[test]
fn test_render_to_string() {
use std::io::Write;
let string = render_to_string(|| panic!(), |w| write!(w, "test")).unwrap();
assert_eq!(string, "test".to_owned());
}
}