Jak utworzyć mint z rozszerzeniem Confidential Transfer
Rozszerzenie Confidential Transfer umożliwia prywatne przelewy tokenów poprzez dodanie dodatkowego stanu do mint account. W tej sekcji wyjaśniono, jak utworzyć token mint z włączonym tym rozszerzeniem.
Poniższy diagram przedstawia kroki związane z tworzeniem mintu z rozszerzeniem Confidential Transfer:
Stan Mint Confidential Transfer
Rozszerzenie dodaje stan ConfidentialTransferMint do mint account:
#[repr(C)]#[derive(Clone, Copy, Debug, Default, PartialEq, Pod, Zeroable)]pub struct ConfidentialTransferMint {/// Authority to modify the `ConfidentialTransferMint` configuration and to/// approve new accounts (if `auto_approve_new_accounts` is true)////// The legacy Token Multisig account is not supported as the authoritypub authority: OptionalNonZeroPubkey,/// Indicate if newly configured accounts must be approved by the/// `authority` before they may be used by the user.////// * If `true`, no approval is required and new accounts may be used/// immediately/// * If `false`, the authority must approve newly configured accounts (see/// `ConfidentialTransferInstruction::ConfigureAccount`)pub auto_approve_new_accounts: PodBool,/// Authority to decode any transfer amount in a confidential transfer.pub auditor_elgamal_pubkey: OptionalNonZeroElGamalPubkey,}
ConfidentialTransferMint zawiera trzy pola konfiguracyjne:
-
authority: Konto posiadające uprawnienia do zmiany ustawień poufnych przelewów dla mintu oraz zatwierdzania nowych kont poufnych, jeśli automatyczne zatwierdzanie jest wyłączone.
-
auto_approve_new_accounts: Gdy ustawione na true, użytkownicy mogą tworzyć token accounts z domyślnie włączonymi przelewami poufnymi. Gdy ustawione na false, authority musi zatwierdzić każdy nowy token account przed jego użyciem do przelewów poufnych.
-
auditor_elgamal_pubkey: Opcjonalny audytor, który może odszyfrowywać kwoty przelewów w poufnych transakcjach, zapewniając mechanizm zgodności przy jednoczesnym zachowaniu prywatności wobec ogółu użytkowników.
Wymagane Instrukcje
Utworzenie mintu z włączonym Confidential Transfer wymaga trzech instrukcji w jednej transakcji:
-
Utwórz Mint Account: Wywołaj instrukcję
CreateAccountSystem Program, aby utworzyć mint account. -
Zainicjalizuj Rozszerzenie Confidential Transfer: Wywołaj instrukcję ConfidentialTransferInstruction::InitializeMint Token Extensions Program, aby skonfigurować stan
ConfidentialTransferMintdla mintu. -
Zainicjalizuj Mint: Wywołaj instrukcję
Instruction::InitializeMintToken Extensions Program, aby zainicjalizować standardowy stan mintu.
Choć można napisać te instrukcje ręcznie, biblioteka spl_token_client
udostępnia metodę create_mint, która buduje i wysyła transakcję ze wszystkimi
trzema instrukcjami w jednym wywołaniu funkcji, jak pokazano w poniższym
przykładzie.
Przykładowy kod
Poniższy kod demonstruje, jak utworzyć mint z rozszerzeniem Confidential Transfer.
Poufne transfery są zależne od programu ZK ElGamal Proof, który jest włączony na
mainnet i devnet. Standardowy solana-test-validator nie włącza go, ale lokalne
validator z forkiem mainnet, takie jak Surfpool, tak.
Uruchom przykład na jednej z tych sieci (kod używa devnet) z doładowanym kontem
płatnika i zastąp przykładowe adresy mint i konta własnymi.
Rust
fn main() -> Result<()> {let rpc_client = RpcClient::new_with_commitment(String::from("https://api.devnet.solana.com"),CommitmentConfig::confirmed(),);let payer = load_keypair()?;let mint = Keypair::new();let decimals: u8 = 2;// Allocate space for a mint that carries the ConfidentialTransferMint// extension, then fund it for rent exemption.let space =ExtensionType::try_calculate_account_len::<Mint>(&[ExtensionType::ConfidentialTransferMint])?;let rent = rpc_client.get_minimum_balance_for_rent_exemption(space)?;// The auditor ElGamal key lets the issuer decrypt transfer amounts for// compliance. Persist this key. Pass `None` to create a mint with no auditor.let auditor = ElGamalKeypair::new_rand();let auditor_pubkey: PodElGamalPubkey = (*auditor.pubkey()).into();let create_account_ix = system_instruction::create_account(&payer.pubkey(),&mint.pubkey(),rent,space as u64,&spl_token_2022::id(),);// The confidential-transfer extension must be initialized before the base// mint and cannot be added later.let init_confidential_ix = initialize_confidential_transfer_mint(&spl_token_2022::id(),&mint.pubkey(),Some(payer.pubkey()), // authority that can update confidential settingstrue, // auto-approve new accountsSome(auditor_pubkey),)?;let init_mint_ix = initialize_mint_base(&spl_token_2022::id(),&mint.pubkey(),&payer.pubkey(), // mint authorityNone, // freeze authoritydecimals,)?;let blockhash = rpc_client.get_latest_blockhash()?;let transaction = Transaction::new_signed_with_payer(&[create_account_ix, init_confidential_ix, init_mint_ix],Some(&payer.pubkey()),&[&payer, &mint],blockhash,);let signature = rpc_client.send_and_confirm_transaction(&transaction)?;println!("Created confidential mint {}: {signature}", mint.pubkey());Ok(())}fn load_keypair() -> Result<Keypair> {let keypair_path = dirs::home_dir().context("could not find home directory")?.join(".config/solana/id.json");let bytes: Vec<u8> = serde_json::from_reader(std::fs::File::open(keypair_path)?)?;let mut secret = [0u8; 32];secret.copy_from_slice(&bytes[0..32]);Ok(Keypair::new_from_array(secret))}
Typescript
const client = await createClient().use(signerFromFile(join(homedir(), ".config/solana/id.json"))).use(solanaRpc({rpcUrl: "https://api.devnet.solana.com"}));const payer = client.payer;const mint = await generateKeyPairSigner();// The auditor ElGamal key lets the issuer decrypt transfer amounts for// compliance. Persist it; omit `auditorElgamalPubkey` to create a mint with no// auditor.const auditor = await deriveElGamalKeypairForOwnerMint({signer: payer,owner: payer.address,mint: mint.address});const plan = getCreateMintInstructionPlan({payer,newMint: mint,decimals: 2,mintAuthority: payer,extensions: [{__kind: "ConfidentialTransferMint",authority: some(payer.address),autoApproveNewAccounts: true,auditorElgamalPubkey: some(auditor.elgamalPubkey)}]});const result = await client.sendTransaction(plan);console.log(`Created confidential mint ${mint.address}: ${result.context.signature}`);
Is this page helpful?