Как предоставить полезные ошибки компилятора в процедурном макросе?
Я разрабатываю собственный синтаксический анализатор HTML, используя proc_macro
а также syn
, Образец:
#[derive(Debug)]
struct BlockElement {
stag: Ident,
child: Vec<Element>,
ctag: Ident
}
impl Synom for BlockElement {
named!(parse -> Self, do_parse!(
punct!(<) >>
stag: syn!(Ident) >>
punct!(>) >>
child: syn!(ElementList) >>
punct!(<) >>
punct!(/) >>
ctag: syn!(Ident) >>
punct!(>) >>
(BlockElement { stag, child: child.inner, ctag })
));
}
Хотя я знаю, как выдавать ошибки, используя Span
после анализа я не могу понять, как это сделать во время анализа. Это просто ошибки с failed to parse anything
, Как точно определить, где произошел сбой, и дать соответствующую ошибку?
1 ответ
Обновив ваш макрос для использования Syn 0.15 или выше, вы получите полезные сообщения об ошибках анализа без дополнительных усилий.
extern crate proc_macro;
use self::proc_macro::TokenStream;
use syn::parse::{Parse, ParseStream, Result};
use syn::{parse_macro_input, Ident, Token};
// FIXME
type Element = Ident;
struct ElementList {
inner: Vec<Element>,
}
impl Parse for ElementList {
fn parse(input: ParseStream) -> Result<Self> {
let mut list = Vec::new();
while let Some(element) = input.parse()? {
list.push(element);
}
Ok(ElementList { inner: list })
}
}
struct BlockElement {
stag: Ident,
child: Vec<Element>,
ctag: Ident
}
impl Parse for BlockElement {
fn parse(input: ParseStream) -> Result<Self> {
input.parse::<Token![<]>()?;
let stag: Ident = input.parse()?;
input.parse::<Token![>]>()?;
let child: ElementList = input.parse()?;
input.parse::<Token![<]>()?;
input.parse::<Token![/]>()?;
let ctag: Ident = input.parse()?;
input.parse::<Token![>]>()?;
Ok(BlockElement { stag, child: child.inner, ctag })
}
}
#[proc_macro]
pub fn html_syntax(input: TokenStream) -> TokenStream {
let _input = parse_macro_input!(input as BlockElement);
// TODO
TokenStream::new()
}
Вот пример ошибки разбора ввода. Такие ошибки будут указывать, где входные данные не удалось проанализировать и какой токен (-ы) ожидал макрос в этой позиции.
fn main() {
html_syntax!(<em>syn quote proc_macro2<em>);
}
error: expected `/`
--> src/main.rs
|
| html_syntax!(<em>syn quote proc_macro2<em>);
| ^^