Techdots

March 22, 2025

Advanced Cypress Strategies: Optimizing Test Performance and Reducing Flakiness

Are you struggling with flaky tests and slow execution times in your Cypress test suites? End-to-end (E2E) testing with Cypress offers powerful capabilities, but without proper optimization, flakiness and performance bottlenecks can hinder your CI/CD pipelines. Flaky tests—those that produce inconsistent results—can lead to increased debugging time and unreliable test outcomes. 

This guide dives into advanced strategies for optimizing test performance and reducing flakiness in Cypress, tailored for senior QA engineers aiming to enhance their automation workflows. From leveraging retry-ability and network stubbing to implementing parallelization and minimizing UI interactions, we’ll explore best practices to make your Cypress tests faster, more reliable, and CI/CD-ready. 

Whether you’re dealing with asynchronous operations, dynamic UI behavior, or unreliable network calls, these techniques will help you achieve stable and efficient test automation. Let’s transform your Cypress tests into a robust, high-performing asset for your development pipeline.

1. Understanding Flakiness in Cypress Tests

Flakiness in Cypress tests often stems from asynchronous operations, dynamic UI behavior, or unreliable network calls. Some common causes include:

  • Unstable test data
  • Improper element waiting strategies
  • Inconsistent API responses
  • Excessive reliance on UI interactions

Addressing these issues is critical to improving test reliability and ensuring consistent results in your CI/CD pipelines.

2. Enhancing Test Reliability

a. Implementing Retry-ability

Cypress automatically retries certain commands, but fine-tuning retry strategies improves test stability. Use cy.retry() to explicitly retry failed commands.

Example:

cy.get('.notification', { timeout: 10000 }).should('be.visible');
If the element does not appear immediately, Cypress retries until the timeout expires.

b. Using Proper Waiting Strategies

Avoid using cy.wait() with hardcoded delays. Instead, leverage cy.intercept() for network stubbing and dynamic waits.

Example:

cy.intercept('GET', '/api/data').as('getData');

cy.wait('@getData').its('response.statusCode').should('eq', 200);

This ensures the test proceeds only after receiving a successful API response.

c. Network Stubbing to Improve Speed

Stub API responses instead of waiting for real backend calls.

Example:

cy.intercept('GET', '/api/users', { fixture: 'users.json' }).as('getUsers');

cy.visit('/dashboard');

cy.wait('@getUsers');

This reduces test execution time and eliminates API flakiness.

3. Optimizing Test Execution

a. Parallelization for Faster CI/CD Execution

Running tests in parallel significantly improves efficiency. Most CI/CD tools like GitHub Actions and CircleCI support parallelization.

GitHub Actions Example:

jobs:

  cypress-run:

    runs-on: ubuntu-latest

    steps:

      - name: Install dependencies

        run: npm install

      - name: Run tests

        run: npx cypress run --record --parallel

This distributes tests across multiple machines for faster execution.

b. Load Balancing in CI/CD

Use Cypress Dashboard’s test parallelization feature to balance test execution dynamically.

CircleCI Example:

parallelism: 4

steps:

  - run: npx cypress run --record --parallel --group ci-tests

This ensures an even distribution of test workloads across available CI/CD resources.

4. Custom Command Optimizations

Creating custom Cypress commands reduces redundancy and speeds up test execution.

Example:

Cypress.Commands.add('login', (username, password) => {

  cy.request('POST', '/api/login', { username, password }).then((resp) => {

    window.localStorage.setItem('token', resp.body.token);

  });

});

This bypasses the UI login process, making tests faster and more reliable.

5. Minimizing UI Interactions

Reducing unnecessary UI interactions significantly improves test speed.

  • Use API calls instead of UI forms
  • Skip non-essential steps (e.g., animations, pop-ups)
  • Use cy.invoke() for direct property manipulation

Example: cy.get('.modal').invoke('hide'); // Avoids waiting for UI animations

Conclusion

By implementing these advanced Cypress strategies, QA engineers can dramatically improve test performance and reliability. Effective retry-ability, network stubbing, parallel execution, and minimizing UI interactions reduce flakiness while optimizing test speed. These best practices ensure stable, efficient test automation in CI/CD environments, making Cypress a powerful tool for modern development pipelines.

For expert assistance in implementing these strategies and optimizing your Cypress test suites, you should get in touch with TechDots today. Visit https://www.techdots.dev/ to learn more about how they can help you achieve seamless test automation.

Ready to start a project?

Let’s work together to ensure your digital space is inclusive and compliant. Reach out to our team and start building an application that works for everyone.

Book Meeting