ut_rpc.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. #!/usr/bin/env python3
  2. """
  3. test.unit_tests_d.ut_rpc: RPC unit test for the MMGen suite
  4. """
  5. import time
  6. from mmgen.common import *
  7. from mmgen.protocol import init_proto
  8. from mmgen.rpc import rpc_init
  9. from mmgen.daemon import CoinDaemon
  10. from mmgen.proto.xmr.rpc import MoneroRPCClient,MoneroWalletRPCClient
  11. from mmgen.proto.xmr.daemon import MoneroWalletDaemon
  12. def cfg_file_auth_test(proto,d,bad_auth=False):
  13. m = 'missing credentials' if bad_auth else f'credentials from {d.cfg_file}'
  14. qmsg(cyan(f'\n Testing authentication with {m}:'))
  15. time.sleep(0.1) # race condition
  16. d.remove_datadir() # removes cookie file to force authentication from cfg file
  17. os.makedirs(d.network_datadir)
  18. if not bad_auth:
  19. cf = os.path.join(d.datadir,d.cfg_file)
  20. with open(cf,'a') as fp:
  21. fp.write('\nrpcuser = ut_rpc\nrpcpassword = ut_rpc_passw0rd\n')
  22. d.flag.keep_cfg_file = True
  23. d.start()
  24. if bad_auth:
  25. os.rename(d.auth_cookie_fn,d.auth_cookie_fn+'.bak')
  26. try:
  27. async_run(rpc_init(proto))
  28. except Exception as e:
  29. vmsg(yellow(str(e)))
  30. else:
  31. die(3,'No error on missing credentials!')
  32. os.rename(d.auth_cookie_fn+'.bak',d.auth_cookie_fn)
  33. else:
  34. rpc = async_run(rpc_init(proto))
  35. assert rpc.auth.user == 'ut_rpc', f'{rpc.auth.user}: user is not ut_rpc!'
  36. d.stop()
  37. def print_daemon_info(rpc):
  38. if rpc.proto.base_proto == 'Monero':
  39. msg(f"""
  40. DAEMON VERSION: {rpc.daemon_version} [{rpc.daemon_version_str}]
  41. NETWORK: {rpc.proto.coin} {rpc.proto.network.upper()}
  42. """.rstrip())
  43. else:
  44. msg(f"""
  45. DAEMON VERSION: {rpc.daemon_version} [{rpc.daemon_version_str}]
  46. CAPS: {rpc.caps}
  47. NETWORK: {rpc.proto.coin} {rpc.proto.network.upper()}
  48. CHAIN: {rpc.chain}
  49. BLOCKCOUNT: {rpc.blockcount}
  50. CUR_DATE: {rpc.cur_date} [{make_timestr(rpc.cur_date)}]
  51. """.rstrip())
  52. if rpc.proto.base_proto == 'Bitcoin':
  53. def fmt_dict(d):
  54. return '\n ' + '\n '.join( pp_fmt(d).split('\n') ) + '\n'
  55. msg(f"""
  56. NETWORKINFO: {fmt_dict(rpc.cached["networkinfo"])}
  57. BLOCKCHAININFO: {fmt_dict(rpc.cached["blockchaininfo"])}
  58. DEPLOYMENTINFO: {fmt_dict(rpc.cached["deploymentinfo"])}
  59. """.rstrip())
  60. msg('')
  61. def do_msg(rpc,backend):
  62. bname = type(rpc.backend).__name__
  63. qmsg(' Testing backend {!r}{}'.format( bname, '' if backend == bname else f' [{backend}]' ))
  64. class init_test:
  65. async def btc(proto,backend,daemon):
  66. rpc = await rpc_init(proto,backend,daemon)
  67. do_msg(rpc,backend)
  68. bh = (await rpc.call('getblockchaininfo',timeout=300))['bestblockhash']
  69. await rpc.gathered_call('getblock',((bh,),(bh,1)),timeout=300)
  70. await rpc.gathered_call(None,(('getblock',(bh,)),('getblock',(bh,1))),timeout=300)
  71. return rpc
  72. async def bch(proto,backend,daemon):
  73. rpc = await rpc_init(proto,backend,daemon)
  74. do_msg(rpc,backend)
  75. return rpc
  76. ltc = bch
  77. async def eth(proto,backend,daemon):
  78. rpc = await rpc_init(proto,backend,daemon)
  79. do_msg(rpc,backend)
  80. await rpc.call('eth_blockNumber',timeout=300)
  81. return rpc
  82. etc = eth
  83. def run_test(network_ids,test_cf_auth=False,daemon_ids=None):
  84. def do_test(d):
  85. if not opt.no_daemon_stop:
  86. d.stop()
  87. if not opt.no_daemon_autostart:
  88. d.remove_datadir()
  89. d.start()
  90. for n,backend in enumerate(g.autoset_opts['rpc_backend'].choices):
  91. test = getattr(init_test,d.proto.coin.lower())
  92. rpc = async_run(test(d.proto,backend,d))
  93. if not n and opt.verbose:
  94. print_daemon_info(rpc)
  95. if not opt.no_daemon_stop:
  96. d.stop()
  97. if test_cf_auth and gc.platform != 'win':
  98. cfg_file_auth_test(d.proto,d)
  99. cfg_file_auth_test(d.proto,d,bad_auth=True)
  100. qmsg('')
  101. for network_id in network_ids:
  102. proto = init_proto(network_id=network_id)
  103. ids = (lambda x:
  104. set(daemon_ids) & set(x) if daemon_ids else x
  105. )(CoinDaemon.get_daemon_ids(proto.coin))
  106. for daemon_id in ids:
  107. do_test( CoinDaemon(proto=proto,test_suite=True,daemon_id=daemon_id) )
  108. return True
  109. class unit_tests:
  110. altcoin_deps = ('ltc','bch','geth','erigon','openethereum','parity','xmrwallet')
  111. win_skip = ('xmrwallet',) # FIXME - wallet doesn't open
  112. arm_skip = ('openethereum','parity') # no prebuilt binaries for ARM
  113. def btc(self,name,ut):
  114. return run_test(['btc','btc_tn'],test_cf_auth=True)
  115. def ltc(self,name,ut):
  116. return run_test(['ltc','ltc_tn'],test_cf_auth=True)
  117. def bch(self,name,ut):
  118. return run_test(['bch','bch_tn'],test_cf_auth=True)
  119. def geth(self,name,ut):
  120. return run_test(['eth_tn','eth_rt'],daemon_ids=['geth']) # mainnet returns EIP-155 error on empty blockchain
  121. def erigon(self,name,ut):
  122. return run_test(['eth','eth_tn','eth_rt'],daemon_ids=['erigon'])
  123. def openethereum(self,name,ut):
  124. return run_test(['eth','eth_tn','eth_rt'],daemon_ids=['openethereum'])
  125. def parity(self,name,ut):
  126. return run_test(['etc'])
  127. def xmrwallet(self,name,ut):
  128. def test_monerod_rpc(md):
  129. rpc = MoneroRPCClient(
  130. proto = md.proto,
  131. host = md.host,
  132. port = md.rpc_port,
  133. user = None,
  134. passwd = None,
  135. daemon = md,
  136. )
  137. if opt.verbose:
  138. print_daemon_info(rpc)
  139. rpc.call_raw('get_height')
  140. rpc.call('get_last_block_header')
  141. async def run():
  142. networks = init_proto('xmr').networks
  143. daemons = [(
  144. CoinDaemon(proto=proto,test_suite=True),
  145. MoneroWalletDaemon(
  146. proto = proto,
  147. test_suite = True,
  148. wallet_dir = 'test/trash2',
  149. passwd = 'ut_rpc_passw0rd' )
  150. ) for proto in (init_proto( 'xmr', network=network ) for network in networks) ]
  151. for md,wd in daemons:
  152. if not opt.no_daemon_autostart:
  153. md.start()
  154. wd.start()
  155. test_monerod_rpc(md)
  156. c = MoneroWalletRPCClient(daemon=wd)
  157. fn = f'monero-{wd.network}-junk-wallet'
  158. qmsg(f'Creating {wd.network} wallet')
  159. c.call(
  160. 'restore_deterministic_wallet',
  161. filename = fn,
  162. password = 'foo',
  163. seed = xmrseed().fromhex('beadface'*8,tostr=True) )
  164. qmsg(f'Opening {wd.network} wallet')
  165. c.call( 'open_wallet', filename=fn, password='foo' )
  166. for md,wd in daemons:
  167. wd.wait = False
  168. await wd.rpc.stop_daemon()
  169. if not opt.no_daemon_stop:
  170. md.wait = False
  171. await md.rpc.stop_daemon()
  172. gmsg('OK')
  173. from mmgen.xmrseed import xmrseed
  174. import shutil
  175. shutil.rmtree('test/trash2',ignore_errors=True)
  176. os.makedirs('test/trash2')
  177. async_run(run())
  178. return True