You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					425 lines
				
				14 KiB
			
		
		
			
		
	
	
					425 lines
				
				14 KiB
			| 
											3 years ago
										 | 'use strict'; | ||
|  | 
 | ||
|  | var utils = require('./../utils'); | ||
|  | var settle = require('./../core/settle'); | ||
|  | var buildFullPath = require('../core/buildFullPath'); | ||
|  | var buildURL = require('./../helpers/buildURL'); | ||
|  | var http = require('http'); | ||
|  | var https = require('https'); | ||
|  | var httpFollow = require('follow-redirects').http; | ||
|  | var httpsFollow = require('follow-redirects').https; | ||
|  | var url = require('url'); | ||
|  | var zlib = require('zlib'); | ||
|  | var VERSION = require('./../env/data').version; | ||
|  | var transitionalDefaults = require('../defaults/transitional'); | ||
|  | var AxiosError = require('../core/AxiosError'); | ||
|  | var CanceledError = require('../cancel/CanceledError'); | ||
|  | 
 | ||
|  | var isHttps = /https:?/; | ||
|  | 
 | ||
|  | var supportedProtocols = [ 'http:', 'https:', 'file:' ]; | ||
|  | 
 | ||
|  | /** | ||
|  |  * | ||
|  |  * @param {http.ClientRequestArgs} options | ||
|  |  * @param {AxiosProxyConfig} proxy | ||
|  |  * @param {string} location | ||
|  |  */ | ||
|  | function setProxy(options, proxy, location) { | ||
|  |   options.hostname = proxy.host; | ||
|  |   options.host = proxy.host; | ||
|  |   options.port = proxy.port; | ||
|  |   options.path = location; | ||
|  | 
 | ||
|  |   // Basic proxy authorization
 | ||
|  |   if (proxy.auth) { | ||
|  |     var base64 = Buffer.from(proxy.auth.username + ':' + proxy.auth.password, 'utf8').toString('base64'); | ||
|  |     options.headers['Proxy-Authorization'] = 'Basic ' + base64; | ||
|  |   } | ||
|  | 
 | ||
|  |   // If a proxy is used, any redirects must also pass through the proxy
 | ||
|  |   options.beforeRedirect = function beforeRedirect(redirection) { | ||
|  |     redirection.headers.host = redirection.host; | ||
|  |     setProxy(redirection, proxy, redirection.href); | ||
|  |   }; | ||
|  | } | ||
|  | 
 | ||
