/testcontainers-cypress

Testcontainers module for running Cypress tests

Primary LanguageJavaApache License 2.0Apache-2.0

Cypress Testcontainer

Goal

The goal of this project is to make it easy to start Cypress tests via Testcontainers.

Example usage

Setup Cypress

  1. Create src/test/e2e directory in your project.

  2. Run npm init in that directory. This will generate a package.json file.

  3. Run npm install cypress --save-dev to install Cypress as a development dependency.

  4. Run npx cypress open to start the Cypress application. It will notice that this is the first startup and add some example tests.

  5. Run npm install cypress-multi-reporters mocha mochawesome --save-dev to install Mochawesome as a test reporter. Testcontainers-cypress will use that to parse the results of the Cypress tests.

  6. Update cypress.config.js as follows:

    const { defineConfig } = require('cypress')
    
    module.exports = defineConfig({
      e2e: {
        baseUrl: 'http://localhost:8080/',
        reporter: 'cypress-multi-reporters',
        reporterOptions: {
          configFile: 'reporter-config.json'
        }
      }
    })
  7. Create a reporter-config.json file (next to cypress.json) and ensure it contains:

    {
      "reporterEnabled": "spec, mochawesome",
      "mochawesomeReporterOptions": {
        "reportDir": "cypress/reports/mochawesome",
        "overwrite": false,
        "html": false,
        "json": true
      }
    }
Tip

Add the following to your .gitignore to avoid accidental commits:

node_modules
src/test/e2e/cypress/reports
src/test/e2e/cypress/videos
src/test/e2e/cypress/screenshots

Add dependency

Maven

Use this dependency if you use Maven:

<dependency>
    <groupId>io.github.wimdeblauwe</groupId>
    <artifactId>testcontainers-cypress</artifactId>
    <version>${tc-cypress.version}</version>
    <scope>test</scope>
</dependency>

Add src/test/e2e as a test resource directory:

<project>
    <build>
        ...
        <testResources>
            <testResource>
                <directory>src/test/resources</directory>
            </testResource>
            <testResource>
                <directory>src/test/e2e</directory>
                <targetPath>e2e</targetPath>
            </testResource>
        </testResources>
    </build>
</project>

Gradle

For Gradle, use the following dependency:

testImplementation 'io.github.wimdeblauwe:testcontainers-cypress:${tc-cypress.version}'

Process src/test/e2e as a test resource directory and copy it into buid/resources/test/e2e.

processTestResources {
    from("src/test/e2e") {
        exclude 'node_modules'
        into("e2e")
    }
}

The default GatherTestResultsStrategy assumes a Maven project and therefore you need to create a custom strategy that fits the Gradle layout.

MochawesomeGatherTestResultsStrategy gradleTestResultStrategy = new MochawesomeGatherTestResultsStrategy(
    FileSystems.getDefault().getPath("build", "resources", "test", "e2e", "cypress", "reports", "mochawesome"));

new CypressContainer()
     .withGatherTestResultsStrategy(gradleTestResultStrategy)

Usage with a @SpringBootTest

The library is not tied to Spring Boot, but I will use the example of a @SpringBootTest to explain how to use it.

Suppose you have a Spring Boot application that has server-side rendered templates using Thymeleaf, and you want to write some UI tests using Cypress. We want to drive all this from a JUnit based test, so we do the following:

  1. Have Spring Boot start the complete application in a test. This is easy using the @SpringBootTest annotation on a JUnit test.

  2. Expose the web port that was opened towards Testcontainers so that Cypress that is running in a Docker container can access our started web application.

  3. Start the Docker container to run the Cypress tests.

  4. Wait for the tests to be done and report the results to JUnit.

Start by writing the following JUnit test:

@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT) //(1)
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE) //(2)
public class CypressEndToEndTests {

    @LocalServerPort //(3)
    private int port;

