This file describes the utility functions and also provides details and tips for writing test scripts.
Tests can be run locally (from a cloned repository) or directly from GitHub. To facilitate testing tests, a script may define the repository to be used, instead of the default - the master
branch of the hifi_tests
repository. This script provides the suppor for this functionality.
The script provides two methods.
getRepositoryPath
is used by thenitpick.js
script (described in the next section).createAutoTester
is used by each test script to provide an interface tonitpick.js
.
nitpick.js
contains a number of utility methods that facilitate writing tests. This file describes the folder structure, the boilerplate required to create a test and documents the functionality of the nitpick.js
script.
The tests are designed to run from GitHub and do not need to be downloaded. They can also be run locally (e.g. while developing tests).
The testing philosophy has 4 major objectives:
- Exact reproduction of the snapshots to allow automatic comparison of actual results with the expected images.
- Manual use of the tests for development purposes.
- Clear and concise syntax for writing tests by developers
- Simple execution of the tests by testers.
To achieve these objectives, the testing implementation includes the following design decisions:
-
To the maximum extent possible - snapshots are taken with the secondary camera. This enables independent positioning, as well as control of the snapshot image size.
-
The test coordinate system is defined by the avatar's position (the avatar is rotated to point down the Z axis). The avatar's position is defined as the hip position - therefore: the test position is defined as a point a fixed height below the avatar (i.e. the assumed position of the avatar's feet) and the camera position is a fixed height above the avatar (i.e. the avatar's assumed eye position).
-
Two modes of advancing steps are provided - manual and automatic.
- In manual mode, pressing the
n
key will advance to the next step - In automatic mode, steps are advanced every 2 seconds. The test may modify this value during setup.
- In manual mode, pressing the
-
"Expected Images" are created by running the test, and then using the
nitpick
tool (documented here).
A test case consists of initialization code followed by a series of steps. In general, the steps are in pairs:
- Create the image and position the camera
- Take the snapshot
It is suggested to set the following properties for all entities:
- lifetime: 60,
- position: Vec3.sum(MyAvatar.position, some offset*),
- userData: JSON.stringify({ grabbableKey: { grabbable: false } })
An origin frame can be defined to create and position the assets required for the test relative to this origin.
The origin frame also helps defining the Validation Camera position
To define the origin frame, use the defineOriginFrame()
method before any steps in the test case
The Origin Frame is then automatically computed from the actual MyAvatar position.
The orientation of the frame is world aligned. The position of the frame is always 1m below the MyAvatar position (regardless of its size). The position of the Validation Camera is then defined as 1.76m up from the origin position.
To validate a test, we are capturing snapshots from a 'Validation Camera'. The view from this camera can be captured from the primary or the secondary views currently available on the rendering engine of Interface depending on the constraints of the test. The actual camera used for capturing the validation image is an implementation issue (ideally they should work the same). During the test, we will refer to 'validation Camera'
The initial location of the Validation Camera is automatically updated when calling 'defineOriginFrame'
The Validation camera can be accessed and moved around in the Origin Frame space using the following functions
- validationCamera_setTranslation(offset vec3)
- validationCamera_translate(offset vec3)
- validationCamera_setRotation(rotation vec3)
The top level folder of importance is tests
. The relevant contents of this folder are:
testsOutline.md
- an outline of all available tests. This file is created bynitpick.exe
.testRecursive.js
- this is a script that can be run in Interface. It executes all the applicable tests in the folder hierarchy (i.e. those tests named test.js).
A similar file is located in each relevant sub-folder. These files are created by nitpick.exe
.
utils
- a set of utility scripts, includingnitpick.js
.- A number of folders containing the actual tests.
Each test folder has (at least) 3 scripts and a number of images.
test.js
- the actual test script. Running this script runs the test in manual mode.testAuto.js
- runs the test in automatic mode. This file is the same for all tests.test.md
- this file is a description of the test. It is created bynitpick.exe
from the test script and the expected images.
The expected images themselves are created in 2 stages: running the test and then running nitpick.exe
to create the images from the test results.
The boilerplate code is enclosed in an if statement to ensure it is only called once.
if (typeof PATH_TO_THE_REPO_PATH_UTILS_FILE === 'undefined') {
// Boilerplate code
}
The first 2 lines of the boilerplate code define the GitHub repository. This allows changing the repository for testing purposes.
PATH_TO_THE_REPO_PATH_UTILS_FILE = "https://raw.githubusercontent.com/highfidelity/hifi_tests/master/tests/utils/branchUtils.js";
Script.include(PATH_TO_THE_REPO_PATH_UTILS_FILE);
The next line loads the nitpick script (the script contains a module)
var nitpick = createAutoTester(Script.resolvePath("."));
The test itself is written in the perform method:
nitpick.perform("<test description string>", Script.resolvePath("."), "secondary", undefined, function(testType) {
// set up zones, objects and other entities
// create steps
var result = nitpick.runTest(testType);
});
Note that "secondary" may be replaced by "primary". This is needed for tests that require the primary camera.
The next parameter that is marked undefined
can optionally be replaced with a list of one or more filters, indicating when the test should run and/or what images it should produce on each platform. This uses a special syntax which is explained in a later section.
The nitpick.runTest(testType);
call is the line that requests the execution of the steps.
As described above, steps usually come in pairs.
The following is an example showing the idea:
nitpick.addStep("Move to blue zone", function () {
var offset = { x: 0.0, y: 0.0, z: -10.0 };
MyAvatar.position = Vec3.sum(avatarOriginPosition, offset);
validationCamera_setTranslation(offset);
Entities.editEntity(sphere, { position: Vec3.sum(MyAvatar.position, SPHERE_OFFSET) });
});
nitpick.addStepSnapshot("Blue zone, dark ambient light");
The first step moves forward 10 metres (the avatar is looking down the Z axis).
As previously mentioned, the test boilerplate contains an undefined
parameter after the "secondary"/"primary" camera option. This allows the test creator to determine when the test is run, and/or how the test result images are labeled. For example, the parameter could be replaced by [["high"]]
to indicate that the test should only run when the current performance tier for Interface is set to High. Or alternatively, the parameter could become [["", "tier"]]
to indicate that the result images should include the performance tier during testing in the image name.
Some more examples are included below:
[[""]] /* Run on all profiles (equivalent to undefined) */
[["mac"]] /* Run only on Mac */
[["mac.amd"]] /* Run only on Mac with an AMD GPU */
[["mid,high"]] /* Run only on the mid/high performance tier */
[["windows,mac,linux.amd,nvidia"]] /* Run only on Windows/Mac/Linux AND run only when AMD or Nvidia GPU present */
[["low,mid.windows"], ["android"], ["mac.intel"]] /* Run on either: 1) Windows in low/mid performance tier, 2) Android, or 3) A Mac with Intel GPU */
[["", "tier.os.gpu"]] /* Run on all profiles. Include the tier, OS, and GPU detected during the test in the image name. */
[["android", "os"], ["windows,mac,linux", "tier.gpu"]] /* Run on Android and include the OS in the image name. Or, run on Windows/Mac/Linux and include the tier and GPU in the image name. */
The filter syntax works as follows:
- Each inner pair of square brackets is a filter. If the current test profile (combination of tier, OS, and GPU) matches at least one of the filters, then the test will be run.
- Multiple filters are separated by commas per standard javascript list syntax.
- Each filter consists of a filter string, and optionally an image name string.
- The filter string contains zero or more restrictions, separated by periods (
.
). All restrictions must match for the filter to match the test profile. Possible restrictions are:- For tier:
low
,mid
,high
- For OS:
windows
,mac
,linux
,android
- For GPU:
amd
,nvidia
,intel
- For tier:
- Each restriction can optionally be made broader. Rather than specifying only one tier/OS/GPU, multiple valid options can be separated by commas (
,
). - The image name string contains zero or more of:
tier
,os
, orgpu
, in that order, separated by periods (.
). Here are some examples of filters and what reference images they correspond to:[[/*Any filter string*/, ""]]
-ExpectedImage_00000.png
(default behavior)[["low,high", "tier"]]
-ExpectedImage_low-high_00000.png
(test does not run on mid tier)[["", "tier.os.gpu"]]
- Many images, for exampleExpectedImage_low_windows_intel_00000.png
nitpick.js
provides the perform
method, used in all tests. This method accepts 4 parameters:
- The name of the script (a string)
- The path to the test. This is used so we can include the path in the snapshots' names.
- Which camera to use for validation. This is "secondary" or "primary"
- A single lambda function. This function is run immediately, unless tests are being run in recursive mode. This function accepts a single parameter "auto" or "manual", defining the mode that will be used when the test is executed.
Recursive mode is selected by calling nitpick.enableRecursive();
before calling perform.
Recursive tests are usually performed automatically as follows:
// This is an automatically generated file, created by nitpick on Jun 1 2018, 11:24
user = "highfidelity/";
repository = "hifi_tests/";
branch = "master/";
var nitpick = Script.require("https://github.com/highfidelity/hifi_tests/blob/master/tests/utils/nitpick.js?raw=true");
nitpick.enableRecursive();
nitpick.enableAuto();
Script.include("https://github.com/highfidelity/hifi_tests/blob/master/tests/content/entity/zone/zoneOrientation/test.js?raw=true");
Script.include("https://github.com/highfidelity/hifi_tests/blob/master/tests/content/entity/zone/create/test.js?raw=true");
Script.include("https://github.com/highfidelity/hifi_tests/blob/master/tests/content/entity/zone/ambientLightZoneEffects/test.js?raw=true");
Script.include("https://github.com/highfidelity/hifi_tests/blob/master/tests/content/entity/zone/ambientLightInheritance/test.js?raw=true");
nitpick.runRecursive();
Note that this code is generated automatically by the nitpick
tool.
As shown in the snippet - auto mode is selected by nitpick.enableAuto();
, manual is the default.
Also note that recursive is distinct from auto/manual.
perform
creates a TestCase
object with the parameters.
If not in recursive mode, this is executed immediately . In recursive mode, the object is pushed on to testCases
.
This array is executed by the nitpick.runRecursive();
method. This method checks for the completion of the previous test,
every second. If the test is complete and there are more tests then the next test case is run.
This is the method that initiates execution of the test, in either manual or auto mode.
A number of methods provide information on the GitHub repository the tests are currently being run from. "Default" refers to running tests from the hifi_tests repository belonging to the highfidelity user, on the master branch.
Note that all results include the final '/'.
- module.exports.getRepositoryPath() provides the repository, default is the path to hifi_tests
- module.exports.getTestsRootPath() provides the path to the top-level tests folder
- module.exports.getUtilsRootPath() provides the path to the top-level utils folder
- module.exports.getAssetsRootPath() provides the path to the top-level assets folder.