ct_xmr_autosign.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, a command-line cryptocurrency wallet
  4. # Copyright (C)2013-2024 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_py_d.ct_xmr_autosign: xmr autosigning tests for the cmdtest.py test suite
  12. """
  13. import os,time,re,shutil
  14. from pathlib import Path
  15. from mmgen.color import purple,gray
  16. from mmgen.util import fmt,async_run
  17. from ..include.common import cfg,imsg,silence,end_silence
  18. from .common import get_file_with_ext
  19. from .ct_xmrwallet import CmdTestXMRWallet
  20. from .ct_autosign import CmdTestAutosignThreaded
  21. def make_burn_addr():
  22. from mmgen.tool.coin import tool_cmd
  23. return tool_cmd(
  24. cfg = cfg,
  25. cmdname = 'privhex2addr',
  26. proto = cfg._proto,
  27. mmtype = 'monero' ).privhex2addr('beadcafe'*8)
  28. class CmdTestXMRAutosign(CmdTestXMRWallet,CmdTestAutosignThreaded):
  29. """
  30. Monero autosigning operations
  31. """
  32. tmpdir_nums = [39]
  33. # ct_xmrwallet attrs:
  34. tx_relay_user = 'miner'
  35. user_data = (
  36. ('miner', '98831F3A', False, 130, '1', []),
  37. ('alice', 'FE3C6545', True, 150, '1-2', []),
  38. )
  39. # ct_autosign attrs:
  40. coins = ['xmr']
  41. cmd_group = (
  42. ('daemon_version', 'checking daemon version'),
  43. ('create_tmp_wallets', 'creating temporary online wallets for Alice'),
  44. ('new_account_alice', 'adding an account to Alice’s tmp wallet'),
  45. ('new_address_alice', 'adding an address to Alice’s tmp wallet'),
  46. ('new_address_alice_label', 'adding an address to Alice’s tmp wallet (with label)'),
  47. ('dump_tmp_wallets', 'dumping Alice’s tmp wallets'),
  48. ('delete_tmp_wallets', 'deleting Alice’s tmp wallets'),
  49. ('autosign_setup', 'autosign setup with Alice’s seed'),
  50. ('create_watchonly_wallets', 'creating online (watch-only) wallets for Alice'),
  51. ('delete_tmp_dump_files', 'deleting Alice’s dump files'),
  52. ('gen_kafiles', 'generating key-address files for Miner'),
  53. ('create_wallets_miner', 'creating Monero wallets for Miner'),
  54. ('mine_initial_coins', 'mining initial coins'),
  55. ('fund_alice1', 'sending funds to Alice (wallet #1)'),
  56. ('fund_alice2', 'sending funds to Alice (wallet #2)'),
  57. ('autosign_start_thread', 'starting autosign wait loop'),
  58. ('create_transfer_tx1', 'creating a transfer TX'),
  59. ('submit_transfer_tx1', 'submitting the transfer TX'),
  60. ('resubmit_transfer_tx1', 'resubmitting the transfer TX'),
  61. ('export_outputs1', 'exporting outputs from Alice’s watch-only wallet #1'),
  62. ('import_key_images1', 'importing signed key images into Alice’s online wallets'),
  63. ('sync_chkbal1', 'syncing Alice’s wallet #1'),
  64. ('create_transfer_tx2', 'creating a transfer TX (for relaying via proxy)'),
  65. ('submit_transfer_tx2', 'submitting the transfer TX (relaying via proxy)'),
  66. ('sync_chkbal2', 'syncing Alice’s wallets and checking balance'),
  67. ('dump_wallets', 'dumping Alice’s wallets'),
  68. ('delete_wallets', 'deleting Alice’s wallets'),
  69. ('restore_wallets', 'creating online (watch-only) wallets for Alice'),
  70. ('delete_dump_files', 'deleting Alice’s dump files'),
  71. ('export_outputs2', 'exporting outputs from Alice’s watch-only wallets'),
  72. ('import_key_images2', 'importing signed key images into Alice’s online wallets'),
  73. ('sync_chkbal3', 'syncing Alice’s wallets and checking balance'),
  74. ('txlist', 'listing Alice’s submitted transactions'),
  75. ('check_tx_dirs', 'cleaning and checking signable file directories'),
  76. ('autosign_kill_thread', 'stopping autosign wait loop'),
  77. ('stop_daemons', 'stopping all wallet and coin daemons'),
  78. ('view', 'viewing Alice’s wallet in offline mode (wallet #1)'),
  79. ('listview', 'list-viewing Alice’s wallet in offline mode (wallet #2)'),
  80. )
  81. def __init__(self,trunner,cfgs,spawn):
  82. CmdTestAutosignThreaded.__init__(self,trunner,cfgs,spawn)
  83. CmdTestXMRWallet.__init__(self,trunner,cfgs,spawn)
  84. if trunner is None:
  85. return
  86. from mmgen.cfg import Config
  87. self.cfg = Config({
  88. 'coin': 'XMR',
  89. 'outdir': self.users['alice'].udir,
  90. 'wallet_dir': self.users['alice'].udir,
  91. 'wallet_rpc_password': 'passwOrd',
  92. 'test_suite': True,
  93. })
  94. self.burn_addr = make_burn_addr()
  95. self.opts.append('--xmrwallets={}'.format(self.users['alice'].kal_range)) # mmgen-autosign opts
  96. self.autosign_opts = ['--autosign'] # mmgen-xmrwallet opts
  97. self.spawn_env['MMGEN_TEST_SUITE_XMR_AUTOSIGN'] = '1'
  98. def create_tmp_wallets(self):
  99. self.spawn('',msg_only=True)
  100. data = self.users['alice']
  101. from mmgen.wallet import Wallet
  102. from mmgen.xmrwallet import MoneroWalletOps,xmrwallet_uargs
  103. from mmgen.addrlist import KeyAddrList
  104. silence()
  105. kal = KeyAddrList(
  106. cfg = self.cfg,
  107. proto = self.proto,
  108. addr_idxs = '1-2',
  109. seed = Wallet(cfg,data.mmwords).seed,
  110. skip_chksum_msg = True,
  111. key_address_validity_check = False )
  112. kal.file.write(ask_overwrite=False)
  113. fn = get_file_with_ext(data.udir,'akeys')
  114. m = MoneroWalletOps.create(
  115. self.cfg,
  116. xmrwallet_uargs(fn, '1-2', None))
  117. async_run(m.main())
  118. async_run(m.stop_wallet_daemon())
  119. end_silence()
  120. return 'ok'
  121. def _new_addr_alice(self,*args):
  122. data = self.users['alice']
  123. return self.new_addr_alice(
  124. *args,
  125. kafile = get_file_with_ext(data.udir,'akeys') )
  126. def new_account_alice(self):
  127. return self._new_addr_alice(
  128. '2',
  129. 'start',
  130. r'Creating new account for wallet .*2.* with label .*‘xmrwallet new account .*y/N\): ')
  131. def new_address_alice(self):
  132. return self._new_addr_alice(
  133. '2:1',
  134. 'continue',
  135. r'Creating new address for wallet .*2.*, account .*#1.* with label .*‘xmrwallet new address .*y/N\): ')
  136. def new_address_alice_label(self):
  137. return self._new_addr_alice(
  138. '2:1,Alice’s new address',
  139. 'stop',
  140. r'Creating new address for wallet .*2.*, account .*#1.* with label .*‘Alice’s new address .*y/N\): ')
  141. def dump_tmp_wallets(self):
  142. return self._dump_wallets(autosign=False)
  143. def dump_wallets(self):
  144. return self._dump_wallets(autosign=True)
  145. def _dump_wallets(self,autosign):
  146. data = self.users['alice']
  147. self.insert_device_online()
  148. t = self.spawn(
  149. 'mmgen-xmrwallet',
  150. self.extra_opts
  151. + [f'--wallet-dir={data.udir}', f'--daemon=localhost:{data.md.rpc_port}']
  152. + (self.autosign_opts if autosign else [])
  153. + ['dump']
  154. + ([] if autosign else [get_file_with_ext(data.udir,'akeys')]) )
  155. t.expect('2 wallets dumped')
  156. self.remove_device_online()
  157. return t
  158. def _delete_files(self,*ext_list):
  159. data = self.users['alice']
  160. self.spawn('',msg_only=True)
  161. for ext in ext_list:
  162. get_file_with_ext(data.udir,ext,no_dot=True,delete_all=True)
  163. return 'ok'
  164. def delete_tmp_wallets(self):
  165. return self._delete_files( 'MoneroWallet', 'MoneroWallet.keys', '.akeys' )
  166. def delete_wallets(self):
  167. return self._delete_files( 'MoneroWatchOnlyWallet', '.keys', '.address.txt' )
  168. def delete_tmp_dump_files(self):
  169. return self._delete_files( '.dump' )
  170. def delete_dump_files(self):
  171. return self._delete_files( '.dump' )
  172. def fund_alice1(self):
  173. return self.fund_alice(wallet=1,check_bal=False)
  174. def fund_alice2(self):
  175. return self.fund_alice(wallet=2)
  176. def autosign_setup(self):
  177. self.do_mount_online()
  178. self.asi_online.xmr_dir.mkdir(exist_ok=True)
  179. (self.asi_online.xmr_dir / 'old.vkeys').touch()
  180. self.do_umount_online()
  181. self.insert_device()
  182. t = self.run_setup(
  183. mn_type = 'mmgen',
  184. mn_file = self.users['alice'].mmwords,
  185. use_dfl_wallet = None )
  186. t.expect('Continue with Monero setup? (Y/n): ','y')
  187. t.written_to_file('View keys')
  188. self.remove_device()
  189. return t
  190. def create_watchonly_wallets(self):
  191. self.insert_device_online()
  192. t = self.create_wallets('alice', op='restore')
  193. t.read() # required!
  194. self.remove_device_online()
  195. return t
  196. def restore_wallets(self):
  197. return self.create_watchonly_wallets()
  198. def _create_transfer_tx(self,amt):
  199. self.insert_device_online()
  200. t = self.do_op('transfer','alice',f'1:0:{self.burn_addr},{amt}',no_relay=True,do_ret=True)
  201. t.read() # required!
  202. self.remove_device_online()
  203. return t
  204. def create_transfer_tx1(self):
  205. return self._create_transfer_tx('0.124')
  206. def create_transfer_tx2(self):
  207. self.do_mount_online()
  208. get_file_with_ext(self.asi_online.xmr_tx_dir,'rawtx',delete_all=True)
  209. get_file_with_ext(self.asi_online.xmr_tx_dir,'sigtx',delete_all=True)
  210. self.do_umount_online()
  211. return self._create_transfer_tx('0.257')
  212. def _xmr_autosign_op(
  213. self,
  214. op,
  215. desc = None,
  216. signable_desc = None,
  217. ext = None,
  218. wallet_arg = None,
  219. add_opts = [],
  220. wait_signed = False):
  221. if wait_signed:
  222. self._wait_signed(signable_desc)
  223. data = self.users['alice']
  224. args = (
  225. self.extra_opts
  226. + self.autosign_opts
  227. + [f'--wallet-dir={data.udir}', f'--daemon=localhost:{data.md.rpc_port}']
  228. + add_opts
  229. + [ op ]
  230. + ([get_file_with_ext(self.asi.xmr_tx_dir,ext)] if ext else [])
  231. + ([wallet_arg] if wallet_arg else [])
  232. )
  233. desc_pfx = f'{desc}, ' if desc else ''
  234. return self.spawn( 'mmgen-xmrwallet', args, extra_desc=f'({desc_pfx}Alice)' )
  235. def _sync_chkbal(self,wallet_arg,bal_chk_func):
  236. return self.sync_wallets(
  237. 'alice',
  238. op = 'sync',
  239. wallets = wallet_arg,
  240. bal_chk_func = bal_chk_func )
  241. def sync_chkbal1(self):
  242. return self._sync_chkbal( '1', lambda n,b,ub: b == ub and 1 < b < 1.12 )
  243. # 1.234567891234 - 0.124 = 1.110567891234 (minus fees)
  244. def sync_chkbal2(self):
  245. return self._sync_chkbal( '1', lambda n,b,ub: b == ub and 0.8 < b < 0.86 )
  246. # 1.234567891234 - 0.124 - 0.257 = 0.853567891234 (minus fees)
  247. def sync_chkbal3(self):
  248. return self._sync_chkbal(
  249. '1-2',
  250. lambda n,b,ub: b == ub and ((n == 1 and 0.8 < b < 0.86) or (n == 2 and b > 1.23)) )
  251. def _mine_chk(self,desc):
  252. bal_type = {'locked':'b','unlocked':'ub'}[desc]
  253. return self.mine_chk(
  254. 'alice', 1, 0,
  255. lambda x: 0 < getattr(x,bal_type) < 1.234567891234,
  256. f'{desc} balance 0 < 1.234567891234' )
  257. def submit_transfer_tx1(self):
  258. return self._submit_transfer_tx()
  259. def resubmit_transfer_tx1(self):
  260. return self._submit_transfer_tx(
  261. relay_parm = self.tx_relay_daemon_proxy_parm,
  262. op = 'resubmit',
  263. check_bal = False)
  264. def submit_transfer_tx2(self):
  265. return self._submit_transfer_tx( relay_parm=self.tx_relay_daemon_parm )
  266. def _submit_transfer_tx(self,relay_parm=None,ext=None,op='submit',check_bal=True):
  267. self.insert_device_online()
  268. t = self._xmr_autosign_op(
  269. op = op,
  270. add_opts = [f'--tx-relay-daemon={relay_parm}'] if relay_parm else [],
  271. ext = ext,
  272. signable_desc = 'transaction',
  273. wait_signed = op == 'submit')
  274. t.expect( f'{op.capitalize()} transaction? (y/N): ', 'y' )
  275. t.written_to_file('Submitted transaction')
  276. self.remove_device_online()
  277. if check_bal:
  278. t.ok()
  279. return self._mine_chk('unlocked')
  280. else:
  281. return t
  282. def _export_outputs(self,wallet_arg,add_opts=[]):
  283. self.insert_device_online()
  284. t = self._xmr_autosign_op(
  285. op = 'export-outputs',
  286. wallet_arg = wallet_arg,
  287. add_opts = add_opts )
  288. t.written_to_file('Wallet outputs')
  289. self.remove_device_online()
  290. return t
  291. def export_outputs1(self):
  292. return self._export_outputs('1',['--rescan-blockchain'])
  293. def export_outputs2(self):
  294. return self._export_outputs('1-2')
  295. def _import_key_images(self,wallet_arg):
  296. self.insert_device_online()
  297. t = self._xmr_autosign_op(
  298. op = 'import-key-images',
  299. wallet_arg = wallet_arg,
  300. signable_desc = 'wallet outputs',
  301. wait_signed = True)
  302. t.read()
  303. self.remove_device_online()
  304. return t
  305. def import_key_images1(self):
  306. return self._import_key_images(None)
  307. def import_key_images2(self):
  308. return self._import_key_images(None)
  309. def txlist(self):
  310. self.insert_device_online()
  311. t = self.spawn( 'mmgen-xmrwallet', self.autosign_opts + ['txlist'] )
  312. t.match_expect_list([
  313. 'SUBMITTED',
  314. 'Network','Submitted',
  315. 'Transfer 1:0','-> ext',
  316. 'Transfer 1:0','-> ext'
  317. ])
  318. self.remove_device_online()
  319. return t
  320. def check_tx_dirs(self):
  321. self.do_mount_online()
  322. before = '\n'.join(self._gen_listing())
  323. self.do_umount_online()
  324. t = self.spawn('mmgen-autosign', self.opts + ['clean'])
  325. t.read()
  326. self.do_mount_online()
  327. after = '\n'.join(self._gen_listing())
  328. self.do_umount_online()
  329. imsg(f'\nBefore cleaning:\n{before}')
  330. imsg(f'\nAfter cleaning:\n{after}')
  331. pat = r'xmr/tx: \s*\S+\.subtx \S+\.subtx\s+xmr/outputs:\s*$'
  332. assert re.search( pat, after, re.DOTALL ), f'regex search for {pat} failed'
  333. return t
  334. def view(self):
  335. return self.sync_wallets('alice', op='view', wallets='1')
  336. def listview(self):
  337. return self.sync_wallets('alice', op='listview', wallets='2')