UNPKG

52.2 kBJavaScriptView Raw
1/**
2 * @class TheClient
3 */
4'use strict';
5
6var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
7
8var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
9
10var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
11
12var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray"));
13
14var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties"));
15
16var _objectSpread2 = _interopRequireDefault(require("@babel/runtime/helpers/objectSpread"));
17
18var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
19
20var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
21
22var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
23
24var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
25
26var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
27
28var argx = require('argx');
29
30var asleep = require('asleep');
31
32var cookies = require('browser-cookies');
33
34var _require = require('bstorage'),
35 restore = _require.restore,
36 save = _require.save;
37
38var _require2 = require('bwindow'),
39 _get = _require2.get;
40
41var qs = require('qs');
42
43var _require3 = require('rfunc-client/shim/browser'),
44 RFuncClient = _require3.RFuncClient;
45
46var io = require('socket.io-client');
47
48var _require4 = require('the-check'),
49 isBrowser = _require4.isBrowser,
50 isProduction = _require4.isProduction,
51 unlessProduction = _require4.unlessProduction;
52
53var _require5 = require('the-pack'),
54 ThePack = _require5.ThePack;
55
56var _require6 = require('url'),
57 resolveUrl = _require6.resolve;
58
59var uuid = require('uuid');
60
61var IOEvents = require('./constants/IOEvents');
62
63var _require7 = require('./helpers'),
64 asController = _require7.asController,
65 debugController = _require7.debugController,
66 debugStream = _require7.debugStream,
67 parseClientUrl = _require7.parseClientUrl;
68
69var _require8 = require('./mixins'),
70 infoMix = _require8.infoMix,
71 pingPongMix = _require8.pingPongMix,
72 streamMix = _require8.streamMix;
73
74var debug = require('debug')('the:client');
75
76var NAMESPACE = '/rpc';
77var TheClientBase = [pingPongMix, infoMix, streamMix].reduce(function (Class, mix) {
78 return mix(Class);
79}, RFuncClient);
80
81var _ref = new ThePack({}),
82 decode = _ref.decode,
83 encode = _ref.encode;
84/** @lends TheClient */
85
86
87var TheClient =
88/*#__PURE__*/
89function (_TheClientBase) {
90 (0, _inherits2.default)(TheClient, _TheClientBase);
91 (0, _createClass2.default)(TheClient, null, [{
92 key: "for",
93 // noinspection ReservedWordAsName
94
95 /**
96 * Create the client instance
97 * @param {string} namespace
98 * @param {Object} [config={}]
99 * @returns {TheClient}
100 */
101 value: function _for() {
102 var namespace = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'default';
103 var config = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
104 var key = [TheClient.CID_KEY, namespace].join('/').trim();
105 var cid = restore(key) || TheClient.newCID();
106 var client = new TheClient((0, _objectSpread2.default)({}, config, {
107 cid: cid
108 }));
109 var isBrowser = !!_get('document');
110
111 if (isBrowser) {
112 save(key, cid);
113 cookies.set(key, cid, {});
114 }
115
116 return client;
117 }
118 }]);
119
120 function TheClient(url, config) {
121 var _this;
122
123 (0, _classCallCheck2.default)(this, TheClient);
124 var args = argx(arguments);
125 url = args.shift('string');
126 config = args.pop('object') || {};
127
128 if (!url) {
129 url = parseClientUrl(config);
130 }
131
132 var _config = config,
133 _config$cid = _config.cid,
134 cid = _config$cid === void 0 ? TheClient.newCID() : _config$cid,
135 _config$forceNewSocke = _config.forceNewSocket,
136 forceNewSocket = _config$forceNewSocke === void 0 ? false : _config$forceNewSocke,
137 onGone = _config.onGone,
138 _config$version = _config.version,
139 version = _config$version === void 0 ? 'unknown' : _config$version,
140 restOptions = (0, _objectWithoutProperties2.default)(_config, ["cid", "forceNewSocket", "onGone", "version"]);
141
142 if (!url) {
143 throw new Error("[TheClient] Failed to parse urls with args ".concat(JSON.stringify(arguments)));
144 }
145
146 _this = (0, _possibleConstructorReturn2.default)(this, (0, _getPrototypeOf2.default)(TheClient).call(this, url, restOptions));
147 _this._onGone = onGone;
148 _this._forceNewSocket = forceNewSocket;
149 _this._gone = false;
150 _this._controllers = {};
151 _this._cid = cid;
152 _this._version = version;
153 _this._socket = null;
154 _this._closed = false;
155 return _this;
156 }
157
158 (0, _createClass2.default)(TheClient, [{
159 key: "assertNotClosed",
160 value: function assertNotClosed() {
161 if (this.closed) {
162 throw new Error("[TheClient] Already closed!");
163 }
164 }
165 }, {
166 key: "handleCallback",
167 value: function handleCallback(controllerName, handleName, data) {
168 var values = decode(data);
169 var controller = this._controllers[controllerName];
170
171 if (!controller) {
172 console.warn("[TheClient] Callback controller not found: ".concat(controllerName));
173 return;
174 }
175
176 var callback = controller.callbacks[handleName];
177
178 if (!callback) {
179 console.warn("[TheClient] Callback controller not found: ".concat(controllerName));
180 return;
181 }
182
183 unlessProduction(function () {
184 console.groupCollapsed("[TheClient] Callback `".concat(controllerName, ".").concat(handleName, "()`"));
185 console.log('Signature', "`".concat(controllerName, ".").concat(handleName, "()`"));
186 console.log('Arguments', values);
187 console.groupEnd();
188 });
189 callback.apply(void 0, (0, _toConsumableArray2.default)(values));
190 }
191 }, {
192 key: "markAsGone",
193 value: function markAsGone() {
194 if (this._gone) {
195 return;
196 }
197
198 this._onGone && this._onGone();
199 this._gone = true;
200 }
201 }, {
202 key: "close",
203 value: function () {
204 var _close = (0, _asyncToGenerator2.default)(
205 /*#__PURE__*/
206 _regenerator.default.mark(function _callee() {
207 var socket;
208 return _regenerator.default.wrap(function _callee$(_context) {
209 while (1) {
210 switch (_context.prev = _context.next) {
211 case 0:
212 this._closed = true;
213 socket = this._socket;
214
215 if (socket) {
216 socket.close();
217 }
218
219 case 3:
220 case "end":
221 return _context.stop();
222 }
223 }
224 }, _callee, this);
225 }));
226
227 function close() {
228 return _close.apply(this, arguments);
229 }
230
231 return close;
232 }()
233 /**
234 * Invoke a method
235 * @param {string} moduleName
236 * @param {string} methodName - Name of method to invoke
237 * @param {...*} params - Parameters
238 */
239
240 }, {
241 key: "invoke",
242 value: function () {
243 var _invoke = (0, _asyncToGenerator2.default)(
244 /*#__PURE__*/
245 _regenerator.default.mark(function _callee2(moduleName, methodName) {
246 var _len,
247 params,
248 _key,
249 cid,
250 socket,
251 iid,
252 _args2 = arguments;
253
254 return _regenerator.default.wrap(function _callee2$(_context2) {
255 while (1) {
256 switch (_context2.prev = _context2.next) {
257 case 0:
258 for (_len = _args2.length, params = new Array(_len > 2 ? _len - 2 : 0), _key = 2; _key < _len; _key++) {
259 params[_key - 2] = _args2[_key];
260 }
261
262 this.assertNotClosed();
263 cid = this.cid, socket = this.socket;
264 iid = uuid.v4(); // Invocation id
265
266 _context2.next = 6;
267 return new Promise(function (resolve, reject) {
268 var keptGoneTimer = -1;
269
270 var _onReturn = function onReturn(returned) {
271 if (!_onReturn) {
272 return null;
273 }
274
275 returned = decode(returned);
276
277 if (returned.iid !== iid) {
278 return;
279 }
280
281 socket.off(IOEvents.RPC_RETURN, _onReturn);
282 _onReturn = null;
283 _onKeep = null;
284 clearTimeout(keptGoneTimer);
285 var _returned = returned,
286 data = _returned.data,
287 errors = _returned.errors,
288 ok = _returned.ok;
289 debug('rpc return', moduleName, methodName, returned);
290
291 if (ok) {
292 resolve(data);
293 } else {
294 var e = errors[0];
295 reject(e.message || e);
296 }
297 };
298
299 var _onKeep = function onKeep(kept) {
300 if (!_onKeep) {
301 return;
302 }
303
304 kept = decode(kept);
305
306 if (kept.iid !== iid) {
307 return;
308 }
309
310 socket.off(IOEvents.RPC_KEEP, _onKeep);
311 _onKeep = null;
312 debug('rpc keep', moduleName, methodName);
313 clearTimeout(keptGoneTimer);
314 var _kept = kept,
315 duration = _kept.duration;
316 keptGoneTimer = setTimeout(function () {
317 // TODO throw error?
318 console.error("[TheClient] RPC call seems gone: `".concat(moduleName, ".").concat(methodName, "()`"));
319 }, duration * 2);
320 };
321
322 socket.on(IOEvents.RPC_RETURN, _onReturn);
323 socket.emit(IOEvents.RPC_CALL, encode({
324 cid: cid,
325 iid: iid,
326 methodName: methodName,
327 moduleName: moduleName,
328 params: params
329 }));
330 debug('rpc call', moduleName, methodName, params);
331 });
332
333 case 6:
334 return _context2.abrupt("return", _context2.sent);
335
336 case 7:
337 case "end":
338 return _context2.stop();
339 }
340 }
341 }, _callee2, this);
342 }));
343
344 function invoke(_x, _x2) {
345 return _invoke.apply(this, arguments);
346 }
347
348 return invoke;
349 }()
350 }, {
351 key: "newSocket",
352 value: function () {
353 var _newSocket = (0, _asyncToGenerator2.default)(
354 /*#__PURE__*/
355 _regenerator.default.mark(function _callee3() {
356 var _this2 = this;
357
358 var query, socket;
359 return _regenerator.default.wrap(function _callee3$(_context3) {
360 while (1) {
361 switch (_context3.prev = _context3.next) {
362 case 0:
363 this.assertNotClosed();
364 query = qs.stringify(this.scope);
365 socket = io(resolveUrl(this.baseUrl, "".concat(NAMESPACE, "?").concat(query)), {
366 forceNew: this._forceNewSocket
367 });
368 socket.on('disconnect', function () {
369 debug('disconnect');
370
371 if (_this2.closed) {
372 return;
373 }
374
375 var goneTimer = setTimeout(function () {
376 return _this2.markAsGone();
377 }, 2 * 1000 + 2 * 1000 * Math.random());
378
379 var cancelGone = function cancelGone() {
380 debug('cancelGone');
381 clearTimeout(goneTimer);
382 socket.off('connect', cancelGone);
383 socket.off('reconnect', cancelGone);
384 };
385
386 socket.once('connect', cancelGone);
387 socket.once('reconnect', cancelGone);
388 socket.connect();
389 });
390 _context3.next = 6;
391 return new Promise(function (resolve, reject) {
392 socket.on('connect', function () {
393 debug('connect');
394 resolve(socket);
395 });
396 socket.on('error', function (e) {
397 debug('error', e);
398 reject(e);
399 });
400 });
401
402 case 6:
403 return _context3.abrupt("return", socket);
404
405 case 7:
406 case "end":
407 return _context3.stop();
408 }
409 }
410 }, _callee3, this);
411 }));
412
413 function newSocket() {
414 return _newSocket.apply(this, arguments);
415 }
416
417 return newSocket;
418 }()
419 /**
420 * Create an stream to server
421 * @param {string} name
422 * @param {Object} params - Stream params
423 * @param {Object} [options={}] - Optional setting
424 * @param {Boolean} [options.debug] - With debug mode
425 * @returns {*}
426 */
427
428 }, {
429 key: "stream",
430 value: function () {
431 var _stream = (0, _asyncToGenerator2.default)(
432 /*#__PURE__*/
433 _regenerator.default.mark(function _callee4(name) {
434 var params,
435 options,
436 _options$debug,
437 debug,
438 stream,
439 _args4 = arguments;
440
441 return _regenerator.default.wrap(function _callee4$(_context4) {
442 while (1) {
443 switch (_context4.prev = _context4.next) {
444 case 0:
445 params = _args4.length > 1 && _args4[1] !== undefined ? _args4[1] : {};
446 options = _args4.length > 2 && _args4[2] !== undefined ? _args4[2] : {};
447 this.assertNotClosed();
448 _options$debug = options.debug, debug = _options$debug === void 0 ? !isProduction() && isBrowser() : _options$debug;
449
450 if (this._socket) {
451 _context4.next = 8;
452 break;
453 }
454
455 _context4.next = 7;
456 return this.newSocket();
457
458 case 7:
459 this._socket = _context4.sent;
460
461 case 8:
462 _context4.next = 10;
463 return this.openStream(name, params);
464
465 case 10:
466 stream = _context4.sent;
467 return _context4.abrupt("return", debug ? debugStream(stream) : stream);
468
469 case 12:
470 case "end":
471 return _context4.stop();
472 }
473 }
474 }, _callee4, this);
475 }));
476
477 function stream(_x3) {
478 return _stream.apply(this, arguments);
479 }
480
481 return stream;
482 }()
483 /**
484 * Use a controller module
485 * @param {string} controllerName - Module name
486 * @param {Object} [options={}] - Optional setting
487 * @param {Boolean} [options.debug] - With debug mode
488 * @returns {*}
489 */
490
491 }, {
492 key: "use",
493 value: function () {
494 var _use = (0, _asyncToGenerator2.default)(
495 /*#__PURE__*/
496 _regenerator.default.mark(function _callee5(controllerName) {
497 var _this3 = this;
498
499 var options,
500 _options$debug2,
501 debug,
502 controller,
503 instance,
504 spec,
505 cid,
506 _args5 = arguments;
507
508 return _regenerator.default.wrap(function _callee5$(_context5) {
509 while (1) {
510 switch (_context5.prev = _context5.next) {
511 case 0:
512 options = _args5.length > 1 && _args5[1] !== undefined ? _args5[1] : {};
513 this.assertNotClosed();
514 _options$debug2 = options.debug, debug = _options$debug2 === void 0 ? !isProduction() && isBrowser() : _options$debug2;
515 controller = this._controllers[controllerName];
516
517 if (this._socket) {
518 _context5.next = 8;
519 break;
520 }
521
522 _context5.next = 7;
523 return this.newSocket();
524
525 case 7:
526 this._socket = _context5.sent;
527
528 case 8:
529 if (controller) {
530 _context5.next = 19;
531 break;
532 }
533
534 _context5.next = 11;
535 return this.connect(controllerName);
536
537 case 11:
538 instance = _context5.sent;
539 _context5.next = 14;
540 return this.describe(controllerName);
541
542 case 14:
543 spec = _context5.sent;
544 cid = this.cid;
545 controller = asController(instance, spec, {
546 cid: cid
547 }, {
548 onToggleHandler: function onToggleHandler(handlerName, enabled) {
549 var socket = _this3.socket;
550 var event = [IOEvents.CLIENT_CALLBACK, cid, controllerName, handlerName].join('/');
551
552 if (enabled) {
553 socket.on(event, function (data) {
554 _this3.handleCallback(controllerName, handlerName, data);
555 });
556 } else {
557 socket.off(event);
558 }
559 }
560 });
561
562 if (debug) {
563 controller = debugController(controller);
564 }
565
566 this._controllers[controllerName] = controller;
567
568 case 19:
569 return _context5.abrupt("return", controller);
570
571 case 20:
572 case "end":
573 return _context5.stop();
574 }
575 }
576 }, _callee5, this);
577 }));
578
579 function use(_x4) {
580 return _use.apply(this, arguments);
581 }
582
583 return use;
584 }()
585 /**
586 * Use all controller modules
587 * @param {Object} [options={}] - Optional setting
588 * @returns {Promise<Object>}
589 */
590
591 }, {
592 key: "useAll",
593 value: function () {
594 var _useAll = (0, _asyncToGenerator2.default)(
595 /*#__PURE__*/
596 _regenerator.default.mark(function _callee6() {
597 var options,
598 serverInfo,
599 controllers,
600 controllerSpecs,
601 _iteratorNormalCompletion,
602 _didIteratorError,
603 _iteratorError,
604 _iterator,
605 _step,
606 _ref3,
607 methods,
608 name,
609 _args6 = arguments;
610
611 return _regenerator.default.wrap(function _callee6$(_context6) {
612 while (1) {
613 switch (_context6.prev = _context6.next) {
614 case 0:
615 options = _args6.length > 0 && _args6[0] !== undefined ? _args6[0] : {};
616 _context6.next = 3;
617 return this.serverInfo();
618
619 case 3:
620 serverInfo = _context6.sent;
621 controllers = {};
622 controllerSpecs = serverInfo.controllers;
623 _iteratorNormalCompletion = true;
624 _didIteratorError = false;
625 _iteratorError = undefined;
626 _context6.prev = 9;
627 _iterator = controllerSpecs[Symbol.iterator]();
628
629 case 11:
630 if (_iteratorNormalCompletion = (_step = _iterator.next()).done) {
631 _context6.next = 21;
632 break;
633 }
634
635 _ref3 = _step.value;
636 methods = _ref3.methods, name = _ref3.name;
637 this.specs[name] = {
638 methods: methods,
639 name: name
640 };
641 _context6.next = 17;
642 return this.use(name, options);
643
644 case 17:
645 controllers[name] = _context6.sent;
646
647 case 18:
648 _iteratorNormalCompletion = true;
649 _context6.next = 11;
650 break;
651
652 case 21:
653 _context6.next = 27;
654 break;
655
656 case 23:
657 _context6.prev = 23;
658 _context6.t0 = _context6["catch"](9);
659 _didIteratorError = true;
660 _iteratorError = _context6.t0;
661
662 case 27:
663 _context6.prev = 27;
664 _context6.prev = 28;
665
666 if (!_iteratorNormalCompletion && _iterator.return != null) {
667 _iterator.return();
668 }
669
670 case 30:
671 _context6.prev = 30;
672
673 if (!_didIteratorError) {
674 _context6.next = 33;
675 break;
676 }
677
678 throw _iteratorError;
679
680 case 33:
681 return _context6.finish(30);
682
683 case 34:
684 return _context6.finish(27);
685
686 case 35:
687 unlessProduction(function () {
688 if (typeof Proxy === 'undefined') {
689 return controllers;
690 }
691
692 return new Proxy(controllers, {
693 get: function get(target, key) {
694 var has = key in target;
695
696 if (!has) {
697 console.warn("[TheClient] Unknown controller name: \"".concat(key, "\""));
698 }
699
700 return target[key];
701 }
702 });
703 });
704 return _context6.abrupt("return", controllers);
705
706 case 37:
707 case "end":
708 return _context6.stop();
709 }
710 }
711 }, _callee6, this, [[9, 23, 27, 35], [28,, 30, 34]]);
712 }));
713
714 function useAll() {
715 return _useAll.apply(this, arguments);
716 }
717
718 return useAll;
719 }()
720 }, {
721 key: "cid",
722 get: function get() {
723 return this._cid;
724 }
725 }, {
726 key: "closed",
727 get: function get() {
728 return this._closed;
729 }
730 }, {
731 key: "scope",
732 get: function get() {
733 var cid = this._cid,
734 rpc = this._rpc,
735 version = this._version;
736
737 var language = _get('navigator.language');
738
739 return {
740 /** Caller key */
741 callerKey: rpc && rpc.as,
742
743 /** Client ID */
744 cid: cid,
745
746 /** Host of client */
747 host: _get('location.host'),
748
749 /** Detected lang */
750 lang: language && language.split('-')[0],
751
752 /** Connecting protocol */
753 protocol: _get('location.protocol'),
754
755 /** Client instance version number */
756 v: version,
757
758 /** Via client */
759 via: 'client'
760 };
761 }
762 }, {
763 key: "socket",
764 get: function get() {
765 return this._socket;
766 }
767 }]);
768 return TheClient;
769}(TheClientBase);
770
771TheClient.RPC_ACTOR_NAME = 'rpc';
772TheClient.CID_KEY = 'the:cid';
773
774TheClient.newCID = function () {
775 return uuid.v4();
776};
777
778module.exports = TheClient;
779//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TheClient.js"],"names":["argx","require","asleep","cookies","restore","save","get","qs","RFuncClient","io","isBrowser","isProduction","unlessProduction","ThePack","resolveUrl","resolve","uuid","IOEvents","asController","debugController","debugStream","parseClientUrl","infoMix","pingPongMix","streamMix","debug","NAMESPACE","TheClientBase","reduce","Class","mix","decode","encode","TheClient","namespace","config","key","CID_KEY","join","trim","cid","newCID","client","set","url","args","arguments","shift","pop","forceNewSocket","onGone","version","restOptions","Error","JSON","stringify","_onGone","_forceNewSocket","_gone","_controllers","_cid","_version","_socket","_closed","closed","controllerName","handleName","data","values","controller","console","warn","callback","callbacks","groupCollapsed","log","groupEnd","socket","close","moduleName","methodName","params","assertNotClosed","iid","v4","Promise","reject","keptGoneTimer","onReturn","returned","off","RPC_RETURN","onKeep","clearTimeout","errors","ok","e","message","kept","RPC_KEEP","duration","setTimeout","error","on","emit","RPC_CALL","query","scope","baseUrl","forceNew","goneTimer","markAsGone","Math","random","cancelGone","once","connect","name","options","newSocket","openStream","stream","instance","describe","spec","onToggleHandler","handlerName","enabled","event","CLIENT_CALLBACK","handleCallback","serverInfo","controllers","controllerSpecs","methods","specs","use","Proxy","target","has","rpc","_rpc","language","callerKey","as","host","lang","split","protocol","v","via","RPC_ACTOR_NAME","module","exports"],"mappings":"AAAA;;;AAGA;;;;;;;;;;;;;;;;;;;;;;;;AAEA,IAAMA,IAAI,GAAGC,OAAO,CAAC,MAAD,CAApB;;AACA,IAAMC,MAAM,GAAGD,OAAO,CAAC,QAAD,CAAtB;;AACA,IAAME,OAAO,GAAGF,OAAO,CAAC,iBAAD,CAAvB;;eAC0BA,OAAO,CAAC,UAAD,C;IAAzBG,O,YAAAA,O;IAASC,I,YAAAA,I;;gBACDJ,OAAO,CAAC,SAAD,C;IAAfK,I,aAAAA,G;;AACR,IAAMC,EAAE,GAAGN,OAAO,CAAC,IAAD,CAAlB;;gBACwBA,OAAO,CAAC,2BAAD,C;IAAvBO,W,aAAAA,W;;AACR,IAAMC,EAAE,GAAGR,OAAO,CAAC,kBAAD,CAAlB;;gBACsDA,OAAO,CAAC,WAAD,C;IAArDS,S,aAAAA,S;IAAWC,Y,aAAAA,Y;IAAcC,gB,aAAAA,gB;;gBACbX,OAAO,CAAC,UAAD,C;IAAnBY,O,aAAAA,O;;gBACwBZ,OAAO,CAAC,KAAD,C;IAAtBa,U,aAATC,O;;AACR,IAAMC,IAAI,GAAGf,OAAO,CAAC,MAAD,CAApB;;AACA,IAAMgB,QAAQ,GAAGhB,OAAO,CAAC,sBAAD,CAAxB;;gBAMIA,OAAO,CAAC,WAAD,C;IAJTiB,Y,aAAAA,Y;IACAC,e,aAAAA,e;IACAC,W,aAAAA,W;IACAC,c,aAAAA,c;;gBAE0CpB,OAAO,CAAC,UAAD,C;IAA3CqB,O,aAAAA,O;IAASC,W,aAAAA,W;IAAaC,S,aAAAA,S;;AAC9B,IAAMC,KAAK,GAAGxB,OAAO,CAAC,OAAD,CAAP,CAAiB,YAAjB,CAAd;;AACA,IAAMyB,SAAS,GAAG,MAAlB;AAEA,IAAMC,aAAa,GAAG,CACpBJ,WADoB,EAEpBD,OAFoB,EAGpBE,SAHoB,EAIpBI,MAJoB,CAIb,UAACC,KAAD,EAAQC,GAAR;AAAA,SAAgBA,GAAG,CAACD,KAAD,CAAnB;AAAA,CAJa,EAIerB,WAJf,CAAtB;;WAM2B,IAAIK,OAAJ,CAAY,EAAZ,C;IAAnBkB,M,QAAAA,M;IAAQC,M,QAAAA,M;AAEhB;;;IACMC,S;;;;;;AACJ;;AACA;;;;;;2BAMgD;AAAA,UAApCC,SAAoC,uEAAxB,SAAwB;AAAA,UAAbC,MAAa,uEAAJ,EAAI;AAC9C,UAAMC,GAAG,GAAG,CAACH,SAAS,CAACI,OAAX,EAAoBH,SAApB,EAA+BI,IAA/B,CAAoC,GAApC,EAAyCC,IAAzC,EAAZ;AACA,UAAMC,GAAG,GAAGpC,OAAO,CAACgC,GAAD,CAAP,IAAgBH,SAAS,CAACQ,MAAV,EAA5B;AACA,UAAMC,MAAM,GAAG,IAAIT,SAAJ,iCAAmBE,MAAnB;AAA2BK,QAAAA,GAAG,EAAHA;AAA3B,SAAf;AACA,UAAM9B,SAAS,GAAG,CAAC,CAACJ,IAAG,CAAC,UAAD,CAAvB;;AACA,UAAII,SAAJ,EAAe;AACbL,QAAAA,IAAI,CAAC+B,GAAD,EAAMI,GAAN,CAAJ;AACArC,QAAAA,OAAO,CAACwC,GAAR,CAAYP,GAAZ,EAAiBI,GAAjB,EAAsB,EAAtB;AACD;;AACD,aAAOE,MAAP;AACD;;;AAED,qBAAaE,GAAb,EAAkBT,MAAlB,EAA0B;AAAA;;AAAA;AACxB,QAAMU,IAAI,GAAG7C,IAAI,CAAC8C,SAAD,CAAjB;AACAF,IAAAA,GAAG,GAAGC,IAAI,CAACE,KAAL,CAAW,QAAX,CAAN;AACAZ,IAAAA,MAAM,GAAGU,IAAI,CAACG,GAAL,CAAS,QAAT,KAAsB,EAA/B;;AACA,QAAI,CAACJ,GAAL,EAAU;AACRA,MAAAA,GAAG,GAAGvB,cAAc,CAACc,MAAD,CAApB;AACD;;AANuB,kBAapBA,MAboB;AAAA,8BAQtBK,GARsB;AAAA,QAQtBA,GARsB,4BAQhBP,SAAS,CAACQ,MAAV,EARgB;AAAA,wCAStBQ,cATsB;AAAA,QAStBA,cATsB,sCASL,KATK;AAAA,QAUtBC,MAVsB,WAUtBA,MAVsB;AAAA,kCAWtBC,OAXsB;AAAA,QAWtBA,OAXsB,gCAWZ,SAXY;AAAA,QAYnBC,WAZmB;;AAcxB,QAAI,CAACR,GAAL,EAAU;AACR,YAAM,IAAIS,KAAJ,sDAAwDC,IAAI,CAACC,SAAL,CAAeT,SAAf,CAAxD,EAAN;AACD;;AACD,+GAAMF,GAAN,EAAWQ,WAAX;AACA,UAAKI,OAAL,GAAeN,MAAf;AACA,UAAKO,eAAL,GAAuBR,cAAvB;AACA,UAAKS,KAAL,GAAa,KAAb;AACA,UAAKC,YAAL,GAAoB,EAApB;AACA,UAAKC,IAAL,GAAYpB,GAAZ;AACA,UAAKqB,QAAL,GAAgBV,OAAhB;AACA,UAAKW,OAAL,GAAe,IAAf;AACA,UAAKC,OAAL,GAAe,KAAf;AAzBwB;AA0BzB;;;;sCAmCkB;AACjB,UAAI,KAAKC,MAAT,EAAiB;AACf,cAAM,IAAIX,KAAJ,+BAAN;AACD;AACF;;;mCAEeY,c,EAAgBC,U,EAAYC,I,EAAM;AAChD,UAAMC,MAAM,GAAGrC,MAAM,CAACoC,IAAD,CAArB;AACA,UAAME,UAAU,GAAG,KAAKV,YAAL,CAAkBM,cAAlB,CAAnB;;AACA,UAAI,CAACI,UAAL,EAAiB;AACfC,QAAAA,OAAO,CAACC,IAAR,sDAA2DN,cAA3D;AACA;AACD;;AACD,UAAMO,QAAQ,GAAGH,UAAU,CAACI,SAAX,CAAqBP,UAArB,CAAjB;;AACA,UAAI,CAACM,QAAL,EAAe;AACbF,QAAAA,OAAO,CAACC,IAAR,sDAA2DN,cAA3D;AACA;AACD;;AACDrD,MAAAA,gBAAgB,CAAC,YAAM;AACrB0D,QAAAA,OAAO,CAACI,cAAR,iCAAiDT,cAAjD,cAAmEC,UAAnE;AACAI,QAAAA,OAAO,CAACK,GAAR,CAAY,WAAZ,aAA8BV,cAA9B,cAAgDC,UAAhD;AACAI,QAAAA,OAAO,CAACK,GAAR,CAAY,WAAZ,EAAyBP,MAAzB;AACAE,QAAAA,OAAO,CAACM,QAAR;AACD,OALe,CAAhB;AAMAJ,MAAAA,QAAQ,MAAR,0CAAYJ,MAAZ;AACD;;;iCAEa;AACZ,UAAI,KAAKV,KAAT,EAAgB;AACd;AACD;;AACD,WAAKF,OAAL,IAAgB,KAAKA,OAAL,EAAhB;AACA,WAAKE,KAAL,GAAa,IAAb;AACD;;;;;;;;;;;;AAGC,qBAAKK,OAAL,GAAe,IAAf;AACMc,gBAAAA,M,GAAS,KAAKf,O;;AACpB,oBAAIe,MAAJ,EAAY;AACVA,kBAAAA,MAAM,CAACC,KAAP;AACD;;;;;;;;;;;;;;;;AAGH;;;;;;;;;;;;kDAMcC,U,EAAYC,U;;;;;;;;;;;;;2CAAeC,M;AAAAA,kBAAAA,M;;;AACvC,qBAAKC,eAAL;AACQ1C,gBAAAA,G,GAAgB,I,CAAhBA,G,EAAKqC,M,GAAW,I,CAAXA,M;AACPM,gBAAAA,G,GAAMnE,IAAI,CAACoE,EAAL,E,EAAU;;;uBAET,IAAIC,OAAJ,CAAY,UAACtE,OAAD,EAAUuE,MAAV,EAAqB;AAC5C,sBAAIC,aAAa,GAAG,CAAC,CAArB;;AACA,sBAAIC,SAAQ,GAAG,kBAACC,QAAD,EAAc;AAC3B,wBAAI,CAACD,SAAL,EAAe;AACb,6BAAO,IAAP;AACD;;AACDC,oBAAAA,QAAQ,GAAG1D,MAAM,CAAC0D,QAAD,CAAjB;;AACA,wBAAIA,QAAQ,CAACN,GAAT,KAAiBA,GAArB,EAA0B;AACxB;AACD;;AACDN,oBAAAA,MAAM,CAACa,GAAP,CAAWzE,QAAQ,CAAC0E,UAApB,EAAgCH,SAAhC;AACAA,oBAAAA,SAAQ,GAAG,IAAX;AACAI,oBAAAA,OAAM,GAAG,IAAT;AACAC,oBAAAA,YAAY,CAACN,aAAD,CAAZ;AAX2B,oCAYEE,QAZF;AAAA,wBAYnBtB,IAZmB,aAYnBA,IAZmB;AAAA,wBAYb2B,MAZa,aAYbA,MAZa;AAAA,wBAYLC,EAZK,aAYLA,EAZK;AAa3BtE,oBAAAA,KAAK,CAAC,YAAD,EAAesD,UAAf,EAA2BC,UAA3B,EAAuCS,QAAvC,CAAL;;AACA,wBAAIM,EAAJ,EAAQ;AACNhF,sBAAAA,OAAO,CAACoD,IAAD,CAAP;AACD,qBAFD,MAEO;AACL,0BAAM6B,CAAC,GAAGF,MAAM,CAAC,CAAD,CAAhB;AACAR,sBAAAA,MAAM,CAACU,CAAC,CAACC,OAAF,IAAaD,CAAd,CAAN;AACD;AACF,mBApBD;;AAqBA,sBAAIJ,OAAM,GAAG,gBAACM,IAAD,EAAU;AACrB,wBAAI,CAACN,OAAL,EAAa;AACX;AACD;;AACDM,oBAAAA,IAAI,GAAGnE,MAAM,CAACmE,IAAD,CAAb;;AACA,wBAAIA,IAAI,CAACf,GAAL,KAAaA,GAAjB,EAAsB;AACpB;AACD;;AACDN,oBAAAA,MAAM,CAACa,GAAP,CAAWzE,QAAQ,CAACkF,QAApB,EAA8BP,OAA9B;AACAA,oBAAAA,OAAM,GAAG,IAAT;AACAnE,oBAAAA,KAAK,CAAC,UAAD,EAAasD,UAAb,EAAyBC,UAAzB,CAAL;AACAa,oBAAAA,YAAY,CAACN,aAAD,CAAZ;AAXqB,gCAYAW,IAZA;AAAA,wBAYbE,QAZa,SAYbA,QAZa;AAarBb,oBAAAA,aAAa,GAAGc,UAAU,CAAC,YAAM;AAC/B;AACA/B,sBAAAA,OAAO,CAACgC,KAAR,6CAAoDvB,UAApD,cAAkEC,UAAlE;AACD,qBAHyB,EAGvBoB,QAAQ,GAAG,CAHY,CAA1B;AAID,mBAjBD;;AAkBAvB,kBAAAA,MAAM,CAAC0B,EAAP,CAAUtF,QAAQ,CAAC0E,UAAnB,EAA+BH,SAA/B;AAEAX,kBAAAA,MAAM,CAAC2B,IAAP,CAAYvF,QAAQ,CAACwF,QAArB,EAA+BzE,MAAM,CAAC;AACpCQ,oBAAAA,GAAG,EAAHA,GADoC;AAEpC2C,oBAAAA,GAAG,EAAHA,GAFoC;AAGpCH,oBAAAA,UAAU,EAAVA,UAHoC;AAIpCD,oBAAAA,UAAU,EAAVA,UAJoC;AAKpCE,oBAAAA,MAAM,EAANA;AALoC,mBAAD,CAArC;AAOAxD,kBAAAA,KAAK,CAAC,UAAD,EAAasD,UAAb,EAAyBC,UAAzB,EAAqCC,MAArC,CAAL;AACD,iBAnDY,C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAuDb,qBAAKC,eAAL;AACMwB,gBAAAA,K,GAAQnG,EAAE,CAACgD,SAAH,CAAa,KAAKoD,KAAlB,C;AACR9B,gBAAAA,M,GAASpE,EAAE,CAACK,UAAU,CAAC,KAAK8F,OAAN,YAAkBlF,SAAlB,cAA+BgF,KAA/B,EAAX,EAAoD;AACnEG,kBAAAA,QAAQ,EAAE,KAAKpD;AADoD,iBAApD,C;AAGjBoB,gBAAAA,MAAM,CAAC0B,EAAP,CAAU,YAAV,EAAwB,YAAM;AAC5B9E,kBAAAA,KAAK,CAAC,YAAD,CAAL;;AACA,sBAAI,MAAI,CAACuC,MAAT,EAAiB;AACf;AACD;;AACD,sBAAM8C,SAAS,GAAGT,UAAU,CAC1B;AAAA,2BAAM,MAAI,CAACU,UAAL,EAAN;AAAA,mBAD0B,EAE1B,IAAI,IAAJ,GAAW,IAAI,IAAJ,GAAWC,IAAI,CAACC,MAAL,EAFI,CAA5B;;AAIA,sBAAMC,UAAU,GAAG,SAAbA,UAAa,GAAM;AACvBzF,oBAAAA,KAAK,CAAC,YAAD,CAAL;AACAoE,oBAAAA,YAAY,CAACiB,SAAD,CAAZ;AACAjC,oBAAAA,MAAM,CAACa,GAAP,CAAW,SAAX,EAAsBwB,UAAtB;AACArC,oBAAAA,MAAM,CAACa,GAAP,CAAW,WAAX,EAAwBwB,UAAxB;AACD,mBALD;;AAMArC,kBAAAA,MAAM,CAACsC,IAAP,CAAY,SAAZ,EAAuBD,UAAvB;AACArC,kBAAAA,MAAM,CAACsC,IAAP,CAAY,WAAZ,EAAyBD,UAAzB;AACArC,kBAAAA,MAAM,CAACuC,OAAP;AACD,iBAlBD;;uBAmBM,IAAI/B,OAAJ,CAAY,UAACtE,OAAD,EAAUuE,MAAV,EAAqB;AACrCT,kBAAAA,MAAM,CAAC0B,EAAP,CAAU,SAAV,EAAqB,YAAM;AACzB9E,oBAAAA,KAAK,CAAC,SAAD,CAAL;AACAV,oBAAAA,OAAO,CAAC8D,MAAD,CAAP;AACD,mBAHD;AAIAA,kBAAAA,MAAM,CAAC0B,EAAP,CAAU,OAAV,EAAmB,UAACP,CAAD,EAAO;AACxBvE,oBAAAA,KAAK,CAAC,OAAD,EAAUuE,CAAV,CAAL;AACAV,oBAAAA,MAAM,CAACU,CAAD,CAAN;AACD,mBAHD;AAID,iBATK,C;;;kDAUCnB,M;;;;;;;;;;;;;;;;AAGT;;;;;;;;;;;;;;kDAQcwC,I;;;;;;;;;;;;AAAMpC,gBAAAA,M,8DAAS,E;AAAIqC,gBAAAA,O,8DAAU,E;AACzC,qBAAKpC,eAAL;iCAGIoC,O,CADF7F,K,EAAAA,K,+BAAQ,CAACd,YAAY,EAAb,IAAmBD,SAAS,E;;oBAEjC,KAAKoD,O;;;;;;uBACa,KAAKyD,SAAL,E;;;AAArB,qBAAKzD,O;;;;uBAEc,KAAK0D,UAAL,CAAgBH,IAAhB,EAAsBpC,MAAtB,C;;;AAAfwC,gBAAAA,M;kDACChG,KAAK,GAAGL,WAAW,CAACqG,MAAD,CAAd,GAAyBA,M;;;;;;;;;;;;;;;;AAGvC;;;;;;;;;;;;;kDAOWxD,c;;;;;;;;;;;;;;;;AAAgBqD,gBAAAA,O,8DAAU,E;AACnC,qBAAKpC,eAAL;kCAGIoC,O,CADF7F,K,EAAAA,K,gCAAQ,CAACd,YAAY,EAAb,IAAmBD,SAAS,E;AAElC2D,gBAAAA,U,GAAa,KAAKV,YAAL,CAAkBM,cAAlB,C;;oBACZ,KAAKH,O;;;;;;uBACa,KAAKyD,SAAL,E;;;AAArB,qBAAKzD,O;;;oBAEFO,U;;;;;;uBACoB,KAAK+C,OAAL,CAAanD,cAAb,C;;;AAAjByD,gBAAAA,Q;;uBACa,KAAKC,QAAL,CAAc1D,cAAd,C;;;AAAb2D,gBAAAA,I;AACEpF,gBAAAA,G,GAAQ,I,CAARA,G;AACR6B,gBAAAA,UAAU,GAAGnD,YAAY,CAACwG,QAAD,EAAWE,IAAX,EAAiB;AAAEpF,kBAAAA,GAAG,EAAHA;AAAF,iBAAjB,EAA0B;AACjDqF,kBAAAA,eAAe,EAAE,yBAACC,WAAD,EAAcC,OAAd,EAA0B;AAAA,wBACjClD,MADiC,GACtB,MADsB,CACjCA,MADiC;AAEzC,wBAAMmD,KAAK,GAAG,CAAC/G,QAAQ,CAACgH,eAAV,EAA2BzF,GAA3B,EAAgCyB,cAAhC,EAAgD6D,WAAhD,EAA6DxF,IAA7D,CAAkE,GAAlE,CAAd;;AACA,wBAAIyF,OAAJ,EAAa;AACXlD,sBAAAA,MAAM,CAAC0B,EAAP,CAAUyB,KAAV,EAAiB,UAAC7D,IAAD,EAAU;AACzB,wBAAA,MAAI,CAAC+D,cAAL,CAAoBjE,cAApB,EAAoC6D,WAApC,EAAiD3D,IAAjD;AACD,uBAFD;AAGD,qBAJD,MAIO;AACLU,sBAAAA,MAAM,CAACa,GAAP,CAAWsC,KAAX;AACD;AACF;AAXgD,iBAA1B,CAAzB;;AAaA,oBAAIvG,KAAJ,EAAW;AACT4C,kBAAAA,UAAU,GAAGlD,eAAe,CAACkD,UAAD,CAA5B;AACD;;AACD,qBAAKV,YAAL,CAAkBM,cAAlB,IAAoCI,UAApC;;;kDAGKA,U;;;;;;;;;;;;;;;;AAGT;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAKciD,gBAAAA,O,8DAAU,E;;uBACG,KAAKa,UAAL,E;;;AAAnBA,gBAAAA,U;AACAC,gBAAAA,W,GAAc,E;AACdC,gBAAAA,e,GAAkBF,UAAU,CAACC,W;;;;;4BACHC,e;;;;;;;;;AAAnBC,gBAAAA,O,SAAAA,O,EAASjB,I,SAAAA,I;AACpB,qBAAKkB,KAAL,CAAWlB,IAAX,IAAmB;AAAEiB,kBAAAA,OAAO,EAAPA,OAAF;AAAWjB,kBAAAA,IAAI,EAAJA;AAAX,iBAAnB;;uBAC0B,KAAKmB,GAAL,CAASnB,IAAT,EAAeC,OAAf,C;;;AAA1Bc,gBAAAA,WAAW,CAACf,IAAD,C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAEbzG,gBAAAA,gBAAgB,CAAC,YAAM;AACrB,sBAAI,OAAO6H,KAAP,KAAiB,WAArB,EAAkC;AAChC,2BAAOL,WAAP;AACD;;AACD,yBAAO,IAAIK,KAAJ,CAAUL,WAAV,EAAuB;AAC5B9H,oBAAAA,GAD4B,eACvBoI,MADuB,EACftG,GADe,EACV;AAChB,0BAAMuG,GAAG,GAAGvG,GAAG,IAAIsG,MAAnB;;AACA,0BAAI,CAACC,GAAL,EAAU;AACRrE,wBAAAA,OAAO,CAACC,IAAR,kDAAsDnC,GAAtD;AACD;;AACD,6BAAOsG,MAAM,CAACtG,GAAD,CAAb;AACD;AAP2B,mBAAvB,CAAP;AASD,iBAbe,CAAhB;kDAcOgG,W;;;;;;;;;;;;;;;;;;wBA5QE;AACT,aAAO,KAAKxE,IAAZ;AACD;;;wBAEa;AACZ,aAAO,KAAKG,OAAZ;AACD;;;wBAEY;AAAA,UACGvB,GADH,GACyC,IADzC,CACHoB,IADG;AAAA,UACcgF,GADd,GACyC,IADzC,CACQC,IADR;AAAA,UAC6B1F,OAD7B,GACyC,IADzC,CACmBU,QADnB;;AAEX,UAAMiF,QAAQ,GAAGxI,IAAG,CAAC,oBAAD,CAApB;;AACA,aAAO;AACL;AACAyI,QAAAA,SAAS,EAAEH,GAAG,IAAIA,GAAG,CAACI,EAFjB;;AAGL;AACAxG,QAAAA,GAAG,EAAEA,GAJA;;AAKL;AACAyG,QAAAA,IAAI,EAAE3I,IAAG,CAAC,eAAD,CANJ;;AAOL;AACA4I,QAAAA,IAAI,EAAEJ,QAAQ,IAAIA,QAAQ,CAACK,KAAT,CAAe,GAAf,EAAoB,CAApB,CARb;;AASL;AACAC,QAAAA,QAAQ,EAAE9I,IAAG,CAAC,mBAAD,CAVR;;AAWL;AACA+I,QAAAA,CAAC,EAAElG,OAZE;;AAaL;AACAmG,QAAAA,GAAG,EAAE;AAdA,OAAP;AAgBD;;;wBAEa;AACZ,aAAO,KAAKxF,OAAZ;AACD;;;EA/EqBnC,a;;AAgUxBM,SAAS,CAACsH,cAAV,GAA2B,KAA3B;AACAtH,SAAS,CAACI,OAAV,GAAoB,SAApB;;AAEAJ,SAAS,CAACQ,MAAV,GAAmB;AAAA,SAAMzB,IAAI,CAACoE,EAAL,EAAN;AAAA,CAAnB;;AAEAoE,MAAM,CAACC,OAAP,GAAiBxH,SAAjB","sourceRoot":"../lib","sourcesContent":["/**\n * @class TheClient\n */\n'use strict'\n\nconst argx = require('argx')\nconst asleep = require('asleep')\nconst cookies = require('browser-cookies')\nconst { restore, save } = require('bstorage')\nconst { get } = require('bwindow')\nconst qs = require('qs')\nconst { RFuncClient } = require('rfunc-client/shim/browser')\nconst io = require('socket.io-client')\nconst { isBrowser, isProduction, unlessProduction } = require('the-check')\nconst { ThePack } = require('the-pack')\nconst { resolve: resolveUrl } = require('url')\nconst uuid = require('uuid')\nconst IOEvents = require('./constants/IOEvents')\nconst {\n  asController,\n  debugController,\n  debugStream,\n  parseClientUrl,\n} = require('./helpers')\nconst { infoMix, pingPongMix, streamMix } = require('./mixins')\nconst debug = require('debug')('the:client')\nconst NAMESPACE = '/rpc'\n\nconst TheClientBase = [\n  pingPongMix,\n  infoMix,\n  streamMix,\n].reduce((Class, mix) => mix(Class), RFuncClient)\n\nconst { decode, encode } = new ThePack({})\n\n/** @lends TheClient */\nclass TheClient extends TheClientBase {\n  // noinspection ReservedWordAsName\n  /**\n   * Create the client instance\n   * @param {string} namespace\n   * @param {Object} [config={}]\n   * @returns {TheClient}\n   */\n  static for (namespace = 'default', config = {}) {\n    const key = [TheClient.CID_KEY, namespace].join('/').trim()\n    const cid = restore(key) || TheClient.newCID()\n    const client = new TheClient({ ...config, cid })\n    const isBrowser = !!get('document')\n    if (isBrowser) {\n      save(key, cid)\n      cookies.set(key, cid, {})\n    }\n    return client\n  }\n\n  constructor (url, config) {\n    const args = argx(arguments)\n    url = args.shift('string')\n    config = args.pop('object') || {}\n    if (!url) {\n      url = parseClientUrl(config)\n    }\n    const {\n      cid = TheClient.newCID(),\n      forceNewSocket = false,\n      onGone,\n      version = 'unknown',\n      ...restOptions\n    } = config\n    if (!url) {\n      throw new Error(`[TheClient] Failed to parse urls with args ${JSON.stringify(arguments)}`)\n    }\n    super(url, restOptions)\n    this._onGone = onGone\n    this._forceNewSocket = forceNewSocket\n    this._gone = false\n    this._controllers = {}\n    this._cid = cid\n    this._version = version\n    this._socket = null\n    this._closed = false\n  }\n\n  get cid () {\n    return this._cid\n  }\n\n  get closed () {\n    return this._closed\n  }\n\n  get scope () {\n    const { _cid: cid, _rpc: rpc, _version: version } = this\n    const language = get('navigator.language')\n    return {\n      /** Caller key */\n      callerKey: rpc && rpc.as,\n      /** Client ID */\n      cid: cid,\n      /** Host of client */\n      host: get('location.host'),\n      /** Detected lang */\n      lang: language && language.split('-')[0],\n      /** Connecting protocol */\n      protocol: get('location.protocol'),\n      /** Client instance version number */\n      v: version,\n      /** Via client */\n      via: 'client',\n    }\n  }\n\n  get socket () {\n    return this._socket\n  }\n\n  assertNotClosed () {\n    if (this.closed) {\n      throw new Error(`[TheClient] Already closed!`)\n    }\n  }\n\n  handleCallback (controllerName, handleName, data) {\n    const values = decode(data)\n    const controller = this._controllers[controllerName]\n    if (!controller) {\n      console.warn(`[TheClient] Callback controller not found: ${controllerName}`)\n      return\n    }\n    const callback = controller.callbacks[handleName]\n    if (!callback) {\n      console.warn(`[TheClient] Callback controller not found: ${controllerName}`)\n      return\n    }\n    unlessProduction(() => {\n      console.groupCollapsed(`[TheClient] Callback \\`${controllerName}.${handleName}()\\``)\n      console.log('Signature', `\\`${controllerName}.${handleName}()\\``)\n      console.log('Arguments', values)\n      console.groupEnd()\n    })\n    callback(...values)\n  }\n\n  markAsGone () {\n    if (this._gone) {\n      return\n    }\n    this._onGone && this._onGone()\n    this._gone = true\n  }\n\n  async close () {\n    this._closed = true\n    const socket = this._socket\n    if (socket) {\n      socket.close()\n    }\n  }\n\n  /**\n   * Invoke a method\n   * @param {string} moduleName\n   * @param {string} methodName - Name of method to invoke\n   * @param {...*} params - Parameters\n   */\n  async invoke (moduleName, methodName, ...params) {\n    this.assertNotClosed()\n    const { cid, socket } = this\n    const iid = uuid.v4() // Invocation id\n\n    return await new Promise((resolve, reject) => {\n      let keptGoneTimer = -1\n      let onReturn = (returned) => {\n        if (!onReturn) {\n          return null\n        }\n        returned = decode(returned)\n        if (returned.iid !== iid) {\n          return\n        }\n        socket.off(IOEvents.RPC_RETURN, onReturn)\n        onReturn = null\n        onKeep = null\n        clearTimeout(keptGoneTimer)\n        const { data, errors, ok } = returned\n        debug('rpc return', moduleName, methodName, returned)\n        if (ok) {\n          resolve(data)\n        } else {\n          const e = errors[0]\n          reject(e.message || e)\n        }\n      }\n      let onKeep = (kept) => {\n        if (!onKeep) {\n          return\n        }\n        kept = decode(kept)\n        if (kept.iid !== iid) {\n          return\n        }\n        socket.off(IOEvents.RPC_KEEP, onKeep)\n        onKeep = null\n        debug('rpc keep', moduleName, methodName)\n        clearTimeout(keptGoneTimer)\n        const { duration } = kept\n        keptGoneTimer = setTimeout(() => {\n          // TODO throw error?\n          console.error(`[TheClient] RPC call seems gone: \\`${moduleName}.${methodName}()\\``)\n        }, duration * 2)\n      }\n      socket.on(IOEvents.RPC_RETURN, onReturn)\n\n      socket.emit(IOEvents.RPC_CALL, encode({\n        cid,\n        iid,\n        methodName,\n        moduleName,\n        params,\n      }))\n      debug('rpc call', moduleName, methodName, params)\n    })\n  }\n\n  async newSocket () {\n    this.assertNotClosed()\n    const query = qs.stringify(this.scope)\n    const socket = io(resolveUrl(this.baseUrl, `${NAMESPACE}?${query}`), {\n      forceNew: this._forceNewSocket,\n    })\n    socket.on('disconnect', () => {\n      debug('disconnect')\n      if (this.closed) {\n        return\n      }\n      const goneTimer = setTimeout(\n        () => this.markAsGone(),\n        2 * 1000 + 2 * 1000 * Math.random()\n      )\n      const cancelGone = () => {\n        debug('cancelGone')\n        clearTimeout(goneTimer)\n        socket.off('connect', cancelGone)\n        socket.off('reconnect', cancelGone)\n      }\n      socket.once('connect', cancelGone)\n      socket.once('reconnect', cancelGone)\n      socket.connect()\n    })\n    await new Promise((resolve, reject) => {\n      socket.on('connect', () => {\n        debug('connect')\n        resolve(socket)\n      })\n      socket.on('error', (e) => {\n        debug('error', e)\n        reject(e)\n      })\n    })\n    return socket\n  }\n\n  /**\n   * Create an stream to server\n   * @param {string} name\n   * @param {Object} params - Stream params\n   * @param {Object} [options={}] - Optional setting\n   * @param {Boolean} [options.debug] - With debug mode\n   * @returns {*}\n   */\n  async stream (name, params = {}, options = {}) {\n    this.assertNotClosed()\n    const {\n      debug = !isProduction() && isBrowser(),\n    } = options\n    if (!this._socket) {\n      this._socket = await this.newSocket()\n    }\n    const stream = await this.openStream(name, params)\n    return debug ? debugStream(stream) : stream\n  }\n\n  /**\n   * Use a controller module\n   * @param {string} controllerName - Module name\n   * @param {Object} [options={}] - Optional setting\n   * @param {Boolean} [options.debug] - With debug mode\n   * @returns {*}\n   */\n  async use (controllerName, options = {}) {\n    this.assertNotClosed()\n    const {\n      debug = !isProduction() && isBrowser(),\n    } = options\n    let controller = this._controllers[controllerName]\n    if (!this._socket) {\n      this._socket = await this.newSocket()\n    }\n    if (!controller) {\n      const instance = await this.connect(controllerName)\n      const spec = await this.describe(controllerName)\n      const { cid } = this\n      controller = asController(instance, spec, { cid }, {\n        onToggleHandler: (handlerName, enabled) => {\n          const { socket } = this\n          const event = [IOEvents.CLIENT_CALLBACK, cid, controllerName, handlerName].join('/')\n          if (enabled) {\n            socket.on(event, (data) => {\n              this.handleCallback(controllerName, handlerName, data)\n            })\n          } else {\n            socket.off(event)\n          }\n        },\n      })\n      if (debug) {\n        controller = debugController(controller)\n      }\n      this._controllers[controllerName] = controller\n    }\n\n    return controller\n  }\n\n  /**\n   * Use all controller modules\n   * @param {Object} [options={}] - Optional setting\n   * @returns {Promise<Object>}\n   */\n  async useAll (options = {}) {\n    const serverInfo = await this.serverInfo()\n    const controllers = {}\n    const controllerSpecs = serverInfo.controllers\n    for (const { methods, name } of controllerSpecs) {\n      this.specs[name] = { methods, name }\n      controllers[name] = await this.use(name, options)\n    }\n    unlessProduction(() => {\n      if (typeof Proxy === 'undefined') {\n        return controllers\n      }\n      return new Proxy(controllers, {\n        get (target, key) {\n          const has = key in target\n          if (!has) {\n            console.warn(`[TheClient] Unknown controller name: \"${key}\"`)\n          }\n          return target[key]\n        },\n      })\n    })\n    return controllers\n  }\n}\n\nTheClient.RPC_ACTOR_NAME = 'rpc'\nTheClient.CID_KEY = 'the:cid'\n\nTheClient.newCID = () => uuid.v4()\n\nmodule.exports = TheClient\n"]}
\No newline at end of file