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.
- package/build/index.js +1 -9
- package/build/lib/android-helpers.js +10 -194
- package/build/lib/android-helpers.js.map +1 -1
- package/build/lib/bootstrap.js +4 -45
- package/build/lib/bootstrap.js.map +1 -1
- package/build/lib/commands/actions.js +3 -78
- package/build/lib/commands/actions.js.map +1 -1
- package/build/lib/commands/alert.js +3 -10
- package/build/lib/commands/alert.js.map +1 -1
- package/build/lib/commands/app-management.js +4 -32
- package/build/lib/commands/app-management.js.map +1 -1
- package/build/lib/commands/context.js +6 -78
- package/build/lib/commands/context.js.map +1 -1
- package/build/lib/commands/coverage.js +1 -6
- package/build/lib/commands/coverage.js.map +1 -1
- package/build/lib/commands/element.js +4 -47
- package/build/lib/commands/element.js.map +1 -1
- package/build/lib/commands/emu-console.js +2 -7
- package/build/lib/commands/emu-console.js.map +1 -1
- package/build/lib/commands/execute.js +1 -13
- package/build/lib/commands/execute.js.map +1 -1
- package/build/lib/commands/file-actions.js +3 -44
- package/build/lib/commands/file-actions.js.map +1 -1
- package/build/lib/commands/find.js +3 -16
- package/build/lib/commands/find.js.map +1 -1
- package/build/lib/commands/general.js +4 -78
- package/build/lib/commands/general.js.map +1 -1
- package/build/lib/commands/ime.js +3 -14
- package/build/lib/commands/ime.js.map +1 -1
- package/build/lib/commands/index.js +3 -27
- package/build/lib/commands/index.js.map +1 -1
- package/build/lib/commands/intent.js +1 -40
- package/build/lib/commands/intent.js.map +1 -1
- package/build/lib/commands/log.js +3 -35
- package/build/lib/commands/log.js.map +1 -1
- package/build/lib/commands/media-projection.js +2 -47
- package/build/lib/commands/media-projection.js.map +1 -1
- package/build/lib/commands/network.js +4 -40
- package/build/lib/commands/network.js.map +1 -1
- package/build/lib/commands/performance.js +5 -57
- package/build/lib/commands/performance.js.map +1 -1
- package/build/lib/commands/recordscreen.js +1 -59
- package/build/lib/commands/recordscreen.js.map +1 -1
- package/build/lib/commands/shell.js +1 -15
- package/build/lib/commands/shell.js.map +1 -1
- package/build/lib/commands/streamscreen.js +7 -74
- package/build/lib/commands/streamscreen.js.map +1 -1
- package/build/lib/commands/system-bars.js +1 -23
- package/build/lib/commands/system-bars.js.map +1 -1
- package/build/lib/commands/touch.js +13 -79
- package/build/lib/commands/touch.js.map +1 -1
- package/build/lib/desired-caps.js +3 -4
- package/build/lib/desired-caps.js.map +1 -1
- package/build/lib/driver.js +8 -97
- package/build/lib/driver.js.map +1 -1
- package/build/lib/logger.js +1 -5
- package/build/lib/logger.js.map +1 -1
- package/build/lib/uiautomator.js +3 -24
- package/build/lib/uiautomator.js.map +1 -1
- package/build/lib/unlock-helpers.js +1 -61
- package/build/lib/unlock-helpers.js.map +1 -1
- package/build/lib/utils.js +1 -7
- package/build/lib/utils.js.map +1 -1
- package/build/lib/webview-helpers.js +1 -94
- package/build/lib/webview-helpers.js.map +1 -1
- package/lib/android-helpers.js +2 -1
- 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,
|
|
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":"
|
|
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` +
|
|
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
|
-
|
|
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
|
|
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":";;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAEA,MAAMA,QAAQ,GAAG,EAAjB;AAEA,MAAMC,sBAAsB,GAAG,CAA/B;AACA,MAAMC,4BAA4B,GAAG,IAArC;AACA,MAAMC,gBAAgB,GAAI,iBAAgBC,eAAA,CAAOC,SAAP,KAAqB,MAArB,GAA8B,EAAG,EAA3E;AACA,MAAMC,kBAAkB,GAAI,kBAAiBF,eAAA,CAAOC,SAAP,KAAqB,MAArB,GAA8B,EAAG,EAA9E;AACA,MAAME,oBAAoB,GAAG;EAC3BC,UAAU,EAAE,WADe;EAE3BC,SAAS,EAAE,iBAFgB;EAG3BC,OAAO,EAAE,kBAHkB;EAI3BC,aAAa,EAAE,kBAJY;EAK3BC,YAAY,EAAE;AALa,CAA7B;AAOA,MAAMC,mBAAmB,GAAG,cAA5B;AACA,MAAMC,gBAAgB,GAAG,uEAAzB;AACA,MAAMC,YAAY,GAAG,WAArB;AACA,MAAMC,QAAQ,GAAG,WAAjB;AACA,MAAMC,YAAY,GAAG,IAArB;AACA,MAAMC,eAAe,GAAG,EAAxB;AACA,MAAMC,eAAe,GAAG,OAAxB;AACA,MAAMC,eAAe,GAAG,oCAAxB;AAEA,MAAMC,4BAA4B,GAAG,sBAArC;;AAEA,SAASC,qBAAT,CAAgCC,UAAhC,EAA4CC,IAA5C,EAAkD;EAChD,OAAOC,eAAA,CAAOC,SAAP,CAAkB,GAAEH,UAAW,GAAd,GAAmBI,eAAA,CAAEC,QAAF,CAAWJ,IAAX,EAAiB;IAC1DK,MAAM,EAAE,CADkD;IAE1DC,QAAQ,EAAE;EAFgD,CAAjB,CAApC,CAAP;AAID;;AAED,eAAeC,2BAAf,CAA4CC,GAA5C,EAAiD;EAC/C,IAAI,CAACL,eAAA,CAAEM,IAAF,CAAO,MAAMD,GAAG,CAACE,KAAJ,CAAU,CAAC,OAAD,EAAUrB,mBAAV,CAAV,CAAb,CAAL,EAA8D;IAC5D,MAAM,IAAIsB,KAAJ,CACH,iBAAgBtB,mBAAoB,oDADjC,CAAN;EAED;;EAED,MAAMuB,sBAAsB,GAAG,EAA/B;;EACA,KAAK,MAAMC,UAAX,IAAyB,CAAClC,gBAAD,EAAmBG,kBAAnB,CAAzB,EAAiE;IAC/D8B,sBAAsB,CAACE,IAAvB,CAA4B,CAAC,YAAY;MACvC,IAAI;QACF,MAAMC,WAAA,CAAGC,KAAH,CAASH,UAAT,CAAN;MACD,CAFD,CAEE,OAAOI,CAAP,EAAU;QACV,MAAM,IAAIN,KAAJ,CAAW,QAAOE,UAAW,4DAAnB,GACb,OAAMvB,gBAAiB,yCADpB,CAAN;MAED;IACF,CAP2B,GAA5B;EAQD;;EACD,MAAM4B,iBAAA,CAAEC,GAAF,CAAMP,sBAAN,CAAN;EAEA,MAAMQ,mBAAmB,GAAG,EAA5B;;EACA,KAAK,MAAM,CAACC,IAAD,EAAOC,OAAP,CAAX,IAA8BnB,eAAA,CAAEoB,OAAF,CAAUxC,oBAAV,CAA9B,EAA+D;IAC7DqC,mBAAmB,CAACN,IAApB,CAAyB,CAAC,YAAY;MACpC,MAAM;QAACU;MAAD,IAAW,MAAM,IAAAC,kBAAA,EAAK3C,kBAAL,EAAyB,CAACuC,IAAD,CAAzB,CAAvB;;MACA,IAAI,CAAClB,eAAA,CAAEuB,QAAF,CAAWF,MAAX,EAAmBF,OAAnB,CAAL,EAAkC;QAChC,MAAM,IAAIX,KAAJ,CACH,kCAAiCU,IAAK,WAAUC,OAAQ,6BAAzD,GACC,OAAMhC,gBAAiB,yCAFpB,CAAN;MAGD;IACF,CAPwB,GAAzB;EAQD;;EACD,MAAM4B,iBAAA,CAAEC,GAAF,CAAMC,mBAAN,CAAN;AACD;;AAED,eAAeO,aAAf,CAA8BnB,GAA9B,EAAmCoB,GAAG,GAAG,IAAzC,EAA+C;EAC7C,MAAMC,MAAM,GAAG,MAAMrB,GAAG,CAACE,KAAJ,CAAU,CAAC,SAAD,EAAY,SAAZ,CAAV,CAArB;EACA,MAAMoB,MAAM,GAAG,EAAf;;EACA,KAAK,MAAM,CAACC,GAAD,EAAMC,OAAN,CAAX,IAA6B,CAC3B,CAAC,OAAD,EAAU,qBAAV,CAD2B,EAE3B,CAAC,QAAD,EAAW,sBAAX,CAF2B,EAG3B,CAAC,KAAD,EAAQ,aAAR,CAH2B,CAA7B,EAIG;IACD,MAAMC,KAAK,GAAGD,OAAO,CAACP,IAAR,CAAaI,MAAb,CAAd;;IACA,IAAI,CAACI,KAAL,EAAY;MACVL,GAAG,SAAH,IAAAA,GAAG,WAAH,YAAAA,GAAG,CAAEM,KAAL,CAAWL,MAAX;MACA,MAAM,IAAIlB,KAAJ,CAAW,2BAA0BoB,GAAI,gCAA/B,GACb,wCADG,CAAN;IAED;;IACDD,MAAM,CAACC,GAAD,CAAN,GAAcI,QAAQ,CAACF,KAAK,CAAC,CAAD,CAAN,EAAW,EAAX,CAAtB;EACD;;EACDH,MAAM,CAAC9B,IAAP,GAAcQ,GAAG,CAAC4B,WAAlB;EACA,OAAON,MAAP;AACD;;AAED,eAAeO,uBAAf,CAAwC7B,GAAxC,EAA6CoB,GAA7C,EAAkDU,UAAlD,EAA8DC,IAAI,GAAG,EAArE,EAAyE;EACvE,MAAM;IACJC,KADI;IAEJC,MAFI;IAGJC;EAHI,IAIFH,IAJJ;EAKA,MAAMI,aAAa,GAAGR,QAAQ,CAACK,KAAD,EAAQ,EAAR,CAAR,IAAuBF,UAAU,CAACE,KAAxD;EACA,MAAMI,cAAc,GAAGT,QAAQ,CAACM,MAAD,EAAS,EAAT,CAAR,IAAwBH,UAAU,CAACG,MAA1D;EACA,MAAMI,eAAe,GAAGV,QAAQ,CAACO,OAAD,EAAU,EAAV,CAAR,IAAyB/C,eAAjD;EACA,IAAImD,eAAe,GAAGzD,mBAAmB,GACtC,uBADmB,GAGnB,iBAAgBZ,sBAAuB,EAH1C;;EAIA,IAAI+D,KAAK,IAAIC,MAAb,EAAqB;IACnBK,eAAe,IAAK,WAAUH,aAAc,IAAGC,cAAe,EAA9D;EACD;;EACD,IAAIF,OAAJ,EAAa;IACXI,eAAe,IAAK,eAAcD,eAAgB,EAAlD;EACD;;EACD,MAAME,OAAO,GAAG,CACd,GAAGvC,GAAG,CAACwC,UAAJ,CAAeC,WADJ,EAEd,UAFc,EAKb,kBAAiBH,eAAgB,UALpB,CAAhB;EAOA,MAAMI,eAAe,GAAG,IAAAC,oBAAA,EAAM3C,GAAG,CAACwC,UAAJ,CAAeI,IAArB,EAA2BL,OAA3B,CAAxB;EACAG,eAAe,CAACG,EAAhB,CAAmB,MAAnB,EAA2B,CAACC,IAAD,EAAOC,MAAP,KAAkB;IAC3C3B,GAAG,CAACM,KAAJ,CAAW,6CAA4CoB,IAAK,YAAWC,MAAO,EAA9E;EACD,CAFD;EAIA,IAAIC,SAAS,GAAG,KAAhB;EACA,MAAMC,qBAAqB,GAAG3D,qBAAqB,CAACT,mBAAD,EAAsBiD,UAAU,CAACtC,IAAjC,CAAnD;;EACA,MAAM0D,cAAc,GAAIC,KAAD,IAAW;IAChC,MAAMC,MAAM,GAAGD,KAAK,CAACE,QAAN,EAAf;;IACA,IAAI1D,eAAA,CAAEM,IAAF,CAAOmD,MAAP,CAAJ,EAAoB;MAClBH,qBAAqB,CAACvB,KAAtB,CAA4B0B,MAA5B;IACD;EACF,CALD;;EAMAV,eAAe,CAACU,MAAhB,CAAuBP,EAAvB,CAA0B,MAA1B,EAAkCK,cAAlC;;EAEA,MAAMI,eAAe,GAAIH,KAAD,IAAW;IACjC,IAAI,CAACH,SAAL,EAAgB;MACdA,SAAS,GAAG,CAACrD,eAAA,CAAE4D,OAAF,CAAUJ,KAAV,CAAb;IACD;EACF,CAJD;;EAKAT,eAAe,CAAC1B,MAAhB,CAAuB6B,EAAvB,CAA0B,MAA1B,EAAkCS,eAAlC;;EAEA,IAAI;IACFlC,GAAG,CAACoC,IAAJ,CAAU,8BAA6BC,aAAA,CAAKC,KAAL,CAAW,CAAC1D,GAAG,CAACwC,UAAJ,CAAeI,IAAhB,EAAsB,GAAGL,OAAzB,CAAX,CAA8C,EAArF;IACA,MAAM,IAAAoB,0BAAA,EAAiB,MAAMX,SAAvB,EAAkC;MACtCY,MAAM,EAAE1F,4BAD8B;MAEtC2F,UAAU,EAAE;IAF0B,CAAlC,CAAN;EAID,CAND,CAME,OAAOpD,CAAP,EAAU;IACVW,GAAG,CAAC0C,aAAJ,CACG,8DAA6DrD,CAAC,CAACsD,OAAQ,EAD1E;EAED,CATD,SASU;IACRrB,eAAe,CAACU,MAAhB,CAAuBY,cAAvB,CAAsC,MAAtC,EAA8Cd,cAA9C;IACAR,eAAe,CAAC1B,MAAhB,CAAuBgD,cAAvB,CAAsC,MAAtC,EAA8CV,eAA9C;EACD;;EACD,OAAOZ,eAAP;AACD;;AAED,eAAeuB,qBAAf,CAAsCC,mBAAtC,EAA2DpC,UAA3D,EAAuEV,GAAvE,EAA4EW,IAAI,GAAG,EAAnF,EAAuF;EACrF,MAAM;IACJC,KADI;IAEJC,MAFI;IAGJkC,OAHI;IAIJC,OAJI;IAKJC,gBALI;IAMJC;EANI,IAOFvC,IAPJ;EAQA,MAAMI,aAAa,GAAGR,QAAQ,CAACK,KAAD,EAAQ,EAAR,CAAR,IAAuBF,UAAU,CAACE,KAAxD;EACA,MAAMI,cAAc,GAAGT,QAAQ,CAACM,MAAD,EAAS,EAAT,CAAR,IAAwBH,UAAU,CAACG,MAA1D;EACA,MAAMsC,iBAAiB,GAAG,IAAIC,wBAAJ,CAAerG,gBAAf,EAAiC,CACzD,IADyD,EAEzD,OAFyD,EAEhD,MAFgD,EAGzD,GAHyD,EAGpD,kBACF,SAAQkG,gBAAgB,GAAGI,IAAI,CAACC,GAAL,CAASvC,aAAT,EAAwBC,cAAxB,CAAH,GAA6CD,aAAc,GADjF,GAEF,UAASkC,gBAAgB,GAAGI,IAAI,CAACC,GAAL,CAASvC,aAAT,EAAwBC,cAAxB,CAAH,GAA6CA,cAAe,GAFnF,GAGF,aAAYN,UAAU,CAAC6C,GAAI,KAHzB,GAIH,kBAPuD,EAQzD,GARyD,EAQpD,WARoD,EASzD,GATyD,EASpD,OAToD,EAS3C,kBAT2C,EAUzD,GAVyD,EAUpD,YAVoD,EAWzD,GAXyD,EAWpD,OAXoD,EAW3C,kBAX2C,EAYzD,GAZyD,EAYpD,SAZoD,EAYxC,WAAUR,OAAQ,EAZsB,EAazD,GAbyD,EAapD,cAboD,EAanC,YAAW/E,eAAgB,EAbQ,EAczD,GAdyD,EAcpD,eAdoD,EAclC,QAAOJ,QAAS,EAdkB,EAcd,QAAOoF,OAAQ,EAdD,CAAjC,EAevB;IACDQ,KAAK,EAAE,CAACV,mBAAmB,CAAClD,MAArB,EAA6B,MAA7B,EAAqC,MAArC;EADN,CAfuB,CAA1B;EAkBAuD,iBAAiB,CAAC1B,EAAlB,CAAqB,MAArB,EAA6B,CAACC,IAAD,EAAOC,MAAP,KAAkB;IAC7C3B,GAAG,CAACM,KAAJ,CAAW,+CAA8CoB,IAAK,YAAWC,MAAO,EAAhF;EACD,CAFD;EAGA,MAAM8B,eAAe,GAAGvF,qBAAqB,CAAC,KAAD,EAAQwC,UAAU,CAACtC,IAAnB,CAA7C;;EACA,MAAMsF,iBAAiB,GAAG,CAAC9D,MAAD,EAASoC,MAAT,KAAoB;IAC5C,IAAIzD,eAAA,CAAEM,IAAF,CAAOmD,MAAM,IAAIpC,MAAjB,CAAJ,EAA8B;MAC5B6D,eAAe,CAACnD,KAAhB,CAAsB0B,MAAM,IAAIpC,MAAhC;IACD;EACF,CAJD;;EAKAuD,iBAAiB,CAAC1B,EAAlB,CAAqB,QAArB,EAA+BiC,iBAA/B;EACA,IAAIC,OAAO,GAAG,KAAd;;EACA,IAAI;IACF3D,GAAG,CAACoC,IAAJ,CAAU,gCAA+Be,iBAAiB,CAACS,GAAI,EAA/D;IACA,MAAMT,iBAAiB,CAACU,KAAlB,CAAwB,CAAxB,CAAN;IACA,MAAM,IAAAtB,0BAAA,EAAiB,YAAY;MACjC,IAAI;QACF,OAAO,CAAC,MAAM,IAAAuB,4BAAA,EAAgBd,OAAhB,EAAyBpF,QAAzB,CAAP,MAA+C,MAAtD;MACD,CAFD,CAEE,OAAOmG,GAAP,EAAY;QACZ,OAAO,KAAP;MACD;IACF,CANK,EAMH;MACDvB,MAAM,EAAE1F,4BADP;MAED2F,UAAU,EAAE;IAFX,CANG,CAAN;EAUD,CAbD,CAaE,OAAOpD,CAAP,EAAU;IACVsE,OAAO,GAAG,IAAV;IACA3D,GAAG,CAAC0C,aAAJ,CACG,+DAA8DrD,CAAC,CAACsD,OAAQ,EAD3E;EAED,CAjBD,SAiBU;IACR,IAAI,CAACO,kBAAD,IAAuBS,OAA3B,EAAoC;MAClCR,iBAAiB,CAACP,cAAlB,CAAiC,QAAjC,EAA2Cc,iBAA3C;IACD;EACF;;EACD,OAAOP,iBAAP;AACD;;AAED,SAASa,oBAAT,CAA+BC,GAA/B,EAAoC;EAClC,OAAOA,GAAG,CAACC,OAAJ,CAAY,iBAAZ,KACFD,GAAG,CAACE,MAAJ,CAAWC,aADT,IAEFH,GAAG,CAACI,UAAJ,CAAeD,aAFb,IAGFH,GAAG,CAACI,UAAJ,CAAeF,MAAf,CAAsBC,aAH3B;AAID;;AA2CDxH,QAAQ,CAAC0H,0BAAT,GAAsC,eAAeA,0BAAf,CAA2CC,OAAO,GAAG,EAArD,EAAyD;EAC7F,KAAKC,oBAAL,CAA0BvG,4BAA1B;EAEA,MAAM;IACJ2C,KADI;IAEJC,MAFI;IAGJC,OAHI;IAIJ2D,IAAI,GAAG9G,YAJH;IAKJ+G,IAAI,GAAG7G,YALH;IAMJ8G,QANI;IAOJ3B,OAAO,GAAGnF,YAAY,GAAG,CAPrB;IAQJkF,OAAO,GAAGjF,eARN;IASJmF,gBAAgB,GAAG,KATf;IAUJC,kBAAkB,GAAG;EAVjB,IAWFqB,OAXJ;;EAaA,IAAIhG,eAAA,CAAEqG,WAAF,CAAc,KAAKC,qBAAnB,CAAJ,EAA+C;IAC7C,MAAMlG,2BAA2B,CAAC,KAAKC,GAAN,CAAjC;EACD;;EACD,IAAI,CAACL,eAAA,CAAE4D,OAAF,CAAU,KAAK0C,qBAAf,CAAL,EAA4C;IAC1C,KAAK7E,GAAL,CAASoC,IAAT,CAAe,mDAAD,GACX,4CADH;IAEA;EACD;;EACD,IAAI,CAAC,MAAM,IAAA0B,4BAAA,EAAgBY,IAAhB,EAAsBD,IAAtB,CAAP,MAAwC,MAA5C,EAAoD;IAClD,KAAKzE,GAAL,CAASoC,IAAT,CAAe,aAAYsC,IAAK,OAAMD,IAAK,YAA7B,GACX,kDADH;IAEA;EACD;;EACD,IAAI,CAAC,MAAM,IAAAX,4BAAA,EAAgBd,OAAhB,EAAyBpF,QAAzB,CAAP,MAA+C,MAAnD,EAA2D;IACzD,KAAKoC,GAAL,CAAS0C,aAAT,CAAwB,aAAYM,OAAQ,OAAMpF,QAAS,YAApC,GACpB,0DADH;EAED;;EACD,KAAKiH,qBAAL,GAA6B,IAA7B;EAEA,MAAMnE,UAAU,GAAG,MAAMX,aAAa,CAAC,KAAKnB,GAAN,EAAW,KAAKoB,GAAhB,CAAtC;EACA,MAAM8C,mBAAmB,GAAG,MAAMrC,uBAAuB,CAAC,KAAK7B,GAAN,EAAW,KAAKoB,GAAhB,EAAqBU,UAArB,EAAiC;IACxFE,KADwF;IAExFC,MAFwF;IAGxFC;EAHwF,CAAjC,CAAzD;EAKA,IAAIqC,iBAAJ;;EACA,IAAI;IACFA,iBAAiB,GAAG,MAAMN,qBAAqB,CAACC,mBAAD,EAAsBpC,UAAtB,EAAkC,KAAKV,GAAvC,EAA4C;MACzFY,KADyF;MAEzFC,MAFyF;MAGzFkC,OAHyF;MAIzFC,OAJyF;MAKzFC,gBALyF;MAMzFC;IANyF,CAA5C,CAA/C;EAQD,CATD,CASE,OAAO7D,CAAP,EAAU;IACV,IAAIyD,mBAAmB,CAACgC,IAApB,CAAyB,CAAzB,CAAJ,EAAiC;MAC/BhC,mBAAmB,CAACgC,IAApB;IACD;;IACD,MAAMzF,CAAN;EACD;;EAED,IAAI0F,WAAJ;EACA,IAAIC,WAAJ;;EACA,IAAI;IACF,MAAM,IAAI1F,iBAAJ,CAAM,CAAC2F,OAAD,EAAUC,MAAV,KAAqB;MAC/BH,WAAW,GAAGI,YAAA,CAAIC,gBAAJ,CAAqBpC,OAArB,EAA8BpF,QAA9B,EAAwC,MAAM;QAC1D,KAAKoC,GAAL,CAASoC,IAAT,CAAe,mDAAkDxE,QAAS,IAAGoF,OAAQ,EAArF;QACAgC,WAAW,GAAGK,aAAA,CAAKC,YAAL,CAAkB,CAACrB,GAAD,EAAMsB,GAAN,KAAc;UAC5C,MAAMnB,aAAa,GAAGJ,oBAAoB,CAACC,GAAD,CAA1C;;UACA,MAAMuB,eAAe,GAAGC,YAAA,CAAIC,KAAJ,CAAUzB,GAAG,CAACwB,GAAd,EAAmBd,QAA3C;;UACA,KAAK3E,GAAL,CAASoC,IAAT,CAAe,mDAAkDgC,aAAc,GAAjE,GACX,IAAGH,GAAG,CAACC,OAAJ,CAAY,YAAZ,KAA6B,oBAAqB,QAAOsB,eAAgB,EAD/E;;UAGA,IAAIb,QAAQ,IAAIa,eAAe,KAAKb,QAApC,EAA8C;YAC5C,KAAK3E,GAAL,CAASoC,IAAT,CAAc,4EAAd;YACAmD,GAAG,CAACI,SAAJ,CAAc,GAAd,EAAmB;cACjBC,UAAU,EAAE,OADK;cAEjB,gBAAgB;YAFC,CAAnB;YAIAL,GAAG,CAACM,KAAJ,CAAW,IAAGL,eAAgB,qCAA9B;YACAD,GAAG,CAACO,GAAJ;YACA;UACD;;UAED,KAAK9F,GAAL,CAASoC,IAAT,CAAc,0BAAd;UACAmD,GAAG,CAACI,SAAJ,CAAc,GAAd,EAAmB;YACjB,iBAAiB,2EADA;YAEjBI,MAAM,EAAE,UAFS;YAGjBH,UAAU,EAAE,OAHK;YAIjB,gBAAiB,uCAAsC5H,eAAgB;UAJtD,CAAnB;UAOA+G,WAAW,CAACiB,IAAZ,CAAiBT,GAAjB;QACD,CA1Ba,CAAd;QA2BAP,WAAW,CAACvD,EAAZ,CAAe,OAAf,EAAyBpC,CAAD,IAAO;UAC7B,KAAKW,GAAL,CAASiG,IAAT,CAAc5G,CAAd;UACA6F,MAAM,CAAC7F,CAAD,CAAN;QACD,CAHD;QAIA2F,WAAW,CAACvD,EAAZ,CAAe,OAAf,EAAwB,MAAM;UAC5B,KAAKzB,GAAL,CAASM,KAAT,CAAgB,0BAAyBmE,IAAK,IAAGC,IAAK,kBAAtD;QACD,CAFD;QAGAM,WAAW,CAACvD,EAAZ,CAAe,WAAf,EAA4B,MAAM;UAChC,KAAKzB,GAAL,CAASoC,IAAT,CAAe,+CAA8CqC,IAAK,IAAGC,IAAK,EAA1E;UACAO,OAAO;QACR,CAHD;QAIAD,WAAW,CAACkB,MAAZ,CAAmBxB,IAAnB,EAAyBD,IAAzB;MACD,CAzCa,CAAd;MA0CAM,WAAW,CAACtD,EAAZ,CAAe,OAAf,EAAyBpC,CAAD,IAAO;QAC7B,KAAKW,GAAL,CAASmG,KAAT,CAAe9G,CAAf;QACA6F,MAAM,CAAC7F,CAAD,CAAN;MACD,CAHD;IAID,CA/CK,EA+CH+G,OA/CG,CA+CKtJ,4BA/CL,EAgDH,iDAAgDA,4BAA6B,IAhD1E,CAAN;EAiDD,CAlDD,CAkDE,OAAOuC,CAAP,EAAU;IACV,IAAIyD,mBAAmB,CAACgC,IAApB,CAAyB,CAAzB,CAAJ,EAAiC;MAC/BhC,mBAAmB,CAACgC,IAApB;IACD;;IACD,IAAI3B,iBAAiB,CAACkD,SAAtB,EAAiC;MAC/B,MAAMlD,iBAAiB,CAACmD,IAAlB,EAAN;IACD;;IACD,IAAIvB,WAAJ,EAAiB;MACfA,WAAW,CAACwB,OAAZ;IACD;;IACD,IAAIvB,WAAW,IAAIA,WAAW,CAACwB,SAA/B,EAA0C;MACxCxB,WAAW,CAACyB,KAAZ;IACD;;IACD,MAAMpH,CAAN;EACD;;EAED,KAAKwF,qBAAL,GAA6B;IAC3B/B,mBAD2B;IAE3BK,iBAF2B;IAG3B4B,WAH2B;IAI3BC;EAJ2B,CAA7B;AAMD,CApID;;AA0IApI,QAAQ,CAAC8J,yBAAT,GAAqC,eAAeA,yBAAf,GAA8D;EACjG,IAAInI,eAAA,CAAE4D,OAAF,CAAU,KAAK0C,qBAAf,CAAJ,EAA2C;IACzC,IAAI,CAACtG,eAAA,CAAEqG,WAAF,CAAc,KAAKC,qBAAnB,CAAL,EAAgD;MAC9C,KAAK7E,GAAL,CAASM,KAAT,CAAgB,2DAAhB;IACD;;IACD;EACD;;EAED,MAAM;IACJwC,mBADI;IAEJK,iBAFI;IAGJ4B,WAHI;IAIJC;EAJI,IAKF,KAAKH,qBALT;;EAOA,IAAI;IACFE,WAAW,CAACe,GAAZ;;IACA,IAAId,WAAW,CAACwB,SAAhB,EAA2B;MACzBxB,WAAW,CAACyB,KAAZ;IACD;;IACD,IAAI3D,mBAAmB,CAACgC,IAApB,CAAyB,CAAzB,CAAJ,EAAiC;MAC/BhC,mBAAmB,CAACgC,IAApB,CAAyB,QAAzB;IACD;;IACD,IAAI3B,iBAAiB,CAACkD,SAAtB,EAAiC;MAC/B,IAAI;QACF,MAAMlD,iBAAiB,CAACmD,IAAlB,CAAuB,QAAvB,CAAN;MACD,CAFD,CAEE,OAAOjH,CAAP,EAAU;QACV,KAAKW,GAAL,CAASiG,IAAT,CAAc5G,CAAd;;QACA,IAAI;UACF,MAAM8D,iBAAiB,CAACmD,IAAlB,CAAuB,SAAvB,CAAN;QACD,CAFD,CAEE,OAAOK,EAAP,EAAW;UACX,KAAK3G,GAAL,CAASmG,KAAT,CAAeQ,EAAf;QACD;MACF;IACF;;IACD,KAAK3G,GAAL,CAASoC,IAAT,CAAe,2DAAf;EACD,CArBD,SAqBU;IACR,KAAKyC,qBAAL,GAA6B,IAA7B;EACD;AACF,CAvCD;;eA0CejI,Q"}
|
|
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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"}
|