"use strict"; var __assign = (this && this.__assign) || Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; exports.__esModule = true; var cacheUtils = require("./cache-utils"); var FileCache_1 = require("./FileCache"); var MemoryCache_1 = require("./MemoryCache"); var http_1 = require("http"); var zlib_1 = require("zlib"); var url_1 = require("url"); var stream_1 = require("stream"); var https_1 = require("https"); var Response = require("http-response-object"); exports.Response = Response; var caseless = require('caseless'); var fileCache = new FileCache_1["default"](__dirname + '/cache'); var memoryCache = new MemoryCache_1["default"](); function requestProtocol(protocol, options, callback) { if (protocol === 'http') { return http_1.request(options, callback); } else if (protocol === 'https') { return https_1.request(options, callback); } throw new Error('Unsupported protocol ' + protocol); } function request(method, url, options, callback) { if (typeof options === 'function') { callback = options; options = null; } if (options === null || options === undefined) { options = {}; } if (typeof options !== 'object') { throw new TypeError('options must be an object (or null)'); } if (typeof callback !== 'function') { throw new TypeError('callback must be a function'); } return _request(method, ((url && typeof url === 'object') ? url.href : url), options, callback); } function _request(method, url, options, callback) { var start = Date.now(); if (typeof method !== 'string') { throw new TypeError('The method must be a string.'); } if (typeof url !== 'string') { throw new TypeError('The URL/path must be a string or a URL object.'); } method = method.toUpperCase(); var urlObject = url_1.parse(url); var protocol = (urlObject.protocol || '').replace(/\:$/, ''); if (protocol !== 'http' && protocol !== 'https') { throw new TypeError('The protocol "' + protocol + '" is not supported, cannot load "' + url + '"'); } var rawHeaders = options.headers || {}; var headers = caseless(rawHeaders); if (urlObject.auth) { headers.set('Authorization', 'Basic ' + (Buffer.from(urlObject.auth)).toString('base64')); } var agent = 'agent' in options ? options.agent : false; var cache = options.cache; if (typeof cache === 'string') { if (cache === 'file') { cache = fileCache; } else if (cache === 'memory') { cache = memoryCache; } } if (cache && !(typeof cache === 'object' && typeof cache.getResponse === 'function' && typeof cache.setResponse === 'function' && typeof cache.invalidateResponse === 'function')) { throw new TypeError(cache + ' is not a valid cache, caches must have `getResponse`, `setResponse` and `invalidateResponse` methods.'); } var ignoreFailedInvalidation = options.ignoreFailedInvalidation; if (options.duplex !== undefined && typeof options.duplex !== 'boolean') { throw new Error('expected options.duplex to be a boolean if provided'); } var duplex = options.duplex !== undefined ? options.duplex : !(method === 'GET' || method === 'DELETE' || method === 'HEAD'); var unsafe = !(method === 'GET' || method === 'OPTIONS' || method === 'HEAD'); if (options.gzip) { headers.set('Accept-Encoding', headers.has('Accept-Encoding') ? headers.get('Accept-Encoding') + ',gzip,deflate' : 'gzip,deflate'); return _request(method, url, { allowRedirectHeaders: options.allowRedirectHeaders, duplex: duplex, headers: rawHeaders, agent: agent, followRedirects: options.followRedirects, retry: options.retry, retryDelay: options.retryDelay, maxRetries: options.maxRetries, cache: cache, timeout: options.timeout }, function (err, res) { if (err) return callback(err); if (!res) return callback(new Error('Response should not be undefined if there is no error.')); var newHeaders = __assign({}, res.headers); var newBody = res.body; switch (newHeaders['content-encoding']) { case 'gzip': delete newHeaders['content-encoding']; newBody = res.body.pipe(zlib_1.createGunzip()); break; case 'deflate': delete newHeaders['content-encoding']; newBody = res.body.pipe(zlib_1.createInflate()); break; } return callback(err, new Response(res.statusCode, newHeaders, newBody, res.url)); }); } if (options.followRedirects) { return _request(method, url, { allowRedirectHeaders: options.allowRedirectHeaders, duplex: duplex, headers: rawHeaders, agent: agent, retry: options.retry, retryDelay: options.retryDelay, maxRetries: options.maxRetries, cache: cache, timeout: options.timeout }, function (err, res) { if (err) return callback(err); if (!res) return callback(new Error('Response should not be undefined if there is no error.')); if (options.followRedirects && isRedirect(res.statusCode)) { // prevent leakage of file handles res.body.resume(); if (method === 'DELETE' && res.statusCode === 303) { // 303 See Other should convert to GET for duplex // requests and for DELETE method = 'GET'; } if (options.maxRedirects === 0) { var err_1 = new Error('Maximum number of redirects exceeded'); err_1.res = res; return callback(err_1, res); } options = __assign({}, options, { duplex: false, maxRedirects: options.maxRedirects && options.maxRedirects !== Infinity ? options.maxRedirects - 1 : options.maxRedirects }); // don't maintain headers through redirects // This fixes a problem where a POST to http://example.com // might result in a GET to http://example.co.uk that includes "content-length" // as a header var headers_1 = caseless(options.headers); var redirectHeaders = {}; if (options.allowRedirectHeaders) { for (var i = 0; i < options.allowRedirectHeaders.length; i++) { var headerName = options.allowRedirectHeaders[i]; var headerValue = headers_1.get(headerName); if (headerValue) { redirectHeaders[headerName] = headerValue; } } } options.headers = redirectHeaders; var location = res.headers.location; if (typeof location !== 'string') { return callback(new Error('Cannot redirect to non string location: ' + location)); } return request(duplex ? 'GET' : method, url_1.resolve(url, location), options, callback); } else { return callback(null, res); } }); } if (cache && method === 'GET' && !duplex) { var timestamp_1 = Date.now(); return cache.getResponse(url, function (err, cachedResponse) { if (err) { console.warn('Error reading from cache: ' + err.message); } var isMatch = !!(cachedResponse && cacheUtils.isMatch(rawHeaders, cachedResponse)); if (cachedResponse && (options.isMatch ? options.isMatch(rawHeaders, cachedResponse, isMatch) : isMatch)) { var isExpired = cacheUtils.isExpired(cachedResponse); if (!(options.isExpired ? options.isExpired(cachedResponse, isExpired) : isExpired)) { var res = new Response(cachedResponse.statusCode, cachedResponse.headers, cachedResponse.body, url); res.fromCache = true; res.fromNotModified = false; return callback(null, res); } else { if (cachedResponse.headers['etag']) { headers.set('If-None-Match', cachedResponse.headers['etag']); } if (cachedResponse.headers['last-modified']) { headers.set('If-Modified-Since', cachedResponse.headers['last-modified']); } } } request('GET', url, { allowRedirectHeaders: options.allowRedirectHeaders, headers: rawHeaders, retry: options.retry, retryDelay: options.retryDelay, maxRetries: options.maxRetries, agent: agent, timeout: options.timeout }, function (err, res) { if (err) return callback(err); if (!res) return callback(new Error('Response should not be undefined if there is no error.')); if (res.statusCode === 304 && cachedResponse) { // Not Modified // prevent leakage of file handles res.body.resume(); var resultBody = cachedResponse.body; var c = cache; if (c.updateResponseHeaders) { c.updateResponseHeaders(url, { headers: res.headers, requestTimestamp: timestamp_1 }); } else { var cachedResponseBody_1 = new stream_1.PassThrough(); var newResultBody_1 = new stream_1.PassThrough(); resultBody.on('data', function (data) { cachedResponseBody_1.write(data); newResultBody_1.write(data); }); resultBody.on('end', function () { cachedResponseBody_1.end(); newResultBody_1.end(); }); resultBody = newResultBody_1; cache.setResponse(url, { statusCode: cachedResponse.statusCode, headers: res.headers, body: cachedResponseBody_1, requestHeaders: cachedResponse.requestHeaders, requestTimestamp: timestamp_1 }); } var response = new Response(cachedResponse.statusCode, cachedResponse.headers, resultBody, url); response.fromCache = true; response.fromNotModified = true; return callback(null, response); } // prevent leakage of file handles cachedResponse && cachedResponse.body.resume(); var canCache = cacheUtils.canCache(res); if (options.canCache ? options.canCache(res, canCache) : canCache) { var cachedResponseBody_2 = new stream_1.PassThrough(); var resultResponseBody_1 = new stream_1.PassThrough(); res.body.on('data', function (data) { cachedResponseBody_2.write(data); resultResponseBody_1.write(data); }); res.body.on('end', function () { cachedResponseBody_2.end(); resultResponseBody_1.end(); }); var resultResponse = new Response(res.statusCode, res.headers, resultResponseBody_1, url); cache.setResponse(url, { statusCode: res.statusCode, headers: res.headers, body: cachedResponseBody_2, requestHeaders: rawHeaders, requestTimestamp: timestamp_1 }); return callback(null, resultResponse); } else { return callback(null, res); } }); }); } function attempt(n) { return _request(method, url, { allowRedirectHeaders: options.allowRedirectHeaders, headers: rawHeaders, agent: agent, timeout: options.timeout }, function (err, res) { var retry = err || !res || res.statusCode >= 400; if (typeof options.retry === 'function') { retry = options.retry(err, res, n + 1); } if (n >= (options.maxRetries || 5)) { retry = false; } if (retry) { var delay = options.retryDelay; if (typeof delay === 'function') { delay = delay(err, res, n + 1); } delay = delay || 200; setTimeout(function () { attempt(n + 1); }, delay); } else { callback(err, res); } }); } if (options.retry && method === 'GET' && !duplex) { return attempt(0); } var responded = false; var timeout = null; var req = requestProtocol(protocol, { host: urlObject.hostname, port: urlObject.port == null ? undefined : +urlObject.port, path: urlObject.path, method: method, headers: rawHeaders, agent: agent }, function (res) { var end = Date.now(); if (responded) return res.resume(); responded = true; if (timeout !== null) clearTimeout(timeout); var result = new Response(res.statusCode || 0, res.headers, res, url); if (cache && unsafe && res.statusCode && res.statusCode < 400) { cache.invalidateResponse(url, function (err) { if (err && !ignoreFailedInvalidation) { callback(new Error('Error invalidating the cache for' + url + ': ' + err.message), result); } else { callback(null, result); } }); } else { callback(null, result); } }).on('error', function (err) { if (responded) return; responded = true; if (timeout !== null) clearTimeout(timeout); callback(err); }); function onTimeout() { if (responded) return; responded = true; if (timeout !== null) clearTimeout(timeout); req.abort(); var duration = Date.now() - start; var err = new Error('Request timed out after ' + duration + 'ms'); err.timeout = true; err.duration = duration; callback(err); } if (options.socketTimeout) { req.setTimeout(options.socketTimeout, onTimeout); } if (options.timeout) { timeout = setTimeout(onTimeout, options.timeout); } if (duplex) { return req; } else { req.end(); } return undefined; } function isRedirect(statusCode) { return statusCode === 301 || statusCode === 302 || statusCode === 303 || statusCode === 307 || statusCode === 308; } exports["default"] = request; module.exports = request; module.exports["default"] = request; module.exports.Response = Response;