xmr_autosign.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  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.xmr_autosign: xmr autosigning tests for the cmdtest.py test suite
  12. """
  13. import re, asyncio
  14. from mmgen.color import blue, cyan, brown
  15. from ..include.common import imsg, silence, end_silence, strip_ansi_escapes
  16. from .include.common import get_file_with_ext, cleanup_env
  17. from .xmrwallet import CmdTestXMRWallet
  18. from .autosign import CmdTestAutosignThreaded
  19. def make_burn_addr(cfg):
  20. from mmgen.tool.coin import tool_cmd
  21. return tool_cmd(
  22. cfg = cfg,
  23. cmdname = 'privhex2addr',
  24. proto = cfg._proto,
  25. mmtype = 'monero').privhex2addr('beadcafe'*8)
  26. class CmdTestXMRAutosign(CmdTestXMRWallet, CmdTestAutosignThreaded):
  27. """
  28. Monero autosigning operations (xmrwallet compat mode)
  29. """
  30. tmpdir_nums = [39]
  31. # xmrwallet attrs:
  32. tx_relay_user = 'miner'
  33. # user sid autosign port_shift kal_range add_coind_args
  34. user_data = (
  35. ('miner', '98831F3A', False, 130, '1', []),
  36. ('alice', 'FE3C6545', True, 150, '1-2', []))
  37. # autosign attrs:
  38. coins = ['xmr']
  39. compat = True
  40. cmd_group = (
  41. ('daemon_version', 'checking daemon version'),
  42. ('create_tmp_wallets', 'creating temporary online wallets for Alice'),
  43. ('new_account_alice', 'adding an account to Alice’s tmp wallet'),
  44. ('new_address_alice', 'adding an address to Alice’s tmp wallet'),
  45. ('new_address_alice_label', 'adding an address to Alice’s tmp wallet (with label)'),
  46. ('dump_tmp_wallets', 'dumping Alice’s tmp wallets'),
  47. ('dump_tmp_wallets_json', 'dumping Alice’s tmp wallets to JSON format'),
  48. ('delete_tmp_wallets', 'deleting Alice’s tmp wallets'),
  49. ('gen_kafile_miner', 'generating key-address file for Miner'),
  50. ('create_wallet_miner', 'creating Monero wallet for Miner'),
  51. ('mine_initial_coins', 'mining initial coins'),
  52. ('autosign_setup', 'autosign setup with Alice’s seed'),
  53. ('autosign_xmr_setup', 'autosign setup (creation of Monero signing wallets)'),
  54. ('create_watchonly_wallets', 'creating watch-only wallets from Alice’s wallet dumps'),
  55. ('delete_tmp_dump_files', 'deleting Alice’s dump files'),
  56. ('fund_alice1', 'sending funds to Alice (wallet #1)'),
  57. ('check_bal_alice1', 'mining, checking balance (wallet #1)'),
  58. ('fund_alice2', 'sending funds to Alice (wallet #2)'),
  59. ('check_bal_alice2', 'mining, checking balance (wallet #2)'),
  60. ('wait_loop_start', 'starting autosign wait loop'),
  61. ('export_outputs1', 'exporting outputs from Alice’s watch-only wallet #1'),
  62. ('create_transfer_tx1', 'creating a transfer TX'),
  63. ('submit_transfer_tx1', 'submitting the transfer TX'),
  64. ('resubmit_transfer_tx1', 'resubmitting the transfer TX'),
  65. ('export_outputs2', 'exporting outputs from Alice’s watch-only wallet #1'),
  66. ('import_key_images1', 'importing signed key images into Alice’s online wallets'),
  67. ('sync_chkbal1', 'syncing Alice’s wallet #1'),
  68. ('abort_tx1', 'aborting the current transaction (error)'),
  69. ('create_transfer_tx2', 'creating a transfer TX (for relaying via proxy)'),
  70. ('abort_tx2', 'aborting the current transaction (OK, unsigned)'),
  71. ('create_transfer_tx2a', 'creating the transfer TX again'),
  72. ('submit_transfer_tx2', 'submitting the transfer TX (relaying via proxy)'),
  73. ('sync_chkbal2', 'syncing Alice’s wallets and checking balance'),
  74. ('dump_wallets', 'dumping Alice’s wallets'),
  75. ('delete_wallets', 'deleting Alice’s wallets'),
  76. ('restore_wallets', 'creating online (watch-only) wallets for Alice'),
  77. ('delete_dump_files', 'deleting Alice’s dump files'),
  78. ('export_outputs3', 'exporting outputs from Alice’s watch-only wallets'),
  79. ('import_key_images2', 'importing signed key images into Alice’s online wallets'),
  80. ('sync_chkbal3', 'syncing Alice’s wallets and checking balance'),
  81. ('wait_loop_kill', 'stopping autosign wait loop'),
  82. ('stop_daemons', 'stopping all wallet and coin daemons'),
  83. ('view', 'viewing Alice’s wallet in offline mode (wallet #1)'),
  84. ('listview', 'list-viewing Alice’s wallet in offline mode (wallet #2)'),
  85. ('txlist', 'listing Alice’s submitted transactions'),
  86. ('txview', 'viewing Alice’s submitted transactions'),
  87. ('txview_all', 'viewing all raw, signed and submitted transactions'),
  88. ('check_tx_dirs', 'cleaning and checking signable file directories'),
  89. )
  90. def __init__(self, cfg, trunner, cfgs, spawn):
  91. CmdTestAutosignThreaded.__init__(self, cfg, trunner, cfgs, spawn)
  92. CmdTestXMRWallet.__init__(self, cfg, trunner, cfgs, spawn)
  93. if trunner is None:
  94. return
  95. from mmgen.cfg import Config
  96. self.alice_cfg = Config({
  97. 'coin': 'XMR',
  98. 'outdir': self.users['alice'].udir,
  99. 'wallet_rpc_password': 'passwOrd',
  100. 'test_suite': True,
  101. } | ({
  102. 'alice': True,
  103. 'compat': True
  104. } if self.compat else {
  105. 'wallet_dir': self.users['alice'].udir
  106. }))
  107. self.burn_addr = make_burn_addr(cfg)
  108. self.opts.append('--xmrwallets={}'.format(self.users['alice'].kal_range)) # mmgen-autosign opts
  109. self.autosign_opts = ['--autosign'] # mmgen-xmrwallet opts
  110. self.spawn_env['MMGEN_TEST_SUITE_XMR_AUTOSIGN'] = '1'
  111. def create_tmp_wallets(self):
  112. self.spawn(msg_only=True)
  113. data = self.users['alice']
  114. from mmgen.wallet import Wallet
  115. from mmgen.xmrwallet import op
  116. from mmgen.addrlist import KeyAddrList
  117. silence()
  118. kal = KeyAddrList(
  119. cfg = self.alice_cfg,
  120. proto = self.proto,
  121. addr_idxs = '1-2',
  122. seed = Wallet(self.alice_cfg, fn=data.mmwords).seed,
  123. skip_chksum_msg = True,
  124. key_address_validity_check = False)
  125. kal.file.write(ask_overwrite=False)
  126. fn = get_file_with_ext(data.udir, 'akeys')
  127. m = op('create', self.alice_cfg, fn, '1-2')
  128. asyncio.run(m.main())
  129. asyncio.run(m.stop_wallet_daemon())
  130. end_silence()
  131. return 'ok'
  132. def _new_addr_alice(self, *args):
  133. data = self.users['alice']
  134. return self.new_addr_alice(
  135. *args,
  136. kafile = get_file_with_ext(data.udir, 'akeys'))
  137. def new_account_alice(self):
  138. return self._new_addr_alice(
  139. '2',
  140. 'start',
  141. r'Creating new account for wallet .*2.* with label .*‘xmrwallet new account .*y/N\): ')
  142. def new_address_alice(self):
  143. return self._new_addr_alice(
  144. '2:1',
  145. 'continue',
  146. r'Creating new address for wallet .*2.*, account .*#1.* with label .*‘xmrwallet new address .*y/N\): ')
  147. def new_address_alice_label(self):
  148. return self._new_addr_alice(
  149. '2:1,Alice’s new address',
  150. 'stop',
  151. r'Creating new address for wallet .*2.*, account .*#1.* with label .*‘Alice’s new address .*y/N\): ')
  152. def dump_tmp_wallets(self):
  153. return self._dump_wallets(autosign=False)
  154. def dump_tmp_wallets_json(self):
  155. return self._dump_wallets(autosign=False, op='dump_json')
  156. def dump_wallets(self):
  157. return self._dump_wallets(autosign=True)
  158. def _dump_wallets(self, autosign, op='dump'):
  159. data = self.users['alice']
  160. self.insert_device_online()
  161. t = self.spawn(
  162. 'mmgen-xmrwallet',
  163. self.extra_opts
  164. + (['--alice', '--compat'] if self.compat else [f'--wallet-dir={data.udir}'])
  165. + [f'--daemon=localhost:{data.md.rpc_port}']
  166. + (self.autosign_opts if autosign else [])
  167. + [op]
  168. + ([] if autosign else [get_file_with_ext(data.udir, 'akeys')]),
  169. env = cleanup_env(self.cfg))
  170. t.expect('2 wallets dumped')
  171. res = t.read()
  172. if op == 'dump_json':
  173. import json
  174. data = json.loads(re.sub('Stopping.*', '', strip_ansi_escapes(res)).strip())
  175. self.remove_device_online()
  176. return t
  177. def _delete_files(self, *ext_list):
  178. data = self.users['alice']
  179. self.spawn(msg_only=True)
  180. wdir = data.wd.wallet_dir if self.compat else data.udir
  181. for ext in ext_list:
  182. get_file_with_ext(wdir, ext, no_dot=True, delete_all=True)
  183. return 'ok'
  184. def delete_tmp_wallets(self):
  185. return self._delete_files('MoneroWallet', 'MoneroWallet.keys', '.akeys')
  186. def delete_wallets(self):
  187. return self._delete_files('MoneroWatchOnlyWallet', '.keys', '.address.txt')
  188. def delete_tmp_dump_files(self):
  189. return self._delete_files('.dump')
  190. def gen_kafile_miner(self):
  191. return self.gen_kafiles(['miner'])
  192. def create_wallet_miner(self):
  193. return self.create_wallets_miner()
  194. def delete_dump_files(self):
  195. return self._delete_files('.dump')
  196. async def fund_alice1(self):
  197. return await self.fund_alice(wallet=1)
  198. async def check_bal_alice1(self):
  199. return await self.check_bal_alice(wallet=1)
  200. async def fund_alice2(self):
  201. return await self.fund_alice(wallet=2)
  202. async def check_bal_alice2(self):
  203. return await self.check_bal_alice(wallet=2)
  204. def autosign_setup(self):
  205. return self.run_setup(
  206. mn_type = 'mmgen',
  207. mn_file = self.users['alice'].mmwords,
  208. use_dfl_wallet = None,
  209. expect_args = ['Continue with Monero setup? (Y/n): ', 'n'])
  210. def autosign_xmr_setup(self):
  211. self.insert_device_online()
  212. self.do_mount_online()
  213. self.asi_online.xmr_dir.mkdir(exist_ok=True)
  214. (self.asi_online.xmr_dir / 'old.vkeys').touch()
  215. self.do_umount_online()
  216. self.remove_device_online()
  217. self.insert_device()
  218. t = self.spawn('mmgen-autosign', self.opts + ['xmr_setup'], no_passthru_opts=True)
  219. t.written_to_file('View keys')
  220. t.read()
  221. self.remove_device()
  222. return t
  223. def create_watchonly_wallets(self):
  224. return self._create_wallets('restore')
  225. def restore_wallets(self):
  226. return self._create_wallets('restore')
  227. def _create_wallets(self, op='create'):
  228. self.insert_device_online()
  229. t = self.create_wallets('alice', op=op)
  230. t.read() # required!
  231. self.remove_device_online()
  232. return t
  233. def _create_transfer_tx(self, amt, add_opts=[]):
  234. self.insert_device_online()
  235. t = self.do_op(
  236. 'transfer',
  237. 'alice',
  238. f'1:0:{self.burn_addr},{amt}',
  239. no_relay = True,
  240. do_ret = True,
  241. add_opts = add_opts)
  242. t.read() # required!
  243. self.remove_device_online()
  244. return t
  245. def create_transfer_tx1(self):
  246. return self._create_transfer_tx('0.124', add_opts=['--priority=2'])
  247. def create_transfer_tx2(self):
  248. return self._create_transfer_tx('0.257')
  249. create_transfer_tx2a = create_transfer_tx2
  250. def _abort_tx(self, expect, send=None, exit_val=None):
  251. self.insert_device_online()
  252. t = self.spawn('mmgen-xmrwallet', ['--autosign', 'abort'], exit_val=exit_val)
  253. t.expect(expect)
  254. if send:
  255. t.send(send)
  256. t.read() # required!
  257. self.remove_device_online()
  258. return t
  259. def abort_tx1(self):
  260. return self._abort_tx('No unsent transactions present', exit_val=2)
  261. def abort_tx2(self):
  262. return self._abort_tx('(y/N): ', 'y')
  263. def _xmr_autosign_op(
  264. self,
  265. op,
  266. desc = None,
  267. signable_desc = None,
  268. ext = None,
  269. wallet_arg = None,
  270. add_opts = [],
  271. wait_signed = False):
  272. if wait_signed:
  273. self._wait_signed(signable_desc)
  274. data = self.users['alice']
  275. args = (
  276. self.extra_opts
  277. + self.autosign_opts
  278. + (['--alice', '--compat'] if self.compat else [f'--wallet-dir={data.udir}'])
  279. + [f'--daemon=localhost:{data.md.rpc_port}']
  280. + add_opts
  281. + [op]
  282. + ([get_file_with_ext(self.asi.xmr_tx_dir, ext)] if ext else [])
  283. + ([wallet_arg] if wallet_arg else []))
  284. desc_pfx = f'{desc}, ' if desc else ''
  285. self.insert_device_online() # device must be removed by calling method
  286. return self.spawn('mmgen-xmrwallet', args, extra_desc=f'({desc_pfx}Alice)')
  287. def _sync_chkbal(self, wallet_arg, bal_chk_func):
  288. return self.sync_wallets(
  289. 'alice',
  290. op = 'sync',
  291. wallets = wallet_arg,
  292. bal_chk_func = bal_chk_func)
  293. def sync_chkbal1(self):
  294. return self._sync_chkbal('1', lambda n, b, ub: b == ub and 1 < b < 1.12)
  295. # 1.234567891234 - 0.124 = 1.110567891234 (minus fees)
  296. def sync_chkbal2(self):
  297. return self._sync_chkbal('1', lambda n, b, ub: b == ub and 0.8 < b < 0.86)
  298. # 1.234567891234 - 0.124 - 0.257 = 0.853567891234 (minus fees)
  299. def sync_chkbal3(self):
  300. return self._sync_chkbal(
  301. '1-2',
  302. lambda n, b, ub: b == ub and ((n == 1 and 0.8 < b < 0.86) or (n == 2 and b > 1.23)))
  303. async def submit_transfer_tx1(self):
  304. return await self._submit_transfer_tx()
  305. async def resubmit_transfer_tx1(self):
  306. return await self._submit_transfer_tx(
  307. relay_parm = self.tx_relay_daemon_proxy_parm,
  308. op = 'resubmit',
  309. check_bal = False)
  310. async def submit_transfer_tx2(self):
  311. return await self._submit_transfer_tx(relay_parm=self.tx_relay_daemon_parm)
  312. async def _submit_transfer_tx(self, relay_parm=None, ext=None, op='submit', check_bal=True):
  313. t = self._xmr_autosign_op(
  314. op = op,
  315. add_opts = [f'--tx-relay-daemon={relay_parm}'] if relay_parm else [],
  316. ext = ext,
  317. signable_desc = 'transaction',
  318. wait_signed = op == 'submit')
  319. t.expect(f'{op.capitalize()} transaction? (y/N): ', 'y')
  320. t.written_to_file('Submitted transaction')
  321. t.read()
  322. self.remove_device_online() # device was inserted by _xmr_autosign_op()
  323. if check_bal:
  324. t.ok()
  325. return await self.mine_chk(
  326. 'alice', 1, 0,
  327. lambda x: 0 < x.ub < 1.234567891234,
  328. 'unlocked balance 0 < 1.234567891234')
  329. else:
  330. return t
  331. def _export_outputs(self, wallet_arg, op, add_opts=[]):
  332. t = self._xmr_autosign_op(
  333. op = op,
  334. wallet_arg = wallet_arg,
  335. add_opts = add_opts)
  336. t.written_to_file('Wallet outputs')
  337. t.read()
  338. self.remove_device_online() # device was inserted by _xmr_autosign_op()
  339. return t
  340. def export_outputs1(self):
  341. return self._export_outputs('1', op='export-outputs')
  342. def export_outputs2(self): # NB: --rescan-spent does not work with testnet/stagenet
  343. return self._export_outputs('1', op='export-outputs-sign', add_opts=['--rescan-blockchain'])
  344. def export_outputs3(self):
  345. return self._export_outputs('1-2', op='export-outputs-sign')
  346. def _import_key_images(self, wallet_arg):
  347. t = self._xmr_autosign_op(
  348. op = 'import-key-images',
  349. wallet_arg = wallet_arg,
  350. signable_desc = 'wallet outputs',
  351. wait_signed = True)
  352. t.read()
  353. self.remove_device_online() # device was inserted by _xmr_autosign_op()
  354. return t
  355. def import_key_images1(self):
  356. return self._import_key_images(None)
  357. def import_key_images2(self):
  358. return self._import_key_images(None)
  359. def txlist(self):
  360. self.insert_device_online()
  361. t = self.spawn('mmgen-xmrwallet', self.autosign_opts + ['txlist'])
  362. t.match_expect_list([
  363. 'SUBMITTED',
  364. 'Network', 'Submitted',
  365. 'transfer 1:0', '-> ext',
  366. 'transfer 1:0', '-> ext'
  367. ])
  368. t.read()
  369. self.remove_device_online()
  370. return t
  371. def txview(self):
  372. self.insert_device_online()
  373. t = self.spawn('mmgen-xmrwallet', self.autosign_opts + ['txview'])
  374. t.read()
  375. self.remove_device_online()
  376. return t
  377. def txview_all(self):
  378. self.spawn(msg_only=True)
  379. self.insert_device()
  380. self.do_mount()
  381. imsg(blue('Opening transaction directory: ') + cyan(f'{self.asi.xmr_tx_dir}'))
  382. for fn in self.asi.xmr_tx_dir.iterdir():
  383. imsg('\n' + brown(f'Viewing ‘{fn.name}’'))
  384. self.spawn('mmgen-xmrwallet', ['txview', str(fn)], no_msg=True).read()
  385. imsg('')
  386. self.do_umount()
  387. self.remove_device()
  388. return 'ok'
  389. def check_tx_dirs(self):
  390. self.insert_device()
  391. self.do_mount()
  392. before = '\n'.join(self._gen_listing())
  393. self.do_umount()
  394. self.remove_device()
  395. self.insert_device()
  396. t = self.spawn('mmgen-autosign', self.opts + ['clean'])
  397. t.read()
  398. self.remove_device()
  399. self.insert_device()
  400. self.do_mount()
  401. after = '\n'.join(self._gen_listing())
  402. self.do_umount()
  403. self.remove_device()
  404. imsg(f'\nBefore cleaning:\n{before}')
  405. imsg(f'\nAfter cleaning:\n{after}')
  406. pat = r'xmr/tx: \s*\S+\.subtx \S+\.subtx\s+xmr/outputs:\s*$'
  407. assert re.search(pat, after, re.DOTALL), f'regex search for {pat} failed'
  408. return t
  409. def view(self):
  410. return self.sync_wallets('alice', op='view', wallets='1')
  411. def listview(self):
  412. return self.sync_wallets('alice', op='listview', wallets='2')
  413. class CmdTestXMRAutosignNoCompat(CmdTestXMRAutosign):
  414. """
  415. Monero autosigning operations (non-xmrwallet compat mode)
  416. """
  417. compat = False