| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | /* | 
					
						
							|  |  |  | 	This file is part of go-ethereum | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go-ethereum 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go-ethereum 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 General Public License for more details. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | 	along with go-ethereum.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @authors | 
					
						
							|  |  |  |  * 	Gustav Simonsson <gustav.simonsson@gmail.com> | 
					
						
							|  |  |  |  * @date 2015 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"encoding/hex" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	"flag" | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"log" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	"runtime" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/cmd/utils" | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core" | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	types "github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/ethdb" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/event" | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/rlp" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Account struct { | 
					
						
							|  |  |  | 	Balance string | 
					
						
							|  |  |  | 	Code    string | 
					
						
							|  |  |  | 	Nonce   string | 
					
						
							|  |  |  | 	Storage map[string]string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type BlockHeader struct { | 
					
						
							|  |  |  | 	Bloom            string | 
					
						
							|  |  |  | 	Coinbase         string | 
					
						
							|  |  |  | 	Difficulty       string | 
					
						
							|  |  |  | 	ExtraData        string | 
					
						
							|  |  |  | 	GasLimit         string | 
					
						
							|  |  |  | 	GasUsed          string | 
					
						
							|  |  |  | 	MixHash          string | 
					
						
							|  |  |  | 	Nonce            string | 
					
						
							|  |  |  | 	Number           string | 
					
						
							|  |  |  | 	ParentHash       string | 
					
						
							|  |  |  | 	ReceiptTrie      string | 
					
						
							|  |  |  | 	SeedHash         string | 
					
						
							|  |  |  | 	StateRoot        string | 
					
						
							|  |  |  | 	Timestamp        string | 
					
						
							|  |  |  | 	TransactionsTrie string | 
					
						
							|  |  |  | 	UncleHash        string | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | type Tx struct { | 
					
						
							|  |  |  | 	Data     string | 
					
						
							|  |  |  | 	GasLimit string | 
					
						
							|  |  |  | 	GasPrice string | 
					
						
							|  |  |  | 	Nonce    string | 
					
						
							|  |  |  | 	R        string | 
					
						
							|  |  |  | 	S        string | 
					
						
							|  |  |  | 	To       string | 
					
						
							|  |  |  | 	V        string | 
					
						
							|  |  |  | 	Value    string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Block struct { | 
					
						
							|  |  |  | 	BlockHeader  BlockHeader | 
					
						
							|  |  |  | 	Rlp          string | 
					
						
							|  |  |  | 	Transactions []Tx | 
					
						
							|  |  |  | 	UncleHeaders []string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Test struct { | 
					
						
							|  |  |  | 	Blocks             []Block | 
					
						
							|  |  |  | 	GenesisBlockHeader BlockHeader | 
					
						
							|  |  |  | 	Pre                map[string]Account | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func main() { | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	flag.Usage = func() { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, "%s <testfile>\n", os.Args[0]) | 
					
						
							|  |  |  | 		flag.PrintDefaults() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	runtime.GOMAXPROCS(runtime.NumCPU()) | 
					
						
							|  |  |  | 	logger.AddLogSystem(logger.NewStdLogSystem(os.Stderr, log.LstdFlags, logger.DebugDetailLevel)) | 
					
						
							|  |  |  | 	defer func() { logger.Flush() }() | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	if len(os.Args) < 2 { | 
					
						
							|  |  |  | 		utils.Fatalf("Please specify a test file as the first argument.") | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	blocks, err := loadBlocksFromTestFile(os.Args[1]) | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 		utils.Fatalf("Could not load blocks: %v", err) | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	chain := memchain() | 
					
						
							|  |  |  | 	chain.ResetWithGenesisBlock(blocks[0]) | 
					
						
							|  |  |  | 	if err = chain.InsertChain(types.Blocks{blocks[1]}); err != nil { | 
					
						
							|  |  |  | 		utils.Fatalf("Error: %v", err) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		fmt.Println("PASS") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | func memchain() *core.ChainManager { | 
					
						
							| 
									
										
										
										
											2015-03-09 23:25:46 +01:00
										 |  |  | 	blockdb, err := ethdb.NewMemDatabase() | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		utils.Fatalf("Could not create in-memory database: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-09 23:25:46 +01:00
										 |  |  | 	statedb, err := ethdb.NewMemDatabase() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		utils.Fatalf("Could not create in-memory database: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return core.NewChainManager(blockdb, statedb, new(event.TypeMux)) | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func loadBlocksFromTestFile(filePath string) (blocks types.Blocks, err error) { | 
					
						
							|  |  |  | 	fileContent, err := ioutil.ReadFile(filePath) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-09 16:58:06 +01:00
										 |  |  | 	bt := make(map[string]Test) | 
					
						
							|  |  |  | 	if err = json.Unmarshal(fileContent, &bt); err != nil { | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TODO: support multiple blocks; loop over all blocks | 
					
						
							|  |  |  | 	gbh := new(types.Header) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Let's use slighlty different namings for the same things, because that's awesome. | 
					
						
							|  |  |  | 	gbh.ParentHash, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.ParentHash) | 
					
						
							|  |  |  | 	gbh.UncleHash, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.UncleHash) | 
					
						
							|  |  |  | 	gbh.Coinbase, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.Coinbase) | 
					
						
							|  |  |  | 	gbh.Root, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.StateRoot) | 
					
						
							|  |  |  | 	gbh.TxHash, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.TransactionsTrie) | 
					
						
							|  |  |  | 	gbh.ReceiptHash, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.ReceiptTrie) | 
					
						
							|  |  |  | 	gbh.Bloom, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.Bloom) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gbh.MixDigest, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.MixHash) | 
					
						
							| 
									
										
										
										
											2015-03-14 23:39:45 +01:00
										 |  |  | 	//gbh.SeedHash, err = hex_decode(bt["SimpleTx"].GenesisBlockHeader.SeedHash) | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	d, _ := new(big.Int).SetString(bt["SimpleTx"].GenesisBlockHeader.Difficulty, 10) | 
					
						
							|  |  |  | 	gbh.Difficulty = d | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	n, _ := new(big.Int).SetString(bt["SimpleTx"].GenesisBlockHeader.Number, 10) | 
					
						
							|  |  |  | 	gbh.Number = n | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gl, _ := new(big.Int).SetString(bt["SimpleTx"].GenesisBlockHeader.GasLimit, 10) | 
					
						
							|  |  |  | 	gbh.GasLimit = gl | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gu, _ := new(big.Int).SetString(bt["SimpleTx"].GenesisBlockHeader.GasUsed, 10) | 
					
						
							|  |  |  | 	gbh.GasUsed = gu | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ts, _ := new(big.Int).SetString(bt["SimpleTx"].GenesisBlockHeader.Timestamp, 0) | 
					
						
							|  |  |  | 	gbh.Time = ts.Uint64() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	extra, err := hex_decode(bt["SimpleTx"].GenesisBlockHeader.ExtraData) | 
					
						
							|  |  |  | 	gbh.Extra = string(extra) // TODO: change ExtraData to byte array | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nonce, _ := hex_decode(bt["SimpleTx"].GenesisBlockHeader.Nonce) | 
					
						
							|  |  |  | 	gbh.Nonce = nonce | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gb := types.NewBlockWithHeader(gbh) | 
					
						
							| 
									
										
										
										
											2015-03-06 03:25:57 +01:00
										 |  |  | 	//gb.uncles = *new([]*types.Header) | 
					
						
							|  |  |  | 	//gb.transactions = *new(types.Transactions) | 
					
						
							|  |  |  | 	gb.Td = new(big.Int) | 
					
						
							| 
									
										
										
										
											2015-03-03 18:41:51 +01:00
										 |  |  | 	gb.Reward = new(big.Int) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	testBlock := new(types.Block) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rlpBytes, err := hex_decode(bt["SimpleTx"].Blocks[0].Rlp) | 
					
						
							|  |  |  | 	err = rlp.Decode(bytes.NewReader(rlpBytes), &testBlock) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	blocks = types.Blocks{ | 
					
						
							|  |  |  | 		gb, | 
					
						
							|  |  |  | 		testBlock, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func hex_decode(s string) (res []byte, err error) { | 
					
						
							|  |  |  | 	return hex.DecodeString(strings.TrimPrefix(s, "0x")) | 
					
						
							|  |  |  | } |