267 lines
8.5 KiB
Go
267 lines
8.5 KiB
Go
package types
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
tmbytes "github.com/tendermint/tendermint/libs/bytes"
|
|
tmtypes "github.com/tendermint/tendermint/types"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
|
"github.com/cosmos/cosmos-sdk/client/tx"
|
|
cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
authsigning "github.com/cosmos/cosmos-sdk/x/auth/signing"
|
|
|
|
"github.com/cosmos/ethermint/crypto/ethsecp256k1"
|
|
evmtypes "github.com/cosmos/ethermint/x/evm/types"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
|
ethtypes "github.com/ethereum/go-ethereum/core/types"
|
|
)
|
|
|
|
// RawTxToEthTx returns a evm MsgEthereum transaction from raw tx bytes.
|
|
func RawTxToEthTx(clientCtx client.Context, bz []byte) (*evmtypes.MsgEthereumTx, error) {
|
|
tx, err := clientCtx.TxConfig.TxDecoder()(bz)
|
|
if err != nil {
|
|
return nil, sdkerrors.Wrap(sdkerrors.ErrJSONUnmarshal, err.Error())
|
|
}
|
|
|
|
ethTx, ok := tx.(*evmtypes.MsgEthereumTx)
|
|
if !ok {
|
|
return nil, fmt.Errorf("invalid transaction type %T, expected %T", tx, evmtypes.MsgEthereumTx{})
|
|
}
|
|
return ethTx, nil
|
|
}
|
|
|
|
// NewTransaction returns a transaction that will serialize to the RPC
|
|
// representation, with the given location metadata set (if available).
|
|
func NewTransaction(tx *evmtypes.MsgEthereumTx, txHash, blockHash common.Hash, blockNumber, index uint64) (*Transaction, error) {
|
|
// Verify signature and retrieve sender address
|
|
from, err := tx.VerifySig(tx.ChainID())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
rpcTx := &Transaction{
|
|
From: from,
|
|
Gas: hexutil.Uint64(tx.Data.GasLimit),
|
|
GasPrice: (*hexutil.Big)(tx.Data.Price.BigInt()),
|
|
Hash: txHash,
|
|
Input: hexutil.Bytes(tx.Data.Payload),
|
|
Nonce: hexutil.Uint64(tx.Data.AccountNonce),
|
|
To: tx.To(),
|
|
Value: (*hexutil.Big)(tx.Data.Amount.BigInt()),
|
|
V: (*hexutil.Big)(new(big.Int).SetBytes(tx.Data.V)),
|
|
R: (*hexutil.Big)(new(big.Int).SetBytes(tx.Data.R)),
|
|
S: (*hexutil.Big)(new(big.Int).SetBytes(tx.Data.S)),
|
|
}
|
|
|
|
if blockHash != (common.Hash{}) {
|
|
rpcTx.BlockHash = &blockHash
|
|
rpcTx.BlockNumber = (*hexutil.Big)(new(big.Int).SetUint64(blockNumber))
|
|
rpcTx.TransactionIndex = (*hexutil.Uint64)(&index)
|
|
}
|
|
|
|
return rpcTx, nil
|
|
}
|
|
|
|
// EthBlockFromTendermint returns a JSON-RPC compatible Ethereum blockfrom a given Tendermint block.
|
|
func EthBlockFromTendermint(clientCtx client.Context, queryClient *QueryClient, block *tmtypes.Block) (map[string]interface{}, error) {
|
|
gasLimit, err := BlockMaxGasFromConsensusParams(context.Background(), clientCtx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
transactions, gasUsed, err := EthTransactionsFromTendermint(clientCtx, block.Txs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
req := &evmtypes.QueryBlockBloomRequest{}
|
|
|
|
// use height 0 for querying the latest block height
|
|
res, err := queryClient.BlockBloom(ContextWithHeight(0), req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bloom := ethtypes.BytesToBloom(res.Bloom)
|
|
|
|
return FormatBlock(block.Header, block.Size(), block.Hash(), gasLimit, gasUsed, transactions, bloom), nil
|
|
}
|
|
|
|
// EthHeaderFromTendermint is an util function that returns an Ethereum Header
|
|
// from a tendermint Header.
|
|
func EthHeaderFromTendermint(header tmtypes.Header) *ethtypes.Header {
|
|
return ðtypes.Header{
|
|
ParentHash: common.BytesToHash(header.LastBlockID.Hash.Bytes()),
|
|
UncleHash: common.Hash{},
|
|
Coinbase: common.Address{},
|
|
Root: common.BytesToHash(header.AppHash),
|
|
TxHash: common.BytesToHash(header.DataHash),
|
|
ReceiptHash: common.Hash{},
|
|
Difficulty: nil,
|
|
Number: big.NewInt(header.Height),
|
|
Time: uint64(header.Time.Unix()),
|
|
Extra: nil,
|
|
MixDigest: common.Hash{},
|
|
Nonce: ethtypes.BlockNonce{},
|
|
}
|
|
}
|
|
|
|
// EthTransactionsFromTendermint returns a slice of ethereum transaction hashes and the total gas usage from a set of
|
|
// tendermint block transactions.
|
|
func EthTransactionsFromTendermint(clientCtx client.Context, txs []tmtypes.Tx) ([]common.Hash, *big.Int, error) {
|
|
transactionHashes := []common.Hash{}
|
|
gasUsed := big.NewInt(0)
|
|
|
|
for _, tx := range txs {
|
|
ethTx, err := RawTxToEthTx(clientCtx, tx)
|
|
if err != nil {
|
|
// continue to next transaction in case it's not a MsgEthereumTx
|
|
continue
|
|
}
|
|
// TODO: Remove gas usage calculation if saving gasUsed per block
|
|
gasUsed.Add(gasUsed, big.NewInt(int64(ethTx.GetGas())))
|
|
transactionHashes = append(transactionHashes, common.BytesToHash(tx.Hash()))
|
|
}
|
|
|
|
return transactionHashes, gasUsed, nil
|
|
}
|
|
|
|
// BlockMaxGasFromConsensusParams returns the gas limit for the latest block from the chain consensus params.
|
|
func BlockMaxGasFromConsensusParams(ctx context.Context, clientCtx client.Context) (int64, error) {
|
|
resConsParams, err := clientCtx.Client.ConsensusParams(ctx, nil)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
gasLimit := resConsParams.ConsensusParams.Block.MaxGas
|
|
if gasLimit == -1 {
|
|
// Sets gas limit to max uint32 to not error with javascript dev tooling
|
|
// This -1 value indicating no block gas limit is set to max uint64 with geth hexutils
|
|
// which errors certain javascript dev tooling which only supports up to 53 bits
|
|
gasLimit = int64(^uint32(0))
|
|
}
|
|
|
|
return gasLimit, nil
|
|
}
|
|
|
|
// FormatBlock creates an ethereum block from a tendermint header and ethereum-formatted
|
|
// transactions.
|
|
func FormatBlock(
|
|
header tmtypes.Header, size int, curBlockHash tmbytes.HexBytes, gasLimit int64,
|
|
gasUsed *big.Int, transactions interface{}, bloom ethtypes.Bloom,
|
|
) map[string]interface{} {
|
|
if len(header.DataHash) == 0 {
|
|
header.DataHash = tmbytes.HexBytes(common.Hash{}.Bytes())
|
|
}
|
|
|
|
return map[string]interface{}{
|
|
"number": hexutil.Uint64(header.Height),
|
|
"hash": hexutil.Bytes(curBlockHash),
|
|
"parentHash": hexutil.Bytes(header.LastBlockID.Hash),
|
|
"nonce": hexutil.Uint64(0), // PoW specific
|
|
"sha3Uncles": common.Hash{}, // No uncles in Tendermint
|
|
"logsBloom": bloom,
|
|
"transactionsRoot": hexutil.Bytes(header.DataHash),
|
|
"stateRoot": hexutil.Bytes(header.AppHash),
|
|
"miner": common.Address{},
|
|
"mixHash": common.Hash{},
|
|
"difficulty": 0,
|
|
"totalDifficulty": 0,
|
|
"extraData": hexutil.Uint64(0),
|
|
"size": hexutil.Uint64(size),
|
|
"gasLimit": hexutil.Uint64(gasLimit), // Static gas limit
|
|
"gasUsed": (*hexutil.Big)(gasUsed),
|
|
"timestamp": hexutil.Uint64(header.Time.Unix()),
|
|
"transactions": transactions.([]common.Hash),
|
|
"uncles": []string{},
|
|
"receiptsRoot": common.Hash{},
|
|
}
|
|
}
|
|
|
|
// GetKeyByAddress returns the private key matching the given address. If not found it returns false.
|
|
func GetKeyByAddress(keys []ethsecp256k1.PrivKey, address common.Address) (key *ethsecp256k1.PrivKey, exist bool) {
|
|
for _, key := range keys {
|
|
if bytes.Equal(key.PubKey().Address().Bytes(), address.Bytes()) {
|
|
return &key, true
|
|
}
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
// BuildEthereumTx builds and signs a Cosmos transaction from a MsgEthereumTx and returns the tx
|
|
func BuildEthereumTx(
|
|
clientCtx client.Context,
|
|
msgs []sdk.Msg,
|
|
accNumber, seq, gasLimit uint64,
|
|
fees sdk.Coins,
|
|
privKey cryptotypes.PrivKey,
|
|
) ([]byte, error) {
|
|
signMode := clientCtx.TxConfig.SignModeHandler().DefaultMode()
|
|
signerData := authsigning.SignerData{
|
|
ChainID: clientCtx.ChainID,
|
|
AccountNumber: accNumber,
|
|
Sequence: seq,
|
|
}
|
|
|
|
// Create a TxBuilder
|
|
txBuilder := clientCtx.TxConfig.NewTxBuilder()
|
|
if err := txBuilder.SetMsgs(msgs...); err != nil {
|
|
return nil, err
|
|
|
|
}
|
|
txBuilder.SetFeeAmount(fees)
|
|
txBuilder.SetGasLimit(gasLimit)
|
|
|
|
// sign with the private key
|
|
sigV2, err := tx.SignWithPrivKey(
|
|
signMode, signerData,
|
|
txBuilder, privKey, clientCtx.TxConfig, seq,
|
|
)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := txBuilder.SetSignatures(sigV2); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
txBytes, err := clientCtx.TxConfig.TxEncoder()(txBuilder.GetTx())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return txBytes, nil
|
|
}
|
|
|
|
// GetBlockCumulativeGas returns the cumulative gas used on a block up to a given
|
|
// transaction index. The returned gas used includes the gas from both the SDK and
|
|
// EVM module transactions.
|
|
func GetBlockCumulativeGas(clientCtx client.Context, block *tmtypes.Block, idx int) uint64 {
|
|
var gasUsed uint64
|
|
txDecoder := clientCtx.TxConfig.TxDecoder()
|
|
|
|
for i := 0; i < idx && i < len(block.Txs); i++ {
|
|
txi, err := txDecoder(block.Txs[i])
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
switch tx := txi.(type) {
|
|
case *evmtypes.MsgEthereumTx:
|
|
gasUsed += tx.GetGas()
|
|
case sdk.FeeTx:
|
|
gasUsed += tx.GetGas()
|
|
}
|
|
}
|
|
return gasUsed
|
|
}
|