@appium/base-driver 8.7.2 → 9.0.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.
Files changed (121) hide show
  1. package/build/lib/basedriver/capabilities.d.ts +11 -163
  2. package/build/lib/basedriver/capabilities.d.ts.map +1 -1
  3. package/build/lib/basedriver/capabilities.js +354 -236
  4. package/build/lib/basedriver/capabilities.js.map +1 -1
  5. package/build/lib/basedriver/commands/event.d.ts +7 -6
  6. package/build/lib/basedriver/commands/event.d.ts.map +1 -1
  7. package/build/lib/basedriver/commands/event.js +55 -35
  8. package/build/lib/basedriver/commands/event.js.map +1 -1
  9. package/build/lib/basedriver/commands/execute.d.ts +7 -6
  10. package/build/lib/basedriver/commands/execute.d.ts.map +1 -1
  11. package/build/lib/basedriver/commands/execute.js +66 -58
  12. package/build/lib/basedriver/commands/execute.js.map +1 -1
  13. package/build/lib/basedriver/commands/find.d.ts +9 -7
  14. package/build/lib/basedriver/commands/find.d.ts.map +1 -1
  15. package/build/lib/basedriver/commands/find.js +102 -54
  16. package/build/lib/basedriver/commands/find.js.map +1 -1
  17. package/build/lib/basedriver/commands/index.d.ts +3 -7
  18. package/build/lib/basedriver/commands/index.d.ts.map +1 -1
  19. package/build/lib/basedriver/commands/index.js +30 -33
  20. package/build/lib/basedriver/commands/index.js.map +1 -1
  21. package/build/lib/basedriver/commands/log.d.ts +8 -9
  22. package/build/lib/basedriver/commands/log.d.ts.map +1 -1
  23. package/build/lib/basedriver/commands/log.js +54 -38
  24. package/build/lib/basedriver/commands/log.js.map +1 -1
  25. package/build/lib/basedriver/commands/session.d.ts +7 -6
  26. package/build/lib/basedriver/commands/session.d.ts.map +1 -1
  27. package/build/lib/basedriver/commands/session.js +46 -39
  28. package/build/lib/basedriver/commands/session.js.map +1 -1
  29. package/build/lib/basedriver/commands/settings.d.ts +7 -7
  30. package/build/lib/basedriver/commands/settings.d.ts.map +1 -1
  31. package/build/lib/basedriver/commands/settings.js +35 -28
  32. package/build/lib/basedriver/commands/settings.js.map +1 -1
  33. package/build/lib/basedriver/commands/timeout.d.ts +7 -5
  34. package/build/lib/basedriver/commands/timeout.d.ts.map +1 -1
  35. package/build/lib/basedriver/commands/timeout.js +144 -162
  36. package/build/lib/basedriver/commands/timeout.js.map +1 -1
  37. package/build/lib/basedriver/core.d.ts +6 -157
  38. package/build/lib/basedriver/core.d.ts.map +1 -1
  39. package/build/lib/basedriver/core.js +361 -230
  40. package/build/lib/basedriver/core.js.map +1 -1
  41. package/build/lib/basedriver/desired-caps.js +80 -110
  42. package/build/lib/basedriver/desired-caps.js.map +1 -1
  43. package/build/lib/basedriver/device-settings.js +57 -62
  44. package/build/lib/basedriver/device-settings.js.map +1 -1
  45. package/build/lib/basedriver/driver.d.ts +21 -267
  46. package/build/lib/basedriver/driver.d.ts.map +1 -1
  47. package/build/lib/basedriver/driver.js +362 -258
  48. package/build/lib/basedriver/driver.js.map +1 -1
  49. package/build/lib/basedriver/helpers.js +500 -495
  50. package/build/lib/basedriver/helpers.js.map +1 -1
  51. package/build/lib/basedriver/logger.d.ts +1 -1
  52. package/build/lib/basedriver/logger.d.ts.map +1 -1
  53. package/build/lib/basedriver/logger.js +5 -15
  54. package/build/lib/basedriver/logger.js.map +1 -1
  55. package/build/lib/constants.js +14 -14
  56. package/build/lib/constants.js.map +1 -1
  57. package/build/lib/express/crash.js +8 -15
  58. package/build/lib/express/crash.js.map +1 -1
  59. package/build/lib/express/express-logging.js +49 -59
  60. package/build/lib/express/express-logging.js.map +1 -1
  61. package/build/lib/express/idempotency.js +125 -177
  62. package/build/lib/express/idempotency.js.map +1 -1
  63. package/build/lib/express/logger.d.ts +1 -1
  64. package/build/lib/express/logger.d.ts.map +1 -1
  65. package/build/lib/express/logger.js +5 -15
  66. package/build/lib/express/logger.js.map +1 -1
  67. package/build/lib/express/middleware.js +82 -107
  68. package/build/lib/express/middleware.js.map +1 -1
  69. package/build/lib/express/server.d.ts +17 -5
  70. package/build/lib/express/server.d.ts.map +1 -1
  71. package/build/lib/express/server.js +259 -224
  72. package/build/lib/express/server.js.map +1 -1
  73. package/build/lib/express/static.js +64 -81
  74. package/build/lib/express/static.js.map +1 -1
  75. package/build/lib/express/websocket.js +115 -87
  76. package/build/lib/express/websocket.js.map +1 -1
  77. package/build/lib/helpers/capabilities.d.ts +1 -59
  78. package/build/lib/helpers/capabilities.d.ts.map +1 -1
  79. package/build/lib/helpers/capabilities.js +72 -69
  80. package/build/lib/helpers/capabilities.js.map +1 -1
  81. package/build/lib/index.js +64 -180
  82. package/build/lib/index.js.map +1 -1
  83. package/build/lib/jsonwp-proxy/protocol-converter.js +215 -227
  84. package/build/lib/jsonwp-proxy/protocol-converter.js.map +1 -1
  85. package/build/lib/jsonwp-proxy/proxy.d.ts.map +1 -1
  86. package/build/lib/jsonwp-proxy/proxy.js +355 -393
  87. package/build/lib/jsonwp-proxy/proxy.js.map +1 -1
  88. package/build/lib/jsonwp-status/status.js +119 -130
  89. package/build/lib/jsonwp-status/status.js.map +1 -1
  90. package/build/lib/protocol/errors.d.ts +135 -32
  91. package/build/lib/protocol/errors.d.ts.map +1 -1
  92. package/build/lib/protocol/errors.js +871 -919
  93. package/build/lib/protocol/errors.js.map +1 -1
  94. package/build/lib/protocol/helpers.js +37 -37
  95. package/build/lib/protocol/helpers.js.map +1 -1
  96. package/build/lib/protocol/index.js +22 -109
  97. package/build/lib/protocol/index.js.map +1 -1
  98. package/build/lib/protocol/protocol.js +394 -350
  99. package/build/lib/protocol/protocol.js.map +1 -1
  100. package/build/lib/protocol/routes.d.ts +1238 -4
  101. package/build/lib/protocol/routes.d.ts.map +1 -1
  102. package/build/lib/protocol/routes.js +964 -1327
  103. package/build/lib/protocol/routes.js.map +1 -1
  104. package/build/lib/protocol/validators.js +32 -39
  105. package/build/lib/protocol/validators.js.map +1 -1
  106. package/build/tsconfig.tsbuildinfo +1 -1
  107. package/lib/basedriver/capabilities.js +80 -39
  108. package/lib/basedriver/commands/event.js +10 -5
  109. package/lib/basedriver/commands/execute.js +14 -9
  110. package/lib/basedriver/commands/find.js +18 -12
  111. package/lib/basedriver/commands/index.js +21 -16
  112. package/lib/basedriver/commands/log.js +24 -18
  113. package/lib/basedriver/commands/session.js +10 -5
  114. package/lib/basedriver/commands/settings.js +9 -6
  115. package/lib/basedriver/commands/timeout.js +10 -4
  116. package/lib/basedriver/core.js +2 -3
  117. package/lib/basedriver/driver.js +29 -18
  118. package/lib/express/server.js +6 -3
  119. package/lib/protocol/errors.js +155 -44
  120. package/lib/protocol/routes.js +11 -7
  121. package/package.json +14 -16
