2020-01-17 23:16:01 +00:00
|
|
|
// VulcanizeDB
|
|
|
|
// Copyright © 2019 Vulcanize
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2020-01-16 23:21:49 +00:00
|
|
|
package eth
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-01-21 19:12:35 +00:00
|
|
|
"math/big"
|
2020-01-16 23:21:49 +00:00
|
|
|
|
2020-08-31 15:47:06 +00:00
|
|
|
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth"
|
|
|
|
"github.com/vulcanize/ipld-eth-server/pkg/shared"
|
2020-05-01 21:25:58 +00:00
|
|
|
|
2020-01-21 19:12:35 +00:00
|
|
|
"github.com/ethereum/go-ethereum"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2020-01-16 23:21:49 +00:00
|
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
2020-01-21 19:12:35 +00:00
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
2020-01-16 23:21:49 +00:00
|
|
|
"github.com/ethereum/go-ethereum/rpc"
|
|
|
|
)
|
|
|
|
|
2020-06-30 00:16:52 +00:00
|
|
|
// APIName is the namespace for the watcher's eth api
|
2020-01-16 23:21:49 +00:00
|
|
|
const APIName = "eth"
|
|
|
|
|
2020-06-30 00:16:52 +00:00
|
|
|
// APIVersion is the version of the watcher's eth api
|
2020-01-16 23:21:49 +00:00
|
|
|
const APIVersion = "0.0.1"
|
|
|
|
|
|
|
|
type PublicEthAPI struct {
|
2020-05-01 21:25:58 +00:00
|
|
|
B *Backend
|
2020-01-16 23:21:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewPublicEthAPI creates a new PublicEthAPI with the provided underlying Backend
|
2020-01-17 23:16:01 +00:00
|
|
|
func NewPublicEthAPI(b *Backend) *PublicEthAPI {
|
2020-01-16 23:21:49 +00:00
|
|
|
return &PublicEthAPI{
|
2020-05-01 21:25:58 +00:00
|
|
|
B: b,
|
2020-01-16 23:21:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// BlockNumber returns the block number of the chain head.
|
|
|
|
func (pea *PublicEthAPI) BlockNumber() hexutil.Uint64 {
|
2020-05-01 21:25:58 +00:00
|
|
|
number, _ := pea.B.Retriever.RetrieveLastBlockNumber()
|
2020-01-16 23:21:49 +00:00
|
|
|
return hexutil.Uint64(number)
|
|
|
|
}
|
|
|
|
|
2020-01-21 19:12:35 +00:00
|
|
|
// GetLogs returns logs matching the given argument that are stored within the state.
|
|
|
|
//
|
|
|
|
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_getlogs
|
|
|
|
func (pea *PublicEthAPI) GetLogs(ctx context.Context, crit ethereum.FilterQuery) ([]*types.Log, error) {
|
|
|
|
// Convert FilterQuery into ReceiptFilter
|
|
|
|
addrStrs := make([]string, len(crit.Addresses))
|
|
|
|
for i, addr := range crit.Addresses {
|
|
|
|
addrStrs[i] = addr.String()
|
|
|
|
}
|
|
|
|
topicStrSets := make([][]string, 4)
|
|
|
|
for i, topicSet := range crit.Topics {
|
|
|
|
if i > 3 {
|
2020-02-20 22:13:19 +00:00
|
|
|
// don't allow more than 4 topics
|
2020-01-21 19:12:35 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
for _, topic := range topicSet {
|
|
|
|
topicStrSets[i] = append(topicStrSets[i], topic.String())
|
|
|
|
}
|
|
|
|
}
|
2020-02-03 18:22:29 +00:00
|
|
|
filter := ReceiptFilter{
|
2020-04-02 03:34:06 +00:00
|
|
|
LogAddresses: addrStrs,
|
|
|
|
Topics: topicStrSets,
|
2020-01-21 19:12:35 +00:00
|
|
|
}
|
2020-05-01 21:25:58 +00:00
|
|
|
|
|
|
|
// Begin tx
|
|
|
|
tx, err := pea.B.DB.Beginx()
|
2020-01-21 19:12:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-01 21:25:58 +00:00
|
|
|
defer func() {
|
|
|
|
if p := recover(); p != nil {
|
|
|
|
shared.Rollback(tx)
|
|
|
|
panic(p)
|
|
|
|
} else if err != nil {
|
|
|
|
shared.Rollback(tx)
|
|
|
|
} else {
|
|
|
|
err = tx.Commit()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-01-21 19:12:35 +00:00
|
|
|
// If we have a blockhash to filter on, fire off single retrieval query
|
|
|
|
if crit.BlockHash != nil {
|
2020-05-01 21:25:58 +00:00
|
|
|
rctCIDs, err := pea.B.Retriever.RetrieveRctCIDs(tx, filter, 0, crit.BlockHash, nil)
|
2020-01-21 19:12:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-01 21:25:58 +00:00
|
|
|
rctIPLDs, err := pea.B.Fetcher.FetchRcts(tx, rctCIDs)
|
2020-01-21 19:12:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-01 21:25:58 +00:00
|
|
|
if err := tx.Commit(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-01-21 19:12:35 +00:00
|
|
|
return extractLogsOfInterest(rctIPLDs, filter.Topics)
|
|
|
|
}
|
|
|
|
// Otherwise, create block range from criteria
|
|
|
|
// nil values are filled in; to request a single block have both ToBlock and FromBlock equal that number
|
|
|
|
startingBlock := crit.FromBlock
|
|
|
|
endingBlock := crit.ToBlock
|
|
|
|
if startingBlock == nil {
|
2020-05-01 21:25:58 +00:00
|
|
|
startingBlockInt, err := pea.B.Retriever.RetrieveFirstBlockNumber()
|
2020-01-21 19:12:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
startingBlock = big.NewInt(startingBlockInt)
|
|
|
|
}
|
|
|
|
if endingBlock == nil {
|
2020-05-01 21:25:58 +00:00
|
|
|
endingBlockInt, err := pea.B.Retriever.RetrieveLastBlockNumber()
|
2020-01-21 19:12:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
endingBlock = big.NewInt(endingBlockInt)
|
|
|
|
}
|
|
|
|
start := startingBlock.Int64()
|
|
|
|
end := endingBlock.Int64()
|
2020-08-31 15:47:06 +00:00
|
|
|
allRctCIDs := make([]eth.ReceiptModel, 0)
|
2020-01-21 19:12:35 +00:00
|
|
|
for i := start; i <= end; i++ {
|
2020-05-01 21:25:58 +00:00
|
|
|
rctCIDs, err := pea.B.Retriever.RetrieveRctCIDs(tx, filter, i, nil, nil)
|
2020-01-21 19:12:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
allRctCIDs = append(allRctCIDs, rctCIDs...)
|
|
|
|
}
|
2020-05-01 21:25:58 +00:00
|
|
|
rctIPLDs, err := pea.B.Fetcher.FetchRcts(tx, allRctCIDs)
|
2020-05-01 16:07:47 +00:00
|
|
|
if err != nil {
|
2020-01-21 19:12:35 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-01 16:07:47 +00:00
|
|
|
if err := tx.Commit(); err != nil {
|
2020-01-21 19:12:35 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-01 21:25:58 +00:00
|
|
|
logs, err := extractLogsOfInterest(rctIPLDs, filter.Topics)
|
|
|
|
return logs, err // need to return err variable so that we return the err = tx.Commit() assignment in the defer
|
2020-01-21 19:12:35 +00:00
|
|
|
}
|
|
|
|
|
2020-01-26 19:55:26 +00:00
|
|
|
// GetHeaderByNumber returns the requested canonical block header.
|
|
|
|
// * When blockNr is -1 the chain head is returned.
|
|
|
|
// * We cannot support pending block calls since we do not have an active miner
|
|
|
|
func (pea *PublicEthAPI) GetHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (map[string]interface{}, error) {
|
2020-05-01 21:25:58 +00:00
|
|
|
header, err := pea.B.HeaderByNumber(ctx, number)
|
2020-01-26 19:55:26 +00:00
|
|
|
if header != nil && err == nil {
|
|
|
|
return pea.rpcMarshalHeader(header)
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetBlockByNumber returns the requested canonical block.
|
|
|
|
// * When blockNr is -1 the chain head is returned.
|
|
|
|
// * We cannot support pending block calls since we do not have an active miner
|
|
|
|
// * When fullTx is true all transactions in the block are returned, otherwise
|
|
|
|
// only the transaction hash is returned.
|
|
|
|
func (pea *PublicEthAPI) GetBlockByNumber(ctx context.Context, number rpc.BlockNumber, fullTx bool) (map[string]interface{}, error) {
|
2020-05-01 21:25:58 +00:00
|
|
|
block, err := pea.B.BlockByNumber(ctx, number)
|
2020-01-26 19:55:26 +00:00
|
|
|
if block != nil && err == nil {
|
|
|
|
return pea.rpcMarshalBlock(block, true, fullTx)
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetBlockByHash returns the requested block. When fullTx is true all transactions in the block are returned in full
|
|
|
|
// detail, otherwise only the transaction hash is returned.
|
|
|
|
func (pea *PublicEthAPI) GetBlockByHash(ctx context.Context, hash common.Hash, fullTx bool) (map[string]interface{}, error) {
|
2020-05-01 21:25:58 +00:00
|
|
|
block, err := pea.B.BlockByHash(ctx, hash)
|
2020-01-26 19:55:26 +00:00
|
|
|
if block != nil {
|
|
|
|
return pea.rpcMarshalBlock(block, true, fullTx)
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetTransactionByHash returns the transaction for the given hash
|
2020-08-31 15:47:06 +00:00
|
|
|
// eth ipld-eth-server cannot currently handle pending/tx_pool txs
|
2020-01-26 19:55:26 +00:00
|
|
|
func (pea *PublicEthAPI) GetTransactionByHash(ctx context.Context, hash common.Hash) (*RPCTransaction, error) {
|
|
|
|
// Try to return an already finalized transaction
|
2020-05-01 21:25:58 +00:00
|
|
|
tx, blockHash, blockNumber, index, err := pea.B.GetTransaction(ctx, hash)
|
2020-01-26 19:55:26 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if tx != nil {
|
2020-02-20 22:13:19 +00:00
|
|
|
return NewRPCTransaction(tx, blockHash, blockNumber, index), nil
|
2020-01-26 19:55:26 +00:00
|
|
|
}
|
|
|
|
// Transaction unknown, return as such
|
|
|
|
return nil, nil
|
|
|
|
}
|