system-testing 1.0.73 → 1.0.74
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +51 -1
- package/build/drivers/appium-driver.d.ts +97 -0
- package/build/drivers/appium-driver.js +316 -0
- package/build/drivers/selenium-driver.d.ts +36 -0
- package/build/drivers/selenium-driver.js +44 -0
- package/build/drivers/webdriver-driver.d.ts +173 -0
- package/build/drivers/webdriver-driver.js +462 -0
- package/build/system-test-browser-helper.d.ts +4 -0
- package/build/system-test-browser-helper.js +28 -3
- package/build/system-test.d.ts +80 -10
- package/build/system-test.js +148 -299
- package/build/use-system-test.js +6 -1
- package/package.json +10 -3
package/README.md
CHANGED
|
@@ -62,10 +62,40 @@ describe("Sign in page", () => {
|
|
|
62
62
|
})
|
|
63
63
|
```
|
|
64
64
|
|
|
65
|
+
### Driver selection
|
|
66
|
+
|
|
67
|
+
SystemTest uses Selenium by default. To use Appium instead, pass a driver config when creating the instance:
|
|
68
|
+
|
|
69
|
+
```js
|
|
70
|
+
await SystemTest.run({
|
|
71
|
+
driver: {
|
|
72
|
+
type: "appium",
|
|
73
|
+
options: {
|
|
74
|
+
serverArgs: {
|
|
75
|
+
useDrivers: ["uiautomator2"],
|
|
76
|
+
port: 4723
|
|
77
|
+
},
|
|
78
|
+
capabilities: {
|
|
79
|
+
platformName: "Android",
|
|
80
|
+
"appium:automationName": "UiAutomator2",
|
|
81
|
+
"appium:deviceName": "Android Emulator",
|
|
82
|
+
"appium:app": "/path/to/app.apk"
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}, async (systemTest) => {
|
|
87
|
+
await systemTest.findByTestId("loginScreen")
|
|
88
|
+
})
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
If you already run an Appium server, provide `serverUrl` instead of `serverArgs`. By default, `findByTestId` uses the Appium `accessibility id` strategy. To use CSS instead (for web contexts), set `options.testIdStrategy` to `"css"` and optionally `options.testIdAttribute` (defaults to `"data-testid"`).
|
|
92
|
+
|
|
65
93
|
### Using `useSystemTest` in your Expo app
|
|
66
94
|
|
|
67
95
|
`useSystemTest` wires your Expo app to the system-testing runner: it listens for WebSocket commands, initializes the browser helper, and lets tests navigate or reset state. Add it near the root layout of your Expo Router app (for example in `_layout.tsx` or a top-level provider component).
|
|
68
96
|
|
|
97
|
+
To enable system tests in native builds, set `EXPO_PUBLIC_SYSTEM_TEST=true` at build time (and optionally `EXPO_PUBLIC_SYSTEM_TEST_HOST` to reach the test runner from a device/emulator). For native Appium runs, set `SYSTEM_TEST_HOST=native` in the test environment and point Appium at your APK.
|
|
98
|
+
|
|
69
99
|
Minimal example:
|
|
70
100
|
|
|
71
101
|
```jsx
|
|
@@ -104,12 +134,32 @@ Notes:
|
|
|
104
134
|
- If you need scoundrel remote evaluation, wait for `systemTestBrowserHelper` and register your classes there, as shown in the commented snippet above.
|
|
105
135
|
- Add a root wrapper with `testID="systemTestingComponent"` (and optionally `data-focussed="true"`) around your app so the runner has a stable element to detect and scope selectors against.
|
|
106
136
|
- From your tests, use `await systemTest.getScoundrelClient()` to obtain the browser Scoundrel client for remote evaluation.
|
|
137
|
+
- `useSystemTest` calls `useRouter()` from `expo-router`. If you are not using Expo Router, install `expo-router` or provide your own guard to avoid navigation errors.
|
|
138
|
+
|
|
139
|
+
### Root path and `blankText`
|
|
140
|
+
|
|
141
|
+
`SystemTest.run()` visits `SystemTest.rootPath` (defaults to `/blank?systemTest=true`) and waits for an element with `testID="blankText"` inside the focused `systemTestingComponent`. If your app does not have a `/blank` route, set a custom root path and ensure the element exists on that screen.
|
|
142
|
+
|
|
143
|
+
Example setup:
|
|
144
|
+
|
|
145
|
+
```js
|
|
146
|
+
import SystemTest from "system-testing/build/system-test.js"
|
|
147
|
+
|
|
148
|
+
SystemTest.rootPath = "/?platform=web&systemTest=true"
|
|
149
|
+
```
|
|
150
|
+
|
|
151
|
+
```jsx
|
|
152
|
+
<View testID="systemTestingComponent" dataSet={{focussed: "true"}}>
|
|
153
|
+
<Text testID="blankText">Blank</Text>
|
|
154
|
+
{children}
|
|
155
|
+
</View>
|
|
156
|
+
```
|
|
107
157
|
|
|
108
158
|
### Base selector and focused container
|
|
109
159
|
|
|
110
160
|
System tests scope selectors to the active screen by default. The app marks the active layout container with `data-focussed="true"` on the element with `data-testid="systemTestingComponent"`. In the dummy app, the root layout wraps the navigator and sets `data-focussed="true"` once so the base selector stays stable across screens.
|
|
111
161
|
|
|
112
|
-
`SystemTest.find` and `SystemTest.findByTestID` use a base selector that targets the focused container:
|
|
162
|
+
`SystemTest.find` and `SystemTest.findByTestId` (alias `findByTestID`) use a base selector that targets the focused container:
|
|
113
163
|
|
|
114
164
|
```css
|
|
115
165
|
[data-testid='systemTestingComponent'][data-focussed='true']
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {object} AppiumDriverOptions
|
|
3
|
+
* @property {string} [serverUrl] Remote Appium server URL to connect to.
|
|
4
|
+
* @property {Record<string, any>} [serverArgs] Options passed to the Appium server.
|
|
5
|
+
* @property {string[]} [useDrivers] Appium driver names to load when starting the server.
|
|
6
|
+
* @property {Record<string, any>} [capabilities] Desired capabilities for the session.
|
|
7
|
+
* @property {string} [browserName] Browser name for web sessions.
|
|
8
|
+
* @property {"accessibilityId"|"css"|"id"} [testIdStrategy] Strategy for resolving test IDs.
|
|
9
|
+
* @property {string} [testIdAttribute] Attribute name when using the CSS test ID strategy.
|
|
10
|
+
*/
|
|
11
|
+
/**
|
|
12
|
+
* @typedef {object} FindArgs
|
|
13
|
+
* @property {number} [timeout] Override timeout for lookup.
|
|
14
|
+
* @property {boolean} [visible] Whether to require elements to be visible.
|
|
15
|
+
* @property {boolean} [useBaseSelector] Whether to scope by the base selector.
|
|
16
|
+
*/
|
|
17
|
+
/**
|
|
18
|
+
* Appium driver backed by the Appium server package.
|
|
19
|
+
*/
|
|
20
|
+
export default class AppiumDriver extends WebDriverDriver {
|
|
21
|
+
serverUrl: any;
|
|
22
|
+
appiumServer: import("@appium/types").AppiumServer;
|
|
23
|
+
/**
|
|
24
|
+
* @param {string} testId
|
|
25
|
+
* @param {FindArgs} [args]
|
|
26
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
27
|
+
*/
|
|
28
|
+
findByAccessibilityId(testId: string, args?: FindArgs): Promise<import("selenium-webdriver").WebElement>;
|
|
29
|
+
/**
|
|
30
|
+
* @param {string} testId
|
|
31
|
+
* @param {FindArgs} [args]
|
|
32
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
33
|
+
*/
|
|
34
|
+
allByAccessibilityId(testId: string, args?: FindArgs): Promise<import("selenium-webdriver").WebElement[]>;
|
|
35
|
+
/**
|
|
36
|
+
* @param {string} testId
|
|
37
|
+
* @param {FindArgs} [args]
|
|
38
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
39
|
+
*/
|
|
40
|
+
findById(testId: string, args?: FindArgs): Promise<import("selenium-webdriver").WebElement>;
|
|
41
|
+
/**
|
|
42
|
+
* @param {string} testId
|
|
43
|
+
* @param {FindArgs} [args]
|
|
44
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
45
|
+
*/
|
|
46
|
+
allById(testId: string, args?: FindArgs): Promise<import("selenium-webdriver").WebElement[]>;
|
|
47
|
+
/**
|
|
48
|
+
* @param {Record<string, any>} serverArgs
|
|
49
|
+
* @returns {string}
|
|
50
|
+
*/
|
|
51
|
+
buildServerUrl(serverArgs: Record<string, any>): string;
|
|
52
|
+
}
|
|
53
|
+
export type AppiumDriverOptions = {
|
|
54
|
+
/**
|
|
55
|
+
* Remote Appium server URL to connect to.
|
|
56
|
+
*/
|
|
57
|
+
serverUrl?: string;
|
|
58
|
+
/**
|
|
59
|
+
* Options passed to the Appium server.
|
|
60
|
+
*/
|
|
61
|
+
serverArgs?: Record<string, any>;
|
|
62
|
+
/**
|
|
63
|
+
* Appium driver names to load when starting the server.
|
|
64
|
+
*/
|
|
65
|
+
useDrivers?: string[];
|
|
66
|
+
/**
|
|
67
|
+
* Desired capabilities for the session.
|
|
68
|
+
*/
|
|
69
|
+
capabilities?: Record<string, any>;
|
|
70
|
+
/**
|
|
71
|
+
* Browser name for web sessions.
|
|
72
|
+
*/
|
|
73
|
+
browserName?: string;
|
|
74
|
+
/**
|
|
75
|
+
* Strategy for resolving test IDs.
|
|
76
|
+
*/
|
|
77
|
+
testIdStrategy?: "accessibilityId" | "css" | "id";
|
|
78
|
+
/**
|
|
79
|
+
* Attribute name when using the CSS test ID strategy.
|
|
80
|
+
*/
|
|
81
|
+
testIdAttribute?: string;
|
|
82
|
+
};
|
|
83
|
+
export type FindArgs = {
|
|
84
|
+
/**
|
|
85
|
+
* Override timeout for lookup.
|
|
86
|
+
*/
|
|
87
|
+
timeout?: number;
|
|
88
|
+
/**
|
|
89
|
+
* Whether to require elements to be visible.
|
|
90
|
+
*/
|
|
91
|
+
visible?: boolean;
|
|
92
|
+
/**
|
|
93
|
+
* Whether to scope by the base selector.
|
|
94
|
+
*/
|
|
95
|
+
useBaseSelector?: boolean;
|
|
96
|
+
};
|
|
97
|
+
import WebDriverDriver from "./webdriver-driver.js";
|
|
@@ -0,0 +1,316 @@
|
|
|
1
|
+
import { Builder, By } from "selenium-webdriver";
|
|
2
|
+
import timeout from "awaitery/build/timeout.js";
|
|
3
|
+
import WebDriverDriver from "./webdriver-driver.js";
|
|
4
|
+
/**
|
|
5
|
+
* @typedef {object} AppiumDriverOptions
|
|
6
|
+
* @property {string} [serverUrl] Remote Appium server URL to connect to.
|
|
7
|
+
* @property {Record<string, any>} [serverArgs] Options passed to the Appium server.
|
|
8
|
+
* @property {string[]} [useDrivers] Appium driver names to load when starting the server.
|
|
9
|
+
* @property {Record<string, any>} [capabilities] Desired capabilities for the session.
|
|
10
|
+
* @property {string} [browserName] Browser name for web sessions.
|
|
11
|
+
* @property {"accessibilityId"|"css"|"id"} [testIdStrategy] Strategy for resolving test IDs.
|
|
12
|
+
* @property {string} [testIdAttribute] Attribute name when using the CSS test ID strategy.
|
|
13
|
+
*/
|
|
14
|
+
/**
|
|
15
|
+
* @typedef {object} FindArgs
|
|
16
|
+
* @property {number} [timeout] Override timeout for lookup.
|
|
17
|
+
* @property {boolean} [visible] Whether to require elements to be visible.
|
|
18
|
+
* @property {boolean} [useBaseSelector] Whether to scope by the base selector.
|
|
19
|
+
*/
|
|
20
|
+
/**
|
|
21
|
+
* Appium driver backed by the Appium server package.
|
|
22
|
+
*/
|
|
23
|
+
export default class AppiumDriver extends WebDriverDriver {
|
|
24
|
+
/**
|
|
25
|
+
* @returns {Promise<void>}
|
|
26
|
+
*/
|
|
27
|
+
async start() {
|
|
28
|
+
const serverArgs = { ...(this.options.serverArgs ?? {}) };
|
|
29
|
+
if (this.options.useDrivers && !serverArgs.useDrivers) {
|
|
30
|
+
serverArgs.useDrivers = this.options.useDrivers;
|
|
31
|
+
}
|
|
32
|
+
if (this.options.serverUrl) {
|
|
33
|
+
this.serverUrl = this.options.serverUrl;
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
const appiumModule = await import("appium");
|
|
37
|
+
const appiumMain = appiumModule.main ?? appiumModule.default?.main;
|
|
38
|
+
if (!appiumMain) {
|
|
39
|
+
throw new Error("Appium main() is unavailable from the appium package");
|
|
40
|
+
}
|
|
41
|
+
this.appiumServer = await appiumMain(serverArgs);
|
|
42
|
+
this.serverUrl = this.buildServerUrl(serverArgs);
|
|
43
|
+
}
|
|
44
|
+
const capabilities = this.options.capabilities ? { ...this.options.capabilities } : {};
|
|
45
|
+
if (this.options.browserName && !capabilities.browserName) {
|
|
46
|
+
capabilities.browserName = this.options.browserName;
|
|
47
|
+
}
|
|
48
|
+
if (capabilities.browserName === undefined) {
|
|
49
|
+
capabilities.browserName = "";
|
|
50
|
+
}
|
|
51
|
+
const builder = new Builder().usingServer(this.serverUrl);
|
|
52
|
+
if (Object.keys(capabilities).length > 0) {
|
|
53
|
+
builder.withCapabilities(capabilities);
|
|
54
|
+
}
|
|
55
|
+
const webDriver = await builder.build();
|
|
56
|
+
this.setWebDriver(webDriver);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* @returns {Promise<void>}
|
|
60
|
+
*/
|
|
61
|
+
async stop() {
|
|
62
|
+
try {
|
|
63
|
+
await super.stop();
|
|
64
|
+
}
|
|
65
|
+
finally {
|
|
66
|
+
if (this.appiumServer?.close) {
|
|
67
|
+
await timeout({ timeout: this.getTimeouts(), errorMessage: "timeout while closing Appium server" }, async () => await this.appiumServer.close());
|
|
68
|
+
}
|
|
69
|
+
this.appiumServer = undefined;
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* @returns {Promise<string[]>}
|
|
74
|
+
*/
|
|
75
|
+
async getBrowserLogs() {
|
|
76
|
+
const platformName = this.options.capabilities?.platformName;
|
|
77
|
+
const browserName = this.options.capabilities?.browserName ?? this.options.browserName;
|
|
78
|
+
const isAndroid = typeof platformName === "string" && platformName.toLowerCase() === "android";
|
|
79
|
+
if (isAndroid && !browserName) {
|
|
80
|
+
let entries;
|
|
81
|
+
try {
|
|
82
|
+
entries = await this.getWebDriver().manage().logs().get("logcat");
|
|
83
|
+
}
|
|
84
|
+
catch {
|
|
85
|
+
return [];
|
|
86
|
+
}
|
|
87
|
+
const logcatLogs = [];
|
|
88
|
+
for (const entry of entries) {
|
|
89
|
+
const messageMatch = entry.message.match(/^(.+) (\d+):(\d+) (.+)$/);
|
|
90
|
+
const message = messageMatch ? messageMatch[4] : entry.message;
|
|
91
|
+
logcatLogs.push(`${entry.level.name}: ${message}`);
|
|
92
|
+
}
|
|
93
|
+
return logcatLogs;
|
|
94
|
+
}
|
|
95
|
+
return await super.getBrowserLogs();
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Finds a single element by test ID.
|
|
99
|
+
* @param {string} testId
|
|
100
|
+
* @param {FindArgs} [args]
|
|
101
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
102
|
+
*/
|
|
103
|
+
async findByTestId(testId, args) {
|
|
104
|
+
const testIdStrategy = this.options.testIdStrategy ?? "accessibilityId";
|
|
105
|
+
if (testIdStrategy === "css") {
|
|
106
|
+
const testIdAttribute = this.options.testIdAttribute ?? "data-testid";
|
|
107
|
+
return await this.find(`[${testIdAttribute}='${testId}']`, args);
|
|
108
|
+
}
|
|
109
|
+
if (testIdStrategy === "id") {
|
|
110
|
+
return await this.findById(testId, args);
|
|
111
|
+
}
|
|
112
|
+
return await this.findByAccessibilityId(testId, args);
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Finds a single element by test ID.
|
|
116
|
+
* @param {string} testID
|
|
117
|
+
* @param {FindArgs} [args]
|
|
118
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
119
|
+
*/
|
|
120
|
+
async findByTestID(testID, args) {
|
|
121
|
+
return await this.findByTestId(testID, args);
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* @param {string} testId
|
|
125
|
+
* @param {FindArgs} [args]
|
|
126
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
127
|
+
*/
|
|
128
|
+
async findByAccessibilityId(testId, args = {}) {
|
|
129
|
+
const startTime = Date.now();
|
|
130
|
+
let elements = [];
|
|
131
|
+
try {
|
|
132
|
+
elements = await this.allByAccessibilityId(testId, args);
|
|
133
|
+
}
|
|
134
|
+
catch (error) {
|
|
135
|
+
// Re-throw to recover stack trace
|
|
136
|
+
if (error instanceof Error) {
|
|
137
|
+
if (error.message.startsWith("Wait timed out after")) {
|
|
138
|
+
elements = [];
|
|
139
|
+
}
|
|
140
|
+
throw new Error(`${error.constructor.name} - ${error.message} (accessibility id: ${testId})`);
|
|
141
|
+
}
|
|
142
|
+
else {
|
|
143
|
+
throw new Error(`${typeof error} - ${error} (accessibility id: ${testId})`);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
if (elements.length > 1) {
|
|
147
|
+
throw new Error(`More than 1 elements (${elements.length}) was found by accessibility id: ${testId}`);
|
|
148
|
+
}
|
|
149
|
+
if (!elements[0]) {
|
|
150
|
+
const elapsedSeconds = (Date.now() - startTime) / 1000;
|
|
151
|
+
throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by accessibility id: ${testId}`);
|
|
152
|
+
}
|
|
153
|
+
return elements[0];
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* @param {string} testId
|
|
157
|
+
* @param {FindArgs} [args]
|
|
158
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
159
|
+
*/
|
|
160
|
+
async allByAccessibilityId(testId, args = {}) {
|
|
161
|
+
const { visible = true, timeout, ...restArgs } = args;
|
|
162
|
+
const restArgsKeys = Object.keys(restArgs).filter((key) => key !== "useBaseSelector");
|
|
163
|
+
let actualTimeout;
|
|
164
|
+
if (timeout === undefined) {
|
|
165
|
+
actualTimeout = this._driverTimeouts;
|
|
166
|
+
}
|
|
167
|
+
else {
|
|
168
|
+
actualTimeout = timeout;
|
|
169
|
+
}
|
|
170
|
+
if (restArgsKeys.length > 0)
|
|
171
|
+
throw new Error(`Unknown arguments: ${restArgsKeys.join(", ")}`);
|
|
172
|
+
const startTime = Date.now();
|
|
173
|
+
const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0);
|
|
174
|
+
const getElements = async () => {
|
|
175
|
+
const foundElements = await this.getWebDriver().findElements(new By("accessibility id", testId));
|
|
176
|
+
if (visible !== true && visible !== false) {
|
|
177
|
+
return foundElements;
|
|
178
|
+
}
|
|
179
|
+
const filteredElements = [];
|
|
180
|
+
for (const element of foundElements) {
|
|
181
|
+
const isDisplayed = await element.isDisplayed();
|
|
182
|
+
if (visible && !isDisplayed)
|
|
183
|
+
continue;
|
|
184
|
+
if (!visible && isDisplayed)
|
|
185
|
+
continue;
|
|
186
|
+
filteredElements.push(element);
|
|
187
|
+
}
|
|
188
|
+
return filteredElements;
|
|
189
|
+
};
|
|
190
|
+
let elements = [];
|
|
191
|
+
while (true) {
|
|
192
|
+
const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft();
|
|
193
|
+
try {
|
|
194
|
+
if (timeLeft == 0) {
|
|
195
|
+
elements = await getElements();
|
|
196
|
+
}
|
|
197
|
+
else {
|
|
198
|
+
await this.getWebDriver().wait(async () => {
|
|
199
|
+
elements = await getElements();
|
|
200
|
+
return elements.length > 0;
|
|
201
|
+
}, timeLeft);
|
|
202
|
+
}
|
|
203
|
+
break;
|
|
204
|
+
}
|
|
205
|
+
catch (error) {
|
|
206
|
+
if (error instanceof Error && error.constructor.name === "TimeoutError" && getTimeLeft() > 0) {
|
|
207
|
+
continue;
|
|
208
|
+
}
|
|
209
|
+
throw new Error(`Couldn't get elements with accessibility id: ${testId}: ${error instanceof Error ? error.message : error}`);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
return elements;
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* @param {string} testId
|
|
216
|
+
* @param {FindArgs} [args]
|
|
217
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
218
|
+
*/
|
|
219
|
+
async findById(testId, args = {}) {
|
|
220
|
+
const startTime = Date.now();
|
|
221
|
+
let elements = [];
|
|
222
|
+
try {
|
|
223
|
+
elements = await this.allById(testId, args);
|
|
224
|
+
}
|
|
225
|
+
catch (error) {
|
|
226
|
+
// Re-throw to recover stack trace
|
|
227
|
+
if (error instanceof Error) {
|
|
228
|
+
if (error.message.startsWith("Wait timed out after")) {
|
|
229
|
+
elements = [];
|
|
230
|
+
}
|
|
231
|
+
throw new Error(`${error.constructor.name} - ${error.message} (id: ${testId})`);
|
|
232
|
+
}
|
|
233
|
+
else {
|
|
234
|
+
throw new Error(`${typeof error} - ${error} (id: ${testId})`);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
if (elements.length > 1) {
|
|
238
|
+
throw new Error(`More than 1 elements (${elements.length}) was found by id: ${testId}`);
|
|
239
|
+
}
|
|
240
|
+
if (!elements[0]) {
|
|
241
|
+
const elapsedSeconds = (Date.now() - startTime) / 1000;
|
|
242
|
+
throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by id: ${testId}`);
|
|
243
|
+
}
|
|
244
|
+
return elements[0];
|
|
245
|
+
}
|
|
246
|
+
/**
|
|
247
|
+
* @param {string} testId
|
|
248
|
+
* @param {FindArgs} [args]
|
|
249
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
250
|
+
*/
|
|
251
|
+
async allById(testId, args = {}) {
|
|
252
|
+
const { visible = true, timeout, ...restArgs } = args;
|
|
253
|
+
const restArgsKeys = Object.keys(restArgs).filter((key) => key !== "useBaseSelector");
|
|
254
|
+
let actualTimeout;
|
|
255
|
+
if (timeout === undefined) {
|
|
256
|
+
actualTimeout = this._driverTimeouts;
|
|
257
|
+
}
|
|
258
|
+
else {
|
|
259
|
+
actualTimeout = timeout;
|
|
260
|
+
}
|
|
261
|
+
if (restArgsKeys.length > 0)
|
|
262
|
+
throw new Error(`Unknown arguments: ${restArgsKeys.join(", ")}`);
|
|
263
|
+
const startTime = Date.now();
|
|
264
|
+
const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0);
|
|
265
|
+
const getElements = async () => {
|
|
266
|
+
const foundElements = await this.getWebDriver().findElements(By.id(testId));
|
|
267
|
+
if (visible !== true && visible !== false) {
|
|
268
|
+
return foundElements;
|
|
269
|
+
}
|
|
270
|
+
const filteredElements = [];
|
|
271
|
+
for (const element of foundElements) {
|
|
272
|
+
const isDisplayed = await element.isDisplayed();
|
|
273
|
+
if (visible && !isDisplayed)
|
|
274
|
+
continue;
|
|
275
|
+
if (!visible && isDisplayed)
|
|
276
|
+
continue;
|
|
277
|
+
filteredElements.push(element);
|
|
278
|
+
}
|
|
279
|
+
return filteredElements;
|
|
280
|
+
};
|
|
281
|
+
let elements = [];
|
|
282
|
+
while (true) {
|
|
283
|
+
const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft();
|
|
284
|
+
try {
|
|
285
|
+
if (timeLeft == 0) {
|
|
286
|
+
elements = await getElements();
|
|
287
|
+
}
|
|
288
|
+
else {
|
|
289
|
+
await this.getWebDriver().wait(async () => {
|
|
290
|
+
elements = await getElements();
|
|
291
|
+
return elements.length > 0;
|
|
292
|
+
}, timeLeft);
|
|
293
|
+
}
|
|
294
|
+
break;
|
|
295
|
+
}
|
|
296
|
+
catch (error) {
|
|
297
|
+
if (error instanceof Error && error.constructor.name === "TimeoutError" && getTimeLeft() > 0) {
|
|
298
|
+
continue;
|
|
299
|
+
}
|
|
300
|
+
throw new Error(`Couldn't get elements with id: ${testId}: ${error instanceof Error ? error.message : error}`);
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
return elements;
|
|
304
|
+
}
|
|
305
|
+
/**
|
|
306
|
+
* @param {Record<string, any>} serverArgs
|
|
307
|
+
* @returns {string}
|
|
308
|
+
*/
|
|
309
|
+
buildServerUrl(serverArgs) {
|
|
310
|
+
const address = serverArgs.address ?? "127.0.0.1";
|
|
311
|
+
const port = serverArgs.port ?? 4723;
|
|
312
|
+
const basePath = serverArgs.basePath ? (serverArgs.basePath.startsWith("/") ? serverArgs.basePath : `/${serverArgs.basePath}`) : "";
|
|
313
|
+
return `http://${address}:${port}${basePath}`;
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"appium-driver.js","sourceRoot":"","sources":["../../src/drivers/appium-driver.js"],"names":[],"mappings":"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","sourcesContent":["import {Builder, By} from \"selenium-webdriver\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport WebDriverDriver from \"./webdriver-driver.js\"\n\n/**\n * @typedef {object} AppiumDriverOptions\n * @property {string} [serverUrl] Remote Appium server URL to connect to.\n * @property {Record<string, any>} [serverArgs] Options passed to the Appium server.\n * @property {string[]} [useDrivers] Appium driver names to load when starting the server.\n * @property {Record<string, any>} [capabilities] Desired capabilities for the session.\n * @property {string} [browserName] Browser name for web sessions.\n * @property {\"accessibilityId\"|\"css\"|\"id\"} [testIdStrategy] Strategy for resolving test IDs.\n * @property {string} [testIdAttribute] Attribute name when using the CSS test ID strategy.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean} [visible] Whether to require elements to be visible.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n\n/**\n * Appium driver backed by the Appium server package.\n */\nexport default class AppiumDriver extends WebDriverDriver {\n  /**\n   * @returns {Promise<void>}\n   */\n  async start() {\n    const serverArgs = {...(this.options.serverArgs ?? {})}\n\n    if (this.options.useDrivers && !serverArgs.useDrivers) {\n      serverArgs.useDrivers = this.options.useDrivers\n    }\n\n    if (this.options.serverUrl) {\n      this.serverUrl = this.options.serverUrl\n    } else {\n      const appiumModule = await import(\"appium\")\n      const appiumMain = appiumModule.main ?? appiumModule.default?.main\n\n      if (!appiumMain) {\n        throw new Error(\"Appium main() is unavailable from the appium package\")\n      }\n\n      this.appiumServer = await appiumMain(serverArgs)\n      this.serverUrl = this.buildServerUrl(serverArgs)\n    }\n\n    const capabilities = this.options.capabilities ? {...this.options.capabilities} : {}\n\n    if (this.options.browserName && !capabilities.browserName) {\n      capabilities.browserName = this.options.browserName\n    }\n    if (capabilities.browserName === undefined) {\n      capabilities.browserName = \"\"\n    }\n\n    const builder = new Builder().usingServer(this.serverUrl)\n\n    if (Object.keys(capabilities).length > 0) {\n      builder.withCapabilities(capabilities)\n    }\n\n    const webDriver = await builder.build()\n\n    this.setWebDriver(webDriver)\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async stop() {\n    try {\n      await super.stop()\n    } finally {\n      if (this.appiumServer?.close) {\n        await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while closing Appium server\"}, async () => await this.appiumServer.close())\n      }\n\n      this.appiumServer = undefined\n    }\n  }\n\n  /**\n   * @returns {Promise<string[]>}\n   */\n  async getBrowserLogs() {\n    const platformName = this.options.capabilities?.platformName\n    const browserName = this.options.capabilities?.browserName ?? this.options.browserName\n    const isAndroid = typeof platformName === \"string\" && platformName.toLowerCase() === \"android\"\n\n    if (isAndroid && !browserName) {\n      let entries\n\n      try {\n        entries = await this.getWebDriver().manage().logs().get(\"logcat\")\n      } catch {\n        return []\n      }\n\n      const logcatLogs = []\n\n      for (const entry of entries) {\n        const messageMatch = entry.message.match(/^(.+) (\\d+):(\\d+) (.+)$/)\n        const message = messageMatch ? messageMatch[4] : entry.message\n\n        logcatLogs.push(`${entry.level.name}: ${message}`)\n      }\n\n      return logcatLogs\n    }\n\n    return await super.getBrowserLogs()\n  }\n\n  /**\n   * Finds a single element by test ID.\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestId(testId, args) {\n    const testIdStrategy = this.options.testIdStrategy ?? \"accessibilityId\"\n\n    if (testIdStrategy === \"css\") {\n      const testIdAttribute = this.options.testIdAttribute ?? \"data-testid\"\n      return await this.find(`[${testIdAttribute}='${testId}']`, args)\n    }\n    if (testIdStrategy === \"id\") {\n      return await this.findById(testId, args)\n    }\n\n    return await this.findByAccessibilityId(testId, args)\n  }\n\n  /**\n   * Finds a single element by test ID.\n   * @param {string} testID\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestID(testID, args) {\n    return await this.findByTestId(testID, args)\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByAccessibilityId(testId, args = {}) {\n    const startTime = Date.now()\n    let elements = []\n\n    try {\n      elements = await this.allByAccessibilityId(testId, args)\n    } catch (error) {\n      // Re-throw to recover stack trace\n      if (error instanceof Error) {\n        if (error.message.startsWith(\"Wait timed out after\")) {\n          elements = []\n        }\n\n        throw new Error(`${error.constructor.name} - ${error.message} (accessibility id: ${testId})`)\n      } else {\n        throw new Error(`${typeof error} - ${error} (accessibility id: ${testId})`)\n      }\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by accessibility id: ${testId}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by accessibility id: ${testId}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async allByAccessibilityId(testId, args = {}) {\n    const {visible = true, timeout, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs).filter((key) => key !== \"useBaseSelector\")\n    let actualTimeout\n\n    if (timeout === undefined) {\n      actualTimeout = this._driverTimeouts\n    } else {\n      actualTimeout = timeout\n    }\n\n    if (restArgsKeys.length > 0) throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n\n    const startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(new By(\"accessibility id\", testId))\n\n      if (visible !== true && visible !== false) {\n        return foundElements\n      }\n\n      const filteredElements = []\n\n      for (const element of foundElements) {\n        const isDisplayed = await element.isDisplayed()\n\n        if (visible && !isDisplayed) continue\n        if (!visible && isDisplayed) continue\n\n        filteredElements.push(element)\n      }\n\n      return filteredElements\n    }\n    let elements = []\n\n    while (true) {\n      const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft()\n\n      try {\n        if (timeLeft == 0) {\n          elements = await getElements()\n        } else {\n          await this.getWebDriver().wait(async () => {\n            elements = await getElements()\n\n            return elements.length > 0\n          }, timeLeft)\n        }\n\n        break\n      } catch (error) {\n        if (error instanceof Error && error.constructor.name === \"TimeoutError\" && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw new Error(`Couldn't get elements with accessibility id: ${testId}: ${error instanceof Error ? error.message : error}`)\n      }\n    }\n\n    return elements\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findById(testId, args = {}) {\n    const startTime = Date.now()\n    let elements = []\n\n    try {\n      elements = await this.allById(testId, args)\n    } catch (error) {\n      // Re-throw to recover stack trace\n      if (error instanceof Error) {\n        if (error.message.startsWith(\"Wait timed out after\")) {\n          elements = []\n        }\n\n        throw new Error(`${error.constructor.name} - ${error.message} (id: ${testId})`)\n      } else {\n        throw new Error(`${typeof error} - ${error} (id: ${testId})`)\n      }\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by id: ${testId}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by id: ${testId}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async allById(testId, args = {}) {\n    const {visible = true, timeout, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs).filter((key) => key !== \"useBaseSelector\")\n    let actualTimeout\n\n    if (timeout === undefined) {\n      actualTimeout = this._driverTimeouts\n    } else {\n      actualTimeout = timeout\n    }\n\n    if (restArgsKeys.length > 0) throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n\n    const startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(By.id(testId))\n\n      if (visible !== true && visible !== false) {\n        return foundElements\n      }\n\n      const filteredElements = []\n\n      for (const element of foundElements) {\n        const isDisplayed = await element.isDisplayed()\n\n        if (visible && !isDisplayed) continue\n        if (!visible && isDisplayed) continue\n\n        filteredElements.push(element)\n      }\n\n      return filteredElements\n    }\n    let elements = []\n\n    while (true) {\n      const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft()\n\n      try {\n        if (timeLeft == 0) {\n          elements = await getElements()\n        } else {\n          await this.getWebDriver().wait(async () => {\n            elements = await getElements()\n\n            return elements.length > 0\n          }, timeLeft)\n        }\n\n        break\n      } catch (error) {\n        if (error instanceof Error && error.constructor.name === \"TimeoutError\" && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw new Error(`Couldn't get elements with id: ${testId}: ${error instanceof Error ? error.message : error}`)\n      }\n    }\n\n    return elements\n  }\n\n  /**\n   * @param {Record<string, any>} serverArgs\n   * @returns {string}\n   */\n  buildServerUrl(serverArgs) {\n    const address = serverArgs.address ?? \"127.0.0.1\"\n    const port = serverArgs.port ?? 4723\n    const basePath = serverArgs.basePath ? (serverArgs.basePath.startsWith(\"/\") ? serverArgs.basePath : `/${serverArgs.basePath}`) : \"\"\n\n    return `http://${address}:${port}${basePath}`\n  }\n}\n"]}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {object} SeleniumDriverOptions
|
|
3
|
+
* @property {string} [browserName] Browser name used by the WebDriver session.
|
|
4
|
+
* @property {string[]} [chromeArguments] Chrome CLI arguments.
|
|
5
|
+
* @property {import("selenium-webdriver/chrome.js").Options} [chromeOptions] Preconfigured Chrome options instance.
|
|
6
|
+
* @property {Record<string, any>} [capabilities] Extra WebDriver capabilities.
|
|
7
|
+
* @property {Record<string, any>} [loggingPrefs] Logging preferences for browser logs.
|
|
8
|
+
*/
|
|
9
|
+
/**
|
|
10
|
+
* Selenium WebDriver implementation.
|
|
11
|
+
*/
|
|
12
|
+
export default class SeleniumDriver extends WebDriverDriver {
|
|
13
|
+
}
|
|
14
|
+
export type SeleniumDriverOptions = {
|
|
15
|
+
/**
|
|
16
|
+
* Browser name used by the WebDriver session.
|
|
17
|
+
*/
|
|
18
|
+
browserName?: string;
|
|
19
|
+
/**
|
|
20
|
+
* Chrome CLI arguments.
|
|
21
|
+
*/
|
|
22
|
+
chromeArguments?: string[];
|
|
23
|
+
/**
|
|
24
|
+
* Preconfigured Chrome options instance.
|
|
25
|
+
*/
|
|
26
|
+
chromeOptions?: import("selenium-webdriver/chrome.js").Options;
|
|
27
|
+
/**
|
|
28
|
+
* Extra WebDriver capabilities.
|
|
29
|
+
*/
|
|
30
|
+
capabilities?: Record<string, any>;
|
|
31
|
+
/**
|
|
32
|
+
* Logging preferences for browser logs.
|
|
33
|
+
*/
|
|
34
|
+
loggingPrefs?: Record<string, any>;
|
|
35
|
+
};
|
|
36
|
+
import WebDriverDriver from "./webdriver-driver.js";
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { Builder } from "selenium-webdriver";
|
|
2
|
+
import chrome from "selenium-webdriver/chrome.js";
|
|
3
|
+
import WebDriverDriver from "./webdriver-driver.js";
|
|
4
|
+
/**
|
|
5
|
+
* @typedef {object} SeleniumDriverOptions
|
|
6
|
+
* @property {string} [browserName] Browser name used by the WebDriver session.
|
|
7
|
+
* @property {string[]} [chromeArguments] Chrome CLI arguments.
|
|
8
|
+
* @property {import("selenium-webdriver/chrome.js").Options} [chromeOptions] Preconfigured Chrome options instance.
|
|
9
|
+
* @property {Record<string, any>} [capabilities] Extra WebDriver capabilities.
|
|
10
|
+
* @property {Record<string, any>} [loggingPrefs] Logging preferences for browser logs.
|
|
11
|
+
*/
|
|
12
|
+
/**
|
|
13
|
+
* Selenium WebDriver implementation.
|
|
14
|
+
*/
|
|
15
|
+
export default class SeleniumDriver extends WebDriverDriver {
|
|
16
|
+
/**
|
|
17
|
+
* @returns {Promise<void>}
|
|
18
|
+
*/
|
|
19
|
+
async start() {
|
|
20
|
+
const chromeOptions = this.options.chromeOptions ? this.options.chromeOptions : new chrome.Options();
|
|
21
|
+
const chromeArguments = this.options.chromeArguments ?? [
|
|
22
|
+
"--disable-dev-shm-usage",
|
|
23
|
+
"--disable-gpu",
|
|
24
|
+
"--headless=new",
|
|
25
|
+
"--no-sandbox",
|
|
26
|
+
"--window-size=1920,1080"
|
|
27
|
+
];
|
|
28
|
+
for (const argument of chromeArguments) {
|
|
29
|
+
chromeOptions.addArguments(argument);
|
|
30
|
+
}
|
|
31
|
+
const builder = new Builder().forBrowser(this.options.browserName ?? "chrome").setChromeOptions(chromeOptions);
|
|
32
|
+
const capabilities = builder.getCapabilities();
|
|
33
|
+
const loggingPrefs = this.options.loggingPrefs ?? { browser: "ALL" };
|
|
34
|
+
capabilities.set("goog:loggingPrefs", loggingPrefs);
|
|
35
|
+
if (this.options.capabilities) {
|
|
36
|
+
for (const [key, value] of Object.entries(this.options.capabilities)) {
|
|
37
|
+
capabilities.set(key, value);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
const webDriver = await builder.build();
|
|
41
|
+
this.setWebDriver(webDriver);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZW5pdW0tZHJpdmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RyaXZlcnMvc2VsZW5pdW0tZHJpdmVyLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBQyxPQUFPLEVBQUMsTUFBTSxvQkFBb0IsQ0FBQTtBQUMxQyxPQUFPLE1BQU0sTUFBTSw4QkFBOEIsQ0FBQTtBQUNqRCxPQUFPLGVBQWUsTUFBTSx1QkFBdUIsQ0FBQTtBQUVuRDs7Ozs7OztHQU9HO0FBRUg7O0dBRUc7QUFDSCxNQUFNLENBQUMsT0FBTyxPQUFPLGNBQWUsU0FBUSxlQUFlO0lBQ3pEOztPQUVHO0lBQ0gsS0FBSyxDQUFDLEtBQUs7UUFDVCxNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQyxDQUFDLElBQUksTUFBTSxDQUFDLE9BQU8sRUFBRSxDQUFBO1FBQ3BHLE1BQU0sZUFBZSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsZUFBZSxJQUFJO1lBQ3RELHlCQUF5QjtZQUN6QixlQUFlO1lBQ2YsZ0JBQWdCO1lBQ2hCLGNBQWM7WUFDZCx5QkFBeUI7U0FDMUIsQ0FBQTtRQUVELEtBQUssTUFBTSxRQUFRLElBQUksZUFBZSxFQUFFLENBQUM7WUFDdkMsYUFBYSxDQUFDLFlBQVksQ0FBQyxRQUFRLENBQUMsQ0FBQTtRQUN0QyxDQUFDO1FBRUQsTUFBTSxPQUFPLEdBQUcsSUFBSSxPQUFPLEVBQUUsQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLElBQUksUUFBUSxDQUFDLENBQUMsZ0JBQWdCLENBQUMsYUFBYSxDQUFDLENBQUE7UUFDOUcsTUFBTSxZQUFZLEdBQUcsT0FBTyxDQUFDLGVBQWUsRUFBRSxDQUFBO1FBRTlDLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsWUFBWSxJQUFJLEVBQUMsT0FBTyxFQUFFLEtBQUssRUFBQyxDQUFBO1FBQ2xFLFlBQVksQ0FBQyxHQUFHLENBQUMsbUJBQW1CLEVBQUUsWUFBWSxDQUFDLENBQUE7UUFFbkQsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxDQUFDO1lBQzlCLEtBQUssTUFBTSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztnQkFDckUsWUFBWSxDQUFDLEdBQUcsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUE7WUFDOUIsQ0FBQztRQUNILENBQUM7UUFFRCxNQUFNLFNBQVMsR0FBRyxNQUFNLE9BQU8sQ0FBQyxLQUFLLEVBQUUsQ0FBQTtRQUV2QyxJQUFJLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxDQUFBO0lBQzlCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7QnVpbGRlcn0gZnJvbSBcInNlbGVuaXVtLXdlYmRyaXZlclwiXG5pbXBvcnQgY2hyb21lIGZyb20gXCJzZWxlbml1bS13ZWJkcml2ZXIvY2hyb21lLmpzXCJcbmltcG9ydCBXZWJEcml2ZXJEcml2ZXIgZnJvbSBcIi4vd2ViZHJpdmVyLWRyaXZlci5qc1wiXG5cbi8qKlxuICogQHR5cGVkZWYge29iamVjdH0gU2VsZW5pdW1Ecml2ZXJPcHRpb25zXG4gKiBAcHJvcGVydHkge3N0cmluZ30gW2Jyb3dzZXJOYW1lXSBCcm93c2VyIG5hbWUgdXNlZCBieSB0aGUgV2ViRHJpdmVyIHNlc3Npb24uXG4gKiBAcHJvcGVydHkge3N0cmluZ1tdfSBbY2hyb21lQXJndW1lbnRzXSBDaHJvbWUgQ0xJIGFyZ3VtZW50cy5cbiAqIEBwcm9wZXJ0eSB7aW1wb3J0KFwic2VsZW5pdW0td2ViZHJpdmVyL2Nocm9tZS5qc1wiKS5PcHRpb25zfSBbY2hyb21lT3B0aW9uc10gUHJlY29uZmlndXJlZCBDaHJvbWUgb3B0aW9ucyBpbnN0YW5jZS5cbiAqIEBwcm9wZXJ0eSB7UmVjb3JkPHN0cmluZywgYW55Pn0gW2NhcGFiaWxpdGllc10gRXh0cmEgV2ViRHJpdmVyIGNhcGFiaWxpdGllcy5cbiAqIEBwcm9wZXJ0eSB7UmVjb3JkPHN0cmluZywgYW55Pn0gW2xvZ2dpbmdQcmVmc10gTG9nZ2luZyBwcmVmZXJlbmNlcyBmb3IgYnJvd3NlciBsb2dzLlxuICovXG5cbi8qKlxuICogU2VsZW5pdW0gV2ViRHJpdmVyIGltcGxlbWVudGF0aW9uLlxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBTZWxlbml1bURyaXZlciBleHRlbmRzIFdlYkRyaXZlckRyaXZlciB7XG4gIC8qKlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTx2b2lkPn1cbiAgICovXG4gIGFzeW5jIHN0YXJ0KCkge1xuICAgIGNvbnN0IGNocm9tZU9wdGlvbnMgPSB0aGlzLm9wdGlvbnMuY2hyb21lT3B0aW9ucyA/IHRoaXMub3B0aW9ucy5jaHJvbWVPcHRpb25zIDogbmV3IGNocm9tZS5PcHRpb25zKClcbiAgICBjb25zdCBjaHJvbWVBcmd1bWVudHMgPSB0aGlzLm9wdGlvbnMuY2hyb21lQXJndW1lbnRzID8/IFtcbiAgICAgIFwiLS1kaXNhYmxlLWRldi1zaG0tdXNhZ2VcIixcbiAgICAgIFwiLS1kaXNhYmxlLWdwdVwiLFxuICAgICAgXCItLWhlYWRsZXNzPW5ld1wiLFxuICAgICAgXCItLW5vLXNhbmRib3hcIixcbiAgICAgIFwiLS13aW5kb3ctc2l6ZT0xOTIwLDEwODBcIlxuICAgIF1cblxuICAgIGZvciAoY29uc3QgYXJndW1lbnQgb2YgY2hyb21lQXJndW1lbnRzKSB7XG4gICAgICBjaHJvbWVPcHRpb25zLmFkZEFyZ3VtZW50cyhhcmd1bWVudClcbiAgICB9XG5cbiAgICBjb25zdCBidWlsZGVyID0gbmV3IEJ1aWxkZXIoKS5mb3JCcm93c2VyKHRoaXMub3B0aW9ucy5icm93c2VyTmFtZSA/PyBcImNocm9tZVwiKS5zZXRDaHJvbWVPcHRpb25zKGNocm9tZU9wdGlvbnMpXG4gICAgY29uc3QgY2FwYWJpbGl0aWVzID0gYnVpbGRlci5nZXRDYXBhYmlsaXRpZXMoKVxuXG4gICAgY29uc3QgbG9nZ2luZ1ByZWZzID0gdGhpcy5vcHRpb25zLmxvZ2dpbmdQcmVmcyA/PyB7YnJvd3NlcjogXCJBTExcIn1cbiAgICBjYXBhYmlsaXRpZXMuc2V0KFwiZ29vZzpsb2dnaW5nUHJlZnNcIiwgbG9nZ2luZ1ByZWZzKVxuXG4gICAgaWYgKHRoaXMub3B0aW9ucy5jYXBhYmlsaXRpZXMpIHtcbiAgICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKHRoaXMub3B0aW9ucy5jYXBhYmlsaXRpZXMpKSB7XG4gICAgICAgIGNhcGFiaWxpdGllcy5zZXQoa2V5LCB2YWx1ZSlcbiAgICAgIH1cbiAgICB9XG5cbiAgICBjb25zdCB3ZWJEcml2ZXIgPSBhd2FpdCBidWlsZGVyLmJ1aWxkKClcblxuICAgIHRoaXMuc2V0V2ViRHJpdmVyKHdlYkRyaXZlcilcbiAgfVxufVxuIl19
|