electrum

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

commit 382f69edd9a27dd37744d4eac3d3c8164afc0ae4
parent 05342c553727707a25641b64dcb78f748cfd3e70
Author: ThomasV <thomasv@electrum.org>
Date:   Fri, 23 Mar 2018 19:48:03 +0100

Merge pull request #4094 from SomberNight/labels_plugin_catch_exc

labels plugin: better exception handling
Diffstat:
Mplugins/labels/labels.py | 89++++++++++++++++++++++++++++++++++++++++++++++---------------------------------
Mplugins/labels/qt.py | 21++++++++++++++++-----
2 files changed, 68 insertions(+), 42 deletions(-)

diff --git a/plugins/labels/labels.py b/plugins/labels/labels.py @@ -45,7 +45,7 @@ class LabelsPlugin(BasePlugin): @hook def set_label(self, wallet, item, label): - if not wallet in self.wallets: + if wallet not in self.wallets: return if not item: return @@ -55,7 +55,7 @@ class LabelsPlugin(BasePlugin): "walletNonce": nonce, "externalId": self.encode(wallet, item), "encryptedLabel": self.encode(wallet, label)} - t = threading.Thread(target=self.do_request, + t = threading.Thread(target=self.do_request_safe, args=["POST", "/label", False, bundle]) t.setDaemon(True) t.start() @@ -78,8 +78,18 @@ class LabelsPlugin(BasePlugin): raise BaseException(response["error"]) return response + def do_request_safe(self, *args, **kwargs): + try: + self.do_request(*args, **kwargs) + except BaseException as e: + #traceback.print_exc(file=sys.stderr) + self.print_error('error doing request') + def push_thread(self, wallet): - wallet_id = self.wallets[wallet][2] + wallet_data = self.wallets.get(wallet, None) + if not wallet_data: + raise Exception('Wallet {} not loaded'.format(wallet)) + wallet_id = wallet_data[2] bundle = {"labels": [], "walletId": wallet_id, "walletNonce": self.get_nonce(wallet)} @@ -95,42 +105,47 @@ class LabelsPlugin(BasePlugin): self.do_request("POST", "/labels", True, bundle) def pull_thread(self, wallet, force): - wallet_id = self.wallets[wallet][2] + wallet_data = self.wallets.get(wallet, None) + if not wallet_data: + raise Exception('Wallet {} not loaded'.format(wallet)) + wallet_id = wallet_data[2] nonce = 1 if force else self.get_nonce(wallet) - 1 self.print_error("asking for labels since nonce", nonce) + response = self.do_request("GET", ("/labels/since/%d/for/%s" % (nonce, wallet_id) )) + if response["labels"] is None: + self.print_error('no new labels') + return + result = {} + for label in response["labels"]: + try: + key = self.decode(wallet, label["externalId"]) + value = self.decode(wallet, label["encryptedLabel"]) + except: + continue + try: + json.dumps(key) + json.dumps(value) + except: + self.print_error('error: no json', key) + continue + result[key] = value + + for key, value in result.items(): + if force or not wallet.labels.get(key): + wallet.labels[key] = value + + self.print_error("received %d labels" % len(response)) + # do not write to disk because we're in a daemon thread + wallet.storage.put('labels', wallet.labels) + self.set_nonce(wallet, response["nonce"] + 1) + self.on_pulled(wallet) + + def pull_thread_safe(self, wallet, force): try: - response = self.do_request("GET", ("/labels/since/%d/for/%s" % (nonce, wallet_id) )) - if response["labels"] is None: - self.print_error('no new labels') - return - result = {} - for label in response["labels"]: - try: - key = self.decode(wallet, label["externalId"]) - value = self.decode(wallet, label["encryptedLabel"]) - except: - continue - try: - json.dumps(key) - json.dumps(value) - except: - self.print_error('error: no json', key) - continue - result[key] = value - - for key, value in result.items(): - if force or not wallet.labels.get(key): - wallet.labels[key] = value - - self.print_error("received %d labels" % len(response)) - # do not write to disk because we're in a daemon thread - wallet.storage.put('labels', wallet.labels) - self.set_nonce(wallet, response["nonce"] + 1) - self.on_pulled(wallet) - - except Exception as e: - traceback.print_exc(file=sys.stderr) - self.print_error("could not retrieve labels") + self.pull_thread(wallet, force) + except BaseException as e: + # traceback.print_exc(file=sys.stderr) + self.print_error('could not retrieve labels') def start_wallet(self, wallet): nonce = self.get_nonce(wallet) @@ -144,7 +159,7 @@ class LabelsPlugin(BasePlugin): wallet_id = hashlib.sha256(mpk).hexdigest() self.wallets[wallet] = (password, iv, wallet_id) # If there is an auth token we can try to actually start syncing - t = threading.Thread(target=self.pull_thread, args=(wallet, False)) + t = threading.Thread(target=self.pull_thread_safe, args=(wallet, False)) t.setDaemon(True) t.start() diff --git a/plugins/labels/qt.py b/plugins/labels/qt.py @@ -1,4 +1,6 @@ from functools import partial +import traceback +import sys from PyQt5.QtGui import * from PyQt5.QtCore import * @@ -37,10 +39,12 @@ class Plugin(LabelsPlugin): hbox.addWidget(QLabel("Label sync options:")) upload = ThreadedButton("Force upload", partial(self.push_thread, wallet), - partial(self.done_processing, d)) + partial(self.done_processing_success, d), + partial(self.done_processing_error, d)) download = ThreadedButton("Force download", partial(self.pull_thread, wallet, True), - partial(self.done_processing, d)) + partial(self.done_processing_success, d), + partial(self.done_processing_error, d)) vbox = QVBoxLayout() vbox.addWidget(upload) vbox.addWidget(download) @@ -54,13 +58,20 @@ class Plugin(LabelsPlugin): def on_pulled(self, wallet): self.obj.labels_changed_signal.emit(wallet) - def done_processing(self, dialog, result): + def done_processing_success(self, dialog, result): dialog.show_message(_("Your labels have been synchronised.")) + def done_processing_error(self, dialog, result): + traceback.print_exception(*result, file=sys.stderr) + dialog.show_error(_("Error synchronising labels") + ':\n' + str(result[:2])) + @hook - def on_new_window(self, window): + def load_wallet(self, wallet, window): + # FIXME if the user just enabled the plugin, this hook won't be called + # as the wallet is already loaded, and hence the plugin will be in + # a non-functional state for that window self.obj.labels_changed_signal.connect(window.update_tabs) - self.start_wallet(window.wallet) + self.start_wallet(wallet) @hook def on_close_window(self, window):