|
|
- package state
-
- import (
- . "github.com/tendermint/tendermint/binary"
- . "github.com/tendermint/tendermint/blocks"
- . "github.com/tendermint/tendermint/common"
- . "github.com/tendermint/tendermint/config"
- db_ "github.com/tendermint/tendermint/db"
-
- "bytes"
- "testing"
- "time"
- )
-
- func randAccountDetail(id uint64, status byte) (*AccountDetail, *PrivAccount) {
- privAccount := GenPrivAccount()
- privAccount.Id = id
- account := privAccount.Account
- return &AccountDetail{
- Account: account,
- Sequence: RandUInt(),
- Balance: RandUInt64() + 1000, // At least 1000.
- Status: status,
- }, privAccount
- }
-
- // The first numValidators accounts are validators.
- func randGenesisState(numAccounts int, numValidators int) (*State, []*PrivAccount) {
- db := db_.NewMemDB()
- accountDetails := make([]*AccountDetail, numAccounts)
- privAccounts := make([]*PrivAccount, numAccounts)
- for i := 0; i < numAccounts; i++ {
- if i < numValidators {
- accountDetails[i], privAccounts[i] =
- randAccountDetail(uint64(i), AccountStatusBonded)
- } else {
- accountDetails[i], privAccounts[i] =
- randAccountDetail(uint64(i), AccountStatusNominal)
- }
- }
- s0 := GenesisState(db, time.Now(), accountDetails)
- s0.Save()
- return s0, privAccounts
- }
-
- func TestCopyState(t *testing.T) {
- // Generate a state
- s0, _ := randGenesisState(10, 5)
- 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.
- accDet := s0.GetAccountDetail(0)
- accDet.Balance += 1
- // The account balance shouldn't have changed yet.
- if s0.GetAccountDetail(0).Balance == accDet.Balance {
- t.Error("Account balance changed unexpectedly")
- }
- // Setting, however, should change the balance.
- s0.SetAccountDetail(accDet)
- if s0.GetAccountDetail(0).Balance != accDet.Balance {
- t.Error("Account balance wasn't set")
- }
- // How 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 TestGenesisSaveLoad(t *testing.T) {
-
- // Generate a state, save & load it.
- s0, _ := randGenesisState(10, 5)
- // Mutate the state to append one empty block.
- block := &Block{
- Header: Header{
- Network: Config.Network,
- Height: 1,
- Time: s0.LastBlockTime.Add(time.Minute),
- Fees: 0,
- LastBlockHash: s0.LastBlockHash,
- LastBlockParts: s0.LastBlockParts,
- StateHash: nil,
- },
- Validation: Validation{},
- Data: Data{
- Txs: []Tx{},
- },
- }
- blockParts := NewPartSetFromData(BinaryBytes(block))
- // The second argument to AppendBlock() is false,
- // which sets Block.Header.StateHash.
- err := s0.Copy().AppendBlock(block, blockParts.Header(), false)
- if err != nil {
- t.Error("Error appending initial block:", err)
- }
- if len(block.Header.StateHash) == 0 {
- t.Error("Expected StateHash but got nothing.")
- }
- // Now append the block to s0.
- // This time we also check the StateHash (as computed above).
- err = s0.AppendBlock(block, blockParts.Header(), true)
- if err != nil {
- t.Error("Error appending initial block:", err)
- }
-
- // Save s0
- s0.Save()
-
- // Sanity check s0
- //s0.DB.(*db_.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()) {
- // The BondedValidators hash should have changed because
- // each AppendBlock() calls IncrementAccum(),
- // changing each validator's Accum.
- t.Error("BondedValidators hash should have changed")
- }
- 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.accountDetails.Hash(), s1.accountDetails.Hash()) {
- t.Error("AccountDetail mismatch")
- }
- }
-
- func TestTxSequence(t *testing.T) {
-
- state, privAccounts := randGenesisState(3, 1)
- acc1 := state.GetAccountDetail(1) // Non-validator
-
- // Try executing a SendTx with various sequence numbers.
- stxProto := SendTx{
- BaseTx: BaseTx{
- Sequence: acc1.Sequence + 1,
- Fee: 0},
- To: 2,
- 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++ {
- stxCopy := stxProto
- stx := &stxCopy
- stx.Sequence = uint(int(acc1.Sequence) + i)
- privAccounts[1].Sign(stx)
- stateCopy := state.Copy()
- err := stateCopy.ExecTx(stx)
- if i >= 1 {
- // Sequence is good.
- if err != nil {
- t.Errorf("Expected good sequence to pass")
- }
- // Check accDet.Sequence.
- newAcc1 := stateCopy.GetAccountDetail(1)
- if newAcc1.Sequence != stx.Sequence {
- t.Errorf("Expected account sequence to change")
- }
- } else {
- // Sequence is bad.
- if err == nil {
- t.Errorf("Expected bad sequence to fail")
- }
- // Check accDet.Sequence. (shouldn't have changed)
- newAcc1 := stateCopy.GetAccountDetail(1)
- if newAcc1.Sequence != acc1.Sequence {
- t.Errorf("Expected account sequence to not change")
- }
- }
- }
- }
-
- func TestTxs(t *testing.T) {
-
- state, privAccounts := randGenesisState(3, 1)
-
- acc0 := state.GetAccountDetail(0) // Validator
- acc1 := state.GetAccountDetail(1) // Non-validator
- acc2 := state.GetAccountDetail(2) // Non-validator
-
- // SendTx.
- {
- state := state.Copy()
- stx := &SendTx{
- BaseTx: BaseTx{
- Sequence: acc1.Sequence + 1,
- Fee: 0},
- To: 2,
- Amount: 1,
- }
- privAccounts[1].Sign(stx)
- err := state.ExecTx(stx)
- if err != nil {
- t.Errorf("Got error in executing send transaction, %v", err)
- }
- newAcc1 := state.GetAccountDetail(1)
- if acc1.Balance-1 != newAcc1.Balance {
- t.Errorf("Unexpected newAcc1 balance. Expected %v, got %v",
- acc1.Balance-1, newAcc1.Balance)
- }
- newAcc2 := state.GetAccountDetail(2)
- if acc2.Balance+1 != newAcc2.Balance {
- t.Errorf("Unexpected newAcc2 balance. Expected %v, got %v",
- acc2.Balance+1, newAcc2.Balance)
- }
- }
-
- // TODO: test overflows.
-
- // SendTx should fail for bonded validators.
- {
- state := state.Copy()
- stx := &SendTx{
- BaseTx: BaseTx{
- Sequence: acc0.Sequence + 1,
- Fee: 0},
- To: 2,
- Amount: 1,
- }
- privAccounts[0].Sign(stx)
- err := state.ExecTx(stx)
- if err == nil {
- t.Errorf("Expected error, SendTx should fail for bonded validators")
- }
- }
-
- // TODO: test for unbonding validators.
-
- // BondTx.
- {
- state := state.Copy()
- btx := &BondTx{
- BaseTx: BaseTx{
- Sequence: acc1.Sequence + 1,
- Fee: 0},
- }
- privAccounts[1].Sign(btx)
- err := state.ExecTx(btx)
- if err != nil {
- t.Errorf("Got error in executing bond transaction, %v", err)
- }
- newAcc1 := state.GetAccountDetail(1)
- if acc1.Balance != newAcc1.Balance {
- t.Errorf("Unexpected newAcc1 balance. Expected %v, got %v",
- acc1.Balance, newAcc1.Balance)
- }
- if newAcc1.Status != AccountStatusBonded {
- t.Errorf("Unexpected newAcc1 status.")
- }
- _, acc1Val := state.BondedValidators.GetById(acc1.Id)
- if acc1Val == nil {
- t.Errorf("acc1Val not present")
- }
- if acc1Val.BondHeight != state.LastBlockHeight {
- t.Errorf("Unexpected bond height. Expected %v, got %v",
- state.LastBlockHeight, acc1Val.BondHeight)
- }
- if acc1Val.VotingPower != acc1.Balance {
- t.Errorf("Unexpected voting power. Expected %v, got %v",
- acc1Val.VotingPower, acc1.Balance)
- }
- if acc1Val.Accum != 0 {
- t.Errorf("Unexpected accum. Expected 0, got %v",
- acc1Val.Accum)
- }
- }
-
- // TODO UnbondTx.
- // TODO NameTx.
-
- }
|