@@ -1,419 +1,381 @@
1
1
  "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.default = exports.JWProxy = void 0;
7
-
8
- require("source-map-support/register");
9
-
10
- var _lodash = _interopRequireDefault(require("lodash"));
11
-
12
- var _support = require("@appium/support");
13
-
14
- var _axios = _interopRequireDefault(require("axios"));
15
-
16
- var _status = require("../jsonwp-status/status");
17
-
18
- var _errors = require("../protocol/errors");
19
-
20
- var _protocol = require("../protocol");
21
-
22
- var _constants = require("../constants");
23
-
24
- var _protocolConverter = _interopRequireDefault(require("./protocol-converter"));
25
-
26
- var _helpers = require("../protocol/helpers");
27
-
28
- var _http = _interopRequireDefault(require("http"));
29
-
30
- var _https = _interopRequireDefault(require("https"));
31
-
32
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
33
-
34
- const DEFAULT_LOG = _support.logger.getLogger('WD Proxy');
35
-
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.JWProxy = void 0;
7
+ const lodash_1 = __importDefault(require("lodash"));
8
+ const support_1 = require("@appium/support");
9
+ const axios_1 = __importDefault(require("axios"));
10
+ const status_1 = require("../jsonwp-status/status");
11
+ const errors_1 = require("../protocol/errors");
12
+ const protocol_1 = require("../protocol");
13
+ const constants_1 = require("../constants");
14
+ const protocol_converter_1 = __importDefault(require("./protocol-converter"));
15
+ const helpers_1 = require("../protocol/helpers");
16
+ const http_1 = __importDefault(require("http"));
17
+ const https_1 = __importDefault(require("https"));
18
+ const DEFAULT_LOG = support_1.logger.getLogger('WD Proxy');
36
19
  const DEFAULT_REQUEST_TIMEOUT = 240000;
37
20
  const COMPACT_ERROR_PATTERNS = [/\bECONNREFUSED\b/, /socket hang up/];
