forked from cerc-io/laconicd-deprecated
196 lines
5.5 KiB
Go
196 lines
5.5 KiB
Go
package backend
|
|
|
|
import (
|
|
"bufio"
|
|
"math/big"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/cerc-io/laconicd/crypto/ethsecp256k1"
|
|
|
|
dbm "github.com/tendermint/tm-db"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
|
"github.com/cosmos/cosmos-sdk/crypto/keyring"
|
|
"github.com/cosmos/cosmos-sdk/server"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/ethereum/go-ethereum/common"
|
|
ethtypes "github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/stretchr/testify/suite"
|
|
tmrpctypes "github.com/tendermint/tendermint/rpc/core/types"
|
|
|
|
"github.com/cerc-io/laconicd/app"
|
|
"github.com/cerc-io/laconicd/crypto/hd"
|
|
"github.com/cerc-io/laconicd/encoding"
|
|
"github.com/cerc-io/laconicd/indexer"
|
|
"github.com/cerc-io/laconicd/rpc/backend/mocks"
|
|
rpctypes "github.com/cerc-io/laconicd/rpc/types"
|
|
"github.com/cerc-io/laconicd/tests"
|
|
evmtypes "github.com/cerc-io/laconicd/x/evm/types"
|
|
)
|
|
|
|
type BackendTestSuite struct {
|
|
suite.Suite
|
|
backend *Backend
|
|
acc sdk.AccAddress
|
|
signer keyring.Signer
|
|
}
|
|
|
|
func TestBackendTestSuite(t *testing.T) {
|
|
suite.Run(t, new(BackendTestSuite))
|
|
}
|
|
|
|
const ChainID = "ethermint_9000-1"
|
|
|
|
// SetupTest is executed before every BackendTestSuite test
|
|
func (suite *BackendTestSuite) SetupTest() {
|
|
ctx := server.NewDefaultContext()
|
|
ctx.Viper.Set("telemetry.global-labels", []interface{}{})
|
|
|
|
baseDir := suite.T().TempDir()
|
|
nodeDirName := "node"
|
|
clientDir := filepath.Join(baseDir, nodeDirName, "evmoscli")
|
|
keyRing, err := suite.generateTestKeyring(clientDir)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// Create Account with set sequence
|
|
suite.acc = sdk.AccAddress(tests.GenerateAddress().Bytes())
|
|
accounts := map[string]client.TestAccount{}
|
|
accounts[suite.acc.String()] = client.TestAccount{
|
|
Address: suite.acc,
|
|
Num: uint64(1),
|
|
Seq: uint64(1),
|
|
}
|
|
|
|
priv, err := ethsecp256k1.GenerateKey()
|
|
suite.signer = tests.NewSigner(priv)
|
|
suite.Require().NoError(err)
|
|
|
|
encodingConfig := encoding.MakeConfig(app.ModuleBasics)
|
|
clientCtx := client.Context{}.WithChainID(ChainID).
|
|
WithHeight(1).
|
|
WithTxConfig(encodingConfig.TxConfig).
|
|
WithKeyringDir(clientDir).
|
|
WithKeyring(keyRing).
|
|
WithAccountRetriever(client.TestAccountRetriever{Accounts: accounts})
|
|
|
|
allowUnprotectedTxs := false
|
|
idxer := indexer.NewKVIndexer(dbm.NewMemDB(), ctx.Logger, clientCtx)
|
|
|
|
suite.backend = NewBackend(ctx, ctx.Logger, clientCtx, allowUnprotectedTxs, idxer)
|
|
suite.backend.queryClient.QueryClient = mocks.NewEVMQueryClient(suite.T())
|
|
suite.backend.clientCtx.Client = mocks.NewClient(suite.T())
|
|
suite.backend.queryClient.FeeMarket = mocks.NewFeeMarketQueryClient(suite.T())
|
|
suite.backend.ctx = rpctypes.ContextWithHeight(1)
|
|
|
|
// Add codec
|
|
encCfg := encoding.MakeConfig(app.ModuleBasics)
|
|
suite.backend.clientCtx.Codec = encCfg.Codec
|
|
|
|
}
|
|
|
|
// buildEthereumTx returns an example legacy Ethereum transaction
|
|
func (suite *BackendTestSuite) buildEthereumTx() (*evmtypes.MsgEthereumTx, []byte) {
|
|
msgEthereumTx := evmtypes.NewTx(
|
|
suite.backend.chainID,
|
|
uint64(0),
|
|
&common.Address{},
|
|
big.NewInt(0),
|
|
100000,
|
|
big.NewInt(1),
|
|
nil,
|
|
nil,
|
|
nil,
|
|
nil,
|
|
)
|
|
|
|
// A valid msg should have empty `From`
|
|
msgEthereumTx.From = ""
|
|
|
|
txBuilder := suite.backend.clientCtx.TxConfig.NewTxBuilder()
|
|
err := txBuilder.SetMsgs(msgEthereumTx)
|
|
suite.Require().NoError(err)
|
|
|
|
bz, err := suite.backend.clientCtx.TxConfig.TxEncoder()(txBuilder.GetTx())
|
|
suite.Require().NoError(err)
|
|
return msgEthereumTx, bz
|
|
}
|
|
|
|
// buildFormattedBlock returns a formatted block for testing
|
|
func (suite *BackendTestSuite) buildFormattedBlock(
|
|
blockRes *tmrpctypes.ResultBlockResults,
|
|
resBlock *tmrpctypes.ResultBlock,
|
|
fullTx bool,
|
|
tx *evmtypes.MsgEthereumTx,
|
|
validator sdk.AccAddress,
|
|
baseFee *big.Int,
|
|
) map[string]interface{} {
|
|
header := resBlock.Block.Header
|
|
gasLimit := int64(^uint32(0)) // for `MaxGas = -1` (DefaultConsensusParams)
|
|
gasUsed := new(big.Int).SetUint64(uint64(blockRes.TxsResults[0].GasUsed))
|
|
|
|
root := common.Hash{}.Bytes()
|
|
receipt := ethtypes.NewReceipt(root, false, gasUsed.Uint64())
|
|
bloom := ethtypes.CreateBloom(ethtypes.Receipts{receipt})
|
|
|
|
ethRPCTxs := []interface{}{}
|
|
if tx != nil {
|
|
if fullTx {
|
|
rpcTx, err := rpctypes.NewRPCTransaction(
|
|
tx.AsTransaction(),
|
|
common.BytesToHash(header.Hash()),
|
|
uint64(header.Height),
|
|
uint64(0),
|
|
baseFee,
|
|
suite.backend.chainID,
|
|
)
|
|
suite.Require().NoError(err)
|
|
ethRPCTxs = []interface{}{rpcTx}
|
|
} else {
|
|
ethRPCTxs = []interface{}{common.HexToHash(tx.Hash)}
|
|
}
|
|
}
|
|
|
|
return rpctypes.FormatBlock(
|
|
header,
|
|
resBlock.Block.Size(),
|
|
gasLimit,
|
|
gasUsed,
|
|
ethRPCTxs,
|
|
bloom,
|
|
common.BytesToAddress(validator.Bytes()),
|
|
baseFee,
|
|
)
|
|
}
|
|
|
|
func (suite *BackendTestSuite) generateTestKeyring(clientDir string) (keyring.Keyring, error) {
|
|
buf := bufio.NewReader(os.Stdin)
|
|
encCfg := encoding.MakeConfig(app.ModuleBasics)
|
|
return keyring.New(sdk.KeyringServiceName(), keyring.BackendTest, clientDir, buf, encCfg.Codec, []keyring.Option{hd.EthSecp256k1Option()}...)
|
|
}
|
|
|
|
func (suite *BackendTestSuite) signAndEncodeEthTx(msgEthereumTx *evmtypes.MsgEthereumTx) []byte {
|
|
from, priv := tests.NewAddrKey()
|
|
signer := tests.NewSigner(priv)
|
|
|
|
queryClient := suite.backend.queryClient.QueryClient.(*mocks.EVMQueryClient)
|
|
RegisterParamsWithoutHeader(queryClient, 1)
|
|
|
|
ethSigner := ethtypes.LatestSigner(suite.backend.ChainConfig())
|
|
msgEthereumTx.From = from.String()
|
|
err := msgEthereumTx.Sign(ethSigner, signer)
|
|
suite.Require().NoError(err)
|
|
|
|
tx, err := msgEthereumTx.BuildTx(suite.backend.clientCtx.TxConfig.NewTxBuilder(), "aphoton")
|
|
suite.Require().NoError(err)
|
|
|
|
txEncoder := suite.backend.clientCtx.TxConfig.TxEncoder()
|
|
txBz, err := txEncoder(tx)
|
|
suite.Require().NoError(err)
|
|
|
|
return txBz
|
|
}
|