UNPKG

builder-util-runtime

Version:

HTTP utilities. Used by [electron-builder](https://github.com/electron-userland/electron-builder).

428 lines (426 loc) 17.6 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.safeStringifyJson = exports.configureRequestOptions = exports.safeGetHeader = exports.DigestTransform = exports.configureRequestUrl = exports.configureRequestOptionsFromUrl = exports.HttpExecutor = exports.parseJson = exports.HttpError = exports.createHttpError = void 0; const crypto_1 = require("crypto"); const debug_1 = require("debug"); const fs_1 = require("fs"); const stream_1 = require("stream"); const url_1 = require("url"); const CancellationToken_1 = require("./CancellationToken"); const index_1 = require("./index"); const ProgressCallbackTransform_1 = require("./ProgressCallbackTransform"); const debug = (0, debug_1.default)("electron-builder"); function createHttpError(response, description = null) { return new HttpError(response.statusCode || -1, `${response.statusCode} ${response.statusMessage}` + (description == null ? "" : "\n" + JSON.stringify(description, null, " ")) + "\nHeaders: " + safeStringifyJson(response.headers), description); } exports.createHttpError = createHttpError; const HTTP_STATUS_CODES = new Map([ [429, "Too many requests"], [400, "Bad request"], [403, "Forbidden"], [404, "Not found"], [405, "Method not allowed"], [406, "Not acceptable"], [408, "Request timeout"], [413, "Request entity too large"], [500, "Internal server error"], [502, "Bad gateway"], [503, "Service unavailable"], [504, "Gateway timeout"], [505, "HTTP version not supported"], ]); class HttpError extends Error { constructor(statusCode, message = `HTTP error: ${HTTP_STATUS_CODES.get(statusCode) || statusCode}`, description = null) { super(message); this.statusCode = statusCode; this.description = description; this.name = "HttpError"; this.code = `HTTP_ERROR_${statusCode}`; } isServerError() { return this.statusCode >= 500 && this.statusCode <= 599; } } exports.HttpError = HttpError; function parseJson(result) { return result.then(it => (it == null || it.length === 0 ? null : JSON.parse(it))); } exports.parseJson = parseJson; class HttpExecutor { constructor() { this.maxRedirects = 10; } request(options, cancellationToken = new CancellationToken_1.CancellationToken(), data) { configureRequestOptions(options); const json = data == null ? undefined : JSON.stringify(data); const encodedData = json ? Buffer.from(json) : undefined; if (encodedData != null) { debug(json); const { headers, ...opts } = options; options = { method: "post", headers: { "Content-Type": "application/json", "Content-Length": encodedData.length, ...headers, }, ...opts, }; } return this.doApiRequest(options, cancellationToken, it => it.end(encodedData)); } doApiRequest(options, cancellationToken, requestProcessor, redirectCount = 0) { if (debug.enabled) { debug(`Request: ${safeStringifyJson(options)}`); } return cancellationToken.createPromise((resolve, reject, onCancel) => { const request = this.createRequest(options, (response) => { try { this.handleResponse(response, options, cancellationToken, resolve, reject, redirectCount, requestProcessor); } catch (e) { reject(e); } }); this.addErrorAndTimeoutHandlers(request, reject, options.timeout); this.addRedirectHandlers(request, options, reject, redirectCount, options => { this.doApiRequest(options, cancellationToken, requestProcessor, redirectCount).then(resolve).catch(reject); }); requestProcessor(request, reject); onCancel(() => request.abort()); }); } // noinspection JSUnusedLocalSymbols // eslint-disable-next-line addRedirectHandlers(request, options, reject, redirectCount, handler) { // not required for NodeJS } addErrorAndTimeoutHandlers(request, reject, timeout = 60 * 1000) { this.addTimeOutHandler(request, reject, timeout); request.on("error", reject); request.on("aborted", () => { reject(new Error("Request has been aborted by the server")); }); } handleResponse(response, options, cancellationToken, resolve, reject, redirectCount, requestProcessor) { var _a; if (debug.enabled) { debug(`Response: ${response.statusCode} ${response.statusMessage}, request options: ${safeStringifyJson(options)}`); } // we handle any other >= 400 error on request end (read detailed message in the response body) if (response.statusCode === 404) { // error is clear, we don't need to read detailed error description reject(createHttpError(response, `method: ${options.method || "GET"} url: ${options.protocol || "https:"}//${options.hostname}${options.port ? `:${options.port}` : ""}${options.path} Please double check that your authentication token is correct. Due to security reasons, actual status maybe not reported, but 404. `)); return; } else if (response.statusCode === 204) { // on DELETE request resolve(); return; } const code = (_a = response.statusCode) !== null && _a !== void 0 ? _a : 0; const shouldRedirect = code >= 300 && code < 400; const redirectUrl = safeGetHeader(response, "location"); if (shouldRedirect && redirectUrl != null) { if (redirectCount > this.maxRedirects) { reject(this.createMaxRedirectError()); return; } this.doApiRequest(HttpExecutor.prepareRedirectUrlOptions(redirectUrl, options), cancellationToken, requestProcessor, redirectCount).then(resolve).catch(reject); return; } response.setEncoding("utf8"); let data = ""; response.on("error", reject); response.on("data", (chunk) => (data += chunk)); response.on("end", () => { try { if (response.statusCode != null && response.statusCode >= 400) { const contentType = safeGetHeader(response, "content-type"); const isJson = contentType != null && (Array.isArray(contentType) ? contentType.find(it => it.includes("json")) != null : contentType.includes("json")); reject(createHttpError(response, `method: ${options.method || "GET"} url: ${options.protocol || "https:"}//${options.hostname}${options.port ? `:${options.port}` : ""}${options.path} Data: ${isJson ? JSON.stringify(JSON.parse(data)) : data} `)); } else { resolve(data.length === 0 ? null : data); } } catch (e) { reject(e); } }); } async downloadToBuffer(url, options) { return await options.cancellationToken.createPromise((resolve, reject, onCancel) => { const responseChunks = []; const requestOptions = { headers: options.headers || undefined, // because PrivateGitHubProvider requires HttpExecutor.prepareRedirectUrlOptions logic, so, we need to redirect manually redirect: "manual", }; configureRequestUrl(url, requestOptions); configureRequestOptions(requestOptions); this.doDownload(requestOptions, { destination: null, options, onCancel, callback: error => { if (error == null) { resolve(Buffer.concat(responseChunks)); } else { reject(error); } }, responseHandler: (response, callback) => { let receivedLength = 0; response.on("data", (chunk) => { receivedLength += chunk.length; if (receivedLength > 524288000) { callback(new Error("Maximum allowed size is 500 MB")); return; } responseChunks.push(chunk); }); response.on("end", () => { callback(null); }); }, }, 0); }); } doDownload(requestOptions, options, redirectCount) { const request = this.createRequest(requestOptions, (response) => { if (response.statusCode >= 400) { options.callback(new Error(`Cannot download "${requestOptions.protocol || "https:"}//${requestOptions.hostname}${requestOptions.path}", status ${response.statusCode}: ${response.statusMessage}`)); return; } // It is possible for the response stream to fail, e.g. when a network is lost while // response stream is in progress. Stop waiting and reject so consumer can catch the error. response.on("error", options.callback); // this code not relevant for Electron (redirect event instead handled) const redirectUrl = safeGetHeader(response, "location"); if (redirectUrl != null) { if (redirectCount < this.maxRedirects) { this.doDownload(HttpExecutor.prepareRedirectUrlOptions(redirectUrl, requestOptions), options, redirectCount++); } else { options.callback(this.createMaxRedirectError()); } return; } if (options.responseHandler == null) { configurePipes(options, response); } else { options.responseHandler(response, options.callback); } }); this.addErrorAndTimeoutHandlers(request, options.callback, requestOptions.timeout); this.addRedirectHandlers(request, requestOptions, options.callback, redirectCount, requestOptions => { this.doDownload(requestOptions, options, redirectCount++); }); request.end(); } createMaxRedirectError() { return new Error(`Too many redirects (> ${this.maxRedirects})`); } addTimeOutHandler(request, callback, timeout) { request.on("socket", (socket) => { socket.setTimeout(timeout, () => { request.abort(); callback(new Error("Request timed out")); }); }); } static prepareRedirectUrlOptions(redirectUrl, options) { const newOptions = configureRequestOptionsFromUrl(redirectUrl, { ...options }); const headers = newOptions.headers; if (headers === null || headers === void 0 ? void 0 : headers.authorization) { const parsedNewUrl = new url_1.URL(redirectUrl); if (parsedNewUrl.hostname.endsWith(".amazonaws.com") || parsedNewUrl.searchParams.has("X-Amz-Credential")) { delete headers.authorization; } } return newOptions; } static retryOnServerError(task, maxRetries = 3) { for (let attemptNumber = 0;; attemptNumber++) { try { return task(); } catch (e) { if (attemptNumber < maxRetries && ((e instanceof HttpError && e.isServerError()) || e.code === "EPIPE")) { continue; } throw e; } } } } exports.HttpExecutor = HttpExecutor; function configureRequestOptionsFromUrl(url, options) { const result = configureRequestOptions(options); configureRequestUrl(new url_1.URL(url), result); return result; } exports.configureRequestOptionsFromUrl = configureRequestOptionsFromUrl; function configureRequestUrl(url, options) { options.protocol = url.protocol; options.hostname = url.hostname; if (url.port) { options.port = url.port; } else if (options.port) { delete options.port; } options.path = url.pathname + url.search; } exports.configureRequestUrl = configureRequestUrl; class DigestTransform extends stream_1.Transform { // noinspection JSUnusedGlobalSymbols get actual() { return this._actual; } constructor(expected, algorithm = "sha512", encoding = "base64") { super(); this.expected = expected; this.algorithm = algorithm; this.encoding = encoding; this._actual = null; this.isValidateOnEnd = true; this.digester = (0, crypto_1.createHash)(algorithm); } // noinspection JSUnusedGlobalSymbols _transform(chunk, encoding, callback) { this.digester.update(chunk); callback(null, chunk); } // noinspection JSUnusedGlobalSymbols _flush(callback) { this._actual = this.digester.digest(this.encoding); if (this.isValidateOnEnd) { try { this.validate(); } catch (e) { callback(e); return; } } callback(null); } validate() { if (this._actual == null) { throw (0, index_1.newError)("Not finished yet", "ERR_STREAM_NOT_FINISHED"); } if (this._actual !== this.expected) { throw (0, index_1.newError)(`${this.algorithm} checksum mismatch, expected ${this.expected}, got ${this._actual}`, "ERR_CHECKSUM_MISMATCH"); } return null; } } exports.DigestTransform = DigestTransform; function checkSha2(sha2Header, sha2, callback) { if (sha2Header != null && sha2 != null && sha2Header !== sha2) { callback(new Error(`checksum mismatch: expected ${sha2} but got ${sha2Header} (X-Checksum-Sha2 header)`)); return false; } return true; } function safeGetHeader(response, headerKey) { const value = response.headers[headerKey]; if (value == null) { return null; } else if (Array.isArray(value)) { // electron API return value.length === 0 ? null : value[value.length - 1]; } else { return value; } } exports.safeGetHeader = safeGetHeader; function configurePipes(options, response) { if (!checkSha2(safeGetHeader(response, "X-Checksum-Sha2"), options.options.sha2, options.callback)) { return; } const streams = []; if (options.options.onProgress != null) { const contentLength = safeGetHeader(response, "content-length"); if (contentLength != null) { streams.push(new ProgressCallbackTransform_1.ProgressCallbackTransform(parseInt(contentLength, 10), options.options.cancellationToken, options.options.onProgress)); } } const sha512 = options.options.sha512; if (sha512 != null) { streams.push(new DigestTransform(sha512, "sha512", sha512.length === 128 && !sha512.includes("+") && !sha512.includes("Z") && !sha512.includes("=") ? "hex" : "base64")); } else if (options.options.sha2 != null) { streams.push(new DigestTransform(options.options.sha2, "sha256", "hex")); } const fileOut = (0, fs_1.createWriteStream)(options.destination); streams.push(fileOut); let lastStream = response; for (const stream of streams) { stream.on("error", (error) => { fileOut.close(); if (!options.options.cancellationToken.cancelled) { options.callback(error); } }); lastStream = lastStream.pipe(stream); } fileOut.on("finish", () => { ; fileOut.close(options.callback); }); } function configureRequestOptions(options, token, method) { if (method != null) { options.method = method; } options.headers = { ...options.headers }; const headers = options.headers; if (token != null) { ; headers.authorization = token.startsWith("Basic") || token.startsWith("Bearer") ? token : `token ${token}`; } if (headers["User-Agent"] == null) { headers["User-Agent"] = "electron-builder"; } if (method == null || method === "GET" || headers["Cache-Control"] == null) { headers["Cache-Control"] = "no-cache"; } // do not specify for node (in any case we use https module) if (options.protocol == null && process.versions.electron != null) { options.protocol = "https:"; } return options; } exports.configureRequestOptions = configureRequestOptions; function safeStringifyJson(data, skippedNames) { return JSON.stringify(data, (name, value) => { if (name.endsWith("Authorization") || name.endsWith("authorization") || name.endsWith("Password") || name.endsWith("PASSWORD") || name.endsWith("Token") || name.includes("password") || name.includes("token") || (skippedNames != null && skippedNames.has(name))) { return "<stripped sensitive data>"; } return value; }, 2); } exports.safeStringifyJson = safeStringifyJson; //# sourceMappingURL=httpExecutor.js.map