WebSocketTransport.js 8.6 KB

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