1use std::borrow::Cow;
2use std::fmt;
3
4use proc_macro::{Delimiter, Group, Ident, Literal, Punct, Spacing, Span, TokenStream, TokenTree};
5
6trait WithSpan {
7 fn with_span(self, span: Span) -> Self;
8}
9
10impl WithSpan for TokenTree {
11 fn with_span(mut self, span: Span) -> Self {
12 self.set_span(span);
13 self
14 }
15}
16
17pub(crate) enum Error {
18 MissingComponent {
19 name: &'static str,
20 span_start: Option<Span>,
21 span_end: Option<Span>,
22 },
23 InvalidComponent {
24 name: &'static str,
25 value: String,
26 span_start: Option<Span>,
27 span_end: Option<Span>,
28 },
29 #[cfg(any(feature = "formatting", feature = "parsing"))]
30 ExpectedString {
31 span_start: Option<Span>,
32 span_end: Option<Span>,
33 },
34 UnexpectedToken {
35 tree: TokenTree,
36 },
37 UnexpectedEndOfInput,
38 Custom {
39 message: Cow<'static, str>,
40 span_start: Option<Span>,
41 span_end: Option<Span>,
42 },
43}
44
45impl fmt::Display for Error {
46 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
47 match self {
48 Self::MissingComponent { name, .. } => write!(f, "missing component: {name}"),
49 Self::InvalidComponent { name, value, .. } => {
50 write!(f, "invalid component: {name} was {value}")
51 }
52 #[cfg(any(feature = "formatting", feature = "parsing"))]
53 Self::ExpectedString { .. } => f.write_str("expected string literal"),
54 Self::UnexpectedToken { tree } => write!(f, "unexpected token: {tree}"),
55 Self::UnexpectedEndOfInput => f.write_str("unexpected end of input"),
56 Self::Custom { message, .. } => f.write_str(message),
57 }
58 }
59}
60
61impl Error {
62 fn span_start(&self) -> Span {
63 match self {
64 Self::MissingComponent { span_start, .. }
65 | Self::InvalidComponent { span_start, .. }
66 | Self::Custom { span_start, .. } => *span_start,
67 #[cfg(any(feature = "formatting", feature = "parsing"))]
68 Self::ExpectedString { span_start, .. } => *span_start,
69 Self::UnexpectedToken { tree } => Some(tree.span()),
70 Self::UnexpectedEndOfInput => Some(Span::mixed_site()),
71 }
72 .unwrap_or_else(Span::mixed_site)
73 }
74
75 fn span_end(&self) -> Span {
76 match self {
77 Self::MissingComponent { span_end, .. }
78 | Self::InvalidComponent { span_end, .. }
79 | Self::Custom { span_end, .. } => *span_end,
80 #[cfg(any(feature = "formatting", feature = "parsing"))]
81 Self::ExpectedString { span_end, .. } => *span_end,
82 Self::UnexpectedToken { tree, .. } => Some(tree.span()),
83 Self::UnexpectedEndOfInput => Some(Span::mixed_site()),
84 }
85 .unwrap_or_else(|| self.span_start())
86 }
87
88 pub(crate) fn to_compile_error(&self) -> TokenStream {
89 let (start, end) = (self.span_start(), self.span_end());
90
91 [
92 TokenTree::from(Punct::new(':', Spacing::Joint)).with_span(start),
93 TokenTree::from(Punct::new(':', Spacing::Alone)).with_span(start),
94 TokenTree::from(Ident::new("core", start)),
95 TokenTree::from(Punct::new(':', Spacing::Joint)).with_span(start),
96 TokenTree::from(Punct::new(':', Spacing::Alone)).with_span(start),
97 TokenTree::from(Ident::new("compile_error", start)),
98 TokenTree::from(Punct::new('!', Spacing::Alone)).with_span(start),
99 TokenTree::from(Group::new(
100 Delimiter::Parenthesis,
101 TokenStream::from(
102 TokenTree::from(Literal::string(&self.to_string())).with_span(end),
103 ),
104 ))
105 .with_span(end),
106 ]
107 .iter()
108 .cloned()
109 .collect()
110 }
111
112 #[cfg(all(feature = "serde", any(feature = "formatting", feature = "parsing")))]
114 pub(crate) fn to_compile_error_standalone(&self) -> TokenStream {
115 let end = self.span_end();
116 self.to_compile_error()
117 .into_iter()
118 .chain(std::iter::once(
119 TokenTree::from(Punct::new(';', Spacing::Alone)).with_span(end),
120 ))
121 .collect()
122 }
123}