UNPKG

50.3 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(data) {
168 data = decode(data);
169 var _data = data,
170 controllerName = _data.controller,
171 name = _data.name,
172 values = _data.values;
173 var controller = this._controllers[controllerName];
174
175 if (!controller) {
176 return;
177 }
178
179 var callback = controller.callbacks[name];
180
181 if (!callback) {
182 return;
183 }
184
185 unlessProduction(function () {
186 console.groupCollapsed("[TheClient] Callback `".concat(controllerName, ".").concat(name, "()`"));
187 console.log('Signature', "`".concat(controllerName, ".").concat(name, "()`"));
188 console.log('Arguments', values);
189 console.groupEnd();
190 });
191 callback.apply(void 0, (0, _toConsumableArray2.default)(values));
192 }
193 }, {
194 key: "markAsGone",
195 value: function markAsGone() {
196 if (this._gone) {
197 return;
198 }
199
200 this._onGone && this._onGone();
201 this._gone = true;
202 }
203 }, {
204 key: "close",
205 value: function () {
206 var _close = (0, _asyncToGenerator2.default)(
207 /*#__PURE__*/
208 _regenerator.default.mark(function _callee() {
209 var socket;
210 return _regenerator.default.wrap(function _callee$(_context) {
211 while (1) {
212 switch (_context.prev = _context.next) {
213 case 0:
214 this._closed = true;
215 socket = this._socket;
216
217 if (socket) {
218 socket.close();
219 }
220
221 case 3:
222 case "end":
223 return _context.stop();
224 }
225 }
226 }, _callee, this);
227 }));
228
229 return function close() {
230 return _close.apply(this, arguments);
231 };
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 return function invoke(_x, _x2) {
345 return _invoke.apply(this, arguments);
346 };
347 }()
348 }, {
349 key: "newSocket",
350 value: function () {
351 var _newSocket = (0, _asyncToGenerator2.default)(
352 /*#__PURE__*/
353 _regenerator.default.mark(function _callee3() {
354 var _this2 = this;
355
356 var query, socket;
357 return _regenerator.default.wrap(function _callee3$(_context3) {
358 while (1) {
359 switch (_context3.prev = _context3.next) {
360 case 0:
361 this.assertNotClosed();
362 query = qs.stringify(this.scope);
363 socket = io(resolveUrl(this.baseUrl, "".concat(NAMESPACE, "?").concat(query)), {
364 forceNew: this._forceNewSocket
365 });
366 socket.on(IOEvents.CLIENT_CALLBACK, function (data) {
367 return _this2.handleCallback(data);
368 });
369 socket.on('disconnect', function () {
370 debug('disconnect');
371
372 if (_this2.closed) {
373 return;
374 }
375
376 var goneTimer = setTimeout(function () {
377 return _this2.markAsGone();
378 }, 2 * 1000 + 2 * 1000 * Math.random());
379
380 var cancelGone = function cancelGone() {
381 debug('cancelGone');
382 clearTimeout(goneTimer);
383 socket.off('connect', cancelGone);
384 socket.off('reconnect', cancelGone);
385 };
386
387 socket.once('connect', cancelGone);
388 socket.once('reconnect', cancelGone);
389 socket.connect();
390 });
391 _context3.next = 7;
392 return new Promise(function (resolve, reject) {
393 socket.on('connect', function () {
394 debug('connect');
395 resolve(socket);
396 });
397 socket.on('error', function (e) {
398 debug('error', e);
399 reject(e);
400 });
401 });
402
403 case 7:
404 return _context3.abrupt("return", socket);
405
406 case 8:
407 case "end":
408 return _context3.stop();
409 }
410 }
411 }, _callee3, this);
412 }));
413
414 return function newSocket() {
415 return _newSocket.apply(this, arguments);
416 };
417 }()
418 /**
419 * Create an stream to server
420 * @param {string} name
421 * @param {Object} params - Stream params
422 * @param {Object} [options={}] - Optional setting
423 * @param {Boolean} [options.debug] - With debug mode
424 * @returns {*}
425 */
426
427 }, {
428 key: "stream",
429 value: function () {
430 var _stream = (0, _asyncToGenerator2.default)(
431 /*#__PURE__*/
432 _regenerator.default.mark(function _callee4(name) {
433 var params,
434 options,
435 _options$debug,
436 debug,
437 stream,
438 _args4 = arguments;
439
440 return _regenerator.default.wrap(function _callee4$(_context4) {
441 while (1) {
442 switch (_context4.prev = _context4.next) {
443 case 0:
444 params = _args4.length > 1 && _args4[1] !== undefined ? _args4[1] : {};
445 options = _args4.length > 2 && _args4[2] !== undefined ? _args4[2] : {};
446 this.assertNotClosed();
447 _options$debug = options.debug, debug = _options$debug === void 0 ? !isProduction() && isBrowser() : _options$debug;
448
449 if (this._socket) {
450 _context4.next = 8;
451 break;
452 }
453
454 _context4.next = 7;
455 return this.newSocket();
456
457 case 7:
458 this._socket = _context4.sent;
459
460 case 8:
461 _context4.next = 10;
462 return this.openStream(name, params);
463
464 case 10:
465 stream = _context4.sent;
466 return _context4.abrupt("return", debug ? debugStream(stream) : stream);
467
468 case 12:
469 case "end":
470 return _context4.stop();
471 }
472 }
473 }, _callee4, this);
474 }));
475
476 return function stream(_x3) {
477 return _stream.apply(this, arguments);
478 };
479 }()
480 /**
481 * Use a controller module
482 * @param {string} controllerName - Module name
483 * @param {Object} [options={}] - Optional setting
484 * @param {Boolean} [options.debug] - With debug mode
485 * @returns {*}
486 */
487
488 }, {
489 key: "use",
490 value: function () {
491 var _use = (0, _asyncToGenerator2.default)(
492 /*#__PURE__*/
493 _regenerator.default.mark(function _callee5(controllerName) {
494 var options,
495 _options$debug2,
496 debug,
497 controller,
498 instance,
499 spec,
500 cid,
501 _args5 = arguments;
502
503 return _regenerator.default.wrap(function _callee5$(_context5) {
504 while (1) {
505 switch (_context5.prev = _context5.next) {
506 case 0:
507 options = _args5.length > 1 && _args5[1] !== undefined ? _args5[1] : {};
508 this.assertNotClosed();
509 _options$debug2 = options.debug, debug = _options$debug2 === void 0 ? !isProduction() && isBrowser() : _options$debug2;
510 controller = this._controllers[controllerName];
511
512 if (this._socket) {
513 _context5.next = 8;
514 break;
515 }
516
517 _context5.next = 7;
518 return this.newSocket();
519
520 case 7:
521 this._socket = _context5.sent;
522
523 case 8:
524 if (controller) {
525 _context5.next = 19;
526 break;
527 }
528
529 _context5.next = 11;
530 return this.connect(controllerName);
531
532 case 11:
533 instance = _context5.sent;
534 _context5.next = 14;
535 return this.describe(controllerName);
536
537 case 14:
538 spec = _context5.sent;
539 cid = this.cid;
540 controller = asController(instance, spec, {
541 cid: cid
542 });
543
544 if (debug) {
545 controller = debugController(controller);
546 }
547
548 this._controllers[controllerName] = controller;
549
550 case 19:
551 return _context5.abrupt("return", controller);
552
553 case 20:
554 case "end":
555 return _context5.stop();
556 }
557 }
558 }, _callee5, this);
559 }));
560
561 return function use(_x4) {
562 return _use.apply(this, arguments);
563 };
564 }()
565 /**
566 * Use all controller modules
567 * @param {Object} [options={}] - Optional setting
568 * @returns {Promise<Object>}
569 */
570
571 }, {
572 key: "useAll",
573 value: function () {
574 var _useAll = (0, _asyncToGenerator2.default)(
575 /*#__PURE__*/
576 _regenerator.default.mark(function _callee6() {
577 var options,
578 serverInfo,
579 controllers,
580 controllerSpecs,
581 _iteratorNormalCompletion,
582 _didIteratorError,
583 _iteratorError,
584 _iterator,
585 _step,
586 _ref3,
587 methods,
588 name,
589 _args6 = arguments;
590
591 return _regenerator.default.wrap(function _callee6$(_context6) {
592 while (1) {
593 switch (_context6.prev = _context6.next) {
594 case 0:
595 options = _args6.length > 0 && _args6[0] !== undefined ? _args6[0] : {};
596 _context6.next = 3;
597 return this.serverInfo();
598
599 case 3:
600 serverInfo = _context6.sent;
601 controllers = {};
602 controllerSpecs = serverInfo.controllers;
603 _iteratorNormalCompletion = true;
604 _didIteratorError = false;
605 _iteratorError = undefined;
606 _context6.prev = 9;
607 _iterator = controllerSpecs[Symbol.iterator]();
608
609 case 11:
610 if (_iteratorNormalCompletion = (_step = _iterator.next()).done) {
611 _context6.next = 21;
612 break;
613 }
614
615 _ref3 = _step.value;
616 methods = _ref3.methods, name = _ref3.name;
617 this.specs[name] = {
618 methods: methods,
619 name: name
620 };
621 _context6.next = 17;
622 return this.use(name, options);
623
624 case 17:
625 controllers[name] = _context6.sent;
626
627 case 18:
628 _iteratorNormalCompletion = true;
629 _context6.next = 11;
630 break;
631
632 case 21:
633 _context6.next = 27;
634 break;
635
636 case 23:
637 _context6.prev = 23;
638 _context6.t0 = _context6["catch"](9);
639 _didIteratorError = true;
640 _iteratorError = _context6.t0;
641
642 case 27:
643 _context6.prev = 27;
644 _context6.prev = 28;
645
646 if (!_iteratorNormalCompletion && _iterator.return != null) {
647 _iterator.return();
648 }
649
650 case 30:
651 _context6.prev = 30;
652
653 if (!_didIteratorError) {
654 _context6.next = 33;
655 break;
656 }
657
658 throw _iteratorError;
659
660 case 33:
661 return _context6.finish(30);
662
663 case 34:
664 return _context6.finish(27);
665
666 case 35:
667 unlessProduction(function () {
668 if (typeof Proxy === 'undefined') {
669 return controllers;
670 }
671
672 return new Proxy(controllers, {
673 get: function get(target, key) {
674 var has = key in target;
675
676 if (!has) {
677 console.warn("[TheClient] Unknown controller name: \"".concat(key, "\""));
678 }
679
680 return target[key];
681 }
682 });
683 });
684 return _context6.abrupt("return", controllers);
685
686 case 37:
687 case "end":
688 return _context6.stop();
689 }
690 }
691 }, _callee6, this, [[9, 23, 27, 35], [28,, 30, 34]]);
692 }));
693
694 return function useAll() {
695 return _useAll.apply(this, arguments);
696 };
697 }()
698 }, {
699 key: "cid",
700 get: function get() {
701 return this._cid;
702 }
703 }, {
704 key: "closed",
705 get: function get() {
706 return this._closed;
707 }
708 }, {
709 key: "scope",
710 get: function get() {
711 var cid = this._cid,
712 rpc = this._rpc,
713 version = this._version;
714
715 var language = _get('navigator.language');
716
717 return {
718 /** Caller key */
719 callerKey: rpc && rpc.as,
720
721 /** Client ID */
722 cid: cid,
723
724 /** Host of client */
725 host: _get('location.host'),
726
727 /** Detected lang */
728 lang: language && language.split('-')[0],
729
730 /** Connecting protocol */
731 protocol: _get('location.protocol'),
732
733 /** Client instance version number */
734 v: version,
735
736 /** Via client */
737 via: 'client'
738 };
739 }
740 }, {
741 key: "socket",
742 get: function get() {
743 return this._socket;
744 }
745 }]);
746 return TheClient;
747}(TheClientBase);
748
749TheClient.RPC_ACTOR_NAME = 'rpc';
750TheClient.CID_KEY = 'the:cid';
751
752TheClient.newCID = function () {
753 return uuid.v4();
754};
755
756module.exports = TheClient;
757//# 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","data","controllerName","controller","name","values","callback","callbacks","console","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","CLIENT_CALLBACK","handleCallback","goneTimer","markAsGone","Math","random","cancelGone","once","connect","options","newSocket","openStream","stream","instance","describe","spec","serverInfo","controllers","controllerSpecs","methods","specs","use","Proxy","target","has","warn","rpc","_rpc","language","callerKey","as","host","lang","split","protocol","v","via","RPC_ACTOR_NAME","module","exports"],"mappings":"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","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 (data) {\n    data = decode(data)\n    const { controller: controllerName, name, values } = data\n    const controller = this._controllers[controllerName]\n    if (!controller) {\n      return\n    }\n    const callback = controller.callbacks[name]\n    if (!callback) {\n      return\n    }\n    unlessProduction(() => {\n      console.groupCollapsed(`[TheClient] Callback \\`${controllerName}.${name}()\\``)\n      console.log('Signature', `\\`${controllerName}.${name}()\\``)\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(IOEvents.CLIENT_CALLBACK, (data) => this.handleCallback(data))\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      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