Wallet Guide
This guide is meant for wallet developers who want to support Token-2022.
Since wallets have very different internals for managing token account state and connections to blockchains, this guide will focus on the very specific changes required, without only vague mentions of code design.
Motivation
Wallet developers are accustomed to only including one token program used for all tokens.
To properly support Token-2022, wallet developers must make code changes.
Important note: if you do not wish to support Token-2022, you do not need to do anything. The wallet will not load Token-2022 accounts, and transactions created by the wallet will fail loudly if using Token-2022 incorrectly.
Most likely, transactions will fail with ProgramError::IncorrectProgramId
when trying to target the Token program with Token-2022 accounts.
Prerequisites
When testing locally, be sure to use at least solana-test-validator
version
1.14.17, which includes the Token-2022 program by default. This comes bundled
with version 2.3.0 of the spl-token
CLI, which also supports Token-2022.
Setup
You'll need some Token-2022 tokens for testing. First, create a mint with an extension. We'll use the "Mint Close Authority" extension:
The extension is important because it will test that your wallet properly handles larger mint accounts.
Next, create an account for your test wallet:
With the --owner
parameter, the new account is an associated token account,
which includes the "Immutable Owner" account extension. This way, you'll also
test larger token accounts.
Finally, mint some tokens:
It's also helpful for your test wallet to have some SOL, so be sure to transfer some:
Finally, you can save all of these accounts in a directory to be re-used for testing:
This way, whenever you want to restart your test validator, you can simply run:
Structure of this Guide
We'll go through the required code changes to support Token-2022 in your wallet, using only little code snippets. This work was done for the Backpack wallet in PR #3976, but as mentioned earlier, the actual code changes may look very different for your wallet.
Part I: Fetch Token-2022 Accounts
In addition to normal Token accounts, your wallet must also fetch Token-2022
accounts. Typically, wallets use the getTokenAccountsByOwner
RPC endpoint once
to fetch the accounts.
For Token-2022, you simply need to add one more call to get the additional accounts:
Merge the two responses, and you're good to go! If you can see your test account, then you've done it correctly.
If there are issues, your wallet may be deserializing the token account too strictly, so be sure to relax any restriction that the data size must be equal to 165 bytes.
Part II: Use the Token Program Id for Instructions
If you try to transfer or burn a Token-2022 token, you will likely receive an error because the wallet is trying to send an instruction to Token instead of Token-2022.
Here are two possible ways to resolve the problem.
Option 1: Store the token account's owner during fetch
In the first part, we fetched all of the token accounts and threw away the program id associated with the account. Instead of always targeting the Token program, we need to target the right program for that token.
If we store the program id for each token account, then we can re-use that information when we need to transfer or burn.
Option 2: Fetch the program owner before transfer / burn
This approach introduces one more network call, but may be simpler to integrate.
Before creating an instruction, you can fetch the mint, source account, or
destination account from the network, and pull out its owner
field.
Part III: Use the Token Program Id for Associated Token Accounts
Whenever we derive an associated token account, we must use the correct token program id. Currently, most implementations hardcode the token program id. Instead, you must add the program id as a parameter:
If you're creating associated token accounts, you'll also need to pass the
token program id, which currently defaults to TOKEN_PROGRAM_ID
:
With these three parts done, your wallet will provide basic support for Token-2022!