如何在Node.js中正确取消http请求?

时间:2019-06-11 10:35:04

标签: javascript node.js httprequest cancellation

我需要在Node.js中实现可取消的客户端HTTP请求,而不使用外部库。我提供了一个Promise对象-cancellationPromise-当从外部请求取消时,该对象将被拒绝。这就是我知道我可能需要打电话给request.abort()的方式。

问题是,仅在request.abort()仍待处理且https.request对象尚不可用时,才应该调用response吗?

或者,即使我已经获得了response对象并正在处理响应数据,也应该像下面的代码中那样调用它吗?在那种情况下,这会阻止更多response.on('data')事件的发生吗?

  async simpleHttpRequest(url, oauthToken, cancellationPromise) {
    let cancelled = null;
    let oncancel = null;

    cancellationPromise.catch(error => { 
      cancelled = error; oncancel && oncancel(error) });

    try {
      const response = await new Promise((resolve, reject) => {
        const request = https.request(
          url.toString(), 
          { 
            method: 'GET', 
            headers: { 'Authorization': `Bearer ${oauthToken}` }        
          }, 
          resolve);

        oncancel = error => request.abort();
        request.on('error', reject);
        request.end();
      });

      if (cancelled) throw cancelled;

      // do I need "oncancel = null" here?
      // or should I still allow to call request.abort() while fetching the response's data?

      // oncancel = null;

      try {
        // read the response 
        const chunks = await new Promise((resolve, reject) => {
          response.on('error', reject);  
          const chunks = [];
          response.on('data', data => chunks.push(data));
          response.on('end', () => resolve(chunks));
        });

        if (cancelled) throw cancelled;
        const data = JSON.parse(chunks.join(''));
        return data;
      }
      finally {
        response.resume();
      }
    }
    finally {
      oncancel = null;
    }
  }

1 个答案:

答案 0 :(得分:6)

这取决于您要通过中止请求来实现什么。

只是一些背景。 HTTP 1无法“取消”它发送的请求,然后等待响应。您无法“回滚”您所做的请求。您需要进行分布式事务。 (Further reading.)与MDN developer document states一样:

  

如果已发送请求,则XMLHttpRequest.abort()方法将中止该请求。请求中止后,其readyState更改为XMLHttpRequest.UNSENT(0),并且请求的状态码设置为0。

基本上,您可以停止响应被应用程序处理。另一个应用程序可能会(如果您在发送给它后调用了abort(),则仍然可以完成其处理。

从问题的角度来看:

  

问题是,仅在request.abort()仍处于待处理状态且响应对象尚不可用时,才应该调用https.request吗?

TL.DR .:仅从您的应用程序角度来看很重要。当我浏览您的代码时,我认为它会很好地工作。