-
Notifications
You must be signed in to change notification settings - Fork 1.2k
Writing Performance Tests
To add perf tests for the sdk/<service>/<service-sdk>
package, follow the steps below.
-
Create a new folder for the perf tests.
Path-
sdk/<service>/perf-tests/<service-sdk>
(Create the
perf-tests
folder if that doesn't exist) -
This new perf test project will be managed by the rush infrastructure in the repository, with the package name
@azure-tests/<service-sdk>
. To allow rush to manage the project, add the following entry inrush.json
{ "packageName": "@azure-tests/perf-<service-sdk>", "projectFolder": "sdk/<service>/perf-tests/<service-sdk>", "versionPolicyName": "test" }
-
Tests will live under
sdk/<service>/perf-tests/<service-sdk>/test
-
Add a
package.json
such as example-perf-package.json atsdk/<service>/perf-tests/<service-sdk>
folder.Make sure to import your
<service-sdk>
and thetest-utils-perfstress
project."dependencies": { "@azure/<service-sdk>": "^<version-in-master-branch>", "@azure/test-utils-perfstress": "^1.0.0" }
Note:
"@azure/test-utils-perfstress"
is not a published npm package.Set the name of the package and mark it as private.
"name": "@azure-tests/perf-<service-sdk>", "private": true,
-
Run
rush update
and commit the changes to thepnpm-lock
file. -
Copy the
tsconfig.json
,sample.env
(and.env
) files that are present at thesdk/<service>/<service-sdk>
tosdk/<service>/perf-tests/<service-sdk>
.TSCONFIG
- Modify the "extends" string in the copied tsconfig by adding ".." since the perf tests project is located a level below the actual SDK.
- Set the
compilerOptions.module
tocommonjs
in thetsconfig
to allow running the tests withts-node
.
In the end, your tsconfig may look something like below.
{ "extends": "../../../../tsconfig.package", "compilerOptions": { "module": "CommonJS", "declarationDir": "./typings/latest", "lib": ["ES6", "ESNext.AsyncIterable"], "noEmit": true }, "compileOnSave": true, "include": ["./test/**/*.ts"] }
(Skip this section if your service does not have or does not care about a track-1 version.)
-
If there is an old major version of your package that needs to be compared, create the folder as
sdk/<service>/perf-tests/<service-sdk>-track-1
-
It is expected that the track-1 perf tests are counterparts of track-2 tests, so they need to have the same names as specified in the track-2 tests for convenience.
-
Add a
package.json
such as example-track-1-perf-package.json atsdk/<service>/perf-tests/<service-sdk>
folder.Make sure to import your
<service-sdk>
and thetest-utils-perfstress
project."dependencies": { "@azure/<service-sdk>": "^<latest-track-1-version>", "@azure/test-utils-perfstress": "file:../../../test-utils/perfstress/azure-test-utils-perfstress-1.0.0.tgz", }
Set the name of the package and mark it as private.
"name": "@azure-tests/perf-<service-sdk>-track-1", "private": true,
Note: Track-1 packages will not be managed by
rush
, insteadnpm
will be used to manage/run the track-1 tests, you can copy the readme such as the storage-blob-perf-tests-track-1-readme for instructions.Make sure to add the "setup" step in package.json.
"setup": "node ../../../../common/tools/perf-tests-track-1-setup.js",
-
Run
rush update
followed bynpm run setup
to be able to use the perf framework for track-1 perf tests.npm run setup
installs the dependencies specified inpackage.json
-
Repeat the step 6 from the previous section for the track-1 too to get the
tsconfig.json
,sample.env
(and.env
) files.
Add an index.spec.ts
at sdk/<service>/perf-tests/<service-sdk>/test/
.
import { PerfStressProgram, selectPerfStressTest } from "@azure/test-utils-perfstress";
import { `ServiceNameAPI1Name`Test } from "./api1-name.spec";
import { `ServiceNameAPI2Name`Test } from "./api2-name.spec";
// Expects the .env file at the same level
import * as dotenv from "dotenv";
dotenv.config();
console.log("=== Starting the perfStress test ===");
const perfStressProgram = new PerfStressProgram(selectPerfStressTest([`ServiceNameAPIName`Test, `ServiceNameAPIName2`Test]));
perfStressProgram.run();
Base class would have all the common code that would be repeated for each of the tests - common code such as creating the client, creating a base resource, etc.
Create a new file such as serviceName.spec.ts
at sdk/<service>/perf-tests/<service-sdk>/test/
.
import { PerfStressTest, getEnvVar } from "@azure/test-utils-perfstress";
import {
ServiceNameClient
} from "@azure/<service-sdk>";
export abstract class `ServiceName`Test<TOptions = {}> extends PerfStressTest<TOptions> {
serviceNameClient: ServiceNameClient;
constructor() {
super();
// Setting up the serviceNameClient
}
public async globalSetup() {
// .createResources() using serviceNameClient
}
public async globalCleanup() {
// .deleteResources() using serviceNameClient
}
}
Following code shows how the individual perf test files would look like.
import { ServiceNameClient } from "@azure/<service-sdk>";
import { PerfStressOptionDictionary, drainStream } from "@azure/test-utils-perfstress";
import { `ServiceName`Test } from "./serviceNameTest.spec";
export class `ServiceNameAPIName`Test extends ServiceNameTest {
// The next section talks about the custom options that you can provide for a test
public options: PerfStressOptionDictionary = {};
serviceNameClient: `ServiceName`Client;
constructor() {
super();
// Setting up the client
}
public async globalSetup() {
await super.globalSetup(); // Calling base class' setup
// Add any additional setup
}
async runAsync(): Promise<void> {
// call the method on `serviceNameClient` that you're interested in testing
}
}
It is not mandatory to have separate base class and test classes. If there is nothing common among the testing scenarios of your service, feel free to merge base class with the test class to only have a single test class instead.
As seen in the previous section, you can specify custom options along with the default options from the performance framework. You can access the options in the class using this.parsedOptions
.
Parsed options include the default options such as duration, iterations, parallel, etc offered by the perf framework as well as the custom options provided in the TestClass.
interface `ServiceNameAPIName`TestOptions {
newOption: number;
}
export class `ServiceNameAPIName`Test extends ServiceNameTest<`ServiceNameAPIName`TestOptions> {
public options: PerfStressOptionDictionary<`ServiceNameAPIName`TestOptions> = {
newOption: {
required: true,
description: "A new option",
shortName: "sz",
longName: "newOption",
defaultValue: 10240
}
};
async runAsync(): Promise<void> {
// You can leverage the parsedOptions in the setup or globalSetup or runAsync methods as shown below.
// this.parsedOptions.duration.value!
// this.parsedOptions.newOption.value!
}
}
To run a particular test, use npm run perf-test:node
- takes the test class name as the argument along with the command line arguments you may provide.
- Run
npm run perf-test:node -- TestClassName --warmup 2 --duration 7 --iterations 2 --parallel 50
Refer to storage-blob-perf-tests-readme and storage-blob-perf-tests-readme-track-1 and have similar set of instructions for your perf project.
Example: Currently @azure/<service-sdk>
is at 12.4.0 on master and you want to test version 12.2.0
- In the track 2 perf tests project, update dependency
@azure/<service-sdk>
version inpackage.json
to12.2.0
- Add a new exception in
common\config\rush\common-versions.json
underallowedAlternativeVersions
"@azure/<service-sdk>": [..., "12.2.0"]
-
rush update
(generates a new pnpm-lock file) - Navigate to
sdk\storage\perf-tests\<service-sdk>
rush build -t perf-<service-sdk>
- Run the tests as suggested before, example
npm run perf-test:node -- TestClassName --warmup 2 --duration 7 --iterations 2 --parallel 50