In your Angular application if you are using RxJS Debounce and running Cypress test you may have run into times that your tests are not consistently getting past the debounce wait time and appear like they are flaky tests.

Debounce is a way to wait X number of milliseconds for something to happen before continuing such waiting for a user to stop typing in a field before making an API call. This way you are not making an API call for each character typed into the field.

In Cypress, you could just use a wait statement to get past the debounce time but adding time based wait statements in Cypress is an anti-pattern.

Instead in Cypress you should use the cy.clock() and cy.tick() commands to be able to forward the virtual time and cause debounce to fire. However, I found it was not consistently getting past the debounce. RxJS was acting like we had not waited for the debounce time.

Luckily, after much troubleshooting the solution ended up being quite simple and only involved test code changes.

Broken Code

In the code below, after the cy.tick(501) we would find that the call to the API never happened so it would fail on the wait statement. We also noticed that this tended to pass when running locally on our dev machines but would fail on our CI servers. This made it really difficult to replicate and troubleshoot since it appeared to be some kind of a race condition where our dev machines would run slower than our CI servers were.

cy.clock();
cy.server();
 cy.route({
        url: '/api/v1/collections/*',
        method: 'GET',
        status: 200,
        response: 'false'
      }).as('nameCheck');

cy.get('[data-cy="collectionTitle"]')
        .type(txtMax255)
        .tick(501)
        .wait("@nameCheck");

Working Code

The fix was pretty easy once we understood what we happening. With cy.clock() it overrides the native global functions related to time allowing them to be controlled synchronously via cy.tick(). This includes controlling:

  • setTimeout
  • clearTimeout
  • setInterval
  • clearInterval
  • Date objects

Overriding the setTimeout, clearTimeout, setInterval, and clearInterval is where the issue came in since those are key to the way debounce works. So we just needed to set the date and override the Date object.

In the working code below, the only change that we made from the broken code above was to change cy.clock() to cy.clock(new Date(), ['Date']); and then our debounce worked as expected.

cy.clock(new Date(), ['Date']);
cy.server();
 cy.route({
        url: '/api/v1/collections/*',
        method: 'GET',
        status: 200,
        response: 'false'
      }).as('nameCheck');

cy.get('[data-cy="collectionTitle"]')
        .type(txtMax255)
        .tick(501)
        .wait("@nameCheck");

Now all of our debounce calls are working as expected in our Cypress tests.