Keep your components blazingly fast and ship your React app with no doubt.
π Native Env: benchmark your components in the environment they usually run in.
π° CI Ready: get it to work in your Pull Requests, it's really built for that.
π― Stable Results even in unsteady envs like GitHub runners and other virtual machines.
Install Perftool using preferred package manager.
# Yarn
yarn add --dev @salutejs/perftool
# NPM
npm i -D @salutejs/perftool
# PNPM
pnpm add -D @salutejs/perftool
Let's get started by writing a test for a component. First, create a Component.tsx
module:
import React from 'react';
const Component = ({ prop }) => <div style="color: red;">{prop}</div>;
export default Component;
Then, create a module Component.perf.tsx
. This is an entrypoint for Perftool to obtain the component:
import React from 'react';
import Component from './Component';
export function Default() {
return <Component prop="value" />;
}
Now run Perftool in preview mode to test the build and check if the component renders correctly.
npx perftool --preview *.perf.tsx
Then, we can proceed to the actual run with following command:
npx perftool -o base.json *.perf.tsx
After Perftool finishes the run, base.json
report file will appear in the current directory.
Now let's change our component, so it takes noticeably more amount of time to render it:
import React from 'react';
const Component = ({ prop }) => (
<div>
{[...Array(1000)].map(() => (
<div style="color: red;">{prop}</div>
))}
</div>
);
export default Component;
Run Perftool one more time with changed output filename:
npx perftool -o changed.json *.perf.tsx
After Perftool finishes the run, changed.json
report file will appear in the current directory.
Finally, let's do the comparison of our reports:
npx perftool-compare -o result.json changed.json base.json
After Perftool does the comparison, result.json
report file will appear in the current directory and an error will be thrown owing to bad significant change in the reported metrics.
π Well done! You just successfully done your first performance test with Perftool!
To make an additional configuration, create perftool.config.mts
file in the root of your project, or provide a config file path with -c <path>
option when running Perftool. Here's example config:
import type { Config } from '@salutejs/perftool';
const config: Config = {
retries: 30,
include: ['src/**/*.perf.tsx'],
displayIntermediateCalculations: false,
failOnSignificantChanges: false,
modifyWebpackConfig(conf) {
/**
* Customize the build if needed
*/
return conf;
},
};
export default config;
Check out all the config settings and description here.
Here's example config.
There are no fancy-github-hosted docs right now, so this section consists of useful links and some nice perftool feature specs.
You can run code before Perftool renders the component. This code will be run in the browser, so at this stage you can mock data through the global object.
import React from 'react';
import Component from './Component';
export function Default() {
return <Component prop="value" propX={window.mock} />;
}
Default.beforeTest = () => {
window.mock = 123;
};
Using the intercept method, you can intercept requests and return prepared data. Useful if you need to mock some endpoint, image or file, or cancel some request (see typings).
import React from 'react';
import { intercept } from '@salutejs/perftool';
import Component from './Component';
export function Default() {
return <Component prop="value" />;
}
Default.beforeTest = async () => {
await intercept({
method: 'POST',
source: '**/api/foo*', // glob-pattern
response: 'src/utils/perftool/fixtures/foo.json', // file path
});
await intercept({
method: 'GET',
responseType: 'json',
source: '**/api/bar*',
response: { foo: 'bar' },
});
await intercept({
// any http method
responseType: 'abort',
source: '**/api/baz*',
});
};
Using the setViewport method, you can specify the size of the viewport for rendering the current component. See typings
import React from 'react';
import { setViewport } from '@salutejs/perftool';
import Component from './Component';
export function Default() {
return <Component prop="value" />;
}
Default.beforeTest = async () => {
await setViewport('desktop'); // 'desktop' and 'touch' are built-in presets
};
Most often the following situation will occur: everything is fine, no degradation. In this case, there is no need to do anything - we will assume that nothing bad happened. Now let's look at the case when perftool-compare threw an error. The result of the performance test is a comparison of the results of runs in the base and feature branches. Here is an example of such a report, most of the fields have been omitted for readability:
{
"hasSignificantNegativeChanges": true,
"result": {
"Component.perf.tsx#Default": {
"render": {
"mean": {
"old": [10, 2],
"new": [20, 4],
"change": {
"difference": 10,
"percentage": 100,
"significanceRank": "high"
}
}
}
}
}
}
The top-level part of the report is a field with the boolean data type hasSignificantNegativeChanges, which means whether there are components in the report with statistically significant negative changes. Its value is determined based on the comparison results for each component and metric. In the config you can specify which specific metrics will be aggregated in hasSignificantNegativeChanges.
In this case, we detect hasSignificantNegativeChanges: true, and now we need to find the reason. We are looking for a metric with change.significanceRank === 'high'
and difference > 0
. This can be one or more components/metrics. Once we have found all the candidates for degradation, we move on to the next step.
Now you need to check whether the component that failed the test has actually changed. If the component was not directly or indirectly affected in this branch, then the result is false-positive and can be ignored - sometimes on machines where jobs are run, a load change occurs, which can lead to incorrect results.
At this stage, you can begin searching for degradation in the changed component. If you are not sure if degradation actually happened, you can run a check one more time.
See here.
We follow Node maintenance cycle.
Node | 14 | 16 | 18 | 20 |
---|---|---|---|---|
β | β | β | β |
React | <= 16.8 | ^16.8 | ^17 | ^18 |
---|---|---|---|---|
β | β | β | β |
Windows support is not currently planned.
OS | Windows | MacOS | Linux |
---|---|---|---|
β | β | β |
Check out Contribution guide. All types of contributions are encouraged and valued.
Perftool is MIT licensed.