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
#![cfg_attr(not(feature = "default"), allow(dead_code))]
#[cfg(feature = "Clone")]
pub mod clone;
#[cfg(feature = "Copy")]
pub mod copy;
#[cfg(feature = "Debug")]
pub mod debug;
#[cfg(feature = "Default")]
pub mod default;
#[cfg(feature = "Deref")]
pub mod deref;
#[cfg(feature = "DerefMut")]
pub mod deref_mut;
#[cfg(feature = "Eq")]
pub mod eq;
#[cfg(feature = "Hash")]
pub mod hash;
#[cfg(feature = "Ord")]
pub mod ord;
#[cfg(feature = "PartialEq")]
pub mod partial_eq;
#[cfg(feature = "PartialOrd")]
pub mod partial_ord;
use std::str::FromStr;
use crate::Trait;
use proc_macro2::TokenStream;
use quote::{quote, ToTokens};
use syn::{
self, punctuated::Punctuated, token::Comma, DeriveInput, Expr, GenericParam, LitStr, Meta,
Path, WhereClause, WherePredicate,
};
pub trait TraitHandler {
fn trait_meta_handler(
ast: &DeriveInput,
tokens: &mut TokenStream,
traits: &[Trait],
meta: &Meta,
);
}
#[inline]
pub fn create_path_from_lit_str(s: &LitStr) -> Option<Path> {
let s = s.value();
let s = s.trim();
if s.is_empty() {
None
} else {
let tokens = TokenStream::from_str(s).unwrap();
Some(syn::parse2(tokens).unwrap())
}
}
#[inline]
pub fn create_path_string_from_lit_str(s: &LitStr) -> Option<String> {
create_path_from_lit_str(s).map(|path| path.into_token_stream().to_string().replace(' ', ""))
}
#[inline]
pub fn create_expr_from_lit_str(s: &LitStr) -> Option<Expr> {
let s = s.value();
let s = s.trim();
if s.is_empty() {
None
} else {
let tokens = TokenStream::from_str(s).unwrap();
Some(syn::parse2(tokens).unwrap())
}
}
#[inline]
pub fn create_expr_string_from_lit_str(s: &LitStr) -> Option<String> {
create_expr_from_lit_str(s).map(|expr| expr.into_token_stream().to_string().replace(' ', ""))
}
#[inline]
pub fn create_where_predicates_from_lit_str(
s: &LitStr,
) -> Option<Punctuated<WherePredicate, Comma>> {
let s = s.value();
let s = s.trim();
if s.is_empty() {
None
} else {
let s = format!("where {}", s);
let tokens = TokenStream::from_str(&s).unwrap();
let where_clause: WhereClause = syn::parse2(tokens).unwrap();
Some(where_clause.predicates)
}
}
#[inline]
pub fn create_where_predicates_from_generic_parameters(
p: &Punctuated<GenericParam, Comma>,
bound_trait: &Path,
) -> Punctuated<WherePredicate, Comma> {
let mut where_predicates = Punctuated::new();
for param in p.iter() {
if let GenericParam::Type(typ) = param {
let ident = &typ.ident;
where_predicates.push(syn::parse2(quote! { #ident: #bound_trait }).unwrap());
}
}
where_predicates
}