38
- const {
39
- MJSONWP,
40
- W3C
41
- } = _constants.PROTOCOLS;
42
- const ALLOWED_OPTS = ['scheme', 'server', 'port', 'base', 'reqBasePath', 'sessionId', 'timeout', 'log', 'keepAlive'];
43
-
21
+ const { MJSONWP, W3C } = constants_1.PROTOCOLS;
22
+ const ALLOWED_OPTS = [
23
+ 'scheme',
24
+ 'server',
25
+ 'port',
26
+ 'base',
27
+ 'reqBasePath',
28
+ 'sessionId',
29
+ 'timeout',
30
+ 'log',
31
+ 'keepAlive',
32
+ ];
44
33
  class JWProxy {
45
- scheme;
46
- server;
47
- port;
48
- base;
49
- reqBasePath;
50
- sessionId;
51
- timeout;
52
-
53
- constructor(opts = {}) {
54
- opts = _lodash.default.pick(opts, ALLOWED_OPTS);
55
-
56
- const options = _lodash.default.defaults(_lodash.default.omit(opts, 'log'), {
57
- scheme: 'http',
58
- server: 'localhost',
59
- port: 4444,
60
- base: _constants.DEFAULT_BASE_PATH,
61
- reqBasePath: _constants.DEFAULT_BASE_PATH,
62
- sessionId: null,
63
- timeout: DEFAULT_REQUEST_TIMEOUT
64
- });
65
-
66
- options.scheme = options.scheme.toLowerCase();
67
- Object.assign(this, options);
68
- this._activeRequests = [];
69
- this._downstreamProtocol = null;
70
- const agentOpts = {
71
- keepAlive: opts.keepAlive ?? true,
72
- maxSockets: 10,
73
- maxFreeSockets: 5
74
- };
75
- this.httpAgent = new _http.default.Agent(agentOpts);
76
- this.httpsAgent = new _https.default.Agent(agentOpts);
77
- this.protocolConverter = new _protocolConverter.default(this.proxy.bind(this), opts.log);
78
- this._log = opts.log;
79
- }
80
-
81
- get log() {
82
- return this._log ?? DEFAULT_LOG;
83
- }
84
-
85
- async request(requestConfig) {
86
- const reqPromise = (0, _axios.default)(requestConfig);
87
-
88
- this._activeRequests.push(reqPromise);
89
-
90
- try {
91
- return await reqPromise;
92
- } finally {
93
- _lodash.default.pull(this._activeRequests, reqPromise);
34
+ constructor(opts = {}) {
35
+ opts = lodash_1.default.pick(opts, ALLOWED_OPTS);
36
+ // omit 'log' in the defaults assignment here because 'log' is a getter and we are going to set
37
+ // it to this._log (which lies behind the getter) further down
38
+ const options = lodash_1.default.defaults(lodash_1.default.omit(opts, 'log'), {
39
+ scheme: 'http',
40
+ server: 'localhost',
41
+ port: 4444,
42
+ base: constants_1.DEFAULT_BASE_PATH,
43
+ reqBasePath: constants_1.DEFAULT_BASE_PATH,
44
+ sessionId: null,
45
+ timeout: DEFAULT_REQUEST_TIMEOUT,
46
+ });
47
+ options.scheme = options.scheme.toLowerCase();
48
+ Object.assign(this, options);
49
+ this._activeRequests = [];
50
+ this._downstreamProtocol = null;
51
+ const agentOpts = {
52
+ keepAlive: opts.keepAlive ?? true,
53
+ maxSockets: 10,
54
+ maxFreeSockets: 5,
55
+ };
56
+ this.httpAgent = new http_1.default.Agent(agentOpts);
57
+ this.httpsAgent = new https_1.default.Agent(agentOpts);
58
+ this.protocolConverter = new protocol_converter_1.default(this.proxy.bind(this), opts.log);
59
+ this._log = opts.log;
94
60
  }
95
- }
96
-
97
- getActiveRequestsCount() {
98
- return this._activeRequests.length;
99
- }
100
-
101
- cancelActiveRequests() {
102
- this._activeRequests = [];
103
- }
104
-
105
- endpointRequiresSessionId(endpoint) {
106
- return !_lodash.default.includes(['/session', '/sessions', '/status'], endpoint);
107
- }
108
-
109
- set downstreamProtocol(value) {
110
- this._downstreamProtocol = value;
111
- this.protocolConverter.downstreamProtocol = value;
112
- }
113
-
114
- get downstreamProtocol() {
115
- return this._downstreamProtocol;
116
- }
117
-
118
- getUrlForProxy(url) {
119
- if (url === '') {
120
- url = '/';
61
+ get log() {
62
+ return this._log ?? DEFAULT_LOG;
121
63
  }
122
-
123
- const proxyBase = `${this.scheme}://${this.server}:${this.port}${this.base}`;
124
- const endpointRe = '(/(session|status))';
125
- let remainingUrl = '';
126
-
127
- if (/^http/.test(url)) {
128
- const first = new RegExp(`(https?://.+)${endpointRe}`).exec(url);
129
-
130
- if (!first) {
131
- throw new Error('Got a complete url but could not extract JWP endpoint');
132
- }
133
-
134
- remainingUrl = url.replace(first[1], '');
135
- } else if (new RegExp('^/').test(url)) {
136
- remainingUrl = url;
137
- } else {
138
- throw new Error(`Did not know what to do with url '${url}'`);
64
+ /**
65
+ * Performs requests to the downstream server
66
+ *
67
+ * @private - Do not call this method directly,
68
+ * it uses client-specific arguments and responses!
69
+ *
70
+ * @param {import('axios').AxiosRequestConfig} requestConfig
71
+ * @returns {Promise<import('axios').AxiosResponse>}
72
+ */
73
+ async request(requestConfig) {
74
+ const reqPromise = (0, axios_1.default)(requestConfig);
75
+ this._activeRequests.push(reqPromise);
76
+ try {
77
+ return await reqPromise;
78
+ }
79
+ finally {
80
+ lodash_1.default.pull(this._activeRequests, reqPromise);
81
+ }
139
82
  }
140
-
141
- const stripPrefixRe = new RegExp('^.*?(/(session|status).*)$');
142
-
143
- if (stripPrefixRe.test(remainingUrl)) {
144
- remainingUrl = stripPrefixRe.exec(remainingUrl)[1];
83
+ getActiveRequestsCount() {
84
+ return this._activeRequests.length;
145
85
  }
146
-
147
- if (!new RegExp(endpointRe).test(remainingUrl)) {
148
- remainingUrl = `/session/${this.sessionId}${remainingUrl}`;
86
+ cancelActiveRequests() {
87
+ this._activeRequests = [];
149
88
  }
150
-
151
- const requiresSessionId = this.endpointRequiresSessionId(remainingUrl);
152
-
153
- if (requiresSessionId && this.sessionId === null) {
154
- throw new Error('Trying to proxy a session command without session id');
89
+ endpointRequiresSessionId(endpoint) {
90
+ return !lodash_1.default.includes(['/session', '/sessions', '/status'], endpoint);
155
91
  }
156
-
157
- const sessionBaseRe = new RegExp('^/session/([^/]+)');
158
-
159
- if (sessionBaseRe.test(remainingUrl)) {
160
- if (this.sessionId === null) {
161
- throw new ReferenceError(`Session ID is not set, but saw a URL path referencing a session (${remainingUrl}). This may be a bug in your client.`);
162
- }
163
-
164
- const match = sessionBaseRe.exec(remainingUrl);
165
- remainingUrl = remainingUrl.replace(match[1], this.sessionId);
166
- } else if (requiresSessionId) {
167
- throw new Error(`Could not find :session section for url: ${remainingUrl}`);
92
+ set downstreamProtocol(value) {
93
+ this._downstreamProtocol = value;
94
+ this.protocolConverter.downstreamProtocol = value;
168
95
  }
169
-
170
- remainingUrl = remainingUrl.replace(/\/$/, '');
171
- return proxyBase + remainingUrl;
172
- }
173
-
174
- async proxy(url, method, body = null) {
175
- method = method.toUpperCase();
176
- const newUrl = this.getUrlForProxy(url);
177
-
178
- const truncateBody = content => _lodash.default.truncate(_lodash.default.isString(content) ? content : JSON.stringify(content), {
179
- length: _constants.MAX_LOG_BODY_LENGTH
180
- });
181
-
182
- const reqOpts = {
183
- url: newUrl,
184
- method,
185
- headers: {
186
- 'content-type': 'application/json; charset=utf-8',
187
- 'user-agent': 'appium',
188
- accept: 'application/json, */*'
189
- },
190
- proxy: false,
191
- timeout: this.timeout,
192
- httpAgent: this.httpAgent,
193
- httpsAgent: this.httpsAgent
194
- };
195
-
196
- if (_support.util.hasValue(body) && method !== 'GET') {
197
- if (typeof body !== 'object') {
198
- try {
199
- reqOpts.data = JSON.parse(body);
200
- } catch (e) {
201
- throw new Error(`Cannot interpret the request body as valid JSON: ${truncateBody(body)}`);
202
- }
203
- } else {
204
- reqOpts.data = body;
205
- }
96
+ get downstreamProtocol() {
97
+ return this._downstreamProtocol;
206
98
  }
207
-
208
- this.log.debug(`Proxying [${method} ${url || '/'}] to [${method} ${newUrl}] ` + (reqOpts.data ? `with body: ${truncateBody(reqOpts.data)}` : 'with no body'));
209
-
210
- const throwProxyError = error => {
211
- const err = new Error(`The request to ${url} has failed`);
212
- err.response = {
213
- data: error,
214
- status: 500
215
- };
216
- throw err;
217
- };
218
-
219
- let isResponseLogged = false;
220
-
221
- try {
222
- const {
223
- data,
224
- status,
225
- headers
226
- } = await this.request(reqOpts);
227
-
228
- if (!_lodash.default.isPlainObject(data)) {
229
- throwProxyError(data);
230
- }
231
-
232
- this.log.debug(`Got response with status ${status}: ${truncateBody(data)}`);
233
- isResponseLogged = true;
234
- const isSessionCreationRequest = /\/session$/.test(url) && method === 'POST';
235
-
236
- if (isSessionCreationRequest) {
237
- if (status === 200) {
238
- this.sessionId = data.sessionId || (data.value || {}).sessionId;
99
+ getUrlForProxy(url) {
100
+ if (url === '') {
101
+ url = '/';
239
102
  }
240
-
241
- this.downstreamProtocol = this.getProtocolFromResBody(data);
242
- this.log.info(`Determined the downstream protocol as '${this.downstreamProtocol}'`);
243
- }
244
-
245
- if (_lodash.default.has(data, 'status') && parseInt(data.status, 10) !== 0) {
246
- throwProxyError(data);
247
- }
248
-
249
- const res = {
250
- statusCode: status,
251
- headers,
252
- body: data
253
- };
254
- return [res, data];
255
- } catch (e) {
256
- var _e$response, _e$response2;
257
-
258
- let proxyErrorMsg = e.message;
259
-
260
- if (_support.util.hasValue(e.response)) {
261
- if (!isResponseLogged) {
262
- const error = truncateBody(e.response.data);
263
- this.log.info(_support.util.hasValue(e.response.status) ? `Got response with status ${e.response.status}: ${error}` : `Got response with unknown status: ${error}`);
103
+ const proxyBase = `${this.scheme}://${this.server}:${this.port}${this.base}`;
104
+ const endpointRe = '(/(session|status))';
105
+ let remainingUrl = '';
106
+ if (/^http/.test(url)) {
107
+ const first = new RegExp(`(https?://.+)${endpointRe}`).exec(url);
108
+ if (!first) {
109
+ throw new Error('Got a complete url but could not extract JWP endpoint');
110
+ }
111
+ remainingUrl = url.replace(first[1], '');
264
112
  }
265
- } else {
266
- proxyErrorMsg = `Could not proxy command to the remote server. Original error: ${e.message}`;
267
-
268
- if (COMPACT_ERROR_PATTERNS.some(p => p.test(e.message))) {
269
- this.log.info(e.message);
270
- } else {
271
- this.log.info(e.stack);
113
+ else if (new RegExp('^/').test(url)) {
114
+ remainingUrl = url;
272
115
  }
273
- }
274
-
275
- throw new _errors.errors.ProxyRequestError(proxyErrorMsg, (_e$response = e.response) === null || _e$response === void 0 ? void 0 : _e$response.data, (_e$response2 = e.response) === null || _e$response2 === void 0 ? void 0 : _e$response2.status);
276
- }
277
- }
278
-
279
- getProtocolFromResBody(resObj) {
280
- if (_lodash.default.isInteger(resObj.status)) {
281
- return MJSONWP;
282
- }
283
-
284
- if (!_lodash.default.isUndefined(resObj.value)) {
285
- return W3C;
286
- }
287
- }
288
-
289
- requestToCommandName(url, method) {
290
- const extractCommandName = pattern => {
291
- const pathMatch = pattern.exec(url);
292
-
293
- if (pathMatch) {
294
- return (0, _protocol.routeToCommandName)(pathMatch[1], method, this.reqBasePath);
295
- }
296
- };
297
-
298
- let commandName = (0, _protocol.routeToCommandName)(url, method, this.reqBasePath);
299
-
300
- if (!commandName && _lodash.default.includes(url, `${this.reqBasePath}/session/`)) {
301
- commandName = extractCommandName(new RegExp(`${_lodash.default.escapeRegExp(this.reqBasePath)}/session/[^/]+(.+)`));
116
+ else {
117
+ throw new Error(`Did not know what to do with url '${url}'`);
118
+ }
119
+ const stripPrefixRe = new RegExp('^.*?(/(session|status).*)$');
120
+ if (stripPrefixRe.test(remainingUrl)) {
121
+ remainingUrl = /** @type {RegExpExecArray} */ (stripPrefixRe.exec(remainingUrl))[1];
122
+ }
123
+ if (!new RegExp(endpointRe).test(remainingUrl)) {
124
+ remainingUrl = `/session/${this.sessionId}${remainingUrl}`;
125
+ }
126
+ const requiresSessionId = this.endpointRequiresSessionId(remainingUrl);
127
+ if (requiresSessionId && this.sessionId === null) {
128
+ throw new Error('Trying to proxy a session command without session id');
129
+ }
130
+ const sessionBaseRe = new RegExp('^/session/([^/]+)');
131
+ if (sessionBaseRe.test(remainingUrl)) {
132
+ if (this.sessionId === null) {
133
+ throw new ReferenceError(`Session ID is not set, but saw a URL path referencing a session (${remainingUrl}). This may be a bug in your client.`);
134
+ }
135
+ // we have something like /session/:id/foobar, so we need to replace
136
+ // the session id
137
+ const match = sessionBaseRe.exec(remainingUrl);
138
+ // TODO: if `requiresSessionId` is `false` and `sessionId` is `null`, this is a bug.
139
+ // are we sure `sessionId` is not `null`?
140
+ remainingUrl = remainingUrl.replace(
141
+ /** @type {RegExpExecArray} */ (match)[1],
142
+ /** @type {string} */ (this.sessionId));
143
+ }
144
+ else if (requiresSessionId) {
145
+ throw new Error(`Could not find :session section for url: ${remainingUrl}`);
146
+ }
147
+ remainingUrl = remainingUrl.replace(/\/$/, ''); // can't have trailing slashes
148
+ return proxyBase + remainingUrl;
302
149
  }
303
-
304
- if (!commandName && _lodash.default.includes(url, this.reqBasePath)) {
305
- commandName = extractCommandName(new RegExp(`${_lodash.default.escapeRegExp(this.reqBasePath)}(/.+)`));
150
+ async proxy(url, method, body = null) {
151
+ method = method.toUpperCase();
152
+ const newUrl = this.getUrlForProxy(url);
153
+ const truncateBody = (content) => lodash_1.default.truncate(lodash_1.default.isString(content) ? content : JSON.stringify(content), {
154
+ length: constants_1.MAX_LOG_BODY_LENGTH,
155
+ });
156
+ /** @type {import('axios').AxiosRequestConfig} */
157
+ const reqOpts = {
158
+ url: newUrl,
159
+ method,
160
+ headers: {
161
+ 'content-type': 'application/json; charset=utf-8',
162
+ 'user-agent': 'appium',
163
+ accept: 'application/json, */*',
164
+ },
165
+ proxy: false,
166
+ timeout: this.timeout,
167
+ httpAgent: this.httpAgent,
168
+ httpsAgent: this.httpsAgent,
169
+ };
170
+ // GET methods shouldn't have any body. Most servers are OK with this, but WebDriverAgent throws 400 errors
171
+ if (support_1.util.hasValue(body) && method !== 'GET') {
172
+ if (typeof body !== 'object') {
173
+ try {
174
+ reqOpts.data = JSON.parse(body);
175
+ }
176
+ catch (e) {
177
+ throw new Error(`Cannot interpret the request body as valid JSON: ${truncateBody(body)}`);
178
+ }
179
+ }
180
+ else {
181
+ reqOpts.data = body;
182
+ }
183
+ }
184
+ this.log.debug(`Proxying [${method} ${url || '/'}] to [${method} ${newUrl}] ` +
185
+ (reqOpts.data ? `with body: ${truncateBody(reqOpts.data)}` : 'with no body'));
186
+ const throwProxyError = (error) => {
187
+ const err = /** @type {ProxyError} */ (new Error(`The request to ${url} has failed`));
188
+ err.response = {
189
+ data: error,
190
+ status: 500,
191
+ };
192
+ throw err;
193
+ };
194
+ let isResponseLogged = false;
195
+ try {
196
+ const { data, status, headers } = await this.request(reqOpts);
197
+ // `data` might be really big
198
+ // Be careful while handling it to avoid memory leaks
199
+ if (!lodash_1.default.isPlainObject(data)) {
200
+ // The response should be a valid JSON object
201
+ // If it cannot be coerced to an object then the response is wrong
202
+ throwProxyError(data);
203
+ }
204
+ this.log.debug(`Got response with status ${status}: ${truncateBody(data)}`);
205
+ isResponseLogged = true;
206
+ const isSessionCreationRequest = /\/session$/.test(url) && method === 'POST';
207
+ if (isSessionCreationRequest) {
208
+ if (status === 200) {
209
+ this.sessionId = data.sessionId || (data.value || {}).sessionId;
210
+ }
211
+ this.downstreamProtocol = this.getProtocolFromResBody(data);
212
+ this.log.info(`Determined the downstream protocol as '${this.downstreamProtocol}'`);
213
+ }
214
+ if (lodash_1.default.has(data, 'status') && parseInt(data.status, 10) !== 0) {
215
+ // Some servers, like chromedriver may return response code 200 for non-zero JSONWP statuses
216
+ throwProxyError(data);
217
+ }
218
+ const res = { statusCode: status, headers, body: data };
219
+ return [res, data];
220
+ }
221
+ catch (e) {
222
+ // We only consider an error unexpected if this was not
223
+ // an async request module error or if the response cannot be cast to
224
+ // a valid JSON
225
+ let proxyErrorMsg = e.message;
226
+ if (support_1.util.hasValue(e.response)) {
227
+ if (!isResponseLogged) {
228
+ const error = truncateBody(e.response.data);
229
+ this.log.info(support_1.util.hasValue(e.response.status)
230
+ ? `Got response with status ${e.response.status}: ${error}`
231
+ : `Got response with unknown status: ${error}`);
232
+ }
233
+ }
234
+ else {
235
+ proxyErrorMsg = `Could not proxy command to the remote server. Original error: ${e.message}`;
236
+ if (COMPACT_ERROR_PATTERNS.some((p) => p.test(e.message))) {
237
+ this.log.info(e.message);
238
+ }
239
+ else {
240
+ this.log.info(e.stack);
241
+ }
242
+ }
243
+ throw new errors_1.errors.ProxyRequestError(proxyErrorMsg, e.response?.data, e.response?.status);
244
+ }
306
245
  }
307
-
308
- return commandName;
309
- }
310
-
311
- async proxyCommand(url, method, body = null) {
312
- const commandName = this.requestToCommandName(url, method);
313
-
314
- if (!commandName) {
315
- return await this.proxy(url, method, body);
246
+ getProtocolFromResBody(resObj) {
247
+ if (lodash_1.default.isInteger(resObj.status)) {
248
+ return MJSONWP;
249
+ }
250
+ if (!lodash_1.default.isUndefined(resObj.value)) {
251
+ return W3C;
252
+ }
316
253
  }
317
-
318
- this.log.debug(`Matched '${url}' to command name '${commandName}'`);
319
- return await this.protocolConverter.convertAndProxy(commandName, url, method, body);
320
- }
321
-
322
- async command(url, method, body = null) {
323
- let response;
324
- let resBodyObj;
325
-
326
- try {
327
- [response, resBodyObj] = await this.proxyCommand(url, method, body);
328
- } catch (err) {
329
- if ((0, _errors.isErrorType)(err, _errors.errors.ProxyRequestError)) {
330
- throw err.getActualError();
331
- }
332
-
333
- throw new _errors.errors.UnknownError(err.message);
254
+ /**
255
+ *
256
+ * @param {string} url
257
+ * @param {import('@appium/types').HTTPMethod} method
258
+ * @returns {string|undefined}
259
+ */
260
+ requestToCommandName(url, method) {
261
+ /**
262
+ *
263
+ * @param {RegExp} pattern
264
+ * @returns {string|undefined}
265
+ */
266
+ const extractCommandName = (pattern) => {
267
+ const pathMatch = pattern.exec(url);
268
+ if (pathMatch) {
269
+ return (0, protocol_1.routeToCommandName)(pathMatch[1], method, this.reqBasePath);
270
+ }
271
+ };
272
+ let commandName = (0, protocol_1.routeToCommandName)(url, method, this.reqBasePath);
273
+ if (!commandName && lodash_1.default.includes(url, `${this.reqBasePath}/session/`)) {
274
+ commandName = extractCommandName(new RegExp(`${lodash_1.default.escapeRegExp(this.reqBasePath)}/session/[^/]+(.+)`));
275
+ }
276
+ if (!commandName && lodash_1.default.includes(url, this.reqBasePath)) {
277
+ commandName = extractCommandName(new RegExp(`${lodash_1.default.escapeRegExp(this.reqBasePath)}(/.+)`));
278
+ }
279
+ return commandName;
334
280
  }
335
-
336
- const protocol = this.getProtocolFromResBody(resBodyObj);
337
-
338
- if (protocol === MJSONWP) {
339
- if (response.statusCode === 200 && resBodyObj.status === 0) {
340
- return resBodyObj.value;
341
- }
342
-
343
- const status = parseInt(resBodyObj.status, 10);
344
-
345
- if (!isNaN(status) && status !== 0) {
346
- let message = resBodyObj.value;
347
-
348
- if (_lodash.default.has(message, 'message')) {
349
- message = message.message;
281
+ async proxyCommand(url, method, body = null) {
282
+ const commandName = this.requestToCommandName(url, method);
283
+ if (!commandName) {
284
+ return await this.proxy(url, method, body);
350
285
  }
351
-
352
- throw (0, _errors.errorFromMJSONWPStatusCode)(status, _lodash.default.isEmpty(message) ? (0, _status.getSummaryByCode)(status) : message);
353
- }
354
- } else if (protocol === W3C) {
355
- if (response.statusCode < 300) {
356
- return resBodyObj.value;
357
- }
358
-
359
- if (_lodash.default.isPlainObject(resBodyObj.value) && resBodyObj.value.error) {
360
- throw (0, _errors.errorFromW3CJsonCode)(resBodyObj.value.error, resBodyObj.value.message, resBodyObj.value.stacktrace);
361
- }
362
- } else if (response.statusCode === 200) {
363
- return resBodyObj;
286
+ this.log.debug(`Matched '${url}' to command name '${commandName}'`);
287
+ return await this.protocolConverter.convertAndProxy(commandName, url, method, body);
364
288
  }
365
-
366
- throw new _errors.errors.UnknownError(`Did not know what to do with response code '${response.statusCode}' ` + `and response body '${_lodash.default.truncate(JSON.stringify(resBodyObj), {
367
- length: 300
368
- })}'`);
369
- }
370
-
371
- getSessionIdFromUrl(url) {
372
- const match = url.match(/\/session\/([^/]+)/);
373
- return match ? match[1] : null;
374
- }
375
-
376
- async proxyReqRes(req, res) {
377
- let statusCode;
378
- let resBodyObj;
379
-
380
- try {
381
- let response;
382
- [response, resBodyObj] = await this.proxyCommand(req.originalUrl, req.method, req.body);
383
- res.headers = response.headers;
384
- statusCode = response.statusCode;
385
- } catch (err) {
386
- [statusCode, resBodyObj] = (0, _errors.getResponseForW3CError)((0, _errors.isErrorType)(err, _errors.errors.ProxyRequestError) ? err.getActualError() : err);
289
+ async command(url, method, body = null) {
290
+ let response;
291
+ let resBodyObj;
292
+ try {
293
+ [response, resBodyObj] = await this.proxyCommand(url, method, body);
294
+ }
295
+ catch (err) {
296
+ if ((0, errors_1.isErrorType)(err, errors_1.errors.ProxyRequestError)) {
297
+ throw err.getActualError();
298
+ }
299
+ throw new errors_1.errors.UnknownError(err.message);
300
+ }
301
+ const protocol = this.getProtocolFromResBody(resBodyObj);
302
+ if (protocol === MJSONWP) {
303
+ // Got response in MJSONWP format
304
+ if (response.statusCode === 200 && resBodyObj.status === 0) {
305
+ return resBodyObj.value;
306
+ }
307
+ const status = parseInt(resBodyObj.status, 10);
308
+ if (!isNaN(status) && status !== 0) {
309
+ let message = resBodyObj.value;
310
+ if (lodash_1.default.has(message, 'message')) {
311
+ message = message.message;
312
+ }
313
+ throw (0, errors_1.errorFromMJSONWPStatusCode)(status, lodash_1.default.isEmpty(message) ? (0, status_1.getSummaryByCode)(status) : message);
314
+ }
315
+ }
316
+ else if (protocol === W3C) {
317
+ // Got response in W3C format
318
+ if (response.statusCode < 300) {
319
+ return resBodyObj.value;
320
+ }
321
+ if (lodash_1.default.isPlainObject(resBodyObj.value) && resBodyObj.value.error) {
322
+ throw (0, errors_1.errorFromW3CJsonCode)(resBodyObj.value.error, resBodyObj.value.message, resBodyObj.value.stacktrace);
323
+ }
324
+ }
325
+ else if (response.statusCode === 200) {
326
+ // Unknown protocol. Keeping it because of the backward compatibility
327
+ return resBodyObj;
328
+ }
329
+ throw new errors_1.errors.UnknownError(`Did not know what to do with response code '${response.statusCode}' ` +
330
+ `and response body '${lodash_1.default.truncate(JSON.stringify(resBodyObj), {
331
+ length: 300,
332
+ })}'`);
387
333
  }
388
-
389
- res.set('content-type', 'application/json; charset=utf-8');
390
-
391
- if (!_lodash.default.isPlainObject(resBodyObj)) {
392
- const error = new _errors.errors.UnknownError(`The downstream server response with the status code ${statusCode} is not a valid JSON object: ` + _lodash.default.truncate(`${resBodyObj}`, {
393
- length: 300
394
- }));
395
- [statusCode, resBodyObj] = (0, _errors.getResponseForW3CError)(error);
334
+ getSessionIdFromUrl(url) {
335
+ const match = url.match(/\/session\/([^/]+)/);
336
+ return match ? match[1] : null;
396
337
  }
397
-
398
- if (_lodash.default.has(resBodyObj, 'sessionId')) {
399
- const reqSessionId = this.getSessionIdFromUrl(req.originalUrl);
400
-
401
- if (reqSessionId) {
402
- this.log.info(`Replacing sessionId ${resBodyObj.sessionId} with ${reqSessionId}`);
403
- resBodyObj.sessionId = reqSessionId;
404
- } else if (this.sessionId) {
405
- this.log.info(`Replacing sessionId ${resBodyObj.sessionId} with ${this.sessionId}`);
406
- resBodyObj.sessionId = this.sessionId;
407
- }
338
+ async proxyReqRes(req, res) {
339
+ // ! this method must not throw any exceptions
340
+ // ! make sure to call res.send before return
341
+ let statusCode;
342
+ let resBodyObj;
343
+ try {
344
+ let response;
345
+ [response, resBodyObj] = await this.proxyCommand(req.originalUrl, req.method, req.body);
346
+ res.headers = response.headers;
347
+ statusCode = response.statusCode;
348
+ }
349
+ catch (err) {
350
+ [statusCode, resBodyObj] = (0, errors_1.getResponseForW3CError)((0, errors_1.isErrorType)(err, errors_1.errors.ProxyRequestError) ? err.getActualError() : err);
351
+ }
352
+ res.set('content-type', 'application/json; charset=utf-8');
353
+ if (!lodash_1.default.isPlainObject(resBodyObj)) {
354
+ const error = new errors_1.errors.UnknownError(`The downstream server response with the status code ${statusCode} is not a valid JSON object: ` +
355
+ lodash_1.default.truncate(`${resBodyObj}`, { length: 300 }));
356
+ [statusCode, resBodyObj] = (0, errors_1.getResponseForW3CError)(error);
357
+ }
358
+ // if the proxied response contains a sessionId that the downstream
359
+ // driver has generated, we don't want to return that to the client.
360
+ // Instead, return the id from the request or from current session
361
+ if (lodash_1.default.has(resBodyObj, 'sessionId')) {
362
+ const reqSessionId = this.getSessionIdFromUrl(req.originalUrl);
363
+ if (reqSessionId) {
364
+ this.log.info(`Replacing sessionId ${resBodyObj.sessionId} with ${reqSessionId}`);
365
+ resBodyObj.sessionId = reqSessionId;
366
+ }
367
+ else if (this.sessionId) {
368
+ this.log.info(`Replacing sessionId ${resBodyObj.sessionId} with ${this.sessionId}`);
369
+ resBodyObj.sessionId = this.sessionId;
370
+ }
371
+ }
372
+ resBodyObj.value = (0, helpers_1.formatResponseValue)(resBodyObj.value);
373
+ res.status(statusCode).send(JSON.stringify((0, helpers_1.formatStatus)(resBodyObj)));
408
374
  }
409
-
410
- resBodyObj.value = (0, _helpers.formatResponseValue)(resBodyObj.value);
411
- res.status(statusCode).send(JSON.stringify((0, _helpers.formatStatus)(resBodyObj)));
412
- }
413
-
414
375
  }
415
-
416
376
  exports.JWProxy = JWProxy;
417
- var _default = JWProxy;
418
- exports.default = _default;
419
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["DEFAULT_LOG","logger","getLogger","DEFAULT_REQUEST_TIMEOUT","COMPACT_ERROR_PATTERNS","MJSONWP","W3C","PROTOCOLS","ALLOWED_OPTS","JWProxy","scheme","server","port","base","reqBasePath","sessionId","timeout","constructor","opts","_","pick","options","defaults","omit","DEFAULT_BASE_PATH","toLowerCase","Object","assign","_activeRequests","_downstreamProtocol","agentOpts","keepAlive","maxSockets","maxFreeSockets","httpAgent","http","Agent","httpsAgent","https","protocolConverter","ProtocolConverter","proxy","bind","log","_log","request","requestConfig","reqPromise","axios","push","pull","getActiveRequestsCount","length","cancelActiveRequests","endpointRequiresSessionId","endpoint","includes","downstreamProtocol","value","getUrlForProxy","url","proxyBase","endpointRe","remainingUrl","test","first","RegExp","exec","Error","replace","stripPrefixRe","requiresSessionId","sessionBaseRe","ReferenceError","match","method","body","toUpperCase","newUrl","truncateBody","content","truncate","isString","JSON","stringify","MAX_LOG_BODY_LENGTH","reqOpts","headers","accept","util","hasValue","data","parse","e","debug","throwProxyError","error","err","response","status","isResponseLogged","isPlainObject","isSessionCreationRequest","getProtocolFromResBody","info","has","parseInt","res","statusCode","proxyErrorMsg","message","some","p","stack","errors","ProxyRequestError","resObj","isInteger","isUndefined","requestToCommandName","extractCommandName","pattern","pathMatch","routeToCommandName","commandName","escapeRegExp","proxyCommand","convertAndProxy","command","resBodyObj","isErrorType","getActualError","UnknownError","protocol","isNaN","errorFromMJSONWPStatusCode","isEmpty","getSummaryByCode","errorFromW3CJsonCode","stacktrace","getSessionIdFromUrl","proxyReqRes","req","originalUrl","getResponseForW3CError","set","reqSessionId","formatResponseValue","send","formatStatus"],"sources":["../../../lib/jsonwp-proxy/proxy.js"],"sourcesContent":["import _ from 'lodash';\nimport {logger, util} from '@appium/support';\nimport axios from 'axios';\nimport {getSummaryByCode} from '../jsonwp-status/status';\nimport {\n  errors,\n  isErrorType,\n  errorFromMJSONWPStatusCode,\n  errorFromW3CJsonCode,\n  getResponseForW3CError,\n} from '../protocol/errors';\nimport {routeToCommandName} from '../protocol';\nimport {MAX_LOG_BODY_LENGTH, DEFAULT_BASE_PATH, PROTOCOLS} from '../constants';\nimport ProtocolConverter from './protocol-converter';\nimport {formatResponseValue, formatStatus} from '../protocol/helpers';\nimport http from 'http';\nimport https from 'https';\n\nconst DEFAULT_LOG = logger.getLogger('WD Proxy');\nconst DEFAULT_REQUEST_TIMEOUT = 240000;\nconst COMPACT_ERROR_PATTERNS = [/\\bECONNREFUSED\\b/, /socket hang up/];\n\nconst {MJSONWP, W3C} = PROTOCOLS;\n\nconst ALLOWED_OPTS = [\n  'scheme',\n  'server',\n  'port',\n  'base',\n  'reqBasePath',\n  'sessionId',\n  'timeout',\n  'log',\n  'keepAlive',\n];\n\nclass JWProxy {\n  /** @type {string} */\n  scheme;\n  /** @type {string} */\n  server;\n  /** @type {number} */\n  port;\n  /** @type {string} */\n  base;\n  /** @type {string} */\n  reqBasePath;\n  /** @type {string?} */\n  sessionId;\n  /** @type {number} */\n  timeout;\n\n  constructor(opts = {}) {\n    opts = _.pick(opts, ALLOWED_OPTS);\n\n    // omit 'log' in the defaults assignment here because 'log' is a getter and we are going to set\n    // it to this._log (which lies behind the getter) further down\n    const options = _.defaults(_.omit(opts, 'log'), {\n      scheme: 'http',\n      server: 'localhost',\n      port: 4444,\n      base: DEFAULT_BASE_PATH,\n      reqBasePath: DEFAULT_BASE_PATH,\n      sessionId: null,\n      timeout: DEFAULT_REQUEST_TIMEOUT,\n    });\n    options.scheme = options.scheme.toLowerCase();\n    Object.assign(this, options);\n\n    this._activeRequests = [];\n    this._downstreamProtocol = null;\n    const agentOpts = {\n      keepAlive: opts.keepAlive ?? true,\n      maxSockets: 10,\n      maxFreeSockets: 5,\n    };\n    this.httpAgent = new http.Agent(agentOpts);\n    this.httpsAgent = new https.Agent(agentOpts);\n    this.protocolConverter = new ProtocolConverter(this.proxy.bind(this), opts.log);\n    this._log = opts.log;\n  }\n\n  get log() {\n    return this._log ?? DEFAULT_LOG;\n  }\n\n  /**\n   * Performs requests to the downstream server\n   *\n   * @private - Do not call this method directly,\n   * it uses client-specific arguments and responses!\n   *\n   * @param {import('axios').AxiosRequestConfig} requestConfig\n   * @returns {Promise<import('axios').AxiosResponse>}\n   */\n  async request(requestConfig) {\n    const reqPromise = axios(requestConfig);\n    this._activeRequests.push(reqPromise);\n    try {\n      return await reqPromise;\n    } finally {\n      _.pull(this._activeRequests, reqPromise);\n    }\n  }\n\n  getActiveRequestsCount() {\n    return this._activeRequests.length;\n  }\n\n  cancelActiveRequests() {\n    this._activeRequests = [];\n  }\n\n  endpointRequiresSessionId(endpoint) {\n    return !_.includes(['/session', '/sessions', '/status'], endpoint);\n  }\n\n  set downstreamProtocol(value) {\n    this._downstreamProtocol = value;\n    this.protocolConverter.downstreamProtocol = value;\n  }\n\n  get downstreamProtocol() {\n    return this._downstreamProtocol;\n  }\n\n  getUrlForProxy(url) {\n    if (url === '') {\n      url = '/';\n    }\n    const proxyBase = `${this.scheme}://${this.server}:${this.port}${this.base}`;\n    const endpointRe = '(/(session|status))';\n    let remainingUrl = '';\n    if (/^http/.test(url)) {\n      const first = new RegExp(`(https?://.+)${endpointRe}`).exec(url);\n      if (!first) {\n        throw new Error('Got a complete url but could not extract JWP endpoint');\n      }\n      remainingUrl = url.replace(first[1], '');\n    } else if (new RegExp('^/').test(url)) {\n      remainingUrl = url;\n    } else {\n      throw new Error(`Did not know what to do with url '${url}'`);\n    }\n\n    const stripPrefixRe = new RegExp('^.*?(/(session|status).*)$');\n    if (stripPrefixRe.test(remainingUrl)) {\n      remainingUrl = /** @type {RegExpExecArray} */ (stripPrefixRe.exec(remainingUrl))[1];\n    }\n\n    if (!new RegExp(endpointRe).test(remainingUrl)) {\n      remainingUrl = `/session/${this.sessionId}${remainingUrl}`;\n    }\n\n    const requiresSessionId = this.endpointRequiresSessionId(remainingUrl);\n\n    if (requiresSessionId && this.sessionId === null) {\n      throw new Error('Trying to proxy a session command without session id');\n    }\n\n    const sessionBaseRe = new RegExp('^/session/([^/]+)');\n    if (sessionBaseRe.test(remainingUrl)) {\n      if (this.sessionId === null) {\n        throw new ReferenceError(\n          `Session ID is not set, but saw a URL path referencing a session (${remainingUrl}). This may be a bug in your client.`\n        );\n      }\n      // we have something like /session/:id/foobar, so we need to replace\n      // the session id\n      const match = sessionBaseRe.exec(remainingUrl);\n      // TODO: if `requiresSessionId` is `false` and `sessionId` is `null`, this is a bug.\n      // are we sure `sessionId` is not `null`?\n      remainingUrl = remainingUrl.replace(\n        /** @type {RegExpExecArray} */ (match)[1],\n        /** @type {string} */ (this.sessionId)\n      );\n    } else if (requiresSessionId) {\n      throw new Error(`Could not find :session section for url: ${remainingUrl}`);\n    }\n    remainingUrl = remainingUrl.replace(/\\/$/, ''); // can't have trailing slashes\n\n    return proxyBase + remainingUrl;\n  }\n\n  async proxy(url, method, body = null) {\n    method = method.toUpperCase();\n    const newUrl = this.getUrlForProxy(url);\n    const truncateBody = (content) =>\n      _.truncate(_.isString(content) ? content : JSON.stringify(content), {\n        length: MAX_LOG_BODY_LENGTH,\n      });\n    /** @type {import('axios').AxiosRequestConfig} */\n    const reqOpts = {\n      url: newUrl,\n      method,\n      headers: {\n        'content-type': 'application/json; charset=utf-8',\n        'user-agent': 'appium',\n        accept: 'application/json, */*',\n      },\n      proxy: false,\n      timeout: this.timeout,\n      httpAgent: this.httpAgent,\n      httpsAgent: this.httpsAgent,\n    };\n    // GET methods shouldn't have any body. Most servers are OK with this, but WebDriverAgent throws 400 errors\n    if (util.hasValue(body) && method !== 'GET') {\n      if (typeof body !== 'object') {\n        try {\n          reqOpts.data = JSON.parse(body);\n        } catch (e) {\n          throw new Error(`Cannot interpret the request body as valid JSON: ${truncateBody(body)}`);\n        }\n      } else {\n        reqOpts.data = body;\n      }\n    }\n\n    this.log.debug(\n      `Proxying [${method} ${url || '/'}] to [${method} ${newUrl}] ` +\n        (reqOpts.data ? `with body: ${truncateBody(reqOpts.data)}` : 'with no body')\n    );\n\n    const throwProxyError = (error) => {\n      const err = /** @type {ProxyError} */ (new Error(`The request to ${url} has failed`));\n      err.response = {\n        data: error,\n        status: 500,\n      };\n      throw err;\n    };\n    let isResponseLogged = false;\n    try {\n      const {data, status, headers} = await this.request(reqOpts);\n      // `data` might be really big\n      // Be careful while handling it to avoid memory leaks\n      if (!_.isPlainObject(data)) {\n        // The response should be a valid JSON object\n        // If it cannot be coerced to an object then the response is wrong\n        throwProxyError(data);\n      }\n      this.log.debug(`Got response with status ${status}: ${truncateBody(data)}`);\n      isResponseLogged = true;\n      const isSessionCreationRequest = /\\/session$/.test(url) && method === 'POST';\n      if (isSessionCreationRequest) {\n        if (status === 200) {\n          this.sessionId = data.sessionId || (data.value || {}).sessionId;\n        }\n        this.downstreamProtocol = this.getProtocolFromResBody(data);\n        this.log.info(`Determined the downstream protocol as '${this.downstreamProtocol}'`);\n      }\n      if (_.has(data, 'status') && parseInt(data.status, 10) !== 0) {\n        // Some servers, like chromedriver may return response code 200 for non-zero JSONWP statuses\n        throwProxyError(data);\n      }\n      const res = {statusCode: status, headers, body: data};\n      return [res, data];\n    } catch (e) {\n      // We only consider an error unexpected if this was not\n      // an async request module error or if the response cannot be cast to\n      // a valid JSON\n      let proxyErrorMsg = e.message;\n      if (util.hasValue(e.response)) {\n        if (!isResponseLogged) {\n          const error = truncateBody(e.response.data);\n          this.log.info(\n            util.hasValue(e.response.status)\n              ? `Got response with status ${e.response.status}: ${error}`\n              : `Got response with unknown status: ${error}`\n          );\n        }\n      } else {\n        proxyErrorMsg = `Could not proxy command to the remote server. Original error: ${e.message}`;\n        if (COMPACT_ERROR_PATTERNS.some((p) => p.test(e.message))) {\n          this.log.info(e.message);\n        } else {\n          this.log.info(e.stack);\n        }\n      }\n      throw new errors.ProxyRequestError(proxyErrorMsg, e.response?.data, e.response?.status);\n    }\n  }\n\n  getProtocolFromResBody(resObj) {\n    if (_.isInteger(resObj.status)) {\n      return MJSONWP;\n    }\n    if (!_.isUndefined(resObj.value)) {\n      return W3C;\n    }\n  }\n\n  /**\n   *\n   * @param {string} url\n   * @param {import('@appium/types').HTTPMethod} method\n   * @returns {string|undefined}\n   */\n  requestToCommandName(url, method) {\n    /**\n     *\n     * @param {RegExp} pattern\n     * @returns {string|undefined}\n     */\n    const extractCommandName = (pattern) => {\n      const pathMatch = pattern.exec(url);\n      if (pathMatch) {\n        return routeToCommandName(pathMatch[1], method, this.reqBasePath);\n      }\n    };\n    let commandName = routeToCommandName(url, method, this.reqBasePath);\n    if (!commandName && _.includes(url, `${this.reqBasePath}/session/`)) {\n      commandName = extractCommandName(\n        new RegExp(`${_.escapeRegExp(this.reqBasePath)}/session/[^/]+(.+)`)\n      );\n    }\n    if (!commandName && _.includes(url, this.reqBasePath)) {\n      commandName = extractCommandName(new RegExp(`${_.escapeRegExp(this.reqBasePath)}(/.+)`));\n    }\n    return commandName;\n  }\n\n  async proxyCommand(url, method, body = null) {\n    const commandName = this.requestToCommandName(url, method);\n    if (!commandName) {\n      return await this.proxy(url, method, body);\n    }\n    this.log.debug(`Matched '${url}' to command name '${commandName}'`);\n\n    return await this.protocolConverter.convertAndProxy(commandName, url, method, body);\n  }\n\n  async command(url, method, body = null) {\n    let response;\n    let resBodyObj;\n    try {\n      [response, resBodyObj] = await this.proxyCommand(url, method, body);\n    } catch (err) {\n      if (isErrorType(err, errors.ProxyRequestError)) {\n        throw err.getActualError();\n      }\n      throw new errors.UnknownError(err.message);\n    }\n    const protocol = this.getProtocolFromResBody(resBodyObj);\n    if (protocol === MJSONWP) {\n      // Got response in MJSONWP format\n      if (response.statusCode === 200 && resBodyObj.status === 0) {\n        return resBodyObj.value;\n      }\n      const status = parseInt(resBodyObj.status, 10);\n      if (!isNaN(status) && status !== 0) {\n        let message = resBodyObj.value;\n        if (_.has(message, 'message')) {\n          message = message.message;\n        }\n        throw errorFromMJSONWPStatusCode(\n          status,\n          _.isEmpty(message) ? getSummaryByCode(status) : message\n        );\n      }\n    } else if (protocol === W3C) {\n      // Got response in W3C format\n      if (response.statusCode < 300) {\n        return resBodyObj.value;\n      }\n      if (_.isPlainObject(resBodyObj.value) && resBodyObj.value.error) {\n        throw errorFromW3CJsonCode(\n          resBodyObj.value.error,\n          resBodyObj.value.message,\n          resBodyObj.value.stacktrace\n        );\n      }\n    } else if (response.statusCode === 200) {\n      // Unknown protocol. Keeping it because of the backward compatibility\n      return resBodyObj;\n    }\n    throw new errors.UnknownError(\n      `Did not know what to do with response code '${response.statusCode}' ` +\n        `and response body '${_.truncate(JSON.stringify(resBodyObj), {\n          length: 300,\n        })}'`\n    );\n  }\n\n  getSessionIdFromUrl(url) {\n    const match = url.match(/\\/session\\/([^/]+)/);\n    return match ? match[1] : null;\n  }\n\n  async proxyReqRes(req, res) {\n    // ! this method must not throw any exceptions\n    // ! make sure to call res.send before return\n    let statusCode;\n    let resBodyObj;\n    try {\n      let response;\n      [response, resBodyObj] = await this.proxyCommand(req.originalUrl, req.method, req.body);\n      res.headers = response.headers;\n      statusCode = response.statusCode;\n    } catch (err) {\n      [statusCode, resBodyObj] = getResponseForW3CError(\n        isErrorType(err, errors.ProxyRequestError) ? err.getActualError() : err\n      );\n    }\n    res.set('content-type', 'application/json; charset=utf-8');\n    if (!_.isPlainObject(resBodyObj)) {\n      const error = new errors.UnknownError(\n        `The downstream server response with the status code ${statusCode} is not a valid JSON object: ` +\n          _.truncate(`${resBodyObj}`, {length: 300})\n      );\n      [statusCode, resBodyObj] = getResponseForW3CError(error);\n    }\n\n    // if the proxied response contains a sessionId that the downstream\n    // driver has generated, we don't want to return that to the client.\n    // Instead, return the id from the request or from current session\n    if (_.has(resBodyObj, 'sessionId')) {\n      const reqSessionId = this.getSessionIdFromUrl(req.originalUrl);\n      if (reqSessionId) {\n        this.log.info(`Replacing sessionId ${resBodyObj.sessionId} with ${reqSessionId}`);\n        resBodyObj.sessionId = reqSessionId;\n      } else if (this.sessionId) {\n        this.log.info(`Replacing sessionId ${resBodyObj.sessionId} with ${this.sessionId}`);\n        resBodyObj.sessionId = this.sessionId;\n      }\n    }\n    resBodyObj.value = formatResponseValue(resBodyObj.value);\n    res.status(statusCode).send(JSON.stringify(formatStatus(resBodyObj)));\n  }\n}\n\nexport {JWProxy};\nexport default JWProxy;\n\n/**\n * @typedef {Error & {response: {data: import('type-fest').JsonObject, status: import('http-status-codes').StatusCodes}}} ProxyError\n */\n"],"mappings":"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"}
377
+ exports.default = JWProxy;
378
+ /**
379
+ * @typedef {Error & {response: {data: import('type-fest').JsonObject, status: import('http-status-codes').StatusCodes}}} ProxyError
380
+ */
381
+ //# sourceMappingURL=proxy.js.map