Zusammenfassung
Verwenden Sie invoke_signed, wenn das aufrufende Programm im Namen einer
PDA signieren muss, die es besitzt. Die Runtime leitet PDA-Pubkeys aus den
bereitgestellten Signer-Seeds ab und fügt sie vor der Berechtigungsprüfung zur
Menge der gültigen Signer hinzu.
CPIs mit PDA-Signern
Wenn ein CPI einen PDA-Signer erfordert, verwenden Sie
invoke_signed
mit den Signer-Seeds, die zur Ableitung der
PDA verwendet werden. Details dazu, wie die Runtime
PDA-Signaturen verifiziert, finden Sie unter
PDA-Signierung.
pub fn invoke_signed(instruction: &Instruction,account_infos: &[AccountInfo],signers_seeds: &[&[&[u8]]],) -> ProgramResult {// --snip--invoke_signed_unchecked(instruction, account_infos, signers_seeds)}
Die folgenden Beispiele zeigen einen CPI mit PDA-Signern unter Verwendung von Anchor und Native Rust. Jedes Beispiel enthält eine einzelne Anweisung zum Übertragen von SOL von einer PDA zu einem Empfänger-Konten, unter Verwendung eines von der PDA signierten CPI.
Anchor
Die folgenden Beispiele zeigen zwei Ansätze zur Implementierung von CPIs in einem Anchor-Programm. Die Beispiele sind funktional gleichwertig, demonstrieren jedoch unterschiedliche Abstraktionsebenen.
- Beispiel 1: Verwendet Anchors
CpiContextund Hilfsfunktion. - Beispiel 2: Verwendet die
system_instruction::transfer-Funktion aus demsolana_program-Crate. (Beispiel 1 ist eine Abstraktion dieser Implementierung.) - Beispiel 3: Konstruiert die CPI-Anweisung manuell. Dieser Ansatz ist nützlich, wenn kein Crate verfügbar ist, um die Anweisung zu erstellen, die Sie aufrufen möchten.
use anchor_lang::prelude::*;use anchor_lang::system_program::{transfer, Transfer};declare_id!("BrcdB9sV7z9DvF9rDHG263HUxXgJM3iCQdF36TcxbFEn");#[program]pub mod cpi {use super::*;pub fn sol_transfer(ctx: Context<SolTransfer>, amount: u64) -> Result<()> {let from_pubkey = ctx.accounts.pda_account.to_account_info();let to_pubkey = ctx.accounts.recipient.to_account_info();let program_id = ctx.accounts.system_program.to_account_info();let seed = to_pubkey.key();let bump_seed = ctx.bumps.pda_account;let signer_seeds: &[&[&[u8]]] = &[&[b"pda", seed.as_ref(), &[bump_seed]]];let cpi_context = CpiContext::new(program_id,Transfer {from: from_pubkey,to: to_pubkey,},).with_signer(signer_seeds);transfer(cpi_context, amount)?;Ok(())}}#[derive(Accounts)]pub struct SolTransfer<'info> {#[account(mut,seeds = [b"pda", recipient.key().as_ref()],bump,)]pda_account: SystemAccount<'info>,#[account(mut)]recipient: SystemAccount<'info>,system_program: Program<'info, System>,}
Rust
Das folgende Beispiel führt einen CPI mit PDA-Signierern aus einem in nativem Rust geschriebenen Programm durch. Es enthält eine einzelne Anweisung, die SOL von einem PDA-Konto auf ein anderes überträgt. Der CPI wird vom PDA-Konto signiert. (Die Testdatei verwendet LiteSVM, um das Programm zu testen.)
use borsh::BorshDeserialize;use solana_program::{account_info::AccountInfo,entrypoint,entrypoint::ProgramResult,program::invoke_signed,program_error::ProgramError,pubkey::Pubkey,system_instruction,};// Declare program entrypointentrypoint!(process_instruction);// Define program instructions#[derive(BorshDeserialize)]enum ProgramInstruction {SolTransfer { amount: u64 },}impl ProgramInstruction {fn unpack(input: &[u8]) -> Result<Self, ProgramError> {Self::try_from_slice(input).map_err(|_| ProgramError::InvalidInstructionData)}}pub fn process_instruction(program_id: &Pubkey,accounts: &[AccountInfo],instruction_data: &[u8],) -> ProgramResult {// Deserialize instruction datalet instruction = ProgramInstruction::unpack(instruction_data)?;// Process instructionmatch instruction {ProgramInstruction::SolTransfer { amount } => {// Parse accountslet [pda_account_info, recipient_info, system_program_info] = accounts else {return Err(ProgramError::NotEnoughAccountKeys);};// Derive PDA and verify it matches the account provided by clientlet recipient_pubkey = recipient_info.key;let seeds = &[b"pda", recipient_pubkey.as_ref()];let (expected_pda, bump_seed) = Pubkey::find_program_address(seeds, program_id);if expected_pda != *pda_account_info.key {return Err(ProgramError::InvalidArgument);}// Create the transfer instructionlet transfer_ix = system_instruction::transfer(pda_account_info.key,recipient_info.key,amount,);// Create signer seeds for PDAlet signer_seeds: &[&[&[u8]]] = &[&[b"pda", recipient_pubkey.as_ref(), &[bump_seed]]];// Invoke the transfer instruction with PDA as signerinvoke_signed(&transfer_ix,&[pda_account_info.clone(),recipient_info.clone(),system_program_info.clone(),],signer_seeds,)?;Ok(())}}}
Is this page helpful?