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
|
use wast::parser::{self, Parse, ParseBuffer, Parser, Result};
pub struct Comments<'a> {
comments: Vec<&'a str>,
}
impl<'a> Parse<'a> for Comments<'a> {
fn parse(parser: Parser<'a>) -> Result<Comments<'a>> {
let comments = parser.step(|mut cursor| {
let mut comments = Vec::new();
loop {
let (comment, c) = match cursor.comment()? {
Some(pair) => pair,
None => break,
};
cursor = c;
comments.push(if comment.starts_with(";;") {
&comment[2..]
} else {
&comment[2..comment.len() - 2]
});
}
Ok((comments, cursor))
})?;
Ok(Comments { comments })
}
}
pub struct Documented<'a, T> {
comments: Comments<'a>,
item: T,
}
impl<'a, T: Parse<'a>> Parse<'a> for Documented<'a, T> {
fn parse(parser: Parser<'a>) -> Result<Self> {
let comments = parser.parse()?;
let item = parser.parens(T::parse)?;
Ok(Documented { comments, item })
}
}
#[test]
fn parse_comments() -> anyhow::Result<()> {
let buf = ParseBuffer::new(
r#"
;; hello
(; again ;)
(module)
"#,
)?;
let d: Documented<wast::core::Module> = parser::parse(&buf)?;
assert_eq!(d.comments.comments, vec![" hello", " again "]);
drop(d.item);
let buf = ParseBuffer::new(
r#"
;; this
(; is
on
multiple;)
;; lines
(func)
"#,
)?;
let d: Documented<wast::core::Func> = parser::parse(&buf)?;
assert_eq!(
d.comments.comments,
vec![" this", " is\non\nmultiple", " lines"]
);
drop(d.item);
Ok(())
}
|