From b07b28799d422d03ee3bd32f676ed794007b8f8f Mon Sep 17 00:00:00 2001 From: Nikolay Eskov Date: Wed, 16 Oct 2024 19:10:57 +0300 Subject: [PATCH] Ride generating balance test (#1514) * Refactored a bit 'TestGeneratingBalanceValuesForNewestFunctions'. * Add comment about 'checkSenderBalance' argument for 'ride.NewEnvironmentWithWrappedState' function. * Add 'LastFeature' function. * Add 'TestGeneratingBalanceValuesInRide'. Analogous to the https://github.com/wavesplatform/Waves/pull/3963. --- pkg/settings/features.go | 11 ++ pkg/state/common_test.go | 56 +++++- pkg/state/script_caller.go | 5 +- pkg/state/state_test.go | 358 ++++++++++++++++++++++++++++++------- 4 files changed, 359 insertions(+), 71 deletions(-) diff --git a/pkg/settings/features.go b/pkg/settings/features.go index 8a3dd4dca..2fa5f46c2 100644 --- a/pkg/settings/features.go +++ b/pkg/settings/features.go @@ -60,3 +60,14 @@ var FeaturesInfo = map[Feature]FeatureInfo{ BoostBlockReward: {true, "Boost Block Reward"}, InvokeExpression: {false, "InvokeExpression"}, } + +// LastFeature returns the last implemented feature. +func LastFeature() Feature { + var f Feature + for feature, info := range FeaturesInfo { + if info.Implemented { + f = max(f, feature) + } + } + return f +} diff --git a/pkg/state/common_test.go b/pkg/state/common_test.go index 1135ea276..440ea0f33 100644 --- a/pkg/state/common_test.go +++ b/pkg/state/common_test.go @@ -357,8 +357,9 @@ func createStorageObjects(t *testing.T, amend bool) *testStorageObjects { } type testStorageObjectsOptions struct { - Amend bool - Settings *settings.BlockchainSettings + Amend bool + Settings *settings.BlockchainSettings + CalculateHashes bool } func createStorageObjectsWithOptions(t *testing.T, options testStorageObjectsOptions) *testStorageObjects { @@ -388,7 +389,7 @@ func createStorageObjectsWithOptions(t *testing.T, options testStorageObjectsOpt hs, err := newHistoryStorage(db, dbBatch, stateDB, options.Amend) require.NoError(t, err) - entities, err := newBlockchainEntitiesStorage(hs, options.Settings, rw, false) + entities, err := newBlockchainEntitiesStorage(hs, options.Settings, rw, options.CalculateHashes) require.NoError(t, err) return &testStorageObjects{db, dbBatch, rw, hs, stateDB, options.Settings, entities} @@ -454,6 +455,12 @@ func (s *testStorageObjects) finishBlock(t *testing.T, blockID proto.BlockID) { assert.NoError(t, err, "finishBlock() failed") } +func (s *testStorageObjects) addBlockAndDo(t *testing.T, blockID proto.BlockID, f func(proto.BlockID)) { + s.prepareAndStartBlock(t, blockID) + f(blockID) + s.finishBlock(t, blockID) +} + func (s *testStorageObjects) addBlocks(t *testing.T, blocksNum int) { ids := genRandBlockIds(t, blocksNum) for _, id := range ids { @@ -504,6 +511,39 @@ func (s *testStorageObjects) createSmartAsset(t *testing.T, assetID crypto.Diges s.flush(t) } +func (s *testStorageObjects) setWavesBalance( + t *testing.T, + addr proto.WavesAddress, + bp balanceProfile, + blockID proto.BlockID, +) { + wb := newWavesValueFromProfile(bp) + err := s.entities.balances.setWavesBalance(addr.ID(), wb, blockID) + assert.NoError(t, err, "setWavesBalance() failed") +} + +func (s *testStorageObjects) transferWaves( + t *testing.T, + from, to proto.WavesAddress, + amount uint64, + blockID proto.BlockID, +) { + fromBP, err := s.entities.balances.newestWavesBalance(from.ID()) + require.NoError(t, err, "newestWavesBalance() failed") + if fromBalance := fromBP.spendableBalance(); fromBalance < amount { + require.Failf(t, "transferWaves()", "not enough balance at account '%s': %d < %d", + from.String(), fromBalance, amount, + ) + } + fromBP.balance -= amount + s.setWavesBalance(t, from, fromBP, blockID) + + toBalance, err := s.entities.balances.newestWavesBalance(to.ID()) + require.NoError(t, err, "newestWavesBalance() failed") + toBalance.balance += amount + s.setWavesBalance(t, to, toBalance, blockID) +} + func storeScriptByAddress( stor *blockchainEntitiesStorage, scheme proto.Scheme, @@ -542,11 +582,15 @@ func (s *testStorageObjects) setScript(t *testing.T, pk crypto.PublicKey, script require.NoError(t, err) } -func (s *testStorageObjects) activateFeature(t *testing.T, featureID int16) { - s.addBlock(t, blockID0) +func (s *testStorageObjects) activateFeatureWithBlock(t *testing.T, featureID int16, blockID proto.BlockID) { activationReq := &activatedFeaturesRecord{1} - err := s.entities.features.activateFeature(featureID, activationReq, blockID0) + err := s.entities.features.activateFeature(featureID, activationReq, blockID) assert.NoError(t, err, "activateFeature() failed") +} + +func (s *testStorageObjects) activateFeature(t *testing.T, featureID int16) { + s.addBlock(t, blockID0) + s.activateFeatureWithBlock(t, featureID, blockID0) s.flush(t) } diff --git a/pkg/state/script_caller.go b/pkg/state/script_caller.go index 0f7b82d31..5aa93e91c 100644 --- a/pkg/state/script_caller.go +++ b/pkg/state/script_caller.go @@ -408,10 +408,13 @@ func (a *scriptCaller) invokeFunctionByEthereumTx( } // Since V5 we have to create environment with wrapped state to which we put attached payments if tree.LibVersion >= ast.LibV5 { + const checkSenderBalance = false // skip initial payments validation for eth tx, see PR #965 for more info //TODO: Update last argument of the followinxg call with new feature activation flag or // something else depending on NODE-2531 issue resolution in scala implementation. isPbTx := proto.IsProtobufTx(tx) - env, err = ride.NewEnvironmentWithWrappedState(env, a.state, scriptPayments, sender, isPbTx, tree.LibVersion, false) + env, err = ride.NewEnvironmentWithWrappedState(env, a.state, scriptPayments, sender, + isPbTx, tree.LibVersion, checkSenderBalance, + ) if err != nil { return nil, proto.FunctionCall{}, errors.Wrap(err, "failed to create RIDE environment with wrapped state") } diff --git a/pkg/state/state_test.go b/pkg/state/state_test.go index 768e7817c..105b053c8 100644 --- a/pkg/state/state_test.go +++ b/pkg/state/state_test.go @@ -2,6 +2,8 @@ package state import ( "context" + "encoding/binary" + stderrs "errors" "fmt" "math/big" "path/filepath" @@ -18,7 +20,11 @@ import ( "github.com/wavesplatform/gowaves/pkg/importer" "github.com/wavesplatform/gowaves/pkg/keyvalue" "github.com/wavesplatform/gowaves/pkg/proto" + "github.com/wavesplatform/gowaves/pkg/ride" + "github.com/wavesplatform/gowaves/pkg/ride/ast" + ridec "github.com/wavesplatform/gowaves/pkg/ride/compiler" "github.com/wavesplatform/gowaves/pkg/settings" + "github.com/wavesplatform/gowaves/pkg/types" ) const ( @@ -469,62 +475,61 @@ type timeMock struct{} func (timeMock) Now() time.Time { return time.Now().UTC() } -func TestGeneratingBalanceValuesForNewestFunctions(t *testing.T) { - createMockStateManager := func(t *testing.T, bs *settings.BlockchainSettings) (*stateManager, *testStorageObjects) { - const ( - handleAmend = true - calculateHashes = true - enableLightNode = false - verificationGoroutinesNum = 2 - provideExtendedAPI = true - ) - toOpts := testStorageObjectsOptions{Amend: handleAmend, Settings: bs} - to := createStorageObjectsWithOptions(t, toOpts) - - stor, err := newBlockchainEntitiesStorage(to.hs, to.settings, to.rw, calculateHashes) - require.NoError(t, err, "newBlockchainEntitiesStorage() failed") - - blockStorageDir := t.TempDir() - atxParams := &addressTransactionsParams{ - dir: blockStorageDir, - batchedStorMemLimit: proto.KiB, - batchedStorMaxKeys: AddressTransactionsMaxKeys, - maxFileSize: 2 * proto.KiB, - providesData: provideExtendedAPI, - } - atx, err := newAddressTransactions(to.db, to.stateDB, to.rw, atxParams, handleAmend) - require.NoError(t, err, "newAddressTransactions() failed") - - state := &stateManager{ - mu: new(sync.RWMutex), - lastBlock: atomic.Value{}, - genesis: new(proto.Block), // stub - stateDB: to.stateDB, - stor: stor, - rw: to.rw, - settings: to.settings, - cv: nil, // filled in later - appender: nil, // filled in later - atx: atx, - verificationGoroutinesNum: verificationGoroutinesNum, - newBlocks: newNewBlocks(to.rw, to.settings), - enableLightNode: enableLightNode, - } - snapshotApplier := newBlockSnapshotsApplier(nil, newSnapshotApplierStorages(stor, to.rw)) - appender, err := newTxAppender( - state, - state.rw, - state.stor, - state.settings, - state.stateDB, - state.atx, - &snapshotApplier, - ) - require.NoError(t, err, "newTxAppender() failed") - state.appender = appender - state.cv = consensus.NewValidator(state, state.settings, timeMock{}) - return state, to +func createMockStateManager(t *testing.T, bs *settings.BlockchainSettings) (*stateManager, *testStorageObjects) { + const ( + handleAmend = true + calculateHashes = false + enableLightNode = false + verificationGoroutinesNum = 2 + provideExtendedAPI = true + ) + toOpts := testStorageObjectsOptions{Amend: handleAmend, Settings: bs, CalculateHashes: calculateHashes} + to := createStorageObjectsWithOptions(t, toOpts) + stor := to.entities + + blockStorageDir := t.TempDir() + atxParams := &addressTransactionsParams{ + dir: blockStorageDir, + batchedStorMemLimit: proto.KiB, + batchedStorMaxKeys: AddressTransactionsMaxKeys, + maxFileSize: 2 * proto.KiB, + providesData: provideExtendedAPI, } + atx, err := newAddressTransactions(to.db, to.stateDB, to.rw, atxParams, handleAmend) + require.NoError(t, err, "newAddressTransactions() failed") + + state := &stateManager{ + mu: new(sync.RWMutex), + lastBlock: atomic.Value{}, + genesis: new(proto.Block), // stub + stateDB: to.stateDB, + stor: stor, + rw: to.rw, + settings: to.settings, + cv: nil, // filled in later + appender: nil, // filled in later + atx: atx, + verificationGoroutinesNum: verificationGoroutinesNum, + newBlocks: newNewBlocks(to.rw, to.settings), + enableLightNode: enableLightNode, + } + snapshotApplier := newBlockSnapshotsApplier(nil, newSnapshotApplierStorages(stor, to.rw)) + appender, err := newTxAppender( + state, + state.rw, + state.stor, + state.settings, + state.stateDB, + state.atx, + &snapshotApplier, + ) + require.NoError(t, err, "newTxAppender() failed") + state.appender = appender + state.cv = consensus.NewValidator(state, state.settings, timeMock{}) + return state, to +} + +func TestGeneratingBalanceValuesForNewestFunctions(t *testing.T) { const ( initialBalance = 100 changedBalance = 200 @@ -538,17 +543,13 @@ func TestGeneratingBalanceValuesForNewestFunctions(t *testing.T) { // add initial balance at first block testObj.addBlock(t, blockID0) - initialBP := newWavesValueFromProfile(balanceProfile{initialBalance, 0, 0}) - err := state.stor.balances.setWavesBalance(addr.ID(), initialBP, blockID0) // height 1 - require.NoError(t, err, "setWavesBalance() failed") + testObj.setWavesBalance(t, addr, balanceProfile{initialBalance, 0, 0}, blockID0) // height 1 // add changed balance at second block testObj.addBlock(t, blockID1) - changedBP := newWavesValueFromProfile(balanceProfile{changedBalance, 0, 0}) - err = state.stor.balances.setWavesBalance(addr.ID(), changedBP, blockID1) // height 2 - require.NoError(t, err, "setWavesBalance() failed") - - testObj.addBlocks(t, blocksToApply-2) // add 998 random blocks, 2 blocks have already been added - + testObj.setWavesBalance(t, addr, balanceProfile{changedBalance, 0, 0}, blockID1) // height 2 + // add 998 random blocks, 2 blocks have already been added + testObj.addBlocks(t, blocksToApply-2) + // check blockchain height nh, err := state.NewestHeight() require.NoError(t, err, "NewestHeight() failed") require.Equal(t, uint64(blocksToApply), nh) // sanity check, blockchain height should be 1000 @@ -640,3 +641,232 @@ func TestGeneratingBalanceValuesForNewestFunctions(t *testing.T) { assert.Equal(t, uint64(changedBalance), newGB) // result should be the same }) } + +type stateForEnv interface { + StateInfo + types.EnrichedSmartState +} + +func createNewRideEnv( + t *testing.T, + state stateForEnv, + dApp, caller proto.WavesAddress, + rootLibV ast.LibraryVersion, +) *ride.EvaluationEnvironment { + blockV5, err := state.IsActivated(int16(settings.BlockV5)) + require.NoError(t, err, "IsActivated() failed for feature BlockV5") + rideV6, err := state.IsActivated(int16(settings.RideV6)) + require.NoError(t, err, "IsActivated() failed for feature RideV6") + consensusImprovements, err := state.IsActivated(int16(settings.ConsensusImprovements)) + require.NoError(t, err, "IsActivated() failed for feature ConsensusImprovements") + blockRewardDistribution, err := state.IsActivated(int16(settings.BlockRewardDistribution)) + require.NoError(t, err, "IsActivated() failed for feature BlockRewardDistribution") + lightNode, err := state.IsActivated(int16(settings.LightNode)) + require.NoError(t, err, "IsActivated() failed for feature LightNode") + bs, err := state.BlockchainSettings() + require.NoError(t, err, "BlockchainSettings() failed") + var ( + internalPaymentsValidationHeight = bs.InternalInvokePaymentsValidationAfterHeight + paymentsFixAfterHeight = bs.PaymentsFixAfterHeight + ) + origEnv, err := ride.NewEnvironment( + bs.AddressSchemeCharacter, + state, + internalPaymentsValidationHeight, paymentsFixAfterHeight, + blockV5, rideV6, consensusImprovements, blockRewardDistribution, lightNode, + ) + require.NoError(t, err, "ride.NewEnvironment() failed") + origEnv.SetThisFromAddress(dApp) + complexity, err := ride.MaxChainInvokeComplexityByVersion(rootLibV) + require.NoError(t, err, "MaxChainInvokeComplexityByVersion() failed") + origEnv.SetLimit(complexity) + const ( + isProtobufTransaction = true // assume that transaction is protobuf + checkSenderBalance = true // check initial sender balance for payments + ) + var payments proto.ScriptPayments // no payments + env, err := ride.NewEnvironmentWithWrappedState(origEnv, state, + payments, caller, isProtobufTransaction, rootLibV, checkSenderBalance, + ) + require.NoError(t, err, "ride.NewEnvironmentWithWrappedState() failed") + return env +} + +// TestGeneratingBalanceValuesInRide tests that generating balance values are calculated correctly in Ride scripts. +// It's analogous to RideGeneratingBalanceSpec in scala node tests. +func TestGeneratingBalanceValuesInRide(t *testing.T) { + createTestScript := func(t *testing.T, libV ast.LibraryVersion) *ast.Tree { + //nolint:lll // keep original formatting of the script + const scriptTemplate = ` + {-# STDLIB_VERSION %d #-} + {-# CONTENT_TYPE DAPP #-} + {-# SCRIPT_TYPE ACCOUNT #-} + + @Callable(i) + func assertBalances( + expectedRegularBalance: Int, + expectedAvailableBalance: Int, + expectedEffectiveBalance: Int, + expectedGeneratingBalance: Int + ) = { + let actualRegularBalance = wavesBalance(this).regular + let actualAvailableBalance = wavesBalance(this).available + let actualEffectiveBalance = wavesBalance(this).effective + let actualGeneratingBalance = wavesBalance(this).generating + + strict checkRegular = if (actualRegularBalance != expectedRegularBalance) + then throw("Expected Regular balance to be: " + toString(expectedRegularBalance) + ", But got: " + toString(actualRegularBalance)) + else unit + + strict checkAvailable = if (actualAvailableBalance != expectedAvailableBalance) + then throw("Expected Available balance to be: " + toString(expectedAvailableBalance) + ", But got: " + toString(actualAvailableBalance)) + else unit + + strict checkEffective = if (actualEffectiveBalance != expectedEffectiveBalance) + then throw("Expected Effective balance to be: " + toString(expectedEffectiveBalance) + ", But got: " + toString(actualEffectiveBalance)) + else unit + + strict checkGenerating = if (actualGeneratingBalance != expectedGeneratingBalance) + then throw("Expected Generating balance to be: " + toString(expectedGeneratingBalance) + ", But got: " + toString(actualGeneratingBalance)) + else unit + ([], unit) + }` + scriptSrc := fmt.Sprintf(scriptTemplate, libV) + tree, errs := ridec.CompileToTree(scriptSrc) + require.NoError(t, stderrs.Join(errs...), "ride.CompileToTree() failed") + return tree + } + doTest := func(t *testing.T, state *stateManager, testObj *testStorageObjects, libV ast.LibraryVersion) { + // create test accounts + dApp, err := proto.NewKeyPair(binary.BigEndian.AppendUint32(nil, 999)) + require.NoError(t, err, "NewKeyPair() failed") + anotherAccount, err := proto.NewKeyPair(binary.BigEndian.AppendUint32(nil, 1)) + require.NoError(t, err, "NewKeyPair() failed") + // create test addresses + bs, bsErr := state.BlockchainSettings() + require.NoError(t, bsErr, "BlockchainSettings() failed") + caller, aErr := anotherAccount.Addr(bs.AddressSchemeCharacter) + require.NoError(t, aErr, "Addr() failed") + dAppAddr, aErr := dApp.Addr(bs.AddressSchemeCharacter) + require.NoError(t, aErr, "Addr() failed") + // create test script + tree := createTestScript(t, libV) + // create assertion function for the current state + assertBalances := func(t *testing.T, regular, available, effective, generating int64) { + fc := proto.NewFunctionCall("assertBalances", proto.Arguments{ + proto.NewIntegerArgument(regular), + proto.NewIntegerArgument(available), + proto.NewIntegerArgument(effective), + proto.NewIntegerArgument(generating), + }) + env := createNewRideEnv(t, state, dAppAddr, caller, libV) + _, err = ride.CallFunction(env, tree, fc) + require.NoError(t, err, "ride.CallFunction() failed") + } + assertHeight := func(t *testing.T, expectedHeight int) { + nh, hErr := state.NewestHeight() + require.NoError(t, hErr, "NewestHeight() failed") + require.Equal(t, proto.Height(expectedHeight), nh) + } + assertHeight(t, 1) // check that height is 1 + // set initial balance for dApp and another account + const ( + initialDAppBalance = 100 * proto.PriceConstant + initialAnotherAccountBalance = 500 * proto.PriceConstant + firstTransferAmount = 10 * proto.PriceConstant + secondTransferAmount = 50 * proto.PriceConstant + ) + testObj.setWavesBalance(t, dAppAddr, balanceProfile{initialDAppBalance, 0, 0}, blockID0) // height 1 + testObj.setWavesBalance(t, caller, balanceProfile{initialAnotherAccountBalance, 0, 0}, blockID0) // height 1 + + dAppBalance := int64(initialDAppBalance) + testObj.addBlockAndDo(t, blockID1, func(_ proto.BlockID) { // height 2 + assertBalances(t, dAppBalance, dAppBalance, dAppBalance, dAppBalance) + }) + assertHeight(t, 2) // check that height is 2 + + testObj.addBlockAndDo(t, blockID2, func(blockID proto.BlockID) { // height 3 + testObj.transferWaves(t, caller, dAppAddr, firstTransferAmount, blockID) // transfer 10 waves from caller to dApp + dAppBalance += firstTransferAmount // update dApp balance + }) + assertHeight(t, 3) // check that height is 3 + // add 997 blocks + testObj.addBlocks(t, 1000-3) // add 997 blocks + assertHeight(t, 1000) // check that height is 1000 + + // Block 1001 + // This assertion tells us that the generating balance + // is not being updated until the block 1002, which is expected, + // because 10 waves was sent on height = 3, + // and until height 1002 the balance is not updated + // (...the lowest of the last 1000 blocks, including 3 and 1002) + testObj.addBlockAndDo(t, blockID3, func(_ proto.BlockID) { // height 1000 + assertBalances(t, dAppBalance, dAppBalance, dAppBalance, initialDAppBalance) + }) + assertHeight(t, 1001) // check that height is 1001 + + // Block 1002 + testObj.addBlockAndDo(t, genBlockId(42), func(blockID proto.BlockID) { // height 1001 + // This assertion tells us that the generating balance + // was already updated after 10 waves was sent on height = 3 + assertBalances(t, dAppBalance, dAppBalance, dAppBalance, dAppBalance) + testObj.transferWaves(t, dAppAddr, caller, secondTransferAmount, blockID) // transfer 50 waves from dApp to caller + dAppBalance -= secondTransferAmount // update dApp balance + // This assertion tells us that the generating balance + // was updated by a transaction in this block. + assertBalances(t, dAppBalance, dAppBalance, dAppBalance, dAppBalance) + }) + assertHeight(t, 1002) // check that height is 1002 + } + t.Run("The generating balance is affected by transactions in the current block", func(t *testing.T) { + generateFeaturesList := func(targetFeature settings.Feature) []settings.Feature { + var feats []settings.Feature + for f := settings.SmallerMinimalGeneratingBalance; f <= targetFeature; f++ { + feats = append(feats, f) + } + return feats + } + activateFeatures := func(t *testing.T, testObj *testStorageObjects, feats []settings.Feature, id proto.BlockID) { + for _, f := range feats { + testObj.activateFeatureWithBlock(t, int16(f), id) + } + } + createMockState := func(t *testing.T, targetFeature settings.Feature) (*stateManager, *testStorageObjects) { + sets := settings.MustDefaultCustomSettings() + sets.LightNodeBlockFieldsAbsenceInterval = 0 // disable absence interval for Light Node + sets.GenerationBalanceDepthFrom50To1000AfterHeight = 1 // set from the first height + state, testObj := createMockStateManager(t, sets) + featuresList := generateFeaturesList(targetFeature) + testObj.addBlock(t, blockID0) // add "genesis" block, height 1 + activateFeatures(t, testObj, featuresList, blockID0) // activate features at height 1 + testObj.flush(t) // write changes to the storage + return state, testObj + } + t.Run("RideV5, STDLIB_VERSION 5", func(t *testing.T) { + state, testObj := createMockState(t, settings.RideV5) + doTest(t, state, testObj, ast.LibV5) + }) + t.Run("RideV6, STDLIB_VERSION 6", func(t *testing.T) { + state, testObj := createMockState(t, settings.RideV6) + doTest(t, state, testObj, ast.LibV6) + }) + t.Run("ConsensusImprovements, STDLIB_VERSION 6", func(t *testing.T) { + state, testObj := createMockState(t, settings.ConsensusImprovements) + doTest(t, state, testObj, ast.LibV6) + }) + // scala name is "ContinuationTransaction, STDLIB_VERSION 6", it means that all possible features are activated + t.Run("AllFeatures, STDLIB_VERSION 6", func(t *testing.T) { + state, testObj := createMockState(t, settings.LastFeature()) // all features are activated + doTest(t, state, testObj, ast.LibV6) + }) + t.Run("BlockRewardDistribution, STDLIB_VERSION 7", func(t *testing.T) { + state, testObj := createMockState(t, settings.BlockRewardDistribution) + doTest(t, state, testObj, ast.LibV7) + }) + // scala name is "TransactionStateSnapshot, STDLIB_VERSION 8", TransactionStateSnapshot == LightNode + t.Run("LightNode, STDLIB_VERSION 8", func(t *testing.T) { + state, testObj := createMockState(t, settings.LightNode) + doTest(t, state, testObj, ast.LibV8) + }) + }) +}