cosmos-sdk/state/execution_test.go

315 lines
11 KiB
Go
Raw Normal View History

package state
import (
"testing"
"github.com/stretchr/testify/assert"
abci "github.com/tendermint/abci/types"
"github.com/tendermint/basecoin/types"
2017-05-01 07:03:54 -07:00
"github.com/tendermint/tmlibs/log"
)
2017-04-13 18:33:39 -07:00
//--------------------------------------------------------
// test environment is a bunch of lists of accountns
type execTest struct {
chainID string
store types.KVStore
state *State
accIn types.PrivAccount
accOut types.PrivAccount
2017-04-13 18:33:39 -07:00
}
func newExecTest() *execTest {
et := &execTest{
chainID: "test_chain_id",
}
et.reset()
return et
}
func (et *execTest) signTx(tx *types.SendTx, accsIn ...types.PrivAccount) {
types.SignTx(et.chainID, tx, accsIn...)
2017-04-13 18:33:39 -07:00
}
// make tx from accsIn to et.accOut
2017-04-17 16:53:06 -07:00
func (et *execTest) getTx(seq int, accOut types.PrivAccount, accsIn ...types.PrivAccount) *types.SendTx {
return types.GetTx(seq, accOut, accsIn...)
2017-04-13 18:33:39 -07:00
}
// returns the final balance and expected balance for input and output accounts
func (et *execTest) exec(tx *types.SendTx, checkTx bool) (res abci.Result, inGot, inExp, outGot, outExp types.Coins) {
initBalIn := et.state.GetAccount(et.accIn.Account.PubKey.Address()).Balance
initBalOut := et.state.GetAccount(et.accOut.Account.PubKey.Address()).Balance
2017-04-13 18:33:39 -07:00
res = ExecTx(et.state, nil, tx, checkTx, nil)
2017-04-13 18:33:39 -07:00
endBalIn := et.state.GetAccount(et.accIn.Account.PubKey.Address()).Balance
endBalOut := et.state.GetAccount(et.accOut.Account.PubKey.Address()).Balance
decrBalInExp := tx.Outputs[0].Coins.Plus(types.Coins{tx.Fee}) //expected decrease in balance In
return res, endBalIn, initBalIn.Minus(decrBalInExp), endBalOut, initBalOut.Plus(tx.Outputs[0].Coins)
2017-04-13 18:33:39 -07:00
}
func (et *execTest) acc2State(accs ...types.PrivAccount) {
2017-04-13 18:33:39 -07:00
for _, acc := range accs {
et.state.SetAccount(acc.Account.PubKey.Address(), &acc.Account)
}
}
//reset everything. state is empty
func (et *execTest) reset() {
et.accIn = types.MakeAcc("foo")
et.accOut = types.MakeAcc("bar")
2017-04-13 18:33:39 -07:00
et.store = types.NewMemKVStore()
et.state = NewState(et.store)
et.state.SetLogger(log.TestingLogger())
2017-04-13 18:33:39 -07:00
et.state.SetChainID(et.chainID)
2017-04-17 14:47:00 -07:00
// NOTE we dont run acc2State here
// so we can test non-existing accounts
2017-04-13 18:33:39 -07:00
}
//--------------------------------------------------------
2017-03-23 17:51:50 -07:00
func TestGetInputs(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//nil submissions
acc, res := getInputs(nil, nil)
2017-03-28 13:32:55 -07:00
assert.True(res.IsOK(), "getInputs: error on nil submission")
2017-03-23 17:51:50 -07:00
assert.Zero(len(acc), "getInputs: accounts returned on nil submission")
//test getInputs for registered, non-registered account
2017-03-28 13:32:55 -07:00
et.reset()
txs := types.Accs2TxInputs(1, et.accIn)
2017-03-28 13:32:55 -07:00
acc, res = getInputs(et.state, txs)
assert.True(res.IsErr(), "getInputs: expected error when using getInput with non-registered Input")
2017-03-23 17:51:50 -07:00
et.acc2State(et.accIn)
2017-03-28 13:32:55 -07:00
acc, res = getInputs(et.state, txs)
assert.True(res.IsOK(), "getInputs: expected to getInput from registered Input")
2017-03-23 17:51:50 -07:00
//test sending duplicate accounts
2017-03-28 13:32:55 -07:00
et.reset()
et.acc2State(et.accIn, et.accIn, et.accIn)
txs = types.Accs2TxInputs(1, et.accIn, et.accIn, et.accIn)
2017-03-28 13:32:55 -07:00
acc, res = getInputs(et.state, txs)
2017-03-23 17:51:50 -07:00
assert.True(res.IsErr(), "getInputs: expected error when sending duplicate accounts")
}
func TestGetOrMakeOutputs(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//nil submissions
acc, res := getOrMakeOutputs(nil, nil, nil)
2017-03-28 13:32:55 -07:00
assert.True(res.IsOK(), "getOrMakeOutputs: error on nil submission")
2017-03-23 17:51:50 -07:00
assert.Zero(len(acc), "getOrMakeOutputs: accounts returned on nil submission")
//test sending duplicate accounts
2017-03-28 13:32:55 -07:00
et.reset()
txs := types.Accs2TxOutputs(et.accIn, et.accIn, et.accIn)
2017-03-28 13:32:55 -07:00
_, res = getOrMakeOutputs(et.state, nil, txs)
2017-03-23 17:51:50 -07:00
assert.True(res.IsErr(), "getOrMakeOutputs: expected error when sending duplicate accounts")
2017-04-17 14:47:00 -07:00
//test sending to existing/new account
2017-03-28 13:32:55 -07:00
et.reset()
txs1 := types.Accs2TxOutputs(et.accIn)
txs2 := types.Accs2TxOutputs(et.accOut)
2017-03-23 17:51:50 -07:00
et.acc2State(et.accIn)
2017-03-28 13:32:55 -07:00
_, res = getOrMakeOutputs(et.state, nil, txs1)
assert.True(res.IsOK(), "getOrMakeOutputs: error when sending to existing account")
2017-03-23 17:51:50 -07:00
2017-03-28 13:32:55 -07:00
mapRes2, res := getOrMakeOutputs(et.state, nil, txs2)
assert.True(res.IsOK(), "getOrMakeOutputs: error when sending to new account")
2017-03-23 17:51:50 -07:00
//test the map results
_, map2ok := mapRes2[string(txs2[0].Address)]
assert.True(map2ok, "getOrMakeOutputs: account output does not contain new account map item")
}
func TestValidateInputsBasic(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//validate input basic
txs := types.Accs2TxInputs(1, et.accIn)
2017-03-23 17:51:50 -07:00
res := validateInputsBasic(txs)
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "validateInputsBasic: expected no error on good tx input. Error: %v", res.Error())
2017-03-23 17:51:50 -07:00
txs[0].Coins[0].Amount = 0
res = validateInputsBasic(txs)
assert.True(res.IsErr(), "validateInputsBasic: expected error on bad tx input")
2017-03-23 17:51:50 -07:00
}
2017-03-23 17:51:50 -07:00
func TestValidateInputsAdvanced(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-04-17 16:53:06 -07:00
//create three temp accounts for the test
accIn1 := types.MakeAcc("foox")
accIn2 := types.MakeAcc("fooy")
accIn3 := types.MakeAcc("fooz")
2017-03-23 17:51:50 -07:00
//validate inputs advanced
2017-04-17 16:53:06 -07:00
txs := et.getTx(1, et.accOut, accIn1, accIn2, accIn3)
2017-04-17 16:53:06 -07:00
et.acc2State(accIn1, accIn2, accIn3, et.accOut)
2017-03-28 13:32:55 -07:00
accMap, res := getInputs(et.state, txs.Inputs)
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "validateInputsAdvanced: error retrieving accMap. Error: %v", res.Error())
2017-03-28 13:32:55 -07:00
signBytes := txs.SignBytes(et.chainID)
2017-03-23 17:51:50 -07:00
//test bad case, unsigned
totalCoins, res := validateInputsAdvanced(accMap, signBytes, txs.Inputs)
assert.True(res.IsErr(), "validateInputsAdvanced: expected an error on an unsigned tx input")
//test good case sgined
2017-04-17 16:53:06 -07:00
et.signTx(txs, accIn1, accIn2, accIn3, et.accOut)
2017-03-23 17:51:50 -07:00
totalCoins, res = validateInputsAdvanced(accMap, signBytes, txs.Inputs)
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "validateInputsAdvanced: expected no error on good tx input. Error: %v", res.Error())
2017-04-17 17:10:31 -07:00
txsTotalCoins := txs.Inputs[0].Coins.
Plus(txs.Inputs[1].Coins).
Plus(txs.Inputs[2].Coins)
2017-04-17 16:53:06 -07:00
assert.True(totalCoins.IsEqual(txsTotalCoins),
"ValidateInputsAdvanced: transaction total coins are not equal: got %v, expected %v", txsTotalCoins, totalCoins)
2017-03-23 17:51:50 -07:00
}
func TestValidateInputAdvanced(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//validate input advanced
2017-04-17 16:53:06 -07:00
txs := et.getTx(1, et.accOut, et.accIn)
et.acc2State(et.accIn, et.accOut)
2017-03-28 13:32:55 -07:00
signBytes := txs.SignBytes(et.chainID)
2017-03-23 17:51:50 -07:00
//unsigned case
res := validateInputAdvanced(&et.accIn.Account, signBytes, txs.Inputs[0])
2017-03-23 17:51:50 -07:00
assert.True(res.IsErr(), "validateInputAdvanced: expected error on tx input without signature")
//good signed case
et.signTx(txs, et.accIn, et.accOut)
res = validateInputAdvanced(&et.accIn.Account, signBytes, txs.Inputs[0])
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "validateInputAdvanced: expected no error on good tx input. Error: %v", res.Error())
2017-03-23 17:51:50 -07:00
//bad sequence case
2017-04-17 16:53:06 -07:00
et.accIn.Sequence = 1
et.signTx(txs, et.accIn, et.accOut)
res = validateInputAdvanced(&et.accIn.Account, signBytes, txs.Inputs[0])
2017-04-17 16:53:06 -07:00
assert.Equal(abci.CodeType_BaseInvalidSequence, res.Code, "validateInputAdvanced: expected error on tx input with bad sequence")
et.accIn.Sequence = 0 //restore sequence
2017-03-23 17:51:50 -07:00
//bad balance case
2017-04-17 16:53:06 -07:00
et.accIn.Balance = types.Coins{{"mycoin", 2}}
et.signTx(txs, et.accIn, et.accOut)
res = validateInputAdvanced(&et.accIn.Account, signBytes, txs.Inputs[0])
2017-04-17 16:53:06 -07:00
assert.Equal(abci.CodeType_BaseInsufficientFunds, res.Code,
"validateInputAdvanced: expected error on tx input with insufficient funds %v", et.accIn.Sequence)
2017-03-23 17:51:50 -07:00
}
func TestValidateOutputsAdvanced(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//validateOutputsBasic
txs := types.Accs2TxOutputs(et.accIn)
2017-03-23 17:51:50 -07:00
res := validateOutputsBasic(txs)
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "validateOutputsBasic: expected no error on good tx output. Error: %v", res.Error())
2017-03-23 17:51:50 -07:00
txs[0].Coins[0].Amount = 0
res = validateOutputsBasic(txs)
2017-04-17 16:53:06 -07:00
assert.True(res.IsErr(), "validateInputBasic: expected error on bad tx output. Error: %v", res.Error())
2017-03-23 17:51:50 -07:00
}
func TestSumOutput(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//SumOutput
txs := types.Accs2TxOutputs(et.accIn, et.accOut)
2017-03-23 17:51:50 -07:00
total := sumOutputs(txs)
assert.True(total.IsEqual(txs[0].Coins.Plus(txs[1].Coins)), "sumOutputs: total coins are not equal")
}
func TestAdjustBy(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//adjustByInputs/adjustByOutputs
//sending transaction from accIn to accOut
initBalIn := et.accIn.Account.Balance
initBalOut := et.accOut.Account.Balance
et.acc2State(et.accIn, et.accOut)
2017-03-23 17:51:50 -07:00
txIn := types.Accs2TxInputs(1, et.accIn)
txOut := types.Accs2TxOutputs(et.accOut)
2017-03-28 13:32:55 -07:00
accMap, _ := getInputs(et.state, txIn)
accMap, _ = getOrMakeOutputs(et.state, accMap, txOut)
2017-03-23 17:51:50 -07:00
2017-03-28 13:32:55 -07:00
adjustByInputs(et.state, accMap, txIn)
adjustByOutputs(et.state, accMap, txOut, false)
2017-03-23 17:51:50 -07:00
endBalIn := accMap[string(et.accIn.Account.PubKey.Address())].Balance
endBalOut := accMap[string(et.accOut.Account.PubKey.Address())].Balance
decrBalIn := initBalIn.Minus(endBalIn)
incrBalOut := endBalOut.Minus(initBalOut)
2017-03-23 17:51:50 -07:00
assert.True(decrBalIn.IsEqual(txIn[0].Coins),
2017-04-17 16:53:06 -07:00
"adjustByInputs: total coins are not equal. diff: %v, tx: %v", decrBalIn.String(), txIn[0].Coins.String())
assert.True(incrBalOut.IsEqual(txOut[0].Coins),
2017-04-17 16:53:06 -07:00
"adjustByInputs: total coins are not equal. diff: %v, tx: %v", incrBalOut.String(), txOut[0].Coins.String())
2017-03-23 17:51:50 -07:00
}
func TestExecTx(t *testing.T) {
assert := assert.New(t)
2017-03-28 13:32:55 -07:00
et := newExecTest()
2017-03-23 17:51:50 -07:00
//ExecTx
2017-04-17 16:53:06 -07:00
txs := et.getTx(1, et.accOut, et.accIn)
et.acc2State(et.accIn)
et.acc2State(et.accOut)
et.signTx(txs, et.accIn)
2017-03-23 17:51:50 -07:00
//Bad Balance
et.accIn.Balance = types.Coins{{"mycoin", 2}}
et.acc2State(et.accIn)
2017-03-28 13:32:55 -07:00
res, _, _, _, _ := et.exec(txs, true)
2017-04-17 16:53:06 -07:00
assert.True(res.IsErr(), "ExecTx/Bad CheckTx: Expected error return from ExecTx, returned: %v", res)
res, balIn, balInExp, balOut, balOutExp := et.exec(txs, false)
2017-04-17 16:53:06 -07:00
assert.True(res.IsErr(), "ExecTx/Bad DeliverTx: Expected error return from ExecTx, returned: %v", res)
assert.False(balIn.IsEqual(balInExp),
2017-04-17 16:53:06 -07:00
"ExecTx/Bad DeliverTx: balance shouldn't be equal for accIn: got %v, expected: %v", balIn, balInExp)
assert.False(balOut.IsEqual(balOutExp),
2017-04-17 16:53:06 -07:00
"ExecTx/Bad DeliverTx: balance shouldn't be equal for accOut: got %v, expected: %v", balOut, balOutExp)
2017-03-23 17:51:50 -07:00
//Regular CheckTx
2017-03-28 13:32:55 -07:00
et.reset()
et.acc2State(et.accIn)
et.acc2State(et.accOut)
2017-03-28 13:32:55 -07:00
res, _, _, _, _ = et.exec(txs, true)
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "ExecTx/Good CheckTx: Expected OK return from ExecTx, Error: %v", res)
2017-03-23 17:51:50 -07:00
//Regular DeliverTx
2017-03-28 13:32:55 -07:00
et.reset()
et.acc2State(et.accIn)
et.acc2State(et.accOut)
res, balIn, balInExp, balOut, balOutExp = et.exec(txs, false)
2017-04-17 16:53:06 -07:00
assert.True(res.IsOK(), "ExecTx/Good DeliverTx: Expected OK return from ExecTx, Error: %v", res)
assert.True(balIn.IsEqual(balInExp),
2017-04-17 16:53:06 -07:00
"ExecTx/good DeliverTx: unexpected change in input balance, got: %v, expected: %v", balIn, balInExp)
assert.True(balOut.IsEqual(balOutExp),
2017-04-17 16:53:06 -07:00
"ExecTx/good DeliverTx: unexpected change in output balance, got: %v, expected: %v", balOut, balOutExp)
2017-03-28 13:32:55 -07:00
}