From 532eafecf5c31f6e26932c62009992cdf454b568 Mon Sep 17 00:00:00 2001 From: dCorral <55594560+dcorral@users.noreply.github.com> Date: Mon, 6 Mar 2023 09:05:15 +0100 Subject: [PATCH] Test framework: Initial impl of exhaustive state verification with rollback (#1713) * Adds rollback decorator * Sets decorator in main tests. Commiting for testing CI. * Fixes index error accessing self.nodes list * Fix typo. * Revert "Sets decorator in main tests." This reverts commit ba8b4b6153d75949bc64e6217c15f446762d114a. * Add rollback parameter detection in decorator * Fix lint error and test_runner nomeclature * Update test_runner.py * Fix lint * Fix rollback_to call * Remove function rollback parameter Rename decorator and fix tests * Rename decorator * Add decorator warning and comments * Fix lint error --------- Co-authored-by: Prasanna Loganathar --- test/functional/feature_framework_rollback.py | 137 ++++++++++++++++++ .../test_framework/test_framework.py | 37 +++++ test/functional/test_runner.py | 1 + 3 files changed, 175 insertions(+) create mode 100755 test/functional/feature_framework_rollback.py diff --git a/test/functional/feature_framework_rollback.py b/test/functional/feature_framework_rollback.py new file mode 100755 index 0000000000..40b26c6543 --- /dev/null +++ b/test/functional/feature_framework_rollback.py @@ -0,0 +1,137 @@ +#!/usr/bin/env python3 +# Copyright (c) 2014-2019 The Bitcoin Core developers +# Copyright (c) DeFi Blockchain Developers +# Distributed under the MIT software license, see the accompanying +# file LICENSE or http://www.opensource.org/licenses/mit-license.php. +"""Test Loan - loan basics.""" + +from test_framework.test_framework import DefiTestFramework +from test_framework.util import assert_equal, assert_raises_rpc_error + +class RollbackFrameworkTest (DefiTestFramework): + def set_test_params(self): + self.num_nodes = 4 + self.setup_clean_chain = True + self.extra_args = [ + ['-txnotokens=0', '-amkheight=50', '-bayfrontheight=50', '-bayfrontgardensheight=1', '-fortcanningheight=50', '-eunosheight=50', '-fortcanninghillheight=50', '-fortcanningparkheight=50', '-fortcanningroadheight=50', '-fortcanningcrunchheight=50', '-fortcanningspringheight=50', '-fortcanninggreatworldheight=250', '-grandcentralheight=254', '-grandcentralepilogueheight=350', '-regtest-minttoken-simulate-mainnet=1', '-txindex=1'], + ['-txnotokens=0', '-amkheight=50', '-bayfrontheight=50', '-bayfrontgardensheight=1', '-fortcanningheight=50', '-eunosheight=50', '-fortcanninghillheight=50', '-fortcanningparkheight=50', '-fortcanningroadheight=50', '-fortcanningcrunchheight=50', '-fortcanningspringheight=50', '-fortcanninggreatworldheight=250', '-grandcentralheight=254', '-grandcentralepilogueheight=350', '-regtest-minttoken-simulate-mainnet=1', '-txindex=1'], + ['-txnotokens=0', '-amkheight=50', '-bayfrontheight=50', '-bayfrontgardensheight=1', '-fortcanningheight=50', '-eunosheight=50', '-fortcanninghillheight=50', '-fortcanningparkheight=50', '-fortcanningroadheight=50', '-fortcanningcrunchheight=50', '-fortcanningspringheight=50', '-fortcanninggreatworldheight=250', '-grandcentralheight=254', '-grandcentralepilogueheight=350', '-regtest-minttoken-simulate-mainnet=1', '-txindex=1'], + ['-txnotokens=0', '-amkheight=50', '-bayfrontheight=50', '-bayfrontgardensheight=1', '-fortcanningheight=50', '-eunosheight=50', '-fortcanninghillheight=50', '-fortcanningparkheight=50', '-fortcanningroadheight=50', '-fortcanningcrunchheight=50', '-fortcanningspringheight=50', '-fortcanninggreatworldheight=250', '-grandcentralheight=254', '-grandcentralepilogueheight=350', '-regtest-minttoken-simulate-mainnet=1', '-txindex=1']] + + + def init_chain(self): + print("Generating initial chain...") + self.nodes[0].generate(100) + self.sync_blocks() + + def set_accounts(self, rollback=True): + self.account0 = self.nodes[0].get_genesis_keys().ownerAuthAddress + self.account1 = self.nodes[1].getnewaddress("", "legacy") + self.account2 = self.nodes[2].get_genesis_keys().ownerAuthAddress + self.account3 = self.nodes[3].get_genesis_keys().ownerAuthAddress + self.nodes[1].generate(20) + self.sync_blocks() + + self.nodes[0].sendtoaddress(self.account1, 10) + self.nodes[0].sendtoaddress(self.account2, 10) + self.nodes[0].sendtoaddress(self.account3, 10) + self.nodes[0].generate(1) + self.sync_blocks() + + @DefiTestFramework.capture_rollback_verify + def set_accounts_with_rollback(self): + self.set_accounts() + + + def create_tokens(self, rollback=None): + self.symbolBTC = "BTC" + self.symbolDOGE = "DOGE" + + self.nodes[0].createtoken({ + "symbol": self.symbolBTC, + "name": "BTC token", + "isDAT": True, + "collateralAddress": self.account0 + }) + + self.nodes[0].generate(1) + self.sync_blocks() + + self.nodes[0].minttokens(["2@" + self.symbolBTC]) + + self.nodes[0].createtoken({ + "symbol": self.symbolDOGE, + "name": "DOGE token", + "isDAT": True, + "collateralAddress": self.account3 + }) + + self.nodes[0].generate(1) + self.sync_blocks() + + self.idBTC = list(self.nodes[0].gettoken(self.symbolBTC).keys())[0] + self.idDOGE = list(self.nodes[0].gettoken(self.symbolDOGE).keys())[0] + + assert_raises_rpc_error(-32600, "called before GrandCentral height", self.nodes[0].burntokens, { + 'amounts': "1@" + self.symbolBTC, + 'from': self.account0, + }) + + self.nodes[0].generate(254 - self.nodes[0].getblockcount()) + self.sync_blocks() + + # DAT owner can mint + self.nodes[3].minttokens(["1@" + self.symbolDOGE]) + self.nodes[3].generate(1) + self.sync_blocks() + + @DefiTestFramework.capture_rollback_verify + def create_tokens_with_rollback(self): + self.create_tokens() + + def mint_extra(self, rollback=None): + self.nodes[3].minttokens(["1@" + self.symbolDOGE]) + self.nodes[3].generate(1) + self.sync_blocks() + + @DefiTestFramework.capture_rollback_verify + def mint_extra_with_rollback(self): + self.mint_extra() + + def run_test(self): + + self.init_chain() + height = self.nodes[0].getblockcount() # block 100 + + # rollback + self.set_accounts_with_rollback() + height1 = self.nodes[1].getblockcount() + assert_equal(height, height1) + + # no rollback + self.set_accounts(rollback=False) + height2 = self.nodes[3].getblockcount() + assert(height != height2) + + # rollback + self.create_tokens_with_rollback() + height3 = self.nodes[0].getblockcount() + assert_equal(height2, height3) + + # no rollback + self.create_tokens() + height4 = self.nodes[0].getblockcount() + assert(height3 != height4) + + # rollback + self.mint_extra_with_rollback() + height5 = self.nodes[0].getblockcount() + assert_equal(height5, height4) + + # no rollback + self.mint_extra() + height6 = self.nodes[0].getblockcount() + assert(height6 != height5) + +if __name__ == '__main__': + RollbackFrameworkTest().main() diff --git a/test/functional/test_framework/test_framework.py b/test/functional/test_framework/test_framework.py index 2598d6a11a..a4dc40f5dc 100755 --- a/test/functional/test_framework/test_framework.py +++ b/test/functional/test_framework/test_framework.py @@ -105,6 +105,25 @@ def __init__(self): assert hasattr(self, "num_nodes"), "Test must set self.num_nodes in set_test_params()" + # Captures the chain data, does a rollback and checks data has been restored + def _check_rollback(self, func, *args, **kwargs): + init_height = self.nodes[0].getblockcount() + init_data = self._get_chain_data() + result = func(self, *args, **kwargs) + self.rollback_to(init_height) + final_data = self._get_chain_data() + final_height = self.nodes[0].getblockcount() + assert(init_data == final_data) + assert(init_height == final_height) + return result + + # WARNING: This decorator uses _get_chain_data() internally which can be an expensive call if used in large test scenarios. + @classmethod + def capture_rollback_verify(cls, func): + def wrapper(self, *args, **kwargs): + return self._check_rollback(func, *args, **kwargs) + return wrapper + def main(self): """Main function. This should not be overridden by the subclass test scripts.""" @@ -440,6 +459,24 @@ def rollback_to(self, block, nodes=None): for x in connections[node]: connect_nodes(node, x) + # build the data obj to be checked pre and post rollback + def _get_chain_data(self): + return [ + self.nodes[0].logaccountbalances(), + self.nodes[0].logstoredinterests(), + self.nodes[0].listvaults(), + self.nodes[0].listtokens(), + self.nodes[0].listgovs(), + self.nodes[0].listmasternodes(), + self.nodes[0].listaccounthistory(), + self.nodes[0].getburninfo(), + self.nodes[0].getloaninfo(), + self.nodes[0].listanchors(), + self.nodes[0].listgovproposals(), + self.nodes[0].listburnhistory(), + self.nodes[0].listcommunitybalances() + ] + def run_test(self): """Tests must override this method to define test logic""" raise NotImplementedError diff --git a/test/functional/test_runner.py b/test/functional/test_runner.py index af1fdbfaaf..f7cc843775 100755 --- a/test/functional/test_runner.py +++ b/test/functional/test_runner.py @@ -318,6 +318,7 @@ 'feature_update_mn.py', 'feature_block_reward.py', 'feature_negative_interest.py', + 'feature_framework_rollback.py', 'rpc_getstoredinterest.py', 'feature_dusd_loans.py', # Don't append tests at the end to avoid merge conflicts