     @Test
    void runCypressTests() throws InterruptedException, IOException, TimeoutException {

        Testcontainers.exposeHostPorts(port); //(4)

        try (CypressContainer container = new CypressContainer().withLocalServerPort(port)) { //(5)
            container.start();
            CypressTestResults testResults = container.getTestResults(); //(6)

            if (testResults.getNumberOfFailingTests() > 0) {
                fail("There was a failure running the Cypress tests!\n\n" + testResults); //(7)
            }
        }
    }
}
  1. Have Spring Boot start the full application on a random port.

  2. Tell Spring Boot to not configure a test database, Because we use a real database (via Testcontainers obviously :-) ).

  3. Have Spring inject the random port that was used when starting the application.

  4. Ensures that the container will be able to access the Spring Boot application that is started via @SpringBootTest

  5. Create the CypressContainer and pass in the port so the base URL that Cypress will use is correct.

  6. Wait on the tests and get the results.

  7. Check if there have been failures in Cypress. If so, fail the test.

JUnit 5 dynamic tests

If you are using JUnit 5, then you can use a @TestFactory annotated method so that it looks like there is a JUnit test for each of the Cypress tests.

@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
public class CypressEndToEndTests {

    @LocalServerPort
    private int port;

     @TestFactory // (1)
    List<DynamicContainer> runCypressTests() throws InterruptedException, IOException, TimeoutException {

        Testcontainers.exposeHostPorts(port);

        try (CypressContainer container = new CypressContainer().withLocalServerPort(port)) {
            container.start();
            CypressTestResults testResults = container.getTestResults();

             return convertToJUnitDynamicTests(testResults); // (2)
        }
    }

    @NotNull
    private List<DynamicContainer> convertToJUnitDynamicTests(CypressTestResults testResults) {
        List<DynamicContainer> dynamicContainers = new ArrayList<>();
        List<CypressTestSuite> suites = testResults.getSuites();
        for (CypressTestSuite suite : suites) {
            createContainerFromSuite(dynamicContainers, suite);
        }
        return dynamicContainers;
    }

    private void createContainerFromSuite(List<DynamicContainer> dynamicContainers, CypressTestSuite suite) {
        List<DynamicTest> dynamicTests = new ArrayList<>();
        for (CypressTest test : suite.getTests()) {
            dynamicTests.add(DynamicTest.dynamicTest(test.getDescription(), () -> {
                if (!test.isSuccess()) {
                    LOGGER.error(test.getErrorMessage());
                    LOGGER.error(test.getStackTrace());
                }
                assertTrue(test.isSuccess());
            }));
        }
        dynamicContainers.add(DynamicContainer.dynamicContainer(suite.getTitle(), dynamicTests));
    }
}
  1. Use the @TestFactory annotated as opposed to the @Test method

  2. Use the CypressTestResults to generate DynamicTest and DynamicContainer instances

If the Cypress tests look like this:

context('Todo tests', () => {
   it('should show a message if there are no todo items', () => {
       cy.request('POST', '/api/integration-test/clear-all-todos');
       cy.visit('/todos');
       cy.get('h1').contains('TODO list');
       cy.get('#empty-todos-message').contains('There are no todo items');
   });

   it('should show all todo items', () => {
       cy.request('POST', '/api/integration-test/prepare-todo-list-items');
       cy.visit('/todos');
       cy.get('h1').contains('TODO list');
       cy.get('#todo-items-list')
           .children()
           .should('have.length', 2)
           .should('contain', 'Add Cypress tests')
           .and('contain', 'Write blog post');
   })
});

Then running the JUnit test will show this in the IDE:

Cypress tests in JUnit with IntelliJ

This makes it a lot easier to see which Cypress test has failed.

Configuration options

The CypressContainer instance can be customized with the following options:

Method Description Default

CypressContainer(String dockerImageName)

Allows to specify the docker image to use

cypress/included:4.0.1

withLocalServerPort(int port)

