// Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; var __generator = (this && this.__generator) || function (thisArg, body) { var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; function verb(n) { return function (v) { return step([n, v]); }; } function step(op) { if (f) throw new TypeError("Generator is already executing."); while (_) try { 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; if (y = 0, t) op = [op[0] & 2, t.value]; switch (op[0]) { case 0: case 1: t = op; break; case 4: _.label++; return { value: op[1], done: false }; case 5: _.label++; y = op[1]; op = [0]; continue; case 7: op = _.ops.pop(); _.trys.pop(); continue; default: if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } if (t[2]) _.ops.pop(); _.trys.pop(); continue; } op = body.call(thisArg, _); } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; } }; import { LogLevel } from "./ILogger"; import { HttpTransportType, TransferFormat } from "./ITransport"; import { LongPollingTransport } from "./LongPollingTransport"; import { Arg, createLogger } from "./Utils"; import DefaultRequest from "./DefualtRequest"; import { ResponseType } from "./wx-request/model/ResponseType"; import { WxSocketTransport } from "./WxSocketTransport"; var MAX_REDIRECTS = 100; var WxSocketModule = WxSocketTransport; var LongPollingModule = LongPollingTransport; /** @private */ var HttpConnection = /** @class */ (function () { function HttpConnection(url, options) { if (options === void 0) { options = {}; } this.features = {}; Arg.isRequired(url, "url"); this.logger = createLogger(options.logger); options = options || {}; // ! 这里修改为自定义解析 和 默认传入 全路径方式 this.baseUrl = options.resolveUrl ? options.resolveUrl(url) : this.resolveUrl(url); options.logMessageContent = options.logMessageContent || false; // ! 修改 options 参数赋值方式 if (!options.WxSocket && wx) { options.WxSocket = WxSocketModule; } if (!options.LongPolling) { options.LongPolling = LongPollingModule; } this.request = options.request || new DefaultRequest({}, this.logger); this.connectionState = 2 /* Disconnected */; this.options = options; this.onreceive = null; this.onclose = null; } HttpConnection.prototype.start = function (transferFormat) { transferFormat = transferFormat || TransferFormat.Binary; Arg.isIn(transferFormat, TransferFormat, "transferFormat"); this.logger.log(LogLevel.Debug, "Starting connection with transfer format '" + TransferFormat[transferFormat] + "'.", TransferFormat); if (this.connectionState !== 2 /* Disconnected */) { return Promise.reject(new Error("Cannot start a connection that is not in the 'Disconnected' state. state is " + this.connectionState)); } this.connectionState = 0 /* Connecting */; this.startPromise = this.startInternal(transferFormat); return this.startPromise; }; HttpConnection.prototype.send = function (data) { if (this.connectionState !== 1 /* Connected */) { throw new Error("Cannot send data if the connection is not in the 'Connected' State."); } // Transport will not be null if state is connected return this.transport.send(data); }; HttpConnection.prototype.stop = function (error) { return __awaiter(this, void 0, void 0, function () { var e_1; return __generator(this, function (_a) { switch (_a.label) { case 0: this.connectionState = 2 /* Disconnected */; // Set error as soon as possible otherwise there is a race between // the transport closing and providing an error and the error from a close message // We would prefer the close message error. this.stopError = error; _a.label = 1; case 1: _a.trys.push([1, 3, , 4]); return [4 /*yield*/, this.startPromise]; case 2: _a.sent(); return [3 /*break*/, 4]; case 3: e_1 = _a.sent(); return [3 /*break*/, 4]; case 4: if (!this.transport) return [3 /*break*/, 6]; return [4 /*yield*/, this.transport.stop()]; case 5: _a.sent(); this.transport = undefined; _a.label = 6; case 6: return [2 /*return*/]; } }); }); }; HttpConnection.prototype.startInternal = function (transferFormat) { return __awaiter(this, void 0, void 0, function () { var url, negotiateResponse, redirects, _loop_1, this_1, state_1, e_2; var _this = this; return __generator(this, function (_a) { switch (_a.label) { case 0: url = this.baseUrl; this.accessTokenFactory = this.options.accessTokenFactory; this.socketUrlFactory = this.options.socketUrlFactory; _a.label = 1; case 1: _a.trys.push([1, 12, , 13]); if (!this.options.skipNegotiation) return [3 /*break*/, 5]; if (!(this.options.transport === HttpTransportType.WebSockets)) return [3 /*break*/, 3]; // No need to add a connection ID in this case this.transport = this.constructTransport(HttpTransportType.WebSockets); // We should just call connect directly in this case. // No fallback or negotiate in this case. return [4 /*yield*/, this.transport.connect({ url: url, header: {}, protocols: [], tcpNoDelay: true, transferFormat: transferFormat })]; case 2: // We should just call connect directly in this case. // No fallback or negotiate in this case. _a.sent(); return [3 /*break*/, 4]; case 3: throw Error("Negotiation can only be skipped when using the WxSocket transport directly."); case 4: return [3 /*break*/, 11]; case 5: negotiateResponse = null; redirects = 0; _loop_1 = function () { var accessToken_1; return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, this_1.getNegotiationResponse(url)]; case 1: negotiateResponse = _a.sent(); // the user tries to stop the connection when it is being started if (this_1.connectionState === 2 /* Disconnected */) { return [2 /*return*/, { value: void 0 }]; } if (negotiateResponse.error) { throw Error(negotiateResponse.error); } if (negotiateResponse.ProtocolVersion) { throw Error("检测到尝试连接到一个 非 ASP.NET Core 服务器。此客户端仅支持连接到ASP.NET Core 服务器。. See https://aka.ms/signalr-core-differences for details."); } if (negotiateResponse.url) { url = negotiateResponse.url; } if (negotiateResponse.accessToken) { accessToken_1 = negotiateResponse.accessToken; // ! 通过 /negotiate 接口返回的assessToken 仅支持 accessTokenFactory(),如果实现了 socketUrlFactory(),会忽略掉这个token this_1.accessTokenFactory = function () { return accessToken_1; }; } redirects++; return [2 /*return*/]; } }); }; this_1 = this; _a.label = 6; case 6: return [5 /*yield**/, _loop_1()]; case 7: state_1 = _a.sent(); if (typeof state_1 === "object") return [2 /*return*/, state_1.value]; _a.label = 8; case 8: if (negotiateResponse.url && redirects < MAX_REDIRECTS) return [3 /*break*/, 6]; _a.label = 9; case 9: if (redirects === MAX_REDIRECTS && negotiateResponse.url) { throw Error("Negotiate redirection limit exceeded. -fy : 超出协商重定向限制"); } return [4 /*yield*/, this.createTransport(url, this.options.transport, negotiateResponse, transferFormat)]; case 10: _a.sent(); _a.label = 11; case 11: if (this.transport instanceof LongPollingTransport) { this.features.inherentKeepAlive = true; } this.transport.onreceive = this.onreceive; this.transport.onclose = function (e) { return _this.stopConnection(e); }; // only change the state if we were connecting to not overwrite // the state if the connection is already marked as Disconnected this.changeState(0 /* Connecting */, 1 /* Connected */); return [2 /*return*/]; case 12: e_2 = _a.sent(); this.logger.log(LogLevel.Error, "Failed to start the connection: ", e_2); this.connectionState = 2 /* Disconnected */; this.transport = undefined; throw e_2; case 13: return [2 /*return*/]; } }); }); }; HttpConnection.prototype.getNegotiationResponse = function (url) { return __awaiter(this, void 0, void 0, function () { var headers, token, negotiateUrl, response, e_3; var _a; return __generator(this, function (_b) { switch (_b.label) { case 0: if (!this.accessTokenFactory) return [3 /*break*/, 2]; return [4 /*yield*/, this.accessTokenFactory()]; case 1: token = _b.sent(); if (token) { headers = (_a = {}, _a["Authorization"] = "Bearer " + token, _a); } _b.label = 2; case 2: negotiateUrl = this.resolveNegotiateUrl(url); this.logger.log(LogLevel.Debug, "Sending negotiation request: " + negotiateUrl + "."); _b.label = 3; case 3: _b.trys.push([3, 5, , 6]); return [4 /*yield*/, this.request.post(negotiateUrl, {}, { headers: headers, responseType: ResponseType.TEXT })]; case 4: response = _b.sent(); if (response.statusCode !== 200) { throw Error("Unexpected status code returned from negotiate " + response.statusCode); } return [2 /*return*/, JSON.parse(response.data)]; case 5: e_3 = _b.sent(); this.logger.log(LogLevel.Error, "Failed to complete negotiation with the server: ", e_3); throw e_3; case 6: return [2 /*return*/]; } }); }); }; HttpConnection.prototype.createConnectUrl = function (url, connectionId) { if (!connectionId) { return url; } return url + (url.indexOf("?") === -1 ? "?" : "&") + ("id=" + connectionId); }; HttpConnection.prototype.createTransport = function (url, requestedTransport, negotiateResponse, requestedTransferFormat) { return __awaiter(this, void 0, void 0, function () { var connectUrl, transports, _i, transports_1, endpoint, transport, ex_1; return __generator(this, function (_a) { switch (_a.label) { case 0: connectUrl = this.createConnectUrl(url, negotiateResponse.connectionId); if (!this.isITransport(requestedTransport)) return [3 /*break*/, 2]; this.logger.log(LogLevel.Debug, "Connection was provided an instance of ITransport, using that directly."); this.transport = requestedTransport; return [4 /*yield*/, this.transport.connect({ url: connectUrl, transferFormat: requestedTransferFormat })]; case 1: _a.sent(); // only change the state if we were connecting to not overwrite // the state if the connection is already marked as Disconnected this.changeState(0 /* Connecting */, 1 /* Connected */); return [2 /*return*/]; case 2: transports = negotiateResponse.availableTransports || []; _i = 0, transports_1 = transports; _a.label = 3; case 3: if (!(_i < transports_1.length)) return [3 /*break*/, 9]; endpoint = transports_1[_i]; this.connectionState = 0 /* Connecting */; transport = this.resolveTransport(endpoint, requestedTransport, requestedTransferFormat); if (!(typeof transport === "number")) return [3 /*break*/, 8]; this.transport = this.constructTransport(transport); if (!!negotiateResponse.connectionId) return [3 /*break*/, 5]; return [4 /*yield*/, this.getNegotiationResponse(url)]; case 4: negotiateResponse = _a.sent(); connectUrl = this.createConnectUrl(url, negotiateResponse.connectionId); _a.label = 5; case 5: _a.trys.push([5, 7, , 8]); return [4 /*yield*/, this.transport.connect({ url: connectUrl, transferFormat: requestedTransferFormat })]; case 6: _a.sent(); this.changeState(0 /* Connecting */, 1 /* Connected */); return [2 /*return*/]; case 7: ex_1 = _a.sent(); this.logger.log(LogLevel.Error, "Failed to start the transport '" + HttpTransportType[transport] + "':", ex_1); this.connectionState = 2 /* Disconnected */; negotiateResponse.connectionId = undefined; return [3 /*break*/, 8]; case 8: _i++; return [3 /*break*/, 3]; case 9: throw new Error("Unable to initialize any of the available transports."); } }); }); }; /** * * @description 这里对原来的实例化方式进行了改写,如果传入的是实例化完成的 Transport ,将直接返回 * 如果是传入继承 Transport的 class,将执行 new Transport(options) * - 这里对原生的多项入参合并成了options(这点差异需要注意) * @private * @param {HttpTransportType} transport * @returns * @memberof HttpConnection */ HttpConnection.prototype.constructTransport = function (transport) { var _a = this.options, WxSocket = _a.WxSocket, LongPolling = _a.LongPolling, wxSocketTransportOptions = _a.wxSocketTransportOptions, longPollingTransportOptions = _a.longPollingTransportOptions; switch (transport) { case HttpTransportType.WebSockets: // wx socket 方式 if (WxSocket instanceof WxSocketTransport) { return WxSocket; } else { return new WxSocket(wxSocketTransportOptions ? wxSocketTransportOptions : { // token 工厂 accessTokenFactory: this.accessTokenFactory, // socket 单独实现一个socket url factory(用于后端改了 accecc_token 参数名的场景) socketUrlFactory: this.socketUrlFactory, // logger logger: this.logger, logMessageContent: this.options.logMessageContent || false, /** 是否允许替换socket连接 * * 小程序 版本 < 1.7.0 时, 最多允许存在一个socket连接, 此参数用于是否允许在这个情况下,替换这个打开的socket */ allowReplaceSocket: true, /** 是否启用消息队列缓存连接建立前消息,并在建立连接后发送 */ enableMessageQueue: this.options.enableMessageQueue == undefined ? true : this.options.enableMessageQueue, /** 重连设置 */ reconnect: { enable: true, max: 3 } }); } case HttpTransportType.LongPolling: // 长轮询方式 if (LongPolling instanceof LongPollingTransport) { return LongPolling; } else { return new LongPolling(longPollingTransportOptions ? longPollingTransportOptions : { request: this.request, accessTokenFactory: this.accessTokenFactory, logger: this.logger, logMessageContent: this.options.logMessageContent || false }); } default: throw new Error("Unknown transport: " + transport + "."); } }; HttpConnection.prototype.resolveTransport = function (endpoint, requestedTransport, requestedTransferFormat) { var transport = HttpTransportType[endpoint.transport]; if (transport === null || transport === undefined) { this.logger.log(LogLevel.Debug, "Skipping transport '" + endpoint.transport + "' because it is not supported by this client."); } else { var transferFormats = endpoint.transferFormats.map(function (s) { return TransferFormat[s]; }); if (transportMatches(requestedTransport, transport)) { if (transferFormats.indexOf(requestedTransferFormat) >= 0) { if (transport === HttpTransportType.WebSockets && !this.options.WxSocket) { this.logger.log(LogLevel.Debug, "Skipping transport '" + HttpTransportType[transport] + "' because it is not supported in your environment.'"); } else { this.logger.log(LogLevel.Debug, "Selecting transport '" + HttpTransportType[transport] + "'."); return transport; } } else { this.logger.log(LogLevel.Debug, "Skipping transport '" + HttpTransportType[transport] + "' because it does not support the requested transfer format '" + TransferFormat[requestedTransferFormat] + "'."); } } else { this.logger.log(LogLevel.Debug, "Skipping transport '" + HttpTransportType[transport] + "' because it was disabled by the client."); } } return null; }; HttpConnection.prototype.isITransport = function (transport) { return transport && typeof transport === "object" && "connect" in transport; }; HttpConnection.prototype.changeState = function (from, to) { if (this.connectionState === from) { this.connectionState = to; return true; } return false; }; HttpConnection.prototype.stopConnection = function (error) { this.transport = undefined; // If we have a stopError, it takes precedence over the error from the transport error = this.stopError || error; if (error) { this.logger.log(LogLevel.Error, "Connection disconnected with error '" + error + "'."); } else { this.logger.log(LogLevel.Information, "Connection disconnected."); } this.connectionState = 2 /* Disconnected */; if (this.onclose) { this.onclose(error); } }; /** * ! 由于小程序内必须指定 BaseUrl 关系,这里如果不是全路径的话,暂时直接抛出异常 * @param url */ HttpConnection.prototype.resolveUrl = function (url) { // startsWith is not supported in IE if (url.lastIndexOf("https://", 0) === 0 || url.lastIndexOf("http://", 0) === 0) { return url; } else { throw new Error("HttpConnection. \u89E3\u6790url\u9519\u8BEF,\u5C0F\u7A0B\u5E8F\u5185\u9700\u8981\u4F20\u5165\u5168\u8DEF\u5F84 ->link: " + url); } }; HttpConnection.prototype.resolveNegotiateUrl = function (url) { var index = url.indexOf("?"); var negotiateUrl = url.substring(0, index === -1 ? url.length : index); if (negotiateUrl[negotiateUrl.length - 1] !== "/") { negotiateUrl += "/"; } negotiateUrl += "negotiate"; negotiateUrl += index === -1 ? "" : url.substring(index); return negotiateUrl; }; return HttpConnection; }()); export { HttpConnection }; function transportMatches(requestedTransport, actualTransport) { return !requestedTransport || (actualTransport & requestedTransport) !== 0; }