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
120
121
122
123
use proc_macro::TokenStream;
use quote::quote;
use syn::{parse_macro_input, spanned::Spanned, Data, DeriveInput, Error, Result};
#[proc_macro_derive(LevelDBOrm, attributes(level_db_key))]
pub fn derive(input: TokenStream) -> TokenStream {
let input = parse_macro_input!(input as DeriveInput);
derive_orm(input).map_or_else(|e| TokenStream::from(e.to_compile_error()), |r| r)
}
fn derive_orm(input: DeriveInput) -> Result<TokenStream> {
let span = input.span();
let fields = if let Data::Struct(data_struct) = input.data {
data_struct.fields
} else {
return Err(Error::new(span, "LevelDBOrm macro for Struct only"));
};
let keys = parse::parse_level_db_key(&input.attrs)
.ok_or_else(|| Error::new(span, "Need attributr: `#[level_db_key(key1, key2, ...)]`"))?;
let key_types = parse::parse_key_types(&keys, &fields)?;
let ident = input.ident;
let res = quote!(
impl<'a> leveldb_orm::KeyOrm<'a> for #ident {
type KeyType = (#(#key_types,)*);
type KeyTypeRef = (#(&'a #key_types,)*);
#[inline]
fn key(&self) -> std::result::Result<leveldb_orm::EncodedKey<Self>, Box<dyn std::error::Error>> {
Self::encode_key((#(&self.#keys,)*))
}
}
);
Ok(res.into())
}
mod parse {
use syn::{
parse_quote, Attribute, Error, Fields, Ident, Meta, MetaList, NestedMeta, Path, Result,
Type,
};
pub fn parse_level_db_key(attrs: &[Attribute]) -> Option<Vec<Ident>> {
attrs.iter().find_map(|attr| {
attr.parse_meta().ok().and_then(|meta| {
let path_name: Path = parse_quote!(level_db_key);
if meta.path() == &path_name {
if let Meta::List(MetaList { nested, .. }) = meta {
let keys = nested.iter().filter_map(parse_key).collect::<Vec<_>>();
if keys.is_empty() {
None
} else {
Some(keys)
}
} else {
None
}
} else {
None
}
})
})
}
fn parse_key(nested_meta: &NestedMeta) -> Option<Ident> {
if let NestedMeta::Meta(Meta::Path(Path { segments, .. })) = nested_meta {
segments.first().map(|seg| seg.ident.clone())
} else {
None
}
}
pub fn parse_key_types(keys: &[Ident], fields: &Fields) -> Result<Vec<Type>> {
let mut res = vec![];
for key in keys {
let ty = fields
.iter()
.find_map(|field| {
field
.ident
.as_ref()
.filter(|ident| ident == &key)
.map(|_| field.ty.clone())
})
.ok_or_else(|| {
Error::new(
key.span(),
format!("level_db_key: \"{key}\" not found in struct.",),
)
})?;
res.push(ty);
}
Ok(res)
}
}