| 
									
										
										
										
											2019-07-22 12:17:27 +03:00
										 |  |  | // Copyright 2018 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/>. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | package ethash | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							|  |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	"net/http/httptest" | 
					
						
							| 
									
										
										
										
											2021-03-26 18:30:10 +01:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/internal/testlog" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Tests whether remote HTTP servers are correctly notified of new work. | 
					
						
							|  |  |  | func TestRemoteNotify(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	// Start a simple web server to capture notifications. | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	sink := make(chan [3]string) | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { | 
					
						
							|  |  |  | 		blob, err := ioutil.ReadAll(req.Body) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to read miner notification: %v", err) | 
					
						
							| 
									
										
										
										
											2018-09-03 16:59:23 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 		var work [3]string | 
					
						
							|  |  |  | 		if err := json.Unmarshal(blob, &work); err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to unmarshal miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		sink <- work | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | 	defer server.Close() | 
					
						
							| 
									
										
										
										
											2018-09-03 16:59:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	// Create the custom ethash engine. | 
					
						
							|  |  |  | 	ethash := NewTester([]string{server.URL}, false) | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	defer ethash.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	// Stream a work task and ensure the notification bubbles out. | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	header := &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(100)} | 
					
						
							|  |  |  | 	block := types.NewBlockWithHeader(header) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 	ethash.Seal(nil, block, nil, nil) | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	select { | 
					
						
							|  |  |  | 	case work := <-sink: | 
					
						
							| 
									
										
										
										
											2018-08-23 21:02:57 +08:00
										 |  |  | 		if want := ethash.SealHash(header).Hex(); work[0] != want { | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 			t.Errorf("work packet hash mismatch: have %s, want %s", work[0], want) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if want := common.BytesToHash(SeedHash(header.Number.Uint64())).Hex(); work[1] != want { | 
					
						
							|  |  |  | 			t.Errorf("work packet seed mismatch: have %s, want %s", work[1], want) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		target := new(big.Int).Div(new(big.Int).Lsh(big.NewInt(1), 256), header.Difficulty) | 
					
						
							|  |  |  | 		if want := common.BytesToHash(target.Bytes()).Hex(); work[2] != want { | 
					
						
							|  |  |  | 			t.Errorf("work packet target mismatch: have %s, want %s", work[2], want) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-09-03 16:59:23 +02:00
										 |  |  | 	case <-time.After(3 * time.Second): | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 		t.Fatalf("notification timed out") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 18:30:10 +01:00
										 |  |  | // Tests whether remote HTTP servers are correctly notified of new work. (Full pending block body / --miner.notify.full) | 
					
						
							|  |  |  | func TestRemoteNotifyFull(t *testing.T) { | 
					
						
							|  |  |  | 	// Start a simple web server to capture notifications. | 
					
						
							|  |  |  | 	sink := make(chan map[string]interface{}) | 
					
						
							|  |  |  | 	server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { | 
					
						
							|  |  |  | 		blob, err := ioutil.ReadAll(req.Body) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to read miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var work map[string]interface{} | 
					
						
							|  |  |  | 		if err := json.Unmarshal(blob, &work); err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to unmarshal miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		sink <- work | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | 	defer server.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create the custom ethash engine. | 
					
						
							|  |  |  | 	config := Config{ | 
					
						
							|  |  |  | 		PowMode:    ModeTest, | 
					
						
							|  |  |  | 		NotifyFull: true, | 
					
						
							|  |  |  | 		Log:        testlog.Logger(t, log.LvlWarn), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ethash := New(config, []string{server.URL}, false) | 
					
						
							|  |  |  | 	defer ethash.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Stream a work task and ensure the notification bubbles out. | 
					
						
							|  |  |  | 	header := &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(100)} | 
					
						
							|  |  |  | 	block := types.NewBlockWithHeader(header) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ethash.Seal(nil, block, nil, nil) | 
					
						
							|  |  |  | 	select { | 
					
						
							|  |  |  | 	case work := <-sink: | 
					
						
							|  |  |  | 		if want := "0x" + strconv.FormatUint(header.Number.Uint64(), 16); work["number"] != want { | 
					
						
							|  |  |  | 			t.Errorf("pending block number mismatch: have %v, want %v", work["number"], want) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if want := "0x" + header.Difficulty.Text(16); work["difficulty"] != want { | 
					
						
							|  |  |  | 			t.Errorf("pending block difficulty mismatch: have %s, want %s", work["difficulty"], want) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	case <-time.After(3 * time.Second): | 
					
						
							|  |  |  | 		t.Fatalf("notification timed out") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | // Tests that pushing work packages fast to the miner doesn't cause any data race | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | // issues in the notifications. | 
					
						
							|  |  |  | func TestRemoteMultiNotify(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	// Start a simple web server to capture notifications. | 
					
						
							| 
									
										
										
										
											2018-08-17 18:12:39 +03:00
										 |  |  | 	sink := make(chan [3]string, 64) | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { | 
					
						
							|  |  |  | 		blob, err := ioutil.ReadAll(req.Body) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to read miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var work [3]string | 
					
						
							|  |  |  | 		if err := json.Unmarshal(blob, &work); err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to unmarshal miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		sink <- work | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | 	defer server.Close() | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	// Create the custom ethash engine. | 
					
						
							|  |  |  | 	ethash := NewTester([]string{server.URL}, false) | 
					
						
							|  |  |  | 	ethash.config.Log = testlog.Logger(t, log.LvlWarn) | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	defer ethash.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-25 11:01:03 -05:00
										 |  |  | 	// Provide a results reader. | 
					
						
							|  |  |  | 	// Otherwise the unread results will be logged asynchronously | 
					
						
							|  |  |  | 	// and this can happen after the test is finished, causing a panic. | 
					
						
							|  |  |  | 	results := make(chan *types.Block, cap(sink)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 	// Stream a lot of work task and ensure all the notifications bubble out. | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	for i := 0; i < cap(sink); i++ { | 
					
						
							|  |  |  | 		header := &types.Header{Number: big.NewInt(int64(i)), Difficulty: big.NewInt(100)} | 
					
						
							|  |  |  | 		block := types.NewBlockWithHeader(header) | 
					
						
							| 
									
										
										
										
											2020-05-25 11:01:03 -05:00
										 |  |  | 		ethash.Seal(nil, block, results, nil) | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 	for i := 0; i < cap(sink); i++ { | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-sink: | 
					
						
							| 
									
										
										
										
											2020-05-25 11:01:03 -05:00
										 |  |  | 			<-results | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 		case <-time.After(10 * time.Second): | 
					
						
							| 
									
										
										
										
											2018-08-08 12:15:08 +03:00
										 |  |  | 			t.Fatalf("notification %d timed out", i) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 18:30:10 +01:00
										 |  |  | // Tests that pushing work packages fast to the miner doesn't cause any data race | 
					
						
							|  |  |  | // issues in the notifications. Full pending block body / --miner.notify.full) | 
					
						
							|  |  |  | func TestRemoteMultiNotifyFull(t *testing.T) { | 
					
						
							|  |  |  | 	// Start a simple web server to capture notifications. | 
					
						
							|  |  |  | 	sink := make(chan map[string]interface{}, 64) | 
					
						
							|  |  |  | 	server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { | 
					
						
							|  |  |  | 		blob, err := ioutil.ReadAll(req.Body) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to read miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var work map[string]interface{} | 
					
						
							|  |  |  | 		if err := json.Unmarshal(blob, &work); err != nil { | 
					
						
							|  |  |  | 			t.Errorf("failed to unmarshal miner notification: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		sink <- work | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | 	defer server.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create the custom ethash engine. | 
					
						
							|  |  |  | 	config := Config{ | 
					
						
							|  |  |  | 		PowMode:    ModeTest, | 
					
						
							|  |  |  | 		NotifyFull: true, | 
					
						
							|  |  |  | 		Log:        testlog.Logger(t, log.LvlWarn), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ethash := New(config, []string{server.URL}, false) | 
					
						
							|  |  |  | 	defer ethash.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Provide a results reader. | 
					
						
							|  |  |  | 	// Otherwise the unread results will be logged asynchronously | 
					
						
							|  |  |  | 	// and this can happen after the test is finished, causing a panic. | 
					
						
							|  |  |  | 	results := make(chan *types.Block, cap(sink)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Stream a lot of work task and ensure all the notifications bubble out. | 
					
						
							|  |  |  | 	for i := 0; i < cap(sink); i++ { | 
					
						
							|  |  |  | 		header := &types.Header{Number: big.NewInt(int64(i)), Difficulty: big.NewInt(100)} | 
					
						
							|  |  |  | 		block := types.NewBlockWithHeader(header) | 
					
						
							|  |  |  | 		ethash.Seal(nil, block, results, nil) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < cap(sink); i++ { | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-sink: | 
					
						
							|  |  |  | 			<-results | 
					
						
							|  |  |  | 		case <-time.After(10 * time.Second): | 
					
						
							|  |  |  | 			t.Fatalf("notification %d timed out", i) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | // Tests whether stale solutions are correctly processed. | 
					
						
							|  |  |  | func TestStaleSubmission(t *testing.T) { | 
					
						
							|  |  |  | 	ethash := NewTester(nil, true) | 
					
						
							|  |  |  | 	defer ethash.Close() | 
					
						
							|  |  |  | 	api := &API{ethash} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fakeNonce, fakeDigest := types.BlockNonce{0x01, 0x02, 0x03}, common.HexToHash("deadbeef") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	testcases := []struct { | 
					
						
							|  |  |  | 		headers     []*types.Header | 
					
						
							|  |  |  | 		submitIndex int | 
					
						
							|  |  |  | 		submitRes   bool | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		// Case1: submit solution for the latest mining package | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			[]*types.Header{ | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xa}), Number: big.NewInt(1), Difficulty: big.NewInt(100000000)}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			0, | 
					
						
							|  |  |  | 			true, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		// Case2: submit solution for the previous package but have same parent. | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			[]*types.Header{ | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xb}), Number: big.NewInt(2), Difficulty: big.NewInt(100000000)}, | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xb}), Number: big.NewInt(2), Difficulty: big.NewInt(100000001)}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			0, | 
					
						
							|  |  |  | 			true, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		// Case3: submit stale but acceptable solution | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			[]*types.Header{ | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xc}), Number: big.NewInt(3), Difficulty: big.NewInt(100000000)}, | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xd}), Number: big.NewInt(9), Difficulty: big.NewInt(100000000)}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			0, | 
					
						
							|  |  |  | 			true, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		// Case4: submit very old solution | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			[]*types.Header{ | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xe}), Number: big.NewInt(10), Difficulty: big.NewInt(100000000)}, | 
					
						
							|  |  |  | 				{ParentHash: common.BytesToHash([]byte{0xf}), Number: big.NewInt(17), Difficulty: big.NewInt(100000000)}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			0, | 
					
						
							|  |  |  | 			false, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	results := make(chan *types.Block, 16) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for id, c := range testcases { | 
					
						
							|  |  |  | 		for _, h := range c.headers { | 
					
						
							|  |  |  | 			ethash.Seal(nil, types.NewBlockWithHeader(h), results, nil) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if res := api.SubmitWork(fakeNonce, ethash.SealHash(c.headers[c.submitIndex]), fakeDigest); res != c.submitRes { | 
					
						
							|  |  |  | 			t.Errorf("case %d submit result mismatch, want %t, get %t", id+1, c.submitRes, res) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !c.submitRes { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case res := <-results: | 
					
						
							|  |  |  | 			if res.Header().Nonce != fakeNonce { | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 				t.Errorf("case %d block nonce mismatch, want %x, get %x", id+1, fakeNonce, res.Header().Nonce) | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if res.Header().MixDigest != fakeDigest { | 
					
						
							| 
									
										
										
										
											2019-11-28 10:51:57 +01:00
										 |  |  | 				t.Errorf("case %d block digest mismatch, want %x, get %x", id+1, fakeDigest, res.Header().MixDigest) | 
					
						
							| 
									
										
										
										
											2018-08-28 21:59:05 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if res.Header().Difficulty.Uint64() != c.headers[c.submitIndex].Difficulty.Uint64() { | 
					
						
							|  |  |  | 				t.Errorf("case %d block difficulty mismatch, want %d, get %d", id+1, c.headers[c.submitIndex].Difficulty, res.Header().Difficulty) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if res.Header().Number.Uint64() != c.headers[c.submitIndex].Number.Uint64() { | 
					
						
							|  |  |  | 				t.Errorf("case %d block number mismatch, want %d, get %d", id+1, c.headers[c.submitIndex].Number.Uint64(), res.Header().Number.Uint64()) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if res.Header().ParentHash != c.headers[c.submitIndex].ParentHash { | 
					
						
							|  |  |  | 				t.Errorf("case %d block parent hash mismatch, want %s, get %s", id+1, c.headers[c.submitIndex].ParentHash.Hex(), res.Header().ParentHash.Hex()) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		case <-time.NewTimer(time.Second).C: | 
					
						
							|  |  |  | 			t.Errorf("case %d fetch ethash result timeout", id+1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |