2020-02-04 00:07:15 +01:00
|
|
|
# Copyright: Ankitects Pty Ltd and contributors
|
|
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
|
2020-02-04 02:41:20 +01:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2020-02-04 00:07:15 +01:00
|
|
|
import time
|
|
|
|
from concurrent.futures import Future
|
|
|
|
from copy import copy
|
|
|
|
from dataclasses import dataclass
|
2020-02-05 03:38:36 +01:00
|
|
|
from typing import List, Optional, Union
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
import aqt
|
|
|
|
from anki import hooks
|
|
|
|
from anki.lang import _
|
|
|
|
from anki.media import media_paths_from_col_path
|
|
|
|
from anki.rsbackend import (
|
2020-02-04 06:16:11 +01:00
|
|
|
DBError,
|
2020-02-04 00:07:15 +01:00
|
|
|
Interrupted,
|
|
|
|
MediaSyncDownloadedChanges,
|
|
|
|
MediaSyncDownloadedFiles,
|
|
|
|
MediaSyncProgress,
|
|
|
|
MediaSyncRemovedFiles,
|
|
|
|
MediaSyncUploaded,
|
2020-02-04 03:34:44 +01:00
|
|
|
NetworkError,
|
2020-02-04 10:39:31 +01:00
|
|
|
NetworkErrorKind,
|
2020-02-04 00:07:15 +01:00
|
|
|
Progress,
|
|
|
|
ProgressKind,
|
2020-02-04 10:39:31 +01:00
|
|
|
SyncError,
|
|
|
|
SyncErrorKind,
|
2020-02-04 00:07:15 +01:00
|
|
|
)
|
|
|
|
from anki.types import assert_impossible
|
|
|
|
from anki.utils import intTime
|
|
|
|
from aqt import gui_hooks
|
2020-02-04 02:48:51 +01:00
|
|
|
from aqt.qt import QDialog, QDialogButtonBox, QPushButton
|
2020-02-04 03:34:44 +01:00
|
|
|
from aqt.utils import showWarning
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class MediaSyncState:
|
|
|
|
downloaded_changes: int = 0
|
|
|
|
downloaded_files: int = 0
|
|
|
|
uploaded_files: int = 0
|
|
|
|
uploaded_removals: int = 0
|
|
|
|
removed_files: int = 0
|
|
|
|
|
|
|
|
|
2020-02-04 03:26:10 +01:00
|
|
|
LogEntry = Union[MediaSyncState, str]
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class LogEntryWithTime:
|
|
|
|
time: int
|
|
|
|
entry: LogEntry
|
|
|
|
|
|
|
|
|
|
|
|
class MediaSyncer:
|
2020-02-04 03:26:10 +01:00
|
|
|
def __init__(self, mw: aqt.main.AnkiQt):
|
|
|
|
self.mw = mw
|
2020-02-04 00:07:15 +01:00
|
|
|
self._sync_state: Optional[MediaSyncState] = None
|
|
|
|
self._log: List[LogEntryWithTime] = []
|
|
|
|
self._want_stop = False
|
|
|
|
hooks.rust_progress_callback.append(self._on_rust_progress)
|
2020-02-05 02:55:14 +01:00
|
|
|
gui_hooks.media_sync_did_start_or_stop.append(self._on_start_stop)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
def _on_rust_progress(self, proceed: bool, progress: Progress) -> bool:
|
|
|
|
if progress.kind != ProgressKind.MediaSyncProgress:
|
|
|
|
return proceed
|
|
|
|
|
|
|
|
self._update_state(progress.val)
|
|
|
|
self._log_and_notify(copy(self._sync_state))
|
|
|
|
|
|
|
|
if self._want_stop:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return proceed
|
|
|
|
|
|
|
|
def _update_state(self, progress: MediaSyncProgress) -> None:
|
|
|
|
if isinstance(progress, MediaSyncDownloadedChanges):
|
|
|
|
self._sync_state.downloaded_changes += progress.changes
|
|
|
|
elif isinstance(progress, MediaSyncDownloadedFiles):
|
|
|
|
self._sync_state.downloaded_files += progress.files
|
|
|
|
elif isinstance(progress, MediaSyncUploaded):
|
|
|
|
self._sync_state.uploaded_files += progress.files
|
|
|
|
self._sync_state.uploaded_removals += progress.deletions
|
|
|
|
elif isinstance(progress, MediaSyncRemovedFiles):
|
|
|
|
self._sync_state.removed_files += progress.files
|
|
|
|
|
2020-02-04 03:26:10 +01:00
|
|
|
def start(self) -> None:
|
2020-02-04 00:07:15 +01:00
|
|
|
"Start media syncing in the background, if it's not already running."
|
|
|
|
if self._sync_state is not None:
|
|
|
|
return
|
|
|
|
|
2020-02-04 03:26:10 +01:00
|
|
|
hkey = self.mw.pm.sync_key()
|
|
|
|
if hkey is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
if not self.mw.pm.media_syncing_enabled():
|
|
|
|
self._log_and_notify(_("Media syncing disabled."))
|
|
|
|
return
|
|
|
|
|
|
|
|
self._log_and_notify(_("Media sync starting..."))
|
2020-02-04 00:07:15 +01:00
|
|
|
self._sync_state = MediaSyncState()
|
|
|
|
self._want_stop = False
|
2020-02-05 02:55:14 +01:00
|
|
|
gui_hooks.media_sync_did_start_or_stop(True)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
2020-02-04 03:26:10 +01:00
|
|
|
(media_folder, media_db) = media_paths_from_col_path(self.mw.col.path)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
def run() -> None:
|
2020-02-04 06:16:11 +01:00
|
|
|
self.mw.col.backend.sync_media(
|
|
|
|
hkey, media_folder, media_db, self._endpoint()
|
|
|
|
)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
2020-02-04 03:26:10 +01:00
|
|
|
self.mw.taskman.run_in_background(run, self._on_finished)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
2020-02-04 03:46:57 +01:00
|
|
|
def _endpoint(self) -> str:
|
|
|
|
shard = self.mw.pm.sync_shard()
|
|
|
|
if shard is not None:
|
|
|
|
shard_str = str(shard)
|
|
|
|
else:
|
|
|
|
shard_str = ""
|
|
|
|
return f"https://sync{shard_str}.ankiweb.net/msync/"
|
|
|
|
|
2020-02-04 00:07:15 +01:00
|
|
|
def _log_and_notify(self, entry: LogEntry) -> None:
|
|
|
|
entry_with_time = LogEntryWithTime(time=intTime(), entry=entry)
|
|
|
|
self._log.append(entry_with_time)
|
2020-02-04 03:26:10 +01:00
|
|
|
self.mw.taskman.run_on_main(
|
2020-02-04 00:07:15 +01:00
|
|
|
lambda: gui_hooks.media_sync_did_progress(entry_with_time)
|
|
|
|
)
|
|
|
|
|
|
|
|
def _on_finished(self, future: Future) -> None:
|
|
|
|
self._sync_state = None
|
2020-02-05 02:55:14 +01:00
|
|
|
gui_hooks.media_sync_did_start_or_stop(False)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
exc = future.exception()
|
|
|
|
if exc is not None:
|
2020-02-04 03:26:10 +01:00
|
|
|
self._handle_sync_error(exc)
|
|
|
|
else:
|
|
|
|
self._log_and_notify(_("Media sync complete."))
|
|
|
|
|
|
|
|
def _handle_sync_error(self, exc: BaseException):
|
2020-02-04 06:16:11 +01:00
|
|
|
if isinstance(exc, Interrupted):
|
|
|
|
self._log_and_notify(_("Media sync aborted."))
|
|
|
|
return
|
|
|
|
|
|
|
|
self._log_and_notify(_("Media sync failed."))
|
2020-02-04 10:39:31 +01:00
|
|
|
if isinstance(exc, SyncError):
|
|
|
|
kind = exc.kind()
|
|
|
|
if kind == SyncErrorKind.AUTH_FAILED:
|
|
|
|
self.mw.pm.set_sync_key(None)
|
|
|
|
showWarning(
|
|
|
|
_("AnkiWeb ID or password was incorrect; please try again.")
|
|
|
|
)
|
|
|
|
elif kind == SyncErrorKind.SERVER_ERROR:
|
|
|
|
showWarning(
|
|
|
|
_(
|
|
|
|
"AnkiWeb encountered a problem. Please try again in a few minutes."
|
|
|
|
)
|
|
|
|
)
|
2020-02-05 04:35:40 +01:00
|
|
|
elif kind == SyncErrorKind.MEDIA_CHECK_REQUIRED:
|
|
|
|
showWarning(_("Please use the Tools>Check Media menu option."))
|
|
|
|
elif kind == SyncErrorKind.RESYNC_REQUIRED:
|
|
|
|
showWarning(
|
|
|
|
_(
|
|
|
|
"Please sync again, and post on the support forum if this message keeps appearing."
|
|
|
|
)
|
|
|
|
)
|
2020-02-04 10:39:31 +01:00
|
|
|
else:
|
|
|
|
showWarning(_("Unexpected error: {}").format(str(exc)))
|
2020-02-04 03:34:44 +01:00
|
|
|
elif isinstance(exc, NetworkError):
|
2020-02-04 10:39:31 +01:00
|
|
|
nkind = exc.kind()
|
|
|
|
if nkind in (NetworkErrorKind.OFFLINE, NetworkErrorKind.TIMEOUT):
|
|
|
|
showWarning(
|
|
|
|
_("Syncing failed; please check your internet connection.")
|
|
|
|
+ "\n\n"
|
|
|
|
+ _("Detailed error: {}").format(str(exc))
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
showWarning(_("Unexpected error: {}").format(str(exc)))
|
2020-02-04 06:16:11 +01:00
|
|
|
elif isinstance(exc, DBError):
|
|
|
|
showWarning(_("Problem accessing the media database: {}").format(str(exc)))
|
2020-02-04 00:07:15 +01:00
|
|
|
else:
|
2020-02-04 03:26:10 +01:00
|
|
|
raise exc
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
def entries(self) -> List[LogEntryWithTime]:
|
|
|
|
return self._log
|
|
|
|
|
|
|
|
def abort(self) -> None:
|
2020-02-04 03:26:10 +01:00
|
|
|
if not self.is_syncing():
|
|
|
|
return
|
|
|
|
self._log_and_notify(_("Media sync aborting..."))
|
2020-02-04 00:07:15 +01:00
|
|
|
self._want_stop = True
|
|
|
|
|
2020-02-04 02:41:20 +01:00
|
|
|
def is_syncing(self) -> bool:
|
|
|
|
return self._sync_state is not None
|
|
|
|
|
2020-02-05 02:55:14 +01:00
|
|
|
def _on_start_stop(self, running: bool):
|
2020-02-05 03:23:15 +01:00
|
|
|
self.mw.toolbar.set_sync_active(running) # type: ignore
|
2020-02-04 03:26:10 +01:00
|
|
|
|
|
|
|
def show_sync_log(self):
|
|
|
|
aqt.dialogs.open("sync_log", self.mw, self)
|
|
|
|
|
2020-02-05 03:23:15 +01:00
|
|
|
def show_diag_until_finished(self):
|
|
|
|
# nothing to do if not syncing
|
|
|
|
if not self.is_syncing():
|
|
|
|
return
|
|
|
|
|
|
|
|
diag: MediaSyncDialog = aqt.dialogs.open("sync_log", self.mw, self, True)
|
|
|
|
diag.exec_()
|
|
|
|
|
2020-02-05 03:38:36 +01:00
|
|
|
def seconds_since_last_sync(self) -> int:
|
|
|
|
if self.is_syncing():
|
|
|
|
return 0
|
|
|
|
|
|
|
|
if self._log:
|
|
|
|
last = self._log[-1].time
|
|
|
|
else:
|
|
|
|
last = 0
|
|
|
|
return intTime() - last
|
|
|
|
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
class MediaSyncDialog(QDialog):
|
2020-02-04 02:41:20 +01:00
|
|
|
silentlyClose = True
|
|
|
|
|
2020-02-05 03:23:15 +01:00
|
|
|
def __init__(
|
|
|
|
self, mw: aqt.main.AnkiQt, syncer: MediaSyncer, close_when_done: bool = False
|
|
|
|
) -> None:
|
2020-02-04 02:41:20 +01:00
|
|
|
super().__init__(mw)
|
|
|
|
self.mw = mw
|
2020-02-04 00:07:15 +01:00
|
|
|
self._syncer = syncer
|
2020-02-05 03:23:15 +01:00
|
|
|
self._close_when_done = close_when_done
|
2020-02-04 00:07:15 +01:00
|
|
|
self.form = aqt.forms.synclog.Ui_Dialog()
|
|
|
|
self.form.setupUi(self)
|
|
|
|
self.abort_button = QPushButton(_("Abort"))
|
|
|
|
self.abort_button.clicked.connect(self._on_abort) # type: ignore
|
2020-02-04 03:26:10 +01:00
|
|
|
self.abort_button.setAutoDefault(False)
|
2020-02-04 00:07:15 +01:00
|
|
|
self.form.buttonBox.addButton(self.abort_button, QDialogButtonBox.ActionRole)
|
|
|
|
|
|
|
|
gui_hooks.media_sync_did_progress.append(self._on_log_entry)
|
2020-02-05 03:23:15 +01:00
|
|
|
gui_hooks.media_sync_did_start_or_stop.append(self._on_start_stop)
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
self.form.plainTextEdit.setPlainText(
|
|
|
|
"\n".join(self._entry_to_text(x) for x in syncer.entries())
|
|
|
|
)
|
2020-02-04 02:41:20 +01:00
|
|
|
self.show()
|
|
|
|
|
2020-02-05 03:23:15 +01:00
|
|
|
def reject(self) -> None:
|
|
|
|
if self._close_when_done and self._syncer.is_syncing():
|
|
|
|
# closing while syncing on close starts an abort
|
|
|
|
self._on_abort()
|
|
|
|
return
|
2020-02-04 02:41:20 +01:00
|
|
|
|
|
|
|
aqt.dialogs.markClosed("sync_log")
|
2020-02-05 03:23:15 +01:00
|
|
|
QDialog.reject(self)
|
2020-02-04 02:41:20 +01:00
|
|
|
|
2020-02-05 03:23:15 +01:00
|
|
|
def reopen(self, mw, syncer, close_when_done: bool = False) -> None:
|
|
|
|
self._close_when_done = close_when_done
|
2020-02-04 02:41:20 +01:00
|
|
|
self.show()
|
2020-02-04 00:07:15 +01:00
|
|
|
|
|
|
|
def _on_abort(self, *args) -> None:
|
|
|
|
self._syncer.abort()
|
|
|
|
self.abort_button.setHidden(True)
|
|
|
|
|
|
|
|
def _time_and_text(self, stamp: int, text: str) -> str:
|
|
|
|
asctime = time.asctime(time.localtime(stamp))
|
|
|
|
return f"{asctime}: {text}"
|
|
|
|
|
|
|
|
def _entry_to_text(self, entry: LogEntryWithTime):
|
2020-02-04 03:26:10 +01:00
|
|
|
if isinstance(entry.entry, str):
|
|
|
|
txt = entry.entry
|
2020-02-04 00:07:15 +01:00
|
|
|
elif isinstance(entry.entry, MediaSyncState):
|
|
|
|
txt = self._logentry_to_text(entry.entry)
|
|
|
|
else:
|
|
|
|
assert_impossible(entry.entry)
|
|
|
|
return self._time_and_text(entry.time, txt)
|
|
|
|
|
|
|
|
def _logentry_to_text(self, e: MediaSyncState) -> str:
|
|
|
|
return _(
|
|
|
|
"Added: %(a_up)s ↑, %(a_dwn)s ↓, Removed: %(r_up)s ↑, %(r_dwn)s ↓, Checked: %(chk)s"
|
|
|
|
) % dict(
|
|
|
|
a_up=e.uploaded_files,
|
|
|
|
a_dwn=e.downloaded_files,
|
|
|
|
r_up=e.uploaded_removals,
|
|
|
|
r_dwn=e.removed_files,
|
|
|
|
chk=e.downloaded_changes,
|
|
|
|
)
|
|
|
|
|
|
|
|
def _on_log_entry(self, entry: LogEntryWithTime):
|
|
|
|
self.form.plainTextEdit.appendPlainText(self._entry_to_text(entry))
|
2020-02-04 03:26:10 +01:00
|
|
|
if not self._syncer.is_syncing():
|
|
|
|
self.abort_button.setHidden(True)
|
2020-02-05 03:23:15 +01:00
|
|
|
|
|
|
|
def _on_start_stop(self, running: bool) -> None:
|
|
|
|
if not running and self._close_when_done:
|
|
|
|
aqt.dialogs.markClosed("sync_log")
|
|
|
|
self._close_when_done = False
|
|
|
|
self.close()
|