#!/usr/bin/env python3 # # MMGen Wallet, a terminal-based cryptocurrency wallet # Copyright (C)2013-2025 The MMGen Project # Licensed under the GNU General Public License, Version 3: # https://www.gnu.org/licenses # Public project repositories: # https://github.com/mmgen/mmgen-wallet # https://gitlab.com/mmgen/mmgen-wallet """ test.cmdtest_d.ethbump: Ethereum transaction bumping tests for the cmdtest.py test suite """ import sys, time, asyncio, json from mmgen.cfg import Config from mmgen.protocol import init_proto from mmgen.util import ymsg, suf from ..include.common import imsg, omsg_r from .include.common import cleanup_env, dfl_words_file from .include.runner import CmdTestRunner from .httpd.thornode import ThornodeServer from .ethdev import CmdTestEthdev, CmdTestEthdevMethods, dfl_sid from .regtest import CmdTestRegtest from .swap import CmdTestSwapMethods thornode_server = ThornodeServer() burn_addr = 'beefcafe22' * 4 method_template = """ def {name}(self): self.spawn(log_only=True) return ethbump_ltc.run_test("{ltc_name}", sub=True) """ class CmdTestEthBumpMethods: def init_block_period(self): d = self.daemon bp = self.cfg.devnet_block_period or self.dfl_devnet_block_period[d.id] d.usr_coind_args = { 'reth': [f'--dev.block-time={bp}s'], 'geth': [f'--dev.period={bp}'] }[d.id] self.devnet_block_period = bp def _txcreate(self, args, acct): self.get_file_with_ext('rawtx', delete_all=True) return self.txcreate(args, acct=acct, interactive_fee='0.9G', fee_info_data=('0.0000189', '0.9')) def _txsign(self, has_label=True): self.get_file_with_ext('sigtx', delete_all=True) return self.txsign(has_label=has_label) def _txsend(self, has_label=True): return self.txsend(has_label=has_label) def _txbump_feebump(self, *args, **kwargs): self.get_file_with_ext('rawtx', delete_all=True) return self._txbump(*args, **kwargs) def _txbump_new_outputs(self, *, args, fee, add_opts=[]): self.get_file_with_ext('rawtx', delete_all=True) ext = '{}.regtest.sigtx'.format('-α' if self.cfg.debug_utf8 else '') txfile = self.get_file_with_ext(ext, no_dot=True) return self.txbump_ui_common( self.spawn('mmgen-txbump', self.eth_opts + add_opts + args + [txfile]), fee = fee, fee_desc = 'or gas price', bad_fee = '0.9G') def _token_txcreate(self, *, args, cmd='txcreate'): self.get_file_with_ext('sigtx', delete_all=True) t = self._create_token_tx(cmd=cmd, fee='1.3G', args=args, add_opts=self.eth_opts) t.expect('to confirm: ', 'YES\n') t.written_to_file('Sent transaction') return t async def _wait_for_block(self, require_seen=True): self.spawn(msg_only=True) if self.devnet_block_period: empty_pools_seen = 0 tx_seen = False while True: await asyncio.sleep(1) t = self.spawn( 'mmgen-cli', ['--regtest=1', 'txpool_content'], env = cleanup_env(self.cfg), no_msg = True, silent = True) res = json.loads(t.read().strip()) if p := res['pending']: imsg(f'Pool has {len(p)} transaction{suf(p)}') if self.tr.quiet: omsg_r('+') tx_seen = True else: imsg('Pool is empty') if self.tr.quiet: omsg_r('+') if tx_seen or not require_seen: break empty_pools_seen += 1 if empty_pools_seen > 5: m = ('\nTransaction pool empty! Try increasing the block period with the' ' --devnet-block-period option (current value is {})') ymsg(m.format(self.devnet_block_period)) sys.exit(1) return 'ok' class CmdTestEthBump(CmdTestEthBumpMethods, CmdTestEthdev, CmdTestSwapMethods): 'Ethereum transaction bumping operations' networks = ('eth',) tmpdir_nums = [42] dfl_devnet_block_period = {'geth': 7, 'reth': 9} cmd_group_in = ( ('subgroup.ltc_init', []), ('subgroup.eth_init', []), ('subgroup.feebump', ['eth_init']), ('subgroup.new_outputs', ['eth_init']), ('subgroup.swap_feebump', ['ltc_init', 'eth_init']), ('subgroup.swap_new_outputs', ['ltc_init', 'eth_init']), ('subgroup.token_init', ['eth_init']), ('subgroup.token_feebump', ['token_init']), ('subgroup.token_new_outputs', ['token_init']), ('stop', 'stopping daemon'), ) cmd_subgroups = { 'eth_init': ( 'initializing ETH tracking wallet', ('setup', 'dev mode transaction bumping tests for Ethereum (start daemon)'), ('addrgen', 'generating addresses'), ('addrimport', 'importing addresses'), ('addrimport_dev_addr', 'importing dev faucet address ‘Ox00a329c..’'), ('fund_dev_address', 'funding the default (Parity dev) address'), ('fund_mmgen_address1', 'spend from dev address to address :1)'), ('fund_mmgen_address2', 'spend from dev address to address :11)'), ('fund_mmgen_address3', 'spend from dev address to address :21)'), ('wait1', 'waiting for block'), ), 'ltc_init': ( 'initializing LTC tracking wallet', ('ltc_setup', 'regtest (Bob and Alice) mode setup'), ('ltc_walletconv_bob', 'wallet generation (Bob)'), ('ltc_addrgen_bob', 'address generation (Bob)'), ('ltc_addrimport_bob', 'importing Bob’s addresses'), ), 'feebump': ( 'creating, signing, sending, bumping and resending a transaction (fee-bump only)', ('txcreate1', 'creating a transaction (send to burn address)'), ('txsign1', 'signing the transaction'), ('txsend1', 'sending the transaction'), ('txbump1', 'creating a replacement transaction (fee-bump)'), ('txbump1sign', 'signing the replacement transaction'), ('txbump1send', 'sending the replacement transaction'), ('wait2', 'waiting for block'), ('bal1', 'checking the balance'), ), 'new_outputs': ( 'creating, signing, sending, bumping and resending a transaction (new outputs)', ('txcreate2', 'creating a transaction (send to burn address)'), ('txsign2', 'signing the transaction'), ('txsend2', 'sending the transaction'), ('txbump2', 'creating a replacement transaction (new outputs)'), ('txbump2sign', 'signing the replacement transaction'), ('txbump2send', 'sending the replacement transaction'), ('wait3', 'waiting for block'), ('bal2', 'checking the balance'), ), 'swap_feebump': ( 'creating, signing, sending, bumping and resending a swap transaction (fee-bump only)', ('swaptxcreate1', 'creating a swap transaction (from address :11)'), ('swaptxsign1', 'signing the transaction'), ('swaptxsend1', 'sending the transaction'), ('swaptxbump1', 'creating a replacement swap transaction (fee-bump)'), ('swaptxbump1sign', 'signing the replacement transaction'), ('swaptxbump1send', 'sending the replacement transaction'), ('wait4', 'waiting for block'), ('bal3', 'checking the balance'), ), 'swap_new_outputs': ( 'creating, signing, sending, bumping and resending a swap transaction (new output)', ('swaptxcreate2', 'creating a swap transaction (from address :21)'), ('swaptxsign2', 'signing the transaction'), ('swaptxsend2', 'sending the transaction'), ('swaptxbump2', 'creating a replacement swap transaction (new output)'), ('swaptxbump2sign', 'signing the replacement transaction'), ('swaptxbump2send', 'sending the replacement transaction'), ('wait5', 'waiting for block'), ('bal4', 'checking the balance'), ), 'token_init': ( 'initializing token wallets', ('token_compile1', 'compiling ERC20 token #1'), ('token_deploy_a', 'deploying ERC20 token MM1 (SafeMath)'), ('token_deploy_b', 'deploying ERC20 token MM1 (Owned)'), ('token_deploy_c', 'deploying ERC20 token MM1 (Token)'), ('wait_reth1', 'waiting for block'), ('token_fund_user', 'transferring token funds from dev to user'), ('wait6', 'waiting for block'), ('token_addrgen', 'generating token addresses'), ('token_addrimport', 'importing token addresses using token address (MM1)'), ('token_bal1', 'the token balance'), ), 'token_feebump': ( 'creating, signing, sending, bumping and resending a token transaction (fee-bump only)', ('token_txdo1', 'creating, signing and sending a token transaction'), ('token_txbump1', 'bumping the token transaction (fee-bump)'), ('wait7', 'waiting for block'), ('token_bal2', 'the token balance'), ), 'token_new_outputs': ( 'creating, signing, sending, bumping and resending a token transaction (new outputs)', ('token_txdo2', 'creating, signing and sending a token transaction'), ('token_txbump2', 'creating a replacement token transaction (new outputs)'), ('token_txbump2sign', 'signing the replacement transaction'), ('token_txbump2send', 'sending the replacement transaction'), ('wait8', 'waiting for block'), ('token_bal3', 'the token balance'), ) } ltc_tests = [c[0] for v in tuple(cmd_subgroups.values()) + (cmd_group_in,) for c in v if isinstance(c, tuple) and c[0].startswith('ltc_')] exec(''.join(method_template.format(name=k, ltc_name=k.removeprefix('ltc_')) for k in ltc_tests)) def __init__(self, cfg, trunner, cfgs, spawn): CmdTestEthdev.__init__(self, cfg, trunner, cfgs, spawn) if not trunner: return global ethbump_ltc cfg = Config({ '_clone': trunner.cfg, 'coin': 'ltc', 'resume': None, 'resume_after': None, 'exit_after': None, 'log': None}) t = trunner ethbump_ltc = CmdTestRunner(cfg, t.repo_root, t.data_dir, t.trash_dir, t.trash_dir2) ethbump_ltc.init_group('ethbump_ltc') thornode_server.start() def fund_mmgen_address1(self): return self._fund_mmgen_address(arg=f'{dfl_sid}:E:1,100000') def fund_mmgen_address2(self): return self._fund_mmgen_address(arg=f'{dfl_sid}:E:11,100000') def fund_mmgen_address3(self): return self._fund_mmgen_address(arg=f'{dfl_sid}:E:21,100000') def txcreate1(self): return self._txcreate(args=[f'{burn_addr},987'], acct='1') def txbump1(self): return self._txbump_feebump(fee='1.3G', ext='{}.regtest.sigtx') def txcreate2(self): return self._txcreate(args=[f'{burn_addr},789'], acct='1') def txbump2(self): return self._txbump_new_outputs(args=[f'{dfl_sid}:E:2,777'], fee='1.3G') def swaptxcreate1(self): return self._swaptxcreate_ui_common( self._swaptxcreate(['ETH', '12.34567', 'LTC', f'{dfl_sid}:B:3']), inputs = 4) def swaptxsign1(self): return self._swaptxsign() def swaptxsend1(self): return self._swaptxsend() def swaptxbump1(self): return self._swaptxbump('41.1G') def swaptxbump2(self): return self._swaptxbump('1.9G', output_args=[f'{dfl_sid}:E:12,4444.3333']) def bal1(self): return self._bal_check(pat=rf'{dfl_sid}:E:1\s+99012\.9999727\s') def bal2(self): return self._bal_check(pat=rf'{dfl_sid}:E:2\s+777\s') def bal3(self): return self._bal_check(pat=rf'{dfl_sid}:E:11\s+99987\.653431389777251448\s') def bal4(self): return self._bal_check(pat=rf'{dfl_sid}:E:12\s+4444\.3333\s') async def token_deploy_a(self): return await self._token_deploy_math(num=1, get_receipt=False) async def token_deploy_b(self): return await self._token_deploy_owned(num=1, get_receipt=False) async def token_deploy_c(self): return await self._token_deploy_token(num=1, get_receipt=False) def token_fund_user(self): return self._token_transfer_ops(op='fund_user', mm_idxs=[1], get_receipt=False) def token_addrgen(self): return self._token_addrgen(mm_idxs=[1], naddrs=5) def token_addrimport(self): return self._token_addrimport('token_addr1', '1-5', expect='5/5') def token_bal1(self): return self._token_bal_check(pat=rf'{dfl_sid}:E:1\s+1000\s') def token_txdo1(self): return self._token_txcreate(cmd='txdo', args=[f'{dfl_sid}:E:2,1.23456', dfl_words_file]) def token_txbump1(self): t = self._txbump_feebump( fee = '60G', ext = '{}.regtest.sigtx', add_opts = ['--token=MM1'], add_args = [dfl_words_file]) t.expect('to confirm: ', 'YES\n') t.written_to_file('Signed transaction') return t def token_bal2(self): return self._token_bal_check(pat=rf'{dfl_sid}:E:2\s+1\.23456') def token_txdo2(self): return self._token_txcreate(cmd='txdo', args=[f'{dfl_sid}:E:3,5.4321', dfl_words_file]) def token_txbump2(self): return self._txbump_new_outputs( args = [f'{dfl_sid}:E:4,6.54321'], fee = '1.6G', add_opts = ['--token=mm1']) def token_txbump2sign(self): return self._txsign(has_label=False) def token_txbump2send(self): return self._txsend(has_label=False) def token_bal3(self): return self._token_bal_check(pat=rf'{dfl_sid}:E:4\s+6\.54321') def wait_reth1(self): return self._wait_for_block() if self.daemon.id == 'reth' else 'silent' wait1 = wait2 = wait3 = wait4 = wait5 = wait6 = wait7 = wait8 = CmdTestEthBumpMethods._wait_for_block txsign1 = txsign2 = txbump1sign = txbump2sign = CmdTestEthBumpMethods._txsign txsend1 = txsend2 = txbump1send = txbump2send = CmdTestEthBumpMethods._txsend swaptxcreate2 = swaptxcreate1 swaptxsign2 = swaptxsign1 swaptxsend2 = swaptxsend1 swaptxbump1sign = swaptxbump2sign = token_txbump2sign swaptxbump1send = swaptxbump2send = token_txbump2send class CmdTestEthBumpLTC(CmdTestRegtest, CmdTestSwapMethods): network = ('ltc',) tmpdir_nums = [43] cmd_group_in = CmdTestRegtest.cmd_group_in + ( ('walletconv_bob', 'LTC wallet generation'), ('addrgen_bob', 'LTC address generation'), ('addrimport_bob', 'importing LTC addresses'), )