Tracking memory leaks using Cypress

Dev Gravity
Tracking memory leaks using Cypress

TLDR;

  1. Monitor memory inside an E2E test
  2. Try a fix
  3. Run the test again
  4. Compare memory usage between runs
  5. GOTO step 1.

Last week, we were notified by our colleague Christiano that the Gravity application was facing quite a huge problem of memory leaks. As a result, we decided to investigate the issue further.

Tracking memory leaks using Cypress

Although we were not able to reach such peaks as the ones Christiano was facing, a quick look at the “Memory” tab of the Chrome inspector was enough to ensure the issue was present. The situation prompted us to look for an efficient way to address the problem.

One of the biggest changes we had released in the previous days was to use a beta version of gravity-data-collector. This new version was collecting screenshots of the visited pages alongside the video recording. We swiftly deactivated the collect and checked again for memory leaks and saw that the issue disappeared.

Now was the time for the real investigation.

Tracking down the issue

Limitations of the naive approach

The original response to handle the problem (navigate on the application and keep an eye on the memory tab of the Chrome inspector) was good enough to easily see if the problem was gone by the quick fix, but was not as precise as needed to pinpoint the issue in the code.

Tracking memory leaks using Cypress -chrome Memory Inspector

Chrome inspector is a nice tool to get an overall idea of the memory consumption of the web page, but it faces two limitations:

  • the memory tab gives the memory consumption in real-time, but no history
  • when activating the recording of the memory consumption (and when the app has a memory leak), the tab tends to crash, which is not really helpful.

Another issue we were facing was that the memory usage of the app tended to vary quite a lot (between 500Mo and 2Go) between two exploratory tests.

To pinpoint the problem, we needed a solution that was:

  • repeatable
  • limiting discrepancies
  • providing an historic of the memory consumption during the navigation

Hopefully, we already had an automated E2E test runner set up on Gravity. Therefore, all we had to do was to provide some capabilities for monitoring the memory consumption.

Gravity - AI-Powered Quality Intelligence

Reading memory usage inside Cypress

Thanks to this article by Donald Le, we were provided with the basic code to read memory usage.
As with the memory tool of Chrome Inspector, this is just providing real-time memory usage, but we need to have trends to check for memory leaks. In order to do so, three tasks were added to Cypress:

