solana-go/programs/tokenregistry/instruction.go

132 lines
3.5 KiB
Go

package tokenregistry
import (
"bytes"
"encoding/binary"
"fmt"
"github.com/dfuse-io/solana-go/text"
bin "github.com/dfuse-io/binary"
"github.com/dfuse-io/solana-go"
)
func init() {
solana.RegisterInstructionDecoder(ProgramID(), registryDecodeInstruction)
}
func registryDecodeInstruction(accounts []*solana.AccountMeta, rawInstruction *solana.CompiledInstruction) (interface{}, error) {
inst, err := DecodeInstruction(accounts, rawInstruction)
if err != nil {
return nil, err
}
return inst, nil
}
func DecodeInstruction(accounts []*solana.AccountMeta, compiledInstruction *solana.CompiledInstruction) (*Instruction, error) {
var inst Instruction
if err := bin.NewDecoder(compiledInstruction.Data).Decode(&inst); err != nil {
return nil, fmt.Errorf("unable to decode instruction for serum program: %w", err)
}
if v, ok := inst.Impl.(solana.AccountSettable); ok {
err := v.SetAccounts(accounts, compiledInstruction.Accounts)
if err != nil {
return nil, fmt.Errorf("unable to set accounts for instruction: %w", err)
}
}
return &inst, nil
}
func NewRegisterTokenInstruction(logo Logo, name Name, symbol Symbol, website Website, tokenMetaKey, ownerKey, tokenKey solana.PublicKey) *Instruction {
return &Instruction{
BaseVariant: bin.BaseVariant{
TypeID: 0,
Impl: &RegisterToken{
Logo: logo,
Name: name,
Website: website,
Symbol: symbol,
Accounts: &RegisterTokenAccounts{
TokenMeta: &solana.AccountMeta{tokenMetaKey, false, true},
Owner: &solana.AccountMeta{ownerKey, true, false},
Token: &solana.AccountMeta{tokenKey, false, false},
},
},
},
}
}
type Instruction struct {
bin.BaseVariant
}
func (i *Instruction) Accounts() (out []*solana.AccountMeta) {
switch i.TypeID {
case 0:
accounts := i.Impl.(*RegisterToken).Accounts
out = []*solana.AccountMeta{accounts.TokenMeta, accounts.Owner, accounts.Token}
}
return
}
func (i *Instruction) ProgramID() solana.PublicKey {
return ProgramID()
}
func (i *Instruction) Data() ([]byte, error) {
buf := new(bytes.Buffer)
if err := bin.NewEncoder(buf).Encode(i); err != nil {
return nil, fmt.Errorf("unable to encode instruction: %w", err)
}
return buf.Bytes(), nil
}
var InstructionDefVariant = bin.NewVariantDefinition(bin.Uint32TypeIDEncoding, []bin.VariantType{
{"register_token", (*RegisterToken)(nil)},
})
func (i *Instruction) TextEncode(encoder *text.Encoder, option *text.Option) error {
return encoder.Encode(i.Impl, option)
}
func (i *Instruction) UnmarshalBinary(decoder *bin.Decoder) (err error) {
return i.BaseVariant.UnmarshalBinaryVariant(decoder, InstructionDefVariant)
}
func (i *Instruction) MarshalBinary(encoder *bin.Encoder) error {
err := encoder.WriteUint32(i.TypeID, binary.LittleEndian)
if err != nil {
return fmt.Errorf("unable to write variant type: %w", err)
}
return encoder.Encode(i.Impl)
}
type RegisterTokenAccounts struct {
TokenMeta *solana.AccountMeta `text:"linear,notype"`
Owner *solana.AccountMeta `text:"linear,notype"`
Token *solana.AccountMeta `text:"linear,notype"`
}
type RegisterToken struct {
Logo Logo
Name Name
Website Website
Symbol Symbol
Accounts *RegisterTokenAccounts `bin:"-"`
}
func (i *RegisterToken) SetAccounts(accounts []*solana.AccountMeta, instructionActIdx []uint8) error {
if len(instructionActIdx) < 9 {
return fmt.Errorf("insuficient account")
}
i.Accounts = &RegisterTokenAccounts{
TokenMeta: accounts[instructionActIdx[0]],
Owner: accounts[instructionActIdx[1]],
Token: accounts[instructionActIdx[2]],
}
return nil
}