mirror of https://github.com/ethereum/go-ethereum
Merge pull request #1701 from karalabe/eth62-sync-rebase
eth: implement eth/62 synchronization logicpull/1732/merge
commit
6ec13e7e2b
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,45 @@ |
|||||||
|
// Copyright 2015 The go-ethereum Authors
|
||||||
|
// This file is part of the go-ethereum library.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// The go-ethereum library 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 Lesser General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU Lesser General Public License
|
||||||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
// Contains the metrics collected by the downloader.
|
||||||
|
|
||||||
|
package downloader |
||||||
|
|
||||||
|
import ( |
||||||
|
"github.com/ethereum/go-ethereum/metrics" |
||||||
|
) |
||||||
|
|
||||||
|
var ( |
||||||
|
hashInMeter = metrics.NewMeter("eth/downloader/hashes/in") |
||||||
|
hashReqTimer = metrics.NewTimer("eth/downloader/hashes/req") |
||||||
|
hashDropMeter = metrics.NewMeter("eth/downloader/hashes/drop") |
||||||
|
hashTimeoutMeter = metrics.NewMeter("eth/downloader/hashes/timeout") |
||||||
|
|
||||||
|
blockInMeter = metrics.NewMeter("eth/downloader/blocks/in") |
||||||
|
blockReqTimer = metrics.NewTimer("eth/downloader/blocks/req") |
||||||
|
blockDropMeter = metrics.NewMeter("eth/downloader/blocks/drop") |
||||||
|
blockTimeoutMeter = metrics.NewMeter("eth/downloader/blocks/timeout") |
||||||
|
|
||||||
|
headerInMeter = metrics.NewMeter("eth/downloader/headers/in") |
||||||
|
headerReqTimer = metrics.NewTimer("eth/downloader/headers/req") |
||||||
|
headerDropMeter = metrics.NewMeter("eth/downloader/headers/drop") |
||||||
|
headerTimeoutMeter = metrics.NewMeter("eth/downloader/headers/timeout") |
||||||
|
|
||||||
|
bodyInMeter = metrics.NewMeter("eth/downloader/bodies/in") |
||||||
|
bodyReqTimer = metrics.NewTimer("eth/downloader/bodies/req") |
||||||
|
bodyDropMeter = metrics.NewMeter("eth/downloader/bodies/drop") |
||||||
|
bodyTimeoutMeter = metrics.NewMeter("eth/downloader/bodies/timeout") |
||||||
|
) |
@ -0,0 +1,522 @@ |
|||||||
|
package eth |
||||||
|
|
||||||
|
import ( |
||||||
|
"fmt" |
||||||
|
"math/big" |
||||||
|
"math/rand" |
||||||
|
"testing" |
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common" |
||||||
|
"github.com/ethereum/go-ethereum/core" |
||||||
|
"github.com/ethereum/go-ethereum/core/state" |
||||||
|
"github.com/ethereum/go-ethereum/core/types" |
||||||
|
"github.com/ethereum/go-ethereum/crypto" |
||||||
|
"github.com/ethereum/go-ethereum/eth/downloader" |
||||||
|
"github.com/ethereum/go-ethereum/ethdb" |
||||||
|
"github.com/ethereum/go-ethereum/p2p" |
||||||
|
"github.com/ethereum/go-ethereum/params" |
||||||
|
) |
||||||
|
|
||||||
|
// Tests that hashes can be retrieved from a remote chain by hashes in reverse
|
||||||
|
// order.
|
||||||
|
func TestGetBlockHashes61(t *testing.T) { testGetBlockHashes(t, 61) } |
||||||
|
|
||||||
|
func testGetBlockHashes(t *testing.T, protocol int) { |
||||||
|
pm := newTestProtocolManager(downloader.MaxHashFetch+15, nil, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Create a batch of tests for various scenarios
|
||||||
|
limit := downloader.MaxHashFetch |
||||||
|
tests := []struct { |
||||||
|
origin common.Hash |
||||||
|
number int |
||||||
|
result int |
||||||
|
}{ |
||||||
|
{common.Hash{}, 1, 0}, // Make sure non existent hashes don't return results
|
||||||
|
{pm.chainman.Genesis().Hash(), 1, 0}, // There are no hashes to retrieve up from the genesis
|
||||||
|
{pm.chainman.GetBlockByNumber(5).Hash(), 5, 5}, // All the hashes including the genesis requested
|
||||||
|
{pm.chainman.GetBlockByNumber(5).Hash(), 10, 5}, // More hashes than available till the genesis requested
|
||||||
|
{pm.chainman.GetBlockByNumber(100).Hash(), 10, 10}, // All hashes available from the middle of the chain
|
||||||
|
{pm.chainman.CurrentBlock().Hash(), 10, 10}, // All hashes available from the head of the chain
|
||||||
|
{pm.chainman.CurrentBlock().Hash(), limit, limit}, // Request the maximum allowed hash count
|
||||||
|
{pm.chainman.CurrentBlock().Hash(), limit + 1, limit}, // Request more than the maximum allowed hash count
|
||||||
|
} |
||||||
|
// Run each of the tests and verify the results against the chain
|
||||||
|
for i, tt := range tests { |
||||||
|
// Assemble the hash response we would like to receive
|
||||||
|
resp := make([]common.Hash, tt.result) |
||||||
|
if len(resp) > 0 { |
||||||
|
from := pm.chainman.GetBlock(tt.origin).NumberU64() - 1 |
||||||
|
for j := 0; j < len(resp); j++ { |
||||||
|
resp[j] = pm.chainman.GetBlockByNumber(uint64(int(from) - j)).Hash() |
||||||
|
} |
||||||
|
} |
||||||
|
// Send the hash request and verify the response
|
||||||
|
p2p.Send(peer.app, 0x03, getBlockHashesData{tt.origin, uint64(tt.number)}) |
||||||
|
if err := p2p.ExpectMsg(peer.app, 0x04, resp); err != nil { |
||||||
|
t.Errorf("test %d: block hashes mismatch: %v", i, err) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Tests that hashes can be retrieved from a remote chain by numbers in forward
|
||||||
|
// order.
|
||||||
|
func TestGetBlockHashesFromNumber61(t *testing.T) { testGetBlockHashesFromNumber(t, 61) } |
||||||
|
|
||||||
|
func testGetBlockHashesFromNumber(t *testing.T, protocol int) { |
||||||
|
pm := newTestProtocolManager(downloader.MaxHashFetch+15, nil, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Create a batch of tests for various scenarios
|
||||||
|
limit := downloader.MaxHashFetch |
||||||
|
tests := []struct { |
||||||
|
origin uint64 |
||||||
|
number int |
||||||
|
result int |
||||||
|
}{ |
||||||
|
{pm.chainman.CurrentBlock().NumberU64() + 1, 1, 0}, // Out of bounds requests should return empty
|
||||||
|
{pm.chainman.CurrentBlock().NumberU64(), 1, 1}, // Make sure the head hash can be retrieved
|
||||||
|
{pm.chainman.CurrentBlock().NumberU64() - 4, 5, 5}, // All hashes, including the head hash requested
|
||||||
|
{pm.chainman.CurrentBlock().NumberU64() - 4, 10, 5}, // More hashes requested than available till the head
|
||||||
|
{pm.chainman.CurrentBlock().NumberU64() - 100, 10, 10}, // All hashes available from the middle of the chain
|
||||||
|
{0, 10, 10}, // All hashes available from the root of the chain
|
||||||
|
{0, limit, limit}, // Request the maximum allowed hash count
|
||||||
|
{0, limit + 1, limit}, // Request more than the maximum allowed hash count
|
||||||
|
{0, 1, 1}, // Make sure the genesis hash can be retrieved
|
||||||
|
} |
||||||
|
// Run each of the tests and verify the results against the chain
|
||||||
|
for i, tt := range tests { |
||||||
|
// Assemble the hash response we would like to receive
|
||||||
|
resp := make([]common.Hash, tt.result) |
||||||
|
for j := 0; j < len(resp); j++ { |
||||||
|
resp[j] = pm.chainman.GetBlockByNumber(tt.origin + uint64(j)).Hash() |
||||||
|
} |
||||||
|
// Send the hash request and verify the response
|
||||||
|
p2p.Send(peer.app, 0x08, getBlockHashesFromNumberData{tt.origin, uint64(tt.number)}) |
||||||
|
if err := p2p.ExpectMsg(peer.app, 0x04, resp); err != nil { |
||||||
|
t.Errorf("test %d: block hashes mismatch: %v", i, err) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Tests that blocks can be retrieved from a remote chain based on their hashes.
|
||||||
|
func TestGetBlocks61(t *testing.T) { testGetBlocks(t, 61) } |
||||||
|
|
||||||
|
func testGetBlocks(t *testing.T, protocol int) { |
||||||
|
pm := newTestProtocolManager(downloader.MaxHashFetch+15, nil, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Create a batch of tests for various scenarios
|
||||||
|
limit := downloader.MaxBlockFetch |
||||||
|
tests := []struct { |
||||||
|
random int // Number of blocks to fetch randomly from the chain
|
||||||
|
explicit []common.Hash // Explicitly requested blocks
|
||||||
|
available []bool // Availability of explicitly requested blocks
|
||||||
|
expected int // Total number of existing blocks to expect
|
||||||
|
}{ |
||||||
|
{1, nil, nil, 1}, // A single random block should be retrievable
|
||||||
|
{10, nil, nil, 10}, // Multiple random blocks should be retrievable
|
||||||
|
{limit, nil, nil, limit}, // The maximum possible blocks should be retrievable
|
||||||
|
{limit + 1, nil, nil, limit}, // No more that the possible block count should be returned
|
||||||
|
{0, []common.Hash{pm.chainman.Genesis().Hash()}, []bool{true}, 1}, // The genesis block should be retrievable
|
||||||
|
{0, []common.Hash{pm.chainman.CurrentBlock().Hash()}, []bool{true}, 1}, // The chains head block should be retrievable
|
||||||
|
{0, []common.Hash{common.Hash{}}, []bool{false}, 0}, // A non existent block should not be returned
|
||||||
|
|
||||||
|
// Existing and non-existing blocks interleaved should not cause problems
|
||||||
|
{0, []common.Hash{ |
||||||
|
common.Hash{}, |
||||||
|
pm.chainman.GetBlockByNumber(1).Hash(), |
||||||
|
common.Hash{}, |
||||||
|
pm.chainman.GetBlockByNumber(10).Hash(), |
||||||
|
common.Hash{}, |
||||||
|
pm.chainman.GetBlockByNumber(100).Hash(), |
||||||
|
common.Hash{}, |
||||||
|
}, []bool{false, true, false, true, false, true, false}, 3}, |
||||||
|
} |
||||||
|
// Run each of the tests and verify the results against the chain
|
||||||
|
for i, tt := range tests { |
||||||
|
// Collect the hashes to request, and the response to expect
|
||||||
|
hashes, seen := []common.Hash{}, make(map[int64]bool) |
||||||
|
blocks := []*types.Block{} |
||||||
|
|
||||||
|
for j := 0; j < tt.random; j++ { |
||||||
|
for { |
||||||
|
num := rand.Int63n(int64(pm.chainman.CurrentBlock().NumberU64())) |
||||||
|
if !seen[num] { |
||||||
|
seen[num] = true |
||||||
|
|
||||||
|
block := pm.chainman.GetBlockByNumber(uint64(num)) |
||||||
|
hashes = append(hashes, block.Hash()) |
||||||
|
if len(blocks) < tt.expected { |
||||||
|
blocks = append(blocks, block) |
||||||
|
} |
||||||
|
break |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
for j, hash := range tt.explicit { |
||||||
|
hashes = append(hashes, hash) |
||||||
|
if tt.available[j] && len(blocks) < tt.expected { |
||||||
|
blocks = append(blocks, pm.chainman.GetBlock(hash)) |
||||||
|
} |
||||||
|
} |
||||||
|
// Send the hash request and verify the response
|
||||||
|
p2p.Send(peer.app, 0x05, hashes) |
||||||
|
if err := p2p.ExpectMsg(peer.app, 0x06, blocks); err != nil { |
||||||
|
t.Errorf("test %d: blocks mismatch: %v", i, err) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Tests that block headers can be retrieved from a remote chain based on user queries.
|
||||||
|
func TestGetBlockHeaders62(t *testing.T) { testGetBlockHeaders(t, 62) } |
||||||
|
func TestGetBlockHeaders63(t *testing.T) { testGetBlockHeaders(t, 63) } |
||||||
|
func TestGetBlockHeaders64(t *testing.T) { testGetBlockHeaders(t, 64) } |
||||||
|
|
||||||
|
func testGetBlockHeaders(t *testing.T, protocol int) { |
||||||
|
pm := newTestProtocolManager(downloader.MaxHashFetch+15, nil, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Create a "random" unknown hash for testing
|
||||||
|
var unknown common.Hash |
||||||
|
for i, _ := range unknown { |
||||||
|
unknown[i] = byte(i) |
||||||
|
} |
||||||
|
// Create a batch of tests for various scenarios
|
||||||
|
limit := uint64(downloader.MaxHeaderFetch) |
||||||
|
tests := []struct { |
||||||
|
query *getBlockHeadersData // The query to execute for header retrieval
|
||||||
|
expect []common.Hash // The hashes of the block whose headers are expected
|
||||||
|
}{ |
||||||
|
// A single random block should be retrievable by hash and number too
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Hash: pm.chainman.GetBlockByNumber(limit / 2).Hash()}, Amount: 1}, |
||||||
|
[]common.Hash{pm.chainman.GetBlockByNumber(limit / 2).Hash()}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: limit / 2}, Amount: 1}, |
||||||
|
[]common.Hash{pm.chainman.GetBlockByNumber(limit / 2).Hash()}, |
||||||
|
}, |
||||||
|
// Multiple headers should be retrievable in both directions
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: limit / 2}, Amount: 3}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(limit / 2).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 + 1).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 + 2).Hash(), |
||||||
|
}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: limit / 2}, Amount: 3, Reverse: true}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(limit / 2).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 - 1).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 - 2).Hash(), |
||||||
|
}, |
||||||
|
}, |
||||||
|
// Multiple headers with skip lists should be retrievable
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: limit / 2}, Skip: 3, Amount: 3}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(limit / 2).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 + 4).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 + 8).Hash(), |
||||||
|
}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: limit / 2}, Skip: 3, Amount: 3, Reverse: true}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(limit / 2).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 - 4).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(limit/2 - 8).Hash(), |
||||||
|
}, |
||||||
|
}, |
||||||
|
// The chain endpoints should be retrievable
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: 0}, Amount: 1}, |
||||||
|
[]common.Hash{pm.chainman.GetBlockByNumber(0).Hash()}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: pm.chainman.CurrentBlock().NumberU64()}, Amount: 1}, |
||||||
|
[]common.Hash{pm.chainman.CurrentBlock().Hash()}, |
||||||
|
}, |
||||||
|
// Ensure protocol limits are honored
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: pm.chainman.CurrentBlock().NumberU64() - 1}, Amount: limit + 10, Reverse: true}, |
||||||
|
pm.chainman.GetBlockHashesFromHash(pm.chainman.CurrentBlock().Hash(), limit), |
||||||
|
}, |
||||||
|
// Check that requesting more than available is handled gracefully
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: pm.chainman.CurrentBlock().NumberU64() - 4}, Skip: 3, Amount: 3}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(pm.chainman.CurrentBlock().NumberU64() - 4).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(pm.chainman.CurrentBlock().NumberU64()).Hash(), |
||||||
|
}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: 4}, Skip: 3, Amount: 3, Reverse: true}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(4).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(0).Hash(), |
||||||
|
}, |
||||||
|
}, |
||||||
|
// Check that requesting more than available is handled gracefully, even if mid skip
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: pm.chainman.CurrentBlock().NumberU64() - 4}, Skip: 2, Amount: 3}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(pm.chainman.CurrentBlock().NumberU64() - 4).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(pm.chainman.CurrentBlock().NumberU64() - 1).Hash(), |
||||||
|
}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: 4}, Skip: 2, Amount: 3, Reverse: true}, |
||||||
|
[]common.Hash{ |
||||||
|
pm.chainman.GetBlockByNumber(4).Hash(), |
||||||
|
pm.chainman.GetBlockByNumber(1).Hash(), |
||||||
|
}, |
||||||
|
}, |
||||||
|
// Check that non existing headers aren't returned
|
||||||
|
{ |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Hash: unknown}, Amount: 1}, |
||||||
|
[]common.Hash{}, |
||||||
|
}, { |
||||||
|
&getBlockHeadersData{Origin: hashOrNumber{Number: pm.chainman.CurrentBlock().NumberU64() + 1}, Amount: 1}, |
||||||
|
[]common.Hash{}, |
||||||
|
}, |
||||||
|
} |
||||||
|
// Run each of the tests and verify the results against the chain
|
||||||
|
for i, tt := range tests { |
||||||
|
// Collect the headers to expect in the response
|
||||||
|
headers := []*types.Header{} |
||||||
|
for _, hash := range tt.expect { |
||||||
|
headers = append(headers, pm.chainman.GetBlock(hash).Header()) |
||||||
|
} |
||||||
|
// Send the hash request and verify the response
|
||||||
|
p2p.Send(peer.app, 0x03, tt.query) |
||||||
|
if err := p2p.ExpectMsg(peer.app, 0x04, headers); err != nil { |
||||||
|
t.Errorf("test %d: headers mismatch: %v", i, err) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Tests that block contents can be retrieved from a remote chain based on their hashes.
|
||||||
|
func TestGetBlockBodies62(t *testing.T) { testGetBlockBodies(t, 62) } |
||||||
|
func TestGetBlockBodies63(t *testing.T) { testGetBlockBodies(t, 63) } |
||||||
|
func TestGetBlockBodies64(t *testing.T) { testGetBlockBodies(t, 64) } |
||||||
|
|
||||||
|
func testGetBlockBodies(t *testing.T, protocol int) { |
||||||
|
pm := newTestProtocolManager(downloader.MaxBlockFetch+15, nil, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Create a batch of tests for various scenarios
|
||||||
|
limit := downloader.MaxBlockFetch |
||||||
|
tests := []struct { |
||||||
|
random int // Number of blocks to fetch randomly from the chain
|
||||||
|
explicit []common.Hash // Explicitly requested blocks
|
||||||
|
available []bool // Availability of explicitly requested blocks
|
||||||
|
expected int // Total number of existing blocks to expect
|
||||||
|
}{ |
||||||
|
{1, nil, nil, 1}, // A single random block should be retrievable
|
||||||
|
{10, nil, nil, 10}, // Multiple random blocks should be retrievable
|
||||||
|
{limit, nil, nil, limit}, // The maximum possible blocks should be retrievable
|
||||||
|
{limit + 1, nil, nil, limit}, // No more that the possible block count should be returned
|
||||||
|
{0, []common.Hash{pm.chainman.Genesis().Hash()}, []bool{true}, 1}, // The genesis block should be retrievable
|
||||||
|
{0, []common.Hash{pm.chainman.CurrentBlock().Hash()}, []bool{true}, 1}, // The chains head block should be retrievable
|
||||||
|
{0, []common.Hash{common.Hash{}}, []bool{false}, 0}, // A non existent block should not be returned
|
||||||
|
|
||||||
|
// Existing and non-existing blocks interleaved should not cause problems
|
||||||
|
{0, []common.Hash{ |
||||||
|
common.Hash{}, |
||||||
|
pm.chainman.GetBlockByNumber(1).Hash(), |
||||||
|
common.Hash{}, |
||||||
|
pm.chainman.GetBlockByNumber(10).Hash(), |
||||||
|
common.Hash{}, |
||||||
|
pm.chainman.GetBlockByNumber(100).Hash(), |
||||||
|
common.Hash{}, |
||||||
|
}, []bool{false, true, false, true, false, true, false}, 3}, |
||||||
|
} |
||||||
|
// Run each of the tests and verify the results against the chain
|
||||||
|
for i, tt := range tests { |
||||||
|
// Collect the hashes to request, and the response to expect
|
||||||
|
hashes, seen := []common.Hash{}, make(map[int64]bool) |
||||||
|
bodies := []*blockBody{} |
||||||
|
|
||||||
|
for j := 0; j < tt.random; j++ { |
||||||
|
for { |
||||||
|
num := rand.Int63n(int64(pm.chainman.CurrentBlock().NumberU64())) |
||||||
|
if !seen[num] { |
||||||
|
seen[num] = true |
||||||
|
|
||||||
|
block := pm.chainman.GetBlockByNumber(uint64(num)) |
||||||
|
hashes = append(hashes, block.Hash()) |
||||||
|
if len(bodies) < tt.expected { |
||||||
|
bodies = append(bodies, &blockBody{Transactions: block.Transactions(), Uncles: block.Uncles()}) |
||||||
|
} |
||||||
|
break |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
for j, hash := range tt.explicit { |
||||||
|
hashes = append(hashes, hash) |
||||||
|
if tt.available[j] && len(bodies) < tt.expected { |
||||||
|
block := pm.chainman.GetBlock(hash) |
||||||
|
bodies = append(bodies, &blockBody{Transactions: block.Transactions(), Uncles: block.Uncles()}) |
||||||
|
} |
||||||
|
} |
||||||
|
// Send the hash request and verify the response
|
||||||
|
p2p.Send(peer.app, 0x05, hashes) |
||||||
|
if err := p2p.ExpectMsg(peer.app, 0x06, bodies); err != nil { |
||||||
|
t.Errorf("test %d: bodies mismatch: %v", i, err) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Tests that the node state database can be retrieved based on hashes.
|
||||||
|
func TestGetNodeData63(t *testing.T) { testGetNodeData(t, 63) } |
||||||
|
func TestGetNodeData64(t *testing.T) { testGetNodeData(t, 64) } |
||||||
|
|
||||||
|
func testGetNodeData(t *testing.T, protocol int) { |
||||||
|
// Define three accounts to simulate transactions with
|
||||||
|
acc1Key, _ := crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a") |
||||||
|
acc2Key, _ := crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee") |
||||||
|
acc1Addr := crypto.PubkeyToAddress(acc1Key.PublicKey) |
||||||
|
acc2Addr := crypto.PubkeyToAddress(acc2Key.PublicKey) |
||||||
|
|
||||||
|
// Create a chain generator with some simple transactions (blatantly stolen from @fjl/chain_makerts_test)
|
||||||
|
generator := func(i int, block *core.BlockGen) { |
||||||
|
switch i { |
||||||
|
case 0: |
||||||
|
// In block 1, the test bank sends account #1 some ether.
|
||||||
|
tx, _ := types.NewTransaction(block.TxNonce(testBankAddress), acc1Addr, big.NewInt(10000), params.TxGas, nil, nil).SignECDSA(testBankKey) |
||||||
|
block.AddTx(tx) |
||||||
|
case 1: |
||||||
|
// In block 2, the test bank sends some more ether to account #1.
|
||||||
|
// acc1Addr passes it on to account #2.
|
||||||
|
tx1, _ := types.NewTransaction(block.TxNonce(testBankAddress), acc1Addr, big.NewInt(1000), params.TxGas, nil, nil).SignECDSA(testBankKey) |
||||||
|
tx2, _ := types.NewTransaction(block.TxNonce(acc1Addr), acc2Addr, big.NewInt(1000), params.TxGas, nil, nil).SignECDSA(acc1Key) |
||||||
|
block.AddTx(tx1) |
||||||
|
block.AddTx(tx2) |
||||||
|
case 2: |
||||||
|
// Block 3 is empty but was mined by account #2.
|
||||||
|
block.SetCoinbase(acc2Addr) |
||||||
|
block.SetExtra([]byte("yeehaw")) |
||||||
|
case 3: |
||||||
|
// Block 4 includes blocks 2 and 3 as uncle headers (with modified extra data).
|
||||||
|
b2 := block.PrevBlock(1).Header() |
||||||
|
b2.Extra = []byte("foo") |
||||||
|
block.AddUncle(b2) |
||||||
|
b3 := block.PrevBlock(2).Header() |
||||||
|
b3.Extra = []byte("foo") |
||||||
|
block.AddUncle(b3) |
||||||
|
} |
||||||
|
} |
||||||
|
// Assemble the test environment
|
||||||
|
pm := newTestProtocolManager(4, generator, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Fetch for now the entire chain db
|
||||||
|
hashes := []common.Hash{} |
||||||
|
for _, key := range pm.chaindb.(*ethdb.MemDatabase).Keys() { |
||||||
|
hashes = append(hashes, common.BytesToHash(key)) |
||||||
|
} |
||||||
|
p2p.Send(peer.app, 0x0d, hashes) |
||||||
|
msg, err := peer.app.ReadMsg() |
||||||
|
if err != nil { |
||||||
|
t.Fatalf("failed to read node data response: %v", err) |
||||||
|
} |
||||||
|
if msg.Code != 0x0e { |
||||||
|
t.Fatalf("response packet code mismatch: have %x, want %x", msg.Code, 0x0c) |
||||||
|
} |
||||||
|
var data [][]byte |
||||||
|
if err := msg.Decode(&data); err != nil { |
||||||
|
t.Fatalf("failed to decode response node data: %v", err) |
||||||
|
} |
||||||
|
// Verify that all hashes correspond to the requested data, and reconstruct a state tree
|
||||||
|
for i, want := range hashes { |
||||||
|
if hash := crypto.Sha3Hash(data[i]); hash != want { |
||||||
|
fmt.Errorf("data hash mismatch: have %x, want %x", hash, want) |
||||||
|
} |
||||||
|
} |
||||||
|
statedb, _ := ethdb.NewMemDatabase() |
||||||
|
for i := 0; i < len(data); i++ { |
||||||
|
statedb.Put(hashes[i].Bytes(), data[i]) |
||||||
|
} |
||||||
|
accounts := []common.Address{testBankAddress, acc1Addr, acc2Addr} |
||||||
|
for i := uint64(0); i <= pm.chainman.CurrentBlock().NumberU64(); i++ { |
||||||
|
trie := state.New(pm.chainman.GetBlockByNumber(i).Root(), statedb) |
||||||
|
|
||||||
|
for j, acc := range accounts { |
||||||
|
bw := pm.chainman.State().GetBalance(acc) |
||||||
|
bh := trie.GetBalance(acc) |
||||||
|
|
||||||
|
if (bw != nil && bh == nil) || (bw == nil && bh != nil) { |
||||||
|
t.Errorf("test %d, account %d: balance mismatch: have %v, want %v", i, j, bh, bw) |
||||||
|
} |
||||||
|
if bw != nil && bh != nil && bw.Cmp(bw) != 0 { |
||||||
|
t.Errorf("test %d, account %d: balance mismatch: have %v, want %v", i, j, bh, bw) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Tests that the transaction receipts can be retrieved based on hashes.
|
||||||
|
func TestGetReceipt63(t *testing.T) { testGetReceipt(t, 63) } |
||||||
|
func TestGetReceipt64(t *testing.T) { testGetReceipt(t, 64) } |
||||||
|
|
||||||
|
func testGetReceipt(t *testing.T, protocol int) { |
||||||
|
// Define three accounts to simulate transactions with
|
||||||
|
acc1Key, _ := crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a") |
||||||
|
acc2Key, _ := crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee") |
||||||
|
acc1Addr := crypto.PubkeyToAddress(acc1Key.PublicKey) |
||||||
|
acc2Addr := crypto.PubkeyToAddress(acc2Key.PublicKey) |
||||||
|
|
||||||
|
// Create a chain generator with some simple transactions (blatantly stolen from @fjl/chain_makerts_test)
|
||||||
|
generator := func(i int, block *core.BlockGen) { |
||||||
|
switch i { |
||||||
|
case 0: |
||||||
|
// In block 1, the test bank sends account #1 some ether.
|
||||||
|
tx, _ := types.NewTransaction(block.TxNonce(testBankAddress), acc1Addr, big.NewInt(10000), params.TxGas, nil, nil).SignECDSA(testBankKey) |
||||||
|
block.AddTx(tx) |
||||||
|
case 1: |
||||||
|
// In block 2, the test bank sends some more ether to account #1.
|
||||||
|
// acc1Addr passes it on to account #2.
|
||||||
|
tx1, _ := types.NewTransaction(block.TxNonce(testBankAddress), acc1Addr, big.NewInt(1000), params.TxGas, nil, nil).SignECDSA(testBankKey) |
||||||
|
tx2, _ := types.NewTransaction(block.TxNonce(acc1Addr), acc2Addr, big.NewInt(1000), params.TxGas, nil, nil).SignECDSA(acc1Key) |
||||||
|
block.AddTx(tx1) |
||||||
|
block.AddTx(tx2) |
||||||
|
case 2: |
||||||
|
// Block 3 is empty but was mined by account #2.
|
||||||
|
block.SetCoinbase(acc2Addr) |
||||||
|
block.SetExtra([]byte("yeehaw")) |
||||||
|
case 3: |
||||||
|
// Block 4 includes blocks 2 and 3 as uncle headers (with modified extra data).
|
||||||
|
b2 := block.PrevBlock(1).Header() |
||||||
|
b2.Extra = []byte("foo") |
||||||
|
block.AddUncle(b2) |
||||||
|
b3 := block.PrevBlock(2).Header() |
||||||
|
b3.Extra = []byte("foo") |
||||||
|
block.AddUncle(b3) |
||||||
|
} |
||||||
|
} |
||||||
|
// Assemble the test environment
|
||||||
|
pm := newTestProtocolManager(4, generator, nil) |
||||||
|
peer, _ := newTestPeer("peer", protocol, pm, true) |
||||||
|
defer peer.close() |
||||||
|
|
||||||
|
// Collect the hashes to request, and the response to expect
|
||||||
|
hashes := []common.Hash{} |
||||||
|
for i := uint64(0); i <= pm.chainman.CurrentBlock().NumberU64(); i++ { |
||||||
|
for _, tx := range pm.chainman.GetBlockByNumber(i).Transactions() { |
||||||
|
hashes = append(hashes, tx.Hash()) |
||||||
|
} |
||||||
|
} |
||||||
|
receipts := make([]*types.Receipt, len(hashes)) |
||||||
|
for i, hash := range hashes { |
||||||
|
receipts[i] = core.GetReceipt(pm.chaindb, hash) |
||||||
|
} |
||||||
|
// Send the hash request and verify the response
|
||||||
|
p2p.Send(peer.app, 0x0f, hashes) |
||||||
|
if err := p2p.ExpectMsg(peer.app, 0x10, receipts); err != nil { |
||||||
|
t.Errorf("receipts mismatch: %v", err) |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,147 @@ |
|||||||
|
// This file contains some shares testing functionality, common to multiple
|
||||||
|
// different files and modules being tested.
|
||||||
|
|
||||||
|
package eth |
||||||
|
|
||||||
|
import ( |
||||||
|
"crypto/rand" |
||||||
|
"math/big" |
||||||
|
"sync" |
||||||
|
"testing" |
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common" |
||||||
|
"github.com/ethereum/go-ethereum/core" |
||||||
|
"github.com/ethereum/go-ethereum/core/types" |
||||||
|
"github.com/ethereum/go-ethereum/crypto" |
||||||
|
"github.com/ethereum/go-ethereum/ethdb" |
||||||
|
"github.com/ethereum/go-ethereum/event" |
||||||
|
"github.com/ethereum/go-ethereum/p2p" |
||||||
|
"github.com/ethereum/go-ethereum/p2p/discover" |
||||||
|
) |
||||||
|
|
||||||
|
var ( |
||||||
|
testBankKey, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") |
||||||
|
testBankAddress = crypto.PubkeyToAddress(testBankKey.PublicKey) |
||||||
|
testBankFunds = big.NewInt(1000000) |
||||||
|
) |
||||||
|
|
||||||
|
// newTestProtocolManager creates a new protocol manager for testing purposes,
|
||||||
|
// with the given number of blocks already known, and potential notification
|
||||||
|
// channels for different events.
|
||||||
|
func newTestProtocolManager(blocks int, generator func(int, *core.BlockGen), newtx chan<- []*types.Transaction) *ProtocolManager { |
||||||
|
var ( |
||||||
|
evmux = new(event.TypeMux) |
||||||
|
pow = new(core.FakePow) |
||||||
|
db, _ = ethdb.NewMemDatabase() |
||||||
|
genesis = core.WriteGenesisBlockForTesting(db, testBankAddress, testBankFunds) |
||||||
|
chainman, _ = core.NewChainManager(db, pow, evmux) |
||||||
|
blockproc = core.NewBlockProcessor(db, pow, chainman, evmux) |
||||||
|
) |
||||||
|
chainman.SetProcessor(blockproc) |
||||||
|
if _, err := chainman.InsertChain(core.GenerateChain(genesis, db, blocks, generator)); err != nil { |
||||||
|
panic(err) |
||||||
|
} |
||||||
|
pm := NewProtocolManager(NetworkId, evmux, &testTxPool{added: newtx}, pow, chainman, db) |
||||||
|
pm.Start() |
||||||
|
return pm |
||||||
|
} |
||||||
|
|
||||||
|
// testTxPool is a fake, helper transaction pool for testing purposes
|
||||||
|
type testTxPool struct { |
||||||
|
pool []*types.Transaction // Collection of all transactions
|
||||||
|
added chan<- []*types.Transaction // Notification channel for new transactions
|
||||||
|
|
||||||
|
lock sync.RWMutex // Protects the transaction pool
|
||||||
|
} |
||||||
|
|
||||||
|
// AddTransactions appends a batch of transactions to the pool, and notifies any
|
||||||
|
// listeners if the addition channel is non nil
|
||||||
|
func (p *testTxPool) AddTransactions(txs []*types.Transaction) { |
||||||
|
p.lock.Lock() |
||||||
|
defer p.lock.Unlock() |
||||||
|
|
||||||
|
p.pool = append(p.pool, txs...) |
||||||
|
if p.added != nil { |
||||||
|
p.added <- txs |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// GetTransactions returns all the transactions known to the pool
|
||||||
|
func (p *testTxPool) GetTransactions() types.Transactions { |
||||||
|
p.lock.RLock() |
||||||
|
defer p.lock.RUnlock() |
||||||
|
|
||||||
|
txs := make([]*types.Transaction, len(p.pool)) |
||||||
|
copy(txs, p.pool) |
||||||
|
|
||||||
|
return txs |
||||||
|
} |
||||||
|
|
||||||
|
// newTestTransaction create a new dummy transaction.
|
||||||
|
func newTestTransaction(from *crypto.Key, nonce uint64, datasize int) *types.Transaction { |
||||||
|
tx := types.NewTransaction(nonce, common.Address{}, big.NewInt(0), big.NewInt(100000), big.NewInt(0), make([]byte, datasize)) |
||||||
|
tx, _ = tx.SignECDSA(from.PrivateKey) |
||||||
|
|
||||||
|
return tx |
||||||
|
} |
||||||
|
|
||||||
|
// testPeer is a simulated peer to allow testing direct network calls.
|
||||||
|
type testPeer struct { |
||||||
|
net p2p.MsgReadWriter // Network layer reader/writer to simulate remote messaging
|
||||||
|
app *p2p.MsgPipeRW // Application layer reader/writer to simulate the local side
|
||||||
|
*peer |
||||||
|
} |
||||||
|
|
||||||
|
// newTestPeer creates a new peer registered at the given protocol manager.
|
||||||
|
func newTestPeer(name string, version int, pm *ProtocolManager, shake bool) (*testPeer, <-chan error) { |
||||||
|
// Create a message pipe to communicate through
|
||||||
|
app, net := p2p.MsgPipe() |
||||||
|
|
||||||
|
// Generate a random id and create the peer
|
||||||
|
var id discover.NodeID |
||||||
|
rand.Read(id[:]) |
||||||
|
|
||||||
|
peer := pm.newPeer(version, NetworkId, p2p.NewPeer(id, name, nil), net) |
||||||
|
|
||||||
|
// Start the peer on a new thread
|
||||||
|
errc := make(chan error, 1) |
||||||
|
go func() { |
||||||
|
pm.newPeerCh <- peer |
||||||
|
errc <- pm.handle(peer) |
||||||
|
}() |
||||||
|
tp := &testPeer{ |
||||||
|
app: app, |
||||||
|
net: net, |
||||||
|
peer: peer, |
||||||
|
} |
||||||
|
// Execute any implicitly requested handshakes and return
|
||||||
|
if shake { |
||||||
|
td, head, genesis := pm.chainman.Status() |
||||||
|
tp.handshake(nil, td, head, genesis) |
||||||
|
} |
||||||
|
return tp, errc |
||||||
|
} |
||||||
|
|
||||||
|
// handshake simulates a trivial handshake that expects the same state from the
|
||||||
|
// remote side as we are simulating locally.
|
||||||
|
func (p *testPeer) handshake(t *testing.T, td *big.Int, head common.Hash, genesis common.Hash) { |
||||||
|
msg := &statusData{ |
||||||
|
ProtocolVersion: uint32(p.version), |
||||||
|
NetworkId: uint32(NetworkId), |
||||||
|
TD: td, |
||||||
|
CurrentBlock: head, |
||||||
|
GenesisBlock: genesis, |
||||||
|
} |
||||||
|
if err := p2p.ExpectMsg(p.app, StatusMsg, msg); err != nil { |
||||||
|
t.Fatalf("status recv: %v", err) |
||||||
|
} |
||||||
|
if err := p2p.Send(p.app, StatusMsg, msg); err != nil { |
||||||
|
t.Fatalf("status send: %v", err) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// close terminates the local side of the peer, notifying the remote protocol
|
||||||
|
// manager of termination.
|
||||||
|
func (p *testPeer) close() { |
||||||
|
p.app.Close() |
||||||
|
} |
Loading…
Reference in new issue