package state import ( "github.com/tendermint/tendermint/account" _ "github.com/tendermint/tendermint/config/tendermint_test" "github.com/tendermint/tendermint/types" "bytes" "testing" "time" ) func execTxWithState(state *State, tx types.Tx, runCall bool) error { cache := NewBlockCache(state) err := ExecTx(cache, tx, runCall, nil) if err != nil { return err } else { cache.Sync() return nil } } func TestCopyState(t *testing.T) { // Generate a random state s0, privAccounts, _ := RandGenesisState(10, true, 1000, 5, true, 1000) s0Hash := s0.Hash() if len(s0Hash) == 0 { t.Error("Expected state hash") } // Check hash of copy s0Copy := s0.Copy() if !bytes.Equal(s0Hash, s0Copy.Hash()) { t.Error("Expected state copy hash to be the same") } // Mutate the original; hash should change. acc0Address := privAccounts[0].PubKey.Address() acc := s0.GetAccount(acc0Address) acc.Balance += 1 // The account balance shouldn't have changed yet. if s0.GetAccount(acc0Address).Balance == acc.Balance { t.Error("Account balance changed unexpectedly") } // Setting, however, should change the balance. s0.UpdateAccount(acc) if s0.GetAccount(acc0Address).Balance != acc.Balance { t.Error("Account balance wasn't set") } // Now that the state changed, the hash should change too. if bytes.Equal(s0Hash, s0.Hash()) { t.Error("Expected state hash to have changed") } // The s0Copy shouldn't have changed though. if !bytes.Equal(s0Hash, s0Copy.Hash()) { t.Error("Expected state copy hash to have not changed") } } func makeBlock(t *testing.T, state *State, commits []types.Commit, txs []types.Tx) *types.Block { block := &types.Block{ Header: &types.Header{ Network: "tendermint_test", Height: state.LastBlockHeight + 1, Time: state.LastBlockTime.Add(time.Minute), Fees: 0, NumTxs: uint(len(txs)), LastBlockHash: state.LastBlockHash, LastBlockParts: state.LastBlockParts, StateHash: nil, }, Validation: &types.Validation{ Commits: commits, }, Data: &types.Data{ Txs: txs, }, } // Fill in block StateHash err := state.SetBlockStateHash(block) if err != nil { t.Error("Error appending initial block:", err) } if len(block.Header.StateHash) == 0 { t.Error("Expected StateHash but got nothing.") } return block } func TestGenesisSaveLoad(t *testing.T) { // Generate a state, save & load it. s0, _, _ := RandGenesisState(10, true, 1000, 5, true, 1000) // Make complete block and blockParts block := makeBlock(t, s0, nil, nil) blockParts := block.MakePartSet() // Now append the block to s0. err := ExecBlock(s0, block, blockParts.Header()) if err != nil { t.Error("Error appending initial block:", err) } // Save s0 s0.Save() // Sanity check s0 //s0.DB.(*dbm.MemDB).Print() if s0.BondedValidators.TotalVotingPower() == 0 { t.Error("s0 BondedValidators TotalVotingPower should not be 0") } if s0.LastBlockHeight != 1 { t.Error("s0 LastBlockHeight should be 1, got", s0.LastBlockHeight) } // Load s1 s1 := LoadState(s0.DB) // Compare height & blockHash if s0.LastBlockHeight != s1.LastBlockHeight { t.Error("LastBlockHeight mismatch") } if !bytes.Equal(s0.LastBlockHash, s1.LastBlockHash) { t.Error("LastBlockHash mismatch") } // Compare state merkle trees if s0.BondedValidators.Size() != s1.BondedValidators.Size() { t.Error("BondedValidators Size mismatch") } if s0.BondedValidators.TotalVotingPower() != s1.BondedValidators.TotalVotingPower() { t.Error("BondedValidators TotalVotingPower mismatch") } if !bytes.Equal(s0.BondedValidators.Hash(), s1.BondedValidators.Hash()) { t.Error("BondedValidators hash mismatch") } if s0.UnbondingValidators.Size() != s1.UnbondingValidators.Size() { t.Error("UnbondingValidators Size mismatch") } if s0.UnbondingValidators.TotalVotingPower() != s1.UnbondingValidators.TotalVotingPower() { t.Error("UnbondingValidators TotalVotingPower mismatch") } if !bytes.Equal(s0.UnbondingValidators.Hash(), s1.UnbondingValidators.Hash()) { t.Error("UnbondingValidators hash mismatch") } if !bytes.Equal(s0.accounts.Hash(), s1.accounts.Hash()) { t.Error("Accounts mismatch") } if !bytes.Equal(s0.validatorInfos.Hash(), s1.validatorInfos.Hash()) { t.Error("Accounts mismatch") } } func TestTxSequence(t *testing.T) { state, privAccounts, _ := RandGenesisState(3, true, 1000, 1, true, 1000) acc0 := state.GetAccount(privAccounts[0].PubKey.Address()) acc0PubKey := privAccounts[0].PubKey acc1 := state.GetAccount(privAccounts[1].PubKey.Address()) // Try executing a SendTx with various sequence numbers. makeSendTx := func(sequence uint) *types.SendTx { return &types.SendTx{ Inputs: []*types.TxInput{ &types.TxInput{ Address: acc0.Address, Amount: 1, Sequence: sequence, PubKey: acc0PubKey, }, }, Outputs: []*types.TxOutput{ &types.TxOutput{ Address: acc1.Address, Amount: 1, }, }, } } // Test a variety of sequence numbers for the tx. // The tx should only pass when i == 1. for i := -1; i < 3; i++ { sequence := acc0.Sequence + uint(i) tx := makeSendTx(sequence) tx.Inputs[0].Signature = privAccounts[0].Sign(tx) stateCopy := state.Copy() err := execTxWithState(stateCopy, tx, true) if i == 1 { // Sequence is good. if err != nil { t.Errorf("Expected good sequence to pass: %v", err) } // Check acc.Sequence. newAcc0 := stateCopy.GetAccount(acc0.Address) if newAcc0.Sequence != sequence { t.Errorf("Expected account sequence to change to %v, got %v", sequence, newAcc0.Sequence) } } else { // Sequence is bad. if err == nil { t.Errorf("Expected bad sequence to fail") } // Check acc.Sequence. (shouldn't have changed) newAcc0 := stateCopy.GetAccount(acc0.Address) if newAcc0.Sequence != acc0.Sequence { t.Errorf("Expected account sequence to not change from %v, got %v", acc0.Sequence, newAcc0.Sequence) } } } } // TODO: test overflows. // TODO: test for unbonding validators. func TestTxs(t *testing.T) { state, privAccounts, _ := RandGenesisState(3, true, 1000, 1, true, 1000) //val0 := state.GetValidatorInfo(privValidators[0].Address) acc0 := state.GetAccount(privAccounts[0].PubKey.Address()) acc0PubKey := privAccounts[0].PubKey acc1 := state.GetAccount(privAccounts[1].PubKey.Address()) // SendTx. { state := state.Copy() tx := &types.SendTx{ Inputs: []*types.TxInput{ &types.TxInput{ Address: acc0.Address, Amount: 1, Sequence: acc0.Sequence + 1, PubKey: acc0PubKey, }, }, Outputs: []*types.TxOutput{ &types.TxOutput{ Address: acc1.Address, Amount: 1, }, }, } tx.Inputs[0].Signature = privAccounts[0].Sign(tx) err := execTxWithState(state, tx, true) if err != nil { t.Errorf("Got error in executing send transaction, %v", err) } newAcc0 := state.GetAccount(acc0.Address) if acc0.Balance-1 != newAcc0.Balance { t.Errorf("Unexpected newAcc0 balance. Expected %v, got %v", acc0.Balance-1, newAcc0.Balance) } newAcc1 := state.GetAccount(acc1.Address) if acc1.Balance+1 != newAcc1.Balance { t.Errorf("Unexpected newAcc1 balance. Expected %v, got %v", acc1.Balance+1, newAcc1.Balance) } } // BondTx. { state := state.Copy() tx := &types.BondTx{ PubKey: acc0PubKey.(account.PubKeyEd25519), Inputs: []*types.TxInput{ &types.TxInput{ Address: acc0.Address, Amount: 1, Sequence: acc0.Sequence + 1, PubKey: acc0PubKey, }, }, UnbondTo: []*types.TxOutput{ &types.TxOutput{ Address: acc0.Address, Amount: 1, }, }, } tx.Signature = privAccounts[0].Sign(tx).(account.SignatureEd25519) tx.Inputs[0].Signature = privAccounts[0].Sign(tx) err := execTxWithState(state, tx, true) if err != nil { t.Errorf("Got error in executing bond transaction, %v", err) } newAcc0 := state.GetAccount(acc0.Address) if newAcc0.Balance != acc0.Balance-1 { t.Errorf("Unexpected newAcc0 balance. Expected %v, got %v", acc0.Balance-1, newAcc0.Balance) } _, acc0Val := state.BondedValidators.GetByAddress(acc0.Address) if acc0Val == nil { t.Errorf("acc0Val not present") } if acc0Val.BondHeight != state.LastBlockHeight+1 { t.Errorf("Unexpected bond height. Expected %v, got %v", state.LastBlockHeight, acc0Val.BondHeight) } if acc0Val.VotingPower != 1 { t.Errorf("Unexpected voting power. Expected %v, got %v", acc0Val.VotingPower, acc0.Balance) } if acc0Val.Accum != 0 { t.Errorf("Unexpected accum. Expected 0, got %v", acc0Val.Accum) } } // TODO UnbondTx. } func TestAddValidator(t *testing.T) { // Generate a state, save & load it. s0, privAccounts, privValidators := RandGenesisState(10, false, 1000, 1, false, 1000) // The first privAccount will become a validator acc0 := privAccounts[0] bondTx := &types.BondTx{ PubKey: acc0.PubKey.(account.PubKeyEd25519), Inputs: []*types.TxInput{ &types.TxInput{ Address: acc0.Address, Amount: 1000, Sequence: 1, PubKey: acc0.PubKey, }, }, UnbondTo: []*types.TxOutput{ &types.TxOutput{ Address: acc0.Address, Amount: 1000, }, }, } bondTx.Signature = acc0.Sign(bondTx).(account.SignatureEd25519) bondTx.Inputs[0].Signature = acc0.Sign(bondTx) // Make complete block and blockParts block0 := makeBlock(t, s0, nil, []types.Tx{bondTx}) block0Parts := block0.MakePartSet() // Sanity check if s0.BondedValidators.Size() != 1 { t.Error("Expected there to be 1 validators before bondTx") } // Now append the block to s0. err := ExecBlock(s0, block0, block0Parts.Header()) if err != nil { t.Error("Error appending initial block:", err) } // Must save before further modification s0.Save() // Test new validator set if s0.BondedValidators.Size() != 2 { t.Error("Expected there to be 2 validators after bondTx") } // The validation for the next block should only require 1 signature // (the new validator wasn't active for block0) commit0 := &types.Vote{ Height: 1, Round: 0, Type: types.VoteTypeCommit, BlockHash: block0.Hash(), BlockParts: block0Parts.Header(), } privValidators[0].SignVote(commit0) block1 := makeBlock(t, s0, []types.Commit{ types.Commit{ Address: privValidators[0].Address, Round: 0, Signature: commit0.Signature, }, }, nil, ) block1Parts := block1.MakePartSet() err = ExecBlock(s0, block1, block1Parts.Header()) if err != nil { t.Error("Error appending secondary block:", err) } }