1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-25 23:01:02 +00:00

feat(core): show fee rate when replacing transaction

This commit is contained in:
grdddj 2022-08-29 15:51:48 +02:00 committed by matejcik
parent d5b0650cc2
commit c53177fe89
7 changed files with 47 additions and 28 deletions

View File

@ -0,0 +1 @@
Show fee rate when replacing transaction

View File

@ -249,6 +249,7 @@ class BasicApprover(Approver):
total = self.total_in - self.change_out total = self.total_in - self.change_out
spending = total - self.external_in spending = total - self.external_in
tx_size_vB = self.weight.get_virtual_size() tx_size_vB = self.weight.get_virtual_size()
fee_rate = fee / tx_size_vB
# fee_threshold = (coin.maxfee per byte * tx size) # fee_threshold = (coin.maxfee per byte * tx size)
fee_threshold = (self.coin.maxfee_kb / 1000) * tx_size_vB fee_threshold = (self.coin.maxfee_kb / 1000) * tx_size_vB
@ -295,14 +296,14 @@ class BasicApprover(Approver):
# coming entirely from the user's own funds and from decreases of external outputs. # coming entirely from the user's own funds and from decreases of external outputs.
# We consider the decreases as belonging to the user. # We consider the decreases as belonging to the user.
await helpers.confirm_modify_fee( await helpers.confirm_modify_fee(
fee - orig_fee, fee, self.coin, self.amount_unit fee - orig_fee, fee, fee_rate, self.coin, self.amount_unit
) )
elif spending > orig_spending: elif spending > orig_spending:
# PayJoin and user is spending more: Show the increase in the user's contribution # PayJoin and user is spending more: Show the increase in the user's contribution
# to the fee, ignoring any contribution from external inputs. Decreasing of # to the fee, ignoring any contribution from external inputs. Decreasing of
# external outputs is not allowed in PayJoin, so there is no need to handle those. # external outputs is not allowed in PayJoin, so there is no need to handle those.
await helpers.confirm_modify_fee( await helpers.confirm_modify_fee(
spending - orig_spending, fee, self.coin, self.amount_unit spending - orig_spending, fee, fee_rate, self.coin, self.amount_unit
) )
else: else:
# PayJoin and user is not spending more: When new external inputs are involved and # PayJoin and user is not spending more: When new external inputs are involved and
@ -317,7 +318,6 @@ class BasicApprover(Approver):
) )
if not self.external_in: if not self.external_in:
fee_rate = fee / tx_size_vB
await helpers.confirm_total( await helpers.confirm_total(
total, fee, fee_rate, self.coin, self.amount_unit total, fee, fee_rate, self.coin, self.amount_unit
) )

View File

