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 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
use std::fmt; use beef::lean::Cow; use quote::quote; use proc_macro2::{Span, TokenStream}; use quote::{quote_spanned, ToTokens, TokenStreamExt}; pub type Result<T> = std::result::Result<T, Error>; pub struct Errors { collected: Vec<SpannedError>, } impl Default for Errors { fn default() -> Self { Errors { collected: Vec::new(), } } } impl Errors { pub fn err<M>(&mut self, message: M, span: Span) -> &mut Self where M: Into<Cow<'static, str>>, { self.collected.push(SpannedError { message: message.into(), span, }); self } pub fn render(self) -> Option<TokenStream> { let errors = self.collected; match errors.len() { 0 => None, _ => Some(quote! { fn _logos_derive_compile_errors() { #(#errors)* } }) } } } pub struct Error(Cow<'static, str>); #[derive(Debug)] pub struct SpannedError { message: Cow<'static, str>, span: Span, } impl Error { pub fn new<M>(message: M) -> Self where M: Into<Cow<'static, str>>, { Error(message.into()) } pub fn span(self, span: Span) -> SpannedError { SpannedError { message: self.0, span, } } } impl fmt::Display for Error { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { self.0.fmt(f) } } impl fmt::Debug for Error { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Display::fmt(self, f) } } impl From<regex_syntax::Error> for Error { fn from(err: regex_syntax::Error) -> Error { Error(err.to_string().into()) } } impl From<&'static str> for Error { fn from(err: &'static str) -> Error { Error(err.into()) } } impl From<String> for Error { fn from(err: String) -> Error { Error(err.into()) } } impl From<Error> for Cow<'static, str> { fn from(err: Error) -> Self { err.0 } } impl ToTokens for SpannedError { fn to_tokens(&self, tokens: &mut TokenStream) { let message = &*self.message; tokens.append_all( quote_spanned!(self.span => { compile_error!(#message) }) ) } }