@appium/base-driver 8.5.7 → 8.6.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 (62) hide show
  1. package/build/lib/basedriver/capabilities.js +2 -2
  2. package/build/lib/basedriver/commands/log.js +2 -4
  3. package/build/lib/basedriver/core.d.ts +8 -0
  4. package/build/lib/basedriver/core.d.ts.map +1 -1
  5. package/build/lib/basedriver/core.js +5 -1
  6. package/build/lib/basedriver/helpers.js +2 -2
  7. package/build/lib/express/server.d.ts +69 -7
  8. package/build/lib/express/server.d.ts.map +1 -1
  9. package/build/lib/express/server.js +18 -16
  10. package/build/lib/express/websocket.d.ts +12 -30
  11. package/build/lib/express/websocket.d.ts.map +1 -1
  12. package/build/lib/express/websocket.js +6 -4
  13. package/build/lib/index.d.ts +11 -180
  14. package/build/lib/index.js +107 -80
  15. package/build/lib/jsonwp-proxy/protocol-converter.js +2 -4
  16. package/build/lib/jsonwp-proxy/proxy.js +3 -7
  17. package/build/lib/protocol/protocol.d.ts +6 -4
  18. package/build/lib/protocol/protocol.d.ts.map +1 -1
  19. package/build/lib/protocol/protocol.js +5 -7
  20. package/build/lib/protocol/routes.js +2 -6
  21. package/build/tsconfig.tsbuildinfo +1 -1
  22. package/lib/basedriver/core.js +11 -0
  23. package/lib/express/server.js +124 -36
  24. package/lib/express/websocket.js +13 -11
  25. package/lib/index.js +16 -62
  26. package/lib/protocol/protocol.js +8 -1
  27. package/package.json +16 -14
  28. package/build/test/basedriver/README.md +0 -5
  29. package/build/test/basedriver/driver-e2e-tests.js +0 -413
  30. package/build/test/basedriver/driver-tests.js +0 -572
  31. package/build/test/basedriver/index.js +0 -26
  32. package/build/test/e2e/basedriver/driver.e2e.spec.js +0 -15
  33. package/build/test/e2e/basedriver/helpers.e2e.spec.js +0 -192
  34. package/build/test/e2e/basedriver/websockets.e2e.spec.js +0 -87
  35. package/build/test/e2e/express/server.e2e.spec.js +0 -159
  36. package/build/test/e2e/jsonwp-proxy/proxy.e2e.spec.js +0 -59
  37. package/build/test/e2e/protocol/fake-driver.js +0 -163
  38. package/build/test/e2e/protocol/helpers.js +0 -25
  39. package/build/test/e2e/protocol/protocol.e2e.spec.js +0 -1186
  40. package/build/test/helpers.js +0 -55
  41. package/build/test/unit/basedriver/capabilities.spec.js +0 -672
  42. package/build/test/unit/basedriver/capability.spec.js +0 -353
  43. package/build/test/unit/basedriver/commands/event.spec.js +0 -110
  44. package/build/test/unit/basedriver/commands/log.spec.js +0 -92
  45. package/build/test/unit/basedriver/device-settings.spec.js +0 -97
  46. package/build/test/unit/basedriver/driver.spec.js +0 -15
  47. package/build/test/unit/basedriver/helpers.spec.js +0 -151
  48. package/build/test/unit/basedriver/timeout.spec.js +0 -135
  49. package/build/test/unit/express/server.spec.js +0 -155
  50. package/build/test/unit/express/static.spec.js +0 -26
  51. package/build/test/unit/jsonwp-proxy/mock-request.js +0 -91
  52. package/build/test/unit/jsonwp-proxy/protocol-converter.spec.js +0 -171
  53. package/build/test/unit/jsonwp-proxy/proxy.spec.js +0 -292
  54. package/build/test/unit/jsonwp-proxy/url.spec.js +0 -165
  55. package/build/test/unit/jsonwp-status/status.spec.js +0 -34
  56. package/build/test/unit/protocol/errors.spec.js +0 -390
  57. package/build/test/unit/protocol/routes.spec.js +0 -80
  58. package/build/test/unit/protocol/validator.spec.js +0 -149
  59. package/test/basedriver/README.md +0 -5
  60. package/test/basedriver/driver-e2e-tests.js +0 -386
  61. package/test/basedriver/driver-tests.js +0 -624
  62. package/test/basedriver/index.js +0 -6
@@ -19,13 +19,75 @@ export type StartServerOpts = {
19
19
  */
20
20
  hostname?: string | undefined;
21
21
  };
22
- export function server(opts?: {}): Promise<any>;
23
- export function configureServer({ app, addRoutes, allowCors, basePath, extraMethodMap, }: {
24
- app: any;
25
- addRoutes: any;
22
+ export type AppiumServer = import('@appium/types').AppiumServer;
23
+ export type AppiumServerSocket = import('@appium/types').AppiumServerSocket;
24
+ export type MethodMap = import('@appium/types').MethodMap;
25
+ /**
26
+ * Options for {@linkcode configureHttp }
27
+ */
28
+ export type ConfigureHttpOpts = {
29
+ /**
30
+ * - HTTP server instance
31
+ */
32
+ httpServer: import('http').Server;
33
+ /**
34
+ * - Rejection function from `Promise` constructor
35
+ */
36
+ reject: (error?: any) => void;
37
+ /**
38
+ * - Keep-alive timeout in milliseconds
39
+ */
40
+ keepAliveTimeout: number;
41
+ };
42
+ /**
43
+ * Options for {@linkcode server }
44
+ */
45
+ export type ServerOpts = {
46
+ routeConfiguringFunction: RouteConfiguringFunction;
47
+ port: number;
48
+ hostname?: string | undefined;
26
49
  allowCors?: boolean | undefined;
27
50
  basePath?: string | undefined;
28
- extraMethodMap?: {} | undefined;
29
- }): void;
30
- export function normalizeBasePath(basePath: any): any;
51
+ extraMethodMap?: MethodMap | undefined;
52
+ serverUpdaters?: import("@appium/types").UpdateServerCallback[] | undefined;
53
+ keepAliveTimeout?: number | undefined;
54
+ };
55
+ /**
56
+ * A function which configures routes
57
+ */
58
+ export type RouteConfiguringFunction = (app: import('express').Express, opts?: RouteConfiguringFunctionOpts | undefined) => void;
59
+ /**
60
+ * Options for a {@linkcode RouteConfiguringFunction }
61
+ */
62
+ export type RouteConfiguringFunctionOpts = {
63
+ basePath?: string | undefined;
64
+ extraMethodMap?: MethodMap | undefined;
65
+ };
66
+ /**
67
+ * Options for {@linkcode configureServer }
68
+ */
69
+ export type ConfigureServerOpts = {
70
+ app: import('express').Express;
71
+ addRoutes: RouteConfiguringFunction;
72
+ allowCors?: boolean | undefined;
73
+ basePath?: string | undefined;
74
+ extraMethodMap?: MethodMap | undefined;
75
+ };
76
+ /**
77
+ *
78
+ * @param {ServerOpts} opts
79
+ * @returns {Promise<AppiumServer>}
80
+ */
81
+ export function server(opts: ServerOpts): Promise<AppiumServer>;
82
+ /**
83
+ * Sets up some Express middleware and stuff
84
+ * @param {ConfigureServerOpts} opts
85
+ */
86
+ export function configureServer({ app, addRoutes, allowCors, basePath, extraMethodMap, }: ConfigureServerOpts): void;
87
+ /**
88
+ * Normalize base path string
89
+ * @param {string} basePath
90
+ * @returns {string}
91
+ */
92
+ export function normalizeBasePath(basePath: string): string;
31
93
  //# sourceMappingURL=server.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../../lib/express/server.js"],"names":[],"mappings":";;;;;;;gBAyOc,OAAO,MAAM,EAAE,MAAM;;;;UACrB,MAAM;;;;sBACN,MAAM;;;;;;AA3MpB,gDA8CC;AAED;;;;;;SA6CC;AAqFD,sDAgBC"}
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../../lib/express/server.js"],"names":[],"mappings":";;;;;;;gBA6Qc,OAAO,MAAM,EAAE,MAAM;;;;UACrB,MAAM;;;;sBACN,MAAM;;;;;;2BAKP,OAAO,eAAe,EAAE,YAAY;iCACpC,OAAO,eAAe,EAAE,kBAAkB;wBAC1C,OAAO,eAAe,EAAE,SAAS;;;;;;;;gBAMhC,OAAO,MAAM,EAAE,MAAM;;;;qBACZ,GAAG,KAAK,IAAI;;;;sBACrB,MAAM;;;;;;8BAMN,wBAAwB;UACxB,MAAM;;;;;;;;;;;6CAYT,OAAO,SAAS,EAAE,OAAO,sDAEvB,IAAI;;;;;;;;;;;;SAaH,OAAO,SAAS,EAAE,OAAO;eACzB,wBAAwB;;;;;AAjStC;;;;GAIG;AACH,6BAHW,UAAU,GACR,QAAQ,YAAY,CAAC,CAqDjC;AAED;;;GAGG;AACH,0FAFW,mBAAmB,QA+C7B;AAsGD;;;;GAIG;AACH,4CAHW,MAAM,GACJ,MAAM,CAkBlB"}
@@ -45,11 +45,11 @@ var _events = require("events");
45
45
 
