ThomasV
7 years ago
committed by
GitHub
4 changed files with 222 additions and 3 deletions
@ -0,0 +1,181 @@ |
|||
#!/usr/bin/env python |
|||
# |
|||
# Electrum - lightweight Bitcoin client |
|||
# |
|||
# Permission is hereby granted, free of charge, to any person |
|||
# obtaining a copy of this software and associated documentation files |
|||
# (the "Software"), to deal in the Software without restriction, |
|||
# including without limitation the rights to use, copy, modify, merge, |
|||
# publish, distribute, sublicense, and/or sell copies of the Software, |
|||
# and to permit persons to whom the Software is furnished to do so, |
|||
# subject to the following conditions: |
|||
# |
|||
# The above copyright notice and this permission notice shall be |
|||
# included in all copies or substantial portions of the Software. |
|||
# |
|||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
|||
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
|||
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND |
|||
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS |
|||
# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN |
|||
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN |
|||
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
|||
# SOFTWARE. |
|||
import json |
|||
import locale |
|||
import platform |
|||
import traceback |
|||
|
|||
import requests |
|||
from PyQt5.QtCore import QObject |
|||
import PyQt5.QtCore as QtCore |
|||
from PyQt5.QtGui import QIcon |
|||
from PyQt5.QtWidgets import * |
|||
|
|||
from electrum.i18n import _ |
|||
import sys |
|||
from electrum import ELECTRUM_VERSION |
|||
|
|||
issue_template = """<h2>Traceback</h2> |
|||
<pre> |
|||
{traceback} |
|||
</pre> |
|||
|
|||
<h2>Additional information</h2> |
|||
<ul> |
|||
<li>Electrum version: {electrum_version}</li> |
|||
<li>Operating system: {os}</li> |
|||
<li>Wallet type: {wallet_type}</li> |
|||
<li>Locale: {locale}</li> |
|||
</ul> |
|||
""" |
|||
report_server = "https://crashhub.electrum.org/crash" |
|||
|
|||
|
|||
class Exception_Window(QWidget): |
|||
_active_window = None |
|||
|
|||
def __init__(self, main_window, exctype, value, tb): |
|||
self.exc_args = (exctype, value, tb) |
|||
self.main_window = main_window |
|||
QWidget.__init__(self) |
|||
self.setWindowTitle('Electrum - ' + _('An Error Occured')) |
|||
self.setMinimumSize(600, 300) |
|||
|
|||
main_box = QVBoxLayout() |
|||
|
|||
heading = QLabel('<h2>' + _('Sorry!') + '</h2>') |
|||
main_box.addWidget(heading) |
|||
main_box.addWidget(QLabel(_('Something went wrong while executing Electrum.'))) |
|||
|
|||
main_box.addWidget(QLabel( |
|||
_('To help us diagnose and fix the problem, you can send us a bug report that contains useful debug ' |
|||
'information:'))) |
|||
|
|||
collapse_info = QPushButton(_("Show report contents")) |
|||
collapse_info.clicked.connect(lambda: QMessageBox.about(self, "Report contents", self.get_report_string())) |
|||
main_box.addWidget(collapse_info) |
|||
|
|||
main_box.addWidget(QLabel(_("Please briefly describe what led to the error (optional):"))) |
|||
|
|||
self.description_textfield = QTextEdit() |
|||
self.description_textfield.setFixedHeight(50) |
|||
main_box.addWidget(self.description_textfield) |
|||
|
|||
main_box.addWidget(QLabel(_("Do you want to send this report?"))) |
|||
|
|||
buttons = QHBoxLayout() |
|||
|
|||
report_button = QPushButton(_('Send Bug Report')) |
|||
report_button.clicked.connect(self.send_report) |
|||
report_button.setIcon(QIcon(":icons/tab_send.png")) |
|||
buttons.addWidget(report_button) |
|||
|
|||
never_button = QPushButton(_('Never')) |
|||
never_button.clicked.connect(self.show_never) |
|||
buttons.addWidget(never_button) |
|||
|
|||
close_button = QPushButton(_('Not Now')) |
|||
close_button.clicked.connect(self.close) |
|||
buttons.addWidget(close_button) |
|||
|
|||
main_box.addLayout(buttons) |
|||
|
|||
self.setLayout(main_box) |
|||
self.show() |
|||
|
|||
def send_report(self): |
|||
report = self.get_traceback_info() |
|||
report.update(self.get_additional_info()) |
|||
report = json.dumps(report) |
|||
response = requests.post(report_server, data=report) |
|||
QMessageBox.about(self, "Crash report", response.text) |
|||
self.close() |
|||
|
|||
def on_close(self): |
|||
Exception_Window._active_window = None |
|||
sys.__excepthook__(*self.exc_args) |
|||
self.close() |
|||
|
|||
def show_never(self): |
|||
self.main_window.config.set_key("show_crash_reporter", False) |
|||
self.close() |
|||
|
|||
def closeEvent(self, event): |
|||
self.on_close() |
|||
event.accept() |
|||
|
|||
def get_traceback_info(self): |
|||
exc_string = str(self.exc_args[1]) |
|||
stack = traceback.extract_tb(self.exc_args[2]) |
|||
readable_trace = "".join(traceback.format_list(stack)) |
|||
id = { |
|||
"file": stack[-1].filename, |
|||
"name": stack[-1].name, |
|||
"type": self.exc_args[0].__name__ |
|||
} |
|||
return { |
|||
"exc_string": exc_string, |
|||
"stack": readable_trace, |
|||
"id": id |
|||
} |
|||
|
|||
def get_additional_info(self): |
|||
args = { |
|||
"electrum_version": ELECTRUM_VERSION, |
|||
"os": platform.platform(), |
|||
"wallet_type": "unknown", |
|||
"locale": locale.getdefaultlocale()[0], |
|||
"description": self.description_textfield.toPlainText() |
|||
} |
|||
try: |
|||
args["wallet_type"] = self.main_window.wallet.wallet_type |
|||
except: |
|||
# Maybe the wallet isn't loaded yet |
|||
pass |
|||
return args |
|||
|
|||
def get_report_string(self): |
|||
info = self.get_additional_info() |
|||
info["traceback"] = "".join(traceback.format_exception(*self.exc_args)) |
|||
return issue_template.format(**info) |
|||
|
|||
|
|||
def _show_window(*args): |
|||
if not Exception_Window._active_window: |
|||
Exception_Window._active_window = Exception_Window(*args) |
|||
|
|||
|
|||
class Exception_Hook(QObject): |
|||
_report_exception = QtCore.pyqtSignal(object, object, object, object) |
|||
|
|||
def __init__(self, main_window, *args, **kwargs): |
|||
super(Exception_Hook, self).__init__(*args, **kwargs) |
|||
if not main_window.config.get("show_crash_reporter", default=True): |
|||
return |
|||
self.main_window = main_window |
|||
sys.excepthook = self.handler |
|||
self._report_exception.connect(_show_window) |
|||
|
|||
def handler(self, *args): |
|||
self._report_exception.emit(self.main_window, *args) |
Loading…
Reference in new issue