Skip to content

Commit 6a6d2a9

Browse files
jopemachineszmarczaksindresorhus
authored
Support AbortController (#2020)
Co-authored-by: Szymon Marczak <[email protected]> Co-authored-by: Sindre Sorhus <[email protected]>
1 parent 3207061 commit 6a6d2a9

File tree

6 files changed

+356
-0
lines changed

6 files changed

+356
-0
lines changed

documentation/2-options.md

+22
Original file line numberDiff line numberDiff line change
@@ -209,6 +209,28 @@ await got('https://httpbin.org/anything');
209209
#### **Note:**
210210
> - If you're passing an absolute URL as `url`, you need to set `prefixUrl` to an empty string.
211211
212+
### `signal`
213+
214+
**Type: [`AbortSignal`](https://developer.mozilla.org/en-US/docs/Web/API/AbortSignal)**
215+
216+
You can abort the `request` using [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).
217+
218+
*Requires Node.js 16 or later.*
219+
220+
```js
221+
import got from 'got';
222+
223+
const abortController = new AbortController();
224+
225+
const request = got('https://httpbin.org/anything', {
226+
signal: abortController.signal
227+
});
228+
229+
setTimeout(() => {
230+
abortController.abort();
231+
}, 100);
232+
```
233+
212234
### `method`
213235

214236
**Type: `string`**\

documentation/8-errors.md

+6
Original file line numberDiff line numberDiff line change
@@ -97,3 +97,9 @@ When the request is aborted with `promise.cancel()`.
9797
**Code: `ERR_RETRYING`**
9898

9999
Always triggers a new retry when thrown.
100+
101+
### `AbortError`
102+
103+
**Code: `ERR_ABORTED`**
104+
105+
When the request is aborted with [AbortController.abort()](https://developer.mozilla.org/en-US/docs/Web/API/AbortController/abort).

source/core/errors.ts

+11
Original file line numberDiff line numberDiff line change
@@ -170,3 +170,14 @@ export class RetryError extends RequestError {
170170
this.code = 'ERR_RETRYING';
171171
}
172172
}
173+
174+
/**
175+
An error to be thrown when the request is aborted by AbortController.
176+
*/
177+
export class AbortError extends RequestError {
178+
constructor(request: Request) {
179+
super('This operation was aborted.', {}, request);
180+
this.code = 'ERR_ABORTED';
181+
this.name = 'AbortError';
182+
}
183+
}

source/core/index.ts

+9
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ import {
3232
TimeoutError,
3333
UploadError,
3434
CacheError,
35+
AbortError,
3536
} from './errors.js';
3637
import type {PlainResponse} from './response.js';
3738
import type {PromiseCookieJar, NativeRequestOptions, RetryOptions} from './options.js';
@@ -241,6 +242,14 @@ export default class Request extends Duplex implements RequestEvents<Request> {
241242
return;
242243
}
243244

245+
if (this.options.signal?.aborted) {
246+
this.destroy(new AbortError(this));
247+
}
248+
249+
this.options.signal?.addEventListener('abort', () => {
250+
this.destroy(new AbortError(this));
251+
});
252+
244253
// Important! If you replace `body` in a handler with another stream, make sure it's readable first.
245254
// The below is run only once.
246255
const {body} = this.options;

source/core/options.ts

+34
Original file line numberDiff line numberDiff line change
@@ -827,6 +827,7 @@ const defaultInternals: Options['_internals'] = {
827827
},
828828
setHost: true,
829829
maxHeaderSize: undefined,
830+
signal: undefined,
830831
enableUnixSockets: true,
831832
};
832833

@@ -1489,6 +1490,38 @@ export default class Options {
14891490
}
14901491
}
14911492

1493+
/**
1494+
You can abort the `request` using [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).
1495+
1496+
*Requires Node.js 16 or later.*
1497+
1498+
@example
1499+
```
1500+
import got from 'got';
1501+
1502+
const abortController = new AbortController();
1503+
1504+
const request = got('https://httpbin.org/anything', {
1505+
signal: abortController.signal
1506+
});
1507+
1508+
setTimeout(() => {
1509+
abortController.abort();
1510+
}, 100);
1511+
```
1512+
*/
1513+
// TODO: Replace `any` with `AbortSignal` when targeting Node 16.
1514+
get signal(): any | undefined {
1515+
return this._internals.signal;
1516+
}
1517+
1518+
// TODO: Replace `any` with `AbortSignal` when targeting Node 16.
1519+
set signal(value: any | undefined) {
1520+
assert.object(value);
1521+
1522+
this._internals.signal = value;
1523+
}
1524+
14921525
/**
14931526
Ignore invalid cookies instead of throwing an error.
14941527
Only useful when the `cookieJar` option has been set. Not recommended.
@@ -2488,5 +2521,6 @@ export default class Options {
24882521
Object.freeze(options.retry.methods);
24892522
Object.freeze(options.retry.statusCodes);
24902523
Object.freeze(options.context);
2524+
Object.freeze(options.signal);
24912525
}
24922526
}

0 commit comments

Comments
 (0)