1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-28 16:21:03 +00:00

stellar: refactoring

This commit is contained in:
Tomas Susanka 2018-06-06 16:02:13 +02:00
parent 2af33a6893
commit 501ea6bb2c
8 changed files with 216 additions and 203 deletions

View File

@ -9,3 +9,8 @@ REVIEWER = Jan Pochyla <jan.pochyla@satoshilabs.com>
----- -----
TODO TODO
# Stellar transactions consist of sha256 of:
# - sha256(network passphrase)
# - 4-byte unsigned big-endian int type constant (2 for tx)
# - public key

View File

@ -34,6 +34,7 @@ op_wire_types = [
wire_types.StellarSetOptionsOp, wire_types.StellarSetOptionsOp,
] ]
# https://github.com/stellar/go/blob/e0ffe19f58879d3c31e2976b97a5bf10e13a337b/xdr/xdr_generated.go#L584
ASSET_TYPE_NATIVE = const(0) ASSET_TYPE_NATIVE = const(0)
ASSET_TYPE_ALPHANUM4 = const(1) ASSET_TYPE_ALPHANUM4 = const(1)
ASSET_TYPE_ALPHANUM12 = const(2) ASSET_TYPE_ALPHANUM12 = const(2)

View File

@ -36,5 +36,5 @@ async def _show(ctx, address: str):
cancel_style=ui.BTN_KEY) cancel_style=ui.BTN_KEY)
def _split_address(address: str): def _split_address(address: str): # todo merge with NEM
return chunks(address, 17) return chunks(address, 17)

View File

@ -1,4 +1,4 @@
from apps.common.confirm import * from apps.common.confirm import require_confirm, require_hold_to_confirm
from apps.stellar import consts from apps.stellar import consts
from apps.stellar import helpers from apps.stellar import helpers
from trezor import ui from trezor import ui
@ -66,6 +66,7 @@ def format_address(pubkey: bytes) -> str:
return helpers.address_from_public_key(pubkey) return helpers.address_from_public_key(pubkey)
# todo merge with nem
def split(text): def split(text):
return utils.chunks(text, 17) return utils.chunks(text, 17)

View File

@ -1,44 +1,46 @@
from apps.stellar.operations.serialize import * from apps.stellar.operations import serialize
from apps.stellar.operations.layout import * from apps.stellar.operations import layout
from apps.stellar import consts
from apps.stellar import writers
async def operation(ctx, w, op): async def operation(ctx, w, op):
if op.source_account: if op.source_account:
await confirm_source_account(ctx, op.source_account) await layout.confirm_source_account(ctx, op.source_account)
serialize_account(w, op.source_account) serialize.serialize_account(w, op.source_account)
write_uint32(w, get_op_code(op)) writers.write_uint32(w, consts.get_op_code(op))
if isinstance(op, StellarAccountMergeOp): if isinstance(op, serialize.StellarAccountMergeOp):
await confirm_account_merge_op(ctx, op) await layout.confirm_account_merge_op(ctx, op)
serialize_account_merge_op(w, op) serialize.serialize_account_merge_op(w, op)
elif isinstance(op, StellarAllowTrustOp): elif isinstance(op, serialize.StellarAllowTrustOp):
await confirm_allow_trust_op(ctx, op) await layout.confirm_allow_trust_op(ctx, op)
serialize_allow_trust_op(w, op) serialize.serialize_allow_trust_op(w, op)
elif isinstance(op, StellarBumpSequenceOp): elif isinstance(op, serialize.StellarBumpSequenceOp):
await confirm_bump_sequence_op(ctx, op) await layout.confirm_bump_sequence_op(ctx, op)
serialize_bump_sequence_op(w, op) serialize.serialize_bump_sequence_op(w, op)
elif isinstance(op, StellarChangeTrustOp): elif isinstance(op, serialize.StellarChangeTrustOp):
await confirm_change_trust_op(ctx, op) await layout.confirm_change_trust_op(ctx, op)
serialize_change_trust_op(w, op) serialize.serialize_change_trust_op(w, op)
elif isinstance(op, StellarCreateAccountOp): elif isinstance(op, serialize.StellarCreateAccountOp):
await confirm_create_account_op(ctx, op) await layout.confirm_create_account_op(ctx, op)
serialize_create_account_op(w, op) serialize.serialize_create_account_op(w, op)
elif isinstance(op, StellarCreatePassiveOfferOp): elif isinstance(op, serialize.StellarCreatePassiveOfferOp):
await confirm_create_passive_offer_op(ctx, op) await layout.confirm_create_passive_offer_op(ctx, op)
serialize_create_passive_offer_op(w, op) serialize.serialize_create_passive_offer_op(w, op)
elif isinstance(op, StellarManageDataOp): elif isinstance(op, serialize.StellarManageDataOp):
await confirm_manage_data_op(ctx, op) await layout.confirm_manage_data_op(ctx, op)
serialize_manage_data_op(w, op) serialize.serialize_manage_data_op(w, op)
elif isinstance(op, StellarManageOfferOp): elif isinstance(op, serialize.StellarManageOfferOp):
await confirm_manage_offer_op(ctx, op) await layout.confirm_manage_offer_op(ctx, op)
serialize_manage_offer_op(w, op) serialize.serialize_manage_offer_op(w, op)
elif isinstance(op, StellarPathPaymentOp): elif isinstance(op, serialize.StellarPathPaymentOp):
await confirm_path_payment_op(ctx, op) await layout.confirm_path_payment_op(ctx, op)
serialize_path_payment_op(w, op) serialize.serialize_path_payment_op(w, op)
elif isinstance(op, StellarPaymentOp): elif isinstance(op, serialize.StellarPaymentOp):
await confirm_payment_op(ctx, op) await layout.confirm_payment_op(ctx, op)
serialize_payment_op(w, op) serialize.serialize_payment_op(w, op)
elif isinstance(op, StellarSetOptionsOp): elif isinstance(op, serialize.StellarSetOptionsOp):
await confirm_set_options_op(ctx, op) await layout.confirm_set_options_op(ctx, op)
serialize_set_options_op(w, op) serialize.serialize_set_options_op(w, op)
else: else:
raise ValueError('Stellar: unknown operation') raise ValueError('serialize.Stellar: unknown operation')

