2 releases
0.1.1 | Mar 13, 2025 |
---|---|
0.1.0 | Mar 6, 2025 |
#14 in #procedural-macro
285 downloads per month
29KB
447 lines
anchor-decoder
A procedural macro to help with decoding Solana accounts and instructions for programs written with the Anchor framework.
Note: this tool relies on IDLs generated by Anchor 0.30.0. Fortunately, there's a CLI command to help convert legacy IDLs to the new format.
Usage
Add the crate to your Cargo.toml
, and point the macro at your IDL:
use anchor_decoder::anchor_idl;
#[anchor_idl("./idl.json")]
pub const ID: Pubkey = crate::ID;
You can also call the macro multiple times in the same files by using separate modules.
pub mod production {
use anchor_decoder_gen::anchor_idl;
#[anchor_idl("./production.json")]
pub const ID: Pubkey = crate::ID;
}
pub mod staging {
use anchor_decoder_gen::anchor_idl;
#[anchor_idl("./staging.json")]
pub const ID: Pubkey = crate::ID;
}
Using Decoders
I recommend creating separate crates for each parser. You can then install and import crates as needed. Assume you have a separate crate called program_decoder
. You can then use the decoders as follows:
Decode instructions
use program_decoder::{decode_instruction, DecodedInstruction, Instruction1, Instruction2};
let ix: CompiledInstruction = ...;
let account_keys: Vec<Pubkey> = ...;
match decode_instruction(&ix.data) {
// matches the decoded instructions, with or without ix args
Some(DecodedInstruction::Instruction1) => {
// convert the array of accounts into a map based on IDL definition
let accounts_map = Instruction1::map_accounts(account_keys);
println!("Instruction1: {:?}", accounts_map);
}
Some(DecodedInstruction::Instruction2(decoded)) => {
let accounts_map = Instruction2::map_accounts(account_keys);
// can now access Instruction2 arg fields
println!("Instruction2: {:?}, data: {:?}", accounts_map, decoded);
}
// handle the case when no match arms are found
None => {
println!("Failed to decode PROgraM1111111111111111111111111 instruction")
}
};
Decode accounts
use program_decoder::{decode_account, DecodedAccount, Account1};
let account_data: Vec<u8> = ...;
let account_key: Pubkey = ...;
// account decoder shown here, but it works the same for `decode_instruction`
match decode_account(&account_data) {
Some(DecodedAccount::Account1(decoded)) => {
// can now access Account1 struct fields
println!("Account1: {:?}", decoded);
}
// handle the case when no match arms are found
None => println!("Failed to decode PROgraM1111111111111111111111111 account: {:?}", account_key),
}
There are more examples in the examples/ directory.
License
The project is licensed under Apache 2.0.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion by you, as defined in the Apache-2.0 license, shall be licensed as above, without any additional terms or conditions.
Contribution
Any contributions are welcome. If you notice there's an IDL that doesn't breaks the macro, please open an issue or submit a pull request.
Dependencies
~4.5MB
~86K SLoC