help.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  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. tool.help: Help screen routines for the 'mmgen-tool' utility
  20. """
  21. from .. import main_tool
  22. from .common import tool_cmd_base
  23. def main_help():
  24. from ..util import capfirst
  25. from ..util2 import pretty_format
  26. def do():
  27. for clsname, cmdlist in main_tool.mods.items():
  28. cls = main_tool.get_mod_cls(clsname)
  29. cls_docstr = cls.__doc__.strip()
  30. yield capfirst(cls_docstr.split('\n')[0].strip()) + ':'
  31. yield ''
  32. if '\n' in cls_docstr:
  33. for line in cls_docstr.split('\n')[2:]:
  34. yield ' ' + line.lstrip('\t')
  35. yield ''
  36. max_w = max(map(len, cmdlist))
  37. for cmdname in cmdlist:
  38. code = getattr(cls, cmdname)
  39. if code.__doc__:
  40. yield ' {:{}} - {}'.format(
  41. cmdname,
  42. max_w,
  43. pretty_format(
  44. code.__doc__.strip().split('\n')[0].strip(),
  45. width = 79-(max_w+7),
  46. pfx = ' '*(max_w+5)).lstrip()
  47. )
  48. yield ''
  49. return '\n'.join(do())
  50. def gen_tool_usage():
  51. from ..util import capfirst
  52. m1 = """
  53. GENERAL USAGE INFORMATION FOR MMGEN-TOOL COMMANDS
  54. Arguments with only type specified in square brackets are required
  55. Arguments with both type and default value specified in square brackets are
  56. optional and must be specified in the form ‘name=value’
  57. For more detailed usage information for a particular tool command, type
  58. ‘mmgen-tool help <command name>’
  59. """
  60. m2 = """
  61. To force a command to read from STDIN instead of file (for commands taking
  62. a filename as their first argument), substitute "-" for the filename.
  63. EXAMPLES:
  64. Generate a random LTC Bech32 public/private keypair:
  65. $ mmgen-tool -r0 --coin=ltc --type=bech32 randpair
  66. Generate a DASH address with compressed public key from the supplied WIF key:
  67. $ mmgen-tool --coin=dash --type=compressed wif2addr XJkVRC3eGKurc9Uzx1wfQoio3yqkmaXVqLMTa6y7s3M3jTBnmxfw
  68. Generate a well-known burn address:
  69. $ mmgen-tool hextob58chk 000000000000000000000000000000000000000000
  70. Generate a random 12-word seed phrase:
  71. $ mmgen-tool -r0 mn_rand128 fmt=bip39
  72. Same as above, but get additional entropy from user:
  73. $ mmgen-tool mn_rand128 fmt=bip39
  74. Encode bytes from a file to base 58:
  75. $ mmgen-tool bytestob58 /etc/timezone pad=20
  76. Reverse a hex string:
  77. $ mmgen-tool hexreverse "deadbeefcafe"
  78. Same as above, but supply input via STDIN:
  79. $ echo "deadbeefcafe" | mmgen-tool hexreverse -
  80. """
  81. for line in m1.lstrip().split('\n'):
  82. yield line.lstrip('\t')
  83. for clsname, cmdlist in main_tool.mods.items():
  84. cls = main_tool.get_mod_cls(clsname)
  85. cls_docstr = cls.__doc__.strip()
  86. yield ''
  87. yield ' {}:'.format(capfirst(cls_docstr.split('\n')[0].strip()))
  88. yield ''
  89. if '\n' in cls_docstr:
  90. for line in cls_docstr.split('\n')[2:]:
  91. yield ' ' + line.lstrip('\t')
  92. yield ''
  93. max_w = max(map(len, cmdlist))
  94. for cmdname in cmdlist:
  95. yield ' {a:{w}} {b}'.format(
  96. a = cmdname,
  97. b = main_tool.create_call_sig(cmdname, cls, as_string=True),
  98. w = max_w)
  99. yield ''
  100. for line in m2.rstrip().split('\n'):
  101. yield line.lstrip('\t')
  102. def gen_tool_cmd_usage(mod, cmdname):
  103. from ..cfg import gc
  104. from ..util import capfirst
  105. cls = main_tool.get_mod_cls(mod)
  106. docstr = getattr(cls, cmdname).__doc__.strip()
  107. args, kwargs, kwargs_types, _, ann = main_tool.create_call_sig(cmdname, cls)
  108. ARGS = 'ARG' if len(args) == 1 else 'ARGS' if args else ''
  109. KWARGS = 'KEYWORD ARG' if len(kwargs) == 1 else 'KEYWORD ARGS' if kwargs else ''
  110. yield capfirst(docstr.split('\n')[0].strip())
  111. yield ''
  112. yield 'USAGE: {b} [OPTS] {c}{d}{e}'.format(
  113. b = gc.prog_name,
  114. c = cmdname,
  115. d = f' {ARGS}' if ARGS else '',
  116. e = f' [{KWARGS}]' if KWARGS else '')
  117. if args:
  118. max_w = max(len(k[0]) for k in args)
  119. yield ''
  120. yield f'Required {ARGS} (type shown in square brackets):'
  121. yield ''
  122. for argname, argtype in args:
  123. have_sstr = ann.get(argname) == 'sstr'
  124. yield ' {a:{w}} [{b}]{c}{d}'.format(
  125. a = argname,
  126. b = argtype,
  127. c = " (use '-' to read from STDIN)" if have_sstr else '',
  128. d = ' ' + ann[argname] if isinstance(ann.get(argname), str) and not have_sstr else '',
  129. w = max_w)
  130. if kwargs:
  131. max_w = max(len(k) for k in kwargs)
  132. max_w2 = max(len(kwargs_types[k].__name__) + len(repr(kwargs[k])) for k in kwargs) + 3
  133. yield ''
  134. yield f'Optional {KWARGS} (type and default value shown in square brackets):'
  135. yield ''
  136. for argname in kwargs:
  137. yield ' {a:{w}} {b:{w2}} {c}'.format(
  138. a = argname,
  139. b = '[{}={}]'.format(kwargs_types[argname].__name__, repr(kwargs[argname])),
  140. c = capfirst(ann[argname]) if isinstance(ann.get(argname), str) else '',
  141. w = max_w,
  142. w2 = max_w2).rstrip()
  143. if '\n' in docstr:
  144. for line in docstr.split('\n')[1:]:
  145. yield line.lstrip('\t')
  146. def usage(cmdname=None, *, exit_val=1):
  147. from ..util import Msg, die
  148. if cmdname:
  149. for mod, cmdlist in main_tool.mods.items():
  150. if cmdname in cmdlist:
  151. Msg('\n'.join(gen_tool_cmd_usage(mod, cmdname)))
  152. break
  153. else:
  154. die(1, f'{cmdname!r}: no such tool command')
  155. else:
  156. from ..ui import do_pager
  157. do_pager('\n'.join(gen_tool_usage()) + '\n')
  158. import sys
  159. sys.exit(exit_val)
  160. class tool_cmd(tool_cmd_base):
  161. "help/usage commands"
  162. def help(self, command_name=''):
  163. "display usage information for a single command or all commands"
  164. usage(command_name, exit_val=0)
  165. def usage(self, command_name=''):
  166. "display usage information for a single command or all commands"
  167. usage(command_name, exit_val=0)