system-testing 1.0.34 → 1.0.36
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 +5 -3
- 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 +19 -26
- package/build/system-test.js +696 -0
- package/build/use-system-test.js +79 -0
- package/package.json +6 -6
- package/dist/index.js +0 -3
- package/dist/index.js.map +0 -1
- package/dist/system-test-browser-helper.js +0 -237
- 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
|
|
@@ -5,10 +5,10 @@ export default class SystemTestBrowserHelper {
|
|
|
5
5
|
events: EventEmitter<any>;
|
|
6
6
|
startScoundrel(): Promise<void>;
|
|
7
7
|
scoundrelWs: WebSocket;
|
|
8
|
-
scoundrelClientWebSocket:
|
|
9
|
-
scoundrelClient:
|
|
8
|
+
scoundrelClientWebSocket: ClientWebSocket;
|
|
9
|
+
scoundrelClient: Client;
|
|
10
10
|
waitForScoundrelStarted(): Promise<any>;
|
|
11
|
-
getScoundrel():
|
|
11
|
+
getScoundrel(): Client;
|
|
12
12
|
connectOnError(): void;
|
|
13
13
|
connectUnhandledRejection(): void;
|
|
14
14
|
/**
|
|
@@ -104,3 +104,5 @@ export default class SystemTestBrowserHelper {
|
|
|
104
104
|
}
|
|
105
105
|
import SystemTestCommunicator from "./system-test-communicator.js";
|
|
106
106
|
import EventEmitter from "events";
|
|
107
|
+
import ClientWebSocket from "scoundrel-remote-eval/build/client/connections/web-socket/index.js";
|
|
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 "events";
|
|
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 \"events\"\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
|
|
@@ -67,8 +66,8 @@ export default class SystemTest {
|
|
|
67
66
|
*/
|
|
68
67
|
startScoundrel(): void;
|
|
69
68
|
wss: import("ws").Server<typeof import("ws").default, typeof import("node:http").IncomingMessage>;
|
|
70
|
-
serverWebSocket:
|
|
71
|
-
server:
|
|
69
|
+
serverWebSocket: ServerWebSocket;
|
|
70
|
+
server: Server;
|
|
72
71
|
/**
|
|
73
72
|
* @returns {void}
|
|
74
73
|
*/
|
|
@@ -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
|
|
@@ -294,4 +285,6 @@ export default class SystemTest {
|
|
|
294
285
|
dismissTo(path: string): Promise<void>;
|
|
295
286
|
}
|
|
296
287
|
import SystemTestCommunicator from "./system-test-communicator.js";
|
|
288
|
+
import ServerWebSocket from "scoundrel-remote-eval/build/server/connections/web-socket/index.js";
|
|
289
|
+
import Server from "scoundrel-remote-eval/build/server/index.js";
|
|
297
290
|
import SystemTestHttpServer from "./system-test-http-server.js";
|