View File

@ -1,7 +1,7 @@
from apps.stellar.layout import * from apps.stellar.layout import split, format_address, format_amount, ui, trim_to_rows, require_confirm
from apps.stellar import consts
from trezor.messages import ButtonRequestType from trezor.messages import ButtonRequestType
from trezor.ui.text import Text from trezor.ui.text import Text
from trezor.messages.StellarAccountMergeOp import StellarAccountMergeOp from trezor.messages.StellarAccountMergeOp import StellarAccountMergeOp
from trezor.messages.StellarAssetType import StellarAssetType from trezor.messages.StellarAssetType import StellarAssetType
from trezor.messages.StellarAllowTrustOp import StellarAllowTrustOp from trezor.messages.StellarAllowTrustOp import StellarAllowTrustOp
@ -27,12 +27,12 @@ async def confirm_source_account(ctx, source_account: bytes):
async def confirm_allow_trust_op(ctx, op: StellarAllowTrustOp): async def confirm_allow_trust_op(ctx, op: StellarAllowTrustOp):
if op.is_authorized: if op.is_authorized:
text = 'Allow' text = 'Allow Trust'
else: else:
text = 'Revoke' text = 'Revoke Trust'
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, text + ' Trust', ui.BOLD, text,
ui.NORMAL, "of '" + op.asset_code + "' by:", ui.NORMAL, 'of %s by:' % op.asset_code,
ui.MONO, *split(trim_to_rows(format_address(op.trusted_account), 3)), ui.MONO, *split(trim_to_rows(format_address(op.trusted_account), 3)),
icon_color=ui.GREEN) icon_color=ui.GREEN)
@ -59,13 +59,13 @@ async def confirm_bump_sequence_op(ctx, op: StellarBumpSequenceOp):
async def confirm_change_trust_op(ctx, op: StellarChangeTrustOp): async def confirm_change_trust_op(ctx, op: StellarChangeTrustOp):
if op.limit == 0: if op.limit == 0:
text = 'Delete' text = 'Delete Trust'
else: else:
text = 'Add' text = 'Add Trust'
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, text + ' Trust', ui.BOLD, text,
ui.NORMAL, 'Asset: ' + op.asset.code, ui.NORMAL, 'Asset: %s' % op.asset.code,
ui.NORMAL, 'Amount: ' + format_amount(op.limit, ticker=False), ui.NORMAL, 'Amount: %s' % format_amount(op.limit, ticker=False),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
await confirm_asset_issuer(ctx, op.asset) await confirm_asset_issuer(ctx, op.asset)
@ -74,7 +74,7 @@ async def confirm_change_trust_op(ctx, op: StellarChangeTrustOp):
async def confirm_create_account_op(ctx, op: StellarCreateAccountOp): async def confirm_create_account_op(ctx, op: StellarCreateAccountOp):
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, 'Create Account', ui.BOLD, 'Create Account',
ui.NORMAL, 'with ' + format_amount(op.starting_balance), ui.NORMAL, 'with %s' % format_amount(op.starting_balance),
ui.MONO, *split(trim_to_rows(format_address(op.new_account), 3)), ui.MONO, *split(trim_to_rows(format_address(op.new_account), 3)),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
@ -96,32 +96,30 @@ async def confirm_manage_offer_op(ctx, op: StellarManageOfferOp):
text = 'Delete' text = 'Delete'
else: else:
text = 'Update' text = 'Update'
text += ' #' + str(op.offer_id) text += ' #%d' % op.offer_id
await _confirm_offer(ctx, text, op) await _confirm_offer(ctx, text, op)
# todo scale? this is inconsistent with T1 (op.amount should? use divisibility)
async def _confirm_offer(ctx, text, op): async def _confirm_offer(ctx, text, op):
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, text, ui.BOLD, text,
ui.NORMAL, 'Sell ' + str(op.amount) + ' ' + op.selling_asset.code, ui.NORMAL, 'Sell %s %s' % (format_amount(op.amount, ticker=False), op.selling_asset.code),
ui.NORMAL, 'For ' + str(op.price_n / op.price_d), ui.NORMAL, 'For %f' % (op.price_n / op.price_d),
ui.NORMAL, 'Per ' + format_asset_code(op.buying_asset), ui.NORMAL, 'Per %s' % format_asset_code(op.buying_asset),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
await confirm_asset_issuer(ctx, op.selling_asset) await confirm_asset_issuer(ctx, op.selling_asset)
await confirm_asset_issuer(ctx, op.buying_asset) await confirm_asset_issuer(ctx, op.buying_asset)
# todo done
async def confirm_manage_data_op(ctx, op: StellarManageDataOp): async def confirm_manage_data_op(ctx, op: StellarManageDataOp):
from trezor.crypto.hashlib import sha256 from trezor.crypto.hashlib import sha256
if op.value: if op.value:
text = 'Set' title = 'Set'
else: else:
text = 'Clear' title = 'Clear'
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, text + ' data value key', ui.BOLD, '%s data value key' % title,
ui.MONO, *split(op.key), ui.MONO, *split(op.key),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
@ -129,7 +127,7 @@ async def confirm_manage_data_op(ctx, op: StellarManageDataOp):
digest = sha256(op.value).digest() digest = sha256(op.value).digest()
digest_str = hexlify(digest).decode() digest_str = hexlify(digest).decode()
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, ' Value (SHA-256):', ui.BOLD, 'Value (SHA-256):',
ui.MONO, *split(digest_str), ui.MONO, *split(digest_str),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
@ -137,8 +135,8 @@ async def confirm_manage_data_op(ctx, op: StellarManageDataOp):
async def confirm_path_payment_op(ctx, op: StellarPathPaymentOp): async def confirm_path_payment_op(ctx, op: StellarPathPaymentOp):
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, 'Path Pay ' + format_amount(op.destination_amount, ticker=False), ui.BOLD, 'Path Pay %s' % format_amount(op.destination_amount, ticker=False),
ui.BOLD, format_asset_code(op.destination_asset) + ' to:', ui.BOLD, '%s to:' % format_asset_code(op.destination_asset),
ui.MONO, *split(trim_to_rows(format_address(op.destination_account), 3)), ui.MONO, *split(trim_to_rows(format_address(op.destination_account), 3)),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
@ -157,8 +155,8 @@ async def confirm_path_payment_op(ctx, op: StellarPathPaymentOp):
async def confirm_payment_op(ctx, op: StellarPaymentOp): async def confirm_payment_op(ctx, op: StellarPaymentOp):
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, 'Pay ' + format_amount(op.amount, ticker=False), ui.BOLD, 'Pay %s' % format_amount(op.amount, ticker=False),
ui.BOLD, format_asset_code(op.asset) + ' to:', ui.BOLD, '%s to:' % format_asset_code(op.asset),
ui.MONO, *split(trim_to_rows(format_address(op.destination_account), 3)), ui.MONO, *split(trim_to_rows(format_address(op.destination_account), 3)),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
@ -201,23 +199,22 @@ async def confirm_set_options_op(ctx, op: StellarSetOptionsOp):
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
if op.signer_type is not None: if op.signer_type is not None:
if op.signer_weight > 0: if op.signer_weight > 0:
text = 'Add Signer' text = 'Add Signer (%s)'
else: else:
text = 'Remove Signer' text = 'Remove Signer (%s)'
if op.signer_type == consts.SIGN_TYPE_ACCOUNT: if op.signer_type == consts.SIGN_TYPE_ACCOUNT:
text += ' (acc)'
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, text, ui.BOLD, text % 'acc',
ui.MONO, *split(format_address(op.signer_key)), ui.MONO, *split(format_address(op.signer_key)),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
elif op.signer_type in [consts.SIGN_TYPE_PRE_AUTH, consts.SIGN_TYPE_HASH]: elif op.signer_type in (consts.SIGN_TYPE_PRE_AUTH, consts.SIGN_TYPE_HASH):
if op.signer_type == consts.SIGN_TYPE_PRE_AUTH: if op.signer_type == consts.SIGN_TYPE_PRE_AUTH:
text += ' (auth)' signer_type = 'auth'
else: else:
text += ' (hash)' signer_type = 'hash'
content = Text('Confirm operation', ui.ICON_CONFIRM, content = Text('Confirm operation', ui.ICON_CONFIRM,
ui.BOLD, text, ui.BOLD, text % signer_type,
ui.MONO, *split(hexlify(op.signer_key).decode()), ui.MONO, *split(hexlify(op.signer_key).decode()),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)
@ -225,20 +222,20 @@ async def confirm_set_options_op(ctx, op: StellarSetOptionsOp):
raise ValueError('Stellar: invalid signer type') raise ValueError('Stellar: invalid signer type')
def _format_thresholds(op: StellarSetOptionsOp) -> (): def _format_thresholds(op: StellarSetOptionsOp) -> tuple:
text = () text = ()
if op.master_weight is not None: if op.master_weight is not None:
text += ('Master Weight: ' + str(op.master_weight), ) text += ('Master Weight: %d' % op.master_weight, )
if op.low_threshold is not None: if op.low_threshold is not None:
text += ('Low: ' + str(op.low_threshold), ) text += ('Low: %d' % op.low_threshold, )
if op.medium_threshold is not None: if op.medium_threshold is not None:
text += ('Medium: ' + str(op.medium_threshold), ) text += ('Medium: %d' % op.medium_threshold, )
if op.high_threshold is not None: if op.high_threshold is not None:
text += ('High: ' + str(op.high_threshold), ) text += ('High: %d' % op.high_threshold, )
return text return text
def _format_flags(flags: int) -> (): def _format_flags(flags: int) -> tuple:
if flags > consts.FLAGS_MAX_SIZE: if flags > consts.FLAGS_MAX_SIZE:
raise ValueError('Stellar: invalid') raise ValueError('Stellar: invalid')
text = () text = ()
@ -261,7 +258,7 @@ async def confirm_asset_issuer(ctx, asset: StellarAssetType):
if asset is None or asset.type == consts.ASSET_TYPE_NATIVE: if asset is None or asset.type == consts.ASSET_TYPE_NATIVE:
return return
content = Text('Confirm issuer', ui.ICON_CONFIRM, content = Text('Confirm issuer', ui.ICON_CONFIRM,
ui.BOLD, asset.code + ' issuer:', ui.BOLD, '%s issuer:' % asset.code,
ui.MONO, *split(format_address(asset.issuer)), ui.MONO, *split(format_address(asset.issuer)),
icon_color=ui.GREEN) icon_color=ui.GREEN)
await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput) await require_confirm(ctx, content, ButtonRequestType.ConfirmOutput)