46
46
  const KEEP_ALIVE_TIMEOUT_MS = 10 * 60 * 1000;
47
47
 
48
- async function server(opts = {}) {
48
+ async function server(opts) {
49
49
  const {
50
50
  routeConfiguringFunction,
51
51
  port,
52
- hostname = null,
52
+ hostname,
53
53
  allowCors = true,
54
54
  basePath = _constants.DEFAULT_BASE_PATH,
55
55
  extraMethodMap = {},
@@ -62,7 +62,7 @@ async function server(opts = {}) {
62
62
 
63
63
  return await new _bluebird.default(async (resolve, reject) => {
64
64
  try {
65
- configureHttp({
65
+ const appiumServer = configureHttp({
66
66
  httpServer,
67
67
  reject,
68
68
  keepAliveTimeout
@@ -76,7 +76,7 @@ async function server(opts = {}) {
76
76
  });
77
77
 
78
78
  for (const updater of serverUpdaters) {
79
- await updater(app, httpServer);
79
+ await updater(app, appiumServer);
80
80
  }
81
81
 
82
82
  app.all('*', _middleware.catch404Handler);
@@ -86,7 +86,7 @@ async function server(opts = {}) {
86
86
  port,
87
87
  keepAliveTimeout
88
88
  });
89
- resolve(httpServer);
89
+ resolve(appiumServer);
90
90
  } catch (err) {
91
91
  reject(err);
92
92
  }
@@ -144,13 +144,14 @@ function configureHttp({
144
144
  notifier: new _events.EventEmitter(),
145
145
  closed: false
146
146
  };
147
- httpServer.addWebSocketHandler = _websocket.addWebSocketHandler;
148
- httpServer.removeWebSocketHandler = _websocket.removeWebSocketHandler;
149
- httpServer.removeAllWebSocketHandlers = _websocket.removeAllWebSocketHandlers;
150
- httpServer.getWebSocketHandlers = _websocket.getWebSocketHandlers;
151
- const close = httpServer.close.bind(httpServer);
152
-
153
- httpServer.close = async () => await new _bluebird.default((resolve, reject) => {
147
+ const appiumServer = httpServer;
148
+ appiumServer.addWebSocketHandler = _websocket.addWebSocketHandler;
149
+ appiumServer.removeWebSocketHandler = _websocket.removeWebSocketHandler;
150
+ appiumServer.removeAllWebSocketHandlers = _websocket.removeAllWebSocketHandlers;
151
+ appiumServer.getWebSocketHandlers = _websocket.getWebSocketHandlers;
152
+ const close = appiumServer.close.bind(appiumServer);
153
+
154
+ appiumServer.close = async () => await new _bluebird.default((resolve, reject) => {
154
155
  serverState.closed = true;
155
156
  serverState.notifier.emit('shutdown');
156
157
 
@@ -166,7 +167,7 @@ function configureHttp({
166
167
  });
167
168
  });
168
169
 
169
- httpServer.on('error', err => {
170
+ appiumServer.on('error', err => {
170
171
  if (err.code === 'EADDRNOTAVAIL') {
171
172
  _logger.default.error('Could not start REST http interface listener. ' + 'Requested address is not available.');
172
173
  } else {
@@ -175,7 +176,7 @@ function configureHttp({
175
176
 
176
177
  reject(err);
177
178
  });
178
- httpServer.on('connection', socket => {
179
+ appiumServer.on('connection', socket => {
179
180
  socket.setTimeout(keepAliveTimeout);
180
181
  socket.on('error', reject);
181
182
 
@@ -187,7 +188,7 @@ function configureHttp({
187
188
  socket.once('close', () => serverState.notifier.removeListener('shutdown', destroy));
188
189
  serverState.notifier.once('shutdown', destroy);
189
190
  });
190
- httpServer.on('request', function (req, res) {
191
+ appiumServer.on('request', function (req, res) {
191
192
  const socket = req.connection || req.socket;
192
193
  socket._openReqCount++;
193
194
  res.on('finish', function () {
@@ -198,6 +199,7 @@ function configureHttp({
198
199
  }
199
200
  });
200
201
  });
202
+ return appiumServer;
201
203
  }
202
204
 
203
205
  async function startServer({
@@ -229,4 +231,4 @@ function normalizeBasePath(basePath) {
229
231
 
230
232
  return basePath;
231
233
  }
232
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["KEEP_ALIVE_TIMEOUT_MS","server","opts","routeConfiguringFunction","port","hostname","allowCors","basePath","DEFAULT_BASE_PATH","extraMethodMap","serverUpdaters","keepAliveTimeout","app","express","httpServer","http","createServer","B","resolve","reject","configureHttp","configureServer","addRoutes","updater","all","catch404Handler","startServer","err","normalizeBasePath","use","endLogFormatter","favicon","path","STATIC_DIR","static","produceError","produceCrash","allowCrossDomain","allowCrossDomainAsyncExecute","handleIdempotency","fixPythonContentType","defaultToJSONContentType","bodyParser","urlencoded","extended","methodOverride","catchAllHandler","json","limit","startLogFormatter","welcome","guineaPig","guineaPigScrollable","guineaPigAppBanner","serverState","notifier","EventEmitter","closed","addWebSocketHandler","removeWebSocketHandler","removeAllWebSocketHandlers","getWebSocketHandlers","close","bind","emit","log","info","on","code","error","socket","setTimeout","destroy","_openReqCount","once","removeListener","req","res","connection","start","promisify","listen","context","startPromise","headersTimeout","_","isString","Error","replace"],"sources":["../../../lib/express/server.js"],"sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport express from 'express';\nimport http from 'http';\nimport favicon from 'serve-favicon';\nimport bodyParser from 'body-parser';\nimport methodOverride from 'method-override';\nimport log from './logger';\nimport {startLogFormatter, endLogFormatter} from './express-logging';\nimport {\n  allowCrossDomain,\n  fixPythonContentType,\n  defaultToJSONContentType,\n  catchAllHandler,\n  allowCrossDomainAsyncExecute,\n  handleIdempotency,\n  catch404Handler,\n} from './middleware';\nimport {guineaPig, guineaPigScrollable, guineaPigAppBanner, welcome, STATIC_DIR} from './static';\nimport {produceError, produceCrash} from './crash';\nimport {\n  addWebSocketHandler,\n  removeWebSocketHandler,\n  removeAllWebSocketHandlers,\n  getWebSocketHandlers,\n} from './websocket';\nimport B from 'bluebird';\nimport {DEFAULT_BASE_PATH} from '../constants';\nimport {EventEmitter} from 'events';\n\nconst KEEP_ALIVE_TIMEOUT_MS = 10 * 60 * 1000; // 10 minutes\n\nasync function server(opts = {}) {\n  const {\n    routeConfiguringFunction,\n    port,\n    hostname = null,\n    allowCors = true,\n    basePath = DEFAULT_BASE_PATH,\n    extraMethodMap = {},\n    serverUpdaters = [],\n    keepAliveTimeout = KEEP_ALIVE_TIMEOUT_MS,\n  } = opts;\n\n  // create the actual http server\n  const app = express();\n  const httpServer = http.createServer(app);\n  return await new B(async (resolve, reject) => {\n    // we put an async function as the promise constructor because we want some things to happen in\n    // serial (application of plugin updates, for example). But we still need to use a promise here\n    // because some elements of server start failure only happen in httpServer listeners. So the\n    // way we resolve it is to use an async function here but to wrap all the inner logic in\n    // try/catch so any errors can be passed to reject.\n    try {\n      configureHttp({httpServer, reject, keepAliveTimeout});\n      configureServer({\n        app,\n        addRoutes: routeConfiguringFunction,\n        allowCors,\n        basePath,\n        extraMethodMap,\n      });\n      // allow extensions to update the app and http server objects\n      for (const updater of serverUpdaters) {\n        await updater(app, httpServer);\n      }\n\n      // once all configurations and updaters have been applied, make sure to set up a catchall\n      // handler so that anything unknown 404s. But do this after everything else since we don't\n      // want to block extensions' ability to add routes if they want.\n      app.all('*', catch404Handler);\n\n      await startServer({httpServer, hostname, port, keepAliveTimeout});\n      resolve(httpServer);\n    } catch (err) {\n      reject(err);\n    }\n  });\n}\n\nfunction configureServer({\n  app,\n  addRoutes,\n  allowCors = true,\n  basePath = DEFAULT_BASE_PATH,\n  extraMethodMap = {},\n}) {\n  basePath = normalizeBasePath(basePath);\n\n  app.use(endLogFormatter);\n\n  // set up static assets\n  app.use(favicon(path.resolve(STATIC_DIR, 'favicon.ico')));\n  app.use(express.static(STATIC_DIR));\n\n  // crash routes, for testing\n  app.use(`${basePath}/produce_error`, produceError);\n  app.use(`${basePath}/crash`, produceCrash);\n\n  // add middlewares\n  if (allowCors) {\n    app.use(allowCrossDomain);\n  } else {\n    app.use(allowCrossDomainAsyncExecute(basePath));\n  }\n  app.use(handleIdempotency);\n  app.use(fixPythonContentType(basePath));\n  app.use(defaultToJSONContentType);\n  app.use(bodyParser.urlencoded({extended: true}));\n  app.use(methodOverride());\n  app.use(catchAllHandler);\n\n  // make sure appium never fails because of a file size upload limit\n  app.use(bodyParser.json({limit: '1gb'}));\n\n  // set up start logging (which depends on bodyParser doing its thing)\n  app.use(startLogFormatter);\n\n  addRoutes(app, {basePath, extraMethodMap});\n\n  // dynamic routes for testing, etc.\n  app.all('/welcome', welcome);\n  app.all('/test/guinea-pig', guineaPig);\n  app.all('/test/guinea-pig-scrollable', guineaPigScrollable);\n  app.all('/test/guinea-pig-app-banner', guineaPigAppBanner);\n}\n\nfunction configureHttp({httpServer, reject, keepAliveTimeout}) {\n  const serverState = {\n    notifier: new EventEmitter(),\n    closed: false,\n  };\n  httpServer.addWebSocketHandler = addWebSocketHandler;\n  httpServer.removeWebSocketHandler = removeWebSocketHandler;\n  httpServer.removeAllWebSocketHandlers = removeAllWebSocketHandlers;\n  httpServer.getWebSocketHandlers = getWebSocketHandlers;\n\n  // http.Server.close() only stops new connections, but we need to wait until\n  // all connections are closed and the `close` event is emitted\n  const close = httpServer.close.bind(httpServer);\n  httpServer.close = async () =>\n    await new B((resolve, reject) => {\n      // https://github.com/nodejs/node-v0.x-archive/issues/9066#issuecomment-124210576\n      serverState.closed = true;\n      serverState.notifier.emit('shutdown');\n      log.info('Waiting until the server is closed');\n      httpServer.on('close', () => {\n        log.info('Received server close event');\n        resolve();\n      });\n      close((err) => {\n        if (err) reject(err); // eslint-disable-line curly\n      });\n    });\n\n  httpServer.on('error', (err) => {\n    if (err.code === 'EADDRNOTAVAIL') {\n      log.error(\n        'Could not start REST http interface listener. ' + 'Requested address is not available.'\n      );\n    } else {\n      log.error(\n        'Could not start REST http interface listener. The requested ' +\n          'port may already be in use. Please make sure there is no ' +\n          'other instance of this server running already.'\n      );\n    }\n    reject(err);\n  });\n\n  httpServer.on('connection', (socket) => {\n    socket.setTimeout(keepAliveTimeout);\n    socket.on('error', reject);\n\n    function destroy() {\n      socket.destroy();\n    }\n    socket._openReqCount = 0;\n    socket.once('close', () => serverState.notifier.removeListener('shutdown', destroy));\n    serverState.notifier.once('shutdown', destroy);\n  });\n\n  httpServer.on('request', function (req, res) {\n    const socket = req.connection || req.socket;\n    socket._openReqCount++;\n    res.on('finish', function () {\n      socket._openReqCount--;\n      if (serverState.closed && socket._openReqCount === 0) {\n        socket.destroy();\n      }\n    });\n  });\n}\n\n/**\n *\n * @param {StartServerOpts} opts\n */\nasync function startServer({httpServer, port, hostname, keepAliveTimeout}) {\n  // If the hostname is omitted, the server will accept\n  // connections on any IP address\n  /** @type {(port: number, hostname?: string) => B<http.Server>} */\n  const start = B.promisify(httpServer.listen, {context: httpServer});\n  const startPromise = start(port, hostname);\n  httpServer.keepAliveTimeout = keepAliveTimeout;\n  // headers timeout must be greater than keepAliveTimeout\n  httpServer.headersTimeout = keepAliveTimeout + 5 * 1000;\n  await startPromise;\n}\n\nfunction normalizeBasePath(basePath) {\n  if (!_.isString(basePath)) {\n    throw new Error(`Invalid path prefix ${basePath}`);\n  }\n\n  // ensure the path prefix does not end in '/', since our method map\n  // starts all paths with '/'\n  basePath = basePath.replace(/\\/$/, '');\n\n  // likewise, ensure the path prefix does always START with /, unless the path\n  // is empty meaning no base path at all\n  if (basePath !== '' && basePath[0] !== '/') {\n    basePath = `/${basePath}`;\n  }\n\n  return basePath;\n}\n\nexport {server, configureServer, normalizeBasePath};\n\n/**\n * Options for {@linkcode startServer}.\n * @typedef StartServerOpts\n * @property {import('http').Server} httpServer - HTTP server instance\n * @property {number} port - Port to run on\n * @property {number} keepAliveTimeout - Keep-alive timeout in milliseconds\n * @property {string} [hostname] - Optional hostname\n */\n"],"mappings":"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"}
234
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["KEEP_ALIVE_TIMEOUT_MS","server","opts","routeConfiguringFunction","port","hostname","allowCors","basePath","DEFAULT_BASE_PATH","extraMethodMap","serverUpdaters","keepAliveTimeout","app","express","httpServer","http","createServer","B","resolve","reject","appiumServer","configureHttp","configureServer","addRoutes","updater","all","catch404Handler","startServer","err","normalizeBasePath","use","endLogFormatter","favicon","path","STATIC_DIR","static","produceError","produceCrash","allowCrossDomain","allowCrossDomainAsyncExecute","handleIdempotency","fixPythonContentType","defaultToJSONContentType","bodyParser","urlencoded","extended","methodOverride","catchAllHandler","json","limit","startLogFormatter","welcome","guineaPig","guineaPigScrollable","guineaPigAppBanner","serverState","notifier","EventEmitter","closed","addWebSocketHandler","removeWebSocketHandler","removeAllWebSocketHandlers","getWebSocketHandlers","close","bind","emit","log","info","on","code","error","socket","setTimeout","destroy","_openReqCount","once","removeListener","req","res","connection","start","promisify","listen","context","startPromise","headersTimeout","_","isString","Error","replace"],"sources":["../../../lib/express/server.js"],"sourcesContent":["import _ from 'lodash';\nimport path from 'path';\nimport express from 'express';\nimport http from 'http';\nimport favicon from 'serve-favicon';\nimport bodyParser from 'body-parser';\nimport methodOverride from 'method-override';\nimport log from './logger';\nimport {startLogFormatter, endLogFormatter} from './express-logging';\nimport {\n  allowCrossDomain,\n  fixPythonContentType,\n  defaultToJSONContentType,\n  catchAllHandler,\n  allowCrossDomainAsyncExecute,\n  handleIdempotency,\n  catch404Handler,\n} from './middleware';\nimport {guineaPig, guineaPigScrollable, guineaPigAppBanner, welcome, STATIC_DIR} from './static';\nimport {produceError, produceCrash} from './crash';\nimport {\n  addWebSocketHandler,\n  removeWebSocketHandler,\n  removeAllWebSocketHandlers,\n  getWebSocketHandlers,\n} from './websocket';\nimport B from 'bluebird';\nimport {DEFAULT_BASE_PATH} from '../constants';\nimport {EventEmitter} from 'events';\n\nconst KEEP_ALIVE_TIMEOUT_MS = 10 * 60 * 1000; // 10 minutes\n\n/**\n *\n * @param {ServerOpts} opts\n * @returns {Promise<AppiumServer>}\n */\nasync function server(opts) {\n  const {\n    routeConfiguringFunction,\n    port,\n    hostname,\n    allowCors = true,\n    basePath = DEFAULT_BASE_PATH,\n    extraMethodMap = {},\n    serverUpdaters = [],\n    keepAliveTimeout = KEEP_ALIVE_TIMEOUT_MS,\n  } = opts;\n\n  // create the actual http server\n  const app = express();\n  const httpServer = http.createServer(app);\n  return await new B(async (resolve, reject) => {\n    // we put an async function as the promise constructor because we want some things to happen in\n    // serial (application of plugin updates, for example). But we still need to use a promise here\n    // because some elements of server start failure only happen in httpServer listeners. So the\n    // way we resolve it is to use an async function here but to wrap all the inner logic in\n    // try/catch so any errors can be passed to reject.\n    try {\n      const appiumServer = configureHttp({\n        httpServer,\n        reject,\n        keepAliveTimeout,\n      });\n      configureServer({\n        app,\n        addRoutes: routeConfiguringFunction,\n        allowCors,\n        basePath,\n        extraMethodMap,\n      });\n      // allow extensions to update the app and http server objects\n      for (const updater of serverUpdaters) {\n        await updater(app, appiumServer);\n      }\n\n      // once all configurations and updaters have been applied, make sure to set up a catchall\n      // handler so that anything unknown 404s. But do this after everything else since we don't\n      // want to block extensions' ability to add routes if they want.\n      app.all('*', catch404Handler);\n\n      await startServer({httpServer, hostname, port, keepAliveTimeout});\n\n      resolve(appiumServer);\n    } catch (err) {\n      reject(err);\n    }\n  });\n}\n\n/**\n * Sets up some Express middleware and stuff\n * @param {ConfigureServerOpts} opts\n */\nfunction configureServer({\n  app,\n  addRoutes,\n  allowCors = true,\n  basePath = DEFAULT_BASE_PATH,\n  extraMethodMap = {},\n}) {\n  basePath = normalizeBasePath(basePath);\n\n  app.use(endLogFormatter);\n\n  // set up static assets\n  app.use(favicon(path.resolve(STATIC_DIR, 'favicon.ico')));\n  app.use(express.static(STATIC_DIR));\n\n  // crash routes, for testing\n  app.use(`${basePath}/produce_error`, produceError);\n  app.use(`${basePath}/crash`, produceCrash);\n\n  // add middlewares\n  if (allowCors) {\n    app.use(allowCrossDomain);\n  } else {\n    app.use(allowCrossDomainAsyncExecute(basePath));\n  }\n  app.use(handleIdempotency);\n  app.use(fixPythonContentType(basePath));\n  app.use(defaultToJSONContentType);\n  app.use(bodyParser.urlencoded({extended: true}));\n  app.use(methodOverride());\n  app.use(catchAllHandler);\n\n  // make sure appium never fails because of a file size upload limit\n  app.use(bodyParser.json({limit: '1gb'}));\n\n  // set up start logging (which depends on bodyParser doing its thing)\n  app.use(startLogFormatter);\n\n  addRoutes(app, {basePath, extraMethodMap});\n\n  // dynamic routes for testing, etc.\n  app.all('/welcome', welcome);\n  app.all('/test/guinea-pig', guineaPig);\n  app.all('/test/guinea-pig-scrollable', guineaPigScrollable);\n  app.all('/test/guinea-pig-app-banner', guineaPigAppBanner);\n}\n\n/**\n * Monkeypatches the `http.Server` instance and returns a {@linkcode AppiumServer}.\n * This function _mutates_ the `httpServer` parameter.\n * @param {ConfigureHttpOpts} opts\n * @returns {AppiumServer}\n */\nfunction configureHttp({httpServer, reject, keepAliveTimeout}) {\n  const serverState = {\n    notifier: new EventEmitter(),\n    closed: false,\n  };\n  // TS does not love monkeypatching.\n  const appiumServer = /** @type {AppiumServer} */ (/** @type {unknown} */ (httpServer));\n  appiumServer.addWebSocketHandler = addWebSocketHandler;\n  appiumServer.removeWebSocketHandler = removeWebSocketHandler;\n  appiumServer.removeAllWebSocketHandlers = removeAllWebSocketHandlers;\n  appiumServer.getWebSocketHandlers = getWebSocketHandlers;\n\n  // http.Server.close() only stops new connections, but we need to wait until\n  // all connections are closed and the `close` event is emitted\n  const close = appiumServer.close.bind(appiumServer);\n  appiumServer.close = async () =>\n    await new B((resolve, reject) => {\n      // https://github.com/nodejs/node-v0.x-archive/issues/9066#issuecomment-124210576\n      serverState.closed = true;\n      serverState.notifier.emit('shutdown');\n      log.info('Waiting until the server is closed');\n      httpServer.on('close', () => {\n        log.info('Received server close event');\n        resolve();\n      });\n      close((err) => {\n        if (err) reject(err); // eslint-disable-line curly\n      });\n    });\n\n  appiumServer.on(\n    'error',\n    /** @param {NodeJS.ErrnoException} err */ (err) => {\n      if (err.code === 'EADDRNOTAVAIL') {\n        log.error(\n          'Could not start REST http interface listener. ' + 'Requested address is not available.'\n        );\n      } else {\n        log.error(\n          'Could not start REST http interface listener. The requested ' +\n            'port may already be in use. Please make sure there is no ' +\n            'other instance of this server running already.'\n        );\n      }\n      reject(err);\n    }\n  );\n\n  appiumServer.on(\n    'connection',\n    /** @param {AppiumServerSocket} socket */ (socket) => {\n      socket.setTimeout(keepAliveTimeout);\n      socket.on('error', reject);\n\n      function destroy() {\n        socket.destroy();\n      }\n      socket._openReqCount = 0;\n      socket.once('close', () => serverState.notifier.removeListener('shutdown', destroy));\n      serverState.notifier.once('shutdown', destroy);\n    }\n  );\n\n  appiumServer.on('request', function (req, res) {\n    const socket = /** @type {AppiumServerSocket} */ (req.connection || req.socket);\n    socket._openReqCount++;\n    res.on('finish', function () {\n      socket._openReqCount--;\n      if (serverState.closed && socket._openReqCount === 0) {\n        socket.destroy();\n      }\n    });\n  });\n\n  return appiumServer;\n}\n\n/**\n * Starts an {@linkcode AppiumServer}\n * @param {StartServerOpts} opts\n * @returns {Promise<void>}\n */\nasync function startServer({httpServer, port, hostname, keepAliveTimeout}) {\n  // If the hostname is omitted, the server will accept\n  // connections on any IP address\n  /** @type {(port: number, hostname?: string) => B<http.Server>} */\n  const start = B.promisify(httpServer.listen, {context: httpServer});\n  const startPromise = start(port, hostname);\n  httpServer.keepAliveTimeout = keepAliveTimeout;\n  // headers timeout must be greater than keepAliveTimeout\n  httpServer.headersTimeout = keepAliveTimeout + 5 * 1000;\n  await startPromise;\n}\n\n/**\n * Normalize base path string\n * @param {string} basePath\n * @returns {string}\n */\nfunction normalizeBasePath(basePath) {\n  if (!_.isString(basePath)) {\n    throw new Error(`Invalid path prefix ${basePath}`);\n  }\n\n  // ensure the path prefix does not end in '/', since our method map\n  // starts all paths with '/'\n  basePath = basePath.replace(/\\/$/, '');\n\n  // likewise, ensure the path prefix does always START with /, unless the path\n  // is empty meaning no base path at all\n  if (basePath !== '' && basePath[0] !== '/') {\n    basePath = `/${basePath}`;\n  }\n\n  return basePath;\n}\n\nexport {server, configureServer, normalizeBasePath};\n\n/**\n * Options for {@linkcode startServer}.\n * @typedef StartServerOpts\n * @property {import('http').Server} httpServer - HTTP server instance\n * @property {number} port - Port to run on\n * @property {number} keepAliveTimeout - Keep-alive timeout in milliseconds\n * @property {string} [hostname] - Optional hostname\n */\n\n/**\n * @typedef {import('@appium/types').AppiumServer} AppiumServer\n * @typedef {import('@appium/types').AppiumServerSocket} AppiumServerSocket\n * @typedef {import('@appium/types').MethodMap} MethodMap\n */\n\n/**\n * Options for {@linkcode configureHttp}\n * @typedef ConfigureHttpOpts\n * @property {import('http').Server} httpServer - HTTP server instance\n * @property {(error?: any) => void} reject - Rejection function from `Promise` constructor\n * @property {number} keepAliveTimeout - Keep-alive timeout in milliseconds\n */\n\n/**\n * Options for {@linkcode server}\n * @typedef ServerOpts\n * @property {RouteConfiguringFunction} routeConfiguringFunction\n * @property {number} port\n * @property {string} [hostname]\n * @property {boolean} [allowCors]\n * @property {string} [basePath]\n * @property {MethodMap} [extraMethodMap]\n * @property {import('@appium/types').UpdateServerCallback[]} [serverUpdaters]\n * @property {number} [keepAliveTimeout]\n */\n\n/**\n * A function which configures routes\n * @callback RouteConfiguringFunction\n * @param {import('express').Express} app\n * @param {RouteConfiguringFunctionOpts} [opts]\n * @returns {void}\n */\n\n/**\n * Options for a {@linkcode RouteConfiguringFunction}\n * @typedef RouteConfiguringFunctionOpts\n * @property {string} [basePath]\n * @property {MethodMap} [extraMethodMap]\n */\n\n/**\n * Options for {@linkcode configureServer}\n * @typedef ConfigureServerOpts\n * @property {import('express').Express} app\n * @property {RouteConfiguringFunction} addRoutes\n * @property {boolean} [allowCors]\n * @property {string} [basePath]\n * @property {MethodMap} [extraMethodMap]\n */\n"],"mappings":"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"}
@@ -1,67 +1,49 @@
1
- export type WebSocketServer = import('http').Server & {
2
- webSocketsMapping: Record<string, import('ws').Server>;
3
- };
1
+ export type AppiumServer = import('@appium/types').AppiumServer;
4
2
  /**
5
3
  * Adds websocket handler to express server instance.
6
4
  * It is expected this function is called in Express
7
5
  * server instance context.
8
6
  *
9
- * @this {WebSocketServer} - An instance of express HTTP server.
7
+ * @this {AppiumServer} - An instance of express HTTP server.
10
8
  * @param {string} handlerPathname - Web socket endpoint path starting with
11
9
  * a single slash character. It is recommended to always add
12
10
  * DEFAULT_WS_PATHNAME_PREFIX to all web socket pathnames.
13
- * @param {Object} handlerServer - WebSocket server instance. See
11
+ * @param {import('ws').Server} handlerServer - WebSocket server instance. See
14
12
  * https://github.com/websockets/ws/pull/885 for more details
15
13
  * on how to configure the handler properly.
14
+ * @returns {Promise<void>}
16
15
  */
17
- export function addWebSocketHandler(this: WebSocketServer, handlerPathname: string, handlerServer: any): Promise<void>;
18
- export class addWebSocketHandler {
19
- /**
20
- * Adds websocket handler to express server instance.
21
- * It is expected this function is called in Express
22
- * server instance context.
23
- *
24
- * @this {WebSocketServer} - An instance of express HTTP server.
25
- * @param {string} handlerPathname - Web socket endpoint path starting with
26
- * a single slash character. It is recommended to always add
27
- * DEFAULT_WS_PATHNAME_PREFIX to all web socket pathnames.
28
- * @param {Object} handlerServer - WebSocket server instance. See
29
- * https://github.com/websockets/ws/pull/885 for more details
30
- * on how to configure the handler properly.
31
- */
32
- constructor(this: WebSocketServer, handlerPathname: string, handlerServer: any);
33
- webSocketsMapping: {};
34
- }
16
+ export function addWebSocketHandler(this: import("@appium/types").AppiumServer, handlerPathname: string, handlerServer: import('ws').Server): Promise<void>;
35
17
  /**
36
18
  * Removes existing websocket handler from express server instance.
37
19
  * The call is ignored if the given `handlerPathname` handler
38
20
  * is not present in the handlers list.
39
21
  * It is expected this function is called in Express
40
22
  * server instance context.
41
- *
23
+ * @this {AppiumServer}
42
24
  * @param {string} handlerPathname - Websocket endpoint path.
43
25
  * @returns {Promise<boolean>} true if the handlerPathname was found and deleted
44
26
  */
45
- export function removeWebSocketHandler(handlerPathname: string): Promise<boolean>;
27
+ export function removeWebSocketHandler(this: import("@appium/types").AppiumServer, handlerPathname: string): Promise<boolean>;
46
28
  /**
47
29
  * Removes all existing websocket handler from express server instance.
48
30
  * It is expected this function is called in Express
49
31
  * server instance context.
50
- *
32
+ * @this {AppiumServer}
51
33
  * @returns {Promise<boolean>} true if at least one handler has been deleted
52
34
  */
53
- export function removeAllWebSocketHandlers(): Promise<boolean>;
35
+ export function removeAllWebSocketHandlers(this: import("@appium/types").AppiumServer): Promise<boolean>;
54
36
  /**
55
37
  * Returns web socket handlers registered for the given server
56
38
  * instance.
57
39
  * It is expected this function is called in Express
58
40
  * server instance context.
59
41
  *
42
+ * @this {AppiumServer}
60
43
  * @param {string?} [keysFilter] - Only include pathnames with given
61
44
  * `keysFilter` value if set. All pairs will be included by default.
62
- * @returns {Promise<Object>} pathnames to websocket server isntances mapping
63
- * matching the search criteria or an empty object otherwise.
45
+ * @returns {Promise<Record<string, import('ws').Server>>} pathnames to websocket server instances mapping matching the search criteria or an empty object otherwise.
64
46
  */
65
- export function getWebSocketHandlers(keysFilter?: string | null | undefined): Promise<any>;
47
+ export function getWebSocketHandlers(this: import("@appium/types").AppiumServer, keysFilter?: string | null | undefined): Promise<Record<string, import('ws').Server>>;
66
48
  export const DEFAULT_WS_PATHNAME_PREFIX: "/ws";
67
49
  //# sourceMappingURL=websocket.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"websocket.d.ts","sourceRoot":"","sources":["../../../lib/express/websocket.js"],"names":[],"mappings":"8BAkIa,OAAO,MAAM,EAAE,MAAM,GAAG;IAAC,iBAAiB,EAAE,OAAO,MAAM,EAAC,OAAO,IAAI,EAAE,MAAM,CAAC,CAAA;CAAC;AA5H5F;;;;;;;;;;;;GAYG;AAEH,4EARW,MAAM,qCAgChB;;IAtCD;;;;;;;;;;;;OAYG;IAEH,oDARW,MAAM,sBAgChB;IAtBG,sBAA2B;;AAiD/B;;;;;;;;;GASG;AAEH,wDAJW,MAAM,GACJ,QAAQ,OAAO,CAAC,CAqB5B;AAED;;;;;;GAMG;AACH,8CAFa,QAAQ,OAAO,CAAC,CAY5B;AAzED;;;;;;;;;;GAUG;AAEH,8EAJa,YAAe,CAe3B;AAjED,+CAAyC"}
1
+ {"version":3,"file":"websocket.d.ts","sourceRoot":"","sources":["../../../lib/express/websocket.js"],"names":[],"mappings":"2BAoIa,OAAO,eAAe,EAAE,YAAY;AA7HjD;;;;;;;;;;;;;GAaG;AACH,iGARW,MAAM,iBAGN,OAAO,IAAI,EAAE,MAAM,GAGjB,QAAQ,IAAI,CAAC,CA2BzB;AA2BD;;;;;;;;;GASG;AAEH,oGAJW,MAAM,GACJ,QAAQ,OAAO,CAAC,CAqB5B;AAED;;;;;;GAMG;AACH,wFAFa,QAAQ,OAAO,CAAC,CAY5B;AAzED;;;;;;;;;;GAUG;AAEH,0HAHa,QAAQ,OAAO,MAAM,EAAE,OAAO,IAAI,EAAE,MAAM,CAAC,CAAC,CAcxD;AAlED,+CAAyC"}
@@ -23,9 +23,11 @@ const DEFAULT_WS_PATHNAME_PREFIX = '/ws';
23
23
  exports.DEFAULT_WS_PATHNAME_PREFIX = DEFAULT_WS_PATHNAME_PREFIX;
24
24
 
25
25
  async function addWebSocketHandler(handlerPathname, handlerServer) {
26
+ const server = this;
27
+
26
28
  if (_lodash.default.isUndefined(this.webSocketsMapping)) {
27
- this.webSocketsMapping = {};
28
- this.on('upgrade', (request, socket, head) => {
29
+ server.webSocketsMapping = {};
30
+ server.on('upgrade', (request, socket, head) => {
29
31
  let currentPathname;
30
32
 
31
33
  try {
@@ -34,7 +36,7 @@ async function addWebSocketHandler(handlerPathname, handlerServer) {
34
36
  currentPathname = request.url;
35
37
  }
36
38
 
37
- for (const [pathname, wsServer] of _lodash.default.toPairs(this.webSocketsMapping)) {
39
+ for (const [pathname, wsServer] of _lodash.default.toPairs(server.webSocketsMapping)) {
38
40
  if (currentPathname === pathname) {
39
41
  wsServer.handleUpgrade(request, socket, head, ws => {
40
42
  wsServer.emit('connection', ws, request);
@@ -95,4 +97,4 @@ async function removeAllWebSocketHandlers() {
95
97
 
96
98
  return _lodash.default.some(await _bluebird.default.all(_lodash.default.keys(this.webSocketsMapping).map(pathname => this.removeWebSocketHandler(pathname))));
97
99
  }
98
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["DEFAULT_WS_PATHNAME_PREFIX","addWebSocketHandler","handlerPathname","handlerServer","_","isUndefined","webSocketsMapping","on","request","socket","head","currentPathname","URL","url","pathname","wsServer","toPairs","handleUpgrade","ws","emit","destroy","getWebSocketHandlers","keysFilter","isEmpty","reduce","acc","isString","includes","removeWebSocketHandler","close","client","clients","terminate","ign","removeAllWebSocketHandlers","some","B","all","keys","map"],"sources":["../../../lib/express/websocket.js"],"sourcesContent":["import _ from 'lodash';\nimport {URL} from 'url';\nimport B from 'bluebird';\n\nconst DEFAULT_WS_PATHNAME_PREFIX = '/ws';\n\n/**\n * Adds websocket handler to express server instance.\n * It is expected this function is called in Express\n * server instance context.\n *\n * @this {WebSocketServer} - An instance of express HTTP server.\n * @param {string} handlerPathname - Web socket endpoint path starting with\n * a single slash character. It is recommended to always add\n * DEFAULT_WS_PATHNAME_PREFIX to all web socket pathnames.\n * @param {Object} handlerServer - WebSocket server instance. See\n * https://github.com/websockets/ws/pull/885 for more details\n * on how to configure the handler properly.\n */\n// eslint-disable-next-line require-await\nasync function addWebSocketHandler(handlerPathname, handlerServer) {\n  if (_.isUndefined(this.webSocketsMapping)) {\n    this.webSocketsMapping = {};\n    // https://github.com/websockets/ws/pull/885\n    this.on('upgrade', (request, socket, head) => {\n      let currentPathname;\n      try {\n        // @ts-expect-error\n        currentPathname = new URL(request.url).pathname;\n      } catch {\n        currentPathname = request.url;\n      }\n      for (const [pathname, wsServer] of _.toPairs(this.webSocketsMapping)) {\n        if (currentPathname === pathname) {\n          wsServer.handleUpgrade(request, socket, head, (ws) => {\n            wsServer.emit('connection', ws, request);\n          });\n          return;\n        }\n      }\n      socket.destroy();\n    });\n  }\n  this.webSocketsMapping[handlerPathname] = handlerServer;\n}\n\n/**\n * Returns web socket handlers registered for the given server\n * instance.\n * It is expected this function is called in Express\n * server instance context.\n *\n * @param {string?} [keysFilter] - Only include pathnames with given\n * `keysFilter` value if set. All pairs will be included by default.\n * @returns {Promise<Object>} pathnames to websocket server isntances mapping\n * matching the search criteria or an empty object otherwise.\n */\n// eslint-disable-next-line require-await\nasync function getWebSocketHandlers(keysFilter = null) {\n  if (_.isEmpty(this.webSocketsMapping)) {\n    return {};\n  }\n\n  return _.toPairs(this.webSocketsMapping).reduce((acc, [pathname, wsServer]) => {\n    if (!_.isString(keysFilter) || pathname.includes(keysFilter)) {\n      acc[pathname] = wsServer;\n    }\n    return acc;\n  }, {});\n}\n\n/**\n * Removes existing websocket handler from express server instance.\n * The call is ignored if the given `handlerPathname` handler\n * is not present in the handlers list.\n * It is expected this function is called in Express\n * server instance context.\n *\n * @param {string} handlerPathname - Websocket endpoint path.\n * @returns {Promise<boolean>} true if the handlerPathname was found and deleted\n */\n// eslint-disable-next-line require-await\nasync function removeWebSocketHandler(handlerPathname) {\n  const wsServer = this.webSocketsMapping?.[handlerPathname];\n  if (!wsServer) {\n    return false;\n  }\n\n  try {\n    wsServer.close();\n    for (const client of wsServer.clients || []) {\n      client.terminate();\n    }\n    return true;\n  } catch (ign) {\n    // ignore\n  } finally {\n    delete this.webSocketsMapping[handlerPathname];\n  }\n  return false;\n}\n\n/**\n * Removes all existing websocket handler from express server instance.\n * It is expected this function is called in Express\n * server instance context.\n *\n * @returns {Promise<boolean>} true if at least one handler has been deleted\n */\nasync function removeAllWebSocketHandlers() {\n  if (_.isEmpty(this.webSocketsMapping)) {\n    return false;\n  }\n\n  return _.some(\n    await B.all(\n      _.keys(this.webSocketsMapping).map((pathname) => this.removeWebSocketHandler(pathname))\n    )\n  );\n}\n\nexport {\n  addWebSocketHandler,\n  removeWebSocketHandler,\n  removeAllWebSocketHandlers,\n  getWebSocketHandlers,\n  DEFAULT_WS_PATHNAME_PREFIX,\n};\n\n/**\n * @typedef {import('http').Server & {webSocketsMapping: Record<string,import('ws').Server>}} WebSocketServer\n */\n"],"mappings":";;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AAEA,MAAMA,0BAA0B,GAAG,KAAnC;;;AAgBA,eAAeC,mBAAf,CAAmCC,eAAnC,EAAoDC,aAApD,EAAmE;EACjE,IAAIC,eAAA,CAAEC,WAAF,CAAc,KAAKC,iBAAnB,CAAJ,EAA2C;IACzC,KAAKA,iBAAL,GAAyB,EAAzB;IAEA,KAAKC,EAAL,CAAQ,SAAR,EAAmB,CAACC,OAAD,EAAUC,MAAV,EAAkBC,IAAlB,KAA2B;MAC5C,IAAIC,eAAJ;;MACA,IAAI;QAEFA,eAAe,GAAG,IAAIC,QAAJ,CAAQJ,OAAO,CAACK,GAAhB,EAAqBC,QAAvC;MACD,CAHD,CAGE,MAAM;QACNH,eAAe,GAAGH,OAAO,CAACK,GAA1B;MACD;;MACD,KAAK,MAAM,CAACC,QAAD,EAAWC,QAAX,CAAX,IAAmCX,eAAA,CAAEY,OAAF,CAAU,KAAKV,iBAAf,CAAnC,EAAsE;QACpE,IAAIK,eAAe,KAAKG,QAAxB,EAAkC;UAChCC,QAAQ,CAACE,aAAT,CAAuBT,OAAvB,EAAgCC,MAAhC,EAAwCC,IAAxC,EAA+CQ,EAAD,IAAQ;YACpDH,QAAQ,CAACI,IAAT,CAAc,YAAd,EAA4BD,EAA5B,EAAgCV,OAAhC;UACD,CAFD;UAGA;QACD;MACF;;MACDC,MAAM,CAACW,OAAP;IACD,CAjBD;EAkBD;;EACD,KAAKd,iBAAL,CAAuBJ,eAAvB,IAA0CC,aAA1C;AACD;;AAcD,eAAekB,oBAAf,CAAoCC,UAAU,GAAG,IAAjD,EAAuD;EACrD,IAAIlB,eAAA,CAAEmB,OAAF,CAAU,KAAKjB,iBAAf,CAAJ,EAAuC;IACrC,OAAO,EAAP;EACD;;EAED,OAAOF,eAAA,CAAEY,OAAF,CAAU,KAAKV,iBAAf,EAAkCkB,MAAlC,CAAyC,CAACC,GAAD,EAAM,CAACX,QAAD,EAAWC,QAAX,CAAN,KAA+B;IAC7E,IAAI,CAACX,eAAA,CAAEsB,QAAF,CAAWJ,UAAX,CAAD,IAA2BR,QAAQ,CAACa,QAAT,CAAkBL,UAAlB,CAA/B,EAA8D;MAC5DG,GAAG,CAACX,QAAD,CAAH,GAAgBC,QAAhB;IACD;;IACD,OAAOU,GAAP;EACD,CALM,EAKJ,EALI,CAAP;AAMD;;AAaD,eAAeG,sBAAf,CAAsC1B,eAAtC,EAAuD;EAAA;;EACrD,MAAMa,QAAQ,4BAAG,KAAKT,iBAAR,0DAAG,sBAAyBJ,eAAzB,CAAjB;;EACA,IAAI,CAACa,QAAL,EAAe;IACb,OAAO,KAAP;EACD;;EAED,IAAI;IACFA,QAAQ,CAACc,KAAT;;IACA,KAAK,MAAMC,MAAX,IAAqBf,QAAQ,CAACgB,OAAT,IAAoB,EAAzC,EAA6C;MAC3CD,MAAM,CAACE,SAAP;IACD;;IACD,OAAO,IAAP;EACD,CAND,CAME,OAAOC,GAAP,EAAY,CAEb,CARD,SAQU;IACR,OAAO,KAAK3B,iBAAL,CAAuBJ,eAAvB,CAAP;EACD;;EACD,OAAO,KAAP;AACD;;AASD,eAAegC,0BAAf,GAA4C;EAC1C,IAAI9B,eAAA,CAAEmB,OAAF,CAAU,KAAKjB,iBAAf,CAAJ,EAAuC;IACrC,OAAO,KAAP;EACD;;EAED,OAAOF,eAAA,CAAE+B,IAAF,CACL,MAAMC,iBAAA,CAAEC,GAAF,CACJjC,eAAA,CAAEkC,IAAF,CAAO,KAAKhC,iBAAZ,EAA+BiC,GAA/B,CAAoCzB,QAAD,IAAc,KAAKc,sBAAL,CAA4Bd,QAA5B,CAAjD,CADI,CADD,CAAP;AAKD"}
100
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["DEFAULT_WS_PATHNAME_PREFIX","addWebSocketHandler","handlerPathname","handlerServer","server","_","isUndefined","webSocketsMapping","on","request","socket","head","currentPathname","URL","url","pathname","wsServer","toPairs","handleUpgrade","ws","emit","destroy","getWebSocketHandlers","keysFilter","isEmpty","reduce","acc","isString","includes","removeWebSocketHandler","close","client","clients","terminate","ign","removeAllWebSocketHandlers","some","B","all","keys","map"],"sources":["../../../lib/express/websocket.js"],"sourcesContent":["/* eslint-disable require-await */\nimport _ from 'lodash';\nimport {URL} from 'url';\nimport B from 'bluebird';\n\nconst DEFAULT_WS_PATHNAME_PREFIX = '/ws';\n\n/**\n * Adds websocket handler to express server instance.\n * It is expected this function is called in Express\n * server instance context.\n *\n * @this {AppiumServer} - An instance of express HTTP server.\n * @param {string} handlerPathname - Web socket endpoint path starting with\n * a single slash character. It is recommended to always add\n * DEFAULT_WS_PATHNAME_PREFIX to all web socket pathnames.\n * @param {import('ws').Server} handlerServer - WebSocket server instance. See\n * https://github.com/websockets/ws/pull/885 for more details\n * on how to configure the handler properly.\n * @returns {Promise<void>}\n */\nasync function addWebSocketHandler(handlerPathname, handlerServer) {\n  const server = /** @type {AppiumServer} */ (this);\n  if (_.isUndefined(this.webSocketsMapping)) {\n    server.webSocketsMapping = {};\n    // https://github.com/websockets/ws/pull/885\n    server.on('upgrade', (request, socket, head) => {\n      let currentPathname;\n      try {\n        // @ts-expect-error\n        currentPathname = new URL(request.url).pathname;\n      } catch {\n        currentPathname = request.url;\n      }\n      for (const [pathname, wsServer] of _.toPairs(server.webSocketsMapping)) {\n        if (currentPathname === pathname) {\n          wsServer.handleUpgrade(request, socket, head, (ws) => {\n            wsServer.emit('connection', ws, request);\n          });\n          return;\n        }\n      }\n      socket.destroy();\n    });\n  }\n  this.webSocketsMapping[handlerPathname] = handlerServer;\n}\n\n/**\n * Returns web socket handlers registered for the given server\n * instance.\n * It is expected this function is called in Express\n * server instance context.\n *\n * @this {AppiumServer}\n * @param {string?} [keysFilter] - Only include pathnames with given\n * `keysFilter` value if set. All pairs will be included by default.\n * @returns {Promise<Record<string, import('ws').Server>>} pathnames to websocket server instances mapping matching the search criteria or an empty object otherwise.\n */\n// eslint-disable-next-line require-await\nasync function getWebSocketHandlers(keysFilter = null) {\n  if (_.isEmpty(this.webSocketsMapping)) {\n    return {};\n  }\n\n  return _.toPairs(this.webSocketsMapping).reduce((acc, [pathname, wsServer]) => {\n    if (!_.isString(keysFilter) || pathname.includes(keysFilter)) {\n      acc[pathname] = wsServer;\n    }\n    return acc;\n  }, {});\n}\n\n/**\n * Removes existing websocket handler from express server instance.\n * The call is ignored if the given `handlerPathname` handler\n * is not present in the handlers list.\n * It is expected this function is called in Express\n * server instance context.\n * @this {AppiumServer}\n * @param {string} handlerPathname - Websocket endpoint path.\n * @returns {Promise<boolean>} true if the handlerPathname was found and deleted\n */\n// eslint-disable-next-line require-await\nasync function removeWebSocketHandler(handlerPathname) {\n  const wsServer = this.webSocketsMapping?.[handlerPathname];\n  if (!wsServer) {\n    return false;\n  }\n\n  try {\n    wsServer.close();\n    for (const client of wsServer.clients || []) {\n      client.terminate();\n    }\n    return true;\n  } catch (ign) {\n    // ignore\n  } finally {\n    delete this.webSocketsMapping[handlerPathname];\n  }\n  return false;\n}\n\n/**\n * Removes all existing websocket handler from express server instance.\n * It is expected this function is called in Express\n * server instance context.\n * @this {AppiumServer}\n * @returns {Promise<boolean>} true if at least one handler has been deleted\n */\nasync function removeAllWebSocketHandlers() {\n  if (_.isEmpty(this.webSocketsMapping)) {\n    return false;\n  }\n\n  return _.some(\n    await B.all(\n      _.keys(this.webSocketsMapping).map((pathname) => this.removeWebSocketHandler(pathname))\n    )\n  );\n}\n\nexport {\n  addWebSocketHandler,\n  removeWebSocketHandler,\n  removeAllWebSocketHandlers,\n  getWebSocketHandlers,\n  DEFAULT_WS_PATHNAME_PREFIX,\n};\n\n/**\n * @typedef {import('@appium/types').AppiumServer} AppiumServer\n */\n"],"mappings":";;;;;;;;;;;;;;;AACA;;AACA;;AACA;;AAEA,MAAMA,0BAA0B,GAAG,KAAnC;;;AAgBA,eAAeC,mBAAf,CAAmCC,eAAnC,EAAoDC,aAApD,EAAmE;EACjE,MAAMC,MAAM,GAAgC,IAA5C;;EACA,IAAIC,eAAA,CAAEC,WAAF,CAAc,KAAKC,iBAAnB,CAAJ,EAA2C;IACzCH,MAAM,CAACG,iBAAP,GAA2B,EAA3B;IAEAH,MAAM,CAACI,EAAP,CAAU,SAAV,EAAqB,CAACC,OAAD,EAAUC,MAAV,EAAkBC,IAAlB,KAA2B;MAC9C,IAAIC,eAAJ;;MACA,IAAI;QAEFA,eAAe,GAAG,IAAIC,QAAJ,CAAQJ,OAAO,CAACK,GAAhB,EAAqBC,QAAvC;MACD,CAHD,CAGE,MAAM;QACNH,eAAe,GAAGH,OAAO,CAACK,GAA1B;MACD;;MACD,KAAK,MAAM,CAACC,QAAD,EAAWC,QAAX,CAAX,IAAmCX,eAAA,CAAEY,OAAF,CAAUb,MAAM,CAACG,iBAAjB,CAAnC,EAAwE;QACtE,IAAIK,eAAe,KAAKG,QAAxB,EAAkC;UAChCC,QAAQ,CAACE,aAAT,CAAuBT,OAAvB,EAAgCC,MAAhC,EAAwCC,IAAxC,EAA+CQ,EAAD,IAAQ;YACpDH,QAAQ,CAACI,IAAT,CAAc,YAAd,EAA4BD,EAA5B,EAAgCV,OAAhC;UACD,CAFD;UAGA;QACD;MACF;;MACDC,MAAM,CAACW,OAAP;IACD,CAjBD;EAkBD;;EACD,KAAKd,iBAAL,CAAuBL,eAAvB,IAA0CC,aAA1C;AACD;;AAcD,eAAemB,oBAAf,CAAoCC,UAAU,GAAG,IAAjD,EAAuD;EACrD,IAAIlB,eAAA,CAAEmB,OAAF,CAAU,KAAKjB,iBAAf,CAAJ,EAAuC;IACrC,OAAO,EAAP;EACD;;EAED,OAAOF,eAAA,CAAEY,OAAF,CAAU,KAAKV,iBAAf,EAAkCkB,MAAlC,CAAyC,CAACC,GAAD,EAAM,CAACX,QAAD,EAAWC,QAAX,CAAN,KAA+B;IAC7E,IAAI,CAACX,eAAA,CAAEsB,QAAF,CAAWJ,UAAX,CAAD,IAA2BR,QAAQ,CAACa,QAAT,CAAkBL,UAAlB,CAA/B,EAA8D;MAC5DG,GAAG,CAACX,QAAD,CAAH,GAAgBC,QAAhB;IACD;;IACD,OAAOU,GAAP;EACD,CALM,EAKJ,EALI,CAAP;AAMD;;AAaD,eAAeG,sBAAf,CAAsC3B,eAAtC,EAAuD;EAAA;;EACrD,MAAMc,QAAQ,4BAAG,KAAKT,iBAAR,0DAAG,sBAAyBL,eAAzB,CAAjB;;EACA,IAAI,CAACc,QAAL,EAAe;IACb,OAAO,KAAP;EACD;;EAED,IAAI;IACFA,QAAQ,CAACc,KAAT;;IACA,KAAK,MAAMC,MAAX,IAAqBf,QAAQ,CAACgB,OAAT,IAAoB,EAAzC,EAA6C;MAC3CD,MAAM,CAACE,SAAP;IACD;;IACD,OAAO,IAAP;EACD,CAND,CAME,OAAOC,GAAP,EAAY,CAEb,CARD,SAQU;IACR,OAAO,KAAK3B,iBAAL,CAAuBL,eAAvB,CAAP;EACD;;EACD,OAAO,KAAP;AACD;;AASD,eAAeiC,0BAAf,GAA4C;EAC1C,IAAI9B,eAAA,CAAEmB,OAAF,CAAU,KAAKjB,iBAAf,CAAJ,EAAuC;IACrC,OAAO,KAAP;EACD;;EAED,OAAOF,eAAA,CAAE+B,IAAF,CACL,MAAMC,iBAAA,CAAEC,GAAF,CACJjC,eAAA,CAAEkC,IAAF,CAAO,KAAKhC,iBAAZ,EAA+BiC,GAA/B,CAAoCzB,QAAD,IAAc,KAAKc,sBAAL,CAA4Bd,QAA5B,CAAjD,CADI,CADD,CAAP;AAKD"}
@@ -1,184 +1,15 @@
1
1
  export { DriverCore } from "./basedriver/core";
2
+ export { DeviceSettings } from "./basedriver/device-settings";
3
+ export { BaseDriver };
2
4
  export default BaseDriver;
3
- export { PREFIXED_APPIUM_OPTS_CAP } from "./basedriver/capabilities";
5
+ export * from "./protocol";
6
+ export { errorFromMJSONWPStatusCode as errorFromCode } from "./protocol";
7
+ export { STATIC_DIR } from "./express/static";
8
+ export { JWProxy } from "./jsonwp-proxy/proxy";
9
+ export { DEFAULT_WS_PATHNAME_PREFIX } from "./express/websocket";
4
10
  import { BaseDriver } from "./basedriver/driver";
5
- import { DeviceSettings } from "./basedriver/device-settings";
6
- export const routeConfiguringFunction: typeof protocol.routeConfiguringFunction;
7
- export const errors: {
8
- NotYetImplementedError: typeof import("./protocol/errors").NotYetImplementedError;
9
- NotImplementedError: typeof import("./protocol/errors").NotImplementedError;
10
- BadParametersError: typeof import("./protocol/errors").BadParametersError;
11
- InvalidArgumentError: typeof import("./protocol/errors").InvalidArgumentError;
12
- NoSuchDriverError: typeof import("./protocol/errors").NoSuchDriverError;
13
- NoSuchElementError: typeof import("./protocol/errors").NoSuchElementError;
14
- UnknownCommandError: typeof import("./protocol/errors").UnknownCommandError;
15
- StaleElementReferenceError: typeof import("./protocol/errors").StaleElementReferenceError;
16
- ElementNotVisibleError: typeof import("./protocol/errors").ElementNotVisibleError;
17
- InvalidElementStateError: typeof import("./protocol/errors").InvalidElementStateError;
18
- UnknownError: typeof import("./protocol/errors").UnknownError;
19
- ElementIsNotSelectableError: typeof import("./protocol/errors").ElementIsNotSelectableError;
20
- ElementClickInterceptedError: typeof import("./protocol/errors").ElementClickInterceptedError;
21
- ElementNotInteractableError: typeof import("./protocol/errors").ElementNotInteractableError;
22
- InsecureCertificateError: typeof import("./protocol/errors").InsecureCertificateError;
23
- JavaScriptError: typeof import("./protocol/errors").JavaScriptError;
24
- XPathLookupError: typeof import("./protocol/errors").XPathLookupError;
25
- TimeoutError: typeof import("./protocol/errors").TimeoutError;
26
- NoSuchWindowError: typeof import("./protocol/errors").NoSuchWindowError;
27
- NoSuchCookieError: typeof import("./protocol/errors").NoSuchCookieError;
28
- InvalidCookieDomainError: typeof import("./protocol/errors").InvalidCookieDomainError;
29
- InvalidCoordinatesError: typeof import("./protocol/errors").InvalidCoordinatesError;
30
- UnableToSetCookieError: typeof import("./protocol/errors").UnableToSetCookieError;
31
- UnexpectedAlertOpenError: typeof import("./protocol/errors").UnexpectedAlertOpenError;
32
- NoAlertOpenError: typeof import("./protocol/errors").NoAlertOpenError;
33
- ScriptTimeoutError: typeof import("./protocol/errors").ScriptTimeoutError;
34
- InvalidElementCoordinatesError: typeof import("./protocol/errors").InvalidElementCoordinatesError;
35
- IMENotAvailableError: typeof import("./protocol/errors").IMENotAvailableError;
36
- IMEEngineActivationFailedError: typeof import("./protocol/errors").IMEEngineActivationFailedError;
37
- InvalidSelectorError: typeof import("./protocol/errors").InvalidSelectorError;
38
- SessionNotCreatedError: typeof import("./protocol/errors").SessionNotCreatedError;
39
- MoveTargetOutOfBoundsError: typeof import("./protocol/errors").MoveTargetOutOfBoundsError;
40
- NoSuchAlertError: typeof import("./protocol/errors").NoSuchAlertError;
41
- NoSuchContextError: typeof import("./protocol/errors").NoSuchContextError;
42
- InvalidContextError: typeof import("./protocol/errors").InvalidContextError;
43
- NoSuchFrameError: typeof import("./protocol/errors").NoSuchFrameError;
44
- UnableToCaptureScreen: typeof import("./protocol/errors").UnableToCaptureScreen;
45
- UnknownMethodError: typeof import("./protocol/errors").UnknownMethodError;
46
- UnsupportedOperationError: typeof import("./protocol/errors").UnsupportedOperationError;
47
- ProxyRequestError: typeof import("./protocol/errors").ProxyRequestError;
48
- };
49
- export const isErrorType: typeof protocol.isErrorType;
50
- import { PROTOCOLS } from "./constants";
51
- export const errorFromMJSONWPStatusCode: typeof protocol.errorFromMJSONWPStatusCode;
52
- export const errorFromW3CJsonCode: typeof protocol.errorFromW3CJsonCode;
53
- export const determineProtocol: typeof protocol.determineProtocol;
54
- export const ALL_COMMANDS: any[];
55
- export const METHOD_MAP: import("./protocol/routes").MethodMap;
56
- export const routeToCommandName: typeof protocol.routeToCommandName;
57
- export const NO_SESSION_ID_COMMANDS: string[];
58
- export const isSessionCommand: typeof protocol.isSessionCommand;
59
- import { DEFAULT_BASE_PATH } from "./constants";
60
- export const CREATE_SESSION_COMMAND: "createSession";
61
- export const DELETE_SESSION_COMMAND: "deleteSession";
62
- export const GET_STATUS_COMMAND: "getStatus";
63
- export const STATIC_DIR: string;
64
- export const server: typeof serverIndex.server;
65
- export const normalizeBasePath: typeof serverIndex.normalizeBasePath;
66
- export const JWProxy: typeof proxyIndex.JWProxy;
67
- export const statusCodes: {
68
- Success: {
69
- code: number;
70
- summary: string;
71
- };
72
- NoSuchDriver: {
73
- code: number;
74
- summary: string;
75
- };
76
- NoSuchElement: {
77
- code: number;
78
- summary: string;
79
- };
80
- NoSuchFrame: {
81
- code: number;
82
- summary: string;
83
- };
84
- UnknownCommand: {
85
- code: number;
86
- summary: string;
87
- };
88
- StaleElementReference: {
89
- code: number;
90
- summary: string;
91
- };
92
- ElementNotVisible: {
93
- code: number;
94
- summary: string;
95
- };
96
- InvalidElementState: {
97
- code: number;
98
- summary: string;
99
- };
100
- UnknownError: {
101
- code: number;
102
- summary: string;
103
- };
104
- ElementIsNotSelectable: {
105
- code: number;
106
- summary: string;
107
- };
108
- JavaScriptError: {
109
- code: number;
110
- summary: string;
111
- };
112
- XPathLookupError: {
113
- code: number;
114
- summary: string;
115
- };
116
- Timeout: {
117
- code: number;
118
- summary: string;
119
- };
120
- NoSuchWindow: {
121
- code: number;
122
- summary: string;
123
- };
124
- InvalidCookieDomain: {
125
- code: number;
126
- summary: string;
127
- };
128
- UnableToSetCookie: {
129
- code: number;
130
- summary: string;
131
- };
132
- UnexpectedAlertOpen: {
133
- code: number;
134
- summary: string;
135
- };
136
- NoAlertOpenError: {
137
- code: number;
138
- summary: string;
139
- };
140
- ScriptTimeout: {
141
- code: number;
142
- summary: string;
143
- };
144
- InvalidElementCoordinates: {
145
- code: number;
146
- summary: string;
147
- };
148
- IMENotAvailable: {
149
- code: number;
150
- summary: string;
151
- };
152
- IMEEngineActivationFailed: {
153
- code: number;
154
- summary: string;
155
- };
156
- InvalidSelector: {
157
- code: number;
158
- summary: string;
159
- };
160
- SessionNotCreatedException: {
161
- code: number;
162
- summary: string;
163
- };
164
- MoveTargetOutOfBounds: {
165
- code: number;
166
- summary: string;
167
- };
168
- NoSuchContext: {
169
- code: number;
170
- summary: string;
171
- };
172
- };
173
- export const getSummaryByCode: typeof statusIndex.getSummaryByCode;
174
- export const processCapabilities: typeof caps.processCapabilities;
175
- export const isStandardCap: typeof caps.isStandardCap;
176
- export const validateCaps: typeof caps.validateCaps;
177
- export const DEFAULT_WS_PATHNAME_PREFIX: "/ws";
178
- import * as protocol from "./protocol";
179
- import * as serverIndex from "./express/server";
180
- import * as proxyIndex from "./jsonwp-proxy/proxy";
181
- import * as statusIndex from "./jsonwp-status/status";
182
- import * as caps from "./basedriver/capabilities";
183
- export { BaseDriver, DeviceSettings, PROTOCOLS, errorFromMJSONWPStatusCode as errorFromCode, DEFAULT_BASE_PATH };
11
+ export { DEFAULT_BASE_PATH, PROTOCOLS } from "./constants";
12
+ export { server, normalizeBasePath } from "./express/server";
13
+ export { getSummaryByCode, codes as statusCodes } from "./jsonwp-status/status";
14
+ export { PREFIXED_APPIUM_OPTS_CAP, processCapabilities, isStandardCap, validateCaps } from "./basedriver/capabilities";
184
15
  //# sourceMappingURL=index.d.ts.map