273 lines
9.1 KiB
Go
273 lines
9.1 KiB
Go
// 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/>.
|
|
|
|
package eth_test
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"math/big"
|
|
|
|
"github.com/ethereum/go-ethereum"
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
. "github.com/onsi/ginkgo"
|
|
. "github.com/onsi/gomega"
|
|
|
|
vulcCore "github.com/vulcanize/vulcanizedb/pkg/core"
|
|
"github.com/vulcanize/vulcanizedb/pkg/eth"
|
|
"github.com/vulcanize/vulcanizedb/pkg/fakes"
|
|
)
|
|
|
|
var _ = Describe("Geth blockchain", func() {
|
|
var (
|
|
mockClient *fakes.MockEthClient
|
|
blockChain *eth.BlockChain
|
|
mockRpcClient *fakes.MockRpcClient
|
|
mockTransactionConverter *fakes.MockTransactionConverter
|
|
node vulcCore.Node
|
|
)
|
|
|
|
BeforeEach(func() {
|
|
mockClient = fakes.NewMockEthClient()
|
|
mockRpcClient = fakes.NewMockRpcClient()
|
|
mockTransactionConverter = fakes.NewMockTransactionConverter()
|
|
node = vulcCore.Node{}
|
|
blockChain = eth.NewBlockChain(mockClient, mockRpcClient, node, mockTransactionConverter)
|
|
})
|
|
|
|
Describe("getting a block", func() {
|
|
It("fetches block from ethClient", func() {
|
|
mockClient.SetBlockByNumberReturnBlock(types.NewBlockWithHeader(&types.Header{}))
|
|
blockNumber := int64(100)
|
|
|
|
_, err := blockChain.GetBlockByNumber(blockNumber)
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockClient.AssertBlockByNumberCalledWith(context.Background(), big.NewInt(blockNumber))
|
|
})
|
|
|
|
It("returns err if ethClient returns err", func() {
|
|
mockClient.SetBlockByNumberErr(fakes.FakeError)
|
|
|
|
_, err := blockChain.GetBlockByNumber(100)
|
|
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(MatchError(fakes.FakeError))
|
|
})
|
|
})
|
|
|
|
Describe("getting a header", func() {
|
|
Describe("default/mainnet", func() {
|
|
It("fetches header from ethClient", func() {
|
|
blockNumber := int64(100)
|
|
mockClient.SetHeaderByNumberReturnHeader(&types.Header{Number: big.NewInt(blockNumber)})
|
|
|
|
_, err := blockChain.GetHeaderByNumber(blockNumber)
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockClient.AssertHeaderByNumberCalledWith(context.Background(), big.NewInt(blockNumber))
|
|
})
|
|
|
|
It("returns err if ethClient returns err", func() {
|
|
mockClient.SetHeaderByNumberErr(fakes.FakeError)
|
|
|
|
_, err := blockChain.GetHeaderByNumber(100)
|
|
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(MatchError(fakes.FakeError))
|
|
})
|
|
|
|
It("fetches headers with multiple blocks", func() {
|
|
_, err := blockChain.GetHeadersByNumbers([]int64{100, 99})
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockRpcClient.AssertBatchCalledWith("eth_getBlockByNumber", 2)
|
|
})
|
|
})
|
|
|
|
Describe("POA/Kovan", func() {
|
|
It("fetches header from rpcClient", func() {
|
|
node.NetworkID = vulcCore.KOVAN_NETWORK_ID
|
|
blockNumber := hexutil.Big(*big.NewInt(100))
|
|
mockRpcClient.SetReturnPOAHeader(vulcCore.POAHeader{Number: &blockNumber})
|
|
blockChain = eth.NewBlockChain(mockClient, mockRpcClient, node, fakes.NewMockTransactionConverter())
|
|
|
|
_, err := blockChain.GetHeaderByNumber(100)
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockRpcClient.AssertCallContextCalledWith(context.Background(), &vulcCore.POAHeader{}, "eth_getBlockByNumber")
|
|
})
|
|
|
|
It("returns err if rpcClient returns err", func() {
|
|
node.NetworkID = vulcCore.KOVAN_NETWORK_ID
|
|
mockRpcClient.SetCallContextErr(fakes.FakeError)
|
|
blockChain = eth.NewBlockChain(mockClient, mockRpcClient, node, fakes.NewMockTransactionConverter())
|
|
|
|
_, err := blockChain.GetHeaderByNumber(100)
|
|
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(MatchError(fakes.FakeError))
|
|
})
|
|
|
|
It("returns error if returned header is empty", func() {
|
|
node.NetworkID = vulcCore.KOVAN_NETWORK_ID
|
|
blockChain = eth.NewBlockChain(mockClient, mockRpcClient, node, fakes.NewMockTransactionConverter())
|
|
|
|
_, err := blockChain.GetHeaderByNumber(100)
|
|
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(MatchError(eth.ErrEmptyHeader))
|
|
})
|
|
|
|
It("returns multiple headers with multiple blocknumbers", func() {
|
|
node.NetworkID = vulcCore.KOVAN_NETWORK_ID
|
|
blockNumber := hexutil.Big(*big.NewInt(100))
|
|
mockRpcClient.SetReturnPOAHeaders([]vulcCore.POAHeader{{Number: &blockNumber}})
|
|
|
|
_, err := blockChain.GetHeadersByNumbers([]int64{100, 99})
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockRpcClient.AssertBatchCalledWith("eth_getBlockByNumber", 2)
|
|
})
|
|
})
|
|
})
|
|
|
|
Describe("getting logs with default FilterQuery", func() {
|
|
It("fetches logs from ethClient", func() {
|
|
mockClient.SetFilterLogsReturnLogs([]types.Log{{}})
|
|
contract := vulcCore.Contract{Hash: common.BytesToHash([]byte{1, 2, 3, 4, 5}).Hex()}
|
|
startingBlockNumber := big.NewInt(1)
|
|
endingBlockNumber := big.NewInt(2)
|
|
|
|
_, err := blockChain.GetFullSyncLogs(contract, startingBlockNumber, endingBlockNumber)
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
expectedQuery := ethereum.FilterQuery{
|
|
FromBlock: startingBlockNumber,
|
|
ToBlock: endingBlockNumber,
|
|
Addresses: []common.Address{common.HexToAddress(contract.Hash)},
|
|
}
|
|
mockClient.AssertFilterLogsCalledWith(context.Background(), expectedQuery)
|
|
})
|
|
|
|
It("returns err if ethClient returns err", func() {
|
|
mockClient.SetFilterLogsErr(fakes.FakeError)
|
|
contract := vulcCore.Contract{Hash: common.BytesToHash([]byte{1, 2, 3, 4, 5}).Hex()}
|
|
startingBlockNumber := big.NewInt(1)
|
|
endingBlockNumber := big.NewInt(2)
|
|
|
|
_, err := blockChain.GetFullSyncLogs(contract, startingBlockNumber, endingBlockNumber)
|
|
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(MatchError(fakes.FakeError))
|
|
})
|
|
})
|
|
|
|
Describe("getting logs with a custom FilterQuery", func() {
|
|
It("fetches logs from ethClient", func() {
|
|
mockClient.SetFilterLogsReturnLogs([]types.Log{{}})
|
|
address := common.HexToAddress("0x")
|
|
startingBlockNumber := big.NewInt(1)
|
|
endingBlockNumber := big.NewInt(2)
|
|
topic := common.HexToHash("0x")
|
|
query := ethereum.FilterQuery{
|
|
FromBlock: startingBlockNumber,
|
|
ToBlock: endingBlockNumber,
|
|
Addresses: []common.Address{address},
|
|
Topics: [][]common.Hash{{topic}},
|
|
}
|
|
|
|
_, err := blockChain.GetEthLogsWithCustomQuery(query)
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockClient.AssertFilterLogsCalledWith(context.Background(), query)
|
|
})
|
|
|
|
It("returns err if ethClient returns err", func() {
|
|
mockClient.SetFilterLogsErr(fakes.FakeError)
|
|
contract := vulcCore.Contract{Hash: common.BytesToHash([]byte{1, 2, 3, 4, 5}).Hex()}
|
|
startingBlockNumber := big.NewInt(1)
|
|
endingBlockNumber := big.NewInt(2)
|
|
query := ethereum.FilterQuery{
|
|
FromBlock: startingBlockNumber,
|
|
ToBlock: endingBlockNumber,
|
|
Addresses: []common.Address{common.HexToAddress(contract.Hash)},
|
|
Topics: nil,
|
|
}
|
|
|
|
_, err := blockChain.GetEthLogsWithCustomQuery(query)
|
|
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(MatchError(fakes.FakeError))
|
|
})
|
|
})
|
|
|
|
Describe("getting transactions", func() {
|
|
It("fetches transaction for each hash", func() {
|
|
_, err := blockChain.GetTransactions([]common.Hash{{}, {}})
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
mockRpcClient.AssertBatchCalledWith("eth_getTransactionByHash", 2)
|
|
})
|
|
|
|
It("converts transaction indexes from hex to int", func() {
|
|
_, err := blockChain.GetTransactions([]common.Hash{{}, {}})
|
|
|
|
Expect(err).NotTo(HaveOccurred())
|
|
Expect(mockTransactionConverter.ConvertHeaderTransactionIndexToIntCalled).To(BeTrue())
|
|
})
|
|
})
|
|
|
|
Describe("getting the most recent block number", func() {
|
|
It("fetches latest header from ethClient", func() {
|
|
blockNumber := int64(100)
|
|
mockClient.SetHeaderByNumberReturnHeader(&types.Header{Number: big.NewInt(blockNumber)})
|
|
|
|
result, err := blockChain.LastBlock()
|
|
Expect(err).NotTo(HaveOccurred())
|
|
|
|
mockClient.AssertHeaderByNumberCalledWith(context.Background(), nil)
|
|
Expect(result).To(Equal(big.NewInt(blockNumber)))
|
|
})
|
|
})
|
|
|
|
Describe("getting an account balance", func() {
|
|
It("fetches the balance for a given account address at a given block height", func() {
|
|
balance := big.NewInt(100000)
|
|
mockClient.SetBalanceAt(balance)
|
|
|
|
result, err := blockChain.GetAccountBalance(common.HexToAddress("0x40"), big.NewInt(100))
|
|
Expect(err).NotTo(HaveOccurred())
|
|
|
|
mockClient.AssertBalanceAtCalled(context.Background(), common.HexToAddress("0x40"), big.NewInt(100))
|
|
Expect(result).To(Equal(balance))
|
|
})
|
|
|
|
It("fails if the client returns an error", func() {
|
|
balance := big.NewInt(100000)
|
|
mockClient.SetBalanceAt(balance)
|
|
setErr := errors.New("testError")
|
|
mockClient.SetBalanceAtErr(setErr)
|
|
|
|
_, err := blockChain.GetAccountBalance(common.HexToAddress("0x40"), big.NewInt(100))
|
|
Expect(err).To(HaveOccurred())
|
|
Expect(err).To(Equal(setErr))
|
|
})
|
|
})
|
|
})
|