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
						
					
					
				| '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);
 | |
|     }
 | |
|   });
 | |
| };
 |