cmdtest.py: new ethbump test

Testing:

    # with Go-Ethereum:
    $ test/cmdtest.py --coin=eth ethbump

    # or with Rust Ethereum:
    $ test/cmdtest.py --coin=eth --daemon-id=reth ethbump
This commit is contained in:
The MMGen Project 2025-03-29 09:30:16 +00:00
commit b7446566a2
Signed by: mmgen
GPG key ID: 3F8B1861E32B7DA2
7 changed files with 271 additions and 4 deletions

View file

@ -1,6 +1,6 @@
```text
MMGEN-TXBUMP: Create, and optionally send and sign, a replacement transaction
on networks that support replace-by-fee (RBF)
on supporting networks
USAGE: mmgen-txbump [opts] [MMGen TX file] [seed source] ...
mmgen-txbump [opts] [ADDR,AMT ... | DATA_SPEC] ADDR [MMGen TX file] [seed source] ...
OPTIONS:
@ -116,5 +116,5 @@
MMGenWallet .mmdat wallet,w
PlainHexSeedFile .hex hex,rawhex,plainhex
MMGEN v15.1.dev18 March 2025 MMGEN-TXBUMP(1)
MMGEN-WALLET 15.1.dev25 March 2025 MMGEN-TXBUMP(1)
```

View file

@ -21,6 +21,7 @@
‘etherscan’). This is done via a publicly accessible web
page, so no API key or registration is required
-q, --quiet Suppress warnings; overwrite files without prompting
-r, --receipt Print the receipt of the sent transaction (Ethereum only)
-s, --status Get status of a sent transaction (or current transaction,
whether sent or unsent, when used with --autosign)
-t, --test Test whether the transaction can be sent without sending it
@ -28,5 +29,5 @@
-x, --proxy P Connect to TX proxy via SOCKS5 proxy ‘P’ (host:port)
-y, --yes Answer 'yes' to prompts, suppress non-essential output
MMGEN v15.1.dev20 March 2025 MMGEN-TXSEND(1)
MMGEN-WALLET 15.1.dev25 March 2025 MMGEN-TXSEND(1)
```

View file

@ -1 +1 @@
15.1.dev24
15.1.dev25

View file

@ -129,6 +129,7 @@ opts_data = {
debugging only)
-e, --exact-output Show the exact output of the MMGen script(s) being run
-G, --exclude-groups=G Exclude the specified command groups (comma-separated)
-k, --devnet-block-period=N Block time for Ethereum devnet bump tests
-l, --list-cmds List the test scripts available commands
-L, --list-cmd-groups List the test scripts command groups and subgroups
-g, --list-current-cmd-groups List command groups for current configuration

258
test/cmdtest_d/ethbump.py Executable file
View file

@ -0,0 +1,258 @@
#!/usr/bin/env python3
#
# MMGen Wallet, a terminal-based cryptocurrency wallet
# Copyright (C)2013-2025 The MMGen Project <mmgen@tuta.io>
# 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.util import ymsg, suf
from .ethdev import CmdTestEthdev, CmdTestEthdevMethods, dfl_sid
from ..include.common import imsg, omsg_r
from .include.common import cleanup_env, dfl_words_file
burn_addr = 'beefcafe22' * 4
class CmdTestEthBumpMethods:
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(CmdTestEthdev, CmdTestEthdevMethods, CmdTestEthBumpMethods):
'Ethereum transaction bumping operations'
networks = ('eth',)
tmpdir_nums = [42]
dfl_devnet_block_period = 7
cmd_group_in = (
('setup', 'dev mode transaction bumping tests for Ethereum (start daemon)'),
('subgroup.init', []),
('subgroup.feebump', ['init']),
('subgroup.new_outputs', ['init']),
('subgroup.token_init', ['init']),
('subgroup.token_feebump', ['token_init']),
('subgroup.token_new_outputs', ['token_init']),
('stop', 'stopping daemon'),
)
cmd_subgroups = CmdTestEthdev.cmd_subgroups | {
'init': (
'initializing wallets',
('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_address', 'creating a transaction (spend from dev address to address :1)'),
('wait2', 'waiting for block'),
),
'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'),
('wait3', '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'),
('wait4', 'waiting for block'),
('bal2', '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'),
)
}
def __init__(self, cfg, trunner, cfgs, spawn):
self.devnet_block_period = cfg.devnet_block_period or self.dfl_devnet_block_period
CmdTestEthdev.__init__(self, cfg, trunner, cfgs, spawn)
def fund_mmgen_address(self):
return self._fund_mmgen_address(arg=f'{dfl_sid}:E:1,98765.4321')
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 bal1(self):
return self._bal_check(pat=rf'{dfl_sid}:E:1\s+97778\.4320727\s')
def bal2(self):
return self._bal_check(pat=rf'{dfl_sid}:E:2\s+777\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:1\s+998.76544\s.*\s{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

View file

@ -353,6 +353,7 @@ class CmdTestEthdev(CmdTestBase, CmdTestShared, CmdTestEthdevMethods):
color = True
menu_prompt = 'efresh balance:\b'
input_sels_prompt = 'to spend from: '
devnet_block_period = None
bals = lambda self, k: {
'1': [ ('98831F3A:E:1', '123.456')],
@ -728,6 +729,10 @@ class CmdTestEthdev(CmdTestBase, CmdTestShared, CmdTestEthdevMethods):
if not d.id in ('geth', 'erigon'):
d.stop(silent=True)
d.remove_datadir()
if d.id in ('geth', 'reth'):
if bp := self.devnet_block_period:
d.usr_coind_args = [
f'--dev.block-time={bp}s' if d.id == 'reth' else f'--dev.period={bp}']
d.start(silent=self.tr.quiet)
rpc = await self.rpc
imsg(f'Daemon: {rpc.daemon.coind_name} v{rpc.daemon_version_str}')

View file

@ -42,6 +42,7 @@ cmd_groups_dfl = {
'ethswap': ('CmdTestEthSwap', {}),
# 'chainsplit': ('CmdTestChainsplit', {}),
'ethdev': ('CmdTestEthdev', {}),
'ethbump': ('CmdTestEthBump', {}),
'xmrwallet': ('CmdTestXMRWallet', {}),
'xmr_autosign': ('CmdTestXMRAutosign', {}),
}
@ -243,6 +244,7 @@ cfgs = { # addr_idx_lists (except 31, 32, 33, 34) must contain exactly 8 address
'39': {}, # xmr_autosign
'40': {}, # cfgfile
'41': {}, # opts
'42': {}, # ethbump
'47': {}, # ethswap
'48': {}, # ethswap_eth
'49': {}, # autosign_automount