|
|
@@ -19,8 +19,10 @@ package core
|
|
|
import (
|
|
|
"encoding/json"
|
|
|
"fmt"
|
|
|
+ "io"
|
|
|
+ "io/ioutil"
|
|
|
"math/big"
|
|
|
- "os"
|
|
|
+ "strings"
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
"github.com/ethereum/go-ethereum/core/state"
|
|
|
@@ -28,51 +30,71 @@ import (
|
|
|
"github.com/ethereum/go-ethereum/params"
|
|
|
)
|
|
|
|
|
|
-// GenesisBlock creates a genesis block with the given nonce.
|
|
|
-func GenesisBlock(nonce uint64, db common.Database) *types.Block {
|
|
|
- var accounts map[string]struct {
|
|
|
- Balance string
|
|
|
- Code string
|
|
|
- }
|
|
|
- err := json.Unmarshal(GenesisAccounts, &accounts)
|
|
|
+// WriteGenesisBlock writes the genesis block to the database as block number 0
|
|
|
+func WriteGenesisBlock(stateDb, blockDb common.Database, reader io.Reader) (*types.Block, error) {
|
|
|
+ contents, err := ioutil.ReadAll(reader)
|
|
|
if err != nil {
|
|
|
- fmt.Println("unable to decode genesis json data:", err)
|
|
|
- os.Exit(1)
|
|
|
+ return nil, err
|
|
|
}
|
|
|
- statedb := state.New(common.Hash{}, db)
|
|
|
- for addr, account := range accounts {
|
|
|
- codedAddr := common.Hex2Bytes(addr)
|
|
|
- accountState := statedb.CreateAccount(common.BytesToAddress(codedAddr))
|
|
|
- accountState.SetBalance(common.Big(account.Balance))
|
|
|
- accountState.SetCode(common.FromHex(account.Code))
|
|
|
- statedb.UpdateStateObject(accountState)
|
|
|
+
|
|
|
+ var genesis struct {
|
|
|
+ Nonce string
|
|
|
+ Timestamp string
|
|
|
+ ParentHash string
|
|
|
+ ExtraData string
|
|
|
+ GasLimit string
|
|
|
+ Difficulty string
|
|
|
+ Mixhash string
|
|
|
+ Coinbase string
|
|
|
+ Alloc map[string]struct {
|
|
|
+ Code string
|
|
|
+ Storage map[string]string
|
|
|
+ Balance string
|
|
|
+ }
|
|
|
}
|
|
|
- statedb.Sync()
|
|
|
|
|
|
+ if err := json.Unmarshal(contents, &genesis); err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+
|
|
|
+ statedb := state.New(common.Hash{}, stateDb)
|
|
|
+ for addr, account := range genesis.Alloc {
|
|
|
+ address := common.HexToAddress(addr)
|
|
|
+ statedb.AddBalance(address, common.String2Big(account.Balance))
|
|
|
+ statedb.SetCode(address, common.Hex2Bytes(account.Code))
|
|
|
+ for key, value := range account.Storage {
|
|
|
+ statedb.SetState(address, common.HexToHash(key), common.HexToHash(value))
|
|
|
+ }
|
|
|
+ }
|
|
|
+ statedb.SyncObjects()
|
|
|
+
|
|
|
+ difficulty := common.String2Big(genesis.Difficulty)
|
|
|
block := types.NewBlock(&types.Header{
|
|
|
- Difficulty: params.GenesisDifficulty,
|
|
|
- GasLimit: params.GenesisGasLimit,
|
|
|
- Nonce: types.EncodeNonce(nonce),
|
|
|
+ Nonce: types.EncodeNonce(common.String2Big(genesis.Nonce).Uint64()),
|
|
|
+ Time: common.String2Big(genesis.Timestamp).Uint64(),
|
|
|
+ ParentHash: common.HexToHash(genesis.ParentHash),
|
|
|
+ Extra: common.Hex2Bytes(genesis.ExtraData),
|
|
|
+ GasLimit: common.String2Big(genesis.GasLimit),
|
|
|
+ Difficulty: difficulty,
|
|
|
+ MixDigest: common.HexToHash(genesis.Mixhash),
|
|
|
+ Coinbase: common.HexToAddress(genesis.Coinbase),
|
|
|
Root: statedb.Root(),
|
|
|
}, nil, nil, nil)
|
|
|
- block.Td = params.GenesisDifficulty
|
|
|
- return block
|
|
|
-}
|
|
|
+ block.Td = difficulty
|
|
|
|
|
|
-var GenesisAccounts = []byte(`{
|
|
|
- "0000000000000000000000000000000000000001": {"balance": "1"},
|
|
|
- "0000000000000000000000000000000000000002": {"balance": "1"},
|
|
|
- "0000000000000000000000000000000000000003": {"balance": "1"},
|
|
|
- "0000000000000000000000000000000000000004": {"balance": "1"},
|
|
|
- "dbdbdb2cbd23b783741e8d7fcf51e459b497e4a6": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "e4157b34ea9615cfbde6b4fda419828124b70c78": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "b9c015918bdaba24b4ff057a92a3873d6eb201be": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "6c386a4b26f73c802f34673f7248bb118f97424a": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "cd2a3d9f938e13cd947ec05abc7fe734df8dd826": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "2ef47100e0787b915105fd5e3f4ff6752079d5cb": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "e6716f9544a56c530d868e4bfbacb172315bdead": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
- "1a26338f0d905e295fccb71fa9ea849ffa12aaf4": {"balance": "1606938044258990275541962092341162602522202993782792835301376"}
|
|
|
-}`)
|
|
|
+ statedb.Sync()
|
|
|
+
|
|
|
+ err = WriteBlock(blockDb, block)
|
|
|
+ if err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+ err = WriteHead(blockDb, block)
|
|
|
+ if err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+
|
|
|
+ return block, nil
|
|
|
+}
|
|
|
|
|
|
// GenesisBlockForTesting creates a block in which addr has the given wei balance.
|
|
|
// The state trie of the block is written to db.
|
|
|
@@ -90,3 +112,39 @@ func GenesisBlockForTesting(db common.Database, addr common.Address, balance *bi
|
|
|
block.Td = params.GenesisDifficulty
|
|
|
return block
|
|
|
}
|
|
|
+
|
|
|
+func WriteGenesisBlockForTesting(db common.Database, addr common.Address, balance *big.Int) *types.Block {
|
|
|
+ testGenesis := fmt.Sprintf(`{
|
|
|
+ "nonce":"0x%x",
|
|
|
+ "gasLimit":"0x%x",
|
|
|
+ "difficulty":"0x%x",
|
|
|
+ "alloc": {
|
|
|
+ "0x%x":{"balance":"0x%x"}
|
|
|
+ }
|
|
|
+}`, types.EncodeNonce(0), params.GenesisGasLimit.Bytes(), params.GenesisDifficulty.Bytes(), addr, balance.Bytes())
|
|
|
+ block, _ := WriteGenesisBlock(db, db, strings.NewReader(testGenesis))
|
|
|
+ return block
|
|
|
+}
|
|
|
+
|
|
|
+func WriteTestNetGenesisBlock(stateDb, blockDb common.Database, nonce uint64) (*types.Block, error) {
|
|
|
+ testGenesis := fmt.Sprintf(`{
|
|
|
+ "nonce":"0x%x",
|
|
|
+ "gasLimit":"0x%x",
|
|
|
+ "difficulty":"0x%x",
|
|
|
+ "alloc": {
|
|
|
+ "0000000000000000000000000000000000000001": {"balance": "1"},
|
|
|
+ "0000000000000000000000000000000000000002": {"balance": "1"},
|
|
|
+ "0000000000000000000000000000000000000003": {"balance": "1"},
|
|
|
+ "0000000000000000000000000000000000000004": {"balance": "1"},
|
|
|
+ "dbdbdb2cbd23b783741e8d7fcf51e459b497e4a6": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "e4157b34ea9615cfbde6b4fda419828124b70c78": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "b9c015918bdaba24b4ff057a92a3873d6eb201be": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "6c386a4b26f73c802f34673f7248bb118f97424a": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "cd2a3d9f938e13cd947ec05abc7fe734df8dd826": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "2ef47100e0787b915105fd5e3f4ff6752079d5cb": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "e6716f9544a56c530d868e4bfbacb172315bdead": {"balance": "1606938044258990275541962092341162602522202993782792835301376"},
|
|
|
+ "1a26338f0d905e295fccb71fa9ea849ffa12aaf4": {"balance": "1606938044258990275541962092341162602522202993782792835301376"}
|
|
|
+ }
|
|
|
+}`, types.EncodeNonce(nonce), params.GenesisGasLimit.Bytes(), params.GenesisDifficulty.Bytes())
|
|
|
+ return WriteGenesisBlock(stateDb, blockDb, strings.NewReader(testGenesis))
|
|
|
+}
|