[random test failure] JSWebAssertTest::testJsWebAssert

Created on 24 March 2025, 13 days ago

Problem/Motivation

Drupal\FunctionalJavascriptTests\Tests\JSWebAssertTest::testJsWebAssert sometimes fails randomly as per the output below.

    JSWeb Assert (Drupal\FunctionalJavascriptTests\Tests\JSWebAssert)
     ✘ Js web assert
       ┐
       β”œ WebDriver\Exception\StaleElementReference: stale element reference: stale element not found in the current frame
       β”œ   (Session info: chrome-headless-shell=127.0.6533.119)                                                          
       β”‚
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/php-webdriver/lib/WebDriver/Exception.php:198
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/php-webdriver/lib/WebDriver/AbstractWebDriver.php:227
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/php-webdriver/lib/WebDriver/Execute.php:56
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/php-webdriver/lib/WebDriver/Session.php:530
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/mink-selenium2-driver/src/Selenium2Driver.php:353
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/mink-selenium2-driver/src/Selenium2Driver.php:315
       β”‚ /builds/issue/drupal-1822440/vendor/lullabot/mink-selenium2-driver/src/Selenium2Driver.php:621
       β”‚ /builds/issue/drupal-1822440/vendor/behat/mink/src/Element/NodeElement.php:142
       β”‚ /builds/issue/drupal-1822440/core/tests/Drupal/FunctionalJavascriptTests/Tests/JSWebAssertTest.php:124
       β”΄

Here's the relevant part of the test, the last line being the one that fails ($result->getAttribute('id')):

// Ensure that the javascript has replaced the element 1100 times.
$assert_session->waitForText('New Text!! 1100');
$result = $page->find('named', ['id', 'test_text']);
$this->assertSame('test_text', $result->getAttribute('id'));

The waitForText call returns TRUE or FALSE, but we don't check which we get. If the element doesn't exist we continue anyway.

Furthermore, it's possible the controller logic is flawed:

// Clear the interval after 1.1 seconds as this is shorter than the 10 seconds
// JSWebAssert::waitForElementVisible() waits for.

The comments suggest the the loop should take 1.1 seconds to complete, but in my browser it feels more like 5 seconds. I added some profiling and it varies from 4300 ms to 4600 ms for me. It's not unthinkable that this could take more than 10 seconds in some CI environments.

So in the case of a slow runner, waitForText is possibly exceeding the threshold. Then $page->find finds an element from loop number 1050 (for example). Then by the time we try to fetch the id attribute, the element has been replaced a further 50 times.

Steps to reproduce

https://git.drupalcode.org/issue/drupal-1822440/-/jobs/4730776

Proposed resolution

  1. Refactor the controller so there's no way it takes 10+ seconds, but still requires a wait
  2. Assert that we've actually found an element after waiting, possibly also assert it doesn't exist first
  3. Ensure the loop has stopped by the time we try to fetch the attribute from the element

Remaining tasks

User interface changes

Introduced terminology

API changes

Data model changes

Release notes snippet

πŸ› Bug report
Status

Active

Version

11.0 πŸ”₯

Component

phpunit

Created by

πŸ‡¦πŸ‡ΊAustralia mstrelan

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Merge Requests

Comments & Activities

Production build 0.71.5 2024