Skip to content

Latest commit

 

History

History
151 lines (95 loc) · 6.25 KB

README.md

File metadata and controls

151 lines (95 loc) · 6.25 KB

pytest-anki

a simple pytest plugin for testing Anki add-ons

This project is a rewrite of krassowski/anki_testing as a pytest plugin, with a number of added convenience features and adjustments for more recent Anki releases.

Disclaimer

Project State

This is still very much a work-in-progress. Neither the API, nor the implementation are set in stone.

Platform Support

pytest-anki has only been tested on Linux so far.

Installation

  1. Set up an Anki development environment and add your local Anki source folder to your PYTHONPATH (so that both the anki and aqt packages can be resolved by Python, see run_tests.sh for an example).

    The minimum supported Anki version currently is 2.1.17

  2. Install pytest-anki into your testing environment:

    pip install --upgrade git+https://github.com/glutanimate/pytest-anki.git

Usage

Basic Use

In your tests add:

def test_my_addon(anki_session):
    # add some tests in here

The anki_session fixture yields an AnkiSession object that gives you access to the following attributes:

app {AnkiApp} -- Anki QApplication instance
mw {AnkiQt} -- Anki QMainWindow instance
user {str} -- User profile name (e.g. "User 1")
base {str} -- Path to Anki base directory

This allows you to perform actions like loading or unloading Anki profiles, e.g.:

def test_my_addon(anki_session):
    anki_session.mw.loadProfile()
    assert anki_session.mw.col is not None

Finally, assuming that your tests are located in a tests folder at the root of your project, you can then run your tests with:

python3 -m pytest tests

(also see the sample script under tools)

Configuring the Anki Session

You can customize the Anki session context with a series of arguments that can be passed to the anki_session fixture using indirect parametrization, e.g.

@pytest.mark.parametrize("anki_session", [dict(profile_name="foo")], indirect=True)
def test_my_addon(anki_session):
    assert anki_session.user == "foo"

A full list of supported arguments follows below:

Keyword Arguments:
    base_path {str} -- Path to write Anki base folder to
                        (default: {tempfile.gettempdir()})
    base_name {str} -- Base folder name (default: {"anki_base"})
    profile_name {str} -- User profile name (default: {"__Temporary Test User__"})
    keep_profile {bool} -- Whether to preserve profile at context exit
                            (default: {False})
    load_profile {bool} -- Whether to return an Anki session with the user profile
                            and collection fully preloaded (default: {False})
    lang {str} -- Language to use for the user profile (default: {"en_US"})

Other Features

pytest-anki also provides a convenient context manager called profile_loaded that simplifies testing your add-ons at different profile load states:

from pytest_anki import AnkiSession  # used here for type annotations
from pytest_anki import profile_loaded

def test_my_addon(anki_session: AnkiSession):
    assert anki_session.mw.col is None  # profile / collection not yet loaded

    with profile_loaded(anki_session.mw):
        assert anki_session.mw.col is not None  # loaded
    
    assert anki_session.mw.col is None  # safely unloaded again

Additional helper functions and context managers are also available. Please refer to the source code for the latest feature-set.

Other Notes

Running your test in an Anki environment is expensive and introduces an additional layer of confounding factors, so I would recommend avoiding the anki_session fixture as much as possible, mocking away Anki runtime dependencies where you can. The anki_session fixture is in many ways more suited towards end-to-end testing rather more fundamental tests in the test pyramid.

If you do use anki_session, I would highly recommend running your tests in separate subprocesses using pytest-forked. Because of the way Anki works (e.g. in terms of monkey-patching, etc.) exiting out of the anki_session fixture is never quite clean, and so you run the risk of state persisting across to your next tests. This could lead to unexpected behavior, or worse still, your tests crashing. Forking a new subprocess for each test bypasses these limitations.

Running a test in a separate subprocess is as easy as decorating it with pytest.mark.forked:

@pytest.mark.forked
def test_my_addon(anki_session):
    # add some tests in here

Automated Testing

pytest-anki is designed to work well with continuous integration systems such as GitHub actions. For an example see pytest-anki's own GitHub workflows.

Troubleshooting

pytest hanging when using xvfb

Especially if you run your tests headlessly with xvfb, you might run into cases where pytest will sometimes appear to hang. Oftentimes this is due to blocking non-dismissable prompts that Anki your add-on code might invoke in some scenarios. If you suspect that might be the case, my advice would be to temporarily bypass xvfb locally via pytest --no-xvfb to debug the issue.

License and Credits

pytest-anki is

Copyright © Ankitects Pty Ltd and contributors

Copyright © 2017-2019 Michal Krassowski (krassowski)

Copyright © 2019-2020 Aristotelis P. (glutanimate)

All credits for the original idea for this project go to Michal. pytest-anki would not exist without his work.

pytest-anki is free and open-source software. Its source-code is released under the GNU AGPLv3 license, extended by a number of additional terms. For more information please see the license file that accompanies this program.

This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY. Please see the license file for more details.