View File

@ -1,3 +1,5 @@
from apps.stellar import writers
from apps.stellar import consts
from trezor.messages.StellarAccountMergeOp import StellarAccountMergeOp from trezor.messages.StellarAccountMergeOp import StellarAccountMergeOp
from trezor.messages.StellarAssetType import StellarAssetType from trezor.messages.StellarAssetType import StellarAssetType
from trezor.messages.StellarAllowTrustOp import StellarAllowTrustOp from trezor.messages.StellarAllowTrustOp import StellarAllowTrustOp
@ -10,156 +12,154 @@ from trezor.messages.StellarManageOfferOp import StellarManageOfferOp
from trezor.messages.StellarPathPaymentOp import StellarPathPaymentOp from trezor.messages.StellarPathPaymentOp import StellarPathPaymentOp
from trezor.messages.StellarPaymentOp import StellarPaymentOp from trezor.messages.StellarPaymentOp import StellarPaymentOp
from trezor.messages.StellarSetOptionsOp import StellarSetOptionsOp from trezor.messages.StellarSetOptionsOp import StellarSetOptionsOp
from apps.stellar.consts import *
from apps.stellar.writers import *
def serialize_account_merge_op(w, msg: StellarAccountMergeOp): def serialize_account_merge_op(w, msg: StellarAccountMergeOp):
write_pubkey(w, msg.destination_account) writers.write_pubkey(w, msg.destination_account)
def serialize_allow_trust_op(w, msg: StellarAllowTrustOp): def serialize_allow_trust_op(w, msg: StellarAllowTrustOp):
# trustor account (the account being allowed to access the asset) # trustor account (the account being allowed to access the asset)
write_pubkey(w, msg.trusted_account) writers.write_pubkey(w, msg.trusted_account)
write_uint32(w, msg.asset_type) writers.write_uint32(w, msg.asset_type)
_serialize_asset_code(w, msg.asset_type, msg.asset_code) _serialize_asset_code(w, msg.asset_type, msg.asset_code)
write_bool(w, msg.is_authorized) writers.write_bool(w, msg.is_authorized)
def serialize_bump_sequence_op(w, msg: StellarBumpSequenceOp): def serialize_bump_sequence_op(w, msg: StellarBumpSequenceOp):
write_uint64(w, msg.bump_to) writers.write_uint64(w, msg.bump_to)
def serialize_change_trust_op(w, msg: StellarChangeTrustOp): def serialize_change_trust_op(w, msg: StellarChangeTrustOp):
_serialize_asset(w, msg.asset) _serialize_asset(w, msg.asset)
write_uint64(w, msg.limit) writers.write_uint64(w, msg.limit)
def serialize_create_account_op(w, msg: StellarCreateAccountOp): def serialize_create_account_op(w, msg: StellarCreateAccountOp):
write_pubkey(w, msg.new_account) writers.write_pubkey(w, msg.new_account)
write_uint64(w, msg.starting_balance) writers.write_uint64(w, msg.starting_balance)
def serialize_create_passive_offer_op(w, msg: StellarCreatePassiveOfferOp): def serialize_create_passive_offer_op(w, msg: StellarCreatePassiveOfferOp):
_serialize_asset(w, msg.selling_asset) _serialize_asset(w, msg.selling_asset)
_serialize_asset(w, msg.buying_asset) _serialize_asset(w, msg.buying_asset)
write_uint64(w, msg.amount) writers.write_uint64(w, msg.amount)
write_uint32(w, msg.price_n) writers.write_uint32(w, msg.price_n)
write_uint32(w, msg.price_d) writers.write_uint32(w, msg.price_d)
def serialize_manage_data_op(w, msg: StellarManageDataOp): def serialize_manage_data_op(w, msg: StellarManageDataOp):
if len(msg.key) > 64: if len(msg.key) > 64:
raise ValueError('Stellar: max length of a key is 64 bytes') raise ValueError('Stellar: max length of a key is 64 bytes')
write_string(w, msg.key) writers.write_string(w, msg.key)
write_bool(w, bool(msg.value)) writers.write_bool(w, bool(msg.value))
if msg.value: if msg.value:
write_uint32(w, len(msg.value)) writers.write_uint32(w, len(msg.value))
write_bytes(w, msg.value) writers.write_bytes(w, msg.value)
def serialize_manage_offer_op(w, msg: StellarManageOfferOp): def serialize_manage_offer_op(w, msg: StellarManageOfferOp):
_serialize_asset(w, msg.selling_asset) _serialize_asset(w, msg.selling_asset)
_serialize_asset(w, msg.buying_asset) _serialize_asset(w, msg.buying_asset)
write_uint64(w, msg.amount) # amount to sell writers.write_uint64(w, msg.amount) # amount to sell
write_uint32(w, msg.price_n) # numerator writers.write_uint32(w, msg.price_n) # numerator
write_uint32(w, msg.price_d) # denominator writers.write_uint32(w, msg.price_d) # denominator
write_uint64(w, msg.offer_id) writers.write_uint64(w, msg.offer_id)
def serialize_path_payment_op(w, msg: StellarPathPaymentOp): def serialize_path_payment_op(w, msg: StellarPathPaymentOp):
_serialize_asset(w, msg.send_asset) _serialize_asset(w, msg.send_asset)
write_uint64(w, msg.send_max) writers.write_uint64(w, msg.send_max)
write_pubkey(w, msg.destination_account) writers.write_pubkey(w, msg.destination_account)
_serialize_asset(w, msg.destination_asset) _serialize_asset(w, msg.destination_asset)
write_uint64(w, msg.destination_amount) writers.write_uint64(w, msg.destination_amount)
write_uint32(w, len(msg.paths)) writers.write_uint32(w, len(msg.paths))
for p in msg.paths: for p in msg.paths:
_serialize_asset(w, p) _serialize_asset(w, p)
def serialize_payment_op(w, msg: StellarPaymentOp): def serialize_payment_op(w, msg: StellarPaymentOp):
write_pubkey(w, msg.destination_account) writers.write_pubkey(w, msg.destination_account)
_serialize_asset(w, msg.asset) _serialize_asset(w, msg.asset)
write_uint64(w, msg.amount) writers.write_uint64(w, msg.amount)
def serialize_set_options_op(w, msg: StellarSetOptionsOp): def serialize_set_options_op(w, msg: StellarSetOptionsOp):
# inflation destination # inflation destination
write_bool(w, bool(msg.inflation_destination_account)) writers.write_bool(w, bool(msg.inflation_destination_account))
if msg.inflation_destination_account: if msg.inflation_destination_account:
write_pubkey(w, msg.inflation_destination_account) writers.write_pubkey(w, msg.inflation_destination_account)
# clear flags # clear flags
write_bool(w, bool(msg.clear_flags)) writers.write_bool(w, bool(msg.clear_flags))
if msg.clear_flags: if msg.clear_flags:
write_uint32(w, msg.clear_flags) writers.write_uint32(w, msg.clear_flags)
# set flags # set flags
write_bool(w, bool(msg.set_flags)) writers.write_bool(w, bool(msg.set_flags))
if msg.set_flags: if msg.set_flags:
write_uint32(w, msg.set_flags) writers.write_uint32(w, msg.set_flags)
# account thresholds # account thresholds
write_bool(w, bool(msg.master_weight)) writers.write_bool(w, bool(msg.master_weight))
if msg.master_weight: if msg.master_weight:
write_uint32(w, msg.master_weight) writers.write_uint32(w, msg.master_weight)
write_bool(w, bool(msg.low_threshold)) writers.write_bool(w, bool(msg.low_threshold))
if msg.low_threshold: if msg.low_threshold:
write_uint32(w, msg.low_threshold) writers.write_uint32(w, msg.low_threshold)
write_bool(w, bool(msg.medium_threshold)) writers.write_bool(w, bool(msg.medium_threshold))
if msg.medium_threshold: if msg.medium_threshold:
write_uint32(w, msg.medium_threshold) writers.write_uint32(w, msg.medium_threshold)
write_bool(w, bool(msg.high_threshold)) writers.write_bool(w, bool(msg.high_threshold))
if msg.high_threshold: if msg.high_threshold:
write_uint32(w, msg.high_threshold) writers.write_uint32(w, msg.high_threshold)
# home domain # home domain
write_bool(w, bool(msg.home_domain)) writers.write_bool(w, bool(msg.home_domain))
if msg.home_domain: if msg.home_domain:
if len(msg.home_domain) > 32: if len(msg.home_domain) > 32:
raise ValueError('Stellar: max length of a home domain is 32 bytes') raise ValueError('Stellar: max length of a home domain is 32 bytes')
write_string(w, msg.home_domain) writers.write_string(w, msg.home_domain)
# signer # signer
write_bool(w, bool(msg.signer_type)) writers.write_bool(w, bool(msg.signer_type))
if msg.signer_type: if msg.signer_type:
# signer type # signer type
write_uint32(w, msg.signer_type) writers.write_uint32(w, msg.signer_type)
write_bytes(w, msg.signer_key) writers.write_bytes(w, msg.signer_key)
write_uint32(w, msg.signer_weight) writers.write_uint32(w, msg.signer_weight)
def serialize_account(w, source_account: bytes): def serialize_account(w, source_account: bytes):
if source_account is None: if source_account is None:
write_bool(w, False) writers.write_bool(w, False)
return return
write_pubkey(w, source_account) writers.write_pubkey(w, source_account)
def _serialize_asset_code(w, asset_type: int, asset_code: str): def _serialize_asset_code(w, asset_type: int, asset_code: str):
code = bytearray(asset_code) code = bytearray(asset_code)
if asset_type == ASSET_TYPE_NATIVE: if asset_type == consts.ASSET_TYPE_NATIVE:
return # nothing is needed return # nothing is needed
elif asset_type == ASSET_TYPE_ALPHANUM4: elif asset_type == consts.ASSET_TYPE_ALPHANUM4:
# pad with zeros to 4 chars # pad with zeros to 4 chars
write_bytes(w, code + bytearray([0] * (4 - len(code)))) writers.write_bytes(w, code + bytearray([0] * (4 - len(code))))
elif asset_type == ASSET_TYPE_ALPHANUM12: elif asset_type == consts.ASSET_TYPE_ALPHANUM12:
# pad with zeros to 12 chars # pad with zeros to 12 chars
write_bytes(w, code + bytearray([0] * (12 - len(code)))) writers.write_bytes(w, code + bytearray([0] * (12 - len(code))))
else: else:
raise ValueError('Stellar: invalid asset type') raise ValueError('Stellar: invalid asset type')
def _serialize_asset(w, asset: StellarAssetType): def _serialize_asset(w, asset: StellarAssetType):
if asset is None: if asset is None:
write_uint32(w, 0) writers.write_uint32(w, 0)
return return
write_uint32(w, asset.type) writers.write_uint32(w, asset.type)
_serialize_asset_code(w, asset.type, asset.code) _serialize_asset_code(w, asset.type, asset.code)
write_pubkey(w, asset.issuer) writers.write_pubkey(w, asset.issuer)

