📩Sending Transactions on Solana
Optimize your transactions to minimize confirmation latency and maximize delivery rates.
Helius' staked connections guarantee 100% transaction delivery with minimal confirmation times.
Summary
Optimize your landing rates by following these best practices:
Use staked connections.
Set competitive priority fees.
Set minimal compute units.
Rebroadcast transactions until confirmation.
Want to go deeper? We cover all fundamentals in this blog post if you prefer to optimize transactions yourself.
Staked Connections
We offer two ways to access staked connections via any paid plan:
Mainnet Endpoint (default)
Requirements:
Priority fees meet or exceed the
recommended
value provided by the Priority Fee APISet
maxRetries
to 0
Endpoint:
https://mainnet.helius-rpc.com/?api-key=xxx
Cost: 1 credit per request
Staked Endpoint (premium)
Transactions will be sent over staked connections via an optimized network (Asia, Europe, North America), minimizing latency and confirmation times. The fastest way to send transactions on Solana.
Endpoint:
https://staked.helius-rpc.com?api-key=xxx
Cost: 500 credits per request (subject to change)
Usage metrics display staked endpoint calls assendTransactionWithStake
to avoid collisions with regular sendTransaction.
Best Practices
We recommend the following:
Optimizing a transaction's compute unit (CU) usage
As of 1.18, transactions requiring fewer CUs will be given higher priority. In our Node.js SDK, the
getComputeUnits
helper method can be used to simulate a transaction based on a set of instructions and address lookup tables, returning the amount of CUs consumed. Similarly, our Rust SDK has theget_compute_units
helper method providing the same functionality.
Implementing priority fees and calculating them dynamically
Implementing robust retry logic
Set
maxRetries
to 0
Using durable nonces for transactions that are time-insensitive
Send transactions via Staked Connections
Sending Smart Transactions
Both the Helius Node.js and Rust SDKs can send smart transactions. This new method builds and sends an optimized transaction while handling its confirmation status. Users can configure the transaction's send options, such as whether the transaction should skip preflight checks.
At the most basic level, users must supply their keypair and the instructions they wish to execute, and we handle the rest.
We:
Fetch the latest blockhash
Build the initial transaction
Simulate the initial transaction to fetch the compute units consumed
Set the compute unit limit to the compute units consumed in the previous step, with some margin
Get the Helius recommended priority fee via our Priority Fee API
Set the priority fee (microlamports per compute unit) as the Helius recommended fee
Adds a small safety buffer fee in case the recommended value changes in the next few seconds
Build and send the optimized transaction
Return the transaction signature, if successful
Requiring the recommended value (or higher) for our staked connections ensures that Helius sends high-quality transactions and that we wont be rate-limited by validators.
This method is designed to be the easiest way to build, send, and land a transaction on Solana. Note that by using the Helius recommended fee, transactions sent by Helius users on one of our paid shared plans will be routed through our staked connections, guaranteeing near 100% transaction delivery.
Node.js SDK
The sendSmartTransaction
method is available in our Helius Node.js SDK for versions >= 1.3.2. To update to a more recent version of the SDK, run npm update helius-sdk
.
The following example transfers SOL to an account of your choice. It leverages sendSmartTransaction
to send an optimized transaction that does not skip preflight checks
Rust SDK
The send_smart_transaction
method is available in our Rust SDK for versions >= 0.1.5. To update to a more recent version of the SDK, run cargo update helius
.
The following example transfers 0.01 SOL to an account of your choice. It leverages send_smart_transaction
to send an optimized transaction that skips preflight checks and retries twice, if necessary:
Sending Transactions Without the SDK
We recommend sending smart transactions with one of our SDKs but the same functionality can be achieved without using one. Both the Node.js SDK and Rust SDK are open-source, so the underlying code for the send smart transaction functionality can be viewed anytime.
Prepare and Build the Initial Transaction
First, prepare and build the initial transaction. This includes creating a new transaction with a set of instructions, adding the recent blockhash, and assigning a fee payer. For versioned transactions, create a TransactionMessage
and compile it with lookup tables if any are present. Then, create a new versioned transaction and sign it — this is necessary for the next step when we simulate the transaction, as the transaction must be signed.
For example, if we wanted to prepare a versioned transaction:
Optimize the Transaction's Compute Unit (CU) Usage
To optimize the transaction's compute unit (CU) usage, we can use the simulateTransaction
RPC method to simulate the transaction. Simulating the transaction will return the amount of CUs used, so we can use this value to set our compute limit accordingly. It's recommended to use a test transaction with the desired instructions first, plus an instruction that sets the compute limit to 1.4m CUs. This is done to ensure the transaction simulation succeeds. For example:
It is also recommended to add a bit of margin to ensure the transaction executes without any issues. We can do so by setting the following:
Then, create an instruction that sets the compute unit limit to this value and add it to your array of instructions:
Serialize and Encode the Transaction
This is relatively straightforward. First, to serialize the transaction, both Transaction
and VersionedTransaction
types have a .serialize()
method. Then use the bs58 package to encode the transaction. Your code should look something like bs58.encode(txt.serialize());
Setting the Right Priority Fee
First, use the Priority Fee API to get the priority fee estimate. We want to pass in our transaction and get the Helius recommended fee via the recommended
parameter:
Then, create an instruction that sets the compute unit price to this value, and add that instruction to your previous instructions:
Build and Send the Optimized Transaction
This step is almost a repeat of the first step. However, the array of initial instructions has been altered to add two instructions to set the compute unit limit and price optimally. Now, send the transaction. It doesn't matter if you send with or without preflight checks or change any other send options — the transaction will be routed through our staked connections.
Polling the Transaction's Status and Rebroadcasting
While staked connections will forward a transaction directly to the leader, it is still possible for the transaction to be dropped in the Banking Stage. It is recommended that users employ their own rebroadcasting logic rather than rely on the RPC to retry the transaction for them.
The sendTransaction
RPC method has a maxRetries
parameter that can be set to override the RPC's default retry logic, giving developers more control over the retry process. It is a common pattern to fetch the current blockhash via getLatestBlockhash
, store the lastValidBlockHeight
, and retry the transaction until the blockhash expires. It is crucial to only re-sign a transaction when the blockhash is no longer valid, or else it is possible for both transactions to be accepted by the network.
Once a transaction is sent, it is important to poll its confirmation status to see whether the network has processed and confirmed it before retrying. Use the getSignatureStatuses
RPC method to check a list of transactions' confirmation status. The @solana/web3.js SDK also has a getSignatureStatus
method on its Connection
class to fetch the current status of a given signature.
How sendSmartTransaction
Handles Polling and Rebroadcasting
sendSmartTransaction
Handles Polling and RebroadcastingThe sendSmartTransaction
method has a timeout period of 60 seconds. Since a blockhash is valid for 150 slots, and assuming perfect 400ms slots, we can reasonably assume a transaction's blockhash will be invalid after one minute. The method sends the transaction and polls its transaction signature using this timeout period:
txtSig
is set to the signature of the transaction that was just sent. The method then uses the pollTransactionConfirmation()
method to poll the transaction's confirmation status. This method checks a transaction's status every five seconds for a maximum of three times. If the transaction is not confirmed during this time, an error is returned:
We continue sending the transaction, polling its confirmation status, and retrying it until a minute has elapsed. If the transaction has not been confirmed at this time, an error is thrown.
Last updated