| 'use strict'; |
| |
| var utils = require('./../utils'); |
| var settle = require('./../core/settle'); |
| 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 pkg = require('./../../package.json'); |
| var Buffer = require('buffer').Buffer; |
| var createError = require('../core/createError'); |
| var enhanceError = require('../core/enhanceError'); |
| |
| /*eslint consistent-return:0*/ |
| module.exports = function httpAdapter(config) { |
| return new Promise(function dispatchHttpRequest(resolve, reject) { |
| var data = config.data; |
| var headers = config.headers; |
| var timer; |
| var aborted = false; |
| |
| // Set User-Agent (required by some servers) |
| // Only set header if it hasn't been set in config |
| // See https://github.com/mzabriskie/axios/issues/69 |
| if (!headers['User-Agent'] && !headers['user-agent']) { |
| headers['User-Agent'] = 'axios/' + pkg.version; |
| } |
| |
| if (data && !utils.isStream(data)) { |
| if (utils.isArrayBuffer(data)) { |
| data = new Buffer(new Uint8Array(data)); |
| } else if (utils.isString(data)) { |
| data = new Buffer(data, 'utf-8'); |
| } else { |
| return reject(createError( |
| 'Data after transformation must be a string, an ArrayBuffer, or a Stream', |
| config |
| )); |
| } |
| |
| // Add Content-Length header if data exists |
| 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 parsed = url.parse(config.url); |
| var protocol = parsed.protocol || 'http:'; |
| |
| if (!auth && parsed.auth) { |
| var urlAuth = parsed.auth.split(':'); |
| var urlUsername = urlAuth[0] || ''; |
| var urlPassword = urlAuth[1] || ''; |
| auth = urlUsername + ':' + urlPassword; |
| } |
| |
| if (auth) { |
| delete headers.Authorization; |
| } |
| |
| var isHttps = protocol === 'https:'; |
| var agent = isHttps ? config.httpsAgent : config.httpAgent; |
| |
| var options = { |
| hostname: parsed.hostname, |
| port: parsed.port, |
| path: buildURL(parsed.path, config.params, config.paramsSerializer).replace(/^\?/, ''), |
| method: config.method, |
| headers: headers, |
| agent: agent, |
| auth: auth |
| }; |
| |
| var proxy = config.proxy; |
| if (!proxy) { |
| var proxyEnv = protocol.slice(0, -1) + '_proxy'; |
| var proxyUrl = process.env[proxyEnv] || process.env[proxyEnv.toUpperCase()]; |
| if (proxyUrl) { |
| var parsedProxyUrl = url.parse(proxyUrl); |
| proxy = { |
| host: parsedProxyUrl.hostname, |
| port: parsedProxyUrl.port |
| }; |
| |
| if (parsedProxyUrl.auth) { |
| var proxyUrlAuth = parsedProxyUrl.auth.split(':'); |
| proxy.auth = { |
| username: proxyUrlAuth[0], |
| password: proxyUrlAuth[1] |
| }; |
| } |
| } |
| } |
| |
| if (proxy) { |
| options.hostname = proxy.host; |
| options.host = proxy.host; |
| options.headers.host = parsed.hostname + (parsed.port ? ':' + parsed.port : ''); |
| options.port = proxy.port; |
| options.path = protocol + '//' + parsed.hostname + (parsed.port ? ':' + parsed.port : '') + options.path; |
| |
| // Basic proxy authorization |
| if (proxy.auth) { |
| var base64 = new Buffer(proxy.auth.username + ':' + proxy.auth.password, 'utf8').toString('base64'); |
| options.headers['Proxy-Authorization'] = 'Basic ' + base64; |
| } |
| } |
| |
| var transport; |
| if (config.maxRedirects === 0) { |
| transport = isHttps ? https : http; |
| } else { |
| if (config.maxRedirects) { |
| options.maxRedirects = config.maxRedirects; |
| } |
| transport = isHttps ? httpsFollow : httpFollow; |
| } |
| |
| // Create the request |
| var req = transport.request(options, function handleResponse(res) { |
| if (aborted) return; |
| |
| // Response has been received so kill timer that handles request timeout |
| clearTimeout(timer); |
| timer = null; |
| |
| // uncompress the response body transparently if required |
| var stream = res; |
| 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: req |
| }; |
| |
| if (config.responseType === 'stream') { |
| response.data = stream; |
| settle(resolve, reject, response); |
| } else { |
| var responseBuffer = []; |
| stream.on('data', function handleStreamData(chunk) { |
| responseBuffer.push(chunk); |
| |
| // make sure the content length is not over the maxContentLength if specified |
| if (config.maxContentLength > -1 && Buffer.concat(responseBuffer).length > config.maxContentLength) { |
| reject(createError('maxContentLength size of ' + config.maxContentLength + ' exceeded', config)); |
| } |
| }); |
| |
| stream.on('error', function handleStreamError(err) { |
| if (aborted) return; |
| reject(enhanceError(err, config)); |
| }); |
| |
| stream.on('end', function handleStreamEnd() { |
| var responseData = Buffer.concat(responseBuffer); |
| if (config.responseType !== 'arraybuffer') { |
| responseData = responseData.toString('utf8'); |
| } |
| |
| response.data = responseData; |
| settle(resolve, reject, response); |
| }); |
| } |
| }); |
| |
| // Handle errors |
| req.on('error', function handleRequestError(err) { |
| if (aborted) return; |
| reject(enhanceError(err, config)); |
| }); |
| |
| // Handle request timeout |
| if (config.timeout && !timer) { |
| timer = setTimeout(function handleRequestTimeout() { |
| req.abort(); |
| reject(createError('timeout of ' + config.timeout + 'ms exceeded', config, 'ECONNABORTED')); |
| aborted = true; |
| }, config.timeout); |
| } |
| |
| if (config.cancelToken) { |
| // Handle cancellation |
| config.cancelToken.promise.then(function onCanceled(cancel) { |
| if (aborted) { |
| return; |
| } |
| |
| req.abort(); |
| reject(cancel); |
| aborted = true; |
| }); |
| } |
| |
| // Send the request |
| if (utils.isStream(data)) { |
| data.pipe(req); |
| } else { |
| req.end(data); |
| } |
| }); |
| }; |