mirror of
https://github.com/ankitects/anki.git
synced 2025-09-19 22:42:25 -04:00

While something we probably don't want to encourage much of, this may enable some previously-unshared add-ons. https://forums.ankiweb.net/t/bundling-numpy-in-an-add-on/62669/5 The 'uv add' command is transaction, so if an add-on tries to inject incompatible dependencies into the environment, the venv will be left as-is. And each Anki upgrade/downgrade resets the requirements, so the requested packages shouldn't cause errors down the line. Sample add-on: import subprocess from aqt import mw from aqt.operations import QueryOp from aqt.qt import QAction from aqt.utils import showInfo def ensure_spacy(col): print("trying to import spacy") try: import spacy print("successful import") return except Exception as e: print("error importing:", e) print("attempting add") try: from aqt.package import add_python_requirements as add except Exception as e: raise Exception(f"package unavailable, can't install: {e}") # be explicit about version, or Anki beta users will get # a beta wheel that may break (success, output) = add(["spacy==3.8.7", "https://github.com/explosion/spacy-models/releases/download/ko_core_news_sm-3.8.0/ko_core_news_sm-3.8.0-py3-none-any.whl"]) if not success: raise Exception(f"adding failed: {output}") print("success") # alterantively: # from aqt.package import venv_binary # subprocess.run([venv_binary("spacy"), "download", "ko_core_news_sm"], check=True) # print("model added") # large packages will freeze for a while on first import on macOS import spacy print("spacy import successful") def activate_spacy(): def on_success(res): mw.progress.single_shot(1000, lambda: showInfo("Spacy installed")) QueryOp(parent=mw, op=ensure_spacy, success=on_success).with_progress("Installing spacy...").run_in_background() action = QAction("Activate Spacy", mw) action.triggered.connect(activate_spacy) mw.form.menuTools.addAction(action)
184 lines
5 KiB
Python
184 lines
5 KiB
Python
# Copyright: Ankitects Pty Ltd and contributors
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
from __future__ import annotations
|
|
|
|
import contextlib
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
from pathlib import Path
|
|
from typing import Any
|
|
|
|
from anki.utils import pointVersion
|
|
from aqt import mw
|
|
from aqt.qt import QAction
|
|
from aqt.utils import askUser, is_mac, is_win, showInfo
|
|
|
|
|
|
def launcher_executable() -> str | None:
|
|
"""Return the path to the Anki launcher executable."""
|
|
return os.getenv("ANKI_LAUNCHER")
|
|
|
|
|
|
def uv_binary() -> str | None:
|
|
"""Return the path to the uv binary."""
|
|
return os.environ.get("ANKI_LAUNCHER_UV")
|
|
|
|
|
|
def launcher_root() -> str | None:
|
|
"""Return the path to the launcher root directory (AnkiProgramFiles)."""
|
|
return os.environ.get("UV_PROJECT")
|
|
|
|
|
|
def venv_binary(cmd: str) -> str | None:
|
|
"""Return the path to a binary in the launcher's venv."""
|
|
root = launcher_root()
|
|
if not root:
|
|
return None
|
|
|
|
root_path = Path(root)
|
|
if is_win:
|
|
binary_path = root_path / ".venv" / "Scripts" / cmd
|
|
else:
|
|
binary_path = root_path / ".venv" / "bin" / cmd
|
|
|
|
return str(binary_path)
|
|
|
|
|
|
def add_python_requirements(reqs: list[str]) -> tuple[bool, str]:
|
|
"""Add Python requirements to the launcher venv using uv add.
|
|
|
|
Returns (success, output)"""
|
|
|
|
binary = uv_binary()
|
|
if not binary:
|
|
return (False, "Not in packaged build.")
|
|
|
|
uv_cmd = [binary, "add"] + reqs
|
|
result = subprocess.run(uv_cmd, capture_output=True, text=True, check=False)
|
|
|
|
if result.returncode == 0:
|
|
root = launcher_root()
|
|
if root:
|
|
sync_marker = Path(root) / ".sync_complete"
|
|
sync_marker.touch()
|
|
return (True, result.stdout)
|
|
else:
|
|
return (False, result.stderr)
|
|
|
|
|
|
def trigger_launcher_run() -> None:
|
|
"""Bump the mtime on pyproject.toml in the local data directory to trigger an update on next run."""
|
|
try:
|
|
root = launcher_root()
|
|
if not root:
|
|
return
|
|
|
|
pyproject_path = Path(root) / "pyproject.toml"
|
|
|
|
if pyproject_path.exists():
|
|
# Touch the file to update its mtime
|
|
pyproject_path.touch()
|
|
except Exception as e:
|
|
print(e)
|
|
|
|
|
|
def update_and_restart() -> None:
|
|
"""Update and restart Anki using the launcher."""
|
|
launcher = launcher_executable()
|
|
assert launcher
|
|
|
|
trigger_launcher_run()
|
|
|
|
with contextlib.suppress(ResourceWarning):
|
|
env = os.environ.copy()
|
|
creationflags = 0
|
|
if sys.platform == "win32":
|
|
creationflags = (
|
|
subprocess.CREATE_NEW_PROCESS_GROUP | subprocess.DETACHED_PROCESS
|
|
)
|
|
subprocess.Popen(
|
|
[launcher],
|
|
start_new_session=True,
|
|
stdin=subprocess.DEVNULL,
|
|
stdout=subprocess.DEVNULL,
|
|
stderr=subprocess.DEVNULL,
|
|
env=env,
|
|
creationflags=creationflags,
|
|
)
|
|
|
|
mw.app.quit()
|
|
|
|
|
|
def confirm_then_upgrade():
|
|
if not askUser("Change to a different Anki version?"):
|
|
return
|
|
update_and_restart()
|
|
|
|
|
|
# return modified command array that points to bundled command, and return
|
|
# required environment
|
|
def _packagedCmd(cmd: list[str]) -> tuple[Any, dict[str, str]]:
|
|
cmd = cmd[:]
|
|
env = os.environ.copy()
|
|
# keep LD_LIBRARY_PATH when in snap environment
|
|
if "LD_LIBRARY_PATH" in env and "SNAP" not in env:
|
|
del env["LD_LIBRARY_PATH"]
|
|
|
|
# Try to find binary in anki-audio package for Windows/Mac
|
|
if is_win or is_mac:
|
|
try:
|
|
import anki_audio
|
|
|
|
audio_pkg_path = Path(anki_audio.__file__).parent
|
|
if is_win:
|
|
packaged_path = audio_pkg_path / (cmd[0] + ".exe")
|
|
else: # is_mac
|
|
packaged_path = audio_pkg_path / cmd[0]
|
|
|
|
if packaged_path.exists():
|
|
cmd[0] = str(packaged_path)
|
|
return cmd, env
|
|
except ImportError:
|
|
# anki-audio not available, fall back to old behavior
|
|
pass
|
|
|
|
packaged_path = Path(sys.prefix) / cmd[0]
|
|
if packaged_path.exists():
|
|
cmd[0] = str(packaged_path)
|
|
|
|
return cmd, env
|
|
|
|
|
|
def setup():
|
|
if pointVersion() >= 250600:
|
|
return
|
|
if not launcher_executable():
|
|
return
|
|
|
|
# Add action to tools menu
|
|
action = QAction("Upgrade/Downgrade", mw)
|
|
action.triggered.connect(confirm_then_upgrade)
|
|
mw.form.menuTools.addAction(action)
|
|
|
|
# Monkey-patch audio tools to use anki-audio
|
|
if is_win or is_mac:
|
|
import aqt
|
|
import aqt.sound
|
|
|
|
aqt.sound._packagedCmd = _packagedCmd
|
|
|
|
# Inject launcher functions into launcher module
|
|
import aqt.package
|
|
|
|
aqt.package.launcher_executable = launcher_executable
|
|
aqt.package.update_and_restart = update_and_restart
|
|
aqt.package.trigger_launcher_run = trigger_launcher_run
|
|
aqt.package.uv_binary = uv_binary
|
|
aqt.package.launcher_root = launcher_root
|
|
aqt.package.venv_binary = venv_binary
|
|
aqt.package.add_python_requirements = add_python_requirements
|
|
|
|
|
|
setup()
|