bump.py 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  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. tx.bump: transaction bump class
  12. """
  13. from .new_swap import NewSwap
  14. from .completed import Completed
  15. from ..util import msg, ymsg, is_int, die
  16. from ..color import pink
  17. class Bump(Completed, NewSwap):
  18. desc = 'fee-bumped transaction'
  19. ext = 'rawtx'
  20. bump_output_idx = None
  21. is_bump = True
  22. swap_attrs = (
  23. 'is_swap',
  24. 'swap_proto',
  25. 'swap_quote_expiry',
  26. 'swap_recv_addr_mmid')
  27. def __init__(self, *, check_sent, new_outputs, **kwargs):
  28. super().__init__(**kwargs)
  29. self.new_outputs = new_outputs
  30. self.orig_rel_fee = self.get_orig_rel_fee()
  31. if new_outputs:
  32. from .base import Base
  33. if self.is_swap:
  34. for attr in self.swap_attrs:
  35. setattr(self, attr, getattr(Base, attr))
  36. self.outputs = self.OutputList(self)
  37. self.cfg = kwargs['cfg'] # must use current cfg opts, not those from orig_tx
  38. if not self.is_replaceable():
  39. die(1, f'Transaction {self.txid} is not replaceable')
  40. # If sending, require original tx to be sent
  41. if check_sent and not self.coin_txid:
  42. die(1, f'Transaction {self.txid!r} was not broadcast to the network')
  43. self.coin_txid = ''
  44. self.sent_timestamp = None
  45. async def get_inputs(self, outputs_sum):
  46. return True
  47. def check_bumped_fee_ok(self, abs_fee):
  48. orig = int(self.orig_rel_fee)
  49. new = int(self.fee_abs2rel(abs_fee))
  50. if new <= orig:
  51. ymsg('New fee ({b} {d}) <= original fee ({a} {d}). Please choose a higher fee'.format(
  52. a=orig, b=new, d=self.rel_fee_disp))
  53. return False
  54. return True
  55. async def create_feebump(self, silent):
  56. from ..rpc import rpc_init
  57. self.rpc = await rpc_init(self.cfg, self.proto)
  58. msg('Creating replacement transaction')
  59. self.check_sufficient_funds_for_bump()
  60. output_idx = self.choose_output()
  61. if not silent:
  62. msg('Minimum fee for new transaction: {} {} ({} {})'.format(
  63. self.min_fee.hl(),
  64. self.proto.coin,
  65. pink(self.fee_abs2rel(self.min_fee)),
  66. self.rel_fee_disp))
  67. if self.is_swap:
  68. self.recv_proto = self.check_swap_memo().proto
  69. self.process_swap_options()
  70. fee_hint = await self.update_vault_output(self.send_amt)
  71. else:
  72. fee_hint = None
  73. self.usr_fee = self.get_usr_fee_interactive(
  74. fee = self.cfg.fee or fee_hint,
  75. desc = 'User-selected' if self.cfg.fee else 'Recommended' if fee_hint else None)
  76. self.bump_fee(output_idx, self.usr_fee)
  77. assert self.fee <= self.proto.max_tx_fee
  78. if not self.cfg.yes:
  79. self.add_comment() # edits an existing comment
  80. await self.create_serialized()
  81. self.add_timestamp()
  82. self.add_blockcount()
  83. self.cfg._util.qmsg('Fee successfully increased')
  84. def check_sufficient_funds_for_bump(self):
  85. if not [o.amt for o in self.outputs if o.amt >= self.min_fee]:
  86. die(1,
  87. 'Transaction cannot be bumped.\n' +
  88. f'All outputs contain less than the minimum fee ({self.min_fee} {self.coin})')
  89. def choose_output(self):
  90. def check_sufficient_funds(o_amt):
  91. if o_amt < self.min_fee:
  92. msg(f'Minimum fee ({self.min_fee} {self.coin}) is greater than output amount ({o_amt} {self.coin})')
  93. return False
  94. return True
  95. if len(self.nondata_outputs) == 1:
  96. if check_sufficient_funds(self.nondata_outputs[0].amt):
  97. self.bump_output_idx = 0
  98. return 0
  99. else:
  100. die(1, 'Insufficient funds to bump transaction')
  101. init_reply = self.cfg.output_to_reduce
  102. chg_idx = self.chg_idx
  103. while True:
  104. if init_reply is None:
  105. from ..ui import line_input
  106. m = 'Choose an output to deduct the fee from (Hit ENTER for the change output): '
  107. reply = line_input(self.cfg, m) or 'c'
  108. else:
  109. reply, init_reply = init_reply, None
  110. if chg_idx is None and not is_int(reply):
  111. msg('Output must be an integer')
  112. elif chg_idx is not None and not is_int(reply) and reply != 'c':
  113. msg("Output must be an integer, or 'c' for the change output")
  114. else:
  115. idx = chg_idx if reply == 'c' else (int(reply) - 1)
  116. if idx < 0 or idx >= len(self.outputs):
  117. msg(f'Output must be in the range 1-{len(self.outputs)}')
  118. else:
  119. o_amt = self.outputs[idx].amt
  120. cm = ' (change output)' if chg_idx == idx else ''
  121. prompt = f'Fee will be deducted from output {idx+1}{cm} ({o_amt} {self.coin})'
  122. if check_sufficient_funds(o_amt):
  123. if self.cfg.yes:
  124. msg(prompt)
  125. else:
  126. from ..ui import keypress_confirm
  127. if not keypress_confirm(self.cfg, prompt+'. OK?', default_yes=True):
  128. continue
  129. self.bump_output_idx = idx
  130. return idx