webdriverio-automation
Version:
WebdriverIO-Automation android ios project
406 lines (301 loc) • 46 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
var _logger = _interopRequireDefault(require("./logger"));
var _lodash = _interopRequireDefault(require("lodash"));
var _bplistCreator = _interopRequireDefault(require("bplist-creator"));
var _bplistParser = _interopRequireDefault(require("bplist-parser"));
var _bufferpack = _interopRequireDefault(require("bufferpack"));
var _bluebird = _interopRequireDefault(require("bluebird"));
var _remoteDebugger = require("./remote-debugger");
var _uuidJs = _interopRequireDefault(require("uuid-js"));
var _net = _interopRequireDefault(require("net"));
var _remoteDebuggerMessageHandler = _interopRequireDefault(require("./remote-debugger-message-handler"));
var _remoteMessages = _interopRequireDefault(require("./remote-messages"));
class RemoteDebuggerRpcClient {
constructor(opts = {}) {
const _opts$host = opts.host,
host = _opts$host === void 0 ? '::1' : _opts$host,
_opts$port = opts.port,
port = _opts$port === void 0 ? _remoteDebugger.REMOTE_DEBUGGER_PORT : _opts$port,
socketPath = opts.socketPath,
_opts$specialMessageH = opts.specialMessageHandlers,
specialMessageHandlers = _opts$specialMessageH === void 0 ? {} : _opts$specialMessageH,
messageProxy = opts.messageProxy;
this.host = host;
this.port = port;
this.socketPath = socketPath;
this.messageProxy = messageProxy;
this.socket = null;
this.connected = false;
this.connId = _uuidJs.default.create().toString();
this.senderId = _uuidJs.default.create().toString();
this.curMsgId = 0;
this.received = Buffer.alloc(0);
this.readPos = 0;
this.specialMessageHandlers = specialMessageHandlers;
this.messageHandler = null;
}
connect() {
var _this = this;
return (0, _asyncToGenerator2.default)(function* () {
_this.messageHandler = new _remoteDebuggerMessageHandler.default(_this.specialMessageHandlers);
if (_this.socketPath) {
if (_this.messageProxy) {
_logger.default.debug(`Connecting to remote debugger via proxy through unix domain socket: '${_this.messageProxy}'`);
_this.socket = _net.default.connect(_this.messageProxy);
_this.socket.once('connect', () => {
_logger.default.debug(`Forwarding the actual web inspector socket to the proxy: '${_this.socketPath}'`);
_this.socket.write(JSON.stringify({
socketPath: _this.socketPath
}));
});
} else {
_logger.default.debug(`Connecting to remote debugger through unix domain socket: '${_this.socketPath}'`);
_this.socket = _net.default.connect(_this.socketPath);
}
} else {
if (_this.messageProxy) {
_this.port = _this.messageProxy;
}
_logger.default.debug(`Connecting to remote debugger ${_this.messageProxy ? 'via proxy ' : ''}through TCP: ${_this.host}:${_this.port}`);
_this.socket = new _net.default.Socket({
type: 'tcp6'
});
_this.socket.connect(_this.port, _this.host);
}
_this.socket.setNoDelay(true);
_this.socket.on('close', () => {
if (_this.connected) {
_logger.default.debug('Debugger socket disconnected');
}
_this.connected = false;
_this.socket = null;
});
_this.socket.on('end', () => {
_this.connected = false;
});
_this.socket.on('data', _this.receive.bind(_this));
return yield new _bluebird.default((resolve, reject) => {
_this.socket.on('connect', () => {
_logger.default.debug(`Debugger socket connected`);
_this.connected = true;
resolve();
});
_this.socket.on('error', err => {
if (_this.connected) {
_logger.default.error(`Socket error: ${err.message}`);
_this.connected = false;
}
reject(err);
});
});
})();
}
disconnect() {
var _this2 = this;
return (0, _asyncToGenerator2.default)(function* () {
if (_this2.isConnected()) {
_logger.default.debug('Disconnecting from remote debugger');
_this2.socket.destroy();
}
_this2.connected = false;
})();
}
isConnected() {
return this.connected;
}
setSpecialMessageHandler(key, errorHandler, handler) {
this.messageHandler.setSpecialMessageHandler(key, errorHandler, handler);
}
getSpecialMessageHandler(key) {
return this.messageHandler.getSpecialMessageHandler(key);
}
setDataMessageHandler(key, errorHandler, handler) {
this.messageHandler.setDataMessageHandler(key, errorHandler, handler);
}
allowNavigationWithoutReload(allow = true) {
this.messageHandler.allowNavigationWithoutReload(allow);
}
selectApp(appIdKey, applicationConnectedHandler) {
var _this3 = this;
return (0, _asyncToGenerator2.default)(function* () {
return yield new _bluebird.default((resolve, reject) => {
let onAppChange = dict => {
let oldAppIdKey = dict.WIRHostApplicationIdentifierKey;
let correctAppIdKey = dict.WIRApplicationIdentifierKey;
if (oldAppIdKey && correctAppIdKey !== oldAppIdKey) {
_logger.default.debug(`We were notified we might have connected to the wrong app. ` + `Using id ${correctAppIdKey} instead of ${oldAppIdKey}`);
}
applicationConnectedHandler(dict);
reject(new Error('New application has connected'));
};
_this3.setSpecialMessageHandler('_rpc_applicationConnected:', reject, onAppChange);
return (0, _asyncToGenerator2.default)(function* () {
let _ref2 = yield _this3.send('connectToApp', {
appIdKey
}),
_ref3 = (0, _slicedToArray2.default)(_ref2, 2),
connectedAppIdKey = _ref3[0],
pageDict = _ref3[1];
if (_lodash.default.isEmpty(pageDict)) {
let msg = 'Empty page dictionary received';
_logger.default.debug(msg);
reject(new Error(msg));
} else {
resolve([connectedAppIdKey, pageDict]);
}
})();
}).finally(() => {
_this3.setSpecialMessageHandler('_rpc_applicationConnected:', null, applicationConnectedHandler);
});
})();
}
send(command, opts = {}) {
var _this4 = this;
return (0, _asyncToGenerator2.default)(function* () {
let onSocketError;
return new _bluebird.default((resolve, reject) => {
opts = _lodash.default.defaults({
connId: _this4.connId,
senderId: _this4.senderId
}, opts);
let data = (0, _remoteMessages.default)(command, opts);
let socketCb = _lodash.default.noop;
onSocketError = exception => {
if (_this4.connected) {
_logger.default.error(`Socket error: ${exception.message}`);
}
reject(exception);
};
_this4.socket.on('error', onSocketError);
if (_this4.messageHandler.hasSpecialMessageHandler(data.__selector)) {
let specialMessageHandler = _this4.getSpecialMessageHandler(data.__selector);
_this4.setSpecialMessageHandler(data.__selector, reject, function (...args) {
_logger.default.debug(`Received response from socket send: '${_lodash.default.truncate(JSON.stringify(args), {
length: 50
})}'`);
specialMessageHandler(...args);
if (this.messageHandler.hasSpecialMessageHandler(data.__selector)) {
this.setSpecialMessageHandler(data.__selector, null, specialMessageHandler);
}
resolve(args);
}.bind(_this4));
} else if (data.__argument && data.__argument.WIRSocketDataKey) {
_this4.curMsgId++;
const errorHandler = function errorHandler(err) {
const msg = `Remote debugger error with code '${err.code}': ${err.message}`;
reject(new Error(msg));
};
_this4.setDataMessageHandler(_this4.curMsgId.toString(), errorHandler, value => {
const msg = _lodash.default.truncate(_lodash.default.isString(value) ? value : JSON.stringify(value), {
length: 50
});
_logger.default.debug(`Received data response from socket send: '${msg}'`);
_logger.default.debug(`Original command: ${command}`);
resolve(value);
});
data.__argument.WIRSocketDataKey.id = _this4.curMsgId;
data.__argument.WIRSocketDataKey = Buffer.from(JSON.stringify(data.__argument.WIRSocketDataKey));
} else {
socketCb = resolve;
}
_logger.default.debug(`Sending '${data.__selector}' message to remote debugger`);
let plist;
try {
plist = (0, _bplistCreator.default)(data);
} catch (e) {
let msg = `Could not create binary plist from data: ${e.message}`;
_logger.default.error(msg);
return reject(new Error(msg));
}
if (_this4.socket && _this4.connected) {
_this4.socket.cork();
try {
_this4.socket.write(_bufferpack.default.pack('L', [plist.length]));
_this4.socket.write(plist, socketCb);
} finally {
_this4.socket.uncork();
}
} else {
let msg = 'Attempted to write data to socket after it was closed!';
_logger.default.error(msg);
reject(new Error(msg));
}
}).finally(() => {
_this4.socket.removeListener('error', onSocketError);
});
})();
}
receive(data) {
this.received = Buffer.concat([this.received, data]);
let dataLeftOver = true;
while (dataLeftOver) {
let oldReadPos = this.readPos;
let prefix = this.received.slice(this.readPos, this.readPos + 4);
let msgLength;
try {
msgLength = _bufferpack.default.unpack('L', prefix)[0];
} catch (e) {
_logger.default.error(`Buffer could not unpack: ${e}`);
return;
}
this.readPos += 4;
if (this.received.length < msgLength + this.readPos) {
this.readPos = oldReadPos;
break;
}
let body = this.received.slice(this.readPos, msgLength + this.readPos);
let plist;
try {
plist = _bplistParser.default.parseBuffer(body);
} catch (e) {
_logger.default.error(`Error parsing binary plist: ${e}`);
return;
}
if (plist.length === 1) {
plist = plist[0];
}
var _arr = ['WIRMessageDataKey', 'WIRDestinationKey', 'WIRSocketDataKey'];
for (var _i = 0; _i < _arr.length; _i++) {
let key = _arr[_i];
if (!_lodash.default.isUndefined(plist[key])) {
plist[key] = plist[key].toString("utf8");
}
}
this.readPos += msgLength;
let leftOver = this.received.length - this.readPos;
if (leftOver !== 0) {
let chunk = Buffer.alloc(leftOver);
this.received.copy(chunk, 0, this.readPos);
this.received = chunk;
} else {
this.received = Buffer.alloc(0);
dataLeftOver = false;
}
this.readPos = 0;
if (plist) {
this.messageHandler.handleMessage(plist);
}
}
}
setTimelineEventHandler(timelineEventHandler) {
this.timelineEventHandler = timelineEventHandler;
this.messageHandler.setTimelineEventHandler(timelineEventHandler);
}
setConsoleLogEventHandler(consoleEventHandler) {
this.consoleEventHandler = consoleEventHandler;
this.messageHandler.setConsoleLogEventHandler(consoleEventHandler);
}
setNetworkLogEventHandler(networkEventHandler) {
this.networkEventHandler = networkEventHandler;
this.messageHandler.setNetworkEventHandler(networkEventHandler);
}
}
exports.default = RemoteDebuggerRpcClient;require('source-map-support').install();
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/remote-debugger-rpc-client.js"],"names":["RemoteDebuggerRpcClient","constructor","opts","host","port","REMOTE_DEBUGGER_PORT","socketPath","specialMessageHandlers","messageProxy","socket","connected","connId","UUID","create","toString","senderId","curMsgId","received","Buffer","alloc","readPos","messageHandler","connect","RpcMessageHandler","log","debug","net","once","write","JSON","stringify","Socket","type","setNoDelay","on","receive","bind","Promise","resolve","reject","err","error","message","disconnect","isConnected","destroy","setSpecialMessageHandler","key","errorHandler","handler","getSpecialMessageHandler","setDataMessageHandler","allowNavigationWithoutReload","allow","selectApp","appIdKey","applicationConnectedHandler","onAppChange","dict","oldAppIdKey","WIRHostApplicationIdentifierKey","correctAppIdKey","WIRApplicationIdentifierKey","Error","send","connectedAppIdKey","pageDict","_","isEmpty","msg","finally","command","onSocketError","defaults","data","socketCb","noop","exception","hasSpecialMessageHandler","__selector","specialMessageHandler","args","truncate","length","__argument","WIRSocketDataKey","code","value","isString","id","from","plist","e","cork","bufferpack","pack","uncork","removeListener","concat","dataLeftOver","oldReadPos","prefix","slice","msgLength","unpack","body","bplistParser","parseBuffer","isUndefined","leftOver","chunk","copy","handleMessage","setTimelineEventHandler","timelineEventHandler","setConsoleLogEventHandler","consoleEventHandler","setNetworkLogEventHandler","networkEventHandler","setNetworkEventHandler"],"mappings":";;;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAGe,MAAMA,uBAAN,CAA8B;AAC3CC,EAAAA,WAAW,CAAEC,IAAI,GAAG,EAAT,EAAa;AAAA,uBAOlBA,IAPkB,CAEpBC,IAFoB;AAAA,UAEpBA,IAFoB,2BAEb,KAFa;AAAA,uBAOlBD,IAPkB,CAGpBE,IAHoB;AAAA,UAGpBA,IAHoB,2BAGbC,oCAHa;AAAA,UAIpBC,UAJoB,GAOlBJ,IAPkB,CAIpBI,UAJoB;AAAA,kCAOlBJ,IAPkB,CAKpBK,sBALoB;AAAA,UAKpBA,sBALoB,sCAKK,EALL;AAAA,UAMpBC,YANoB,GAOlBN,IAPkB,CAMpBM,YANoB;AAUtB,SAAKL,IAAL,GAAYA,IAAZ;AACA,SAAKC,IAAL,GAAYA,IAAZ;AACA,SAAKE,UAAL,GAAkBA,UAAlB;AACA,SAAKE,YAAL,GAAoBA,YAApB;AAEA,SAAKC,MAAL,GAAc,IAAd;AACA,SAAKC,SAAL,GAAiB,KAAjB;AACA,SAAKC,MAAL,GAAcC,gBAAKC,MAAL,GAAcC,QAAd,EAAd;AACA,SAAKC,QAAL,GAAgBH,gBAAKC,MAAL,GAAcC,QAAd,EAAhB;AACA,SAAKE,QAAL,GAAgB,CAAhB;AACA,SAAKC,QAAL,GAAgBC,MAAM,CAACC,KAAP,CAAa,CAAb,CAAhB;AACA,SAAKC,OAAL,GAAe,CAAf;AAGA,SAAKb,sBAAL,GAA8BA,sBAA9B;AACA,SAAKc,cAAL,GAAsB,IAAtB;AACD;;AAEKC,EAAAA,OAAN,GAAiB;AAAA;;AAAA;AACf,MAAA,KAAI,CAACD,cAAL,GAAsB,IAAIE,qCAAJ,CAAsB,KAAI,CAAChB,sBAA3B,CAAtB;;AAGA,UAAI,KAAI,CAACD,UAAT,EAAqB;AACnB,YAAI,KAAI,CAACE,YAAT,EAAuB;AAErBgB,0BAAIC,KAAJ,CAAW,wEAAuE,KAAI,CAACjB,YAAa,GAApG;;AACA,UAAA,KAAI,CAACC,MAAL,GAAciB,aAAIJ,OAAJ,CAAY,KAAI,CAACd,YAAjB,CAAd;;AAGA,UAAA,KAAI,CAACC,MAAL,CAAYkB,IAAZ,CAAiB,SAAjB,EAA4B,MAAM;AAChCH,4BAAIC,KAAJ,CAAW,6DAA4D,KAAI,CAACnB,UAAW,GAAvF;;AACA,YAAA,KAAI,CAACG,MAAL,CAAYmB,KAAZ,CAAkBC,IAAI,CAACC,SAAL,CAAe;AAACxB,cAAAA,UAAU,EAAE,KAAI,CAACA;AAAlB,aAAf,CAAlB;AACD,WAHD;AAKD,SAXD,MAWO;AAELkB,0BAAIC,KAAJ,CAAW,8DAA6D,KAAI,CAACnB,UAAW,GAAxF;;AACA,UAAA,KAAI,CAACG,MAAL,GAAciB,aAAIJ,OAAJ,CAAY,KAAI,CAAChB,UAAjB,CAAd;AACD;AACF,OAjBD,MAiBO;AACL,YAAI,KAAI,CAACE,YAAT,EAAuB;AAErB,UAAA,KAAI,CAACJ,IAAL,GAAY,KAAI,CAACI,YAAjB;AACD;;AAGDgB,wBAAIC,KAAJ,CAAW,iCAAgC,KAAI,CAACjB,YAAL,GAAoB,YAApB,GAAmC,EAAG,gBAAe,KAAI,CAACL,IAAK,IAAG,KAAI,CAACC,IAAK,EAAvH;;AACA,QAAA,KAAI,CAACK,MAAL,GAAc,IAAIiB,aAAIK,MAAR,CAAe;AAACC,UAAAA,IAAI,EAAE;AAAP,SAAf,CAAd;;AACA,QAAA,KAAI,CAACvB,MAAL,CAAYa,OAAZ,CAAoB,KAAI,CAAClB,IAAzB,EAA+B,KAAI,CAACD,IAApC;AACD;;AAED,MAAA,KAAI,CAACM,MAAL,CAAYwB,UAAZ,CAAuB,IAAvB;;AACA,MAAA,KAAI,CAACxB,MAAL,CAAYyB,EAAZ,CAAe,OAAf,EAAwB,MAAM;AAC5B,YAAI,KAAI,CAACxB,SAAT,EAAoB;AAClBc,0BAAIC,KAAJ,CAAU,8BAAV;AACD;;AACD,QAAA,KAAI,CAACf,SAAL,GAAiB,KAAjB;AACA,QAAA,KAAI,CAACD,MAAL,GAAc,IAAd;AACD,OAND;;AAOA,MAAA,KAAI,CAACA,MAAL,CAAYyB,EAAZ,CAAe,KAAf,EAAsB,MAAM;AAC1B,QAAA,KAAI,CAACxB,SAAL,GAAiB,KAAjB;AACD,OAFD;;AAGA,MAAA,KAAI,CAACD,MAAL,CAAYyB,EAAZ,CAAe,MAAf,EAAuB,KAAI,CAACC,OAAL,CAAaC,IAAb,CAAkB,KAAlB,CAAvB;;AAGA,mBAAa,IAAIC,iBAAJ,CAAY,CAACC,OAAD,EAAUC,MAAV,KAAqB;AAE5C,QAAA,KAAI,CAAC9B,MAAL,CAAYyB,EAAZ,CAAe,SAAf,EAA0B,MAAM;AAC9BV,0BAAIC,KAAJ,CAAW,2BAAX;;AACA,UAAA,KAAI,CAACf,SAAL,GAAiB,IAAjB;AAEA4B,UAAAA,OAAO;AACR,SALD;;AAMA,QAAA,KAAI,CAAC7B,MAAL,CAAYyB,EAAZ,CAAe,OAAf,EAAyBM,GAAD,IAAS;AAC/B,cAAI,KAAI,CAAC9B,SAAT,EAAoB;AAClBc,4BAAIiB,KAAJ,CAAW,iBAAgBD,GAAG,CAACE,OAAQ,EAAvC;;AACA,YAAA,KAAI,CAAChC,SAAL,GAAiB,KAAjB;AACD;;AAGD6B,UAAAA,MAAM,CAACC,GAAD,CAAN;AACD,SARD;AASD,OAjBY,CAAb;AA/Ce;AAiEhB;;AAEKG,EAAAA,UAAN,GAAoB;AAAA;;AAAA;AAClB,UAAI,MAAI,CAACC,WAAL,EAAJ,EAAwB;AACtBpB,wBAAIC,KAAJ,CAAU,oCAAV;;AACA,QAAA,MAAI,CAAChB,MAAL,CAAYoC,OAAZ;AACD;;AACD,MAAA,MAAI,CAACnC,SAAL,GAAiB,KAAjB;AALkB;AAMnB;;AAEDkC,EAAAA,WAAW,GAAI;AACb,WAAO,KAAKlC,SAAZ;AACD;;AAEDoC,EAAAA,wBAAwB,CAAEC,GAAF,EAAOC,YAAP,EAAqBC,OAArB,EAA8B;AACpD,SAAK5B,cAAL,CAAoByB,wBAApB,CAA6CC,GAA7C,EAAkDC,YAAlD,EAAgEC,OAAhE;AACD;;AAEDC,EAAAA,wBAAwB,CAAEH,GAAF,EAAO;AAC7B,WAAO,KAAK1B,cAAL,CAAoB6B,wBAApB,CAA6CH,GAA7C,CAAP;AACD;;AAEDI,EAAAA,qBAAqB,CAAEJ,GAAF,EAAOC,YAAP,EAAqBC,OAArB,EAA8B;AACjD,SAAK5B,cAAL,CAAoB8B,qBAApB,CAA0CJ,GAA1C,EAA+CC,YAA/C,EAA6DC,OAA7D;AACD;;AAEDG,EAAAA,4BAA4B,CAAEC,KAAK,GAAG,IAAV,EAAgB;AAC1C,SAAKhC,cAAL,CAAoB+B,4BAApB,CAAiDC,KAAjD;AACD;;AAEKC,EAAAA,SAAN,CAAiBC,QAAjB,EAA2BC,2BAA3B,EAAwD;AAAA;;AAAA;AACtD,mBAAa,IAAInB,iBAAJ,CAAY,CAACC,OAAD,EAAUC,MAAV,KAAqB;AAI5C,YAAIkB,WAAW,GAAIC,IAAD,IAAU;AAE1B,cAAIC,WAAW,GAAGD,IAAI,CAACE,+BAAvB;AACA,cAAIC,eAAe,GAAGH,IAAI,CAACI,2BAA3B;;AAIA,cAAIH,WAAW,IAAIE,eAAe,KAAKF,WAAvC,EAAoD;AAClDnC,4BAAIC,KAAJ,CAAW,6DAAD,GACC,YAAWoC,eAAgB,eAAcF,WAAY,EADhE;AAED;;AAEDH,UAAAA,2BAA2B,CAACE,IAAD,CAA3B;AACAnB,UAAAA,MAAM,CAAC,IAAIwB,KAAJ,CAAU,+BAAV,CAAD,CAAN;AACD,SAdD;;AAeA,QAAA,MAAI,CAACjB,wBAAL,CAA8B,4BAA9B,EAA4DP,MAA5D,EAAoEkB,WAApE;;AAGA,eAAO,gCAAC,aAAY;AAAA,4BACwB,MAAI,CAACO,IAAL,CAAU,cAAV,EAA0B;AAClET,YAAAA;AADkE,WAA1B,CADxB;AAAA;AAAA,cACbU,iBADa;AAAA,cACMC,QADN;;AAOlB,cAAIC,gBAAEC,OAAF,CAAUF,QAAV,CAAJ,EAAyB;AACvB,gBAAIG,GAAG,GAAG,gCAAV;;AACA7C,4BAAIC,KAAJ,CAAU4C,GAAV;;AACA9B,YAAAA,MAAM,CAAC,IAAIwB,KAAJ,CAAUM,GAAV,CAAD,CAAN;AACD,WAJD,MAIO;AACL/B,YAAAA,OAAO,CAAC,CAAC2B,iBAAD,EAAoBC,QAApB,CAAD,CAAP;AACD;AACF,SAdM,GAAP;AAeD,OArCY,EAqCVI,OArCU,CAqCF,MAAM;AAEf,QAAA,MAAI,CAACxB,wBAAL,CAA8B,4BAA9B,EAA4D,IAA5D,EAAkEU,2BAAlE;AACD,OAxCY,CAAb;AADsD;AA0CvD;;AAEKQ,EAAAA,IAAN,CAAYO,OAAZ,EAAqBrE,IAAI,GAAG,EAA5B,EAAgC;AAAA;;AAAA;AAE9B,UAAIsE,aAAJ;AAEA,aAAO,IAAInC,iBAAJ,CAAY,CAACC,OAAD,EAAUC,MAAV,KAAqB;AAKtCrC,QAAAA,IAAI,GAAGiE,gBAAEM,QAAF,CAAW;AAAC9D,UAAAA,MAAM,EAAE,MAAI,CAACA,MAAd;AAAsBI,UAAAA,QAAQ,EAAE,MAAI,CAACA;AAArC,SAAX,EAA2Db,IAA3D,CAAP;AACA,YAAIwE,IAAI,GAAG,6BAAiBH,OAAjB,EAA0BrE,IAA1B,CAAX;AAIA,YAAIyE,QAAQ,GAAGR,gBAAES,IAAjB;;AAGAJ,QAAAA,aAAa,GAAIK,SAAD,IAAe;AAC7B,cAAI,MAAI,CAACnE,SAAT,EAAoB;AAClBc,4BAAIiB,KAAJ,CAAW,iBAAgBoC,SAAS,CAACnC,OAAQ,EAA7C;AACD;;AAGDH,UAAAA,MAAM,CAACsC,SAAD,CAAN;AACD,SAPD;;AAQA,QAAA,MAAI,CAACpE,MAAL,CAAYyB,EAAZ,CAAe,OAAf,EAAwBsC,aAAxB;;AACA,YAAI,MAAI,CAACnD,cAAL,CAAoByD,wBAApB,CAA6CJ,IAAI,CAACK,UAAlD,CAAJ,EAAmE;AAGjE,cAAIC,qBAAqB,GAAG,MAAI,CAAC9B,wBAAL,CAA8BwB,IAAI,CAACK,UAAnC,CAA5B;;AACA,UAAA,MAAI,CAACjC,wBAAL,CAA8B4B,IAAI,CAACK,UAAnC,EAA+CxC,MAA/C,EAAuD,UAAU,GAAG0C,IAAb,EAAmB;AACxEzD,4BAAIC,KAAJ,CAAW,wCAAuC0C,gBAAEe,QAAF,CAAWrD,IAAI,CAACC,SAAL,CAAemD,IAAf,CAAX,EAAiC;AAACE,cAAAA,MAAM,EAAE;AAAT,aAAjC,CAA+C,GAAjG;;AAGAH,YAAAA,qBAAqB,CAAC,GAAGC,IAAJ,CAArB;;AACA,gBAAI,KAAK5D,cAAL,CAAoByD,wBAApB,CAA6CJ,IAAI,CAACK,UAAlD,CAAJ,EAAmE;AAEjE,mBAAKjC,wBAAL,CAA8B4B,IAAI,CAACK,UAAnC,EAA+C,IAA/C,EAAqDC,qBAArD;AACD;;AAED1C,YAAAA,OAAO,CAAC2C,IAAD,CAAP;AACD,WAXsD,CAWrD7C,IAXqD,CAWhD,MAXgD,CAAvD;AAYD,SAhBD,MAgBO,IAAIsC,IAAI,CAACU,UAAL,IAAmBV,IAAI,CAACU,UAAL,CAAgBC,gBAAvC,EAAyD;AAG9D,UAAA,MAAI,CAACrE,QAAL;;AAEA,gBAAMgC,YAAY,GAAG,SAAfA,YAAe,CAAUR,GAAV,EAAe;AAClC,kBAAM6B,GAAG,GAAI,oCAAmC7B,GAAG,CAAC8C,IAAK,MAAK9C,GAAG,CAACE,OAAQ,EAA1E;AACAH,YAAAA,MAAM,CAAC,IAAIwB,KAAJ,CAAUM,GAAV,CAAD,CAAN;AACD,WAHD;;AAKA,UAAA,MAAI,CAAClB,qBAAL,CAA2B,MAAI,CAACnC,QAAL,CAAcF,QAAd,EAA3B,EAAqDkC,YAArD,EAAoEuC,KAAD,IAAW;AAC5E,kBAAMlB,GAAG,GAAGF,gBAAEe,QAAF,CAAWf,gBAAEqB,QAAF,CAAWD,KAAX,IAAoBA,KAApB,GAA4B1D,IAAI,CAACC,SAAL,CAAeyD,KAAf,CAAvC,EAA8D;AAACJ,cAAAA,MAAM,EAAE;AAAT,aAA9D,CAAZ;;AACA3D,4BAAIC,KAAJ,CAAW,6CAA4C4C,GAAI,GAA3D;;AACA7C,4BAAIC,KAAJ,CAAW,qBAAoB8C,OAAQ,EAAvC;;AACAjC,YAAAA,OAAO,CAACiD,KAAD,CAAP;AACD,WALD;;AAMAb,UAAAA,IAAI,CAACU,UAAL,CAAgBC,gBAAhB,CAAiCI,EAAjC,GAAsC,MAAI,CAACzE,QAA3C;AACA0D,UAAAA,IAAI,CAACU,UAAL,CAAgBC,gBAAhB,GACInE,MAAM,CAACwE,IAAP,CAAY7D,IAAI,CAACC,SAAL,CAAe4C,IAAI,CAACU,UAAL,CAAgBC,gBAA/B,CAAZ,CADJ;AAED,SAnBM,MAmBA;AAGLV,UAAAA,QAAQ,GAAGrC,OAAX;AACD;;AAEDd,wBAAIC,KAAJ,CAAW,YAAWiD,IAAI,CAACK,UAAW,8BAAtC;;AAGA,YAAIY,KAAJ;;AACA,YAAI;AACFA,UAAAA,KAAK,GAAG,4BAAajB,IAAb,CAAR;AACD,SAFD,CAEE,OAAOkB,CAAP,EAAU;AACV,cAAIvB,GAAG,GAAI,4CAA2CuB,CAAC,CAAClD,OAAQ,EAAhE;;AACAlB,0BAAIiB,KAAJ,CAAU4B,GAAV;;AACA,iBAAO9B,MAAM,CAAC,IAAIwB,KAAJ,CAAUM,GAAV,CAAD,CAAb;AACD;;AAED,YAAI,MAAI,CAAC5D,MAAL,IAAe,MAAI,CAACC,SAAxB,EAAmC;AAIjC,UAAA,MAAI,CAACD,MAAL,CAAYoF,IAAZ;;AACA,cAAI;AACF,YAAA,MAAI,CAACpF,MAAL,CAAYmB,KAAZ,CAAkBkE,oBAAWC,IAAX,CAAgB,GAAhB,EAAqB,CAACJ,KAAK,CAACR,MAAP,CAArB,CAAlB;;AACA,YAAA,MAAI,CAAC1E,MAAL,CAAYmB,KAAZ,CAAkB+D,KAAlB,EAAyBhB,QAAzB;AACD,WAHD,SAGU;AACR,YAAA,MAAI,CAAClE,MAAL,CAAYuF,MAAZ;AACD;AACF,SAXD,MAWO;AACL,cAAI3B,GAAG,GAAG,wDAAV;;AACA7C,0BAAIiB,KAAJ,CAAU4B,GAAV;;AACA9B,UAAAA,MAAM,CAAC,IAAIwB,KAAJ,CAAUM,GAAV,CAAD,CAAN;AACD;AACF,OA3FM,EA4FNC,OA5FM,CA4FE,MAAM;AAEb,QAAA,MAAI,CAAC7D,MAAL,CAAYwF,cAAZ,CAA2B,OAA3B,EAAoCzB,aAApC;AACD,OA/FM,CAAP;AAJ8B;AAoG/B;;AAEDrC,EAAAA,OAAO,CAAEuC,IAAF,EAAQ;AAEb,SAAKzD,QAAL,GAAgBC,MAAM,CAACgF,MAAP,CAAc,CAAC,KAAKjF,QAAN,EAAgByD,IAAhB,CAAd,CAAhB;AACA,QAAIyB,YAAY,GAAG,IAAnB;;AAGA,WAAOA,YAAP,EAAqB;AAEnB,UAAIC,UAAU,GAAG,KAAKhF,OAAtB;AAIA,UAAIiF,MAAM,GAAG,KAAKpF,QAAL,CAAcqF,KAAd,CAAoB,KAAKlF,OAAzB,EAAkC,KAAKA,OAAL,GAAe,CAAjD,CAAb;AAEA,UAAImF,SAAJ;;AACA,UAAI;AACFA,QAAAA,SAAS,GAAGT,oBAAWU,MAAX,CAAkB,GAAlB,EAAuBH,MAAvB,EAA+B,CAA/B,CAAZ;AACD,OAFD,CAEE,OAAOT,CAAP,EAAU;AACVpE,wBAAIiB,KAAJ,CAAW,4BAA2BmD,CAAE,EAAxC;;AACA;AACD;;AAGD,WAAKxE,OAAL,IAAgB,CAAhB;;AAIA,UAAI,KAAKH,QAAL,CAAckE,MAAd,GAAuBoB,SAAS,GAAG,KAAKnF,OAA5C,EAAqD;AACnD,aAAKA,OAAL,GAAegF,UAAf;AACA;AACD;;AAGD,UAAIK,IAAI,GAAG,KAAKxF,QAAL,CAAcqF,KAAd,CAAoB,KAAKlF,OAAzB,EAAkCmF,SAAS,GAAG,KAAKnF,OAAnD,CAAX;AAGA,UAAIuE,KAAJ;;AACA,UAAI;AACFA,QAAAA,KAAK,GAAGe,sBAAaC,WAAb,CAAyBF,IAAzB,CAAR;AACD,OAFD,CAEE,OAAOb,CAAP,EAAU;AACVpE,wBAAIiB,KAAJ,CAAW,+BAA8BmD,CAAE,EAA3C;;AACA;AACD;;AAGD,UAAID,KAAK,CAACR,MAAN,KAAiB,CAArB,EAAwB;AACtBQ,QAAAA,KAAK,GAAGA,KAAK,CAAC,CAAD,CAAb;AACD;;AAzCkB,iBA2CH,CAAC,mBAAD,EAAsB,mBAAtB,EAA2C,kBAA3C,CA3CG;;AA2CnB,+CAAgF;AAA3E,YAAI5C,GAAG,WAAP;;AACH,YAAI,CAACoB,gBAAEyC,WAAF,CAAcjB,KAAK,CAAC5C,GAAD,CAAnB,CAAL,EAAgC;AAC9B4C,UAAAA,KAAK,CAAC5C,GAAD,CAAL,GAAa4C,KAAK,CAAC5C,GAAD,CAAL,CAAWjC,QAAX,CAAoB,MAApB,CAAb;AACD;AACF;;AAGD,WAAKM,OAAL,IAAgBmF,SAAhB;AAGA,UAAIM,QAAQ,GAAG,KAAK5F,QAAL,CAAckE,MAAd,GAAuB,KAAK/D,OAA3C;;AAGA,UAAIyF,QAAQ,KAAK,CAAjB,EAAoB;AAElB,YAAIC,KAAK,GAAG5F,MAAM,CAACC,KAAP,CAAa0F,QAAb,CAAZ;AACA,aAAK5F,QAAL,CAAc8F,IAAd,CAAmBD,KAAnB,EAA0B,CAA1B,EAA6B,KAAK1F,OAAlC;AACA,aAAKH,QAAL,GAAgB6F,KAAhB;AACD,OALD,MAKO;AAEL,aAAK7F,QAAL,GAAgBC,MAAM,CAACC,KAAP,CAAa,CAAb,CAAhB;AACAgF,QAAAA,YAAY,GAAG,KAAf;AACD;;AAGD,WAAK/E,OAAL,GAAe,CAAf;;AAGA,UAAIuE,KAAJ,EAAW;AACT,aAAKtE,cAAL,CAAoB2F,aAApB,CAAkCrB,KAAlC;AACD;AACF;AACF;;AAEDsB,EAAAA,uBAAuB,CAAEC,oBAAF,EAAwB;AAC7C,SAAKA,oBAAL,GAA4BA,oBAA5B;AACA,SAAK7F,cAAL,CAAoB4F,uBAApB,CAA4CC,oBAA5C;AACD;;AAEDC,EAAAA,yBAAyB,CAAEC,mBAAF,EAAuB;AAC9C,SAAKA,mBAAL,GAA2BA,mBAA3B;AACA,SAAK/F,cAAL,CAAoB8F,yBAApB,CAA8CC,mBAA9C;AACD;;AAEDC,EAAAA,yBAAyB,CAAEC,mBAAF,EAAuB;AAC9C,SAAKA,mBAAL,GAA2BA,mBAA3B;AACA,SAAKjG,cAAL,CAAoBkG,sBAApB,CAA2CD,mBAA3C;AACD;;AA9W0C","sourcesContent":["import log from './logger';\nimport _ from 'lodash';\nimport bplistCreate from 'bplist-creator';\nimport bplistParser from 'bplist-parser';\nimport bufferpack from 'bufferpack';\nimport Promise from 'bluebird';\nimport { REMOTE_DEBUGGER_PORT } from './remote-debugger';\nimport UUID from 'uuid-js';\nimport net from 'net';\nimport RpcMessageHandler from './remote-debugger-message-handler';\nimport getRemoteCommand from './remote-messages';\n\n\nexport default class RemoteDebuggerRpcClient {\n  constructor (opts = {}) {\n    const {\n      host = '::1',\n      port = REMOTE_DEBUGGER_PORT,\n      socketPath,\n      specialMessageHandlers = {},\n      messageProxy,\n    } = opts;\n\n    // host/port config for TCP communication, socketPath for unix domain sockets\n    this.host = host;\n    this.port = port;\n    this.socketPath = socketPath;\n    this.messageProxy = messageProxy;\n\n    this.socket = null;\n    this.connected = false;\n    this.connId = UUID.create().toString();\n    this.senderId = UUID.create().toString();\n    this.curMsgId = 0;\n    this.received = Buffer.alloc(0);\n    this.readPos = 0;\n\n    // message handlers\n    this.specialMessageHandlers = specialMessageHandlers;\n    this.messageHandler = null;\n  }\n\n  async connect () {\n    this.messageHandler = new RpcMessageHandler(this.specialMessageHandlers);\n\n    // create socket and handle its messages\n    if (this.socketPath) {\n      if (this.messageProxy) {\n        // unix domain socket via proxy\n        log.debug(`Connecting to remote debugger via proxy through unix domain socket: '${this.messageProxy}'`);\n        this.socket = net.connect(this.messageProxy);\n\n        // Forward the actual socketPath to the proxy\n        this.socket.once('connect', () => {\n          log.debug(`Forwarding the actual web inspector socket to the proxy: '${this.socketPath}'`);\n          this.socket.write(JSON.stringify({socketPath: this.socketPath}));\n        });\n\n      } else {\n        // unix domain socket\n        log.debug(`Connecting to remote debugger through unix domain socket: '${this.socketPath}'`);\n        this.socket = net.connect(this.socketPath);\n      }\n    } else {\n      if (this.messageProxy) {\n        // connect to the proxy instead of the remote debugger directly\n        this.port = this.messageProxy;\n      }\n\n      // tcp socket\n      log.debug(`Connecting to remote debugger ${this.messageProxy ? 'via proxy ' : ''}through TCP: ${this.host}:${this.port}`);\n      this.socket = new net.Socket({type: 'tcp6'});\n      this.socket.connect(this.port, this.host);\n    }\n\n    this.socket.setNoDelay(true);\n    this.socket.on('close', () => {\n      if (this.connected) {\n        log.debug('Debugger socket disconnected');\n      }\n      this.connected = false;\n      this.socket = null;\n    });\n    this.socket.on('end', () => {\n      this.connected = false;\n    });\n    this.socket.on('data', this.receive.bind(this));\n\n    // connect the socket\n    return await new Promise((resolve, reject) => {\n      // only resolve this function when we are actually connected\n      this.socket.on('connect', () => {\n        log.debug(`Debugger socket connected`);\n        this.connected = true;\n\n        resolve();\n      });\n      this.socket.on('error', (err) => {\n        if (this.connected) {\n          log.error(`Socket error: ${err.message}`);\n          this.connected = false;\n        }\n\n        // the connection was refused, so reject the connect promise\n        reject(err);\n      });\n    });\n  }\n\n  async disconnect () { // eslint-disable-line require-await\n    if (this.isConnected()) {\n      log.debug('Disconnecting from remote debugger');\n      this.socket.destroy();\n    }\n    this.connected = false;\n  }\n\n  isConnected () {\n    return this.connected;\n  }\n\n  setSpecialMessageHandler (key, errorHandler, handler) {\n    this.messageHandler.setSpecialMessageHandler(key, errorHandler, handler);\n  }\n\n  getSpecialMessageHandler (key) {\n    return this.messageHandler.getSpecialMessageHandler(key);\n  }\n\n  setDataMessageHandler (key, errorHandler, handler) {\n    this.messageHandler.setDataMessageHandler(key, errorHandler, handler);\n  }\n\n  allowNavigationWithoutReload (allow = true) {\n    this.messageHandler.allowNavigationWithoutReload(allow);\n  }\n\n  async selectApp (appIdKey, applicationConnectedHandler) {\n    return await new Promise((resolve, reject) => {\n      // local callback, temporarily added as callback to\n      // `_rpc_applicationConnected:` remote debugger response\n      // to handle the initial connection\n      let onAppChange = (dict) => {\n        // from the dictionary returned, get the ids\n        let oldAppIdKey = dict.WIRHostApplicationIdentifierKey;\n        let correctAppIdKey = dict.WIRApplicationIdentifierKey;\n\n        // if this is a report of a proxy redirect from the remote debugger\n        // we want to update our dictionary and get a new app id\n        if (oldAppIdKey && correctAppIdKey !== oldAppIdKey) {\n          log.debug(`We were notified we might have connected to the wrong app. ` +\n                    `Using id ${correctAppIdKey} instead of ${oldAppIdKey}`);\n        }\n\n        applicationConnectedHandler(dict);\n        reject(new Error('New application has connected'));\n      };\n      this.setSpecialMessageHandler('_rpc_applicationConnected:', reject, onAppChange);\n\n      // do the actual connecting to the app\n      return (async () => {\n        let [connectedAppIdKey, pageDict] = await this.send('connectToApp', {\n          appIdKey\n        });\n\n        // sometimes the connect logic happens, but with an empty dictionary\n        // which leads to the remote debugger getting disconnected, and into a loop\n        if (_.isEmpty(pageDict)) {\n          let msg = 'Empty page dictionary received';\n          log.debug(msg);\n          reject(new Error(msg));\n        } else {\n          resolve([connectedAppIdKey, pageDict]);\n        }\n      })();\n    }).finally(() => {\n      // no matter what, we want to restore the handler that was changed.\n      this.setSpecialMessageHandler('_rpc_applicationConnected:', null, applicationConnectedHandler);\n    });\n  }\n\n  async send (command, opts = {}) { // eslint-disable-line require-await\n    // error listener, which needs to be removed after the promise is resolved\n    let onSocketError;\n\n    return new Promise((resolve, reject) => {\n      // promise to be resolved whenever remote debugger\n      // replies to our request\n\n      // retrieve the correct command to send\n      opts = _.defaults({connId: this.connId, senderId: this.senderId}, opts);\n      let data = getRemoteCommand(command, opts);\n\n      // most of the time we don't care when socket.write does\n      // so give it an empty function\n      let socketCb = _.noop;\n\n      // handle socket problems\n      onSocketError = (exception) => {\n        if (this.connected) {\n          log.error(`Socket error: ${exception.message}`);\n        }\n\n        // the connection was refused, so reject the connect promise\n        reject(exception);\n      };\n      this.socket.on('error', onSocketError);\n      if (this.messageHandler.hasSpecialMessageHandler(data.__selector)) {\n        // special replies will return any number of arguments\n        // temporarily wrap with promise handling\n        let specialMessageHandler = this.getSpecialMessageHandler(data.__selector);\n        this.setSpecialMessageHandler(data.__selector, reject, function (...args) {\n          log.debug(`Received response from socket send: '${_.truncate(JSON.stringify(args), {length: 50})}'`);\n\n          // call the original listener, and put it back, if necessary\n          specialMessageHandler(...args);\n          if (this.messageHandler.hasSpecialMessageHandler(data.__selector)) {\n            // this means that the system has not removed this listener\n            this.setSpecialMessageHandler(data.__selector, null, specialMessageHandler);\n          }\n\n          resolve(args);\n        }.bind(this));\n      } else if (data.__argument && data.__argument.WIRSocketDataKey) {\n        // keep track of the messages coming and going using\n        // a simple sequential id\n        this.curMsgId++;\n\n        const errorHandler = function (err) {\n          const msg = `Remote debugger error with code '${err.code}': ${err.message}`;\n          reject(new Error(msg));\n        };\n\n        this.setDataMessageHandler(this.curMsgId.toString(), errorHandler, (value) => {\n          const msg = _.truncate(_.isString(value) ? value : JSON.stringify(value), {length: 50});\n          log.debug(`Received data response from socket send: '${msg}'`);\n          log.debug(`Original command: ${command}`);\n          resolve(value);\n        });\n        data.__argument.WIRSocketDataKey.id = this.curMsgId;\n        data.__argument.WIRSocketDataKey =\n            Buffer.from(JSON.stringify(data.__argument.WIRSocketDataKey));\n      } else {\n        // we want to immediately resolve this socket.write\n        // any long term callbacks will do their business in the background\n        socketCb = resolve;\n      }\n\n      log.debug(`Sending '${data.__selector}' message to remote debugger`);\n\n      // remote debugger expects a binary plist as data\n      let plist;\n      try {\n        plist = bplistCreate(data);\n      } catch (e) {\n        let msg = `Could not create binary plist from data: ${e.message}`;\n        log.error(msg);\n        return reject(new Error(msg));\n      }\n\n      if (this.socket && this.connected) {\n        // cork and uncork in order to not buffer the write\n        // on some systems this is necessary or the server\n        // gets confused.\n        this.socket.cork();\n        try {\n          this.socket.write(bufferpack.pack('L', [plist.length]));\n          this.socket.write(plist, socketCb);\n        } finally {\n          this.socket.uncork();\n        }\n      } else {\n        let msg = 'Attempted to write data to socket after it was closed!';\n        log.error(msg);\n        reject(new Error(msg));\n      }\n    })\n    .finally(() => {\n      // remove this listener, so we don't exhaust the system\n      this.socket.removeListener('error', onSocketError);\n    });\n  }\n\n  receive (data) {\n    // Append this new data to the existing Buffer\n    this.received = Buffer.concat([this.received, data]);\n    let dataLeftOver = true;\n\n    // Parse multiple messages in the same packet\n    while (dataLeftOver) {\n      // Store a reference to where we were\n      let oldReadPos = this.readPos;\n\n      // Read the prefix (plist length) to see how far to read next\n      // It's always 4 bytes long\n      let prefix = this.received.slice(this.readPos, this.readPos + 4);\n\n      let msgLength;\n      try {\n        msgLength = bufferpack.unpack('L', prefix)[0];\n      } catch (e) {\n        log.error(`Buffer could not unpack: ${e}`);\n        return;\n      }\n\n      // Jump forward 4 bytes\n      this.readPos += 4;\n\n      // Is there enough data here?\n      // If not, jump back to our original position and gtfo\n      if (this.received.length < msgLength + this.readPos) {\n        this.readPos = oldReadPos;\n        break;\n      }\n\n      // Extract the main body of the message (where the plist should be)\n      let body = this.received.slice(this.readPos, msgLength + this.readPos);\n\n      // Extract the plist\n      let plist;\n      try {\n        plist = bplistParser.parseBuffer(body);\n      } catch (e) {\n        log.error(`Error parsing binary plist: ${e}`);\n        return;\n      }\n\n      // bplistParser.parseBuffer returns an array\n      if (plist.length === 1) {\n        plist = plist[0];\n      }\n\n      for (let key of ['WIRMessageDataKey', 'WIRDestinationKey', 'WIRSocketDataKey']) {\n        if (!_.isUndefined(plist[key])) {\n          plist[key] = plist[key].toString(\"utf8\");\n        }\n      }\n\n      // Jump forward the length of the plist\n      this.readPos += msgLength;\n\n      // Calculate how much buffer is left\n      let leftOver = this.received.length - this.readPos;\n\n      // Is there some left over?\n      if (leftOver !== 0) {\n        // Copy what's left over into a new buffer, and save it for next time\n        let chunk = Buffer.alloc(leftOver);\n        this.received.copy(chunk, 0, this.readPos);\n        this.received = chunk;\n      } else {\n        // Otherwise, empty the buffer and get out of the loop\n        this.received = Buffer.alloc(0);\n        dataLeftOver = false;\n      }\n\n      // Reset the read position\n      this.readPos = 0;\n\n      // Now do something with the plist\n      if (plist) {\n        this.messageHandler.handleMessage(plist);\n      }\n    }\n  }\n\n  setTimelineEventHandler (timelineEventHandler) {\n    this.timelineEventHandler = timelineEventHandler;\n    this.messageHandler.setTimelineEventHandler(timelineEventHandler);\n  }\n\n  setConsoleLogEventHandler (consoleEventHandler) {\n    this.consoleEventHandler = consoleEventHandler;\n    this.messageHandler.setConsoleLogEventHandler(consoleEventHandler);\n  }\n\n  setNetworkLogEventHandler (networkEventHandler) {\n    this.networkEventHandler = networkEventHandler;\n    this.messageHandler.setNetworkEventHandler(networkEventHandler);\n  }\n}\n"],"file":"lib/remote-debugger-rpc-client.js","sourceRoot":"../.."}