|
@ -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 |
|
|