appium-android-driver 5.13.1 → 5.13.2
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/CHANGELOG.md +7 -0
- package/build/index.js +43 -40
- package/build/lib/android-helpers.d.ts +136 -0
- package/build/lib/android-helpers.d.ts.map +1 -0
- package/build/lib/android-helpers.js +760 -679
- package/build/lib/android-helpers.js.map +1 -1
- package/build/lib/bootstrap.d.ts +29 -0
- package/build/lib/bootstrap.d.ts.map +1 -0
- package/build/lib/bootstrap.js +192 -179
- package/build/lib/bootstrap.js.map +1 -1
- package/build/lib/commands/actions.d.ts +209 -0
- package/build/lib/commands/actions.d.ts.map +1 -0
- package/build/lib/commands/actions.js +327 -265
- package/build/lib/commands/actions.js.map +1 -1
- package/build/lib/commands/alert.d.ts +10 -0
- package/build/lib/commands/alert.d.ts.map +1 -0
- package/build/lib/commands/alert.js +12 -18
- package/build/lib/commands/alert.js.map +1 -1
- package/build/lib/commands/app-management.d.ts +314 -0
- package/build/lib/commands/app-management.d.ts.map +1 -0
- package/build/lib/commands/app-management.js +278 -110
- package/build/lib/commands/app-management.js.map +1 -1
- package/build/lib/commands/context.d.ts +94 -0
- package/build/lib/commands/context.d.ts.map +1 -0
- package/build/lib/commands/context.js +412 -260
- package/build/lib/commands/context.js.map +1 -1
- package/build/lib/commands/coverage.d.ts +5 -0
- package/build/lib/commands/coverage.d.ts.map +1 -0
- package/build/lib/commands/coverage.js +14 -17
- package/build/lib/commands/coverage.js.map +1 -1
- package/build/lib/commands/element.d.ts +36 -0
- package/build/lib/commands/element.d.ts.map +1 -0
- package/build/lib/commands/element.js +97 -127
- package/build/lib/commands/element.js.map +1 -1
- package/build/lib/commands/emu-console.d.ts +49 -0
- package/build/lib/commands/emu-console.d.ts.map +1 -0
- package/build/lib/commands/emu-console.js +36 -25
- package/build/lib/commands/emu-console.js.map +1 -1
- package/build/lib/commands/execute.d.ts +6 -0
- package/build/lib/commands/execute.d.ts.map +1 -0
- package/build/lib/commands/execute.js +68 -69
- package/build/lib/commands/execute.js.map +1 -1
- package/build/lib/commands/file-actions.d.ts +129 -0
- package/build/lib/commands/file-actions.d.ts.map +1 -0
- package/build/lib/commands/file-actions.js +321 -178
- package/build/lib/commands/file-actions.js.map +1 -1
- package/build/lib/commands/find.d.ts +13 -0
- package/build/lib/commands/find.d.ts.map +1 -0
- package/build/lib/commands/find.js +69 -51
- package/build/lib/commands/find.js.map +1 -1
- package/build/lib/commands/general.d.ts +133 -0
- package/build/lib/commands/general.d.ts.map +1 -0
- package/build/lib/commands/general.js +275 -216
- package/build/lib/commands/general.js.map +1 -1
- package/build/lib/commands/ime.d.ts +11 -0
- package/build/lib/commands/ime.d.ts.map +1 -0
- package/build/lib/commands/ime.js +27 -33
- package/build/lib/commands/ime.js.map +1 -1
- package/build/lib/commands/index.d.ts +3 -0
- package/build/lib/commands/index.d.ts.map +1 -0
- package/build/lib/commands/index.js +32 -35
- package/build/lib/commands/index.js.map +1 -1
- package/build/lib/commands/intent.d.ts +418 -0
- package/build/lib/commands/intent.d.ts.map +1 -0
- package/build/lib/commands/intent.js +281 -151
- package/build/lib/commands/intent.js.map +1 -1
- package/build/lib/commands/keyboard.d.ts +6 -0
- package/build/lib/commands/keyboard.d.ts.map +1 -0
- package/build/lib/commands/keyboard.js +6 -14
- package/build/lib/commands/keyboard.js.map +1 -1
- package/build/lib/commands/log.d.ts +45 -0
- package/build/lib/commands/log.d.ts.map +1 -0
- package/build/lib/commands/log.js +117 -103
- package/build/lib/commands/log.js.map +1 -1
- package/build/lib/commands/media-projection.d.ts +144 -0
- package/build/lib/commands/media-projection.d.ts.map +1 -0
- package/build/lib/commands/media-projection.js +228 -171
- package/build/lib/commands/media-projection.js.map +1 -1
- package/build/lib/commands/network.d.ts +139 -0
- package/build/lib/commands/network.d.ts.map +1 -0
- package/build/lib/commands/network.js +249 -181
- package/build/lib/commands/network.js.map +1 -1
- package/build/lib/commands/performance.d.ts +101 -0
- package/build/lib/commands/performance.d.ts.map +1 -0
- package/build/lib/commands/performance.js +390 -236
- package/build/lib/commands/performance.js.map +1 -1
- package/build/lib/commands/permissions.d.ts +93 -0
- package/build/lib/commands/permissions.d.ts.map +1 -0
- package/build/lib/commands/permissions.js +133 -93
- package/build/lib/commands/permissions.js.map +1 -1
- package/build/lib/commands/recordscreen.d.ts +194 -0
- package/build/lib/commands/recordscreen.d.ts.map +1 -0
- package/build/lib/commands/recordscreen.js +293 -224
- package/build/lib/commands/recordscreen.js.map +1 -1
- package/build/lib/commands/shell.d.ts +8 -0
- package/build/lib/commands/shell.d.ts.map +1 -0
- package/build/lib/commands/shell.js +38 -43
- package/build/lib/commands/shell.js.map +1 -1
- package/build/lib/commands/streamscreen.d.ts +104 -0
- package/build/lib/commands/streamscreen.d.ts.map +1 -0
- package/build/lib/commands/streamscreen.js +364 -305
- package/build/lib/commands/streamscreen.js.map +1 -1
- package/build/lib/commands/system-bars.d.ts +100 -0
- package/build/lib/commands/system-bars.d.ts.map +1 -0
- package/build/lib/commands/system-bars.js +148 -90
- package/build/lib/commands/system-bars.js.map +1 -1
- package/build/lib/commands/touch.d.ts +30 -0
- package/build/lib/commands/touch.d.ts.map +1 -0
- package/build/lib/commands/touch.js +311 -287
- package/build/lib/commands/touch.js.map +1 -1
- package/build/lib/desired-caps.d.ts +353 -0
- package/build/lib/desired-caps.d.ts.map +1 -0
- package/build/lib/desired-caps.js +291 -292
- package/build/lib/desired-caps.js.map +1 -1
- package/build/lib/driver.d.ts +430 -0
- package/build/lib/driver.d.ts.map +1 -0
- package/build/lib/driver.js +449 -384
- package/build/lib/driver.js.map +1 -1
- package/build/lib/logger.d.ts +3 -0
- package/build/lib/logger.d.ts.map +1 -0
- package/build/lib/logger.js +5 -11
- package/build/lib/logger.js.map +1 -1
- package/build/lib/method-map.d.ts +389 -0
- package/build/lib/method-map.d.ts.map +1 -0
- package/build/lib/method-map.js +220 -394
- package/build/lib/method-map.js.map +1 -1
- package/build/lib/stubs.d.ts +8 -0
- package/build/lib/stubs.d.ts.map +1 -0
- package/build/lib/stubs.js +5 -0
- package/build/lib/stubs.js.map +1 -0
- package/build/lib/uiautomator.d.ts +24 -0
- package/build/lib/uiautomator.d.ts.map +1 -0
- package/build/lib/uiautomator.js +86 -82
- package/build/lib/uiautomator.js.map +1 -1
- package/build/lib/unlock-helpers.d.ts +38 -0
- package/build/lib/unlock-helpers.d.ts.map +1 -0
- package/build/lib/unlock-helpers.js +228 -204
- package/build/lib/unlock-helpers.js.map +1 -1
- package/build/lib/utils.d.ts +11 -0
- package/build/lib/utils.d.ts.map +1 -0
- package/build/lib/utils.js +23 -18
- package/build/lib/utils.js.map +1 -1
- package/build/lib/webview-helpers.d.ts +223 -0
- package/build/lib/webview-helpers.d.ts.map +1 -0
- package/build/lib/webview-helpers.js +476 -298
- package/build/lib/webview-helpers.js.map +1 -1
- package/index.js +3 -1
- package/lib/android-helpers.js +2 -1
- package/lib/stubs.ts +8 -0
- package/lib/unlock-helpers.js +2 -2
- package/package.json +23 -14
|
@@ -1,32 +1,29 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
require("
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
var _asyncbox = require("asyncbox");
|
|
17
|
-
var _child_process = require("child_process");
|
|
18
|
-
var _url = _interopRequireDefault(require("url"));
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
const lodash_1 = __importDefault(require("lodash"));
|
|
7
|
+
const support_1 = require("@appium/support");
|
|
8
|
+
const teen_process_1 = require("teen_process");
|
|
9
|
+
const portscanner_1 = require("portscanner");
|
|
10
|
+
const http_1 = __importDefault(require("http"));
|
|
11
|
+
const net_1 = __importDefault(require("net"));
|
|
12
|
+
const bluebird_1 = __importDefault(require("bluebird"));
|
|
13
|
+
const asyncbox_1 = require("asyncbox");
|
|
14
|
+
const child_process_1 = require("child_process");
|
|
15
|
+
const url_1 = __importDefault(require("url"));
|
|
19
16
|
const commands = {};
|
|
20
17
|
const RECORDING_INTERVAL_SEC = 5;
|
|
21
18
|
const STREAMING_STARTUP_TIMEOUT_MS = 5000;
|
|
22
|
-
const GSTREAMER_BINARY = `gst-launch-1.0${
|
|
23
|
-
const GST_INSPECT_BINARY = `gst-inspect-1.0${
|
|
19
|
+
const GSTREAMER_BINARY = `gst-launch-1.0${support_1.system.isWindows() ? '.exe' : ''}`;
|
|
20
|
+
const GST_INSPECT_BINARY = `gst-inspect-1.0${support_1.system.isWindows() ? '.exe' : ''}`;
|
|
24
21
|
const REQUIRED_GST_PLUGINS = {
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
22
|
+
avdec_h264: 'gst-libav',
|
|
23
|
+
h264parse: 'gst-plugins-bad',
|
|
24
|
+
jpegenc: 'gst-plugins-good',
|
|
25
|
+
tcpserversink: 'gst-plugins-base',
|
|
26
|
+
multipartmux: 'gst-plugins-good',
|
|
30
27
|
};
|
|
31
28
|
const SCREENRECORD_BINARY = 'screenrecord';
|
|
32
29
|
const GST_TUTORIAL_URL = 'https://gstreamer.freedesktop.org/documentation/installing/index.html';
|
|
@@ -34,316 +31,378 @@ const DEFAULT_HOST = '127.0.0.1';
|
|
|
34
31
|
const TCP_HOST = '127.0.0.1';
|
|
35
32
|
const DEFAULT_PORT = 8093;
|
|
36
33
|
const DEFAULT_QUALITY = 70;
|
|
37
|
-
const DEFAULT_BITRATE = 4000000;
|
|
34
|
+
const DEFAULT_BITRATE = 4000000; // 4 Mbps
|
|
38
35
|
const BOUNDARY_STRING = '--2ae9746887f170b8cf7c271047ce314c';
|
|
39
36
|
const ADB_SCREEN_STREAMING_FEATURE = 'adb_screen_streaming';
|
|
40
37
|
function createStreamingLogger(streamName, udid) {
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
38
|
+
return support_1.logger.getLogger(`${streamName}@` + lodash_1.default.truncate(udid, {
|
|
39
|
+
length: 8,
|
|
40
|
+
omission: '',
|
|
41
|
+
}));
|
|
45
42
|
}
|
|
46
43
|
async function verifyStreamingRequirements(adb) {
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
moduleCheckPromises
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
44
|
+
if (!lodash_1.default.trim(await adb.shell(['which', SCREENRECORD_BINARY]))) {
|
|
45
|
+
throw new Error(`The required '${SCREENRECORD_BINARY}' binary is not available on the device under test`);
|
|
46
|
+
}
|
|
47
|
+
const gstreamerCheckPromises = [];
|
|
48
|
+
for (const binaryName of [GSTREAMER_BINARY, GST_INSPECT_BINARY]) {
|
|
49
|
+
gstreamerCheckPromises.push((async () => {
|
|
50
|
+
try {
|
|
51
|
+
await support_1.fs.which(binaryName);
|
|
52
|
+
}
|
|
53
|
+
catch (e) {
|
|
54
|
+
throw new Error(`The '${binaryName}' binary is not available in the PATH on the host system. ` +
|
|
55
|
+
`See ${GST_TUTORIAL_URL} for more details on how to install it.`);
|
|
56
|
+
}
|
|
57
|
+
})());
|
|
58
|
+
}
|
|
59
|
+
await bluebird_1.default.all(gstreamerCheckPromises);
|
|
60
|
+
const moduleCheckPromises = [];
|
|
61
|
+
for (const [name, modName] of lodash_1.default.toPairs(REQUIRED_GST_PLUGINS)) {
|
|
62
|
+
moduleCheckPromises.push((async () => {
|
|
63
|
+
const { stdout } = await (0, teen_process_1.exec)(GST_INSPECT_BINARY, [name]);
|
|
64
|
+
if (!lodash_1.default.includes(stdout, modName)) {
|
|
65
|
+
throw new Error(`The required GStreamer plugin '${name}' from '${modName}' module is not installed. ` +
|
|
66
|
+
`See ${GST_TUTORIAL_URL} for more details on how to install it.`);
|
|
67
|
+
}
|
|
68
|
+
})());
|
|
69
|
+
}
|
|
70
|
+
await bluebird_1.default.all(moduleCheckPromises);
|
|
73
71
|
}
|
|
74
72
|
async function getDeviceInfo(adb, log = null) {
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
73
|
+
const output = await adb.shell(['dumpsys', 'display']);
|
|
74
|
+
const result = {};
|
|
75
|
+
for (const [key, pattern] of [
|
|
76
|
+
['width', /\bdeviceWidth=(\d+)/],
|
|
77
|
+
['height', /\bdeviceHeight=(\d+)/],
|
|
78
|
+
['fps', /\bfps=(\d+)/],
|
|
79
|
+
]) {
|
|
80
|
+
const match = pattern.exec(output);
|
|
81
|
+
if (!match) {
|
|
82
|
+
log?.debug(output);
|
|
83
|
+
throw new Error(`Cannot parse the device ${key} from the adb command output. ` +
|
|
84
|
+
`Check the server log for more details.`);
|
|
85
|
+
}
|
|
86
|
+
result[key] = parseInt(match[1], 10);
|
|
82
87
|
}
|
|
83
|
-
result
|
|
84
|
-
|
|
85
|
-
result.udid = adb.curDeviceId;
|
|
86
|
-
return result;
|
|
88
|
+
result.udid = adb.curDeviceId;
|
|
89
|
+
return result;
|
|
87
90
|
}
|
|
88
91
|
async function initDeviceStreamingProc(adb, log, deviceInfo, opts = {}) {
|
|
89
|
-
|
|
90
|
-
width,
|
|
91
|
-
height,
|
|
92
|
-
bitRate
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
screenRecordCmd += ` --size=${adjustedWidth}x${adjustedHeight}`;
|
|
100
|
-
}
|
|
101
|
-
if (bitRate) {
|
|
102
|
-
screenRecordCmd += ` --bit-rate=${adjustedBitrate}`;
|
|
103
|
-
}
|
|
104
|
-
const adbArgs = [...adb.executable.defaultArgs, 'exec-out', `while true; do ${screenRecordCmd} -; done`];
|
|
105
|
-
const deviceStreaming = (0, _child_process.spawn)(adb.executable.path, adbArgs);
|
|
106
|
-
deviceStreaming.on('exit', (code, signal) => {
|
|
107
|
-
log.debug(`Device streaming process exited with code ${code}, signal ${signal}`);
|
|
108
|
-
});
|
|
109
|
-
let isStarted = false;
|
|
110
|
-
const deviceStreamingLogger = createStreamingLogger(SCREENRECORD_BINARY, deviceInfo.udid);
|
|
111
|
-
const errorsListener = chunk => {
|
|
112
|
-
const stderr = chunk.toString();
|
|
113
|
-
if (_lodash.default.trim(stderr)) {
|
|
114
|
-
deviceStreamingLogger.debug(stderr);
|
|
92
|
+
const { width, height, bitRate, } = opts;
|
|
93
|
+
const adjustedWidth = parseInt(width, 10) || deviceInfo.width;
|
|
94
|
+
const adjustedHeight = parseInt(height, 10) || deviceInfo.height;
|
|
95
|
+
const adjustedBitrate = parseInt(bitRate, 10) || DEFAULT_BITRATE;
|
|
96
|
+
let screenRecordCmd = SCREENRECORD_BINARY +
|
|
97
|
+
` --output-format=h264` +
|
|
98
|
+
// 5 seconds is fine to detect rotation changes
|
|
99
|
+
` --time-limit=${RECORDING_INTERVAL_SEC}`;
|
|
100
|
+
if (width || height) {
|
|
101
|
+
screenRecordCmd += ` --size=${adjustedWidth}x${adjustedHeight}`;
|
|
115
102
|
}
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
const startupListener = chunk => {
|
|
119
|
-
if (!isStarted) {
|
|
120
|
-
isStarted = !_lodash.default.isEmpty(chunk);
|
|
103
|
+
if (bitRate) {
|
|
104
|
+
screenRecordCmd += ` --bit-rate=${adjustedBitrate}`;
|
|
121
105
|
}
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
106
|
+
const adbArgs = [
|
|
107
|
+
...adb.executable.defaultArgs,
|
|
108
|
+
'exec-out',
|
|
109
|
+
// The loop is required, because by default the maximum record duration
|
|
110
|
+
// for screenrecord is always limited
|
|
111
|
+
`while true; do ${screenRecordCmd} -; done`,
|
|
112
|
+
];
|
|
113
|
+
const deviceStreaming = (0, child_process_1.spawn)(adb.executable.path, adbArgs);
|
|
114
|
+
deviceStreaming.on('exit', (code, signal) => {
|
|
115
|
+
log.debug(`Device streaming process exited with code ${code}, signal ${signal}`);
|
|
129
116
|
});
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
117
|
+
let isStarted = false;
|
|
118
|
+
const deviceStreamingLogger = createStreamingLogger(SCREENRECORD_BINARY, deviceInfo.udid);
|
|
119
|
+
const errorsListener = (chunk) => {
|
|
120
|
+
const stderr = chunk.toString();
|
|
121
|
+
if (lodash_1.default.trim(stderr)) {
|
|
122
|
+
deviceStreamingLogger.debug(stderr);
|
|
123
|
+
}
|
|
124
|
+
};
|
|
125
|
+
deviceStreaming.stderr.on('data', errorsListener);
|
|
126
|
+
const startupListener = (chunk) => {
|
|
127
|
+
if (!isStarted) {
|
|
128
|
+
isStarted = !lodash_1.default.isEmpty(chunk);
|
|
129
|
+
}
|
|
130
|
+
};
|
|
131
|
+
deviceStreaming.stdout.on('data', startupListener);
|
|
132
|
+
try {
|
|
133
|
+
log.info(`Starting device streaming: ${support_1.util.quote([adb.executable.path, ...adbArgs])}`);
|
|
134
|
+
await (0, asyncbox_1.waitForCondition)(() => isStarted, {
|
|
135
|
+
waitMs: STREAMING_STARTUP_TIMEOUT_MS,
|
|
136
|
+
intervalMs: 300,
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
catch (e) {
|
|
140
|
+
log.errorAndThrow(`Cannot start the screen streaming process. Original error: ${e.message}`);
|
|
141
|
+
}
|
|
142
|
+
finally {
|
|
143
|
+
deviceStreaming.stderr.removeListener('data', errorsListener);
|
|
144
|
+
deviceStreaming.stdout.removeListener('data', startupListener);
|
|
145
|
+
}
|
|
146
|
+
return deviceStreaming;
|
|
137
147
|
}
|
|
138
148
|
async function initGstreamerPipeline(deviceStreamingProc, deviceInfo, log, opts = {}) {
|
|
139
|
-
|
|
140
|
-
width,
|
|
141
|
-
height,
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
let didFail = false;
|
|
163
|
-
try {
|
|
164
|
-
log.info(`Starting GStreamer pipeline: ${gstreamerPipeline.rep}`);
|
|
165
|
-
await gstreamerPipeline.start(0);
|
|
166
|
-
await (0, _asyncbox.waitForCondition)(async () => {
|
|
167
|
-
try {
|
|
168
|
-
return (await (0, _portscanner.checkPortStatus)(tcpPort, TCP_HOST)) === 'open';
|
|
169
|
-
} catch (ign) {
|
|
170
|
-
return false;
|
|
171
|
-
}
|
|
172
|
-
}, {
|
|
173
|
-
waitMs: STREAMING_STARTUP_TIMEOUT_MS,
|
|
174
|
-
intervalMs: 300
|
|
149
|
+
const { width, height, quality, tcpPort, considerRotation, logPipelineDetails, } = opts;
|
|
150
|
+
const adjustedWidth = parseInt(width, 10) || deviceInfo.width;
|
|
151
|
+
const adjustedHeight = parseInt(height, 10) || deviceInfo.height;
|
|
152
|
+
const gstreamerPipeline = new teen_process_1.SubProcess(GSTREAMER_BINARY, [
|
|
153
|
+
'-v',
|
|
154
|
+
'fdsrc', 'fd=0',
|
|
155
|
+
'!', 'video/x-h264,' +
|
|
156
|
+
`width=${considerRotation ? Math.max(adjustedWidth, adjustedHeight) : adjustedWidth},` +
|
|
157
|
+
`height=${considerRotation ? Math.max(adjustedWidth, adjustedHeight) : adjustedHeight},` +
|
|
158
|
+
`framerate=${deviceInfo.fps}/1,` +
|
|
159
|
+
'byte-stream=true',
|
|
160
|
+
'!', 'h264parse',
|
|
161
|
+
'!', 'queue', 'leaky=downstream',
|
|
162
|
+
'!', 'avdec_h264',
|
|
163
|
+
'!', 'queue', 'leaky=downstream',
|
|
164
|
+
'!', 'jpegenc', `quality=${quality}`,
|
|
165
|
+
'!', 'multipartmux', `boundary=${BOUNDARY_STRING}`,
|
|
166
|
+
'!', 'tcpserversink', `host=${TCP_HOST}`, `port=${tcpPort}`,
|
|
167
|
+
], {
|
|
168
|
+
stdio: [deviceStreamingProc.stdout, 'pipe', 'pipe']
|
|
169
|
+
});
|
|
170
|
+
gstreamerPipeline.on('exit', (code, signal) => {
|
|
171
|
+
log.debug(`Pipeline streaming process exited with code ${code}, signal ${signal}`);
|
|
175
172
|
});
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
173
|
+
const gstreamerLogger = createStreamingLogger('gst', deviceInfo.udid);
|
|
174
|
+
const gstOutputListener = (stdout, stderr) => {
|
|
175
|
+
if (lodash_1.default.trim(stderr || stdout)) {
|
|
176
|
+
gstreamerLogger.debug(stderr || stdout);
|
|
177
|
+
}
|
|
178
|
+
};
|
|
179
|
+
gstreamerPipeline.on('output', gstOutputListener);
|
|
180
|
+
let didFail = false;
|
|
181
|
+
try {
|
|
182
|
+
log.info(`Starting GStreamer pipeline: ${gstreamerPipeline.rep}`);
|
|
183
|
+
await gstreamerPipeline.start(0);
|
|
184
|
+
await (0, asyncbox_1.waitForCondition)(async () => {
|
|
185
|
+
try {
|
|
186
|
+
return (await (0, portscanner_1.checkPortStatus)(tcpPort, TCP_HOST)) === 'open';
|
|
187
|
+
}
|
|
188
|
+
catch (ign) {
|
|
189
|
+
return false;
|
|
190
|
+
}
|
|
191
|
+
}, {
|
|
192
|
+
waitMs: STREAMING_STARTUP_TIMEOUT_MS,
|
|
193
|
+
intervalMs: 300,
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
catch (e) {
|
|
197
|
+
didFail = true;
|
|
198
|
+
log.errorAndThrow(`Cannot start the screen streaming pipeline. Original error: ${e.message}`);
|
|
182
199
|
}
|
|
183
|
-
|
|
184
|
-
|
|
200
|
+
finally {
|
|
201
|
+
if (!logPipelineDetails || didFail) {
|
|
202
|
+
gstreamerPipeline.removeListener('output', gstOutputListener);
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
return gstreamerPipeline;
|
|
185
206
|
}
|
|
186
207
|
function extractRemoteAddress(req) {
|
|
187
|
-
|
|
208
|
+
return req.headers['x-forwarded-for']
|
|
209
|
+
|| req.socket.remoteAddress
|
|
210
|
+
|| req.connection.remoteAddress
|
|
211
|
+
|| req.connection.socket.remoteAddress;
|
|
188
212
|
}
|
|
213
|
+
/**
|
|
214
|
+
* @typedef {Object} StartScreenStreamingOptions
|
|
215
|
+
*
|
|
216
|
+
* @property {?number} width - The scaled width of the device's screen. If unset then the script will assign it
|
|
217
|
+
* to the actual screen width measured in pixels.
|
|
218
|
+
* @property {?number} height - The scaled height of the device's screen. If unset then the script will assign it
|
|
219
|
+
* to the actual screen height measured in pixels.
|
|
220
|
+
* @property {?number} bitRate - The video bit rate for the video, in bits per second.
|
|
221
|
+
* The default value is 4000000 (4 Mb/s). You can increase the bit rate to improve video quality,
|
|
222
|
+
* but doing so results in larger movie files.
|
|
223
|
+
* @property {?string} host [127.0.0.1] - The IP address/host name to start the MJPEG server on.
|
|
224
|
+
* You can set it to `0.0.0.0` to trigger the broadcast on all available network interfaces.
|
|
225
|
+
* @property {?string} pathname - The HTTP request path the MJPEG server should be available on.
|
|
226
|
+
* If unset then any pathname on the given `host`/`port` combination will work. Note that the value
|
|
227
|
+
* should always start with a single slash: `/`
|
|
228
|
+
* @property {?number} tcpPort [8094] - The port number to start the internal TCP MJPEG broadcast on.
|
|
229
|
+
* This type of broadcast always starts on the loopback interface (`127.0.0.1`).
|
|
230
|
+
* @property {?number} port [8093] - The port number to start the MJPEG server on.
|
|
231
|
+
* @property {?number} quality [70] - The quality value for the streamed JPEG images.
|
|
232
|
+
* This number should be in range [1, 100], where 100 is the best quality.
|
|
233
|
+
* @property {?boolean} considerRotation [false] - If set to `true` then GStreamer pipeline will
|
|
234
|
+
* increase the dimensions of the resulting images to properly fit images in both landscape and
|
|
235
|
+
* portrait orientations. Set it to `true` if the device rotation is not going to be the same during the
|
|
236
|
+
* broadcasting session.
|
|
237
|
+
* @property {?boolean} logPipelineDetails [false] - Whether to log GStreamer pipeline events into
|
|
238
|
+
* the standard log output. Might be useful for debugging purposes.
|
|
239
|
+
*/
|
|
240
|
+
/**
|
|
241
|
+
* Starts device screen broadcast by creating MJPEG server.
|
|
242
|
+
* Multiple calls to this method have no effect unless the previous streaming
|
|
243
|
+
* session is stopped.
|
|
244
|
+
* This method only works if the `adb_screen_streaming` feature is
|
|
245
|
+
* enabled on the server side.
|
|
246
|
+
*
|
|
247
|
+
* @param {?StartScreenStreamingOptions} options - The available options.
|
|
248
|
+
* @throws {Error} If screen streaming has failed to start or
|
|
249
|
+
* is not supported on the host system or
|
|
250
|
+
* the corresponding server feature is not enabled.
|
|
251
|
+
*/
|
|
189
252
|
commands.mobileStartScreenStreaming = async function mobileStartScreenStreaming(options = {}) {
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
bitRate,
|
|
195
|
-
host = DEFAULT_HOST,
|
|
196
|
-
port = DEFAULT_PORT,
|
|
197
|
-
pathname,
|
|
198
|
-
tcpPort = DEFAULT_PORT + 1,
|
|
199
|
-
quality = DEFAULT_QUALITY,
|
|
200
|
-
considerRotation = false,
|
|
201
|
-
logPipelineDetails = false
|
|
202
|
-
} = options;
|
|
203
|
-
if (_lodash.default.isUndefined(this._screenStreamingProps)) {
|
|
204
|
-
await verifyStreamingRequirements(this.adb);
|
|
205
|
-
}
|
|
206
|
-
if (!_lodash.default.isEmpty(this._screenStreamingProps)) {
|
|
207
|
-
this.log.info(`The screen streaming session is already running. ` + `Stop it first in order to start a new one.`);
|
|
208
|
-
return;
|
|
209
|
-
}
|
|
210
|
-
if ((await (0, _portscanner.checkPortStatus)(port, host)) === 'open') {
|
|
211
|
-
this.log.info(`The port #${port} at ${host} is busy. ` + `Assuming the screen streaming is already running`);
|
|
212
|
-
return;
|
|
213
|
-
}
|
|
214
|
-
if ((await (0, _portscanner.checkPortStatus)(tcpPort, TCP_HOST)) === 'open') {
|
|
215
|
-
this.log.errorAndThrow(`The port #${tcpPort} at ${TCP_HOST} is busy. ` + `Make sure there are no leftovers from previous sessions.`);
|
|
216
|
-
}
|
|
217
|
-
this._screenStreamingProps = null;
|
|
218
|
-
const deviceInfo = await getDeviceInfo(this.adb, this.log);
|
|
219
|
-
const deviceStreamingProc = await initDeviceStreamingProc(this.adb, this.log, deviceInfo, {
|
|
220
|
-
width,
|
|
221
|
-
height,
|
|
222
|
-
bitRate
|
|
223
|
-
});
|
|
224
|
-
let gstreamerPipeline;
|
|
225
|
-
try {
|
|
226
|
-
gstreamerPipeline = await initGstreamerPipeline(deviceStreamingProc, deviceInfo, this.log, {
|
|
227
|
-
width,
|
|
228
|
-
height,
|
|
229
|
-
quality,
|
|
230
|
-
tcpPort,
|
|
231
|
-
considerRotation,
|
|
232
|
-
logPipelineDetails
|
|
233
|
-
});
|
|
234
|
-
} catch (e) {
|
|
235
|
-
if (deviceStreamingProc.kill(0)) {
|
|
236
|
-
deviceStreamingProc.kill();
|
|
253
|
+
this.ensureFeatureEnabled(ADB_SCREEN_STREAMING_FEATURE);
|
|
254
|
+
const { width, height, bitRate, host = DEFAULT_HOST, port = DEFAULT_PORT, pathname, tcpPort = DEFAULT_PORT + 1, quality = DEFAULT_QUALITY, considerRotation = false, logPipelineDetails = false, } = options;
|
|
255
|
+
if (lodash_1.default.isUndefined(this._screenStreamingProps)) {
|
|
256
|
+
await verifyStreamingRequirements(this.adb);
|
|
237
257
|
}
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
try {
|
|
243
|
-
await new _bluebird.default((resolve, reject) => {
|
|
244
|
-
mjpegSocket = _net.default.createConnection(tcpPort, TCP_HOST, () => {
|
|
245
|
-
this.log.info(`Successfully connected to MJPEG stream at tcp://${TCP_HOST}:${tcpPort}`);
|
|
246
|
-
mjpegServer = _http.default.createServer((req, res) => {
|
|
247
|
-
const remoteAddress = extractRemoteAddress(req);
|
|
248
|
-
const currentPathname = _url.default.parse(req.url).pathname;
|
|
249
|
-
this.log.info(`Got an incoming screen bradcasting request from ${remoteAddress} ` + `(${req.headers['user-agent'] || 'User Agent unknown'}) at ${currentPathname}`);
|
|
250
|
-
if (pathname && currentPathname !== pathname) {
|
|
251
|
-
this.log.info('Rejecting the broadcast request since it does not match the given pathname');
|
|
252
|
-
res.writeHead(404, {
|
|
253
|
-
Connection: 'close',
|
|
254
|
-
'Content-Type': 'text/plain; charset=utf-8'
|
|
255
|
-
});
|
|
256
|
-
res.write(`'${currentPathname}' did not match any known endpoints`);
|
|
257
|
-
res.end();
|
|
258
|
-
return;
|
|
259
|
-
}
|
|
260
|
-
this.log.info('Starting MJPEG broadcast');
|
|
261
|
-
res.writeHead(200, {
|
|
262
|
-
'Cache-Control': 'no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0',
|
|
263
|
-
Pragma: 'no-cache',
|
|
264
|
-
Connection: 'close',
|
|
265
|
-
'Content-Type': `multipart/x-mixed-replace; boundary=${BOUNDARY_STRING}`
|
|
266
|
-
});
|
|
267
|
-
mjpegSocket.pipe(res);
|
|
268
|
-
});
|
|
269
|
-
mjpegServer.on('error', e => {
|
|
270
|
-
this.log.warn(e);
|
|
271
|
-
reject(e);
|
|
272
|
-
});
|
|
273
|
-
mjpegServer.on('close', () => {
|
|
274
|
-
this.log.debug(`MJPEG server at http://${host}:${port} has been closed`);
|
|
275
|
-
});
|
|
276
|
-
mjpegServer.on('listening', () => {
|
|
277
|
-
this.log.info(`Successfully started MJPEG server at http://${host}:${port}`);
|
|
278
|
-
resolve();
|
|
279
|
-
});
|
|
280
|
-
mjpegServer.listen(port, host);
|
|
281
|
-
});
|
|
282
|
-
mjpegSocket.on('error', e => {
|
|
283
|
-
this.log.error(e);
|
|
284
|
-
reject(e);
|
|
285
|
-
});
|
|
286
|
-
}).timeout(STREAMING_STARTUP_TIMEOUT_MS, `Cannot connect to the streaming server within ${STREAMING_STARTUP_TIMEOUT_MS}ms`);
|
|
287
|
-
} catch (e) {
|
|
288
|
-
if (deviceStreamingProc.kill(0)) {
|
|
289
|
-
deviceStreamingProc.kill();
|
|
258
|
+
if (!lodash_1.default.isEmpty(this._screenStreamingProps)) {
|
|
259
|
+
this.log.info(`The screen streaming session is already running. ` +
|
|
260
|
+
`Stop it first in order to start a new one.`);
|
|
261
|
+
return;
|
|
290
262
|
}
|
|
291
|
-
if (
|
|
292
|
-
|
|
263
|
+
if ((await (0, portscanner_1.checkPortStatus)(port, host)) === 'open') {
|
|
264
|
+
this.log.info(`The port #${port} at ${host} is busy. ` +
|
|
265
|
+
`Assuming the screen streaming is already running`);
|
|
266
|
+
return;
|
|
293
267
|
}
|
|
294
|
-
if (
|
|
295
|
-
|
|
268
|
+
if ((await (0, portscanner_1.checkPortStatus)(tcpPort, TCP_HOST)) === 'open') {
|
|
269
|
+
this.log.errorAndThrow(`The port #${tcpPort} at ${TCP_HOST} is busy. ` +
|
|
270
|
+
`Make sure there are no leftovers from previous sessions.`);
|
|
296
271
|
}
|
|
297
|
-
|
|
298
|
-
|
|
272
|
+
this._screenStreamingProps = null;
|
|
273
|
+
const deviceInfo = await getDeviceInfo(this.adb, this.log);
|
|
274
|
+
const deviceStreamingProc = await initDeviceStreamingProc(this.adb, this.log, deviceInfo, {
|
|
275
|
+
width,
|
|
276
|
+
height,
|
|
277
|
+
bitRate,
|
|
278
|
+
});
|
|
279
|
+
let gstreamerPipeline;
|
|
280
|
+
try {
|
|
281
|
+
gstreamerPipeline = await initGstreamerPipeline(deviceStreamingProc, deviceInfo, this.log, {
|
|
282
|
+
width,
|
|
283
|
+
height,
|
|
284
|
+
quality,
|
|
285
|
+
tcpPort,
|
|
286
|
+
considerRotation,
|
|
287
|
+
logPipelineDetails,
|
|
288
|
+
});
|
|
299
289
|
}
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
mjpegSocket,
|
|
306
|
-
mjpegServer
|
|
307
|
-
};
|
|
308
|
-
};
|
|
309
|
-
commands.mobileStopScreenStreaming = async function mobileStopScreenStreaming() {
|
|
310
|
-
if (_lodash.default.isEmpty(this._screenStreamingProps)) {
|
|
311
|
-
if (!_lodash.default.isUndefined(this._screenStreamingProps)) {
|
|
312
|
-
this.log.debug(`Screen streaming is not running. There is nothing to stop`);
|
|
290
|
+
catch (e) {
|
|
291
|
+
if (deviceStreamingProc.kill(0)) {
|
|
292
|
+
deviceStreamingProc.kill();
|
|
293
|
+
}
|
|
294
|
+
throw e;
|
|
313
295
|
}
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
296
|
+
let mjpegSocket;
|
|
297
|
+
let mjpegServer;
|
|
298
|
+
try {
|
|
299
|
+
await new bluebird_1.default((resolve, reject) => {
|
|
300
|
+
mjpegSocket = net_1.default.createConnection(tcpPort, TCP_HOST, () => {
|
|
301
|
+
this.log.info(`Successfully connected to MJPEG stream at tcp://${TCP_HOST}:${tcpPort}`);
|
|
302
|
+
mjpegServer = http_1.default.createServer((req, res) => {
|
|
303
|
+
const remoteAddress = extractRemoteAddress(req);
|
|
304
|
+
const currentPathname = url_1.default.parse(req.url).pathname;
|
|
305
|
+
this.log.info(`Got an incoming screen bradcasting request from ${remoteAddress} ` +
|
|
306
|
+
`(${req.headers['user-agent'] || 'User Agent unknown'}) at ${currentPathname}`);
|
|
307
|
+
if (pathname && currentPathname !== pathname) {
|
|
308
|
+
this.log.info('Rejecting the broadcast request since it does not match the given pathname');
|
|
309
|
+
res.writeHead(404, {
|
|
310
|
+
Connection: 'close',
|
|
311
|
+
'Content-Type': 'text/plain; charset=utf-8',
|
|
312
|
+
});
|
|
313
|
+
res.write(`'${currentPathname}' did not match any known endpoints`);
|
|
314
|
+
res.end();
|
|
315
|
+
return;
|
|
316
|
+
}
|
|
317
|
+
this.log.info('Starting MJPEG broadcast');
|
|
318
|
+
res.writeHead(200, {
|
|
319
|
+
'Cache-Control': 'no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0',
|
|
320
|
+
Pragma: 'no-cache',
|
|
321
|
+
Connection: 'close',
|
|
322
|
+
'Content-Type': `multipart/x-mixed-replace; boundary=${BOUNDARY_STRING}`
|
|
323
|
+
});
|
|
324
|
+
mjpegSocket.pipe(res);
|
|
325
|
+
});
|
|
326
|
+
mjpegServer.on('error', (e) => {
|
|
327
|
+
this.log.warn(e);
|
|
328
|
+
reject(e);
|
|
329
|
+
});
|
|
330
|
+
mjpegServer.on('close', () => {
|
|
331
|
+
this.log.debug(`MJPEG server at http://${host}:${port} has been closed`);
|
|
332
|
+
});
|
|
333
|
+
mjpegServer.on('listening', () => {
|
|
334
|
+
this.log.info(`Successfully started MJPEG server at http://${host}:${port}`);
|
|
335
|
+
resolve();
|
|
336
|
+
});
|
|
337
|
+
mjpegServer.listen(port, host);
|
|
338
|
+
});
|
|
339
|
+
mjpegSocket.on('error', (e) => {
|
|
340
|
+
this.log.error(e);
|
|
341
|
+
reject(e);
|
|
342
|
+
});
|
|
343
|
+
}).timeout(STREAMING_STARTUP_TIMEOUT_MS, `Cannot connect to the streaming server within ${STREAMING_STARTUP_TIMEOUT_MS}ms`);
|
|
326
344
|
}
|
|
327
|
-
|
|
328
|
-
|
|
345
|
+
catch (e) {
|
|
346
|
+
if (deviceStreamingProc.kill(0)) {
|
|
347
|
+
deviceStreamingProc.kill();
|
|
348
|
+
}
|
|
349
|
+
if (gstreamerPipeline.isRunning) {
|
|
350
|
+
await gstreamerPipeline.stop();
|
|
351
|
+
}
|
|
352
|
+
if (mjpegSocket) {
|
|
353
|
+
mjpegSocket.destroy();
|
|
354
|
+
}
|
|
355
|
+
if (mjpegServer && mjpegServer.listening) {
|
|
356
|
+
mjpegServer.close();
|
|
357
|
+
}
|
|
358
|
+
throw e;
|
|
329
359
|
}
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
360
|
+
this._screenStreamingProps = {
|
|
361
|
+
deviceStreamingProc,
|
|
362
|
+
gstreamerPipeline,
|
|
363
|
+
mjpegSocket,
|
|
364
|
+
mjpegServer,
|
|
365
|
+
};
|
|
366
|
+
};
|
|
367
|
+
/**
|
|
368
|
+
* Stop screen streaming.
|
|
369
|
+
* If no screen streaming server has been started then nothing is done.
|
|
370
|
+
*/
|
|
371
|
+
commands.mobileStopScreenStreaming = async function mobileStopScreenStreaming( /* options = {} */) {
|
|
372
|
+
if (lodash_1.default.isEmpty(this._screenStreamingProps)) {
|
|
373
|
+
if (!lodash_1.default.isUndefined(this._screenStreamingProps)) {
|
|
374
|
+
this.log.debug(`Screen streaming is not running. There is nothing to stop`);
|
|
339
375
|
}
|
|
340
|
-
|
|
376
|
+
return;
|
|
377
|
+
}
|
|
378
|
+
const { deviceStreamingProc, gstreamerPipeline, mjpegSocket, mjpegServer, } = this._screenStreamingProps;
|
|
379
|
+
try {
|
|
380
|
+
mjpegSocket.end();
|
|
381
|
+
if (mjpegServer.listening) {
|
|
382
|
+
mjpegServer.close();
|
|
383
|
+
}
|
|
384
|
+
if (deviceStreamingProc.kill(0)) {
|
|
385
|
+
deviceStreamingProc.kill('SIGINT');
|
|
386
|
+
}
|
|
387
|
+
if (gstreamerPipeline.isRunning) {
|
|
388
|
+
try {
|
|
389
|
+
await gstreamerPipeline.stop('SIGINT');
|
|
390
|
+
}
|
|
391
|
+
catch (e) {
|
|
392
|
+
this.log.warn(e);
|
|
393
|
+
try {
|
|
394
|
+
await gstreamerPipeline.stop('SIGKILL');
|
|
395
|
+
}
|
|
396
|
+
catch (e1) {
|
|
397
|
+
this.log.error(e1);
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
}
|
|
401
|
+
this.log.info(`Successfully terminated the screen streaming MJPEG server`);
|
|
402
|
+
}
|
|
403
|
+
finally {
|
|
404
|
+
this._screenStreamingProps = null;
|
|
341
405
|
}
|
|
342
|
-
this.log.info(`Successfully terminated the screen streaming MJPEG server`);
|
|
343
|
-
} finally {
|
|
344
|
-
this._screenStreamingProps = null;
|
|
345
|
-
}
|
|
346
406
|
};
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","_interopRequireDefault","require","_support","_teen_process","_portscanner","_http","_net","_bluebird","_asyncbox","_child_process","_url","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","_default","exports","default"],"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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"}
|
|
407
|
+
exports.default = commands;
|
|
408
|
+
//# sourceMappingURL=streamscreen.js.map
|