Sponsored Transactions
One of the unique features of the Stacks blockchain is that it has native support for "sponsored transactions". A sponsored transaction is signed by a user, but the fee is paid for by a third party. This allows for improved user experiences where a user doesn't need to own Stacks in order to make transactions.
Implementing sponsored transactions
In most cases, the flow for a sponsored transactions is:
- When signing a transaction, include
sponsored: true
as one of the transaction options - After the transaction is signed, the callback includes a
txRaw
property, which is a hex-encoded transaction. This transaction is not broadcasted to the blockchain yet. - Pass the
txRaw
encoded transaction to an API route - In the API route, sign the transaction using a different private key. The account controlled by this private key is paying the fee.
Signing transactions
For more information about signing transactions, check out the docs for contract calls.
In order to implement sponsored transactions, you must include the sponsored: true
parameter. Each of the transaction signing functions support this parameter. When this option is included, the transaction payload is signed slightly differently. Additionally, the transaction is not immediately broadcasted.
import { useOpenContractCall } from '@micro-stacks/react';
import { useCallback } from 'react';
export const ContractCall = () => {
const { openContractCall } = useOpenContractCall();
const signTx = useCallback(() => {
openContractCall({
...params, // all the usual parameters
sponsored: true,
});
}, []);
return (
<button onClick={signTx}>Sign Transaction</button>
);
}
Server-side sponsoring transactions
When implementing sponsored transactions, we need a backend API that supports signing a transaction as the sponsor. To do this, use the sponsorTransaction
function from micro-stacks/transactions
. We'll need to provide the transaction and our private key as parameters.
You'll need a private key in order to sign the transaction as a sponsor. In this example, the private key is configured as an environment variable.
import { deserializeTransaction, sponsorTransaction, broadcastTransaction } from 'micro-stacks/transactions';
export const sponsorHex = async (txHex: string) => {
const tranaction = deserializeTransaction(txHex);
const sponsorPrivateKey = process.env.SPONSOR_PRIVATE_KEY;
if (!sponsorPrivateKey) throw new Error('SPONSOR_PRIVATE_KEY is required');
const network = new StacksTestnet();
const sponsoredTx = await sponsorTransaction({
transaction,
sponsorPrivateKey: sponsorPrivateKey,
network,
fee: '1000',
})
const result = await broadcastTransaction(sponsoredTx, network);
if ('error' in result) {
throw new Error(`Broadcast failed: ${result.error}`);
}
return result.txid;
}
The sponsorTransaction
parameters are:
transaction
: AStacksTransaction
instance. This must already be signed by the original sendersponsorPrivateKey
: a hex private key, which corresponds to the account paying the transaction feenetwork
: theStacksNetwork
we want to interface withfee
: optionally, a fee can be provided. Iffee
is omitted, then a fee estimation is used.sponsorNonce
: optionally, we can specify the nonce used. This is helpful in cases where our backend needs to handle many consecutive sponsored transactions.
Putting it all together
Now that we have a backend API setup to handle sponsoring, we can use it after signing the transaction.
import { useOpenContractCall } from '@micro-stacks/react';
import { useCallback } from 'react';
export const ContractCall = () => {
const { openContractCall } = useOpenContractCall();
const signTx = useCallback(() => {
const receipt = openContractCall({
...params, // all the usual parameters
sponsored: true,
});
if (typeof receipt === 'undefined') return; // if the popup was closed
// Send it to our API
const result = await fetch('/api/sponsor', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ txHex }),
})
// get the txid from the response
const { txid } = await result.json();
console.log('Transaction submitted!', txid);
}, [openContractCall]);
return (
<button onClick={signTx}>Sign Transaction</button>
);
}
Advanced Usage
Authorizing sponsored transactions
Most of the time, you'll want to only sponsor specific transactions. For example, you might want to only sponsor your app's contracts, or only sponsor transactions for specific users. You can implement that kind of authorization before signing.
export function verifyContractCall(tx: StacksTransaction) {
// only contract call types
if (tx.payload.payloadType !== 2) return false;
const myContract = createAddress('SP123.my-contract');
// validate contract address
if (tx.payload.contractAddress !== myContract) return false;
// maybe authorize the signer
if (!verifySigner(tx.auth.spendingCondition.signer)) return false;
return true;
}
export async function sponsorTx(tx: StacksTransaction) {
if (!verifyContractCall(tx)) {
throw new Error('Unauthorized');
}
// continue sponsoring
}
Managing sponsor nonce
In production, you might want to keep track of your sponsor account's nonce in a custom way. By default, sponsorTransaction
fetches the sponsor's nonce from the network. If you have many requests to sponsor transactions coming in simultaneously, you should keep track of your nonce using some central data store.
export async function sponsorTx(tx: StacksTransaction) {
// some method to fetch and update your next nonce
const nonce = atomicFetchNextNonce();
await sponsorTransaction({
sponsorNonce: nonce,
// other params
});
}