appium-android-driver 5.4.2 → 5.4.4

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 (67) hide show
  1. package/build/index.js +1 -9
  2. package/build/lib/android-helpers.js +10 -194
  3. package/build/lib/android-helpers.js.map +1 -1
  4. package/build/lib/bootstrap.js +4 -45
  5. package/build/lib/bootstrap.js.map +1 -1
  6. package/build/lib/commands/actions.js +3 -78
  7. package/build/lib/commands/actions.js.map +1 -1
  8. package/build/lib/commands/alert.js +3 -10
  9. package/build/lib/commands/alert.js.map +1 -1
  10. package/build/lib/commands/app-management.js +4 -32
  11. package/build/lib/commands/app-management.js.map +1 -1
  12. package/build/lib/commands/context.js +6 -78
  13. package/build/lib/commands/context.js.map +1 -1
  14. package/build/lib/commands/coverage.js +1 -6
  15. package/build/lib/commands/coverage.js.map +1 -1
  16. package/build/lib/commands/element.js +4 -47
  17. package/build/lib/commands/element.js.map +1 -1
  18. package/build/lib/commands/emu-console.js +2 -7
  19. package/build/lib/commands/emu-console.js.map +1 -1
  20. package/build/lib/commands/execute.js +1 -13
  21. package/build/lib/commands/execute.js.map +1 -1
  22. package/build/lib/commands/file-actions.js +3 -44
  23. package/build/lib/commands/file-actions.js.map +1 -1
  24. package/build/lib/commands/find.js +3 -16
  25. package/build/lib/commands/find.js.map +1 -1
  26. package/build/lib/commands/general.js +4 -78
  27. package/build/lib/commands/general.js.map +1 -1
  28. package/build/lib/commands/ime.js +3 -14
  29. package/build/lib/commands/ime.js.map +1 -1
  30. package/build/lib/commands/index.js +3 -27
  31. package/build/lib/commands/index.js.map +1 -1
  32. package/build/lib/commands/intent.js +1 -40
  33. package/build/lib/commands/intent.js.map +1 -1
  34. package/build/lib/commands/log.js +3 -35
  35. package/build/lib/commands/log.js.map +1 -1
  36. package/build/lib/commands/media-projection.js +2 -47
  37. package/build/lib/commands/media-projection.js.map +1 -1
  38. package/build/lib/commands/network.js +4 -40
  39. package/build/lib/commands/network.js.map +1 -1
  40. package/build/lib/commands/performance.js +5 -57
  41. package/build/lib/commands/performance.js.map +1 -1
  42. package/build/lib/commands/recordscreen.js +1 -59
  43. package/build/lib/commands/recordscreen.js.map +1 -1
  44. package/build/lib/commands/shell.js +1 -15
  45. package/build/lib/commands/shell.js.map +1 -1
  46. package/build/lib/commands/streamscreen.js +7 -74
  47. package/build/lib/commands/streamscreen.js.map +1 -1
  48. package/build/lib/commands/system-bars.js +1 -23
  49. package/build/lib/commands/system-bars.js.map +1 -1
  50. package/build/lib/commands/touch.js +13 -79
  51. package/build/lib/commands/touch.js.map +1 -1
  52. package/build/lib/desired-caps.js +3 -4
  53. package/build/lib/desired-caps.js.map +1 -1
  54. package/build/lib/driver.js +8 -97
  55. package/build/lib/driver.js.map +1 -1
  56. package/build/lib/logger.js +1 -5
  57. package/build/lib/logger.js.map +1 -1
  58. package/build/lib/uiautomator.js +3 -24
  59. package/build/lib/uiautomator.js.map +1 -1
  60. package/build/lib/unlock-helpers.js +1 -61
  61. package/build/lib/unlock-helpers.js.map +1 -1
  62. package/build/lib/utils.js +1 -7
  63. package/build/lib/utils.js.map +1 -1
  64. package/build/lib/webview-helpers.js +1 -94
  65. package/build/lib/webview-helpers.js.map +1 -1
  66. package/lib/android-helpers.js +2 -1
  67. package/package.json +2 -2
@@ -1,26 +1,18 @@
1
1
  "use strict";
2
2
 
3
3
  var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
4
  Object.defineProperty(exports, "__esModule", {
6
5
  value: true
7
6
  });
8
7
  exports.default = exports.commands = void 0;
9
-
10
8
  require("source-map-support/register");
11
-
12
9
  var _lodash = _interopRequireDefault(require("lodash"));
13
-
14
10
  var _teen_process = require("teen_process");
15
-
16
11
  var _support = require("appium/support");
17
-
18
12
  var _driver = require("appium/driver");
19
-
20
13
  const ADB_SHELL_FEATURE = 'adb_shell';
21
14
  let commands = {};
22
15
  exports.commands = commands;
