Browse Source

Segwit support

2.9.x
BTChip 8 years ago
parent
commit
a610e21279
  1. 116
      plugins/ledger/ledger.py

116
plugins/ledger/ledger.py

@ -11,6 +11,7 @@ from electrum.bitcoin import TYPE_ADDRESS, int_to_hex, var_int
from electrum.i18n import _ from electrum.i18n import _
from electrum.plugins import BasePlugin, hook from electrum.plugins import BasePlugin, hook
from electrum.keystore import Hardware_KeyStore, parse_xpubkey from electrum.keystore import Hardware_KeyStore, parse_xpubkey
from electrum.transaction import push_script
from ..hw_wallet import HW_PluginBase from ..hw_wallet import HW_PluginBase
from electrum.util import format_satoshis_plain, print_error, is_verbose from electrum.util import format_satoshis_plain, print_error, is_verbose
@ -172,6 +173,9 @@ class Ledger_KeyStore(Hardware_KeyStore):
self.signing = False self.signing = False
self.cfg = d.get('cfg', {'mode':0,'pair':''}) self.cfg = d.get('cfg', {'mode':0,'pair':''})
def is_segwit(self):
return self.plugin.segwit
def dump(self): def dump(self):
obj = Hardware_KeyStore.dump(self) obj = Hardware_KeyStore.dump(self)
obj['cfg'] = self.cfg obj['cfg'] = self.cfg
@ -268,6 +272,7 @@ class Ledger_KeyStore(Hardware_KeyStore):
output = None output = None
outputAmount = None outputAmount = None
p2shTransaction = False p2shTransaction = False
segwitTransaction = True
reorganize = False reorganize = False
pin = "" pin = ""
self.get_client() # prompt for the PIN before displaying the dialog if necessary self.get_client() # prompt for the PIN before displaying the dialog if necessary
@ -281,6 +286,9 @@ class Ledger_KeyStore(Hardware_KeyStore):
if txin['type'] in ['p2sh']: if txin['type'] in ['p2sh']:
p2shTransaction = True p2shTransaction = True
if txin['type'] in ['p2wpkh-p2sh']:
segwitTransaction = True
pubkeys, x_pubkeys = tx.get_sorted_pubkeys(txin) pubkeys, x_pubkeys = tx.get_sorted_pubkeys(txin)
for i, x_pubkey in enumerate(x_pubkeys): for i, x_pubkey in enumerate(x_pubkeys):
if x_pubkey in derivations: if x_pubkey in derivations:
@ -291,7 +299,12 @@ class Ledger_KeyStore(Hardware_KeyStore):
else: else:
self.give_error("No matching x_key for sign_transaction") # should never happen self.give_error("No matching x_key for sign_transaction") # should never happen
inputs.append([txin['prev_tx'].raw, txin['prevout_n'], txin.get('redeemScript'), txin['prevout_hash'], signingPos ]) redeemScript = txin.get('redeemScript')
if segwitTransaction and not redeemScript:
pkh = bitcoin.hash_160(pubkeys[0].decode('hex')).encode('hex')
redeemScript = '76a9' + push_script(pkh) + '88ac'
inputs.append([txin['prev_tx'].raw, txin['prevout_n'], redeemScript, txin['prevout_hash'], signingPos ])
inputsPaths.append(hwAddress) inputsPaths.append(hwAddress)
pubKeys.append(pubkeys) pubKeys.append(pubkeys)
@ -330,7 +343,14 @@ class Ledger_KeyStore(Hardware_KeyStore):
try: try:
# Get trusted inputs from the original transactions # Get trusted inputs from the original transactions
for utxo in inputs: for utxo in inputs:
if not p2shTransaction: if segwitTransaction:
txtmp = bitcoinTransaction(bytearray(utxo[0].decode('hex')))
tmp = utxo[3].decode('hex')[::-1].encode('hex')
tmp += int_to_hex(utxo[1], 4)
tmp += str(txtmp.outputs[utxo[1]].amount).encode('hex')
chipInputs.append({'value' : tmp.decode('hex'), 'witness' : True})
redeemScripts.append(bytearray(utxo[2].decode('hex')))
elif not p2shTransaction:
txtmp = bitcoinTransaction(bytearray(utxo[0].decode('hex'))) txtmp = bitcoinTransaction(bytearray(utxo[0].decode('hex')))
chipInputs.append(self.get_client().getTrustedInput(txtmp, utxo[1])) chipInputs.append(self.get_client().getTrustedInput(txtmp, utxo[1]))
redeemScripts.append(txtmp.outputs[utxo[1]].script) redeemScripts.append(txtmp.outputs[utxo[1]].script)
@ -345,19 +365,12 @@ class Ledger_KeyStore(Hardware_KeyStore):
inputIndex = 0 inputIndex = 0
rawTx = tx.serialize() rawTx = tx.serialize()
self.get_client().enableAlternate2fa(False) self.get_client().enableAlternate2fa(False)
while inputIndex < len(inputs): if segwitTransaction:
self.get_client().startUntrustedTransaction(firstTransaction, inputIndex, self.get_client().startUntrustedTransaction(True, inputIndex,
chipInputs, redeemScripts[inputIndex]) chipInputs, redeemScripts[inputIndex])
if not p2shTransaction: outputData = self.get_client().finalizeInputFull(txOutput)
outputData = self.get_client().finalizeInput(output, format_satoshis_plain(outputAmount), outputData['outputData'] = txOutput
format_satoshis_plain(tx.get_fee()), changePath, bytearray(rawTx.decode('hex'))) transactionOutput = outputData['outputData']
reorganize = True
else:
outputData = self.get_client().finalizeInputFull(txOutput)
outputData['outputData'] = txOutput
if firstTransaction:
transactionOutput = outputData['outputData']
if outputData['confirmationNeeded']: if outputData['confirmationNeeded']:
outputData['address'] = output outputData['address'] = output
self.handler.clear_dialog() self.handler.clear_dialog()
@ -366,14 +379,44 @@ class Ledger_KeyStore(Hardware_KeyStore):
raise UserWarning() raise UserWarning()
if pin != 'paired': if pin != 'paired':
self.handler.show_message(_("Confirmed. Signing Transaction...")) self.handler.show_message(_("Confirmed. Signing Transaction..."))
else: while inputIndex < len(inputs):
# Sign input with the provided PIN singleInput = [ chipInputs[inputIndex] ]
self.get_client().startUntrustedTransaction(False, 0,
singleInput, redeemScripts[inputIndex])
inputSignature = self.get_client().untrustedHashSign(inputsPaths[inputIndex], pin) inputSignature = self.get_client().untrustedHashSign(inputsPaths[inputIndex], pin)
inputSignature[0] = 0x30 # force for 1.4.9+ inputSignature[0] = 0x30 # force for 1.4.9+
signatures.append(inputSignature) signatures.append(inputSignature)
inputIndex = inputIndex + 1 inputIndex = inputIndex + 1
if pin != 'paired': else:
firstTransaction = False while inputIndex < len(inputs):
self.get_client().startUntrustedTransaction(firstTransaction, inputIndex,
chipInputs, redeemScripts[inputIndex])
if not p2shTransaction:
outputData = self.get_client().finalizeInput(output, format_satoshis_plain(outputAmount),
format_satoshis_plain(tx.get_fee()), changePath, bytearray(rawTx.decode('hex')))
reorganize = True
else:
outputData = self.get_client().finalizeInputFull(txOutput)
outputData['outputData'] = txOutput
if firstTransaction:
transactionOutput = outputData['outputData']
if outputData['confirmationNeeded']:
outputData['address'] = output
self.handler.clear_dialog()
pin = self.handler.get_auth( outputData ) # does the authenticate dialog and returns pin
if not pin:
raise UserWarning()
if pin != 'paired':
self.handler.show_message(_("Confirmed. Signing Transaction..."))
else:
# Sign input with the provided PIN
inputSignature = self.get_client().untrustedHashSign(inputsPaths[inputIndex], pin)
inputSignature[0] = 0x30 # force for 1.4.9+
signatures.append(inputSignature)
inputIndex = inputIndex + 1
if pin != 'paired':
firstTransaction = False
except UserWarning: except UserWarning:
self.handler.show_error(_('Cancelled by user')) self.handler.show_error(_('Cancelled by user'))
return return
@ -385,22 +428,27 @@ class Ledger_KeyStore(Hardware_KeyStore):
# Reformat transaction # Reformat transaction
inputIndex = 0 inputIndex = 0
while inputIndex < len(inputs): if segwitTransaction:
if p2shTransaction: for txin in tx.inputs():
signaturesPack = [signatures[inputIndex]] * len(pubKeys[inputIndex]) txin['signatures'] = [str(signatures[inputIndex][0:-1]).encode('hex')]
inputScript = get_p2sh_input_script(redeemScripts[inputIndex], signaturesPack) inputIndex = inputIndex + 1
preparedTrustedInputs.append([ ("\x00" * 4) + chipInputs[inputIndex]['value'], inputScript ])
else:
inputScript = get_regular_input_script(signatures[inputIndex], pubKeys[inputIndex][0].decode('hex'))
preparedTrustedInputs.append([ chipInputs[inputIndex]['value'], inputScript ])
inputIndex = inputIndex + 1
updatedTransaction = format_transaction(transactionOutput, preparedTrustedInputs)
updatedTransaction = hexlify(updatedTransaction)
if reorganize:
tx.update(updatedTransaction)
else: else:
tx.update_signatures(updatedTransaction) while inputIndex < len(inputs):
if p2shTransaction:
signaturesPack = [signatures[inputIndex]] * len(pubKeys[inputIndex])
inputScript = get_p2sh_input_script(redeemScripts[inputIndex], signaturesPack)
preparedTrustedInputs.append([ ("\x00" * 4) + chipInputs[inputIndex]['value'], inputScript ])
else:
inputScript = get_regular_input_script(signatures[inputIndex], pubKeys[inputIndex][0].decode('hex'))
preparedTrustedInputs.append([ chipInputs[inputIndex]['value'], inputScript ])
inputIndex = inputIndex + 1
updatedTransaction = format_transaction(transactionOutput, preparedTrustedInputs)
updatedTransaction = hexlify(updatedTransaction)
if reorganize:
tx.update(updatedTransaction)
else:
tx.update_signatures(updatedTransaction)
self.signing = False self.signing = False
@ -418,6 +466,7 @@ class LedgerPlugin(HW_PluginBase):
] ]
def __init__(self, parent, config, name): def __init__(self, parent, config, name):
self.segwit = config.get("segwit")
HW_PluginBase.__init__(self, parent, config, name) HW_PluginBase.__init__(self, parent, config, name)
if self.libraries_available: if self.libraries_available:
self.device_manager().register_devices(self.DEVICE_IDS) self.device_manager().register_devices(self.DEVICE_IDS)
@ -469,7 +518,6 @@ class LedgerPlugin(HW_PluginBase):
#assert self.main_thread != threading.current_thread() #assert self.main_thread != threading.current_thread()
devmgr = self.device_manager() devmgr = self.device_manager()
handler = keystore.handler handler = keystore.handler
handler = keystore.handler
with devmgr.hid_lock: with devmgr.hid_lock:
client = devmgr.client_for_keystore(self, handler, keystore, force_pair) client = devmgr.client_for_keystore(self, handler, keystore, force_pair)
# returns the client for a given keystore. can use xpub # returns the client for a given keystore. can use xpub

Loading…
Cancel
Save