Yhteenveto
Käytä invoke-funktiota CPI:hin, joissa kaikki vaaditut allekirjoittajat
ovat jo allekirjoittaneet alkuperäisen transaktion. PDA-allekirjoitusta ei
tarvita. Allekirjoittaja- ja kirjoitusoikeudet siirtyvät kutsujalta
kutsuttavalle.
CPI:t ilman PDA-allekirjoittajia
Kun CPI ei vaadi PDA-allekirjoittajia, käytetään
invoke-funktiota.
invoke-funktio kutsuu invoke_signed-funktiota tyhjällä
signers_seeds-taulukolla. Tyhjä allekirjoittajataulukko osoittaa, että
PDA:ita ei tarvita allekirjoitukseen.
pub fn invoke(instruction: &Instruction, account_infos: &[AccountInfo]) -> ProgramResult {invoke_signed(instruction, account_infos, &[])}
Alla olevat esimerkit tekevät CPI:n käyttäen Anchoria ja Native Rustia. Ne sisältävät yhden instruktion, joka siirtää SOL:ia tililtä toiselle.
Anchor
Seuraavat esimerkit näyttävät kaksi lähestymistapaa CPI:iden toteuttamiseen Anchor-ohjelmassa. Esimerkit ovat toiminnallisesti samanarvoisia, mutta havainnollistavat eri abstraktiotasoja.
- Esimerkki 1: Käyttää Anchorin
CpiContext-makroa ja apufunktiota. - Esimerkki 2: Käyttää
system_instruction::transfer-funktiotasolana_program-cratesta. - Esimerkki 3: Rakentaa CPI-instruktion manuaalisesti. Tämä lähestymistapa on hyödyllinen, kun ei ole olemassa cratea, joka auttaisi rakentamaan haluamasi instruktion.
use anchor_lang::prelude::*;use anchor_lang::system_program::{transfer, Transfer};declare_id!("9AvUNHjxscdkiKQ8tUn12QCMXtcnbR9BVGq3ULNzFMRi");#[program]pub mod cpi {use super::*;pub fn sol_transfer(ctx: Context<SolTransfer>, amount: u64) -> Result<()> {let from_pubkey = ctx.accounts.sender.to_account_info();let to_pubkey = ctx.accounts.recipient.to_account_info();let program_id = ctx.accounts.system_program.to_account_info();let cpi_context = CpiContext::new(program_id,Transfer {from: from_pubkey,to: to_pubkey,},);transfer(cpi_context, amount)?;Ok(())}}#[derive(Accounts)]pub struct SolTransfer<'info> {#[account(mut)]sender: Signer<'info>,#[account(mut)]recipient: SystemAccount<'info>,system_program: Program<'info, System>,}
Rust
Seuraava esimerkki näyttää, kuinka tehdä CPI Native Rustilla kirjoitetusta ohjelmasta. Se sisältää yhden instruktion, joka siirtää SOL:ia tililtä toiselle. (Testitiedosto käyttää LiteSVM-kirjastoa ohjelman testaamiseen.)
use borsh::BorshDeserialize;use solana_program::{account_info::AccountInfo,entrypoint,entrypoint::ProgramResult,program::invoke,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 [sender_info, recipient_info, system_program_info] = accounts else {return Err(ProgramError::NotEnoughAccountKeys);};// Verify the sender is a signerif !sender_info.is_signer {return Err(ProgramError::MissingRequiredSignature);}// Create and invoke the transfer instructionlet transfer_ix = system_instruction::transfer(sender_info.key,recipient_info.key,amount,);invoke(&transfer_ix,&[sender_info.clone(),recipient_info.clone(),system_program_info.clone(),],)?;Ok(())}}}
Is this page helpful?