TypeScript
setupNodeEvents(on: Cypress.PluginEvents): void {
    on('task', {
    'memoryUsage:reset': () => {
        memoryUsageStorage = []
        return null
    },
    'memoryUsage:add': (value: number) => {
        memoryUsageStorage.push(value)
        return null
    },
    'memoryUsage:write': (args?: Partial<{ testName: string }>) => {
        fs.appendFileSync(
            path.join(
                __dirname,
                'performanceMonitoring',
                `${args?.testName || 'memoryUsage'}.csv`
            ),
            `${memoryUsageStorage.join(';')}\n`
        )
        return null
    },
})

The three tasks are pretty simple:

  • 1: memoryUsage:reset resets the memory storage
  • 2: memoryUsage:add(value) adds a value in the memory storage
  • 3: memoryUsage:write appends a new line in a CSV file with all recorded memory usage during the test

We also needed a simple Cypress command to add the memory reading:

TypeScript
Cypress.Commands.add('getMemoryUsage', () => {
  cy.window().then((window) => {
    // @ts-ignore
    cy.task('memoryUsage:add', window.performance.memory.usedJSHeapSize)
  })
})

Now, all we have to do is to write (or reuse) an E2E test that navigates inside the application and monitors the memory usage. Our test looks something like this:

TypeScript
it('navigates in the application', () => {
  cy.visit('http://localhost:3002')
  cy.login()

  cy.task('memoryUsage:reset')

  navigateInUserJourneys()
  navigateInPages()
  navigateInSessions()
  navigateInUserJourneys()

  cy.task('memoryUsage:write', { testName: Cypress.spec.name }).then(() => {
    // Some assertions to ensure the test correctly finished
  })
})

function navigateInPages() {
  cy.getMemoryUsage()
  cy.get('[data-testid="Exploration.Pages"]').click()
  cy.getMemoryUsage()

  cy.get('.pages-page-list')
  cy.getMemoryUsage()

  cy.get('.page-trends-page__view-selector').click()
  cy.get('[data-value="treeMap"]').click()
  cy.get('.sitemap')
  cy.getMemoryUsage()

  cy.get('.page-trends-page__view-selector').click()
  cy.get('[data-value="quadrant"]').click()
  cy.get('.gravity-quadrant')
  cy.getMemoryUsage()

  cy.get('.page-trends-page__view-selector').click()
  cy.get('[data-value="list"]').click()
  cy.getMemoryUsage()

  cy.get('.pages-page-list tbody tr').first().click()
  cy.getMemoryUsage()

  cy.get('.page-heatmap')
  cy.getMemoryUsage()

  cy.get('.page-page__view-selector').click()
  cy.get('[data-value="interactions"]').click()

  cy.get('.page-page__view-selector')
  cy.getMemoryUsage()

  cy.get('.page-page__view-selector').click()
  cy.get('[data-value="elements"]').click()

  cy.get('.action-summary__wrapper')
  cy.getMemoryUsage()
}

As you can see, the test will:

  • reset the memory storage at the beginning
  • after each action, save a reading of the memory
  • write the recording to a CSV file by the end of the test

This way, we were able to launch Cypress, click a dozen times on the replay button and we would get a CSV file that allowed us to gather some insights about the memory leaks.

Tracking memory leaks using Cypress, executions Replay Button

In this graph, the horizontal axis represents the time and the vertical one represents the memory consumption. Each line is a test execution.

One problem we found with this approach was that it seemed than the more we were running the tests, the more memory was consumed. In order to fix this, a simple bash script was written in order to use a fresh instance of Cypress for each run:

TypeScript
#!/bin/bash
filename=$1

echo "-- Running $filename"
exportName="performanceMonitoring/$(basename $filename).csv"
touch "$exportName"

while [ "$(cat $exportName | wc -l)" -lt 15 ]; do
  npm run cypress:run:perf -- --spec "$filename"
done;

What this script does is that it runs the same Cypress test until the CSV file reaches 15 lines. The choice of 15 executions was a compromise between gathering enough data and not spending 3 hours waiting for those data.

With this approach, we were finally able to get some more consistent readings:

15 executions

Generalizing the performance test to pinpoint the issue

We were aware that our data collector was causing the issue. This was probably due to the new screenshot feature, but we wanted to ensure our assumption was true.
In order to do so, we wrote four versions of the same test but with different options for setting up the data collector:

TypeScript
// noDataCollection.perf.cy.ts
describe('Performance tests', () => {
  it('Navigate in Gravity - no collect', () => {
    perfE2E({
      sessionRecording: false,
      videoRecording: false,
    })
  })
})

// noRecording.perf.cy.ts
describe('Performance tests', () => {
  it('Navigate in Gravity - no recording', () => {
    perfE2E({
      sessionRecording: true,
      videoRecording: false,
    })
  })
})

// recordVideoOnly.cy.ts
describe('Performance tests', () => {
  it('Navigate in Gravity - full collect', () => {
    perfE2E({
      sessionRecording: true,
      videoRecording: true,
      snapshotRecording: false,
    })
  })
})

// fullCollect.perf.cy.ts
describe('Performance tests', () => {
  it('Navigate in Gravity - full collect', () => {
    perfE2E({
      sessionRecording: true,
      videoRecording: true,
    })
  })
})

We also updated our bash script so it would run each test 15 times and generate 4 different CSV files:

Bash
#!/bin/bash

echo "Running performance tests"

echo "-- Cleaning up existing monitoring"
rm -f performanceMonitoring/*.csv

for filename in cypress/perf/*.perf.cy.ts; do
  echo "-- Running $filename"
  exportName="performanceMonitoring/$(basename $filename).csv"
  touch "$exportName"

  while [ "$(cat $exportName | wc -l)" -lt 15 ]; do
    npm run cypress:run:perf -- --spec "$filename"
  done;
done;

Now, all we had to do was run this script, grab a coffee or two and throw our nice CSV files inside our favorite spreadsheet editor (or, by lack of a favorite, one that is available), and validate our hypothesis:

CSV files

As we have a test which does not use the data collector at all, we could also use this one as a reference to understand the added memory usage with the data collector activated:

data collector activated

For each measure, we divide the value by the one in the reference test. This means that the closer to 1 we are, the less extra memory is consumed.

Rinse & repeat – finding the faulty code

Now than we know that the screenshot handling is responsible for our memory leak, it is time to digg more inside the code and find the culprit for the memory leak. We will not digg much into that part, but what we basically did was disconnect parts of the feature (starting by the end, in our case sending the screenshot to Gravity), run the test, copy the data in the spreadsheet, and see if the memory usage is closer to the normal.

Performing this task is quite tedious, and numerous coffees have been consumed while waiting for the tests to be executed. However, we were finally able to get something closer to memory usage without collecting:

Rinse & repeat - finding the faulty code

We were able to find that our memory leak was caused by this quite innocent-looking piece of code:

TypeScript
const node = rebuild(snapshot, {
  doc: inDocument,
  cache: createCache(),
  mirror: createMirror(),
})
if (node) return new XMLSerializer().serializeToString(node)

The problem did not come from the XMLSerializer nor the rebuild function. It was the assignment of the node variable that caused the issue. It was not properly picked up by the garbage collector after exiting the function. So, after each screenshot, a copy of the dom was stored inside the memory. Once this was found, it was quite easy to fix: we destroyed all the children of the node after transforming the screenshot to a string and then we “forced” its deletion (with quotes around the forcing, as there is now a real way to force the garbage collector to do its job).

TypeScript
const data: { node?: Node | null } = {}

data.node = rebuild(snapshot, {
  doc: inDocument,
  cache: createCache(),
  mirror: createMirror(),
})
if (data.node) {
  const serialized = new XMLSerializer().serializeToString(data.node)
  removeAllChildren(data.node)
  delete data.node

  return serialized
}

Enhancements and other usages

Thanks to this solution, we have been able to quickly find and fix our memory leak. More importantly, we now have some code base than we will be able to use if we are facing again this kind of issue.

There are obviously a number of things that could be improved:

  • use perfomance.measureUserAgentSpecificMemory instead of usedJSHeapSize: this solution should enable a finer read of the memory distribution. We did not use it as we were missing some headers in Gravity to activate it. That being said, although usedJSHeapSize might not be precise, it has the advantage to be consistent, which is what we are looking for when improving memory usage.
  • automate the graph generation: it’s not that long to import the CSV file, add it to the spreadsheet and update the graph references, but repetition is bad. It would be nice to have a simpler tool that just reads the CSV and just show the graphs.

The solution could also be used in some other ways to improve code quality. We could imagine a CI task that runs tests with the latest deployed version and the one to be deployed and stops the CD pipeline if the memory usage grows more than a given threshold.

We talked a lot about Gravity in this article, so if you are interested in learning more about it and it’s features you can learn more here: https://www.smartesting.com/en/gravity/ and you can also book a demo with Julien:

Stay tuned!

Aligning Continuous Testing with Customer Journeys through AI

Aligning Continuous Testing with Customer Journeys through AI

Gravity Testing

Businesses are under unprecedented pressure to innovate quickly and deploy new features while ensuring high standards through continuous testing. Development…

Boost Your Agile Testing Efficiency in Jira with Yest and Xray

Boost Your Agile Testing Efficiency in Jira with Yest and Xray

Testing Yest

In agile development, efficient testing and collaboration are key. Although methodologies and ceremonies guide the process, choosing the best tools…

AutomationStar 2024

AutomationStar 2024: A Thrilling Experience for Gravity

Gravity News News Gravity

Last month, the Smartesting team had the pleasure of attending the AutomationStar conference in the beautiful city of Vienna. This…