main_xmrwallet.py 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2024 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. mmgen-xmrwallet: Perform various Monero wallet and transacting operations for
  20. addresses in an MMGen XMR key-address file
  21. """
  22. import asyncio
  23. from .cfg import gc,Config
  24. from .util import die,fmt_dict
  25. from .xmrwallet import (
  26. MoneroWalletOps,
  27. xmrwallet_uarg_info,
  28. xmrwallet_uargs,
  29. tx_priorities
  30. )
  31. opts_data = {
  32. 'sets': [
  33. ('autosign',True,'watch_only',True),
  34. ('autosign_mountpoint',bool,'autosign',True),
  35. ('autosign_mountpoint',bool,'watch_only',True),
  36. ],
  37. 'text': {
  38. 'desc': """Perform various Monero wallet and transacting operations for
  39. addresses in an MMGen XMR key-address file""",
  40. 'usage2': [
  41. '[opts] create | sync | list | view | listview | dump | restore [xmr_keyaddrfile] [wallets]',
  42. '[opts] label [xmr_keyaddrfile] LABEL_SPEC',
  43. '[opts] new [xmr_keyaddrfile] NEW_ADDRESS_SPEC',
  44. '[opts] transfer [xmr_keyaddrfile] TRANSFER_SPEC',
  45. '[opts] sweep | sweep_all [xmr_keyaddrfile] SWEEP_SPEC',
  46. '[opts] submit [TX_file]',
  47. '[opts] relay <TX_file>',
  48. '[opts] resubmit | abort (for use with --autosign only)',
  49. '[opts] txview | txlist [TX_file] ...',
  50. '[opts] export-outputs | export-outputs-sign | import-key-images [wallets]',
  51. ],
  52. 'options': """
  53. -h, --help Print this help message
  54. --, --longhelp Print help message for long options (common
  55. options)
  56. -a, --autosign Use appropriate outdir and other params for
  57. autosigning operations (implies --watch-only).
  58. When this option is in effect, filename argu-
  59. ments must be omitted, as files are located
  60. automatically.
  61. -f, --priority=N Specify an integer priority ‘N’ for inclusion
  62. of a transaction in the blockchain (higher
  63. number means higher fee). Valid parameters:
  64. {tp}. If option
  65. is omitted, the default priority will be used
  66. -F, --full-address Print addresses in full instead of truncating
  67. -m, --autosign-mountpoint=P Specify the autosign mountpoint (defaults to
  68. ‘/mnt/mmgen_autosign’, implies --autosign)
  69. -b, --rescan-blockchain Rescan the blockchain if wallet fails to sync
  70. -d, --outdir=D Save transaction files to directory 'D'
  71. instead of the working directory
  72. -D, --daemon=H:P Connect to the monerod at {D}
  73. -e, --skip-empty-accounts Skip display of empty accounts in wallets
  74. where applicable
  75. -E, --skip-empty-addresses Skip display of used empty addresses in
  76. wallets where applicable
  77. -k, --use-internal-keccak-module Force use of the internal keccak module
  78. -p, --hash-preset=P Use scrypt hash preset 'P' for password
  79. hashing (default: '{gc.dfl_hash_preset}')
  80. -P, --rescan-spent Perform a rescan of spent outputs. Used only
  81. with the ‘export-outputs-sign’ operation
  82. -R, --tx-relay-daemon=H:P[:H:P] Relay transactions via a monerod specified by
  83. {R}
  84. -r, --restore-height=H Scan from height 'H' when creating wallets.
  85. Use special value ‘current’ to create empty
  86. wallet at current blockchain height.
  87. -R, --no-relay Save transaction to file instead of relaying
  88. -s, --no-start-wallet-daemon Don’t start the wallet daemon at startup
  89. -S, --no-stop-wallet-daemon Don’t stop the wallet daemon at exit
  90. -W, --watch-only Create or operate on watch-only wallets
  91. -w, --wallet-dir=D Output or operate on wallets in directory 'D'
  92. instead of the working directory
  93. -U, --wallet-rpc-user=user Wallet RPC username (currently: {cfg.monero_wallet_rpc_user!r})
  94. -P, --wallet-rpc-password=pass Wallet RPC password (currently: [scrubbed])
  95. """,
  96. 'notes': """
  97. {xmrwallet_help}
  98. """
  99. },
  100. 'code': {
  101. 'options': lambda cfg,s: s.format(
  102. D=xmrwallet_uarg_info['daemon'].annot,
  103. R=xmrwallet_uarg_info['tx_relay_daemon'].annot,
  104. cfg=cfg,
  105. gc=gc,
  106. tp=fmt_dict(tx_priorities,fmt='equal_compact')
  107. ),
  108. 'notes': lambda help_mod,s: s.format(
  109. xmrwallet_help = help_mod('xmrwallet')
  110. )
  111. }
  112. }
  113. cfg = Config(opts_data=opts_data, init_opts={'coin':'xmr'})
  114. cmd_args = cfg._args
  115. if cmd_args and cfg.autosign and (
  116. cmd_args[0] in (
  117. MoneroWalletOps.kafile_arg_ops
  118. + ('export-outputs', 'export-outputs-sign', 'import-key-images', 'txview', 'txlist')
  119. )
  120. or len(cmd_args) == 1 and cmd_args[0] in ('submit', 'resubmit', 'abort')
  121. ):
  122. cmd_args.insert(1,None)
  123. if len(cmd_args) < 2:
  124. cfg._opts.usage()
  125. op = cmd_args.pop(0)
  126. infile = cmd_args.pop(0)
  127. wallets = spec = None
  128. if op in ('relay', 'submit', 'resubmit', 'abort'):
  129. if len(cmd_args) != 0:
  130. cfg._opts.usage()
  131. elif op in ('txview','txlist'):
  132. infile = [infile] + cmd_args
  133. elif op in ('create','sync','list','view','listview','dump','restore'): # kafile_arg_ops
  134. if len(cmd_args) > 1:
  135. cfg._opts.usage()
  136. wallets = cmd_args.pop(0) if cmd_args else None
  137. elif op in ('new', 'transfer', 'sweep', 'sweep_all', 'label'):
  138. if len(cmd_args) != 1:
  139. cfg._opts.usage()
  140. spec = cmd_args[0]
  141. elif op in ('export-outputs', 'export-outputs-sign', 'import-key-images'):
  142. if not cfg.autosign: # --autosign only for now - TODO
  143. die(f'--autosign must be used with command {op!r}')
  144. if len(cmd_args) > 1:
  145. cfg._opts.usage()
  146. wallets = cmd_args.pop(0) if cmd_args else None
  147. else:
  148. die(1,f'{op!r}: unrecognized operation')
  149. op_cls = getattr(MoneroWalletOps,op.replace('-','_'))
  150. m = op_cls(cfg, xmrwallet_uargs(infile, wallets, spec))
  151. if asyncio.run(m.main()):
  152. m.post_main_success()
  153. else:
  154. m.post_main_failure()