ct_xmr_autosign.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  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 yellow,purple,gray
  16. from mmgen.util import fmt,async_run
  17. from ..include.common import (
  18. cfg,
  19. oqmsg,
  20. oqmsg_r,
  21. imsg,
  22. silence,
  23. end_silence
  24. )
  25. from .common import get_file_with_ext
  26. from .ct_xmrwallet import CmdTestXMRWallet
  27. from .ct_autosign import CmdTestAutosignBase
  28. def make_burn_addr():
  29. from mmgen.tool.coin import tool_cmd
  30. return tool_cmd(
  31. cfg = cfg,
  32. cmdname = 'privhex2addr',
  33. proto = cfg._proto,
  34. mmtype = 'monero' ).privhex2addr('beadcafe'*8)
  35. class CmdTestXMRAutosign(CmdTestXMRWallet,CmdTestAutosignBase):
  36. """
  37. Monero autosigning operations
  38. """
  39. tmpdir_nums = [39]
  40. # ct_xmrwallet attrs:
  41. user_data = (
  42. ('miner', '98831F3A', False, 130, '1', []),
  43. ('alice', 'FE3C6545', True, 150, '1-2', []),
  44. )
  45. # ct_autosign attrs:
  46. coins = ['xmr']
  47. daemon_coins = []
  48. txfile_coins = []
  49. live = False
  50. simulate_led = False
  51. bad_tx_count = 0
  52. tx_relay_user = 'miner'
  53. no_insert_check = False
  54. win_skip = True
  55. cmd_group = (
  56. ('daemon_version', 'checking daemon version'),
  57. ('create_tmp_wallets', 'creating temporary online wallets for Alice'),
  58. ('new_account_alice', 'adding an account to Alice’s tmp wallet'),
  59. ('new_address_alice', 'adding an address to Alice’s tmp wallet'),
  60. ('new_address_alice_label', 'adding an address to Alice’s tmp wallet (with label)'),
  61. ('dump_tmp_wallets', 'dumping Alice’s tmp wallets'),
  62. ('delete_tmp_wallets', 'deleting Alice’s tmp wallets'),
  63. ('autosign_clean', 'cleaning signable file directories'),
  64. ('autosign_setup', 'autosign setup with Alice’s seed'),
  65. ('create_watchonly_wallets', 'creating online (watch-only) wallets for Alice'),
  66. ('delete_tmp_dump_files', 'deleting Alice’s dump files'),
  67. ('gen_kafiles', 'generating key-address files for Miner'),
  68. ('create_wallets_miner', 'creating Monero wallets for Miner'),
  69. ('mine_initial_coins', 'mining initial coins'),
  70. ('fund_alice1', 'sending funds to Alice (wallet #1)'),
  71. ('fund_alice2', 'sending funds to Alice (wallet #2)'),
  72. ('autosign_start_thread', 'starting autosign wait loop'),
  73. ('create_transfer_tx1', 'creating a transfer TX'),
  74. ('submit_transfer_tx1', 'submitting the transfer TX'),
  75. ('resubmit_transfer_tx1', 'resubmitting the transfer TX'),
  76. ('export_outputs1', 'exporting outputs from Alice’s watch-only wallet #1'),
  77. ('import_key_images1', 'importing signed key images into Alice’s online wallets'),
  78. ('sync_chkbal1', 'syncing Alice’s wallet #1'),
  79. ('create_transfer_tx2', 'creating a transfer TX (for relaying via proxy)'),
  80. ('submit_transfer_tx2', 'submitting the transfer TX (relaying via proxy)'),
  81. ('sync_chkbal2', 'syncing Alice’s wallets and checking balance'),
  82. ('dump_wallets', 'dumping Alice’s wallets'),
  83. ('delete_wallets', 'deleting Alice’s wallets'),
  84. ('restore_wallets', 'creating online (watch-only) wallets for Alice'),
  85. ('delete_dump_files', 'deleting Alice’s dump files'),
  86. ('export_outputs2', 'exporting outputs from Alice’s watch-only wallets'),
  87. ('import_key_images2', 'importing signed key images into Alice’s online wallets'),
  88. ('sync_chkbal3', 'syncing Alice’s wallets and checking balance'),
  89. ('txlist', 'listing Alice’s submitted transactions'),
  90. ('check_tx_dirs', 'cleaning and checking signable file directories'),
  91. ('autosign_kill_thread', 'stopping autosign wait loop'),
  92. ('stop_daemons', 'stopping all wallet and coin daemons'),
  93. ('view', 'viewing Alice’s wallet in offline mode (wallet #1)'),
  94. ('listview', 'list-viewing Alice’s wallet in offline mode (wallet #2)'),
  95. )
  96. def __init__(self,trunner,cfgs,spawn):
  97. CmdTestAutosignBase.__init__(self,trunner,cfgs,spawn)
  98. CmdTestXMRWallet.__init__(self,trunner,cfgs,spawn)
  99. if trunner is None:
  100. return
  101. from mmgen.cfg import Config
  102. self.cfg = Config({
  103. 'coin': 'XMR',
  104. 'outdir': self.users['alice'].udir,
  105. 'wallet_dir': self.users['alice'].udir,
  106. 'wallet_rpc_password': 'passwOrd',
  107. 'test_suite': True,
  108. })
  109. self.burn_addr = make_burn_addr()
  110. self.opts.append('--xmrwallets={}'.format( self.users['alice'].kal_range )) # mmgen-autosign opts
  111. self.autosign_opts = [f'--autosign-mountpoint={self.mountpoint}'] # mmgen-xmrwallet opts
  112. self.tx_count = 1
  113. self.spawn_env['MMGEN_TEST_SUITE_XMR_AUTOSIGN'] = '1'
  114. def create_tmp_wallets(self):
  115. self.spawn('',msg_only=True)
  116. data = self.users['alice']
  117. from mmgen.wallet import Wallet
  118. from mmgen.xmrwallet import MoneroWalletOps,xmrwallet_uargs
  119. from mmgen.addrlist import KeyAddrList
  120. silence()
  121. kal = KeyAddrList(
  122. cfg = self.cfg,
  123. proto = self.proto,
  124. addr_idxs = '1-2',
  125. seed = Wallet(cfg,data.mmwords).seed,
  126. skip_chksum_msg = True,
  127. key_address_validity_check = False )
  128. kal.file.write(ask_overwrite=False)
  129. fn = get_file_with_ext(data.udir,'akeys')
  130. m = MoneroWalletOps.create(
  131. self.cfg,
  132. xmrwallet_uargs(fn, '1-2', None))
  133. async_run(m.main())
  134. async_run(m.stop_wallet_daemon())
  135. end_silence()
  136. return 'ok'
  137. def _new_addr_alice(self,*args):
  138. data = self.users['alice']
  139. return self.new_addr_alice(
  140. *args,
  141. kafile = get_file_with_ext(data.udir,'akeys') )
  142. def new_account_alice(self):
  143. return self._new_addr_alice(
  144. '2',
  145. 'start',
  146. r'Creating new account for wallet .*2.* with label .*‘xmrwallet new account .*y/N\): ')
  147. def new_address_alice(self):
  148. return self._new_addr_alice(
  149. '2:1',
  150. 'continue',
  151. r'Creating new address for wallet .*2.*, account .*#1.* with label .*‘xmrwallet new address .*y/N\): ')
  152. def new_address_alice_label(self):
  153. return self._new_addr_alice(
  154. '2:1,Alice’s new address',
  155. 'stop',
  156. r'Creating new address for wallet .*2.*, account .*#1.* with label .*‘Alice’s new address .*y/N\): ')
  157. def dump_tmp_wallets(self):
  158. return self._dump_wallets(autosign=False)
  159. def dump_wallets(self):
  160. return self._dump_wallets(autosign=True)
  161. def _dump_wallets(self,autosign):
  162. data = self.users['alice']
  163. t = self.spawn(
  164. 'mmgen-xmrwallet',
  165. self.extra_opts
  166. + [f'--wallet-dir={data.udir}', f'--daemon=localhost:{data.md.rpc_port}']
  167. + (self.autosign_opts if autosign else [])
  168. + ['dump']
  169. + ([] if autosign else [get_file_with_ext(data.udir,'akeys')]) )
  170. t.expect('2 wallets dumped')
  171. return t
  172. def _delete_files(self,*ext_list):
  173. data = self.users['alice']
  174. self.spawn('',msg_only=True)
  175. for ext in ext_list:
  176. get_file_with_ext(data.udir,ext,no_dot=True,delete_all=True)
  177. return 'ok'
  178. def delete_tmp_wallets(self):
  179. return self._delete_files( 'MoneroWallet', 'MoneroWallet.keys', '.akeys' )
  180. def delete_wallets(self):
  181. return self._delete_files( 'MoneroWatchOnlyWallet', '.keys', '.address.txt' )
  182. def delete_tmp_dump_files(self):
  183. return self._delete_files( '.dump' )
  184. def delete_dump_files(self):
  185. return self._delete_files( '.dump' )
  186. def fund_alice1(self):
  187. return self.fund_alice(wallet=1,check_bal=False)
  188. def fund_alice2(self):
  189. return self.fund_alice(wallet=2)
  190. def autosign_setup(self):
  191. self.insert_device()
  192. Path(self.autosign_xmr_dir).mkdir(parents=True,exist_ok=True)
  193. Path(self.autosign_xmr_dir,'old.vkeys').touch()
  194. t = self.run_setup(
  195. mn_type = 'mmgen',
  196. mn_file = self.users['alice'].mmwords,
  197. use_dfl_wallet = None )
  198. t.expect('Continue with Monero setup? (Y/n): ','y')
  199. t.written_to_file('View keys')
  200. return t
  201. def autosign_start_thread(self):
  202. if self.asi.dev_disk_path.exists():
  203. self.asi.dev_disk_path.unlink()
  204. def run():
  205. t = self.spawn('mmgen-autosign', self.opts + ['wait'], direct_exec=True)
  206. self.write_to_tmpfile('autosign_thread_pid',str(t.ep.pid))
  207. import threading
  208. threading.Thread( target=run, name='Autosign wait loop' ).start()
  209. time.sleep(0.2)
  210. return 'silent'
  211. def autosign_kill_thread(self):
  212. self.spawn('',msg_only=True)
  213. pid = int(self.read_from_tmpfile('autosign_thread_pid'))
  214. self.delete_tmpfile('autosign_thread_pid')
  215. from signal import SIGTERM
  216. imsg(purple(f'Killing autosign wait loop [PID {pid}]'))
  217. try:
  218. os.kill(pid,SIGTERM)
  219. except:
  220. imsg(yellow(f'{pid}: no such process'))
  221. return 'ok'
  222. def create_watchonly_wallets(self):
  223. return self.create_wallets( 'alice', op='restore' )
  224. def restore_wallets(self):
  225. return self.create_watchonly_wallets()
  226. def _create_transfer_tx(self,amt):
  227. t = self.do_op('transfer','alice',f'1:0:{self.burn_addr},{amt}',no_relay=True,do_ret=True)
  228. t.read() # required!
  229. self.insert_device()
  230. return t
  231. def create_transfer_tx1(self):
  232. return self._create_transfer_tx('0.124')
  233. def create_transfer_tx2(self):
  234. get_file_with_ext(self.asi.xmr_tx_dir,'rawtx',delete_all=True)
  235. get_file_with_ext(self.asi.xmr_tx_dir,'sigtx',delete_all=True)
  236. return self._create_transfer_tx('0.257')
  237. def _wait_signed(self,dtype):
  238. oqmsg_r(gray(f'→ offline wallet{"s" if dtype.endswith("s") else ""} signing {dtype}'))
  239. while True:
  240. oqmsg_r(gray('.'))
  241. if not self.asi.dev_disk_path.exists():
  242. break
  243. time.sleep(0.5)
  244. oqmsg(gray('done'))
  245. def _xmr_autosign_op(
  246. self,
  247. op,
  248. desc = None,
  249. dtype = None,
  250. ext = None,
  251. wallet_arg = None,
  252. add_opts = [],
  253. wait_signed = False):
  254. if wait_signed:
  255. self._wait_signed(dtype)
  256. data = self.users['alice']
  257. args = (
  258. self.extra_opts
  259. + self.autosign_opts
  260. + [f'--wallet-dir={data.udir}', f'--daemon=localhost:{data.md.rpc_port}']
  261. + add_opts
  262. + [ op ]
  263. + ([get_file_with_ext(self.asi.xmr_tx_dir,ext)] if ext else [])
  264. + ([wallet_arg] if wallet_arg else [])
  265. )
  266. desc_pfx = f'{desc}, ' if desc else ''
  267. return self.spawn( 'mmgen-xmrwallet', args, extra_desc=f'({desc_pfx}Alice)' )
  268. def _sync_chkbal(self,wallet_arg,bal_chk_func):
  269. return self.sync_wallets(
  270. 'alice',
  271. op = 'sync',
  272. wallets = wallet_arg,
  273. bal_chk_func = bal_chk_func )
  274. def sync_chkbal1(self):
  275. return self._sync_chkbal( '1', lambda n,b,ub: b == ub and 1 < b < 1.12 )
  276. # 1.234567891234 - 0.124 = 1.110567891234 (minus fees)
  277. def sync_chkbal2(self):
  278. return self._sync_chkbal( '1', lambda n,b,ub: b == ub and 0.8 < b < 0.86 )
  279. # 1.234567891234 - 0.124 - 0.257 = 0.853567891234 (minus fees)
  280. def sync_chkbal3(self):
  281. return self._sync_chkbal(
  282. '1-2',
  283. lambda n,b,ub: b == ub and ((n == 1 and 0.8 < b < 0.86) or (n == 2 and b > 1.23)) )
  284. def _mine_chk(self,desc):
  285. bal_type = {'locked':'b','unlocked':'ub'}[desc]
  286. return self.mine_chk(
  287. 'alice', 1, 0,
  288. lambda x: 0 < getattr(x,bal_type) < 1.234567891234,
  289. f'{desc} balance 0 < 1.234567891234' )
  290. def submit_transfer_tx1(self):
  291. return self._submit_transfer_tx( ext='sigtx' )
  292. def resubmit_transfer_tx1(self):
  293. return self._submit_transfer_tx(
  294. relay_parm = self.tx_relay_daemon_proxy_parm,
  295. op = 'resubmit',
  296. check_bal = False)
  297. def submit_transfer_tx2(self):
  298. return self._submit_transfer_tx( relay_parm=self.tx_relay_daemon_parm )
  299. def _submit_transfer_tx(self,relay_parm=None,ext=None,op='submit',check_bal=True):
  300. t = self._xmr_autosign_op(
  301. op = op,
  302. add_opts = [f'--tx-relay-daemon={relay_parm}'] if relay_parm else [],
  303. ext = ext,
  304. dtype = 'transaction',
  305. wait_signed = op == 'submit' )
  306. t.expect( f'{op.capitalize()} transaction? (y/N): ', 'y' )
  307. t.written_to_file('Submitted transaction')
  308. if check_bal:
  309. t.ok()
  310. return self._mine_chk('unlocked')
  311. else:
  312. return t
  313. def _export_outputs(self,wallet_arg,add_opts=[]):
  314. t = self._xmr_autosign_op(
  315. op = 'export-outputs',
  316. wallet_arg = wallet_arg,
  317. add_opts = add_opts )
  318. t.written_to_file('Wallet outputs')
  319. t.read() # required!
  320. self.insert_device()
  321. return t
  322. def export_outputs1(self):
  323. return self._export_outputs('1',['--rescan-blockchain'])
  324. def export_outputs2(self):
  325. return self._export_outputs('1-2')
  326. def _import_key_images(self,wallet_arg):
  327. t = self._xmr_autosign_op(
  328. op = 'import-key-images',
  329. wallet_arg = wallet_arg,
  330. dtype = 'wallet outputs',
  331. wait_signed = True )
  332. return t
  333. def import_key_images1(self):
  334. return self._import_key_images(None)
  335. def import_key_images2(self):
  336. return self._import_key_images(None)
  337. def create_fake_tx_files(self):
  338. imsg('Creating fake transaction files')
  339. self.asi.msg_dir.mkdir(exist_ok=True)
  340. self.asi.xmr_dir.mkdir(exist_ok=True)
  341. self.asi.xmr_tx_dir.mkdir(exist_ok=True)
  342. self.asi.xmr_outputs_dir.mkdir(exist_ok=True)
  343. for fn in (
  344. 'a.rawtx', 'a.sigtx',
  345. 'b.rawtx', 'b.sigtx',
  346. 'c.rawtx',
  347. 'd.sigtx',
  348. ):
  349. (self.asi.tx_dir / fn).touch()
  350. for fn in (
  351. 'a.rawmsg.json', 'a.sigmsg.json',
  352. 'b.rawmsg.json',
  353. 'c.sigmsg.json',
  354. 'd.rawmsg.json', 'd.sigmsg.json',
  355. ):
  356. (self.asi.msg_dir / fn).touch()
  357. for fn in (
  358. 'a.rawtx', 'a.sigtx', 'a.subtx',
  359. 'b.rawtx', 'b.sigtx',
  360. 'c.subtx',
  361. 'd.rawtx', 'd.subtx',
  362. 'e.rawtx',
  363. 'f.sigtx','f.subtx',
  364. ):
  365. (self.asi.xmr_tx_dir / fn).touch()
  366. for fn in (
  367. 'a.raw', 'a.sig',
  368. 'b.raw',
  369. 'c.sig',
  370. ):
  371. (self.asi.xmr_outputs_dir / fn).touch()
  372. return 'ok'
  373. def _gen_listing(self):
  374. for k in ('tx_dir','msg_dir','xmr_tx_dir','xmr_outputs_dir'):
  375. d = getattr(self.asi,k)
  376. if d.is_dir():
  377. yield '{:12} {}'.format(
  378. str(Path(*d.parts[4:])) + ':',
  379. ' '.join(sorted(i.name for i in d.iterdir()))).strip()
  380. def autosign_clean(self):
  381. self.create_fake_tx_files()
  382. before = '\n'.join(self._gen_listing())
  383. t = self.spawn('mmgen-autosign', self.opts + ['clean'])
  384. out = t.read()
  385. after = '\n'.join(self._gen_listing())
  386. chk = """
  387. tx: a.sigtx b.sigtx c.rawtx d.sigtx
  388. msg: a.sigmsg.json b.rawmsg.json c.sigmsg.json d.sigmsg.json
  389. xmr/tx: a.subtx b.sigtx c.subtx d.subtx e.rawtx f.subtx
  390. xmr/outputs:
  391. """
  392. shutil.rmtree(self.asi.mountpoint)
  393. self.asi.tx_dir.mkdir(parents=True)
  394. imsg(f'\nBefore cleaning:\n{before}')
  395. imsg(f'\nAfter cleaning:\n{after}')
  396. assert '13 files shredded' in out
  397. assert after + '\n' == fmt(chk), f'\n{after}\n!=\n{fmt(chk)}'
  398. return t
  399. def txlist(self):
  400. t = self.spawn( 'mmgen-xmrwallet', self.autosign_opts + ['txlist'] )
  401. t.match_expect_list([
  402. 'SUBMITTED',
  403. 'Network','Submitted',
  404. 'Transfer 1:0','-> ext',
  405. 'Transfer 1:0','-> ext'
  406. ])
  407. return t
  408. def check_tx_dirs(self):
  409. before = '\n'.join(self._gen_listing())
  410. t = self.spawn('mmgen-autosign', self.opts + ['clean'])
  411. t.read()
  412. after = '\n'.join(self._gen_listing())
  413. imsg(f'\nBefore cleaning:\n{before}')
  414. imsg(f'\nAfter cleaning:\n{after}')
  415. pat = r'xmr/tx: \s*\S+\.subtx \S+\.subtx\s+xmr/outputs:\s*$'
  416. assert re.search( pat, after, re.DOTALL ), f'regex search for {pat} failed'
  417. return t
  418. def view(self):
  419. return self.sync_wallets('alice', op='view', wallets='1')
  420. def listview(self):
  421. return self.sync_wallets('alice', op='listview', wallets='2')