From ffbae8671909c8b7894b4404d29072a00c1f1ae1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Federico=20Kunze=20K=C3=BCllmer?= <31522760+fedekunze@users.noreply.github.com> Date: Mon, 28 Feb 2022 08:42:09 -0300 Subject: [PATCH 1/5] fix: remove hardcoded pubkey from tx simulation (#11282) ## Description Closes: #11283 --- ### Author Checklist *All items are required. Please add a note to the item if the item is not applicable and please add links to any relevant follow up issues.* I have... - [x] included the correct [type prefix](https://github.com/commitizen/conventional-commit-types/blob/v3.0.0/index.json) in the PR title - [ ] added `!` to the type prefix if API or client breaking change - [x] targeted the correct branch (see [PR Targeting](https://github.com/cosmos/cosmos-sdk/blob/master/CONTRIBUTING.md#pr-targeting)) - [x] provided a link to the relevant issue or specification - [x] followed the guidelines for [building modules](https://github.com/cosmos/cosmos-sdk/blob/master/docs/building-modules) - [x] included the necessary unit and integration [tests](https://github.com/cosmos/cosmos-sdk/blob/master/CONTRIBUTING.md#testing) - [x] added a changelog entry to `CHANGELOG.md` - [ ] included comments for [documenting Go code](https://blog.golang.org/godoc) - [ ] updated the relevant documentation or specification - [ ] reviewed "Files changed" and left comments if necessary - [ ] confirmed all CI checks have passed ### Reviewers Checklist *All items are required. Please add a note if the item is not applicable and please add your handle next to the items reviewed if you only reviewed selected items.* I have... - [ ] confirmed the correct [type prefix](https://github.com/commitizen/conventional-commit-types/blob/v3.0.0/index.json) in the PR title - [ ] confirmed `!` in the type prefix if API or client breaking change - [ ] confirmed all author checklist items have been addressed - [ ] reviewed state machine logic - [ ] reviewed API design and naming - [ ] reviewed documentation is accurate - [ ] reviewed tests and test coverage - [ ] manually tested (if applicable) (cherry picked from commit e6571906043b6751951a42b6546431b1c38b05bd) # Conflicts: # client/tx/factory.go # client/tx/tx_test.go --- CHANGELOG.md | 1 + client/tx/factory.go | 162 +++++++++++++++++++++++++++++++++++++++++++ client/tx/tx_test.go | 116 ++++++++++++++++++++++++++----- 3 files changed, 261 insertions(+), 18 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a739b8767615..135b0949910c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -47,6 +47,7 @@ Ref: https://keepachangelog.com/en/1.0.0/ ### Bug Fixes +* (client) [\#11283](https://github.com/cosmos/cosmos-sdk/issues/11283) Support multiple keys for tx simulation and setting automatic gas for txs. * (store) [\#11177](https://github.com/cosmos/cosmos-sdk/pull/11177) Update the prune `everything` strategy to store the last two heights. * (store) [\#11117](https://github.com/cosmos/cosmos-sdk/pull/11117) Fix data race in store trace component diff --git a/client/tx/factory.go b/client/tx/factory.go index 2cb951922d0a..8bf053905f0e 100644 --- a/client/tx/factory.go +++ b/client/tx/factory.go @@ -6,6 +6,11 @@ import ( "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/client/flags" "github.com/cosmos/cosmos-sdk/crypto/keyring" +<<<<<<< HEAD +======= + "github.com/cosmos/cosmos-sdk/crypto/keys/secp256k1" + cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types" +>>>>>>> e65719060 (fix: remove hardcoded pubkey from tx simulation (#11282)) sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/tx/signing" ) @@ -189,3 +194,160 @@ func (f Factory) WithTimeoutHeight(height uint64) Factory { f.timeoutHeight = height return f } +<<<<<<< HEAD +======= + +// BuildUnsignedTx builds a transaction to be signed given a set of messages. +// Once created, the fee, memo, and messages are set. +func (f Factory) BuildUnsignedTx(msgs ...sdk.Msg) (client.TxBuilder, error) { + if f.offline && f.generateOnly { + if f.chainID != "" { + return nil, fmt.Errorf("chain ID cannot be used when offline and generate-only flags are set") + } + } else if f.chainID == "" { + return nil, fmt.Errorf("chain ID required but not specified") + } + + fees := f.fees + + if !f.gasPrices.IsZero() { + if !fees.IsZero() { + return nil, errors.New("cannot provide both fees and gas prices") + } + + glDec := sdk.NewDec(int64(f.gas)) + + // Derive the fees based on the provided gas prices, where + // fee = ceil(gasPrice * gasLimit). + fees = make(sdk.Coins, len(f.gasPrices)) + + for i, gp := range f.gasPrices { + fee := gp.Amount.Mul(glDec) + fees[i] = sdk.NewCoin(gp.Denom, fee.Ceil().RoundInt()) + } + } + + tx := f.txConfig.NewTxBuilder() + + if err := tx.SetMsgs(msgs...); err != nil { + return nil, err + } + + tx.SetMemo(f.memo) + tx.SetFeeAmount(fees) + tx.SetGasLimit(f.gas) + tx.SetTimeoutHeight(f.TimeoutHeight()) + + return tx, nil +} + +// PrintUnsignedTx will generate an unsigned transaction and print it to the writer +// specified by ctx.Output. If simulation was requested, the gas will be +// simulated and also printed to the same writer before the transaction is +// printed. +func (f Factory) PrintUnsignedTx(clientCtx client.Context, msgs ...sdk.Msg) error { + if f.SimulateAndExecute() { + if clientCtx.Offline { + return errors.New("cannot estimate gas in offline mode") + } + + _, adjusted, err := CalculateGas(clientCtx, f, msgs...) + if err != nil { + return err + } + + f = f.WithGas(adjusted) + _, _ = fmt.Fprintf(os.Stderr, "%s\n", GasEstimateResponse{GasEstimate: f.Gas()}) + } + + unsignedTx, err := f.BuildUnsignedTx(msgs...) + if err != nil { + return err + } + + json, err := clientCtx.TxConfig.TxJSONEncoder()(unsignedTx.GetTx()) + if err != nil { + return err + } + + return clientCtx.PrintString(fmt.Sprintf("%s\n", json)) +} + +// BuildSimTx creates an unsigned tx with an empty single signature and returns +// the encoded transaction or an error if the unsigned transaction cannot be +// built. +func (f Factory) BuildSimTx(msgs ...sdk.Msg) ([]byte, error) { + txb, err := f.BuildUnsignedTx(msgs...) + if err != nil { + return nil, err + } + + // use the first element from the list of keys in order to generate a valid + // pubkey that supports multiple algorithms + + var ( + ok bool + pk cryptotypes.PubKey = &secp256k1.PubKey{} // use default public key type + ) + + if f.keybase != nil { + records, _ := f.keybase.List() + if len(records) == 0 { + return nil, errors.New("cannot build signature for simulation, key records slice is empty") + } + + // take the first record just for simulation purposes + pk, ok = records[0].PubKey.GetCachedValue().(cryptotypes.PubKey) + if !ok { + return nil, errors.New("cannot build signature for simulation, failed to convert proto Any to public key") + } + } + + // Create an empty signature literal as the ante handler will populate with a + // sentinel pubkey. + sig := signing.SignatureV2{ + PubKey: pk, + Data: &signing.SingleSignatureData{ + SignMode: f.signMode, + }, + Sequence: f.Sequence(), + } + if err := txb.SetSignatures(sig); err != nil { + return nil, err + } + + return f.txConfig.TxEncoder()(txb.GetTx()) +} + +// Prepare ensures the account defined by ctx.GetFromAddress() exists and +// if the account number and/or the account sequence number are zero (not set), +// they will be queried for and set on the provided Factory. A new Factory with +// the updated fields will be returned. +func (f Factory) Prepare(clientCtx client.Context) (Factory, error) { + fc := f + + from := clientCtx.GetFromAddress() + + if err := fc.accountRetriever.EnsureExists(clientCtx, from); err != nil { + return fc, err + } + + initNum, initSeq := fc.accountNumber, fc.sequence + if initNum == 0 || initSeq == 0 { + num, seq, err := fc.accountRetriever.GetAccountNumberSequence(clientCtx, from) + if err != nil { + return fc, err + } + + if initNum == 0 { + fc = fc.WithAccountNumber(num) + } + + if initSeq == 0 { + fc = fc.WithSequence(seq) + } + } + + return fc, nil +} +>>>>>>> e65719060 (fix: remove hardcoded pubkey from tx simulation (#11282)) diff --git a/client/tx/tx_test.go b/client/tx/tx_test.go index b095d05b9e5d..8440d5993f2e 100644 --- a/client/tx/tx_test.go +++ b/client/tx/tx_test.go @@ -45,6 +45,7 @@ func (m mockContext) Invoke(grpcCtx gocontext.Context, method string, req, reply return nil } + func (mockContext) NewStream(gocontext.Context, *grpc.StreamDesc, string, ...grpc.CallOption) (grpc.ClientStream, error) { panic("not implemented") } @@ -96,6 +97,14 @@ func TestCalculateGas(t *testing.T) { func TestBuildSimTx(t *testing.T) { txCfg := NewTestTxConfig() + encCfg := simapp.MakeTestEncodingConfig() + + kb, err := keyring.New(t.Name(), "test", t.TempDir(), nil, encCfg.Codec) + require.NoError(t, err) + + path := hd.CreateHDPath(118, 0, 0).String() + _, _, err = kb.NewMnemonic("test_key1", keyring.English, path, keyring.DefaultBIP39Passphrase, hd.Secp256k1) + require.NoError(t, err) txf := tx.Factory{}. WithTxConfig(txCfg). @@ -104,7 +113,8 @@ func TestBuildSimTx(t *testing.T) { WithFees("50stake"). WithMemo("memo"). WithChainID("test-chain"). - WithSignMode(txCfg.SignModeHandler().DefaultMode()) + WithSignMode(txCfg.SignModeHandler().DefaultMode()). + WithKeybase(kb) msg := banktypes.NewMsgSend(sdk.AccAddress("from"), sdk.AccAddress("to"), nil) bz, err := tx.BuildSimTx(txf, msg) @@ -113,13 +123,23 @@ func TestBuildSimTx(t *testing.T) { } func TestBuildUnsignedTx(t *testing.T) { + encCfg := simapp.MakeTestEncodingConfig() + kb, err := keyring.New(t.Name(), "test", t.TempDir(), nil, encCfg.Codec) + require.NoError(t, err) + + path := hd.CreateHDPath(118, 0, 0).String() + + _, _, err = kb.NewMnemonic("test_key1", keyring.English, path, keyring.DefaultBIP39Passphrase, hd.Secp256k1) + require.NoError(t, err) + txf := tx.Factory{}. WithTxConfig(NewTestTxConfig()). WithAccountNumber(50). WithSequence(23). WithFees("50stake"). WithMemo("memo"). - WithChainID("test-chain") + WithChainID("test-chain"). + WithKeybase(kb) msg := banktypes.NewMsgSend(sdk.AccAddress("from"), sdk.AccAddress("to"), nil) tx, err := tx.BuildUnsignedTx(txf, msg) @@ -137,8 +157,8 @@ func TestSign(t *testing.T) { kr, err := keyring.New(t.Name(), "test", t.TempDir(), nil) requireT.NoError(err) - var from1 = "test_key1" - var from2 = "test_key2" + from1 := "test_key1" + from2 := "test_key2" // create a new key using a mnemonic generator and test if we can reuse seed to recreate that account _, seed, err := kr.NewMnemonic(from1, keyring.English, path, keyring.DefaultBIP39Passphrase, hd.Secp256k1) @@ -185,24 +205,49 @@ func TestSign(t *testing.T) { expectedPKs []cryptotypes.PubKey matchingSigs []int // if not nil, check matching signature against old ones. }{ - {"should fail if txf without keyring", - txfNoKeybase, txb, from1, true, nil, nil}, - {"should fail for non existing key", - txfAmino, txb, "unknown", true, nil, nil}, - {"amino: should succeed with keyring", - txfAmino, txbSimple, from1, true, []cryptotypes.PubKey{pubKey1}, nil}, - {"direct: should succeed with keyring", - txfDirect, txbSimple, from1, true, []cryptotypes.PubKey{pubKey1}, nil}, + { + "should fail if txf without keyring", + txfNoKeybase, txb, from1, true, nil, nil, + }, + { + "should fail for non existing key", + txfAmino, txb, "unknown", true, nil, nil, + }, + { + "amino: should succeed with keyring", + txfAmino, txbSimple, from1, true, + []cryptotypes.PubKey{pubKey1}, + nil, + }, + { + "direct: should succeed with keyring", + txfDirect, txbSimple, from1, true, + []cryptotypes.PubKey{pubKey1}, + nil, + }, /**** test double sign Amino mode ****/ - {"amino: should sign multi-signers tx", - txfAmino, txb, from1, true, []cryptotypes.PubKey{pubKey1}, nil}, - {"amino: should append a second signature and not overwrite", - txfAmino, txb, from2, false, []cryptotypes.PubKey{pubKey1, pubKey2}, []int{0, 0}}, - {"amino: should overwrite a signature", - txfAmino, txb, from2, true, []cryptotypes.PubKey{pubKey2}, []int{1, 0}}, + { + "amino: should sign multi-signers tx", + txfAmino, txb, from1, true, + []cryptotypes.PubKey{pubKey1}, + nil, + }, + { + "amino: should append a second signature and not overwrite", + txfAmino, txb, from2, false, + []cryptotypes.PubKey{pubKey1, pubKey2}, + []int{0, 0}, + }, + { + "amino: should overwrite a signature", + txfAmino, txb, from2, true, + []cryptotypes.PubKey{pubKey2}, + []int{1, 0}, + }, /**** test double sign Direct mode +<<<<<<< HEAD signing transaction with more than 2 signers should fail in DIRECT mode ****/ {"direct: should fail to append a signature with different mode", txfDirect, txb, from1, false, []cryptotypes.PubKey{}, nil}, @@ -210,6 +255,41 @@ func TestSign(t *testing.T) { txfDirect, txb2, from1, false, []cryptotypes.PubKey{}, nil}, {"direct: should fail to overwrite multi-signers tx", txfDirect, txb2, from1, true, []cryptotypes.PubKey{}, nil}, +======= + signing transaction with 2 or more DIRECT signers should fail in DIRECT mode ****/ + { + "direct: should append a DIRECT signature with existing AMINO", + // txb already has 1 AMINO signature + txfDirect, txb, from1, false, + []cryptotypes.PubKey{pubKey2, pubKey1}, + nil, + }, + { + "direct: should add single DIRECT sig in multi-signers tx", + txfDirect, txb2, from1, false, + []cryptotypes.PubKey{pubKey1}, + nil, + }, + { + "direct: should fail to append 2nd DIRECT sig in multi-signers tx", + txfDirect, txb2, from2, false, + []cryptotypes.PubKey{}, + nil, + }, + { + "amino: should append 2nd AMINO sig in multi-signers tx with 1 DIRECT sig", + // txb2 already has 1 DIRECT signature + txfAmino, txb2, from2, false, + []cryptotypes.PubKey{}, + nil, + }, + { + "direct: should overwrite multi-signers tx with DIRECT sig", + txfDirect, txb2, from1, true, + []cryptotypes.PubKey{pubKey1}, + nil, + }, +>>>>>>> e65719060 (fix: remove hardcoded pubkey from tx simulation (#11282)) } var prevSigs []signingtypes.SignatureV2 for _, tc := range testCases { From ea9bacc552f67330d4a1e6a52357bb9101045044 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Federico=20Kunze=20K=C3=BCllmer?= Date: Mon, 28 Feb 2022 16:50:26 +0100 Subject: [PATCH 2/5] fix conflicts --- CHANGELOG.md | 1 - client/tx/factory.go | 6 ------ client/tx/tx_test.go | 10 ---------- 3 files changed, 17 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 135b0949910c..ccbac7f8ec3b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -44,7 +44,6 @@ Ref: https://keepachangelog.com/en/1.0.0/ * [\#11124](https://github.com/cosmos/cosmos-sdk/pull/11124) Add `GetAllVersions` to application store * (x/auth) [\#10880](https://github.com/cosmos/cosmos-sdk/pull/10880) Added a new query to the tx query service that returns a block with transactions fully decoded. - ### Bug Fixes * (client) [\#11283](https://github.com/cosmos/cosmos-sdk/issues/11283) Support multiple keys for tx simulation and setting automatic gas for txs. diff --git a/client/tx/factory.go b/client/tx/factory.go index 8bf053905f0e..71269e09e514 100644 --- a/client/tx/factory.go +++ b/client/tx/factory.go @@ -6,11 +6,8 @@ import ( "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/client/flags" "github.com/cosmos/cosmos-sdk/crypto/keyring" -<<<<<<< HEAD -======= "github.com/cosmos/cosmos-sdk/crypto/keys/secp256k1" cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types" ->>>>>>> e65719060 (fix: remove hardcoded pubkey from tx simulation (#11282)) sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/tx/signing" ) @@ -194,8 +191,6 @@ func (f Factory) WithTimeoutHeight(height uint64) Factory { f.timeoutHeight = height return f } -<<<<<<< HEAD -======= // BuildUnsignedTx builds a transaction to be signed given a set of messages. // Once created, the fee, memo, and messages are set. @@ -350,4 +345,3 @@ func (f Factory) Prepare(clientCtx client.Context) (Factory, error) { return fc, nil } ->>>>>>> e65719060 (fix: remove hardcoded pubkey from tx simulation (#11282)) diff --git a/client/tx/tx_test.go b/client/tx/tx_test.go index 8440d5993f2e..833fe05025a4 100644 --- a/client/tx/tx_test.go +++ b/client/tx/tx_test.go @@ -247,15 +247,6 @@ func TestSign(t *testing.T) { }, /**** test double sign Direct mode -<<<<<<< HEAD - signing transaction with more than 2 signers should fail in DIRECT mode ****/ - {"direct: should fail to append a signature with different mode", - txfDirect, txb, from1, false, []cryptotypes.PubKey{}, nil}, - {"direct: should fail to sign multi-signers tx", - txfDirect, txb2, from1, false, []cryptotypes.PubKey{}, nil}, - {"direct: should fail to overwrite multi-signers tx", - txfDirect, txb2, from1, true, []cryptotypes.PubKey{}, nil}, -======= signing transaction with 2 or more DIRECT signers should fail in DIRECT mode ****/ { "direct: should append a DIRECT signature with existing AMINO", @@ -289,7 +280,6 @@ func TestSign(t *testing.T) { []cryptotypes.PubKey{pubKey1}, nil, }, ->>>>>>> e65719060 (fix: remove hardcoded pubkey from tx simulation (#11282)) } var prevSigs []signingtypes.SignatureV2 for _, tc := range testCases { From 637accded300a7bc2c67c8fb461be94cb41fc696 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Federico=20Kunze=20K=C3=BCllmer?= Date: Mon, 28 Feb 2022 16:59:35 +0100 Subject: [PATCH 3/5] fix --- client/tx/tx.go | 57 ++------------------------------------------ client/tx/tx_test.go | 6 ++--- 2 files changed, 4 insertions(+), 59 deletions(-) diff --git a/client/tx/tx.go b/client/tx/tx.go index 8e20be369bfd..90c49514fa1a 100644 --- a/client/tx/tx.go +++ b/client/tx/tx.go @@ -14,7 +14,6 @@ import ( "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/client/flags" "github.com/cosmos/cosmos-sdk/client/input" - "github.com/cosmos/cosmos-sdk/crypto/keys/secp256k1" cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types" sdk "github.com/cosmos/cosmos-sdk/types" sdkerrors "github.com/cosmos/cosmos-sdk/types/errors" @@ -221,66 +220,14 @@ func WriteGeneratedTxResponse( // transaction is initially created via the provided factory's generator. Once // created, the fee, memo, and messages are set. func BuildUnsignedTx(txf Factory, msgs ...sdk.Msg) (client.TxBuilder, error) { - if txf.chainID == "" { - return nil, fmt.Errorf("chain ID required but not specified") - } - - fees := txf.fees - - if !txf.gasPrices.IsZero() { - if !fees.IsZero() { - return nil, errors.New("cannot provide both fees and gas prices") - } - - glDec := sdk.NewDec(int64(txf.gas)) - - // Derive the fees based on the provided gas prices, where - // fee = ceil(gasPrice * gasLimit). - fees = make(sdk.Coins, len(txf.gasPrices)) - - for i, gp := range txf.gasPrices { - fee := gp.Amount.Mul(glDec) - fees[i] = sdk.NewCoin(gp.Denom, fee.Ceil().RoundInt()) - } - } - - tx := txf.txConfig.NewTxBuilder() - - if err := tx.SetMsgs(msgs...); err != nil { - return nil, err - } - - tx.SetMemo(txf.memo) - tx.SetFeeAmount(fees) - tx.SetGasLimit(txf.gas) - tx.SetTimeoutHeight(txf.TimeoutHeight()) - - return tx, nil + return txf.BuildUnsignedTx(msgs...) } // BuildSimTx creates an unsigned tx with an empty single signature and returns // the encoded transaction or an error if the unsigned transaction cannot be // built. func BuildSimTx(txf Factory, msgs ...sdk.Msg) ([]byte, error) { - txb, err := BuildUnsignedTx(txf, msgs...) - if err != nil { - return nil, err - } - - // Create an empty signature literal as the ante handler will populate with a - // sentinel pubkey. - sig := signing.SignatureV2{ - PubKey: &secp256k1.PubKey{}, - Data: &signing.SingleSignatureData{ - SignMode: txf.signMode, - }, - Sequence: txf.Sequence(), - } - if err := txb.SetSignatures(sig); err != nil { - return nil, err - } - - return txf.txConfig.TxEncoder()(txb.GetTx()) + return txf.BuildSimTx(msgs...) } // CalculateGas simulates the execution of a transaction and returns the diff --git a/client/tx/tx_test.go b/client/tx/tx_test.go index 833fe05025a4..f79fb894a075 100644 --- a/client/tx/tx_test.go +++ b/client/tx/tx_test.go @@ -97,9 +97,8 @@ func TestCalculateGas(t *testing.T) { func TestBuildSimTx(t *testing.T) { txCfg := NewTestTxConfig() - encCfg := simapp.MakeTestEncodingConfig() - kb, err := keyring.New(t.Name(), "test", t.TempDir(), nil, encCfg.Codec) + kb, err := keyring.New(t.Name(), "test", t.TempDir(), nil) require.NoError(t, err) path := hd.CreateHDPath(118, 0, 0).String() @@ -123,8 +122,7 @@ func TestBuildSimTx(t *testing.T) { } func TestBuildUnsignedTx(t *testing.T) { - encCfg := simapp.MakeTestEncodingConfig() - kb, err := keyring.New(t.Name(), "test", t.TempDir(), nil, encCfg.Codec) + kb, err := keyring.New(t.Name(), "test", t.TempDir(), nil) require.NoError(t, err) path := hd.CreateHDPath(118, 0, 0).String() From 61469a0526ddab26dc271bf3dad3b62725ae2105 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Federico=20Kunze=20K=C3=BCllmer?= Date: Mon, 28 Feb 2022 17:33:14 +0100 Subject: [PATCH 4/5] fix build --- client/tx/factory.go | 28 +++++++++++----------------- 1 file changed, 11 insertions(+), 17 deletions(-) diff --git a/client/tx/factory.go b/client/tx/factory.go index 71269e09e514..13860bba11e2 100644 --- a/client/tx/factory.go +++ b/client/tx/factory.go @@ -1,6 +1,10 @@ package tx import ( + "errors" + "fmt" + "os" + "github.com/spf13/pflag" "github.com/cosmos/cosmos-sdk/client" @@ -195,11 +199,7 @@ func (f Factory) WithTimeoutHeight(height uint64) Factory { // BuildUnsignedTx builds a transaction to be signed given a set of messages. // Once created, the fee, memo, and messages are set. func (f Factory) BuildUnsignedTx(msgs ...sdk.Msg) (client.TxBuilder, error) { - if f.offline && f.generateOnly { - if f.chainID != "" { - return nil, fmt.Errorf("chain ID cannot be used when offline and generate-only flags are set") - } - } else if f.chainID == "" { + if f.chainID == "" { return nil, fmt.Errorf("chain ID required but not specified") } @@ -280,22 +280,16 @@ func (f Factory) BuildSimTx(msgs ...sdk.Msg) ([]byte, error) { // use the first element from the list of keys in order to generate a valid // pubkey that supports multiple algorithms - var ( - ok bool - pk cryptotypes.PubKey = &secp256k1.PubKey{} // use default public key type - ) + var pk cryptotypes.PubKey = &secp256k1.PubKey{} // use default public key type if f.keybase != nil { - records, _ := f.keybase.List() - if len(records) == 0 { - return nil, errors.New("cannot build signature for simulation, key records slice is empty") + infos, _ := f.keybase.List() + if len(infos) == 0 { + return nil, errors.New("cannot build signature for simulation, key infos slice is empty") } - // take the first record just for simulation purposes - pk, ok = records[0].PubKey.GetCachedValue().(cryptotypes.PubKey) - if !ok { - return nil, errors.New("cannot build signature for simulation, failed to convert proto Any to public key") - } + // take the first info record just for simulation purposes + pk = infos[0].GetPubKey() } // Create an empty signature literal as the ante handler will populate with a From cfd3c74ccfa8154fdf040fe4994f9a41dcb707c8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Federico=20Kunze=20K=C3=BCllmer?= Date: Wed, 9 Mar 2022 17:00:20 +0100 Subject: [PATCH 5/5] fix tests --- client/tx/tx_test.go | 32 +++++++++----------------------- 1 file changed, 9 insertions(+), 23 deletions(-) diff --git a/client/tx/tx_test.go b/client/tx/tx_test.go index f79fb894a075..9a4c60d07708 100644 --- a/client/tx/tx_test.go +++ b/client/tx/tx_test.go @@ -136,8 +136,7 @@ func TestBuildUnsignedTx(t *testing.T) { WithSequence(23). WithFees("50stake"). WithMemo("memo"). - WithChainID("test-chain"). - WithKeybase(kb) + WithChainID("test-chain") msg := banktypes.NewMsgSend(sdk.AccAddress("from"), sdk.AccAddress("to"), nil) tx, err := tx.BuildUnsignedTx(txf, msg) @@ -187,6 +186,7 @@ func TestSign(t *testing.T) { WithSignMode(signingtypes.SignMode_SIGN_MODE_LEGACY_AMINO_JSON) msg1 := banktypes.NewMsgSend(info1.GetAddress(), sdk.AccAddress("to"), nil) msg2 := banktypes.NewMsgSend(info2.GetAddress(), sdk.AccAddress("to"), nil) + txb, err := tx.BuildUnsignedTx(txfNoKeybase, msg1, msg2) requireT.NoError(err) txb2, err := tx.BuildUnsignedTx(txfNoKeybase, msg1, msg2) @@ -245,43 +245,29 @@ func TestSign(t *testing.T) { }, /**** test double sign Direct mode - signing transaction with 2 or more DIRECT signers should fail in DIRECT mode ****/ + signing transaction with more than 2 signers should fail in DIRECT mode ****/ { - "direct: should append a DIRECT signature with existing AMINO", - // txb already has 1 AMINO signature + "direct: should fail to append a signature with different mode", txfDirect, txb, from1, false, - []cryptotypes.PubKey{pubKey2, pubKey1}, - nil, - }, - { - "direct: should add single DIRECT sig in multi-signers tx", - txfDirect, txb2, from1, false, - []cryptotypes.PubKey{pubKey1}, - nil, - }, - { - "direct: should fail to append 2nd DIRECT sig in multi-signers tx", - txfDirect, txb2, from2, false, []cryptotypes.PubKey{}, nil, }, { - "amino: should append 2nd AMINO sig in multi-signers tx with 1 DIRECT sig", - // txb2 already has 1 DIRECT signature - txfAmino, txb2, from2, false, + "direct: should fail to sign multi-signers tx", + txfDirect, txb2, from1, false, []cryptotypes.PubKey{}, nil, }, { - "direct: should overwrite multi-signers tx with DIRECT sig", + "direct: should fail to overwrite multi-signers tx", txfDirect, txb2, from1, true, - []cryptotypes.PubKey{pubKey1}, + []cryptotypes.PubKey{}, nil, }, } var prevSigs []signingtypes.SignatureV2 for _, tc := range testCases { - t.Run(tc.name, func(t *testing.T) { + t.Run(tc.name, func(_ *testing.T) { err = tx.Sign(tc.txf, tc.from, tc.txb, tc.overwrite) if len(tc.expectedPKs) == 0 { requireT.Error(err)