system-testing 1.0.35 → 1.0.37
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/build/index.js +3 -0
- package/{dist → build}/system-test-browser-helper.d.ts +2 -2
- package/build/system-test-browser-helper.js +238 -0
- package/build/system-test-communicator.js +105 -0
- package/build/system-test-http-server.js +66 -0
- package/{dist → build}/system-test.d.ts +15 -24
- package/build/system-test.js +696 -0
- package/build/use-system-test.js +79 -0
- package/package.json +10 -6
- package/dist/index.js +0 -3
- package/dist/index.js.map +0 -1
- package/dist/system-test-browser-helper.js +0 -238
- package/dist/system-test-browser-helper.js.map +0 -1
- package/dist/system-test-communicator.js +0 -105
- package/dist/system-test-communicator.js.map +0 -1
- package/dist/system-test-http-server.js +0 -66
- package/dist/system-test-http-server.js.map +0 -1
- package/dist/system-test.js +0 -705
- package/dist/system-test.js.map +0 -1
- package/dist/use-system-test.js +0 -79
- package/dist/use-system-test.js.map +0 -1
- /package/{dist → build}/index.d.ts +0 -0
- /package/{dist → build}/system-test-communicator.d.ts +0 -0
- /package/{dist → build}/system-test-http-server.d.ts +0 -0
- /package/{dist → build}/use-system-test.d.ts +0 -0
package/build/index.js
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
const stub = "Hello world";
|
|
2
|
+
export { stub };
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXguanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxJQUFJLEdBQUcsYUFBYSxDQUFBO0FBRTFCLE9BQU8sRUFBQyxJQUFJLEVBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImNvbnN0IHN0dWIgPSBcIkhlbGxvIHdvcmxkXCJcblxuZXhwb3J0IHtzdHVifVxuIl19
|
|
@@ -2,7 +2,7 @@ export default class SystemTestBrowserHelper {
|
|
|
2
2
|
static current(): SystemTestBrowserHelper;
|
|
3
3
|
communicator: SystemTestCommunicator;
|
|
4
4
|
_enabled: boolean;
|
|
5
|
-
events: EventEmitter<any>;
|
|
5
|
+
events: EventEmitter<string | symbol, any>;
|
|
6
6
|
startScoundrel(): Promise<void>;
|
|
7
7
|
scoundrelWs: WebSocket;
|
|
8
8
|
scoundrelClientWebSocket: ClientWebSocket;
|
|
@@ -103,6 +103,6 @@ export default class SystemTestBrowserHelper {
|
|
|
103
103
|
sendQuery(sql: string): Promise<Array<Record<string, any>>>;
|
|
104
104
|
}
|
|
105
105
|
import SystemTestCommunicator from "./system-test-communicator.js";
|
|
106
|
-
import EventEmitter from "
|
|
106
|
+
import { EventEmitter } from "eventemitter3";
|
|
107
107
|
import ClientWebSocket from "scoundrel-remote-eval/build/client/connections/web-socket/index.js";
|
|
108
108
|
import Client from "scoundrel-remote-eval/build/client/index.js";
|
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import Client from "scoundrel-remote-eval/build/client/index.js";
|
|
3
|
+
import ClientWebSocket from "scoundrel-remote-eval/build/client/connections/web-socket/index.js";
|
|
4
|
+
import { digg } from "diggerize";
|
|
5
|
+
import { EventEmitter } from "eventemitter3";
|
|
6
|
+
import SystemTestCommunicator from "./system-test-communicator.js";
|
|
7
|
+
/** @type {{systemTestBrowserHelper: SystemTestBrowserHelper | null}} */
|
|
8
|
+
const shared = {
|
|
9
|
+
systemTestBrowserHelper: null
|
|
10
|
+
};
|
|
11
|
+
export default class SystemTestBrowserHelper {
|
|
12
|
+
static current() {
|
|
13
|
+
if (!shared.systemTestBrowserHelper) {
|
|
14
|
+
throw new Error("No current SystemTestBrowserHelper set");
|
|
15
|
+
}
|
|
16
|
+
return shared.systemTestBrowserHelper;
|
|
17
|
+
}
|
|
18
|
+
constructor() {
|
|
19
|
+
/**
|
|
20
|
+
* @param {any[]} args
|
|
21
|
+
* @returns {void}
|
|
22
|
+
*/
|
|
23
|
+
this.fakeConsoleError = (...args) => {
|
|
24
|
+
this.communicator.sendCommand({ type: "console.error", value: this.consoleLogMessage(args) });
|
|
25
|
+
if (this.originalConsoleError) {
|
|
26
|
+
return this.originalConsoleError(...args);
|
|
27
|
+
}
|
|
28
|
+
};
|
|
29
|
+
/**
|
|
30
|
+
* @param {any[]} args
|
|
31
|
+
* @returns {void}
|
|
32
|
+
*/
|
|
33
|
+
this.fakeConsoleLog = (...args) => {
|
|
34
|
+
this.communicator.sendCommand({ type: "console.log", value: this.consoleLogMessage(args) });
|
|
35
|
+
if (this.originalConsoleLog) {
|
|
36
|
+
return this.originalConsoleLog(...args);
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
/**
|
|
40
|
+
* @param {{data: {path: string, type: string}}} args
|
|
41
|
+
* @returns {Promise<{result: string} | void>}
|
|
42
|
+
*/
|
|
43
|
+
this.onCommand = async ({ data }) => {
|
|
44
|
+
if (data.type == "initialize") {
|
|
45
|
+
this.events.emit("initialize");
|
|
46
|
+
if (this._onInitializeCallback) {
|
|
47
|
+
await this._onInitializeCallback();
|
|
48
|
+
}
|
|
49
|
+
return { result: "initialized" };
|
|
50
|
+
}
|
|
51
|
+
else if (data.type == "visit") {
|
|
52
|
+
this.events.emit("navigate", { path: data.path });
|
|
53
|
+
}
|
|
54
|
+
else if (data.type == "dismissTo") {
|
|
55
|
+
this.events.emit("dismissTo", { path: data.path });
|
|
56
|
+
}
|
|
57
|
+
else {
|
|
58
|
+
throw new Error(`Unknown command type for SystemTestBrowserHelper: ${data.type}`);
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
this.communicator = new SystemTestCommunicator({ parent: this, onCommand: this.onCommand });
|
|
62
|
+
this._enabled = false;
|
|
63
|
+
this.events = new EventEmitter();
|
|
64
|
+
shared.systemTestBrowserHelper = this;
|
|
65
|
+
this.startScoundrel();
|
|
66
|
+
}
|
|
67
|
+
async startScoundrel() {
|
|
68
|
+
this.scoundrelWs = new WebSocket("http://localhost:8090");
|
|
69
|
+
// @ts-expect-error
|
|
70
|
+
this.scoundrelClientWebSocket = new ClientWebSocket(this.scoundrelWs);
|
|
71
|
+
await this.scoundrelClientWebSocket.waitForOpened();
|
|
72
|
+
this.scoundrelClient = new Client(this.scoundrelClientWebSocket);
|
|
73
|
+
this.events.emit("scoundrelStarted");
|
|
74
|
+
}
|
|
75
|
+
waitForScoundrelStarted() {
|
|
76
|
+
return new Promise((resolve) => {
|
|
77
|
+
if (this.scoundrelClient) {
|
|
78
|
+
resolve(undefined);
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
this.events.once("scoundrelStarted", () => {
|
|
82
|
+
resolve(undefined);
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
getScoundrel() {
|
|
88
|
+
if (!this.scoundrelClient) {
|
|
89
|
+
throw new Error("Scoundrel client is not started yet");
|
|
90
|
+
}
|
|
91
|
+
return this.scoundrelClient;
|
|
92
|
+
}
|
|
93
|
+
connectOnError() {
|
|
94
|
+
window.addEventListener("error", (event) => {
|
|
95
|
+
this.handleError({
|
|
96
|
+
type: "error",
|
|
97
|
+
error: event.error,
|
|
98
|
+
errorClass: event.error?.name,
|
|
99
|
+
file: event.filename,
|
|
100
|
+
line: event.lineno,
|
|
101
|
+
message: event.message || "Unknown error",
|
|
102
|
+
url: window.location.href
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
connectUnhandledRejection() {
|
|
107
|
+
window.addEventListener("unhandledrejection", (event) => {
|
|
108
|
+
this.handleError({
|
|
109
|
+
type: "unhandledrejection",
|
|
110
|
+
error: event.reason,
|
|
111
|
+
errorClass: "UnhandledRejection",
|
|
112
|
+
message: event.reason.message || event.reason || "Unhandled promise rejection without a message",
|
|
113
|
+
url: window.location.href
|
|
114
|
+
});
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* @param {object} data
|
|
119
|
+
* @param {string} [data.backtrace]
|
|
120
|
+
* @param {Error} [data.error]
|
|
121
|
+
* @param {string} [data.errorClass]
|
|
122
|
+
* @param {string} [data.file]
|
|
123
|
+
* @param {number} [data.line]
|
|
124
|
+
* @param {string} [data.message]
|
|
125
|
+
* @param {string} [data.type]
|
|
126
|
+
* @param {string} [data.url]
|
|
127
|
+
* @returns {void}
|
|
128
|
+
*/
|
|
129
|
+
handleError(data) {
|
|
130
|
+
let backtrace;
|
|
131
|
+
if (data.error && data.error.stack) {
|
|
132
|
+
backtrace = data.error.stack.split("\n");
|
|
133
|
+
backtrace.shift();
|
|
134
|
+
backtrace = backtrace.join("\n");
|
|
135
|
+
}
|
|
136
|
+
else if (data.file) {
|
|
137
|
+
backtrace = `${data.file}:${data.line}`;
|
|
138
|
+
}
|
|
139
|
+
data.backtrace = backtrace;
|
|
140
|
+
this.communicator.sendCommand(data);
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* @returns {void}
|
|
144
|
+
*/
|
|
145
|
+
connectWebSocket() {
|
|
146
|
+
this.ws = new WebSocket("ws://localhost:1985");
|
|
147
|
+
this.communicator.ws = this.ws;
|
|
148
|
+
this.ws.addEventListener("error", digg(this, "communicator", "onError"));
|
|
149
|
+
this.ws.addEventListener("open", digg(this, "communicator", "onOpen"));
|
|
150
|
+
this.ws.addEventListener("message", (event) => this.communicator.onMessage(event.data));
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* @returns {void}
|
|
154
|
+
*/
|
|
155
|
+
enableOnBrowser() {
|
|
156
|
+
this._enabled = true;
|
|
157
|
+
this.connectWebSocket();
|
|
158
|
+
this.connectOnError();
|
|
159
|
+
this.connectUnhandledRejection();
|
|
160
|
+
this.overrideConsoleLog();
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* @returns {boolean}
|
|
164
|
+
*/
|
|
165
|
+
getEnabled() { return this._enabled; }
|
|
166
|
+
/**
|
|
167
|
+
* @returns {EventEmitter}
|
|
168
|
+
*/
|
|
169
|
+
getEvents() { return this.events; }
|
|
170
|
+
/**
|
|
171
|
+
* @param {any} arg
|
|
172
|
+
* @param {any[]} [scannedObjects]
|
|
173
|
+
* @returns {any}
|
|
174
|
+
*/
|
|
175
|
+
consoleLogMessage(arg, scannedObjects = []) {
|
|
176
|
+
if (Array.isArray(arg)) {
|
|
177
|
+
if (scannedObjects.includes(arg)) {
|
|
178
|
+
return "[recursive]";
|
|
179
|
+
}
|
|
180
|
+
else {
|
|
181
|
+
scannedObjects.push(arg);
|
|
182
|
+
}
|
|
183
|
+
const result = [];
|
|
184
|
+
for (const value of arg) {
|
|
185
|
+
result.push(this.consoleLogMessage(value, scannedObjects));
|
|
186
|
+
}
|
|
187
|
+
return result;
|
|
188
|
+
}
|
|
189
|
+
else if (Object.prototype.toString.call(arg) === '[object Object]') {
|
|
190
|
+
if (scannedObjects.includes(arg)) {
|
|
191
|
+
return "[recursive]";
|
|
192
|
+
}
|
|
193
|
+
else {
|
|
194
|
+
scannedObjects.push(arg);
|
|
195
|
+
}
|
|
196
|
+
/** @type {Record<string, any>} */
|
|
197
|
+
const result = {};
|
|
198
|
+
for (const key in arg) {
|
|
199
|
+
result[key] = this.consoleLogMessage(arg[key], scannedObjects);
|
|
200
|
+
}
|
|
201
|
+
return result;
|
|
202
|
+
}
|
|
203
|
+
else if (typeof arg == "object") {
|
|
204
|
+
return `[object ${arg?.constructor?.name}]`;
|
|
205
|
+
}
|
|
206
|
+
else {
|
|
207
|
+
return arg;
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* @param {function() : void} callback
|
|
212
|
+
* @returns {void}
|
|
213
|
+
*/
|
|
214
|
+
onInitialize(callback) {
|
|
215
|
+
this._onInitializeCallback = callback;
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* @returns {void}
|
|
219
|
+
*/
|
|
220
|
+
overrideConsoleLog() {
|
|
221
|
+
if (this.originalConsoleError || this.originalConsoleLog) {
|
|
222
|
+
throw new Error("Console methods has already been overridden!");
|
|
223
|
+
}
|
|
224
|
+
this.originalConsoleError = console.error;
|
|
225
|
+
this.originalConsoleLog = console.log;
|
|
226
|
+
console.error = this.fakeConsoleError;
|
|
227
|
+
console.log = this.fakeConsoleLog;
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* @param {string} sql
|
|
231
|
+
* @returns {Promise<Array<Record<string, any>>>}
|
|
232
|
+
*/
|
|
233
|
+
async sendQuery(sql) {
|
|
234
|
+
// @ts-expect-error
|
|
235
|
+
return await this.communicator.sendCommand({ type: "query", sql });
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-browser-helper.js","sourceRoot":"","sources":["../src/system-test-browser-helper.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Client from \"scoundrel-remote-eval/build/client/index.js\"\nimport ClientWebSocket from \"scoundrel-remote-eval/build/client/connections/web-socket/index.js\"\nimport {digg} from \"diggerize\"\nimport {EventEmitter} from \"eventemitter3\"\n\nimport SystemTestCommunicator from \"./system-test-communicator.js\"\n\n/** @type {{systemTestBrowserHelper: SystemTestBrowserHelper | null}} */\nconst shared = {\n  systemTestBrowserHelper: null\n}\n\nexport default class SystemTestBrowserHelper {\n  static current() {\n    if (!shared.systemTestBrowserHelper) {\n      throw new Error(\"No current SystemTestBrowserHelper set\")\n    }\n\n    return shared.systemTestBrowserHelper\n  }\n\n  constructor() {\n    this.communicator = new SystemTestCommunicator({parent: this, onCommand: this.onCommand})\n    this._enabled = false\n    this.events = new EventEmitter()\n\n    shared.systemTestBrowserHelper = this\n\n    this.startScoundrel()\n  }\n\n  async startScoundrel() {\n    this.scoundrelWs = new WebSocket(\"http://localhost:8090\")\n\n    // @ts-expect-error\n    this.scoundrelClientWebSocket = new ClientWebSocket(this.scoundrelWs)\n\n    await this.scoundrelClientWebSocket.waitForOpened()\n\n    this.scoundrelClient = new Client(this.scoundrelClientWebSocket)\n    this.events.emit(\"scoundrelStarted\")\n  }\n\n  waitForScoundrelStarted() {\n    return new Promise((resolve) => {\n      if (this.scoundrelClient) {\n        resolve(undefined)\n      } else {\n        this.events.once(\"scoundrelStarted\", () => {\n          resolve(undefined)\n        })\n      }\n    })\n  }\n\n  getScoundrel() {\n    if (!this.scoundrelClient) {\n      throw new Error(\"Scoundrel client is not started yet\")\n    }\n\n    return this.scoundrelClient\n  }\n\n  connectOnError() {\n    window.addEventListener(\"error\", (event) => {\n      this.handleError({\n        type: \"error\",\n        error: event.error,\n        errorClass: event.error?.name,\n        file: event.filename,\n        line: event.lineno,\n        message: event.message || \"Unknown error\",\n        url: window.location.href\n      })\n    })\n  }\n\n  connectUnhandledRejection() {\n    window.addEventListener(\"unhandledrejection\", (event) => {\n      this.handleError({\n        type: \"unhandledrejection\",\n        error: event.reason,\n        errorClass: \"UnhandledRejection\",\n        message: event.reason.message || event.reason || \"Unhandled promise rejection without a message\",\n        url: window.location.href\n      })\n    })\n  }\n\n  /**\n   * @param {object} data\n   * @param {string} [data.backtrace]\n   * @param {Error} [data.error]\n   * @param {string} [data.errorClass]\n   * @param {string} [data.file]\n   * @param {number} [data.line]\n   * @param {string} [data.message]\n   * @param {string} [data.type]\n   * @param {string} [data.url]\n   * @returns {void}\n   */\n  handleError(data) {\n    let backtrace\n\n    if (data.error && data.error.stack) {\n      backtrace = data.error.stack.split(\"\\n\")\n      backtrace.shift()\n      backtrace = backtrace.join(\"\\n\")\n    } else if (data.file) {\n      backtrace = `${data.file}:${data.line}`\n    }\n\n    data.backtrace = backtrace\n\n    this.communicator.sendCommand(data)\n  }\n\n  /**\n   * @returns {void}\n   */\n  connectWebSocket() {\n    this.ws = new WebSocket(\"ws://localhost:1985\")\n    this.communicator.ws = this.ws\n    this.ws.addEventListener(\"error\", digg(this, \"communicator\", \"onError\"))\n    this.ws.addEventListener(\"open\", digg(this, \"communicator\", \"onOpen\"))\n    this.ws.addEventListener(\"message\", (event) => this.communicator.onMessage(event.data))\n  }\n\n  /**\n   * @returns {void}\n   */\n  enableOnBrowser() {\n    this._enabled = true\n    this.connectWebSocket()\n    this.connectOnError()\n    this.connectUnhandledRejection()\n    this.overrideConsoleLog()\n  }\n\n  /**\n   * @returns {boolean}\n   */\n  getEnabled() { return this._enabled }\n\n  /**\n   * @returns {EventEmitter}\n   */\n  getEvents() { return this.events }\n\n  /**\n   * @param {any[]} args\n   * @returns {void}\n   */\n  fakeConsoleError = (...args) => {\n    this.communicator.sendCommand({type: \"console.error\", value: this.consoleLogMessage(args)})\n\n    if (this.originalConsoleError) {\n      return this.originalConsoleError(...args)\n    }\n  }\n\n  /**\n   * @param {any[]} args\n   * @returns {void}\n   */\n  fakeConsoleLog = (...args) => {\n    this.communicator.sendCommand({type: \"console.log\", value: this.consoleLogMessage(args)})\n\n    if (this.originalConsoleLog) {\n      return this.originalConsoleLog(...args)\n    }\n  }\n\n  /**\n   * @param {any} arg\n   * @param {any[]} [scannedObjects]\n   * @returns {any}\n   */\n  consoleLogMessage(arg, scannedObjects = []) {\n    if (Array.isArray(arg)) {\n      if (scannedObjects.includes(arg)) {\n        return \"[recursive]\"\n      } else {\n        scannedObjects.push(arg)\n      }\n\n      const result = []\n\n      for (const value of arg) {\n        result.push(this.consoleLogMessage(value, scannedObjects))\n      }\n\n      return result\n    } else if (Object.prototype.toString.call(arg) === '[object Object]') {\n      if (scannedObjects.includes(arg)) {\n        return \"[recursive]\"\n      } else {\n        scannedObjects.push(arg)\n      }\n\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key in arg) {\n        result[key] = this.consoleLogMessage(arg[key], scannedObjects)\n      }\n\n      return result\n    } else if (typeof arg == \"object\") {\n      return `[object ${arg?.constructor?.name}]`\n    } else {\n      return arg\n    }\n  }\n\n  /**\n   * @param {{data: {path: string, type: string}}} args\n   * @returns {Promise<{result: string} | void>}\n   */\n  onCommand = async ({data}) => {\n    if (data.type == \"initialize\") {\n      this.events.emit(\"initialize\")\n\n      if (this._onInitializeCallback) {\n        await this._onInitializeCallback()\n      }\n\n      return {result: \"initialized\"}\n    } else if (data.type == \"visit\") {\n      this.events.emit(\"navigate\", {path: data.path})\n    } else if (data.type == \"dismissTo\") {\n      this.events.emit(\"dismissTo\", {path: data.path})\n    } else {\n      throw new Error(`Unknown command type for SystemTestBrowserHelper: ${data.type}`)\n    }\n  }\n\n  /**\n   * @param {function() : void} callback\n   * @returns {void}\n   */\n  onInitialize(callback) {\n    this._onInitializeCallback = callback\n  }\n\n  /**\n   * @returns {void}\n   */\n  overrideConsoleLog() {\n    if (this.originalConsoleError || this.originalConsoleLog) {\n      throw new Error(\"Console methods has already been overridden!\")\n    }\n\n    this.originalConsoleError = console.error\n    this.originalConsoleLog = console.log\n\n    console.error = this.fakeConsoleError\n    console.log = this.fakeConsoleLog\n  }\n\n  /**\n   * @param {string} sql\n   * @returns {Promise<Array<Record<string, any>>>}\n   */\n  async sendQuery(sql) {\n    // @ts-expect-error\n    return await this.communicator.sendCommand({type: \"query\", sql})\n  }\n}\n"]}
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
export default class SystemTestCommunicator {
|
|
3
|
+
/**
|
|
4
|
+
* @param {object} args
|
|
5
|
+
* @param {(args: Record<string, any>) => Promise<{result: string} | void>} args.onCommand
|
|
6
|
+
* @param {object} [args.parent]
|
|
7
|
+
*/
|
|
8
|
+
constructor({ onCommand, parent }) {
|
|
9
|
+
/** @type {Record<string, {resolve: (data: any) => void, reject: (data: any) => void}>} */
|
|
10
|
+
this._responses = {};
|
|
11
|
+
/** @type {Record<string, any>} */
|
|
12
|
+
this._sendQueue = [];
|
|
13
|
+
this._sendQueueCount = 0;
|
|
14
|
+
/** @type {WebSocket | null} */
|
|
15
|
+
this.ws = null;
|
|
16
|
+
/** @param {Error} error */
|
|
17
|
+
this.onError = (error) => {
|
|
18
|
+
console.error("onWebSocketClientError", error);
|
|
19
|
+
};
|
|
20
|
+
/** @param {string} rawData */
|
|
21
|
+
this.onMessage = async (rawData) => {
|
|
22
|
+
/** @type {{data: any, id: number, type: string, isTrusted?: boolean}} */
|
|
23
|
+
const data = JSON.parse(rawData);
|
|
24
|
+
if (data.isTrusted) {
|
|
25
|
+
// Ignore
|
|
26
|
+
}
|
|
27
|
+
else if (data.type == "command") {
|
|
28
|
+
try {
|
|
29
|
+
const result = await this.onCommand({ data: data.data });
|
|
30
|
+
this.respond(data.id, { result });
|
|
31
|
+
}
|
|
32
|
+
catch (error) {
|
|
33
|
+
if (error instanceof Error) {
|
|
34
|
+
this.respond(data.id, { error: error.message });
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
this.respond(data.id, { error: error });
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
else if (data.type == "response") {
|
|
42
|
+
const response = this._responses[data.id];
|
|
43
|
+
if (!response) {
|
|
44
|
+
throw new Error(`No such response: ${data.id}`);
|
|
45
|
+
}
|
|
46
|
+
delete this._responses[data.id];
|
|
47
|
+
if (data.data.error) {
|
|
48
|
+
response.reject(data.data.error);
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
response.resolve(data.data.result);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
throw new Error(`Unknown type for SystemTestCommunicator: ${data.type}: ${JSON.stringify(data)}`);
|
|
56
|
+
}
|
|
57
|
+
};
|
|
58
|
+
this.onOpen = () => {
|
|
59
|
+
this.flushSendQueue();
|
|
60
|
+
};
|
|
61
|
+
this.onCommand = onCommand;
|
|
62
|
+
this.parent = parent;
|
|
63
|
+
}
|
|
64
|
+
flushSendQueue() {
|
|
65
|
+
while (this._sendQueue.length !== 0) {
|
|
66
|
+
const data = this._sendQueue.shift();
|
|
67
|
+
if (!this.ws || this.ws.readyState !== 1) {
|
|
68
|
+
throw new Error("WebSocket is not open");
|
|
69
|
+
}
|
|
70
|
+
this.ws.send(JSON.stringify(data));
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* @param {Record<string, any>} data
|
|
75
|
+
* @returns {void}
|
|
76
|
+
*/
|
|
77
|
+
send(data) {
|
|
78
|
+
this._sendQueue.push(data);
|
|
79
|
+
if (this.ws.readyState == 1) {
|
|
80
|
+
this.flushSendQueue();
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Sends a command and returns a promise that resolves with the response.
|
|
85
|
+
* @param {Record<string, any>} data - The command data to send.
|
|
86
|
+
* @returns {Promise<void>} A promise that resolves with the response data.
|
|
87
|
+
*/
|
|
88
|
+
sendCommand(data) {
|
|
89
|
+
return new Promise((resolve, reject) => {
|
|
90
|
+
const id = this._sendQueueCount;
|
|
91
|
+
this._sendQueueCount += 1;
|
|
92
|
+
this._responses[id] = { resolve, reject };
|
|
93
|
+
this.send({ type: "command", id, data });
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* @param {number} id
|
|
98
|
+
* @param {Record<string, any>} data
|
|
99
|
+
* @returns {void}
|
|
100
|
+
*/
|
|
101
|
+
respond(id, data) {
|
|
102
|
+
this.send({ type: "response", id, data });
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-communicator.js","sourceRoot":"","sources":["../src/system-test-communicator.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nexport default class SystemTestCommunicator {\n  /** @type {Record<string, {resolve: (data: any) => void, reject: (data: any) => void}>} */\n  _responses = {}\n\n  /** @type {Record<string, any>} */\n  _sendQueue = []\n\n  _sendQueueCount = 0\n\n  /** @type {WebSocket | null} */\n  ws = null\n\n  /**\n   * @param {object} args\n   * @param {(args: Record<string, any>) => Promise<{result: string} | void>} args.onCommand\n   * @param {object} [args.parent]\n   */\n  constructor({onCommand, parent}) {\n    this.onCommand = onCommand\n    this.parent = parent\n  }\n\n  flushSendQueue() {\n    while (this._sendQueue.length !== 0) {\n      const data = this._sendQueue.shift()\n\n      if (!this.ws || this.ws.readyState !== 1) {\n        throw new Error(\"WebSocket is not open\")\n      }\n\n      this.ws.send(JSON.stringify(data))\n    }\n  }\n\n  /** @param {Error} error */\n  onError = (error) => {\n    console.error(\"onWebSocketClientError\", error)\n  }\n\n  /** @param {string} rawData */\n  onMessage = async (rawData) => {\n    /** @type {{data: any, id: number, type: string, isTrusted?: boolean}} */\n    const data = JSON.parse(rawData)\n\n    if (data.isTrusted) {\n      // Ignore\n    } else if (data.type == \"command\") {\n      try {\n        const result = await this.onCommand({data: data.data})\n\n        this.respond(data.id, {result})\n      } catch (error) {\n        if (error instanceof Error) {\n          this.respond(data.id, {error: error.message})\n        } else {\n          this.respond(data.id, {error: error})\n        }\n      }\n    } else if (data.type == \"response\") {\n      const response = this._responses[data.id]\n\n      if (!response) {\n        throw new Error(`No such response: ${data.id}`)\n      }\n\n      delete this._responses[data.id]\n\n      if (data.data.error) {\n        response.reject(data.data.error)\n      } else {\n        response.resolve(data.data.result)\n      }\n    } else {\n      throw new Error(`Unknown type for SystemTestCommunicator: ${data.type}: ${JSON.stringify(data)}`)\n    }\n  }\n\n  onOpen = () => {\n    this.flushSendQueue()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  send(data) {\n    this._sendQueue.push(data)\n\n    if (this.ws.readyState == 1) {\n      this.flushSendQueue()\n    }\n  }\n\n  /**\n   * Sends a command and returns a promise that resolves with the response.\n   * @param {Record<string, any>} data - The command data to send.\n   * @returns {Promise<void>} A promise that resolves with the response data.\n   */\n  sendCommand(data) {\n    return new Promise((resolve, reject) => {\n      const id = this._sendQueueCount\n\n      this._sendQueueCount += 1\n      this._responses[id] = {resolve, reject}\n\n      this.send({type: \"command\", id, data})\n    })\n  }\n\n  /**\n   * @param {number} id\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  respond(id, data) {\n    this.send({type: \"response\", id, data})\n  }\n}\n"]}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import fs from "node:fs/promises";
|
|
3
|
+
import http from "node:http";
|
|
4
|
+
import mime from "mime";
|
|
5
|
+
import url from "url";
|
|
6
|
+
export default class SystemTestHttpServer {
|
|
7
|
+
constructor() {
|
|
8
|
+
/**
|
|
9
|
+
* @param {http.IncomingMessage} request
|
|
10
|
+
* @param {http.ServerResponse} response
|
|
11
|
+
* @returns {Promise<void>}
|
|
12
|
+
*/
|
|
13
|
+
this.onHttpServerRequest = async (request, response) => {
|
|
14
|
+
if (!request.url) {
|
|
15
|
+
response.statusCode = 400;
|
|
16
|
+
response.end("Bad Request");
|
|
17
|
+
return;
|
|
18
|
+
}
|
|
19
|
+
const parsedUrl = url.parse(request.url);
|
|
20
|
+
let filePath = `${process.cwd()}/dist${parsedUrl.pathname}`;
|
|
21
|
+
if (filePath.endsWith("/")) {
|
|
22
|
+
filePath += "index.html";
|
|
23
|
+
}
|
|
24
|
+
let fileExists;
|
|
25
|
+
try {
|
|
26
|
+
await fs.stat(filePath);
|
|
27
|
+
fileExists = true;
|
|
28
|
+
}
|
|
29
|
+
catch (_error) { // eslint-disable-line no-unused-vars
|
|
30
|
+
fileExists = false;
|
|
31
|
+
}
|
|
32
|
+
if (!fileExists) {
|
|
33
|
+
filePath = `${process.cwd()}/dist/index.html`;
|
|
34
|
+
}
|
|
35
|
+
const fileContent = await fs.readFile(filePath);
|
|
36
|
+
const mimeType = mime.getType(filePath);
|
|
37
|
+
response.statusCode = 200;
|
|
38
|
+
if (mimeType) {
|
|
39
|
+
response.setHeader("Content-Type", mimeType);
|
|
40
|
+
}
|
|
41
|
+
response.end(fileContent);
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
/** @returns {void} */
|
|
45
|
+
close() {
|
|
46
|
+
if (!this.httpServer) {
|
|
47
|
+
throw new Error("HTTP server is not initialized");
|
|
48
|
+
}
|
|
49
|
+
this.httpServer.close();
|
|
50
|
+
}
|
|
51
|
+
/** @returns {Promise<void>} */
|
|
52
|
+
async start() {
|
|
53
|
+
this.basePath = await fs.realpath(`${__dirname}/../..`);
|
|
54
|
+
await this.startHttpServer();
|
|
55
|
+
}
|
|
56
|
+
/** @returns {Promise<void>} */
|
|
57
|
+
startHttpServer() {
|
|
58
|
+
return new Promise((resolve) => {
|
|
59
|
+
this.httpServer = http.createServer(this.onHttpServerRequest);
|
|
60
|
+
this.httpServer.listen(1984, "localhost", () => {
|
|
61
|
+
resolve();
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3lzdGVtLXRlc3QtaHR0cC1zZXJ2ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvc3lzdGVtLXRlc3QtaHR0cC1zZXJ2ZXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsWUFBWTtBQUVaLE9BQU8sRUFBRSxNQUFNLGtCQUFrQixDQUFBO0FBQ2pDLE9BQU8sSUFBSSxNQUFNLFdBQVcsQ0FBQTtBQUM1QixPQUFPLElBQUksTUFBTSxNQUFNLENBQUE7QUFDdkIsT0FBTyxHQUFHLE1BQU0sS0FBSyxDQUFBO0FBRXJCLE1BQU0sQ0FBQyxPQUFPLE9BQU8sb0JBQW9CO0lBQXpDO1FBVUU7Ozs7V0FJRztRQUNILHdCQUFtQixHQUFHLEtBQUssRUFBRSxPQUFPLEVBQUUsUUFBUSxFQUFFLEVBQUU7WUFDaEQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQztnQkFDakIsUUFBUSxDQUFDLFVBQVUsR0FBRyxHQUFHLENBQUE7Z0JBQ3pCLFFBQVEsQ0FBQyxHQUFHLENBQUMsYUFBYSxDQUFDLENBQUE7Z0JBQzNCLE9BQU07WUFDUixDQUFDO1lBRUQsTUFBTSxTQUFTLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLENBQUE7WUFDeEMsSUFBSSxRQUFRLEdBQUcsR0FBRyxPQUFPLENBQUMsR0FBRyxFQUFFLFFBQVEsU0FBUyxDQUFDLFFBQVEsRUFBRSxDQUFBO1lBRTNELElBQUksUUFBUSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDO2dCQUMzQixRQUFRLElBQUksWUFBWSxDQUFBO1lBQzFCLENBQUM7WUFFRCxJQUFJLFVBQVUsQ0FBQTtZQUVkLElBQUksQ0FBQztnQkFDSCxNQUFNLEVBQUUsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUE7Z0JBQ3ZCLFVBQVUsR0FBRyxJQUFJLENBQUE7WUFDbkIsQ0FBQztZQUFDLE9BQU8sTUFBTSxFQUFFLENBQUMsQ0FBQyxxQ0FBcUM7Z0JBQ3RELFVBQVUsR0FBRyxLQUFLLENBQUE7WUFDcEIsQ0FBQztZQUVELElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztnQkFDaEIsUUFBUSxHQUFHLEdBQUcsT0FBTyxDQUFDLEdBQUcsRUFBRSxrQkFBa0IsQ0FBQTtZQUMvQyxDQUFDO1lBRUQsTUFBTSxXQUFXLEdBQUcsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxDQUFBO1lBQy9DLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLENBQUE7WUFFdkMsUUFBUSxDQUFDLFVBQVUsR0FBRyxHQUFHLENBQUE7WUFFekIsSUFBSSxRQUFRLEVBQUUsQ0FBQztnQkFDYixRQUFRLENBQUMsU0FBUyxDQUFDLGNBQWMsRUFBRSxRQUFRLENBQUMsQ0FBQTtZQUM5QyxDQUFDO1lBRUQsUUFBUSxDQUFDLEdBQUcsQ0FBQyxXQUFXLENBQUMsQ0FBQTtRQUMzQixDQUFDLENBQUE7SUFpQkgsQ0FBQztJQXBFQyxzQkFBc0I7SUFDdEIsS0FBSztRQUNILElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDckIsTUFBTSxJQUFJLEtBQUssQ0FBQyxnQ0FBZ0MsQ0FBQyxDQUFBO1FBQ25ELENBQUM7UUFFRCxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxDQUFBO0lBQ3pCLENBQUM7SUE4Q0QsK0JBQStCO0lBQy9CLEtBQUssQ0FBQyxLQUFLO1FBQ1QsSUFBSSxDQUFDLFFBQVEsR0FBRyxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsR0FBRyxTQUFTLFFBQVEsQ0FBQyxDQUFBO1FBQ3ZELE1BQU0sSUFBSSxDQUFDLGVBQWUsRUFBRSxDQUFBO0lBQzlCLENBQUM7SUFFRCwrQkFBK0I7SUFDL0IsZUFBZTtRQUNiLE9BQU8sSUFBSSxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsRUFBRTtZQUM3QixJQUFJLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLG1CQUFtQixDQUFDLENBQUE7WUFDN0QsSUFBSSxDQUFDLFVBQVUsQ0FBQyxNQUFNLENBQUMsSUFBSSxFQUFFLFdBQVcsRUFBRSxHQUFHLEVBQUU7Z0JBQzdDLE9BQU8sRUFBRSxDQUFBO1lBQ1gsQ0FBQyxDQUFDLENBQUE7UUFDSixDQUFDLENBQUMsQ0FBQTtJQUNKLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8vIEB0cy1jaGVja1xuXG5pbXBvcnQgZnMgZnJvbSBcIm5vZGU6ZnMvcHJvbWlzZXNcIlxuaW1wb3J0IGh0dHAgZnJvbSBcIm5vZGU6aHR0cFwiXG5pbXBvcnQgbWltZSBmcm9tIFwibWltZVwiXG5pbXBvcnQgdXJsIGZyb20gXCJ1cmxcIlxuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBTeXN0ZW1UZXN0SHR0cFNlcnZlciB7XG4gIC8qKiBAcmV0dXJucyB7dm9pZH0gKi9cbiAgY2xvc2UoKSB7XG4gICAgaWYgKCF0aGlzLmh0dHBTZXJ2ZXIpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcIkhUVFAgc2VydmVyIGlzIG5vdCBpbml0aWFsaXplZFwiKVxuICAgIH1cblxuICAgIHRoaXMuaHR0cFNlcnZlci5jbG9zZSgpXG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIHtodHRwLkluY29taW5nTWVzc2FnZX0gcmVxdWVzdFxuICAgKiBAcGFyYW0ge2h0dHAuU2VydmVyUmVzcG9uc2V9IHJlc3BvbnNlXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fVxuICAgKi9cbiAgb25IdHRwU2VydmVyUmVxdWVzdCA9IGFzeW5jIChyZXF1ZXN0LCByZXNwb25zZSkgPT4ge1xuICAgIGlmICghcmVxdWVzdC51cmwpIHtcbiAgICAgIHJlc3BvbnNlLnN0YXR1c0NvZGUgPSA0MDBcbiAgICAgIHJlc3BvbnNlLmVuZChcIkJhZCBSZXF1ZXN0XCIpXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBjb25zdCBwYXJzZWRVcmwgPSB1cmwucGFyc2UocmVxdWVzdC51cmwpXG4gICAgbGV0IGZpbGVQYXRoID0gYCR7cHJvY2Vzcy5jd2QoKX0vZGlzdCR7cGFyc2VkVXJsLnBhdGhuYW1lfWBcblxuICAgIGlmIChmaWxlUGF0aC5lbmRzV2l0aChcIi9cIikpIHtcbiAgICAgIGZpbGVQYXRoICs9IFwiaW5kZXguaHRtbFwiXG4gICAgfVxuXG4gICAgbGV0IGZpbGVFeGlzdHNcblxuICAgIHRyeSB7XG4gICAgICBhd2FpdCBmcy5zdGF0KGZpbGVQYXRoKVxuICAgICAgZmlsZUV4aXN0cyA9IHRydWVcbiAgICB9IGNhdGNoIChfZXJyb3IpIHsgLy8gZXNsaW50LWRpc2FibGUtbGluZSBuby11bnVzZWQtdmFyc1xuICAgICAgZmlsZUV4aXN0cyA9IGZhbHNlXG4gICAgfVxuXG4gICAgaWYgKCFmaWxlRXhpc3RzKSB7XG4gICAgICBmaWxlUGF0aCA9IGAke3Byb2Nlc3MuY3dkKCl9L2Rpc3QvaW5kZXguaHRtbGBcbiAgICB9XG5cbiAgICBjb25zdCBmaWxlQ29udGVudCA9IGF3YWl0IGZzLnJlYWRGaWxlKGZpbGVQYXRoKVxuICAgIGNvbnN0IG1pbWVUeXBlID0gbWltZS5nZXRUeXBlKGZpbGVQYXRoKVxuXG4gICAgcmVzcG9uc2Uuc3RhdHVzQ29kZSA9IDIwMFxuXG4gICAgaWYgKG1pbWVUeXBlKSB7XG4gICAgICByZXNwb25zZS5zZXRIZWFkZXIoXCJDb250ZW50LVR5cGVcIiwgbWltZVR5cGUpXG4gICAgfVxuXG4gICAgcmVzcG9uc2UuZW5kKGZpbGVDb250ZW50KVxuICB9XG5cbiAgLyoqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAqL1xuICBhc3luYyBzdGFydCgpIHtcbiAgICB0aGlzLmJhc2VQYXRoID0gYXdhaXQgZnMucmVhbHBhdGgoYCR7X19kaXJuYW1lfS8uLi8uLmApXG4gICAgYXdhaXQgdGhpcy5zdGFydEh0dHBTZXJ2ZXIoKVxuICB9XG5cbiAgLyoqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAqL1xuICBzdGFydEh0dHBTZXJ2ZXIoKSB7XG4gICAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiB7XG4gICAgICB0aGlzLmh0dHBTZXJ2ZXIgPSBodHRwLmNyZWF0ZVNlcnZlcih0aGlzLm9uSHR0cFNlcnZlclJlcXVlc3QpXG4gICAgICB0aGlzLmh0dHBTZXJ2ZXIubGlzdGVuKDE5ODQsIFwibG9jYWxob3N0XCIsICgpID0+IHtcbiAgICAgICAgcmVzb2x2ZSgpXG4gICAgICB9KVxuICAgIH0pXG4gIH1cbn1cbiJdfQ==
|
|
@@ -34,6 +34,7 @@ export default class SystemTest {
|
|
|
34
34
|
_started: boolean;
|
|
35
35
|
_driverTimeouts: number;
|
|
36
36
|
_timeouts: number;
|
|
37
|
+
/** @returns {SystemTestCommunicator} */
|
|
37
38
|
getCommunicator(): SystemTestCommunicator;
|
|
38
39
|
_host: string;
|
|
39
40
|
_port: number;
|
|
@@ -45,9 +46,7 @@ export default class SystemTest {
|
|
|
45
46
|
* @returns {string | undefined}
|
|
46
47
|
*/
|
|
47
48
|
getBaseSelector(): string | undefined;
|
|
48
|
-
/**
|
|
49
|
-
* @returns {import("selenium-webdriver").WebDriver}
|
|
50
|
-
*/
|
|
49
|
+
/** @returns {import("selenium-webdriver").WebDriver} */
|
|
51
50
|
getDriver(): import("selenium-webdriver").WebDriver;
|
|
52
51
|
/**
|
|
53
52
|
* Sets the base selector for scoping element searches
|
|
@@ -76,7 +75,7 @@ export default class SystemTest {
|
|
|
76
75
|
/**
|
|
77
76
|
* Finds all elements by CSS selector
|
|
78
77
|
* @param {string} selector
|
|
79
|
-
* @param {object} args
|
|
78
|
+
* @param {object} [args]
|
|
80
79
|
* @param {number} [args.timeout]
|
|
81
80
|
* @param {boolean} [args.visible]
|
|
82
81
|
* @param {boolean} [args.useBaseSelector]
|
|
@@ -96,17 +95,17 @@ export default class SystemTest {
|
|
|
96
95
|
/**
|
|
97
96
|
* Finds a single element by CSS selector
|
|
98
97
|
* @param {string} selector
|
|
99
|
-
* @param {object} args
|
|
98
|
+
* @param {object} [args]
|
|
100
99
|
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
101
100
|
*/
|
|
102
101
|
find(selector: string, args?: object): Promise<import("selenium-webdriver").WebElement>;
|
|
103
102
|
/**
|
|
104
103
|
* Finds a single element by test ID
|
|
105
104
|
* @param {string} testID
|
|
106
|
-
* @param {object} args
|
|
105
|
+
* @param {object} [args]
|
|
107
106
|
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
108
107
|
*/
|
|
109
|
-
findByTestID(testID: string, args
|
|
108
|
+
findByTestID(testID: string, args?: object): Promise<import("selenium-webdriver").WebElement>;
|
|
110
109
|
/**
|
|
111
110
|
* @param {string|import("selenium-webdriver").WebElement} elementOrIdentifier
|
|
112
111
|
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
@@ -124,20 +123,16 @@ export default class SystemTest {
|
|
|
124
123
|
* @returns {Promise<string[]>}
|
|
125
124
|
*/
|
|
126
125
|
getBrowserLogs(): Promise<string[]>;
|
|
127
|
-
/**
|
|
128
|
-
* @returns {Promise<string>}
|
|
129
|
-
*/
|
|
126
|
+
/** @returns {Promise<string>} */
|
|
130
127
|
getCurrentUrl(): Promise<string>;
|
|
131
|
-
/**
|
|
132
|
-
* @returns {number}
|
|
133
|
-
*/
|
|
128
|
+
/** @returns {number} */
|
|
134
129
|
getTimeouts(): number;
|
|
135
130
|
/**
|
|
136
131
|
* Interacts with an element by calling a method on it with the given arguments.
|
|
137
132
|
* Retrying on ElementNotInteractableError.
|
|
138
|
-
* @param {import("selenium-webdriver").WebElement|string} elementOrIdentifier
|
|
139
|
-
* @param {string} methodName
|
|
140
|
-
* @param {...any} args
|
|
133
|
+
* @param {import("selenium-webdriver").WebElement|string} elementOrIdentifier The element or a CSS selector to find the element.
|
|
134
|
+
* @param {string} methodName The method name to call on the element.
|
|
135
|
+
* @param {...any} args Arguments to pass to the method.
|
|
141
136
|
* @returns {Promise<any>}
|
|
142
137
|
*/
|
|
143
138
|
interact(elementOrIdentifier: import("selenium-webdriver").WebElement | string, methodName: string, ...args: any[]): Promise<any>;
|
|
@@ -149,11 +144,11 @@ export default class SystemTest {
|
|
|
149
144
|
expectNoElement(selector: string): Promise<void>;
|
|
150
145
|
/**
|
|
151
146
|
* @param {string} selector
|
|
152
|
-
* @param {object} args
|
|
147
|
+
* @param {object} [args]
|
|
153
148
|
* @param {boolean} [args.useBaseSelector]
|
|
154
149
|
* @returns {Promise<void>}
|
|
155
150
|
*/
|
|
156
|
-
waitForNoSelector(selector: string, args
|
|
151
|
+
waitForNoSelector(selector: string, args?: {
|
|
157
152
|
useBaseSelector?: boolean;
|
|
158
153
|
}): Promise<void>;
|
|
159
154
|
/**
|
|
@@ -167,9 +162,7 @@ export default class SystemTest {
|
|
|
167
162
|
* @returns {Promise<void>}
|
|
168
163
|
*/
|
|
169
164
|
expectNotificationMessage(expectedNotificationMessage: string): Promise<void>;
|
|
170
|
-
/**
|
|
171
|
-
* @returns {Promise<void>}
|
|
172
|
-
*/
|
|
165
|
+
/** @returns {Promise<void>} */
|
|
173
166
|
dismissNotificationMessages(): Promise<void>;
|
|
174
167
|
/**
|
|
175
168
|
* Indicates whether the system test has been started
|
|
@@ -249,9 +242,7 @@ export default class SystemTest {
|
|
|
249
242
|
*/
|
|
250
243
|
onWebSocketConnection: (ws: WebSocket) => Promise<void>;
|
|
251
244
|
ws: WebSocket;
|
|
252
|
-
/**
|
|
253
|
-
* @returns {void}
|
|
254
|
-
*/
|
|
245
|
+
/** @returns {void} */
|
|
255
246
|
onWebSocketClose: () => void;
|
|
256
247
|
/**
|
|
257
248
|
* Handles an error reported from the browser
|