electrum

Electrum Bitcoin wallet
git clone https://git.parazyd.org/electrum
Log | Files | Refs | Submodules

commit e6e587b7da58a4fb3da451fef4af8fddd8715d01
parent 2adbbee5fe2da2bcf753a4ef27ddb4a86cf6dfae
Author: SomberNight <somber.night@protonmail.com>
Date:   Fri, 29 May 2020 18:58:30 +0200

trustedcoin: use psbt format on wire. rm psbt to legacy tx conversion.

closes #6123

Diffstat:
Delectrum/plugins/trustedcoin/legacy_tx_format.py | 106-------------------------------------------------------------------------------
Melectrum/plugins/trustedcoin/trustedcoin.py | 5++---
Melectrum/tests/test_transaction.py | 63+--------------------------------------------------------------
3 files changed, 3 insertions(+), 171 deletions(-)

diff --git a/electrum/plugins/trustedcoin/legacy_tx_format.py b/electrum/plugins/trustedcoin/legacy_tx_format.py @@ -1,106 +0,0 @@ -# Copyright (C) 2018 The Electrum developers -# Distributed under the MIT software license, see the accompanying -# file LICENCE or http://www.opensource.org/licenses/mit-license.php - -import copy -from typing import Union - -from electrum import bitcoin -from electrum.bitcoin import push_script, int_to_hex, var_int -from electrum.transaction import (Transaction, PartialTransaction, PartialTxInput, - multisig_script, construct_witness) -from electrum.keystore import BIP32_KeyStore -from electrum.wallet import Multisig_Wallet - - -ELECTRUM_PARTIAL_TXN_HEADER_MAGIC = b'EPTF\xff' -PARTIAL_FORMAT_VERSION = b'\x00' -NO_SIGNATURE = b'\xff' - - -def get_xpubkey(keystore: BIP32_KeyStore, c, i) -> str: - def encode_path_int(path_int) -> str: - if path_int < 0xffff: - hex = bitcoin.int_to_hex(path_int, 2) - else: - hex = 'ffff' + bitcoin.int_to_hex(path_int, 4) - return hex - - s = ''.join(map(encode_path_int, (c, i))) - return 'ff' + bitcoin.DecodeBase58Check(keystore.xpub).hex() + s - - -def serialize_tx_in_legacy_format(tx: PartialTransaction, *, wallet: Multisig_Wallet) -> str: - assert isinstance(tx, PartialTransaction) - - # copy tx so we don't mutate the input arg - # monkey-patch method of tx instance to change serialization - tx = copy.deepcopy(tx) - - def get_siglist(txin: 'PartialTxInput', *, estimate_size=False): - if txin.is_coinbase_input(): - return [], [] - if estimate_size: - try: - pubkey_size = len(txin.pubkeys[0]) - except IndexError: - pubkey_size = 33 # guess it is compressed - num_pubkeys = max(1, len(txin.pubkeys)) - pk_list = ["00" * pubkey_size] * num_pubkeys - # we assume that signature will be 0x48 bytes long - num_sig = max(txin.num_sig, num_pubkeys) - sig_list = [ "00" * 0x48 ] * num_sig - else: - pk_list = ["" for pk in txin.pubkeys] - for ks in wallet.get_keystores(): - my_pubkey, full_path = ks.find_my_pubkey_in_txinout(txin) - x_pubkey = get_xpubkey(ks, full_path[-2], full_path[-1]) - pubkey_index = txin.pubkeys.index(my_pubkey) - pk_list[pubkey_index] = x_pubkey - assert all(pk_list) - sig_list = [txin.part_sigs.get(pubkey, NO_SIGNATURE).hex() for pubkey in txin.pubkeys] - return pk_list, sig_list - - def input_script(self, txin: PartialTxInput, *, estimate_size=False) -> str: - assert estimate_size is False - pubkeys, sig_list = get_siglist(txin, estimate_size=estimate_size) - script = ''.join(push_script(x) for x in sig_list) - if txin.script_type == 'p2sh': - # put op_0 before script - script = '00' + script - redeem_script = multisig_script(pubkeys, txin.num_sig) - script += push_script(redeem_script) - return script - elif txin.script_type == 'p2wsh': - return '' - raise Exception(f"unexpected type {txin.script_type}") - tx.input_script = input_script.__get__(tx, PartialTransaction) - - def serialize_witness(self, txin: PartialTxInput, *, estimate_size=False): - assert estimate_size is False - if txin.witness is not None: - return txin.witness.hex() - if txin.is_coinbase_input(): - return '' - assert isinstance(txin, PartialTxInput) - if not self.is_segwit_input(txin): - return '00' - pubkeys, sig_list = get_siglist(txin, estimate_size=estimate_size) - if txin.script_type == 'p2wsh': - witness_script = multisig_script(pubkeys, txin.num_sig) - witness = construct_witness([0] + sig_list + [witness_script]) - else: - raise Exception(f"unexpected type {txin.script_type}") - if txin.is_complete() or estimate_size: - partial_format_witness_prefix = '' - else: - input_value = int_to_hex(txin.value_sats(), 8) - witness_version = int_to_hex(0, 2) - partial_format_witness_prefix = var_int(0xffffffff) + input_value + witness_version - return partial_format_witness_prefix + witness - tx.serialize_witness = serialize_witness.__get__(tx, PartialTransaction) - - buf = ELECTRUM_PARTIAL_TXN_HEADER_MAGIC.hex() - buf += PARTIAL_FORMAT_VERSION.hex() - buf += tx.serialize_to_network() - return buf diff --git a/electrum/plugins/trustedcoin/trustedcoin.py b/electrum/plugins/trustedcoin/trustedcoin.py @@ -49,8 +49,6 @@ from electrum.network import Network from electrum.base_wizard import BaseWizard, WizardWalletPasswordSetting from electrum.logging import Logger -from .legacy_tx_format import serialize_tx_in_legacy_format - def get_signing_xpub(xtype): if not constants.net.TESTNET: @@ -345,7 +343,8 @@ class Wallet_2fa(Multisig_Wallet): return otp = int(otp) long_user_id, short_id = self.get_user_id() - raw_tx = serialize_tx_in_legacy_format(tx, wallet=self) + raw_tx = tx.serialize_as_bytes().hex() + assert raw_tx[:10] == "70736274ff", f"bad magic. {raw_tx[:10]}" try: r = server.sign(short_id, raw_tx, otp) except TrustedCoinException as e: diff --git a/electrum/tests/test_transaction.py b/electrum/tests/test_transaction.py @@ -3,16 +3,8 @@ from typing import NamedTuple, Union from electrum import transaction, bitcoin from electrum.transaction import convert_raw_tx_to_hex, tx_from_any, Transaction, PartialTransaction from electrum.util import bh2u, bfh -from electrum import keystore -from electrum import bip32 -from electrum.mnemonic import seed_type -from electrum.simple_config import SimpleConfig - -from electrum.plugins.trustedcoin import trustedcoin -from electrum.plugins.trustedcoin.legacy_tx_format import serialize_tx_in_legacy_format - -from . import ElectrumTestCase, TestCaseForTestnet +from . import ElectrumTestCase signed_blob = '01000000012a5c9a94fcde98f5581cd00162c60a13936ceb75389ea65bf38633b424eb4031000000006c493046022100a82bbc57a0136751e5433f41cf000b3f1a99c6744775e76ec764fb78c54ee100022100f9e80b7de89de861dc6fb0c1429d5da72c2b6b2ee2406bc9bfb1beedd729d985012102e61d176da16edd1d258a200ad9759ef63adf8e14cd97f53227bae35cdb84d2f6ffffffff0140420f00000000001976a914230ac37834073a42146f11ef8414ae929feaafc388ac00000000' v2_blob = "0200000001191601a44a81e061502b7bfbc6eaa1cef6d1e6af5308ef96c9342f71dbf4b9b5000000006b483045022100a6d44d0a651790a477e75334adfb8aae94d6612d01187b2c02526e340a7fd6c8022028bdf7a64a54906b13b145cd5dab21a26bd4b85d6044e9b97bceab5be44c2a9201210253e8e0254b0c95776786e40984c1aa32a7d03efa6bdacdea5f421b774917d346feffffff026b20fa04000000001976a914024db2e87dd7cfd0e5f266c5f212e21a31d805a588aca0860100000000001976a91421919b94ae5cefcdf0271191459157cdb41c4cbf88aca6240700" @@ -848,56 +840,3 @@ class TestTransaction(ElectrumTestCase): self._run_naive_tests_on_tx(raw_tx, txid) # txns from Bitcoin Core ends <--- - - -class TestLegacyPartialTxFormat(TestCaseForTestnet): - - def setUp(self): - super().setUp() - self.config = SimpleConfig({'electrum_path': self.electrum_path}) - - def test_trustedcoin_legacy_2fa_psbt_to_legacy_partial_tx(self): - from .test_wallet_vertical import WalletIntegrityHelper - seed_words = 'kiss live scene rude gate step hip quarter bunker oxygen motor glove' - self.assertEqual(seed_type(seed_words), '2fa') - - xprv1, xpub1, xprv2, xpub2 = trustedcoin.TrustedCoinPlugin.xkeys_from_seed(seed_words, '') - ks1 = keystore.from_xprv(xprv1) - ks2 = keystore.from_xprv(xprv2) - long_user_id, short_id = trustedcoin.get_user_id( - {'x1/': {'xpub': xpub1}, - 'x2/': {'xpub': xpub2}}) - xtype = bip32.xpub_type(xpub1) - xpub3 = trustedcoin.make_xpub(trustedcoin.get_signing_xpub(xtype), long_user_id) - ks3 = keystore.from_xpub(xpub3) - - wallet = WalletIntegrityHelper.create_multisig_wallet([ks1, ks2, ks3], '2of3', config=self.config) - - tx = tx_from_any('cHNidP8BAJQCAAAAAcqqxrXrkW4wZ9AiT5QvszHOHc+0Axz7R555Qdz5XkCYAQAAAAD9////A6CGAQAAAAAAFgAU+fBLRlKk9v89xVEm2xJ0kG1wcvNMCwMAAAAAABepFPKffLiXEB3Gmv1Y35uy5bTUM59Nh0ANAwAAAAAAGXapFPriyJZefiOenIisUU3nDewLDxYIiKwSKxgATwEENYfPAAAAAAAAAAAAnOMnCVq57ruCJ7c38H6PtmrwS48+kcQJPEh70w/ofCQCDSEN062A0pw2JKkYltX2G3th8zLexPfEVDGu74BeD6cEcH3xxE8BBDWHzwGCB4l2gAAAAJOfYJjOAH6kksFOokIboP3+8Gwhlzlxhl5uY7zokvfcAmGy8e8txy0wkx69/TgZFOMe1aZc2g1HCwrRQ9M9+Ph7CIIHiXYAAACATwEENYfPAYIHiXaAAAABb6EovcClpG/Hrxr9IF22IHGR1MQFG27b0GQTzcCxot8Dak5MvnvEZt1lN4TIazd0m+w+goApzqNMFWkJVv1hV28IggeJdgEAAIAAAQDfAgAAAAGcKHw7enlMh6IibIkEeKQlL5pUR2wKv6GC1NTd6KY8ggEAAABqRzBEAiBNHsG9H5z10eHHsIOe4kFdvnZK38E7Jx+Cmru14SdQ/gIgWngNYj/F8qHAhkdlU+BgY5ktAL2MeIUoqIJKXudcFRMBIQNU856KX8nmKx8+nbIRwjpRAvyMWroJGz+F6ADwzYv/GP3///8CnJ0HAAAAAAAZdqkUsQVAb+BbDci+RMeDa6WBLb9nTOiIrCChBwAAAAAAF6kULBYX0k+TbkDRSw3ylOy3u6rXUzeHEisYACICA/5C2rWHGOoEE/fI3mk83u4izhmx3DTAu916SCRUZcWiSDBFAiEA0Dw1yyk7Adp74Ndxztr6iR7V1wpnfPzNaWcTVva+vtwCIAsqV2xM0cZCSAdWzh/WYKyvC6UmGTowmeH4HN0BrSCTAQEEaVIhAgkfC02KswAWpdHAiCSeAog/rYFg8G+lNYithZhlCj5iIQNfL4JjuzYI1sxO4DvUy41lxNcK9xBJ8F+/7kl4gyof0iED/kLatYcY6gQT98jeaTze7iLOGbHcNMC73XpIJFRlxaJTriIGA/5C2rWHGOoEE/fI3mk83u4izhmx3DTAu916SCRUZcWiEIIHiXYAAACAAAAAAAAAAAAiBgIJHwtNirMAFqXRwIgkngKIP62BYPBvpTWIrYWYZQo+YhCCB4l2AQAAgAAAAAAAAAAAIgYDXy+CY7s2CNbMTuA71MuNZcTXCvcQSfBfv+5JeIMqH9IMcH3xxAAAAAAAAAAAAAABAGlSIQIqtnn9ouM3xAq7wIID09cdKpb9u/OMkI97kuU3wcTv/yEDTNFHFnZ1xmKGRQzFaUAT9DeDk2NdeWSrilc8w9BKjU4hA1joWoIBRYfeqDPrX/uT45hWkO5Lph7zLVsorVqhYXN/U64iAgNM0UcWdnXGYoZFDMVpQBP0N4OTY115ZKuKVzzD0EqNThCCB4l2AAAAgAEAAAAAAAAAIgICKrZ5/aLjN8QKu8CCA9PXHSqW/bvzjJCPe5LlN8HE7/8QggeJdgEAAIABAAAAAAAAACICA1joWoIBRYfeqDPrX/uT45hWkO5Lph7zLVsorVqhYXN/DHB98cQBAAAAAAAAAAAA') - tx.add_info_from_wallet(wallet) - raw_tx = serialize_tx_in_legacy_format(tx, wallet=wallet) - self.assertEqual('45505446ff000200000001caaac6b5eb916e3067d0224f942fb331ce1dcfb4031cfb479e7941dcf95e409801000000fd53010001ff01ff483045022100d03c35cb293b01da7be0d771cedafa891ed5d70a677cfccd69671356f6bebedc02200b2a576c4cd1c642480756ce1fd660acaf0ba526193a3099e1f81cdd01ad2093014d0201524c53ff043587cf0182078976800000016fa128bdc0a5a46fc7af1afd205db6207191d4c4051b6edbd06413cdc0b1a2df036a4e4cbe7bc466dd653784c86b37749bec3e828029cea34c15690956fd61576f000000004c53ff043587cf0000000000000000009ce327095ab9eebb8227b737f07e8fb66af04b8f3e91c4093c487bd30fe87c24020d210dd3ad80d29c3624a91896d5f61b7b61f332dec4f7c45431aeef805e0fa7000000004c53ff043587cf018207897680000000939f6098ce007ea492c14ea2421ba0fdfef06c21973971865e6e63bce892f7dc0261b2f1ef2dc72d30931ebdfd381914e31ed5a65cda0d470b0ad143d33df8f87b0000000053aefdffffff03a086010000000000160014f9f04b4652a4f6ff3dc55126db1274906d7072f34c0b03000000000017a914f29f7cb897101dc69afd58df9bb2e5b4d4339f4d87400d0300000000001976a914fae2c8965e7e239e9c88ac514de70dec0b0f160888ac122b1800', - raw_tx) - - def test_trustedcoin_segwit_2fa_psbt_to_legacy_partial_tx(self): - from .test_wallet_vertical import WalletIntegrityHelper - seed_words = 'universe topic remind silver february ranch shine worth innocent cattle enhance wise' - self.assertEqual(seed_type(seed_words), '2fa_segwit') - - xprv1, xpub1, xprv2, xpub2 = trustedcoin.TrustedCoinPlugin.xkeys_from_seed(seed_words, '') - ks1 = keystore.from_xprv(xprv1) - ks2 = keystore.from_xprv(xprv2) - long_user_id, short_id = trustedcoin.get_user_id( - {'x1/': {'xpub': xpub1}, - 'x2/': {'xpub': xpub2}}) - xtype = bip32.xpub_type(xpub1) - xpub3 = trustedcoin.make_xpub(trustedcoin.get_signing_xpub(xtype), long_user_id) - ks3 = keystore.from_xpub(xpub3) - - wallet = WalletIntegrityHelper.create_multisig_wallet([ks1, ks2, ks3], '2of3', config=self.config) - - tx = tx_from_any('70736274ff01009f020000000187c4646ca690b397e357b23b2137030691a90a068a6690834d340b4be84acd6a0100000000fdffffff03a0860100000000001600148bc9d947e4d0addc2f4c34b8371034eb47b3d305140c030000000000220020a6087c4f84a55dc39014729a18e08955139d4384559d1fd2a48a1d95d746a425400d0300000000001976a914c13fd6294d1be7b9410a5538f4b4ef10fc594ee788ac132b18004f0102575483000000000000000000d644dcdd7a4acb432355e010ac4a5f955940c82a9767731b7d89dc42eb3cdac40272faa2f98b76cc9d655ef281c07acda75c30a990dc3a527b6cf92a8f5a6b8a80044bf212094f010257548301d24cd5918000000199c4d6c893fa1addec6b0121181137e25c38194333b8c57215b3cf1e6d7e7af102a23ddae698bb6095b8b8035bdd1e94479f66c29679d81931e0fe07aa436149ee08d24cd591010000804f010257548301d24cd59180000000b60a85089ad0850a6ffe8590a3e15064e63eefd020813df2e4a6b3209ce3df5f02c8b14f2917d557cd482970c114c3e3457e09d256397802277a2e4d1519ab9f8c08d24cd591000000800001012b20a1070000000000220020a948d7fa6abbb97e31779ae54383012b413d53821c7fd394900f6b443c61deee22020307a3c41d07ed976d65e213e823d02840937475e709b41253e85e970e3cb1667447304402202f7be5fad398f1a3576293339f2274d227af17798690aa1ad00ff83d96725cc5022000cb2e9fff5be23a862718c665a5035f172783e9b3158eae83673f6c59adc3c001010569522102a9dcb570e8280c741f09032c158095b7aa3b0ce401ada030f2d47b999f020606210307a3c41d07ed976d65e213e823d02840937475e709b41253e85e970e3cb166742103521b0a45e042f08ccd03af47fd88bb207b5414e0e30bb8799fca311a06323a1953ae22060307a3c41d07ed976d65e213e823d02840937475e709b41253e85e970e3cb1667410d24cd591000000800000000001000000220603521b0a45e042f08ccd03af47fd88bb207b5414e0e30bb8799fca311a06323a1910d24cd591010000800000000001000000220602a9dcb570e8280c741f09032c158095b7aa3b0ce401ada030f2d47b999f0206060c4bf212090000000001000000000001016952210205829f9522577122ca9ca9beb67f94cf2fe0ad0d17e052a110701cf4128d339c21022cca282cfdd9cc1f387d3098661d68bc9c8a39ac6bd72c30db579a7857d0859b2102d90e3c8973844b9cc0b38494c48515a319eec3ac2489f96f6f55e6aa6912a60853ae220202d90e3c8973844b9cc0b38494c48515a319eec3ac2489f96f6f55e6aa6912a60810d24cd5910000008001000000000000002202022cca282cfdd9cc1f387d3098661d68bc9c8a39ac6bd72c30db579a7857d0859b10d24cd59101000080010000000000000022020205829f9522577122ca9ca9beb67f94cf2fe0ad0d17e052a110701cf4128d339c0c4bf2120901000000000000000000') - tx.add_info_from_wallet(wallet) - raw_tx = serialize_tx_in_legacy_format(tx, wallet=wallet) - self.assertEqual('45505446ff000200000000010187c4646ca690b397e357b23b2137030691a90a068a6690834d340b4be84acd6a0100000000fdffffff03a0860100000000001600148bc9d947e4d0addc2f4c34b8371034eb47b3d305140c030000000000220020a6087c4f84a55dc39014729a18e08955139d4384559d1fd2a48a1d95d746a425400d0300000000001976a914c13fd6294d1be7b9410a5538f4b4ef10fc594ee788acfeffffffff20a10700000000000000050001ff47304402202f7be5fad398f1a3576293339f2274d227af17798690aa1ad00ff83d96725cc5022000cb2e9fff5be23a862718c665a5035f172783e9b3158eae83673f6c59adc3c00101fffd0201524c53ff02575483000000000000000000d644dcdd7a4acb432355e010ac4a5f955940c82a9767731b7d89dc42eb3cdac40272faa2f98b76cc9d655ef281c07acda75c30a990dc3a527b6cf92a8f5a6b8a80000001004c53ff0257548301d24cd59180000000b60a85089ad0850a6ffe8590a3e15064e63eefd020813df2e4a6b3209ce3df5f02c8b14f2917d557cd482970c114c3e3457e09d256397802277a2e4d1519ab9f8c000001004c53ff0257548301d24cd5918000000199c4d6c893fa1addec6b0121181137e25c38194333b8c57215b3cf1e6d7e7af102a23ddae698bb6095b8b8035bdd1e94479f66c29679d81931e0fe07aa436149ee0000010053ae132b1800', - raw_tx)