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
use std::borrow::Cow;
use xmlparser::{ElementEnd, Token};
use crate::{xml_unescape, XmlError, XmlReader, XmlResult};
pub fn read_text<'a>(reader: &mut XmlReader<'a>, tag: &'a str) -> XmlResult<Cow<'a, str>> {
let mut res = None;
while let Some(token) = reader.next() {
match token? {
Token::ElementEnd {
end: ElementEnd::Open,
..
}
| Token::Attribute { .. } => (),
Token::Text { text } => {
res = Some(xml_unescape(text.as_str())?);
}
Token::ElementEnd {
end: ElementEnd::Close(_, _),
span,
} => {
let span = span.as_str();
if tag == &span[2..span.len() - 1] {
break;
} else {
return Err(XmlError::TagMismatch {
expected: tag.to_owned(),
found: span[2..span.len() - 1].to_owned(),
});
}
}
token => {
return Err(XmlError::UnexpectedToken {
token: format!("{:?}", token),
});
}
}
}
Ok(res.unwrap_or_default())
}
#[test]
fn test_read_text() -> XmlResult<()> {
use xmlparser::Tokenizer;
let mut reader = Tokenizer::from("<parent></parent>").peekable();
assert!(reader.next().is_some());
assert_eq!(read_text(&mut reader, "parent")?, "");
assert!(reader.next().is_none());
let mut reader = Tokenizer::from("<parent>text</parent>").peekable();
assert!(reader.next().is_some());
assert_eq!(read_text(&mut reader, "parent")?, "text");
assert!(reader.next().is_none());
let mut reader = Tokenizer::from("<parent attr=\"value\">text</parent>").peekable();
assert!(reader.next().is_some());
assert_eq!(read_text(&mut reader, "parent")?, "text");
assert!(reader.next().is_none());
Ok(())
}