We are improving detox API as we go along, sometimes these changes require us to break the API in order for it to make more sense. These migration guides refer to breaking changes. If a newer version has no entries in this document, it means it does not require special migration steps. Refer to the release notes of the later builds to learn about their improvements and changes.
It is recommended to change "name" string to "device" object in your configurations, like shown below:
Before:
{
"ios.sim.debug": {
"type": "ios.simulator",
"name": "iPhone 11 Pro"
},
"android.emu.release": {
"type": "android.emulator",
"name": "Nexus_5X_API_29"
},
"android.att.release": {
"type": "android.attached",
"name": "YOGAA1BBB412"
}
}
After:
{
"ios.sim.debug": {
"type": "ios.simulator",
"device": { // one of these or a combination of them
"id": "D53474CF-7DD1-4673-8517-E75DAD6C34D6",
"type": "iPhone 11 Pro",
"name": "MySim",
"os": "iOS 13.0",
}
},
"android.emu.release": {
"type": "android.emulator",
"device": { // only avdName is supported at the moment
"avdName": "Nexus_5X_API_29",
}
},
"android.att.release": {
"type": "android.attached",
"device": { // only adbName is supported at the moment
"adbName": "YOGAA1BBB412",
}
}
}
Detox 14.0.0 drops support for iOS 9.x simulators, and thus it also drops support for any API that is deprecated in iOS 10 and above. This includes legacy remote and local notifications handling API. These APIs have been deprecated since iOS 10, and we believe we've given app developers enough time to use the modern APIs. Make sure you transition to the UserNotifications framework as soon as possible.
Please note that for React Native apps, PushNotificationIOS (RCTPushNotificationManager
) is severely outdated and does not support these modern APIs. It is recommended to transition to a more modern solution. While it is sad that such an important app feature is let to stagnate so much by Facebook, it cannot be the concern of Detox. It is up to RN users to keep their apps up to date with the latest Apple APIs.
Our own React Native notifications solution supports these modern APIs.
See #1514.
This is only relevant to those running Detox using Jest
as the test runner!
In 12.7.0
we've greatly improved our support in Jest - trying to tackle these two caveats which hold developers back from embracing it:
- Jest file-level summary logs take precedence over 'plain' output, which makes them and all other logs (e.g. user in-test logging) seem cluttered.
- Plain logs output is batched, and thus often does not show in real-time as the test is run. This is particularly annoying when running tests on the local computer.
- Jest offeres no spec-level logging => no way to tell what's running "right now" and which test created what log-outputs.
Put in simple words, Jest is optimized for running tests concurrently using multiple workers. This isn't the case when writing/debugging tests on a local machine.
In 12.7.0
we've worked out a configuration scheme that aims at solving these by streamlining all test-related outputs. Please follow the updated Jest installation guide, to set it up.
Starting Detox 12.5.0
, we ship Android with precompiled sources under a .aar
file. The complete configuration process is thoroughly described in the Android setup guide - but it mostly fits new projects. For existing projects, migrating is strongly recommended; Here's the diff:
Root settings.gradle
file:
-include ':detox'
-project(':detox').projectDir = new File(rootProject.projectDir, '../node_modules/detox/android/detox')
Root buildscript (i.e. build.gradle
):
allprojects {
repositories {
// ...
+ maven {
+ url "$rootDir/../node_modules/detox/Detox-android"
+ }
}
}
App buildscript (i.e. app/build.gradle
):
dependencies {
- androidTestImplementation(project(path: ":detox"))
+ androidTestImplementation('com.wix:detox:+') { transitive = true }
}
If you have Detox Proguard rules integrated into the app/build.gradle
, be sure to switch to an explicit search path:
buildTypes {
release {
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- proguardFile "${project(':detox').projectDir}/proguard-rules-app.pro"
+ proguardFile "${rootProject.projectDir}/../node_modules/detox/android/detox/proguard-rules-app.pro"
}
}
The deprecation of "specs"
(in package.json
) introduced in 12.1.0 is no longer relevant.
It is valid now, like it was before, but from now on the semantics has been slightly changed -
it acts as a fallback for the default root for your Detox e2e specs, in cases when
you don't specify it explicitly, e.g.:
detox test # translates to: mocha <...args> e2e
detox test e2e/01.sanity.test.js # translates to: mocha <...args> e2e/01.sanity.test.js
Between 12.1.x and 12.3.x, it was buggy and used to work like this:
detox test # translates to: mocha <...args> e2e
detox test e2e/01.sanity.test.js # translates to: mocha <...args> e2e e2e/01.sanity.test.js
This is not a breaking change yet, but starting from [email protected]
you'll start seeing warnings like:
detox[21201] WARN: [deprecation.js] Beware: -f, --file will be removed in the next version of Detox.
detox[21201] WARN: [deprecation.js] See the migration guide:
https://github.com/wix/Detox/blob/master/docs/Guide.Migration.md#migrating-from-detox-120x-to-121x
In the next major version --file
and --specs
will be treated as unknown arguments
and therefore passed as-is to your appropriate test runner. That allows to avoid name
conflict with the respective --file
option in Mocha runner itself and other potential
collisions.
So, if you have been using CLI arguments like --file e2e
or
--specs e2e
, please drop the preceding --file
and --specs
, so that:
detox test --file e2e/01.sanity.test.js
becomes:
detox test e2e/01.sanity.test.js
UPDATE: It was decided not to deprecate "specs"
in package.json
, so the text below
is not relevant to a large extent. Please ignore the guide below.
To get rid of this warning:
find"specs"
or"file"
entry in your project'spackage.json
and empty it (e.g."e2e"
⟶""
);update yourdetox test
scripts — make sure they have an explicit path to your Detox tests folder, e.g.detox test e2e
.
For example, if it were a package.json
before:
{
"name": "your-project",
"scripts": {
"e2e:ios": "detox test -c ios.simulator.release"
},
"detox": {
"specs": "e2e"
}
}
Then this is how it should look like afterwards:
{
"name": "your-project",
"scripts": {
"e2e:ios": "detox test -c ios.simulator.release e2e"
},
"detox": {
"specs": ""
}
}
Notice that we appended e2e
to the e2e:ios
test script and
emptied "specs"
property in detox
configuration.
In a case if you had no "specs"
property in your detox
configuration
in package.json
, then please add it temporarily like this:
{
"specs": ""
}
The new version explicity requires Xcode 10.1 or higher in order to run tests on iOS (#1229).
React Native versions older than 0.46 are no longer supported, so the missingDimentsionStrategy
can be removed from android/app/build.gradle
:
android {
defaultConfig {
// ...
- missingDimensionStrategy "minReactNative", "minReactNative46"
}
}
android/app/build.gradle
android {
defaultConfig {
// ...
- testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+ testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
}
dependencies {
implementation "com.facebook.react:react-native:+" // From node_modules
androidTestImplementation(project(path: ":detox"))
androidTestImplementation 'junit:junit:4.12'
- androidTestImplementation 'com.android.support.test:runner:1.0.2'
- androidTestImplementation 'com.android.support.test:rules:1.0.2'
Rewrite your DetoxTest.java
file according to the updated Android setup guide (step 4).
If your project does not already use Kotlin, add the Kotlin Gradle-plugin to your classpath in android/build.gradle
:
buildscript {
// ...
ext.kotlinVersion = '1.3.0'
dependencies: {
// ...
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlinVersion"
}
}
Note: most guides advise of defining a global kotlinVersion
constant - as in this example, but that is not mandatory.
IMPORTANT: Detox aims at a playing fair with your app, and so it allows you to explicitly define the kotlin version for it to use - so as to align it with your own; Please do so - in your root android/build.gradle
configuration file:
buildscript {
ext.kotlinVersion = '1.3.0' // Your app's version
ext.detoxKotlinVersion = ext.kotlinVersion // Detox' version: should be 1.1.0 or higher!
}
Note that Detox has been tested for version 1.1.0 of Kotlin, and higher!
Detox 9.0.0 brings latest Espresso (3.0.2), and React Native 56 support on Android. Espresso 3.0.2 has a few mandatory dependency changes, which break the current setup for Detox users on Android.
Use this to diff to upgrade your dependencies, and follow Android Studio's in-editor guidance/lint support.
android/app/build.gradle
dependencies {
- implementation "com.android.support:appcompat-v7:27.0.2"
+ implementation "com.android.support:appcompat-v7:27.1.1"
implementation "com.facebook.react:react-native:+" // From node_modules
androidTestImplementation(project(path: ":detox"))
androidTestImplementation 'junit:junit:4.12'
- androidTestImplementation 'com.android.support.test:runner:1.0.1'
- androidTestImplementation 'com.android.support.test:rules:1.0.1'
+ androidTestImplementation 'com.android.support.test:runner:1.0.2'
+ androidTestImplementation 'com.android.support.test:rules:1.0.2'
android/build.gradle
dependencies {
- classpath 'com.android.tools.build:gradle:3.0.1'
+ classpath 'com.android.tools.build:gradle:3.1.4'
}
An example for the above changes can be found on demo-react-native
project
More details about Espresso dependencies here
Detox 8.x.x brings support for test artifacts (videos, screenshot, logs), and to learn more about it you can refer to Artifacts documentation and to Detox CLI documentation.
In order for Detox to be able to create artifacts, detox.beforeEach(testSummary)
and detox.afterEach(testSummary)
must be called with a current test summary object (test title, full test name, test status).
Detox 8 introduces adapters for both Mocha and Jest, wrapping the original detox.beforeEach(testSummary)
and detox.afterEach(testSummary)
functions, for easier integration.
you are encouraged to reuse the examples of ./e2e/init.js
for mocha and jest. The gist is brought in the following sections:
const detox = require('detox');
const config = require('../package.json').detox;
const adapter = require('detox/runners/mocha/adapter');
before(async () => {
await detox.init(config);
});
beforeEach(async function () {
await adapter.beforeEach(this);
});
afterEach(async function () {
await adapter.afterEach(this);
});
after(async () => {
await detox.cleanup();
});
NOTICE: Make sure you use ES5 functions in
beforeEach
andafterEach
.this
referes to mocha's test object, using arrow functions will result with failure to to acquire a correctthis
inside the adapter.
// ✗ INCORRECT
beforeEach(() => { /* ... your content ... */ }); // won't work
afterEach(() => { /* ... your content ... */ }); // won't work
// CORRECT
beforeEach(function ( /* ... your content ... */ ) {});
afterEach(function ( /* ... your content ... */ ) {});
const detox = require('detox');
const config = require('../package.json').detox;
const adapter = require('detox/runners/jest/adapter');
jest.setTimeout(120000);
jasmine.getEnv().addReporter(adapter); // don't forget this line
beforeAll(async () => {
await detox.init(config);
});
beforeEach(async () => {
await adapter.beforeEach();
});
afterAll(async () => {
await adapter.afterAll();
await detox.cleanup();
});
NOTICE: Make sure to register the adapter as a Jasmine reporter in
init.js
like this:
jasmine.getEnv().addReporter(adapter);
- Jest adapter requires a hook to
afterAll
:
afterAll(async () => {
await adapter.afterAll();
await detox.cleanup();
});
API of these methods is subject to change in future versions due to complexity behind composing test summary objects (as in the case with Jest test runner). If you have reasons to make direct calls to detox.beforeEach
and detox.afterEach
(e.g. you're adding support for another test runner), please refer to detox object documentation.
The --artifact-location
argument became optional for detox test
in the version 8.x.x.
By default it dynamically creates ./artifacts/{configuration}.{timestamp}
directory in the project folder as soon as it has to save a recorded artifact.
Previously, to enable log recording you just had to specify --artifact-location
arg. Currently, you need to tell that explicitly via a new CLI flag: --record-logs all
or --record-logs failing
.
Notice that --artifact-location
became sensitive to whether you end your directory path with a slash or not. It has the next convention:
- If you want to create automatically a subdirectory with timestamp and configuration name (to avoid file overwrites upon consquent re-runs), specify a path to directory that does not end with a slash.
- Otherwise, if you want to put artifacts straight to the specified directory (in a case where you make a single run only, e.g. on CI), add a slash to the end.
For more information see CLI documentation.
The clearest example for for the 4->5 API changes is the change log of detox's own test suite. Check detox test change log for a real life example.
###Version 5.x.x breaks detox's API in 4 different places
All of the API calls are now promise based, and must use either promise chains or async-await.
Here's an example of async call to tap an element
// <=4.x.x
beforeEach(() => {
element(by.text('Sanity')).tap();
});
// 5.x.x
beforeEach(async () => {
await element(by.text('Sanity')).tap();
});
Same thing with expectations
// <=4.x.x
it('should have welcome screen', () => {
expect(element(by.text('Welcome'))).toBeVisible();
expect(element(by.text('Say Hello'))).toBeVisible();
expect(element(by.text('Say World'))).toBeVisible();
};
// 5.x.x
it('should have welcome screen', async () => {
await expect(element(by.text('Welcome'))).toBeVisible();
await expect(element(by.text('Say Hello'))).toBeVisible();
await expect(element(by.text('Say World'))).toBeVisible();
});
Configure and init detox with just one promise based function.
// <=4.x.x
detox.config(config);
detox.start(done);
// 5.x.x
await detox.init(config);
No need to wait for test result after each test, you can safely remove detox.waitForTestResult
// <=4.x.x
afterEach((done) => {
detox.waitForTestResult(done);
});
cleanup is promise based
// <=4.x.x
detox.cleanup(done);
// 5.x.x
await detox.cleanup();
The global object simulator
is now device
, this change makes sense when thinking about multi-platform tests (Android support).
Along with the new promise based API, this is how we now control the attached device
<=4.x.x | 5.x.x |
---|---|
simulator.reloadReactNativeApp(done) |
await device.reloadReactNative() |
simulator.relaunchApp(done) |
await device.relaunchApp() |
simulator.sendUserNotification(params, done) |
await device.sendUserNotification(params) |
simulator.openURL(url) |
await device.openURL(url) |
In order for our API to support multiple platforms and devices, and to be able to provide a valid command line tool, we changed the the detox configuration scheme (in package.json)
Previous config looked like this:
//<=4.x.x
"detox": {
"session": {
"server": "ws://localhost:8099",
"sessionId": "test"
},
"ios-simulator": {
"app": "ios/build/Build/Products/Release-iphonesimulator/example.app",
"device": "iPhone 7 Plus"
}
}
The new configuration holds a dictionary of configurations
.
- Each configuration must state
type
- currently only simulator is supported app
is nowbinaryPath
build
- [optional] build command (eitherxcodebuild
,react-native run-ios
, etc...), will be later available through detox CLI tool.- session object is not mandatory anymore, if is not provided detox will handle server creation by itself.
//5.x.x
"detox": {
"configurations": {
"ios.sim.release": {
"binaryPath": "ios/build/Build/Products/Release-iphonesimulator/example.app",
"build": "xcodebuild -project ios/example.xcodeproj -scheme example -configuration Release -sdk iphonesimulator -derivedDataPath ios/build",
"type": "ios.simulator",
"name": "iPhone 7 Plus"
}
}
}
You will now be able to run builds and tests from your command line detox build
and detox test
, read more about CLI tools here
If you have integrated with Detox in version 3.x.x, you will need to clean your project from previously generated targets.
-
Use the provided
cleanup_4.0.rb
to remove unneeded changes made with Detox 4.x.x.ruby node_modules/detox/scripts/cleanup_4.0.rb
-
The script will delete previously configured project targets
*_Detox
. The targets are not used by detox anymore since the framework is now injected at runtime and doesn't need to be linked in a different target. -
Make sure to add changes performed by running this script to version control.