automount.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. #!/usr/bin/env python3
  2. #
  3. # MMGen Wallet, a terminal-based cryptocurrency wallet
  4. # Copyright (C)2013-2025 The MMGen Project <mmgen@tuta.io>
  5. # Licensed under the GNU General Public License, Version 3:
  6. # https://www.gnu.org/licenses
  7. # Public project repositories:
  8. # https://github.com/mmgen/mmgen-wallet
  9. # https://gitlab.com/mmgen/mmgen-wallet
  10. """
  11. test.cmdtest_d.automount: autosigning with automount tests for the cmdtest.py test suite
  12. """
  13. import time
  14. from .autosign import CmdTestAutosignThreaded
  15. from .regtest import CmdTestRegtest, rt_pw
  16. from ..include.common import gr_uc, create_addrpairs
  17. class CmdTestAutosignAutomount(CmdTestAutosignThreaded, CmdTestRegtest):
  18. 'automounted transacting operations via regtest mode'
  19. networks = ('btc', 'bch', 'ltc')
  20. tmpdir_nums = [49]
  21. bdb_wallet = True
  22. keylist_passwd = 'abc'
  23. rt_data = {
  24. 'rtFundAmt': {'btc':'500', 'bch':'500', 'ltc':'5500'},
  25. }
  26. bal1_chk = {
  27. 'btc': '502.46',
  28. 'bch': '502.46',
  29. 'ltc': '5502.46'}
  30. bal2_chk = {
  31. 'btc': '493.56992828',
  32. 'bch': '501.22524576',
  33. 'ltc': '5493.56992828'}
  34. cmd_group = (
  35. ('setup', 'regtest mode setup'),
  36. ('walletgen_alice', 'wallet generation (Alice)'),
  37. ('addrgen_alice', 'address generation (Alice)'),
  38. ('addrimport_alice', 'importing Alice’s addresses'),
  39. ('addrimport_alice_non_mmgen', 'importing Alice’s non-MMGen addresses'),
  40. ('fund_alice', 'funding Alice’s wallet'),
  41. ('fund_alice_non_mmgen1', 'funding Alice’s wallet (non-MMGen addr #1)'),
  42. ('fund_alice_non_mmgen2', 'funding Alice’s wallet (non-MMGen addr #2)'),
  43. ('generate', 'mining a block'),
  44. ('alice_bal1', 'checking Alice’s balance'),
  45. ('alice_txcreate1', 'creating a transaction'),
  46. ('alice_add_xmr_compat_tx', 'adding XMR compat TX to txauto dir'),
  47. ('alice_txcreate_bad_have_unsigned', 'creating the transaction again (error)'),
  48. ('alice_run_autosign_setup', 'running ‘autosign setup’ (with default wallet)'),
  49. ('wait_loop_start', 'starting autosign wait loop'),
  50. ('alice_txstatus1', 'getting transaction status (unsigned)'),
  51. ('alice_txstatus2', 'getting transaction status (unsent)'),
  52. ('alice_txcreate_bad_have_unsent', 'creating the transaction again (error)'),
  53. ('alice_txsend1', 'sending a transaction, editing comment'),
  54. ('alice_txstatus3', 'getting transaction status (in mempool)'),
  55. ('alice_txsend_bad_no_unsent', 'sending the transaction again (error)'),
  56. ('generate', 'mining a block'),
  57. ('alice_txstatus4', 'getting transaction status (one confirmation)'),
  58. ('alice_txcreate2', 'creating a transaction'),
  59. ('alice_txsend_abort1', 'aborting the transaction (raw only)'),
  60. ('alice_txsend_abort2', 'aborting the transaction again (error)'),
  61. ('alice_txcreate3', 'creating a transaction'),
  62. ('alice_txsend_abort3', 'aborting the transaction (user exit)'),
  63. ('alice_txsend_abort4', 'aborting the transaction (raw + signed)'),
  64. ('alice_txsend_abort5', 'aborting the transaction again (error)'),
  65. ('generate', 'mining a block'),
  66. ('alice_txcreate4', 'creating a transaction'),
  67. ('alice_txbump1', 'bumping the unsigned transaction (error)'),
  68. ('alice_txbump2', 'bumping the unsent transaction (error)'),
  69. ('alice_txsend2_dump_hex', 'dumping the transaction to hex'),
  70. ('alice_txsend2_cli', 'sending the transaction via cli'),
  71. ('alice_txsend2_mark_sent', 'marking the transaction sent'),
  72. ('alice_txbump3', 'bumping the transaction'),
  73. ('alice_txsend3', 'sending the bumped transaction'),
  74. ('alice_txbump4', 'bumping the transaction (new outputs, fee too low)'),
  75. ('alice_txbump_abort1', 'aborting the transaction'),
  76. ('alice_txbump5', 'bumping the transaction (new outputs)'),
  77. ('alice_txsend5', 'sending the bumped transaction'),
  78. ('alice_txstatus5', 'getting transaction status (in mempool)'),
  79. ('alice_txstatus6', 'getting transaction status (tx_range=0, in mempool)'),
  80. ('alice_txstatus7', 'getting transaction status (tx_range=1, replaced)'),
  81. ('alice_txstatus8', 'getting transaction status (tx_range=3, 2 confirmations)'),
  82. ('alice_txstatus9', 'getting transaction status (tx_range=0-3)'),
  83. ('generate', 'mining a block'),
  84. ('alice_bal2', 'checking Alice’s balance'),
  85. ('wait_loop_kill', 'stopping autosign wait loop'),
  86. ('stop', 'stopping regtest daemon'),
  87. ('txview', 'viewing transactions'),
  88. )
  89. def __init__(self, cfg, trunner, cfgs, spawn):
  90. self.coins = [cfg.coin.lower()]
  91. CmdTestAutosignThreaded.__init__(self, cfg, trunner, cfgs, spawn)
  92. CmdTestRegtest.__init__(self, cfg, trunner, cfgs, spawn)
  93. if trunner is None:
  94. return
  95. self.opts.append('--alice')
  96. self.non_mmgen_addrs = create_addrpairs(self.proto, 'C', 2)
  97. def addrimport_alice_non_mmgen(self):
  98. self.write_to_tmpfile(
  99. 'non_mmgen_addrs',
  100. '\n'.join(e.addr for e in self.non_mmgen_addrs))
  101. return self.spawn(
  102. 'mmgen-addrimport',
  103. ['--alice', '--quiet', '--addrlist', f'{self.tmpdir}/non_mmgen_addrs'])
  104. def fund_alice_non_mmgen1(self):
  105. return self.fund_wallet('alice', '1.23', addr=self.non_mmgen_addrs[0].addr)
  106. def fund_alice_non_mmgen2(self):
  107. return self.fund_wallet('alice', '1.23', addr=self.non_mmgen_addrs[1].addr)
  108. def alice_bal1(self):
  109. return self._user_bal_cli('alice', chk=self.bal1_chk[self.coin])
  110. def alice_add_xmr_compat_tx(self):
  111. self.spawn(msg_only=True)
  112. self.insert_device()
  113. self.do_mount()
  114. from shutil import copyfile
  115. tx_fn = '16E53D-E41192-XMR[0.012345].asubtx'
  116. copyfile(f'test/ref/monero/{tx_fn}', f'{self.asi.txauto_dir}/{tx_fn}')
  117. self.do_umount()
  118. self.remove_device()
  119. return 'ok'
  120. def alice_txcreate1(self):
  121. return self._user_txcreate(
  122. 'alice',
  123. inputs = '1-3',
  124. tweaks = ['confirm_non_mmgen'],
  125. chg_addr = 'C:5',
  126. data_arg = 'data:'+gr_uc[:24])
  127. def alice_txcreate2(self):
  128. return self._user_txcreate('alice', chg_addr='L:5')
  129. alice_txcreate3 = alice_txcreate2
  130. def alice_txcreate4(self):
  131. return self._user_txcreate('alice', chg_addr='L:4', need_rbf=True)
  132. def _alice_txsend_abort(self, err=False, send_resp='y', expect=None, shred_expect=[]):
  133. self.insert_device_online()
  134. t = self.spawn(
  135. 'mmgen-txsend',
  136. ['--quiet', '--abort'],
  137. no_passthru_opts = ['coin'],
  138. exit_val = 2 if err else 1 if send_resp == 'n' else None)
  139. if err:
  140. t.expect(expect)
  141. else:
  142. t.expect('(y/N): ', send_resp)
  143. if expect:
  144. t.expect(expect)
  145. for pat in shred_expect:
  146. t.expect(pat, regex=True)
  147. t.read()
  148. self.remove_device_online()
  149. return t
  150. def alice_txsend_abort1(self):
  151. return self._alice_txsend_abort(shred_expect=['Shredding .*arawtx'])
  152. def alice_txsend_abort2(self):
  153. return self._alice_txsend_abort(err=True, expect='No unsent transactions')
  154. def alice_txsend_abort3(self):
  155. return self._alice_txsend_abort(send_resp='n', expect='Exiting at user request')
  156. def alice_txsend_abort4(self):
  157. self._wait_signed('transaction')
  158. return self._alice_txsend_abort(shred_expect=[r'Shredding .*arawtx', r'Shredding .*asigtx'])
  159. alice_txsend_abort5 = alice_txsend_abort2
  160. def alice_txcreate_bad_have_unsigned(self):
  161. return self._user_txcreate('alice', chg_addr='C:5', exit_val=2, expect_str='already present')
  162. def alice_txcreate_bad_have_unsent(self):
  163. return self._user_txcreate('alice', chg_addr='C:5', exit_val=2, expect_str='unsent transaction')
  164. def alice_run_autosign_setup(self):
  165. from mmgen.crypto import Crypto
  166. from mmgen.cfg import Config
  167. new_cfg = Config({'_clone': self.cfg, 'usr_randchars': 0, 'hash_preset': '1'})
  168. enc_data = Crypto(new_cfg).mmgen_encrypt(
  169. '\n'.join(e.wif for e in self.non_mmgen_addrs).encode(), passwd=self.keylist_passwd)
  170. self.write_to_tmpfile('non_mmgen_keys.mmenc', enc_data, binary=True)
  171. return self.run_setup(
  172. mn_type = 'default',
  173. use_dfl_wallet = True,
  174. wallet_passwd = rt_pw,
  175. add_opts = [f'--keys-from-file={self.tmpdir}/non_mmgen_keys.mmenc'],
  176. keylist_passwd = self.keylist_passwd)
  177. def alice_txsend1(self):
  178. return self._user_txsend('alice', comment='This one’s worth a comment', no_wait=True)
  179. def alice_txsend2_dump_hex(self):
  180. return self._user_txsend('alice', need_rbf=True, dump_hex=True)
  181. def alice_txsend2_cli(self):
  182. if not self.proto.cap('rbf'):
  183. return 'skip'
  184. return self._user_dump_hex_send_cli('alice')
  185. def alice_txsend2_mark_sent(self):
  186. return self._user_txsend('alice', need_rbf=True, mark_sent=True)
  187. def alice_txsend3(self):
  188. return self._user_txsend('alice', need_rbf=True)
  189. def alice_txsend5(self):
  190. return self._user_txsend('alice', need_rbf=True)
  191. def _alice_txstatus(
  192. self,
  193. expect,
  194. exit_val = None,
  195. need_rbf = False,
  196. tx_range = None,
  197. verbose = True,
  198. batch = False):
  199. if need_rbf and not self.proto.cap('rbf'):
  200. return 'skip'
  201. self.insert_device_online()
  202. t = self.spawn(
  203. 'mmgen-txsend',
  204. ['--alice', '--autosign', '--status']
  205. + (['--verbose'] if verbose else [])
  206. + ([] if tx_range is None else [tx_range]),
  207. no_passthru_opts = ['coin'],
  208. exit_val = exit_val)
  209. t.expect(expect, regex=True)
  210. if not (exit_val or batch):
  211. t.expect('view: ', 'n')
  212. t.read()
  213. self.remove_device_online()
  214. return t
  215. def alice_txstatus1(self):
  216. return self._alice_txstatus('unsigned', 1)
  217. def alice_txstatus2(self):
  218. self._wait_signed('transaction')
  219. return self._alice_txstatus('unsent', 1)
  220. def alice_txstatus3(self):
  221. return self._alice_txstatus('in mempool', 0)
  222. def alice_txstatus4(self):
  223. return self._alice_txstatus('1 confirmation', 0)
  224. def alice_txstatus5(self):
  225. return self._alice_txstatus('in mempool', need_rbf=True)
  226. def alice_txstatus6(self):
  227. return self._alice_txstatus('in mempool', need_rbf=True, tx_range='0')
  228. def alice_txstatus7(self):
  229. return self._alice_txstatus('replaced', need_rbf=True, tx_range='1')
  230. def alice_txstatus8(self):
  231. return self._alice_txstatus('2 confirmations', need_rbf=True, tx_range='3')
  232. def alice_txstatus9(self):
  233. return self._alice_txstatus(
  234. 'in mempool.*replaced.*replaced.*2 confirmations',
  235. need_rbf = True,
  236. tx_range = '0-3',
  237. verbose = False,
  238. batch = True)
  239. def alice_txsend_bad_no_unsent(self):
  240. self.insert_device_online()
  241. t = self.spawn('mmgen-txsend', ['--quiet', '--autosign'], exit_val=2, no_passthru_opts=['coin'])
  242. t.expect('No unsent transactions')
  243. t.read()
  244. self.remove_device_online()
  245. return t
  246. def _alice_txbump(self, fee_opt=None, output_args=[], bad_tx_expect=None, low_fee_fix=None):
  247. if not self.proto.cap('rbf'):
  248. return 'skip'
  249. self.insert_device_online()
  250. t = self.spawn(
  251. 'mmgen-txbump',
  252. ['--alice', '--autosign']
  253. + ([fee_opt] if fee_opt else [])
  254. + output_args,
  255. exit_val = 1 if bad_tx_expect else None)
  256. if bad_tx_expect:
  257. time.sleep(0.5)
  258. t.expect('Only sent transactions')
  259. t.expect(bad_tx_expect)
  260. else:
  261. if not output_args:
  262. t.expect(r'to deduct the fee from .* change output\): ', '\n', regex=True)
  263. t.expect(r'(Y/n): ', 'y') # output OK?
  264. if low_fee_fix or not fee_opt:
  265. if low_fee_fix:
  266. t.expect('Please choose a higher fee')
  267. t.expect('transaction fee: ', (low_fee_fix or '200s') + '\n')
  268. if output_args:
  269. t.expect(r'(Y/n): ', 'y')
  270. t.expect(r'(Y/n): ', 'y') # fee OK?
  271. t.expect(r'(y/N): ', '\n') # add comment?
  272. t.expect(r'(y/N): ', 'y') # save?
  273. t.read()
  274. self.remove_device_online()
  275. return t
  276. def alice_txbump1(self):
  277. return self._alice_txbump(bad_tx_expect='unsigned transaction')
  278. def alice_txbump2(self):
  279. self._wait_signed('transaction')
  280. return self._alice_txbump(bad_tx_expect='unsent transaction')
  281. def alice_txbump3(self):
  282. return self._alice_txbump()
  283. def alice_txbump4(self):
  284. sid = self._user_sid('alice')
  285. return self._alice_txbump(
  286. fee_opt = '--fee=3s',
  287. output_args = [f'{self.burn_addr},7.654321', f'{sid}:C:1'],
  288. low_fee_fix = '300s')
  289. def alice_txbump_abort1(self):
  290. if not self.proto.cap('rbf'):
  291. return 'skip'
  292. return self._alice_txsend_abort(shred_expect=['Shredding .*arawtx'])
  293. def alice_txbump5(self):
  294. sid = self._user_sid('alice')
  295. return self._alice_txbump(
  296. fee_opt = '--fee=400s',
  297. output_args = ['data:message for posterity', f'{self.burn_addr},7.654321', f'{sid}:C:1'])
  298. def alice_bal2(self):
  299. return self.user_bal('alice', self.bal2_chk[self.coin])