use {
crate::{
codegen::accounts::{generics, ParsedGenerics},
AccountField, AccountsStruct,
},
quote::quote,
};
pub fn generate(accs: &AccountsStruct) -> proc_macro2::TokenStream {
let name = &accs.ident;
let ParsedGenerics {
combined_generics,
trait_generics: _,
struct_generics,
where_clause,
} = generics(accs);
let key_pushes: Vec<proc_macro2::TokenStream> = accs
.fields
.iter()
.filter_map(|af: &AccountField| match af {
AccountField::CompositeField(s) => {
let field_name = &s.ident;
Some(quote! {
keys.extend(self.#field_name.duplicate_mutable_account_keys());
})
}
AccountField::Field(f)
if f.constraints.is_mutable()
&& !f.constraints.is_dup()
&& !f.constraints.is_pure_init() =>
{
match &f.ty {
crate::Ty::Account(_)
| crate::Ty::LazyAccount(_)
| crate::Ty::InterfaceAccount(_)
| crate::Ty::Migration(_) => {
let field_name = &f.ident;
if f.is_optional {
Some(quote! {
if let Some(ref account) = self.#field_name {
keys.push(account.key());
}
})
} else {
Some(quote! {
keys.push(self.#field_name.key());
})
}
}
_ => None,
}
}
_ => None,
})
.collect();
quote! {
#[automatically_derived]
impl<#combined_generics> anchor_lang::DuplicateMutableAccountKeys for #name<#struct_generics> #where_clause {
fn duplicate_mutable_account_keys(&self) -> Vec<anchor_lang::solana_program::pubkey::Pubkey> {
let mut keys = Vec::new();
#(#key_pushes)*
keys
}
}
}
}