mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-17 10:51:00 +00:00
feat(tools): new snippets folder, sign_tx.py and unify_test_files.py scripts
This commit is contained in:
parent
3536d86fa9
commit
8d7b379349
16
tools/snippets/README.md
Normal file
16
tools/snippets/README.md
Normal file
@ -0,0 +1,16 @@
|
||||
# Snippets
|
||||
|
||||
|
||||
This folder is storing non-essential scripts (snippets, gists), which may however come in handy in a year or two.
|
||||
|
||||
These scripts do not need to have a high standard, but each of those should have a short description of what is it doing.
|
||||
|
||||
## sign_tx.py - [file](./sign_tx.py)
|
||||
- signing of BTC transactions that can be spent with currently connected device
|
||||
- need to specify the input UTXOs and the output address(es)
|
||||
- generates a serialized transaction that can be then announced to the network
|
||||
|
||||
## unify_test_files.py - [file](./unify_test_files.py)
|
||||
- enforcing some readability practices in test files and possibly adding information about paths and addresses used
|
||||
- need to specify all the files to be modified and optionally the path-to-address translation file
|
||||
- rewrites the files in place, or just prints the intended changes when run with `-c`
|
125
tools/snippets/sign_tx.py
Normal file
125
tools/snippets/sign_tx.py
Normal file
@ -0,0 +1,125 @@
|
||||
"""
|
||||
Script for quick (and repeatable) creation and signing of
|
||||
specific inputs and outputs inspired by BTC device tests.
|
||||
|
||||
INPUTS and OUTPUTS lists are the only things needed to be modified.
|
||||
|
||||
Similar to tools/build_tx.py, but more suitable for bigger/autogenerated transactions.
|
||||
|
||||
The serialized transaction can then be announced to network at https://tbtc1.trezor.io/sendtx
|
||||
It could be useful to inspect the transaction details at https://live.blockcypher.com/btc/decodetx/
|
||||
|
||||
Usage:
|
||||
- modify INPUTS and OUTPUTS lists to suit the needs
|
||||
- call the script with possible flags - see `python sign_tx.py --help`
|
||||
"""
|
||||
|
||||
import argparse
|
||||
from decimal import Decimal
|
||||
from typing import Dict, List
|
||||
|
||||
import requests
|
||||
|
||||
from trezorlib import btc, messages
|
||||
from trezorlib.client import get_default_client
|
||||
from trezorlib.debuglink import TrezorClientDebugLink
|
||||
from trezorlib.tools import parse_path
|
||||
from trezorlib.transport import enumerate_devices
|
||||
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument(
|
||||
"--autoconfirm",
|
||||
action="store_true",
|
||||
help="Automatically confirm everything on the device.",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--testnet",
|
||||
action="store_true",
|
||||
help="Use BTC testnet instead of mainnet.",
|
||||
)
|
||||
args = parser.parse_args()
|
||||
|
||||
# Can choose autoconfirm everything on the device (in the device-tests-style)
|
||||
# (Suitable for long/repetitive transactions)
|
||||
if args.autoconfirm:
|
||||
print("Autoconfirming everything on the device.")
|
||||
for device in enumerate_devices():
|
||||
try:
|
||||
CLIENT = TrezorClientDebugLink(device, auto_interact=True)
|
||||
break
|
||||
except Exception:
|
||||
pass
|
||||
else:
|
||||
raise RuntimeError("Could not find device")
|
||||
else:
|
||||
CLIENT = get_default_client()
|
||||
# Choosing between Mainnet and Testnet
|
||||
if args.testnet:
|
||||
COIN = "Testnet"
|
||||
URL = "https://tbtc1.trezor.io/api/tx-specific"
|
||||
else:
|
||||
COIN = "Bitcoin"
|
||||
URL = "https://btc1.trezor.io/api/tx-specific"
|
||||
print(f"Operating on {COIN} at {URL}")
|
||||
|
||||
# Specific example of generating and signing a transaction with 255 outputs
|
||||
# (Could be tried on `all all all...` seed on testnet)
|
||||
# (--autoconfirm really helps here)
|
||||
INPUTS = [
|
||||
messages.TxInputType(
|
||||
address_n=parse_path("44h/1h/0h/0/0"), # mvbu1Gdy8SUjTenqerxUaZyYjmveZvt33q
|
||||
amount=1_827_955,
|
||||
prev_hash=bytes.fromhex(
|
||||
"58d56a5d1325cf83543ee4c87fd73a784e4ba1499ced574be359fa2bdcb9ac8e"
|
||||
),
|
||||
prev_index=1,
|
||||
),
|
||||
]
|
||||
count = 255
|
||||
OUTPUTS = [
|
||||
messages.TxOutputType(
|
||||
address="momtnzR3XqXgDSsFmd8gkGxUiHZLde3RmA", # "44h/1h/0h/0/3"
|
||||
amount=(1_827_955 - 10_000) // count,
|
||||
script_type=messages.OutputScriptType.PAYTOADDRESS,
|
||||
)
|
||||
for _ in range(count)
|
||||
]
|
||||
|
||||
|
||||
def get_tx_info(tx_id: str) -> messages.TransactionType:
|
||||
"""Fetch basic transaction info for the signing."""
|
||||
tx_url = f"{URL}/{tx_id}"
|
||||
tx_src = requests.get(tx_url, headers={"user-agent": "tx_cache"}).json(
|
||||
parse_float=Decimal
|
||||
)
|
||||
if "error" in tx_src:
|
||||
raise RuntimeError(tx_src["error"])
|
||||
return btc.from_json(tx_src)
|
||||
|
||||
|
||||
def get_prev_txes(
|
||||
inputs: List[messages.TxInputType],
|
||||
) -> Dict[bytes, messages.TransactionType]:
|
||||
"""Get info for all the previous transactions inputs are depending on."""
|
||||
prev_txes = {}
|
||||
for input in inputs:
|
||||
tx_id = input.prev_hash
|
||||
if tx_id not in prev_txes:
|
||||
prev_txes[tx_id] = get_tx_info(tx_id.hex())
|
||||
|
||||
return prev_txes
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
assert len(INPUTS) > 0, "there are no inputs"
|
||||
assert len(OUTPUTS) > 0, "there are no outputs"
|
||||
if not all(isinstance(inp, messages.TxInputType) for inp in INPUTS):
|
||||
raise RuntimeError("all inputs must be TxInputType")
|
||||
if not all(isinstance(out, messages.TxOutputType) for out in OUTPUTS):
|
||||
raise RuntimeError("all outputs must be TxOutputType")
|
||||
|
||||
_, serialized_tx = btc.sign_tx(
|
||||
CLIENT, COIN, INPUTS, OUTPUTS, prev_txes=get_prev_txes(INPUTS)
|
||||
)
|
||||
print(80 * "-")
|
||||
print(serialized_tx.hex())
|
210
tools/snippets/unify_test_files.py
Normal file
210
tools/snippets/unify_test_files.py
Normal file
@ -0,0 +1,210 @@
|
||||
"""
|
||||
Makes some unifications and improvements in the testing file.
|
||||
For example:
|
||||
- makes sure the paths have the same structure everywhere ("44h/1h/0h/1/0")
|
||||
- parse_path("m/44'/1'/0'/1/1")
|
||||
->
|
||||
- parse_path("44h/1h/0h/1/1")
|
||||
- formats big numbers with underscores for better readability (30_090_000)
|
||||
- amount=30090000,
|
||||
->
|
||||
- amount=30_090_000,
|
||||
- if it encouters a path or address, it tries to find its counterpart
|
||||
and put it as a comment to this line (it requires a translation file)
|
||||
- address_n=parse_path("44h/1h/0h/1/1"),
|
||||
->
|
||||
- address_n=parse_path("44h/1h/0h/1/1"), # mjXZwmEi1z1MzveZrKUAo4DBgbdq4sBYT6
|
||||
...
|
||||
- address="mwue7mokpBRAsJtHqEMcRPanYBmsSmYKvY",
|
||||
->
|
||||
- address="mwue7mokpBRAsJtHqEMcRPanYBmsSmYKvY", # 44h/1h/4h/0/2
|
||||
|
||||
The implementation here relies a lot on regexes, it could be better
|
||||
to use some syntax tree parser like https://github.com/Instagram/LibCST.
|
||||
|
||||
Usage:
|
||||
- specifying TRANSLATION_FILE (optional)
|
||||
- specifying FILES_TO_MODIFY
|
||||
- call the script with possible flags - see `python unify_test_files.py --help`
|
||||
"""
|
||||
import json
|
||||
import os
|
||||
import re
|
||||
from typing import List
|
||||
|
||||
import click
|
||||
|
||||
TRANSLATION_FILE = "address_cache_all_all_seed.json" # might be missing
|
||||
FILES_TO_MODIFY = [
|
||||
"./../../tests/device_tests/bitcoin/test_signtx.py",
|
||||
"./../../tests/device_tests/bitcoin/test_multisig.py",
|
||||
"./../../tests/device_tests/bitcoin/test_signtx_segwit.py",
|
||||
]
|
||||
|
||||
|
||||
class FileUnifier:
|
||||
# Optional "m/" prefix, at least three (\d+[h']/) groups and then take the rest [\dh'/]*
|
||||
PATH_REGEX = r"(?:m/)?(?:\d+[h']/){3,}[\dh'/]*"
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
translation_file: str,
|
||||
files_to_modify: List[str],
|
||||
quiet: bool = False,
|
||||
check_only: bool = False,
|
||||
) -> None:
|
||||
self.files_to_modify = files_to_modify
|
||||
self.quiet = quiet
|
||||
self.check_only = check_only
|
||||
|
||||
# File might not exist, in that case not doing translation
|
||||
# Example content: {"44h/1h/0h/0/1": "mopZWqZZyQc3F2Sy33cvDtJchSAMsnLi7b"}
|
||||
if os.path.isfile(translation_file):
|
||||
with open(translation_file, "r") as file:
|
||||
path_to_address = json.load(file)
|
||||
self.translations = {
|
||||
**path_to_address,
|
||||
**{a: p for p, a in path_to_address.items()},
|
||||
}
|
||||
else:
|
||||
self.translations = {}
|
||||
print(
|
||||
f"{len(self.translations)} translations available (path/address and address/path)\n{80*'*'}"
|
||||
)
|
||||
|
||||
# To be used for reporting purposes and to pass data around easily
|
||||
self.file: str
|
||||
self.line: str
|
||||
self.line_no: int
|
||||
|
||||
def unify_files(self) -> None:
|
||||
for file in self.files_to_modify:
|
||||
self.modify_file(file)
|
||||
|
||||
def modify_file(self, file: str) -> None:
|
||||
"""Read the file, modify lines and save them back into it."""
|
||||
new_lines: List[str] = []
|
||||
self.file = file
|
||||
self.line_no = 1
|
||||
with open(file, "r") as f:
|
||||
for line in f:
|
||||
self.line = line
|
||||
self.modify_line()
|
||||
new_lines.append(self.line)
|
||||
self.line_no += 1
|
||||
|
||||
if not self.check_only:
|
||||
with open(file, "w") as f:
|
||||
f.writelines(new_lines)
|
||||
|
||||
def modify_line(self) -> None:
|
||||
"""What should be done to this line."""
|
||||
# Not interested in whole comment lines - not changing them
|
||||
if self.line.lstrip().startswith("#"):
|
||||
return
|
||||
|
||||
# All modifiers should modify self.line
|
||||
modifiers = [
|
||||
self.path_to_uniform_format,
|
||||
self.path_to_address_translation,
|
||||
self.address_to_path_translation,
|
||||
self.format_long_numbers,
|
||||
]
|
||||
for modifier in modifiers:
|
||||
modifier()
|
||||
|
||||
def path_to_uniform_format(self) -> None:
|
||||
"""Unifies all paths to the same format."""
|
||||
if path_match := re.search(self.PATH_REGEX, self.line):
|
||||
|
||||
def sanitize_path(m: re.Match) -> str:
|
||||
# without "m/" and with "h" instead of "'"
|
||||
path = m[0]
|
||||
if path.startswith("m/"):
|
||||
path = path[2:]
|
||||
return path.replace("'", "h")
|
||||
|
||||
new_line = re.sub(self.PATH_REGEX, sanitize_path, self.line)
|
||||
if new_line != self.line:
|
||||
self.report_change(
|
||||
f"path sanitized - {path_match.group()}",
|
||||
new_line,
|
||||
)
|
||||
self.line = new_line
|
||||
|
||||
def path_to_address_translation(self) -> None:
|
||||
"""Translate path to address according to translations file."""
|
||||
if path_match := re.search(self.PATH_REGEX, self.line):
|
||||
if address := self.translations.get(path_match.group()):
|
||||
# Address might be there from previous run
|
||||
if address not in self.line:
|
||||
new_line = f"{self.line.rstrip()} # {address}\n"
|
||||
self.report_change(
|
||||
f"path translated - {path_match.group()}",
|
||||
new_line,
|
||||
)
|
||||
self.line = new_line
|
||||
|
||||
def address_to_path_translation(self) -> None:
|
||||
"""Translate address to path according to translations file."""
|
||||
address_regex = r"\b\w{33,35}\b"
|
||||
if address_match := re.search(address_regex, self.line):
|
||||
if path := self.translations.get(address_match.group()):
|
||||
# Path might be there from previous run
|
||||
if path not in self.line:
|
||||
new_line = f"{self.line.rstrip()} # {path}\n"
|
||||
self.report_change(
|
||||
f"address translated - {address_match.group()}",
|
||||
new_line,
|
||||
)
|
||||
self.line = new_line
|
||||
|
||||
def format_long_numbers(self) -> None:
|
||||
"""Uses underscore delimiters in long integers."""
|
||||
long_number_regex = r"\d{4,}"
|
||||
if number_match := re.search(long_number_regex, self.line):
|
||||
# Do it only in amount lines
|
||||
if "amount=" in self.line:
|
||||
|
||||
def num_to_underscore(m: re.Match) -> str:
|
||||
# https://stackoverflow.com/questions/9475241/split-string-every-nth-character
|
||||
# https://stackoverflow.com/questions/931092/reverse-a-string-in-python
|
||||
parts_reversed = re.findall(".{1,3}", m[0][::-1])
|
||||
return "_".join(parts_reversed)[::-1]
|
||||
|
||||
new_line = re.sub(long_number_regex, num_to_underscore, self.line)
|
||||
if new_line != self.line:
|
||||
self.report_change(
|
||||
f"long number formatted - {number_match.group()}",
|
||||
new_line,
|
||||
)
|
||||
self.line = new_line
|
||||
|
||||
def report_change(self, info: str, new_line: str) -> None:
|
||||
if self.quiet:
|
||||
return
|
||||
|
||||
would_be = "would be" if self.check_only else ""
|
||||
print(f"{self.file}:{self.line_no} {would_be} changed")
|
||||
print(info)
|
||||
print(self.line.strip())
|
||||
print(f" {would_be} changed to")
|
||||
print(new_line.strip())
|
||||
print(80 * "*")
|
||||
|
||||
|
||||
@click.command()
|
||||
@click.option("-q", "--quiet", is_flag=True, help="Do not report")
|
||||
@click.option("-c", "--check_only", is_flag=True, help="Do not rewrite")
|
||||
def run_unifier(quiet: bool, check_only: bool) -> None:
|
||||
file_unifier = FileUnifier(
|
||||
translation_file=TRANSLATION_FILE,
|
||||
files_to_modify=FILES_TO_MODIFY,
|
||||
quiet=quiet,
|
||||
check_only=check_only,
|
||||
)
|
||||
file_unifier.unify_files()
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
run_unifier()
|
Loading…
Reference in New Issue
Block a user