Ogni transazione Solana richiede SOL per pagare le commissioni di rete. Ma gli utenti che arrivano alla tua applicazione di pagamento si aspettano di effettuare transazioni in stablecoin, non di gestire un secondo saldo di token. L'astrazione delle commissioni elimina questo attrito facendo pagare le commissioni a qualcun altro.
Questa guida copre due livelli:
- Come funziona la sponsorizzazione delle commissioni — la primitiva sottostante di Solana
- Astrazione delle commissioni su larga scala con Kora — un servizio di astrazione delle commissioni pronto per la produzione
Come funziona la sponsorizzazione delle commissioni
Le transazioni Solana hanno un pagatore di commissioni designato, l'account che paga la commissione di rete. Per impostazione predefinita, questo è il primo firmatario. Ma puoi specificare un account diverso come pagatore delle commissioni, consentendo a una terza parte (lo "sponsor") di coprire le commissioni per conto del mittente.
Sia il mittente che lo sponsor devono firmare la transazione:
- Il mittente firma per autorizzare il trasferimento dei propri token
- Lo sponsor firma per autorizzare il pagamento della commissione di rete
Vedi Come funzionano i pagamenti su Solana per i concetti fondamentali sui pagamenti.
I passaggi seguenti mostrano il flusso principale. Vedi la Demo per il codice completo eseguibile.
Crea un account sponsor
Genera una keypair separata per lo sponsor che pagherà le commissioni di transazione. Lo sponsor ha bisogno di SOL per le commissioni ma non deve detenere i token che vengono trasferiti.
const sponsor = (await generateKeypair()).signer;
Crea l'istruzione di trasferimento
Crea l'istruzione di trasferimento dei token con il mittente come autorità. Il mittente possiede i token e deve firmare il trasferimento.
Invia con lo sponsor come pagatore delle commissioni
Usa prepareAndSend con entrambi authority (il mittente che firma il
trasferimento) e feePayer (lo sponsor che paga le commissioni). Entrambi
devono firmare la transazione.
Demo
// Generate keypairs for sender, recipient, and sponsor (fee payer)const sender = (await generateKeypair()).signer;const recipient = (await generateKeypair()).signer;const sponsor = (await generateKeypair()).signer;console.log("Sender Address:", sender.address);console.log("Recipient Address:", recipient.address);console.log("Sponsor Address (Fee Payer):", sponsor.address);// Demo Setup: Create client, mint account, token accounts, and fund with initial tokensconst { client, mint } = await demoSetup(sender, recipient, sponsor);console.log("\nMint Address:", mint.address);// Derive the Associated Token Accounts addresses (ATAs) for sender and recipientconst [senderAta] = await findAssociatedTokenPda({mint: mint.address,owner: sender.address,tokenProgram: TOKEN_2022_PROGRAM_ADDRESS});const [recipientAta] = await findAssociatedTokenPda({mint: mint.address,owner: recipient.address,tokenProgram: TOKEN_2022_PROGRAM_ADDRESS});console.log("Sender Token Account:", senderAta.toString());console.log("Recipient Token Account:", recipientAta.toString());// =============================================================================// Sponsored Token Payment Demo// =============================================================================// Create instruction to transfer tokens from sender to recipient// Transferring 250,000 base units = 0.25 tokens (with 6 decimals)const transferInstruction = getTransferInstruction({source: senderAta,destination: recipientAta,authority: sender, // Pass signer, not just addressamount: 250_000n // 0.25 tokens});// Prepare and send transaction with sponsor as fee payer using @solana/client// The sponsor pays transaction fees, sender signs for the transferconst signature = await client.transaction.prepareAndSend({authority: sender, // Sender signs the transfer instructionfeePayer: sponsor, // Sponsor pays the transaction fees (different account)instructions: [transferInstruction],version: 0});console.log("\n=== Sponsored Token Payment Complete ===");console.log("Transaction Signature:", signature.toString());// Fetch final token account balances using @solana/client SPL token helperconst splToken = client.splToken({mint: mint.address,tokenProgram: "auto"});const senderBalance = await splToken.fetchBalance(sender.address);const recipientBalance = await splToken.fetchBalance(recipient.address);console.log("\nSender Token Account Balance:", senderBalance);console.log("Recipient Token Account Balance:", recipientBalance);// Fetch transaction detailsconst transaction = await client.runtime.rpc.getTransaction(signature, {encoding: "jsonParsed",maxSupportedTransactionVersion: 0}).send();const feePayer = transaction?.transaction.message.accountKeys?.[0];console.log("\nNote: The first account in accountKeys is always the fee payer");console.log("Fee Payer Address:", feePayer);// =============================================================================// Demo Setup Helper Function// =============================================================================
Quando crei un token account per un utente finale, questi può chiuderlo e recuperare i SOL utilizzati per il rent. Considera di addebitare agli utenti la creazione dell'account in stablecoin, oppure includi questo costo nell'economia del tuo prodotto.
Astrazione delle commissioni su larga scala con Kora
La primitiva del pagatore delle commissioni è potente, ma costruire un sistema gasless in produzione richiede di più: gestire i wallet degli sponsor, gestire le conversioni di token (in modo che gli utenti possano "pagare" le commissioni in USDC), limitazione della frequenza e controlli di sicurezza.
Kora gestisce questa complessità. È un server JSON-RPC che fornisce l'astrazione delle commissioni in modo che gli utenti non abbiano mai bisogno di SOL. Puoi sponsorizzare completamente le commissioni o accettare il pagamento delle commissioni in qualsiasi token.
Distribuisci Kora con un singolo comando:
cargo install kora-clikora --config path/to/kora.toml rpc start --signers-config path/to/signers.toml
Quindi usa il client Kora per firmare e inviare transazioni:
pnpm add @solana/kora
import { KoraClient } from "@solana/kora";const kora = new KoraClient({ rpcUrl: "https://your-kora-instance" });const { signature } = await kora.signAndSendTransaction({transaction: base64EncodedTransaction});
Risorse Kora
Guida rapida Kora
Metti in funzione Kora localmente in pochi minuti.
Demo completa delle transazioni
Guida completa all'implementazione delle transazioni con astrazione delle commissioni.
Riferimento API
Metodi JSON-RPC e documentazione SDK.
Guida per operatori di nodi
Distribuisci e configura la tua istanza Kora.
Is this page helpful?