23
-
24
16
  commands.mobileShell = async function mobileShell(opts = {}) {
25
17
  this.ensureFeatureEnabled(ADB_SHELL_FEATURE);
26
18
  const {
@@ -29,14 +21,11 @@ commands.mobileShell = async function mobileShell(opts = {}) {
29
21
  timeout = 20000,
30
22
  includeStderr
31
23
  } = opts;
32
-
33
24
  if (!_lodash.default.isString(command)) {
34
25
  throw new _driver.errors.InvalidArgumentError(`The 'command' argument is mandatory`);
35
26
  }
36
-
37
27
  const adbArgs = [...this.adb.executable.defaultArgs, 'shell', command, ...(_lodash.default.isArray(args) ? args : [args])];
38
28
  this.log.debug(`Running '${this.adb.executable.path} ${_support.util.quote(adbArgs)}'`);
39
-
40
29
  try {
41
30
  const {
42
31
  stdout,
@@ -44,20 +33,17 @@ commands.mobileShell = async function mobileShell(opts = {}) {
44
33
  } = await (0, _teen_process.exec)(this.adb.executable.path, adbArgs, {
45
34
  timeout
46
35
  });
47
-
48
36
  if (includeStderr) {
49
37
  return {
50
38
  stdout,
51
39
  stderr
52
40
  };
53
41
  }
54
-
55
42
  return stdout;
56
43
  } catch (err) {
57
44
  this.log.errorAndThrow(`Cannot execute the '${command}' shell command. ` + `Original error: ${err.message}. ` + `StdOut: ${err.stdout}. StdErr: ${err.stderr}`);
58
45
  }
59
46
  };
60
-
61
47
  var _default = commands;
62
48
  exports.default = _default;
63
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJBREJfU0hFTExfRkVBVFVSRSIsImNvbW1hbmRzIiwibW9iaWxlU2hlbGwiLCJvcHRzIiwiZW5zdXJlRmVhdHVyZUVuYWJsZWQiLCJjb21tYW5kIiwiYXJncyIsInRpbWVvdXQiLCJpbmNsdWRlU3RkZXJyIiwiXyIsImlzU3RyaW5nIiwiZXJyb3JzIiwiSW52YWxpZEFyZ3VtZW50RXJyb3IiLCJhZGJBcmdzIiwiYWRiIiwiZXhlY3V0YWJsZSIsImRlZmF1bHRBcmdzIiwiaXNBcnJheSIsImxvZyIsImRlYnVnIiwicGF0aCIsInV0aWwiLCJxdW90ZSIsInN0ZG91dCIsInN0ZGVyciIsImV4ZWMiLCJlcnIiLCJlcnJvckFuZFRocm93IiwibWVzc2FnZSJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2xpYi9jb21tYW5kcy9zaGVsbC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgXyBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0IHsgZXhlYyB9IGZyb20gJ3RlZW5fcHJvY2Vzcyc7XG5pbXBvcnQgeyB1dGlsIH0gZnJvbSAnYXBwaXVtL3N1cHBvcnQnO1xuaW1wb3J0IHsgZXJyb3JzIH0gZnJvbSAnYXBwaXVtL2RyaXZlcic7XG5cbmNvbnN0IEFEQl9TSEVMTF9GRUFUVVJFID0gJ2FkYl9zaGVsbCc7XG5cbmxldCBjb21tYW5kcyA9IHt9O1xuXG5jb21tYW5kcy5tb2JpbGVTaGVsbCA9IGFzeW5jIGZ1bmN0aW9uIG1vYmlsZVNoZWxsIChvcHRzID0ge30pIHtcbiAgdGhpcy5lbnN1cmVGZWF0dXJlRW5hYmxlZChBREJfU0hFTExfRkVBVFVSRSk7XG5cbiAgY29uc3Qge1xuICAgIGNvbW1hbmQsXG4gICAgYXJncyA9IFtdLFxuICAgIHRpbWVvdXQgPSAyMDAwMCxcbiAgICBpbmNsdWRlU3RkZXJyLFxuICB9ID0gb3B0cztcblxuICBpZiAoIV8uaXNTdHJpbmcoY29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLkludmFsaWRBcmd1bWVudEVycm9yKGBUaGUgJ2NvbW1hbmQnIGFyZ3VtZW50IGlzIG1hbmRhdG9yeWApO1xuICB9XG5cbiAgY29uc3QgYWRiQXJncyA9IFtcbiAgICAuLi50aGlzLmFkYi5leGVjdXRhYmxlLmRlZmF1bHRBcmdzLFxuICAgICdzaGVsbCcsXG4gICAgY29tbWFuZCxcbiAgICAuLi4oXy5pc0FycmF5KGFyZ3MpID8gYXJncyA6IFthcmdzXSlcbiAgXTtcbiAgdGhpcy5sb2cuZGVidWcoYFJ1bm5pbmcgJyR7dGhpcy5hZGIuZXhlY3V0YWJsZS5wYXRofSAke3V0aWwucXVvdGUoYWRiQXJncyl9J2ApO1xuICB0cnkge1xuICAgIGNvbnN0IHtzdGRvdXQsIHN0ZGVycn0gPSBhd2FpdCBleGVjKHRoaXMuYWRiLmV4ZWN1dGFibGUucGF0aCwgYWRiQXJncywge3RpbWVvdXR9KTtcbiAgICBpZiAoaW5jbHVkZVN0ZGVycikge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgc3Rkb3V0LFxuICAgICAgICBzdGRlcnJcbiAgICAgIH07XG4gICAgfVxuICAgIHJldHVybiBzdGRvdXQ7XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIHRoaXMubG9nLmVycm9yQW5kVGhyb3coYENhbm5vdCBleGVjdXRlIHRoZSAnJHtjb21tYW5kfScgc2hlbGwgY29tbWFuZC4gYCArXG4gICAgICBgT3JpZ2luYWwgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9LiBgICtcbiAgICAgIGBTdGRPdXQ6ICR7ZXJyLnN0ZG91dH0uIFN0ZEVycjogJHtlcnIuc3RkZXJyfWApO1xuICB9XG59O1xuXG5leHBvcnQgeyBjb21tYW5kcyB9O1xuZXhwb3J0IGRlZmF1bHQgY29tbWFuZHM7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBRUEsTUFBTUEsaUJBQWlCLEdBQUcsV0FBMUI7QUFFQSxJQUFJQyxRQUFRLEdBQUcsRUFBZjs7O0FBRUFBLFFBQVEsQ0FBQ0MsV0FBVCxHQUF1QixlQUFlQSxXQUFmLENBQTRCQyxJQUFJLEdBQUcsRUFBbkMsRUFBdUM7RUFDNUQsS0FBS0Msb0JBQUwsQ0FBMEJKLGlCQUExQjtFQUVBLE1BQU07SUFDSkssT0FESTtJQUVKQyxJQUFJLEdBQUcsRUFGSDtJQUdKQyxPQUFPLEdBQUcsS0FITjtJQUlKQztFQUpJLElBS0ZMLElBTEo7O0VBT0EsSUFBSSxDQUFDTSxlQUFBLENBQUVDLFFBQUYsQ0FBV0wsT0FBWCxDQUFMLEVBQTBCO0lBQ3hCLE1BQU0sSUFBSU0sY0FBQSxDQUFPQyxvQkFBWCxDQUFpQyxxQ0FBakMsQ0FBTjtFQUNEOztFQUVELE1BQU1DLE9BQU8sR0FBRyxDQUNkLEdBQUcsS0FBS0MsR0FBTCxDQUFTQyxVQUFULENBQW9CQyxXQURULEVBRWQsT0FGYyxFQUdkWCxPQUhjLEVBSWQsSUFBSUksZUFBQSxDQUFFUSxPQUFGLENBQVVYLElBQVYsSUFBa0JBLElBQWxCLEdBQXlCLENBQUNBLElBQUQsQ0FBN0IsQ0FKYyxDQUFoQjtFQU1BLEtBQUtZLEdBQUwsQ0FBU0MsS0FBVCxDQUFnQixZQUFXLEtBQUtMLEdBQUwsQ0FBU0MsVUFBVCxDQUFvQkssSUFBSyxJQUFHQyxhQUFBLENBQUtDLEtBQUwsQ0FBV1QsT0FBWCxDQUFvQixHQUEzRTs7RUFDQSxJQUFJO0lBQ0YsTUFBTTtNQUFDVSxNQUFEO01BQVNDO0lBQVQsSUFBbUIsTUFBTSxJQUFBQyxrQkFBQSxFQUFLLEtBQUtYLEdBQUwsQ0FBU0MsVUFBVCxDQUFvQkssSUFBekIsRUFBK0JQLE9BQS9CLEVBQXdDO01BQUNOO0lBQUQsQ0FBeEMsQ0FBL0I7O0lBQ0EsSUFBSUMsYUFBSixFQUFtQjtNQUNqQixPQUFPO1FBQ0xlLE1BREs7UUFFTEM7TUFGSyxDQUFQO0lBSUQ7O0lBQ0QsT0FBT0QsTUFBUDtFQUNELENBVEQsQ0FTRSxPQUFPRyxHQUFQLEVBQVk7SUFDWixLQUFLUixHQUFMLENBQVNTLGFBQVQsQ0FBd0IsdUJBQXNCdEIsT0FBUSxtQkFBL0IsR0FDcEIsbUJBQWtCcUIsR0FBRyxDQUFDRSxPQUFRLElBRFYsR0FFcEIsV0FBVUYsR0FBRyxDQUFDSCxNQUFPLGFBQVlHLEdBQUcsQ0FBQ0YsTUFBTyxFQUYvQztFQUdEO0FBQ0YsQ0FuQ0Q7O2VBc0NldkIsUSJ9
49
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJBREJfU0hFTExfRkVBVFVSRSIsImNvbW1hbmRzIiwibW9iaWxlU2hlbGwiLCJvcHRzIiwiZW5zdXJlRmVhdHVyZUVuYWJsZWQiLCJjb21tYW5kIiwiYXJncyIsInRpbWVvdXQiLCJpbmNsdWRlU3RkZXJyIiwiXyIsImlzU3RyaW5nIiwiZXJyb3JzIiwiSW52YWxpZEFyZ3VtZW50RXJyb3IiLCJhZGJBcmdzIiwiYWRiIiwiZXhlY3V0YWJsZSIsImRlZmF1bHRBcmdzIiwiaXNBcnJheSIsImxvZyIsImRlYnVnIiwicGF0aCIsInV0aWwiLCJxdW90ZSIsInN0ZG91dCIsInN0ZGVyciIsImV4ZWMiLCJlcnIiLCJlcnJvckFuZFRocm93IiwibWVzc2FnZSJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2xpYi9jb21tYW5kcy9zaGVsbC5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgXyBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0IHsgZXhlYyB9IGZyb20gJ3RlZW5fcHJvY2Vzcyc7XG5pbXBvcnQgeyB1dGlsIH0gZnJvbSAnYXBwaXVtL3N1cHBvcnQnO1xuaW1wb3J0IHsgZXJyb3JzIH0gZnJvbSAnYXBwaXVtL2RyaXZlcic7XG5cbmNvbnN0IEFEQl9TSEVMTF9GRUFUVVJFID0gJ2FkYl9zaGVsbCc7XG5cbmxldCBjb21tYW5kcyA9IHt9O1xuXG5jb21tYW5kcy5tb2JpbGVTaGVsbCA9IGFzeW5jIGZ1bmN0aW9uIG1vYmlsZVNoZWxsIChvcHRzID0ge30pIHtcbiAgdGhpcy5lbnN1cmVGZWF0dXJlRW5hYmxlZChBREJfU0hFTExfRkVBVFVSRSk7XG5cbiAgY29uc3Qge1xuICAgIGNvbW1hbmQsXG4gICAgYXJncyA9IFtdLFxuICAgIHRpbWVvdXQgPSAyMDAwMCxcbiAgICBpbmNsdWRlU3RkZXJyLFxuICB9ID0gb3B0cztcblxuICBpZiAoIV8uaXNTdHJpbmcoY29tbWFuZCkpIHtcbiAgICB0aHJvdyBuZXcgZXJyb3JzLkludmFsaWRBcmd1bWVudEVycm9yKGBUaGUgJ2NvbW1hbmQnIGFyZ3VtZW50IGlzIG1hbmRhdG9yeWApO1xuICB9XG5cbiAgY29uc3QgYWRiQXJncyA9IFtcbiAgICAuLi50aGlzLmFkYi5leGVjdXRhYmxlLmRlZmF1bHRBcmdzLFxuICAgICdzaGVsbCcsXG4gICAgY29tbWFuZCxcbiAgICAuLi4oXy5pc0FycmF5KGFyZ3MpID8gYXJncyA6IFthcmdzXSlcbiAgXTtcbiAgdGhpcy5sb2cuZGVidWcoYFJ1bm5pbmcgJyR7dGhpcy5hZGIuZXhlY3V0YWJsZS5wYXRofSAke3V0aWwucXVvdGUoYWRiQXJncyl9J2ApO1xuICB0cnkge1xuICAgIGNvbnN0IHtzdGRvdXQsIHN0ZGVycn0gPSBhd2FpdCBleGVjKHRoaXMuYWRiLmV4ZWN1dGFibGUucGF0aCwgYWRiQXJncywge3RpbWVvdXR9KTtcbiAgICBpZiAoaW5jbHVkZVN0ZGVycikge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgc3Rkb3V0LFxuICAgICAgICBzdGRlcnJcbiAgICAgIH07XG4gICAgfVxuICAgIHJldHVybiBzdGRvdXQ7XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIHRoaXMubG9nLmVycm9yQW5kVGhyb3coYENhbm5vdCBleGVjdXRlIHRoZSAnJHtjb21tYW5kfScgc2hlbGwgY29tbWFuZC4gYCArXG4gICAgICBgT3JpZ2luYWwgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9LiBgICtcbiAgICAgIGBTdGRPdXQ6ICR7ZXJyLnN0ZG91dH0uIFN0ZEVycjogJHtlcnIuc3RkZXJyfWApO1xuICB9XG59O1xuXG5leHBvcnQgeyBjb21tYW5kcyB9O1xuZXhwb3J0IGRlZmF1bHQgY29tbWFuZHM7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7O0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFFQSxNQUFNQSxpQkFBaUIsR0FBRyxXQUFXO0FBRXJDLElBQUlDLFFBQVEsR0FBRyxDQUFDLENBQUM7QUFBQztBQUVsQkEsUUFBUSxDQUFDQyxXQUFXLEdBQUcsZUFBZUEsV0FBVyxDQUFFQyxJQUFJLEdBQUcsQ0FBQyxDQUFDLEVBQUU7RUFDNUQsSUFBSSxDQUFDQyxvQkFBb0IsQ0FBQ0osaUJBQWlCLENBQUM7RUFFNUMsTUFBTTtJQUNKSyxPQUFPO0lBQ1BDLElBQUksR0FBRyxFQUFFO0lBQ1RDLE9BQU8sR0FBRyxLQUFLO0lBQ2ZDO0VBQ0YsQ0FBQyxHQUFHTCxJQUFJO0VBRVIsSUFBSSxDQUFDTSxlQUFDLENBQUNDLFFBQVEsQ0FBQ0wsT0FBTyxDQUFDLEVBQUU7SUFDeEIsTUFBTSxJQUFJTSxjQUFNLENBQUNDLG9CQUFvQixDQUFFLHFDQUFvQyxDQUFDO0VBQzlFO0VBRUEsTUFBTUMsT0FBTyxHQUFHLENBQ2QsR0FBRyxJQUFJLENBQUNDLEdBQUcsQ0FBQ0MsVUFBVSxDQUFDQyxXQUFXLEVBQ2xDLE9BQU8sRUFDUFgsT0FBTyxFQUNQLElBQUlJLGVBQUMsQ0FBQ1EsT0FBTyxDQUFDWCxJQUFJLENBQUMsR0FBR0EsSUFBSSxHQUFHLENBQUNBLElBQUksQ0FBQyxDQUFDLENBQ3JDO0VBQ0QsSUFBSSxDQUFDWSxHQUFHLENBQUNDLEtBQUssQ0FBRSxZQUFXLElBQUksQ0FBQ0wsR0FBRyxDQUFDQyxVQUFVLENBQUNLLElBQUssSUFBR0MsYUFBSSxDQUFDQyxLQUFLLENBQUNULE9BQU8sQ0FBRSxHQUFFLENBQUM7RUFDOUUsSUFBSTtJQUNGLE1BQU07TUFBQ1UsTUFBTTtNQUFFQztJQUFNLENBQUMsR0FBRyxNQUFNLElBQUFDLGtCQUFJLEVBQUMsSUFBSSxDQUFDWCxHQUFHLENBQUNDLFVBQVUsQ0FBQ0ssSUFBSSxFQUFFUCxPQUFPLEVBQUU7TUFBQ047SUFBTyxDQUFDLENBQUM7SUFDakYsSUFBSUMsYUFBYSxFQUFFO01BQ2pCLE9BQU87UUFDTGUsTUFBTTtRQUNOQztNQUNGLENBQUM7SUFDSDtJQUNBLE9BQU9ELE1BQU07RUFDZixDQUFDLENBQUMsT0FBT0csR0FBRyxFQUFFO0lBQ1osSUFBSSxDQUFDUixHQUFHLENBQUNTLGFBQWEsQ0FBRSx1QkFBc0J0QixPQUFRLG1CQUFrQixHQUNyRSxtQkFBa0JxQixHQUFHLENBQUNFLE9BQVEsSUFBRyxHQUNqQyxXQUFVRixHQUFHLENBQUNILE1BQU8sYUFBWUcsR0FBRyxDQUFDRixNQUFPLEVBQUMsQ0FBQztFQUNuRDtBQUNGLENBQUM7QUFBQyxlQUdhdkIsUUFBUTtBQUFBIn0=
@@ -1 +1 @@
1
- {"version":3,"file":"shell.js","names":["ADB_SHELL_FEATURE","commands","mobileShell","opts","ensureFeatureEnabled","command","args","timeout","includeStderr","_","isString","errors","InvalidArgumentError","adbArgs","adb","executable","defaultArgs","isArray","log","debug","path","util","quote","stdout","stderr","exec","err","errorAndThrow","message"],"sources":["../../../lib/commands/shell.js"],"sourcesContent":["import _ from 'lodash';\nimport { exec } from 'teen_process';\nimport { util } from 'appium/support';\nimport { errors } from 'appium/driver';\n\nconst ADB_SHELL_FEATURE = 'adb_shell';\n\nlet commands = {};\n\ncommands.mobileShell = async function mobileShell (opts = {}) {\n this.ensureFeatureEnabled(ADB_SHELL_FEATURE);\n\n const {\n command,\n args = [],\n timeout = 20000,\n includeStderr,\n } = opts;\n\n if (!_.isString(command)) {\n throw new errors.InvalidArgumentError(`The 'command' argument is mandatory`);\n }\n\n const adbArgs = [\n ...this.adb.executable.defaultArgs,\n 'shell',\n command,\n ...(_.isArray(args) ? args : [args])\n ];\n this.log.debug(`Running '${this.adb.executable.path} ${util.quote(adbArgs)}'`);\n try {\n const {stdout, stderr} = await exec(this.adb.executable.path, adbArgs, {timeout});\n if (includeStderr) {\n return {\n stdout,\n stderr\n };\n }\n return stdout;\n } catch (err) {\n this.log.errorAndThrow(`Cannot execute the '${command}' shell command. ` +\n `Original error: ${err.message}. ` +\n `StdOut: ${err.stdout}. StdErr: ${err.stderr}`);\n }\n};\n\nexport { commands };\nexport default commands;\n"],"mappings":";;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AAEA,MAAMA,iBAAiB,GAAG,WAA1B;AAEA,IAAIC,QAAQ,GAAG,EAAf;;;AAEAA,QAAQ,CAACC,WAAT,GAAuB,eAAeA,WAAf,CAA4BC,IAAI,GAAG,EAAnC,EAAuC;EAC5D,KAAKC,oBAAL,CAA0BJ,iBAA1B;EAEA,MAAM;IACJK,OADI;IAEJC,IAAI,GAAG,EAFH;IAGJC,OAAO,GAAG,KAHN;IAIJC;EAJI,IAKFL,IALJ;;EAOA,IAAI,CAACM,eAAA,CAAEC,QAAF,CAAWL,OAAX,CAAL,EAA0B;IACxB,MAAM,IAAIM,cAAA,CAAOC,oBAAX,CAAiC,qCAAjC,CAAN;EACD;;EAED,MAAMC,OAAO,GAAG,CACd,GAAG,KAAKC,GAAL,CAASC,UAAT,CAAoBC,WADT,EAEd,OAFc,EAGdX,OAHc,EAId,IAAII,eAAA,CAAEQ,OAAF,CAAUX,IAAV,IAAkBA,IAAlB,GAAyB,CAACA,IAAD,CAA7B,CAJc,CAAhB;EAMA,KAAKY,GAAL,CAASC,KAAT,CAAgB,YAAW,KAAKL,GAAL,CAASC,UAAT,CAAoBK,IAAK,IAAGC,aAAA,CAAKC,KAAL,CAAWT,OAAX,CAAoB,GAA3E;;EACA,IAAI;IACF,MAAM;MAACU,MAAD;MAASC;IAAT,IAAmB,MAAM,IAAAC,kBAAA,EAAK,KAAKX,GAAL,CAASC,UAAT,CAAoBK,IAAzB,EAA+BP,OAA/B,EAAwC;MAACN;IAAD,CAAxC,CAA/B;;IACA,IAAIC,aAAJ,EAAmB;MACjB,OAAO;QACLe,MADK;QAELC;MAFK,CAAP;IAID;;IACD,OAAOD,MAAP;EACD,CATD,CASE,OAAOG,GAAP,EAAY;IACZ,KAAKR,GAAL,CAASS,aAAT,CAAwB,uBAAsBtB,OAAQ,mBAA/B,GACpB,mBAAkBqB,GAAG,CAACE,OAAQ,IADV,GAEpB,WAAUF,GAAG,CAACH,MAAO,aAAYG,GAAG,CAACF,MAAO,EAF/C;EAGD;AACF,CAnCD;;eAsCevB,Q"}
1
+ {"version":3,"file":"shell.js","names":["ADB_SHELL_FEATURE","commands","mobileShell","opts","ensureFeatureEnabled","command","args","timeout","includeStderr","_","isString","errors","InvalidArgumentError","adbArgs","adb","executable","defaultArgs","isArray","log","debug","path","util","quote","stdout","stderr","exec","err","errorAndThrow","message"],"sources":["../../../lib/commands/shell.js"],"sourcesContent":["import _ from 'lodash';\nimport { exec } from 'teen_process';\nimport { util } from 'appium/support';\nimport { errors } from 'appium/driver';\n\nconst ADB_SHELL_FEATURE = 'adb_shell';\n\nlet commands = {};\n\ncommands.mobileShell = async function mobileShell (opts = {}) {\n this.ensureFeatureEnabled(ADB_SHELL_FEATURE);\n\n const {\n command,\n args = [],\n timeout = 20000,\n includeStderr,\n } = opts;\n\n if (!_.isString(command)) {\n throw new errors.InvalidArgumentError(`The 'command' argument is mandatory`);\n }\n\n const adbArgs = [\n ...this.adb.executable.defaultArgs,\n 'shell',\n command,\n ...(_.isArray(args) ? args : [args])\n ];\n this.log.debug(`Running '${this.adb.executable.path} ${util.quote(adbArgs)}'`);\n try {\n const {stdout, stderr} = await exec(this.adb.executable.path, adbArgs, {timeout});\n if (includeStderr) {\n return {\n stdout,\n stderr\n };\n }\n return stdout;\n } catch (err) {\n this.log.errorAndThrow(`Cannot execute the '${command}' shell command. ` +\n `Original error: ${err.message}. ` +\n `StdOut: ${err.stdout}. StdErr: ${err.stderr}`);\n }\n};\n\nexport { commands };\nexport default commands;\n"],"mappings":";;;;;;;;AAAA;AACA;AACA;AACA;AAEA,MAAMA,iBAAiB,GAAG,WAAW;AAErC,IAAIC,QAAQ,GAAG,CAAC,CAAC;AAAC;AAElBA,QAAQ,CAACC,WAAW,GAAG,eAAeA,WAAW,CAAEC,IAAI,GAAG,CAAC,CAAC,EAAE;EAC5D,IAAI,CAACC,oBAAoB,CAACJ,iBAAiB,CAAC;EAE5C,MAAM;IACJK,OAAO;IACPC,IAAI,GAAG,EAAE;IACTC,OAAO,GAAG,KAAK;IACfC;EACF,CAAC,GAAGL,IAAI;EAER,IAAI,CAACM,eAAC,CAACC,QAAQ,CAACL,OAAO,CAAC,EAAE;IACxB,MAAM,IAAIM,cAAM,CAACC,oBAAoB,CAAE,qCAAoC,CAAC;EAC9E;EAEA,MAAMC,OAAO,GAAG,CACd,GAAG,IAAI,CAACC,GAAG,CAACC,UAAU,CAACC,WAAW,EAClC,OAAO,EACPX,OAAO,EACP,IAAII,eAAC,CAACQ,OAAO,CAACX,IAAI,CAAC,GAAGA,IAAI,GAAG,CAACA,IAAI,CAAC,CAAC,CACrC;EACD,IAAI,CAACY,GAAG,CAACC,KAAK,CAAE,YAAW,IAAI,CAACL,GAAG,CAACC,UAAU,CAACK,IAAK,IAAGC,aAAI,CAACC,KAAK,CAACT,OAAO,CAAE,GAAE,CAAC;EAC9E,IAAI;IACF,MAAM;MAACU,MAAM;MAAEC;IAAM,CAAC,GAAG,MAAM,IAAAC,kBAAI,EAAC,IAAI,CAACX,GAAG,CAACC,UAAU,CAACK,IAAI,EAAEP,OAAO,EAAE;MAACN;IAAO,CAAC,CAAC;IACjF,IAAIC,aAAa,EAAE;MACjB,OAAO;QACLe,MAAM;QACNC;MACF,CAAC;IACH;IACA,OAAOD,MAAM;EACf,CAAC,CAAC,OAAOG,GAAG,EAAE;IACZ,IAAI,CAACR,GAAG,CAACS,aAAa,CAAE,uBAAsBtB,OAAQ,mBAAkB,GACrE,mBAAkBqB,GAAG,CAACE,OAAQ,IAAG,GACjC,WAAUF,GAAG,CAACH,MAAO,aAAYG,GAAG,CAACF,MAAO,EAAC,CAAC;EACnD;AACF,CAAC;AAAC,eAGavB,QAAQ;AAAA"}
@@ -1,34 +1,21 @@
1
1
  "use strict";
2
2
 
3
3
  var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
4
  Object.defineProperty(exports, "__esModule", {
6
5
  value: true
7
6
  });
8
7
  exports.default = void 0;
9
-
10
8
  require("source-map-support/register");
11
-
12
9
  var _lodash = _interopRequireDefault(require("lodash"));
13
-
14
10
  var _support = require("appium/support");
15
-
16
11
  var _teen_process = require("teen_process");
17
-
18
12
  var _portscanner = require("portscanner");
19
-
20
13
  var _http = _interopRequireDefault(require("http"));
21
-
22
14
  var _net = _interopRequireDefault(require("net"));
23
-
24
15
  var _bluebird = _interopRequireDefault(require("bluebird"));
25
-
26
16
  var _asyncbox = require("asyncbox");
27
-
28
17
  var _child_process = require("child_process");
29
-
30
18
  var _url = _interopRequireDefault(require("url"));
31
-
32
19
  const commands = {};
33
20
  const RECORDING_INTERVAL_SEC = 5;
34
21
  const STREAMING_STARTUP_TIMEOUT_MS = 5000;
@@ -50,21 +37,17 @@ const DEFAULT_QUALITY = 70;
50
37
  const DEFAULT_BITRATE = 4000000;
51
38
  const BOUNDARY_STRING = '--2ae9746887f170b8cf7c271047ce314c';
52
39
  const ADB_SCREEN_STREAMING_FEATURE = 'adb_screen_streaming';
53
-
54
40
  function createStreamingLogger(streamName, udid) {
55
41
  return _support.logger.getLogger(`${streamName}@` + _lodash.default.truncate(udid, {
56
42
  length: 8,
57
43
  omission: ''
58
44
  }));
59
45
  }
60
-
61
46
  async function verifyStreamingRequirements(adb) {
62
47
  if (!_lodash.default.trim(await adb.shell(['which', SCREENRECORD_BINARY]))) {
63
48
  throw new Error(`The required '${SCREENRECORD_BINARY}' binary is not available on the device under test`);
64
49
  }
65
-
66
50
  const gstreamerCheckPromises = [];
67
-
68
51
  for (const binaryName of [GSTREAMER_BINARY, GST_INSPECT_BINARY]) {
69
52
  gstreamerCheckPromises.push((async () => {
70
53
  try {
@@ -74,44 +57,34 @@ async function verifyStreamingRequirements(adb) {
74
57
  }
75
58
  })());
76
59
  }
77
-
78
60
  await _bluebird.default.all(gstreamerCheckPromises);
79
61
  const moduleCheckPromises = [];
80
-
81
62
  for (const [name, modName] of _lodash.default.toPairs(REQUIRED_GST_PLUGINS)) {
82
63
  moduleCheckPromises.push((async () => {
83
64
  const {
84
65
  stdout
85
66
  } = await (0, _teen_process.exec)(GST_INSPECT_BINARY, [name]);
86
-
87
67
  if (!_lodash.default.includes(stdout, modName)) {
88
68
  throw new Error(`The required GStreamer plugin '${name}' from '${modName}' module is not installed. ` + `See ${GST_TUTORIAL_URL} for more details on how to install it.`);
89
69
  }
90
70
  })());
91
71
  }
92
-
93
72
  await _bluebird.default.all(moduleCheckPromises);
94
73
  }
95
-
96
74
  async function getDeviceInfo(adb, log = null) {
97
75
  const output = await adb.shell(['dumpsys', 'display']);
98
76
  const result = {};
99
-
100
77
  for (const [key, pattern] of [['width', /\bdeviceWidth=(\d+)/], ['height', /\bdeviceHeight=(\d+)/], ['fps', /\bfps=(\d+)/]]) {
101
78
  const match = pattern.exec(output);
102
-
103
79
  if (!match) {
104
80
  log === null || log === void 0 ? void 0 : log.debug(output);
105
81
  throw new Error(`Cannot parse the device ${key} from the adb command output. ` + `Check the server log for more details.`);
106
82
  }
107
-
108
83
  result[key] = parseInt(match[1], 10);
109
84
  }
110
-
111
85
  result.udid = adb.curDeviceId;
112
86
  return result;
113
87
  }
114
-
115
88
  async function initDeviceStreamingProc(adb, log, deviceInfo, opts = {}) {
116
89
  const {
117
90
  width,
@@ -121,42 +94,35 @@ async function initDeviceStreamingProc(adb, log, deviceInfo, opts = {}) {
121
94
  const adjustedWidth = parseInt(width, 10) || deviceInfo.width;
122
95
  const adjustedHeight = parseInt(height, 10) || deviceInfo.height;
123
96
  const adjustedBitrate = parseInt(bitRate, 10) || DEFAULT_BITRATE;
124
- let screenRecordCmd = SCREENRECORD_BINARY + ` --output-format=h264` + ` --time-limit=${RECORDING_INTERVAL_SEC}`;
125
-
97
+ let screenRecordCmd = SCREENRECORD_BINARY + ` --output-format=h264` +
98
+ ` --time-limit=${RECORDING_INTERVAL_SEC}`;
126
99
  if (width || height) {
127
100
  screenRecordCmd += ` --size=${adjustedWidth}x${adjustedHeight}`;
128
101
  }
129
-
130
102
  if (bitRate) {
131
103
  screenRecordCmd += ` --bit-rate=${adjustedBitrate}`;
132
104
  }
133
-
134
- const adbArgs = [...adb.executable.defaultArgs, 'exec-out', `while true; do ${screenRecordCmd} -; done`];
105
+ const adbArgs = [...adb.executable.defaultArgs, 'exec-out',
106
+ `while true; do ${screenRecordCmd} -; done`];
135
107
  const deviceStreaming = (0, _child_process.spawn)(adb.executable.path, adbArgs);
136
108
  deviceStreaming.on('exit', (code, signal) => {
137
109
  log.debug(`Device streaming process exited with code ${code}, signal ${signal}`);
138
110
  });
139
111
  let isStarted = false;
140
112
  const deviceStreamingLogger = createStreamingLogger(SCREENRECORD_BINARY, deviceInfo.udid);
141
-
142
113
  const errorsListener = chunk => {
143
114
  const stderr = chunk.toString();
144
-
145
115
  if (_lodash.default.trim(stderr)) {
146
116
  deviceStreamingLogger.debug(stderr);
147
117
  }
148
118
  };
149
-
150
119
  deviceStreaming.stderr.on('data', errorsListener);
151
-
152
120
  const startupListener = chunk => {
153
121
  if (!isStarted) {
154
122
  isStarted = !_lodash.default.isEmpty(chunk);
155
123
  }
156
124
  };
157
-
158
125
  deviceStreaming.stdout.on('data', startupListener);
159
-
160
126
  try {
161
127
  log.info(`Starting device streaming: ${_support.util.quote([adb.executable.path, ...adbArgs])}`);
162
128
  await (0, _asyncbox.waitForCondition)(() => isStarted, {
@@ -169,10 +135,8 @@ async function initDeviceStreamingProc(adb, log, deviceInfo, opts = {}) {
169
135
  deviceStreaming.stderr.removeListener('data', errorsListener);
170
136
  deviceStreaming.stdout.removeListener('data', startupListener);
171
137
  }
172
-
173
138
  return deviceStreaming;
174
139
  }
175
-
176
140
  async function initGstreamerPipeline(deviceStreamingProc, deviceInfo, log, opts = {}) {
177
141
  const {
178
142
  width,
@@ -191,16 +155,13 @@ async function initGstreamerPipeline(deviceStreamingProc, deviceInfo, log, opts
191
155
  log.debug(`Pipeline streaming process exited with code ${code}, signal ${signal}`);
192
156
  });
193
157
  const gstreamerLogger = createStreamingLogger('gst', deviceInfo.udid);
194
-
195
158
  const gstOutputListener = (stdout, stderr) => {
196
159
  if (_lodash.default.trim(stderr || stdout)) {
197
160
  gstreamerLogger.debug(stderr || stdout);
198
161
  }
199
162
  };
200
-
201
163
  gstreamerPipeline.on('output', gstOutputListener);
202
164
  let didFail = false;
203
-
204
165
  try {
205
166
  log.info(`Starting GStreamer pipeline: ${gstreamerPipeline.rep}`);
206
167
  await gstreamerPipeline.start(0);
@@ -222,10 +183,8 @@ async function initGstreamerPipeline(deviceStreamingProc, deviceInfo, log, opts
222
183
  gstreamerPipeline.removeListener('output', gstOutputListener);
223
184
  }
224
185
  }
225
-
226
186
  return gstreamerPipeline;
227
187
  }
228
-
229
188
  function extractRemoteAddress(req) {
230
189
  return req.headers['x-forwarded-for'] || req.socket.remoteAddress || req.connection.remoteAddress || req.connection.socket.remoteAddress;
231
190
  }
@@ -244,25 +203,20 @@ commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(
244
203
  considerRotation = false,
245
204
  logPipelineDetails = false
246
205
  } = options;
247
-
248
206
  if (_lodash.default.isUndefined(this._screenStreamingProps)) {
249
207
  await verifyStreamingRequirements(this.adb);
250
208
  }
251
-
252
209
  if (!_lodash.default.isEmpty(this._screenStreamingProps)) {
253
210
  this.log.info(`The screen streaming session is already running. ` + `Stop it first in order to start a new one.`);
254
211
  return;
255
212
  }
256
-
257
213
  if ((await (0, _portscanner.checkPortStatus)(port, host)) === 'open') {
258
214
  this.log.info(`The port #${port} at ${host} is busy. ` + `Assuming the screen streaming is already running`);
259
215
  return;
260
216
  }
261
-
262
217
  if ((await (0, _portscanner.checkPortStatus)(tcpPort, TCP_HOST)) === 'open') {
263
218
  this.log.errorAndThrow(`The port #${tcpPort} at ${TCP_HOST} is busy. ` + `Make sure there are no leftovers from previous sessions.`);
264
219
  }
265
-
266
220
  this._screenStreamingProps = null;
267
221
  const deviceInfo = await getDeviceInfo(this.adb, this.log);
268
222
  const deviceStreamingProc = await initDeviceStreamingProc(this.adb, this.log, deviceInfo, {
@@ -271,7 +225,6 @@ commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(
271
225
  bitRate
272
226
  });
273
227
  let gstreamerPipeline;
274
-
275
228
  try {
276
229
  gstreamerPipeline = await initGstreamerPipeline(deviceStreamingProc, deviceInfo, this.log, {
277
230
  width,
@@ -285,24 +238,18 @@ commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(
285
238
  if (deviceStreamingProc.kill(0)) {
286
239
  deviceStreamingProc.kill();
287
240
  }
288
-
289
241
  throw e;
290
242
  }
291
-
292
243
  let mjpegSocket;
293
244
  let mjpegServer;
294
-
295
245
  try {
296
246
  await new _bluebird.default((resolve, reject) => {
297
247
  mjpegSocket = _net.default.createConnection(tcpPort, TCP_HOST, () => {
298
248
  this.log.info(`Successfully connected to MJPEG stream at tcp://${TCP_HOST}:${tcpPort}`);
299
249
  mjpegServer = _http.default.createServer((req, res) => {
300
250
  const remoteAddress = extractRemoteAddress(req);
301
-
302
251
  const currentPathname = _url.default.parse(req.url).pathname;
303
-
304
252
  this.log.info(`Got an incoming screen bradcasting request from ${remoteAddress} ` + `(${req.headers['user-agent'] || 'User Agent unknown'}) at ${currentPathname}`);
305
-
306
253
  if (pathname && currentPathname !== pathname) {
307
254
  this.log.info('Rejecting the broadcast request since it does not match the given pathname');
308
255
  res.writeHead(404, {
@@ -313,7 +260,6 @@ commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(
313
260
  res.end();
314
261
  return;
315
262
  }
316
-
317
263
  this.log.info('Starting MJPEG broadcast');
318
264
  res.writeHead(200, {
319
265
  'Cache-Control': 'no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0',
@@ -345,22 +291,17 @@ commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(
345
291
  if (deviceStreamingProc.kill(0)) {
346
292
  deviceStreamingProc.kill();
347
293
  }
348
-
349
294
  if (gstreamerPipeline.isRunning) {
350
295
  await gstreamerPipeline.stop();
351
296
  }
352
-
353
297
  if (mjpegSocket) {
354
298
  mjpegSocket.destroy();
355
299
  }
356
-
357
300
  if (mjpegServer && mjpegServer.listening) {
358
301
  mjpegServer.close();
359
302
  }
360
-
361
303
  throw e;
362
304
  }
363
-
364
305
  this._screenStreamingProps = {
365
306
  deviceStreamingProc,
366
307
  gstreamerPipeline,
@@ -369,39 +310,33 @@ commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(
369
310
  };
370
311
  };
371
312
 
372
- commands.mobileStopScreenStreaming = async function mobileStopScreenStreaming() {
313
+ commands.mobileStopScreenStreaming = async function
314
+ mobileStopScreenStreaming() {
373
315
  if (_lodash.default.isEmpty(this._screenStreamingProps)) {
374
316
  if (!_lodash.default.isUndefined(this._screenStreamingProps)) {
375
317
  this.log.debug(`Screen streaming is not running. There is nothing to stop`);
376
318
  }
377
-
378
319
  return;
379
320
  }
380
-
381
321
  const {
382
322
  deviceStreamingProc,
383
323
  gstreamerPipeline,
384
324
  mjpegSocket,
385
325
  mjpegServer
386
326
  } = this._screenStreamingProps;
387
-
388
327
  try {
389
328
  mjpegSocket.end();
390
-
391
329
  if (mjpegServer.listening) {
392
330
  mjpegServer.close();
393
331
  }
394
-
395
332
  if (deviceStreamingProc.kill(0)) {
396
333
  deviceStreamingProc.kill('SIGINT');
397
334
  }
398
-
399
335
  if (gstreamerPipeline.isRunning) {
400
336
  try {
401
337
  await gstreamerPipeline.stop('SIGINT');
402
338
  } catch (e) {
403
339
  this.log.warn(e);
404
-
405
340
  try {
406
341
  await gstreamerPipeline.stop('SIGKILL');
407
342
  } catch (e1) {
@@ -409,13 +344,11 @@ commands.mobileStopScreenStreaming = async function mobileStopScreenStreaming()
409
344
  }
410
345
  }
411
346
  }
412
-
413
347
  this.log.info(`Successfully terminated the screen streaming MJPEG server`);
414
348
  } finally {
415
349
  this._screenStreamingProps = null;
416
350
  }
417
351
  };
418
-
419
352
  var _default = commands;
420
353
  exports.default = _default;
421
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["commands","RECORDING_INTERVAL_SEC","STREAMING_STARTUP_TIMEOUT_MS","GSTREAMER_BINARY","system","isWindows","GST_INSPECT_BINARY","REQUIRED_GST_PLUGINS","avdec_h264","h264parse","jpegenc","tcpserversink","multipartmux","SCREENRECORD_BINARY","GST_TUTORIAL_URL","DEFAULT_HOST","TCP_HOST","DEFAULT_PORT","DEFAULT_QUALITY","DEFAULT_BITRATE","BOUNDARY_STRING","ADB_SCREEN_STREAMING_FEATURE","createStreamingLogger","streamName","udid","logger","getLogger","_","truncate","length","omission","verifyStreamingRequirements","adb","trim","shell","Error","gstreamerCheckPromises","binaryName","push","fs","which","e","B","all","moduleCheckPromises","name","modName","toPairs","stdout","exec","includes","getDeviceInfo","log","output","result","key","pattern","match","debug","parseInt","curDeviceId","initDeviceStreamingProc","deviceInfo","opts","width","height","bitRate","adjustedWidth","adjustedHeight","adjustedBitrate","screenRecordCmd","adbArgs","executable","defaultArgs","deviceStreaming","spawn","path","on","code","signal","isStarted","deviceStreamingLogger","errorsListener","chunk","stderr","toString","startupListener","isEmpty","info","util","quote","waitForCondition","waitMs","intervalMs","errorAndThrow","message","removeListener","initGstreamerPipeline","deviceStreamingProc","quality","tcpPort","considerRotation","logPipelineDetails","gstreamerPipeline","SubProcess","Math","max","fps","stdio","gstreamerLogger","gstOutputListener","didFail","rep","start","checkPortStatus","ign","extractRemoteAddress","req","headers","socket","remoteAddress","connection","mobileStartScreenStreaming","options","ensureFeatureEnabled","host","port","pathname","isUndefined","_screenStreamingProps","kill","mjpegSocket","mjpegServer","resolve","reject","net","createConnection","http","createServer","res","currentPathname","url","parse","writeHead","Connection","write","end","Pragma","pipe","warn","listen","error","timeout","isRunning","stop","destroy","listening","close","mobileStopScreenStreaming","e1"],"sources":["../../../lib/commands/streamscreen.js"],"sourcesContent":["import _ from 'lodash';\nimport { fs, system, logger, util } from 'appium/support';\nimport { exec, SubProcess } from 'teen_process';\nimport { checkPortStatus } from 'portscanner';\nimport http from 'http';\nimport net from 'net';\nimport B from 'bluebird';\nimport { waitForCondition } from 'asyncbox';\nimport { spawn } from 'child_process';\nimport url from 'url';\n\nconst commands = {};\n\nconst RECORDING_INTERVAL_SEC = 5;\nconst STREAMING_STARTUP_TIMEOUT_MS = 5000;\nconst GSTREAMER_BINARY = `gst-launch-1.0${system.isWindows() ? '.exe' : ''}`;\nconst GST_INSPECT_BINARY = `gst-inspect-1.0${system.isWindows() ? '.exe' : ''}`;\nconst REQUIRED_GST_PLUGINS = {\n  avdec_h264: 'gst-libav',\n  h264parse: 'gst-plugins-bad',\n  jpegenc: 'gst-plugins-good',\n  tcpserversink: 'gst-plugins-base',\n  multipartmux: 'gst-plugins-good',\n};\nconst SCREENRECORD_BINARY = 'screenrecord';\nconst GST_TUTORIAL_URL = 'https://gstreamer.freedesktop.org/documentation/installing/index.html';\nconst DEFAULT_HOST = '127.0.0.1';\nconst TCP_HOST = '127.0.0.1';\nconst DEFAULT_PORT = 8093;\nconst DEFAULT_QUALITY = 70;\nconst DEFAULT_BITRATE = 4000000; // 4 Mbps\nconst BOUNDARY_STRING = '--2ae9746887f170b8cf7c271047ce314c';\n\nconst ADB_SCREEN_STREAMING_FEATURE = 'adb_screen_streaming';\n\nfunction createStreamingLogger (streamName, udid) {\n  return logger.getLogger(`${streamName}@` + _.truncate(udid, {\n    length: 8,\n    omission: '',\n  }));\n}\n\nasync function verifyStreamingRequirements (adb) {\n  if (!_.trim(await adb.shell(['which', SCREENRECORD_BINARY]))) {\n    throw new Error(\n      `The required '${SCREENRECORD_BINARY}' binary is not available on the device under test`);\n  }\n\n  const gstreamerCheckPromises = [];\n  for (const binaryName of [GSTREAMER_BINARY, GST_INSPECT_BINARY]) {\n    gstreamerCheckPromises.push((async () => {\n      try {\n        await fs.which(binaryName);\n      } catch (e) {\n        throw new Error(`The '${binaryName}' binary is not available in the PATH on the host system. ` +\n          `See ${GST_TUTORIAL_URL} for more details on how to install it.`);\n      }\n    })());\n  }\n  await B.all(gstreamerCheckPromises);\n\n  const moduleCheckPromises = [];\n  for (const [name, modName] of _.toPairs(REQUIRED_GST_PLUGINS)) {\n    moduleCheckPromises.push((async () => {\n      const {stdout} = await exec(GST_INSPECT_BINARY, [name]);\n      if (!_.includes(stdout, modName)) {\n        throw new Error(\n          `The required GStreamer plugin '${name}' from '${modName}' module is not installed. ` +\n          `See ${GST_TUTORIAL_URL} for more details on how to install it.`);\n      }\n    })());\n  }\n  await B.all(moduleCheckPromises);\n}\n\nasync function getDeviceInfo (adb, log = null) {\n  const output = await adb.shell(['dumpsys', 'display']);\n  const result = {};\n  for (const [key, pattern] of [\n    ['width', /\\bdeviceWidth=(\\d+)/],\n    ['height', /\\bdeviceHeight=(\\d+)/],\n    ['fps', /\\bfps=(\\d+)/],\n  ]) {\n    const match = pattern.exec(output);\n    if (!match) {\n      log?.debug(output);\n      throw new Error(`Cannot parse the device ${key} from the adb command output. ` +\n        `Check the server log for more details.`);\n    }\n    result[key] = parseInt(match[1], 10);\n  }\n  result.udid = adb.curDeviceId;\n  return result;\n}\n\nasync function initDeviceStreamingProc (adb, log, deviceInfo, opts = {}) {\n  const {\n    width,\n    height,\n    bitRate,\n  } = opts;\n  const adjustedWidth = parseInt(width, 10) || deviceInfo.width;\n  const adjustedHeight = parseInt(height, 10) || deviceInfo.height;\n  const adjustedBitrate = parseInt(bitRate, 10) || DEFAULT_BITRATE;\n  let screenRecordCmd = SCREENRECORD_BINARY +\n    ` --output-format=h264` +\n    // 5 seconds is fine to detect rotation changes\n    ` --time-limit=${RECORDING_INTERVAL_SEC}`;\n  if (width || height) {\n    screenRecordCmd += ` --size=${adjustedWidth}x${adjustedHeight}`;\n  }\n  if (bitRate) {\n    screenRecordCmd += ` --bit-rate=${adjustedBitrate}`;\n  }\n  const adbArgs = [\n    ...adb.executable.defaultArgs,\n    'exec-out',\n    // The loop is required, because by default the maximum record duration\n    // for screenrecord is always limited\n    `while true; do ${screenRecordCmd} -; done`,\n  ];\n  const deviceStreaming = spawn(adb.executable.path, adbArgs);\n  deviceStreaming.on('exit', (code, signal) => {\n    log.debug(`Device streaming process exited with code ${code}, signal ${signal}`);\n  });\n\n  let isStarted = false;\n  const deviceStreamingLogger = createStreamingLogger(SCREENRECORD_BINARY, deviceInfo.udid);\n  const errorsListener = (chunk) => {\n    const stderr = chunk.toString();\n    if (_.trim(stderr)) {\n      deviceStreamingLogger.debug(stderr);\n    }\n  };\n  deviceStreaming.stderr.on('data', errorsListener);\n\n  const startupListener = (chunk) => {\n    if (!isStarted) {\n      isStarted = !_.isEmpty(chunk);\n    }\n  };\n  deviceStreaming.stdout.on('data', startupListener);\n\n  try {\n    log.info(`Starting device streaming: ${util.quote([adb.executable.path, ...adbArgs])}`);\n    await waitForCondition(() => isStarted, {\n      waitMs: STREAMING_STARTUP_TIMEOUT_MS,\n      intervalMs: 300,\n    });\n  } catch (e) {\n    log.errorAndThrow(\n      `Cannot start the screen streaming process. Original error: ${e.message}`);\n  } finally {\n    deviceStreaming.stderr.removeListener('data', errorsListener);\n    deviceStreaming.stdout.removeListener('data', startupListener);\n  }\n  return deviceStreaming;\n}\n\nasync function initGstreamerPipeline (deviceStreamingProc, deviceInfo, log, opts = {}) {\n  const {\n    width,\n    height,\n    quality,\n    tcpPort,\n    considerRotation,\n    logPipelineDetails,\n  } = opts;\n  const adjustedWidth = parseInt(width, 10) || deviceInfo.width;\n  const adjustedHeight = parseInt(height, 10) || deviceInfo.height;\n  const gstreamerPipeline = new SubProcess(GSTREAMER_BINARY, [\n    '-v',\n    'fdsrc', 'fd=0',\n    '!', 'video/x-h264,' +\n      `width=${considerRotation ? Math.max(adjustedWidth, adjustedHeight) : adjustedWidth},` +\n      `height=${considerRotation ? Math.max(adjustedWidth, adjustedHeight) : adjustedHeight},` +\n      `framerate=${deviceInfo.fps}/1,` +\n      'byte-stream=true',\n    '!', 'h264parse',\n    '!', 'queue', 'leaky=downstream',\n    '!', 'avdec_h264',\n    '!', 'queue', 'leaky=downstream',\n    '!', 'jpegenc', `quality=${quality}`,\n    '!', 'multipartmux', `boundary=${BOUNDARY_STRING}`,\n    '!', 'tcpserversink', `host=${TCP_HOST}`, `port=${tcpPort}`,\n  ], {\n    stdio: [deviceStreamingProc.stdout, 'pipe', 'pipe']\n  });\n  gstreamerPipeline.on('exit', (code, signal) => {\n    log.debug(`Pipeline streaming process exited with code ${code}, signal ${signal}`);\n  });\n  const gstreamerLogger = createStreamingLogger('gst', deviceInfo.udid);\n  const gstOutputListener = (stdout, stderr) => {\n    if (_.trim(stderr || stdout)) {\n      gstreamerLogger.debug(stderr || stdout);\n    }\n  };\n  gstreamerPipeline.on('output', gstOutputListener);\n  let didFail = false;\n  try {\n    log.info(`Starting GStreamer pipeline: ${gstreamerPipeline.rep}`);\n    await gstreamerPipeline.start(0);\n    await waitForCondition(async () => {\n      try {\n        return (await checkPortStatus(tcpPort, TCP_HOST)) === 'open';\n      } catch (ign) {\n        return false;\n      }\n    }, {\n      waitMs: STREAMING_STARTUP_TIMEOUT_MS,\n      intervalMs: 300,\n    });\n  } catch (e) {\n    didFail = true;\n    log.errorAndThrow(\n      `Cannot start the screen streaming pipeline. Original error: ${e.message}`);\n  } finally {\n    if (!logPipelineDetails || didFail) {\n      gstreamerPipeline.removeListener('output', gstOutputListener);\n    }\n  }\n  return gstreamerPipeline;\n}\n\nfunction extractRemoteAddress (req) {\n  return req.headers['x-forwarded-for']\n    || req.socket.remoteAddress\n    || req.connection.remoteAddress\n    || req.connection.socket.remoteAddress;\n}\n\n\n/**\n * @typedef {Object} StartScreenStreamingOptions\n *\n * @property {?number} width - The scaled width of the device's screen. If unset then the script will assign it\n * to the actual screen width measured in pixels.\n * @property {?number} height - The scaled height of the device's screen. If unset then the script will assign it\n * to the actual screen height measured in pixels.\n * @property {?number} bitRate - The video bit rate for the video, in bits per second.\n * The default value is 4000000 (4 Mb/s). You can increase the bit rate to improve video quality,\n * but doing so results in larger movie files.\n * @property {?string} host [127.0.0.1] - The IP address/host name to start the MJPEG server on.\n * You can set it to `0.0.0.0` to trigger the broadcast on all available network interfaces.\n * @property {?string} pathname - The HTTP request path the MJPEG server should be available on.\n * If unset then any pathname on the given `host`/`port` combination will work. Note that the value\n * should always start with a single slash: `/`\n * @property {?number} tcpPort [8094] - The port number to start the internal TCP MJPEG broadcast on.\n * This type of broadcast always starts on the loopback interface (`127.0.0.1`).\n * @property {?number} port [8093] - The port number to start the MJPEG server on.\n * @property {?number} quality [70] - The quality value for the streamed JPEG images.\n * This number should be in range [1, 100], where 100 is the best quality.\n * @property {?boolean} considerRotation [false] - If set to `true` then GStreamer pipeline will\n * increase the dimensions of the resulting images to properly fit images in both landscape and\n * portrait orientations. Set it to `true` if the device rotation is not going to be the same during the\n * broadcasting session.\n * @property {?boolean} logPipelineDetails [false] - Whether to log GStreamer pipeline events into\n * the standard log output. Might be useful for debugging purposes.\n */\n\n/**\n * Starts device screen broadcast by creating MJPEG server.\n * Multiple calls to this method have no effect unless the previous streaming\n * session is stopped.\n * This method only works if the `adb_screen_streaming` feature is\n * enabled on the server side.\n *\n * @param {?StartScreenStreamingOptions} options - The available options.\n * @throws {Error} If screen streaming has failed to start or\n * is not supported on the host system or\n * the corresponding server feature is not enabled.\n */\ncommands.mobileStartScreenStreaming = async function mobileStartScreenStreaming (options = {}) {\n  this.ensureFeatureEnabled(ADB_SCREEN_STREAMING_FEATURE);\n\n  const {\n    width,\n    height,\n    bitRate,\n    host = DEFAULT_HOST,\n    port = DEFAULT_PORT,\n    pathname,\n    tcpPort = DEFAULT_PORT + 1,\n    quality = DEFAULT_QUALITY,\n    considerRotation = false,\n    logPipelineDetails = false,\n  } = options;\n\n  if (_.isUndefined(this._screenStreamingProps)) {\n    await verifyStreamingRequirements(this.adb);\n  }\n  if (!_.isEmpty(this._screenStreamingProps)) {\n    this.log.info(`The screen streaming session is already running. ` +\n      `Stop it first in order to start a new one.`);\n    return;\n  }\n  if ((await checkPortStatus(port, host)) === 'open') {\n    this.log.info(`The port #${port} at ${host} is busy. ` +\n      `Assuming the screen streaming is already running`);\n    return;\n  }\n  if ((await checkPortStatus(tcpPort, TCP_HOST)) === 'open') {\n    this.log.errorAndThrow(`The port #${tcpPort} at ${TCP_HOST} is busy. ` +\n      `Make sure there are no leftovers from previous sessions.`);\n  }\n  this._screenStreamingProps = null;\n\n  const deviceInfo = await getDeviceInfo(this.adb, this.log);\n  const deviceStreamingProc = await initDeviceStreamingProc(this.adb, this.log, deviceInfo, {\n    width,\n    height,\n    bitRate,\n  });\n  let gstreamerPipeline;\n  try {\n    gstreamerPipeline = await initGstreamerPipeline(deviceStreamingProc, deviceInfo, this.log, {\n      width,\n      height,\n      quality,\n      tcpPort,\n      considerRotation,\n      logPipelineDetails,\n    });\n  } catch (e) {\n    if (deviceStreamingProc.kill(0)) {\n      deviceStreamingProc.kill();\n    }\n    throw e;\n  }\n\n  let mjpegSocket;\n  let mjpegServer;\n  try {\n    await new B((resolve, reject) => {\n      mjpegSocket = net.createConnection(tcpPort, TCP_HOST, () => {\n        this.log.info(`Successfully connected to MJPEG stream at tcp://${TCP_HOST}:${tcpPort}`);\n        mjpegServer = http.createServer((req, res) => {\n          const remoteAddress = extractRemoteAddress(req);\n          const currentPathname = url.parse(req.url).pathname;\n          this.log.info(`Got an incoming screen bradcasting request from ${remoteAddress} ` +\n            `(${req.headers['user-agent'] || 'User Agent unknown'}) at ${currentPathname}`);\n\n          if (pathname && currentPathname !== pathname) {\n            this.log.info('Rejecting the broadcast request since it does not match the given pathname');\n            res.writeHead(404, {\n              Connection: 'close',\n              'Content-Type': 'text/plain; charset=utf-8',\n            });\n            res.write(`'${currentPathname}' did not match any known endpoints`);\n            res.end();\n            return;\n          }\n\n          this.log.info('Starting MJPEG broadcast');\n          res.writeHead(200, {\n            'Cache-Control': 'no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0',\n            Pragma: 'no-cache',\n            Connection: 'close',\n            'Content-Type': `multipart/x-mixed-replace; boundary=${BOUNDARY_STRING}`\n          });\n\n          mjpegSocket.pipe(res);\n        });\n        mjpegServer.on('error', (e) => {\n          this.log.warn(e);\n          reject(e);\n        });\n        mjpegServer.on('close', () => {\n          this.log.debug(`MJPEG server at http://${host}:${port} has been closed`);\n        });\n        mjpegServer.on('listening', () => {\n          this.log.info(`Successfully started MJPEG server at http://${host}:${port}`);\n          resolve();\n        });\n        mjpegServer.listen(port, host);\n      });\n      mjpegSocket.on('error', (e) => {\n        this.log.error(e);\n        reject(e);\n      });\n    }).timeout(STREAMING_STARTUP_TIMEOUT_MS,\n      `Cannot connect to the streaming server within ${STREAMING_STARTUP_TIMEOUT_MS}ms`);\n  } catch (e) {\n    if (deviceStreamingProc.kill(0)) {\n      deviceStreamingProc.kill();\n    }\n    if (gstreamerPipeline.isRunning) {\n      await gstreamerPipeline.stop();\n    }\n    if (mjpegSocket) {\n      mjpegSocket.destroy();\n    }\n    if (mjpegServer && mjpegServer.listening) {\n      mjpegServer.close();\n    }\n    throw e;\n  }\n\n  this._screenStreamingProps = {\n    deviceStreamingProc,\n    gstreamerPipeline,\n    mjpegSocket,\n    mjpegServer,\n  };\n};\n\n/**\n * Stop screen streaming.\n * If no screen streaming server has been started then nothing is done.\n */\ncommands.mobileStopScreenStreaming = async function mobileStopScreenStreaming (/* options = {} */) {\n  if (_.isEmpty(this._screenStreamingProps)) {\n    if (!_.isUndefined(this._screenStreamingProps)) {\n      this.log.debug(`Screen streaming is not running. There is nothing to stop`);\n    }\n    return;\n  }\n\n  const {\n    deviceStreamingProc,\n    gstreamerPipeline,\n    mjpegSocket,\n    mjpegServer,\n  } = this._screenStreamingProps;\n\n  try {\n    mjpegSocket.end();\n    if (mjpegServer.listening) {\n      mjpegServer.close();\n    }\n    if (deviceStreamingProc.kill(0)) {\n      deviceStreamingProc.kill('SIGINT');\n    }\n    if (gstreamerPipeline.isRunning) {\n      try {\n        await gstreamerPipeline.stop('SIGINT');\n      } catch (e) {\n        this.log.warn(e);\n        try {\n          await gstreamerPipeline.stop('SIGKILL');\n        } catch (e1) {\n          this.log.error(e1);\n        }\n      }\n    }\n    this.log.info(`Successfully terminated the screen streaming MJPEG server`);\n  } finally {\n    this._screenStreamingProps = null;\n  }\n};\n\n\nexport default commands;\n"],"mappings":"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"}
354
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["commands","RECORDING_INTERVAL_SEC","STREAMING_STARTUP_TIMEOUT_MS","GSTREAMER_BINARY","system","isWindows","GST_INSPECT_BINARY","REQUIRED_GST_PLUGINS","avdec_h264","h264parse","jpegenc","tcpserversink","multipartmux","SCREENRECORD_BINARY","GST_TUTORIAL_URL","DEFAULT_HOST","TCP_HOST","DEFAULT_PORT","DEFAULT_QUALITY","DEFAULT_BITRATE","BOUNDARY_STRING","ADB_SCREEN_STREAMING_FEATURE","createStreamingLogger","streamName","udid","logger","getLogger","_","truncate","length","omission","verifyStreamingRequirements","adb","trim","shell","Error","gstreamerCheckPromises","binaryName","push","fs","which","e","B","all","moduleCheckPromises","name","modName","toPairs","stdout","exec","includes","getDeviceInfo","log","output","result","key","pattern","match","debug","parseInt","curDeviceId","initDeviceStreamingProc","deviceInfo","opts","width","height","bitRate","adjustedWidth","adjustedHeight","adjustedBitrate","screenRecordCmd","adbArgs","executable","defaultArgs","deviceStreaming","spawn","path","on","code","signal","isStarted","deviceStreamingLogger","errorsListener","chunk","stderr","toString","startupListener","isEmpty","info","util","quote","waitForCondition","waitMs","intervalMs","errorAndThrow","message","removeListener","initGstreamerPipeline","deviceStreamingProc","quality","tcpPort","considerRotation","logPipelineDetails","gstreamerPipeline","SubProcess","Math","max","fps","stdio","gstreamerLogger","gstOutputListener","didFail","rep","start","checkPortStatus","ign","extractRemoteAddress","req","headers","socket","remoteAddress","connection","mobileStartScreenStreaming","options","ensureFeatureEnabled","host","port","pathname","isUndefined","_screenStreamingProps","kill","mjpegSocket","mjpegServer","resolve","reject","net","createConnection","http","createServer","res","currentPathname","url","parse","writeHead","Connection","write","end","Pragma","pipe","warn","listen","error","timeout","isRunning","stop","destroy","listening","close","mobileStopScreenStreaming","e1"],"sources":["../../../lib/commands/streamscreen.js"],"sourcesContent":["import _ from 'lodash';\nimport { fs, system, logger, util } from 'appium/support';\nimport { exec, SubProcess } from 'teen_process';\nimport { checkPortStatus } from 'portscanner';\nimport http from 'http';\nimport net from 'net';\nimport B from 'bluebird';\nimport { waitForCondition } from 'asyncbox';\nimport { spawn } from 'child_process';\nimport url from 'url';\n\nconst commands = {};\n\nconst RECORDING_INTERVAL_SEC = 5;\nconst STREAMING_STARTUP_TIMEOUT_MS = 5000;\nconst GSTREAMER_BINARY = `gst-launch-1.0${system.isWindows() ? '.exe' : ''}`;\nconst GST_INSPECT_BINARY = `gst-inspect-1.0${system.isWindows() ? '.exe' : ''}`;\nconst REQUIRED_GST_PLUGINS = {\n  avdec_h264: 'gst-libav',\n  h264parse: 'gst-plugins-bad',\n  jpegenc: 'gst-plugins-good',\n  tcpserversink: 'gst-plugins-base',\n  multipartmux: 'gst-plugins-good',\n};\nconst SCREENRECORD_BINARY = 'screenrecord';\nconst GST_TUTORIAL_URL = 'https://gstreamer.freedesktop.org/documentation/installing/index.html';\nconst DEFAULT_HOST = '127.0.0.1';\nconst TCP_HOST = '127.0.0.1';\nconst DEFAULT_PORT = 8093;\nconst DEFAULT_QUALITY = 70;\nconst DEFAULT_BITRATE = 4000000; // 4 Mbps\nconst BOUNDARY_STRING = '--2ae9746887f170b8cf7c271047ce314c';\n\nconst ADB_SCREEN_STREAMING_FEATURE = 'adb_screen_streaming';\n\nfunction createStreamingLogger (streamName, udid) {\n  return logger.getLogger(`${streamName}@` + _.truncate(udid, {\n    length: 8,\n    omission: '',\n  }));\n}\n\nasync function verifyStreamingRequirements (adb) {\n  if (!_.trim(await adb.shell(['which', SCREENRECORD_BINARY]))) {\n    throw new Error(\n      `The required '${SCREENRECORD_BINARY}' binary is not available on the device under test`);\n  }\n\n  const gstreamerCheckPromises = [];\n  for (const binaryName of [GSTREAMER_BINARY, GST_INSPECT_BINARY]) {\n    gstreamerCheckPromises.push((async () => {\n      try {\n        await fs.which(binaryName);\n      } catch (e) {\n        throw new Error(`The '${binaryName}' binary is not available in the PATH on the host system. ` +\n          `See ${GST_TUTORIAL_URL} for more details on how to install it.`);\n      }\n    })());\n  }\n  await B.all(gstreamerCheckPromises);\n\n  const moduleCheckPromises = [];\n  for (const [name, modName] of _.toPairs(REQUIRED_GST_PLUGINS)) {\n    moduleCheckPromises.push((async () => {\n      const {stdout} = await exec(GST_INSPECT_BINARY, [name]);\n      if (!_.includes(stdout, modName)) {\n        throw new Error(\n          `The required GStreamer plugin '${name}' from '${modName}' module is not installed. ` +\n          `See ${GST_TUTORIAL_URL} for more details on how to install it.`);\n      }\n    })());\n  }\n  await B.all(moduleCheckPromises);\n}\n\nasync function getDeviceInfo (adb, log = null) {\n  const output = await adb.shell(['dumpsys', 'display']);\n  const result = {};\n  for (const [key, pattern] of [\n    ['width', /\\bdeviceWidth=(\\d+)/],\n    ['height', /\\bdeviceHeight=(\\d+)/],\n    ['fps', /\\bfps=(\\d+)/],\n  ]) {\n    const match = pattern.exec(output);\n    if (!match) {\n      log?.debug(output);\n      throw new Error(`Cannot parse the device ${key} from the adb command output. ` +\n        `Check the server log for more details.`);\n    }\n    result[key] = parseInt(match[1], 10);\n  }\n  result.udid = adb.curDeviceId;\n  return result;\n}\n\nasync function initDeviceStreamingProc (adb, log, deviceInfo, opts = {}) {\n  const {\n    width,\n    height,\n    bitRate,\n  } = opts;\n  const adjustedWidth = parseInt(width, 10) || deviceInfo.width;\n  const adjustedHeight = parseInt(height, 10) || deviceInfo.height;\n  const adjustedBitrate = parseInt(bitRate, 10) || DEFAULT_BITRATE;\n  let screenRecordCmd = SCREENRECORD_BINARY +\n    ` --output-format=h264` +\n    // 5 seconds is fine to detect rotation changes\n    ` --time-limit=${RECORDING_INTERVAL_SEC}`;\n  if (width || height) {\n    screenRecordCmd += ` --size=${adjustedWidth}x${adjustedHeight}`;\n  }\n  if (bitRate) {\n    screenRecordCmd += ` --bit-rate=${adjustedBitrate}`;\n  }\n  const adbArgs = [\n    ...adb.executable.defaultArgs,\n    'exec-out',\n    // The loop is required, because by default the maximum record duration\n    // for screenrecord is always limited\n    `while true; do ${screenRecordCmd} -; done`,\n  ];\n  const deviceStreaming = spawn(adb.executable.path, adbArgs);\n  deviceStreaming.on('exit', (code, signal) => {\n    log.debug(`Device streaming process exited with code ${code}, signal ${signal}`);\n  });\n\n  let isStarted = false;\n  const deviceStreamingLogger = createStreamingLogger(SCREENRECORD_BINARY, deviceInfo.udid);\n  const errorsListener = (chunk) => {\n    const stderr = chunk.toString();\n    if (_.trim(stderr)) {\n      deviceStreamingLogger.debug(stderr);\n    }\n  };\n  deviceStreaming.stderr.on('data', errorsListener);\n\n  const startupListener = (chunk) => {\n    if (!isStarted) {\n      isStarted = !_.isEmpty(chunk);\n    }\n  };\n  deviceStreaming.stdout.on('data', startupListener);\n\n  try {\n    log.info(`Starting device streaming: ${util.quote([adb.executable.path, ...adbArgs])}`);\n    await waitForCondition(() => isStarted, {\n      waitMs: STREAMING_STARTUP_TIMEOUT_MS,\n      intervalMs: 300,\n    });\n  } catch (e) {\n    log.errorAndThrow(\n      `Cannot start the screen streaming process. Original error: ${e.message}`);\n  } finally {\n    deviceStreaming.stderr.removeListener('data', errorsListener);\n    deviceStreaming.stdout.removeListener('data', startupListener);\n  }\n  return deviceStreaming;\n}\n\nasync function initGstreamerPipeline (deviceStreamingProc, deviceInfo, log, opts = {}) {\n  const {\n    width,\n    height,\n    quality,\n    tcpPort,\n    considerRotation,\n    logPipelineDetails,\n  } = opts;\n  const adjustedWidth = parseInt(width, 10) || deviceInfo.width;\n  const adjustedHeight = parseInt(height, 10) || deviceInfo.height;\n  const gstreamerPipeline = new SubProcess(GSTREAMER_BINARY, [\n    '-v',\n    'fdsrc', 'fd=0',\n    '!', 'video/x-h264,' +\n      `width=${considerRotation ? Math.max(adjustedWidth, adjustedHeight) : adjustedWidth},` +\n      `height=${considerRotation ? Math.max(adjustedWidth, adjustedHeight) : adjustedHeight},` +\n      `framerate=${deviceInfo.fps}/1,` +\n      'byte-stream=true',\n    '!', 'h264parse',\n    '!', 'queue', 'leaky=downstream',\n    '!', 'avdec_h264',\n    '!', 'queue', 'leaky=downstream',\n    '!', 'jpegenc', `quality=${quality}`,\n    '!', 'multipartmux', `boundary=${BOUNDARY_STRING}`,\n    '!', 'tcpserversink', `host=${TCP_HOST}`, `port=${tcpPort}`,\n  ], {\n    stdio: [deviceStreamingProc.stdout, 'pipe', 'pipe']\n  });\n  gstreamerPipeline.on('exit', (code, signal) => {\n    log.debug(`Pipeline streaming process exited with code ${code}, signal ${signal}`);\n  });\n  const gstreamerLogger = createStreamingLogger('gst', deviceInfo.udid);\n  const gstOutputListener = (stdout, stderr) => {\n    if (_.trim(stderr || stdout)) {\n      gstreamerLogger.debug(stderr || stdout);\n    }\n  };\n  gstreamerPipeline.on('output', gstOutputListener);\n  let didFail = false;\n  try {\n    log.info(`Starting GStreamer pipeline: ${gstreamerPipeline.rep}`);\n    await gstreamerPipeline.start(0);\n    await waitForCondition(async () => {\n      try {\n        return (await checkPortStatus(tcpPort, TCP_HOST)) === 'open';\n      } catch (ign) {\n        return false;\n      }\n    }, {\n      waitMs: STREAMING_STARTUP_TIMEOUT_MS,\n      intervalMs: 300,\n    });\n  } catch (e) {\n    didFail = true;\n    log.errorAndThrow(\n      `Cannot start the screen streaming pipeline. Original error: ${e.message}`);\n  } finally {\n    if (!logPipelineDetails || didFail) {\n      gstreamerPipeline.removeListener('output', gstOutputListener);\n    }\n  }\n  return gstreamerPipeline;\n}\n\nfunction extractRemoteAddress (req) {\n  return req.headers['x-forwarded-for']\n    || req.socket.remoteAddress\n    || req.connection.remoteAddress\n    || req.connection.socket.remoteAddress;\n}\n\n\n/**\n * @typedef {Object} StartScreenStreamingOptions\n *\n * @property {?number} width - The scaled width of the device's screen. If unset then the script will assign it\n * to the actual screen width measured in pixels.\n * @property {?number} height - The scaled height of the device's screen. If unset then the script will assign it\n * to the actual screen height measured in pixels.\n * @property {?number} bitRate - The video bit rate for the video, in bits per second.\n * The default value is 4000000 (4 Mb/s). You can increase the bit rate to improve video quality,\n * but doing so results in larger movie files.\n * @property {?string} host [127.0.0.1] - The IP address/host name to start the MJPEG server on.\n * You can set it to `0.0.0.0` to trigger the broadcast on all available network interfaces.\n * @property {?string} pathname - The HTTP request path the MJPEG server should be available on.\n * If unset then any pathname on the given `host`/`port` combination will work. Note that the value\n * should always start with a single slash: `/`\n * @property {?number} tcpPort [8094] - The port number to start the internal TCP MJPEG broadcast on.\n * This type of broadcast always starts on the loopback interface (`127.0.0.1`).\n * @property {?number} port [8093] - The port number to start the MJPEG server on.\n * @property {?number} quality [70] - The quality value for the streamed JPEG images.\n * This number should be in range [1, 100], where 100 is the best quality.\n * @property {?boolean} considerRotation [false] - If set to `true` then GStreamer pipeline will\n * increase the dimensions of the resulting images to properly fit images in both landscape and\n * portrait orientations. Set it to `true` if the device rotation is not going to be the same during the\n * broadcasting session.\n * @property {?boolean} logPipelineDetails [false] - Whether to log GStreamer pipeline events into\n * the standard log output. Might be useful for debugging purposes.\n */\n\n/**\n * Starts device screen broadcast by creating MJPEG server.\n * Multiple calls to this method have no effect unless the previous streaming\n * session is stopped.\n * This method only works if the `adb_screen_streaming` feature is\n * enabled on the server side.\n *\n * @param {?StartScreenStreamingOptions} options - The available options.\n * @throws {Error} If screen streaming has failed to start or\n * is not supported on the host system or\n * the corresponding server feature is not enabled.\n */\ncommands.mobileStartScreenStreaming = async function mobileStartScreenStreaming (options = {}) {\n  this.ensureFeatureEnabled(ADB_SCREEN_STREAMING_FEATURE);\n\n  const {\n    width,\n    height,\n    bitRate,\n    host = DEFAULT_HOST,\n    port = DEFAULT_PORT,\n    pathname,\n    tcpPort = DEFAULT_PORT + 1,\n    quality = DEFAULT_QUALITY,\n    considerRotation = false,\n    logPipelineDetails = false,\n  } = options;\n\n  if (_.isUndefined(this._screenStreamingProps)) {\n    await verifyStreamingRequirements(this.adb);\n  }\n  if (!_.isEmpty(this._screenStreamingProps)) {\n    this.log.info(`The screen streaming session is already running. ` +\n      `Stop it first in order to start a new one.`);\n    return;\n  }\n  if ((await checkPortStatus(port, host)) === 'open') {\n    this.log.info(`The port #${port} at ${host} is busy. ` +\n      `Assuming the screen streaming is already running`);\n    return;\n  }\n  if ((await checkPortStatus(tcpPort, TCP_HOST)) === 'open') {\n    this.log.errorAndThrow(`The port #${tcpPort} at ${TCP_HOST} is busy. ` +\n      `Make sure there are no leftovers from previous sessions.`);\n  }\n  this._screenStreamingProps = null;\n\n  const deviceInfo = await getDeviceInfo(this.adb, this.log);\n  const deviceStreamingProc = await initDeviceStreamingProc(this.adb, this.log, deviceInfo, {\n    width,\n    height,\n    bitRate,\n  });\n  let gstreamerPipeline;\n  try {\n    gstreamerPipeline = await initGstreamerPipeline(deviceStreamingProc, deviceInfo, this.log, {\n      width,\n      height,\n      quality,\n      tcpPort,\n      considerRotation,\n      logPipelineDetails,\n    });\n  } catch (e) {\n    if (deviceStreamingProc.kill(0)) {\n      deviceStreamingProc.kill();\n    }\n    throw e;\n  }\n\n  let mjpegSocket;\n  let mjpegServer;\n  try {\n    await new B((resolve, reject) => {\n      mjpegSocket = net.createConnection(tcpPort, TCP_HOST, () => {\n        this.log.info(`Successfully connected to MJPEG stream at tcp://${TCP_HOST}:${tcpPort}`);\n        mjpegServer = http.createServer((req, res) => {\n          const remoteAddress = extractRemoteAddress(req);\n          const currentPathname = url.parse(req.url).pathname;\n          this.log.info(`Got an incoming screen bradcasting request from ${remoteAddress} ` +\n            `(${req.headers['user-agent'] || 'User Agent unknown'}) at ${currentPathname}`);\n\n          if (pathname && currentPathname !== pathname) {\n            this.log.info('Rejecting the broadcast request since it does not match the given pathname');\n            res.writeHead(404, {\n              Connection: 'close',\n              'Content-Type': 'text/plain; charset=utf-8',\n            });\n            res.write(`'${currentPathname}' did not match any known endpoints`);\n            res.end();\n            return;\n          }\n\n          this.log.info('Starting MJPEG broadcast');\n          res.writeHead(200, {\n            'Cache-Control': 'no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0',\n            Pragma: 'no-cache',\n            Connection: 'close',\n            'Content-Type': `multipart/x-mixed-replace; boundary=${BOUNDARY_STRING}`\n          });\n\n          mjpegSocket.pipe(res);\n        });\n        mjpegServer.on('error', (e) => {\n          this.log.warn(e);\n          reject(e);\n        });\n        mjpegServer.on('close', () => {\n          this.log.debug(`MJPEG server at http://${host}:${port} has been closed`);\n        });\n        mjpegServer.on('listening', () => {\n          this.log.info(`Successfully started MJPEG server at http://${host}:${port}`);\n          resolve();\n        });\n        mjpegServer.listen(port, host);\n      });\n      mjpegSocket.on('error', (e) => {\n        this.log.error(e);\n        reject(e);\n      });\n    }).timeout(STREAMING_STARTUP_TIMEOUT_MS,\n      `Cannot connect to the streaming server within ${STREAMING_STARTUP_TIMEOUT_MS}ms`);\n  } catch (e) {\n    if (deviceStreamingProc.kill(0)) {\n      deviceStreamingProc.kill();\n    }\n    if (gstreamerPipeline.isRunning) {\n      await gstreamerPipeline.stop();\n    }\n    if (mjpegSocket) {\n      mjpegSocket.destroy();\n    }\n    if (mjpegServer && mjpegServer.listening) {\n      mjpegServer.close();\n    }\n    throw e;\n  }\n\n  this._screenStreamingProps = {\n    deviceStreamingProc,\n    gstreamerPipeline,\n    mjpegSocket,\n    mjpegServer,\n  };\n};\n\n/**\n * Stop screen streaming.\n * If no screen streaming server has been started then nothing is done.\n */\ncommands.mobileStopScreenStreaming = async function mobileStopScreenStreaming (/* options = {} */) {\n  if (_.isEmpty(this._screenStreamingProps)) {\n    if (!_.isUndefined(this._screenStreamingProps)) {\n      this.log.debug(`Screen streaming is not running. There is nothing to stop`);\n    }\n    return;\n  }\n\n  const {\n    deviceStreamingProc,\n    gstreamerPipeline,\n    mjpegSocket,\n    mjpegServer,\n  } = this._screenStreamingProps;\n\n  try {\n    mjpegSocket.end();\n    if (mjpegServer.listening) {\n      mjpegServer.close();\n    }\n    if (deviceStreamingProc.kill(0)) {\n      deviceStreamingProc.kill('SIGINT');\n    }\n    if (gstreamerPipeline.isRunning) {\n      try {\n        await gstreamerPipeline.stop('SIGINT');\n      } catch (e) {\n        this.log.warn(e);\n        try {\n          await gstreamerPipeline.stop('SIGKILL');\n        } catch (e1) {\n          this.log.error(e1);\n        }\n      }\n    }\n    this.log.info(`Successfully terminated the screen streaming MJPEG server`);\n  } finally {\n    this._screenStreamingProps = null;\n  }\n};\n\n\nexport default commands;\n"],"mappings":"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"}