automount.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339
  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 (idx=0, in mempool)'),
  80. ('alice_txstatus7', 'getting transaction status (idx=1, replaced)'),
  81. ('alice_txstatus8', 'getting transaction status (idx=3, 2 confirmations)'),
  82. ('generate', 'mining a block'),
  83. ('alice_bal2', 'checking Alice’s balance'),
  84. ('wait_loop_kill', 'stopping autosign wait loop'),
  85. ('stop', 'stopping regtest daemon'),
  86. ('txview', 'viewing transactions'),
  87. )
  88. def __init__(self, cfg, trunner, cfgs, spawn):
  89. self.coins = [cfg.coin.lower()]
  90. CmdTestAutosignThreaded.__init__(self, cfg, trunner, cfgs, spawn)
  91. CmdTestRegtest.__init__(self, cfg, trunner, cfgs, spawn)
  92. if trunner is None:
  93. return
  94. self.opts.append('--alice')
  95. self.non_mmgen_addrs = create_addrpairs(self.proto, 'C', 2)
  96. def addrimport_alice_non_mmgen(self):
  97. self.write_to_tmpfile(
  98. 'non_mmgen_addrs',
  99. '\n'.join(e.addr for e in self.non_mmgen_addrs))
  100. return self.spawn(
  101. 'mmgen-addrimport',
  102. ['--alice', '--quiet', '--addrlist', f'{self.tmpdir}/non_mmgen_addrs'])
  103. def fund_alice_non_mmgen1(self):
  104. return self.fund_wallet('alice', '1.23', addr=self.non_mmgen_addrs[0].addr)
  105. def fund_alice_non_mmgen2(self):
  106. return self.fund_wallet('alice', '1.23', addr=self.non_mmgen_addrs[1].addr)
  107. def alice_bal1(self):
  108. return self._user_bal_cli('alice', chk=self.bal1_chk[self.coin])
  109. def alice_add_xmr_compat_tx(self):
  110. self.spawn(msg_only=True)
  111. self.insert_device()
  112. self.do_mount()
  113. from shutil import copyfile
  114. tx_fn = '16E53D-E41192-XMR[0.012345].asubtx'
  115. copyfile(f'test/ref/monero/{tx_fn}', f'{self.asi.txauto_dir}/{tx_fn}')
  116. self.do_umount()
  117. self.remove_device()
  118. return 'ok'
  119. def alice_txcreate1(self):
  120. return self._user_txcreate(
  121. 'alice',
  122. inputs = '1-3',
  123. tweaks = ['confirm_non_mmgen'],
  124. chg_addr = 'C:5',
  125. data_arg = 'data:'+gr_uc[:24])
  126. def alice_txcreate2(self):
  127. return self._user_txcreate('alice', chg_addr='L:5')
  128. alice_txcreate3 = alice_txcreate2
  129. def alice_txcreate4(self):
  130. return self._user_txcreate('alice', chg_addr='L:4', need_rbf=True)
  131. def _alice_txsend_abort(self, err=False, send_resp='y', expect=None, shred_expect=[]):
  132. self.insert_device_online()
  133. t = self.spawn(
  134. 'mmgen-txsend',
  135. ['--quiet', '--abort'],
  136. no_passthru_opts = ['coin'],
  137. exit_val = 2 if err else 1 if send_resp == 'n' else None)
  138. if err:
  139. t.expect(expect)
  140. else:
  141. t.expect('(y/N): ', send_resp)
  142. if expect:
  143. t.expect(expect)
  144. for pat in shred_expect:
  145. t.expect(pat, regex=True)
  146. t.read()
  147. self.remove_device_online()
  148. return t
  149. def alice_txsend_abort1(self):
  150. return self._alice_txsend_abort(shred_expect=['Shredding .*arawtx'])
  151. def alice_txsend_abort2(self):
  152. return self._alice_txsend_abort(err=True, expect='No unsent transactions')
  153. def alice_txsend_abort3(self):
  154. return self._alice_txsend_abort(send_resp='n', expect='Exiting at user request')
  155. def alice_txsend_abort4(self):
  156. self._wait_signed('transaction')
  157. return self._alice_txsend_abort(shred_expect=[r'Shredding .*arawtx', r'Shredding .*asigtx'])
  158. alice_txsend_abort5 = alice_txsend_abort2
  159. def alice_txcreate_bad_have_unsigned(self):
  160. return self._user_txcreate('alice', chg_addr='C:5', exit_val=2, expect_str='already present')
  161. def alice_txcreate_bad_have_unsent(self):
  162. return self._user_txcreate('alice', chg_addr='C:5', exit_val=2, expect_str='unsent transaction')
  163. def alice_run_autosign_setup(self):
  164. from mmgen.crypto import Crypto
  165. from mmgen.cfg import Config
  166. new_cfg = Config({'_clone': self.cfg, 'usr_randchars': 0, 'hash_preset': '1'})
  167. enc_data = Crypto(new_cfg).mmgen_encrypt(
  168. '\n'.join(e.wif for e in self.non_mmgen_addrs).encode(), passwd=self.keylist_passwd)
  169. self.write_to_tmpfile('non_mmgen_keys.mmenc', enc_data, binary=True)
  170. return self.run_setup(
  171. mn_type = 'default',
  172. use_dfl_wallet = True,
  173. wallet_passwd = rt_pw,
  174. add_opts = [f'--keys-from-file={self.tmpdir}/non_mmgen_keys.mmenc'],
  175. keylist_passwd = self.keylist_passwd)
  176. def alice_txsend1(self):
  177. return self._user_txsend('alice', comment='This one’s worth a comment', no_wait=True)
  178. def alice_txsend2_dump_hex(self):
  179. return self._user_txsend('alice', need_rbf=True, dump_hex=True)
  180. def alice_txsend2_cli(self):
  181. if not self.proto.cap('rbf'):
  182. return 'skip'
  183. return self._user_dump_hex_send_cli('alice')
  184. def alice_txsend2_mark_sent(self):
  185. return self._user_txsend('alice', need_rbf=True, mark_sent=True)
  186. def alice_txsend3(self):
  187. return self._user_txsend('alice', need_rbf=True)
  188. def alice_txsend5(self):
  189. return self._user_txsend('alice', need_rbf=True)
  190. def _alice_txstatus(self, expect, exit_val=None, need_rbf=False, idx=None):
  191. if need_rbf and not self.proto.cap('rbf'):
  192. return 'skip'
  193. self.insert_device_online()
  194. t = self.spawn(
  195. 'mmgen-txsend',
  196. ['--alice', '--autosign', '--status', '--verbose']
  197. + ([] if idx is None else [str(idx)]),
  198. no_passthru_opts = ['coin'],
  199. exit_val = exit_val)
  200. t.expect(expect)
  201. if not exit_val:
  202. t.expect('view: ', 'n')
  203. t.read()
  204. self.remove_device_online()
  205. return t
  206. def alice_txstatus1(self):
  207. return self._alice_txstatus('unsigned', 1)
  208. def alice_txstatus2(self):
  209. self._wait_signed('transaction')
  210. return self._alice_txstatus('unsent', 1)
  211. def alice_txstatus3(self):
  212. return self._alice_txstatus('in mempool', 0)
  213. def alice_txstatus4(self):
  214. return self._alice_txstatus('1 confirmation', 0)
  215. def alice_txstatus5(self):
  216. return self._alice_txstatus('in mempool', need_rbf=True)
  217. def alice_txstatus6(self):
  218. return self._alice_txstatus('in mempool', need_rbf=True, idx=0)
  219. def alice_txstatus7(self):
  220. return self._alice_txstatus('replaced', need_rbf=True, idx=1)
  221. def alice_txstatus8(self):
  222. return self._alice_txstatus('2 confirmations', need_rbf=True, idx=3)
  223. def alice_txsend_bad_no_unsent(self):
  224. self.insert_device_online()
  225. t = self.spawn('mmgen-txsend', ['--quiet', '--autosign'], exit_val=2, no_passthru_opts=['coin'])
  226. t.expect('No unsent transactions')
  227. t.read()
  228. self.remove_device_online()
  229. return t
  230. def _alice_txbump(self, fee_opt=None, output_args=[], bad_tx_expect=None, low_fee_fix=None):
  231. if not self.proto.cap('rbf'):
  232. return 'skip'
  233. self.insert_device_online()
  234. t = self.spawn(
  235. 'mmgen-txbump',
  236. ['--alice', '--autosign']
  237. + ([fee_opt] if fee_opt else [])
  238. + output_args,
  239. exit_val = 1 if bad_tx_expect else None)
  240. if bad_tx_expect:
  241. time.sleep(0.5)
  242. t.expect('Only sent transactions')
  243. t.expect(bad_tx_expect)
  244. else:
  245. if not output_args:
  246. t.expect(r'to deduct the fee from .* change output\): ', '\n', regex=True)
  247. t.expect(r'(Y/n): ', 'y') # output OK?
  248. if low_fee_fix or not fee_opt:
  249. if low_fee_fix:
  250. t.expect('Please choose a higher fee')
  251. t.expect('transaction fee: ', (low_fee_fix or '200s') + '\n')
  252. if output_args:
  253. t.expect(r'(Y/n): ', 'y')
  254. t.expect(r'(Y/n): ', 'y') # fee OK?
  255. t.expect(r'(y/N): ', '\n') # add comment?
  256. t.expect(r'(y/N): ', 'y') # save?
  257. t.read()
  258. self.remove_device_online()
  259. return t
  260. def alice_txbump1(self):
  261. return self._alice_txbump(bad_tx_expect='unsigned transaction')
  262. def alice_txbump2(self):
  263. self._wait_signed('transaction')
  264. return self._alice_txbump(bad_tx_expect='unsent transaction')
  265. def alice_txbump3(self):
  266. return self._alice_txbump()
  267. def alice_txbump4(self):
  268. sid = self._user_sid('alice')
  269. return self._alice_txbump(
  270. fee_opt = '--fee=3s',
  271. output_args = [f'{self.burn_addr},7.654321', f'{sid}:C:1'],
  272. low_fee_fix = '300s')
  273. def alice_txbump_abort1(self):
  274. if not self.proto.cap('rbf'):
  275. return 'skip'
  276. return self._alice_txsend_abort(shred_expect=['Shredding .*arawtx'])
  277. def alice_txbump5(self):
  278. sid = self._user_sid('alice')
  279. return self._alice_txbump(
  280. fee_opt = '--fee=400s',
  281. output_args = ['data:message for posterity', f'{self.burn_addr},7.654321', f'{sid}:C:1'])
  282. def alice_bal2(self):
  283. return self.user_bal('alice', self.bal2_chk[self.coin])