def full_upload(mw: aqt.main.AnkiQt, on_done: Callable[[], None]) -> None: mw.col.close_for_full_sync() def on_timer() -> None: on_full_sync_timer(mw) timer = QTimer(mw) qconnect(timer.timeout, on_timer) timer.start(150) def on_future_done(fut: Future) -> None: timer.stop() mw.col.reopen(after_full_sync=True) mw.reset() try: fut.result() except Exception as err: handle_sync_error(mw, err) return on_done() mw.media_syncer.start() return on_done() mw.taskman.with_progress( lambda: mw.col.full_upload(mw.pm.sync_auth()), on_future_done, label=tr.sync_uploading_to_ankiweb(), )
def full_upload(mw: aqt.main.AnkiQt, on_done: Callable[[], None]) -> None: mw.col.close_for_full_sync() def on_timer(): on_full_sync_timer(mw) timer = QTimer(mw) qconnect(timer.timeout, on_timer) timer.start(150) def on_future_done(fut): timer.stop() mw.col.reopen(after_full_sync=True) mw.reset() try: fut.result() except Exception as err: handle_sync_error(mw, err) return on_done() return on_done() mw.taskman.with_progress( lambda: mw.col.backend.full_upload(mw.pm.sync_auth()), on_future_done, label=tr(TR.SYNC_UPLOADING_TO_ANKIWEB), )
def full_download(mw: aqt.main.AnkiQt, on_done: Callable[[], None]) -> None: def on_timer() -> None: on_full_sync_timer(mw) timer = QTimer(mw) qconnect(timer.timeout, on_timer) timer.start(150) def download() -> None: mw._close_for_full_download() mw.col.full_download(mw.pm.sync_auth()) def on_future_done(fut: Future) -> None: timer.stop() mw.col.reopen(after_full_sync=True) mw.reset() try: fut.result() except Exception as err: handle_sync_error(mw, err) mw.media_syncer.start() return on_done() mw.taskman.with_progress( download, on_future_done, label=tr(TR.SYNC_DOWNLOADING_FROM_ANKIWEB), )
def _start_chase_mode_timer(http_client, chase_mode_context): global _chase_mode_timer _chase_mode_timer = QTimer(chase_mode_context.main_window) job = partial( _fetch_and_display_chase_mode, http_client, chase_mode_context, reraise=False, ) enqueue_job = partial(chase_mode_context.start_job, job) _chase_mode_timer.timeout.connect(enqueue_job) _chase_mode_timer.start(_CHASE_MODE_INTERVAL_MS)
def timer(self, ms, func, repeat, requiresCollection=True): def handler(): if self.inDB or self._levels: # retry in 100ms self.timer(100, func, False, requiresCollection) elif not self.mw.col and requiresCollection: # ignore timer events that fire after collection has been # unloaded print("Ignored progress func as collection unloaded: %s" % repr(func)) else: func() t = QTimer(self.mw) if not repeat: t.setSingleShot(True) t.timeout.connect(handler) t.start(ms) return t
def sync_collection(mw: aqt.main.AnkiQt, on_done: Callable[[], None]) -> None: auth = mw.pm.sync_auth() assert auth def on_timer(): on_normal_sync_timer(mw) timer = QTimer(mw) qconnect(timer.timeout, on_timer) timer.start(150) def on_future_done(fut): mw.col.db.begin() timer.stop() try: out: SyncOutput = fut.result() except Exception as err: handle_sync_error(mw, err) return on_done() mw.pm.set_host_number(out.host_number) if out.server_message: showText(out.server_message) if out.required == out.NO_CHANGES: # all done return on_done() else: full_sync(mw, out, on_done) if not mw.col.basicCheck(): showWarning("Please use Tools>Check Database") return on_done() mw.col.save(trx=False) mw.taskman.with_progress( lambda: mw.col.backend.sync_collection(auth), on_future_done, label=tr(TR.SYNC_CHECKING), immediate=True, )
def sync_collection(mw: aqt.main.AnkiQt, on_done: Callable[[], None]) -> None: auth = mw.pm.sync_auth() if not auth: raise Exception("expected auth") def on_timer() -> None: on_normal_sync_timer(mw) timer = QTimer(mw) qconnect(timer.timeout, on_timer) timer.start(150) def on_future_done(fut: Future) -> None: mw.col.db.begin() timer.stop() try: out: SyncOutput = fut.result() except Exception as err: handle_sync_error(mw, err) return on_done() mw.pm.set_host_number(out.host_number) if out.server_message: showText(out.server_message) if out.required == out.NO_CHANGES: # all done return on_done() else: full_sync(mw, out, on_done) mw.col.save(trx=False) mw.taskman.with_progress( lambda: mw.col.sync_collection(auth), on_future_done, label=tr.sync_checking(), immediate=True, )
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 "Pyaudio not" in error: return showWarning(_("Please install PyAudio")) if "install mplayer" in error: return showWarning( _("Sound and video on cards will not function until mpv or mplayer is installed." )) 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: return showWarning( _("Your computer's storage may be full. Please delete some unneeded files, then try again." )) if "disk I/O error" in error: return showWarning( _("""\ An error occurred while accessing the database. Possible causes: - Antivirus, firewall, backup, or synchronization software may be \ interfering with Anki. Try disabling such software and see if the \ problem goes away. - Your disk may be full. - The Documents/Anki folder may be on a network drive. - Files in the Documents/Anki folder may not be writeable. - Your hard disk may have errors. It's a good idea to run Tools>Check Database to ensure your collection \ is not corrupt. """)) stdText = _("""\ <h1>Error</h1> <p>An error occurred. Please use <b>Tools > Check Database</b> to see if \ that fixes the problem.</p> <p>If problems persist, please report the problem on our \ <a href="https://help.ankiweb.net">support site</a>. Please copy and paste \ the information below into your report.</p>""") pluginText = _("""\ <h1>Error</h1> <p>An error occurred. Please start Anki while holding down the shift \ key, which will temporarily disable the add-ons you have installed.</p> <p>If the issue only occurs when add-ons are enabled, please use the \ Tools>Add-ons menu item to disable some add-ons and restart Anki, \ repeating until you discover the add-on that is causing the problem.</p> <p>When you've discovered the add-on that is causing the problem, please \ report the issue on the <a href="https://help.ankiweb.net/discussions/add-ons/">\ add-ons section</a> of our support site. <p>Debug info:</p> """) if self.mw.addonManager.dirty: txt = pluginText error = supportText() + self._addonText(error) + "\n" + error else: txt = stdText 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 = [ aqt_mw.addonManager.addonName(i) for i in dict.fromkeys(reversed(matches)) ] txt = _("""Add-ons possibly involved: {}\n""") # highlight importance of first add-on: addons[0] = "<b>{}</b>".format(addons[0]) return txt.format(", ".join(addons))