2020-10-12 10:17:34 -04:00
package commands
2020-10-14 11:59:42 -04:00
import (
2021-05-04 12:51:28 +07:00
"bytes"
2020-10-14 11:59:42 -04:00
"context"
2021-07-01 12:29:32 +08:00
"math/big"
2020-10-12 10:17:34 -04:00
2021-07-01 22:31:14 +01:00
"github.com/ledgerwatch/erigon-lib/gointerfaces"
"github.com/ledgerwatch/erigon-lib/gointerfaces/txpool"
"github.com/ledgerwatch/erigon-lib/gointerfaces/types"
2021-05-21 01:25:53 +07:00
"github.com/ledgerwatch/erigon/common"
"github.com/ledgerwatch/erigon/common/hexutil"
"github.com/ledgerwatch/erigon/core/rawdb"
types2 "github.com/ledgerwatch/erigon/core/types"
2021-09-15 14:22:57 +07:00
"github.com/ledgerwatch/erigon/rlp"
2021-05-21 01:25:53 +07:00
"github.com/ledgerwatch/erigon/rpc"
2022-06-14 16:29:49 +03:00
"github.com/ledgerwatch/erigon/turbo/rpchelper"
2020-10-14 11:59:42 -04:00
)
2020-10-24 13:03:52 -04:00
// GetTransactionByHash implements eth_getTransactionByHash. Returns information about a transaction given the transaction's hash.
2022-07-08 20:15:22 -07:00
func ( api * APIImpl ) GetTransactionByHash ( ctx context . Context , txnHash common . Hash ) ( * RPCTransaction , error ) {
2021-04-03 09:26:00 +03:00
tx , err := api . db . BeginRo ( ctx )
2020-10-14 11:59:42 -04:00
if err != nil {
return nil , err
}
defer tx . Rollback ( )
2022-02-10 15:36:24 +07:00
chainConfig , err := api . chainConfig ( tx )
2022-01-06 18:22:59 +07:00
if err != nil {
return nil , err
}
2022-02-10 15:36:24 +07:00
// https://infura.io/docs/ethereum/json-rpc/eth-getTransactionByHash
2022-07-08 20:15:22 -07:00
blockNum , ok , err := api . txnLookup ( ctx , tx , txnHash )
2021-07-11 05:25:21 +00:00
if err != nil {
return nil , err
}
2022-02-10 15:36:24 +07:00
if ok {
block , err := api . blockByNumberWithSenders ( tx , blockNum )
if err != nil {
return nil , err
}
if block == nil {
return nil , nil
}
blockHash := block . Hash ( )
var txnIndex uint64
var txn types2 . Transaction
for i , transaction := range block . Transactions ( ) {
2022-07-08 20:15:22 -07:00
if transaction . Hash ( ) == txnHash {
2022-02-10 15:36:24 +07:00
txn = transaction
txnIndex = uint64 ( i )
break
}
2022-01-07 20:52:38 +07:00
}
2021-07-01 12:29:32 +08:00
2022-02-10 15:36:24 +07:00
// Add GasPrice for the DynamicFeeTransaction
var baseFee * big . Int
if chainConfig . IsLondon ( blockNum ) && blockHash != ( common . Hash { } ) {
baseFee = block . BaseFee ( )
}
2021-07-01 12:29:32 +08:00
2022-03-07 03:24:21 +00:00
// if no transaction was found then we return nil
if txn == nil {
2022-07-08 20:15:22 -07:00
if chainConfig . Bor == nil {
return nil , nil
2022-07-07 01:40:50 -07:00
}
2022-07-08 20:15:22 -07:00
borTx , _ , _ , _ := rawdb . ReadBorTransactionForBlock ( tx , block )
if borTx == nil {
return nil , nil
}
return newRPCBorTransaction ( borTx , txnHash , blockHash , blockNum , uint64 ( len ( block . Transactions ( ) ) ) , baseFee ) , nil
2022-02-10 15:36:24 +07:00
}
2022-03-07 03:24:21 +00:00
return newRPCTransaction ( txn , blockHash , blockNum , txnIndex , baseFee ) , nil
2021-05-04 12:51:28 +07:00
}
2021-07-22 21:21:55 +07:00
curHeader := rawdb . ReadCurrentHeader ( tx )
2021-10-20 07:28:14 +07:00
if curHeader == nil {
return nil , nil
}
2021-07-22 21:21:55 +07:00
2021-05-04 12:51:28 +07:00
// No finalized transaction, try to retrieve it from the pool
2022-07-08 20:15:22 -07:00
reply , err := api . txPool . Transactions ( ctx , & txpool . TransactionsRequest { Hashes : [ ] * types . H256 { gointerfaces . ConvertHashToH256 ( txnHash ) } } )
2021-05-04 12:51:28 +07:00
if err != nil {
return nil , err
}
2021-05-04 22:58:13 +01:00
if len ( reply . RlpTxs [ 0 ] ) > 0 {
2021-09-15 14:22:57 +07:00
s := rlp . NewStream ( bytes . NewReader ( reply . RlpTxs [ 0 ] ) , uint64 ( len ( reply . RlpTxs [ 0 ] ) ) )
2022-02-10 15:36:24 +07:00
txn , err := types2 . DecodeTransaction ( s )
2021-05-04 12:51:28 +07:00
if err != nil {
return nil , err
}
2022-01-18 17:04:05 +03:00
2022-03-07 03:24:21 +00:00
// if no transaction was found in the txpool then we return nil and an error warning that we didn't find the transaction by the hash
2022-01-18 17:04:05 +03:00
if txn == nil {
return nil , nil
}
2021-07-22 21:21:55 +07:00
return newRPCPendingTransaction ( txn , curHeader , chainConfig ) , nil
2021-05-04 12:51:28 +07:00
}
// Transaction unknown, return as such
return nil , nil
}
// GetRawTransactionByHash returns the bytes of the transaction for the given hash.
func ( api * APIImpl ) GetRawTransactionByHash ( ctx context . Context , hash common . Hash ) ( hexutil . Bytes , error ) {
tx , err := api . db . BeginRo ( ctx )
if err != nil {
return nil , err
}
defer tx . Rollback ( )
// https://infura.io/docs/ethereum/json-rpc/eth-getTransactionByHash
2022-01-07 20:52:38 +07:00
blockNum , ok , err := api . txnLookup ( ctx , tx , hash )
2022-01-06 18:22:59 +07:00
if err != nil {
return nil , err
}
2022-01-07 20:52:38 +07:00
if ! ok {
2022-01-06 18:22:59 +07:00
return nil , nil
}
2022-01-07 20:52:38 +07:00
block , err := api . blockByNumberWithSenders ( tx , blockNum )
2021-07-11 05:25:21 +00:00
if err != nil {
return nil , err
}
2022-01-07 20:52:38 +07:00
if block == nil {
return nil , nil
}
var txn types2 . Transaction
for _ , transaction := range block . Transactions ( ) {
if transaction . Hash ( ) == hash {
txn = transaction
break
}
}
2021-05-04 22:58:13 +01:00
if txn != nil {
2021-05-04 12:51:28 +07:00
var buf bytes . Buffer
err = txn . MarshalBinary ( & buf )
return buf . Bytes ( ) , err
}
// No finalized transaction, try to retrieve it from the pool
reply , err := api . txPool . Transactions ( ctx , & txpool . TransactionsRequest { Hashes : [ ] * types . H256 { gointerfaces . ConvertHashToH256 ( hash ) } } )
if err != nil {
return nil , err
}
2021-05-04 22:58:13 +01:00
if len ( reply . RlpTxs [ 0 ] ) > 0 {
2021-05-04 12:51:28 +07:00
return reply . RlpTxs [ 0 ] , nil
}
return nil , nil
2020-10-14 11:59:42 -04:00
}
2020-10-24 13:03:52 -04:00
// GetTransactionByBlockHashAndIndex implements eth_getTransactionByBlockHashAndIndex. Returns information about a transaction given the block's hash and a transaction index.
2020-10-14 11:59:42 -04:00
func ( api * APIImpl ) GetTransactionByBlockHashAndIndex ( ctx context . Context , blockHash common . Hash , txIndex hexutil . Uint64 ) ( * RPCTransaction , error ) {
2021-04-03 09:26:00 +03:00
tx , err := api . db . BeginRo ( ctx )
2020-10-14 11:59:42 -04:00
if err != nil {
return nil , err
}
defer tx . Rollback ( )
2022-07-07 01:40:50 -07:00
chainConfig , err := api . chainConfig ( tx )
if err != nil {
return nil , err
}
2020-10-14 11:59:42 -04:00
// https://infura.io/docs/ethereum/json-rpc/eth-getTransactionByBlockHashAndIndex
2021-09-29 13:51:51 +07:00
block , err := api . blockByHashWithSenders ( tx , blockHash )
2020-10-24 13:57:09 +07:00
if err != nil {
return nil , err
}
2020-10-14 11:59:42 -04:00
if block == nil {
2021-05-26 13:35:39 +03:00
return nil , nil // not error, see https://github.com/ledgerwatch/erigon/issues/1645
2020-10-14 11:59:42 -04:00
}
txs := block . Transactions ( )
2022-07-07 01:40:50 -07:00
if uint64 ( txIndex ) > uint64 ( len ( txs ) ) {
2022-02-02 15:25:16 +01:00
return nil , nil // not error
2022-07-07 01:40:50 -07:00
} else if uint64 ( txIndex ) == uint64 ( len ( txs ) ) {
2022-07-08 20:15:22 -07:00
if chainConfig . Bor == nil {
2022-07-07 01:40:50 -07:00
return nil , nil // not error
}
2022-07-08 20:15:22 -07:00
borTx , _ , _ , _ := rawdb . ReadBorTransactionForBlock ( tx , block )
if borTx == nil {
return nil , nil // not error
}
derivedBorTxHash := types2 . ComputeBorTxHash ( block . NumberU64 ( ) , block . Hash ( ) )
return newRPCBorTransaction ( borTx , derivedBorTxHash , block . Hash ( ) , block . NumberU64 ( ) , uint64 ( txIndex ) , block . BaseFee ( ) ) , nil
2020-10-14 11:59:42 -04:00
}
2021-07-01 12:29:32 +08:00
return newRPCTransaction ( txs [ txIndex ] , block . Hash ( ) , block . NumberU64 ( ) , uint64 ( txIndex ) , block . BaseFee ( ) ) , nil
2020-10-14 11:59:42 -04:00
}
2021-05-04 12:51:28 +07:00
// GetRawTransactionByBlockHashAndIndex returns the bytes of the transaction for the given block hash and index.
func ( api * APIImpl ) GetRawTransactionByBlockHashAndIndex ( ctx context . Context , blockHash common . Hash , index hexutil . Uint ) ( hexutil . Bytes , error ) {
tx , err := api . db . BeginRo ( ctx )
if err != nil {
return nil , err
}
defer tx . Rollback ( )
// https://infura.io/docs/ethereum/json-rpc/eth-getRawTransactionByBlockHashAndIndex
2021-09-29 13:51:51 +07:00
block , err := api . blockByHashWithSenders ( tx , blockHash )
2021-05-04 12:51:28 +07:00
if err != nil {
return nil , err
}
if block == nil {
2021-05-26 13:35:39 +03:00
return nil , nil // not error, see https://github.com/ledgerwatch/erigon/issues/1645
2021-05-04 12:51:28 +07:00
}
return newRPCRawTransactionFromBlockIndex ( block , uint64 ( index ) )
}
2020-10-24 13:03:52 -04:00
// GetTransactionByBlockNumberAndIndex implements eth_getTransactionByBlockNumberAndIndex. Returns information about a transaction given a block number and transaction index.
2020-10-14 11:59:42 -04:00
func ( api * APIImpl ) GetTransactionByBlockNumberAndIndex ( ctx context . Context , blockNr rpc . BlockNumber , txIndex hexutil . Uint ) ( * RPCTransaction , error ) {
2021-04-03 09:26:00 +03:00
tx , err := api . db . BeginRo ( ctx )
2020-10-14 11:59:42 -04:00
if err != nil {
return nil , err
}
defer tx . Rollback ( )
2022-07-07 01:40:50 -07:00
chainConfig , err := api . chainConfig ( tx )
if err != nil {
return nil , err
}
2020-10-14 11:59:42 -04:00
// https://infura.io/docs/ethereum/json-rpc/eth-getTransactionByBlockNumberAndIndex
2022-06-14 16:29:49 +03:00
blockNum , _ , _ , err := rpchelper . GetBlockNumber ( rpc . BlockNumberOrHashWithNumber ( blockNr ) , tx , api . filters )
2020-10-14 11:59:42 -04:00
if err != nil {
return nil , err
}
2021-09-29 13:51:51 +07:00
block , err := api . blockByNumberWithSenders ( tx , blockNum )
2020-10-24 13:57:09 +07:00
if err != nil {
return nil , err
}
2020-10-14 11:59:42 -04:00
if block == nil {
2021-05-26 13:35:39 +03:00
return nil , nil // not error, see https://github.com/ledgerwatch/erigon/issues/1645
2020-10-14 11:59:42 -04:00
}
txs := block . Transactions ( )
2022-07-07 01:40:50 -07:00
if uint64 ( txIndex ) > uint64 ( len ( txs ) ) {
2022-02-02 15:25:16 +01:00
return nil , nil // not error
2022-07-07 01:40:50 -07:00
} else if uint64 ( txIndex ) == uint64 ( len ( txs ) ) {
2022-07-08 20:15:22 -07:00
if chainConfig . Bor == nil {
2022-07-07 01:40:50 -07:00
return nil , nil // not error
}
2022-07-08 20:15:22 -07:00
borTx , _ , _ , _ := rawdb . ReadBorTransactionForBlock ( tx , block )
if borTx == nil {
return nil , nil
}
derivedBorTxHash := types2 . ComputeBorTxHash ( block . NumberU64 ( ) , block . Hash ( ) )
return newRPCBorTransaction ( borTx , derivedBorTxHash , block . Hash ( ) , block . NumberU64 ( ) , uint64 ( txIndex ) , block . BaseFee ( ) ) , nil
2020-10-14 11:59:42 -04:00
}
2021-07-01 12:29:32 +08:00
return newRPCTransaction ( txs [ txIndex ] , block . Hash ( ) , block . NumberU64 ( ) , uint64 ( txIndex ) , block . BaseFee ( ) ) , nil
2020-10-14 11:59:42 -04:00
}
2021-05-04 12:51:28 +07:00
// GetRawTransactionByBlockNumberAndIndex returns the bytes of the transaction for the given block number and index.
func ( api * APIImpl ) GetRawTransactionByBlockNumberAndIndex ( ctx context . Context , blockNr rpc . BlockNumber , index hexutil . Uint ) ( hexutil . Bytes , error ) {
tx , err := api . db . BeginRo ( ctx )
if err != nil {
return nil , err
}
defer tx . Rollback ( )
// https://infura.io/docs/ethereum/json-rpc/eth-getRawTransactionByBlockNumberAndIndex
2021-09-29 13:51:51 +07:00
block , err := api . blockByRPCNumber ( blockNr , tx )
2021-05-04 12:51:28 +07:00
if err != nil {
return nil , err
}
if block == nil {
2021-05-26 13:35:39 +03:00
return nil , nil // not error, see https://github.com/ledgerwatch/erigon/issues/1645
2021-05-04 12:51:28 +07:00
}
return newRPCRawTransactionFromBlockIndex ( block , uint64 ( index ) )
}