1use syn::spanned::Spanned;
2use syn::{Field, Ident, Meta};
34use crate::codegen::ForwardAttrs;
5use crate::options::{
6 AttrsField, Core, DefaultExpression, ForwardAttrsFilter, ParseAttribute, ParseData,
7};
8use crate::util::PathList;
9use crate::{FromField, FromMeta, Result};
1011/// Reusable base for `FromDeriveInput`, `FromVariant`, `FromField`, and other top-level
12/// `From*` traits.
13#[derive(Debug, Clone)]
14pub struct OuterFrom {
15/// The field on the target struct which should receive the type identifier, if any.
16pub ident: Option<Ident>,
1718/// The field on the target struct which should receive the type attributes, if any.
19pub attrs: Option<AttrsField>,
2021pub container: Core,
2223/// The attribute names that should be searched.
24pub attr_names: PathList,
2526/// The attribute names that should be forwarded. The presence of the word with no additional
27 /// filtering will cause _all_ attributes to be cloned and exposed to the struct after parsing.
28pub forward_attrs: Option<ForwardAttrsFilter>,
2930/// Whether or not the container can be made through conversion from the type `Ident`.
31pub from_ident: bool,
32}
3334impl OuterFrom {
35pub fn start(di: &syn::DeriveInput) -> Result<Self> {
36Ok(OuterFrom {
37 container: Core::start(di)?,
38 attrs: Default::default(),
39 ident: Default::default(),
40 attr_names: Default::default(),
41 forward_attrs: Default::default(),
42 from_ident: Default::default(),
43 })
44 }
4546pub fn as_forward_attrs(&self) -> ForwardAttrs<'_> {
47 ForwardAttrs {
48 field: self.attrs.as_ref(),
49 filter: self.forward_attrs.as_ref(),
50 }
51 }
52}
5354impl ParseAttribute for OuterFrom {
55fn parse_nested(&mut self, mi: &Meta) -> Result<()> {
56let path = mi.path();
57if path.is_ident("attributes") {
58self.attr_names = FromMeta::from_meta(mi)?;
59 } else if path.is_ident("forward_attrs") {
60self.forward_attrs = FromMeta::from_meta(mi)?;
61 } else if path.is_ident("from_ident") {
62// HACK: Declaring that a default is present will cause fields to
63 // generate correct code, but control flow isn't that obvious.
64self.container.default = Some(DefaultExpression::Trait {
65// Use the span of the `from_ident` keyword so that errors in generated code
66 // caused by this will point back to the correct location.
67span: path.span(),
68 });
69self.from_ident = true;
70 } else {
71return self.container.parse_nested(mi);
72 }
73Ok(())
74 }
75}
7677impl ParseData for OuterFrom {
78fn parse_field(&mut self, field: &Field) -> Result<()> {
79match field.ident.as_ref().map(|v| v.to_string()).as_deref() {
80Some("ident") => {
81self.ident.clone_from(&field.ident);
82Ok(())
83 }
84Some("attrs") => {
85self.attrs = AttrsField::from_field(field).map(Some)?;
86Ok(())
87 }
88_ => self.container.parse_field(field),
89 }
90 }
9192fn validate_body(&self, errors: &mut crate::error::Accumulator) {
93self.container.validate_body(errors);
94 }
95}