1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-21 04:41:18 +00:00

feat(core): automatically choose best QR encoding

This commit is contained in:
matejcik 2022-03-04 15:33:33 +01:00 committed by matejcik
parent c3b28d8fb4
commit 4e2129e0a0
14 changed files with 66 additions and 30 deletions

View File

@ -0,0 +1 @@
Automatically choose best size and encoding for QR codes.

View File

@ -0,0 +1 @@
Bitcoin bech32 addresses are encoded in lower-case for QR codes.

View File

@ -25,6 +25,6 @@ async def get_address(
address = address_from_public_key(pubkey, HRP)
if msg.show_display:
title = paths.address_n_to_str(msg.address_n)
await show_address(ctx, address=address, address_qr=address, title=title)
await show_address(ctx, address=address, title=title)
return BinanceAddress(address=address)

View File

@ -55,15 +55,15 @@ async def get_address(
address = addresses.get_address(msg.script_type, coin, node, msg.multisig)
address_short = addresses.address_short(coin, address)
address_case_sensitive = True
if coin.segwit and msg.script_type in (
InputScriptType.SPENDWITNESS,
InputScriptType.SPENDTAPROOT,
):
address_qr = address.upper() # bech32 address
address_case_sensitive = False # bech32 address
elif coin.cashaddr_prefix is not None:
address_qr = address.upper() # cashaddr address
else:
address_qr = address # base58 address
address_case_sensitive = False # cashaddr address
mac: bytes | None = None
multisig_xpub_magic = coin.xpub_magic
@ -100,7 +100,7 @@ async def get_address(
await show_address(
ctx,
address=address_short,
address_qr=address_qr,
case_sensitive=address_case_sensitive,
title=title,
multisig_index=multisig_index,
xpubs=_get_xpubs(coin, multisig_xpub_magic, pubnodes),
@ -108,7 +108,10 @@ async def get_address(
else:
title = address_n_to_str(msg.address_n)
await show_address(
ctx, address=address_short, address_qr=address_qr, title=title
ctx,
address=address_short,
case_sensitive=address_case_sensitive,
title=title,
)
return Address(address=address, mac=mac)

View File

@ -32,6 +32,6 @@ async def get_address(
if msg.show_display:
title = paths.address_n_to_str(msg.address_n)
await show_address(ctx, address=address, address_qr=address, title=title)
await show_address(ctx, address=address, title=title)
return EthereumAddress(address=address)

View File

@ -33,7 +33,7 @@ async def get_address(
await show_address(
ctx,
address=address,
address_qr=address.upper(),
case_sensitive=False,
title=title,
network=get_network_str(msg.network),
)

View File

@ -25,6 +25,6 @@ async def get_address(
if msg.show_display:
title = paths.address_n_to_str(msg.address_n)
await show_address(ctx, address=address, address_qr=address, title=title)
await show_address(ctx, address=address, title=title)
return RippleAddress(address=address)

View File

@ -25,8 +25,6 @@ async def get_address(
if msg.show_display:
title = paths.address_n_to_str(msg.address_n)
await show_address(
ctx, address=address, address_qr=address.upper(), title=title
)
await show_address(ctx, address=address, case_sensitive=False, title=title)
return StellarAddress(address=address)

View File

@ -30,6 +30,6 @@ async def get_address(
if msg.show_display:
title = paths.address_n_to_str(msg.address_n)
await show_address(ctx, address=address, address_qr=address, title=title)
await show_address(ctx, address=address, title=title)
return TezosAddress(address=address)

View File

@ -12,5 +12,5 @@ MONO_ADDR_PER_LINE = const(17)
MONO_HEX_PER_LINE = const(18)
QR_X = const(120)
QR_Y = const(115)
QR_SIZE_THRESHOLD = const(63)
QR_Y = const(112)
QR_SIDE_MAX = const(140)

View File

@ -31,7 +31,6 @@ from ...components.tt.text import LINE_WIDTH_PAGINATED, Span, Text
from ...constants.tt import (
MONO_ADDR_PER_LINE,
MONO_HEX_PER_LINE,
QR_SIZE_THRESHOLD,
QR_X,
QR_Y,
TEXT_MAX_LINES,
@ -228,11 +227,11 @@ async def confirm_path_warning(
def _show_qr(
address: str,
case_sensitive: bool,
title: str,
cancel: str = "Address",
) -> Confirm:
QR_COEF = const(4) if len(address) < QR_SIZE_THRESHOLD else const(3)
qr = Qr(address, QR_X, QR_Y, QR_COEF)
qr = Qr(address, case_sensitive, QR_X, QR_Y)
text = Text(title, ui.ICON_RECEIVE, ui.GREEN)
return Confirm(Container(qr, text), cancel=cancel, cancel_style=ButtonDefault)
@ -315,7 +314,9 @@ async def show_xpub(
async def show_address(
ctx: wire.GenericContext,
address: str,
*,
address_qr: str | None = None,
case_sensitive: bool = True,
title: str = "Confirm address",
network: str | None = None,
multisig_index: int | None = None,
@ -344,6 +345,7 @@ async def show_address(
ctx,
_show_qr(
address if address_qr is None else address_qr,
case_sensitive,
title if title_qr is None else title_qr,
cancel="XPUBs" if is_multisig else "Address",
),

View File

@ -134,6 +134,8 @@ async def show_xpub(
async def show_address(
ctx: wire.GenericContext,
address: str,
*,
case_sensitive: bool = True,
address_qr: str | None = None,
title: str = "Confirm address",
network: str | None = None,

View File

@ -1,13 +1,42 @@
from typing import Sequence
from trezor import ui
from trezor.ui.constants import QR_SIDE_MAX
_QR_WIDTHS = (21, 25, 29, 33, 37, 41, 45, 49, 53, 57)
_THRESHOLDS_BINARY = (14, 26, 42, 62, 84, 106, 122, 152, 180, 213)
_THRESHOLDS_ALPHANUM = (20, 38, 61, 90, 122, 154, 178, 221, 262, 311)
def _qr_version_index(data: str, thresholds: Sequence[int]) -> int:
for i, threshold in enumerate(thresholds):
if len(data) <= threshold:
return i
raise ValueError # data too long
def is_alphanum_only(data: str) -> bool:
return all(c in "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ $*+-./:" for c in data)
class Qr(ui.Component):
def __init__(self, data: str, x: int, y: int, scale: int):
def __init__(self, data: str, case_sensitive: bool, x: int, y: int) -> None:
super().__init__()
if case_sensitive and not is_alphanum_only(data):
# must encode in BINARY mode
version_idx = _qr_version_index(data, _THRESHOLDS_BINARY)
else:
# can make the QR code more readable by using the best version
version_idx = _qr_version_index(data, _THRESHOLDS_ALPHANUM)
if len(data) > _THRESHOLDS_BINARY[version_idx]:
data = data.upper()
size = _QR_WIDTHS[version_idx]
self.data = data
self.x = x
self.y = y
self.scale = scale
self.scale = QR_SIDE_MAX // size
def on_render(self) -> None:
if self.repaint:

View File

@ -701,17 +701,17 @@
"TT_bitcoin-test_getaddress_segwit_native.py::test_show_segwit[Testnet-m-86h-1h-0h-0-0-InputScr-821a199d": "c09de07fbbf1e047442180e2facb5482d06a1a428891b875b7dd93c9e4704ae1",
"TT_bitcoin-test_getaddress_segwit_native.py::test_show_segwit[Testnet-m-86h-1h-0h-1-0-InputScr-9d2fa8bc": "c09de07fbbf1e047442180e2facb5482d06a1a428891b875b7dd93c9e4704ae1",
"TT_bitcoin-test_getaddress_segwit_native.py::test_show_segwit[Testnet-m-86h-1h-0h-1-0-InputScr-d5b7f8fc": "4460b43883e6049f719050ee015b01f6dd528c1aa51b96ae97c60230f56bbc7e",
"TT_bitcoin-test_getaddress_show.py::test_show[m-44h-0h-12h-0-0-InputScriptType.SPENDADDRESS-1F-1e4f2f74": "1639a90cd9deccc25478b736c1992bd3f1ef17db95ab7e0ea82451dff0ebb12d",
"TT_bitcoin-test_getaddress_show.py::test_show[m-49h-0h-12h-0-0-InputScriptType.SPENDP2SHWITNES-a986211d": "a2b3af26697700980024d2d9ba1145925d9c3c9dea40f6cc3d4546ea94e6e913",
"TT_bitcoin-test_getaddress_show.py::test_show[m-84h-0h-12h-0-0-InputScriptType.SPENDWITNESS-bc-a5f08dfb": "c3258bfb300e27bb54aec51eaf68008e087191b97794ddb8a1e6ee650fd8a6b2",
"TT_bitcoin-test_getaddress_show.py::test_show[m-44h-0h-12h-0-0-InputScriptType.SPENDADDRESS-1F-1e4f2f74": "f2f31c8493549ddff876e5222b262063a39839c3884ec84d8d1890cc52a0f036",
"TT_bitcoin-test_getaddress_show.py::test_show[m-49h-0h-12h-0-0-InputScriptType.SPENDP2SHWITNES-a986211d": "38ebd99ae3ae7d8f30ce77acb153f6d6637a3a81c86c7e5d038c3e1628ee2509",
"TT_bitcoin-test_getaddress_show.py::test_show[m-84h-0h-12h-0-0-InputScriptType.SPENDWITNESS-bc-a5f08dfb": "48c1738b62a2a04036aa2a521240e192ea025bb98fd6dd2f038bcf2d5af1a214",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_15": "1a656989a6144461d99ea5aae33bb1683c8eafb95eaf5523d21f736df6861043",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_3": "c305f05b829e446d327e69e75ededde3bb76c9cc9f39f5a0f50788874bebd372",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDMULTISIG-0-3-4efd9cf3": "67f2450d6655510cc5f4ce1790af851a4565d4ff54c817738b9e33994ff2843d",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDMULTISIG-0-3-98a7e339": "67f2450d6655510cc5f4ce1790af851a4565d4ff54c817738b9e33994ff2843d",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDP2SHWITNESS--2cf5f03c": "59e21947978d42aa97b9a076e4994b23784a244713441e0d10e18711b1be1eae",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDP2SHWITNESS--5ea18367": "9997e053d6f88b966da59eb7c67bdb47015094d7a7ecfadf8d71476a21a3c4c5",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDWITNESS-2-bc-e70b56ea": "947c8c6a4ae8b5519dc4e9939496ba579288bbb5334625d44bf4f307ebde4ea9",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDWITNESS-2-bc-f3c4650f": "224e62bbc3546a18cc033d7d61d63ae2ed0750a7d8cd9bfd3d0deed18d42de62",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDMULTISIG-0-3-4efd9cf3": "bec64e8a3c7079d97ad5b487914a0da9c6fb541714785772dd67ea89f86a040a",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDMULTISIG-0-3-98a7e339": "bec64e8a3c7079d97ad5b487914a0da9c6fb541714785772dd67ea89f86a040a",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDP2SHWITNESS--2cf5f03c": "cc8a98fb9dd08f15fa27ff0b74b1b147576ffe57e3c56d802e33f0fa0a4c3752",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDP2SHWITNESS--5ea18367": "cd7d53102ee2482e4014c602767550cc3f2a9c1c2833e908a857e7d093b7b3d6",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDWITNESS-2-bc-e70b56ea": "ed9eafcc09412c67de9f086ac79ec33a9b3335b2b2dec721965c75f0bea87dae",
"TT_bitcoin-test_getaddress_show.py::test_show_multisig_xpubs[InputScriptType.SPENDWITNESS-2-bc-f3c4650f": "9ea4ef4f47d568081aed5f03a43a92e307933ed6d05f274cb2961a565b9d7ec7",
"TT_bitcoin-test_getaddress_show.py::test_show_unrecognized_path": "c09de07fbbf1e047442180e2facb5482d06a1a428891b875b7dd93c9e4704ae1",
"TT_bitcoin-test_getownershipproof.py::test_attack_ownership_id": "c09de07fbbf1e047442180e2facb5482d06a1a428891b875b7dd93c9e4704ae1",
"TT_bitcoin-test_getownershipproof.py::test_confirm_ownership_proof": "a79fd462b99f5ffc755917b6d06d3c5a3061c971649cdfa752cbc9ef068e6acf",