diff --git a/packages/pytest-simcore/src/pytest_simcore/helpers/pydantic_extension.py b/packages/pytest-simcore/src/pytest_simcore/helpers/pydantic_extension.py new file mode 100644 index 00000000000..c1252ed8bb4 --- /dev/null +++ b/packages/pytest-simcore/src/pytest_simcore/helpers/pydantic_extension.py @@ -0,0 +1,34 @@ +from pydantic import SecretStr + + +def _mask(value): + """ + Mask the password, showing only the first and last characters + or *** if very short passwords + """ + if len(value) > 2: + masked_value = value[0] + "*" * (len(value) - 2) + value[-1] + else: + # In case of very short passwords + masked_value = "*" * len(value) + return masked_value + + +def _hash(value): + """Uses hash number to mask the password""" + return f"hash:{hash(value)}" + + +class Secret4TestsStr(SecretStr): + """Prints a hint of the secret + TIP: Can be handy for testing + """ + + def _display(self) -> str | bytes: + # SEE overrides _SecretBase._display + value = self.get_secret_value() + return _mask(value) if value else "" + + +assert str(Secret4TestsStr("123456890")) == "1*******0" +assert "1*******0" in repr(Secret4TestsStr("123456890")) diff --git a/services/static-webserver/client/source/class/osparc/product/TIPTeaser.js b/services/static-webserver/client/source/class/osparc/product/TIPTeaser.js index d71ac819f3b..524d5795c7e 100644 --- a/services/static-webserver/client/source/class/osparc/product/TIPTeaser.js +++ b/services/static-webserver/client/source/class/osparc/product/TIPTeaser.js @@ -35,6 +35,11 @@ qx.Class.define("osparc.product.TIPTeaser", { }); this.getChildControl("teaser-text"); + + osparc.utils.Utils.setIdToWidget(this, "tipTeaserWindow"); + + const closeBtn = this.getChildControl("close-button"); + osparc.utils.Utils.setIdToWidget(closeBtn, "tipTeaserWindowCloseBtn"); }, statics: { diff --git a/tests/e2e-playwright/.gitignore b/tests/e2e-playwright/.gitignore index cf83940dd82..23b36998635 100644 --- a/tests/e2e-playwright/.gitignore +++ b/tests/e2e-playwright/.gitignore @@ -1,5 +1,5 @@ -test-results +.e2e-playwright-*.txt assets report.html -.e2e-playwright-*.txt report.xml +test-results diff --git a/tests/e2e-playwright/Makefile b/tests/e2e-playwright/Makefile index 4c8984e9aa5..88a15a845d1 100644 --- a/tests/e2e-playwright/Makefile +++ b/tests/e2e-playwright/Makefile @@ -117,6 +117,7 @@ CLASSIC_TIP_INPUT_FILE := .e2e-playwright-classictip-env.txt $(SLEEPERS_INPUT_FILE) $(JUPYTER_LAB_INPUT_FILE) $(CLASSIC_TIP_INPUT_FILE) $(S4L_INPUT_FILE): @read -p "Enter your product URL: " PRODUCT_URL; \ read -p "Is the product billable [y/n]: " BILLABLE; \ + read -p "Is the product lite [y/n]: " IS_LITE; \ read -p "Is the test running in autoscaled deployment [y/n]: " AUTOSCALED; \ read -p "Enter your username: " USER_NAME; \ read -s -p "Enter your password: " PASSWORD; echo ""; \ @@ -124,6 +125,9 @@ $(SLEEPERS_INPUT_FILE) $(JUPYTER_LAB_INPUT_FILE) $(CLASSIC_TIP_INPUT_FILE) $(S4L if [ "$$BILLABLE" = "y" ]; then \ echo "--product-billable" >> $@; \ fi; \ + if [ "$$IS_LITE" = "y" ]; then \ + echo "--product-lite" >> $@; \ + fi; \ if [ "$$AUTOSCALED" = "y" ]; then \ echo "--autoscaled" >> $@; \ fi; \ @@ -183,4 +187,4 @@ define run_test_on_chrome endef clean: - @rm -rf $(SLEEPERS_INPUT_FILE) $(JUPYTER_LAB_INPUT_FILE) $(CLASSIC_TIP_INPUT_FILE) + -@rm -rf $(SLEEPERS_INPUT_FILE) $(JUPYTER_LAB_INPUT_FILE) $(CLASSIC_TIP_INPUT_FILE) diff --git a/tests/e2e-playwright/README.md b/tests/e2e-playwright/README.md index 0caaa6c6adc..9c8e996b84d 100644 --- a/tests/e2e-playwright/README.md +++ b/tests/e2e-playwright/README.md @@ -1,5 +1,11 @@ + + +## Usage + ### Auto generate new test -`playwright codegen sim4life.io` +``` +playwright codegen sim4life.io +``` ### Run test locally with headed mode ``` @@ -7,16 +13,24 @@ pytest -s tests/sim4life.py --headed --browser chromium --product-billable --pr ``` ### Check test results output -`playwright show-trace test-results/tests-sim4life-py-test-billable-sim4life-chromium/trace.zip` +``` +playwright show-trace test-results/tests-sim4life-py-test-billable-sim4life-chromium/trace.zip +``` ### Run debug mode -`PWDEBUG=1 pytest -s tests/sim4life.py` +``` +PWDEBUG=1 pytest -s tests/sim4life.py +``` ### Run test in different browsers -`pytest -s tests/sim4life.py --tracing on --html=report.html --browser chromium --browser firefox` +``` +pytest -s tests/sim4life.py --tracing on --html=report.html --browser chromium --browser firefox +``` -### or in chrome/msedge -`pytest -s tests/sim4life.py --tracing on --html=report.html --browser-channel chrome` +### or in chrome/ms-edge +``` +pytest -s tests/sim4life.py --tracing on --html=report.html --browser-channel chrome +``` -### Runs in CI +## e2e CI - https://git.speag.com/oSparc/e2e-backend diff --git a/tests/e2e-playwright/tests/conftest.py b/tests/e2e-playwright/tests/conftest.py index 997ac6b7138..1136e4035cb 100644 --- a/tests/e2e-playwright/tests/conftest.py +++ b/tests/e2e-playwright/tests/conftest.py @@ -22,7 +22,7 @@ from playwright.sync_api import APIRequestContext, BrowserContext, Page, WebSocket from playwright.sync_api._generated import Playwright from pydantic import AnyUrl, TypeAdapter -from pytest import Item +from pytest_simcore.helpers.faker_factories import DEFAULT_TEST_PASSWORD from pytest_simcore.helpers.logging_tools import log_context from pytest_simcore.helpers.playwright import ( MINUTE, @@ -36,6 +36,7 @@ decode_socketio_42_message, web_socket_default_log_handler, ) +from pytest_simcore.helpers.pydantic_extension import Secret4TestsStr _PROJECT_CLOSING_TIMEOUT: Final[int] = 10 * MINUTE _OPENING_NEW_EMPTY_PROJECT_MAX_WAIT_TIME: Final[int] = 30 * SECOND @@ -79,6 +80,12 @@ def pytest_addoption(parser: pytest.Parser) -> None: default=False, help="Whether product is billable or not", ) + group.addoption( + "--product-lite", + action="store_true", + default=False, + help="Whether product is lite version or not", + ) group.addoption( "--autoscaled", action="store_true", @@ -116,7 +123,7 @@ def pytest_addoption(parser: pytest.Parser) -> None: # Dictionary to store start times of tests -_test_start_times = {} +_test_start_times: dict[str, datetime.datetime] = {} def pytest_runtest_setup(item): @@ -144,7 +151,7 @@ def _construct_graylog_url( return f"{monitoring_url}/graylog/search?{query}" -def pytest_runtest_makereport(item: Item, call): +def pytest_runtest_makereport(item: pytest.Item, call): """ Hook to add extra information when a test fails. """ @@ -171,7 +178,6 @@ def pytest_runtest_makereport(item: Item, call): ) diagnostics["duration"] = str(end_time - start_time) - # Print the diagnostics report with log_context( logging.WARNING, f"ℹ️ Diagnostics report for {test_name} ---", # noqa: RUF001 @@ -217,23 +223,29 @@ def user_name(request: pytest.FixtureRequest, auto_register: bool, faker: Faker) @pytest.fixture def user_password( request: pytest.FixtureRequest, auto_register: bool, faker: Faker -) -> str: +) -> Secret4TestsStr: if auto_register: - return faker.password(length=12) + return Secret4TestsStr(DEFAULT_TEST_PASSWORD) if osparc_password := request.config.getoption("--password"): assert isinstance(osparc_password, str) - return osparc_password - return os.environ["USER_PASSWORD"] + return Secret4TestsStr(osparc_password) + return Secret4TestsStr(os.environ["USER_PASSWORD"]) @pytest.fixture(scope="session") -def product_billable(request: pytest.FixtureRequest) -> bool: +def is_product_billable(request: pytest.FixtureRequest) -> bool: billable = request.config.getoption("--product-billable") return TypeAdapter(bool).validate_python(billable) @pytest.fixture(scope="session") -def autoscaled(request: pytest.FixtureRequest) -> bool: +def is_product_lite(request: pytest.FixtureRequest) -> bool: + enabled = request.config.getoption("--product-lite") + return TypeAdapter(bool).validate_python(enabled) + + +@pytest.fixture(scope="session") +def is_autoscaled(request: pytest.FixtureRequest) -> bool: autoscaled = request.config.getoption("--autoscaled") return TypeAdapter(bool).validate_python(autoscaled) @@ -280,7 +292,7 @@ def register( page: Page, product_url: AnyUrl, user_name: str, - user_password: str, + user_password: Secret4TestsStr, ) -> Callable[[], AutoRegisteredUser]: def _do() -> AutoRegisteredUser: with log_context( @@ -297,11 +309,13 @@ def _do() -> AutoRegisteredUser: for pass_id in ["registrationPass1Fld", "registrationPass2Fld"]: user_password_box = page.get_by_test_id(pass_id) user_password_box.click() - user_password_box.fill(user_password) + user_password_box.fill(user_password.get_secret_value()) with page.expect_response(re.compile(r"/auth/register")) as response_info: page.get_by_test_id("registrationSubmitBtn").click() assert response_info.value.ok, response_info.value.json() - return AutoRegisteredUser(user_email=user_name, password=user_password) + return AutoRegisteredUser( + user_email=user_name, password=user_password.get_secret_value() + ) return _do @@ -311,7 +325,7 @@ def log_in_and_out( page: Page, product_url: AnyUrl, user_name: str, - user_password: str, + user_password: Secret4TestsStr, auto_register: bool, register: Callable[[], AutoRegisteredUser], ) -> Iterator[WebSocket]: @@ -352,7 +366,7 @@ def log_in_and_out( _user_email_box.fill(user_name) _user_password_box = page.get_by_test_id("loginPasswordFld") _user_password_box.click() - _user_password_box.fill(user_password) + _user_password_box.fill(user_password.get_secret_value()) with page.expect_response(re.compile(r"/login")) as response_info: page.get_by_test_id("loginSubmitBtn").click() assert response_info.value.ok, f"{response_info.value.json()}" @@ -392,7 +406,7 @@ def log_in_and_out( def create_new_project_and_delete( page: Page, log_in_and_out: WebSocket, - product_billable: bool, + is_product_billable: bool, api_request_context: APIRequestContext, product_url: AnyUrl, ) -> Iterator[Callable[[tuple[RunningState], bool], dict[str, Any]]]: @@ -411,7 +425,7 @@ def _( ), "misuse of this fixture! only 1 study can be opened at a time. Otherwise please modify the fixture" with log_context( logging.INFO, - f"Open project in {product_url=} as {product_billable=}", + f"Open project in {product_url=} as {is_product_billable=}", ) as ctx: waiter = SocketIOProjectStateUpdatedWaiter(expected_states=expected_states) timeout = ( @@ -473,7 +487,7 @@ def wait_for_done(response): ... else: open_button.click() - if product_billable: + if is_product_billable: # Open project with default resources page.get_by_test_id("openWithResources").click() project_data = response_info.value.json() @@ -512,7 +526,7 @@ def wait_for_done(response): for project_uuid in created_project_uuids: with log_context( logging.INFO, - f"Delete project with {project_uuid=} in {product_url=} as {product_billable=}", + f"Delete project with {project_uuid=} in {product_url=} as {is_product_billable=}", ): response = api_request_context.delete( f"{product_url}v0/projects/{project_uuid}" diff --git a/tests/e2e-playwright/tests/sim4life/test_sim4life.py b/tests/e2e-playwright/tests/sim4life/test_sim4life.py index b993f262181..96c361bb546 100644 --- a/tests/e2e-playwright/tests/sim4life/test_sim4life.py +++ b/tests/e2e-playwright/tests/sim4life/test_sim4life.py @@ -31,7 +31,7 @@ def test_sim4life( log_in_and_out: WebSocket, service_key: str, use_plus_button: bool, - autoscaled: bool, + is_autoscaled: bool, check_videostreaming: bool, ): if use_plus_button: @@ -49,7 +49,11 @@ def test_sim4life( assert len(node_ids) == 1, "Expected 1 node in the workbench!" resp = wait_for_launched_s4l( - page, node_ids[0], log_in_and_out, autoscaled=autoscaled, copy_workspace=False + page, + node_ids[0], + log_in_and_out, + autoscaled=is_autoscaled, + copy_workspace=False, ) s4l_websocket = resp["websocket"] with web_socket_default_log_handler(s4l_websocket): diff --git a/tests/e2e-playwright/tests/sim4life/test_template.py b/tests/e2e-playwright/tests/sim4life/test_template.py index a4f104a6291..fb9b260c992 100644 --- a/tests/e2e-playwright/tests/sim4life/test_template.py +++ b/tests/e2e-playwright/tests/sim4life/test_template.py @@ -24,7 +24,7 @@ def test_template( create_project_from_template_dashboard: Callable[[str], dict[str, Any]], log_in_and_out: WebSocket, template_id: str, - autoscaled: bool, + is_autoscaled: bool, check_videostreaming: bool, ): project_data = create_project_from_template_dashboard(template_id) @@ -37,7 +37,7 @@ def test_template( assert len(node_ids) == 1, "Expected 1 node in the workbench!" resp = wait_for_launched_s4l( - page, node_ids[0], log_in_and_out, autoscaled=autoscaled, copy_workspace=True + page, node_ids[0], log_in_and_out, autoscaled=is_autoscaled, copy_workspace=True ) s4l_websocket = resp["websocket"] with web_socket_default_log_handler(s4l_websocket): diff --git a/tests/e2e-playwright/tests/tip/test_ti_plan.py b/tests/e2e-playwright/tests/tip/test_ti_plan.py index e721b7f5ab3..cac6bb5b56d 100644 --- a/tests/e2e-playwright/tests/tip/test_ti_plan.py +++ b/tests/e2e-playwright/tests/tip/test_ti_plan.py @@ -86,21 +86,41 @@ def __call__(self, message: str) -> bool: return False -def test_tip( # noqa: PLR0915 +def test_classic_ti_plan( # noqa: PLR0915 page: Page, - create_tip_plan_from_dashboard: Callable[[str], dict[str, Any]], log_in_and_out: WebSocket, - autoscaled: bool, + is_autoscaled: bool, + is_product_lite: bool, + create_tip_plan_from_dashboard: Callable[[str], dict[str, Any]], ): + with log_context(logging.INFO, "Checking 'Access TIP' teaser"): + page.get_by_test_id("userMenuBtn").click() + page.get_by_test_id("userMenuAccessTIPBtn").click() + assert page.get_by_test_id("tipTeaserWindow").is_visible() + page.get_by_test_id("tipTeaserWindowCloseBtn").click() + + # press + button project_data = create_tip_plan_from_dashboard("newTIPlanButton") assert "workbench" in project_data, "Expected workbench to be in project data!" assert isinstance( project_data["workbench"], dict ), "Expected workbench to be a dict!" node_ids: list[str] = list(project_data["workbench"]) - assert len(node_ids) >= 3, "Expected at least 3 nodes in the workbench!" - with log_context(logging.INFO, "Electrode Selector step") as ctx: + if is_product_lite: + expected_number_of_steps = 2 + assert ( + len(node_ids) == expected_number_of_steps + ), f"Expected {expected_number_of_steps=} in the app-mode" + else: + expected_number_of_steps = 3 + assert ( + len(node_ids) >= expected_number_of_steps + ), f"Expected at least {expected_number_of_steps} nodes in the workbench" + + with log_context( + logging.INFO, "Electrode Selector step (1/%s)", expected_number_of_steps + ) as ctx: # NOTE: creating the plan auto-triggers the first service to start, which might already triggers socket events electrode_selector_iframe = wait_for_service_running( page=page, @@ -108,7 +128,7 @@ def test_tip( # noqa: PLR0915 websocket=log_in_and_out, timeout=( _ELECTRODE_SELECTOR_AUTOSCALED_MAX_STARTUP_TIME - if autoscaled + if is_autoscaled else _ELECTRODE_SELECTOR_MAX_STARTUP_TIME ), press_start_button=False, @@ -149,13 +169,15 @@ def test_tip( # noqa: PLR0915 response_body = response.json() ctx.logger.info("the following output was generated: %s", response_body) - with log_context(logging.INFO, "Classic TI step") as ctx: + with log_context( + logging.INFO, "Classic TI step (2/%s)", expected_number_of_steps + ) as ctx: with page.expect_websocket( _JLabWaitForWebSocket(), timeout=_OUTER_EXPECT_TIMEOUT_RATIO * ( _JLAB_AUTOSCALED_MAX_STARTUP_TIME - if autoscaled + if is_autoscaled else _JLAB_MAX_STARTUP_MAX_TIME ), ) as ws_info: @@ -165,7 +187,7 @@ def test_tip( # noqa: PLR0915 websocket=log_in_and_out, timeout=( _JLAB_AUTOSCALED_MAX_STARTUP_TIME - if autoscaled + if is_autoscaled else _JLAB_MAX_STARTUP_MAX_TIME ), press_start_button=False, @@ -193,43 +215,76 @@ def test_tip( # noqa: PLR0915 ) with log_context(logging.INFO, "Create report"): + ti_iframe.get_by_role("button", name="Load Analysis").click() page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) ti_iframe.get_by_role("button", name="Load").nth(1).click() page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) - ti_iframe.get_by_role("button", name="Add to Report (0)").nth(0).click() - page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) - ti_iframe.get_by_role("button", name="Export to S4L").click() - page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) - ti_iframe.get_by_role("button", name="Add to Report (1)").nth(1).click() - page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) - ti_iframe.get_by_role("button", name="Export Report").click() - page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) + + if is_product_lite: + assert ( + not ti_iframe.get_by_role("button", name="Add to Report (0)") + .nth(0) + .is_enabled() + ) + assert not ti_iframe.get_by_role( + "button", name="Export to S4L" + ).is_enabled() + assert not ti_iframe.get_by_role( + "button", name="Export Report" + ).is_enabled() + + else: + ti_iframe.get_by_role("button", name="Add to Report (0)").nth(0).click() + page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) + ti_iframe.get_by_role("button", name="Export to S4L").click() + page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) + ti_iframe.get_by_role("button", name="Add to Report (1)").nth(1).click() + page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) + ti_iframe.get_by_role("button", name="Export Report").click() + page.wait_for_timeout(_JLAB_REPORTING_MAX_TIME) with log_context(logging.INFO, "Check outputs"): - expected_outputs = ["output_1.zip", "TIP_report.pdf", "results.csv"] - text_on_output_button = f"Outputs ({len(expected_outputs)})" - page.get_by_test_id("outputsBtn").get_by_text(text_on_output_button).click() + if is_product_lite: + expected_outputs = ["results.csv"] + text_on_output_button = f"Outputs ({len(expected_outputs)})" + page.get_by_test_id("outputsBtn").get_by_text( + text_on_output_button + ).click() - with log_context(logging.INFO, "Exposure Analysis step"): - with expected_service_running( - page=page, - node_id=node_ids[2], - websocket=log_in_and_out, - timeout=( - _POST_PRO_AUTOSCALED_MAX_STARTUP_TIME - if autoscaled - else _POST_PRO_MAX_STARTUP_TIME - ), - press_start_button=False, - ) as service_running: - app_mode_trigger_next_app(page) - s4l_postpro_iframe = service_running.iframe_locator - assert s4l_postpro_iframe - - with log_context(logging.INFO, "Post process"): - # click on the postpro mode button - s4l_postpro_iframe.get_by_test_id("mode-button-postro").click() - # click on the surface viewer - s4l_postpro_iframe.get_by_test_id("tree-item-ti_field.cache").click() - s4l_postpro_iframe.get_by_test_id("tree-item-SurfaceViewer").nth(0).click() + else: + expected_outputs = ["output_1.zip", "TIP_report.pdf", "results.csv"] + text_on_output_button = f"Outputs ({len(expected_outputs)})" + page.get_by_test_id("outputsBtn").get_by_text( + text_on_output_button + ).click() + + if is_product_lite: + assert expected_number_of_steps == 2 + else: + with log_context( + logging.INFO, "Exposure Analysis step (3/%s)", expected_number_of_steps + ): + with expected_service_running( + page=page, + node_id=node_ids[2], + websocket=log_in_and_out, + timeout=( + _POST_PRO_AUTOSCALED_MAX_STARTUP_TIME + if is_autoscaled + else _POST_PRO_MAX_STARTUP_TIME + ), + press_start_button=False, + ) as service_running: + app_mode_trigger_next_app(page) + s4l_postpro_iframe = service_running.iframe_locator + assert s4l_postpro_iframe + + with log_context(logging.INFO, "Post process"): + # click on the postpro mode button + s4l_postpro_iframe.get_by_test_id("mode-button-postro").click() + # click on the surface viewer + s4l_postpro_iframe.get_by_test_id("tree-item-ti_field.cache").click() + s4l_postpro_iframe.get_by_test_id("tree-item-SurfaceViewer").nth( + 0 + ).click()