package tests import ( "bytes" "context" "encoding/hex" "fmt" "math/big" "reflect" "strings" "testing" "time" "github.com/ethereum/go-ethereum/accounts/abi/bind" "github.com/ethereum/go-ethereum/accounts/abi/bind/backends" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/consensus/XDPoS" contractValidator "github.com/ethereum/go-ethereum/contracts/validator/contract" "github.com/ethereum/go-ethereum/core" . "github.com/ethereum/go-ethereum/core" "github.com/ethereum/go-ethereum/core/types" "github.com/ethereum/go-ethereum/core/vm" "github.com/ethereum/go-ethereum/crypto" "github.com/ethereum/go-ethereum/log" "github.com/ethereum/go-ethereum/rlp" ) type masterNodes map[string]big.Int type signersList map[string]bool const GAP = int(450) var ( acc1Key, _ = crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a") acc2Key, _ = crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee") acc3Key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") voterKey, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee04aefe388d1e14474d32c45c72ce7b7a") acc1Addr = crypto.PubkeyToAddress(acc1Key.PublicKey) //xdc703c4b2bD70c169f5717101CaeE543299Fc946C7 acc2Addr = crypto.PubkeyToAddress(acc2Key.PublicKey) //xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e acc3Addr = crypto.PubkeyToAddress(acc3Key.PublicKey) //xdc71562b71999873DB5b286dF957af199Ec94617F7 voterAddr = crypto.PubkeyToAddress(voterKey.PublicKey) //xdc5F74529C0338546f82389402a01c31fB52c6f434 chainID = int64(1337) ) func debugMessage(backend *backends.SimulatedBackend, signers signersList, t *testing.T) { ms := GetCandidateFromCurrentSmartContract(backend, t) fmt.Println("=== current smart contract") for nodeAddr, cap := range ms { if !strings.Contains(nodeAddr, "000000000000000000000000000000000000") { //remove defaults fmt.Println(nodeAddr, cap) } } fmt.Println("=== this block signer list") for signer := range signers { if !strings.Contains(signer, "000000000000000000000000000000000000") { //remove defaults fmt.Println(signer) } } } func getCommonBackend(t *testing.T) *backends.SimulatedBackend { // initial helper backend contractBackendForSC := backends.NewXDCSimulatedBackend(core.GenesisAlloc{ voterAddr: {Balance: new(big.Int).SetUint64(10000000000)}, }, 10000000) transactOpts := bind.NewKeyedTransactor(voterKey) var candidates []common.Address var caps []*big.Int defalutCap := new(big.Int) defalutCap.SetString("1000000000", 10) for i := 1; i <= 16; i++ { addr := fmt.Sprintf("%02d", i) candidates = append(candidates, common.StringToAddress(addr)) // StringToAddress does not exist caps = append(caps, defalutCap) } acc1Cap, acc2Cap, acc3Cap, voterCap := new(big.Int), new(big.Int), new(big.Int), new(big.Int) acc1Cap.SetString("10000001", 10) acc2Cap.SetString("10000002", 10) acc3Cap.SetString("10000003", 10) voterCap.SetString("1000000000", 10) caps = append(caps, voterCap, acc1Cap, acc2Cap, acc3Cap) candidates = append(candidates, voterAddr, acc1Addr, acc2Addr, acc3Addr) // create validator smart contract validatorSCAddr, _, _, err := contractValidator.DeployXDCValidator( transactOpts, contractBackendForSC, candidates, caps, voterAddr, // first owner, not used big.NewInt(50000), big.NewInt(1), big.NewInt(99), big.NewInt(100), big.NewInt(100), ) if err != nil { t.Fatalf("can't deploy root registry: %v", err) } contractBackendForSC.Commit() // Write into database(state) // Prepare Code and Storage d := time.Now().Add(1000 * time.Millisecond) ctx, cancel := context.WithDeadline(context.Background(), d) defer cancel() code, _ := contractBackendForSC.CodeAt(ctx, validatorSCAddr, nil) storage := make(map[common.Hash]common.Hash) f := func(key, val common.Hash) bool { decode := []byte{} trim := bytes.TrimLeft(val.Bytes(), "\x00") rlp.DecodeBytes(trim, &decode) storage[key] = common.BytesToHash(decode) log.Info("DecodeBytes", "value", val.String(), "decode", storage[key].String()) return true } contractBackendForSC.ForEachStorageAt(ctx, validatorSCAddr, nil, f) // create test backend with smart contract in it contractBackend2 := backends.NewXDCSimulatedBackend(core.GenesisAlloc{ acc1Addr: {Balance: new(big.Int).SetUint64(10000000000)}, acc2Addr: {Balance: new(big.Int).SetUint64(10000000000)}, acc3Addr: {Balance: new(big.Int).SetUint64(10000000000)}, voterAddr: {Balance: new(big.Int).SetUint64(10000000000)}, common.HexToAddress(common.MasternodeVotingSMC): {Balance: new(big.Int).SetUint64(1), Code: code, Storage: storage}, // Binding the MasternodeVotingSMC with newly created 'code' for SC execution }, 10000000) return contractBackend2 } func transferTx(t *testing.T, to common.Address, transferAmount int64) *types.Transaction { t.Logf("Transfering %v to address: %v", transferAmount, to.String()) data := []byte{} gasPrice := big.NewInt(int64(0)) gasLimit := uint64(21000) amount := big.NewInt(transferAmount) nonce := uint64(1) tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) if err != nil { t.Fatal(err) } return signedTX } /* func proposeTX(t *testing.T) *types.Transaction { data := common.Hex2Bytes("012679510000000000000000000000000d3ab14bbad3d99f4203bd7a11acb94882050e7e") //data := []byte{} fmt.Println("data", string(data[:])) gasPrice := big.NewInt(int64(0)) gasLimit := uint64(22680) amountInt := new(big.Int) amount, ok := amountInt.SetString("11000000000000000000000000", 10) if !ok { t.Fatal("big int init failed") } nonce := uint64(0) to := common.HexToAddress("xdc35658f7b2a9e7701e65e7a654659eb1c481d1dc5") tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), acc4Key) if err != nil { t.Fatal(err) } return signedTX } */ func voteTX(gasLimit uint64, nonce uint64, addr string) (*types.Transaction, error) { vote := "6dd7d8ea" // VoteMethod = "0x6dd7d8ea" action := fmt.Sprintf("%s%s%s", vote, "000000000000000000000000", addr[3:]) data := common.Hex2Bytes(action) gasPrice := big.NewInt(int64(0)) amountInt := new(big.Int) amount, ok := amountInt.SetString("60000", 10) if !ok { return nil, fmt.Errorf("big int init failed") } to := common.HexToAddress(common.MasternodeVotingSMC) tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) if err != nil { return nil, err } return signedTX, nil } func UpdateSigner(bc *BlockChain) error { err := bc.UpdateM1() return err } func GetSnapshotSigner(bc *BlockChain, header *types.Header) (signersList, error) { engine := bc.Engine().(*XDPoS.XDPoS) snap, err := engine.GetSnapshot(bc, header) if err != nil { return nil, err } ms := make(signersList) for addr := range snap.Signers { ms[addr.Hex()] = true } return ms, nil } func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testing.T) masterNodes { addr := common.HexToAddress(common.MasternodeVotingSMC) validator, err := contractValidator.NewXDCValidator(addr, backend) if err != nil { t.Fatal(err) } opts := new(bind.CallOpts) candidates, err := validator.GetCandidates(opts) if err != nil { t.Fatal(err) } ms := make(masterNodes) for _, candidate := range candidates { v, err := validator.GetCandidateCap(opts, candidate) if err != nil { t.Fatal(err) } ms[candidate.String()] = *v } return ms } func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block) { // Preparation var err error backend := getCommonBackend(t) blockchain := backend.GetBlockChain() blockchain.Client = backend currentBlock := blockchain.Genesis() // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" block, err := insertBlock(blockchain, i, blockCoinBase, currentBlock, merkleRoot) if err != nil { t.Fatal(err) } currentBlock = block } // Update Signer as there is no previous signer assigned err = UpdateSigner(blockchain) if err != nil { t.Fatal(err) } return blockchain, backend, currentBlock } // insert Block without transcation attached func insertBlock(blockchain *BlockChain, blockNum int, blockCoinBase string, parentBlock *types.Block, root string) (*types.Block, error) { block, err := createXDPoSTestBlock( blockchain, parentBlock.Hash().Hex(), blockCoinBase, blockNum, nil, "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421", common.HexToHash(root), ) if err != nil { return nil, err } err = blockchain.InsertBlock(block) if err != nil { return nil, err } return block, nil } // insert Block with transcation attached func insertBlockTxs(blockchain *BlockChain, blockNum int, blockCoinBase string, parentBlock *types.Block, txs []*types.Transaction, root string) (*types.Block, error) { block, err := createXDPoSTestBlock( blockchain, parentBlock.Hash().Hex(), blockCoinBase, blockNum, txs, "0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c", common.HexToHash(root), ) if err != nil { return nil, err } err = blockchain.InsertBlock(block) if err != nil { return nil, err } return block, nil } func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number int, txs []*types.Transaction, receiptHash string, root common.Hash) (*types.Block, error) { extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation //ReceiptHash = "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421" //Root := "0xc99c095e53ff1afe3b86750affd13c7550a2d24d51fb8e41b3c3ef2ea8274bcc" extraByte, _ := hex.DecodeString(extraSubstring) header := types.Header{ ParentHash: common.HexToHash(parentHash), UncleHash: types.EmptyUncleHash, TxHash: types.EmptyRootHash, // ReceiptHash: types.EmptyRootHash, ReceiptHash: common.HexToHash(receiptHash), Root: root, Coinbase: common.HexToAddress(coinbase), Difficulty: big.NewInt(int64(1)), Number: big.NewInt(int64(number)), GasLimit: 1200000000, Time: uint64(number * 10), Extra: extraByte, } var block *types.Block if len(txs) == 0 { block = types.NewBlockWithHeader(&header) } else { // Prepare Receipt statedb, err := bc.StateAt(bc.GetBlockByNumber(uint64(number - 1)).Root()) //Get parent root if err != nil { return nil, fmt.Errorf("%v when get state", err) } gp := new(GasPool).AddGas(header.GasLimit) // usedGas := uint64(0) var gasUsed = new(uint64) var receipts types.Receipts for i, tx := range txs { statedb.Prepare(tx.Hash(), header.Hash(), i) receipt, _, err := ApplyTransaction(bc.Config(), bc, &header.Coinbase, gp, statedb, &header, tx, gasUsed, vm.Config{}) if err != nil { return nil, fmt.Errorf("%v when applying transaction", err) } receipts = append(receipts, receipt) } header.GasUsed = *gasUsed block = types.NewBlock(&header, txs, nil, receipts) } return block, nil } // Should NOT update signerList if not on the gap block func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, 400) parentSigners, err := GetSnapshotSigner(blockchain, parentBlock.Header()) if err != nil { t.Fatal(err) } t.Logf("Inserting block with propose at 401") blockCoinbaseA := "0xaaa0000000000000000000000000000000000401" tx, err := voteTX(37117, 0, acc1Addr.String()) if err != nil { t.Fatal(err) } //Get from block validator error message merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" blockA, err := insertBlockTxs(blockchain, 401, blockCoinbaseA, parentBlock, []*types.Transaction{tx}, merkleRoot) if err != nil { t.Fatal(err) } signers, err := GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) } if signers[acc1Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("account 1 should NOT sit in the signer list") } eq := reflect.DeepEqual(parentSigners, signers) if eq { t.Logf("Signers unchanged") } else { t.Fatalf("Singers should not be changed!") } } // Should call updateM1 at the gap block, and have the same snapshot values as the parent block if no SM transaction is involved func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { blockchain, _, parentBlock := PrepareXDCTestBlockChain(t, GAP-1) // Insert block 450 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 450) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" block, err := insertBlock(blockchain, 450, blockCoinBase, parentBlock, merkleRoot) if err != nil { t.Fatal(err) } parentSigners, err := GetSnapshotSigner(blockchain, parentBlock.Header()) if err != nil { t.Fatal(err) } signers, err := GetSnapshotSigner(blockchain, block.Header()) if err != nil { t.Fatal(err) } eq := reflect.DeepEqual(parentSigners, signers) if eq { t.Logf("Signers unchanged") } else { t.Fatalf("Singers should not be changed!") } } //Should call updateM1 at gap block, and update the snapshot if there are SM transactions involved func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-2) // Insert first Block 449 t.Logf("Inserting block with propose at 449...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000449" tx, err := voteTX(37117, 0, acc1Addr.String()) if err != nil { t.Fatal(err) } //Get from block validator error message merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" block449, err := insertBlockTxs(blockchain, 449, blockCoinbaseA, parentBlock, []*types.Transaction{tx}, merkleRoot) if err != nil { t.Fatal(err) } parentBlock = block449 signers, err := GetSnapshotSigner(blockchain, block449.Header()) if err != nil { t.Fatal(err) } // At block 449, we should not update signerList. we need to update it till block 450 gap block. // Acc3 is the default account that is on the signerList if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 3 should sit in the signer list") } if signers[acc1Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("account 1 should NOT sit in the signer list") } // Now, let's mine another block to trigger the GAP block signerList update block450CoinbaseAddress := "0xaaa0000000000000000000000000000000000450" merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" block450, err := insertBlock(blockchain, 450, block450CoinbaseAddress, parentBlock, merkleRoot) if err != nil { t.Fatal(err) } signers, err = GetSnapshotSigner(blockchain, block450.Header()) // Now, we voted acc 1 to be in the signerList, which will kick out acc3 because it has less funds if signers[acc3Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("account 3 should NOT sit in the signer list") } if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") } } //Should call updateM1 before gap block, and update the snapshot if there are SM transactions involved func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { blockchain, backend, currentBlock := PrepareXDCTestBlockChain(t, GAP-1) // Insert first Block 450 A t.Logf("Inserting block with propose at 450 A...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000450" tx, err := voteTX(37117, 0, acc1Addr.String()) if err != nil { t.Fatal(err) } //Get from block validator error message merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, currentBlock, []*types.Transaction{tx}, merkleRoot) if err != nil { t.Fatal(err) } signers, err := GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) } if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") } } // Should call updateM1 and update snapshot when a forked block(at gap block number) is inserted back into main chain (Edge case) func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { blockchain, backend, currentBlock := PrepareXDCTestBlockChain(t, GAP-1) // Check initial signer, by default, acc3 is in the signerList signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { t.Fatal(err) } if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc3 should sit in the signer list") } if (signers[acc1Addr.Hex()] == true) || (signers[acc2Addr.Hex()] == true) { debugMessage(backend, signers, t) t.Fatalf("acc1,2should NOT sit in the signer list") } // Insert first Block 450 A t.Logf("Inserting block with propose at 450 A...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000450" tx, err := voteTX(37117, 0, acc1Addr.String()) if err != nil { t.Fatal(err) } merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, currentBlock, []*types.Transaction{tx}, merkleRoot) if err != nil { t.Fatal(err) } signers, err = GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) } if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") } if signers[acc2Addr.Hex()] == true || signers[acc3Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("account 2,3 should NOT sit in the signer list") } // Insert forked Block 450 B t.Logf("Inserting block with propose for acc2 at 450 B...") blockCoinBase450B := "0xbbb0000000000000000000000000000000000450" tx, err = voteTX(37117, 0, acc2Addr.String()) if err != nil { t.Fatal(err) } merkleRoot = "068dfa09d7b4093441c0cc4d9807a71bc586f6101c072d939b214c21cd136eb3" block450B, err := insertBlockTxs(blockchain, 450, blockCoinBase450B, currentBlock, []*types.Transaction{tx}, merkleRoot) if err != nil { t.Fatal(err) } signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { t.Fatal(err) } // Should not run the `updateM1` for forked chain, hence account3 still exit if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 3 should sit in the signer list as previos block result") } if (signers[acc1Addr.Hex()] == true) || (signers[acc2Addr.Hex()] == true) { debugMessage(backend, signers, t) t.Fatalf("acc1,2should NOT sit in the signer list") } //Insert block 451 parent is 451 B t.Logf("Inserting block with propose at 451 B...") blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "068dfa09d7b4093441c0cc4d9807a71bc586f6101c072d939b214c21cd136eb3" block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot) if err != nil { t.Fatal(err) } signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { t.Fatal(err) } if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") } if signers[acc1Addr.Hex()] == true || signers[acc3Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("acc1,3should NOT sit in the signer list") } signers, err = GetSnapshotSigner(blockchain, block451B.Header()) if err != nil { t.Fatal(err) } if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") } if signers[acc1Addr.Hex()] == true || signers[acc3Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("acc1,3should NOT sit in the signer list") } signers, err = GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { t.Fatal(err) } if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc2Addr should sit in the signer list") } if signers[acc1Addr.Hex()] == true || signers[acc3Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("acc1,3should NOT sit in the signer list") } } func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testing.T) { blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-1) state, err := blockchain.State() t.Logf("Account %v have balance of: %v", acc1Addr.String(), state.GetBalance(acc1Addr)) // Check initial signer signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { t.Fatal(err) } if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc3Addr should sit in the signer list") } // Insert first Block 450 A t.Logf("Inserting block with propose and transfer at 450 A...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000450" tx, err := voteTX(58117, 0, acc1Addr.String()) if err != nil { t.Fatal(err) } transferTransaction := transferTx(t, acc1Addr, 999) merkleRoot := "ea465415b60d88429f181fec9fae67c0f19cbf5a4fa10971d96d4faa57d96ffa" blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, parentBlock, []*types.Transaction{tx, transferTransaction}, merkleRoot) if err != nil { t.Fatal(err) } state, err = blockchain.State() t.Log("After transfer transaction at block 450 A, Account 1 have balance of: ", state.GetBalance(acc1Addr)) if state.GetBalance(acc1Addr).Cmp(new(big.Int).SetUint64(10000000999)) != 0 { t.Fatalf("account 1 should have 10000000999 in balance") } signers, err = GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) } if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") } // Insert forked Block 450 B t.Logf("Inserting block with propose at 450 B...") blockCoinBase450B := "0xbbb0000000000000000000000000000000000450" tx, err = voteTX(37117, 0, acc2Addr.String()) if err != nil { t.Fatal(err) } transferTransaction = transferTx(t, acc1Addr, 888) merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" block450B, err := insertBlockTxs(blockchain, 450, blockCoinBase450B, parentBlock, []*types.Transaction{tx, transferTransaction}, merkleRoot) if err != nil { t.Fatal(err) } state, err = blockchain.State() if state.GetBalance(acc1Addr).Cmp(new(big.Int).SetUint64(10000000999)) != 0 { t.Fatalf("account 1 should have 10000000999 in balance as the block is forked, not on the main chain") } signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { t.Fatal(err) } // Should not run the `updateM1` for forked chain, hence account3 still exit if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 3 should sit in the signer list as previos block result") } //Insert block 451 parent is 451 B t.Logf("Inserting block with propose at 451 B...") blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot) if err != nil { t.Fatal(err) } signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { t.Fatal(err) } if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") } signers, err = GetSnapshotSigner(blockchain, block451B.Header()) if err != nil { t.Fatal(err) } if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") } signers, err = GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { t.Fatal(err) } if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc2Addr should sit in the signer list") } state, err = blockchain.State() t.Log("After transfer transaction at block 450 B and the B fork has been merged into main chain, Account 1 have balance of: ", state.GetBalance(acc1Addr)) if state.GetBalance(acc1Addr).Cmp(new(big.Int).SetUint64(10000000888)) != 0 { t.Fatalf("account 1 should have 10000000888 in balance") } } func TestVoteShouldNotBeAffectedByFork(t *testing.T) { blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-1) // Check initial signer, by default, acc3 is in the signerList signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { t.Fatal(err) } if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc3 should sit in the signer list") } if (signers[acc1Addr.Hex()] == true) || (signers[acc2Addr.Hex()] == true) { debugMessage(backend, signers, t) t.Fatalf("acc1,2should NOT sit in the signer list") } // Insert normal blocks 450 A blockCoinBase450A := "0xaaa0000000000000000000000000000000000450" merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" block450A, err := insertBlock(blockchain, 450, blockCoinBase450A, parentBlock, merkleRoot) if err != nil { t.Fatal(err) } // Insert 451 A with vote blockCoinbase451A := "0xaaa0000000000000000000000000000000000451" tx, err := voteTX(37117, 0, acc1Addr.String()) if err != nil { t.Fatal(err) } merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" block451A, err := insertBlockTxs(blockchain, 451, blockCoinbase451A, block450A, []*types.Transaction{tx}, merkleRoot) if err != nil { t.Fatal(err) } // SignerList should be unchanged as the vote happen after GAP block signers, err = GetSnapshotSigner(blockchain, block451A.Header()) if err != nil { t.Fatal(err) } if signers[acc1Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("account 1 should NOT sit in the signer list") } if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 3 should sit in the signer list") } // Now, we going to inject normal blocks of 450B, 451B and 452B. Because it's the longest, it will become the mainchain // Insert forked Block 450 B blockCoinBase450B := "0xbbb0000000000000000000000000000000000450" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" block450B, err := insertBlock(blockchain, 450, blockCoinBase450B, parentBlock, merkleRoot) if err != nil { t.Fatal(err) } blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot) if err != nil { t.Fatal(err) } blockCoinBase452B := "0xbbb0000000000000000000000000000000000452" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" block452B, err := insertBlock(blockchain, 452, blockCoinBase452B, block451B, merkleRoot) if err != nil { t.Fatal(err) } signers, err = GetSnapshotSigner(blockchain, block452B.Header()) if err != nil { t.Fatal(err) } // Should run the `updateM1` for forked chain, but it should not be affected by the voted block 451A which is not on the mainchain anymore if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 3 should sit in the signer list as previos block result") } }