ct_help.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  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.ct_help: helpscreen test group for the cmdtest.py test suite
  12. """
  13. import sys, os, time
  14. from mmgen.util import ymsg
  15. from mmgen.cfg import gc
  16. from .ct_base import CmdTestBase
  17. class CmdTestHelp(CmdTestBase):
  18. 'help, info and usage screens'
  19. networks = ('btc', 'ltc', 'bch', 'eth', 'xmr', 'doge')
  20. passthru_opts = ('daemon_data_dir', 'rpc_port', 'coin', 'testnet')
  21. cmd_group = (
  22. ('usage1', (1, 'usage message (via --usage)', [])),
  23. ('usage2', (1, 'usage message (via --usage, with --coin)', [])),
  24. ('usage3', (1, 'usage message (via bad invocation)', [])),
  25. ('usage4', (1, 'usage message (via bad invocation, with --coin)', [])),
  26. ('version', (1, 'version message', [])),
  27. ('license', (1, 'license message', [])),
  28. ('helpscreens', (1, 'help screens', [])),
  29. ('longhelpscreens', (1, 'help screens (--longhelp)', [])),
  30. ('show_hash_presets', (1, 'info screen (--show-hash-presets)', [])),
  31. ('tool_help', (1, '‘mmgen-tool’ usage screen', [])),
  32. ('tool_cmd_usage', (1, '‘mmgen-tool’ usage screen', [])),
  33. ('test_help', (1, '‘cmdtest.py’ help screens', [])),
  34. ('tooltest_help', (1, '‘tooltest.py’ help screens', [])),
  35. )
  36. def usage1(self):
  37. return self._usage('walletgen', ['--usage'], True, False, 0)
  38. def usage2(self):
  39. return self._usage('tool' if self.coin == 'xmr' else 'txcreate', ['--usage'], True, True, 0)
  40. def usage3(self):
  41. return self._usage('walletgen', ['foo'], True, False, 1)
  42. def usage4(self):
  43. return self._usage('tool' if self.coin == 'xmr' else 'txcreate', [], True, True, 1)
  44. def _usage(self, cmd_arg, args, no_passthru_opts, add_coin_opt, exit_val):
  45. if cmd := (None if self._gen_skiplist(cmd_arg) else cmd_arg):
  46. t = self.spawn(
  47. f'mmgen-{cmd}',
  48. ([f'--coin={self.coin}'] if add_coin_opt else []) + args,
  49. exit_val = exit_val,
  50. no_passthru_opts = no_passthru_opts)
  51. t.expect(f'USAGE: mmgen-{cmd}')
  52. return t
  53. return 'skip'
  54. def version(self):
  55. t = self.spawn('mmgen-tool', ['--version'], exit_val=0)
  56. t.expect('MMGEN-TOOL version')
  57. return t
  58. def license(self):
  59. t = self.spawn(
  60. 'mmgen-walletconv',
  61. ['--stdout', '--in-fmt=hex', '--out-fmt=hex'],
  62. env = {'MMGEN_NO_LICENSE':''},
  63. no_passthru_opts = True)
  64. t.expect('to continue: ', 'w')
  65. t.expect('TERMS AND CONDITIONS') # start of GPL text
  66. if self.cfg.pexpect_spawn:
  67. t.send('G')
  68. t.expect('return for a fee.') # end of GPL text
  69. if self.cfg.pexpect_spawn:
  70. t.send('q')
  71. t.expect('to continue: ', 'c')
  72. t.expect('data: ', 'beadcafe'*4 + '\n')
  73. t.expect('to confirm: ', 'YES\n')
  74. return t
  75. def spawn_chk_expect(self, *args, **kwargs):
  76. expect = kwargs.pop('expect')
  77. t = self.spawn(*args, **kwargs)
  78. t.expect(expect)
  79. if t.pexpect_spawn:
  80. time.sleep(0.4)
  81. t.send('q')
  82. t.read()
  83. t.ok()
  84. t.skip_ok = True
  85. return t
  86. def _gen_skiplist(self, scripts):
  87. def gen(scripts):
  88. if isinstance(scripts, str):
  89. scripts = [scripts]
  90. for script in scripts:
  91. d = gc.cmd_caps_data[script]
  92. if sys.platform == 'win32' and 'w' not in d.platforms:
  93. yield script
  94. elif not (d.use_coin_opt or self.proto.coin.lower() == 'btc'):
  95. yield script
  96. else:
  97. for cap in d.caps:
  98. if cap not in self.proto.mmcaps:
  99. yield script
  100. break
  101. return set(gen(scripts))
  102. def helpscreens(self, arg='--help', scripts=(), expect='USAGE:.*OPTIONS:', pager=True):
  103. scripts = list(scripts or gc.cmd_caps_data)
  104. cmdlist = sorted(set(scripts) - self._gen_skiplist(scripts))
  105. for cmdname in cmdlist:
  106. cmd_caps = gc.cmd_caps_data[cmdname]
  107. assert cmd_caps, cmdname
  108. t = self.spawn(
  109. f'mmgen-{cmdname}',
  110. [arg],
  111. extra_desc = f'(mmgen-{cmdname})',
  112. no_passthru_opts = not cmd_caps.use_coin_opt)
  113. t.expect(expect, regex=True)
  114. if pager and t.pexpect_spawn:
  115. time.sleep(0.2)
  116. t.send('q')
  117. t.read()
  118. t.ok()
  119. t.skip_ok = True
  120. return 'silent'
  121. def longhelpscreens(self):
  122. return self.helpscreens(arg='--longhelp', expect='USAGE:.*GLOBAL OPTIONS:')
  123. def show_hash_presets(self):
  124. return self.helpscreens(
  125. arg = '--show-hash-presets',
  126. scripts = (
  127. 'walletgen', 'walletconv', 'walletchk', 'passchg', 'subwalletgen',
  128. 'addrgen', 'keygen', 'passgen',
  129. 'txdo', 'swaptxdo', 'txsign', 'txbump'),
  130. expect = 'Available parameters.*Preset',
  131. pager = False)
  132. def tool_help(self):
  133. if os.getenv('PYTHONOPTIMIZE') == '2':
  134. ymsg('Skipping tool help with PYTHONOPTIMIZE=2 (no docstrings)')
  135. return 'skip'
  136. for arg in (
  137. 'help',
  138. 'usage',
  139. ):
  140. t = self.spawn_chk_expect(
  141. 'mmgen-tool',
  142. [arg],
  143. extra_desc = f'(mmgen-tool {arg})',
  144. expect = 'GENERAL USAGE')
  145. return t
  146. def tool_cmd_usage(self):
  147. if os.getenv('PYTHONOPTIMIZE') == '2':
  148. ymsg('Skipping tool cmd usage with PYTHONOPTIMIZE=2 (no docstrings)')
  149. return 'skip'
  150. from mmgen.main_tool import mods
  151. for cmdlist in mods.values():
  152. for cmd in cmdlist:
  153. t = self.spawn_chk('mmgen-tool', ['help', cmd], extra_desc=f'({cmd})')
  154. return t
  155. def test_help(self):
  156. for arg, expect in (
  157. ('--help', 'USAGE'),
  158. ('--list-cmds', 'AVAILABLE COMMANDS'),
  159. ('--list-cmd-groups', 'AVAILABLE COMMAND GROUPS')
  160. ):
  161. t = self.spawn_chk_expect(
  162. 'cmdtest.py',
  163. [arg],
  164. cmd_dir = 'test',
  165. extra_desc = f'(cmdtest.py {arg})',
  166. expect = expect)
  167. return t
  168. def tooltest_help(self):
  169. for arg, expect in (
  170. ('--list-cmds', 'Available commands'),
  171. ('--testing-status', 'Testing status')
  172. ):
  173. t = self.spawn_chk_expect(
  174. 'tooltest.py',
  175. [arg],
  176. cmd_dir = 'test',
  177. extra_desc = f'(tooltest.py {arg})',
  178. expect = expect)
  179. return t