8c0d4c8059
and ensure the UI doesn't get temporarily stuck after the command fails
129 lines
4.0 KiB
Python
129 lines
4.0 KiB
Python
# Copyright: Ankitects Pty Ltd and contributors
|
|
# -*- coding: utf-8 -*-
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
import html
|
|
import re
|
|
import sys
|
|
import traceback
|
|
|
|
from markdown import markdown
|
|
|
|
from anki.lang import _
|
|
from aqt import mw
|
|
from aqt.qt import *
|
|
from aqt.utils import TR, showText, showWarning, supportText, tr
|
|
|
|
if not os.environ.get("DEBUG"):
|
|
|
|
def excepthook(etype, val, tb):
|
|
sys.stderr.write(
|
|
"Caught exception:\n%s\n"
|
|
% ("".join(traceback.format_exception(etype, val, tb)))
|
|
)
|
|
|
|
sys.excepthook = excepthook
|
|
|
|
|
|
class ErrorHandler(QObject):
|
|
"Catch stderr and write into buffer."
|
|
ivl = 100
|
|
|
|
errorTimer = pyqtSignal()
|
|
|
|
def __init__(self, mw):
|
|
QObject.__init__(self, mw)
|
|
self.mw = mw
|
|
self.timer = None
|
|
self.errorTimer.connect(self._setTimer)
|
|
self.pool = ""
|
|
self._oldstderr = sys.stderr
|
|
sys.stderr = self
|
|
|
|
def unload(self):
|
|
sys.stderr = self._oldstderr
|
|
sys.excepthook = None
|
|
|
|
def write(self, data):
|
|
# dump to stdout
|
|
sys.stdout.write(data)
|
|
# save in buffer
|
|
self.pool += data
|
|
# and update timer
|
|
self.setTimer()
|
|
|
|
def setTimer(self):
|
|
# we can't create a timer from a different thread, so we post a
|
|
# message to the object on the main thread
|
|
self.errorTimer.emit()
|
|
|
|
def _setTimer(self):
|
|
if not self.timer:
|
|
self.timer = QTimer(self.mw)
|
|
self.timer.timeout.connect(self.onTimeout)
|
|
self.timer.setInterval(self.ivl)
|
|
self.timer.setSingleShot(True)
|
|
self.timer.start()
|
|
|
|
def tempFolderMsg(self):
|
|
return _(
|
|
"""Unable to access Anki media folder. The permissions on \
|
|
your system's temporary folder may be incorrect."""
|
|
)
|
|
|
|
def onTimeout(self):
|
|
error = html.escape(self.pool)
|
|
self.pool = ""
|
|
self.mw.progress.clear()
|
|
if "abortSchemaMod" in error:
|
|
return
|
|
if "10013" in error:
|
|
return showWarning(
|
|
_(
|
|
"Your firewall or antivirus program is preventing Anki from creating a connection to itself. Please add an exception for Anki."
|
|
)
|
|
)
|
|
if "no default input" in error.lower():
|
|
return showWarning(
|
|
_(
|
|
"Please connect a microphone, and ensure "
|
|
"other programs are not using the audio device."
|
|
)
|
|
)
|
|
if "invalidTempFolder" in error:
|
|
return showWarning(self.tempFolderMsg())
|
|
if "Beautiful Soup is not an HTTP client" in error:
|
|
return
|
|
if "database or disk is full" in error or "Errno 28" in error:
|
|
return showWarning(
|
|
_(
|
|
"Your computer's storage may be full. Please delete some unneeded files, then try again."
|
|
)
|
|
)
|
|
if "disk I/O error" in error:
|
|
showWarning(markdown(tr(TR.ERRORS_ACCESSING_DB)))
|
|
return
|
|
|
|
if self.mw.addonManager.dirty:
|
|
txt = markdown(tr(TR.ERRORS_ADDONS_ACTIVE_POPUP))
|
|
error = supportText() + self._addonText(error) + "\n" + error
|
|
else:
|
|
txt = markdown(tr(TR.ERRORS_STANDARD_POPUP))
|
|
error = supportText() + "\n" + error
|
|
|
|
# show dialog
|
|
txt = txt + "<div style='white-space: pre-wrap'>" + error + "</div>"
|
|
showText(txt, type="html", copyBtn=True)
|
|
|
|
def _addonText(self, error):
|
|
matches = re.findall(r"addons21/(.*?)/", error)
|
|
if not matches:
|
|
return ""
|
|
# reverse to list most likely suspect first, dict to deduplicate:
|
|
addons = [
|
|
mw.addonManager.addonName(i) for i in dict.fromkeys(reversed(matches))
|
|
]
|
|
# highlight importance of first add-on:
|
|
addons[0] = "<b>{}</b>".format(addons[0])
|
|
addons_str = ", ".join(addons)
|
|
return tr(TR.ADDONS_POSSIBLY_INVOLVED, addons=addons_str) + "\n"
|