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
use std::str;
macro_rules! escaping_body {
($start:ident, $i:ident, $fmt:ident, $bytes:ident, $quote:expr) => {{
if $start < $i {
$fmt.push_str(unsafe { str::from_utf8_unchecked(&$bytes[$start..$i]) });
}
$fmt.push_str($quote);
$start = $i + 1;
}};
}
const FLAG: u8 = b'>' - b'"';
#[inline]
pub fn escape_html(input: &str) -> String {
let mut start = 0;
let mut output = String::with_capacity(input.len() + input.len() / 2);
let bytes = input.as_bytes();
for (i, b) in bytes.iter().enumerate() {
if b.wrapping_sub(b'"') <= FLAG {
match *b {
b'<' => escaping_body!(start, i, output, bytes, "<"),
b'>' => escaping_body!(start, i, output, bytes, ">"),
b'&' => escaping_body!(start, i, output, bytes, "&"),
b'"' => escaping_body!(start, i, output, bytes, """),
b'\'' => escaping_body!(start, i, output, bytes, "'"),
b'/' => escaping_body!(start, i, output, bytes, "/"),
_ => (),
}
}
}
output.push_str(unsafe { str::from_utf8_unchecked(&bytes[start..]) });
output
}
#[cfg(test)]
mod tests {
use super::escape_html;
#[test]
fn test_escape_html() {
let tests = vec![
(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);
}
}
}