Pending (#571)
* add PendingBlockNumber -1 * increase block times * update bn * get pending balance * additional logic to check for pending state * add multiple balance query * pending state for getTransactionCount * fix lint * add getBlockTransactionCountByNumber code - commented * cleanup test * GetBlockTransactionCountByNumber * cleanup * getBlockByNumber * GetTransactionByBlockNumberAndIndex * conform to namespace changes * exportable FormatBlock method * eth_getTransactionByHash * eth_getTransactionByBlockNumberAndIndex * pending nonce * set nonce for pending and check for invalid * WIP: doCall * add pending tx test * cleanup + refactor * push first tests and init pending * pending changes (#600) * cleanup * updates * more fixes * lint * update call and send * comments and minor changes * add pending tests into sep package * fix latest case for eth_GetBlockTransactionCountByNumber * fix repeating null transactions in queue * remove repeated structs * latestblock case * revert init script back * fix to exportable method * automate pending tests; add make cmd * move and comment out pending call test * fix some golint * fix unlock issue * wip: linter stringer fix? * stringer lint * set arr instead of append * instantiate with length * sep if statement * edit pendingblocknumber note * switch statement * fix and update tests * move tests-pending into tests dir * remove commented test * revert to appending pendingtx * Update tests/utils.go Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> * Update tests/utils.go Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> * Update tests/utils.go Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> * require no err * rename var * check result for eth_sendTransaction * update changelog * update * Update tests/utils.go Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> * Update tests/tests-pending/rpc_pending_test.go Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> * changelog * remove redundant check Co-authored-by: noot <elizabethjbinks@gmail.com> Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com> Co-authored-by: Federico Kunze <federico.kunze94@gmail.com>
This commit is contained in:
parent
602e61adea
commit
cb96bc4ea3
11
CHANGELOG.md
11
CHANGELOG.md
@ -41,6 +41,17 @@ Ref: https://keepachangelog.com/en/1.0.0/
|
|||||||
|
|
||||||
* (evm) [\#661](https://github.com/cosmos/ethermint/pull/661) `Balance` field has been removed from the evm module's `GenesisState`.
|
* (evm) [\#661](https://github.com/cosmos/ethermint/pull/661) `Balance` field has been removed from the evm module's `GenesisState`.
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* (rpc) [\#571](https://github.com/cosmos/ethermint/pull/571) Add pending queries to JSON-RPC calls. This allows for the querying of pending transactions and other relevant information that pertains to the pending state:
|
||||||
|
* `eth_getBalance`
|
||||||
|
* `eth_getTransactionCount`
|
||||||
|
* `eth_getBlockTransactionCountByNumber`
|
||||||
|
* `eth_getBlockByNumber`
|
||||||
|
* `eth_getTransactionByHash`
|
||||||
|
* `eth_getTransactionByBlockNumberAndIndex`
|
||||||
|
* `eth_sendTransaction` - the nonce will automatically update to its pending nonce (when none is explicitly provided)
|
||||||
|
|
||||||
### Improvements
|
### Improvements
|
||||||
|
|
||||||
* (evm) [\#661](https://github.com/cosmos/ethermint/pull/661) Add invariant check for account balance and account nonce.
|
* (evm) [\#661](https://github.com/cosmos/ethermint/pull/661) Add invariant check for account balance and account nonce.
|
||||||
|
5
Makefile
5
Makefile
@ -266,7 +266,10 @@ test-import:
|
|||||||
rm -rf importer/tmp
|
rm -rf importer/tmp
|
||||||
|
|
||||||
test-rpc:
|
test-rpc:
|
||||||
./scripts/integration-test-all.sh -q 1 -z 1 -s 2
|
./scripts/integration-test-all.sh -t "rpc" -q 1 -z 1 -s 2 -m "rpc"
|
||||||
|
|
||||||
|
test-rpc-pending:
|
||||||
|
./scripts/integration-test-all.sh -t "pending" -q 1 -z 1 -s 2 -m "pending"
|
||||||
|
|
||||||
test-contract:
|
test-contract:
|
||||||
@type "npm" 2> /dev/null || (echo 'Npm does not exist. Please install node.js and npm."' && exit 1)
|
@type "npm" 2> /dev/null || (echo 'Npm does not exist. Please install node.js and npm."' && exit 1)
|
||||||
|
15
init.sh
15
init.sh
@ -32,6 +32,21 @@ cat $HOME/.ethermintd/config/genesis.json | jq '.app_state["mint"]["params"]["mi
|
|||||||
# Enable faucet
|
# Enable faucet
|
||||||
cat $HOME/.ethermintd/config/genesis.json | jq '.app_state["faucet"]["enable_faucet"]=true' > $HOME/.ethermintd/config/tmp_genesis.json && mv $HOME/.ethermintd/config/tmp_genesis.json $HOME/.ethermintd/config/genesis.json
|
cat $HOME/.ethermintd/config/genesis.json | jq '.app_state["faucet"]["enable_faucet"]=true' > $HOME/.ethermintd/config/tmp_genesis.json && mv $HOME/.ethermintd/config/tmp_genesis.json $HOME/.ethermintd/config/genesis.json
|
||||||
|
|
||||||
|
# increase block time (?)
|
||||||
|
cat $HOME/.ethermintd/config/genesis.json | jq '.consensus_params["block"]["time_iota_ms"]="30000"' > $HOME/.ethermintd/config/tmp_genesis.json && mv $HOME/.ethermintd/config/tmp_genesis.json $HOME/.ethermintd/config/genesis.json
|
||||||
|
|
||||||
|
if [[ $1 == "pending" ]]; then
|
||||||
|
echo "pending mode on; block times will be set to 30s."
|
||||||
|
# sed -i 's/create_empty_blocks_interval = "0s"/create_empty_blocks_interval = "30s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_propose = "3s"/timeout_propose = "30s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_propose_delta = "500ms"/timeout_propose_delta = "5s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_prevote = "1s"/timeout_prevote = "10s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_prevote_delta = "500ms"/timeout_prevote_delta = "5s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_precommit = "1s"/timeout_precommit = "10s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_precommit_delta = "500ms"/timeout_precommit_delta = "5s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
sed -i 's/timeout_commit = "5s"/timeout_commit = "150s"/g' $HOME/.ethermintd/config/config.toml
|
||||||
|
fi
|
||||||
|
|
||||||
# Allocate genesis accounts (cosmos formatted addresses)
|
# Allocate genesis accounts (cosmos formatted addresses)
|
||||||
ethermintd add-genesis-account $(ethermintcli keys show $KEY -a) 100000000000000000000aphoton
|
ethermintd add-genesis-account $(ethermintcli keys show $KEY -a) 100000000000000000000aphoton
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@ import (
|
|||||||
type Backend interface {
|
type Backend interface {
|
||||||
// Used by block filter; also used for polling
|
// Used by block filter; also used for polling
|
||||||
BlockNumber() (hexutil.Uint64, error)
|
BlockNumber() (hexutil.Uint64, error)
|
||||||
|
LatestBlockNumber() (int64, error)
|
||||||
HeaderByNumber(blockNum rpctypes.BlockNumber) (*ethtypes.Header, error)
|
HeaderByNumber(blockNum rpctypes.BlockNumber) (*ethtypes.Header, error)
|
||||||
HeaderByHash(blockHash common.Hash) (*ethtypes.Header, error)
|
HeaderByHash(blockHash common.Hash) (*ethtypes.Header, error)
|
||||||
GetBlockByNumber(blockNum rpctypes.BlockNumber, fullTx bool) (map[string]interface{}, error)
|
GetBlockByNumber(blockNum rpctypes.BlockNumber, fullTx bool) (map[string]interface{}, error)
|
||||||
@ -60,13 +61,12 @@ func New(clientCtx clientcontext.CLIContext) *EthermintBackend {
|
|||||||
|
|
||||||
// BlockNumber returns the current block number.
|
// BlockNumber returns the current block number.
|
||||||
func (b *EthermintBackend) BlockNumber() (hexutil.Uint64, error) {
|
func (b *EthermintBackend) BlockNumber() (hexutil.Uint64, error) {
|
||||||
// NOTE: using 0 as min and max height returns the blockchain info up to the latest block.
|
blockNumber, err := b.LatestBlockNumber()
|
||||||
info, err := b.clientCtx.Client.BlockchainInfo(0, 0)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return hexutil.Uint64(0), err
|
return hexutil.Uint64(0), err
|
||||||
}
|
}
|
||||||
|
|
||||||
return hexutil.Uint64(info.LastHeight), nil
|
return hexutil.Uint64(blockNumber), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetBlockByNumber returns the block identified by number.
|
// GetBlockByNumber returns the block identified by number.
|
||||||
@ -196,7 +196,7 @@ func (b *EthermintBackend) PendingTransactions() ([]*rpctypes.Transaction, error
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
transactions := make([]*rpctypes.Transaction, pendingTxs.Count)
|
transactions := make([]*rpctypes.Transaction, 0)
|
||||||
for _, tx := range pendingTxs.Txs {
|
for _, tx := range pendingTxs.Txs {
|
||||||
ethTx, err := rpctypes.RawTxToEthTx(b.clientCtx, tx)
|
ethTx, err := rpctypes.RawTxToEthTx(b.clientCtx, tx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -209,10 +209,8 @@ func (b *EthermintBackend) PendingTransactions() ([]*rpctypes.Transaction, error
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
transactions = append(transactions, rpcTx)
|
transactions = append(transactions, rpcTx)
|
||||||
}
|
}
|
||||||
|
|
||||||
return transactions, nil
|
return transactions, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -257,3 +255,14 @@ func (b *EthermintBackend) GetLogs(blockHash common.Hash) ([][]*ethtypes.Log, er
|
|||||||
func (b *EthermintBackend) BloomStatus() (uint64, uint64) {
|
func (b *EthermintBackend) BloomStatus() (uint64, uint64) {
|
||||||
return 4096, 0
|
return 4096, 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// LatestBlockNumber gets the latest block height in int64 format.
|
||||||
|
func (b *EthermintBackend) LatestBlockNumber() (int64, error) {
|
||||||
|
// NOTE: using 0 as min and max height returns the blockchain info up to the latest block.
|
||||||
|
info, err := b.clientCtx.Client.BlockchainInfo(0, 0)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return info.LastHeight, nil
|
||||||
|
}
|
||||||
|
@ -8,6 +8,7 @@ import (
|
|||||||
"math/big"
|
"math/big"
|
||||||
"os"
|
"os"
|
||||||
"sync"
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
|
|
||||||
@ -212,7 +213,7 @@ func (api *PublicEthereumAPI) Accounts() ([]common.Address, error) {
|
|||||||
return addresses, nil
|
return addresses, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// rpctypes.BlockNumber returns the current block number.
|
// BlockNumber returns the current block number.
|
||||||
func (api *PublicEthereumAPI) BlockNumber() (hexutil.Uint64, error) {
|
func (api *PublicEthereumAPI) BlockNumber() (hexutil.Uint64, error) {
|
||||||
api.logger.Debug("eth_blockNumber")
|
api.logger.Debug("eth_blockNumber")
|
||||||
return api.backend.BlockNumber()
|
return api.backend.BlockNumber()
|
||||||
@ -221,7 +222,12 @@ func (api *PublicEthereumAPI) BlockNumber() (hexutil.Uint64, error) {
|
|||||||
// GetBalance returns the provided account's balance up to the provided block number.
|
// GetBalance returns the provided account's balance up to the provided block number.
|
||||||
func (api *PublicEthereumAPI) GetBalance(address common.Address, blockNum rpctypes.BlockNumber) (*hexutil.Big, error) {
|
func (api *PublicEthereumAPI) GetBalance(address common.Address, blockNum rpctypes.BlockNumber) (*hexutil.Big, error) {
|
||||||
api.logger.Debug("eth_getBalance", "address", address, "block number", blockNum)
|
api.logger.Debug("eth_getBalance", "address", address, "block number", blockNum)
|
||||||
clientCtx := api.clientCtx.WithHeight(blockNum.Int64())
|
|
||||||
|
clientCtx := api.clientCtx
|
||||||
|
if !(blockNum == rpctypes.PendingBlockNumber || blockNum == rpctypes.LatestBlockNumber) {
|
||||||
|
clientCtx = api.clientCtx.WithHeight(blockNum.Int64())
|
||||||
|
}
|
||||||
|
|
||||||
res, _, err := clientCtx.QueryWithData(fmt.Sprintf("custom/%s/balance/%s", evmtypes.ModuleName, address.Hex()), nil)
|
res, _, err := clientCtx.QueryWithData(fmt.Sprintf("custom/%s/balance/%s", evmtypes.ModuleName, address.Hex()), nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -234,6 +240,29 @@ func (api *PublicEthereumAPI) GetBalance(address common.Address, blockNum rpctyp
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if blockNum != rpctypes.PendingBlockNumber {
|
||||||
|
return (*hexutil.Big)(val), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// update the address balance with the pending transactions value (if applicable)
|
||||||
|
pendingTxs, err := api.backend.PendingTransactions()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tx := range pendingTxs {
|
||||||
|
if tx == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if tx.From == address {
|
||||||
|
val = new(big.Int).Sub(val, tx.Value.ToInt())
|
||||||
|
}
|
||||||
|
if *tx.To == address {
|
||||||
|
val = new(big.Int).Add(val, tx.Value.ToInt())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return (*hexutil.Big)(val), nil
|
return (*hexutil.Big)(val), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -254,20 +283,16 @@ func (api *PublicEthereumAPI) GetStorageAt(address common.Address, key string, b
|
|||||||
// GetTransactionCount returns the number of transactions at the given address up to the given block number.
|
// GetTransactionCount returns the number of transactions at the given address up to the given block number.
|
||||||
func (api *PublicEthereumAPI) GetTransactionCount(address common.Address, blockNum rpctypes.BlockNumber) (*hexutil.Uint64, error) {
|
func (api *PublicEthereumAPI) GetTransactionCount(address common.Address, blockNum rpctypes.BlockNumber) (*hexutil.Uint64, error) {
|
||||||
api.logger.Debug("eth_getTransactionCount", "address", address, "block number", blockNum)
|
api.logger.Debug("eth_getTransactionCount", "address", address, "block number", blockNum)
|
||||||
clientCtx := api.clientCtx.WithHeight(blockNum.Int64())
|
|
||||||
|
|
||||||
// Get nonce (sequence) from account
|
clientCtx := api.clientCtx
|
||||||
from := sdk.AccAddress(address.Bytes())
|
pending := blockNum == rpctypes.PendingBlockNumber
|
||||||
accRet := authtypes.NewAccountRetriever(clientCtx)
|
|
||||||
|
|
||||||
err := accRet.EnsureExists(from)
|
// pass the given block height to the context if the height is not pending or latest
|
||||||
if err != nil {
|
if !pending && blockNum != rpctypes.LatestBlockNumber {
|
||||||
// account doesn't exist yet, return 0
|
clientCtx = api.clientCtx.WithHeight(blockNum.Int64())
|
||||||
n := hexutil.Uint64(0)
|
|
||||||
return &n, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_, nonce, err := accRet.GetAccountNumberSequence(from)
|
nonce, err := api.accountNonce(clientCtx, address, pending)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -298,17 +323,54 @@ func (api *PublicEthereumAPI) GetBlockTransactionCountByHash(hash common.Hash) *
|
|||||||
return &n
|
return &n
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetBlockTransactionCountByNumber returns the number of transactions in the block identified by number.
|
// GetBlockTransactionCountByNumber returns the number of transactions in the block identified by its height.
|
||||||
func (api *PublicEthereumAPI) GetBlockTransactionCountByNumber(blockNum rpctypes.BlockNumber) *hexutil.Uint {
|
func (api *PublicEthereumAPI) GetBlockTransactionCountByNumber(blockNum rpctypes.BlockNumber) *hexutil.Uint {
|
||||||
api.logger.Debug("eth_getBlockTransactionCountByNumber", "block number", blockNum)
|
api.logger.Debug("eth_getBlockTransactionCountByNumber", "block number", blockNum)
|
||||||
height := blockNum.Int64()
|
|
||||||
|
var (
|
||||||
|
height int64
|
||||||
|
err error
|
||||||
|
txCount hexutil.Uint
|
||||||
|
txs int
|
||||||
|
)
|
||||||
|
|
||||||
|
switch blockNum {
|
||||||
|
case rpctypes.PendingBlockNumber:
|
||||||
|
height, err = api.backend.LatestBlockNumber()
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
resBlock, err := api.clientCtx.Client.Block(&height)
|
resBlock, err := api.clientCtx.Client.Block(&height)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
// get the pending transaction count
|
||||||
|
pendingTxs, err := api.backend.PendingTransactions()
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
txs = len(resBlock.Block.Txs) + len(pendingTxs)
|
||||||
|
case rpctypes.LatestBlockNumber:
|
||||||
|
height, err = api.backend.LatestBlockNumber()
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
resBlock, err := api.clientCtx.Client.Block(&height)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
txs = len(resBlock.Block.Txs)
|
||||||
|
default:
|
||||||
|
height = blockNum.Int64()
|
||||||
|
resBlock, err := api.clientCtx.Client.Block(&height)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
txs = len(resBlock.Block.Txs)
|
||||||
|
}
|
||||||
|
|
||||||
n := hexutil.Uint(len(resBlock.Block.Txs))
|
txCount = hexutil.Uint(txs)
|
||||||
return &n
|
return &txCount
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetUncleCountByBlockHash returns the number of uncles in the block idenfied by hash. Always zero.
|
// GetUncleCountByBlockHash returns the number of uncles in the block idenfied by hash. Always zero.
|
||||||
@ -385,6 +447,11 @@ func (api *PublicEthereumAPI) SendTransaction(args rpctypes.SendTxArgs) (common.
|
|||||||
return common.Hash{}, err
|
return common.Hash{}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := tx.ValidateBasic(); err != nil {
|
||||||
|
api.logger.Debug("tx failed basic validation", "error", err)
|
||||||
|
return common.Hash{}, err
|
||||||
|
}
|
||||||
|
|
||||||
// Sign transaction
|
// Sign transaction
|
||||||
if err := tx.Sign(api.chainIDEpoch, key.ToECDSA()); err != nil {
|
if err := tx.Sign(api.chainIDEpoch, key.ToECDSA()); err != nil {
|
||||||
api.logger.Debug("failed to sign tx", "error", err)
|
api.logger.Debug("failed to sign tx", "error", err)
|
||||||
@ -457,13 +524,13 @@ func (api *PublicEthereumAPI) Call(args rpctypes.CallArgs, blockNr rpctypes.Bloc
|
|||||||
// DoCall performs a simulated call operation through the evmtypes. It returns the
|
// DoCall performs a simulated call operation through the evmtypes. It returns the
|
||||||
// estimated gas used on the operation or an error if fails.
|
// estimated gas used on the operation or an error if fails.
|
||||||
func (api *PublicEthereumAPI) doCall(
|
func (api *PublicEthereumAPI) doCall(
|
||||||
args rpctypes.CallArgs, blockNr rpctypes.BlockNumber, globalGasCap *big.Int,
|
args rpctypes.CallArgs, blockNum rpctypes.BlockNumber, globalGasCap *big.Int,
|
||||||
) (*sdk.SimulationResponse, error) {
|
) (*sdk.SimulationResponse, error) {
|
||||||
// Set height for historical queries
|
|
||||||
clientCtx := api.clientCtx
|
|
||||||
|
|
||||||
if blockNr.Int64() != 0 {
|
clientCtx := api.clientCtx
|
||||||
clientCtx = api.clientCtx.WithHeight(blockNr.Int64())
|
// pass the given block height to the context if the height is not pending or latest
|
||||||
|
if !(blockNum == rpctypes.PendingBlockNumber || blockNum == rpctypes.LatestBlockNumber) {
|
||||||
|
clientCtx = api.clientCtx.WithHeight(blockNum.Int64())
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set sender address or use a default if none specified
|
// Set sender address or use a default if none specified
|
||||||
@ -513,17 +580,29 @@ func (api *PublicEthereumAPI) doCall(
|
|||||||
toAddr = sdk.AccAddress(args.To.Bytes())
|
toAddr = sdk.AccAddress(args.To.Bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var msgs []sdk.Msg
|
||||||
// Create new call message
|
// Create new call message
|
||||||
msg := evmtypes.NewMsgEthermint(0, &toAddr, sdk.NewIntFromBigInt(value), gas,
|
msg := evmtypes.NewMsgEthermint(0, &toAddr, sdk.NewIntFromBigInt(value), gas,
|
||||||
sdk.NewIntFromBigInt(gasPrice), data, sdk.AccAddress(addr.Bytes()))
|
sdk.NewIntFromBigInt(gasPrice), data, sdk.AccAddress(addr.Bytes()))
|
||||||
|
msgs = append(msgs, msg)
|
||||||
|
|
||||||
if err := msg.ValidateBasic(); err != nil {
|
// convert the pending transactions into ethermint msgs
|
||||||
|
if blockNum == rpctypes.PendingBlockNumber {
|
||||||
|
pendingMsgs, err := api.pendingMsgs()
|
||||||
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
msgs = append(msgs, pendingMsgs...)
|
||||||
|
}
|
||||||
|
|
||||||
// Generate tx to be used to simulate (signature isn't needed)
|
// Generate tx to be used to simulate (signature isn't needed)
|
||||||
var stdSig authtypes.StdSignature
|
var stdSig authtypes.StdSignature
|
||||||
tx := authtypes.NewStdTx([]sdk.Msg{msg}, authtypes.StdFee{}, []authtypes.StdSignature{stdSig}, "")
|
stdSigs := []authtypes.StdSignature{stdSig}
|
||||||
|
|
||||||
|
tx := authtypes.NewStdTx(msgs, authtypes.StdFee{}, stdSigs, "")
|
||||||
|
if err := tx.ValidateBasic(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
txEncoder := authclient.GetTxEncoder(clientCtx.Codec)
|
txEncoder := authclient.GetTxEncoder(clientCtx.Codec)
|
||||||
txBytes, err := txEncoder(tx)
|
txBytes, err := txEncoder(tx)
|
||||||
@ -571,14 +650,71 @@ func (api *PublicEthereumAPI) GetBlockByHash(hash common.Hash, fullTx bool) (map
|
|||||||
// GetBlockByNumber returns the block identified by number.
|
// GetBlockByNumber returns the block identified by number.
|
||||||
func (api *PublicEthereumAPI) GetBlockByNumber(blockNum rpctypes.BlockNumber, fullTx bool) (map[string]interface{}, error) {
|
func (api *PublicEthereumAPI) GetBlockByNumber(blockNum rpctypes.BlockNumber, fullTx bool) (map[string]interface{}, error) {
|
||||||
api.logger.Debug("eth_getBlockByNumber", "number", blockNum, "full", fullTx)
|
api.logger.Debug("eth_getBlockByNumber", "number", blockNum, "full", fullTx)
|
||||||
|
|
||||||
|
if blockNum != rpctypes.PendingBlockNumber {
|
||||||
return api.backend.GetBlockByNumber(blockNum, fullTx)
|
return api.backend.GetBlockByNumber(blockNum, fullTx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
height, err := api.backend.LatestBlockNumber()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// latest block info
|
||||||
|
latestBlock, err := api.clientCtx.Client.Block(&height)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// number of pending txs queried from the mempool
|
||||||
|
unconfirmedTxs, err := api.clientCtx.Client.UnconfirmedTxs(1000)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
pendingTxs, gasUsed, err := rpctypes.EthTransactionsFromTendermint(api.clientCtx, unconfirmedTxs.Txs)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return rpctypes.FormatBlock(
|
||||||
|
tmtypes.Header{
|
||||||
|
Version: latestBlock.Block.Version,
|
||||||
|
ChainID: api.clientCtx.ChainID,
|
||||||
|
Height: height + 1,
|
||||||
|
Time: time.Unix(0, 0),
|
||||||
|
LastBlockID: latestBlock.BlockID,
|
||||||
|
ValidatorsHash: latestBlock.Block.NextValidatorsHash,
|
||||||
|
},
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
gasUsed,
|
||||||
|
pendingTxs,
|
||||||
|
ethtypes.Bloom{},
|
||||||
|
), nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
// GetTransactionByHash returns the transaction identified by hash.
|
// GetTransactionByHash returns the transaction identified by hash.
|
||||||
func (api *PublicEthereumAPI) GetTransactionByHash(hash common.Hash) (*rpctypes.Transaction, error) {
|
func (api *PublicEthereumAPI) GetTransactionByHash(hash common.Hash) (*rpctypes.Transaction, error) {
|
||||||
api.logger.Debug("eth_getTransactionByHash", "hash", hash)
|
api.logger.Debug("eth_getTransactionByHash", "hash", hash)
|
||||||
|
|
||||||
tx, err := api.clientCtx.Client.Tx(hash.Bytes(), false)
|
tx, err := api.clientCtx.Client.Tx(hash.Bytes(), false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
// check if the tx is on the mempool
|
||||||
|
pendingTxs, pendingErr := api.PendingTransactions()
|
||||||
|
if pendingErr != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(pendingTxs) != 0 {
|
||||||
|
for _, tx := range pendingTxs {
|
||||||
|
if tx != nil && hash == tx.Hash {
|
||||||
|
return tx, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Return nil for transaction when not found
|
// Return nil for transaction when not found
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
@ -622,7 +758,37 @@ func (api *PublicEthereumAPI) GetTransactionByBlockHashAndIndex(hash common.Hash
|
|||||||
// GetTransactionByBlockNumberAndIndex returns the transaction identified by number and index.
|
// GetTransactionByBlockNumberAndIndex returns the transaction identified by number and index.
|
||||||
func (api *PublicEthereumAPI) GetTransactionByBlockNumberAndIndex(blockNum rpctypes.BlockNumber, idx hexutil.Uint) (*rpctypes.Transaction, error) {
|
func (api *PublicEthereumAPI) GetTransactionByBlockNumberAndIndex(blockNum rpctypes.BlockNumber, idx hexutil.Uint) (*rpctypes.Transaction, error) {
|
||||||
api.logger.Debug("eth_getTransactionByBlockNumberAndIndex", "number", blockNum, "index", idx)
|
api.logger.Debug("eth_getTransactionByBlockNumberAndIndex", "number", blockNum, "index", idx)
|
||||||
height := blockNum.Int64()
|
var (
|
||||||
|
height int64
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
|
||||||
|
switch blockNum {
|
||||||
|
case rpctypes.PendingBlockNumber:
|
||||||
|
// get all the EVM pending txs
|
||||||
|
pendingTxs, err := api.backend.PendingTransactions()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// return if index out of bounds
|
||||||
|
if uint64(idx) >= uint64(len(pendingTxs)) {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// change back to pendingTxs[idx] once pending queue is fixed.
|
||||||
|
return pendingTxs[int(idx)], nil
|
||||||
|
|
||||||
|
case rpctypes.LatestBlockNumber:
|
||||||
|
height, err = api.backend.LatestBlockNumber()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
height = blockNum.Int64()
|
||||||
|
}
|
||||||
|
|
||||||
resBlock, err := api.clientCtx.Client.Block(&height)
|
resBlock, err := api.clientCtx.Client.Block(&height)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -731,7 +897,7 @@ func (api *PublicEthereumAPI) GetTransactionReceipt(hash common.Hash) (map[strin
|
|||||||
// PendingTransactions returns the transactions that are in the transaction pool
|
// PendingTransactions returns the transactions that are in the transaction pool
|
||||||
// and have a from address that is one of the accounts this node manages.
|
// and have a from address that is one of the accounts this node manages.
|
||||||
func (api *PublicEthereumAPI) PendingTransactions() ([]*rpctypes.Transaction, error) {
|
func (api *PublicEthereumAPI) PendingTransactions() ([]*rpctypes.Transaction, error) {
|
||||||
api.logger.Debug("eth_getPendingTransactions")
|
api.logger.Debug("eth_pendingTransactions")
|
||||||
return api.backend.PendingTransactions()
|
return api.backend.PendingTransactions()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -821,8 +987,7 @@ func (api *PublicEthereumAPI) GetProof(address common.Address, storageKeys []str
|
|||||||
// generateFromArgs populates tx message with args (used in RPC API)
|
// generateFromArgs populates tx message with args (used in RPC API)
|
||||||
func (api *PublicEthereumAPI) generateFromArgs(args rpctypes.SendTxArgs) (*evmtypes.MsgEthereumTx, error) {
|
func (api *PublicEthereumAPI) generateFromArgs(args rpctypes.SendTxArgs) (*evmtypes.MsgEthereumTx, error) {
|
||||||
var (
|
var (
|
||||||
nonce uint64
|
nonce, gasLimit uint64
|
||||||
gasLimit uint64
|
|
||||||
err error
|
err error
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -830,31 +995,24 @@ func (api *PublicEthereumAPI) generateFromArgs(args rpctypes.SendTxArgs) (*evmty
|
|||||||
gasPrice := (*big.Int)(args.GasPrice)
|
gasPrice := (*big.Int)(args.GasPrice)
|
||||||
|
|
||||||
if args.GasPrice == nil {
|
if args.GasPrice == nil {
|
||||||
|
|
||||||
// Set default gas price
|
// Set default gas price
|
||||||
// TODO: Change to min gas price from context once available through server/daemon
|
// TODO: Change to min gas price from context once available through server/daemon
|
||||||
gasPrice = big.NewInt(ethermint.DefaultGasPrice)
|
gasPrice = big.NewInt(ethermint.DefaultGasPrice)
|
||||||
}
|
}
|
||||||
|
|
||||||
if args.Nonce == nil {
|
if args.Nonce == nil {
|
||||||
// Get nonce (sequence) from account
|
// get the nonce from the account retriever and the pending transactions
|
||||||
from := sdk.AccAddress(args.From.Bytes())
|
nonce, err = api.accountNonce(api.clientCtx, args.From, true)
|
||||||
accRet := authtypes.NewAccountRetriever(api.clientCtx)
|
|
||||||
|
|
||||||
if api.clientCtx.Keybase == nil {
|
|
||||||
return nil, fmt.Errorf("clientCtx.Keybase is nil")
|
|
||||||
}
|
|
||||||
|
|
||||||
_, nonce, err = accRet.GetAccountNumberSequence(from)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
nonce = (uint64)(*args.Nonce)
|
nonce = (uint64)(*args.Nonce)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
if args.Data != nil && args.Input != nil && !bytes.Equal(*args.Data, *args.Input) {
|
if args.Data != nil && args.Input != nil && !bytes.Equal(*args.Data, *args.Input) {
|
||||||
return nil, errors.New(`both "data" and "input" are set and not equal. Please use "input" to pass transaction call data`)
|
return nil, errors.New("both 'data' and 'input' are set and not equal. Please use 'input' to pass transaction call data")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sets input to either Input or Data, if both are set and not equal error above returns
|
// Sets input to either Input or Data, if both are set and not equal error above returns
|
||||||
@ -891,3 +1049,89 @@ func (api *PublicEthereumAPI) generateFromArgs(args rpctypes.SendTxArgs) (*evmty
|
|||||||
|
|
||||||
return &msg, nil
|
return &msg, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// pendingMsgs constructs an array of sdk.Msg. This method will check pending transactions and convert
|
||||||
|
// those transactions into ethermint messages.
|
||||||
|
func (api *PublicEthereumAPI) pendingMsgs() ([]sdk.Msg, error) {
|
||||||
|
// nolint: prealloc
|
||||||
|
var msgs []sdk.Msg
|
||||||
|
|
||||||
|
pendingTxs, err := api.PendingTransactions()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, pendingTx := range pendingTxs {
|
||||||
|
// NOTE: we have to construct the EVM transaction instead of just casting from the tendermint
|
||||||
|
// transactions because PendingTransactions only checks for MsgEthereumTx messages.
|
||||||
|
|
||||||
|
pendingTo := sdk.AccAddress(pendingTx.To.Bytes())
|
||||||
|
pendingFrom := sdk.AccAddress(pendingTx.From.Bytes())
|
||||||
|
pendingGas, err := hexutil.DecodeUint64(pendingTx.Gas.String())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
pendingValue := pendingTx.Value.ToInt()
|
||||||
|
pendingGasPrice := new(big.Int).SetUint64(ethermint.DefaultGasPrice)
|
||||||
|
if pendingTx.GasPrice != nil {
|
||||||
|
pendingGasPrice = pendingTx.GasPrice.ToInt()
|
||||||
|
}
|
||||||
|
|
||||||
|
pendingData := pendingTx.Input
|
||||||
|
|
||||||
|
msg := evmtypes.NewMsgEthermint(0, &pendingTo, sdk.NewIntFromBigInt(pendingValue), pendingGas,
|
||||||
|
sdk.NewIntFromBigInt(pendingGasPrice), pendingData, pendingFrom)
|
||||||
|
|
||||||
|
msgs = append(msgs, msg)
|
||||||
|
}
|
||||||
|
return msgs, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// accountNonce returns looks up the transaction nonce count for a given address. If the pending boolean
|
||||||
|
// is set to true, it will add to the counter all the uncommitted EVM transactions sent from the address.
|
||||||
|
// NOTE: The function returns no error if the account doesn't exist.
|
||||||
|
func (api *PublicEthereumAPI) accountNonce(
|
||||||
|
clientCtx clientcontext.CLIContext, address common.Address, pending bool,
|
||||||
|
) (uint64, error) {
|
||||||
|
// Get nonce (sequence) from sender account
|
||||||
|
from := sdk.AccAddress(address.Bytes())
|
||||||
|
|
||||||
|
// use a the given client context in case its wrapped with a custom height
|
||||||
|
accRet := authtypes.NewAccountRetriever(clientCtx)
|
||||||
|
|
||||||
|
if err := accRet.EnsureExists(from); err != nil {
|
||||||
|
// account doesn't exist yet, return 0
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
_, nonce, err := accRet.GetAccountNumberSequence(from)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !pending {
|
||||||
|
return nonce, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// the account retriever doesn't include the uncommitted transactions on the nonce so we need to
|
||||||
|
// to manually add them.
|
||||||
|
pendingTxs, err := api.backend.PendingTransactions()
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// add the uncommitted txs to the nonce counter
|
||||||
|
if len(pendingTxs) != 0 {
|
||||||
|
for i := range pendingTxs {
|
||||||
|
if pendingTxs[i] == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if pendingTxs[i].From == address {
|
||||||
|
nonce++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nonce, nil
|
||||||
|
}
|
||||||
|
@ -18,6 +18,9 @@ const (
|
|||||||
|
|
||||||
// EarliestBlockNumber mapping from "earliest" to 1 for tm query (earliest query not supported)
|
// EarliestBlockNumber mapping from "earliest" to 1 for tm query (earliest query not supported)
|
||||||
EarliestBlockNumber = BlockNumber(1)
|
EarliestBlockNumber = BlockNumber(1)
|
||||||
|
|
||||||
|
// PendingBlockNumber mapping from "pending" to -1 for tm query
|
||||||
|
PendingBlockNumber = BlockNumber(-1)
|
||||||
)
|
)
|
||||||
|
|
||||||
// NewBlockNumber creates a new BlockNumber instance.
|
// NewBlockNumber creates a new BlockNumber instance.
|
||||||
@ -45,7 +48,7 @@ func (bn *BlockNumber) UnmarshalJSON(data []byte) error {
|
|||||||
*bn = LatestBlockNumber
|
*bn = LatestBlockNumber
|
||||||
return nil
|
return nil
|
||||||
case "pending":
|
case "pending":
|
||||||
*bn = LatestBlockNumber
|
*bn = PendingBlockNumber
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@ func RawTxToEthTx(clientCtx clientcontext.CLIContext, bz []byte) (*evmtypes.MsgE
|
|||||||
|
|
||||||
ethTx, ok := tx.(evmtypes.MsgEthereumTx)
|
ethTx, ok := tx.(evmtypes.MsgEthereumTx)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, fmt.Errorf("invalid transaction type %T, expected %T", tx, &evmtypes.MsgEthereumTx{})
|
return nil, fmt.Errorf("invalid transaction type %T, expected %T", tx, evmtypes.MsgEthereumTx{})
|
||||||
}
|
}
|
||||||
return ðTx, nil
|
return ðTx, nil
|
||||||
}
|
}
|
||||||
@ -90,7 +90,7 @@ func EthBlockFromTendermint(clientCtx clientcontext.CLIContext, block *tmtypes.B
|
|||||||
|
|
||||||
bloom := bloomRes.Bloom
|
bloom := bloomRes.Bloom
|
||||||
|
|
||||||
return formatBlock(block.Header, block.Size(), gasLimit, gasUsed, transactions, bloom), nil
|
return FormatBlock(block.Header, block.Size(), gasLimit, gasUsed, transactions, bloom), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EthHeaderFromTendermint is an util function that returns an Ethereum Header
|
// EthHeaderFromTendermint is an util function that returns an Ethereum Header
|
||||||
@ -150,7 +150,9 @@ func BlockMaxGasFromConsensusParams(_ context.Context, clientCtx clientcontext.C
|
|||||||
return gasLimit, nil
|
return gasLimit, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func formatBlock(
|
// FormatBlock creates an ethereum block from a tendermint header and ethereum-formatted
|
||||||
|
// transactions.
|
||||||
|
func FormatBlock(
|
||||||
header tmtypes.Header, size int, gasLimit int64,
|
header tmtypes.Header, size int, gasLimit int64,
|
||||||
gasUsed *big.Int, transactions interface{}, bloom ethtypes.Bloom,
|
gasUsed *big.Int, transactions interface{}, bloom ethtypes.Bloom,
|
||||||
) map[string]interface{} {
|
) map[string]interface{} {
|
||||||
|
@ -12,7 +12,6 @@ TEST_QTD=1
|
|||||||
#PORT AND RPC_PORT 3 initial digits, to be concat with a suffix later when node is initialized
|
#PORT AND RPC_PORT 3 initial digits, to be concat with a suffix later when node is initialized
|
||||||
RPC_PORT="854"
|
RPC_PORT="854"
|
||||||
IP_ADDR="0.0.0.0"
|
IP_ADDR="0.0.0.0"
|
||||||
MODE="rpc"
|
|
||||||
|
|
||||||
KEY="mykey"
|
KEY="mykey"
|
||||||
CHAINID="ethermint-2"
|
CHAINID="ethermint-2"
|
||||||
@ -33,7 +32,7 @@ usage() {
|
|||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
while getopts "h?t:q:z:s:" args; do
|
while getopts "h?t:q:z:s:m:" args; do
|
||||||
case $args in
|
case $args in
|
||||||
h|\?)
|
h|\?)
|
||||||
usage;
|
usage;
|
||||||
@ -42,6 +41,7 @@ while getopts "h?t:q:z:s:" args; do
|
|||||||
q ) QTD=${OPTARG};;
|
q ) QTD=${OPTARG};;
|
||||||
z ) TEST_QTD=${OPTARG};;
|
z ) TEST_QTD=${OPTARG};;
|
||||||
s ) SLEEP_TIMEOUT=${OPTARG};;
|
s ) SLEEP_TIMEOUT=${OPTARG};;
|
||||||
|
m ) MODE=${OPTARG};;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
@ -92,6 +92,18 @@ init_func() {
|
|||||||
"$PWD"/build/ethermintd collect-gentxs --home "$DATA_DIR$i"
|
"$PWD"/build/ethermintd collect-gentxs --home "$DATA_DIR$i"
|
||||||
echo "prepare genesis: Run validate-genesis to ensure everything worked and that the genesis file is setup correctly"
|
echo "prepare genesis: Run validate-genesis to ensure everything worked and that the genesis file is setup correctly"
|
||||||
"$PWD"/build/ethermintd validate-genesis --home "$DATA_DIR$i"
|
"$PWD"/build/ethermintd validate-genesis --home "$DATA_DIR$i"
|
||||||
|
|
||||||
|
if [[ $MODE == "pending" ]]; then
|
||||||
|
ls $DATA_DIR$i
|
||||||
|
# sed -i 's/create_empty_blocks_interval = "0s"/create_empty_blocks_interval = "30s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_propose = "3s"/timeout_propose = "30s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_propose_delta = "500ms"/timeout_propose_delta = "2s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_prevote = "1s"/timeout_prevote = "120s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_prevote_delta = "500ms"/timeout_prevote_delta = "2s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_precommit = "1s"/timeout_precommit = "10s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_precommit_delta = "500ms"/timeout_precommit_delta = "2s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
sed -i 's/timeout_commit = "5s"/timeout_commit = "150s"/g' $DATA_DIR$i/config/config.toml
|
||||||
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
start_func() {
|
start_func() {
|
||||||
@ -138,12 +150,17 @@ echo "done sleeping"
|
|||||||
|
|
||||||
set +e
|
set +e
|
||||||
|
|
||||||
if [[ -z $TEST || $TEST == "rpc" ]]; then
|
if [[ -z $TEST || $TEST == "rpc" || $TEST == "pending" ]]; then
|
||||||
|
|
||||||
for i in $(seq 1 "$TEST_QTD"); do
|
for i in $(seq 1 "$TEST_QTD"); do
|
||||||
HOST_RPC=http://$IP_ADDR:$RPC_PORT"$i"
|
HOST_RPC=http://$IP_ADDR:$RPC_PORT"$i"
|
||||||
echo "going to test ethermint node $HOST_RPC ..."
|
echo "going to test ethermint node $HOST_RPC ..."
|
||||||
|
if [[ $MODE == "pending" ]]; then
|
||||||
|
sleep 150
|
||||||
|
MODE=$MODE HOST=$HOST_RPC go test -v ./tests/tests-pending/rpc_pending_test.go
|
||||||
|
else
|
||||||
MODE=$MODE HOST=$HOST_RPC go test ./tests/... -timeout=300s -v -short
|
MODE=$MODE HOST=$HOST_RPC go test ./tests/... -timeout=300s -v -short
|
||||||
|
fi
|
||||||
|
|
||||||
RPC_FAIL=$?
|
RPC_FAIL=$?
|
||||||
done
|
done
|
||||||
|
@ -12,7 +12,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestPersonal_ListAccounts(t *testing.T) {
|
func TestPersonal_ListAccounts(t *testing.T) {
|
||||||
rpcRes := call(t, "personal_listAccounts", []string{})
|
rpcRes := Call(t, "personal_listAccounts", []string{})
|
||||||
|
|
||||||
var res []hexutil.Bytes
|
var res []hexutil.Bytes
|
||||||
err := json.Unmarshal(rpcRes.Result, &res)
|
err := json.Unmarshal(rpcRes.Result, &res)
|
||||||
@ -21,12 +21,12 @@ func TestPersonal_ListAccounts(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestPersonal_NewAccount(t *testing.T) {
|
func TestPersonal_NewAccount(t *testing.T) {
|
||||||
rpcRes := call(t, "personal_newAccount", []string{"password"})
|
rpcRes := Call(t, "personal_newAccount", []string{"password"})
|
||||||
var addr common.Address
|
var addr common.Address
|
||||||
err := json.Unmarshal(rpcRes.Result, &addr)
|
err := json.Unmarshal(rpcRes.Result, &addr)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
rpcRes = call(t, "personal_listAccounts", []string{})
|
rpcRes = Call(t, "personal_listAccounts", []string{})
|
||||||
var res []hexutil.Bytes
|
var res []hexutil.Bytes
|
||||||
err = json.Unmarshal(rpcRes.Result, &res)
|
err = json.Unmarshal(rpcRes.Result, &res)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
@ -34,7 +34,7 @@ func TestPersonal_NewAccount(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestPersonal_Sign(t *testing.T) {
|
func TestPersonal_Sign(t *testing.T) {
|
||||||
rpcRes := call(t, "personal_sign", []interface{}{hexutil.Bytes{0x88}, hexutil.Bytes(from), ""})
|
rpcRes := Call(t, "personal_sign", []interface{}{hexutil.Bytes{0x88}, hexutil.Bytes(from), ""})
|
||||||
|
|
||||||
var res hexutil.Bytes
|
var res hexutil.Bytes
|
||||||
err := json.Unmarshal(rpcRes.Result, &res)
|
err := json.Unmarshal(rpcRes.Result, &res)
|
||||||
@ -49,7 +49,7 @@ func TestPersonal_ImportRawKey(t *testing.T) {
|
|||||||
|
|
||||||
// parse priv key to hex
|
// parse priv key to hex
|
||||||
hexPriv := common.Bytes2Hex(ethcrypto.FromECDSA(privkey))
|
hexPriv := common.Bytes2Hex(ethcrypto.FromECDSA(privkey))
|
||||||
rpcRes := call(t, "personal_importRawKey", []string{hexPriv, "password"})
|
rpcRes := Call(t, "personal_importRawKey", []string{hexPriv, "password"})
|
||||||
|
|
||||||
var res hexutil.Bytes
|
var res hexutil.Bytes
|
||||||
err = json.Unmarshal(rpcRes.Result, &res)
|
err = json.Unmarshal(rpcRes.Result, &res)
|
||||||
@ -63,14 +63,14 @@ func TestPersonal_ImportRawKey(t *testing.T) {
|
|||||||
|
|
||||||
func TestPersonal_EcRecover(t *testing.T) {
|
func TestPersonal_EcRecover(t *testing.T) {
|
||||||
data := hexutil.Bytes{0x88}
|
data := hexutil.Bytes{0x88}
|
||||||
rpcRes := call(t, "personal_sign", []interface{}{data, hexutil.Bytes(from), ""})
|
rpcRes := Call(t, "personal_sign", []interface{}{data, hexutil.Bytes(from), ""})
|
||||||
|
|
||||||
var res hexutil.Bytes
|
var res hexutil.Bytes
|
||||||
err := json.Unmarshal(rpcRes.Result, &res)
|
err := json.Unmarshal(rpcRes.Result, &res)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, 65, len(res))
|
require.Equal(t, 65, len(res))
|
||||||
|
|
||||||
rpcRes = call(t, "personal_ecRecover", []interface{}{data, res})
|
rpcRes = Call(t, "personal_ecRecover", []interface{}{data, res})
|
||||||
var ecrecoverRes common.Address
|
var ecrecoverRes common.Address
|
||||||
err = json.Unmarshal(rpcRes.Result, &ecrecoverRes)
|
err = json.Unmarshal(rpcRes.Result, &ecrecoverRes)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
@ -79,23 +79,23 @@ func TestPersonal_EcRecover(t *testing.T) {
|
|||||||
|
|
||||||
func TestPersonal_UnlockAccount(t *testing.T) {
|
func TestPersonal_UnlockAccount(t *testing.T) {
|
||||||
pswd := "nootwashere"
|
pswd := "nootwashere"
|
||||||
rpcRes := call(t, "personal_newAccount", []string{pswd})
|
rpcRes := Call(t, "personal_newAccount", []string{pswd})
|
||||||
var addr common.Address
|
var addr common.Address
|
||||||
err := json.Unmarshal(rpcRes.Result, &addr)
|
err := json.Unmarshal(rpcRes.Result, &addr)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
// try to sign, should be locked
|
// try to sign, should be locked
|
||||||
_, err = callWithError("personal_sign", []interface{}{hexutil.Bytes{0x88}, addr, ""})
|
_, err = CallWithError("personal_sign", []interface{}{hexutil.Bytes{0x88}, addr, ""})
|
||||||
require.Error(t, err)
|
require.Error(t, err)
|
||||||
|
|
||||||
rpcRes = call(t, "personal_unlockAccount", []interface{}{addr, ""})
|
rpcRes = Call(t, "personal_unlockAccount", []interface{}{addr, ""})
|
||||||
var unlocked bool
|
var unlocked bool
|
||||||
err = json.Unmarshal(rpcRes.Result, &unlocked)
|
err = json.Unmarshal(rpcRes.Result, &unlocked)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.True(t, unlocked)
|
require.True(t, unlocked)
|
||||||
|
|
||||||
// try to sign, should work now
|
// try to sign, should work now
|
||||||
rpcRes = call(t, "personal_sign", []interface{}{hexutil.Bytes{0x88}, addr, pswd})
|
rpcRes = Call(t, "personal_sign", []interface{}{hexutil.Bytes{0x88}, addr, pswd})
|
||||||
var res hexutil.Bytes
|
var res hexutil.Bytes
|
||||||
err = json.Unmarshal(rpcRes.Result, &res)
|
err = json.Unmarshal(rpcRes.Result, &res)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
@ -104,24 +104,24 @@ func TestPersonal_UnlockAccount(t *testing.T) {
|
|||||||
|
|
||||||
func TestPersonal_LockAccount(t *testing.T) {
|
func TestPersonal_LockAccount(t *testing.T) {
|
||||||
pswd := "nootwashere"
|
pswd := "nootwashere"
|
||||||
rpcRes := call(t, "personal_newAccount", []string{pswd})
|
rpcRes := Call(t, "personal_newAccount", []string{pswd})
|
||||||
var addr common.Address
|
var addr common.Address
|
||||||
err := json.Unmarshal(rpcRes.Result, &addr)
|
err := json.Unmarshal(rpcRes.Result, &addr)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
rpcRes = call(t, "personal_unlockAccount", []interface{}{addr, ""})
|
rpcRes = Call(t, "personal_unlockAccount", []interface{}{addr, ""})
|
||||||
var unlocked bool
|
var unlocked bool
|
||||||
err = json.Unmarshal(rpcRes.Result, &unlocked)
|
err = json.Unmarshal(rpcRes.Result, &unlocked)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.True(t, unlocked)
|
require.True(t, unlocked)
|
||||||
|
|
||||||
rpcRes = call(t, "personal_lockAccount", []interface{}{addr})
|
rpcRes = Call(t, "personal_lockAccount", []interface{}{addr})
|
||||||
var locked bool
|
var locked bool
|
||||||
err = json.Unmarshal(rpcRes.Result, &locked)
|
err = json.Unmarshal(rpcRes.Result, &locked)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.True(t, locked)
|
require.True(t, locked)
|
||||||
|
|
||||||
// try to sign, should be locked
|
// try to sign, should be locked
|
||||||
_, err = callWithError("personal_sign", []interface{}{hexutil.Bytes{0x88}, addr, ""})
|
_, err = CallWithError("personal_sign", []interface{}{hexutil.Bytes{0x88}, addr, ""})
|
||||||
require.Error(t, err)
|
require.Error(t, err)
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,6 @@ package tests
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/hex"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/big"
|
"math/big"
|
||||||
@ -34,43 +33,18 @@ const (
|
|||||||
|
|
||||||
var (
|
var (
|
||||||
MODE = os.Getenv("MODE")
|
MODE = os.Getenv("MODE")
|
||||||
HOST = os.Getenv("HOST")
|
|
||||||
|
|
||||||
zeroString = "0x0"
|
|
||||||
from = []byte{}
|
from = []byte{}
|
||||||
|
zeroString = "0x0"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Request struct {
|
|
||||||
Version string `json:"jsonrpc"`
|
|
||||||
Method string `json:"method"`
|
|
||||||
Params interface{} `json:"params"`
|
|
||||||
ID int `json:"id"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type RPCError struct {
|
|
||||||
Code int `json:"code"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
Data interface{} `json:"data,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Response struct {
|
|
||||||
Error *RPCError `json:"error"`
|
|
||||||
ID int `json:"id"`
|
|
||||||
Result json.RawMessage `json:"result,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
if MODE != "rpc" {
|
if MODE != "rpc" {
|
||||||
_, _ = fmt.Fprintln(os.Stdout, "Skipping RPC test")
|
_, _ = fmt.Fprintln(os.Stdout, "Skipping RPC test")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if HOST == "" {
|
|
||||||
HOST = "http://localhost:8545"
|
|
||||||
}
|
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
from, err = getAddress()
|
from, err = GetAddress()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("failed to get account: %s\n", err)
|
fmt.Printf("failed to get account: %s\n", err)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
@ -81,106 +55,19 @@ func TestMain(m *testing.M) {
|
|||||||
os.Exit(code)
|
os.Exit(code)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getAddress() ([]byte, error) {
|
|
||||||
rpcRes, err := callWithError("eth_accounts", []string{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var res []hexutil.Bytes
|
|
||||||
err = json.Unmarshal(rpcRes.Result, &res)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return res[0], nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func createRequest(method string, params interface{}) Request {
|
|
||||||
return Request{
|
|
||||||
Version: "2.0",
|
|
||||||
Method: method,
|
|
||||||
Params: params,
|
|
||||||
ID: 1,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func call(t *testing.T, method string, params interface{}) *Response {
|
|
||||||
req, err := json.Marshal(createRequest(method, params))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
var rpcRes *Response
|
|
||||||
time.Sleep(1 * time.Second)
|
|
||||||
/* #nosec */
|
|
||||||
res, err := http.Post(HOST, "application/json", bytes.NewBuffer(req))
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
decoder := json.NewDecoder(res.Body)
|
|
||||||
rpcRes = new(Response)
|
|
||||||
err = decoder.Decode(&rpcRes)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
err = res.Body.Close()
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.Nil(t, rpcRes.Error)
|
|
||||||
|
|
||||||
return rpcRes
|
|
||||||
}
|
|
||||||
|
|
||||||
func callWithError(method string, params interface{}) (*Response, error) {
|
|
||||||
req, err := json.Marshal(createRequest(method, params))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var rpcRes *Response
|
|
||||||
time.Sleep(1 * time.Second)
|
|
||||||
/* #nosec */
|
|
||||||
res, err := http.Post(HOST, "application/json", bytes.NewBuffer(req))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
decoder := json.NewDecoder(res.Body)
|
|
||||||
rpcRes = new(Response)
|
|
||||||
err = decoder.Decode(&rpcRes)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = res.Body.Close()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if rpcRes.Error != nil {
|
|
||||||
return nil, fmt.Errorf(rpcRes.Error.Message)
|
|
||||||
}
|
|
||||||
|
|
||||||
return rpcRes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// turns a 0x prefixed hex string to a big.Int
|
|
||||||
func hexToBigInt(t *testing.T, in string) *big.Int {
|
|
||||||
s := in[2:]
|
|
||||||
b, err := hex.DecodeString(s)
|
|
||||||
require.NoError(t, err)
|
|
||||||
return big.NewInt(0).SetBytes(b)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestBlockBloom(t *testing.T) {
|
func TestBlockBloom(t *testing.T) {
|
||||||
hash := deployTestContractWithFunction(t)
|
hash := DeployTestContractWithFunction(t, from)
|
||||||
receipt := waitForReceipt(t, hash)
|
receipt := WaitForReceipt(t, hash)
|
||||||
|
|
||||||
number := receipt["blockNumber"].(string)
|
number := receipt["blockNumber"].(string)
|
||||||
param := []interface{}{number, false}
|
param := []interface{}{number, false}
|
||||||
rpcRes := call(t, "eth_getBlockByNumber", param)
|
rpcRes := Call(t, "eth_getBlockByNumber", param)
|
||||||
|
|
||||||
block := make(map[string]interface{})
|
block := make(map[string]interface{})
|
||||||
err := json.Unmarshal(rpcRes.Result, &block)
|
err := json.Unmarshal(rpcRes.Result, &block)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
lb := hexToBigInt(t, block["logsBloom"].(string))
|
lb := HexToBigInt(t, block["logsBloom"].(string))
|
||||||
require.NotEqual(t, big.NewInt(0), lb)
|
require.NotEqual(t, big.NewInt(0), lb)
|
||||||
require.Equal(t, hash.String(), block["transactions"].([]interface{})[0])
|
require.Equal(t, hash.String(), block["transactions"].([]interface{})[0])
|
||||||
}
|
}
|
||||||
@ -189,7 +76,7 @@ func TestEth_GetLogs_NoLogs(t *testing.T) {
|
|||||||
param := make([]map[string][]string, 1)
|
param := make([]map[string][]string, 1)
|
||||||
param[0] = make(map[string][]string)
|
param[0] = make(map[string][]string)
|
||||||
param[0]["topics"] = []string{}
|
param[0]["topics"] = []string{}
|
||||||
rpcRes := call(t, "eth_getLogs", param)
|
rpcRes := Call(t, "eth_getLogs", param)
|
||||||
require.NotNil(t, rpcRes)
|
require.NotNil(t, rpcRes)
|
||||||
require.Nil(t, rpcRes.Error)
|
require.Nil(t, rpcRes.Error)
|
||||||
|
|
||||||
@ -205,7 +92,7 @@ func TestEth_GetLogs_Topics_AB(t *testing.T) {
|
|||||||
t.Skip("skipping TestEth_GetLogs_Topics_AB")
|
t.Skip("skipping TestEth_GetLogs_Topics_AB")
|
||||||
}
|
}
|
||||||
|
|
||||||
rpcRes := call(t, "eth_blockNumber", []string{})
|
rpcRes := Call(t, "eth_blockNumber", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint64
|
var res hexutil.Uint64
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -216,10 +103,10 @@ func TestEth_GetLogs_Topics_AB(t *testing.T) {
|
|||||||
param[0]["topics"] = []string{helloTopic, worldTopic}
|
param[0]["topics"] = []string{helloTopic, worldTopic}
|
||||||
param[0]["fromBlock"] = res.String()
|
param[0]["fromBlock"] = res.String()
|
||||||
|
|
||||||
hash := deployTestContractWithFunction(t)
|
hash := DeployTestContractWithFunction(t, from)
|
||||||
waitForReceipt(t, hash)
|
WaitForReceipt(t, hash)
|
||||||
|
|
||||||
rpcRes = call(t, "eth_getLogs", param)
|
rpcRes = Call(t, "eth_getLogs", param)
|
||||||
|
|
||||||
var logs []*ethtypes.Log
|
var logs []*ethtypes.Log
|
||||||
err = json.Unmarshal(rpcRes.Result, &logs)
|
err = json.Unmarshal(rpcRes.Result, &logs)
|
||||||
@ -234,9 +121,9 @@ func TestEth_GetTransactionCount(t *testing.T) {
|
|||||||
t.Skip("skipping TestEth_GetTransactionCount")
|
t.Skip("skipping TestEth_GetTransactionCount")
|
||||||
}
|
}
|
||||||
|
|
||||||
prev := getNonce(t)
|
prev := GetNonce(t, "latest")
|
||||||
sendTestTransaction(t)
|
SendTestTransaction(t, from)
|
||||||
post := getNonce(t)
|
post := GetNonce(t, "latest")
|
||||||
require.Equal(t, prev, post-1)
|
require.Equal(t, prev, post-1)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -246,10 +133,10 @@ func TestEth_GetTransactionLogs(t *testing.T) {
|
|||||||
t.Skip("skipping TestEth_GetTransactionLogs")
|
t.Skip("skipping TestEth_GetTransactionLogs")
|
||||||
}
|
}
|
||||||
|
|
||||||
hash, _ := deployTestContract(t)
|
hash, _ := DeployTestContract(t, from)
|
||||||
|
|
||||||
param := []string{hash.String()}
|
param := []string{hash.String()}
|
||||||
rpcRes := call(t, "eth_getTransactionLogs", param)
|
rpcRes := Call(t, "eth_getTransactionLogs", param)
|
||||||
|
|
||||||
logs := new([]*ethtypes.Log)
|
logs := new([]*ethtypes.Log)
|
||||||
err := json.Unmarshal(rpcRes.Result, logs)
|
err := json.Unmarshal(rpcRes.Result, logs)
|
||||||
@ -260,7 +147,7 @@ func TestEth_GetTransactionLogs(t *testing.T) {
|
|||||||
func TestEth_protocolVersion(t *testing.T) {
|
func TestEth_protocolVersion(t *testing.T) {
|
||||||
expectedRes := hexutil.Uint(ethermint.ProtocolVersion)
|
expectedRes := hexutil.Uint(ethermint.ProtocolVersion)
|
||||||
|
|
||||||
rpcRes := call(t, "eth_protocolVersion", []string{})
|
rpcRes := Call(t, "eth_protocolVersion", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint
|
var res hexutil.Uint
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -271,7 +158,7 @@ func TestEth_protocolVersion(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_chainId(t *testing.T) {
|
func TestEth_chainId(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_chainId", []string{})
|
rpcRes := Call(t, "eth_chainId", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint
|
var res hexutil.Uint
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -280,7 +167,7 @@ func TestEth_chainId(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_blockNumber(t *testing.T) {
|
func TestEth_blockNumber(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_blockNumber", []string{})
|
rpcRes := Call(t, "eth_blockNumber", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint64
|
var res hexutil.Uint64
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -291,7 +178,7 @@ func TestEth_blockNumber(t *testing.T) {
|
|||||||
|
|
||||||
func TestEth_coinbase(t *testing.T) {
|
func TestEth_coinbase(t *testing.T) {
|
||||||
zeroAddress := hexutil.Bytes(ethcmn.Address{}.Bytes())
|
zeroAddress := hexutil.Bytes(ethcmn.Address{}.Bytes())
|
||||||
rpcRes := call(t, "eth_coinbase", []string{})
|
rpcRes := Call(t, "eth_coinbase", []string{})
|
||||||
|
|
||||||
var res hexutil.Bytes
|
var res hexutil.Bytes
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -302,7 +189,7 @@ func TestEth_coinbase(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_GetBalance(t *testing.T) {
|
func TestEth_GetBalance(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_getBalance", []string{addrA, zeroString})
|
rpcRes := Call(t, "eth_getBalance", []string{addrA, zeroString})
|
||||||
|
|
||||||
var res hexutil.Big
|
var res hexutil.Big
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -318,7 +205,7 @@ func TestEth_GetBalance(t *testing.T) {
|
|||||||
|
|
||||||
func TestEth_GetStorageAt(t *testing.T) {
|
func TestEth_GetStorageAt(t *testing.T) {
|
||||||
expectedRes := hexutil.Bytes{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
|
expectedRes := hexutil.Bytes{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
|
||||||
rpcRes := call(t, "eth_getStorageAt", []string{addrA, fmt.Sprint(addrAStoreKey), zeroString})
|
rpcRes := Call(t, "eth_getStorageAt", []string{addrA, fmt.Sprint(addrAStoreKey), zeroString})
|
||||||
|
|
||||||
var storage hexutil.Bytes
|
var storage hexutil.Bytes
|
||||||
err := storage.UnmarshalJSON(rpcRes.Result)
|
err := storage.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -334,7 +221,7 @@ func TestEth_GetProof(t *testing.T) {
|
|||||||
params[0] = addrA
|
params[0] = addrA
|
||||||
params[1] = []string{fmt.Sprint(addrAStoreKey)}
|
params[1] = []string{fmt.Sprint(addrAStoreKey)}
|
||||||
params[2] = "latest"
|
params[2] = "latest"
|
||||||
rpcRes := call(t, "eth_getProof", params)
|
rpcRes := Call(t, "eth_getProof", params)
|
||||||
require.NotNil(t, rpcRes)
|
require.NotNil(t, rpcRes)
|
||||||
|
|
||||||
var accRes rpctypes.AccountResult
|
var accRes rpctypes.AccountResult
|
||||||
@ -348,7 +235,7 @@ func TestEth_GetProof(t *testing.T) {
|
|||||||
|
|
||||||
func TestEth_GetCode(t *testing.T) {
|
func TestEth_GetCode(t *testing.T) {
|
||||||
expectedRes := hexutil.Bytes{}
|
expectedRes := hexutil.Bytes{}
|
||||||
rpcRes := call(t, "eth_getCode", []string{addrA, zeroString})
|
rpcRes := Call(t, "eth_getCode", []string{addrA, zeroString})
|
||||||
|
|
||||||
var code hexutil.Bytes
|
var code hexutil.Bytes
|
||||||
err := code.UnmarshalJSON(rpcRes.Result)
|
err := code.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -368,13 +255,13 @@ func TestEth_SendTransaction_Transfer(t *testing.T) {
|
|||||||
param[0]["gasLimit"] = "0x5208"
|
param[0]["gasLimit"] = "0x5208"
|
||||||
param[0]["gasPrice"] = "0x55ae82600"
|
param[0]["gasPrice"] = "0x55ae82600"
|
||||||
|
|
||||||
rpcRes := call(t, "eth_sendTransaction", param)
|
rpcRes := Call(t, "eth_sendTransaction", param)
|
||||||
|
|
||||||
var hash hexutil.Bytes
|
var hash hexutil.Bytes
|
||||||
err := json.Unmarshal(rpcRes.Result, &hash)
|
err := json.Unmarshal(rpcRes.Result, &hash)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
receipt := waitForReceipt(t, hash)
|
receipt := WaitForReceipt(t, hash)
|
||||||
require.NotNil(t, receipt)
|
require.NotNil(t, receipt)
|
||||||
require.Equal(t, "0x1", receipt["status"].(string))
|
require.Equal(t, "0x1", receipt["status"].(string))
|
||||||
}
|
}
|
||||||
@ -385,7 +272,7 @@ func TestEth_SendTransaction_ContractDeploy(t *testing.T) {
|
|||||||
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
param[0]["data"] = "0x6080604052348015600f57600080fd5b5060117f775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73889860405160405180910390a2603580604b6000396000f3fe6080604052600080fdfea165627a7a723058206cab665f0f557620554bb45adf266708d2bd349b8a4314bdff205ee8440e3c240029"
|
param[0]["data"] = "0x6080604052348015600f57600080fd5b5060117f775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73889860405160405180910390a2603580604b6000396000f3fe6080604052600080fdfea165627a7a723058206cab665f0f557620554bb45adf266708d2bd349b8a4314bdff205ee8440e3c240029"
|
||||||
|
|
||||||
rpcRes := call(t, "eth_sendTransaction", param)
|
rpcRes := Call(t, "eth_sendTransaction", param)
|
||||||
|
|
||||||
var hash hexutil.Bytes
|
var hash hexutil.Bytes
|
||||||
err := json.Unmarshal(rpcRes.Result, &hash)
|
err := json.Unmarshal(rpcRes.Result, &hash)
|
||||||
@ -396,7 +283,7 @@ func TestEth_NewFilter(t *testing.T) {
|
|||||||
param := make([]map[string][]string, 1)
|
param := make([]map[string][]string, 1)
|
||||||
param[0] = make(map[string][]string)
|
param[0] = make(map[string][]string)
|
||||||
param[0]["topics"] = []string{"0x0000000000000000000000000000000000000000000000000000000012341234"}
|
param[0]["topics"] = []string{"0x0000000000000000000000000000000000000000000000000000000012341234"}
|
||||||
rpcRes := call(t, "eth_newFilter", param)
|
rpcRes := Call(t, "eth_newFilter", param)
|
||||||
|
|
||||||
var ID string
|
var ID string
|
||||||
err := json.Unmarshal(rpcRes.Result, &ID)
|
err := json.Unmarshal(rpcRes.Result, &ID)
|
||||||
@ -404,7 +291,7 @@ func TestEth_NewFilter(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_NewBlockFilter(t *testing.T) {
|
func TestEth_NewBlockFilter(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_newBlockFilter", []string{})
|
rpcRes := Call(t, "eth_newBlockFilter", []string{})
|
||||||
|
|
||||||
var ID string
|
var ID string
|
||||||
err := json.Unmarshal(rpcRes.Result, &ID)
|
err := json.Unmarshal(rpcRes.Result, &ID)
|
||||||
@ -412,7 +299,7 @@ func TestEth_NewBlockFilter(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_GetFilterChanges_BlockFilter(t *testing.T) {
|
func TestEth_GetFilterChanges_BlockFilter(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_newBlockFilter", []string{})
|
rpcRes := Call(t, "eth_newBlockFilter", []string{})
|
||||||
|
|
||||||
var ID string
|
var ID string
|
||||||
err := json.Unmarshal(rpcRes.Result, &ID)
|
err := json.Unmarshal(rpcRes.Result, &ID)
|
||||||
@ -420,7 +307,7 @@ func TestEth_GetFilterChanges_BlockFilter(t *testing.T) {
|
|||||||
|
|
||||||
time.Sleep(5 * time.Second)
|
time.Sleep(5 * time.Second)
|
||||||
|
|
||||||
changesRes := call(t, "eth_getFilterChanges", []string{ID})
|
changesRes := Call(t, "eth_getFilterChanges", []string{ID})
|
||||||
var hashes []ethcmn.Hash
|
var hashes []ethcmn.Hash
|
||||||
err = json.Unmarshal(changesRes.Result, &hashes)
|
err = json.Unmarshal(changesRes.Result, &hashes)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
@ -431,13 +318,13 @@ func TestEth_GetFilterChanges_NoLogs(t *testing.T) {
|
|||||||
param := make([]map[string][]string, 1)
|
param := make([]map[string][]string, 1)
|
||||||
param[0] = make(map[string][]string)
|
param[0] = make(map[string][]string)
|
||||||
param[0]["topics"] = []string{}
|
param[0]["topics"] = []string{}
|
||||||
rpcRes := call(t, "eth_newFilter", param)
|
rpcRes := Call(t, "eth_newFilter", param)
|
||||||
|
|
||||||
var ID string
|
var ID string
|
||||||
err := json.Unmarshal(rpcRes.Result, &ID)
|
err := json.Unmarshal(rpcRes.Result, &ID)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
changesRes := call(t, "eth_getFilterChanges", []string{ID})
|
changesRes := Call(t, "eth_getFilterChanges", []string{ID})
|
||||||
|
|
||||||
var logs []*ethtypes.Log
|
var logs []*ethtypes.Log
|
||||||
err = json.Unmarshal(changesRes.Result, &logs)
|
err = json.Unmarshal(changesRes.Result, &logs)
|
||||||
@ -445,7 +332,7 @@ func TestEth_GetFilterChanges_NoLogs(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_GetFilterChanges_WrongID(t *testing.T) {
|
func TestEth_GetFilterChanges_WrongID(t *testing.T) {
|
||||||
req, err := json.Marshal(createRequest("eth_getFilterChanges", []string{"0x1122334400000077"}))
|
req, err := json.Marshal(CreateRequest("eth_getFilterChanges", []string{"0x1122334400000077"}))
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
var rpcRes *Response
|
var rpcRes *Response
|
||||||
@ -464,28 +351,13 @@ func TestEth_GetFilterChanges_WrongID(t *testing.T) {
|
|||||||
require.NotNil(t, "invalid filter ID", rpcRes.Error.Message)
|
require.NotNil(t, "invalid filter ID", rpcRes.Error.Message)
|
||||||
}
|
}
|
||||||
|
|
||||||
// sendTestTransaction sends a dummy transaction
|
|
||||||
func sendTestTransaction(t *testing.T) hexutil.Bytes {
|
|
||||||
param := make([]map[string]string, 1)
|
|
||||||
param[0] = make(map[string]string)
|
|
||||||
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
|
||||||
param[0]["to"] = "0x1122334455667788990011223344556677889900"
|
|
||||||
param[0]["value"] = "0x1"
|
|
||||||
rpcRes := call(t, "eth_sendTransaction", param)
|
|
||||||
|
|
||||||
var hash hexutil.Bytes
|
|
||||||
err := json.Unmarshal(rpcRes.Result, &hash)
|
|
||||||
require.NoError(t, err)
|
|
||||||
return hash
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestEth_GetTransactionReceipt(t *testing.T) {
|
func TestEth_GetTransactionReceipt(t *testing.T) {
|
||||||
hash := sendTestTransaction(t)
|
hash := SendTestTransaction(t, from)
|
||||||
|
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
|
|
||||||
param := []string{hash.String()}
|
param := []string{hash.String()}
|
||||||
rpcRes := call(t, "eth_getTransactionReceipt", param)
|
rpcRes := Call(t, "eth_getTransactionReceipt", param)
|
||||||
require.Nil(t, rpcRes.Error)
|
require.Nil(t, rpcRes.Error)
|
||||||
|
|
||||||
receipt := make(map[string]interface{})
|
receipt := make(map[string]interface{})
|
||||||
@ -496,34 +368,13 @@ func TestEth_GetTransactionReceipt(t *testing.T) {
|
|||||||
require.Equal(t, []interface{}{}, receipt["logs"].([]interface{}))
|
require.Equal(t, []interface{}{}, receipt["logs"].([]interface{}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// deployTestContract deploys a contract that emits an event in the constructor
|
|
||||||
func deployTestContract(t *testing.T) (hexutil.Bytes, map[string]interface{}) {
|
|
||||||
param := make([]map[string]string, 1)
|
|
||||||
param[0] = make(map[string]string)
|
|
||||||
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
|
||||||
param[0]["data"] = "0x6080604052348015600f57600080fd5b5060117f775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73889860405160405180910390a2603580604b6000396000f3fe6080604052600080fdfea165627a7a723058206cab665f0f557620554bb45adf266708d2bd349b8a4314bdff205ee8440e3c240029"
|
|
||||||
param[0]["gas"] = "0x200000"
|
|
||||||
|
|
||||||
rpcRes := call(t, "eth_sendTransaction", param)
|
|
||||||
|
|
||||||
var hash hexutil.Bytes
|
|
||||||
err := json.Unmarshal(rpcRes.Result, &hash)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
receipt := waitForReceipt(t, hash)
|
|
||||||
require.NotNil(t, receipt, "transaction failed")
|
|
||||||
require.Equal(t, "0x1", receipt["status"].(string))
|
|
||||||
|
|
||||||
return hash, receipt
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestEth_GetTransactionReceipt_ContractDeployment(t *testing.T) {
|
func TestEth_GetTransactionReceipt_ContractDeployment(t *testing.T) {
|
||||||
hash, _ := deployTestContract(t)
|
hash, _ := DeployTestContract(t, from)
|
||||||
|
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
|
|
||||||
param := []string{hash.String()}
|
param := []string{hash.String()}
|
||||||
rpcRes := call(t, "eth_getTransactionReceipt", param)
|
rpcRes := Call(t, "eth_getTransactionReceipt", param)
|
||||||
|
|
||||||
receipt := make(map[string]interface{})
|
receipt := make(map[string]interface{})
|
||||||
err := json.Unmarshal(rpcRes.Result, &receipt)
|
err := json.Unmarshal(rpcRes.Result, &receipt)
|
||||||
@ -535,32 +386,8 @@ func TestEth_GetTransactionReceipt_ContractDeployment(t *testing.T) {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func getTransactionReceipt(t *testing.T, hash hexutil.Bytes) map[string]interface{} {
|
|
||||||
param := []string{hash.String()}
|
|
||||||
rpcRes := call(t, "eth_getTransactionReceipt", param)
|
|
||||||
|
|
||||||
receipt := make(map[string]interface{})
|
|
||||||
err := json.Unmarshal(rpcRes.Result, &receipt)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
return receipt
|
|
||||||
}
|
|
||||||
|
|
||||||
func waitForReceipt(t *testing.T, hash hexutil.Bytes) map[string]interface{} {
|
|
||||||
for i := 0; i < 12; i++ {
|
|
||||||
receipt := getTransactionReceipt(t, hash)
|
|
||||||
if receipt != nil {
|
|
||||||
return receipt
|
|
||||||
}
|
|
||||||
|
|
||||||
time.Sleep(time.Second)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestEth_GetFilterChanges_NoTopics(t *testing.T) {
|
func TestEth_GetFilterChanges_NoTopics(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_blockNumber", []string{})
|
rpcRes := Call(t, "eth_blockNumber", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint64
|
var res hexutil.Uint64
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -572,17 +399,17 @@ func TestEth_GetFilterChanges_NoTopics(t *testing.T) {
|
|||||||
param[0]["fromBlock"] = res.String()
|
param[0]["fromBlock"] = res.String()
|
||||||
|
|
||||||
// instantiate new filter
|
// instantiate new filter
|
||||||
rpcRes = call(t, "eth_newFilter", param)
|
rpcRes = Call(t, "eth_newFilter", param)
|
||||||
require.Nil(t, rpcRes.Error)
|
require.Nil(t, rpcRes.Error)
|
||||||
var ID string
|
var ID string
|
||||||
err = json.Unmarshal(rpcRes.Result, &ID)
|
err = json.Unmarshal(rpcRes.Result, &ID)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
// deploy contract, emitting some event
|
// deploy contract, emitting some event
|
||||||
deployTestContract(t)
|
DeployTestContract(t, from)
|
||||||
|
|
||||||
// get filter changes
|
// get filter changes
|
||||||
changesRes := call(t, "eth_getFilterChanges", []string{ID})
|
changesRes := Call(t, "eth_getFilterChanges", []string{ID})
|
||||||
|
|
||||||
var logs []*ethtypes.Log
|
var logs []*ethtypes.Log
|
||||||
err = json.Unmarshal(changesRes.Result, &logs)
|
err = json.Unmarshal(changesRes.Result, &logs)
|
||||||
@ -606,51 +433,11 @@ var helloTopic = "0x775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73
|
|||||||
// world parameter in Hello event
|
// world parameter in Hello event
|
||||||
var worldTopic = "0x0000000000000000000000000000000000000000000000000000000000000011"
|
var worldTopic = "0x0000000000000000000000000000000000000000000000000000000000000011"
|
||||||
|
|
||||||
func deployTestContractWithFunction(t *testing.T) hexutil.Bytes {
|
|
||||||
// pragma solidity ^0.5.1;
|
|
||||||
|
|
||||||
// contract Test {
|
|
||||||
// event Hello(uint256 indexed world);
|
|
||||||
// event TestEvent(uint256 indexed a, uint256 indexed b);
|
|
||||||
|
|
||||||
// uint256 myStorage;
|
|
||||||
|
|
||||||
// constructor() public {
|
|
||||||
// emit Hello(17);
|
|
||||||
// }
|
|
||||||
|
|
||||||
// function test(uint256 a, uint256 b) public {
|
|
||||||
// myStorage = a;
|
|
||||||
// emit TestEvent(a, b);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
bytecode := "0x608060405234801561001057600080fd5b5060117f775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73889860405160405180910390a260d08061004d6000396000f3fe6080604052348015600f57600080fd5b506004361060285760003560e01c8063eb8ac92114602d575b600080fd5b606060048036036040811015604157600080fd5b8101908080359060200190929190803590602001909291905050506062565b005b8160008190555080827ff3ca124a697ba07e8c5e80bebcfcc48991fc16a63170e8a9206e30508960d00360405160405180910390a3505056fea265627a7a723158201d94d2187aaf3a6790527b615fcc40970febf0385fa6d72a2344848ebd0df3e964736f6c63430005110032"
|
|
||||||
|
|
||||||
param := make([]map[string]string, 1)
|
|
||||||
param[0] = make(map[string]string)
|
|
||||||
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
|
||||||
param[0]["data"] = bytecode
|
|
||||||
param[0]["gas"] = "0x200000"
|
|
||||||
|
|
||||||
rpcRes := call(t, "eth_sendTransaction", param)
|
|
||||||
|
|
||||||
var hash hexutil.Bytes
|
|
||||||
err := json.Unmarshal(rpcRes.Result, &hash)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
receipt := waitForReceipt(t, hash)
|
|
||||||
require.NotNil(t, receipt, "transaction failed")
|
|
||||||
require.Equal(t, "0x1", receipt["status"].(string))
|
|
||||||
|
|
||||||
return hash
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tests topics case where there are topics in first two positions
|
// Tests topics case where there are topics in first two positions
|
||||||
func TestEth_GetFilterChanges_Topics_AB(t *testing.T) {
|
func TestEth_GetFilterChanges_Topics_AB(t *testing.T) {
|
||||||
time.Sleep(time.Second)
|
time.Sleep(time.Second)
|
||||||
|
|
||||||
rpcRes := call(t, "eth_blockNumber", []string{})
|
rpcRes := Call(t, "eth_blockNumber", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint64
|
var res hexutil.Uint64
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -662,15 +449,15 @@ func TestEth_GetFilterChanges_Topics_AB(t *testing.T) {
|
|||||||
param[0]["fromBlock"] = res.String()
|
param[0]["fromBlock"] = res.String()
|
||||||
|
|
||||||
// instantiate new filter
|
// instantiate new filter
|
||||||
rpcRes = call(t, "eth_newFilter", param)
|
rpcRes = Call(t, "eth_newFilter", param)
|
||||||
var ID string
|
var ID string
|
||||||
err = json.Unmarshal(rpcRes.Result, &ID)
|
err = json.Unmarshal(rpcRes.Result, &ID)
|
||||||
require.NoError(t, err, string(rpcRes.Result))
|
require.NoError(t, err, string(rpcRes.Result))
|
||||||
|
|
||||||
deployTestContractWithFunction(t)
|
DeployTestContractWithFunction(t, from)
|
||||||
|
|
||||||
// get filter changes
|
// get filter changes
|
||||||
changesRes := call(t, "eth_getFilterChanges", []string{ID})
|
changesRes := Call(t, "eth_getFilterChanges", []string{ID})
|
||||||
|
|
||||||
var logs []*ethtypes.Log
|
var logs []*ethtypes.Log
|
||||||
err = json.Unmarshal(changesRes.Result, &logs)
|
err = json.Unmarshal(changesRes.Result, &logs)
|
||||||
@ -680,7 +467,7 @@ func TestEth_GetFilterChanges_Topics_AB(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_GetFilterChanges_Topics_XB(t *testing.T) {
|
func TestEth_GetFilterChanges_Topics_XB(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_blockNumber", []string{})
|
rpcRes := Call(t, "eth_blockNumber", []string{})
|
||||||
|
|
||||||
var res hexutil.Uint64
|
var res hexutil.Uint64
|
||||||
err := res.UnmarshalJSON(rpcRes.Result)
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
@ -692,15 +479,15 @@ func TestEth_GetFilterChanges_Topics_XB(t *testing.T) {
|
|||||||
param[0]["fromBlock"] = res.String()
|
param[0]["fromBlock"] = res.String()
|
||||||
|
|
||||||
// instantiate new filter
|
// instantiate new filter
|
||||||
rpcRes = call(t, "eth_newFilter", param)
|
rpcRes = Call(t, "eth_newFilter", param)
|
||||||
var ID string
|
var ID string
|
||||||
err = json.Unmarshal(rpcRes.Result, &ID)
|
err = json.Unmarshal(rpcRes.Result, &ID)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
deployTestContractWithFunction(t)
|
DeployTestContractWithFunction(t, from)
|
||||||
|
|
||||||
// get filter changes
|
// get filter changes
|
||||||
changesRes := call(t, "eth_getFilterChanges", []string{ID})
|
changesRes := Call(t, "eth_getFilterChanges", []string{ID})
|
||||||
|
|
||||||
var logs []*ethtypes.Log
|
var logs []*ethtypes.Log
|
||||||
err = json.Unmarshal(changesRes.Result, &logs)
|
err = json.Unmarshal(changesRes.Result, &logs)
|
||||||
@ -715,20 +502,20 @@ func TestEth_GetFilterChanges_Topics_XXC(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEth_PendingTransactionFilter(t *testing.T) {
|
func TestEth_PendingTransactionFilter(t *testing.T) {
|
||||||
rpcRes := call(t, "eth_newPendingTransactionFilter", []string{})
|
rpcRes := Call(t, "eth_newPendingTransactionFilter", []string{})
|
||||||
|
|
||||||
var ID string
|
var ID string
|
||||||
err := json.Unmarshal(rpcRes.Result, &ID)
|
err := json.Unmarshal(rpcRes.Result, &ID)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
for i := 0; i < 5; i++ {
|
for i := 0; i < 5; i++ {
|
||||||
deployTestContractWithFunction(t)
|
DeployTestContractWithFunction(t, from)
|
||||||
}
|
}
|
||||||
|
|
||||||
time.Sleep(10 * time.Second)
|
time.Sleep(10 * time.Second)
|
||||||
|
|
||||||
// get filter changes
|
// get filter changes
|
||||||
changesRes := call(t, "eth_getFilterChanges", []string{ID})
|
changesRes := Call(t, "eth_getFilterChanges", []string{ID})
|
||||||
require.NotNil(t, changesRes)
|
require.NotNil(t, changesRes)
|
||||||
|
|
||||||
var txs []*hexutil.Bytes
|
var txs []*hexutil.Bytes
|
||||||
@ -738,23 +525,13 @@ func TestEth_PendingTransactionFilter(t *testing.T) {
|
|||||||
require.True(t, len(txs) >= 2, "could not get any txs", "changesRes.Result", string(changesRes.Result))
|
require.True(t, len(txs) >= 2, "could not get any txs", "changesRes.Result", string(changesRes.Result))
|
||||||
}
|
}
|
||||||
|
|
||||||
func getNonce(t *testing.T) hexutil.Uint64 {
|
|
||||||
param := []interface{}{hexutil.Bytes(from), "latest"}
|
|
||||||
rpcRes := call(t, "eth_getTransactionCount", param)
|
|
||||||
|
|
||||||
var nonce hexutil.Uint64
|
|
||||||
err := json.Unmarshal(rpcRes.Result, &nonce)
|
|
||||||
require.NoError(t, err)
|
|
||||||
return nonce
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestEth_EstimateGas(t *testing.T) {
|
func TestEth_EstimateGas(t *testing.T) {
|
||||||
param := make([]map[string]string, 1)
|
param := make([]map[string]string, 1)
|
||||||
param[0] = make(map[string]string)
|
param[0] = make(map[string]string)
|
||||||
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
param[0]["to"] = "0x1122334455667788990011223344556677889900"
|
param[0]["to"] = "0x1122334455667788990011223344556677889900"
|
||||||
param[0]["value"] = "0x1"
|
param[0]["value"] = "0x1"
|
||||||
rpcRes := call(t, "eth_estimateGas", param)
|
rpcRes := Call(t, "eth_estimateGas", param)
|
||||||
require.NotNil(t, rpcRes)
|
require.NotNil(t, rpcRes)
|
||||||
require.NotEmpty(t, rpcRes.Result)
|
require.NotEmpty(t, rpcRes.Result)
|
||||||
|
|
||||||
@ -773,7 +550,7 @@ func TestEth_EstimateGas_ContractDeployment(t *testing.T) {
|
|||||||
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
param[0]["data"] = bytecode
|
param[0]["data"] = bytecode
|
||||||
|
|
||||||
rpcRes := call(t, "eth_estimateGas", param)
|
rpcRes := Call(t, "eth_estimateGas", param)
|
||||||
require.NotNil(t, rpcRes)
|
require.NotNil(t, rpcRes)
|
||||||
require.NotEmpty(t, rpcRes.Result)
|
require.NotEmpty(t, rpcRes.Result)
|
||||||
|
|
||||||
@ -786,7 +563,7 @@ func TestEth_EstimateGas_ContractDeployment(t *testing.T) {
|
|||||||
|
|
||||||
func TestEth_GetBlockByNumber(t *testing.T) {
|
func TestEth_GetBlockByNumber(t *testing.T) {
|
||||||
param := []interface{}{"0x1", false}
|
param := []interface{}{"0x1", false}
|
||||||
rpcRes := call(t, "eth_getBlockByNumber", param)
|
rpcRes := Call(t, "eth_getBlockByNumber", param)
|
||||||
|
|
||||||
block := make(map[string]interface{})
|
block := make(map[string]interface{})
|
||||||
err := json.Unmarshal(rpcRes.Result, &block)
|
err := json.Unmarshal(rpcRes.Result, &block)
|
||||||
|
323
tests/tests-pending/rpc_pending_test.go
Normal file
323
tests/tests-pending/rpc_pending_test.go
Normal file
@ -0,0 +1,323 @@
|
|||||||
|
// This is a test utility for Ethermint's Web3 JSON-RPC services.
|
||||||
|
//
|
||||||
|
// To run these tests please first ensure you have the ethermintd running
|
||||||
|
// and have started the RPC service with `ethermintcli rest-server`.
|
||||||
|
//
|
||||||
|
// You can configure the desired HOST and MODE as well
|
||||||
|
package pending
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"math/big"
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
||||||
|
|
||||||
|
rpctypes "github.com/cosmos/ethermint/rpc/types"
|
||||||
|
util "github.com/cosmos/ethermint/tests"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
addrA = "0xc94770007dda54cF92009BFF0dE90c06F603a09f"
|
||||||
|
addrAStoreKey = 0
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
MODE = os.Getenv("MODE")
|
||||||
|
from = []byte{}
|
||||||
|
)
|
||||||
|
|
||||||
|
type Request struct {
|
||||||
|
Version string `json:"jsonrpc"`
|
||||||
|
Method string `json:"method"`
|
||||||
|
Params interface{} `json:"params"`
|
||||||
|
ID int `json:"id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RPCError struct {
|
||||||
|
Code int `json:"code"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Data interface{} `json:"data,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Response struct {
|
||||||
|
Error *RPCError `json:"error"`
|
||||||
|
ID int `json:"id"`
|
||||||
|
Result json.RawMessage `json:"result,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMain(m *testing.M) {
|
||||||
|
if MODE != "pending" {
|
||||||
|
_, _ = fmt.Fprintln(os.Stdout, "Skipping pending RPC test")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
from, err = util.GetAddress()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("failed to get account: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start all tests
|
||||||
|
code := m.Run()
|
||||||
|
os.Exit(code)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_GetBalance(t *testing.T) {
|
||||||
|
var res hexutil.Big
|
||||||
|
rpcRes := util.Call(t, "eth_getBalance", []string{addrA, "latest"})
|
||||||
|
err := res.UnmarshalJSON(rpcRes.Result)
|
||||||
|
require.NoError(t, err)
|
||||||
|
preTxLatestBalance := res.ToInt()
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBalance", []string{addrA, "pending"})
|
||||||
|
err = res.UnmarshalJSON(rpcRes.Result)
|
||||||
|
require.NoError(t, err)
|
||||||
|
preTxPendingBalance := res.ToInt()
|
||||||
|
|
||||||
|
t.Logf("Got pending balance %s for %s pre tx\n", preTxPendingBalance, addrA)
|
||||||
|
t.Logf("Got latest balance %s for %s pre tx\n", preTxLatestBalance, addrA)
|
||||||
|
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, rpcRes.Error)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBalance", []string{addrA, "pending"})
|
||||||
|
err = res.UnmarshalJSON(rpcRes.Result)
|
||||||
|
require.NoError(t, err)
|
||||||
|
postTxPendingBalance := res.ToInt()
|
||||||
|
t.Logf("Got pending balance %s for %s post tx\n", postTxPendingBalance, addrA)
|
||||||
|
|
||||||
|
require.Equal(t, preTxPendingBalance.Add(preTxPendingBalance, big.NewInt(10)), postTxPendingBalance)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBalance", []string{addrA, "latest"})
|
||||||
|
err = res.UnmarshalJSON(rpcRes.Result)
|
||||||
|
require.NoError(t, err)
|
||||||
|
postTxLatestBalance := res.ToInt()
|
||||||
|
t.Logf("Got latest balance %s for %s post tx\n", postTxLatestBalance, addrA)
|
||||||
|
|
||||||
|
require.Equal(t, preTxLatestBalance, postTxLatestBalance)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_GetTransactionCount(t *testing.T) {
|
||||||
|
prePendingNonce := util.GetNonce(t, "pending")
|
||||||
|
t.Logf("Pending nonce before tx is %d", prePendingNonce)
|
||||||
|
|
||||||
|
currentNonce := util.GetNonce(t, "latest")
|
||||||
|
t.Logf("Current nonce is %d", currentNonce)
|
||||||
|
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
|
||||||
|
txRes := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes.Error)
|
||||||
|
|
||||||
|
pendingNonce := util.GetNonce(t, "pending")
|
||||||
|
latestNonce := util.GetNonce(t, "latest")
|
||||||
|
t.Logf("Latest nonce is %d", latestNonce)
|
||||||
|
require.Equal(t, currentNonce, latestNonce)
|
||||||
|
t.Logf("Pending nonce is %d", pendingNonce)
|
||||||
|
require.NotEqual(t, latestNonce, pendingNonce)
|
||||||
|
|
||||||
|
require.Greater(t, uint64(pendingNonce), uint64(latestNonce))
|
||||||
|
require.Equal(t, uint64(prePendingNonce)+uint64(1), uint64(pendingNonce))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_GetBlockTransactionCountByNumber(t *testing.T) {
|
||||||
|
rpcRes := util.Call(t, "eth_getBlockTransactionCountByNumber", []interface{}{"pending"})
|
||||||
|
var preTxPendingTxCount hexutil.Uint
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &preTxPendingTxCount)
|
||||||
|
require.NoError(t, err)
|
||||||
|
t.Logf("Pre tx pending nonce is %d", preTxPendingTxCount)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBlockTransactionCountByNumber", []interface{}{"latest"})
|
||||||
|
var preTxLatestTxCount hexutil.Uint
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &preTxLatestTxCount)
|
||||||
|
require.NoError(t, err)
|
||||||
|
t.Logf("Pre tx latest nonce is %d", preTxLatestTxCount)
|
||||||
|
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
|
||||||
|
txRes := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes.Error)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBlockTransactionCountByNumber", []interface{}{"pending"})
|
||||||
|
var postTxPendingTxCount hexutil.Uint
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &postTxPendingTxCount)
|
||||||
|
require.NoError(t, err)
|
||||||
|
t.Logf("Post tx pending nonce is %d", postTxPendingTxCount)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBlockTransactionCountByNumber", []interface{}{"latest"})
|
||||||
|
var postTxLatestTxCount hexutil.Uint
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &postTxLatestTxCount)
|
||||||
|
require.NoError(t, err)
|
||||||
|
t.Logf("Post tx latest nonce is %d", postTxLatestTxCount)
|
||||||
|
|
||||||
|
require.Equal(t, uint64(preTxPendingTxCount)+uint64(1), uint64(postTxPendingTxCount))
|
||||||
|
require.NotEqual(t, uint64(postTxPendingTxCount)-uint64(preTxPendingTxCount), uint64(postTxLatestTxCount)-uint64(preTxLatestTxCount))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_GetBlockByNumber(t *testing.T) {
|
||||||
|
rpcRes := util.Call(t, "eth_getBlockByNumber", []interface{}{"latest", true})
|
||||||
|
var preTxLatestBlock map[string]interface{}
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &preTxLatestBlock)
|
||||||
|
require.NoError(t, err)
|
||||||
|
preTxLatestTxs := len(preTxLatestBlock["transactions"].([]interface{}))
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBlockByNumber", []interface{}{"pending", true})
|
||||||
|
var preTxPendingBlock map[string]interface{}
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &preTxPendingBlock)
|
||||||
|
require.NoError(t, err)
|
||||||
|
preTxPendingTxs := len(preTxPendingBlock["transactions"].([]interface{}))
|
||||||
|
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
|
||||||
|
txRes := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes.Error)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBlockByNumber", []interface{}{"pending", true})
|
||||||
|
var postTxPendingBlock map[string]interface{}
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &postTxPendingBlock)
|
||||||
|
require.NoError(t, err)
|
||||||
|
postTxPendingTxs := len(postTxPendingBlock["transactions"].([]interface{}))
|
||||||
|
require.Greater(t, postTxPendingTxs, preTxPendingTxs)
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getBlockByNumber", []interface{}{"latest", true})
|
||||||
|
var postTxLatestBlock map[string]interface{}
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &postTxLatestBlock)
|
||||||
|
require.NoError(t, err)
|
||||||
|
postTxLatestTxs := len(postTxLatestBlock["transactions"].([]interface{}))
|
||||||
|
require.Equal(t, preTxLatestTxs, postTxLatestTxs)
|
||||||
|
|
||||||
|
require.Greater(t, postTxPendingTxs, preTxPendingTxs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_GetTransactionByBlockNumberAndIndex(t *testing.T) {
|
||||||
|
var pendingTx []*rpctypes.Transaction
|
||||||
|
resPendingTxs := util.Call(t, "eth_pendingTransactions", []string{})
|
||||||
|
err := json.Unmarshal(resPendingTxs.Result, &pendingTx)
|
||||||
|
require.NoError(t, err)
|
||||||
|
pendingTxCount := len(pendingTx)
|
||||||
|
|
||||||
|
data := "0x608060405234801561001057600080fd5b5061011e806100206000396000f3fe6080604052348015600f57600080fd5b506004361060285760003560e01c8063bc9c707d14602d575b600080fd5b603360ab565b6040518080602001828103825283818151815260200191508051906020019080838360005b8381101560715780820151818401526020810190506058565b50505050905090810190601f168015609d5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b60606040518060400160405280600681526020017f617261736b61000000000000000000000000000000000000000000000000000081525090509056fea2646970667358221220a31fa4c1ce0b3651fbf5401c511b483c43570c7de4735b5c3b0ad0db30d2573164736f6c63430007050033"
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
param[0]["data"] = data
|
||||||
|
|
||||||
|
txRes := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes.Error)
|
||||||
|
|
||||||
|
rpcRes := util.Call(t, "eth_getTransactionByBlockNumberAndIndex", []interface{}{"pending", "0x" + fmt.Sprintf("%X", pendingTxCount)})
|
||||||
|
var pendingBlockTx map[string]interface{}
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &pendingBlockTx)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// verify the pending tx has all the correct fields from the tx sent.
|
||||||
|
require.NotEmpty(t, pendingBlockTx["hash"])
|
||||||
|
require.Equal(t, pendingBlockTx["value"], "0xa")
|
||||||
|
require.Equal(t, data, pendingBlockTx["input"])
|
||||||
|
|
||||||
|
rpcRes = util.Call(t, "eth_getTransactionByBlockNumberAndIndex", []interface{}{"latest", "0x" + fmt.Sprintf("%X", pendingTxCount)})
|
||||||
|
var latestBlock map[string]interface{}
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &latestBlock)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// verify the pending trasnaction does not exist in the latest block info.
|
||||||
|
require.Empty(t, latestBlock)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_GetTransactionByHash(t *testing.T) {
|
||||||
|
data := "0x608060405234801561001057600080fd5b5061011e806100206000396000f3fe6080604052348015600f57600080fd5b506004361060285760003560e01c806302eb691b14602d575b600080fd5b603360ab565b6040518080602001828103825283818151815260200191508051906020019080838360005b8381101560715780820151818401526020810190506058565b50505050905090810190601f168015609d5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b60606040518060400160405280600d81526020017f617261736b61776173686572650000000000000000000000000000000000000081525090509056fea264697066735822122060917c5c2fab8c058a17afa6d3c1d23a7883b918ea3c7157131ea5b396e1aa7564736f6c63430007050033"
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
param[0]["data"] = data
|
||||||
|
|
||||||
|
txRes := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
var txHash common.Hash
|
||||||
|
err := txHash.UnmarshalJSON(txRes.Result)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
rpcRes := util.Call(t, "eth_getTransactionByHash", []interface{}{txHash})
|
||||||
|
var pendingBlockTx map[string]interface{}
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &pendingBlockTx)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// verify the pending tx has all the correct fields from the tx sent.
|
||||||
|
require.NotEmpty(t, pendingBlockTx)
|
||||||
|
require.NotEmpty(t, pendingBlockTx["hash"])
|
||||||
|
require.Equal(t, pendingBlockTx["value"], "0xa")
|
||||||
|
require.Equal(t, pendingBlockTx["input"], data)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEth_Pending_SendTransaction_PendingNonce(t *testing.T) {
|
||||||
|
currNonce := util.GetNonce(t, "latest")
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", from)
|
||||||
|
param[0]["to"] = addrA
|
||||||
|
param[0]["value"] = "0xA"
|
||||||
|
param[0]["gasLimit"] = "0x5208"
|
||||||
|
param[0]["gasPrice"] = "0x1"
|
||||||
|
|
||||||
|
// first transaction
|
||||||
|
txRes1 := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes1.Error)
|
||||||
|
pendingNonce1 := util.GetNonce(t, "pending")
|
||||||
|
require.Greater(t, uint64(pendingNonce1), uint64(currNonce))
|
||||||
|
|
||||||
|
// second transaction
|
||||||
|
param[0]["to"] = "0x7f0f463c4d57b1bd3e3b79051e6c5ab703e803d9"
|
||||||
|
txRes2 := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes2.Error)
|
||||||
|
pendingNonce2 := util.GetNonce(t, "pending")
|
||||||
|
require.Greater(t, uint64(pendingNonce2), uint64(currNonce))
|
||||||
|
require.Greater(t, uint64(pendingNonce2), uint64(pendingNonce1))
|
||||||
|
|
||||||
|
// third transaction
|
||||||
|
param[0]["to"] = "0x7fb24493808b3f10527e3e0870afeb8a953052d2"
|
||||||
|
txRes3 := util.Call(t, "eth_sendTransaction", param)
|
||||||
|
require.Nil(t, txRes3.Error)
|
||||||
|
pendingNonce3 := util.GetNonce(t, "pending")
|
||||||
|
require.Greater(t, uint64(pendingNonce3), uint64(currNonce))
|
||||||
|
require.Greater(t, uint64(pendingNonce3), uint64(pendingNonce2))
|
||||||
|
}
|
265
tests/utils.go
Normal file
265
tests/utils.go
Normal file
@ -0,0 +1,265 @@
|
|||||||
|
package tests
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/hex"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"math/big"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Request struct {
|
||||||
|
Version string `json:"jsonrpc"`
|
||||||
|
Method string `json:"method"`
|
||||||
|
Params interface{} `json:"params"`
|
||||||
|
ID int `json:"id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RPCError struct {
|
||||||
|
Code int `json:"code"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Data interface{} `json:"data,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Response struct {
|
||||||
|
Error *RPCError `json:"error"`
|
||||||
|
ID int `json:"id"`
|
||||||
|
Result json.RawMessage `json:"result,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
HOST = os.Getenv("HOST")
|
||||||
|
)
|
||||||
|
|
||||||
|
func GetAddress() ([]byte, error) {
|
||||||
|
rpcRes, err := CallWithError("eth_accounts", []string{})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var res []hexutil.Bytes
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &res)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return res[0], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateRequest(method string, params interface{}) Request {
|
||||||
|
return Request{
|
||||||
|
Version: "2.0",
|
||||||
|
Method: method,
|
||||||
|
Params: params,
|
||||||
|
ID: 1,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func Call(t *testing.T, method string, params interface{}) *Response {
|
||||||
|
req, err := json.Marshal(CreateRequest(method, params))
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
var rpcRes *Response
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
/* #nosec */
|
||||||
|
|
||||||
|
if HOST == "" {
|
||||||
|
HOST = "http://localhost:8545"
|
||||||
|
}
|
||||||
|
res, err := http.Post(HOST, "application/json", bytes.NewBuffer(req)) //nolint:gosec
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
decoder := json.NewDecoder(res.Body)
|
||||||
|
rpcRes = new(Response)
|
||||||
|
err = decoder.Decode(&rpcRes)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
err = res.Body.Close()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Nil(t, rpcRes.Error)
|
||||||
|
|
||||||
|
return rpcRes
|
||||||
|
}
|
||||||
|
|
||||||
|
func CallWithError(method string, params interface{}) (*Response, error) {
|
||||||
|
req, err := json.Marshal(CreateRequest(method, params))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var rpcRes *Response
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
/* #nosec */
|
||||||
|
|
||||||
|
if HOST == "" {
|
||||||
|
HOST = "http://localhost:8545"
|
||||||
|
}
|
||||||
|
res, err := http.Post(HOST, "application/json", bytes.NewBuffer(req)) //nolint:gosec
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
decoder := json.NewDecoder(res.Body)
|
||||||
|
rpcRes = new(Response)
|
||||||
|
err = decoder.Decode(&rpcRes)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = res.Body.Close()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if rpcRes.Error != nil {
|
||||||
|
return nil, fmt.Errorf(rpcRes.Error.Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
return rpcRes, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// turns a 0x prefixed hex string to a big.Int
|
||||||
|
func HexToBigInt(t *testing.T, in string) *big.Int {
|
||||||
|
s := in[2:]
|
||||||
|
b, err := hex.DecodeString(s)
|
||||||
|
require.NoError(t, err)
|
||||||
|
return big.NewInt(0).SetBytes(b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// sendTestTransaction sends a dummy transaction
|
||||||
|
func SendTestTransaction(t *testing.T, addr []byte) hexutil.Bytes {
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", addr)
|
||||||
|
param[0]["to"] = "0x1122334455667788990011223344556677889900"
|
||||||
|
param[0]["value"] = "0x1"
|
||||||
|
rpcRes := Call(t, "eth_sendTransaction", param)
|
||||||
|
|
||||||
|
var hash hexutil.Bytes
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &hash)
|
||||||
|
require.NoError(t, err)
|
||||||
|
return hash
|
||||||
|
}
|
||||||
|
|
||||||
|
// deployTestContract deploys a contract that emits an event in the constructor
|
||||||
|
func DeployTestContract(t *testing.T, addr []byte) (hexutil.Bytes, map[string]interface{}) {
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", addr)
|
||||||
|
param[0]["data"] = "0x6080604052348015600f57600080fd5b5060117f775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73889860405160405180910390a2603580604b6000396000f3fe6080604052600080fdfea165627a7a723058206cab665f0f557620554bb45adf266708d2bd349b8a4314bdff205ee8440e3c240029"
|
||||||
|
param[0]["gas"] = "0x200000"
|
||||||
|
|
||||||
|
rpcRes := Call(t, "eth_sendTransaction", param)
|
||||||
|
|
||||||
|
var hash hexutil.Bytes
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &hash)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
receipt := WaitForReceipt(t, hash)
|
||||||
|
require.NotNil(t, receipt, "transaction failed")
|
||||||
|
require.Equal(t, "0x1", receipt["status"].(string))
|
||||||
|
|
||||||
|
return hash, receipt
|
||||||
|
}
|
||||||
|
|
||||||
|
func DeployTestContractWithFunction(t *testing.T, addr []byte) hexutil.Bytes {
|
||||||
|
// pragma solidity ^0.5.1;
|
||||||
|
|
||||||
|
// contract Test {
|
||||||
|
// event Hello(uint256 indexed world);
|
||||||
|
// event TestEvent(uint256 indexed a, uint256 indexed b);
|
||||||
|
|
||||||
|
// uint256 myStorage;
|
||||||
|
|
||||||
|
// constructor() public {
|
||||||
|
// emit Hello(17);
|
||||||
|
// }
|
||||||
|
|
||||||
|
// function test(uint256 a, uint256 b) public {
|
||||||
|
// myStorage = a;
|
||||||
|
// emit TestEvent(a, b);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
bytecode := "0x608060405234801561001057600080fd5b5060117f775a94827b8fd9b519d36cd827093c664f93347070a554f65e4a6f56cd73889860405160405180910390a260d08061004d6000396000f3fe6080604052348015600f57600080fd5b506004361060285760003560e01c8063eb8ac92114602d575b600080fd5b606060048036036040811015604157600080fd5b8101908080359060200190929190803590602001909291905050506062565b005b8160008190555080827ff3ca124a697ba07e8c5e80bebcfcc48991fc16a63170e8a9206e30508960d00360405160405180910390a3505056fea265627a7a723158201d94d2187aaf3a6790527b615fcc40970febf0385fa6d72a2344848ebd0df3e964736f6c63430005110032"
|
||||||
|
|
||||||
|
param := make([]map[string]string, 1)
|
||||||
|
param[0] = make(map[string]string)
|
||||||
|
param[0]["from"] = "0x" + fmt.Sprintf("%x", addr)
|
||||||
|
param[0]["data"] = bytecode
|
||||||
|
param[0]["gas"] = "0x200000"
|
||||||
|
|
||||||
|
rpcRes := Call(t, "eth_sendTransaction", param)
|
||||||
|
|
||||||
|
var hash hexutil.Bytes
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &hash)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
receipt := WaitForReceipt(t, hash)
|
||||||
|
require.NotNil(t, receipt, "transaction failed")
|
||||||
|
require.Equal(t, "0x1", receipt["status"].(string))
|
||||||
|
|
||||||
|
return hash
|
||||||
|
}
|
||||||
|
|
||||||
|
//nolint
|
||||||
|
func GetTransactionReceipt(t *testing.T, hash hexutil.Bytes) map[string]interface{} {
|
||||||
|
param := []string{hash.String()}
|
||||||
|
rpcRes := Call(t, "eth_getTransactionReceipt", param)
|
||||||
|
|
||||||
|
receipt := make(map[string]interface{})
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &receipt)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
return receipt
|
||||||
|
}
|
||||||
|
|
||||||
|
func WaitForReceipt(t *testing.T, hash hexutil.Bytes) map[string]interface{} {
|
||||||
|
for i := 0; i < 12; i++ {
|
||||||
|
receipt := GetTransactionReceipt(t, hash)
|
||||||
|
if receipt != nil {
|
||||||
|
return receipt
|
||||||
|
}
|
||||||
|
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetNonce(t *testing.T, block string) hexutil.Uint64 {
|
||||||
|
from, err := GetAddress()
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
param := []interface{}{hexutil.Bytes(from), block}
|
||||||
|
rpcRes := Call(t, "eth_getTransactionCount", param)
|
||||||
|
|
||||||
|
var nonce hexutil.Uint64
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &nonce)
|
||||||
|
require.NoError(t, err)
|
||||||
|
return nonce
|
||||||
|
}
|
||||||
|
|
||||||
|
func UnlockAllAccounts(t *testing.T) {
|
||||||
|
var accts []common.Address
|
||||||
|
rpcRes := Call(t, "eth_accounts", []map[string]string{})
|
||||||
|
err := json.Unmarshal(rpcRes.Result, &accts)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
for _, acct := range accts {
|
||||||
|
t.Logf("account: %v", acct)
|
||||||
|
rpcRes = Call(t, "personal_unlockAccount", []interface{}{acct, ""})
|
||||||
|
var unlocked bool
|
||||||
|
err = json.Unmarshal(rpcRes.Result, &unlocked)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.True(t, unlocked)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user