Browse Source

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
The MMGen Project 5 days ago
parent
commit
b7446566a2

+ 2 - 2
doc/wiki/commands/command-help-txbump.md

@@ -1,6 +1,6 @@
 ```text
 ```text
   MMGEN-TXBUMP: Create, and optionally send and sign, a replacement transaction
   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] ...
   USAGE:        mmgen-txbump [opts] [MMGen TX file] [seed source] ...
                 mmgen-txbump [opts] [ADDR,AMT ... | DATA_SPEC] ADDR [MMGen TX file] [seed source] ...
                 mmgen-txbump [opts] [ADDR,AMT ... | DATA_SPEC] ADDR [MMGen TX file] [seed source] ...
   OPTIONS:
   OPTIONS:
@@ -116,5 +116,5 @@
     MMGenWallet        .mmdat    wallet,w
     MMGenWallet        .mmdat    wallet,w
     PlainHexSeedFile   .hex      hex,rawhex,plainhex
     PlainHexSeedFile   .hex      hex,rawhex,plainhex
 
 
-  MMGEN v15.1.dev18              March 2025                    MMGEN-TXBUMP(1)
+  MMGEN-WALLET 15.1.dev25        March 2025                    MMGEN-TXBUMP(1)
 ```
 ```

+ 2 - 1
doc/wiki/commands/command-help-txsend.md

@@ -21,6 +21,7 @@
                    ‘etherscan’).  This is done via a publicly accessible web
                    ‘etherscan’).  This is done via a publicly accessible web
                    page, so no API key or registration is required
                    page, so no API key or registration is required
   -q, --quiet      Suppress warnings; overwrite files without prompting
   -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,
   -s, --status     Get status of a sent transaction (or current transaction,
                    whether sent or unsent, when used with --autosign)
                    whether sent or unsent, when used with --autosign)
   -t, --test       Test whether the transaction can be sent without sending it
   -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)
   -x, --proxy P    Connect to TX proxy via SOCKS5 proxy ‘P’ (host:port)
   -y, --yes        Answer 'yes' to prompts, suppress non-essential output
   -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)
 ```
 ```

+ 1 - 1
mmgen/data/version

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

+ 1 - 0
test/cmdtest.py

@@ -129,6 +129,7 @@ opts_data = {
                      debugging only)
                      debugging only)
 -e, --exact-output   Show the exact output of the MMGen script(s) being run
 -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)
 -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 script’s available commands
 -l, --list-cmds      List the test script’s available commands
 -L, --list-cmd-groups List the test script’s command groups and subgroups
 -L, --list-cmd-groups List the test script’s command groups and subgroups
 -g, --list-current-cmd-groups List command groups for current configuration
 -g, --list-current-cmd-groups List command groups for current configuration

+ 258 - 0
test/cmdtest_d/ethbump.py

@@ -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

+ 5 - 0
test/cmdtest_d/ethdev.py

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

+ 2 - 0
test/cmdtest_d/include/cfg.py

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