Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/JetBrains/intellij-ui-test-robot
The library allows you to write and execute UI tests among IntelliJ IDEA. You can test your Plugin.
https://github.com/JetBrains/intellij-ui-test-robot
ide idea-plugin intellij intellij-plugin java kotlin test-automation
Last synced: 3 months ago
JSON representation
The library allows you to write and execute UI tests among IntelliJ IDEA. You can test your Plugin.
- Host: GitHub
- URL: https://github.com/JetBrains/intellij-ui-test-robot
- Owner: JetBrains
- License: apache-2.0
- Created: 2020-10-05T13:06:13.000Z (over 4 years ago)
- Default Branch: master
- Last Pushed: 2024-08-23T19:12:10.000Z (5 months ago)
- Last Synced: 2024-11-05T04:21:06.168Z (3 months ago)
- Topics: ide, idea-plugin, intellij, intellij-plugin, java, kotlin, test-automation
- Language: Kotlin
- Homepage: https://jetbrains-platform.slack.com/archives/C026SVA9MMM
- Size: 1.12 MB
- Stars: 111
- Watchers: 16
- Forks: 32
- Open Issues: 32
-
Metadata Files:
- Readme: README.md
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
![official JetBrains project](https://jb.gg/badges/official.svg)
This library allows you to write and execute UI tests in IntelliJ IDEA. You can use it to test your plugin.
If you have any questions, feel free to post them in our [Slack Channel](https://jetbrains-platform.slack.com/archives/C026SVA9MMM).
## Quick Start
The first thing we need to do is to launch the IDE. Because the `runIdeForUiTests` task is blocking, we can run it as an asynchronous
process:`./gradlew ui-test-example:clean ui-test-example:runIdeForUiTests &`
Next, we can start the tests. Since they run locally, you need to make sure that the Welcome Frame is visible on the screen:
`./gradlew ui-test-example:test`
Alternatively, you can run all tasks at once with the following command:
`./gradlew ui-test-example:clean ui-test-example:runIdeForUiTests & ./gradlew ui-test-example:test`
## Remote-Robot
The Remote-Robot library is inspired by Selenium WebDriver. It supports IntelliJ IDEA since version `2019.2`.
![](docs/simple-schema.png)
It consists of a `remote-robot` client and a `robot-server` plugin:
* `remote-robot` - is a client (test) side library that is used to send commands to the `robot-server` plugin
* `robot-server` - is an IDEA plugin that must be run with the plugin you are developingThe easiest way to start the test system is to execute the `runIdeForUiTests` task (refer to the [Quick Start section](#quick-start) above).
When IDEA is initialized, the `robot-server` plugin starts listening for commands from the UI test client.The `remote-robot` library communicates with the `robot-server` plugin via an HTTP protocol. This connection means you can
launch IDEA on remote machines or in Docker containers to check your plugin within different test environments.### Setup
The last version of the Remote-Robot is `0.11.23`.
In the test project:
```groovy
repositories {
maven { url = "https://packages.jetbrains.team/maven/p/ij/intellij-dependencies" }
}
dependencies {
testImplementation("com.intellij.remoterobot:remote-robot:REMOTE-ROBOT_VERSION")
}
```In the plugin project:
```groovy
runIdeForUiTests {
systemProperty "robot-server.port", "8082" // default port 8580
}downloadRobotServerPlugin {
version = REMOTE-ROBOT_VERSION
}
```By default, the port is set to local, so it cannot be reached from another host. In case you need to make it public, you can
add the `robot-server.host.public` system property to the `runIdeForUiTests` task:```groovy
runIdeForUiTests {
// ......
systemProperty "robot-server.host.public", "true" // port is public
}
```Of course, you can write UI tests in the plugin project.
### Launching
There are two ways of launching IDEA and UI tests:
#### Using [IntelliJ gradle plugin](https://github.com/JetBrains/gradle-intellij-plugin)
First, we need to launch the IDE. Because the `runIdeForUiTests` task is blocking, we can run it as an asynchronous
process:`./gradlew ui-test-example:clean ui-test-example:runIdeForUiTests &`
Next, we can start the tests. Since they run locally, you need to make sure that the Welcome Frame is visible on the screen:
`./gradlew ui-test-example:test`
Alternatively, you can run all tasks at once with the following command:
`./gradlew ui-test-example:clean ui-test-example:runIdeForUiTests & ./gradlew ui-test-example:test`
Check [this project](ui-test-example) as an example.
#### Using ide-launcher
The `ide-launcher` library allows us to launch IDEA directly from the test. To use it, we need to add a dependency to our
project:```groovy
dependencies {
testImplementation("com.intellij.remoterobot:ide-launcher:REMOTE-ROBOT_VERSION")
}
```Next, we can use `IdeLauncher` to start IDEA:
```java
final OkHttpClient client=new OkHttpClient();
final IdeDownloader ideDownloader=new IdeDownloader(client);
ideaProcess = IdeLauncher.INSTANCE.launchIde(
ideDownloader.downloadAndExtract(Ide.IDEA_COMMUNITY, tmpDir),
Map.of("robot-server.port",8082),
List.of(),
List.of(ideDownloader.downloadRobotPlugin(tmpDir), pathToOurPlugin),
tmpDir
);
```
Check [Java](ide-launcher/src/test/java/com/intellij/remoterobot/launcher/LauncherJavaExampleTest.java) and [Kotlin](ide-launcher/src/test/kotlin/com/intellij/remoterobot/launcher/CommandLineProjectTest.kt) examples.#### Useful launch properties
| Property | Value | Description |
|----------------------------------------|---------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| jb.consents.confirmation.enabled | false | Disable the consent dialog. |
| eap.require.license | true | The EAP version requires license the same way as the Release version. This property helps to [avoid EAP login](https://github.com/JetBrains/intellij-ui-test-robot/tree/master/ui-test-example#how-can-i-switch-off-eap-login) on CI. |
| ide.mac.message.dialogs.as.sheets | false | Disable the Sheet dialogs on Mac, they are not recognized by Java Robot. |
| ide.mac.file.chooser.native | false | Disable the Mac native file chooser, it is not recognized by Java Robot.
| jbScreenMenuBar.enabled + apple.laf.useScreenMenuBar | false + false | Disable the Mac native menu, it is not recognized by Java Robot.
| idea.trust.all.projects | true | Disable the **Trust Project** dialog that appears when the project is opened. |
| ide.show.tips.on.startup.default.value | false | Disable the **Tips Of the Day** dialog on startup. |### Create RemoteRobot
In the UI test project:
```java
RemoteRobot remoteRobot = new RemoteRobot("http://127.0.0.1:8082");
```### Searching Components
We use the [`XPath`](https://www.w3.org/TR/xpath-21/) query language to find components. Once IDEA with `robot-server`
has started, you can open the `http://ROBOT-SERVER:PORT` [link](http://127.0.0.1:8082). The page shows the IDEA UI
components hierarchy in HTML format. You can find the component of interest and write an XPath to it, similar to
Selenium WebDriver. There is also a simple XPath generator that can help you write and test your XPaths.![](docs/hierarchy.gif)
For example:
- Define a locator
```java
Locator loginToGitHubLocator = byXpath("//div[@class='MainButton' and @text='Log in to GitHub...']");
```- Find one component
```java
ComponentFixture loginToGitHub = remoteRobot.find(ComponentFixture.class,loginToGitHubLocator);
```- Find many components
```java
List dialogs = remoteRobot.findAll(
ComponentFixture.class,
byXpath("//div[@class='MyDialog']")
);
```### Fixtures
Fixtures support the `PageObject` pattern. There are two basic fixtures:
- `ComponentFixture` is the simplest representation of any real component with basic methods
- `ContainerFixture` extends `ComponentFixture` and allows searching other components within itYou can create your own fixtures:
```java
@DefaultXpath(by = "FlatWelcomeFrame type", xpath = "//div[@class='FlatWelcomeFrame']")
@FixtureName(name = "Welcome Frame")
public class WelcomeFrameFixture extends ContainerFixture {
public WelcomeFrameFixture(@NotNull RemoteRobot remoteRobot, @NotNull RemoteComponent remoteComponent) {
super(remoteRobot, remoteComponent);
}// Create New Project
public ComponentFixture createNewProjectLink() {
return find(ComponentFixture.class, byXpath("//div[@text='Create New Project' and @class='ActionLink']"));
}// Import Project
public ComponentFixture importProjectLink() {
return find(ComponentFixture.class, byXpath("//div[@text='Import Project' and @class='ActionLink']"));
}
}
``````java
// find the custom fixture by its default XPath
WelcomeFrameFixture welcomeFrame=remoteRobot.find(WelcomeFrameFixture.class);
welcomeFrame.createNewProjectLink().click();
```### Remote-Fixtures
We have prepared some basic fixtures:
```groovy
dependencies {
testImplementation("com.intellij.remoterobot:remote-fixtures:REMOTE-ROBOT_VERSION")
}
```The library contains fixtures for most basic UI components. Please check [this package](https://github.com/JetBrains/intellij-ui-test-robot/tree/master/remote-fixtures/src/main/kotlin/com/intellij/remoterobot/fixtures)
to learn more. In case you want to add missing basic fixtures, you are welcome to PR or create
an [issue](https://github.com/JetBrains/intellij-ui-test-robot/issues).### Getting Data From a Real Component
We use the JavaScript [`rhino`](https://github.com/mozilla/rhino) engine to work with components on the IDEA side.
For example, retrieving text from the `ActionLink` component:
```java
public class ActionLinkFixture extends ComponentFixture {
public ActionLinkFixture(@NotNull RemoteRobot remoteRobot, @NotNull RemoteComponent remoteComponent) {
super(remoteRobot, remoteComponent);
}public String text() {
return callJs("component.getText();");
}
}
```We can retrieve data using `RemoteRobot` with the `callJs` method. In this case, there is a `robot` var in the context
of JavaScript execution. The `robot` is an instance of extending
the [`org.assertj.swing.core.Robot`](https://joel-costigliola.github.io/assertj/swing/api/org/assertj/swing/core/Robot.html)
class.When you use the `callJs()` method of a `fixture` object, the `component` argument represents the actual UI component
found (see [Searching Components](#searching-components)) and used to initialize the `ComponentFixture`.The `runJs` method works the same way without any return value:
```java
public void click() {
runJs("const offset = component.getHeight()/2;"+
"robot.click("+
"component, "+
"new Point(offset, offset), "+
"MouseButton.LEFT_BUTTON, 1);"
);
}
```We import some packages to the context before the script is executed:
```java
java.awt
org.assertj.swing.core
org.assertj.swing.fixture
```You can add other packages or classes with
js [methods](https://www-archive.mozilla.org/rhino/apidocs/org/mozilla/javascript/importertoplevel):```java
importClass(java.io.File);
importPackage(java.io);
```Alternatively, you can just use the full path:
```java
Boolean isDumbMode=ideaFtame.callJs(
"com.intellij.openapi.project.DumbService.isDumb(component.project);"
);
```#### Store data between `runJs/callJs` requests
There are `global` and `local` `Map` variables available in the js context:
- `global` is a single map for the whole IDE
- `local` map is defined on a per-fixture basisPlease check [GlobalAndLocalMapExamples](/ui-test-example/src/test/kotlin/org/intellij/examples/simple/plugin/GlobalAndLocalMapExamplesTest.kt) for additional information.
In case you made robot-server-plugin port public, you may want to enable encryption for JavaScript code:
```groovy
runIdeForUiTests {
systemProperty "robot.encryption.enabled", "true"
systemProperty "robot.encryption.password", "secret"
}test {
systemProperty "robot.encryption.password", "secret"
}
```### Text
Sometimes, you may not want to dig through the whole component to determine which field contains the text you need to
reach. If you need to check whether some text is present on the component or you need to click the text, you can
use `fixture` methods:```java
welcomeFrame.findText("Create New Project").click();
assert(welcomeFrame.hasText(startsWith("Version 20")));
List renderedText=welcomeFrame.findAllText()
.stream()
.map(RemoteText::getText)
.collect(Collectors.toList());
```Instead of looking for the text inside the component structure, we render it on a fake `Graphics` to collect text data and
its points.### Screenshots
There are two ways to get a screenshot.
1. Get a screenshot of the whole screen (method of `RemoteRobot` object).
```java
remoteRobot.getScreenshot()
```2. Get component screenshot (method of `Fixture` object).
The `isPaintingMode` parameter allows you to return a new render of the component (set to `false` by default).
This might be helpful when you don't have a complete set of desktop environments or when any other component covers the
component of your interest.```java
someFixture.getScreenshot();
someFixture.getScreenshot(true);
```In both cases, you will get the `BufferedImage` object specified as `.png`.
### Kotlin
If you are familiar with Kotlin, please take a look at
the [kotlin example](/ui-test-example/src/test/kotlin/org/intellij/examples/simple/plugin/CreateCommandLineKotlinTest.kt). You may find it easier to read and use.### Steps Logging
We use the `step` wrapper method to make test logs easy to read. The `StepLogger` example shows how useful it can be.
For instance, by implementing your own `StepProcessor`, you can extend the steps workflow and connect to
the [allure report](https://docs.qameta.io/allure/) framework.### FAQ
[FAQ](/ui-test-example)