| use proc_macro::{Delimiter, Group, Ident, Literal, Punct, Spacing, Span, TokenStream, TokenTree}; |
| use std::fmt::Display; |
| use std::iter::FromIterator; |
| |
| pub type Result<T, E = Error> = std::result::Result<T, E>; |
| |
| pub struct Error { |
| begin: Span, |
| end: Span, |
| msg: String, |
| } |
| |
| impl Error { |
| pub fn new(span: Span, msg: impl Display) -> Self { |
| Self::new2(span, span, msg) |
| } |
| |
| pub fn new2(begin: Span, end: Span, msg: impl Display) -> Self { |
| Error { |
| begin, |
| end, |
| msg: msg.to_string(), |
| } |
| } |
| |
| pub fn group(group: Group, msg: impl Display) -> Self { |
| let mut iter = group.stream().into_iter(); |
| let delimiter = group.span(); |
| let begin = iter.next().map_or(delimiter, |t| t.span()); |
| let end = iter.last().map_or(begin, |t| t.span()); |
| Self::new2(begin, end, msg) |
| } |
| |
| pub fn into_compile_error(self) -> TokenStream { |
| // compile_error! { $msg } |
| TokenStream::from_iter(vec![ |
| TokenTree::Ident(Ident::new("compile_error", self.begin)), |
| TokenTree::Punct({ |
| let mut punct = Punct::new('!', Spacing::Alone); |
| punct.set_span(self.begin); |
| punct |
| }), |
| TokenTree::Group({ |
| let mut group = Group::new(Delimiter::Brace, { |
| TokenStream::from_iter(vec![TokenTree::Literal({ |
| let mut string = Literal::string(&self.msg); |
| string.set_span(self.end); |
| string |
| })]) |
| }); |
| group.set_span(self.end); |
| group |
| }), |
| ]) |
| } |
| } |