|
@ -10,7 +10,7 @@ from base64 import b64encode, b64decode |
|
|
from electrum_gui.qt.password_dialog import make_password_dialog, run_password_dialog |
|
|
from electrum_gui.qt.password_dialog import make_password_dialog, run_password_dialog |
|
|
from electrum_gui.qt.util import ok_cancel_buttons |
|
|
from electrum_gui.qt.util import ok_cancel_buttons |
|
|
from electrum.account import BIP32_Account |
|
|
from electrum.account import BIP32_Account |
|
|
from electrum.bitcoin import EC_KEY, EncodeBase58Check, DecodeBase58Check, public_key_to_bc_address, bc_address_to_hash_160 |
|
|
from electrum.bitcoin import EncodeBase58Check, DecodeBase58Check, public_key_to_bc_address, bc_address_to_hash_160 |
|
|
from electrum.i18n import _ |
|
|
from electrum.i18n import _ |
|
|
from electrum.plugins import BasePlugin |
|
|
from electrum.plugins import BasePlugin |
|
|
from electrum.transaction import deserialize |
|
|
from electrum.transaction import deserialize |
|
@ -137,10 +137,10 @@ class BTChipWallet(NewWallet): |
|
|
firmware = self.client.getFirmwareVersion()['version'].split(".") |
|
|
firmware = self.client.getFirmwareVersion()['version'].split(".") |
|
|
if int(firmware[0]) <> 1 or int(firmware[1]) <> 4: |
|
|
if int(firmware[0]) <> 1 or int(firmware[1]) <> 4: |
|
|
aborted = True |
|
|
aborted = True |
|
|
give_error("Unsupported firmware version") |
|
|
raise Exception("Unsupported firmware version") |
|
|
if int(firmware[2]) < 8: |
|
|
if int(firmware[2]) < 9: |
|
|
aborted = True |
|
|
aborted = True |
|
|
give_error("Please update your firmware - 1.4.8 or higher is necessary") |
|
|
raise Exception("Please update your firmware - 1.4.9 or higher is necessary") |
|
|
try: |
|
|
try: |
|
|
self.client.getOperationMode() |
|
|
self.client.getOperationMode() |
|
|
except BTChipException, e: |
|
|
except BTChipException, e: |
|
@ -156,16 +156,37 @@ class BTChipWallet(NewWallet): |
|
|
raise e |
|
|
raise e |
|
|
if not noPin: |
|
|
if not noPin: |
|
|
# Immediately prompts for the PIN |
|
|
# Immediately prompts for the PIN |
|
|
confirmed, p, pin = self.password_dialog("Enter your BTChip PIN") |
|
|
remaining_attempts = self.client.getVerifyPinRemainingAttempts() |
|
|
|
|
|
if remaining_attempts <> 1: |
|
|
|
|
|
msg = "Enter your BTChip PIN - remaining attempts : " + str(remaining_attempts) |
|
|
|
|
|
else: |
|
|
|
|
|
msg = "Enter your BTChip PIN - WARNING : LAST ATTEMPT. If the PIN is not correct, the dongle will be wiped." |
|
|
|
|
|
confirmed, p, pin = self.password_dialog(msg) |
|
|
if not confirmed: |
|
|
if not confirmed: |
|
|
aborted = True |
|
|
aborted = True |
|
|
give_error('Aborted by user') |
|
|
raise Exception('Aborted by user - please unplug the dongle and plug it again before retrying') |
|
|
pin = pin.encode() |
|
|
pin = pin.encode() |
|
|
self.client.verifyPin(pin) |
|
|
self.client.verifyPin(pin) |
|
|
|
|
|
|
|
|
|
|
|
except BTChipException, e: |
|
|
|
|
|
try: |
|
|
|
|
|
self.client.dongle.close() |
|
|
|
|
|
except: |
|
|
|
|
|
pass |
|
|
|
|
|
self.client = None |
|
|
|
|
|
if (e.sw == 0x6faa): |
|
|
|
|
|
raise Exception("Dongle is temporarily locked - please unplug it and replug it again") |
|
|
|
|
|
if ((e.sw & 0xFFF0) == 0x63c0): |
|
|
|
|
|
raise Exception("Invalid PIN - please unplug the dongle and plug it again before retrying") |
|
|
|
|
|
raise e |
|
|
except Exception, e: |
|
|
except Exception, e: |
|
|
|
|
|
try: |
|
|
|
|
|
self.client.dongle.close() |
|
|
|
|
|
except: |
|
|
|
|
|
pass |
|
|
self.client = None |
|
|
self.client = None |
|
|
if not aborted: |
|
|
if not aborted: |
|
|
give_error("Could not connect to your BTChip dongle. Please verify access permissions or PIN") |
|
|
raise Exception("Could not connect to your BTChip dongle. Please verify access permissions or PIN") |
|
|
else: |
|
|
else: |
|
|
raise e |
|
|
raise e |
|
|
self.client.bad = False |
|
|
self.client.bad = False |
|
@ -232,34 +253,31 @@ class BTChipWallet(NewWallet): |
|
|
give_error("Not supported") |
|
|
give_error("Not supported") |
|
|
|
|
|
|
|
|
def sign_message(self, address, message, password): |
|
|
def sign_message(self, address, message, password): |
|
|
|
|
|
use2FA = False |
|
|
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 |
|
|
if not self.check_proper_device(): |
|
|
if not self.check_proper_device(): |
|
|
give_error('Wrong device or password') |
|
|
give_error('Wrong device or password') |
|
|
address_path = self.address_id(address) |
|
|
address_path = self.address_id(address) |
|
|
waitDialog.start("Signing Message ...") |
|
|
waitDialog.start("Signing Message ...") |
|
|
aborted = False |
|
|
|
|
|
try: |
|
|
try: |
|
|
info = self.get_client().signMessagePrepare(address_path, message) |
|
|
info = self.get_client().signMessagePrepare(address_path, message) |
|
|
pin = "" |
|
|
pin = "" |
|
|
if info['confirmationNeeded']: |
|
|
if info['confirmationNeeded']: |
|
|
# TODO : handle different confirmation types. For the time being only supports keyboard 2FA |
|
|
# TODO : handle different confirmation types. For the time being only supports keyboard 2FA |
|
|
|
|
|
use2FA = True |
|
|
confirmed, p, pin = self.password_dialog() |
|
|
confirmed, p, pin = self.password_dialog() |
|
|
if not confirmed: |
|
|
if not confirmed: |
|
|
aborted = True |
|
|
raise Exception('Aborted by user') |
|
|
give_error('Aborted by user') |
|
|
|
|
|
pin = pin.encode() |
|
|
pin = pin.encode() |
|
|
self.client.bad = True |
|
|
self.client.bad = True |
|
|
self.get_client(True) |
|
|
self.get_client(True) |
|
|
signature = self.get_client().signMessageSign(pin) |
|
|
signature = self.get_client().signMessageSign(pin) |
|
|
except Exception, e: |
|
|
except Exception, e: |
|
|
if not aborted: |
|
|
|
|
|
give_error(e) |
|
|
give_error(e) |
|
|
else: |
|
|
|
|
|
raise e |
|
|
|
|
|
finally: |
|
|
finally: |
|
|
if waitDialog.waiting: |
|
|
if waitDialog.waiting: |
|
|
waitDialog.emit(SIGNAL('dongle_done')) |
|
|
waitDialog.emit(SIGNAL('dongle_done')) |
|
|
self.client.bad = True |
|
|
self.client.bad = use2FA |
|
|
|
|
|
|
|
|
# Parse the ASN.1 signature |
|
|
# Parse the ASN.1 signature |
|
|
|
|
|
|
|
@ -276,21 +294,8 @@ class BTChipWallet(NewWallet): |
|
|
|
|
|
|
|
|
# And convert it |
|
|
# And convert it |
|
|
|
|
|
|
|
|
#Optimization for 1.4.9+ |
|
|
|
|
|
#return b64encode(chr(27 + 4 + (signature[0] & 0x01)) + r + s) |
|
|
|
|
|
|
|
|
|
|
|
for i in range(4): |
|
|
|
|
|
sig = b64encode( chr(27 + i + 4) + r + s) |
|
|
|
|
|
try: |
|
|
|
|
|
EC_KEY.verify_message(address, sig, message) |
|
|
|
|
|
return sig |
|
|
|
|
|
except Exception: |
|
|
|
|
|
continue |
|
|
|
|
|
else: |
|
|
|
|
|
raise Exception("error: cannot sign message") |
|
|
|
|
|
return b64encode(chr(27 + 4 + (signature[0] & 0x01)) + r + s) |
|
|
return b64encode(chr(27 + 4 + (signature[0] & 0x01)) + r + s) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def choose_tx_inputs( self, amount, fixed_fee, num_outputs, domain = None, coins = None ): |
|
|
def choose_tx_inputs( self, amount, fixed_fee, num_outputs, domain = None, coins = None ): |
|
|
# Overloaded to get the fee, as BTChip recomputes the change amount |
|
|
# Overloaded to get the fee, as BTChip recomputes the change amount |
|
|
inputs, total, fee = super(BTChipWallet, self).choose_tx_inputs(amount, fixed_fee, num_outputs, domain, coins) |
|
|
inputs, total, fee = super(BTChipWallet, self).choose_tx_inputs(amount, fixed_fee, num_outputs, domain, coins) |
|
@ -312,7 +317,6 @@ class BTChipWallet(NewWallet): |
|
|
output = None |
|
|
output = None |
|
|
outputAmount = None |
|
|
outputAmount = None |
|
|
use2FA = False |
|
|
use2FA = False |
|
|
aborted = False |
|
|
|
|
|
pin = "" |
|
|
pin = "" |
|
|
# Fetch inputs of the transaction to sign |
|
|
# Fetch inputs of the transaction to sign |
|
|
for txinput in tx.inputs: |
|
|
for txinput in tx.inputs: |
|
@ -366,8 +370,7 @@ class BTChipWallet(NewWallet): |
|
|
waitDialog.emit(SIGNAL('dongle_done')) |
|
|
waitDialog.emit(SIGNAL('dongle_done')) |
|
|
confirmed, p, pin = self.password_dialog() |
|
|
confirmed, p, pin = self.password_dialog() |
|
|
if not confirmed: |
|
|
if not confirmed: |
|
|
aborted = True |
|
|
raise Exception('Aborted by user') |
|
|
give_error('Aborted by user') |
|
|
|
|
|
pin = pin.encode() |
|
|
pin = pin.encode() |
|
|
self.client.bad = True |
|
|
self.client.bad = True |
|
|
self.get_client(True) |
|
|
self.get_client(True) |
|
@ -381,10 +384,7 @@ class BTChipWallet(NewWallet): |
|
|
inputIndex = inputIndex + 1 |
|
|
inputIndex = inputIndex + 1 |
|
|
firstTransaction = False |
|
|
firstTransaction = False |
|
|
except Exception, e: |
|
|
except Exception, e: |
|
|
if not aborted: |
|
|
|
|
|
give_error(e) |
|
|
give_error(e) |
|
|
else: |
|
|
|
|
|
raise e |
|
|
|
|
|
finally: |
|
|
finally: |
|
|
if waitDialog.waiting: |
|
|
if waitDialog.waiting: |
|
|
waitDialog.emit(SIGNAL('dongle_done')) |
|
|
waitDialog.emit(SIGNAL('dongle_done')) |
|
|