From f8716155dd534f696fce1104d08a6c2c0e1b999e Mon Sep 17 00:00:00 2001 From: Silas Lenihan Date: Mon, 30 Dec 2024 12:13:20 -0500 Subject: [PATCH] Updated codec usage --- pkg/solana/chainwriter/ccip_example_config.go | 2 +- pkg/solana/chainwriter/chain_writer.go | 73 +++++++++-------- pkg/solana/chainwriter/chain_writer_test.go | 78 ++++++++++--------- pkg/solana/chainwriter/lookups.go | 1 + 4 files changed, 81 insertions(+), 73 deletions(-) diff --git a/pkg/solana/chainwriter/ccip_example_config.go b/pkg/solana/chainwriter/ccip_example_config.go index adbd4d324..fc46794a8 100644 --- a/pkg/solana/chainwriter/ccip_example_config.go +++ b/pkg/solana/chainwriter/ccip_example_config.go @@ -37,7 +37,7 @@ func TestConfig() { Name: "RegistryTokenState", // In this case, the user configured the lookup table accounts to use a PDALookup, which // generates a list of one of more PDA accounts based on the input parameters. Specifically, - // there will be multple PDA accounts if there are multiple addresses in the message, otherwise, + // there will be multiple PDA accounts if there are multiple addresses in the message, otherwise, // there will only be one PDA account to read from. The PDA account corresponds to the lookup table. Accounts: PDALookups{ Name: "RegistryTokenState", diff --git a/pkg/solana/chainwriter/chain_writer.go b/pkg/solana/chainwriter/chain_writer.go index 4ed9c8a60..e02148d89 100644 --- a/pkg/solana/chainwriter/chain_writer.go +++ b/pkg/solana/chainwriter/chain_writer.go @@ -11,7 +11,6 @@ import ( "github.com/gagliardetto/solana-go/rpc" commoncodec "github.com/smartcontractkit/chainlink-common/pkg/codec" - "github.com/smartcontractkit/chainlink-common/pkg/codec/encodings/binary" "github.com/smartcontractkit/chainlink-common/pkg/logger" "github.com/smartcontractkit/chainlink-common/pkg/services" "github.com/smartcontractkit/chainlink-common/pkg/types" @@ -31,7 +30,8 @@ type SolanaChainWriterService struct { ge fees.Estimator config ChainWriterConfig - codecs map[string]types.Codec + parsed *codec.ParsedTypes + encoder types.Encoder services.StateMachine } @@ -62,48 +62,54 @@ type MethodConfig struct { } func NewSolanaChainWriterService(logger logger.Logger, reader client.Reader, txm txm.TxManager, ge fees.Estimator, config ChainWriterConfig) (*SolanaChainWriterService, error) { - codecs, err := parseIDLCodecs(config) - if err != nil { - return nil, fmt.Errorf("failed to parse IDL codecs: %w", err) - } - - return &SolanaChainWriterService{ + w := SolanaChainWriterService{ lggr: logger, reader: reader, txm: txm, ge: ge, config: config, - codecs: codecs, - }, nil + parsed: &codec.ParsedTypes{EncoderDefs: map[string]codec.Entry{}, DecoderDefs: map[string]codec.Entry{}}, + } + + if err := w.parsePrograms(config); err != nil { + return nil, fmt.Errorf("failed to parse programs: %w", err) + } + + var err error + if w.encoder, err = w.parsed.ToCodec(); err != nil { + return nil, fmt.Errorf("%w: failed to create codec", err) + } + + return &w, nil } -func parseIDLCodecs(config ChainWriterConfig) (map[string]types.Codec, error) { - codecs := make(map[string]types.Codec) +func (s *SolanaChainWriterService) parsePrograms(config ChainWriterConfig) error { for program, programConfig := range config.Programs { var idl codec.IDL if err := json.Unmarshal([]byte(programConfig.IDL), &idl); err != nil { - return nil, fmt.Errorf("failed to unmarshal IDL for program: %s, error: %w", program, err) - } - idlCodec, err := codec.NewIDLInstructionsCodec(idl, binary.LittleEndian()) - if err != nil { - return nil, fmt.Errorf("failed to create codec from IDL for program: %s, error: %w", program, err) + return fmt.Errorf("failed to unmarshal IDL for program: %s, error: %w", program, err) } for method, methodConfig := range programConfig.Methods { - if methodConfig.InputModifications != nil { - modConfig, err := methodConfig.InputModifications.ToModifier(codec.DecoderHooks...) - if err != nil { - return nil, fmt.Errorf("failed to create input modifications for method %s.%s, error: %w", program, method, err) - } - // add mods to codec - idlCodec, err = codec.NewNamedModifierCodec(idlCodec, method, modConfig) - if err != nil { - return nil, fmt.Errorf("failed to create named codec for method %s.%s, error: %w", program, method, err) - } + idlDef, err := codec.FindDefinitionFromIDL(codec.ChainConfigTypeInstructionDef, methodConfig.ChainSpecificName, idl) + if err != nil { + return err } + + inputMod, err := methodConfig.InputModifications.ToModifier(codec.DecoderHooks...) + if err != nil { + return fmt.Errorf("failed to create input modifications for method %s.%s, error: %w", program, method, err) + } + + input, err := codec.CreateCodecEntry(idlDef, methodConfig.ChainSpecificName, idl, inputMod) + if err != nil { + return fmt.Errorf("failed to create codec entry for method %s.%s, error: %w", program, method, err) + } + + s.parsed.EncoderDefs[codec.WrapItemType(true, program, method, "")] = input } - codecs[program] = idlCodec } - return codecs, nil + + return nil } /* @@ -250,16 +256,15 @@ func (s *SolanaChainWriterService) SubmitTransaction(ctx context.Context, contra } } - codec := s.codecs[contractName] - encodedPayload, err := codec.Encode(ctx, args, method) - - discriminator := GetDiscriminator(methodConfig.ChainSpecificName) - encodedPayload = append(discriminator[:], encodedPayload...) + encodedPayload, err := s.encoder.Encode(ctx, args, codec.WrapItemType(true, contractName, method, "")) if err != nil { return errorWithDebugID(fmt.Errorf("error encoding transaction payload: %w", err), debugID) } + discriminator := GetDiscriminator(methodConfig.ChainSpecificName) + encodedPayload = append(discriminator[:], encodedPayload...) + // Fetch derived and static table maps derivedTableMap, staticTableMap, err := s.ResolveLookupTables(ctx, args, methodConfig.LookupTables) if err != nil { diff --git a/pkg/solana/chainwriter/chain_writer_test.go b/pkg/solana/chainwriter/chain_writer_test.go index 03428d080..947674a2f 100644 --- a/pkg/solana/chainwriter/chain_writer_test.go +++ b/pkg/solana/chainwriter/chain_writer_test.go @@ -3,7 +3,6 @@ package chainwriter_test import ( "bytes" "errors" - "io/ioutil" "math/big" "os" "reflect" @@ -27,6 +26,12 @@ import ( txmMocks "github.com/smartcontractkit/chainlink-solana/pkg/solana/txm/mocks" ) +type Arguments struct { + LookupTable solana.PublicKey + Seed1 []byte + Seed2 []byte +} + func TestChainWriter_GetAddresses(t *testing.T) { ctx := tests.Context(t) @@ -87,7 +92,7 @@ func TestChainWriter_GetAddresses(t *testing.T) { PublicKey: chainwriter.AccountConstant{Name: "WriteTest", Address: programID.String()}, Seeds: []chainwriter.Seed{ // extract seed2 for PDA lookup - {Dynamic: chainwriter.AccountLookup{Name: "seed2", Location: "seed2"}}, + {Dynamic: chainwriter.AccountLookup{Name: "Seed2", Location: "Seed2"}}, }, IsSigner: derivedTablePdaLookupMeta.IsSigner, IsWritable: derivedTablePdaLookupMeta.IsWritable, @@ -107,10 +112,10 @@ func TestChainWriter_GetAddresses(t *testing.T) { // correlates to DerivedTable index in account lookup config derivedTablePdaLookupMeta.PublicKey = storedPubKeys[0] - args := map[string]interface{}{ - "lookup_table": accountLookupMeta.PublicKey.Bytes(), - "seed1": seed1, - "seed2": seed2, + args := Arguments{ + LookupTable: accountLookupMeta.PublicKey, + Seed1: seed1, + Seed2: seed2, } accountLookupConfig := []chainwriter.Lookup{ @@ -122,7 +127,7 @@ func TestChainWriter_GetAddresses(t *testing.T) { }, chainwriter.AccountLookup{ Name: "LookupTable", - Location: "lookup_table", + Location: "LookupTable", IsSigner: accountLookupMeta.IsSigner, IsWritable: accountLookupMeta.IsWritable, }, @@ -131,7 +136,7 @@ func TestChainWriter_GetAddresses(t *testing.T) { PublicKey: chainwriter.AccountConstant{Name: "WriteTest", Address: solana.SystemProgramID.String()}, Seeds: []chainwriter.Seed{ // extract seed1 for PDA lookup - {Dynamic: chainwriter.AccountLookup{Name: "seed1", Location: "seed1"}}, + {Dynamic: chainwriter.AccountLookup{Name: "Seed1", Location: "Seed1"}}, }, IsSigner: pdaLookupMeta.IsSigner, IsWritable: pdaLookupMeta.IsWritable, @@ -177,8 +182,8 @@ func TestChainWriter_GetAddresses(t *testing.T) { }) t.Run("resolve addresses for multiple indices from derived lookup table", func(t *testing.T) { - args := map[string]interface{}{ - "seed2": seed2, + args := Arguments{ + Seed2: seed2, } accountLookupConfig := []chainwriter.Lookup{ @@ -202,8 +207,8 @@ func TestChainWriter_GetAddresses(t *testing.T) { }) t.Run("resolve all addresses from derived lookup table if indices not specified", func(t *testing.T) { - args := map[string]interface{}{ - "seed2": seed2, + args := Arguments{ + Seed2: seed2, } accountLookupConfig := []chainwriter.Lookup{ @@ -274,7 +279,7 @@ func TestChainWriter_FilterLookupTableAddresses(t *testing.T) { PublicKey: chainwriter.AccountConstant{Name: "WriteTest", Address: programID.String()}, Seeds: []chainwriter.Seed{ // extract seed1 for PDA lookup - {Dynamic: chainwriter.AccountLookup{Name: "seed1", Location: "seed1"}}, + {Dynamic: chainwriter.AccountLookup{Name: "Seed1", Location: "Seed1"}}, }, IsSigner: true, IsWritable: true, @@ -291,7 +296,7 @@ func TestChainWriter_FilterLookupTableAddresses(t *testing.T) { PublicKey: chainwriter.AccountConstant{Name: "UnusedAccount", Address: unusedProgramID.String()}, Seeds: []chainwriter.Seed{ // extract seed2 for PDA lookup - {Dynamic: chainwriter.AccountLookup{Name: "seed2", Location: "seed2"}}, + {Dynamic: chainwriter.AccountLookup{Name: "Seed2", Location: "Seed2"}}, }, IsSigner: true, IsWritable: true, @@ -305,9 +310,9 @@ func TestChainWriter_FilterLookupTableAddresses(t *testing.T) { StaticLookupTables: []solana.PublicKey{staticLookupTablePubkey1, staticLookupTablePubkey2}, } - args := map[string]interface{}{ - "seed1": seed1, - "seed2": seed2, + args := Arguments{ + Seed1: seed1, + Seed2: seed2, } t.Run("returns filtered map with only relevant addresses required by account lookup config", func(t *testing.T) { @@ -414,19 +419,14 @@ func TestChainWriter_SubmitTransaction(t *testing.T) { staticLookupKeys := chainwriter.CreateTestPubKeys(t, 2) mockFetchLookupTableAddresses(t, rw, staticLookupTablePubkey, staticLookupKeys) - jsonFile, err := os.Open("testContractIDL.json") - require.NoError(t, err) - - defer jsonFile.Close() - - data, err := ioutil.ReadAll(jsonFile) + data, err := os.ReadFile("testContractIDL.json") require.NoError(t, err) testContractIDLJson := string(data) cwConfig := chainwriter.ChainWriterConfig{ Programs: map[string]chainwriter.ProgramConfig{ - "contractReaderInterface": { + "contract_reader_interface": { Methods: map[string]chainwriter.MethodConfig{ "initializeLookupTable": { FromAddress: admin.String(), @@ -440,7 +440,7 @@ func TestChainWriter_SubmitTransaction(t *testing.T) { PublicKey: chainwriter.AccountConstant{Name: "WriteTest", Address: programID.String()}, Seeds: []chainwriter.Seed{ // extract seed2 for PDA lookup - {Dynamic: chainwriter.AccountLookup{Name: "seed2", Location: "seed2"}}, + {Dynamic: chainwriter.AccountLookup{Name: "Seed2", Location: "Seed2"}}, }, IsSigner: false, IsWritable: false, @@ -462,7 +462,7 @@ func TestChainWriter_SubmitTransaction(t *testing.T) { }, chainwriter.AccountLookup{ Name: "LookupTable", - Location: "lookup_table", + Location: "LookupTable", IsSigner: false, IsWritable: false, }, @@ -471,7 +471,7 @@ func TestChainWriter_SubmitTransaction(t *testing.T) { PublicKey: chainwriter.AccountConstant{Name: "WriteTest", Address: solana.SystemProgramID.String()}, Seeds: []chainwriter.Seed{ // extract seed1 for PDA lookup - {Dynamic: chainwriter.AccountLookup{Name: "seed1", Location: "seed1"}}, + {Dynamic: chainwriter.AccountLookup{Name: "Seed1", Location: "Seed1"}}, }, IsSigner: false, IsWritable: false, @@ -514,22 +514,24 @@ func TestChainWriter_SubmitTransaction(t *testing.T) { t.Run("fails to encode payload if args with missing values provided", func(t *testing.T) { txID := uuid.NewString() - args := map[string]interface{}{} - submitErr := cw.SubmitTransaction(ctx, "contractReaderInterface", "initializeLookupTable", args, txID, programID.String(), nil, nil) + type InvalidArgs struct{} + args := InvalidArgs{} + submitErr := cw.SubmitTransaction(ctx, "contract_reader_interface", "initializeLookupTable", args, txID, programID.String(), nil, nil) require.Error(t, submitErr) }) t.Run("fails if invalid contract name provided", func(t *testing.T) { txID := uuid.NewString() - args := map[string]interface{}{} + args := Arguments{} submitErr := cw.SubmitTransaction(ctx, "badContract", "initializeLookupTable", args, txID, programID.String(), nil, nil) require.Error(t, submitErr) }) t.Run("fails if invalid method provided", func(t *testing.T) { txID := uuid.NewString() - args := map[string]interface{}{} - submitErr := cw.SubmitTransaction(ctx, "contractReaderInterface", "badMethod", args, txID, programID.String(), nil, nil) + + args := Arguments{} + submitErr := cw.SubmitTransaction(ctx, "contract_reader_interface", "badMethod", args, txID, programID.String(), nil, nil) require.Error(t, submitErr) }) @@ -555,13 +557,13 @@ func TestChainWriter_SubmitTransaction(t *testing.T) { return true }), &txID, mock.Anything).Return(nil).Once() - args := map[string]interface{}{ - "lookupTable": chainwriter.GetRandomPubKey(t).Bytes(), - "lookup_table": account2.Bytes(), - "seed1": seed1, - "seed2": seed2, + args := Arguments{ + LookupTable: account2, + Seed1: seed1, + Seed2: seed2, } - submitErr := cw.SubmitTransaction(ctx, "contractReaderInterface", "initializeLookupTable", args, txID, programID.String(), nil, nil) + + submitErr := cw.SubmitTransaction(ctx, "contract_reader_interface", "initializeLookupTable", args, txID, programID.String(), nil, nil) require.NoError(t, submitErr) }) } diff --git a/pkg/solana/chainwriter/lookups.go b/pkg/solana/chainwriter/lookups.go index fb8c9cd98..f875ade3a 100644 --- a/pkg/solana/chainwriter/lookups.go +++ b/pkg/solana/chainwriter/lookups.go @@ -12,6 +12,7 @@ import ( "github.com/smartcontractkit/chainlink-solana/pkg/solana/client" ) +// Lookup is an interface that defines a method to resolve an address (or multiple addresses) from a given definition. type Lookup interface { Resolve(ctx context.Context, args any, derivedTableMap map[string]map[string][]*solana.AccountMeta, reader client.Reader) ([]*solana.AccountMeta, error) }