Skip to content

Commit

Permalink
test(release): Add Python script to test UIs (#3232)
Browse files Browse the repository at this point in the history
Authored-by: Marco <[email protected]>
  • Loading branch information
mmicu authored Dec 6, 2023
1 parent 4b36d59 commit 826cf17
Show file tree
Hide file tree
Showing 2 changed files with 382 additions and 1 deletion.
42 changes: 41 additions & 1 deletion .github/workflows/e2e.yml
Original file line number Diff line number Diff line change
Expand Up @@ -152,10 +152,50 @@ jobs:
- name: Run e2e test
run: timeout 2400 bats "e2e/$E2E_TEST"

ui_test:
runs-on: ${{ matrix.os }}
needs: [build_dfx]
strategy:
matrix:
os: [macos-12, ubuntu-20.04, ubuntu-22.04]
steps:
- name: Checking out repo
uses: actions/checkout@v4
- name: Setting up Python
uses: actions/setup-python@v4
with:
python-version: '3.9'
- name: Installing playwright
run: |
pip install playwright==1.40.0
playwright install
playwright install-deps
- name: Download dfx binary
uses: actions/download-artifact@v3
with:
name: dfx-${{ matrix.os }}-rs-${{ hashFiles('rust-toolchain.toml') }}
path: /usr/local/bin
- name: Setup dfx binary
run: chmod +x /usr/local/bin/dfx
- name: Deploy default dfx project
run: |
dfx new e2e_project
cd e2e_project
dfx start --background --clean
dfx deploy 2>&1 | tee deploy.log
echo FRONTEND_URL=$(grep "_frontend:" deploy.log | grep -Eo "(http|https)://[a-zA-Z0-9./?=_&%:-]*") >> $GITHUB_ENV
echo CANDID_URL=$(grep "_backend:" deploy.log | grep -Eo "(http|https)://[a-zA-Z0-9./?=_&%:-]*") >> $GITHUB_ENV
- name: Running the Python script
run: |
python scripts/test-uis.py \
--frontend_url "$FRONTEND_URL" \
--candid_url "$CANDID_URL" \
--browser chromium firefox webkit
aggregate:
name: e2e:required
if: ${{ always() }}
needs: [test, smoke]
needs: [test, smoke, ui_test]
runs-on: ubuntu-latest
steps:
- name: check smoke test result
Expand Down
341 changes: 341 additions & 0 deletions scripts/test-uis.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,341 @@
"""
Automate frontend tests by using Playwright.
The script tests the following UIs:
1. Frontend UI.
2. Candid UI.
Examples:
$ python3 test-uis.py --frontend_url '...' --browser chromium firefox webkit # Only test the frontend UI
$ python3 test-uis.py --candid_url '...' --browser chromium firefox webkit # Only test the Candid UI
$ python3 test-uis.py --frontend_url '...' --candid_url '...' --browser chromium firefox webkit # Test both UIs
"""
import argparse
import logging
import re
import sys
import time
from enum import Enum

from playwright.sync_api import sync_playwright

_CHROMIUM_BROWSER = "chromium"
_FIREFOX_BROWSER = "firefox"
_WEBKIT_BROWSER = "webkit"
_SUPPORTED_BROWSERS = {
_CHROMIUM_BROWSER,
_FIREFOX_BROWSER,
_WEBKIT_BROWSER,
}
_CANDID_UI_WARNINGS_TO_IGNORE = [
("Error", "/index.js"),
("Invalid asm.js: Unexpected token", "/index.js"),
("Expected to find result for path [object Object], but instead found nothing.", "/index.js"),
(
"""
Error: Server returned an error:
Code: 404 (Not Found)
Body: Custom section name not found.
at j.readState (http://localhost:4943/index.js:2:11709)
at async http://localhost:4943/index.js:2:97683
at async Promise.all (index 0)
at async Module.UA (http://localhost:4943/index.js:2:98732)
at async Object.getNames (http://localhost:4943/index.js:2:266156)
at async http://localhost:4943/index.js:2:275479""".strip(),
"/index.js",
),
(
"""
Error: Server returned an error:
Code: 404 (Not Found)
Body: Custom section name not found.""".strip(),
"/index.js",
),
]
_CANDID_UI_ERRORS_TO_IGNORE = [
("Failed to load resource: the server responded with a status of 404 (Not Found)", "/read_state"),
]
# `page.route` does not support additional function parameters
_FRONTEND_URL = None


class _UI(Enum):
CANDID = 1
FRONTEND = 2


def _validate_browsers(browser):
if browser not in _SUPPORTED_BROWSERS:
logging.error(f"Browser {browser} not supported")
sys.exit(1)

return browser


def _get_argument_parser():
parser = argparse.ArgumentParser(description="Test the Frontend and Candid UIs")

parser.add_argument("--frontend_url", help="Frontend UI url")
parser.add_argument("--candid_url", help="Candid UI url")

parser.add_argument(
"--browsers",
nargs="+",
type=_validate_browsers,
help=f"Test against the specified browsers ({_SUPPORTED_BROWSERS})",
)

return parser


def _validate_args(args):
has_err = False

if not args.frontend_url and not args.candid_url:
logging.error('Either "--frontend_url" or "--candid_url" must be specified to start the tests')
has_err = True

if not args.browsers:
logging.error("At least one browser must be specified")
logging.error(f"Possible browsers: {_SUPPORTED_BROWSERS}")
has_err = True

if has_err:
sys.exit(1)


def _get_browser_obj(playwright, browser_name):
if browser_name == _CHROMIUM_BROWSER:
return playwright.chromium
if browser_name == _FIREFOX_BROWSER:
return playwright.firefox
if browser_name == _WEBKIT_BROWSER:
return playwright.webkit

return None


def _check_console_logs(console_logs):
logging.info("Checking console logs")

has_err = False
for log in console_logs:
if log.type not in {"warning", "error"}:
continue

# Skip all `Error with Permissions-Policy header: Unrecognized feature` warnings
perm_policy_warn = "Error with Permissions-Policy header:"
if perm_policy_warn in log.text:
logging.warning(f'Skipping Permissions-Policy warning. log.text="{log.text}"')
continue

url = log.location.get("url")
if not url:
raise RuntimeError(
f'Cannot find "url" during log parsing (log.type={log.type}, log.text="{log.text}", log.location="{log.location}")'
)

for actual_text, endpoint in (
_CANDID_UI_ERRORS_TO_IGNORE if log.type == "error" else _CANDID_UI_WARNINGS_TO_IGNORE
):
if actual_text == log.text.strip() and endpoint in url:
logging.warning(
f'Found {log.type}, but it was expected (log.type="{actual_text}", endpoint="{endpoint}")'
)
break
else:
logging.error(f'Found unexpected console log {log.type}. Text: "{log.text}", url: {url}')

has_err = True

if has_err:
raise RuntimeError("Console has unexpected warnings and/or errors. Check previous logs")

logging.info("Console logs are ok")


def _click_button(page, button):
logging.info(f'Clicking button "{button}"')
page.get_by_role("button", name=button).click()


def _set_text(page, text, value):
logging.info(f'Setting text to "{value}"')
page.get_by_role("textbox", name=text).fill(value)


def _test_frontend_ui_handler(page):
# Set the name & Click the button
name = "my name"
logging.info(f'Setting name "{name}"')
page.get_by_label("Enter your name:").fill(name)
_click_button(page, "Click Me!")

# Check if `#greeting` is populated correctly
greeting_id = "#greeting"
timeout_ms = 60000
greeting_obj = page.wait_for_selector(greeting_id, timeout=timeout_ms)
if greeting_obj:
actual_value = greeting_obj.inner_text()
expected_value = f"Hello, {name}!"
if actual_value == expected_value:
logging.info(f'"{actual_value}" found in "{greeting_id}"')
else:
raise RuntimeError(f'Expected greeting message is "{expected_value}", but found "{actual_value}"')
else:
raise RuntimeError(f"Cannot find {greeting_id} selector")


def _test_candid_ui_handler(page):
# Set the text & Click the "Query" button
text = "hello, world"
_set_text(page, "text", text)
_click_button(page, "Query")

# Check if `#output-list` is populated correctly (after the first click)
output_list_id = "#output-list"
timeout_ms = 60000
_ = page.wait_for_selector(output_list_id, timeout=timeout_ms)

# Reset the text & Click the "Random" button
_set_text(page, "text", "")
_click_button(page, "Random")
# ~

# Check if `#output-list` is populated correctly (after the second click)
#
# NOTE: `#output-list` already exists, so `wait_for_selector` won't work as expected.
# We noticed that, especially for `Ubuntu 20.04` and `Webkit`, the two additional lines
# created once the `Random` button was clicked, were not created properly.
#
# For this reason there is this simple fallback logic that tries to look at the selector
# for more than once by sleeping for some time.
fallback_retries = 10
fallback_sleep_sec = 5
last_err = None
for _ in range(fallback_retries):
try:
output_list_obj = page.wait_for_selector(output_list_id, timeout=timeout_ms)
if not output_list_obj:
raise RuntimeError(f"Cannot find {output_list_id} selector")

output_list_lines = output_list_obj.inner_text().split("\n")
actual_num_lines, expected_num_lines = len(output_list_lines), 4
if actual_num_lines != expected_num_lines:
err = [f"Expected {expected_num_lines} lines of text but found {actual_num_lines}"]
err.append("Lines:")
err.extend(output_list_lines)
raise RuntimeError("\n".join(err))

# Extract random text from third line
random_text = re.search(r'"([^"]*)"', output_list_lines[2])
if not random_text:
raise RuntimeError(f"Cannot extract the random text from the third line: {output_list_lines[2]}")
random_text = random_text.group(1)

for i, text_str in enumerate([text, random_text]):
line1, line2 = (i * 2), (i * 2 + 1)

# First output line
actual_line, expected_line = output_list_lines[line1], f'› greet("{text_str}")'
if actual_line != expected_line:
raise RuntimeError(f"Expected {expected_line} line, but found {actual_line} (line {line1})")
logging.info(f'"{actual_line}" found in {output_list_id} at position {line1}')

# Second output line
actual_line, expected_line = output_list_lines[line2], f'("Hello, {text_str}!")'
if actual_line != expected_line:
raise RuntimeError(f"Expected {expected_line} line, but found {actual_line} (line {line2})")
logging.info(f'"{actual_line}" found in {output_list_id} at position {line2}')

# All good!
last_err = None
logging.info(f"{output_list_id} lines are defined correctly")
break
except RuntimeError as run_err:
last_err = str(run_err)
logging.warning(f"Fallback hit! Sleeping for {fallback_sleep_sec} before continuing")
time.sleep(fallback_sleep_sec)

if last_err:
raise RuntimeError(last_err)


def _handle_route_for_webkit(route):
url = route.request.url.replace("https://", "http://")

headers = None
if any(map(url.endswith, [".css", ".js", ".svg"])):
global _FRONTEND_URL
assert _FRONTEND_URL
headers = {
"referer": _FRONTEND_URL,
}

response = route.fetch(url=url, headers=headers)
assert response.status == 200, f"Expected 200 status code, but got {response.status}. Url: {url}"
route.fulfill(response=response)


def _test_ui(ui, url, handler, browsers):
logging.info(f'Testing "{str(ui)}" at "{url}"')

has_err = False
with sync_playwright() as playwright:
for browser_name in browsers:
logging.info(f'Checking "{browser_name}" browser')
browser = _get_browser_obj(playwright, browser_name)
if not browser:
raise RuntimeError(f"Cannot determine browser object for browser {browser_name}")

try:
browser = browser.launch(headless=True)
context = browser.new_context()
page = context.new_page()

# Attach a listener to the page's console events
console_logs = []
page.on("console", lambda msg: console_logs.append(msg))

# Webkit forces HTTPS:
# - https://github.com/microsoft/playwright/issues/12975
# - https://stackoverflow.com/questions/46394682/safari-keeps-forcing-https-on-localhost
if ui == _UI.FRONTEND and browser_name == _WEBKIT_BROWSER:
global _FRONTEND_URL
_FRONTEND_URL = url
page.route("**/*", _handle_route_for_webkit)

page.goto(url)

handler(page)
_check_console_logs(console_logs)
except Exception as e:
logging.error(f"Error: {str(e)}")
has_err = True
finally:
if context:
context.close()
if browser:
browser.close()

if has_err:
sys.exit(1)


def _main():
args = _get_argument_parser().parse_args()
_validate_args(args)

if args.frontend_url:
_test_ui(_UI.FRONTEND, args.frontend_url, _test_frontend_ui_handler, args.browsers)
if args.candid_url:
_test_ui(_UI.CANDID, args.candid_url, _test_candid_ui_handler, args.browsers)

logging.info("DONE!")


if __name__ == "__main__":
logging.basicConfig(level=logging.INFO, format="%(asctime)s - %(levelname)s - %(message)s")
_main()

0 comments on commit 826cf17

Please sign in to comment.