@ -116,17 +116,24 @@ class UiConfirmModifyFee(UiConfirm):
self, self,
user_fee_change: int, user_fee_change: int,
total_fee_new: int, total_fee_new: int,
fee_rate: float,
coin: CoinInfo, coin: CoinInfo,
amount_unit: AmountUnit, amount_unit: AmountUnit,
): ):
self.user_fee_change = user_fee_change self.user_fee_change = user_fee_change
self.total_fee_new = total_fee_new self.total_fee_new = total_fee_new
self.fee_rate = fee_rate
self.coin = coin self.coin = coin
self.amount_unit = amount_unit self.amount_unit = amount_unit
def confirm_dialog(self, ctx: wire.Context) -> Awaitable[Any]: def confirm_dialog(self, ctx: wire.Context) -> Awaitable[Any]:
return layout.confirm_modify_fee( return layout.confirm_modify_fee(
ctx, self.user_fee_change, self.total_fee_new, self.coin, self.amount_unit ctx,
self.user_fee_change,
self.total_fee_new,
self.fee_rate,
self.coin,
self.amount_unit,
) )
@ -230,8 +237,12 @@ def confirm_modify_output(txo: TxOutput, orig_txo: TxOutput, coin: CoinInfo, amo
return (yield UiConfirmModifyOutput(txo, orig_txo, coin, amount_unit)) return (yield UiConfirmModifyOutput(txo, orig_txo, coin, amount_unit))
def confirm_modify_fee(user_fee_change: int, total_fee_new: int, coin: CoinInfo, amount_unit: AmountUnit) -> Awaitable[Any]: # type: ignore [awaitable-is-generator] def confirm_modify_fee(user_fee_change: int, total_fee_new: int, fee_rate: float, coin: CoinInfo, amount_unit: AmountUnit) -> Awaitable[Any]: # type: ignore [awaitable-is-generator]
return (yield UiConfirmModifyFee(user_fee_change, total_fee_new, coin, amount_unit)) return (
yield UiConfirmModifyFee(
user_fee_change, total_fee_new, fee_rate, coin, amount_unit
)
)
def confirm_total(spending: int, fee: int, fee_rate: float, coin: CoinInfo, amount_unit: AmountUnit) -> Awaitable[None]: # type: ignore [awaitable-is-generator] def confirm_total(spending: int, fee: int, fee_rate: float, coin: CoinInfo, amount_unit: AmountUnit) -> Awaitable[None]: # type: ignore [awaitable-is-generator]

View File

@ -154,6 +154,7 @@ async def confirm_modify_fee(
ctx: wire.Context, ctx: wire.Context,
user_fee_change: int, user_fee_change: int,
total_fee_new: int, total_fee_new: int,
fee_rate: float,
coin: CoinInfo, coin: CoinInfo,
amount_unit: AmountUnit, amount_unit: AmountUnit,
) -> None: ) -> None:
@ -162,6 +163,7 @@ async def confirm_modify_fee(
user_fee_change, user_fee_change,
format_coin_amount(abs(user_fee_change), coin, amount_unit), format_coin_amount(abs(user_fee_change), coin, amount_unit),
format_coin_amount(total_fee_new, coin, amount_unit), format_coin_amount(total_fee_new, coin, amount_unit),
fee_rate_amount=_get_fee_rate_str(fee_rate, coin),
) )
@ -179,6 +181,18 @@ async def confirm_joint_total(
) )
def _get_fee_rate_str(fee_rate: float, coin: CoinInfo) -> str | None:
if fee_rate >= 0:
# Use format_amount to get correct thousands separator -- micropython's built-in
# formatting doesn't add thousands sep to floating point numbers.
# We multiply by 10 to get a fixed-point integer with one decimal place,
# and add 0.5 to round to the nearest integer.
fee_rate_formatted = format_amount(int(fee_rate * 10 + 0.5), 1)
return f"({fee_rate_formatted} sat/{'v' if coin.segwit else ''}B)"
else:
return None
async def confirm_total( async def confirm_total(
ctx: wire.Context, ctx: wire.Context,
spending: int, spending: int,
@ -187,21 +201,11 @@ async def confirm_total(
coin: CoinInfo, coin: CoinInfo,
amount_unit: AmountUnit, amount_unit: AmountUnit,
) -> None: ) -> None:
fee_rate_str: str | None = None
if fee_rate >= 0:
# Use format_amount to get correct thousands separator -- micropython's built-in
# formatting doesn't add thousands sep to floating point numbers.
# We multiply by 10 to get a fixed-point integer with one decimal place,
# and add 0.5 to round to the nearest integer.
fee_rate_formatted = format_amount(int(fee_rate * 10 + 0.5), 1)
fee_rate_str = f"({fee_rate_formatted} sat/{'v' if coin.segwit else ''}B)"
await layouts.confirm_total( await layouts.confirm_total(
ctx, ctx,
total_amount=format_coin_amount(spending, coin, amount_unit), total_amount=format_coin_amount(spending, coin, amount_unit),
fee_amount=format_coin_amount(fee, coin, amount_unit), fee_amount=format_coin_amount(fee, coin, amount_unit),
fee_rate_amount=fee_rate_str, fee_rate_amount=_get_fee_rate_str(fee_rate, coin),
) )

View File

@ -980,6 +980,7 @@ async def confirm_modify_fee(
sign: int, sign: int,
user_fee_change: str, user_fee_change: str,
total_fee_new: str, total_fee_new: str,
fee_rate_amount: str | None = None,
) -> None: ) -> None:
text = Text("Modify fee", ui.ICON_SEND, ui.GREEN, new_lines=False) text = Text("Modify fee", ui.ICON_SEND, ui.GREEN, new_lines=False)
if sign == 0: if sign == 0:
@ -991,9 +992,10 @@ async def confirm_modify_fee(
text.normal("Increase your fee by:\n") text.normal("Increase your fee by:\n")
text.bold(user_fee_change) text.bold(user_fee_change)
text.br() text.br()
text.br_half()
text.normal("Transaction fee:\n") text.normal("Transaction fee:\n")
text.bold(total_fee_new) text.bold(total_fee_new)
if fee_rate_amount is not None:
text.normal("\n" + fee_rate_amount)
await raise_if_cancelled( await raise_if_cancelled(
interact(ctx, HoldToConfirm(text), "modify_fee", ButtonRequestType.SignTx) interact(ctx, HoldToConfirm(text), "modify_fee", ButtonRequestType.SignTx)
) )

View File

@ -743,6 +743,7 @@ async def confirm_modify_fee(
sign: int, sign: int,
user_fee_change: str, user_fee_change: str,
total_fee_new: str, total_fee_new: str,
fee_rate_amount: str | None = None,
) -> None: ) -> None:
result = await interact( result = await interact(
ctx, ctx,

View File

@ -947,20 +947,20 @@
"TT_bitcoin-test_signtx_replacement.py::test_attack_fake_int_input_amount": "1c100ce4b7c1e47e72428f390de0846c1ff933e9f07894872644a369a9422738", "TT_bitcoin-test_signtx_replacement.py::test_attack_fake_int_input_amount": "1c100ce4b7c1e47e72428f390de0846c1ff933e9f07894872644a369a9422738",
"TT_bitcoin-test_signtx_replacement.py::test_attack_false_internal": "1c100ce4b7c1e47e72428f390de0846c1ff933e9f07894872644a369a9422738", "TT_bitcoin-test_signtx_replacement.py::test_attack_false_internal": "1c100ce4b7c1e47e72428f390de0846c1ff933e9f07894872644a369a9422738",
"TT_bitcoin-test_signtx_replacement.py::test_attack_steal_change": "04ba3e05862eef616957f9f0c67f7a1d841a3bc7278c5fd999dfba4ebeee5314", "TT_bitcoin-test_signtx_replacement.py::test_attack_steal_change": "04ba3e05862eef616957f9f0c67f7a1d841a3bc7278c5fd999dfba4ebeee5314",
"TT_bitcoin-test_signtx_replacement.py::test_p2pkh_fee_bump": "9521ab8dcf72d514103ec16410e3ad7d8e384084c3a645178ab191315911cd13", "TT_bitcoin-test_signtx_replacement.py::test_p2pkh_fee_bump": "46f53aa43a1ab3b040716ad7090f602695b77e8250e65b7024d2ace11ffdb37b",
"TT_bitcoin-test_signtx_replacement.py::test_p2tr_fee_bump": "d2adcdaf62181c703628e355ba84ff979a81426429ee487c0059a9f161fbeb25", "TT_bitcoin-test_signtx_replacement.py::test_p2tr_fee_bump": "cd02aade75d2b644867fa98d07fdc6f1b22372286718980c5aac8970c4594168",
"TT_bitcoin-test_signtx_replacement.py::test_p2tr_invalid_signature": "d2adcdaf62181c703628e355ba84ff979a81426429ee487c0059a9f161fbeb25", "TT_bitcoin-test_signtx_replacement.py::test_p2tr_invalid_signature": "cd02aade75d2b644867fa98d07fdc6f1b22372286718980c5aac8970c4594168",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_finalize": "66d0c98fe05fab4e77a516434883f81fa9516464373374097e91a8e3c278a6b3", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_finalize": "90948701923e3f288fdf04a27198644f042fed8993d683dbdd577e676785e5d4",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_in_p2sh_fee_bump_from_external": "d3d97e310fb2d1d29b351bdc35b359aab75b4bba7afd257eb85100031edaacbf", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_in_p2sh_fee_bump_from_external": "9c625a9e2d1895d410e4c1c6a968b8b53deee3700e5b1902f491d1c7401de122",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_in_p2sh_remove_change": "ae9e7bc83227cf6187fb5c689515b66da7bf8fcc58a43128389f6eb7ab35240d", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_in_p2sh_remove_change": "6c4f7638aa2ebf7105b3bb263e2c68c9771581738d6e2aa137df89dfb6d383ee",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_invalid_signature": "66d0c98fe05fab4e77a516434883f81fa9516464373374097e91a8e3c278a6b3", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_invalid_signature": "90948701923e3f288fdf04a27198644f042fed8993d683dbdd577e676785e5d4",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_op_return_fee_bump": "f0d4e746eb927051ae7144fd9cbc63dc26aa61fe79e97eb31e74677ecf65eda9", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_op_return_fee_bump": "a65bb3df6ba4f68c6060efe5eef5fd1e66d6e9457becf6c4e8112fec028d153b",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909659-90000-02483045022100aa1b91-c9b963ae": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909659-90000-02483045022100aa1b91-c9b963ae": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909718-90000-024730440220753f5304-ecb983d1": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909718-90000-024730440220753f5304-ecb983d1": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909800-89859-0248304502210097a42b-7a89e474": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909800-89859-0248304502210097a42b-7a89e474": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909859-89800-02483045022100af3a87-80428fad": "f5be02a50a1876ac0478e37e41bca38c5feb569613dc5b105b95c6bb4763514a", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909859-89800-02483045022100af3a87-80428fad": "546caace6b10e9c9993b12033a8df2627a85d183c1eff2e090c5ff280dd25665",
"TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909859-89859-02483045022100eb74ab-881c7bef": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57", "TT_bitcoin-test_signtx_replacement.py::test_p2wpkh_payjoin[19909859-89859-02483045022100eb74ab-881c7bef": "da3ec44de0435cca3828752a0ba44483b2f087d1ae02c99530be4bcd01b80e57",
"TT_bitcoin-test_signtx_replacement.py::test_tx_meld": "1a5221c169069689cd038a1b493932e42f0c740efad52e3c77bb636786af594e", "TT_bitcoin-test_signtx_replacement.py::test_tx_meld": "a08f8f4b137150d03031a16a615f62ef8134f5599f43e5cd2057aa1a1d144b79",
"TT_bitcoin-test_signtx_segwit.py::test_attack_change_input_address": "f3398d5814422383b9a6fa8cce53948af05e985616c20e7f3aed649a9d3fe1a6", "TT_bitcoin-test_signtx_segwit.py::test_attack_change_input_address": "f3398d5814422383b9a6fa8cce53948af05e985616c20e7f3aed649a9d3fe1a6",
"TT_bitcoin-test_signtx_segwit.py::test_attack_mixed_inputs": "a06e89d99c1e218fba7ceff794cb9dab98bbae666b397996e067fe9f0bd8a009", "TT_bitcoin-test_signtx_segwit.py::test_attack_mixed_inputs": "a06e89d99c1e218fba7ceff794cb9dab98bbae666b397996e067fe9f0bd8a009",
"TT_bitcoin-test_signtx_segwit.py::test_send_multisig_1": "cfd3b3e882067b1e42a36623da3523829bf579399b14f8711e09f7e89aa82d65", "TT_bitcoin-test_signtx_segwit.py::test_send_multisig_1": "cfd3b3e882067b1e42a36623da3523829bf579399b14f8711e09f7e89aa82d65",