use proc_macro2;
use syn;
#[derive(Debug, Default)]
pub struct Input {
pub clone: Option<InputClone>,
pub copy: Option<InputCopy>,
pub debug: Option<InputDebug>,
pub default: Option<InputDefault>,
pub eq: Option<InputEq>,
pub hash: Option<InputHash>,
pub partial_eq: Option<InputPartialEq>,
}
#[derive(Debug, Default)]
pub struct Field {
clone: FieldClone,
copy_bound: Option<Vec<syn::WherePredicate>>,
debug: FieldDebug,
default: FieldDefault,
eq_bound: Option<Vec<syn::WherePredicate>>,
hash: FieldHash,
partial_eq: FieldPartialEq,
}
#[derive(Debug, Default)]
pub struct InputClone {
bounds: Option<Vec<syn::WherePredicate>>,
pub clone_from: bool,
}
#[derive(Debug, Default)]
pub struct InputCopy {
bounds: Option<Vec<syn::WherePredicate>>,
}
#[derive(Debug, Default)]
pub struct InputDebug {
bounds: Option<Vec<syn::WherePredicate>>,
pub transparent: bool,
}
#[derive(Debug, Default)]
pub struct InputDefault {
bounds: Option<Vec<syn::WherePredicate>>,
pub new: bool,
}
#[derive(Debug, Default)]
pub struct InputEq {
bounds: Option<Vec<syn::WherePredicate>>,
}
#[derive(Debug, Default)]
pub struct InputHash {
bounds: Option<Vec<syn::WherePredicate>>,
}
#[derive(Debug, Default)]
pub struct InputPartialEq {
bounds: Option<Vec<syn::WherePredicate>>,
on_enum: bool,
}
#[derive(Debug, Default)]
pub struct FieldClone {
bounds: Option<Vec<syn::WherePredicate>>,
clone_with: Option<syn::Path>,
}
#[derive(Debug, Default)]
pub struct FieldDebug {
bounds: Option<Vec<syn::WherePredicate>>,
format_with: Option<syn::Path>,
ignore: bool,
}
#[derive(Debug, Default)]
pub struct FieldDefault {
bounds: Option<Vec<syn::WherePredicate>>,
pub value: Option<proc_macro2::TokenStream>,
}
#[derive(Debug, Default)]
pub struct FieldHash {
bounds: Option<Vec<syn::WherePredicate>>,
hash_with: Option<syn::Path>,
ignore: bool,
}
#[derive(Debug, Default)]
pub struct FieldPartialEq {
bounds: Option<Vec<syn::WherePredicate>>,
compare_with: Option<syn::Path>,
ignore: bool,
}
macro_rules! for_all_attr {
(for ($name:ident, $value:ident) in $attrs:expr; $($body:tt)*) => {
for meta_items in $attrs.iter().filter_map(|attr| derivative_attribute(attr.parse_meta())) {
for meta_item in meta_items.iter().map(read_items) {
let MetaItem($name, $value) = try!(meta_item);
match $name.to_string().as_ref() {
$($body)*
_ => return Err(format!("unknown trait `{}`", $name)),
}
}
}
};
}
macro_rules! match_attributes {
(let Some($name:ident) = $unwrapped:expr; for $value:ident in $values:expr; $($body:tt)* ) => {
let mut $name = $unwrapped.take().unwrap_or_default();
match_attributes! {
for $value in $values;
$($body)*
}
$unwrapped = Some($name);
};
(for $value:ident in $values:expr; $($body:tt)* ) => {
for (name, $value) in $values {
match name {
Some(ident) => {
match ident.to_string().as_ref() {
$($body)*
_ => return Err(format!("unknown attribute `{}`", ident)),
}
}
None => {
match $value.expect("Expected value to be passed").value().as_ref() {
$($body)*
_ => return Err("unknown attribute".to_string()),
}
}
}
}
};
}
impl Input {
pub fn from_ast(attrs: &[syn::Attribute]) -> Result<Input, String> {
let mut input = Input::default();
for_all_attr! {
for (name, values) in attrs;
"Clone" => {
match_attributes! {
let Some(clone) = input.clone;
for value in values;
"bound" => try!(parse_bound(&mut clone.bounds, value)),
"clone_from" => {
clone.clone_from = try!(parse_boolean_meta_item(value, true, "clone_from"));
}
}
}
"Copy" => {
match_attributes! {
let Some(copy) = input.copy;
for value in values;
"bound" => try!(parse_bound(&mut copy.bounds, value)),
}
}
"Debug" => {
match_attributes! {
let Some(debug) = input.debug;
for value in values;
"bound" => try!(parse_bound(&mut debug.bounds, value)),
"transparent" => {
debug.transparent = try!(parse_boolean_meta_item(value, true, "transparent"));
}
}
}
"Default" => {
match_attributes! {
let Some(default) = input.default;
for value in values;
"bound" => try!(parse_bound(&mut default.bounds, value)),
"new" => {
default.new = try!(parse_boolean_meta_item(value, true, "new"));
}
}
}
"Eq" => {
match_attributes! {
let Some(eq) = input.eq;
for value in values;
"bound" => try!(parse_bound(&mut eq.bounds, value)),
}
}
"Hash" => {
match_attributes! {
let Some(hash) = input.hash;
for value in values;
"bound" => try!(parse_bound(&mut hash.bounds, value)),
}
}
"PartialEq" => {
match_attributes! {
let Some(partial_eq) = input.partial_eq;
for value in values;
"bound" => try!(parse_bound(&mut partial_eq.bounds, value)),
"feature_allow_slow_enum" => {
partial_eq.on_enum = try!(parse_boolean_meta_item(value, true, "feature_allow_slow_enum"));
}
}
}
}
Ok(input)
}
pub fn clone_bound(&self) -> Option<&[syn::WherePredicate]> {
self.clone
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn clone_from(&self) -> bool {
self.clone.as_ref().map_or(false, |d| d.clone_from)
}
pub fn copy_bound(&self) -> Option<&[syn::WherePredicate]> {
self.copy
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn debug_bound(&self) -> Option<&[syn::WherePredicate]> {
self.debug
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn debug_transparent(&self) -> bool {
self.debug.as_ref().map_or(false, |d| d.transparent)
}
pub fn default_bound(&self) -> Option<&[syn::WherePredicate]> {
self.default
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn eq_bound(&self) -> Option<&[syn::WherePredicate]> {
self.eq
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn hash_bound(&self) -> Option<&[syn::WherePredicate]> {
self.hash
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn partial_eq_bound(&self) -> Option<&[syn::WherePredicate]> {
self.partial_eq
.as_ref()
.and_then(|d| d.bounds.as_ref().map(Vec::as_slice))
}
pub fn partial_eq_on_enum(&self) -> bool {
self.partial_eq.as_ref().map_or(false, |d| d.on_enum)
}
}
impl Field {
pub fn from_ast(field: &syn::Field) -> Result<Field, String> {
let mut out = Field::default();
for_all_attr! {
for (name, values) in field.attrs;
"Clone" => {
match_attributes! {
for value in values;
"bound" => try!(parse_bound(&mut out.clone.bounds, value)),
"clone_with" => {
let path = try!(value.ok_or_else(|| "`clone_with` needs a value".to_string()));
out.clone.clone_with = Some(try!(parse_str_lit(&path)));
}
}
}
"Debug" => {
match_attributes! {
for value in values;
"bound" => try!(parse_bound(&mut out.debug.bounds, value)),
"format_with" => {
let path = try!(value.ok_or_else(|| "`format_with` needs a value".to_string()));
out.debug.format_with = Some(try!(parse_str_lit(&path)));
}
"ignore" => {
out.debug.ignore = try!(parse_boolean_meta_item(value, true, "ignore"));
}
}
}
"Default" => {
match_attributes! {
for value in values;
"bound" => try!(parse_bound(&mut out.default.bounds, value)),
"value" => {
let value = try!(value.ok_or_else(|| "`value` needs a value".to_string()));
out.default.value = Some(try!(parse_str_lit(&value)));
}
}
}
"Eq" => {
match_attributes! {
for value in values;
"bound" => try!(parse_bound(&mut out.eq_bound, value)),
}
}
"Hash" => {
match_attributes! {
for value in values;
"bound" => try!(parse_bound(&mut out.hash.bounds, value)),
"hash_with" => {
let path = try!(value.ok_or_else(|| "`hash_with` needs a value".to_string()));
out.hash.hash_with = Some(try!(parse_str_lit(&path)));
}
"ignore" => {
out.hash.ignore = try!(parse_boolean_meta_item(value, true, "ignore"));
}
}
}
"PartialEq" => {
match_attributes! {
for value in values;
"bound" => try!(parse_bound(&mut out.partial_eq.bounds, value)),
"compare_with" => {
let path = try!(value.ok_or_else(|| "`compare_with` needs a value".to_string()));
out.partial_eq.compare_with = Some(try!(parse_str_lit(&path)));
}
"ignore" => {
out.partial_eq.ignore = try!(parse_boolean_meta_item(value, true, "ignore"));
}
}
}
}
Ok(out)
}
pub fn clone_bound(&self) -> Option<&[syn::WherePredicate]> {
self.clone.bounds.as_ref().map(Vec::as_slice)
}
pub fn clone_with(&self) -> Option<&syn::Path> {
self.clone.clone_with.as_ref()
}
pub fn copy_bound(&self) -> Option<&[syn::WherePredicate]> {
self.copy_bound.as_ref().map(Vec::as_slice)
}
pub fn debug_bound(&self) -> Option<&[syn::WherePredicate]> {
self.debug.bounds.as_ref().map(Vec::as_slice)
}
pub fn debug_format_with(&self) -> Option<&syn::Path> {
self.debug.format_with.as_ref()
}
pub fn ignore_debug(&self) -> bool {
self.debug.ignore
}
pub fn ignore_hash(&self) -> bool {
self.hash.ignore
}
pub fn default_bound(&self) -> Option<&[syn::WherePredicate]> {
self.default.bounds.as_ref().map(Vec::as_slice)
}
pub fn default_value(&self) -> Option<&proc_macro2::TokenStream> {
self.default.value.as_ref()
}
pub fn eq_bound(&self) -> Option<&[syn::WherePredicate]> {
self.eq_bound.as_ref().map(Vec::as_slice)
}
pub fn hash_bound(&self) -> Option<&[syn::WherePredicate]> {
self.hash.bounds.as_ref().map(Vec::as_slice)
}
pub fn hash_with(&self) -> Option<&syn::Path> {
self.hash.hash_with.as_ref()
}
pub fn partial_eq_bound(&self) -> Option<&[syn::WherePredicate]> {
self.partial_eq.bounds.as_ref().map(Vec::as_slice)
}
pub fn partial_eq_compare_with(&self) -> Option<&syn::Path> {
self.partial_eq.compare_with.as_ref()
}
pub fn ignore_partial_eq(&self) -> bool {
self.partial_eq.ignore
}
}
struct MetaItem<'a>(
&'a syn::Ident,
Vec<(Option<&'a syn::Ident>, Option<&'a syn::LitStr>)>,
);
fn read_items(item: &syn::NestedMeta) -> Result<MetaItem, String> {
let item = match *item {
syn::NestedMeta::Meta(ref item) => item,
syn::NestedMeta::Literal(..) => {
return Err("Expected meta-item but found literal".to_string());
}
};
match *item {
syn::Meta::Word(ref name) => Ok(MetaItem(name, Vec::new())),
syn::Meta::List(syn::MetaList {
ident: ref name,
nested: ref values,
..
}) => {
let values = try!(values
.iter()
.map(|value| {
if let syn::NestedMeta::Meta(syn::Meta::NameValue(syn::MetaNameValue {
ident: ref name,
lit: ref value,
..
})) = *value
{
let value = try!(ensure_str_lit(&name.to_string(), value));
Ok((Some(name), Some(value)))
} else {
Err("Expected named value".to_string())
}
})
.collect());
Ok(MetaItem(name, values))
}
syn::Meta::NameValue(syn::MetaNameValue {
ident: ref name,
lit: ref value,
..
}) => {
let value = try!(ensure_str_lit(&name.to_string(), value));
Ok(MetaItem(name, vec![(None, Some(value))]))
}
}
}
fn derivative_attribute(
meta: syn::parse::Result<syn::Meta>,
) -> Option<syn::punctuated::Punctuated<syn::NestedMeta, syn::token::Comma>> {
match meta {
Ok(syn::Meta::List(syn::MetaList {
ident: name,
nested: mis,
..
})) => {
if name == "derivative" {
Some(mis)
} else {
None
}
}
_ => None,
}
}
fn parse_boolean_meta_item(
item: Option<&syn::LitStr>,
default: bool,
name: &str,
) -> Result<bool, String> {
let item = item.map(|item| item.value());
match item.as_ref().map(|item| item.as_ref()) {
Some("true") => Ok(true),
Some("false") => Ok(false),
Some(val) => {
if val == name {
Ok(true)
} else {
Err(format!("Invalid value for `{}`: `{}`", name, val))
}
}
None => Ok(default),
}
}
fn parse_bound(
opt_bounds: &mut Option<Vec<syn::WherePredicate>>,
value: Option<&syn::LitStr>,
) -> Result<(), String> {
let bound = try!(value.ok_or_else(|| "`bound` needs a value".to_string()));
let bound_value = bound.value();
*opt_bounds = if !bound_value.is_empty() {
let where_string = syn::LitStr::new(&format!("where {}", bound_value), bound.span());
let bounds = parse_str_lit::<syn::WhereClause>(&where_string)
.map(|wh| wh.predicates.into_iter().collect())
.map_err(|_| "Could not parse `bound`".to_string());
Some(try!(bounds))
} else {
Some(vec![])
};
Ok(())
}
fn parse_str_lit<T>(value: &syn::LitStr) -> Result<T, String>
where
T: syn::parse::Parse,
{
value.parse().map_err(|e| e.to_string())
}
fn ensure_str_lit<'a>(attr_name: &str, lit: &'a syn::Lit) -> Result<&'a syn::LitStr, String> {
if let syn::Lit::Str(ref lit) = *lit {
Ok(lit)
} else {
Err(format!(
"expected derivative {} attribute to be a string: `{} = \"...\"`",
attr_name, attr_name
))
}
}