mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-27 15:51:02 +00:00
feat(core): Check script type match for change-outputs in Bitcoin signing.
This commit is contained in:
parent
9528e2f9eb
commit
102ab3c7d6
@ -43,8 +43,7 @@ class MatchChecker(Generic[T]):
|
||||
|
||||
def attribute_from_tx(self, txio: TxInput | TxOutput) -> T:
|
||||
# Return the attribute from the txio, which is to be used for matching.
|
||||
# If the txio is invalid for matching, then return an object which
|
||||
# evaluates as a boolean False.
|
||||
# If the txio is invalid for matching, then return None.
|
||||
raise NotImplementedError
|
||||
|
||||
def add_input(self, txi: TxInput) -> None:
|
||||
@ -56,7 +55,7 @@ class MatchChecker(Generic[T]):
|
||||
return # There was a mismatch in previous inputs.
|
||||
|
||||
added_attribute = self.attribute_from_tx(txi)
|
||||
if not added_attribute:
|
||||
if added_attribute is None:
|
||||
self.attribute = self.MISMATCH # The added input is invalid for matching.
|
||||
elif self.attribute is self.UNDEFINED:
|
||||
self.attribute = added_attribute # This is the first input.
|
||||
@ -87,7 +86,7 @@ class WalletPathChecker(MatchChecker):
|
||||
def attribute_from_tx(self, txio: TxInput | TxOutput) -> Any:
|
||||
from ..common import BIP32_WALLET_DEPTH
|
||||
|
||||
if len(txio.address_n) < BIP32_WALLET_DEPTH:
|
||||
if len(txio.address_n) <= BIP32_WALLET_DEPTH:
|
||||
return None
|
||||
return txio.address_n[:-BIP32_WALLET_DEPTH]
|
||||
|
||||
@ -99,3 +98,24 @@ class MultisigFingerprintChecker(MatchChecker):
|
||||
if not txio.multisig:
|
||||
return None
|
||||
return multisig.multisig_fingerprint(txio.multisig)
|
||||
|
||||
|
||||
class ScriptTypeChecker(MatchChecker):
|
||||
def attribute_from_tx(self, txio: TxInput | TxOutput) -> Any:
|
||||
from trezor.enums import InputScriptType
|
||||
from trezor.messages import TxOutput
|
||||
from ..common import CHANGE_OUTPUT_TO_INPUT_SCRIPT_TYPES
|
||||
|
||||
if TxOutput.is_type_of(txio):
|
||||
script_type = CHANGE_OUTPUT_TO_INPUT_SCRIPT_TYPES[txio.script_type]
|
||||
else:
|
||||
script_type = txio.script_type
|
||||
|
||||
# SPENDMULTISIG is used only for non-SegWit and is effectively the same as SPENDADDRESS.
|
||||
# For SegWit inputs and outputs multisig is indicated by the presence of the multisig
|
||||
# structure. For both SegWit and non-SegWit we can rely on MultisigFingerprintChecker to
|
||||
# check the multisig structure.
|
||||
if script_type == InputScriptType.SPENDMULTISIG:
|
||||
script_type = InputScriptType.SPENDADDRESS
|
||||
|
||||
return script_type
|
||||
|
@ -58,7 +58,11 @@ class TxInfoBase:
|
||||
def __init__(self, signer: Signer, tx: SignTx | PrevTx) -> None:
|
||||
from trezor.crypto.hashlib import sha256
|
||||
from trezor.utils import HashWriter
|
||||
from .matchcheck import MultisigFingerprintChecker, WalletPathChecker
|
||||
from .matchcheck import (
|
||||
MultisigFingerprintChecker,
|
||||
WalletPathChecker,
|
||||
ScriptTypeChecker,
|
||||
)
|
||||
|
||||
# Checksum of multisig inputs, used to validate change-output.
|
||||
self.multisig_fingerprint = MultisigFingerprintChecker()
|
||||
@ -66,6 +70,9 @@ class TxInfoBase:
|
||||
# Common prefix of input paths, used to validate change-output.
|
||||
self.wallet_path = WalletPathChecker()
|
||||
|
||||
# Common script type, used to validate change-output.
|
||||
self.script_type = ScriptTypeChecker()
|
||||
|
||||
# h_tx_check is used to make sure that the inputs and outputs streamed in
|
||||
# different steps are the same every time, e.g. the ones streamed for approval
|
||||
# in Steps 1 and 2 and the ones streamed for signing legacy inputs in Step 4.
|
||||
@ -90,6 +97,7 @@ class TxInfoBase:
|
||||
|
||||
if not common.input_is_external(txi):
|
||||
self.wallet_path.add_input(txi)
|
||||
self.script_type.add_input(txi)
|
||||
self.multisig_fingerprint.add_input(txi)
|
||||
|
||||
def add_output(self, txo: TxOutput, script_pubkey: bytes) -> None:
|
||||
@ -98,6 +106,7 @@ class TxInfoBase:
|
||||
|
||||
def check_input(self, txi: TxInput) -> None:
|
||||
self.wallet_path.check_input(txi)
|
||||
self.script_type.check_input(txi)
|
||||
self.multisig_fingerprint.check_input(txi)
|
||||
|
||||
def output_is_change(self, txo: TxOutput) -> bool:
|
||||
@ -107,6 +116,7 @@ class TxInfoBase:
|
||||
return False
|
||||
return (
|
||||
self.wallet_path.output_matches(txo)
|
||||
and self.script_type.output_matches(txo)
|
||||
and len(txo.address_n) >= common.BIP32_WALLET_DEPTH
|
||||
and txo.address_n[-2] <= _BIP32_CHANGE_CHAIN
|
||||
and txo.address_n[-1] <= _BIP32_MAX_LAST_ELEMENT
|
||||
|
@ -605,6 +605,8 @@ def test_send_multisig_3_change(client: Client):
|
||||
expected_responses = [
|
||||
request_input(0),
|
||||
request_output(0),
|
||||
messages.ButtonRequest(code=B.ConfirmOutput),
|
||||
(tt, messages.ButtonRequest(code=B.ConfirmOutput)),
|
||||
messages.ButtonRequest(code=B.SignTx),
|
||||
(tt, messages.ButtonRequest(code=B.SignTx)),
|
||||
request_input(0),
|
||||
@ -690,6 +692,8 @@ def test_send_multisig_4_change(client: Client):
|
||||
expected_responses = [
|
||||
request_input(0),
|
||||
request_output(0),
|
||||
messages.ButtonRequest(code=B.ConfirmOutput),
|
||||
(tt, messages.ButtonRequest(code=B.ConfirmOutput)),
|
||||
messages.ButtonRequest(code=B.SignTx),
|
||||
(tt, messages.ButtonRequest(code=B.SignTx)),
|
||||
request_input(0),
|
||||
|
Loading…
Reference in New Issue
Block a user