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
#![crate_type="dylib"]
extern crate regex;
extern crate unicode_names2;
#[macro_use]
extern crate syn;
extern crate proc_macro;
struct CharByName(syn::LitChar);
impl syn::parse::Parse for CharByName {
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
let string: syn::LitStr = input.parse()?;
let name = string.value();
match unicode_names2::character(&name) {
None => Err(syn::Error::new(string.span(), format!("`{}` does not name a character", name))),
Some(c) => Ok(CharByName(syn::LitChar::new(c, string.span()))),
}
}
}
#[proc_macro]
pub fn named_char(stream: proc_macro::TokenStream) -> proc_macro::TokenStream {
let CharByName(c) = parse_macro_input!(stream as CharByName);
proc_macro::TokenTree::Literal(proc_macro::Literal::character(c.value())).into()
}
struct StringWithCharNames(syn::LitStr);
impl syn::parse::Parse for StringWithCharNames {
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
let string: syn::LitStr = input.parse()?;
let names_re = regex::Regex::new(r"\\N\{(.*?)(?:\}|$)").unwrap();
let mut errors = Vec::new();
let new = names_re.replace_all(&string.value(), |c: ®ex::Captures| {
let full = c.at(0).unwrap();
if !full.ends_with("}") {
errors.push(format!("unclosed escape in `named!`: {}", full));
} else {
let name = c.at(1).unwrap();
match unicode_names2::character(name) {
Some(c) => {
return c.to_string()
},
None => {
errors.push(format!("`{}` does not name a character", name));
}
}
}
String::new()
});
if errors.len() > 0 {
Err(syn::Error::new(string.span(), errors.get(0).unwrap()))
}
else {
Ok(StringWithCharNames(syn::LitStr::new(&new, string.span())))
}
}
}
#[proc_macro]
pub fn named(stream: proc_macro::TokenStream) -> proc_macro::TokenStream {
let StringWithCharNames(s) = parse_macro_input!(stream as StringWithCharNames);
proc_macro::TokenTree::Literal(proc_macro::Literal::string(&s.value())).into()
}