Node.js v15.0.0 提供了一个全局实用 API AbortController,用于在选定的基于 Promise API 中发出取消信号。无需引入在所有模块中均可使用,该 API 的实现是基于浏览器中的 Web API AbortController。
通俗的讲 AbortController 表示一个控制器对象,允许我们根据需要中止一个或多个 Web 请求。
下面是一个示例,在 1 秒后会执行 ac.abort() 方法,将会触发 abort 事件,并且仅会触发一次,这可通过 abortSignal.aborted 属性查看前后改变状态。
ac.signal.addEventListener('abort', () => {
console.log('Aborted!');
console.log('ac.signal.aborted:', ac.signal.aborted);
}, { once: true });
setTimeout(() => ac.abort(), 1000)
console.log('ac.signal.aborted:', ac.signal.aborted);
Node.js 中我们可以选择使用 node-fetch 这个请求处理库,传递 signal 给 fetch。
假设这个请求需要等待 5 秒钟,大约在 2 秒钟后执行 abort() 将会中止这个请求。
const ac = new AbortController();
import fetch from 'node-fetch';
const timer = setTimeout(() => ac.abort(), 2000)
try {
const { statusText } = await fetch('http://localhost:3000/api', { signal: ac.signal })
console.log(statusText);
} catch (err) {
console.log(err.name); // AbortError
} finally {
clearTimeout(timer);
}
传递 ac.signal 中止一个正在运行的 Promise,这需要我们为 ac.signal 注册一个 abort 事件,做一些处理。之后在任何地方调用 ac.abort() 中止 Promise。
使用 Promise 表示中止操作的任何 Web 平台 APIs 都必须遵循以下原则:
以下关于 doSomeThingAsync 这个异步 Promise Function 的实现基本上也是遵循的这些规则。
class AbortError extends Error {
constructor(message) {
super(message);
this.name = 'AbortError';
}
}
function doSomethingAsync({ ac }) {
return new Promise((resolve, reject) => {
console.log('task start...');
if (ac.aborted) {
return reject(new AbortError('task handler failed', 'AbortError'));
}
const timer = setTimeout(() => {
console.log('task end...');
resolve(1);
}, 5000);
ac.signal.addEventListener('abort', () => {
clearTimeout(timer);
reject(new AbortError('task handler failed', 'AbortError'));
}, { once: true });
});
}
setTimeout(() => ac.abort(), 2000)
try {
await doSomethingAsync({ ac });
} catch (err) {
console.error(err.name, err.message); // AbortError task handler failed
}
注意:在 Node.js 中目前并没有 DOMException 这个类,我们无法这样做 new DOMException('task handler failed', 'AbortError')
所以我在刚开始先创建了一个 AbortError 类来模拟。
Node.js 中已经有一些异步 API 支持传递 signal,但是它的 DOMException 错误也是在内部通过封装来实现的:
// https://github.com/nodejs/node/blob/f6b1df2226/lib/internal/fs/promises.js#L98
const lazyDOMException = hideStackFrames((message, name) => {
if (DOMException === undefined)
DOMException = internalBinding('messaging').DOMException;
return new DOMException(message, name);
});
// 例如 writeFileHandle
// https://github.com/nodejs/node/blob/f6b1df2226/lib/internal/fs/promises.js#L282
if (signal?.aborted) {
throw lazyDOMException('The operation was aborted', 'AbortError');
}