Utwórz mint

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:

Create Mint with Confidential Transfer Extension

Stan Mint Confidential Transfer

Rozszerzenie dodaje stan ConfidentialTransferMint do mint account:

Confidential Mint State
#[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 authority
pub 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:

  1. Utwórz Mint Account: Wywołaj instrukcję CreateAccount System Program, aby utworzyć mint account.

  2. Zainicjalizuj Rozszerzenie Confidential Transfer: Wywołaj instrukcję ConfidentialTransferInstruction::InitializeMint Token Extensions Program, aby skonfigurować stan ConfidentialTransferMint dla mintu.

  3. Zainicjalizuj Mint: Wywołaj instrukcję Instruction::InitializeMint Token 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 settings
true, // auto-approve new accounts
Some(auditor_pubkey),
)?;
let init_mint_ix = initialize_mint_base(
&spl_token_2022::id(),
&mint.pubkey(),
&payer.pubkey(), // mint authority
None, // freeze authority
decimals,
)?;
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?

Spis treści

Edytuj stronę