2012-12-21 08:51:59 +01:00
|
|
|
# Copyright: Damien Elmes <anki@ichi2.net>
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
2016-08-02 03:51:44 +02:00
|
|
|
import io
|
2017-08-26 07:14:20 +02:00
|
|
|
import json
|
|
|
|
import re
|
2015-01-05 02:47:05 +01:00
|
|
|
import zipfile
|
2017-08-26 07:14:20 +02:00
|
|
|
from send2trash import send2trash
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
from aqt.qt import *
|
2012-12-22 01:17:10 +01:00
|
|
|
from aqt.utils import showInfo, openFolder, isWin, openLink, \
|
2017-02-15 14:38:37 +01:00
|
|
|
askUser, restoreGeom, saveGeom, showWarning, tooltip
|
2012-12-21 08:51:59 +01:00
|
|
|
from zipfile import ZipFile
|
|
|
|
import aqt.forms
|
|
|
|
import aqt
|
|
|
|
from aqt.downloader import download
|
2015-01-05 02:47:05 +01:00
|
|
|
from anki.lang import _
|
2017-08-26 07:14:20 +02:00
|
|
|
from anki.utils import intTime
|
|
|
|
from anki.sync import AnkiRequestsClient
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-02-06 23:21:33 +01:00
|
|
|
class AddonManager:
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
def __init__(self, mw):
|
|
|
|
self.mw = mw
|
2017-08-26 07:14:20 +02:00
|
|
|
self.dirty = False
|
2016-05-31 10:51:40 +02:00
|
|
|
f = self.mw.form
|
2017-08-26 07:14:20 +02:00
|
|
|
f.actionAdd_ons.triggered.connect(self.onAddonsDialog)
|
2012-12-21 08:51:59 +01:00
|
|
|
sys.path.insert(0, self.addonsFolder())
|
2013-05-17 08:32:17 +02:00
|
|
|
if not self.mw.safeMode:
|
|
|
|
self.loadAddons()
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def allAddons(self):
|
|
|
|
l = []
|
|
|
|
for d in os.listdir(self.addonsFolder()):
|
|
|
|
path = self.addonsFolder(d)
|
|
|
|
if not os.path.exists(os.path.join(path, "__init__.py")):
|
|
|
|
continue
|
|
|
|
l.append(d)
|
|
|
|
return l
|
|
|
|
|
|
|
|
def managedAddons(self):
|
|
|
|
return [d for d in self.allAddons()
|
|
|
|
if re.match("^\d+$", d)]
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def addonsFolder(self, dir=None):
|
|
|
|
root = self.mw.pm.addonFolder()
|
|
|
|
if not dir:
|
|
|
|
return root
|
|
|
|
return os.path.join(root, dir)
|
2015-09-27 00:55:15 +02:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
def loadAddons(self):
|
2017-08-26 07:14:20 +02:00
|
|
|
for dir in self.allAddons():
|
|
|
|
meta = self.addonMeta(dir)
|
|
|
|
if meta.get("disabled"):
|
|
|
|
continue
|
|
|
|
self.dirty = True
|
2015-09-27 00:55:15 +02:00
|
|
|
try:
|
2017-08-26 07:14:20 +02:00
|
|
|
__import__(dir)
|
2015-09-27 00:55:15 +02:00
|
|
|
except:
|
2017-08-26 07:14:20 +02:00
|
|
|
showWarning(_("""\
|
|
|
|
An add-on you installed failed to load. If problems persist, please \
|
|
|
|
go to the Tools>Add-ons menu, and disable or delete the add-on.
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
When loading '%(name)s':
|
|
|
|
%(traceback)s
|
|
|
|
""") % dict(name=meta.get("name", dir), traceback=traceback.format_exc()))
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def onAddonsDialog(self):
|
|
|
|
AddonsDialog(self)
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
# Metadata
|
2012-12-21 08:51:59 +01:00
|
|
|
######################################################################
|
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def _addonMetaPath(self, dir):
|
|
|
|
return os.path.join(self.addonsFolder(dir), "meta.json")
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def addonMeta(self, dir):
|
|
|
|
path = self._addonMetaPath(dir)
|
|
|
|
try:
|
|
|
|
return json.load(open(path, encoding="utf8"))
|
|
|
|
except:
|
|
|
|
return dict()
|
|
|
|
|
|
|
|
def writeAddonMeta(self, dir, meta):
|
|
|
|
path = self._addonMetaPath(dir)
|
|
|
|
json.dump(meta, open(path, "w", encoding="utf8"))
|
|
|
|
|
|
|
|
def toggleEnabled(self, dir):
|
|
|
|
meta = self.addonMeta(dir)
|
|
|
|
meta['disabled'] = not meta.get("disabled")
|
|
|
|
self.writeAddonMeta(dir, meta)
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def addonName(self, dir):
|
|
|
|
return self.addonMeta(dir).get("name", dir)
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
# Installing and deleting add-ons
|
2012-12-21 08:51:59 +01:00
|
|
|
######################################################################
|
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def install(self, sid, data, fname):
|
2015-01-05 02:47:05 +01:00
|
|
|
try:
|
2016-08-02 03:51:44 +02:00
|
|
|
z = ZipFile(io.BytesIO(data))
|
2016-02-18 09:49:44 +01:00
|
|
|
except zipfile.BadZipfile:
|
2015-01-05 02:47:05 +01:00
|
|
|
showWarning(_("The download was corrupt. Please try again."))
|
|
|
|
return
|
2017-08-26 07:14:20 +02:00
|
|
|
|
|
|
|
name = os.path.splitext(fname)[0]
|
|
|
|
|
|
|
|
# remove old version first
|
|
|
|
base = self.addonsFolder(sid)
|
|
|
|
if os.path.exists(base):
|
|
|
|
self.deleteAddon(sid)
|
|
|
|
|
|
|
|
# extract
|
|
|
|
os.mkdir(base)
|
2012-12-21 08:51:59 +01:00
|
|
|
for n in z.namelist():
|
|
|
|
if n.endswith("/"):
|
|
|
|
# folder; ignore
|
|
|
|
continue
|
|
|
|
# write
|
|
|
|
z.extract(n, base)
|
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
# write metadata
|
|
|
|
meta = dict(name=name,
|
|
|
|
mod=intTime())
|
|
|
|
self.writeAddonMeta(sid, meta)
|
|
|
|
|
|
|
|
def deleteAddon(self, dir):
|
|
|
|
send2trash(self.addonsFolder(dir))
|
|
|
|
|
|
|
|
# Downloading
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
def downloadIds(self, ids):
|
|
|
|
log = []
|
|
|
|
errs = []
|
|
|
|
self.mw.progress.start(immediate=True)
|
|
|
|
for n in ids:
|
|
|
|
ret = download(self.mw, n)
|
|
|
|
if ret[0] == "error":
|
|
|
|
errs.append(_("Error downloading %(id)s: %(error)s") % dict(id=n, error=ret[1]))
|
|
|
|
continue
|
|
|
|
data, fname = ret
|
|
|
|
self.install(str(n), data, fname)
|
|
|
|
name = os.path.splitext(fname)[0]
|
|
|
|
log.append(_("Downloaded %(fname)s" % dict(fname=name)))
|
|
|
|
self.mw.progress.finish()
|
|
|
|
return log, errs
|
|
|
|
|
|
|
|
# Updating
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
def checkForUpdates(self):
|
|
|
|
client = AnkiRequestsClient()
|
|
|
|
|
|
|
|
# get mod times
|
|
|
|
self.mw.progress.start(immediate=True)
|
|
|
|
try:
|
|
|
|
# ..of enabled items downloaded from ankiweb
|
|
|
|
addons = []
|
|
|
|
for dir in self.managedAddons():
|
|
|
|
meta = self.addonMeta(dir)
|
|
|
|
if not meta.get("disabled"):
|
|
|
|
addons.append(dir)
|
|
|
|
|
|
|
|
mods = []
|
|
|
|
while addons:
|
|
|
|
chunk = addons[:25]
|
|
|
|
del addons[:25]
|
|
|
|
mods.extend(self._getModTimes(client, chunk))
|
|
|
|
return self._updatedIds(mods)
|
|
|
|
finally:
|
|
|
|
self.mw.progress.finish()
|
|
|
|
|
|
|
|
def _getModTimes(self, client, chunk):
|
|
|
|
resp = client.get(
|
|
|
|
aqt.appShared + "updates/" + ",".join(chunk))
|
|
|
|
if resp.status_code == 200:
|
|
|
|
return resp.json()
|
|
|
|
else:
|
|
|
|
raise Exception("Unexpected response code from AnkiWeb: {}".format(resp.status_code))
|
|
|
|
|
|
|
|
def _updatedIds(self, mods):
|
|
|
|
updated = []
|
|
|
|
for dir, ts in mods:
|
|
|
|
sid = str(dir)
|
|
|
|
if self.addonMeta(sid).get("mod") < ts:
|
|
|
|
updated.append(sid)
|
|
|
|
return updated
|
|
|
|
|
|
|
|
# Add-ons Dialog
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
class AddonsDialog(QDialog):
|
|
|
|
|
|
|
|
def __init__(self, addonsManager):
|
|
|
|
self.mgr = addonsManager
|
|
|
|
self.mw = addonsManager.mw
|
|
|
|
|
|
|
|
super().__init__(self.mw)
|
|
|
|
|
|
|
|
f = self.form = aqt.forms.addons.Ui_Dialog()
|
|
|
|
f.setupUi(self)
|
|
|
|
f.getAddons.clicked.connect(self.onGetAddons)
|
|
|
|
f.checkForUpdates.clicked.connect(self.onCheckForUpdates)
|
|
|
|
f.toggleEnabled.clicked.connect(self.onToggleEnabled)
|
|
|
|
f.viewPage.clicked.connect(self.onViewPage)
|
|
|
|
f.viewFiles.clicked.connect(self.onViewFiles)
|
|
|
|
f.delete_2.clicked.connect(self.onDelete)
|
|
|
|
self.redrawAddons()
|
|
|
|
self.show()
|
|
|
|
|
|
|
|
def redrawAddons(self):
|
|
|
|
self.addons = [(self.annotatedName(d), d) for d in self.mgr.allAddons()]
|
|
|
|
self.addons.sort()
|
|
|
|
self.form.addonList.clear()
|
|
|
|
self.form.addonList.addItems([r[0] for r in self.addons])
|
|
|
|
if self.addons:
|
|
|
|
self.form.addonList.setCurrentRow(0)
|
|
|
|
|
|
|
|
def annotatedName(self, dir):
|
|
|
|
meta = self.mgr.addonMeta(dir)
|
|
|
|
buf = self.mgr.addonName(dir)
|
|
|
|
if meta.get('disabled'):
|
|
|
|
buf += _(" (disabled)")
|
|
|
|
return buf
|
|
|
|
|
|
|
|
def selectedAddons(self):
|
|
|
|
idxs = [x.row() for x in self.form.addonList.selectedIndexes()]
|
|
|
|
return [self.addons[idx][1] for idx in idxs]
|
|
|
|
|
|
|
|
def onlyOneSelected(self):
|
|
|
|
dirs = self.selectedAddons()
|
|
|
|
if len(dirs) != 1:
|
|
|
|
showInfo("Please select a single add-on first.")
|
|
|
|
return
|
|
|
|
return dirs[0]
|
|
|
|
|
|
|
|
def onToggleEnabled(self):
|
|
|
|
for dir in self.selectedAddons():
|
|
|
|
self.mgr.toggleEnabled(dir)
|
|
|
|
self.redrawAddons()
|
|
|
|
|
|
|
|
def onViewPage(self):
|
|
|
|
addon = self.onlyOneSelected()
|
|
|
|
if not addon:
|
|
|
|
return
|
|
|
|
if re.match("^\d+$", addon):
|
|
|
|
openLink(aqt.appShared + f"info/{addon}")
|
|
|
|
else:
|
|
|
|
showWarning(_("Add-on was not downloaded from AnkiWeb."))
|
|
|
|
|
|
|
|
def onViewFiles(self):
|
|
|
|
# if nothing selected, open top level folder
|
|
|
|
selected = self.selectedAddons()
|
|
|
|
if not selected:
|
|
|
|
openFolder(self.mgr.addonsFolder())
|
|
|
|
return
|
|
|
|
|
|
|
|
# otherwise require a single selection
|
|
|
|
addon = self.onlyOneSelected()
|
|
|
|
if not addon:
|
|
|
|
return
|
|
|
|
path = self.mgr.addonsFolder(addon)
|
|
|
|
openFolder(path)
|
|
|
|
|
|
|
|
def onDelete(self):
|
|
|
|
selected = self.selectedAddons()
|
|
|
|
if not selected:
|
|
|
|
return
|
|
|
|
if not askUser(_("Delete the %(num)d selected add-ons?") %
|
|
|
|
dict(num=len(selected))):
|
|
|
|
return
|
|
|
|
for dir in selected:
|
|
|
|
self.mgr.deleteAddon(dir)
|
|
|
|
self.redrawAddons()
|
|
|
|
|
|
|
|
def onGetAddons(self):
|
|
|
|
GetAddons(self)
|
|
|
|
|
|
|
|
def onCheckForUpdates(self):
|
|
|
|
updated = self.mgr.checkForUpdates()
|
|
|
|
if not updated:
|
|
|
|
tooltip(_("No updates available."))
|
|
|
|
else:
|
|
|
|
names = [self.mgr.addonName(d) for d in updated]
|
|
|
|
if askUser(_("Update the following add-ons?") +
|
|
|
|
"\n" + "\n".join(names)):
|
|
|
|
log, errs = self.mgr.downloadIds(updated)
|
|
|
|
if log:
|
|
|
|
tooltip("\n".join(log), parent=self)
|
|
|
|
if errs:
|
|
|
|
showWarning("\n".join(errs), parent=self)
|
|
|
|
|
|
|
|
self.redrawAddons()
|
|
|
|
|
|
|
|
# Fetching Add-ons
|
|
|
|
######################################################################
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
class GetAddons(QDialog):
|
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
def __init__(self, dlg):
|
|
|
|
QDialog.__init__(self, dlg)
|
|
|
|
self.addonsDlg = dlg
|
|
|
|
self.mgr = dlg.mgr
|
|
|
|
self.mw = self.mgr.mw
|
2012-12-21 08:51:59 +01:00
|
|
|
self.form = aqt.forms.getaddons.Ui_Dialog()
|
|
|
|
self.form.setupUi(self)
|
|
|
|
b = self.form.buttonBox.addButton(
|
|
|
|
_("Browse"), QDialogButtonBox.ActionRole)
|
2016-05-31 10:51:40 +02:00
|
|
|
b.clicked.connect(self.onBrowse)
|
2014-06-18 20:47:45 +02:00
|
|
|
restoreGeom(self, "getaddons", adjustSize=True)
|
2012-12-21 08:51:59 +01:00
|
|
|
self.exec_()
|
2014-06-18 20:47:45 +02:00
|
|
|
saveGeom(self, "getaddons")
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
def onBrowse(self):
|
2017-08-26 07:14:20 +02:00
|
|
|
openLink(aqt.appShared + "addons/?v=2.1")
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
def accept(self):
|
2017-02-15 06:55:20 +01:00
|
|
|
# get codes
|
|
|
|
try:
|
|
|
|
ids = [int(n) for n in self.form.code.text().split()]
|
|
|
|
except ValueError:
|
|
|
|
showWarning(_("Invalid code."))
|
|
|
|
return
|
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
log, errs = self.mgr.downloadIds(ids)
|
2017-02-15 06:55:20 +01:00
|
|
|
|
2017-08-26 07:14:20 +02:00
|
|
|
if log:
|
|
|
|
tooltip("\n".join(log), parent=self.addonsDlg)
|
|
|
|
if errs:
|
|
|
|
showWarning("\n".join(errs))
|
|
|
|
|
|
|
|
self.addonsDlg.redrawAddons()
|
|
|
|
QDialog.accept(self)
|