WebSocketTransport.js 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. "use strict";
  2. // Copyright (c) .NET Foundation. All rights reserved.
  3. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  4. var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
  5. return new (P || (P = Promise))(function (resolve, reject) {
  6. function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
  7. function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
  8. function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
  9. step((generator = generator.apply(thisArg, _arguments || [])).next());
  10. });
  11. };
  12. var __generator = (this && this.__generator) || function (thisArg, body) {
  13. var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
  14. return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
  15. function verb(n) { return function (v) { return step([n, v]); }; }
  16. function step(op) {
  17. if (f) throw new TypeError("Generator is already executing.");
  18. while (_) try {
  19. if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
  20. if (y = 0, t) op = [op[0] & 2, t.value];
  21. switch (op[0]) {
  22. case 0: case 1: t = op; break;
  23. case 4: _.label++; return { value: op[1], done: false };
  24. case 5: _.label++; y = op[1]; op = [0]; continue;
  25. case 7: op = _.ops.pop(); _.trys.pop(); continue;
  26. default:
  27. if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
  28. if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
  29. if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
  30. if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
  31. if (t[2]) _.ops.pop();
  32. _.trys.pop(); continue;
  33. }
  34. op = body.call(thisArg, _);
  35. } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
  36. if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
  37. }
  38. };
  39. Object.defineProperty(exports, "__esModule", { value: true });
  40. var ILogger_1 = require("./ILogger");
  41. var ITransport_1 = require("./ITransport");
  42. var Utils_1 = require("./Utils");
  43. /** @private */
  44. var WebSocketTransport = /** @class */ (function () {
  45. function WebSocketTransport(httpClient, accessTokenFactory, logger, logMessageContent, webSocketConstructor) {
  46. this.logger = logger;
  47. this.accessTokenFactory = accessTokenFactory;
  48. this.logMessageContent = logMessageContent;
  49. this.webSocketConstructor = webSocketConstructor;
  50. this.httpClient = httpClient;
  51. this.onreceive = null;
  52. this.onclose = null;
  53. }
  54. WebSocketTransport.prototype.connect = function (url, transferFormat) {
  55. return __awaiter(this, void 0, void 0, function () {
  56. var token;
  57. var _this = this;
  58. return __generator(this, function (_a) {
  59. switch (_a.label) {
  60. case 0:
  61. Utils_1.Arg.isRequired(url, "url");
  62. Utils_1.Arg.isRequired(transferFormat, "transferFormat");
  63. Utils_1.Arg.isIn(transferFormat, ITransport_1.TransferFormat, "transferFormat");
  64. this.logger.log(ILogger_1.LogLevel.Trace, "(WebSockets transport) Connecting.");
  65. if (!this.accessTokenFactory) return [3 /*break*/, 2];
  66. return [4 /*yield*/, this.accessTokenFactory()];
  67. case 1:
  68. token = _a.sent();
  69. if (token) {
  70. url += (url.indexOf("?") < 0 ? "?" : "&") + ("access_token=" + encodeURIComponent(token));
  71. }
  72. _a.label = 2;
  73. case 2: return [2 /*return*/, new Promise(function (resolve, reject) {
  74. url = url.replace(/^http/, "ws");
  75. var webSocket;
  76. var cookies = _this.httpClient.getCookieString(url);
  77. if (typeof window === "undefined" && cookies) {
  78. // Only pass cookies when in non-browser environments
  79. webSocket = new _this.webSocketConstructor(url, undefined, {
  80. headers: {
  81. Cookie: "" + cookies,
  82. },
  83. });
  84. }
  85. if (!webSocket) {
  86. // Chrome is not happy with passing 'undefined' as protocol
  87. webSocket = new _this.webSocketConstructor(url);
  88. }
  89. if (transferFormat === ITransport_1.TransferFormat.Binary) {
  90. webSocket.binaryType = "arraybuffer";
  91. }
  92. // tslint:disable-next-line:variable-name
  93. webSocket.onopen = function (_event) {
  94. _this.logger.log(ILogger_1.LogLevel.Information, "WebSocket connected to " + url + ".");
  95. _this.webSocket = webSocket;
  96. resolve();
  97. };
  98. webSocket.onerror = function (event) {
  99. var error = null;
  100. // ErrorEvent is a browser only type we need to check if the type exists before using it
  101. if (typeof ErrorEvent !== "undefined" && event instanceof ErrorEvent) {
  102. error = event.error;
  103. }
  104. reject(error);
  105. };
  106. webSocket.onmessage = function (message) {
  107. _this.logger.log(ILogger_1.LogLevel.Trace, "(WebSockets transport) data received. " + Utils_1.getDataDetail(message.data, _this.logMessageContent) + ".");
  108. if (_this.onreceive) {
  109. _this.onreceive(message.data);
  110. }
  111. };
  112. webSocket.onclose = function (event) { return _this.close(event); };
  113. })];
  114. }
  115. });
  116. });
  117. };
  118. WebSocketTransport.prototype.send = function (data) {
  119. if (this.webSocket && this.webSocket.readyState === this.webSocketConstructor.OPEN) {
  120. this.logger.log(ILogger_1.LogLevel.Trace, "(WebSockets transport) sending data. " + Utils_1.getDataDetail(data, this.logMessageContent) + ".");
  121. this.webSocket.send(data);
  122. return Promise.resolve();
  123. }
  124. return Promise.reject("WebSocket is not in the OPEN state");
  125. };
  126. WebSocketTransport.prototype.stop = function () {
  127. if (this.webSocket) {
  128. // Clear websocket handlers because we are considering the socket closed now
  129. this.webSocket.onclose = function () { };
  130. this.webSocket.onmessage = function () { };
  131. this.webSocket.onerror = function () { };
  132. this.webSocket.close();
  133. this.webSocket = undefined;
  134. // Manually invoke onclose callback inline so we know the HttpConnection was closed properly before returning
  135. // This also solves an issue where websocket.onclose could take 18+ seconds to trigger during network disconnects
  136. this.close(undefined);
  137. }
  138. return Promise.resolve();
  139. };
  140. WebSocketTransport.prototype.close = function (event) {
  141. // webSocket will be null if the transport did not start successfully
  142. this.logger.log(ILogger_1.LogLevel.Trace, "(WebSockets transport) socket closed.");
  143. if (this.onclose) {
  144. if (event && (event.wasClean === false || event.code !== 1000)) {
  145. this.onclose(new Error("WebSocket closed with status code: " + event.code + " (" + event.reason + ")."));
  146. }
  147. else {
  148. this.onclose();
  149. }
  150. }
  151. };
  152. return WebSocketTransport;
  153. }());
  154. exports.WebSocketTransport = WebSocketTransport;
  155. //# sourceMappingURL=WebSocketTransport.js.map