|  | /*eslint consistent-return:0*/ | ||
|  | module.exports = function httpAdapter(config) { | ||
|  |   return new Promise(function dispatchHttpRequest(resolvePromise, rejectPromise) { | ||
|  |     var onCanceled; | ||
|  |     function done() { | ||
|  |       if (config.cancelToken) { | ||
|  |         config.cancelToken.unsubscribe(onCanceled); | ||
|  |       } | ||
|  | 
 | ||
|  |       if (config.signal) { | ||
|  |         config.signal.removeEventListener('abort', onCanceled); | ||
|  |       } | ||
|  |     } | ||
|  |     var resolve = function resolve(value) { | ||
|  |       done(); | ||
|  |       resolvePromise(value); | ||
|  |     }; | ||
|  |     var rejected = false; | ||
|  |     var reject = function reject(value) { | ||
|  |       done(); | ||
|  |       rejected = true; | ||
|  |       rejectPromise(value); | ||
|  |     }; | ||
|  |     var data = config.data; | ||
|  |     var headers = config.headers; | ||
|  |     var headerNames = {}; | ||
|  | 
 | ||
|  |     Object.keys(headers).forEach(function storeLowerName(name) { | ||
|  |       headerNames[name.toLowerCase()] = name; | ||
|  |     }); | ||
|  | 
 | ||
|  |     // Set User-Agent (required by some servers)
 | ||
|  |     // See https://github.com/axios/axios/issues/69
 | ||
|  |     if ('user-agent' in headerNames) { | ||
|  |       // User-Agent is specified; handle case where no UA header is desired
 | ||
|  |       if (!headers[headerNames['user-agent']]) { | ||
|  |         delete headers[headerNames['user-agent']]; | ||
|  |       } | ||
|  |       // Otherwise, use specified value
 | ||
|  |     } else { | ||
|  |       // Only set header if it hasn't been set in config
 | ||
|  |       headers['User-Agent'] = 'axios/' + VERSION; | ||
|  |     } | ||
|  | 
 | ||
|  |     // support for https://www.npmjs.com/package/form-data api
 | ||
|  |     if (utils.isFormData(data) && utils.isFunction(data.getHeaders)) { | ||
|  |       Object.assign(headers, data.getHeaders()); | ||
|  |     } else if (data && !utils.isStream(data)) { | ||
|  |       if (Buffer.isBuffer(data)) { | ||
|  |         // Nothing to do...
 | ||
|  |       } else if (utils.isArrayBuffer(data)) { | ||
|  |         data = Buffer.from(new Uint8Array(data)); | ||
|  |       } else if (utils.isString(data)) { | ||
|  |         data = Buffer.from(data, 'utf-8'); | ||
|  |       } else { | ||
|  |         return reject(new AxiosError( | ||
|  |           'Data after transformation must be a string, an ArrayBuffer, a Buffer, or a Stream', | ||
|  |           AxiosError.ERR_BAD_REQUEST, | ||
|  |           config | ||
|  |         )); | ||
|  |       } | ||
|  | 
 | ||
|  |       if (config.maxBodyLength > -1 && data.length > config.maxBodyLength) { | ||
|  |         return reject(new AxiosError( | ||
|  |           'Request body larger than maxBodyLength limit', | ||
|  |           AxiosError.ERR_BAD_REQUEST, | ||
|  |           config | ||
|  |         )); | ||
|  |       } | ||
|  | 
 | ||
|  |       // Add Content-Length header if data exists
 | ||
|  |       if (!headerNames['content-length']) { | ||
|  |         headers['Content-Length'] = data.length; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     // HTTP basic authentication
 | ||
|  |     var auth = undefined; | ||
|  |     if (config.auth) { | ||
|  |       var username = config.auth.username || ''; | ||
|  |       var password = config.auth.password || ''; | ||
|  |       auth = username + ':' + password; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Parse url
 | ||
|  |     var fullPath = buildFullPath(config.baseURL, config.url); | ||
|  |     var parsed = url.parse(fullPath); | ||
|  |     var protocol = parsed.protocol || supportedProtocols[0]; | ||
|  | 
 | ||
|  |     if (supportedProtocols.indexOf(protocol) === -1) { | ||
|  |       return reject(new AxiosError( | ||
|  |         'Unsupported protocol ' + protocol, | ||
|  |         AxiosError.ERR_BAD_REQUEST, | ||
|  |         config | ||
|  |       )); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!auth && parsed.auth) { | ||
|  |       var urlAuth = parsed.auth.split(':'); | ||
|  |       var urlUsername = urlAuth[0] || ''; | ||
|  |       var urlPassword = urlAuth[1] || ''; | ||
|  |       auth = urlUsername + ':' + urlPassword; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (auth && headerNames.authorization) { | ||
|  |       delete headers[headerNames.authorization]; | ||
|  |     } | ||
|  | 
 | ||
|  |     var isHttpsRequest = isHttps.test(protocol); | ||
|  |     var agent = isHttpsRequest ? config.httpsAgent : config.httpAgent; | ||
|  | 
 | ||
|  |     try { | ||
|  |       buildURL(parsed.path, config.params, config.paramsSerializer).replace(/^\?/, ''); | ||
|  |     } catch (err) { | ||
|  |       var customErr = new Error(err.message); | ||
|  |       customErr.config = config; | ||
|  |       customErr.url = config.url; | ||
|  |       customErr.exists = true; | ||
|  |       reject(customErr); | ||
|  |     } | ||
|  | 
 | ||
|  |     var options = { | ||
|  |       path: buildURL(parsed.path, config.params, config.paramsSerializer).replace(/^\?/, ''), | ||
|  |       method: config.method.toUpperCase(), | ||
|  |       headers: headers, | ||
|  |       agent: agent, | ||
|  |       agents: { http: config.httpAgent, https: config.httpsAgent }, | ||
|  |       auth: auth | ||
|  |     }; | ||
|  | 
 | ||
|  |     if (config.socketPath) { | ||
|  |       options.socketPath = config.socketPath; | ||
|  |     } else { | ||
|  |       options.hostname = parsed.hostname; | ||
|  |       options.port = parsed.port; | ||
|  |     } | ||
|  | 
 | ||
|  |     var proxy = config.proxy; | ||
|  |     if (!proxy && proxy !== false) { | ||
|  |       var proxyEnv = protocol.slice(0, -1) + '_proxy'; | ||
|  |       var proxyUrl = process.env[proxyEnv] || process.env[proxyEnv.toUpperCase()]; | ||
|  |       if (proxyUrl) { | ||
|  |         var parsedProxyUrl = url.parse(proxyUrl); | ||
|  |         var noProxyEnv = process.env.no_proxy || process.env.NO_PROXY; | ||
|  |         var shouldProxy = true; | ||
|  | 
 | ||
|  |         if (noProxyEnv) { | ||
|  |           var noProxy = noProxyEnv.split(',').map(function trim(s) { | ||
|  |             return s.trim(); | ||
|  |           }); | ||
|  | 
 | ||
|  |           shouldProxy = !noProxy.some(function proxyMatch(proxyElement) { | ||
|  |             if (!proxyElement) { | ||
|  |               return false; | ||
|  |             } | ||
|  |             if (proxyElement === '*') { | ||
|  |               return true; | ||
|  |             } | ||
|  |             if (proxyElement[0] === '.' && | ||
|  |                 parsed.hostname.substr(parsed.hostname.length - proxyElement.length) === proxyElement) { | ||
|  |               return true; | ||
|  |             } | ||
|  | 
 | ||
|  |             return parsed.hostname === proxyElement; | ||
|  |           }); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (shouldProxy) { | ||
|  |           proxy = { | ||
|  |             host: parsedProxyUrl.hostname, | ||
|  |             port: parsedProxyUrl.port, | ||
|  |             protocol: parsedProxyUrl.protocol | ||
|  |           }; | ||
|  | 
 | ||
|  |           if (parsedProxyUrl.auth) { | ||
|  |             var proxyUrlAuth = parsedProxyUrl.auth.split(':'); | ||
|  |             proxy.auth = { | ||
|  |               username: proxyUrlAuth[0], | ||
|  |               password: proxyUrlAuth[1] | ||
|  |             }; | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     if (proxy) { | ||
|  |       options.headers.host = parsed.hostname + (parsed.port ? ':' + parsed.port : ''); | ||
|  |       setProxy(options, proxy, protocol + '//' + parsed.hostname + (parsed.port ? ':' + parsed.port : '') + options.path); | ||
|  |     } | ||
|  | 
 | ||
|  |     var transport; | ||
|  |     var isHttpsProxy = isHttpsRequest && (proxy ? isHttps.test(proxy.protocol) : true); | ||
|  |     if (config.transport) { | ||
|  |       transport = config.transport; | ||
|  |     } else if (config.maxRedirects === 0) { | ||
|  |       transport = isHttpsProxy ? https : http; | ||
|  |     } else { | ||
|  |       if (config.maxRedirects) { | ||
|  |         options.maxRedirects = config.maxRedirects; | ||
|  |       } | ||
|  |       if (config.beforeRedirect) { | ||
|  |         options.beforeRedirect = config.beforeRedirect; | ||
|  |       } | ||
|  |       transport = isHttpsProxy ? httpsFollow : httpFollow; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (config.maxBodyLength > -1) { | ||
|  |       options.maxBodyLength = config.maxBodyLength; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (config.insecureHTTPParser) { | ||
|  |       options.insecureHTTPParser = config.insecureHTTPParser; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Create the request
 | ||
|  |     var req = transport.request(options, function handleResponse(res) { | ||
|  |       if (req.aborted) return; | ||
|  | 
 | ||
|  |       // uncompress the response body transparently if required
 | ||
|  |       var stream = res; | ||
|  | 
 | ||
|  |       // return the last request in case of redirects
 | ||
|  |       var lastRequest = res.req || req; | ||
|  | 
 | ||
|  | 
 | ||
|  |       // if no content, is HEAD request or decompress disabled we should not decompress
 | ||
|  |       if (res.statusCode !== 204 && lastRequest.method !== 'HEAD' && config.decompress !== false) { | ||
|  |         switch (res.headers['content-encoding']) { | ||
|  |         /*eslint default-case:0*/ | ||
|  |         case 'gzip': | ||
|  |         case 'compress': | ||
|  |         case 'deflate': | ||
|  |         // add the unzipper to the body stream processing pipeline
 | ||
|  |           stream = stream.pipe(zlib.createUnzip()); | ||
|  | 
 | ||
|  |           // remove the content-encoding in order to not confuse downstream operations
 | ||
|  |           delete res.headers['content-encoding']; | ||
|  |           break; | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       var response = { | ||
|  |         status: res.statusCode, | ||
|  |         statusText: res.statusMessage, | ||
|  |         headers: res.headers, | ||
|  |         config: config, | ||
|  |         request: lastRequest | ||
|  |       }; | ||
|  | 
 | ||
|  |       if (config.responseType === 'stream') { | ||
|  |         response.data = stream; | ||
|  |         settle(resolve, reject, response); | ||
|  |       } else { | ||
|  |         var responseBuffer = []; | ||
|  |         var totalResponseBytes = 0; | ||
|  |         stream.on('data', function handleStreamData(chunk) { | ||
|  |           responseBuffer.push(chunk); | ||
|  |           totalResponseBytes += chunk.length; | ||
|  | 
 | ||
|  |           // make sure the content length is not over the maxContentLength if specified
 | ||
|  |           if (config.maxContentLength > -1 && totalResponseBytes > config.maxContentLength) { | ||
|  |             // stream.destoy() emit aborted event before calling reject() on Node.js v16
 | ||
|  |             rejected = true; | ||
|  |             stream.destroy(); | ||
|  |             reject(new AxiosError('maxContentLength size of ' + config.maxContentLength + ' exceeded', | ||
|  |               AxiosError.ERR_BAD_RESPONSE, config, lastRequest)); | ||
|  |           } | ||
|  |         }); | ||
|  | 
 | ||
|  |         stream.on('aborted', function handlerStreamAborted() { | ||
|  |           if (rejected) { | ||
|  |             return; | ||
|  |           } | ||
|  |           stream.destroy(); | ||
|  |           reject(new AxiosError( | ||
|  |             'maxContentLength size of ' + config.maxContentLength + ' exceeded', | ||
|  |             AxiosError.ERR_BAD_RESPONSE, | ||
|  |             config, | ||
|  |             lastRequest | ||
|  |           )); | ||
|  |         }); | ||
|  | 
 | ||
|  |         stream.on('error', function handleStreamError(err) { | ||
|  |           if (req.aborted) return; | ||
|  |           reject(AxiosError.from(err, null, config, lastRequest)); | ||
|  |         }); | ||
|  | 
 | ||
|  |         stream.on('end', function handleStreamEnd() { | ||
|  |           try { | ||
|  |             var responseData = responseBuffer.length === 1 ? responseBuffer[0] : Buffer.concat(responseBuffer); | ||
|  |             if (config.responseType !== 'arraybuffer') { | ||
|  |               responseData = responseData.toString(config.responseEncoding); | ||
|  |               if (!config.responseEncoding || config.responseEncoding === 'utf8') { | ||
|  |                 responseData = utils.stripBOM(responseData); | ||
|  |               } | ||
|  |             } | ||
|  |             response.data = responseData; | ||
|  |           } catch (err) { | ||
|  |             reject(AxiosError.from(err, null, config, response.request, response)); | ||
|  |           } | ||
|  |           settle(resolve, reject, response); | ||
|  |         }); | ||
|  |       } | ||
|  |     }); | ||
|  | 
 | ||
|  |     // Handle errors
 | ||
|  |     req.on('error', function handleRequestError(err) { | ||
|  |       // @todo remove
 | ||
|  |       // if (req.aborted && err.code !== AxiosError.ERR_FR_TOO_MANY_REDIRECTS) return;
 | ||
|  |       reject(AxiosError.from(err, null, config, req)); | ||
|  |     }); | ||
|  | 
 | ||
|  |     // set tcp keep alive to prevent drop connection by peer
 | ||
|  |     req.on('socket', function handleRequestSocket(socket) { | ||
|  |       // default interval of sending ack packet is 1 minute
 | ||
|  |       socket.setKeepAlive(true, 1000 * 60); | ||
|  |     }); | ||
|  | 
 | ||
|  |     // Handle request timeout
 | ||
|  |     if (config.timeout) { | ||
|  |       // This is forcing a int timeout to avoid problems if the `req` interface doesn't handle other types.
 | ||
|  |       var timeout = parseInt(config.timeout, 10); | ||
|  | 
 | ||
|  |       if (isNaN(timeout)) { | ||
|  |         reject(new AxiosError( | ||
|  |           'error trying to parse `config.timeout` to int', | ||
|  |           AxiosError.ERR_BAD_OPTION_VALUE, | ||
|  |           config, | ||
|  |           req | ||
|  |         )); | ||
|  | 
 | ||
|  |         return; | ||
|  |       } | ||
|  | 
 | ||
|  |       // Sometime, the response will be very slow, and does not respond, the connect event will be block by event loop system.
 | ||
|  |       // And timer callback will be fired, and abort() will be invoked before connection, then get "socket hang up" and code ECONNRESET.
 | ||
|  |       // At this time, if we have a large number of request, nodejs will hang up some socket on background. and the number will up and up.
 | ||
|  |       // And then these socket which be hang up will devoring CPU little by little.
 | ||
|  |       // ClientRequest.setTimeout will be fired on the specify milliseconds, and can make sure that abort() will be fired after connect.
 | ||
|  |       req.setTimeout(timeout, function handleRequestTimeout() { | ||
|  |         req.abort(); | ||
|  |         var transitional = config.transitional || transitionalDefaults; | ||
|  |         reject(new AxiosError( | ||
|  |           'timeout of ' + timeout + 'ms exceeded', | ||
|  |           transitional.clarifyTimeoutError ? AxiosError.ETIMEDOUT : AxiosError.ECONNABORTED, | ||
|  |           config, | ||
|  |           req | ||
|  |         )); | ||
|  |       }); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (config.cancelToken || config.signal) { | ||
|  |       // Handle cancellation
 | ||
|  |       // eslint-disable-next-line func-names
 | ||
|  |       onCanceled = function(cancel) { | ||
|  |         if (req.aborted) return; | ||
|  | 
 | ||
|  |         req.abort(); | ||
|  |         reject(!cancel || (cancel && cancel.type) ? new CanceledError() : cancel); | ||
|  |       }; | ||
|  | 
 | ||
|  |       config.cancelToken && config.cancelToken.subscribe(onCanceled); | ||
|  |       if (config.signal) { | ||
|  |         config.signal.aborted ? onCanceled() : config.signal.addEventListener('abort', onCanceled); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  | 
 | ||
|  |     // Send the request
 | ||
|  |     if (utils.isStream(data)) { | ||
|  |       data.on('error', function handleStreamError(err) { | ||
|  |         reject(AxiosError.from(err, config, null, req)); | ||
|  |       }).pipe(req); | ||
|  |     } else { | ||
|  |       req.end(data); | ||
|  |     } | ||
|  |   }); | ||
|  | }; |