gitee.com/liu-zhao234568/cntest@v1.0.0/eth/catalyst/api_test.go (about) 1 // Copyright 2020 The go-ethereum Authors 2 // This file is part of the go-ethereum library. 3 // 4 // The go-ethereum library is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU Lesser General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // The go-ethereum library is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU Lesser General Public License for more details. 13 // 14 // You should have received a copy of the GNU Lesser General Public License 15 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 16 17 package catalyst 18 19 import ( 20 "math/big" 21 "testing" 22 23 "gitee.com/liu-zhao234568/cntest/consensus/ethash" 24 "gitee.com/liu-zhao234568/cntest/core" 25 "gitee.com/liu-zhao234568/cntest/core/rawdb" 26 "gitee.com/liu-zhao234568/cntest/core/types" 27 "gitee.com/liu-zhao234568/cntest/crypto" 28 "gitee.com/liu-zhao234568/cntest/eth" 29 "gitee.com/liu-zhao234568/cntest/eth/ethconfig" 30 "gitee.com/liu-zhao234568/cntest/node" 31 "gitee.com/liu-zhao234568/cntest/params" 32 ) 33 34 var ( 35 // testKey is a private key to use for funding a tester account. 36 testKey, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") 37 38 // testAddr is the Ethereum address of the tester account. 39 testAddr = crypto.PubkeyToAddress(testKey.PublicKey) 40 41 testBalance = big.NewInt(2e15) 42 ) 43 44 func generateTestChain() (*core.Genesis, []*types.Block) { 45 db := rawdb.NewMemoryDatabase() 46 config := params.AllEthashProtocolChanges 47 genesis := &core.Genesis{ 48 Config: config, 49 Alloc: core.GenesisAlloc{testAddr: {Balance: testBalance}}, 50 ExtraData: []byte("test genesis"), 51 Timestamp: 9000, 52 BaseFee: big.NewInt(params.InitialBaseFee), 53 } 54 generate := func(i int, g *core.BlockGen) { 55 g.OffsetTime(5) 56 g.SetExtra([]byte("test")) 57 } 58 gblock := genesis.ToBlock(db) 59 engine := ethash.NewFaker() 60 blocks, _ := core.GenerateChain(config, gblock, engine, db, 10, generate) 61 blocks = append([]*types.Block{gblock}, blocks...) 62 return genesis, blocks 63 } 64 65 func generateTestChainWithFork(n int, fork int) (*core.Genesis, []*types.Block, []*types.Block) { 66 if fork >= n { 67 fork = n - 1 68 } 69 db := rawdb.NewMemoryDatabase() 70 config := ¶ms.ChainConfig{ 71 ChainID: big.NewInt(1337), 72 HomesteadBlock: big.NewInt(0), 73 EIP150Block: big.NewInt(0), 74 EIP155Block: big.NewInt(0), 75 EIP158Block: big.NewInt(0), 76 ByzantiumBlock: big.NewInt(0), 77 ConstantinopleBlock: big.NewInt(0), 78 PetersburgBlock: big.NewInt(0), 79 IstanbulBlock: big.NewInt(0), 80 MuirGlacierBlock: big.NewInt(0), 81 BerlinBlock: big.NewInt(0), 82 LondonBlock: big.NewInt(0), 83 CatalystBlock: big.NewInt(0), 84 Ethash: new(params.EthashConfig), 85 } 86 genesis := &core.Genesis{ 87 Config: config, 88 Alloc: core.GenesisAlloc{testAddr: {Balance: testBalance}}, 89 ExtraData: []byte("test genesis"), 90 Timestamp: 9000, 91 BaseFee: big.NewInt(params.InitialBaseFee), 92 } 93 generate := func(i int, g *core.BlockGen) { 94 g.OffsetTime(5) 95 g.SetExtra([]byte("test")) 96 } 97 generateFork := func(i int, g *core.BlockGen) { 98 g.OffsetTime(5) 99 g.SetExtra([]byte("testF")) 100 } 101 gblock := genesis.ToBlock(db) 102 engine := ethash.NewFaker() 103 blocks, _ := core.GenerateChain(config, gblock, engine, db, n, generate) 104 blocks = append([]*types.Block{gblock}, blocks...) 105 forkedBlocks, _ := core.GenerateChain(config, blocks[fork], engine, db, n-fork, generateFork) 106 return genesis, blocks, forkedBlocks 107 } 108 109 func TestEth2AssembleBlock(t *testing.T) { 110 genesis, blocks := generateTestChain() 111 n, ethservice := startEthService(t, genesis, blocks[1:9]) 112 defer n.Close() 113 114 api := newConsensusAPI(ethservice) 115 signer := types.NewEIP155Signer(ethservice.BlockChain().Config().ChainID) 116 tx, err := types.SignTx(types.NewTransaction(0, blocks[8].Coinbase(), big.NewInt(1000), params.TxGas, big.NewInt(params.InitialBaseFee), nil), signer, testKey) 117 if err != nil { 118 t.Fatalf("error signing transaction, err=%v", err) 119 } 120 ethservice.TxPool().AddLocal(tx) 121 blockParams := assembleBlockParams{ 122 ParentHash: blocks[8].ParentHash(), 123 Timestamp: blocks[8].Time(), 124 } 125 execData, err := api.AssembleBlock(blockParams) 126 127 if err != nil { 128 t.Fatalf("error producing block, err=%v", err) 129 } 130 131 if len(execData.Transactions) != 1 { 132 t.Fatalf("invalid number of transactions %d != 1", len(execData.Transactions)) 133 } 134 } 135 136 func TestEth2AssembleBlockWithAnotherBlocksTxs(t *testing.T) { 137 genesis, blocks := generateTestChain() 138 n, ethservice := startEthService(t, genesis, blocks[1:9]) 139 defer n.Close() 140 141 api := newConsensusAPI(ethservice) 142 143 // Put the 10th block's tx in the pool and produce a new block 144 api.addBlockTxs(blocks[9]) 145 blockParams := assembleBlockParams{ 146 ParentHash: blocks[9].ParentHash(), 147 Timestamp: blocks[9].Time(), 148 } 149 execData, err := api.AssembleBlock(blockParams) 150 if err != nil { 151 t.Fatalf("error producing block, err=%v", err) 152 } 153 154 if len(execData.Transactions) != blocks[9].Transactions().Len() { 155 t.Fatalf("invalid number of transactions %d != 1", len(execData.Transactions)) 156 } 157 } 158 159 func TestEth2NewBlock(t *testing.T) { 160 genesis, blocks, forkedBlocks := generateTestChainWithFork(10, 4) 161 n, ethservice := startEthService(t, genesis, blocks[1:5]) 162 defer n.Close() 163 164 api := newConsensusAPI(ethservice) 165 for i := 5; i < 10; i++ { 166 p := executableData{ 167 ParentHash: ethservice.BlockChain().CurrentBlock().Hash(), 168 Miner: blocks[i].Coinbase(), 169 StateRoot: blocks[i].Root(), 170 GasLimit: blocks[i].GasLimit(), 171 GasUsed: blocks[i].GasUsed(), 172 Transactions: encodeTransactions(blocks[i].Transactions()), 173 ReceiptRoot: blocks[i].ReceiptHash(), 174 LogsBloom: blocks[i].Bloom().Bytes(), 175 BlockHash: blocks[i].Hash(), 176 Timestamp: blocks[i].Time(), 177 Number: uint64(i), 178 } 179 success, err := api.NewBlock(p) 180 if err != nil || !success.Valid { 181 t.Fatalf("Failed to insert block: %v", err) 182 } 183 } 184 185 exp := ethservice.BlockChain().CurrentBlock().Hash() 186 187 // Introduce the fork point. 188 lastBlockNum := blocks[4].Number() 189 lastBlock := blocks[4] 190 for i := 0; i < 4; i++ { 191 lastBlockNum.Add(lastBlockNum, big.NewInt(1)) 192 p := executableData{ 193 ParentHash: lastBlock.Hash(), 194 Miner: forkedBlocks[i].Coinbase(), 195 StateRoot: forkedBlocks[i].Root(), 196 Number: lastBlockNum.Uint64(), 197 GasLimit: forkedBlocks[i].GasLimit(), 198 GasUsed: forkedBlocks[i].GasUsed(), 199 Transactions: encodeTransactions(blocks[i].Transactions()), 200 ReceiptRoot: forkedBlocks[i].ReceiptHash(), 201 LogsBloom: forkedBlocks[i].Bloom().Bytes(), 202 BlockHash: forkedBlocks[i].Hash(), 203 Timestamp: forkedBlocks[i].Time(), 204 } 205 success, err := api.NewBlock(p) 206 if err != nil || !success.Valid { 207 t.Fatalf("Failed to insert forked block #%d: %v", i, err) 208 } 209 lastBlock, err = insertBlockParamsToBlock(ethservice.BlockChain().Config(), lastBlock.Header(), p) 210 if err != nil { 211 t.Fatal(err) 212 } 213 } 214 215 if ethservice.BlockChain().CurrentBlock().Hash() != exp { 216 t.Fatalf("Wrong head after inserting fork %x != %x", exp, ethservice.BlockChain().CurrentBlock().Hash()) 217 } 218 } 219 220 // startEthService creates a full node instance for testing. 221 func startEthService(t *testing.T, genesis *core.Genesis, blocks []*types.Block) (*node.Node, *eth.Ethereum) { 222 t.Helper() 223 224 n, err := node.New(&node.Config{}) 225 if err != nil { 226 t.Fatal("can't create node:", err) 227 } 228 229 ethcfg := ðconfig.Config{Genesis: genesis, Ethash: ethash.Config{PowMode: ethash.ModeFake}} 230 ethservice, err := eth.New(n, ethcfg) 231 if err != nil { 232 t.Fatal("can't create eth service:", err) 233 } 234 if err := n.Start(); err != nil { 235 t.Fatal("can't start node:", err) 236 } 237 if _, err := ethservice.BlockChain().InsertChain(blocks); err != nil { 238 n.Close() 239 t.Fatal("can't import test blocks:", err) 240 } 241 ethservice.SetEtherbase(testAddr) 242 243 return n, ethservice 244 }