Examples
More examples can be found in the Transfer Hook example tests, as well as the TLV Account Resolution tests.
Initializing Extra Account Metas On-Chain
The
ExtraAccountMetaList
struct is designed to make working with extra account
configurations as seamless as possible.
Using ExtraAccountMetaList::init<T>(..)
, you can initialize a buffer with the
serialized ExtraAccountMeta
configurations by simply providing a mutable
reference to the buffer and a slice of ExtraAccountMeta
. The generic T
is
the instruction whose discriminator the extra account configurations should be
assigned to. In our case, this will be
spl_transfer_hook_interface::instruction::ExecuteInstruction
from the Transfer Hook interface.
Note: All instructions from the SPL Transfer Hook interface implement the trait
SplDiscriminate
, which provides a constant 8-byte discriminator that can be used to create a TLV data entry.
After calling ExtraAccountMetaList::init::<ExecuteInstruction>(..)
on the
mutable account data, the account now stores all of the serialized extra account
configurations for an Execute
instruction!
Resolving Extra Account Metas Off-Chain
When building a transaction with an instruction, either for your transfer hook program directly or for a program that will CPI to your transfer hook program, you must include all required accounts - including the extra accounts.
Below is an example of the logic contained in the Transfer Hook interface's offchain helper.
As you can see from the example, an important concept to remember is which instruction these extra accounts are for. Even though you might be building an instruction for some other program, which may not need them, if that program is going to CPI to your transfer hook program, it needs to have the proper accounts.
Additionally, in order to perform a successful dynamic account resolution, the
proper instruction needs to be provided to align with the instruction that was
configured in the validation account - in this case the Transfer Hook
interface's ExecuteInstruction
. This is why we first create an
ExecuteInstruction
, then resolve the extra accounts for that instruction, and
finally add those accounts to our current instruction.
Resolving Extra Account Metas On-Chain for CPI
During the execution of a program that seeks to CPI to your transfer hook program, even though the additional required accounts were provided by the offchain account resolution, the executing program has to know how to build a CPI instruction with the proper accounts as well!
Below is an example of the logic contained in the Transfer Hook interface's onchain helper.
Although this example may appear more verbose than its offchain counterpart, it's actually doing the exact same steps, just with an instruction and a list of account infos, since CPI requires both.
The key difference between ExtraAccountMetaList::add_to_instruction(..)
and
ExtraAccountMetaList::add_to_cpi_instruction(..)
is that the latter method
will find the corresponding AccountInfo
in the list and add it to
cpi_account_infos
at the same time as it adds the resolved AccountMeta
to
the instruction, ensuring all resolved account keys are present in the
AccountInfo
list.