zl程序教程

您现在的位置是:首页 >  后端

当前栏目

nodejs环境http请求

NodejsHTTP 环境 请求
2023-09-11 14:15:01 时间

axios在nodejs环境使用http或者https模块发送请求。

'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 createError = require('../core/createError');
var enhanceError = require('../core/enhanceError');

var isHttps = /https:?/;

/*eslint consistent-return:0*/
module.exports = function httpAdapter(config) {
  return new Promise(function dispatchHttpRequest(resolvePromise, rejectPromise) {//返回一个promise
    var timer;//定时器
    var resolve = function resolve(value) {//resolve前先将定时器清除
      clearTimeout(timer);
      resolvePromise(value);
    };
    var reject = function reject(value) {//reject前先将定时器清除
      clearTimeout(timer);
      rejectPromise(value);
    };
    var data = config.data;//data参数
    var headers = config.headers;//header请求头对象

    // Set User-Agent (required by some servers)
    // Only set header if it hasn't been set in config
    // See https://github.com/axios/axios/issues/69
    //设置User-Agent请求头
    if (!headers['User-Agent'] && !headers['user-agent']) {
      headers['User-Agent'] = 'axios/' + pkg.version;
    }

    if (data && !utils.isStream(data)) {
      //如果data参数存在且data不是流
      if (Buffer.isBuffer(data)) {//如果是Buffer,不作操作
        // Nothing to do...
      } else if (utils.isArrayBuffer(data)) {//如果是ArrayBuffer
        data = Buffer.from(new Uint8Array(data));
      } else if (utils.isString(data)) {//如果是字符串
        data = Buffer.from(data, 'utf-8');
      } else {//其他情况,抛出错误
        return reject(createError(
          'Data after transformation must be a string, an ArrayBuffer, a Buffer, or a Stream',
          config
        ));
      }

      // Add Content-Length header if data exists
      //如果data存在,添加Content-Length请求头,data长度
      headers['Content-Length'] = data.length;
    }

    // HTTP basic authentication
    var auth = undefined;
    if (config.auth) {//如果传递了auth配置项,把用户名和密码拼起来
      var username = config.auth.username || '';
      var password = config.auth.password || '';
      auth = username + ':' + password;
    }

    // Parse url
    var parsed = url.parse(config.url);//将完整url解析成部分组成的对象
    var protocol = parsed.protocol || 'http:';//协议

    if (!auth && parsed.auth) {//如果url中带有auth且没有传递配置项auth,就使用url中的auth参数
      var urlAuth = parsed.auth.split(':');
      var urlUsername = urlAuth[0] || '';
      var urlPassword = urlAuth[1] || '';
      auth = urlUsername + ':' + urlPassword;
    }

    if (auth) {//有auth参数,从请求头中删除Authorization
      delete headers.Authorization;
    }

    var isHttpsRequest = isHttps.test(protocol);//判断是否是https协议
    var agent = isHttpsRequest ? config.httpsAgent : config.httpAgent;
    //根据判断是否https来获取到对应的httpAgent

    var options = {
      path: buildURL(parsed.path, config.params, config.paramsSerializer).replace(/^\?/, ''),
      method: config.method.toUpperCase(),
      headers: headers,
      agent: agent,
      auth: auth
    };
    //基础选项,path调用buildURL方法创建一个带有查询参数的url

    if (config.socketPath) {//添加socketPath配置到options中
      options.socketPath = config.socketPath;
    } else {//没有socketPath配置,就添加域名和端口号到options中
      options.hostname = parsed.hostname;
      options.port = parsed.port;
    }

    var proxy = config.proxy;//proxy定义代理服务器的域名和端口号
    if (!proxy && proxy !== false) {//没有传递proxy参数
      var proxyEnv = protocol.slice(0, -1) + '_proxy';//协议名加上_proxy的字符串,代理url的环境变量名字
      var proxyUrl = process.env[proxyEnv] || process.env[proxyEnv.toUpperCase()];
      //从环境变量中获取到代理url
      if (proxyUrl) {//如果存在代理url
        var parsedProxyUrl = url.parse(proxyUrl);//将代理url解析
        var noProxyEnv = process.env.no_proxy || process.env.NO_PROXY;//NO_PROXY环境变量
        var shouldProxy = true;

        if (noProxyEnv) {//如果有NO_PROXY环境变量
          var noProxy = noProxyEnv.split(',').map(function trim(s) {
            return s.trim();
          });//处理noProxyEnv,逗号分开变成一个数组,然后去除每一个元素的空格

          shouldProxy = !noProxy.some(function proxyMatch(proxyElement) {
            //循环noProxy,寻找是否有一个元素和请求url的域名相等
            if (!proxyElement) {//当前proxyElement为空,返回
              return false;
            }
            if (proxyElement === '*') {//如果proxyElement是通配符,返回true
              return true;
            }
            if (proxyElement[0] === '.' &&
                parsed.hostname.substr(parsed.hostname.length - proxyElement.length) === proxyElement &&
                proxyElement.match(/\./g).length === parsed.hostname.match(/\./g).length) {
              return true;
            }

            return parsed.hostname === proxyElement;//判断proxyElement与请求url的域名是否相等
          });
        }


        if (shouldProxy) {//如果需要代理
          proxy = {
            host: parsedProxyUrl.hostname,//代理域名
            port: parsedProxyUrl.port//代理端口号
          };

          if (parsedProxyUrl.auth) {//如果代理url拥有auth属性,就给proxy参数加上auth
            var proxyUrlAuth = parsedProxyUrl.auth.split(':');
            proxy.auth = {
              username: proxyUrlAuth[0],
              password: proxyUrlAuth[1]
            };
          }
        }
      }
    }

    if (proxy) {//如果设置了代理参数,根据代理改变options中的配置项
      options.hostname = proxy.host;//主机名
      options.host = proxy.host;//主机
      options.headers.host = parsed.hostname + (parsed.port ? ':' + parsed.port : '');//host请求头
      options.port = proxy.port;//端口号
      options.path = protocol + '//' + parsed.hostname + (parsed.port ? ':' + parsed.port : '') + options.path;//url路径

      // Basic proxy authorization
      if (proxy.auth) {//如果代理中含有auth参数
        var base64 = Buffer.from(proxy.auth.username + ':' + proxy.auth.password, 'utf8').toString('base64');//base64转码用户名和密码
        options.headers['Proxy-Authorization'] = 'Basic ' + base64;//添加对应请求头
      }
    }

    var transport;
    var isHttpsProxy = isHttpsRequest && (proxy ? isHttps.test(proxy.protocol) : true);
    //判断是https协议且含有代理
    if (config.transport) {//如果传递了transport,直接存下
      transport = config.transport;
    } else if (config.maxRedirects === 0) {//如果不允许重定向
      transport = isHttpsProxy ? https : http;//判断使用https模块还是http模块
    } else {//如果允许重定向,使用follow-redirects模块,请求时会自动重定向
      if (config.maxRedirects) {//maxRedirects加入options中
        options.maxRedirects = config.maxRedirects;
      }
      transport = isHttpsProxy ? httpsFollow : httpFollow;
    }

    if (config.maxContentLength && config.maxContentLength > -1) {
      //如果设置了最大正文长度,配置options对应选项
      options.maxBodyLength = config.maxContentLength;
    }

    // Create the request
    //创建请求
    //http.request(option, callback)方法发送http请求,它会返回http.ClientRequest对象
    //通过在http.ClientRequest对象上添加监听器即可获得响应对象
    var req = transport.request(options, function handleResponse(res) {
      //传递给http.request的callback是response事件的监听器,只会触发一次
      //res是http.IncomingMessage对象,可以用来获取响应状态码,响应头和响应数据
      if (req.aborted) return;//如果请求已经被中止,返回

      // uncompress the response body transparently if required
      var stream = res;
      switch (res.headers['content-encoding']) {
        //判断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());//使用zlib压缩解压插件解压响应主体

        // remove the content-encoding in order to not confuse downstream operations
        //移除content-encoding响应头为了避免混淆后面的操作
        delete res.headers['content-encoding'];
        break;//跳出
      }

      // return the last request in case of redirects
      var lastRequest = res.req || req;//最后一次请求的请求对象,因为请求有可能发生了重定向

      var response = {//响应数据初始化
        status: res.statusCode,
        statusText: res.statusMessage,
        headers: res.headers,
        config: config,
        request: lastRequest
      };

      if (config.responseType === 'stream') {//如果请求配置中的responseType是stream格式
        response.data = stream;//response.data赋值为stream
        settle(resolve, reject, response);
        //调用settle来根据响应状态码来决定是否resolve或者reject当前promise
      } else {//如果配置要求的响应格式不是stream流格式
        var responseBuffer = [];
        stream.on('data', function handleStreamData(chunk) {
          //data事件,接收到数据块的时候触发
          responseBuffer.push(chunk);//将数据块插入responseBuffer数组

          // make sure the content length is not over the maxContentLength if specified
          //如果指定了最大响应正文长度,就判断是否响应数据长度超过了这个限制
          if (config.maxContentLength > -1 && Buffer.concat(responseBuffer).length > config.maxContentLength) {
            //如果响应数据长度超出限制长度,reject当前promise
            reject(createError('maxContentLength size of ' + config.maxContentLength + ' exceeded',
              config, null, lastRequest));
          }
        });

        stream.on('error', function handleStreamError(err) {
          //接收流数据块出错时触发error事件
          if (req.aborted) return;//如果请求对象已经中止,返回
          //否则reject当前promise
          reject(enhanceError(err, config, null, lastRequest));
        });

        stream.on('end', function handleStreamEnd() {
          //流数据传输完毕时触发end事件
          var responseData = Buffer.concat(responseBuffer);//将所有数据块连接成一个整体
          if (config.responseType !== 'arraybuffer') {//如果配置中的响应数据类型不是arrayBuffer
            //就调用toString方法转换格式,
            responseData = responseData.toString(config.responseEncodingresponseEncoding);
          }

          response.data = responseData;//响应数据赋值
          settle(resolve, reject, response);
          //调用settle来根据响应状态码来决定是否resolve或者reject当前promise
        });
      }
    });

    // Handle errors
    req.on('error', function handleRequestError(err) {
      //http.ClientRequest对象绑定error事件,发生错误的时候触发
      if (req.aborted) return;//如果请求对象已经中止,返回
      reject(enhanceError(err, config, null, req));//否则reject当前promise
    });

    // Handle request timeout
    if (config.timeout) {//如果配置项中有超时的配置
      timer = setTimeout(function handleRequestTimeout() {
        //新建一个定时器
        req.abort();//调用abort中止请求
        reject(createError('timeout of ' + config.timeout + 'ms exceeded', config, 'ECONNABORTED', req));//reject当前promise
      }, config.timeout);//超时时间过后执行
    }

    if (config.cancelToken) {//如果传递了cancelToken配置
      // Handle cancellation
      //设置了cancelToken的请求,只要用户调用了cancelToken对象的cancel方法,就会执行下面的then回调,中止请求
      config.cancelToken.promise.then(function onCanceled(cancel) {
        //对cancelToken对象上的promise指定resolve后的执行回调
        if (req.aborted) return;//如果请求已经中止,返回

        req.abort();//如果请求没有中止,就调用abort立即中止
        reject(cancel);//reject当前promise
      });
    }

    // Send the request
    if (utils.isStream(data)) {//如果config.data是流数据,就绑定error事件,读取数据出错就reject
      data.on('error', function handleStreamError(err) {
        reject(enhanceError(err, config, null, req));
      }).pipe(req);//将data读取完毕后用pipe传递给http.ClientRequest对象,可读流传递给可写流,用pipe方法
    } else {
      req.end(data);//否则关闭请求
    }
  });
};