regtest.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  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. #
  6. # This program is free software: you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation, either version 3 of the License, or
  9. # (at your option) any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. """
  19. proto.btc.regtest: Coin daemon regression test mode setup and operations
  20. """
  21. import os, shutil, json
  22. from ...util import msg, gmsg, die, capfirst, suf
  23. from ...util2 import cliargs_convert
  24. from ...protocol import init_proto
  25. from ...rpc import rpc_init, json_encoder
  26. from ...objmethods import MMGenObject
  27. from ...daemon import CoinDaemon
  28. def create_data_dir(cfg, data_dir):
  29. try:
  30. os.stat(os.path.join(data_dir, 'regtest'))
  31. except:
  32. pass
  33. else:
  34. from ...ui import keypress_confirm
  35. if keypress_confirm(
  36. cfg,
  37. f'Delete your existing MMGen regtest setup at {data_dir!r} and create a new one?'):
  38. shutil.rmtree(data_dir)
  39. else:
  40. die(1, 'Exiting')
  41. try:
  42. os.makedirs(data_dir)
  43. except:
  44. pass
  45. class MMGenRegtest(MMGenObject):
  46. rpc_user = 'bobandalice'
  47. rpc_password = 'hodltothemoon'
  48. users = ('bob', 'alice', 'carol', 'miner')
  49. coins = ('btc', 'bch', 'ltc')
  50. usr_cmds = (
  51. 'setup',
  52. 'generate',
  53. 'send',
  54. 'start',
  55. 'stop',
  56. 'state',
  57. 'balances',
  58. 'mempool',
  59. 'cli',
  60. 'wallet_cli')
  61. bdb_hdseed = 'beadcafe' * 8
  62. bdb_miner_wif = 'cTyMdQ2BgfAsjopRVZrj7AoEGp97pKfrC2NkqLuwHr4KHfPNAKwp'
  63. bdb_miner_addrs = {
  64. # cTyMdQ2BgfAsjopRVZrj7AoEGp97pKfrC2NkqLuwHr4KHfPNAKwp hdseed=1
  65. 'btc': 'bcrt1qaq8t3pakcftpk095tnqfv5cmmczysls024atnd',
  66. 'ltc': 'rltc1qaq8t3pakcftpk095tnqfv5cmmczysls05c8zyn',
  67. 'bch': 'n2fxhNx27GhHAWQhyuZ5REcBNrJqCJsJ12',
  68. }
  69. def __init__(self, cfg, coin, *, bdb_wallet=False):
  70. self.cfg = cfg
  71. self.coin = coin.lower()
  72. self.bdb_wallet = bdb_wallet
  73. assert self.coin in self.coins, f'{coin!r}: invalid coin for regtest'
  74. self.proto = init_proto(cfg, self.coin, regtest=True, need_amt=True)
  75. self.d = CoinDaemon(
  76. cfg,
  77. network_id = self.coin + '_rt',
  78. test_suite = cfg.test_suite,
  79. opts = ['bdb_wallet'] if bdb_wallet else None)
  80. # Caching creates problems (broken pipe) when recreating + loading wallets,
  81. # so reinstantiate with every call:
  82. @property
  83. async def rpc(self):
  84. return await rpc_init(self.cfg, self.proto, backend=None, daemon=self.d)
  85. @property
  86. async def miner_addr(self):
  87. if not hasattr(self, '_miner_addr'):
  88. self._miner_addr = (
  89. self.bdb_miner_addrs[self.coin] if self.bdb_wallet else
  90. await self.rpc_call('getnewaddress', wallet='miner'))
  91. return self._miner_addr
  92. @property
  93. async def miner_wif(self):
  94. if not hasattr(self, '_miner_wif'):
  95. self._miner_wif = (
  96. self.bdb_miner_wif if self.bdb_wallet else
  97. await self.rpc_call('dumpprivkey', (await self.miner_addr), wallet='miner'))
  98. return self._miner_wif
  99. def create_hdseed_wif(self):
  100. from ...tool.api import tool_api
  101. t = tool_api(self.cfg)
  102. t.init_coin(self.proto.coin, self.proto.network)
  103. t.addrtype = 'compressed' if self.proto.coin == 'BCH' else 'bech32'
  104. return t.hex2wif(self.bdb_hdseed)
  105. async def generate(self, blocks=1, *, silent=False):
  106. blocks = int(blocks)
  107. if self.d.state == 'stopped':
  108. die(1, 'Regtest daemon is not running')
  109. self.d.wait_for_state('ready')
  110. # very slow with descriptor wallet and large block count - 'generatetodescriptor' no better
  111. out = await self.rpc_call(
  112. 'generatetoaddress',
  113. blocks,
  114. await self.miner_addr,
  115. wallet = 'miner')
  116. if len(out) != blocks:
  117. die(4, 'Error generating blocks')
  118. if not silent:
  119. gmsg(f'Mined {blocks} block{suf(blocks)}')
  120. async def create_wallet(self, user):
  121. return await (await self.rpc).icall(
  122. 'createwallet',
  123. wallet_name = user,
  124. blank = user != 'miner' or self.bdb_wallet,
  125. no_keys = user != 'miner',
  126. descriptors = not self.bdb_wallet,
  127. load_on_startup = False)
  128. async def setup(self):
  129. try:
  130. os.makedirs(self.d.datadir)
  131. except:
  132. pass
  133. if self.d.state != 'stopped':
  134. await self.rpc_call('stop')
  135. create_data_dir(self.cfg, self.d.datadir)
  136. gmsg(f'Starting {self.coin.upper()} regtest setup')
  137. self.d.start(silent=True)
  138. for user in ('miner', 'bob', 'alice'):
  139. gmsg(f'Creating {capfirst(user)}’s tracking wallet')
  140. await self.create_wallet(user)
  141. # BCH and LTC daemons refuse to set HD seed with empty blockchain ("in IBD" error),
  142. # so generate a block:
  143. await self.generate(1)
  144. # Unfortunately, we don’t get deterministic output with BCH and LTC even with fixed
  145. # hdseed, as their 'sendtoaddress' calls produce non-deterministic TXIDs due to random
  146. # input ordering and fee estimation.
  147. if self.bdb_wallet:
  148. await (await self.rpc).call(
  149. 'sethdseed',
  150. True,
  151. self.create_hdseed_wif(),
  152. wallet = 'miner')
  153. # Broken litecoind can only mine 431 blocks in regtest mode, so generate just enough
  154. # blocks to fund the test suite. Generation is slow, so divide into chunks:
  155. for n in (100, 100, 100, 92): # 392 blocks
  156. await self.generate(n)
  157. gmsg('Setup complete')
  158. if self.cfg.setup_no_stop_daemon:
  159. msg('Leaving regtest daemon running')
  160. else:
  161. msg('Stopping regtest daemon')
  162. await self.rpc_call('stop')
  163. def init_daemon(self, *, reindex=False):
  164. if reindex:
  165. self.d.usr_coind_args.append('--reindex')
  166. async def start_daemon(self, *, reindex=False, silent=True):
  167. self.init_daemon(reindex=reindex)
  168. self.d.start(silent=silent)
  169. for user in ('miner', 'bob', 'alice'):
  170. msg(f'Loading {capfirst(user)}’s wallet')
  171. await self.rpc_call('loadwallet', user, start_daemon=False)
  172. async def rpc_call(self, *args, wallet=None, start_daemon=True):
  173. if start_daemon and self.d.state == 'stopped':
  174. await self.start_daemon()
  175. return await (await self.rpc).call(*args, wallet=wallet)
  176. async def start(self):
  177. if self.d.state == 'stopped':
  178. await self.start_daemon(silent=False)
  179. else:
  180. msg(f'{self.cfg.coin} regtest daemon already started')
  181. async def stop(self):
  182. if self.d.state == 'stopped':
  183. msg(f'{self.cfg.coin} regtest daemon already stopped')
  184. else:
  185. msg(f'Stopping {self.cfg.coin} regtest daemon')
  186. self.d.stop(silent=True)
  187. def state(self):
  188. msg(self.d.state)
  189. async def balances(self):
  190. bal = {}
  191. users = ('bob', 'alice')
  192. for user in users:
  193. out = await self.rpc_call('listunspent', 0, wallet=user)
  194. bal[user] = sum(self.proto.coin_amt(e['amount']) for e in out)
  195. fs = '{:<16} {:18.8f}'
  196. for user in users:
  197. msg(fs.format(user.capitalize()+"'s balance:", bal[user]))
  198. msg(fs.format('Total balance:', sum(v for k, v in bal.items())))
  199. async def send(self, addr, amt):
  200. gmsg(f'Sending {amt} miner {self.d.coin} to address {addr}')
  201. await self.rpc_call('sendtoaddress', addr, str(amt), wallet='miner')
  202. await self.generate(1)
  203. async def mempool(self):
  204. await self.cli('getrawmempool')
  205. async def cli(self, *args):
  206. ret = await self.rpc_call(*cliargs_convert(args))
  207. print(ret if isinstance(ret, str) else json.dumps(ret, cls=json_encoder, indent=4))
  208. async def wallet_cli(self, wallet, *args):
  209. ret = await self.rpc_call(*cliargs_convert(args), wallet=wallet)
  210. print(ret if isinstance(ret, str) else json.dumps(ret, cls=json_encoder, indent=4))
  211. async def cmd(self, args):
  212. ret = getattr(self, args[0])(*args[1:])
  213. return (await ret) if type(ret).__name__ == 'coroutine' else ret
  214. async def fork(self, coin): # currently disabled
  215. proto = init_proto(self.cfg, coin, testnet=False)
  216. if not [f for f in proto.forks if f[2] == proto.coin.lower() and f[3] is True]:
  217. die(1, f'Coin {proto.coin} is not a replayable fork of coin {coin}')
  218. gmsg(f'Creating fork from coin {coin} to coin {proto.coin}')
  219. source_rt = MMGenRegtest(self.cfg, coin, bdb_wallet=self.bdb_wallet)
  220. try:
  221. os.stat(source_rt.d.datadir)
  222. except:
  223. die(1, f'Source directory {source_rt.d.datadir!r} does not exist!')
  224. # stop the source daemon
  225. if source_rt.d.state != 'stopped':
  226. await source_rt.d.cli('stop')
  227. # stop our daemon
  228. if self.d.state != 'stopped':
  229. await self.rpc_call('stop')
  230. try:
  231. os.makedirs(self.d.datadir)
  232. except:
  233. pass
  234. create_data_dir(self.cfg, self.d.datadir)
  235. os.rmdir(self.d.datadir)
  236. shutil.copytree(source_rt.d.datadir, self.d.datadir, symlinks=True)
  237. await self.start_daemon(reindex=True)
  238. await self.rpc_call('stop')
  239. gmsg(f'Fork {proto.coin} successfully created')