123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771 |
- use crate::*;
- use proc_macro2_diagnostics::SpanDiagnosticExt;
- use quote::quote;
- use syn::Expr;
- pub fn generate(f: &Field) -> proc_macro2::TokenStream {
- let constraints = linearize(&f.constraints);
- let rent = constraints
- .iter()
- .any(|c| matches!(c, Constraint::RentExempt(ConstraintRentExempt::Enforce)))
- .then(|| quote! { let __anchor_rent = Rent::get()?; })
- .unwrap_or_else(|| quote! {});
- let checks: Vec<proc_macro2::TokenStream> = constraints
- .iter()
- .map(|c| generate_constraint(f, c))
- .collect();
- quote! {
- #rent
- #(#checks)*
- }
- }
- pub fn generate_composite(f: &CompositeField) -> proc_macro2::TokenStream {
- let checks: Vec<proc_macro2::TokenStream> = linearize(&f.constraints)
- .iter()
- .filter_map(|c| match c {
- Constraint::Raw(_) => Some(c),
- Constraint::Literal(_) => Some(c),
- _ => panic!("Invariant violation: composite constraints can only be raw or literals"),
- })
- .map(|c| generate_constraint_composite(f, c))
- .collect();
- quote! {
- #(#checks)*
- }
- }
- // Linearizes the constraint group so that constraints with dependencies
- // run after those without.
- pub fn linearize(c_group: &ConstraintGroup) -> Vec<Constraint> {
- let ConstraintGroup {
- init,
- zeroed,
- mutable,
- signer,
- has_one,
- literal,
- raw,
- owner,
- rent_exempt,
- seeds,
- executable,
- state,
- close,
- address,
- associated_token,
- } = c_group.clone();
- let mut constraints = Vec::new();
- if let Some(c) = zeroed {
- constraints.push(Constraint::Zeroed(c));
- }
- if let Some(c) = init {
- constraints.push(Constraint::Init(c));
- }
- if let Some(c) = seeds {
- constraints.push(Constraint::Seeds(c));
- }
- if let Some(c) = associated_token {
- constraints.push(Constraint::AssociatedToken(c));
- }
- if let Some(c) = mutable {
- constraints.push(Constraint::Mut(c));
- }
- if let Some(c) = signer {
- constraints.push(Constraint::Signer(c));
- }
- constraints.append(&mut has_one.into_iter().map(Constraint::HasOne).collect());
- constraints.append(&mut literal.into_iter().map(Constraint::Literal).collect());
- constraints.append(&mut raw.into_iter().map(Constraint::Raw).collect());
- if let Some(c) = owner {
- constraints.push(Constraint::Owner(c));
- }
- if let Some(c) = rent_exempt {
- constraints.push(Constraint::RentExempt(c));
- }
- if let Some(c) = executable {
- constraints.push(Constraint::Executable(c));
- }
- if let Some(c) = state {
- constraints.push(Constraint::State(c));
- }
- if let Some(c) = close {
- constraints.push(Constraint::Close(c));
- }
- if let Some(c) = address {
- constraints.push(Constraint::Address(c));
- }
- constraints
- }
- fn generate_constraint(f: &Field, c: &Constraint) -> proc_macro2::TokenStream {
- match c {
- Constraint::Init(c) => generate_constraint_init(f, c),
- Constraint::Zeroed(c) => generate_constraint_zeroed(f, c),
- Constraint::Mut(c) => generate_constraint_mut(f, c),
- Constraint::HasOne(c) => generate_constraint_has_one(f, c),
- Constraint::Signer(c) => generate_constraint_signer(f, c),
- Constraint::Literal(c) => generate_constraint_literal(c),
- Constraint::Raw(c) => generate_constraint_raw(c),
- Constraint::Owner(c) => generate_constraint_owner(f, c),
- Constraint::RentExempt(c) => generate_constraint_rent_exempt(f, c),
- Constraint::Seeds(c) => generate_constraint_seeds(f, c),
- Constraint::Executable(c) => generate_constraint_executable(f, c),
- Constraint::State(c) => generate_constraint_state(f, c),
- Constraint::Close(c) => generate_constraint_close(f, c),
- Constraint::Address(c) => generate_constraint_address(f, c),
- Constraint::AssociatedToken(c) => generate_constraint_associated_token(f, c),
- }
- }
- fn generate_constraint_composite(_f: &CompositeField, c: &Constraint) -> proc_macro2::TokenStream {
- match c {
- Constraint::Raw(c) => generate_constraint_raw(c),
- Constraint::Literal(c) => generate_constraint_literal(c),
- _ => panic!("Invariant violation"),
- }
- }
- fn generate_constraint_address(f: &Field, c: &ConstraintAddress) -> proc_macro2::TokenStream {
- let field = &f.ident;
- let addr = &c.address;
- let error = generate_custom_error(&c.error, quote! { ConstraintAddress });
- quote! {
- if #field.key() != #addr {
- return Err(#error);
- }
- }
- }
- pub fn generate_constraint_init(f: &Field, c: &ConstraintInitGroup) -> proc_macro2::TokenStream {
- generate_constraint_init_group(f, c)
- }
- pub fn generate_constraint_zeroed(f: &Field, _c: &ConstraintZeroed) -> proc_macro2::TokenStream {
- let field = &f.ident;
- let ty_decl = f.ty_decl();
- let from_account_info = f.from_account_info_unchecked(None);
- quote! {
- let #field: #ty_decl = {
- let mut __data: &[u8] = &#field.try_borrow_data()?;
- let mut __disc_bytes = [0u8; 8];
- __disc_bytes.copy_from_slice(&__data[..8]);
- let __discriminator = u64::from_le_bytes(__disc_bytes);
- if __discriminator != 0 {
- return Err(anchor_lang::__private::ErrorCode::ConstraintZero.into());
- }
- #from_account_info
- };
- }
- }
- pub fn generate_constraint_close(f: &Field, c: &ConstraintClose) -> proc_macro2::TokenStream {
- let field = &f.ident;
- let target = &c.sol_dest;
- quote! {
- if #field.key() == #target.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintClose.into());
- }
- }
- }
- pub fn generate_constraint_mut(f: &Field, c: &ConstraintMut) -> proc_macro2::TokenStream {
- let ident = &f.ident;
- let error = generate_custom_error(&c.error, quote! { ConstraintMut });
- quote! {
- if !#ident.to_account_info().is_writable {
- return Err(#error);
- }
- }
- }
- pub fn generate_constraint_has_one(f: &Field, c: &ConstraintHasOne) -> proc_macro2::TokenStream {
- let target = c.join_target.clone();
- let ident = &f.ident;
- let field = match &f.ty {
- Ty::Loader(_) => quote! {#ident.load()?},
- Ty::AccountLoader(_) => quote! {#ident.load()?},
- _ => quote! {#ident},
- };
- let error = generate_custom_error(&c.error, quote! { ConstraintHasOne });
- quote! {
- if #field.#target != #target.key() {
- return Err(#error);
- }
- }
- }
- pub fn generate_constraint_signer(f: &Field, c: &ConstraintSigner) -> proc_macro2::TokenStream {
- let ident = &f.ident;
- let info = match f.ty {
- Ty::AccountInfo => quote! { #ident },
- Ty::ProgramAccount(_) => quote! { #ident.to_account_info() },
- Ty::Account(_) => quote! { #ident.to_account_info() },
- Ty::Loader(_) => quote! { #ident.to_account_info() },
- Ty::AccountLoader(_) => quote! { #ident.to_account_info() },
- Ty::CpiAccount(_) => quote! { #ident.to_account_info() },
- _ => panic!("Invalid syntax: signer cannot be specified."),
- };
- let error = generate_custom_error(&c.error, quote! { ConstraintSigner });
- quote! {
- if !#info.is_signer {
- return Err(#error);
- }
- }
- }
- pub fn generate_constraint_literal(c: &ConstraintLiteral) -> proc_macro2::TokenStream {
- let lit: proc_macro2::TokenStream = {
- let lit = &c.lit;
- let constraint = lit.value().replace('\"', "");
- let message = format!(
- "Deprecated. Should be used with constraint: #[account(constraint = {})]",
- constraint,
- );
- lit.span().warning(message).emit_as_item_tokens();
- constraint.parse().unwrap()
- };
- quote! {
- if !(#lit) {
- return Err(anchor_lang::__private::ErrorCode::Deprecated.into());
- }
- }
- }
- pub fn generate_constraint_raw(c: &ConstraintRaw) -> proc_macro2::TokenStream {
- let raw = &c.raw;
- let error = generate_custom_error(&c.error, quote! { ConstraintRaw });
- quote! {
- if !(#raw) {
- return Err(#error);
- }
- }
- }
- pub fn generate_constraint_owner(f: &Field, c: &ConstraintOwner) -> proc_macro2::TokenStream {
- let ident = &f.ident;
- let owner_address = &c.owner_address;
- let error = generate_custom_error(&c.error, quote! { ConstraintOwner });
- quote! {
- if #ident.as_ref().owner != &#owner_address {
- return Err(#error);
- }
- }
- }
- pub fn generate_constraint_rent_exempt(
- f: &Field,
- c: &ConstraintRentExempt,
- ) -> proc_macro2::TokenStream {
- let ident = &f.ident;
- let info = quote! {
- #ident.to_account_info()
- };
- match c {
- ConstraintRentExempt::Skip => quote! {},
- ConstraintRentExempt::Enforce => quote! {
- if !__anchor_rent.is_exempt(#info.lamports(), #info.try_data_len()?) {
- return Err(anchor_lang::__private::ErrorCode::ConstraintRentExempt.into());
- }
- },
- }
- }
- fn generate_constraint_init_group(f: &Field, c: &ConstraintInitGroup) -> proc_macro2::TokenStream {
- let payer = {
- let p = &c.payer;
- quote! {
- let payer = #p.to_account_info();
- }
- };
- let seeds_with_nonce = match &c.seeds {
- None => quote! {},
- Some(c) => {
- let s = &mut c.seeds.clone();
- // If the seeds came with a trailing comma, we need to chop it off
- // before we interpolate them below.
- if let Some(pair) = s.pop() {
- s.push_value(pair.into_value());
- }
- let maybe_seeds_plus_comma = (!s.is_empty()).then(|| {
- quote! { #s, }
- });
- let inner = match c.bump.as_ref() {
- // Bump target not given. Use the canonical bump.
- None => {
- quote! {
- [
- #maybe_seeds_plus_comma
- &[
- Pubkey::find_program_address(
- &[#s],
- program_id,
- ).1
- ][..]
- ]
- }
- }
- // Bump target given. Use it.
- Some(b) => quote! {
- [#maybe_seeds_plus_comma &[#b][..]]
- },
- };
- quote! {
- &#inner[..]
- }
- }
- };
- generate_init(f, c.if_needed, seeds_with_nonce, payer, &c.space, &c.kind)
- }
- fn generate_constraint_seeds(f: &Field, c: &ConstraintSeedsGroup) -> proc_macro2::TokenStream {
- let name = &f.ident;
- let s = &mut c.seeds.clone();
- let deriving_program_id = c
- .program_seed
- .clone()
- // If they specified a seeds::program to use when deriving the PDA, use it.
- .map(|program_id| quote! { #program_id })
- // Otherwise fall back to the current program's program_id.
- .unwrap_or(quote! { program_id });
- // If the seeds came with a trailing comma, we need to chop it off
- // before we interpolate them below.
- if let Some(pair) = s.pop() {
- s.push_value(pair.into_value());
- }
- // If the bump is provided with init *and target*, then force it to be the
- // canonical bump.
- if c.is_init && c.bump.is_some() {
- let b = c.bump.as_ref().unwrap();
- quote! {
- let (__program_signer, __bump) = anchor_lang::solana_program::pubkey::Pubkey::find_program_address(
- &[#s],
- &#deriving_program_id,
- );
- if #name.key() != __program_signer {
- return Err(anchor_lang::__private::ErrorCode::ConstraintSeeds.into());
- }
- if __bump != #b {
- return Err(anchor_lang::__private::ErrorCode::ConstraintSeeds.into());
- }
- }
- } else {
- let maybe_seeds_plus_comma = (!s.is_empty()).then(|| {
- quote! { #s, }
- });
- let seeds = match c.bump.as_ref() {
- // Bump target not given. Find it.
- None => {
- quote! {
- [
- #maybe_seeds_plus_comma
- &[
- Pubkey::find_program_address(
- &[#s],
- &#deriving_program_id,
- ).1
- ][..]
- ]
- }
- }
- // Bump target given. Use it.
- Some(b) => {
- quote! {
- [#maybe_seeds_plus_comma &[#b][..]]
- }
- }
- };
- quote! {
- let __program_signer = Pubkey::create_program_address(
- &#seeds[..],
- &#deriving_program_id,
- ).map_err(|_| anchor_lang::__private::ErrorCode::ConstraintSeeds)?;
- if #name.key() != __program_signer {
- return Err(anchor_lang::__private::ErrorCode::ConstraintSeeds.into());
- }
- }
- }
- }
- fn generate_constraint_associated_token(
- f: &Field,
- c: &ConstraintAssociatedToken,
- ) -> proc_macro2::TokenStream {
- let name = &f.ident;
- let wallet_address = &c.wallet;
- let spl_token_mint_address = &c.mint;
- quote! {
- if #name.owner != #wallet_address.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintTokenOwner.into());
- }
- let __associated_token_address = anchor_spl::associated_token::get_associated_token_address(&#wallet_address.key(), &#spl_token_mint_address.key());
- if #name.key() != __associated_token_address {
- return Err(anchor_lang::__private::ErrorCode::ConstraintAssociated.into());
- }
- }
- }
- // `if_needed` is set if account allocation and initialization is optional.
- pub fn generate_init(
- f: &Field,
- if_needed: bool,
- seeds_with_nonce: proc_macro2::TokenStream,
- payer: proc_macro2::TokenStream,
- space: &Option<Expr>,
- kind: &InitKind,
- ) -> proc_macro2::TokenStream {
- let field = &f.ident;
- let ty_decl = f.ty_decl();
- let from_account_info = f.from_account_info_unchecked(Some(kind));
- let if_needed = if if_needed {
- quote! {true}
- } else {
- quote! {false}
- };
- match kind {
- InitKind::Token { owner, mint } => {
- let create_account = generate_create_account(
- field,
- quote! {anchor_spl::token::TokenAccount::LEN},
- quote! {&token_program.key()},
- seeds_with_nonce,
- );
- quote! {
- let #field: #ty_decl = {
- if !#if_needed || #field.as_ref().owner == &anchor_lang::solana_program::system_program::ID {
- // Define payer variable.
- #payer
- // Create the account with the system program.
- #create_account
- // Initialize the token account.
- let cpi_program = token_program.to_account_info();
- let accounts = anchor_spl::token::InitializeAccount {
- account: #field.to_account_info(),
- mint: #mint.to_account_info(),
- authority: #owner.to_account_info(),
- rent: rent.to_account_info(),
- };
- let cpi_ctx = anchor_lang::context::CpiContext::new(cpi_program, accounts);
- anchor_spl::token::initialize_account(cpi_ctx)?;
- }
- let pa: #ty_decl = #from_account_info;
- if !(!#if_needed || #field.as_ref().owner == &anchor_lang::solana_program::system_program::ID) {
- if pa.mint != #mint.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintTokenMint.into());
- }
- if pa.owner != #owner.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintTokenOwner.into());
- }
- }
- pa
- };
- }
- }
- InitKind::AssociatedToken { owner, mint } => {
- quote! {
- let #field: #ty_decl = {
- if !#if_needed || #field.as_ref().owner == &anchor_lang::solana_program::system_program::ID {
- #payer
- let cpi_program = associated_token_program.to_account_info();
- let cpi_accounts = anchor_spl::associated_token::Create {
- payer: payer.to_account_info(),
- associated_token: #field.to_account_info(),
- authority: #owner.to_account_info(),
- mint: #mint.to_account_info(),
- system_program: system_program.to_account_info(),
- token_program: token_program.to_account_info(),
- rent: rent.to_account_info(),
- };
- let cpi_ctx = anchor_lang::context::CpiContext::new(cpi_program, cpi_accounts);
- anchor_spl::associated_token::create(cpi_ctx)?;
- }
- let pa: #ty_decl = #from_account_info;
- if !(!#if_needed || #field.as_ref().owner == &anchor_lang::solana_program::system_program::ID) {
- if pa.mint != #mint.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintTokenMint.into());
- }
- if pa.owner != #owner.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintTokenOwner.into());
- }
- if pa.key() != anchor_spl::associated_token::get_associated_token_address(&#owner.key(), &#mint.key()) {
- return Err(anchor_lang::__private::ErrorCode::AccountNotAssociatedTokenAccount.into());
- }
- }
- pa
- };
- }
- }
- InitKind::Mint {
- owner,
- decimals,
- freeze_authority,
- } => {
- let create_account = generate_create_account(
- field,
- quote! {anchor_spl::token::Mint::LEN},
- quote! {&token_program.key()},
- seeds_with_nonce,
- );
- let freeze_authority = match freeze_authority {
- Some(fa) => quote! { Option::<&anchor_lang::prelude::Pubkey>::Some(&#fa.key()) },
- None => quote! { Option::<&anchor_lang::prelude::Pubkey>::None },
- };
- quote! {
- let #field: #ty_decl = {
- if !#if_needed || #field.as_ref().owner == &anchor_lang::solana_program::system_program::ID {
- // Define payer variable.
- #payer
- // Create the account with the system program.
- #create_account
- // Initialize the mint account.
- let cpi_program = token_program.to_account_info();
- let accounts = anchor_spl::token::InitializeMint {
- mint: #field.to_account_info(),
- rent: rent.to_account_info(),
- };
- let cpi_ctx = anchor_lang::context::CpiContext::new(cpi_program, accounts);
- anchor_spl::token::initialize_mint(cpi_ctx, #decimals, &#owner.key(), #freeze_authority)?;
- }
- let pa: #ty_decl = #from_account_info;
- if !(!#if_needed || #field.as_ref().owner == &anchor_lang::solana_program::system_program::ID) {
- if pa.mint_authority != anchor_lang::solana_program::program_option::COption::Some(#owner.key()) {
- return Err(anchor_lang::__private::ErrorCode::ConstraintMintMintAuthority.into());
- }
- if pa.freeze_authority
- .as_ref()
- .map(|fa| #freeze_authority.as_ref().map(|expected_fa| fa != *expected_fa).unwrap_or(true))
- .unwrap_or(#freeze_authority.is_some()) {
- return Err(anchor_lang::__private::ErrorCode::ConstraintMintFreezeAuthority.into());
- }
- if pa.decimals != #decimals {
- return Err(anchor_lang::__private::ErrorCode::ConstraintMintDecimals.into());
- }
- }
- pa
- };
- }
- }
- InitKind::Program { owner } => {
- let space = match space {
- // If no explicit space param was given, serialize the type to bytes
- // and take the length (with +8 for the discriminator.)
- None => {
- let account_ty = f.account_ty();
- match matches!(f.ty, Ty::Loader(_) | Ty::AccountLoader(_)) {
- false => {
- quote! {
- let space = 8 + #account_ty::default().try_to_vec().unwrap().len();
- }
- }
- true => {
- quote! {
- let space = 8 + anchor_lang::__private::bytemuck::bytes_of(&#account_ty::default()).len();
- }
- }
- }
- }
- // Explicit account size given. Use it.
- Some(s) => quote! {
- let space = #s;
- },
- };
- // Owner of the account being created. If not specified,
- // default to the currently executing program.
- let owner = match owner {
- None => quote! {
- program_id
- },
- Some(o) => quote! {
- &#o
- },
- };
- let pda_check = if !seeds_with_nonce.is_empty() {
- quote! {
- let expected_key = anchor_lang::prelude::Pubkey::create_program_address(
- #seeds_with_nonce,
- #owner
- ).map_err(|_| anchor_lang::__private::ErrorCode::ConstraintSeeds)?;
- if expected_key != #field.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintSeeds.into());
- }
- }
- } else {
- quote! {}
- };
- let create_account =
- generate_create_account(field, quote! {space}, owner.clone(), seeds_with_nonce);
- quote! {
- let #field = {
- let actual_field = #field.to_account_info();
- let actual_owner = actual_field.owner;
- #space
- if !#if_needed || actual_owner == &anchor_lang::solana_program::system_program::ID {
- #payer
- #create_account
- }
- let pa: #ty_decl = #from_account_info;
- if !(!#if_needed || actual_owner == &anchor_lang::solana_program::system_program::ID) {
- if space != actual_field.data_len() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintSpace.into());
- }
- if actual_owner != #owner {
- return Err(anchor_lang::__private::ErrorCode::ConstraintOwner.into());
- }
- {
- let required_lamports = __anchor_rent.minimum_balance(space);
- if pa.to_account_info().lamports() < required_lamports {
- return Err(anchor_lang::__private::ErrorCode::ConstraintRentExempt.into());
- }
- }
- #pda_check
- }
- pa
- };
- }
- }
- }
- }
- // Generated code to create an account with with system program with the
- // given `space` amount of data, owned by `owner`.
- //
- // `seeds_with_nonce` should be given for creating PDAs. Otherwise it's an
- // empty stream.
- pub fn generate_create_account(
- field: &Ident,
- space: proc_macro2::TokenStream,
- owner: proc_macro2::TokenStream,
- seeds_with_nonce: proc_macro2::TokenStream,
- ) -> proc_macro2::TokenStream {
- quote! {
- // If the account being initialized already has lamports, then
- // return them all back to the payer so that the account has
- // zero lamports when the system program's create instruction
- // is eventually called.
- let __current_lamports = #field.lamports();
- if __current_lamports == 0 {
- // Create the token account with right amount of lamports and space, and the correct owner.
- let lamports = __anchor_rent.minimum_balance(#space);
- anchor_lang::solana_program::program::invoke_signed(
- &anchor_lang::solana_program::system_instruction::create_account(
- &payer.key(),
- &#field.key(),
- lamports,
- #space as u64,
- #owner,
- ),
- &[
- payer.to_account_info(),
- #field.to_account_info(),
- system_program.to_account_info(),
- ],
- &[#seeds_with_nonce],
- )?;
- } else {
- // Fund the account for rent exemption.
- let required_lamports = __anchor_rent
- .minimum_balance(#space)
- .max(1)
- .saturating_sub(__current_lamports);
- if required_lamports > 0 {
- anchor_lang::solana_program::program::invoke(
- &anchor_lang::solana_program::system_instruction::transfer(
- &payer.key(),
- &#field.key(),
- required_lamports,
- ),
- &[
- payer.to_account_info(),
- #field.to_account_info(),
- system_program.to_account_info(),
- ],
- )?;
- }
- // Allocate space.
- anchor_lang::solana_program::program::invoke_signed(
- &anchor_lang::solana_program::system_instruction::allocate(
- &#field.key(),
- #space as u64,
- ),
- &[
- #field.to_account_info(),
- system_program.to_account_info(),
- ],
- &[#seeds_with_nonce],
- )?;
- // Assign to the spl token program.
- anchor_lang::solana_program::program::invoke_signed(
- &anchor_lang::solana_program::system_instruction::assign(
- &#field.key(),
- #owner,
- ),
- &[
- #field.to_account_info(),
- system_program.to_account_info(),
- ],
- &[#seeds_with_nonce],
- )?;
- }
- }
- }
- pub fn generate_constraint_executable(
- f: &Field,
- _c: &ConstraintExecutable,
- ) -> proc_macro2::TokenStream {
- let name = &f.ident;
- quote! {
- if !#name.to_account_info().executable {
- return Err(anchor_lang::__private::ErrorCode::ConstraintExecutable.into());
- }
- }
- }
- pub fn generate_constraint_state(f: &Field, c: &ConstraintState) -> proc_macro2::TokenStream {
- let program_target = c.program_target.clone();
- let ident = &f.ident;
- let account_ty = match &f.ty {
- Ty::CpiState(ty) => &ty.account_type_path,
- _ => panic!("Invalid state constraint"),
- };
- quote! {
- // Checks the given state account is the canonical state account for
- // the target program.
- if #ident.key() != anchor_lang::accounts::cpi_state::CpiState::<#account_ty>::address(&#program_target.key()) {
- return Err(anchor_lang::__private::ErrorCode::ConstraintState.into());
- }
- if #ident.as_ref().owner != &#program_target.key() {
- return Err(anchor_lang::__private::ErrorCode::ConstraintState.into());
- }
- }
- }
- fn generate_custom_error(
- custom_error: &Option<Expr>,
- error: proc_macro2::TokenStream,
- ) -> proc_macro2::TokenStream {
- match custom_error {
- Some(error) => quote! { #error.into() },
- None => quote! { anchor_lang::__private::ErrorCode::#error.into() },
- }
- }
|