Common support commands for Cypress when interacting with Drupal.
Table of Contents
- Cypress installed on your local or CI.
- @TODO Document how we have Cypress set up.
Add two entries in composer.json for an install-type and its path:
"installer-types": ["cypress-support"],
"installer-paths": {
// existing entries omitted...
"tests/cypress/cypress/support/{$name}": [
"type:cypress-support"
]
}
In the support folder for where your Cypress tests are located, edit commands.js and add the
following:
// Import commands.js using ES2015 syntax:
import './shrubs/commands'
The Shrubs repository is available via Packagist.
Once you have completed the steps above, run the following command:
composer require kanopi/shrubs
@TODO (See what Paul did on Parks)
Will select the first match from an autocomplete field.
cy.autocomplete('input[data-drupal-selector="edit-field-episode-show-0-target-id"]', 'Term Name')
Gets the value of a ckeditor instance.
cy.ckeditorGet('#edit-body-wrapper').should('contain', 'hello world')
Set the value of a ckeditor instance.
cy.ckeditorType('#field_body-wrapper', 'hello world');
Runs Drush commands in multiple environments. With the correct configuration it can target the following:
- Docksal
- Lando
- DDEV
- Pantheon
- Tugboat
cy.drush('status');
Set these as environment variables or in your cypress.env.json Docksal
{
"DRUSH_IS_DOCKSAL" : true
}Lando
{
"DRUSH_IS_LANDO" : true
}DDEV
{
"DRUSH_IS_DDEV" : true
}Pantheon
In the format of PANTHEON_SITE_ID.ENVIRONMENT_ID
{
"DRUSH_IS_PANTHEON" : "mysite.pr-123"
}Tugboat
DRUSH_IS_TUGBOATis your Tugboat tokenTUGBOAT_INSTANCE_IDis the ID of the specific Tugboat instance that is targeted $TUGBOAT_PREVIEW_ID
{
"DRUSH_IS_TUGBOAT" : "12345abcdef",
"TUGBOAT_INSTANCE_ID" : "1234567890"
}The Tugboat CLI needs a little extra help being installed in AMD64 architecture. For example if you are install the CLI within a CI/CD system like CircleCI or GitHub Actions.
sudo dpkg --add-architecture amd64
sudo apt-get update
sudo apt-get install libc6:amd64 libstdc++6:amd64
wget https://dashboard.tugboatqa.com/cli/linux/tugboat.tar.gz
sudo tar -zxf tugboat.tar.gz -C /usr/local/bin/Login through the default Drupal login form. Sets a default login but also passing custom login details
cy.login(); // login as a default user.
cy.login('user', 'password'); // as a specific user
Assuming there is some other process to create the user.
Uses a Drush one time login links to login as a specific user.
cy.loginOneTimeLink('myusername');
Logs out of the current session
cy.logout();
There a clicks that can generate a blocking ajax request. I.E. Opening modals or slideouts that load content with an ajax request.
The function will wrap an intercept/wait combination around the click to make sure the tests don't continue until the ajax request as completed.
It's also meant to deal with the anti-pattern of using wait() for clicks that trigger ajax requests.
Example
cy.ajaxClick("a.product-name", '/jsonapi/*/**')
This replaces code that would look like this.
const jsonApiRequest5 = 'jsonApiRequest' + Math.random();
cy.intercept('GET', '/jsonapi/*/**').as(jsonApiRequest5)
cy.get('a.product-name').click();
cy.wait('@' + jsonApiRequest5).its('response.statusCode').should('eq', 200)
or
cy.get('a.product-name').click();
cy.wait(5000)
Uploads a file to the media library and selects it in the field.
Can optionally set the type of media uploaded if there is more than one type available.
Files are expected to be in the fixtures folder at the same level as support and
e2e. In most cases, that will be /tests/cypress/cypress/fixtures.
cy.mediaLibraryAdd('#field_media_assets-media-library-wrapper', 'sample.png');
cy.mediaLibraryAdd('#field_media_assets-media-library-wrapper', 'sample.mp3', 'audio');
Open a media browser modal and selects an existing media item
Can optionally set the type of media uploaded if there is more than one type available.
Files are expected to be in the fixtures folder.
cy.mediaLibrarySelect('#field_media_assets-media-library-wrapper', 'sample.png');
cy.mediaLibrarySelect('#field_media_assets-media-library-wrapper', 'sample.png', 'image');
Upload a file through a file field
Files should be in the fixtures folder.
cy.uploadFile('#file-field-wrapper', 'example.png');`
Logs a message and stores it in an internal array to be retrieved later using
cy.logSummary();
cy.logAndStore('This is a log message.');
Outputs all stored log messages at the end of a test run.
cy.logSummary();
Example with after() Hook
describe('Example Test with log summary', () => {
beforeEach(() => {
cy.visit('/example-page');
});
it('should log multiple steps', () => {
cy.logAndStore('Step 1: Visiting the page.');
cy.logAndStore('Step 2: Clicking a button.');
cy.logAndStore('Step 3: Validating output.');
});
after(() => {
cy.logSummary();
});
});
In May 2025 Pantheon added an Interstitial page to projects that are at the Sandbox level. Kanopi has the following work around of overwriting the visit() command.
In e2e.js
/**
* Overwrite the visit() command to support setting headers globally.
*/
Cypress.Commands.overwrite("visit", (originalVisit, url, options = {}) => {
const globalHeaders = Cypress.env('visitHeaders') || {};
// Combine global and specific headers from the unique call of visit().
const headers = Object.assign({}, globalHeaders, options.headers);
// Call the real visit with the merged headers
return originalVisit(url, { ...options, headers });
});In your cypress.config.js you can add the following environment variable.
env: {
"visitHeaders" : {
"Deterrence-Bypass" : "1"
}
}Now when the visit() command is done you dont have to worry about a cookie being set prior.
If you want to set the cookie to bypass the page later you can do so with.
cy.setCookie("Deterrence-Bypass", "1");For issues and support, please use the issue queue at https://www.drupal.org/project/issues/shrubs?categories=All
Current maintainers:
This project is sponsored by: