darling
is a crate for proc macro authors, which enables parsing attributes into structs. It is heavily inspired by serde
both in its internals and in its API.
darling
makes sure they get error markers at the right place in their source, and provides “did you mean” suggestions for misspelled fields.darling
provides a set of traits which can be derived or manually implemented.
FromMeta
is used to extract values from a meta-item in an attribute. Implementations are likely reusable for many libraries, much like FromStr
or serde::Deserialize
. Trait implementations are provided for primitives, some std types, and some syn
types.FromDeriveInput
is implemented or derived by each proc-macro crate which depends on darling
. This is the root for input parsing; it gets access to the identity, generics, and visibility of the target type, and can specify which attribute names should be parsed or forwarded from the input AST.FromField
is implemented or derived by each proc-macro crate which depends on darling
. Structs deriving this trait will get access to the identity (if it exists), type, and visibility of the field.FromVariant
is implemented or derived by each proc-macro crate which depends on darling
. Structs deriving this trait will get access to the identity and contents of the variant, which can be transformed the same as any other darling
input.darling::ast
provides generic types for representing the AST.darling::usage
provides traits and functions for determining where type parameters and lifetimes are used in a struct or enum.darling::util
provides helper types with special FromMeta
implementations, such as IdentList
.#[macro_use] extern crate darling; extern crate syn; #[derive(Default, FromMeta)] #[darling(default)] pub struct Lorem { #[darling(rename = "sit")] ipsum: bool, dolor: Option<String>, } #[derive(FromDeriveInput)] #[darling(from_ident, attributes(my_crate), forward_attrs(allow, doc, cfg))] pub struct MyTraitOpts { ident: syn::Ident, attrs: Vec<syn::Attribute>, lorem: Lorem, }
The above code will then be able to parse this input:
/// A doc comment which will be available in `MyTraitOpts::attrs`. #[derive(MyTrait)] #[my_crate(lorem(dolor = "Hello", ipsum))] pub struct ConsumingType;
Non-derive attribute macros are supported. To parse arguments for attribute macros, derive FromMeta
on the argument receiver type, then pass &syn::AttributeArgs
to the from_list
method. This will produce a normal darling::Result<T>
that can be used the same as a result from parsing a DeriveInput
.
use darling::FromMeta; use syn::{AttributeArgs, ItemFn}; use proc_macro::TokenStream; #[derive(Debug, FromMeta)] pub struct MacroArgs { #[darling(default)] timeout_ms: Option<u16>, path: String, } #[proc_macro_attribute] fn your_attr(args: TokenStream, input: TokenStream) -> TokenStream { let attr_args = parse_macro_input!(args as AttributeArgs); let _input = parse_macro_input!(input as ItemFn); let _args = match MacroArgs::from_list(&attr_args) { Ok(v) => v, Err(e) => { return TokenStream::from(e.write_errors()); } }; // do things with `args` unimplemented!() }
use your_crate::your_attr; #[your_attr(path = "hello", timeout_ms = 15)] fn do_stuff() { println!("Hello"); }
Darling's features are built to work well for real-world projects.
serde
.FromDeriveInput
and FromField
can declare properties named ident
, vis
, ty
, attrs
, and generics
to automatically get copies of the matching values from the input AST. FromDeriveInput
additionally exposes data
to get access to the body of the deriving type, and FromVariant
exposes fields
.#[darling(map="path")]
or #[darling(and_then="path")]
to specify a function that runs on the result of parsing a meta-item field. This can change the return type, which enables you to parse to an intermediate form and convert that to the type you need in your struct.#[darling(skip)]
to mark a field that shouldn't be read from attribute meta-items.#[darling(multiple)]
on a Vec
field to allow that field to appear multiple times in the meta-item. Each occurrence will be pushed into the Vec
.darling::util::SpannedValue
in a struct to get access to that meta item's source code span. This can be used to emit warnings that point at a specific field from your proc macro. In addition, you can use darling::Error::write_errors
to automatically get precise error location details in most cases.