2018-07-16 18:25:15 -07:00
|
|
|
package simulation
|
2018-07-16 18:15:50 -07:00
|
|
|
|
|
|
|
import (
|
2019-06-08 13:55:47 -07:00
|
|
|
"encoding/json"
|
2018-07-17 16:27:51 -07:00
|
|
|
"fmt"
|
2018-11-08 07:32:53 -08:00
|
|
|
"math/rand"
|
2018-09-09 08:34:09 -07:00
|
|
|
"testing"
|
2018-11-09 07:31:56 -08:00
|
|
|
|
2019-04-04 07:36:39 -07:00
|
|
|
abci "github.com/tendermint/tendermint/abci/types"
|
|
|
|
|
2019-02-13 15:01:50 -08:00
|
|
|
"github.com/cosmos/cosmos-sdk/baseapp"
|
2018-11-29 07:17:10 -08:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2018-07-16 18:15:50 -07:00
|
|
|
)
|
|
|
|
|
2019-02-13 15:01:50 -08:00
|
|
|
// assertAll asserts the all invariants against application state
|
|
|
|
func assertAllInvariants(t *testing.T, app *baseapp.BaseApp, invs sdk.Invariants,
|
2019-07-11 03:56:43 -07:00
|
|
|
event string, logWriter LogWriter, allInvariants bool) {
|
2019-02-13 15:01:50 -08:00
|
|
|
|
|
|
|
ctx := app.NewContext(false, abci.Header{Height: app.LastBlockHeight() + 1})
|
|
|
|
|
2019-07-11 03:56:43 -07:00
|
|
|
var broken bool
|
|
|
|
var invariantResults []string
|
2019-02-13 15:01:50 -08:00
|
|
|
for i := 0; i < len(invs); i++ {
|
2019-07-11 03:56:43 -07:00
|
|
|
res, stop := invs[i](ctx)
|
|
|
|
if stop {
|
|
|
|
broken = true
|
|
|
|
invariantResults = append(invariantResults, res)
|
|
|
|
} else if allInvariants {
|
|
|
|
invariantResults = append(invariantResults, res)
|
2019-02-13 15:01:50 -08:00
|
|
|
}
|
|
|
|
}
|
2019-07-11 03:56:43 -07:00
|
|
|
|
|
|
|
if broken {
|
|
|
|
fmt.Printf("Invariants broken after %s\n\n", event)
|
|
|
|
for _, res := range invariantResults {
|
|
|
|
fmt.Printf("%s\n", res)
|
|
|
|
}
|
|
|
|
logWriter.PrintLogs()
|
|
|
|
t.Fatal()
|
|
|
|
}
|
2019-02-13 15:01:50 -08:00
|
|
|
}
|
|
|
|
|
2018-11-07 08:57:53 -08:00
|
|
|
func getTestingMode(tb testing.TB) (testingMode bool, t *testing.T, b *testing.B) {
|
|
|
|
testingMode = false
|
|
|
|
if _t, ok := tb.(*testing.T); ok {
|
|
|
|
t = _t
|
|
|
|
testingMode = true
|
|
|
|
} else {
|
|
|
|
b = tb.(*testing.B)
|
|
|
|
}
|
2019-03-14 11:13:15 -07:00
|
|
|
return testingMode, t, b
|
2018-09-09 08:34:09 -07:00
|
|
|
}
|
2018-11-08 07:32:53 -08:00
|
|
|
|
|
|
|
// getBlockSize returns a block size as determined from the transition matrix.
|
|
|
|
// It targets making average block size the provided parameter. The three
|
|
|
|
// states it moves between are:
|
|
|
|
// - "over stuffed" blocks with average size of 2 * avgblocksize,
|
|
|
|
// - normal sized blocks, hitting avgBlocksize on average,
|
|
|
|
// - and empty blocks, with no txs / only txs scheduled from the past.
|
|
|
|
func getBlockSize(r *rand.Rand, params Params,
|
|
|
|
lastBlockSizeState, avgBlockSize int) (state, blocksize int) {
|
|
|
|
|
|
|
|
// TODO: Make default blocksize transition matrix actually make the average
|
|
|
|
// blocksize equal to avgBlockSize.
|
|
|
|
state = params.BlockSizeTransitionMatrix.NextState(r, lastBlockSizeState)
|
|
|
|
switch state {
|
|
|
|
case 0:
|
|
|
|
blocksize = r.Intn(avgBlockSize * 4)
|
|
|
|
case 1:
|
|
|
|
blocksize = r.Intn(avgBlockSize * 2)
|
|
|
|
default:
|
|
|
|
blocksize = 0
|
|
|
|
}
|
|
|
|
return state, blocksize
|
|
|
|
}
|
2018-11-09 07:31:56 -08:00
|
|
|
|
2019-04-09 18:34:38 -07:00
|
|
|
// PeriodicInvariants returns an array of wrapped Invariants. Where each
|
|
|
|
// invariant function is only executed periodically defined by period and offset.
|
2019-06-28 13:11:27 -07:00
|
|
|
func PeriodicInvariants(invariants []sdk.Invariant, period, offset int) []sdk.Invariant {
|
2019-04-09 18:34:38 -07:00
|
|
|
var outInvariants []sdk.Invariant
|
|
|
|
for _, invariant := range invariants {
|
2019-07-11 03:56:43 -07:00
|
|
|
outInvariant := func(ctx sdk.Context) (string, bool) {
|
2019-04-09 18:34:38 -07:00
|
|
|
if int(ctx.BlockHeight())%period == offset {
|
|
|
|
return invariant(ctx)
|
|
|
|
}
|
2019-07-11 03:56:43 -07:00
|
|
|
return "", false
|
2019-04-09 18:34:38 -07:00
|
|
|
}
|
|
|
|
outInvariants = append(outInvariants, outInvariant)
|
|
|
|
}
|
|
|
|
return outInvariants
|
|
|
|
}
|
2019-06-08 13:55:47 -07:00
|
|
|
|
|
|
|
func mustMarshalJSONIndent(o interface{}) []byte {
|
|
|
|
bz, err := json.MarshalIndent(o, "", " ")
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("failed to JSON encode: %s", err))
|
|
|
|
}
|
|
|
|
|
|
|
|
return bz
|
|
|
|
}
|