@appium/driver-test-support 0.2.0 → 0.3.0

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.
@@ -1,389 +1,417 @@
1
1
  "use strict";
2
-
3
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
- Object.defineProperty(exports, "__esModule", {
6
- value: true
7
- });
8
- exports.createSessionHelpers = createSessionHelpers;
9
- exports.driverE2ETestSuite = driverE2ETestSuite;
10
-
11
- require("source-map-support/register");
12
-
13
- var _lodash = _interopRequireDefault(require("lodash"));
14
-
15
- var _driver = require("appium/driver");
16
-
17
- var _axios = _interopRequireDefault(require("axios"));
18
-
19
- var _bluebird = _interopRequireDefault(require("bluebird"));
20
-
21
- var _helpers = require("./helpers");
22
-
23
- var _chai = _interopRequireDefault(require("chai"));
24
-
25
- var _sinon = _interopRequireDefault(require("sinon"));
26
-
27
- const should = _chai.default.should();
28
-
29
- function createSessionHelpers(port, address = _helpers.TEST_HOST) {
30
- const createAppiumTestURL = (0, _helpers.createAppiumURL)(address, port);
31
- const createSessionURL = createAppiumTestURL(_lodash.default, '');
32
- const newSessionURL = createAppiumTestURL('', 'session');
33
- return {
34
- newSessionURL,
35
- createAppiumTestURL,
36
- postCommand: async (sessionId, cmdName, data = {}, config = {}) => {
37
- var _response$data;
38
-
39
- const url = createAppiumTestURL(sessionId, cmdName);
40
- const response = await _axios.default.post(url, data, config);
41
- return (_response$data = response.data) === null || _response$data === void 0 ? void 0 : _response$data.value;
42
- },
43
- getCommand: async (sessionIdOrCmdName, cmdNameOrConfig, config = {}) => {
44
- var _response$data2;
45
-
46
- if (!_lodash.default.isString(cmdNameOrConfig)) {
47
- config = cmdNameOrConfig;
48
- cmdNameOrConfig = sessionIdOrCmdName;
49
- sessionIdOrCmdName = '';
50
- }
51
-
52
- const response = await (0, _axios.default)({
53
- url: createAppiumTestURL(sessionIdOrCmdName, cmdNameOrConfig),
54
- validateStatus: null,
55
- ...config
56
- });
57
- return (_response$data2 = response.data) === null || _response$data2 === void 0 ? void 0 : _response$data2.value;
58
- },
59
- startSession: async (data, config = {}) => {
60
- var _response$data3;
61
-
62
- data = _lodash.default.defaultsDeep(data, {
63
- capabilities: {
64
- alwaysMatch: {},
65
- firstMatch: [{}]
66
- }
67
- });
68
- const response = await _axios.default.post(newSessionURL, data, config);
69
- return (_response$data3 = response.data) === null || _response$data3 === void 0 ? void 0 : _response$data3.value;
70
- },
71
- endSession: async sessionId => await _axios.default.delete(createSessionURL(sessionId), {
72
- validateStatus: null
73
- }),
74
- getSession: async sessionId => {
75
- var _response$data4;
76
-
77
- const response = await (0, _axios.default)({
78
- url: createSessionURL(sessionId),
79
- validateStatus: null
80
- });
81
- return (_response$data4 = response.data) === null || _response$data4 === void 0 ? void 0 : _response$data4.value;
82
- }
83
- };
84
- }
85
-
86
- function driverE2ETestSuite(DriverClass, defaultCaps = {}) {
87
- let address = defaultCaps['appium:address'] ?? _helpers.TEST_HOST;
88
- let port = defaultCaps['appium:port'];
89
- const className = DriverClass.name || '(unknown driver)';
90
- describe(`BaseDriver E2E (as ${className})`, function () {
91
- let baseServer;
92
- let d;
93
- let newSessionURL;
94
- let startSession;
95
- let getSession;
96
- let endSession;
97
- let getCommand;
98
- let postCommand;
99
- before(async function () {
100
- port = port ?? (await (0, _helpers.getTestPort)());
101
- defaultCaps = { ...defaultCaps,
102
- 'appium:port': port
103
- };
104
- d = new DriverClass({
105
- port,
106
- address
107
- });
108
- baseServer = await (0, _driver.server)({
109
- routeConfiguringFunction: (0, _driver.routeConfiguringFunction)(d),
110
- port,
111
- hostname: address,
112
- cliArgs: {}
113
- });
114
- ({
115
- startSession,
116
- getSession,
117
- endSession,
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.driverE2ETestSuite = exports.createSessionHelpers = void 0;
7
+ const lodash_1 = __importDefault(require("lodash"));
8
+ const driver_1 = require("appium/driver");
9
+ const axios_1 = __importDefault(require("axios"));
10
+ const bluebird_1 = __importDefault(require("bluebird"));
11
+ const helpers_1 = require("./helpers");
12
+ const chai_1 = __importDefault(require("chai"));
13
+ const sinon_1 = __importDefault(require("sinon"));
14
+ const should = chai_1.default.should();
15
+ /**
16
+ * Creates some helper functions for E2E tests to manage sessions.
17
+ * @template [CommandData=unknown]
18
+ * @template [ResponseData=any]
19
+ * @param {number} port - Port on which the server is running. Typically this will be retrieved via `get-port` beforehand
20
+ * @param {string} [address] - Address/host on which the server is running. Defaults to {@linkcode TEST_HOST}
21
+ * @returns {SessionHelpers<CommandData, ResponseData>}
22
+ */
23
+ function createSessionHelpers(port, address = helpers_1.TEST_HOST) {
24
+ const createAppiumTestURL =
25
+ /** @type {import('lodash').CurriedFunction2<string,string,string>} */ ((0, helpers_1.createAppiumURL)(address, port));
26
+ const createSessionURL = createAppiumTestURL(lodash_1.default, '');
27
+ const newSessionURL = createAppiumTestURL('', 'session');
28
+ return /** @type {SessionHelpers<CommandData, ResponseData>} */ ({
118
29
  newSessionURL,
119
- getCommand,
120
- postCommand
121
- } = createSessionHelpers(port, address));
122
- });
123
- after(async function () {
124
- await baseServer.close();
125
- });
126
- describe('session handling', function () {
127
- it('should handle idempotency while creating sessions', async function () {
128
- const sessionIds = [];
129
- let times = 0;
130
-
131
- do {
132
- const {
133
- sessionId
134
- } = await startSession({
135
- capabilities: {
136
- alwaysMatch: defaultCaps
30
+ createAppiumTestURL,
31
+ /**
32
+ *
33
+ * @param {string} sessionId
34
+ * @param {string} cmdName
35
+ * @param {any} [data]
36
+ * @param {AxiosRequestConfig} [config]
37
+ * @returns {Promise<any>}
38
+ */
39
+ postCommand: async (sessionId, cmdName, data = {}, config = {}) => {
40
+ const url = createAppiumTestURL(sessionId, cmdName);
41
+ const response = await axios_1.default.post(url, data, config);
42
+ return response.data?.value;
43
+ },
44
+ /**
45
+ *
46
+ * @param {string} sessionIdOrCmdName
47
+ * @param {string|AxiosRequestConfig} cmdNameOrConfig
48
+ * @param {AxiosRequestConfig} [config]
49
+ * @returns {Promise<any>}
50
+ */
51
+ getCommand: async (sessionIdOrCmdName, cmdNameOrConfig, config = {}) => {
52
+ if (!lodash_1.default.isString(cmdNameOrConfig)) {
53
+ config = cmdNameOrConfig;
54
+ cmdNameOrConfig = sessionIdOrCmdName;
55
+ sessionIdOrCmdName = '';
137
56
  }
138
- }, {
139
- headers: {
140
- 'X-Idempotency-Key': '123456'
141
- },
142
- simple: false,
143
- resolveWithFullResponse: true
144
- });
145
- sessionIds.push(sessionId);
146
- times++;
147
- } while (times < 2);
148
-
149
- _lodash.default.uniq(sessionIds).length.should.equal(1);
150
-
151
- const {
152
- status,
153
- data
154
- } = await endSession(sessionIds[0]);
155
- status.should.equal(200);
156
- should.equal(data.value, null);
157
- });
158
- it('should handle idempotency while creating parallel sessions', async function () {
159
- const reqs = [];
160
- let times = 0;
161
-
162
- do {
163
- reqs.push(startSession({
164
- capabilities: {
165
- alwaysMatch: defaultCaps
166
- }
167
- }, {
168
- headers: {
169
- 'X-Idempotency-Key': '12345'
170
- }
171
- }));
172
- times++;
173
- } while (times < 2);
174
-
175
- const sessionIds = _lodash.default.map(await _bluebird.default.all(reqs), 'sessionId');
176
-
177
- _lodash.default.uniq(sessionIds).length.should.equal(1);
178
-
179
- const {
180
- status,
181
- data
182
- } = await endSession(sessionIds[0]);
183
- status.should.equal(200);
184
- should.equal(data.value, null);
185
- });
186
- it('should create session and retrieve a session id, then delete it', async function () {
187
- let {
188
- status,
189
- data
190
- } = await _axios.default.post(newSessionURL, {
191
- capabilities: {
192
- alwaysMatch: defaultCaps
193
- }
194
- });
195
- status.should.equal(200);
196
- should.exist(data.value.sessionId);
197
- data.value.capabilities.platformName.should.equal(defaultCaps.platformName);
198
- data.value.capabilities.deviceName.should.equal(defaultCaps['appium:deviceName']);
199
- ({
200
- status,
201
- data
202
- } = await endSession(d.sessionId));
203
- status.should.equal(200);
204
- should.equal(data.value, null);
205
- should.equal(d.sessionId, null);
206
- });
57
+ const response = await (0, axios_1.default)({
58
+ url: createAppiumTestURL(sessionIdOrCmdName, cmdNameOrConfig),
59
+ validateStatus: null,
60
+ ...config,
61
+ });
62
+ return response.data?.value;
63
+ },
64
+ /**
65
+ *
66
+ * @param {NewSessionData} data
67
+ * @param {AxiosRequestConfig} [config]
68
+ */
69
+ startSession: async (data, config = {}) => {
70
+ data = lodash_1.default.defaultsDeep(data, {
71
+ capabilities: {
72
+ alwaysMatch: {},
73
+ firstMatch: [{}],
74
+ },
75
+ });
76
+ const response = await axios_1.default.post(newSessionURL, data, config);
77
+ return response.data?.value;
78
+ },
79
+ /**
80
+ *
81
+ * @param {string} sessionId
82
+ */
83
+ endSession: async (sessionId) => await axios_1.default.delete(createSessionURL(sessionId), {
84
+ validateStatus: null,
85
+ }),
86
+ /**
87
+ * @param {string} sessionId
88
+ * @returns {Promise<any>}
89
+ */
90
+ getSession: async (sessionId) => {
91
+ const response = await (0, axios_1.default)({
92
+ url: createSessionURL(sessionId),
93
+ validateStatus: null,
94
+ });
95
+ return response.data?.value;
96
+ },
207
97
  });
208
- it.skip('should throw NYI for commands not implemented', async function () {});
209
- describe('command timeouts', function () {
210
- let originalFindElement, originalFindElements;
211
-
212
- async function startTimeoutSession(timeout) {
213
- const caps = _lodash.default.cloneDeep(defaultCaps);
214
-
215
- caps['appium:newCommandTimeout'] = timeout;
216
- return await startSession({
217
- capabilities: {
218
- alwaysMatch: caps
219
- }
220
- });
221
- }
222
-
223
- before(function () {
224
- originalFindElement = d.findElement;
225
-
226
- d.findElement = function () {
227
- return 'foo';
228
- }.bind(d);
229
-
230
- originalFindElements = d.findElements;
231
-
232
- d.findElements = async function () {
233
- await _bluebird.default.delay(200);
234
- return ['foo'];
235
- }.bind(d);
236
- });
237
- after(function () {
238
- d.findElement = originalFindElement;
239
- d.findElements = originalFindElements;
240
- });
241
- it('should set a default commandTimeout', async function () {
242
- let newSession = await startTimeoutSession();
243
- d.newCommandTimeoutMs.should.be.above(0);
244
- await endSession(newSession.sessionId);
245
- });
246
- it('should timeout on commands using commandTimeout cap', async function () {
247
- let newSession = await startTimeoutSession(0.25);
248
- const sessionId = d.sessionId;
249
- await postCommand(sessionId, 'element', {
250
- using: 'name',
251
- value: 'foo'
252
- });
253
- await _bluebird.default.delay(400);
254
- const value = await getSession(sessionId);
255
- should.equal(value.error, 'invalid session id');
256
- should.equal(d.sessionId, null);
257
- const resp = (await endSession(newSession.sessionId)).data.value;
258
- should.equal(resp === null || resp === void 0 ? void 0 : resp.error, 'invalid session id');
259
- });
260
- it('should not timeout with commandTimeout of false', async function () {
261
- let newSession = await startTimeoutSession(0.1);
262
- let start = Date.now();
263
- const value = await postCommand(d.sessionId, 'elements', {
264
- using: 'name',
265
- value: 'foo'
266
- });
267
- (Date.now() - start).should.be.above(150);
268
- value.should.eql(['foo']);
269
- await endSession(newSession.sessionId);
270
- });
271
- it('should not timeout with commandTimeout of 0', async function () {
272
- var _value$platformName;
273
-
274
- d.newCommandTimeoutMs = 2;
275
- let newSession = await startTimeoutSession(0);
276
- await postCommand(d.sessionId, 'element', {
277
- using: 'name',
278
- value: 'foo'
279
- });
280
- await _bluebird.default.delay(400);
281
- const value = await getSession(d.sessionId);
282
- (_value$platformName = value.platformName) === null || _value$platformName === void 0 ? void 0 : _value$platformName.should.equal(defaultCaps.platformName);
283
- const resp = (await endSession(newSession.sessionId)).data.value;
284
- should.equal(resp, null);
285
- d.newCommandTimeoutMs = 60 * 1000;
286
- });
287
- it('should not timeout if its just the command taking awhile', async function () {
288
- let newSession = await startTimeoutSession(0.25);
289
- const {
290
- sessionId
291
- } = d;
292
- await postCommand(d.sessionId, 'element', {
293
- using: 'name',
294
- value: 'foo'
98
+ }
99
+ exports.createSessionHelpers = createSessionHelpers;
100
+ /**
101
+ * Creates E2E test suites for a driver.
102
+ * @template {Driver} P
103
+ * @param {DriverClass<P>} DriverClass
104
+ * @param {Partial<BaseNSCapabilities>} [defaultCaps]
105
+ */
106
+ function driverE2ETestSuite(DriverClass, defaultCaps = {}) {
107
+ let address = defaultCaps['appium:address'] ?? helpers_1.TEST_HOST;
108
+ let port = defaultCaps['appium:port'];
109
+ const className = DriverClass.name || '(unknown driver)';
110
+ describe(`BaseDriver E2E (as ${className})`, function () {
111
+ let baseServer;
112
+ /** @type {P} */
113
+ let d;
114
+ /**
115
+ * This URL creates a new session
116
+ * @type {string}
117
+ **/
118
+ let newSessionURL;
119
+ /** @type {SessionHelpers['startSession']} */
120
+ let startSession;
121
+ /** @type {SessionHelpers['getSession']} */
122
+ let getSession;
123
+ /** @type {SessionHelpers['endSession']} */
124
+ let endSession;
125
+ /** @type {SessionHelpers['getCommand']} */
126
+ let getCommand;
127
+ /** @type {SessionHelpers['postCommand']} */
128
+ let postCommand;
129
+ before(async function () {
130
+ port = port ?? (await (0, helpers_1.getTestPort)());
131
+ defaultCaps = { ...defaultCaps };
132
+ d = new DriverClass({ port, address });
133
+ baseServer = await (0, driver_1.server)({
134
+ routeConfiguringFunction: (0, driver_1.routeConfiguringFunction)(d),
135
+ port,
136
+ hostname: address,
137
+ // @ts-expect-error
138
+ cliArgs: {},
139
+ });
140
+ ({ startSession, getSession, endSession, newSessionURL, getCommand, postCommand } =
141
+ createSessionHelpers(port, address));
295
142
  });
296
- await _bluebird.default.delay(400);
297
- const value = await getSession(sessionId);
298
- value.error.should.equal('invalid session id');
299
- should.equal(d.sessionId, null);
300
- const resp = (await endSession(newSession.sessionId)).data.value;
301
- resp.error.should.equal('invalid session id');
302
- });
303
- it('should not have a timer running before or after a session', async function () {
304
- should.not.exist(d.noCommandTimer);
305
- let newSession = await startTimeoutSession(0.25);
306
- newSession.sessionId.should.equal(d.sessionId);
307
- should.exist(d.noCommandTimer);
308
- await endSession(newSession.sessionId);
309
- should.not.exist(d.noCommandTimer);
310
- });
311
- });
312
- describe('settings api', function () {
313
- before(function () {
314
- d.settings = new _driver.DeviceSettings({
315
- ignoreUnimportantViews: false
143
+ after(async function () {
144
+ await baseServer.close();
316
145
  });
317
- });
318
- it('should be able to get settings object', function () {
319
- d.settings.getSettings().ignoreUnimportantViews.should.be.false;
320
- });
321
- it('should not reject when `updateSettings` method is not provided', async function () {
322
- await d.settings.update({
323
- ignoreUnimportantViews: true
324
- }).should.not.be.rejected;
325
- });
326
- it('should reject for invalid update object', async function () {
327
- await d.settings.update('invalid json').should.be.rejectedWith('JSON');
328
- });
329
- });
330
- describe('unexpected exits', function () {
331
- let sandbox;
332
- beforeEach(function () {
333
- sandbox = _sinon.default.createSandbox();
334
- });
335
- afterEach(function () {
336
- sandbox.restore();
337
- });
338
- it('should reject a current command when the driver crashes', async function () {
339
- sandbox.stub(d, 'getStatus').callsFake(async function () {
340
- await _bluebird.default.delay(5000);
146
+ describe('session handling', function () {
147
+ it('should handle idempotency while creating sessions', async function () {
148
+ const sessionIds = [];
149
+ let times = 0;
150
+ do {
151
+ const { sessionId } = await startSession({
152
+ capabilities: { alwaysMatch: defaultCaps },
153
+ }, {
154
+ headers: {
155
+ 'X-Idempotency-Key': '123456',
156
+ },
157
+ // XXX: I'm not sure what these are, as they are not documented axios options,
158
+ // nor are they mentioned in our source
159
+ // @ts-expect-error
160
+ simple: false,
161
+ resolveWithFullResponse: true,
162
+ });
163
+ sessionIds.push(sessionId);
164
+ times++;
165
+ } while (times < 2);
166
+ lodash_1.default.uniq(sessionIds).length.should.equal(1);
167
+ const { status, data } = await endSession(sessionIds[0]);
168
+ status.should.equal(200);
169
+ should.equal(data.value, null);
170
+ });
171
+ it('should handle idempotency while creating parallel sessions', async function () {
172
+ const reqs = [];
173
+ let times = 0;
174
+ do {
175
+ reqs.push(startSession({
176
+ capabilities: {
177
+ alwaysMatch: defaultCaps,
178
+ },
179
+ }, {
180
+ headers: {
181
+ 'X-Idempotency-Key': '12345',
182
+ },
183
+ }));
184
+ times++;
185
+ } while (times < 2);
186
+ const sessionIds = lodash_1.default.map(await bluebird_1.default.all(reqs), 'sessionId');
187
+ lodash_1.default.uniq(sessionIds).length.should.equal(1);
188
+ const { status, data } = await endSession(sessionIds[0]);
189
+ status.should.equal(200);
190
+ should.equal(data.value, null);
191
+ });
192
+ it('should create session and retrieve a session id, then delete it', async function () {
193
+ let { status, data } = await axios_1.default.post(newSessionURL, {
194
+ capabilities: {
195
+ alwaysMatch: defaultCaps,
196
+ },
197
+ });
198
+ status.should.equal(200);
199
+ should.exist(data.value.sessionId);
200
+ data.value.capabilities.platformName.should.equal(defaultCaps.platformName);
201
+ data.value.capabilities.deviceName.should.equal(defaultCaps['appium:deviceName']);
202
+ ({ status, data } = await endSession(/** @type {string} */ (d.sessionId)));
203
+ status.should.equal(200);
204
+ should.equal(data.value, null);
205
+ should.equal(d.sessionId, null);
206
+ });
341
207
  });
342
- const reqPromise = getCommand('status', {
343
- validateStatus: null
208
+ it.skip('should throw NYI for commands not implemented', async function () { });
209
+ describe('command timeouts', function () {
210
+ let originalFindElement, originalFindElements;
211
+ /**
212
+ * @param {number} [timeout]
213
+ */
214
+ async function startTimeoutSession(timeout) {
215
+ const caps = lodash_1.default.cloneDeep(defaultCaps);
216
+ caps['appium:newCommandTimeout'] = timeout;
217
+ return await startSession({ capabilities: { alwaysMatch: caps } });
218
+ }
219
+ before(function () {
220
+ originalFindElement = d.findElement;
221
+ d.findElement = function () {
222
+ return 'foo';
223
+ }.bind(d);
224
+ originalFindElements = d.findElements;
225
+ d.findElements = async function () {
226
+ await bluebird_1.default.delay(200);
227
+ return ['foo'];
228
+ }.bind(d);
229
+ });
230
+ after(function () {
231
+ d.findElement = originalFindElement;
232
+ d.findElements = originalFindElements;
233
+ });
234
+ it('should set a default commandTimeout', async function () {
235
+ let newSession = await startTimeoutSession();
236
+ d.newCommandTimeoutMs.should.be.above(0);
237
+ await endSession(newSession.sessionId);
238
+ });
239
+ it('should timeout on commands using commandTimeout cap', async function () {
240
+ let newSession = await startTimeoutSession(0.25);
241
+ const sessionId = /** @type {string} */ (d.sessionId);
242
+ await postCommand(sessionId, 'element', {
243
+ using: 'name',
244
+ value: 'foo',
245
+ });
246
+ await bluebird_1.default.delay(400);
247
+ const value = await getSession(sessionId);
248
+ should.equal(value.error, 'invalid session id');
249
+ should.equal(d.sessionId, null);
250
+ const resp = (await endSession(newSession.sessionId)).data.value;
251
+ should.equal(resp?.error, 'invalid session id');
252
+ });
253
+ it('should not timeout with commandTimeout of false', async function () {
254
+ let newSession = await startTimeoutSession(0.1);
255
+ let start = Date.now();
256
+ const value = await postCommand(/** @type {string} */ (d.sessionId), 'elements', {
257
+ using: 'name',
258
+ value: 'foo',
259
+ });
260
+ (Date.now() - start).should.be.above(150);
261
+ value.should.eql(['foo']);
262
+ await endSession(newSession.sessionId);
263
+ });
264
+ it('should not timeout with commandTimeout of 0', async function () {
265
+ d.newCommandTimeoutMs = 2;
266
+ let newSession = await startTimeoutSession(0);
267
+ await postCommand(/** @type {string} */ (d.sessionId), 'element', {
268
+ using: 'name',
269
+ value: 'foo',
270
+ });
271
+ await bluebird_1.default.delay(400);
272
+ const value = await getSession(/** @type {string} */ (d.sessionId));
273
+ value.platformName?.should.equal(defaultCaps.platformName);
274
+ const resp = (await endSession(newSession.sessionId)).data.value;
275
+ should.equal(resp, null);
276
+ d.newCommandTimeoutMs = 60 * 1000;
277
+ });
278
+ it('should not timeout if its just the command taking awhile', async function () {
279
+ let newSession = await startTimeoutSession(0.25);
280
+ // XXX: race condition: we must build this URL before ...something happens...
281
+ // which causes `d.sessionId` to be missing
282
+ const { sessionId } = d;
283
+ await postCommand(/** @type {string} */ (d.sessionId), 'element', {
284
+ using: 'name',
285
+ value: 'foo',
286
+ });
287
+ await bluebird_1.default.delay(400);
288
+ const value = await getSession(/** @type {string} */ (sessionId));
289
+ /** @type {string} */ (value.error).should.equal('invalid session id');
290
+ should.equal(d.sessionId, null);
291
+ const resp = (await endSession(newSession.sessionId)).data.value;
292
+ /** @type {string} */ ( /** @type { {error: string} } */(resp).error).should.equal('invalid session id');
293
+ });
294
+ it('should not have a timer running before or after a session', async function () {
295
+ // @ts-expect-error
296
+ should.not.exist(d.noCommandTimer);
297
+ let newSession = await startTimeoutSession(0.25);
298
+ newSession.sessionId.should.equal(d.sessionId);
299
+ // @ts-expect-error
300
+ should.exist(d.noCommandTimer);
301
+ await endSession(newSession.sessionId);
302
+ // @ts-expect-error
303
+ should.not.exist(d.noCommandTimer);
304
+ });
344
305
  });
345
- await _bluebird.default.delay(100);
346
- const shutdownEventPromise = new _bluebird.default((resolve, reject) => {
347
- setTimeout(() => reject(new Error('onUnexpectedShutdown event is expected to be fired within 5 seconds timeout')), 5000);
348
- d.onUnexpectedShutdown(resolve);
306
+ describe('settings api', function () {
307
+ before(function () {
308
+ d.settings = new driver_1.DeviceSettings({ ignoreUnimportantViews: false });
309
+ });
310
+ it('should be able to get settings object', function () {
311
+ d.settings.getSettings().ignoreUnimportantViews.should.be.false;
312
+ });
313
+ it('should not reject when `updateSettings` method is not provided', async function () {
314
+ await d.settings.update({ ignoreUnimportantViews: true }).should.not.be.rejected;
315
+ });
316
+ it('should reject for invalid update object', async function () {
317
+ await d.settings
318
+ // @ts-expect-error
319
+ .update('invalid json')
320
+ .should.be.rejectedWith('JSON');
321
+ });
349
322
  });
350
- d.startUnexpectedShutdown(new Error('Crashytimes'));
351
- const value = await reqPromise;
352
- value.message.should.contain('Crashytimes');
353
- await shutdownEventPromise;
354
- });
355
- });
356
- describe('event timings', function () {
357
- it('should not add timings if not using opt-in cap', async function () {
358
- const session = await startSession({
359
- capabilities: {
360
- alwaysMatch: defaultCaps
361
- }
323
+ describe('unexpected exits', function () {
324
+ /** @type {import('sinon').SinonSandbox} */
325
+ let sandbox;
326
+ beforeEach(function () {
327
+ sandbox = sinon_1.default.createSandbox();
328
+ });
329
+ afterEach(function () {
330
+ sandbox.restore();
331
+ });
332
+ it('should reject a current command when the driver crashes', async function () {
333
+ sandbox.stub(d, 'getStatus').callsFake(async function () {
334
+ await bluebird_1.default.delay(5000);
335
+ });
336
+ const reqPromise = getCommand('status', { validateStatus: null });
337
+ // make sure that the request gets to the server before our shutdown
338
+ await bluebird_1.default.delay(100);
339
+ const shutdownEventPromise = new bluebird_1.default((resolve, reject) => {
340
+ setTimeout(() => reject(new Error('onUnexpectedShutdown event is expected to be fired within 5 seconds timeout')), 5000);
341
+ d.onUnexpectedShutdown(resolve);
342
+ });
343
+ d.startUnexpectedShutdown(new Error('Crashytimes'));
344
+ const value = await reqPromise;
345
+ value.message.should.contain('Crashytimes');
346
+ await shutdownEventPromise;
347
+ });
362
348
  });
363
- const res = await getSession(session.sessionId);
364
- should.not.exist(res.events);
365
- await endSession(session.sessionId);
366
- });
367
- it('should add start session timings', async function () {
368
- var _res$events, _res$events2, _res$events3, _res$events4;
369
-
370
- const caps = { ...defaultCaps,
371
- 'appium:eventTimings': true
372
- };
373
- const session = await startSession({
374
- capabilities: {
375
- alwaysMatch: caps
376
- }
349
+ describe('event timings', function () {
350
+ it('should not add timings if not using opt-in cap', async function () {
351
+ const session = await startSession({ capabilities: { alwaysMatch: defaultCaps } });
352
+ const res = await getSession(session.sessionId);
353
+ should.not.exist(res.events);
354
+ await endSession(session.sessionId);
355
+ });
356
+ it('should add start session timings', async function () {
357
+ const caps = { ...defaultCaps, 'appium:eventTimings': true };
358
+ const session = await startSession({ capabilities: { alwaysMatch: caps } });
359
+ const res = await getSession(session.sessionId);
360
+ should.exist(res.events);
361
+ should.exist(res.events?.newSessionRequested);
362
+ should.exist(res.events?.newSessionStarted);
363
+ res.events?.newSessionRequested[0].should.be.a('number');
364
+ res.events?.newSessionStarted[0].should.be.a('number');
365
+ await endSession(session.sessionId);
366
+ });
377
367
  });
378
- const res = await getSession(session.sessionId);
379
- should.exist(res.events);
380
- should.exist((_res$events = res.events) === null || _res$events === void 0 ? void 0 : _res$events.newSessionRequested);
381
- should.exist((_res$events2 = res.events) === null || _res$events2 === void 0 ? void 0 : _res$events2.newSessionStarted);
382
- (_res$events3 = res.events) === null || _res$events3 === void 0 ? void 0 : _res$events3.newSessionRequested[0].should.be.a('number');
383
- (_res$events4 = res.events) === null || _res$events4 === void 0 ? void 0 : _res$events4.newSessionStarted[0].should.be.a('number');
384
- await endSession(session.sessionId);
385
- });
386
368
  });
387
- });
388
369
  }
389
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["should","chai","createSessionHelpers","port","address","TEST_HOST","createAppiumTestURL","createAppiumURL","createSessionURL","_","newSessionURL","postCommand","sessionId","cmdName","data","config","url","response","axios","post","value","getCommand","sessionIdOrCmdName","cmdNameOrConfig","isString","validateStatus","startSession","defaultsDeep","capabilities","alwaysMatch","firstMatch","endSession","delete","getSession","driverE2ETestSuite","DriverClass","defaultCaps","className","name","describe","baseServer","d","before","getTestPort","server","routeConfiguringFunction","hostname","cliArgs","after","close","it","sessionIds","times","headers","simple","resolveWithFullResponse","push","uniq","length","equal","status","reqs","map","B","all","exist","platformName","deviceName","skip","originalFindElement","originalFindElements","startTimeoutSession","timeout","caps","cloneDeep","findElement","bind","findElements","delay","newSession","newCommandTimeoutMs","be","above","using","error","resp","start","Date","now","eql","not","noCommandTimer","settings","DeviceSettings","ignoreUnimportantViews","getSettings","false","update","rejected","rejectedWith","sandbox","beforeEach","sinon","createSandbox","afterEach","restore","stub","callsFake","reqPromise","shutdownEventPromise","resolve","reject","setTimeout","Error","onUnexpectedShutdown","startUnexpectedShutdown","message","contain","session","res","events","newSessionRequested","newSessionStarted","a"],"sources":["../../lib/e2e-suite.js"],"sourcesContent":["import _ from 'lodash';\nimport {server, routeConfiguringFunction, DeviceSettings} from 'appium/driver';\nimport axios from 'axios';\nimport B from 'bluebird';\nimport {TEST_HOST, getTestPort, createAppiumURL} from './helpers';\nimport chai from 'chai';\nimport sinon from 'sinon';\n\nconst should = chai.should();\n\n/**\n * Creates some helper functions for E2E tests to manage sessions.\n * @template [CommandData=unknown]\n * @template [ResponseData=any]\n * @param {number} port - Port on which the server is running. Typically this will be retrieved via `get-port` beforehand\n * @param {string} [address] - Address/host on which the server is running. Defaults to {@linkcode TEST_HOST}\n * @returns {SessionHelpers<CommandData, ResponseData>}\n */\nexport function createSessionHelpers(port, address = TEST_HOST) {\n  const createAppiumTestURL =\n    /** @type {import('lodash').CurriedFunction2<string,string,string>} */ (\n      createAppiumURL(address, port)\n    );\n\n  const createSessionURL = createAppiumTestURL(_, '');\n  const newSessionURL = createAppiumTestURL('', 'session');\n  return /** @type {SessionHelpers<CommandData, ResponseData>} */ ({\n    newSessionURL,\n    createAppiumTestURL,\n    /**\n     *\n     * @param {string} sessionId\n     * @param {string} cmdName\n     * @param {any} [data]\n     * @param {AxiosRequestConfig} [config]\n     * @returns {Promise<any>}\n     */\n    postCommand: async (sessionId, cmdName, data = {}, config = {}) => {\n      const url = createAppiumTestURL(sessionId, cmdName);\n      const response = await axios.post(url, data, config);\n      return response.data?.value;\n    },\n    /**\n     *\n     * @param {string} sessionIdOrCmdName\n     * @param {string|AxiosRequestConfig} cmdNameOrConfig\n     * @param {AxiosRequestConfig} [config]\n     * @returns {Promise<any>}\n     */\n    getCommand: async (sessionIdOrCmdName, cmdNameOrConfig, config = {}) => {\n      if (!_.isString(cmdNameOrConfig)) {\n        config = cmdNameOrConfig;\n        cmdNameOrConfig = sessionIdOrCmdName;\n        sessionIdOrCmdName = '';\n      }\n      const response = await axios({\n        url: createAppiumTestURL(sessionIdOrCmdName, cmdNameOrConfig),\n        validateStatus: null,\n        ...config,\n      });\n      return response.data?.value;\n    },\n    /**\n     *\n     * @param {NewSessionData} data\n     * @param {AxiosRequestConfig} [config]\n     */\n    startSession: async (data, config = {}) => {\n      data = _.defaultsDeep(data, {\n        capabilities: {\n          alwaysMatch: {},\n          firstMatch: [{}],\n        },\n      });\n      const response = await axios.post(newSessionURL, data, config);\n      return response.data?.value;\n    },\n    /**\n     *\n     * @param {string} sessionId\n     */\n    endSession: async (sessionId) =>\n      await axios.delete(createSessionURL(sessionId), {\n        validateStatus: null,\n      }),\n    /**\n     * @param {string} sessionId\n     * @returns {Promise<any>}\n     */\n    getSession: async (sessionId) => {\n      const response = await axios({\n        url: createSessionURL(sessionId),\n        validateStatus: null,\n      });\n      return response.data?.value;\n    },\n  });\n}\n\n/**\n * Creates E2E test suites for a driver.\n * @template {Driver} P\n * @param {DriverClass<P>} DriverClass\n * @param {AppiumW3CCapabilities} [defaultCaps]\n */\nexport function driverE2ETestSuite(DriverClass, defaultCaps = {}) {\n  let address = defaultCaps['appium:address'] ?? TEST_HOST;\n  let port = defaultCaps['appium:port'];\n  const className = DriverClass.name || '(unknown driver)';\n\n  describe(`BaseDriver E2E (as ${className})`, function () {\n    let baseServer;\n    /** @type {P} */\n    let d;\n    /**\n     * This URL creates a new session\n     * @type {string}\n     **/\n    let newSessionURL;\n\n    /** @type {SessionHelpers['startSession']} */\n    let startSession;\n    /** @type {SessionHelpers['getSession']} */\n    let getSession;\n    /** @type {SessionHelpers['endSession']} */\n    let endSession;\n    /** @type {SessionHelpers['getCommand']} */\n    let getCommand;\n    /** @type {SessionHelpers['postCommand']} */\n    let postCommand;\n    before(async function () {\n      port = port ?? (await getTestPort());\n      defaultCaps = {...defaultCaps, 'appium:port': port};\n      d = new DriverClass({port, address});\n      baseServer = await server({\n        routeConfiguringFunction: routeConfiguringFunction(d),\n        port,\n        hostname: address,\n        // @ts-expect-error\n        cliArgs: {},\n      });\n      ({startSession, getSession, endSession, newSessionURL, getCommand, postCommand} =\n        createSessionHelpers(port, address));\n    });\n\n    after(async function () {\n      await baseServer.close();\n    });\n\n    describe('session handling', function () {\n      it('should handle idempotency while creating sessions', async function () {\n        const sessionIds = [];\n        let times = 0;\n        do {\n          const {sessionId} = await startSession(\n            {\n              capabilities: {alwaysMatch: defaultCaps},\n            },\n            {\n              headers: {\n                'X-Idempotency-Key': '123456',\n              },\n              // XXX: I'm not sure what these are, as they are not documented axios options,\n              // nor are they mentioned in our source\n              // @ts-expect-error\n              simple: false,\n              resolveWithFullResponse: true,\n            }\n          );\n\n          sessionIds.push(sessionId);\n          times++;\n        } while (times < 2);\n        _.uniq(sessionIds).length.should.equal(1);\n\n        const {status, data} = await endSession(sessionIds[0]);\n        status.should.equal(200);\n        should.equal(data.value, null);\n      });\n\n      it('should handle idempotency while creating parallel sessions', async function () {\n        const reqs = [];\n        let times = 0;\n        do {\n          reqs.push(\n            startSession(\n              {\n                capabilities: {\n                  alwaysMatch: defaultCaps,\n                },\n              },\n              {\n                headers: {\n                  'X-Idempotency-Key': '12345',\n                },\n              }\n            )\n          );\n          times++;\n        } while (times < 2);\n        const sessionIds = _.map(await B.all(reqs), 'sessionId');\n        _.uniq(sessionIds).length.should.equal(1);\n\n        const {status, data} = await endSession(sessionIds[0]);\n        status.should.equal(200);\n        should.equal(data.value, null);\n      });\n\n      it('should create session and retrieve a session id, then delete it', async function () {\n        let {status, data} = await axios.post(newSessionURL, {\n          capabilities: {\n            alwaysMatch: defaultCaps,\n          },\n        });\n\n        status.should.equal(200);\n        should.exist(data.value.sessionId);\n        data.value.capabilities.platformName.should.equal(defaultCaps.platformName);\n        data.value.capabilities.deviceName.should.equal(defaultCaps['appium:deviceName']);\n\n        ({status, data} = await endSession(/** @type {string} */ (d.sessionId)));\n\n        status.should.equal(200);\n        should.equal(data.value, null);\n        should.equal(d.sessionId, null);\n      });\n    });\n\n    it.skip('should throw NYI for commands not implemented', async function () {});\n\n    describe('command timeouts', function () {\n      let originalFindElement, originalFindElements;\n\n      /**\n       * @param {number} [timeout]\n       */\n      async function startTimeoutSession(timeout) {\n        const caps = _.cloneDeep(defaultCaps);\n        caps['appium:newCommandTimeout'] = timeout;\n        return await startSession({capabilities: {alwaysMatch: caps}});\n      }\n\n      before(function () {\n        originalFindElement = d.findElement;\n        d.findElement = function () {\n          return 'foo';\n        }.bind(d);\n\n        originalFindElements = d.findElements;\n        d.findElements = async function () {\n          await B.delay(200);\n          return ['foo'];\n        }.bind(d);\n      });\n\n      after(function () {\n        d.findElement = originalFindElement;\n        d.findElements = originalFindElements;\n      });\n\n      it('should set a default commandTimeout', async function () {\n        let newSession = await startTimeoutSession();\n        d.newCommandTimeoutMs.should.be.above(0);\n        await endSession(newSession.sessionId);\n      });\n\n      it('should timeout on commands using commandTimeout cap', async function () {\n        let newSession = await startTimeoutSession(0.25);\n        const sessionId = /** @type {string} */ (d.sessionId);\n        await postCommand(sessionId, 'element', {\n          using: 'name',\n          value: 'foo',\n        });\n        await B.delay(400);\n        const value = await getSession(sessionId);\n        should.equal(value.error, 'invalid session id');\n        should.equal(d.sessionId, null);\n        const resp = (await endSession(newSession.sessionId)).data.value;\n        should.equal(resp?.error, 'invalid session id');\n      });\n\n      it('should not timeout with commandTimeout of false', async function () {\n        let newSession = await startTimeoutSession(0.1);\n        let start = Date.now();\n        const value = await postCommand(/** @type {string} */ (d.sessionId), 'elements', {\n          using: 'name',\n          value: 'foo',\n        });\n        (Date.now() - start).should.be.above(150);\n        value.should.eql(['foo']);\n        await endSession(newSession.sessionId);\n      });\n\n      it('should not timeout with commandTimeout of 0', async function () {\n        d.newCommandTimeoutMs = 2;\n        let newSession = await startTimeoutSession(0);\n\n        await postCommand(/** @type {string} */ (d.sessionId), 'element', {\n          using: 'name',\n          value: 'foo',\n        });\n        await B.delay(400);\n        const value = await getSession(/** @type {string} */ (d.sessionId));\n        value.platformName?.should.equal(defaultCaps.platformName);\n        const resp = (await endSession(newSession.sessionId)).data.value;\n        should.equal(resp, null);\n\n        d.newCommandTimeoutMs = 60 * 1000;\n      });\n\n      it('should not timeout if its just the command taking awhile', async function () {\n        let newSession = await startTimeoutSession(0.25);\n        // XXX: race condition: we must build this URL before ...something happens...\n        // which causes `d.sessionId` to be missing\n        const {sessionId} = d;\n\n        await postCommand(/** @type {string} */ (d.sessionId), 'element', {\n          using: 'name',\n          value: 'foo',\n        });\n        await B.delay(400);\n        const value = await getSession(/** @type {string} */ (sessionId));\n        value.error.should.equal('invalid session id');\n        should.equal(d.sessionId, null);\n        const resp = (await endSession(newSession.sessionId)).data.value;\n        /** @type {string} */ (/** @type { {error: string} } */ (resp).error).should.equal(\n          'invalid session id'\n        );\n      });\n\n      it('should not have a timer running before or after a session', async function () {\n        // @ts-expect-error\n        should.not.exist(d.noCommandTimer);\n        let newSession = await startTimeoutSession(0.25);\n        newSession.sessionId.should.equal(d.sessionId);\n        // @ts-expect-error\n        should.exist(d.noCommandTimer);\n        await endSession(newSession.sessionId);\n        // @ts-expect-error\n        should.not.exist(d.noCommandTimer);\n      });\n    });\n\n    describe('settings api', function () {\n      before(function () {\n        d.settings = new DeviceSettings({ignoreUnimportantViews: false});\n      });\n      it('should be able to get settings object', function () {\n        d.settings.getSettings().ignoreUnimportantViews.should.be.false;\n      });\n      it('should not reject when `updateSettings` method is not provided', async function () {\n        await d.settings.update({ignoreUnimportantViews: true}).should.not.be.rejected;\n      });\n      it('should reject for invalid update object', async function () {\n        await d.settings\n          // @ts-expect-error\n          .update('invalid json')\n          .should.be.rejectedWith('JSON');\n      });\n    });\n\n    describe('unexpected exits', function () {\n      /** @type {import('sinon').SinonSandbox} */\n      let sandbox;\n      beforeEach(function () {\n        sandbox = sinon.createSandbox();\n      });\n\n      afterEach(function () {\n        sandbox.restore();\n      });\n\n      it('should reject a current command when the driver crashes', async function () {\n        sandbox.stub(d, 'getStatus').callsFake(async function () {\n          await B.delay(5000);\n        });\n        const reqPromise = getCommand('status', {validateStatus: null});\n        // make sure that the request gets to the server before our shutdown\n        await B.delay(100);\n        const shutdownEventPromise = new B((resolve, reject) => {\n          setTimeout(\n            () =>\n              reject(\n                new Error(\n                  'onUnexpectedShutdown event is expected to be fired within 5 seconds timeout'\n                )\n              ),\n            5000\n          );\n          d.onUnexpectedShutdown(resolve);\n        });\n        d.startUnexpectedShutdown(new Error('Crashytimes'));\n        const value = await reqPromise;\n        value.message.should.contain('Crashytimes');\n        await shutdownEventPromise;\n      });\n    });\n\n    describe('event timings', function () {\n      it('should not add timings if not using opt-in cap', async function () {\n        const session = await startSession({capabilities: {alwaysMatch: defaultCaps}});\n        const res = await getSession(session.sessionId);\n        should.not.exist(res.events);\n        await endSession(session.sessionId);\n      });\n      it('should add start session timings', async function () {\n        const caps = {...defaultCaps, 'appium:eventTimings': true};\n        const session = await startSession({capabilities: {alwaysMatch: caps}});\n        const res = await getSession(session.sessionId);\n        should.exist(res.events);\n        should.exist(res.events?.newSessionRequested);\n        should.exist(res.events?.newSessionStarted);\n        res.events?.newSessionRequested[0].should.be.a('number');\n        res.events?.newSessionStarted[0].should.be.a('number');\n        await endSession(session.sessionId);\n      });\n    });\n  });\n}\n\n/**\n * A {@linkcode DriverClass}, except using the base {@linkcode Driver} type instead of `ExternalDriver`.\n * This allows the suite to work for `BaseDriver`.\n * @template {Driver} P\n * @typedef {import('@appium/types').DriverClass<P>} DriverClass\n */\n\n/**\n * @typedef {import('@appium/types').Capabilities} Capabilities\n * @typedef {import('@appium/types').Driver} Driver\n * @typedef {import('@appium/types').DriverStatic} DriverStatic\n * @typedef {import('@appium/types').AppiumW3CCapabilities} AppiumW3CCapabilities\n * @typedef {import('axios').AxiosRequestConfig} AxiosRequestConfig\n * @typedef {import('@appium/types').SingularSessionData} SingularSessionData\n */\n\n/**\n * @template T,D\n * @typedef {import('axios').AxiosResponse<T, D>} AxiosResponse\n */\n\n/**\n * @typedef NewSessionData\n * @property {import('type-fest').RequireAtLeastOne<import('@appium/types').W3CCapabilities, 'firstMatch'|'alwaysMatch'>} capabilities\n */\n\n/**\n * @typedef NewSessionResponse\n * @property {string} sessionId,\n * @property {import('@appium/types').Capabilities} capabilities\n */\n\n/**\n * Some E2E helpers for making requests and managing sessions\n * See {@linkcode createSessionHelpers}\n * @template [CommandData=unknown]\n * @template [ResponseData=any]\n * @typedef SessionHelpers\n * @property {string} newSessionURL - URL to create a new session. Can be used with raw `axios` requests to fully inspect raw response.  Mostly, this will not be used.\n * @property {(data: NewSessionData, config?: AxiosRequestConfig) => Promise<NewSessionResponse>} startSession - Begin a session\n * @property {(sessionId: string) => Promise<AxiosResponse<{value: {error?: string}?}, {validateStatus: null}>>} endSession - End a session. _Note: resolves with raw response object_\n * @property {(sessionId: string) => Promise<SingularSessionData>} getSession - Get info about a session\n * @property {(sessionId: string, cmdName: string, data?: CommandData, config?: AxiosRequestConfig) => Promise<ResponseData>} postCommand - Send an arbitrary command via `POST`.\n * @property {(sessionIdOrCmdName: string, cmdNameOrConfig: string|AxiosRequestConfig, config?: AxiosRequestConfig) => Promise<ResponseData>} getCommand - Send an arbitrary command via `GET`. Optional `sessionId`.\n */\n"],"mappings":"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"}
370
+ exports.driverE2ETestSuite = driverE2ETestSuite;
371
+ /**
372
+ * A {@linkcode DriverClass}, except using the base {@linkcode Driver} type instead of `ExternalDriver`.
373
+ * This allows the suite to work for `BaseDriver`.
374
+ * @template {Driver} P
375
+ * @typedef {import('@appium/types').DriverClass<P>} DriverClass
376
+ */
377
+ /**
378
+ * @typedef {import('@appium/types').Driver} Driver
379
+ * @typedef {import('@appium/types').Constraints} Constraints
380
+ * @typedef {import('@appium/types').DriverStatic} DriverStatic
381
+ * @typedef {import('@appium/types').StringRecord} StringRecord
382
+ * @typedef {import('@appium/types').BaseDriverCapConstraints} BaseDriverCapConstraints
383
+ * @typedef {import('@appium/types').BaseNSCapabilities} BaseNSCapabilities
384
+ * @typedef {import('axios').AxiosRequestConfig} AxiosRequestConfig
385
+ * @typedef {import('@appium/types').SingularSessionData} SingularSessionData
386
+ */
387
+ /**
388
+ * @template T,D
389
+ * @typedef {import('axios').AxiosResponse<T, D>} AxiosResponse
390
+ */
391
+ /**
392
+ * @template {Constraints} [C=BaseDriverCapConstraints]
393
+ * @template {StringRecord|void} [Extra=void]
394
+ * @typedef NewSessionData
395
+ * @property {import('type-fest').RequireAtLeastOne<import('@appium/types').W3CCapabilities<C, Extra>, 'firstMatch'|'alwaysMatch'>} capabilities
396
+ */
397
+ /**
398
+ * @template {Constraints} [C=BaseDriverCapConstraints]
399
+ * @template {StringRecord|void} [Extra=void]
400
+ * @typedef NewSessionResponse
401
+ * @property {string} sessionId,
402
+ * @property {import('@appium/types').Capabilities<C, Extra>} capabilities
403
+ */
404
+ /**
405
+ * Some E2E helpers for making requests and managing sessions
406
+ * See {@linkcode createSessionHelpers}
407
+ * @template [CommandData=unknown]
408
+ * @template [ResponseData=any]
409
+ * @typedef SessionHelpers
410
+ * @property {string} newSessionURL - URL to create a new session. Can be used with raw `axios` requests to fully inspect raw response. Mostly, this will not be used.
411
+ * @property {(data: NewSessionData, config?: AxiosRequestConfig) => Promise<NewSessionResponse>} startSession - Begin a session
412
+ * @property {(sessionId: string) => Promise<AxiosResponse<{value: {error?: string}?}, {validateStatus: null}>>} endSession - End a session. _Note: resolves with raw response object_
413
+ * @property {(sessionId: string) => Promise<SingularSessionData>} getSession - Get info about a session
414
+ * @property {(sessionId: string, cmdName: string, data?: CommandData, config?: AxiosRequestConfig) => Promise<ResponseData>} postCommand - Send an arbitrary command via `POST`.
415
+ * @property {(sessionIdOrCmdName: string, cmdNameOrConfig: string|AxiosRequestConfig, config?: AxiosRequestConfig) => Promise<ResponseData>} getCommand - Send an arbitrary command via `GET`. Optional `sessionId`.
416
+ */
417
+ //# sourceMappingURL=e2e-suite.js.map