appium-android-driver 5.4.3 → 5.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- 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 +32 -91
- 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/lib/commands/general.js +34 -15
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"log.js","names":["GET_SERVER_LOGS_FEATURE","commands","helpers","extensions","WEBSOCKET_ENDPOINT","sessionId","DEFAULT_WS_PATHNAME_PREFIX","toLogRecord","timestamp","level","message","supportedLogTypes","logcat","description","getter","self","adb","getLogcatLogs","bugreport","output","Date","now","split","os","EOL","map","x","server","ensureFeatureEnabled","log","unwrap","record","_","isEmpty","prefix","mobileStartLogsBroadcast","pathname","getWebSocketHandlers","debug","info","wss","WebSocket","Server","noServer","on","ws","req","remoteIp","headers","connection","remoteAddress","_logcatWebsocketListener","logRecord","readyState","OPEN","send","setLogcatListener","code","reason","removeLogcatListener","ign","closeMsg","toString","addWebSocketHandler","mobileStopLogsBroadcast","removeWebSocketHandler","getLogTypes","nativeLogTypes","BaseDriver","prototype","call","isWebContext","webLogTypes","chromedriver","jwproxy","command","getLog","logType","keys","includes","type","Object","assign"],"sources":["../../../lib/commands/log.js"],"sourcesContent":["import log from '../logger';\nimport os from 'os';\nimport _ from 'lodash';\nimport WebSocket from 'ws';\nimport { DEFAULT_WS_PATHNAME_PREFIX, BaseDriver } from 'appium/driver';\n\nconst GET_SERVER_LOGS_FEATURE = 'get_server_logs';\n\nlet commands = {}, helpers = {}, extensions = {};\n\nconst WEBSOCKET_ENDPOINT = (sessionId) => `${DEFAULT_WS_PATHNAME_PREFIX}/session/${sessionId}/appium/device/logcat`;\n\n// https://github.com/SeleniumHQ/selenium/blob/0d425676b3c9df261dd641917f867d4d5ce7774d/java/client/src/org/openqa/selenium/logging/LogEntry.java\nfunction toLogRecord (timestamp, level, message) {\n return {\n timestamp,\n level,\n message,\n };\n}\n\nextensions.supportedLogTypes = {\n logcat: {\n description: 'Logs for Android applications on real device and emulators via ADB',\n getter: async (self) => await self.adb.getLogcatLogs(),\n },\n bugreport: {\n description: `'adb bugreport' output for advanced issues diagnostic`,\n getter: async (self) => {\n const output = await self.adb.bugreport();\n const timestamp = Date.now();\n return output.split(os.EOL)\n .map((x) => toLogRecord(timestamp, 'ALL', x));\n },\n },\n server: {\n description: 'Appium server logs',\n getter: (self) => {\n self.ensureFeatureEnabled(GET_SERVER_LOGS_FEATURE);\n const timestamp = Date.now();\n return log.unwrap().record\n .map((x) => toLogRecord(timestamp,\n 'ALL',\n _.isEmpty(x.prefix) ? x.message : `[${x.prefix}] ${x.message}`)\n );\n },\n },\n};\n\n/**\n * Starts Android logcat broadcast websocket on the same host and port\n * where Appium server is running at `/ws/session/:sessionId:/appium/logcat` endpoint. The method\n * will return immediately if the web socket is already listening.\n *\n * Each connected websocket listener will receive logcat log lines\n * as soon as they are visible to Appium.\n */\ncommands.mobileStartLogsBroadcast = async function mobileStartLogsBroadcast () {\n const pathname = WEBSOCKET_ENDPOINT(this.sessionId);\n if (!_.isEmpty(await this.server.getWebSocketHandlers(pathname))) {\n log.debug(`The logcat broadcasting web socket server is already listening at ${pathname}`);\n return;\n }\n\n log.info(`Assigning logcat broadcasting web socket server to ${pathname}`);\n // https://github.com/websockets/ws/blob/master/doc/ws.md\n const wss = new WebSocket.Server({\n noServer: true,\n });\n wss.on('connection', (ws, req) => {\n if (req) {\n const remoteIp = _.isEmpty(req.headers['x-forwarded-for'])\n ? req.connection?.remoteAddress\n : req.headers['x-forwarded-for'];\n log.debug(`Established a new logcat listener web socket connection from ${remoteIp}`);\n } else {\n log.debug('Established a new logcat listener web socket connection');\n }\n\n if (_.isEmpty(this._logcatWebsocketListener)) {\n this._logcatWebsocketListener = (logRecord) => {\n if (ws?.readyState === WebSocket.OPEN) {\n ws.send(logRecord.message);\n }\n };\n }\n this.adb.setLogcatListener(this._logcatWebsocketListener);\n\n ws.on('close', (code, reason) => {\n if (!_.isEmpty(this._logcatWebsocketListener)) {\n try {\n this.adb.removeLogcatListener(this._logcatWebsocketListener);\n } catch (ign) {}\n this._logcatWebsocketListener = null;\n }\n\n let closeMsg = 'Logcat listener web socket is closed.';\n if (!_.isEmpty(code)) {\n closeMsg += ` Code: ${code}.`;\n }\n if (!_.isEmpty(reason)) {\n closeMsg += ` Reason: ${reason.toString()}.`;\n }\n log.debug(closeMsg);\n });\n });\n await this.server.addWebSocketHandler(pathname, wss);\n};\n\n/**\n * Stops the previously started logcat broadcasting wesocket server.\n * This method will return immediately if no server is running.\n */\ncommands.mobileStopLogsBroadcast = async function mobileStopLogsBroadcast () {\n const pathname = WEBSOCKET_ENDPOINT(this.sessionId);\n if (_.isEmpty(await this.server.getWebSocketHandlers(pathname))) {\n return;\n }\n\n log.debug('Stopping the logcat broadcasting web socket server');\n await this.server.removeWebSocketHandler(pathname);\n};\n\ncommands.getLogTypes = async function getLogTypes () {\n const nativeLogTypes = await BaseDriver.prototype.getLogTypes.call(this);\n if (this.isWebContext()) {\n const webLogTypes = await this.chromedriver.jwproxy.command('/log/types', 'GET');\n return [...nativeLogTypes, ...webLogTypes];\n }\n return nativeLogTypes;\n};\n\ncommands.getLog = async function getLog (logType) {\n if (this.isWebContext() && !_.keys(this.supportedLogTypes).includes(logType)) {\n return await this.chromedriver.jwproxy.command('/log', 'POST', {type: logType});\n }\n return await BaseDriver.prototype.getLog.call(this, logType);\n};\n\nObject.assign(extensions, commands, helpers);\nexport { commands, helpers };\nexport default extensions;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"log.js","names":["GET_SERVER_LOGS_FEATURE","commands","helpers","extensions","WEBSOCKET_ENDPOINT","sessionId","DEFAULT_WS_PATHNAME_PREFIX","toLogRecord","timestamp","level","message","supportedLogTypes","logcat","description","getter","self","adb","getLogcatLogs","bugreport","output","Date","now","split","os","EOL","map","x","server","ensureFeatureEnabled","log","unwrap","record","_","isEmpty","prefix","mobileStartLogsBroadcast","pathname","getWebSocketHandlers","debug","info","wss","WebSocket","Server","noServer","on","ws","req","remoteIp","headers","connection","remoteAddress","_logcatWebsocketListener","logRecord","readyState","OPEN","send","setLogcatListener","code","reason","removeLogcatListener","ign","closeMsg","toString","addWebSocketHandler","mobileStopLogsBroadcast","removeWebSocketHandler","getLogTypes","nativeLogTypes","BaseDriver","prototype","call","isWebContext","webLogTypes","chromedriver","jwproxy","command","getLog","logType","keys","includes","type","Object","assign"],"sources":["../../../lib/commands/log.js"],"sourcesContent":["import log from '../logger';\nimport os from 'os';\nimport _ from 'lodash';\nimport WebSocket from 'ws';\nimport { DEFAULT_WS_PATHNAME_PREFIX, BaseDriver } from 'appium/driver';\n\nconst GET_SERVER_LOGS_FEATURE = 'get_server_logs';\n\nlet commands = {}, helpers = {}, extensions = {};\n\nconst WEBSOCKET_ENDPOINT = (sessionId) => `${DEFAULT_WS_PATHNAME_PREFIX}/session/${sessionId}/appium/device/logcat`;\n\n// https://github.com/SeleniumHQ/selenium/blob/0d425676b3c9df261dd641917f867d4d5ce7774d/java/client/src/org/openqa/selenium/logging/LogEntry.java\nfunction toLogRecord (timestamp, level, message) {\n return {\n timestamp,\n level,\n message,\n };\n}\n\nextensions.supportedLogTypes = {\n logcat: {\n description: 'Logs for Android applications on real device and emulators via ADB',\n getter: async (self) => await self.adb.getLogcatLogs(),\n },\n bugreport: {\n description: `'adb bugreport' output for advanced issues diagnostic`,\n getter: async (self) => {\n const output = await self.adb.bugreport();\n const timestamp = Date.now();\n return output.split(os.EOL)\n .map((x) => toLogRecord(timestamp, 'ALL', x));\n },\n },\n server: {\n description: 'Appium server logs',\n getter: (self) => {\n self.ensureFeatureEnabled(GET_SERVER_LOGS_FEATURE);\n const timestamp = Date.now();\n return log.unwrap().record\n .map((x) => toLogRecord(timestamp,\n 'ALL',\n _.isEmpty(x.prefix) ? x.message : `[${x.prefix}] ${x.message}`)\n );\n },\n },\n};\n\n/**\n * Starts Android logcat broadcast websocket on the same host and port\n * where Appium server is running at `/ws/session/:sessionId:/appium/logcat` endpoint. The method\n * will return immediately if the web socket is already listening.\n *\n * Each connected websocket listener will receive logcat log lines\n * as soon as they are visible to Appium.\n */\ncommands.mobileStartLogsBroadcast = async function mobileStartLogsBroadcast () {\n const pathname = WEBSOCKET_ENDPOINT(this.sessionId);\n if (!_.isEmpty(await this.server.getWebSocketHandlers(pathname))) {\n log.debug(`The logcat broadcasting web socket server is already listening at ${pathname}`);\n return;\n }\n\n log.info(`Assigning logcat broadcasting web socket server to ${pathname}`);\n // https://github.com/websockets/ws/blob/master/doc/ws.md\n const wss = new WebSocket.Server({\n noServer: true,\n });\n wss.on('connection', (ws, req) => {\n if (req) {\n const remoteIp = _.isEmpty(req.headers['x-forwarded-for'])\n ? req.connection?.remoteAddress\n : req.headers['x-forwarded-for'];\n log.debug(`Established a new logcat listener web socket connection from ${remoteIp}`);\n } else {\n log.debug('Established a new logcat listener web socket connection');\n }\n\n if (_.isEmpty(this._logcatWebsocketListener)) {\n this._logcatWebsocketListener = (logRecord) => {\n if (ws?.readyState === WebSocket.OPEN) {\n ws.send(logRecord.message);\n }\n };\n }\n this.adb.setLogcatListener(this._logcatWebsocketListener);\n\n ws.on('close', (code, reason) => {\n if (!_.isEmpty(this._logcatWebsocketListener)) {\n try {\n this.adb.removeLogcatListener(this._logcatWebsocketListener);\n } catch (ign) {}\n this._logcatWebsocketListener = null;\n }\n\n let closeMsg = 'Logcat listener web socket is closed.';\n if (!_.isEmpty(code)) {\n closeMsg += ` Code: ${code}.`;\n }\n if (!_.isEmpty(reason)) {\n closeMsg += ` Reason: ${reason.toString()}.`;\n }\n log.debug(closeMsg);\n });\n });\n await this.server.addWebSocketHandler(pathname, wss);\n};\n\n/**\n * Stops the previously started logcat broadcasting wesocket server.\n * This method will return immediately if no server is running.\n */\ncommands.mobileStopLogsBroadcast = async function mobileStopLogsBroadcast () {\n const pathname = WEBSOCKET_ENDPOINT(this.sessionId);\n if (_.isEmpty(await this.server.getWebSocketHandlers(pathname))) {\n return;\n }\n\n log.debug('Stopping the logcat broadcasting web socket server');\n await this.server.removeWebSocketHandler(pathname);\n};\n\ncommands.getLogTypes = async function getLogTypes () {\n const nativeLogTypes = await BaseDriver.prototype.getLogTypes.call(this);\n if (this.isWebContext()) {\n const webLogTypes = await this.chromedriver.jwproxy.command('/log/types', 'GET');\n return [...nativeLogTypes, ...webLogTypes];\n }\n return nativeLogTypes;\n};\n\ncommands.getLog = async function getLog (logType) {\n if (this.isWebContext() && !_.keys(this.supportedLogTypes).includes(logType)) {\n return await this.chromedriver.jwproxy.command('/log', 'POST', {type: logType});\n }\n return await BaseDriver.prototype.getLog.call(this, logType);\n};\n\nObject.assign(extensions, commands, helpers);\nexport { commands, helpers };\nexport default extensions;\n"],"mappings":"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"}
|
|
@@ -1,29 +1,20 @@
|
|
|
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 _asyncbox = require("asyncbox");
|
|
15
|
-
|
|
16
11
|
var _support = require("appium/support");
|
|
17
|
-
|
|
18
12
|
var _path = _interopRequireDefault(require("path"));
|
|
19
|
-
|
|
20
13
|
var _bluebird = _interopRequireDefault(require("bluebird"));
|
|
21
|
-
|
|
22
14
|
var _androidHelpers = require("../android-helpers");
|
|
23
|
-
|
|
24
15
|
var _moment = _interopRequireDefault(require("moment"));
|
|
25
|
-
|
|
26
16
|
const commands = {};
|
|
17
|
+
|
|
27
18
|
exports.commands = commands;
|
|
28
19
|
const DEFAULT_EXT = '.mp4';
|
|
29
20
|
const RECORDING_STARTUP_TIMEOUT_MS = 3 * 1000;
|
|
@@ -35,12 +26,10 @@ const RECORDING_ACTION_START = `${_androidHelpers.SETTINGS_HELPER_PKG_ID}.record
|
|
|
35
26
|
const RECORDING_ACTION_STOP = `${_androidHelpers.SETTINGS_HELPER_PKG_ID}.recording.ACTION_STOP`;
|
|
36
27
|
const RECORDINGS_ROOT = `/storage/emulated/0/Android/data/${_androidHelpers.SETTINGS_HELPER_PKG_ID}/files`;
|
|
37
28
|
const DEFAULT_FILENAME_FORMAT = 'YYYY-MM-DDTHH-mm-ss';
|
|
38
|
-
|
|
39
29
|
async function uploadRecordedMedia(localFile, remotePath = null, uploadOptions = {}) {
|
|
40
30
|
if (_lodash.default.isEmpty(remotePath)) {
|
|
41
31
|
return (await _support.util.toInMemoryBase64(localFile)).toString();
|
|
42
32
|
}
|
|
43
|
-
|
|
44
33
|
const {
|
|
45
34
|
user,
|
|
46
35
|
pass,
|
|
@@ -55,45 +44,36 @@ async function uploadRecordedMedia(localFile, remotePath = null, uploadOptions =
|
|
|
55
44
|
fileFieldName,
|
|
56
45
|
formFields
|
|
57
46
|
};
|
|
58
|
-
|
|
59
47
|
if (user && pass) {
|
|
60
48
|
options.auth = {
|
|
61
49
|
user,
|
|
62
50
|
pass
|
|
63
51
|
};
|
|
64
52
|
}
|
|
65
|
-
|
|
66
53
|
await _support.net.uploadFile(localFile, remotePath, options);
|
|
67
54
|
return '';
|
|
68
55
|
}
|
|
69
|
-
|
|
70
56
|
function adjustMediaExtension(name) {
|
|
71
57
|
return _lodash.default.toLower(name).endsWith(DEFAULT_EXT) ? name : `${name}${DEFAULT_EXT}`;
|
|
72
58
|
}
|
|
73
|
-
|
|
74
59
|
async function verifyMediaProjectionRecordingIsSupported(adb) {
|
|
75
60
|
const apiLevel = await adb.getApiLevel();
|
|
76
|
-
|
|
77
61
|
if (apiLevel < MIN_API_LEVEL) {
|
|
78
62
|
throw new Error(`Media projection-based recording is not available on API Level ${apiLevel}. ` + `Minimum required API Level is ${MIN_API_LEVEL}.`);
|
|
79
63
|
}
|
|
80
64
|
}
|
|
81
|
-
|
|
82
65
|
class MediaProjectionRecorder {
|
|
83
66
|
constructor(adb) {
|
|
84
67
|
this.adb = adb;
|
|
85
68
|
}
|
|
86
|
-
|
|
87
69
|
async isRunning() {
|
|
88
70
|
const stdout = await this.adb.shell(['dumpsys', 'activity', 'services', RECORDING_SERVICE_NAME]);
|
|
89
71
|
return stdout.includes(RECORDING_SERVICE_NAME);
|
|
90
72
|
}
|
|
91
|
-
|
|
92
73
|
async start(opts = {}) {
|
|
93
74
|
if (await this.isRunning()) {
|
|
94
75
|
return false;
|
|
95
76
|
}
|
|
96
|
-
|
|
97
77
|
await this.cleanup();
|
|
98
78
|
const {
|
|
99
79
|
filename,
|
|
@@ -102,60 +82,46 @@ class MediaProjectionRecorder {
|
|
|
102
82
|
resolution
|
|
103
83
|
} = opts;
|
|
104
84
|
const args = ['am', 'start', '-n', RECORDING_ACTIVITY_NAME, '-a', RECORDING_ACTION_START];
|
|
105
|
-
|
|
106
85
|
if (filename) {
|
|
107
86
|
args.push('--es', 'filename', filename);
|
|
108
87
|
}
|
|
109
|
-
|
|
110
88
|
if (maxDurationSec) {
|
|
111
89
|
args.push('--es', 'max_duration_sec', `${maxDurationSec}`);
|
|
112
90
|
}
|
|
113
|
-
|
|
114
91
|
if (priority) {
|
|
115
92
|
args.push('--es', 'priority', priority);
|
|
116
93
|
}
|
|
117
|
-
|
|
118
94
|
if (resolution) {
|
|
119
95
|
args.push('--es', 'resolution', resolution);
|
|
120
96
|
}
|
|
121
|
-
|
|
122
97
|
await this.adb.shell(args);
|
|
123
98
|
await new _bluebird.default((resolve, reject) => {
|
|
124
99
|
setTimeout(async () => {
|
|
125
100
|
if (!(await this.isRunning())) {
|
|
126
101
|
return reject(new Error(`The media projection recording is not running after ${RECORDING_STARTUP_TIMEOUT_MS}ms. ` + `Please check the logcat output for more details.`));
|
|
127
102
|
}
|
|
128
|
-
|
|
129
103
|
resolve();
|
|
130
104
|
}, RECORDING_STARTUP_TIMEOUT_MS);
|
|
131
105
|
});
|
|
132
106
|
return true;
|
|
133
107
|
}
|
|
134
|
-
|
|
135
108
|
async cleanup() {
|
|
136
109
|
await this.adb.shell([`rm -f ${RECORDINGS_ROOT}/*`]);
|
|
137
110
|
}
|
|
138
|
-
|
|
139
111
|
async pullRecent() {
|
|
140
112
|
const recordings = await this.adb.ls(RECORDINGS_ROOT, ['-tr']);
|
|
141
|
-
|
|
142
113
|
if (_lodash.default.isEmpty(recordings)) {
|
|
143
114
|
return null;
|
|
144
115
|
}
|
|
145
|
-
|
|
146
116
|
const dstPath = _path.default.join(await _support.tempDir.openDir(), recordings[0]);
|
|
147
|
-
|
|
148
117
|
await this.adb.pull(`${RECORDINGS_ROOT}/${recordings[0]}`, dstPath);
|
|
149
118
|
return dstPath;
|
|
150
119
|
}
|
|
151
|
-
|
|
152
120
|
async stop() {
|
|
153
121
|
if (!(await this.isRunning())) {
|
|
154
122
|
return false;
|
|
155
123
|
}
|
|
156
|
-
|
|
157
124
|
await this.adb.shell(['am', 'start', '-n', RECORDING_ACTIVITY_NAME, '-a', RECORDING_ACTION_STOP]);
|
|
158
|
-
|
|
159
125
|
try {
|
|
160
126
|
await (0, _asyncbox.waitForCondition)(async () => !(await this.isRunning()), {
|
|
161
127
|
waitMs: RECORDING_STOP_TIMEOUT_MS,
|
|
@@ -164,10 +130,8 @@ class MediaProjectionRecorder {
|
|
|
164
130
|
} catch (e) {
|
|
165
131
|
throw new Error(`The attempt to stop the current media projection recording timed out after ` + `${RECORDING_STOP_TIMEOUT_MS}ms`);
|
|
166
132
|
}
|
|
167
|
-
|
|
168
133
|
return true;
|
|
169
134
|
}
|
|
170
|
-
|
|
171
135
|
}
|
|
172
136
|
|
|
173
137
|
commands.mobileStartMediaProjectionRecording = async function mobileStartMediaProjectionRecording(options = {}) {
|
|
@@ -186,13 +150,11 @@ commands.mobileStartMediaProjectionRecording = async function mobileStartMediaPr
|
|
|
186
150
|
maxDurationSec,
|
|
187
151
|
filename: fname
|
|
188
152
|
});
|
|
189
|
-
|
|
190
153
|
if (didStart) {
|
|
191
154
|
this.log.info(`A new media projection recording '${fname}' has been successfully started`);
|
|
192
155
|
} else {
|
|
193
156
|
this.log.info('Another media projection recording is already in progress. There is nothing to start');
|
|
194
157
|
}
|
|
195
|
-
|
|
196
158
|
return didStart;
|
|
197
159
|
};
|
|
198
160
|
|
|
@@ -205,37 +167,30 @@ commands.mobileIsMediaProjectionRecordingRunning = async function mobileIsMediaP
|
|
|
205
167
|
commands.mobileStopMediaProjectionRecording = async function mobileStopMediaProjectionRecording(options = {}) {
|
|
206
168
|
await verifyMediaProjectionRecordingIsSupported(this.adb);
|
|
207
169
|
const recorder = new MediaProjectionRecorder(this.adb);
|
|
208
|
-
|
|
209
170
|
if (await recorder.stop()) {
|
|
210
171
|
this.log.info('Successfully stopped a media projection recording. Pulling the recorded media');
|
|
211
172
|
} else {
|
|
212
173
|
this.log.info('Media projection recording is not running. There is nothing to stop');
|
|
213
174
|
}
|
|
214
|
-
|
|
215
175
|
const recentRecordingPath = await recorder.pullRecent();
|
|
216
|
-
|
|
217
176
|
if (!recentRecordingPath) {
|
|
218
177
|
throw new Error(`No recent media projection recording have been found. Did you start any?`);
|
|
219
178
|
}
|
|
220
|
-
|
|
221
179
|
const {
|
|
222
180
|
remotePath
|
|
223
181
|
} = options;
|
|
224
|
-
|
|
225
182
|
if (_lodash.default.isEmpty(remotePath)) {
|
|
226
183
|
const {
|
|
227
184
|
size
|
|
228
185
|
} = await _support.fs.stat(recentRecordingPath);
|
|
229
186
|
this.log.debug(`The size of the resulting media projection recording is ${_support.util.toReadableSizeString(size)}`);
|
|
230
187
|
}
|
|
231
|
-
|
|
232
188
|
try {
|
|
233
189
|
return await uploadRecordedMedia(recentRecordingPath, remotePath, options);
|
|
234
190
|
} finally {
|
|
235
191
|
await _support.fs.rimraf(_path.default.dirname(recentRecordingPath));
|
|
236
192
|
}
|
|
237
193
|
};
|
|
238
|
-
|
|
239
194
|
var _default = commands;
|
|
240
195
|
exports.default = _default;
|
|
241
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjb21tYW5kcyIsIkRFRkFVTFRfRVhUIiwiUkVDT1JESU5HX1NUQVJUVVBfVElNRU9VVF9NUyIsIlJFQ09SRElOR19TVE9QX1RJTUVPVVRfTVMiLCJNSU5fQVBJX0xFVkVMIiwiUkVDT1JESU5HX1NFUlZJQ0VfTkFNRSIsIlNFVFRJTkdTX0hFTFBFUl9QS0dfSUQiLCJSRUNPUkRJTkdfQUNUSVZJVFlfTkFNRSIsIlJFQ09SRElOR19BQ1RJT05fU1RBUlQiLCJSRUNPUkRJTkdfQUNUSU9OX1NUT1AiLCJSRUNPUkRJTkdTX1JPT1QiLCJERUZBVUxUX0ZJTEVOQU1FX0ZPUk1BVCIsInVwbG9hZFJlY29yZGVkTWVkaWEiLCJsb2NhbEZpbGUiLCJyZW1vdGVQYXRoIiwidXBsb2FkT3B0aW9ucyIsIl8iLCJpc0VtcHR5IiwidXRpbCIsInRvSW5NZW1vcnlCYXNlNjQiLCJ0b1N0cmluZyIsInVzZXIiLCJwYXNzIiwibWV0aG9kIiwiaGVhZGVycyIsImZpbGVGaWVsZE5hbWUiLCJmb3JtRmllbGRzIiwib3B0aW9ucyIsImF1dGgiLCJuZXQiLCJ1cGxvYWRGaWxlIiwiYWRqdXN0TWVkaWFFeHRlbnNpb24iLCJuYW1lIiwidG9Mb3dlciIsImVuZHNXaXRoIiwidmVyaWZ5TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nSXNTdXBwb3J0ZWQiLCJhZGIiLCJhcGlMZXZlbCIsImdldEFwaUxldmVsIiwiRXJyb3IiLCJNZWRpYVByb2plY3Rpb25SZWNvcmRlciIsImNvbnN0cnVjdG9yIiwiaXNSdW5uaW5nIiwic3Rkb3V0Iiwic2hlbGwiLCJpbmNsdWRlcyIsInN0YXJ0Iiwib3B0cyIsImNsZWFudXAiLCJmaWxlbmFtZSIsIm1heER1cmF0aW9uU2VjIiwicHJpb3JpdHkiLCJyZXNvbHV0aW9uIiwiYXJncyIsInB1c2giLCJCIiwicmVzb2x2ZSIsInJlamVjdCIsInNldFRpbWVvdXQiLCJwdWxsUmVjZW50IiwicmVjb3JkaW5ncyIsImxzIiwiZHN0UGF0aCIsInBhdGgiLCJqb2luIiwidGVtcERpciIsIm9wZW5EaXIiLCJwdWxsIiwic3RvcCIsIndhaXRGb3JDb25kaXRpb24iLCJ3YWl0TXMiLCJpbnRlcnZhbE1zIiwiZSIsIm1vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nIiwicmVjb3JkZXIiLCJmbmFtZSIsIm1vbWVudCIsImZvcm1hdCIsImRpZFN0YXJ0IiwibG9nIiwiaW5mbyIsIm1vYmlsZUlzTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nUnVubmluZyIsIm1vYmlsZVN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmciLCJyZWNlbnRSZWNvcmRpbmdQYXRoIiwic2l6ZSIsImZzIiwic3RhdCIsImRlYnVnIiwidG9SZWFkYWJsZVNpemVTdHJpbmciLCJyaW1yYWYiLCJkaXJuYW1lIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL2NvbW1hbmRzL21lZGlhLXByb2plY3Rpb24uanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IF8gZnJvbSAnbG9kYXNoJztcbmltcG9ydCB7IHdhaXRGb3JDb25kaXRpb24gfSBmcm9tICdhc3luY2JveCc7XG5pbXBvcnQgeyB1dGlsLCBmcywgbmV0LCB0ZW1wRGlyIH0gZnJvbSAnYXBwaXVtL3N1cHBvcnQnO1xuaW1wb3J0IHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgQiBmcm9tICdibHVlYmlyZCc7XG5pbXBvcnQgeyBTRVRUSU5HU19IRUxQRVJfUEtHX0lEIH0gZnJvbSAnLi4vYW5kcm9pZC1oZWxwZXJzJztcbmltcG9ydCBtb21lbnQgZnJvbSAnbW9tZW50JztcblxuXG5jb25zdCBjb21tYW5kcyA9IHt9O1xuXG4vLyBodHRwczovL2dpdGh1Yi5jb20vYXBwaXVtL2lvLmFwcGl1bS5zZXR0aW5ncyNpbnRlcm5hbC1hdWRpby0tdmlkZW8tcmVjb3JkaW5nXG5jb25zdCBERUZBVUxUX0VYVCA9ICcubXA0JztcbmNvbnN0IFJFQ09SRElOR19TVEFSVFVQX1RJTUVPVVRfTVMgPSAzICogMTAwMDtcbmNvbnN0IFJFQ09SRElOR19TVE9QX1RJTUVPVVRfTVMgPSAzICogMTAwMDtcbmNvbnN0IE1JTl9BUElfTEVWRUwgPSAyOTtcbmNvbnN0IFJFQ09SRElOR19TRVJWSUNFX05BTUUgPSBgJHtTRVRUSU5HU19IRUxQRVJfUEtHX0lEfS8ucmVjb3JkZXIuUmVjb3JkZXJTZXJ2aWNlYDtcbmNvbnN0IFJFQ09SRElOR19BQ1RJVklUWV9OQU1FID0gYCR7U0VUVElOR1NfSEVMUEVSX1BLR19JRH0vaW8uYXBwaXVtLnNldHRpbmdzLlNldHRpbmdzYDtcbmNvbnN0IFJFQ09SRElOR19BQ1RJT05fU1RBUlQgPSBgJHtTRVRUSU5HU19IRUxQRVJfUEtHX0lEfS5yZWNvcmRpbmcuQUNUSU9OX1NUQVJUYDtcbmNvbnN0IFJFQ09SRElOR19BQ1RJT05fU1RPUCA9IGAke1NFVFRJTkdTX0hFTFBFUl9QS0dfSUR9LnJlY29yZGluZy5BQ1RJT05fU1RPUGA7XG5jb25zdCBSRUNPUkRJTkdTX1JPT1QgPSBgL3N0b3JhZ2UvZW11bGF0ZWQvMC9BbmRyb2lkL2RhdGEvJHtTRVRUSU5HU19IRUxQRVJfUEtHX0lEfS9maWxlc2A7XG5jb25zdCBERUZBVUxUX0ZJTEVOQU1FX0ZPUk1BVCA9ICdZWVlZLU1NLUREVEhILW1tLXNzJztcblxuXG5hc3luYyBmdW5jdGlvbiB1cGxvYWRSZWNvcmRlZE1lZGlhIChsb2NhbEZpbGUsIHJlbW90ZVBhdGggPSBudWxsLCB1cGxvYWRPcHRpb25zID0ge30pIHtcbiAgaWYgKF8uaXNFbXB0eShyZW1vdGVQYXRoKSkge1xuICAgIHJldHVybiAoYXdhaXQgdXRpbC50b0luTWVtb3J5QmFzZTY0KGxvY2FsRmlsZSkpLnRvU3RyaW5nKCk7XG4gIH1cblxuICBjb25zdCB7dXNlciwgcGFzcywgbWV0aG9kLCBoZWFkZXJzLCBmaWxlRmllbGROYW1lLCBmb3JtRmllbGRzfSA9IHVwbG9hZE9wdGlvbnM7XG4gIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgbWV0aG9kOiBtZXRob2QgfHwgJ1BVVCcsXG4gICAgaGVhZGVycyxcbiAgICBmaWxlRmllbGROYW1lLFxuICAgIGZvcm1GaWVsZHMsXG4gIH07XG4gIGlmICh1c2VyICYmIHBhc3MpIHtcbiAgICBvcHRpb25zLmF1dGggPSB7dXNlciwgcGFzc307XG4gIH1cbiAgYXdhaXQgbmV0LnVwbG9hZEZpbGUobG9jYWxGaWxlLCByZW1vdGVQYXRoLCBvcHRpb25zKTtcbiAgcmV0dXJuICcnO1xufVxuXG5mdW5jdGlvbiBhZGp1c3RNZWRpYUV4dGVuc2lvbiAobmFtZSkge1xuICByZXR1cm4gXy50b0xvd2VyKG5hbWUpLmVuZHNXaXRoKERFRkFVTFRfRVhUKSA/IG5hbWUgOiBgJHtuYW1lfSR7REVGQVVMVF9FWFR9YDtcbn1cblxuYXN5bmMgZnVuY3Rpb24gdmVyaWZ5TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nSXNTdXBwb3J0ZWQgKGFkYikge1xuICBjb25zdCBhcGlMZXZlbCA9IGF3YWl0IGFkYi5nZXRBcGlMZXZlbCgpO1xuICBpZiAoYXBpTGV2ZWwgPCBNSU5fQVBJX0xFVkVMKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBNZWRpYSBwcm9qZWN0aW9uLWJhc2VkIHJlY29yZGluZyBpcyBub3QgYXZhaWxhYmxlIG9uIEFQSSBMZXZlbCAke2FwaUxldmVsfS4gYCArXG4gICAgICBgTWluaW11bSByZXF1aXJlZCBBUEkgTGV2ZWwgaXMgJHtNSU5fQVBJX0xFVkVMfS5gKTtcbiAgfVxufVxuXG5cbmNsYXNzIE1lZGlhUHJvamVjdGlvblJlY29yZGVyIHtcbiAgY29uc3RydWN0b3IgKGFkYikge1xuICAgIHRoaXMuYWRiID0gYWRiO1xuICB9XG5cbiAgYXN5bmMgaXNSdW5uaW5nICgpIHtcbiAgICBjb25zdCBzdGRvdXQgPSBhd2FpdCB0aGlzLmFkYi5zaGVsbChbXG4gICAgICAnZHVtcHN5cycsICdhY3Rpdml0eScsICdzZXJ2aWNlcycsIFJFQ09SRElOR19TRVJWSUNFX05BTUVcbiAgICBdKTtcbiAgICByZXR1cm4gc3Rkb3V0LmluY2x1ZGVzKFJFQ09SRElOR19TRVJWSUNFX05BTUUpO1xuICB9XG5cbiAgYXN5bmMgc3RhcnQgKG9wdHMgPSB7fSkge1xuICAgIGlmIChhd2FpdCB0aGlzLmlzUnVubmluZygpKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgYXdhaXQgdGhpcy5jbGVhbnVwKCk7XG4gICAgY29uc3Qge1xuICAgICAgZmlsZW5hbWUsXG4gICAgICBtYXhEdXJhdGlvblNlYyxcbiAgICAgIHByaW9yaXR5LFxuICAgICAgcmVzb2x1dGlvbixcbiAgICB9ID0gb3B0cztcbiAgICBjb25zdCBhcmdzID0gW1xuICAgICAgJ2FtJywgJ3N0YXJ0JyxcbiAgICAgICctbicsIFJFQ09SRElOR19BQ1RJVklUWV9OQU1FLFxuICAgICAgJy1hJywgUkVDT1JESU5HX0FDVElPTl9TVEFSVCxcbiAgICBdO1xuICAgIGlmIChmaWxlbmFtZSkge1xuICAgICAgYXJncy5wdXNoKCctLWVzJywgJ2ZpbGVuYW1lJywgZmlsZW5hbWUpO1xuICAgIH1cbiAgICBpZiAobWF4RHVyYXRpb25TZWMpIHtcbiAgICAgIGFyZ3MucHVzaCgnLS1lcycsICdtYXhfZHVyYXRpb25fc2VjJywgYCR7bWF4RHVyYXRpb25TZWN9YCk7XG4gICAgfVxuICAgIGlmIChwcmlvcml0eSkge1xuICAgICAgYXJncy5wdXNoKCctLWVzJywgJ3ByaW9yaXR5JywgcHJpb3JpdHkpO1xuICAgIH1cbiAgICBpZiAocmVzb2x1dGlvbikge1xuICAgICAgYXJncy5wdXNoKCctLWVzJywgJ3Jlc29sdXRpb24nLCByZXNvbHV0aW9uKTtcbiAgICB9XG4gICAgYXdhaXQgdGhpcy5hZGIuc2hlbGwoYXJncyk7XG4gICAgYXdhaXQgbmV3IEIoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgICAgc2V0VGltZW91dChhc3luYyAoKSA9PiB7XG4gICAgICAgIGlmICghYXdhaXQgdGhpcy5pc1J1bm5pbmcoKSkge1xuICAgICAgICAgIHJldHVybiByZWplY3QobmV3IEVycm9yKFxuICAgICAgICAgICAgYFRoZSBtZWRpYSBwcm9qZWN0aW9uIHJlY29yZGluZyBpcyBub3QgcnVubmluZyBhZnRlciAke1JFQ09SRElOR19TVEFSVFVQX1RJTUVPVVRfTVN9bXMuIGAgK1xuICAgICAgICAgICAgYFBsZWFzZSBjaGVjayB0aGUgbG9nY2F0IG91dHB1dCBmb3IgbW9yZSBkZXRhaWxzLmBcbiAgICAgICAgICApKTtcbiAgICAgICAgfVxuICAgICAgICByZXNvbHZlKCk7XG4gICAgICB9LCBSRUNPUkRJTkdfU1RBUlRVUF9USU1FT1VUX01TKTtcbiAgICB9KTtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfVxuXG4gIGFzeW5jIGNsZWFudXAgKCkge1xuICAgIGF3YWl0IHRoaXMuYWRiLnNoZWxsKFtgcm0gLWYgJHtSRUNPUkRJTkdTX1JPT1R9LypgXSk7XG4gIH1cblxuICBhc3luYyBwdWxsUmVjZW50ICgpIHtcbiAgICBjb25zdCByZWNvcmRpbmdzID0gYXdhaXQgdGhpcy5hZGIubHMoUkVDT1JESU5HU19ST09ULCBbJy10ciddKTtcbiAgICBpZiAoXy5pc0VtcHR5KHJlY29yZGluZ3MpKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBjb25zdCBkc3RQYXRoID0gcGF0aC5qb2luKGF3YWl0IHRlbXBEaXIub3BlbkRpcigpLCByZWNvcmRpbmdzWzBdKTtcbiAgICBhd2FpdCB0aGlzLmFkYi5wdWxsKGAke1JFQ09SRElOR1NfUk9PVH0vJHtyZWNvcmRpbmdzWzBdfWAsIGRzdFBhdGgpO1xuICAgIHJldHVybiBkc3RQYXRoO1xuICB9XG5cbiAgYXN5bmMgc3RvcCAoKSB7XG4gICAgaWYgKCFhd2FpdCB0aGlzLmlzUnVubmluZygpKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgYXdhaXQgdGhpcy5hZGIuc2hlbGwoW1xuICAgICAgJ2FtJywgJ3N0YXJ0JyxcbiAgICAgICctbicsIFJFQ09SRElOR19BQ1RJVklUWV9OQU1FLFxuICAgICAgJy1hJywgUkVDT1JESU5HX0FDVElPTl9TVE9QLFxuICAgIF0pO1xuICAgIHRyeSB7XG4gICAgICBhd2FpdCB3YWl0Rm9yQ29uZGl0aW9uKGFzeW5jICgpID0+ICEoYXdhaXQgdGhpcy5pc1J1bm5pbmcoKSksIHtcbiAgICAgICAgd2FpdE1zOiBSRUNPUkRJTkdfU1RPUF9USU1FT1VUX01TLFxuICAgICAgICBpbnRlcnZhbE1zOiA1MDAsXG4gICAgICB9KTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgIGBUaGUgYXR0ZW1wdCB0byBzdG9wIHRoZSBjdXJyZW50IG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nIHRpbWVkIG91dCBhZnRlciBgICtcbiAgICAgICAgYCR7UkVDT1JESU5HX1NUT1BfVElNRU9VVF9NU31tc2BcbiAgICAgICk7XG4gICAgfVxuICAgIHJldHVybiB0cnVlO1xuICB9XG59XG5cblxuLyoqXG4gKiBAdHlwZWRlZiB7T2JqZWN0fSBTdGFydFJlY29yZGluZ09wdGlvbnNcbiAqXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHJlc29sdXRpb24gTWF4aW11bSBzdXBwb3J0ZWQgcmVzb2x1dGlvbiBvbi1kZXZpY2UgKERldGVjdGVkXG4gKiBhdXRvbWF0aWNhbGx5IGJ5IHRoZSBhcHAgaXRzZWxmKSwgd2hpY2ggdXN1YWxseSBlcXVhbHMgdG8gRnVsbCBIRCAxOTIweDEwODAgb24gbW9zdFxuICogcGhvbmVzIGhvd2V2ZXIgeW91IGNhbiBjaGFuZ2UgaXQgdG8gZm9sbG93aW5nIHN1cHBvcnRlZCByZXNvbHV0aW9uc1xuICogYXMgd2VsbDogXCIxOTIweDEwODBcIiwgXCIxMjgweDcyMFwiLCBcIjcyMHg0ODBcIiwgXCIzMjB4MjQwXCIsIFwiMTc2eDE0NFwiLlxuICogQHByb3BlcnR5IHtudW1iZXI/fSBtYXhEdXJhdGlvblNlYyBbOTAwXSBEZWZhdWx0IHZhbHVlOiA5MDAgc2Vjb25kcyB3aGljaCBtZWFuc1xuICogbWF4aW11bSBhbGxvd2VkIGR1cmF0aW9uIGlzIDE1IG1pbnV0ZSwgeW91IGNhbiBpbmNyZWFzZSBpdCBpZiB5b3VyIHRlc3QgdGFrZXNcbiAqIGxvbmdlciB0aGFuIHRoYXQuXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHByaW9yaXR5IFtoaWdoXSBNZWFucyByZWNvcmRpbmcgdGhyZWFkIHByaW9yaXR5IGlzIG1heGltdW1cbiAqIGhvd2V2ZXIgaWYgeW91IGZhY2UgcGVyZm9ybWFuY2UgZHJvcHMgZHVyaW5nIHRlc3Rpbmcgd2l0aCByZWNvcmRpbmcgZW5hYmxlZCwgeW91XG4gKiBjYW4gcmVkdWNlIHJlY29yZGluZyBwcmlvcml0eSB0byBcIm5vcm1hbFwiIG9yIFwibG93XCIuXG4gKiBAcHJvcGVydHkge3N0cmluZz99IGZpbGVuYW1lIFlvdSBjYW4gdHlwZSByZWNvcmRpbmcgdmlkZW8gZmlsZSBuYW1lIGFzIHlvdSB3YW50LFxuICogYnV0IHJlY29yZGluZyBjdXJyZW50bHkgc3VwcG9ydHMgb25seSBcIm1wNFwiIGZvcm1hdCBzbyB5b3VyIGZpbGVuYW1lIG11c3QgZW5kIHdpdGggXCIubXA0XCIuXG4gKiBBbiBpbnZhbGlkIGZpbGUgbmFtZSB3aWxsIGZhaWwgdG8gc3RhcnQgdGhlIHJlY29yZGluZy5cbiAqIElmIG5vdCBwcm92aWRlZCB0aGVuIHRoZSBjdXJyZW50IHRpbWVzdGFtcCB3aWxsIGJlIHVzZWQgYXMgZmlsZSBuYW1lLlxuICovXG5cbi8qKlxuICogUmVjb3JkIHRoZSBkaXNwbGF5IG9mIGEgcmVhbCBkZXZpY2VzIHJ1bm5pbmcgQW5kcm9pZCAxMCAoQVBJIGxldmVsIDI5KSBhbmQgaGlnaGVyLlxuICogVGhlIHNjcmVlbiBhY3Rpdml0eSBpcyByZWNvcmRlZCB0byBhIE1QRUctNCBmaWxlLiBBdWRpbyBpcyBhbHNvIHJlY29yZGVkIGJ5IGRlZmF1bHRcbiAqIChvbmx5IGZvciBhcHBzIHRoYXQgYWxsb3cgaXQgaW4gdGhlaXIgbWFuaWZlc3RzKS5cbiAqIElmIGFub3RoZXIgcmVjb3JkaW5nIGhhcyBiZWVuIGFscmVhZHkgc3RhcnRlZCB0aGVuIHRoZSBjb21tYW5kIHdpbGwgZXhpdCBzaWxlbnRseS5cbiAqIFRoZSBwcmV2aW91c2x5IHJlY29yZGVkIHZpZGVvIGZpbGUgaXMgZGVsZXRlZCB3aGVuIGEgbmV3IHJlY29yZGluZyBzZXNzaW9uIGlzIHN0YXJ0ZWQuXG4gKiBSZWNvcmRpbmcgY29udGludWVzIGl0IGlzIHN0b3BwZWQgZXhwbGljaXRseSBvciB1bnRpbCB0aGUgdGltZW91dCBoYXBwZW5zLlxuICpcbiAqIEBwYXJhbSB7P1N0YXJ0UmVjb3JkaW5nT3B0aW9uc30gb3B0aW9ucyBBdmFpbGFibGUgb3B0aW9ucy5cbiAqIEByZXR1cm5zIHtib29sZWFufSBUcnVlIGlmIGEgbmV3IHJlY29yZGluZyBoYXMgc3VjY2Vzc2Z1bGx5IHN0YXJ0ZWQuXG4gKiBAdGhyb3dzIHtFcnJvcn0gSWYgcmVjb3JkaW5nIGhhcyBmYWlsZWQgdG8gc3RhcnQgb3IgaXMgbm90IHN1cHBvcnRlZCBvbiB0aGUgZGV2aWNlIHVuZGVyIHRlc3QuXG4gKi9cbmNvbW1hbmRzLm1vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nID0gYXN5bmMgZnVuY3Rpb24gbW9iaWxlU3RhcnRNZWRpYVByb2plY3Rpb25SZWNvcmRpbmcgKG9wdGlvbnMgPSB7fSkge1xuICBhd2FpdCB2ZXJpZnlNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdJc1N1cHBvcnRlZCh0aGlzLmFkYik7XG5cbiAgY29uc3Qge3Jlc29sdXRpb24sIHByaW9yaXR5LCBtYXhEdXJhdGlvblNlYywgZmlsZW5hbWV9ID0gb3B0aW9ucztcbiAgY29uc3QgcmVjb3JkZXIgPSBuZXcgTWVkaWFQcm9qZWN0aW9uUmVjb3JkZXIodGhpcy5hZGIpO1xuICBjb25zdCBmbmFtZSA9IGFkanVzdE1lZGlhRXh0ZW5zaW9uKGZpbGVuYW1lIHx8IG1vbWVudCgpLmZvcm1hdChERUZBVUxUX0ZJTEVOQU1FX0ZPUk1BVCkpO1xuICBjb25zdCBkaWRTdGFydCA9IGF3YWl0IHJlY29yZGVyLnN0YXJ0KHtcbiAgICByZXNvbHV0aW9uLFxuICAgIHByaW9yaXR5LFxuICAgIG1heER1cmF0aW9uU2VjLFxuICAgIGZpbGVuYW1lOiBmbmFtZSxcbiAgfSk7XG4gIGlmIChkaWRTdGFydCkge1xuICAgIHRoaXMubG9nLmluZm8oYEEgbmV3IG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nICcke2ZuYW1lfScgaGFzIGJlZW4gc3VjY2Vzc2Z1bGx5IHN0YXJ0ZWRgKTtcbiAgfSBlbHNlIHtcbiAgICB0aGlzLmxvZy5pbmZvKCdBbm90aGVyIG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nIGlzIGFscmVhZHkgaW4gcHJvZ3Jlc3MuIFRoZXJlIGlzIG5vdGhpbmcgdG8gc3RhcnQnKTtcbiAgfVxuICByZXR1cm4gZGlkU3RhcnQ7XG59O1xuXG4vKipcbiAqIENoZWNrcyBpZiBhIG1lZGlhIHByb2plY3Rpb24tYmFzZWQgcmVjb3JkaW5nIGlzIGN1cnJlbnRseSBydW5uaW5nLlxuICpcbiAqIEByZXR1cm5zIHtib29sZWFufSBUcnVlIGlmIGEgcmVjb3JkaW5nIGlzIGluIHByb2dyZXNzLlxuICogQHRocm93cyB7RXJyb3J9IElmIGEgcmVjb3JkaW5nIGlzIG5vdCBzdXBwb3J0ZWQgb24gdGhlIGRldmljZSB1bmRlciB0ZXN0LlxuICovXG5jb21tYW5kcy5tb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcgPSBhc3luYyBmdW5jdGlvbiBtb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcgKCkge1xuICBhd2FpdCB2ZXJpZnlNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdJc1N1cHBvcnRlZCh0aGlzLmFkYik7XG5cbiAgY29uc3QgcmVjb3JkZXIgPSBuZXcgTWVkaWFQcm9qZWN0aW9uUmVjb3JkZXIodGhpcy5hZGIpO1xuICByZXR1cm4gYXdhaXQgcmVjb3JkZXIuaXNSdW5uaW5nKCk7XG59O1xuXG4vKipcbiAqIEB0eXBlZGVmIHtPYmplY3R9IFN0b3BSZWNvcmRpbmdPcHRpb25zXG4gKlxuICogQHByb3BlcnR5IHtzdHJpbmc/fSByZW1vdGVQYXRoIFRoZSBwYXRoIHRvIHRoZSByZW1vdGUgbG9jYXRpb24sIHdoZXJlIHRoZSByZXN1bHRpbmcgdmlkZW8gc2hvdWxkIGJlIHVwbG9hZGVkLlxuICogVGhlIGZvbGxvd2luZyBwcm90b2NvbHMgYXJlIHN1cHBvcnRlZDogaHR0cC9odHRwcywgZnRwLlxuICogTnVsbCBvciBlbXB0eSBzdHJpbmcgdmFsdWUgKHRoZSBkZWZhdWx0IHNldHRpbmcpIG1lYW5zIHRoZSBjb250ZW50IG9mIHJlc3VsdGluZ1xuICogZmlsZSBzaG91bGQgYmUgZW5jb2RlZCBhcyBCYXNlNjQgYW5kIHBhc3NlZCBhcyB0aGUgZW5kcG9vbnQgcmVzcG9uc2UgdmFsdWUuXG4gKiBBbiBleGNlcHRpb24gd2lsbCBiZSB0aHJvd24gaWYgdGhlIGdlbmVyYXRlZCBtZWRpYSBmaWxlIGlzIHRvbyBiaWcgdG9cbiAqIGZpdCBpbnRvIHRoZSBhdmFpbGFibGUgcHJvY2VzcyBtZW1vcnkuXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHVzZXIgVGhlIG5hbWUgb2YgdGhlIHVzZXIgZm9yIHRoZSByZW1vdGUgYXV0aGVudGljYXRpb24uXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHBhc3MgVGhlIHBhc3N3b3JkIGZvciB0aGUgcmVtb3RlIGF1dGhlbnRpY2F0aW9uLlxuICogQHByb3BlcnR5IHtzdHJpbmc/fSBtZXRob2QgVGhlIGh0dHAgbXVsdGlwYXJ0IHVwbG9hZCBtZXRob2QgbmFtZS4gVGhlICdQVVQnIG9uZSBpcyB1c2VkIGJ5IGRlZmF1bHQuXG4gKiBAcHJvcGVydHkge09iamVjdD99IGhlYWRlcnMgQWRkaXRpb25hbCBoZWFkZXJzIG1hcHBpbmcgZm9yIG11bHRpcGFydCBodHRwKHMpIHVwbG9hZHNcbiAqIEBwcm9wZXJ0eSB7c3RyaW5nP30gZmlsZUZpZWxkTmFtZSBbZmlsZV0gVGhlIG5hbWUgb2YgdGhlIGZvcm0gZmllbGQsIHdoZXJlIHRoZSBmaWxlIGNvbnRlbnQgQkxPQiBzaG91bGQgYmUgc3RvcmVkIGZvclxuICogaHR0cChzKSB1cGxvYWRzXG4gKiBAcHJvcGVydHkge09iamVjdHxBcnJheTxQYWlyPj99IGZvcm1GaWVsZHMgQWRkaXRpb25hbCBmb3JtIGZpZWxkcyBmb3IgbXVsdGlwYXJ0IGh0dHAocykgdXBsb2Fkc1xuICovXG5cbi8qKlxuICogU3RvcCBhIG1lZGlhIHByb2plY3Rpb24tYmFzZWQgcmVjb3JkaW5nLlxuICogSWYgbm8gcmVjb3JkaW5nIGhhcyBiZWVuIHN0YXJ0ZWQgYmVmb3JlIHRoZW4gYW4gZXJyb3IgaXMgdGhyb3duLlxuICogSWYgdGhlIHJlY29yZGluZyBoYXMgYmVlbiBhbHJlYWR5IGZpbmlzaGVkIGJlZm9yZSB0aGlzIEFQSSBoYXMgYmVlbiBjYWxsZWRcbiAqIHRoZW4gdGhlIG1vc3QgcmVjZW50IHJlY29yZGVkIGZpbGUgaXMgcmV0dXJuZWQuXG4gKlxuICogQHBhcmFtIHs/U3RvcFJlY29yZGluZ09wdGlvbnN9IG9wdGlvbnMgQXZhaWxhYmxlIG9wdGlvbnMuXG4gKiBAcmV0dXJucyB7c3RyaW5nfSBCYXNlNjQtZW5jb2RlZCBjb250ZW50IG9mIHRoZSByZWNvcmRlZCBtZWRpYSBmaWxlIGlmICdyZW1vdGVQYXRoJ1xuICogcGFyYW1ldGVyIGlzIGZhbHN5IG9yIGFuIGVtcHR5IHN0cmluZy5cbiAqIEB0aHJvd3Mge0Vycm9yfSBJZiB0aGVyZSB3YXMgYW4gZXJyb3Igd2hpbGUgc3RvcHBpbmcgYSByZWNvcmRpbmcsXG4gKiBmZXRjaGluZyB0aGUgY29udGVudCBvZiB0aGUgcmVtb3RlIG1lZGlhIGZpbGUsXG4gKiBvciBpZiBhIHJlY29yZGluZyBpcyBub3Qgc3VwcG9ydGVkIG9uIHRoZSBkZXZpY2UgdW5kZXIgdGVzdC5cbiAqL1xuY29tbWFuZHMubW9iaWxlU3RvcE1lZGlhUHJvamVjdGlvblJlY29yZGluZyA9IGFzeW5jIGZ1bmN0aW9uIG1vYmlsZVN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmcgKG9wdGlvbnMgPSB7fSkge1xuICBhd2FpdCB2ZXJpZnlNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdJc1N1cHBvcnRlZCh0aGlzLmFkYik7XG5cbiAgY29uc3QgcmVjb3JkZXIgPSBuZXcgTWVkaWFQcm9qZWN0aW9uUmVjb3JkZXIodGhpcy5hZGIpO1xuICBpZiAoYXdhaXQgcmVjb3JkZXIuc3RvcCgpKSB7XG4gICAgdGhpcy5sb2cuaW5mbygnU3VjY2Vzc2Z1bGx5IHN0b3BwZWQgYSBtZWRpYSBwcm9qZWN0aW9uIHJlY29yZGluZy4gUHVsbGluZyB0aGUgcmVjb3JkZWQgbWVkaWEnKTtcbiAgfSBlbHNlIHtcbiAgICB0aGlzLmxvZy5pbmZvKCdNZWRpYSBwcm9qZWN0aW9uIHJlY29yZGluZyBpcyBub3QgcnVubmluZy4gVGhlcmUgaXMgbm90aGluZyB0byBzdG9wJyk7XG4gIH1cbiAgY29uc3QgcmVjZW50UmVjb3JkaW5nUGF0aCA9IGF3YWl0IHJlY29yZGVyLnB1bGxSZWNlbnQoKTtcbiAgaWYgKCFyZWNlbnRSZWNvcmRpbmdQYXRoKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBObyByZWNlbnQgbWVkaWEgcHJvamVjdGlvbiByZWNvcmRpbmcgaGF2ZSBiZWVuIGZvdW5kLiBEaWQgeW91IHN0YXJ0IGFueT9gKTtcbiAgfVxuXG4gIGNvbnN0IHtyZW1vdGVQYXRofSA9IG9wdGlvbnM7XG4gIGlmIChfLmlzRW1wdHkocmVtb3RlUGF0aCkpIHtcbiAgICBjb25zdCB7c2l6ZX0gPSBhd2FpdCBmcy5zdGF0KHJlY2VudFJlY29yZGluZ1BhdGgpO1xuICAgIHRoaXMubG9nLmRlYnVnKGBUaGUgc2l6ZSBvZiB0aGUgcmVzdWx0aW5nIG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nIGlzICR7dXRpbC50b1JlYWRhYmxlU2l6ZVN0cmluZyhzaXplKX1gKTtcbiAgfVxuICB0cnkge1xuICAgIHJldHVybiBhd2FpdCB1cGxvYWRSZWNvcmRlZE1lZGlhKHJlY2VudFJlY29yZGluZ1BhdGgsIHJlbW90ZVBhdGgsIG9wdGlvbnMpO1xuICB9IGZpbmFsbHkge1xuICAgIGF3YWl0IGZzLnJpbXJhZihwYXRoLmRpcm5hbWUocmVjZW50UmVjb3JkaW5nUGF0aCkpO1xuICB9XG59O1xuXG5cbmV4cG9ydCB7IGNvbW1hbmRzIH07XG5leHBvcnQgZGVmYXVsdCBjb21tYW5kcztcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFHQSxNQUFNQSxRQUFRLEdBQUcsRUFBakI7O0FBR0EsTUFBTUMsV0FBVyxHQUFHLE1BQXBCO0FBQ0EsTUFBTUMsNEJBQTRCLEdBQUcsSUFBSSxJQUF6QztBQUNBLE1BQU1DLHlCQUF5QixHQUFHLElBQUksSUFBdEM7QUFDQSxNQUFNQyxhQUFhLEdBQUcsRUFBdEI7QUFDQSxNQUFNQyxzQkFBc0IsR0FBSSxHQUFFQyxzQ0FBdUIsNEJBQXpEO0FBQ0EsTUFBTUMsdUJBQXVCLEdBQUksR0FBRUQsc0NBQXVCLDhCQUExRDtBQUNBLE1BQU1FLHNCQUFzQixHQUFJLEdBQUVGLHNDQUF1Qix5QkFBekQ7QUFDQSxNQUFNRyxxQkFBcUIsR0FBSSxHQUFFSCxzQ0FBdUIsd0JBQXhEO0FBQ0EsTUFBTUksZUFBZSxHQUFJLG9DQUFtQ0osc0NBQXVCLFFBQW5GO0FBQ0EsTUFBTUssdUJBQXVCLEdBQUcscUJBQWhDOztBQUdBLGVBQWVDLG1CQUFmLENBQW9DQyxTQUFwQyxFQUErQ0MsVUFBVSxHQUFHLElBQTVELEVBQWtFQyxhQUFhLEdBQUcsRUFBbEYsRUFBc0Y7RUFDcEYsSUFBSUMsZUFBQSxDQUFFQyxPQUFGLENBQVVILFVBQVYsQ0FBSixFQUEyQjtJQUN6QixPQUFPLENBQUMsTUFBTUksYUFBQSxDQUFLQyxnQkFBTCxDQUFzQk4sU0FBdEIsQ0FBUCxFQUF5Q08sUUFBekMsRUFBUDtFQUNEOztFQUVELE1BQU07SUFBQ0MsSUFBRDtJQUFPQyxJQUFQO0lBQWFDLE1BQWI7SUFBcUJDLE9BQXJCO0lBQThCQyxhQUE5QjtJQUE2Q0M7RUFBN0MsSUFBMkRYLGFBQWpFO0VBQ0EsTUFBTVksT0FBTyxHQUFHO0lB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
|
|
196
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjb21tYW5kcyIsIkRFRkFVTFRfRVhUIiwiUkVDT1JESU5HX1NUQVJUVVBfVElNRU9VVF9NUyIsIlJFQ09SRElOR19TVE9QX1RJTUVPVVRfTVMiLCJNSU5fQVBJX0xFVkVMIiwiUkVDT1JESU5HX1NFUlZJQ0VfTkFNRSIsIlNFVFRJTkdTX0hFTFBFUl9QS0dfSUQiLCJSRUNPUkRJTkdfQUNUSVZJVFlfTkFNRSIsIlJFQ09SRElOR19BQ1RJT05fU1RBUlQiLCJSRUNPUkRJTkdfQUNUSU9OX1NUT1AiLCJSRUNPUkRJTkdTX1JPT1QiLCJERUZBVUxUX0ZJTEVOQU1FX0ZPUk1BVCIsInVwbG9hZFJlY29yZGVkTWVkaWEiLCJsb2NhbEZpbGUiLCJyZW1vdGVQYXRoIiwidXBsb2FkT3B0aW9ucyIsIl8iLCJpc0VtcHR5IiwidXRpbCIsInRvSW5NZW1vcnlCYXNlNjQiLCJ0b1N0cmluZyIsInVzZXIiLCJwYXNzIiwibWV0aG9kIiwiaGVhZGVycyIsImZpbGVGaWVsZE5hbWUiLCJmb3JtRmllbGRzIiwib3B0aW9ucyIsImF1dGgiLCJuZXQiLCJ1cGxvYWRGaWxlIiwiYWRqdXN0TWVkaWFFeHRlbnNpb24iLCJuYW1lIiwidG9Mb3dlciIsImVuZHNXaXRoIiwidmVyaWZ5TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nSXNTdXBwb3J0ZWQiLCJhZGIiLCJhcGlMZXZlbCIsImdldEFwaUxldmVsIiwiRXJyb3IiLCJNZWRpYVByb2plY3Rpb25SZWNvcmRlciIsImNvbnN0cnVjdG9yIiwiaXNSdW5uaW5nIiwic3Rkb3V0Iiwic2hlbGwiLCJpbmNsdWRlcyIsInN0YXJ0Iiwib3B0cyIsImNsZWFudXAiLCJmaWxlbmFtZSIsIm1heER1cmF0aW9uU2VjIiwicHJpb3JpdHkiLCJyZXNvbHV0aW9uIiwiYXJncyIsInB1c2giLCJCIiwicmVzb2x2ZSIsInJlamVjdCIsInNldFRpbWVvdXQiLCJwdWxsUmVjZW50IiwicmVjb3JkaW5ncyIsImxzIiwiZHN0UGF0aCIsInBhdGgiLCJqb2luIiwidGVtcERpciIsIm9wZW5EaXIiLCJwdWxsIiwic3RvcCIsIndhaXRGb3JDb25kaXRpb24iLCJ3YWl0TXMiLCJpbnRlcnZhbE1zIiwiZSIsIm1vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nIiwicmVjb3JkZXIiLCJmbmFtZSIsIm1vbWVudCIsImZvcm1hdCIsImRpZFN0YXJ0IiwibG9nIiwiaW5mbyIsIm1vYmlsZUlzTWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nUnVubmluZyIsIm1vYmlsZVN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmciLCJyZWNlbnRSZWNvcmRpbmdQYXRoIiwic2l6ZSIsImZzIiwic3RhdCIsImRlYnVnIiwidG9SZWFkYWJsZVNpemVTdHJpbmciLCJyaW1yYWYiLCJkaXJuYW1lIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL2NvbW1hbmRzL21lZGlhLXByb2plY3Rpb24uanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IF8gZnJvbSAnbG9kYXNoJztcbmltcG9ydCB7IHdhaXRGb3JDb25kaXRpb24gfSBmcm9tICdhc3luY2JveCc7XG5pbXBvcnQgeyB1dGlsLCBmcywgbmV0LCB0ZW1wRGlyIH0gZnJvbSAnYXBwaXVtL3N1cHBvcnQnO1xuaW1wb3J0IHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgQiBmcm9tICdibHVlYmlyZCc7XG5pbXBvcnQgeyBTRVRUSU5HU19IRUxQRVJfUEtHX0lEIH0gZnJvbSAnLi4vYW5kcm9pZC1oZWxwZXJzJztcbmltcG9ydCBtb21lbnQgZnJvbSAnbW9tZW50JztcblxuXG5jb25zdCBjb21tYW5kcyA9IHt9O1xuXG4vLyBodHRwczovL2dpdGh1Yi5jb20vYXBwaXVtL2lvLmFwcGl1bS5zZXR0aW5ncyNpbnRlcm5hbC1hdWRpby0tdmlkZW8tcmVjb3JkaW5nXG5jb25zdCBERUZBVUxUX0VYVCA9ICcubXA0JztcbmNvbnN0IFJFQ09SRElOR19TVEFSVFVQX1RJTUVPVVRfTVMgPSAzICogMTAwMDtcbmNvbnN0IFJFQ09SRElOR19TVE9QX1RJTUVPVVRfTVMgPSAzICogMTAwMDtcbmNvbnN0IE1JTl9BUElfTEVWRUwgPSAyOTtcbmNvbnN0IFJFQ09SRElOR19TRVJWSUNFX05BTUUgPSBgJHtTRVRUSU5HU19IRUxQRVJfUEtHX0lEfS8ucmVjb3JkZXIuUmVjb3JkZXJTZXJ2aWNlYDtcbmNvbnN0IFJFQ09SRElOR19BQ1RJVklUWV9OQU1FID0gYCR7U0VUVElOR1NfSEVMUEVSX1BLR19JRH0vaW8uYXBwaXVtLnNldHRpbmdzLlNldHRpbmdzYDtcbmNvbnN0IFJFQ09SRElOR19BQ1RJT05fU1RBUlQgPSBgJHtTRVRUSU5HU19IRUxQRVJfUEtHX0lEfS5yZWNvcmRpbmcuQUNUSU9OX1NUQVJUYDtcbmNvbnN0IFJFQ09SRElOR19BQ1RJT05fU1RPUCA9IGAke1NFVFRJTkdTX0hFTFBFUl9QS0dfSUR9LnJlY29yZGluZy5BQ1RJT05fU1RPUGA7XG5jb25zdCBSRUNPUkRJTkdTX1JPT1QgPSBgL3N0b3JhZ2UvZW11bGF0ZWQvMC9BbmRyb2lkL2RhdGEvJHtTRVRUSU5HU19IRUxQRVJfUEtHX0lEfS9maWxlc2A7XG5jb25zdCBERUZBVUxUX0ZJTEVOQU1FX0ZPUk1BVCA9ICdZWVlZLU1NLUREVEhILW1tLXNzJztcblxuXG5hc3luYyBmdW5jdGlvbiB1cGxvYWRSZWNvcmRlZE1lZGlhIChsb2NhbEZpbGUsIHJlbW90ZVBhdGggPSBudWxsLCB1cGxvYWRPcHRpb25zID0ge30pIHtcbiAgaWYgKF8uaXNFbXB0eShyZW1vdGVQYXRoKSkge1xuICAgIHJldHVybiAoYXdhaXQgdXRpbC50b0luTWVtb3J5QmFzZTY0KGxvY2FsRmlsZSkpLnRvU3RyaW5nKCk7XG4gIH1cblxuICBjb25zdCB7dXNlciwgcGFzcywgbWV0aG9kLCBoZWFkZXJzLCBmaWxlRmllbGROYW1lLCBmb3JtRmllbGRzfSA9IHVwbG9hZE9wdGlvbnM7XG4gIGNvbnN0IG9wdGlvbnMgPSB7XG4gICAgbWV0aG9kOiBtZXRob2QgfHwgJ1BVVCcsXG4gICAgaGVhZGVycyxcbiAgICBmaWxlRmllbGROYW1lLFxuICAgIGZvcm1GaWVsZHMsXG4gIH07XG4gIGlmICh1c2VyICYmIHBhc3MpIHtcbiAgICBvcHRpb25zLmF1dGggPSB7dXNlciwgcGFzc307XG4gIH1cbiAgYXdhaXQgbmV0LnVwbG9hZEZpbGUobG9jYWxGaWxlLCByZW1vdGVQYXRoLCBvcHRpb25zKTtcbiAgcmV0dXJuICcnO1xufVxuXG5mdW5jdGlvbiBhZGp1c3RNZWRpYUV4dGVuc2lvbiAobmFtZSkge1xuICByZXR1cm4gXy50b0xvd2VyKG5hbWUpLmVuZHNXaXRoKERFRkFVTFRfRVhUKSA/IG5hbWUgOiBgJHtuYW1lfSR7REVGQVVMVF9FWFR9YDtcbn1cblxuYXN5bmMgZnVuY3Rpb24gdmVyaWZ5TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nSXNTdXBwb3J0ZWQgKGFkYikge1xuICBjb25zdCBhcGlMZXZlbCA9IGF3YWl0IGFkYi5nZXRBcGlMZXZlbCgpO1xuICBpZiAoYXBpTGV2ZWwgPCBNSU5fQVBJX0xFVkVMKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBNZWRpYSBwcm9qZWN0aW9uLWJhc2VkIHJlY29yZGluZyBpcyBub3QgYXZhaWxhYmxlIG9uIEFQSSBMZXZlbCAke2FwaUxldmVsfS4gYCArXG4gICAgICBgTWluaW11bSByZXF1aXJlZCBBUEkgTGV2ZWwgaXMgJHtNSU5fQVBJX0xFVkVMfS5gKTtcbiAgfVxufVxuXG5cbmNsYXNzIE1lZGlhUHJvamVjdGlvblJlY29yZGVyIHtcbiAgY29uc3RydWN0b3IgKGFkYikge1xuICAgIHRoaXMuYWRiID0gYWRiO1xuICB9XG5cbiAgYXN5bmMgaXNSdW5uaW5nICgpIHtcbiAgICBjb25zdCBzdGRvdXQgPSBhd2FpdCB0aGlzLmFkYi5zaGVsbChbXG4gICAgICAnZHVtcHN5cycsICdhY3Rpdml0eScsICdzZXJ2aWNlcycsIFJFQ09SRElOR19TRVJWSUNFX05BTUVcbiAgICBdKTtcbiAgICByZXR1cm4gc3Rkb3V0LmluY2x1ZGVzKFJFQ09SRElOR19TRVJWSUNFX05BTUUpO1xuICB9XG5cbiAgYXN5bmMgc3RhcnQgKG9wdHMgPSB7fSkge1xuICAgIGlmIChhd2FpdCB0aGlzLmlzUnVubmluZygpKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgYXdhaXQgdGhpcy5jbGVhbnVwKCk7XG4gICAgY29uc3Qge1xuICAgICAgZmlsZW5hbWUsXG4gICAgICBtYXhEdXJhdGlvblNlYyxcbiAgICAgIHByaW9yaXR5LFxuICAgICAgcmVzb2x1dGlvbixcbiAgICB9ID0gb3B0cztcbiAgICBjb25zdCBhcmdzID0gW1xuICAgICAgJ2FtJywgJ3N0YXJ0JyxcbiAgICAgICctbicsIFJFQ09SRElOR19BQ1RJVklUWV9OQU1FLFxuICAgICAgJy1hJywgUkVDT1JESU5HX0FDVElPTl9TVEFSVCxcbiAgICBdO1xuICAgIGlmIChmaWxlbmFtZSkge1xuICAgICAgYXJncy5wdXNoKCctLWVzJywgJ2ZpbGVuYW1lJywgZmlsZW5hbWUpO1xuICAgIH1cbiAgICBpZiAobWF4RHVyYXRpb25TZWMpIHtcbiAgICAgIGFyZ3MucHVzaCgnLS1lcycsICdtYXhfZHVyYXRpb25fc2VjJywgYCR7bWF4RHVyYXRpb25TZWN9YCk7XG4gICAgfVxuICAgIGlmIChwcmlvcml0eSkge1xuICAgICAgYXJncy5wdXNoKCctLWVzJywgJ3ByaW9yaXR5JywgcHJpb3JpdHkpO1xuICAgIH1cbiAgICBpZiAocmVzb2x1dGlvbikge1xuICAgICAgYXJncy5wdXNoKCctLWVzJywgJ3Jlc29sdXRpb24nLCByZXNvbHV0aW9uKTtcbiAgICB9XG4gICAgYXdhaXQgdGhpcy5hZGIuc2hlbGwoYXJncyk7XG4gICAgYXdhaXQgbmV3IEIoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgICAgc2V0VGltZW91dChhc3luYyAoKSA9PiB7XG4gICAgICAgIGlmICghYXdhaXQgdGhpcy5pc1J1bm5pbmcoKSkge1xuICAgICAgICAgIHJldHVybiByZWplY3QobmV3IEVycm9yKFxuICAgICAgICAgICAgYFRoZSBtZWRpYSBwcm9qZWN0aW9uIHJlY29yZGluZyBpcyBub3QgcnVubmluZyBhZnRlciAke1JFQ09SRElOR19TVEFSVFVQX1RJTUVPVVRfTVN9bXMuIGAgK1xuICAgICAgICAgICAgYFBsZWFzZSBjaGVjayB0aGUgbG9nY2F0IG91dHB1dCBmb3IgbW9yZSBkZXRhaWxzLmBcbiAgICAgICAgICApKTtcbiAgICAgICAgfVxuICAgICAgICByZXNvbHZlKCk7XG4gICAgICB9LCBSRUNPUkRJTkdfU1RBUlRVUF9USU1FT1VUX01TKTtcbiAgICB9KTtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfVxuXG4gIGFzeW5jIGNsZWFudXAgKCkge1xuICAgIGF3YWl0IHRoaXMuYWRiLnNoZWxsKFtgcm0gLWYgJHtSRUNPUkRJTkdTX1JPT1R9LypgXSk7XG4gIH1cblxuICBhc3luYyBwdWxsUmVjZW50ICgpIHtcbiAgICBjb25zdCByZWNvcmRpbmdzID0gYXdhaXQgdGhpcy5hZGIubHMoUkVDT1JESU5HU19ST09ULCBbJy10ciddKTtcbiAgICBpZiAoXy5pc0VtcHR5KHJlY29yZGluZ3MpKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBjb25zdCBkc3RQYXRoID0gcGF0aC5qb2luKGF3YWl0IHRlbXBEaXIub3BlbkRpcigpLCByZWNvcmRpbmdzWzBdKTtcbiAgICBhd2FpdCB0aGlzLmFkYi5wdWxsKGAke1JFQ09SRElOR1NfUk9PVH0vJHtyZWNvcmRpbmdzWzBdfWAsIGRzdFBhdGgpO1xuICAgIHJldHVybiBkc3RQYXRoO1xuICB9XG5cbiAgYXN5bmMgc3RvcCAoKSB7XG4gICAgaWYgKCFhd2FpdCB0aGlzLmlzUnVubmluZygpKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgYXdhaXQgdGhpcy5hZGIuc2hlbGwoW1xuICAgICAgJ2FtJywgJ3N0YXJ0JyxcbiAgICAgICctbicsIFJFQ09SRElOR19BQ1RJVklUWV9OQU1FLFxuICAgICAgJy1hJywgUkVDT1JESU5HX0FDVElPTl9TVE9QLFxuICAgIF0pO1xuICAgIHRyeSB7XG4gICAgICBhd2FpdCB3YWl0Rm9yQ29uZGl0aW9uKGFzeW5jICgpID0+ICEoYXdhaXQgdGhpcy5pc1J1bm5pbmcoKSksIHtcbiAgICAgICAgd2FpdE1zOiBSRUNPUkRJTkdfU1RPUF9USU1FT1VUX01TLFxuICAgICAgICBpbnRlcnZhbE1zOiA1MDAsXG4gICAgICB9KTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgIGBUaGUgYXR0ZW1wdCB0byBzdG9wIHRoZSBjdXJyZW50IG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nIHRpbWVkIG91dCBhZnRlciBgICtcbiAgICAgICAgYCR7UkVDT1JESU5HX1NUT1BfVElNRU9VVF9NU31tc2BcbiAgICAgICk7XG4gICAgfVxuICAgIHJldHVybiB0cnVlO1xuICB9XG59XG5cblxuLyoqXG4gKiBAdHlwZWRlZiB7T2JqZWN0fSBTdGFydFJlY29yZGluZ09wdGlvbnNcbiAqXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHJlc29sdXRpb24gTWF4aW11bSBzdXBwb3J0ZWQgcmVzb2x1dGlvbiBvbi1kZXZpY2UgKERldGVjdGVkXG4gKiBhdXRvbWF0aWNhbGx5IGJ5IHRoZSBhcHAgaXRzZWxmKSwgd2hpY2ggdXN1YWxseSBlcXVhbHMgdG8gRnVsbCBIRCAxOTIweDEwODAgb24gbW9zdFxuICogcGhvbmVzIGhvd2V2ZXIgeW91IGNhbiBjaGFuZ2UgaXQgdG8gZm9sbG93aW5nIHN1cHBvcnRlZCByZXNvbHV0aW9uc1xuICogYXMgd2VsbDogXCIxOTIweDEwODBcIiwgXCIxMjgweDcyMFwiLCBcIjcyMHg0ODBcIiwgXCIzMjB4MjQwXCIsIFwiMTc2eDE0NFwiLlxuICogQHByb3BlcnR5IHtudW1iZXI/fSBtYXhEdXJhdGlvblNlYyBbOTAwXSBEZWZhdWx0IHZhbHVlOiA5MDAgc2Vjb25kcyB3aGljaCBtZWFuc1xuICogbWF4aW11bSBhbGxvd2VkIGR1cmF0aW9uIGlzIDE1IG1pbnV0ZSwgeW91IGNhbiBpbmNyZWFzZSBpdCBpZiB5b3VyIHRlc3QgdGFrZXNcbiAqIGxvbmdlciB0aGFuIHRoYXQuXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHByaW9yaXR5IFtoaWdoXSBNZWFucyByZWNvcmRpbmcgdGhyZWFkIHByaW9yaXR5IGlzIG1heGltdW1cbiAqIGhvd2V2ZXIgaWYgeW91IGZhY2UgcGVyZm9ybWFuY2UgZHJvcHMgZHVyaW5nIHRlc3Rpbmcgd2l0aCByZWNvcmRpbmcgZW5hYmxlZCwgeW91XG4gKiBjYW4gcmVkdWNlIHJlY29yZGluZyBwcmlvcml0eSB0byBcIm5vcm1hbFwiIG9yIFwibG93XCIuXG4gKiBAcHJvcGVydHkge3N0cmluZz99IGZpbGVuYW1lIFlvdSBjYW4gdHlwZSByZWNvcmRpbmcgdmlkZW8gZmlsZSBuYW1lIGFzIHlvdSB3YW50LFxuICogYnV0IHJlY29yZGluZyBjdXJyZW50bHkgc3VwcG9ydHMgb25seSBcIm1wNFwiIGZvcm1hdCBzbyB5b3VyIGZpbGVuYW1lIG11c3QgZW5kIHdpdGggXCIubXA0XCIuXG4gKiBBbiBpbnZhbGlkIGZpbGUgbmFtZSB3aWxsIGZhaWwgdG8gc3RhcnQgdGhlIHJlY29yZGluZy5cbiAqIElmIG5vdCBwcm92aWRlZCB0aGVuIHRoZSBjdXJyZW50IHRpbWVzdGFtcCB3aWxsIGJlIHVzZWQgYXMgZmlsZSBuYW1lLlxuICovXG5cbi8qKlxuICogUmVjb3JkIHRoZSBkaXNwbGF5IG9mIGEgcmVhbCBkZXZpY2VzIHJ1bm5pbmcgQW5kcm9pZCAxMCAoQVBJIGxldmVsIDI5KSBhbmQgaGlnaGVyLlxuICogVGhlIHNjcmVlbiBhY3Rpdml0eSBpcyByZWNvcmRlZCB0byBhIE1QRUctNCBmaWxlLiBBdWRpbyBpcyBhbHNvIHJlY29yZGVkIGJ5IGRlZmF1bHRcbiAqIChvbmx5IGZvciBhcHBzIHRoYXQgYWxsb3cgaXQgaW4gdGhlaXIgbWFuaWZlc3RzKS5cbiAqIElmIGFub3RoZXIgcmVjb3JkaW5nIGhhcyBiZWVuIGFscmVhZHkgc3RhcnRlZCB0aGVuIHRoZSBjb21tYW5kIHdpbGwgZXhpdCBzaWxlbnRseS5cbiAqIFRoZSBwcmV2aW91c2x5IHJlY29yZGVkIHZpZGVvIGZpbGUgaXMgZGVsZXRlZCB3aGVuIGEgbmV3IHJlY29yZGluZyBzZXNzaW9uIGlzIHN0YXJ0ZWQuXG4gKiBSZWNvcmRpbmcgY29udGludWVzIGl0IGlzIHN0b3BwZWQgZXhwbGljaXRseSBvciB1bnRpbCB0aGUgdGltZW91dCBoYXBwZW5zLlxuICpcbiAqIEBwYXJhbSB7P1N0YXJ0UmVjb3JkaW5nT3B0aW9uc30gb3B0aW9ucyBBdmFpbGFibGUgb3B0aW9ucy5cbiAqIEByZXR1cm5zIHtib29sZWFufSBUcnVlIGlmIGEgbmV3IHJlY29yZGluZyBoYXMgc3VjY2Vzc2Z1bGx5IHN0YXJ0ZWQuXG4gKiBAdGhyb3dzIHtFcnJvcn0gSWYgcmVjb3JkaW5nIGhhcyBmYWlsZWQgdG8gc3RhcnQgb3IgaXMgbm90IHN1cHBvcnRlZCBvbiB0aGUgZGV2aWNlIHVuZGVyIHRlc3QuXG4gKi9cbmNvbW1hbmRzLm1vYmlsZVN0YXJ0TWVkaWFQcm9qZWN0aW9uUmVjb3JkaW5nID0gYXN5bmMgZnVuY3Rpb24gbW9iaWxlU3RhcnRNZWRpYVByb2plY3Rpb25SZWNvcmRpbmcgKG9wdGlvbnMgPSB7fSkge1xuICBhd2FpdCB2ZXJpZnlNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdJc1N1cHBvcnRlZCh0aGlzLmFkYik7XG5cbiAgY29uc3Qge3Jlc29sdXRpb24sIHByaW9yaXR5LCBtYXhEdXJhdGlvblNlYywgZmlsZW5hbWV9ID0gb3B0aW9ucztcbiAgY29uc3QgcmVjb3JkZXIgPSBuZXcgTWVkaWFQcm9qZWN0aW9uUmVjb3JkZXIodGhpcy5hZGIpO1xuICBjb25zdCBmbmFtZSA9IGFkanVzdE1lZGlhRXh0ZW5zaW9uKGZpbGVuYW1lIHx8IG1vbWVudCgpLmZvcm1hdChERUZBVUxUX0ZJTEVOQU1FX0ZPUk1BVCkpO1xuICBjb25zdCBkaWRTdGFydCA9IGF3YWl0IHJlY29yZGVyLnN0YXJ0KHtcbiAgICByZXNvbHV0aW9uLFxuICAgIHByaW9yaXR5LFxuICAgIG1heER1cmF0aW9uU2VjLFxuICAgIGZpbGVuYW1lOiBmbmFtZSxcbiAgfSk7XG4gIGlmIChkaWRTdGFydCkge1xuICAgIHRoaXMubG9nLmluZm8oYEEgbmV3IG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nICcke2ZuYW1lfScgaGFzIGJlZW4gc3VjY2Vzc2Z1bGx5IHN0YXJ0ZWRgKTtcbiAgfSBlbHNlIHtcbiAgICB0aGlzLmxvZy5pbmZvKCdBbm90aGVyIG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nIGlzIGFscmVhZHkgaW4gcHJvZ3Jlc3MuIFRoZXJlIGlzIG5vdGhpbmcgdG8gc3RhcnQnKTtcbiAgfVxuICByZXR1cm4gZGlkU3RhcnQ7XG59O1xuXG4vKipcbiAqIENoZWNrcyBpZiBhIG1lZGlhIHByb2plY3Rpb24tYmFzZWQgcmVjb3JkaW5nIGlzIGN1cnJlbnRseSBydW5uaW5nLlxuICpcbiAqIEByZXR1cm5zIHtib29sZWFufSBUcnVlIGlmIGEgcmVjb3JkaW5nIGlzIGluIHByb2dyZXNzLlxuICogQHRocm93cyB7RXJyb3J9IElmIGEgcmVjb3JkaW5nIGlzIG5vdCBzdXBwb3J0ZWQgb24gdGhlIGRldmljZSB1bmRlciB0ZXN0LlxuICovXG5jb21tYW5kcy5tb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcgPSBhc3luYyBmdW5jdGlvbiBtb2JpbGVJc01lZGlhUHJvamVjdGlvblJlY29yZGluZ1J1bm5pbmcgKCkge1xuICBhd2FpdCB2ZXJpZnlNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdJc1N1cHBvcnRlZCh0aGlzLmFkYik7XG5cbiAgY29uc3QgcmVjb3JkZXIgPSBuZXcgTWVkaWFQcm9qZWN0aW9uUmVjb3JkZXIodGhpcy5hZGIpO1xuICByZXR1cm4gYXdhaXQgcmVjb3JkZXIuaXNSdW5uaW5nKCk7XG59O1xuXG4vKipcbiAqIEB0eXBlZGVmIHtPYmplY3R9IFN0b3BSZWNvcmRpbmdPcHRpb25zXG4gKlxuICogQHByb3BlcnR5IHtzdHJpbmc/fSByZW1vdGVQYXRoIFRoZSBwYXRoIHRvIHRoZSByZW1vdGUgbG9jYXRpb24sIHdoZXJlIHRoZSByZXN1bHRpbmcgdmlkZW8gc2hvdWxkIGJlIHVwbG9hZGVkLlxuICogVGhlIGZvbGxvd2luZyBwcm90b2NvbHMgYXJlIHN1cHBvcnRlZDogaHR0cC9odHRwcywgZnRwLlxuICogTnVsbCBvciBlbXB0eSBzdHJpbmcgdmFsdWUgKHRoZSBkZWZhdWx0IHNldHRpbmcpIG1lYW5zIHRoZSBjb250ZW50IG9mIHJlc3VsdGluZ1xuICogZmlsZSBzaG91bGQgYmUgZW5jb2RlZCBhcyBCYXNlNjQgYW5kIHBhc3NlZCBhcyB0aGUgZW5kcG9vbnQgcmVzcG9uc2UgdmFsdWUuXG4gKiBBbiBleGNlcHRpb24gd2lsbCBiZSB0aHJvd24gaWYgdGhlIGdlbmVyYXRlZCBtZWRpYSBmaWxlIGlzIHRvbyBiaWcgdG9cbiAqIGZpdCBpbnRvIHRoZSBhdmFpbGFibGUgcHJvY2VzcyBtZW1vcnkuXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHVzZXIgVGhlIG5hbWUgb2YgdGhlIHVzZXIgZm9yIHRoZSByZW1vdGUgYXV0aGVudGljYXRpb24uXG4gKiBAcHJvcGVydHkge3N0cmluZz99IHBhc3MgVGhlIHBhc3N3b3JkIGZvciB0aGUgcmVtb3RlIGF1dGhlbnRpY2F0aW9uLlxuICogQHByb3BlcnR5IHtzdHJpbmc/fSBtZXRob2QgVGhlIGh0dHAgbXVsdGlwYXJ0IHVwbG9hZCBtZXRob2QgbmFtZS4gVGhlICdQVVQnIG9uZSBpcyB1c2VkIGJ5IGRlZmF1bHQuXG4gKiBAcHJvcGVydHkge09iamVjdD99IGhlYWRlcnMgQWRkaXRpb25hbCBoZWFkZXJzIG1hcHBpbmcgZm9yIG11bHRpcGFydCBodHRwKHMpIHVwbG9hZHNcbiAqIEBwcm9wZXJ0eSB7c3RyaW5nP30gZmlsZUZpZWxkTmFtZSBbZmlsZV0gVGhlIG5hbWUgb2YgdGhlIGZvcm0gZmllbGQsIHdoZXJlIHRoZSBmaWxlIGNvbnRlbnQgQkxPQiBzaG91bGQgYmUgc3RvcmVkIGZvclxuICogaHR0cChzKSB1cGxvYWRzXG4gKiBAcHJvcGVydHkge09iamVjdHxBcnJheTxQYWlyPj99IGZvcm1GaWVsZHMgQWRkaXRpb25hbCBmb3JtIGZpZWxkcyBmb3IgbXVsdGlwYXJ0IGh0dHAocykgdXBsb2Fkc1xuICovXG5cbi8qKlxuICogU3RvcCBhIG1lZGlhIHByb2plY3Rpb24tYmFzZWQgcmVjb3JkaW5nLlxuICogSWYgbm8gcmVjb3JkaW5nIGhhcyBiZWVuIHN0YXJ0ZWQgYmVmb3JlIHRoZW4gYW4gZXJyb3IgaXMgdGhyb3duLlxuICogSWYgdGhlIHJlY29yZGluZyBoYXMgYmVlbiBhbHJlYWR5IGZpbmlzaGVkIGJlZm9yZSB0aGlzIEFQSSBoYXMgYmVlbiBjYWxsZWRcbiAqIHRoZW4gdGhlIG1vc3QgcmVjZW50IHJlY29yZGVkIGZpbGUgaXMgcmV0dXJuZWQuXG4gKlxuICogQHBhcmFtIHs/U3RvcFJlY29yZGluZ09wdGlvbnN9IG9wdGlvbnMgQXZhaWxhYmxlIG9wdGlvbnMuXG4gKiBAcmV0dXJucyB7c3RyaW5nfSBCYXNlNjQtZW5jb2RlZCBjb250ZW50IG9mIHRoZSByZWNvcmRlZCBtZWRpYSBmaWxlIGlmICdyZW1vdGVQYXRoJ1xuICogcGFyYW1ldGVyIGlzIGZhbHN5IG9yIGFuIGVtcHR5IHN0cmluZy5cbiAqIEB0aHJvd3Mge0Vycm9yfSBJZiB0aGVyZSB3YXMgYW4gZXJyb3Igd2hpbGUgc3RvcHBpbmcgYSByZWNvcmRpbmcsXG4gKiBmZXRjaGluZyB0aGUgY29udGVudCBvZiB0aGUgcmVtb3RlIG1lZGlhIGZpbGUsXG4gKiBvciBpZiBhIHJlY29yZGluZyBpcyBub3Qgc3VwcG9ydGVkIG9uIHRoZSBkZXZpY2UgdW5kZXIgdGVzdC5cbiAqL1xuY29tbWFuZHMubW9iaWxlU3RvcE1lZGlhUHJvamVjdGlvblJlY29yZGluZyA9IGFzeW5jIGZ1bmN0aW9uIG1vYmlsZVN0b3BNZWRpYVByb2plY3Rpb25SZWNvcmRpbmcgKG9wdGlvbnMgPSB7fSkge1xuICBhd2FpdCB2ZXJpZnlNZWRpYVByb2plY3Rpb25SZWNvcmRpbmdJc1N1cHBvcnRlZCh0aGlzLmFkYik7XG5cbiAgY29uc3QgcmVjb3JkZXIgPSBuZXcgTWVkaWFQcm9qZWN0aW9uUmVjb3JkZXIodGhpcy5hZGIpO1xuICBpZiAoYXdhaXQgcmVjb3JkZXIuc3RvcCgpKSB7XG4gICAgdGhpcy5sb2cuaW5mbygnU3VjY2Vzc2Z1bGx5IHN0b3BwZWQgYSBtZWRpYSBwcm9qZWN0aW9uIHJlY29yZGluZy4gUHVsbGluZyB0aGUgcmVjb3JkZWQgbWVkaWEnKTtcbiAgfSBlbHNlIHtcbiAgICB0aGlzLmxvZy5pbmZvKCdNZWRpYSBwcm9qZWN0aW9uIHJlY29yZGluZyBpcyBub3QgcnVubmluZy4gVGhlcmUgaXMgbm90aGluZyB0byBzdG9wJyk7XG4gIH1cbiAgY29uc3QgcmVjZW50UmVjb3JkaW5nUGF0aCA9IGF3YWl0IHJlY29yZGVyLnB1bGxSZWNlbnQoKTtcbiAgaWYgKCFyZWNlbnRSZWNvcmRpbmdQYXRoKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBObyByZWNlbnQgbWVkaWEgcHJvamVjdGlvbiByZWNvcmRpbmcgaGF2ZSBiZWVuIGZvdW5kLiBEaWQgeW91IHN0YXJ0IGFueT9gKTtcbiAgfVxuXG4gIGNvbnN0IHtyZW1vdGVQYXRofSA9IG9wdGlvbnM7XG4gIGlmIChfLmlzRW1wdHkocmVtb3RlUGF0aCkpIHtcbiAgICBjb25zdCB7c2l6ZX0gPSBhd2FpdCBmcy5zdGF0KHJlY2VudFJlY29yZGluZ1BhdGgpO1xuICAgIHRoaXMubG9nLmRlYnVnKGBUaGUgc2l6ZSBvZiB0aGUgcmVzdWx0aW5nIG1lZGlhIHByb2plY3Rpb24gcmVjb3JkaW5nIGlzICR7dXRpbC50b1JlYWRhYmxlU2l6ZVN0cmluZyhzaXplKX1gKTtcbiAgfVxuICB0cnkge1xuICAgIHJldHVybiBhd2FpdCB1cGxvYWRSZWNvcmRlZE1lZGlhKHJlY2VudFJlY29yZGluZ1BhdGgsIHJlbW90ZVBhdGgsIG9wdGlvbnMpO1xuICB9IGZpbmFsbHkge1xuICAgIGF3YWl0IGZzLnJpbXJhZihwYXRoLmRpcm5hbWUocmVjZW50UmVjb3JkaW5nUGF0aCkpO1xuICB9XG59O1xuXG5cbmV4cG9ydCB7IGNvbW1hbmRzIH07XG5leHBvcnQgZGVmYXVsdCBjb21tYW5kcztcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUdBLE1BQU1BLFFBQVEsR0FBRyxDQUFDLENBQUM7O0FBQUM7QUFHcEIsTUFBTUMsV0FBVyxHQUFHLE1BQU07QUFDMUIsTUFBTUMsNEJBQTRCLEdBQUcsQ0FBQyxHQUFHLElBQUk7QUFDN0MsTUFBTUMseUJBQXlCLEdBQUcsQ0FBQyxHQUFHLElBQUk7QUFDMUMsTUFBTUMsYUFBYSxHQUFHLEVBQUU7QUFDeEIsTUFBTUMsc0JBQXNCLEdBQUksR0FBRUMsc0NBQXVCLDRCQUEyQjtBQUNwRixNQUFNQyx1QkFBdUIsR0FBSSxHQUFFRCxzQ0FBdUIsOEJBQTZCO0FBQ3ZGLE1BQU1FLHNCQUFzQixHQUFJLEdBQUVGLHNDQUF1Qix5QkFBd0I7QUFDakYsTUFBTUcscUJBQXFCLEdBQUksR0FBRUgsc0NBQXVCLHdCQUF1QjtBQUMvRSxNQUFNSSxlQUFlLEdBQUksb0NBQW1DSixzQ0FBdUIsUUFBTztBQUMxRixNQUFNSyx1QkFBdUIsR0FBRyxxQkFBcUI7QUFHckQsZUFBZUMsbUJBQW1CLENBQUVDLFNBQVMsRUFBRUMsVUFBVSxHQUFHLElBQUksRUFBRUMsYUFBYSxHQUFHLENBQUMsQ0FBQyxFQUFFO0VBQ3BGLElBQUlDLGVBQUMsQ0FBQ0MsT0FBTyxDQUFDSCxVQUFVLENBQUMsRUFBRTtJQUN6QixPQUFPLENBQUMsTUFBTUksYUFBSSxDQUFDQyxnQkFBZ0IsQ0FBQ04sU0FBUyxDQUFDLEVBQUVPLFFBQVEsRUFBRTtFQUM1RDtFQUVBLE1BQU07SUFBQ0MsSUFBSTtJQUFFQyxJQUFJO0lBQUVDLE1BQU07SUFBRUMsT0FBTztJQUFFQyxhQUFhO0lBQUVDO0VBQVUsQ0FBQyxHQUFHWCxhQUFhO0VBQzlFLE1BQU1ZLE9BQU8s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
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"media-projection.js","names":["commands","DEFAULT_EXT","RECORDING_STARTUP_TIMEOUT_MS","RECORDING_STOP_TIMEOUT_MS","MIN_API_LEVEL","RECORDING_SERVICE_NAME","SETTINGS_HELPER_PKG_ID","RECORDING_ACTIVITY_NAME","RECORDING_ACTION_START","RECORDING_ACTION_STOP","RECORDINGS_ROOT","DEFAULT_FILENAME_FORMAT","uploadRecordedMedia","localFile","remotePath","uploadOptions","_","isEmpty","util","toInMemoryBase64","toString","user","pass","method","headers","fileFieldName","formFields","options","auth","net","uploadFile","adjustMediaExtension","name","toLower","endsWith","verifyMediaProjectionRecordingIsSupported","adb","apiLevel","getApiLevel","Error","MediaProjectionRecorder","constructor","isRunning","stdout","shell","includes","start","opts","cleanup","filename","maxDurationSec","priority","resolution","args","push","B","resolve","reject","setTimeout","pullRecent","recordings","ls","dstPath","path","join","tempDir","openDir","pull","stop","waitForCondition","waitMs","intervalMs","e","mobileStartMediaProjectionRecording","recorder","fname","moment","format","didStart","log","info","mobileIsMediaProjectionRecordingRunning","mobileStopMediaProjectionRecording","recentRecordingPath","size","fs","stat","debug","toReadableSizeString","rimraf","dirname"],"sources":["../../../lib/commands/media-projection.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util, fs, net, tempDir } from 'appium/support';\nimport path from 'path';\nimport B from 'bluebird';\nimport { SETTINGS_HELPER_PKG_ID } from '../android-helpers';\nimport moment from 'moment';\n\n\nconst commands = {};\n\n// https://github.com/appium/io.appium.settings#internal-audio--video-recording\nconst DEFAULT_EXT = '.mp4';\nconst RECORDING_STARTUP_TIMEOUT_MS = 3 * 1000;\nconst RECORDING_STOP_TIMEOUT_MS = 3 * 1000;\nconst MIN_API_LEVEL = 29;\nconst RECORDING_SERVICE_NAME = `${SETTINGS_HELPER_PKG_ID}/.recorder.RecorderService`;\nconst RECORDING_ACTIVITY_NAME = `${SETTINGS_HELPER_PKG_ID}/io.appium.settings.Settings`;\nconst RECORDING_ACTION_START = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_START`;\nconst RECORDING_ACTION_STOP = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_STOP`;\nconst RECORDINGS_ROOT = `/storage/emulated/0/Android/data/${SETTINGS_HELPER_PKG_ID}/files`;\nconst DEFAULT_FILENAME_FORMAT = 'YYYY-MM-DDTHH-mm-ss';\n\n\nasync function uploadRecordedMedia (localFile, remotePath = null, uploadOptions = {}) {\n if (_.isEmpty(remotePath)) {\n return (await util.toInMemoryBase64(localFile)).toString();\n }\n\n const {user, pass, method, headers, fileFieldName, formFields} = uploadOptions;\n const options = {\n method: method || 'PUT',\n headers,\n fileFieldName,\n formFields,\n };\n if (user && pass) {\n options.auth = {user, pass};\n }\n await net.uploadFile(localFile, remotePath, options);\n return '';\n}\n\nfunction adjustMediaExtension (name) {\n return _.toLower(name).endsWith(DEFAULT_EXT) ? name : `${name}${DEFAULT_EXT}`;\n}\n\nasync function verifyMediaProjectionRecordingIsSupported (adb) {\n const apiLevel = await adb.getApiLevel();\n if (apiLevel < MIN_API_LEVEL) {\n throw new Error(`Media projection-based recording is not available on API Level ${apiLevel}. ` +\n `Minimum required API Level is ${MIN_API_LEVEL}.`);\n }\n}\n\n\nclass MediaProjectionRecorder {\n constructor (adb) {\n this.adb = adb;\n }\n\n async isRunning () {\n const stdout = await this.adb.shell([\n 'dumpsys', 'activity', 'services', RECORDING_SERVICE_NAME\n ]);\n return stdout.includes(RECORDING_SERVICE_NAME);\n }\n\n async start (opts = {}) {\n if (await this.isRunning()) {\n return false;\n }\n\n await this.cleanup();\n const {\n filename,\n maxDurationSec,\n priority,\n resolution,\n } = opts;\n const args = [\n 'am', 'start',\n '-n', RECORDING_ACTIVITY_NAME,\n '-a', RECORDING_ACTION_START,\n ];\n if (filename) {\n args.push('--es', 'filename', filename);\n }\n if (maxDurationSec) {\n args.push('--es', 'max_duration_sec', `${maxDurationSec}`);\n }\n if (priority) {\n args.push('--es', 'priority', priority);\n }\n if (resolution) {\n args.push('--es', 'resolution', resolution);\n }\n await this.adb.shell(args);\n await new B((resolve, reject) => {\n setTimeout(async () => {\n if (!await this.isRunning()) {\n return reject(new Error(\n `The media projection recording is not running after ${RECORDING_STARTUP_TIMEOUT_MS}ms. ` +\n `Please check the logcat output for more details.`\n ));\n }\n resolve();\n }, RECORDING_STARTUP_TIMEOUT_MS);\n });\n return true;\n }\n\n async cleanup () {\n await this.adb.shell([`rm -f ${RECORDINGS_ROOT}/*`]);\n }\n\n async pullRecent () {\n const recordings = await this.adb.ls(RECORDINGS_ROOT, ['-tr']);\n if (_.isEmpty(recordings)) {\n return null;\n }\n\n const dstPath = path.join(await tempDir.openDir(), recordings[0]);\n await this.adb.pull(`${RECORDINGS_ROOT}/${recordings[0]}`, dstPath);\n return dstPath;\n }\n\n async stop () {\n if (!await this.isRunning()) {\n return false;\n }\n\n await this.adb.shell([\n 'am', 'start',\n '-n', RECORDING_ACTIVITY_NAME,\n '-a', RECORDING_ACTION_STOP,\n ]);\n try {\n await waitForCondition(async () => !(await this.isRunning()), {\n waitMs: RECORDING_STOP_TIMEOUT_MS,\n intervalMs: 500,\n });\n } catch (e) {\n throw new Error(\n `The attempt to stop the current media projection recording timed out after ` +\n `${RECORDING_STOP_TIMEOUT_MS}ms`\n );\n }\n return true;\n }\n}\n\n\n/**\n * @typedef {Object} StartRecordingOptions\n *\n * @property {string?} resolution Maximum supported resolution on-device (Detected\n * automatically by the app itself), which usually equals to Full HD 1920x1080 on most\n * phones however you can change it to following supported resolutions\n * as well: \"1920x1080\", \"1280x720\", \"720x480\", \"320x240\", \"176x144\".\n * @property {number?} maxDurationSec [900] Default value: 900 seconds which means\n * maximum allowed duration is 15 minute, you can increase it if your test takes\n * longer than that.\n * @property {string?} priority [high] Means recording thread priority is maximum\n * however if you face performance drops during testing with recording enabled, you\n * can reduce recording priority to \"normal\" or \"low\".\n * @property {string?} filename You can type recording video file name as you want,\n * but recording currently supports only \"mp4\" format so your filename must end with \".mp4\".\n * An invalid file name will fail to start the recording.\n * If not provided then the current timestamp will be used as file name.\n */\n\n/**\n * Record the display of a real devices running Android 10 (API level 29) and higher.\n * The screen activity is recorded to a MPEG-4 file. Audio is also recorded by default\n * (only for apps that allow it in their manifests).\n * If another recording has been already started then the command will exit silently.\n * The previously recorded video file is deleted when a new recording session is started.\n * Recording continues it is stopped explicitly or until the timeout happens.\n *\n * @param {?StartRecordingOptions} options Available options.\n * @returns {boolean} True if a new recording has successfully started.\n * @throws {Error} If recording has failed to start or is not supported on the device under test.\n */\ncommands.mobileStartMediaProjectionRecording = async function mobileStartMediaProjectionRecording (options = {}) {\n await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n const {resolution, priority, maxDurationSec, filename} = options;\n const recorder = new MediaProjectionRecorder(this.adb);\n const fname = adjustMediaExtension(filename || moment().format(DEFAULT_FILENAME_FORMAT));\n const didStart = await recorder.start({\n resolution,\n priority,\n maxDurationSec,\n filename: fname,\n });\n if (didStart) {\n this.log.info(`A new media projection recording '${fname}' has been successfully started`);\n } else {\n this.log.info('Another media projection recording is already in progress. There is nothing to start');\n }\n return didStart;\n};\n\n/**\n * Checks if a media projection-based recording is currently running.\n *\n * @returns {boolean} True if a recording is in progress.\n * @throws {Error} If a recording is not supported on the device under test.\n */\ncommands.mobileIsMediaProjectionRecordingRunning = async function mobileIsMediaProjectionRecordingRunning () {\n await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n const recorder = new MediaProjectionRecorder(this.adb);\n return await recorder.isRunning();\n};\n\n/**\n * @typedef {Object} StopRecordingOptions\n *\n * @property {string?} remotePath The path to the remote location, where the resulting video should be uploaded.\n * The following protocols are supported: http/https, ftp.\n * Null or empty string value (the default setting) means the content of resulting\n * file should be encoded as Base64 and passed as the endpoont response value.\n * An exception will be thrown if the generated media file is too big to\n * fit into the available process memory.\n * @property {string?} user The name of the user for the remote authentication.\n * @property {string?} pass The password for the remote authentication.\n * @property {string?} method The http multipart upload method name. The 'PUT' one is used by default.\n * @property {Object?} headers Additional headers mapping for multipart http(s) uploads\n * @property {string?} fileFieldName [file] The name of the form field, where the file content BLOB should be stored for\n * http(s) uploads\n * @property {Object|Array<Pair>?} formFields Additional form fields for multipart http(s) uploads\n */\n\n/**\n * Stop a media projection-based recording.\n * If no recording has been started before then an error is thrown.\n * If the recording has been already finished before this API has been called\n * then the most recent recorded file is returned.\n *\n * @param {?StopRecordingOptions} options Available options.\n * @returns {string} Base64-encoded content of the recorded media file if 'remotePath'\n * parameter is falsy or an empty string.\n * @throws {Error} If there was an error while stopping a recording,\n * fetching the content of the remote media file,\n * or if a recording is not supported on the device under test.\n */\ncommands.mobileStopMediaProjectionRecording = async function mobileStopMediaProjectionRecording (options = {}) {\n await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n const recorder = new MediaProjectionRecorder(this.adb);\n if (await recorder.stop()) {\n this.log.info('Successfully stopped a media projection recording. Pulling the recorded media');\n } else {\n this.log.info('Media projection recording is not running. There is nothing to stop');\n }\n const recentRecordingPath = await recorder.pullRecent();\n if (!recentRecordingPath) {\n throw new Error(`No recent media projection recording have been found. Did you start any?`);\n }\n\n const {remotePath} = options;\n if (_.isEmpty(remotePath)) {\n const {size} = await fs.stat(recentRecordingPath);\n this.log.debug(`The size of the resulting media projection recording is ${util.toReadableSizeString(size)}`);\n }\n try {\n return await uploadRecordedMedia(recentRecordingPath, remotePath, options);\n } finally {\n await fs.rimraf(path.dirname(recentRecordingPath));\n }\n};\n\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"media-projection.js","names":["commands","DEFAULT_EXT","RECORDING_STARTUP_TIMEOUT_MS","RECORDING_STOP_TIMEOUT_MS","MIN_API_LEVEL","RECORDING_SERVICE_NAME","SETTINGS_HELPER_PKG_ID","RECORDING_ACTIVITY_NAME","RECORDING_ACTION_START","RECORDING_ACTION_STOP","RECORDINGS_ROOT","DEFAULT_FILENAME_FORMAT","uploadRecordedMedia","localFile","remotePath","uploadOptions","_","isEmpty","util","toInMemoryBase64","toString","user","pass","method","headers","fileFieldName","formFields","options","auth","net","uploadFile","adjustMediaExtension","name","toLower","endsWith","verifyMediaProjectionRecordingIsSupported","adb","apiLevel","getApiLevel","Error","MediaProjectionRecorder","constructor","isRunning","stdout","shell","includes","start","opts","cleanup","filename","maxDurationSec","priority","resolution","args","push","B","resolve","reject","setTimeout","pullRecent","recordings","ls","dstPath","path","join","tempDir","openDir","pull","stop","waitForCondition","waitMs","intervalMs","e","mobileStartMediaProjectionRecording","recorder","fname","moment","format","didStart","log","info","mobileIsMediaProjectionRecordingRunning","mobileStopMediaProjectionRecording","recentRecordingPath","size","fs","stat","debug","toReadableSizeString","rimraf","dirname"],"sources":["../../../lib/commands/media-projection.js"],"sourcesContent":["import _ from 'lodash';\nimport { waitForCondition } from 'asyncbox';\nimport { util, fs, net, tempDir } from 'appium/support';\nimport path from 'path';\nimport B from 'bluebird';\nimport { SETTINGS_HELPER_PKG_ID } from '../android-helpers';\nimport moment from 'moment';\n\n\nconst commands = {};\n\n// https://github.com/appium/io.appium.settings#internal-audio--video-recording\nconst DEFAULT_EXT = '.mp4';\nconst RECORDING_STARTUP_TIMEOUT_MS = 3 * 1000;\nconst RECORDING_STOP_TIMEOUT_MS = 3 * 1000;\nconst MIN_API_LEVEL = 29;\nconst RECORDING_SERVICE_NAME = `${SETTINGS_HELPER_PKG_ID}/.recorder.RecorderService`;\nconst RECORDING_ACTIVITY_NAME = `${SETTINGS_HELPER_PKG_ID}/io.appium.settings.Settings`;\nconst RECORDING_ACTION_START = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_START`;\nconst RECORDING_ACTION_STOP = `${SETTINGS_HELPER_PKG_ID}.recording.ACTION_STOP`;\nconst RECORDINGS_ROOT = `/storage/emulated/0/Android/data/${SETTINGS_HELPER_PKG_ID}/files`;\nconst DEFAULT_FILENAME_FORMAT = 'YYYY-MM-DDTHH-mm-ss';\n\n\nasync function uploadRecordedMedia (localFile, remotePath = null, uploadOptions = {}) {\n if (_.isEmpty(remotePath)) {\n return (await util.toInMemoryBase64(localFile)).toString();\n }\n\n const {user, pass, method, headers, fileFieldName, formFields} = uploadOptions;\n const options = {\n method: method || 'PUT',\n headers,\n fileFieldName,\n formFields,\n };\n if (user && pass) {\n options.auth = {user, pass};\n }\n await net.uploadFile(localFile, remotePath, options);\n return '';\n}\n\nfunction adjustMediaExtension (name) {\n return _.toLower(name).endsWith(DEFAULT_EXT) ? name : `${name}${DEFAULT_EXT}`;\n}\n\nasync function verifyMediaProjectionRecordingIsSupported (adb) {\n const apiLevel = await adb.getApiLevel();\n if (apiLevel < MIN_API_LEVEL) {\n throw new Error(`Media projection-based recording is not available on API Level ${apiLevel}. ` +\n `Minimum required API Level is ${MIN_API_LEVEL}.`);\n }\n}\n\n\nclass MediaProjectionRecorder {\n constructor (adb) {\n this.adb = adb;\n }\n\n async isRunning () {\n const stdout = await this.adb.shell([\n 'dumpsys', 'activity', 'services', RECORDING_SERVICE_NAME\n ]);\n return stdout.includes(RECORDING_SERVICE_NAME);\n }\n\n async start (opts = {}) {\n if (await this.isRunning()) {\n return false;\n }\n\n await this.cleanup();\n const {\n filename,\n maxDurationSec,\n priority,\n resolution,\n } = opts;\n const args = [\n 'am', 'start',\n '-n', RECORDING_ACTIVITY_NAME,\n '-a', RECORDING_ACTION_START,\n ];\n if (filename) {\n args.push('--es', 'filename', filename);\n }\n if (maxDurationSec) {\n args.push('--es', 'max_duration_sec', `${maxDurationSec}`);\n }\n if (priority) {\n args.push('--es', 'priority', priority);\n }\n if (resolution) {\n args.push('--es', 'resolution', resolution);\n }\n await this.adb.shell(args);\n await new B((resolve, reject) => {\n setTimeout(async () => {\n if (!await this.isRunning()) {\n return reject(new Error(\n `The media projection recording is not running after ${RECORDING_STARTUP_TIMEOUT_MS}ms. ` +\n `Please check the logcat output for more details.`\n ));\n }\n resolve();\n }, RECORDING_STARTUP_TIMEOUT_MS);\n });\n return true;\n }\n\n async cleanup () {\n await this.adb.shell([`rm -f ${RECORDINGS_ROOT}/*`]);\n }\n\n async pullRecent () {\n const recordings = await this.adb.ls(RECORDINGS_ROOT, ['-tr']);\n if (_.isEmpty(recordings)) {\n return null;\n }\n\n const dstPath = path.join(await tempDir.openDir(), recordings[0]);\n await this.adb.pull(`${RECORDINGS_ROOT}/${recordings[0]}`, dstPath);\n return dstPath;\n }\n\n async stop () {\n if (!await this.isRunning()) {\n return false;\n }\n\n await this.adb.shell([\n 'am', 'start',\n '-n', RECORDING_ACTIVITY_NAME,\n '-a', RECORDING_ACTION_STOP,\n ]);\n try {\n await waitForCondition(async () => !(await this.isRunning()), {\n waitMs: RECORDING_STOP_TIMEOUT_MS,\n intervalMs: 500,\n });\n } catch (e) {\n throw new Error(\n `The attempt to stop the current media projection recording timed out after ` +\n `${RECORDING_STOP_TIMEOUT_MS}ms`\n );\n }\n return true;\n }\n}\n\n\n/**\n * @typedef {Object} StartRecordingOptions\n *\n * @property {string?} resolution Maximum supported resolution on-device (Detected\n * automatically by the app itself), which usually equals to Full HD 1920x1080 on most\n * phones however you can change it to following supported resolutions\n * as well: \"1920x1080\", \"1280x720\", \"720x480\", \"320x240\", \"176x144\".\n * @property {number?} maxDurationSec [900] Default value: 900 seconds which means\n * maximum allowed duration is 15 minute, you can increase it if your test takes\n * longer than that.\n * @property {string?} priority [high] Means recording thread priority is maximum\n * however if you face performance drops during testing with recording enabled, you\n * can reduce recording priority to \"normal\" or \"low\".\n * @property {string?} filename You can type recording video file name as you want,\n * but recording currently supports only \"mp4\" format so your filename must end with \".mp4\".\n * An invalid file name will fail to start the recording.\n * If not provided then the current timestamp will be used as file name.\n */\n\n/**\n * Record the display of a real devices running Android 10 (API level 29) and higher.\n * The screen activity is recorded to a MPEG-4 file. Audio is also recorded by default\n * (only for apps that allow it in their manifests).\n * If another recording has been already started then the command will exit silently.\n * The previously recorded video file is deleted when a new recording session is started.\n * Recording continues it is stopped explicitly or until the timeout happens.\n *\n * @param {?StartRecordingOptions} options Available options.\n * @returns {boolean} True if a new recording has successfully started.\n * @throws {Error} If recording has failed to start or is not supported on the device under test.\n */\ncommands.mobileStartMediaProjectionRecording = async function mobileStartMediaProjectionRecording (options = {}) {\n await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n const {resolution, priority, maxDurationSec, filename} = options;\n const recorder = new MediaProjectionRecorder(this.adb);\n const fname = adjustMediaExtension(filename || moment().format(DEFAULT_FILENAME_FORMAT));\n const didStart = await recorder.start({\n resolution,\n priority,\n maxDurationSec,\n filename: fname,\n });\n if (didStart) {\n this.log.info(`A new media projection recording '${fname}' has been successfully started`);\n } else {\n this.log.info('Another media projection recording is already in progress. There is nothing to start');\n }\n return didStart;\n};\n\n/**\n * Checks if a media projection-based recording is currently running.\n *\n * @returns {boolean} True if a recording is in progress.\n * @throws {Error} If a recording is not supported on the device under test.\n */\ncommands.mobileIsMediaProjectionRecordingRunning = async function mobileIsMediaProjectionRecordingRunning () {\n await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n const recorder = new MediaProjectionRecorder(this.adb);\n return await recorder.isRunning();\n};\n\n/**\n * @typedef {Object} StopRecordingOptions\n *\n * @property {string?} remotePath The path to the remote location, where the resulting video should be uploaded.\n * The following protocols are supported: http/https, ftp.\n * Null or empty string value (the default setting) means the content of resulting\n * file should be encoded as Base64 and passed as the endpoont response value.\n * An exception will be thrown if the generated media file is too big to\n * fit into the available process memory.\n * @property {string?} user The name of the user for the remote authentication.\n * @property {string?} pass The password for the remote authentication.\n * @property {string?} method The http multipart upload method name. The 'PUT' one is used by default.\n * @property {Object?} headers Additional headers mapping for multipart http(s) uploads\n * @property {string?} fileFieldName [file] The name of the form field, where the file content BLOB should be stored for\n * http(s) uploads\n * @property {Object|Array<Pair>?} formFields Additional form fields for multipart http(s) uploads\n */\n\n/**\n * Stop a media projection-based recording.\n * If no recording has been started before then an error is thrown.\n * If the recording has been already finished before this API has been called\n * then the most recent recorded file is returned.\n *\n * @param {?StopRecordingOptions} options Available options.\n * @returns {string} Base64-encoded content of the recorded media file if 'remotePath'\n * parameter is falsy or an empty string.\n * @throws {Error} If there was an error while stopping a recording,\n * fetching the content of the remote media file,\n * or if a recording is not supported on the device under test.\n */\ncommands.mobileStopMediaProjectionRecording = async function mobileStopMediaProjectionRecording (options = {}) {\n await verifyMediaProjectionRecordingIsSupported(this.adb);\n\n const recorder = new MediaProjectionRecorder(this.adb);\n if (await recorder.stop()) {\n this.log.info('Successfully stopped a media projection recording. Pulling the recorded media');\n } else {\n this.log.info('Media projection recording is not running. There is nothing to stop');\n }\n const recentRecordingPath = await recorder.pullRecent();\n if (!recentRecordingPath) {\n throw new Error(`No recent media projection recording have been found. Did you start any?`);\n }\n\n const {remotePath} = options;\n if (_.isEmpty(remotePath)) {\n const {size} = await fs.stat(recentRecordingPath);\n this.log.debug(`The size of the resulting media projection recording is ${util.toReadableSizeString(size)}`);\n }\n try {\n return await uploadRecordedMedia(recentRecordingPath, remotePath, options);\n } finally {\n await fs.rimraf(path.dirname(recentRecordingPath));\n }\n};\n\n\nexport { commands };\nexport default commands;\n"],"mappings":"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"}
|