forked from rdmorganiser/rdmo
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
test: add frontend-tests with pytest-playwright
Co-authored-by: David Wallace Fixes: rdmorganiser#715
- Loading branch information
1 parent
7aa821f
commit 49d0a5e
Showing
1 changed file
with
192 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,192 @@ | ||
import os | ||
import re | ||
from dataclasses import dataclass | ||
from urllib.parse import urlparse | ||
|
||
import pytest | ||
|
||
from django.core.management import call_command | ||
|
||
from playwright.sync_api import Page, expect | ||
from pytest_django.live_server_helper import LiveServer | ||
|
||
from rdmo.accounts.utils import set_group_permissions | ||
from rdmo.conditions.models import Condition | ||
from rdmo.core.models import Model | ||
from rdmo.domain.models import Attribute | ||
from rdmo.options.models import Option, OptionSet | ||
from rdmo.questions.models import Catalog, Question, Section | ||
from rdmo.questions.models import Page as PageModel | ||
from rdmo.questions.models.questionset import QuestionSet | ||
from rdmo.tasks.models import Task | ||
from rdmo.views.models import View | ||
|
||
pytestmark = pytest.mark.e2e | ||
|
||
# needed for playwright to run | ||
os.environ.setdefault("DJANGO_ALLOW_ASYNC_UNSAFE", "true") | ||
|
||
|
||
@dataclass | ||
class ModelHelper: | ||
"""Helper class to bundle information about models for test cases.""" | ||
|
||
model: Model | ||
form_field: str = "URI Path" | ||
db_field: str = "uri_path" | ||
has_nested: bool = False | ||
|
||
@property | ||
def url(self) -> str: | ||
return f"{self.model._meta.model_name}s" | ||
|
||
@property | ||
def verbose_name(self) -> str: | ||
"""Return the verbose_name for the model.""" | ||
return self.model._meta.verbose_name | ||
|
||
@property | ||
def verbose_name_plural(self) -> str: | ||
"""Return the verbose_name_plural for the model.""" | ||
return self.model._meta.verbose_name_plural | ||
|
||
|
||
@pytest.fixture(scope="function") | ||
def e2e_tests_django_db_setup(django_db_setup, django_db_blocker, fixtures): | ||
"""Set up database and populate with fixtures, that get restored for every test case.""" | ||
with django_db_blocker.unblock(): | ||
call_command("loaddata", *fixtures) | ||
set_group_permissions() | ||
|
||
|
||
@pytest.fixture(scope="session") | ||
def base_url(live_server: LiveServer) -> str: | ||
"""Enable playwright to address URLs with base URL automatically prefixed.""" | ||
return live_server.url | ||
|
||
|
||
@pytest.fixture | ||
def logged_in_admin_user(e2e_tests_django_db_setup, page: Page) -> Page: | ||
"""Log in as admin user through django login UI, returns logged in page for e2e tests.""" | ||
page.goto("/account/login") | ||
page.get_by_label("Username").fill("admin", timeout=5000) | ||
page.get_by_label("Password").fill("admin") | ||
page.get_by_role("button", name="Login").click() | ||
page.goto("/management") | ||
yield page | ||
|
||
|
||
model_helpers = ( | ||
ModelHelper(Catalog, has_nested=True), | ||
ModelHelper(Section, has_nested=True), | ||
ModelHelper(PageModel, has_nested=True), | ||
ModelHelper(QuestionSet, has_nested=True), | ||
ModelHelper(Question), | ||
ModelHelper( | ||
Attribute, has_nested=True, form_field="Key", db_field="key" | ||
), | ||
ModelHelper(OptionSet, has_nested=True), | ||
ModelHelper(Option), | ||
ModelHelper(Condition), | ||
ModelHelper(Task), | ||
ModelHelper(View), | ||
) | ||
|
||
@pytest.mark.parametrize("helper", model_helpers) | ||
def test_management_navigation(logged_in_admin_user: Page, helper: ModelHelper) -> None: | ||
"""Test that each content type is available through the navigation.""" | ||
page = logged_in_admin_user | ||
expect(page.get_by_role("heading", name="Management")).to_be_visible() | ||
|
||
# click a link in the navigation | ||
name = helper.verbose_name_plural | ||
page.get_by_role("link", name=name, exact=True).click() | ||
|
||
# make sure the browser opened a new page | ||
url_name = name.lower() | ||
url_name = url_name.replace(" ", "") | ||
expect(page).to_have_url(re.compile(rf".*/{url_name}/")) | ||
|
||
|
||
@pytest.mark.parametrize("helper", model_helpers) | ||
def test_management_has_items(logged_in_admin_user: Page, helper: ModelHelper) -> None: | ||
"""Test all items in database are visible in management UI.""" | ||
page = logged_in_admin_user | ||
num_items_in_database = helper.model.objects.count() | ||
page.goto(f"/management/{helper.url}") | ||
items_in_ui = page.locator(".list-group > .list-group-item") | ||
expect(items_in_ui).to_have_count(num_items_in_database) | ||
|
||
|
||
@pytest.mark.parametrize("helper", model_helpers) | ||
def test_management_nested_view( | ||
logged_in_admin_user: Page, helper: ModelHelper | ||
) -> None: | ||
"""For each element type, that has a nested view, click the first example.""" | ||
page = logged_in_admin_user | ||
page.goto(f"/management/{helper.url}") | ||
# Open nested view for element type | ||
if helper.has_nested: | ||
page.get_by_title(f"View {helper.verbose_name} nested").first.click() | ||
expect(page.locator(".panel-default").first).to_be_visible() | ||
expect(page.locator(".panel-default > .panel-body").first).to_be_visible() | ||
|
||
|
||
@pytest.mark.parametrize("helper", model_helpers) | ||
def test_management_create_model( | ||
logged_in_admin_user: Page, helper: ModelHelper | ||
) -> None: | ||
"""Test management UI can create objects in the database.""" | ||
page = logged_in_admin_user | ||
num_objects_at_start = helper.model.objects.count() | ||
page.goto(f"/management/{helper.url}") | ||
# click "New" button | ||
page.get_by_role("button", name="New").click() | ||
# fill mandatory fields | ||
value = "some-value" | ||
page.get_by_label(helper.form_field).fill(value) | ||
if helper.model == Condition: | ||
# conditions need to have a source attribute | ||
source_form = ( | ||
page.locator(".form-group") | ||
.filter(has_text="Source") | ||
.locator(".select-item > .react-select") | ||
) | ||
source_form.click() | ||
page.keyboard.type(Attribute.objects.first().uri) | ||
page.keyboard.press("Enter") | ||
|
||
# save | ||
page.get_by_role("button", name="Create").nth(1).click() | ||
# check if new item is in list | ||
items_in_ui = page.locator(".list-group > .list-group-item") | ||
expect(items_in_ui).to_have_count(num_objects_at_start + 1) | ||
|
||
num_objects_after_save = helper.model.objects.count() | ||
assert num_objects_after_save - num_objects_at_start == 1 | ||
query = {helper.db_field: value} | ||
assert helper.model.objects.get(**query) | ||
|
||
|
||
@pytest.mark.parametrize("helper", model_helpers) | ||
def test_management_edit_model(logged_in_admin_user: Page, helper: ModelHelper) -> None: | ||
page = logged_in_admin_user | ||
page.goto(f"/management/{helper.url}") | ||
# click edit | ||
edit_button_title = f"Edit {helper.verbose_name}" | ||
page.get_by_title(f"{edit_button_title}").first.click() | ||
# edit | ||
page.get_by_label("Comment").click() | ||
comment = "this is a comment." | ||
page.get_by_label("Comment").fill(comment) | ||
# save | ||
page.get_by_role("button", name="Save").nth(1).click() | ||
# click on edit element again | ||
page.get_by_title(f"{edit_button_title}").first.click() | ||
# check the updated comment | ||
comment_locator = page.get_by_label("Comment") | ||
expect(comment_locator).to_have_text(comment) | ||
# compare the updated comment with element object from db | ||
url_id = int(urlparse(page.url).path.rstrip("/").split("/")[-1]) | ||
model_obj = helper.model.objects.get(id=url_id) | ||
assert model_obj.comment == comment |