View File

@ -1,9 +1,8 @@
from apps.common import seed from apps.common import seed
from apps.stellar.writers import * from apps.stellar import writers
from apps.stellar.operations import operation from apps.stellar.operations import operation
from apps.stellar import layout from apps.stellar import layout
from apps.stellar import consts from apps.stellar import consts
from apps.stellar import helpers
from trezor.messages.StellarSignTx import StellarSignTx from trezor.messages.StellarSignTx import StellarSignTx
from trezor.messages.StellarTxOpRequest import StellarTxOpRequest from trezor.messages.StellarTxOpRequest import StellarTxOpRequest
from trezor.messages.StellarSignedTx import StellarSignedTx from trezor.messages.StellarSignedTx import StellarSignedTx
@ -16,73 +15,17 @@ async def sign_tx(ctx, msg: StellarSignTx):
if msg.num_operations == 0: if msg.num_operations == 0:
raise ValueError('Stellar: At least one operation is required') raise ValueError('Stellar: At least one operation is required')
network_passphrase_hash = sha256(msg.network_passphrase).digest()
# Stellar transactions consist of sha256 of:
# - sha256(network passphrase)
# - 4-byte unsigned big-endian int type constant (2 for tx)
# - public key
w = bytearray()
write_bytes(w, network_passphrase_hash)
write_bytes(w, consts.TX_TYPE)
node = await seed.derive_node(ctx, msg.address_n, consts.STELLAR_CURVE) node = await seed.derive_node(ctx, msg.address_n, consts.STELLAR_CURVE)
pubkey = seed.remove_ed25519_public_key_prefix(node.public_key()) pubkey = seed.remove_ed25519_public_key_prefix(node.public_key())
write_pubkey(w, pubkey)
if msg.source_account != pubkey:
raise ValueError('Stellar: source account does not match address_n')
# confirm init w = bytearray()
await layout.require_confirm_init(ctx, pubkey, msg.network_passphrase) await _init(ctx, w, pubkey, msg)
_timebounds(w, msg.timebounds_start, msg.timebounds_end)
write_uint32(w, msg.fee) await _memo(ctx, w, msg)
write_uint64(w, msg.sequence_number) await _operations(ctx, w, msg.num_operations)
await _final(ctx, w, msg)
# timebounds are only present if timebounds_start or timebounds_end is non-zero
if msg.timebounds_start or msg.timebounds_end:
write_bool(w, True)
# timebounds are sent as uint32s since that's all we can display, but they must be hashed as 64bit
write_uint64(w, msg.timebounds_start)
write_uint64(w, msg.timebounds_end)
else:
write_bool(w, False)
write_uint32(w, msg.memo_type)
if msg.memo_type == consts.MEMO_TYPE_NONE:
# nothing is serialized
memo_confirm_text = ''
elif msg.memo_type == consts.MEMO_TYPE_TEXT:
# Text: 4 bytes (size) + up to 28 bytes
if len(msg.memo_text) > 28:
raise ValueError('Stellar: max length of a memo text is 28 bytes')
write_string(w, msg.memo_text)
memo_confirm_text = msg.memo_text
elif msg.memo_type == consts.MEMO_TYPE_ID:
# ID: 64 bit unsigned integer
write_uint64(w, msg.memo_id)
memo_confirm_text = str(msg.memo_id)
elif msg.memo_type in [consts.MEMO_TYPE_HASH, consts.MEMO_TYPE_RETURN]:
# Hash/Return: 32 byte hash
write_bytes(w, bytearray(msg.memo_hash))
memo_confirm_text = hexlify(msg.memo_hash).decode()
else:
raise ValueError('Stellar invalid memo type')
await layout.require_confirm_memo(ctx, msg.memo_type, memo_confirm_text)
write_uint32(w, msg.num_operations)
for i in range(msg.num_operations):
op = await ctx.call(StellarTxOpRequest(), *consts.op_wire_types)
await operation(ctx, w, op)
# 4 null bytes representing a (currently unused) empty union
write_uint32(w, 0)
# final confirm
await layout.require_confirm_final(ctx, msg.fee, msg.num_operations)
# sign # sign
# (note that the signature does not include the 4-byte hint since it can be calculated from the public key)
digest = sha256(w).digest() digest = sha256(w).digest()
signature = ed25519.sign(node.private_key(), digest) signature = ed25519.sign(node.private_key(), digest)
@ -90,6 +33,70 @@ async def sign_tx(ctx, msg: StellarSignTx):
return StellarSignedTx(pubkey, signature) return StellarSignedTx(pubkey, signature)
async def _final(ctx, w: bytearray, msg: StellarSignTx):
# 4 null bytes representing a (currently unused) empty union
writers.write_uint32(w, 0)
# final confirm
await layout.require_confirm_final(ctx, msg.fee, msg.num_operations)
async def _init(ctx, w: bytearray, pubkey: bytes, msg: StellarSignTx):
network_passphrase_hash = sha256(msg.network_passphrase).digest()
writers.write_bytes(w, network_passphrase_hash)
writers.write_bytes(w, consts.TX_TYPE)
writers.write_pubkey(w, pubkey)
if msg.source_account != pubkey:
raise ValueError('Stellar: source account does not match address_n')
writers.write_uint32(w, msg.fee)
writers.write_uint64(w, msg.sequence_number)
# confirm init
await layout.require_confirm_init(ctx, pubkey, msg.network_passphrase)
def _timebounds(w: bytearray, start: int, end: int):
# timebounds are only present if timebounds_start or timebounds_end is non-zero
if start or end:
writers.write_bool(w, True)
# timebounds are sent as uint32s since that's all we can display, but they must be hashed as 64bit
writers.write_uint64(w, start)
writers.write_uint64(w, end)
else:
writers.write_bool(w, False)
async def _operations(ctx, w: bytearray, num_operations: int):
writers.write_uint32(w, num_operations)
for i in range(num_operations):
op = await ctx.call(StellarTxOpRequest(), *consts.op_wire_types)
await operation(ctx, w, op)
async def _memo(ctx, w: bytearray, msg: StellarSignTx):
writers.write_uint32(w, msg.memo_type)
if msg.memo_type == consts.MEMO_TYPE_NONE:
# nothing is serialized
memo_confirm_text = ''
elif msg.memo_type == consts.MEMO_TYPE_TEXT:
# Text: 4 bytes (size) + up to 28 bytes
if len(msg.memo_text) > 28:
raise ValueError('Stellar: max length of a memo text is 28 bytes')
writers.write_string(w, msg.memo_text)
memo_confirm_text = msg.memo_text
elif msg.memo_type == consts.MEMO_TYPE_ID:
# ID: 64 bit unsigned integer
writers.write_uint64(w, msg.memo_id)
memo_confirm_text = str(msg.memo_id)
elif msg.memo_type in (consts.MEMO_TYPE_HASH, consts.MEMO_TYPE_RETURN):
# Hash/Return: 32 byte hash
writers.write_bytes(w, bytearray(msg.memo_hash))
memo_confirm_text = hexlify(msg.memo_hash).decode()
else:
raise ValueError('Stellar invalid memo type')
await layout.require_confirm_memo(ctx, msg.memo_type, memo_confirm_text)
def node_derive(root, address_n: list): def node_derive(root, address_n: list):
node = root.clone() node = root.clone()
node.derive_path(address_n) node.derive_path(address_n)