Command-line Interface
The following explains the instructions available in the Stake Pool Program along with examples using the command-line utility.
Installation
The spl-stake-pool
command-line utility can be used to experiment with SPL
tokens. Once you have Rust installed, run:
Run spl-stake-pool --help
for a full description of available commands.
Configuration
The spl-stake-pool
configuration is shared with the solana
command-line tool.
Current Configuration
Cluster RPC URL
See Solana clusters for cluster-specific RPC URLs
Default Keypair
See Keypair conventions for information on how to setup a keypair if you don't already have one.
Keypair File
Hardware Wallet URL (See URL spec)
Running Locally
If you would like to test a stake pool locally without having to wait for stakes
to activate and deactivate, you can run the stake pool locally using the
solana-test-validator
tool with shorter epochs, and pulling the current program
from devnet.
Stake Pool Manager Examples
Create a stake pool
The stake pool manager controls the stake pool from a high level, and in exchange receives a fee in the form of SPL tokens. The manager sets the fee on creation. Let's create a pool with a 3% fee and a maximum of 1000 validator stake accounts:
The unique stake pool identifier is Zg5YBPAk8RqBR9kaLLSoN5C8Uv7nErBz1WC63HTsCPR
.
The identifier for the stake pool's SPL token mint is
BoNneHKDrX9BHjjvSpPfnQyRjsnc9WFH71v8wrgCd7LB
. The stake pool has full control
over the mint.
The pool creator's fee account identifier is
DgyZrAq88bnG1TNRxpgDQzWXpzEurCvfY2ukKFWBvADQ
. Every epoch, as stake accounts
in the stake pool earn rewards, the program will mint SPL pool tokens
equal to 3% of the gains on that epoch into this account. If no gains were observed,
nothing will be deposited.
The reserve stake account identifier is J5XB7mWpeaUZxZ6ogXT57qSCobczx27vLZYSgfSbZoBB
.
This account holds onto additional stake used when rebalancing between validators.
For a stake pool with 1000 validators, the cost to create a stake pool is less than 0.5 SOL.
The create-pool
command allows setting all of the accounts and keypairs to
pre-generated values, including:
- stake pool, through the
--pool-keypair
flag - validator list, through the
--validator-list-keypair
flag - pool token mint, through the
--mint-keypair
flag - pool reserve stake account, through the
--reserve-keypair
flag
Otherwise, these will all default to newly-generated keypairs.
You can always check out the available options by running spl-stake-pool create-pool -h
.
Create a restricted stake pool
If a manager would like to restrict deposits (stake and SOL) to one key in particular, they can set a deposit authority at creation:
As the output says, the set-funding-authority
can be used to modify or remove
the deposit authority.
As long as the deposit authority is set, SOL and stake deposits must be signed
by 4SnSuUtJGKvk2GYpBwmEsWG53zTurVM8yXGsoiZQyMJn
, so no one else can participate
in the pool. As mentioned earlier, this feature does not prohibit withdrawals,
so anyone with pool tokens will still be able to withdraw from the pool.
Set manager
The stake pool manager may pass their administrator privileges to another account.
At the same time, they may also change the SPL token account that receives fees
every epoch. The mint for the provided token account must be the SPL token mint,
BoNneHKDrX9BHjjvSpPfnQyRjsnc9WFH71v8wrgCd7LB
in our example.
Set fee
The stake pool manager may update any of the fees associated with the stake pool, passing the numerator and denominator for the fraction that make up the fee.
For an epoch fee of 10%, they could run:
In order to protect stake pool depositors from malicious managers, the program applies the new fee after crossing two epoch boundaries, giving a minimum wait time of one full epoch.
For example, if the fee is 1% at epoch 100, and the manager sets it to 10%, the manager will still gain 1% for the rewards earned during epochs 100 and 101. Starting with epoch 102, the manager will earn 10%.
Additionally, to prevent a malicious manager from immediately setting the withdrawal fee to a very high amount, making it practically impossible for users to withdraw, the stake pool program currently enforces a limit of 1.5x increase every two epoch boundaries.
For example, if the current withdrawal fee is 2.5%, the maximum settable fee is 3.75%, and will take effect after two epoch boundaries.
The possible options for the fee type are epoch
, sol-withdrawal
,
stake-withdrawal
, sol-deposit
, and stake-deposit
.
Set referral fee
The stake pool manager may update the referral fee on deposits at any time, passing in a percentage amount.
To set a stake deposit referral fee of 80%, they may run:
For 80%, this means that 20% of the stake deposit fee goes to the manager, and 80% goes to the referrer.
Set staker
In order to manage the stake accounts, the stake pool manager or staker can set the staker authority of the stake pool's managed accounts.
Now, the new staker can perform any normal stake pool operations, including adding and removing validators and rebalancing stake.
Important security note: the stake pool program only gives staking authority to the pool staker and always retains withdraw authority. Therefore, a malicious stake pool staker cannot steal funds from the stake pool.
Note: to avoid "disturbing the manager", the staker can also reassign their stake authority.
Set Funding Authority
To restrict who can interact with the pool, the stake pool manager may require a particular signature on stake deposits, SOL deposits, or SOL withdrawals. This does not make the pool private, since all information is available on-chain, but it restricts who can use the pool.
As an example, let's say a pool wants to restrict all SOL withdrawals.
After running this command, AZ1PgxWSxw4ezX8gvpNgGsr39jJHCwtkaXr1mNMwWWeK
must
sign all SOL withdrawals, otherwise the operation fails.
After some time, if the manager wishes to enable SOL withdrawals, they can remove the restriction:
Now, anyone can withdraw SOL from the stake pool, provided there is enough SOL left in the reserve.
The options for funding authorities are sol-withdraw
, sol-deposit
, and stake-deposit
.
Note: it is impossible to restrict stake withdrawals. This would create an opportunity for malicious pool managers to effectively lock user funds.
Stake Pool Staker Examples
Add a validator to the pool
In order to accommodate large numbers of user deposits into the stake pool, the
stake pool only manages one stake account per validator. To add a new validator
to the stake pool, the staker must use the add-validator
command.
The SOL used to add validators to the pool comes from the stake pool's reserve
account. If there is insufficient SOL in the reserve, the command will fail.
Be sure to use the deposit-sol
command to move some SOL into the pool.
With 10 SOL in the pool, let's add some random validators to the stake pool.
In order to maximize censorship resistance, we want to distribute our SOL to as many validators as possible, so let's add a few more.
We can see the status of a stake account using the Solana command-line utility.
The stake pool creates these special staking accounts with 1 SOL as the required minimum delegation amount. The stake and withdraw authorities are the stake pool withdraw authority, program addresses derived from the stake pool's address.
We can also see the status of the stake pool.
To make reading easier, the tool will not show balances that cannot be touched by
the stake pool. The reserve stake account EN4px2h4gFkYtsQUi4yeCYBrdRM4DoRxCVJyavMXEAm5
actually has an additional balance of 0.002282881 SOL, but since this is the minimum
required amount, it is not shown by the CLI.
Remove validator stake account
If the stake pool staker wants to stop delegating to a vote account, they can totally remove the validator stake account from the stake pool.
As with adding a validator, the validator stake account must have exactly 1.00228288 SOL (1 SOL delegated, 0.00228288 SOL for rent exemption) to be removed.
If that is not the case, the staker must first decrease the stake to that minimum amount.
Let's assume that the validator stake account delegated to
J3xu64PWShcMen99kU3igxtwbke2Nwfo8pkZNRgrq66H
has a total delegated amount of
7.5 SOL. To reduce that number, the staker can run:
Now, let's try to remove validator J3xu64PWShcMen99kU3igxtwbke2Nwfo8pkZNRgrq66H
, with
stake account 5AaobwjccyHnXhFCd24uiX6VqPjXE3Ry4o92fJjqqjAr
.
Unlike a normal withdrawal, the validator stake account is deactivated, and then merged into the reserve during the next epoch.
We can check the deactivating stake account:
Rebalance the stake pool
As time goes on, users will deposit to and withdraw from all of the stake accounts managed by the pool, and the stake pool staker may want to rebalance the stakes.
For example, let's say the staker wants the same delegation to every validator in the pool. When they look at the state of the pool, they see:
This isn't great! The first stake account, EhRbKi4Vhm1oUCGWHiLEMYZqDrHwEd7Jgzgi26QJKvfQ
has too much allocated. For their strategy, the staker wants the 100
SOL to be distributed evenly, meaning 40
in each account. They need
to move 30
to J3xu64PWShcMen99kU3igxtwbke2Nwfo8pkZNRgrq66H
and
38DYMkwYCvsj8TC6cNaEvFHHVDYeWDp1qUgMgyjNqZXk
.
Decrease validator stake
First, they need to decrease the amount on stake account
3k7Nwu9jUSc6SNG11wzufKYoZXRFgxWamheGLYWp5Rvx
, delegated to
EhRbKi4Vhm1oUCGWHiLEMYZqDrHwEd7Jgzgi26QJKvfQ
, by a total of 60
SOL.
They decrease that amount of SOL:
Internally, this instruction splits and deactivates 60 SOL from the
validator stake account 3k7Nwu9jUSc6SNG11wzufKYoZXRFgxWamheGLYWp5Rvx
into a
transient stake account, owned and managed entirely by the stake pool.
Once the stake is deactivated during the next epoch, the update
command will
automatically merge the transient stake account into a reserve stake account,
also entirely owned and managed by the stake pool.
Increase validator stake
Now that the reserve stake account has enough to perform the rebalance, the staker
can increase the stake on the two other validators,
J3xu64PWShcMen99kU3igxtwbke2Nwfo8pkZNRgrq66H
and
38DYMkwYCvsj8TC6cNaEvFHHVDYeWDp1qUgMgyjNqZXk
.
They add 30 SOL to J3xu64PWShcMen99kU3igxtwbke2Nwfo8pkZNRgrq66H
:
And they add 30 SOL to 38DYMkwYCvsj8TC6cNaEvFHHVDYeWDp1qUgMgyjNqZXk
:
Internally, this instruction also uses transient stake accounts. This time, the stake pool splits from the reserve stake, into the transient stake account, then activates it to the appropriate validator.
One to two epochs later, once the transient stakes activate, the update
command
automatically merges the transient stakes into the validator stake account, leaving
a fully rebalanced stake pool:
Due to staking rewards that accrued during the rebalancing process, the pool may not perfectly balanced. This is completely normal.
Set Preferred Deposit / Withdraw Validator
Since a stake pool accepts deposits to any of its stake accounts, and allows withdrawals from any of its stake accounts, it could be used by malicious arbitrageurs looking to maximize returns each epoch.
For example, if a stake pool has 1000 validators, an arbitrageur could stake to any one of those validators. At the end of the epoch, they can check which validator has the best performance, deposit their stake, and immediately withdraw from the highest performing validator. Once rewards are paid out, they can take their valuable stake, and deposit it back for more than they had.
To mitigate this arbitrage, a stake pool staker can set a preferred withdraw or deposit validator. Any deposits or withdrawals must go to the corresponding stake account, making this attack impossible without a lot of funds.
Let's set a preferred deposit validator stake account:
And then let's set the preferred withdraw validator stake account to the same one:
At any time, they may also unset the preferred validator:
The preferred validators are marked in the list
command:
User Examples
List validator stake accounts
In order to deposit into the stake pool, a user must first delegate some stake to one of the validator stake accounts associated with the stake pool. The command-line utility has a special instruction for finding out which vote accounts are already associated with the stake pool.
Deposit SOL
Stake pools accept SOL deposits directly from a normal SOL wallet account, and in exchange mint the appropriate amount of pool tokens.
In return, the stake pool has minted us new pool tokens, representing our share of ownership in the pool. We can double-check our stake pool account using the SPL token command-line utility.
Withdraw SOL
Stake pools allow SOL withdrawals directly from the reserve and into a normal SOL wallet account, and in exchange burns the provided pool tokens.
The stake pool has burned 2 pool tokens, and in return, sent SOL to
7VXPpSxneL6JLj18Naw2gkukXtjBZfbmPh18cnoUCMD8
.
You can check that the pool tokens have been burned:
And you can check that the recipient has been credited:
Deposit stake
Stake pools also accept deposits from active stake accounts, so we must first
create stake accounts and delegate them to one of the validators managed by the
stake pool. Using the list
command from the previous section, we see that
38DYMkwYCvsj8TC6cNaEvFHHVDYeWDp1qUgMgyjNqZXk
is a valid vote account, so let's
create a stake account and delegate our stake there.
Two epochs later, when the stake is fully active and has received one epoch of rewards, we can deposit the stake into the stake pool.
The CLI will default to using the fee payer's Associated Token Account for stake pool tokens and the withdraw authority on the deposited stake account.
Alternatively, you can create an SPL token account yourself and pass it as the
token-receiver
for the command, and specify the withdraw authority on the
stake account using the withdraw-authority
flag.
In return, the stake pool has minted us new pool tokens, representing our share of ownership in the pool. We can double-check our stake pool account using the SPL token command-line utility.
Note on stake deposit fee
Stake pools have separate fees for stake and SOL, so the total fee from depositing a stake account is calculated from the rent-exempt reserve as SOL, and the delegation as stake.
For example, if a stake pool has a stake deposit fee of 1%, and a SOL deposit fee of 5%, and you deposit a stake account with 10 SOL in stake, and .00228288 SOL in rent-exemption, the total fee charged is:
Update
Every epoch, the network pays out rewards to stake accounts managed by the stake pool, increasing the value of pool tokens minted on deposit. In order to calculate the proper value of these stake pool tokens, we must update the total value managed by the stake pool every epoch.
If another user already updated the stake pool balance for the current epoch, we see a different output.
If no one updates the stake pool in the current epoch, all instructions, including deposit and withdraw, will fail. The update instruction is permissionless, so any user can run it before interacting with the pool. As a convenience, the CLI attempts to update before running any instruction on the stake pool.
If the stake pool transient stakes are in an unexpected state, and merges are
not possible, there is the option to only update the stake pool balances without
performing merges using the --no-merge
flag.
Later on, whenever the transient stakes are ready to be merged, it is possible to
force another update in the same epoch using the --force
flag.
Withdraw stake
Whenever the user wants to recover their SOL plus accrued rewards, they can provide their pool tokens in exchange for an activated stake account.
Let's withdraw active staked SOL in exchange for 5 pool tokens.
The stake pool took 5 pool tokens, and in exchange the user received a fully
active stake account, delegated to EhRbKi4Vhm1oUCGWHiLEMYZqDrHwEd7Jgzgi26QJKvfQ
.
Let's double-check the status of the stake account:
Note: this operation cost the user some funds, as they needed to create a new stake account with the minimum rent exemption in order to receive the funds. This allows the user to withdraw any amount of stake pool tokens, even if it is not enough to cover the stake account rent-exemption.
Alternatively, the user can specify an existing uninitialized stake account to
receive their stake using the --stake-receiver
parameter.
By default, the withdraw command uses the token-owner
's associated token account to
source the pool tokens. It's possible to specify the SPL token account using
the --pool-account
flag.
By default, the withdraw command will withdraw from the largest validator stake
accounts in the pool. It's also possible to specify a specific vote account for
the withdraw using the --vote-account
flag.
Note that the associated validator stake account must have enough lamports to satisfy the pool token amount requested.
Special case: exiting pool with a delinquent staker
With the reserve stake, it's possible for a delinquent or malicious staker to
move all stake into the reserve through decrease-validator-stake
, so the
pool tokens will not gain rewards, and the stake pool users will not
be able to withdraw their funds.
To get around this case, it is also possible to withdraw from the stake pool's
reserve, but only if all of the validator stake accounts are at the minimum amount of
1 SOL + stake account rent exemption
.
Special case: removing validator from the pool
Since the funds used to add validators to the pool come from outside deposits, it's possible for a delinquent or malicious staker to make it impossible for users to reclaim their SOL by keeping everything at the minimum amount.
To get around this case, it is also possible to remove a validator from the stake pool
but only if all of the validator stake accounts are at the minimum amount of
1 SOL + stake account rent exemption
.