Set the port where the server is running on. It will use http://host.testcontainers.internal as hostname with the given port as the Cypress base URL. For a @SpringBootTest, pass the injected @LocalServerPort here.

8080

withBaseUrl(String baseUrl)

Set the full server url that will be used as base URL for Cypress.

http://host.testcontainers.internal:8080

withBrowser(String browser)

Set the browser to use when running the tests (E.g. electron, chrome, firefox)

electron

withSpec(String spec)

Sets the test(s) to run. This can be a single test (e.g. cypress/integration/todos.spec.js) or multiple (e.g. cypress/integration/login/**)

By default (meaning not calling this method), all tests are run.

withRecord()

Passes the --record flag on the command line to record the test results on the Cypress Dashboard. The CYPRESS_RECORD_KEY environment variable needs to be set for this to work.

Not enabled by default

withRecord(String recordKey)

Passes the --record flag on the command line to record the test results on the Cypress Dashboard using the given record key.

Not enabled by default

withClasspathResourcePath(String resourcePath)

Set the relative path of where the cypress tests are (the path is the location of where the cypress.json file is)

e2e

withMaximumTotalTestDuration(Duration duration)

Set the maximum timeout for running the Cypress tests.

Duration.ofMinutes(10)

withGatherTestResultsStrategy(GatherTestResultsStrategy strategy)

Set the GatherTestResultsStrategy object that should be used for gathering information on the Cypress tests results.

MochawesomeGatherTestResultsStrategy

withMochawesomeReportsAt(Path path)

Set the path (relative to the root of the project) where the Mochawesome reports are put.

FileSystems.getDefault().getPath("target", "test-classes", "e2e", "cypress", "reports", "mochawesome")

withAutoCleanReports(boolean autoCleanReports)

Set if the Cypress test reports should be automatically cleaned before each run or not.

true

Testcontainers & Cypress versions compatibility

Testcontainers-cypress Testcontainers Cypress

1.9.0

1.19.1

13.3.0

1.8.0

1.17.6

12.9.0

1.7.1

1.17.5

10.11.0

1.7.0

1.17.3

10.7.0

1.6.3

1.17.3

9.7.0

1.6.2

1.16.3

9.1.0

1.6.1

1.16.2

9.1.0

1.6.0

1.16.2

9.1.0

1.5.0

1.16.2

8.7.0

1.4.0

1.16.2

7.7.0

1.3.0

1.15.2

6.8.0

1.2.1

1.15.1

5.6.0

1.2.0

1.15.0

5.6.0

1.1.0

1.15.0

5.5.0

1.0.0

1.14.3

4.12.1

0.7.0

1.14.1

4.5.0

0.6.0

1.13.0

4.3.0

0.5.0

1.12.5

4.0.2

0.4.0

1.12.5

4.0.1

0.3.0

1.12.3

3.8.3

0.2.0

1.12.3

3.8.1

0.1.0

1.12.3

3.8.0

Troubleshooting

When running on a mac with an M1 or M2 chip, only the electron browser is included in the underlying cypress image this project uses. For more info please read this.

There is a workaround, however, which involves following steps:

  1. install rosetta 2: softwareupdate --install-rosetta

  2. Enable rosetta emulation in docker desktop for mac:

    • General > Use virtualization framework

    • Features in Development > Use rosetta for x86/amd64 simulation on apple silicon

  3. Pull the platform specific (linux/amd64) image for the cypress-included which you are using. For example: docker pull --platform linux/amd64 cypress/included:13.3.0

Links to blog or articles that cover testcontainers-cypress:

Example usage of testcontainers cypress

Good introduction on how to get started.

Testcontainers-cypress release 0.4.0

Shows how to run tests on multiple browsers with JUnit

Development

Deployment

Release

Release is done via the Maven Release Plugin:

mvn release:prepare

and

mvn release:perform

Finally, push the local commits and the tag to remote.

Note

Before releasing, run export GPG_TTY=$(tty)