fetch-retrier 0.1.8 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +28 -5
- package/lib/index.d.ts +74 -7
- package/lib/index.js +112 -16
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -2,12 +2,18 @@
|
|
|
2
2
|
|
|
3
3
|
A lightweight wrapper around `fetch` that adds **retries**, **timeout**, and **full jitter** backoff. Useful for calling HTTP APIs that may be rate-limited (429) or temporarily unavailable (5xx).
|
|
4
4
|
|
|
5
|
+
[](https://www.npmjs.com/package/fetch-retrier)
|
|
6
|
+
[](https://www.npmjs.com/package/fetch-retrier)
|
|
7
|
+
[](https://github.com/gammarers-labs/fetch-retrier/actions/workflows/build.yml)
|
|
8
|
+
[](https://github.com/gammarers-labs/fetch-retrier/actions/workflows/release.yml)
|
|
9
|
+
|
|
5
10
|
## Features
|
|
6
11
|
|
|
7
12
|
- **Configurable retries** – Set the maximum number of attempts per request.
|
|
8
13
|
- **Per-request timeout** – Abort requests that exceed a given duration.
|
|
9
14
|
- **Full jitter backoff** – Exponential backoff with random jitter (AWS-style) between retries.
|
|
10
15
|
- **Custom retry predicate** – Control which status codes trigger a retry (default: 429, 500, 502, 503, 504).
|
|
16
|
+
- **External cancellation** – Pass an `AbortSignal` to cancel an in-flight request.
|
|
11
17
|
- **TypeScript** – Exported types for `RequestOptions` and usage in TS/JS.
|
|
12
18
|
|
|
13
19
|
## Installation
|
|
@@ -54,6 +60,7 @@ if (response.ok) {
|
|
|
54
60
|
| `timeoutMs` | `number` | Yes | Timeout in milliseconds for each request. Requests are aborted when this is exceeded. |
|
|
55
61
|
| `baseBackoffMs` | `number` | Yes | Base delay in milliseconds for backoff. Delay is capped at `baseBackoffMs * 2^attempt` and randomized (full jitter). |
|
|
56
62
|
| `headers` | `Record<string, string>` | No | Headers to send with the request. |
|
|
63
|
+
| `signal` | `AbortSignal` | No | Optional external abort signal. If already aborted, `FetchRetrierAlreadyAbortedError` is thrown. If aborted during an attempt, the request is aborted and retried until `retries` is exhausted. |
|
|
57
64
|
| `shouldRetry` | `(response: Response, body: string) => boolean` | No | Custom predicate. Return `true` to retry on this response. Default: retry on status 429, 500, 502, 503, 504. |
|
|
58
65
|
|
|
59
66
|
### Custom retry logic
|
|
@@ -72,13 +79,29 @@ const response = await fetchRetrier('https://api.example.com/data', {
|
|
|
72
79
|
});
|
|
73
80
|
```
|
|
74
81
|
|
|
82
|
+
### Cancellation with `AbortController`
|
|
83
|
+
|
|
84
|
+
```typescript
|
|
85
|
+
const controller = new AbortController();
|
|
86
|
+
|
|
87
|
+
setTimeout(() => controller.abort(), 250);
|
|
88
|
+
|
|
89
|
+
await fetchRetrier('https://api.example.com/data', {
|
|
90
|
+
retries: 3,
|
|
91
|
+
timeoutMs: 5000,
|
|
92
|
+
baseBackoffMs: 250,
|
|
93
|
+
signal: controller.signal,
|
|
94
|
+
});
|
|
95
|
+
```
|
|
96
|
+
|
|
75
97
|
## Retry and error behavior
|
|
76
98
|
|
|
77
99
|
- **Success** – If `response.ok` is true, the response is returned immediately.
|
|
78
|
-
- **Retriable failure** – If the response is not OK and `shouldRetry(response, body)` returns true, the client waits (full jitter backoff) and retries until `retries` is exhausted. On the last attempt,
|
|
79
|
-
- **Non-retriable failure** – If `shouldRetry` returns false,
|
|
80
|
-
- **Timeout** – If a request exceeds `timeoutMs`, it is aborted and retried (same backoff) until `retries` is exhausted
|
|
81
|
-
- **Network/TypeError** – Network errors and related `TypeError`s are retried with backoff; after the last attempt, the error
|
|
100
|
+
- **Retriable failure** – If the response is not OK and `shouldRetry(response, body)` returns true, the client waits (full jitter backoff) and retries until `retries` is exhausted. On the last attempt, `FetchRetrierHttpError` is thrown (includes `status`).
|
|
101
|
+
- **Non-retriable failure** – If `shouldRetry` returns false, `FetchRetrierHttpError` is thrown immediately (e.g. message `Non-retriable HTTP error: 404`).
|
|
102
|
+
- **Timeout** – If a request exceeds `timeoutMs`, it is aborted and retried (same backoff) until `retries` is exhausted; the final failure is `FetchRetrierAbortError`.
|
|
103
|
+
- **Network/TypeError** – Network errors and related `TypeError`s are retried with backoff; after the last attempt, `FetchRetrierNetworkError` is thrown with the original error as `cause`.
|
|
104
|
+
- **Already aborted signal** – If `signal` is already aborted before an attempt starts, `FetchRetrierAlreadyAbortedError` is thrown immediately (no attempt is made).
|
|
82
105
|
|
|
83
106
|
## Requirements
|
|
84
107
|
|
|
@@ -87,4 +110,4 @@ const response = await fetchRetrier('https://api.example.com/data', {
|
|
|
87
110
|
|
|
88
111
|
## License
|
|
89
112
|
|
|
90
|
-
This project is licensed under the Apache-2.0 License.
|
|
113
|
+
This project is licensed under the (Apache-2.0) License.
|
package/lib/index.d.ts
CHANGED
|
@@ -1,21 +1,88 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
2
|
+
* Configuration for {@link fetchRetrier}.
|
|
3
3
|
*/
|
|
4
4
|
export interface RequestOptions {
|
|
5
|
+
/** Optional HTTP headers sent with each attempt. */
|
|
5
6
|
headers?: Record<string, string>;
|
|
7
|
+
/** Maximum number of attempts, including the first. */
|
|
6
8
|
retries: number;
|
|
9
|
+
/** Per-attempt timeout in milliseconds; uses an internal {@link AbortController} when exceeded. */
|
|
7
10
|
timeoutMs: number;
|
|
11
|
+
/**
|
|
12
|
+
* Base backoff in milliseconds for full jitter. The cap for attempt `n` is `baseBackoffMs * 2^n`.
|
|
13
|
+
*/
|
|
8
14
|
baseBackoffMs: number;
|
|
9
15
|
/**
|
|
10
|
-
*
|
|
11
|
-
*
|
|
16
|
+
* Optional external {@link AbortSignal}. When aborted, the in-flight request is aborted; on the
|
|
17
|
+
* final attempt, cancellation surfaces as {@link FetchRetrierAbortError}.
|
|
18
|
+
*/
|
|
19
|
+
signal?: AbortSignal;
|
|
20
|
+
/**
|
|
21
|
+
* Return `true` to schedule another attempt for this non-OK response.
|
|
22
|
+
* Default: retry on status 429, 500, 502, 503, or 504.
|
|
12
23
|
*/
|
|
13
24
|
shouldRetry?: (response: Response, body: string) => boolean;
|
|
14
25
|
}
|
|
26
|
+
/** Error thrown when a request is cancelled by timeout or an external {@link AbortSignal}. */
|
|
27
|
+
export declare class FetchRetrierAbortError extends Error {
|
|
28
|
+
readonly name: string;
|
|
29
|
+
/**
|
|
30
|
+
* @param message - Human-readable reason (default: `'Aborted'`)
|
|
31
|
+
*/
|
|
32
|
+
constructor(message?: string);
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Error thrown when {@link RequestOptions.signal} is already aborted before an attempt starts.
|
|
36
|
+
*/
|
|
37
|
+
export declare class FetchRetrierAlreadyAbortedError extends FetchRetrierAbortError {
|
|
38
|
+
readonly name: string;
|
|
39
|
+
/**
|
|
40
|
+
* @param message - Human-readable reason (default: `'Signal was already aborted'`)
|
|
41
|
+
*/
|
|
42
|
+
constructor(message?: string);
|
|
43
|
+
}
|
|
44
|
+
/** Error thrown when the server returns a non-OK HTTP status and no further retry is performed. */
|
|
45
|
+
export declare class FetchRetrierHttpError extends Error {
|
|
46
|
+
readonly status: number;
|
|
47
|
+
readonly name: string;
|
|
48
|
+
/**
|
|
49
|
+
* @param message - Error description
|
|
50
|
+
* @param status - HTTP status code from the response
|
|
51
|
+
*/
|
|
52
|
+
constructor(message: string, status: number);
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Error thrown when a fetch fails with a network-level error (e.g. DNS failure, connection refused).
|
|
56
|
+
*/
|
|
57
|
+
export declare class FetchRetrierNetworkError extends Error {
|
|
58
|
+
readonly cause?: unknown | undefined;
|
|
59
|
+
readonly name: string;
|
|
60
|
+
/**
|
|
61
|
+
* @param message - Human-readable reason (default: `'Network error'`)
|
|
62
|
+
* @param cause - Original error, if any
|
|
63
|
+
*/
|
|
64
|
+
constructor(message?: string, cause?: unknown | undefined);
|
|
65
|
+
}
|
|
66
|
+
/** Error thrown when an internal invariant fails (should not happen in normal use). */
|
|
67
|
+
export declare class FetchRetrierUnreachableError extends Error {
|
|
68
|
+
readonly name: string;
|
|
69
|
+
/**
|
|
70
|
+
* @param message - Human-readable reason (default: `'Unreachable'`)
|
|
71
|
+
*/
|
|
72
|
+
constructor(message?: string);
|
|
73
|
+
}
|
|
15
74
|
/**
|
|
16
|
-
*
|
|
17
|
-
* @
|
|
18
|
-
*
|
|
19
|
-
*
|
|
75
|
+
* Performs `fetch` with retries, a per-attempt timeout, exponential backoff with full jitter,
|
|
76
|
+
* optional {@link RequestOptions.signal} cancellation, and a configurable retry predicate for
|
|
77
|
+
* non-OK responses.
|
|
78
|
+
*
|
|
79
|
+
* @param url - Request URL
|
|
80
|
+
* @param options - Retries, backoff, timeout, optional abort signal, and optional retry predicate
|
|
81
|
+
* @returns The first successful (OK) {@link Response}
|
|
82
|
+
* @throws {FetchRetrierAlreadyAbortedError} If `options.signal` is already aborted before an attempt
|
|
83
|
+
* @throws {FetchRetrierHttpError} On a non-OK response that is not retried or after the last attempt
|
|
84
|
+
* @throws {FetchRetrierNetworkError} On a network error on the final attempt
|
|
85
|
+
* @throws {FetchRetrierAbortError} On timeout or external abort on the final attempt
|
|
86
|
+
* @throws {FetchRetrierUnreachableError} If the retry loop exits without returning (internal bug)
|
|
20
87
|
*/
|
|
21
88
|
export declare const fetchRetrier: (url: string, options: RequestOptions) => Promise<Response>;
|
package/lib/index.js
CHANGED
|
@@ -1,26 +1,116 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.fetchRetrier = void 0;
|
|
3
|
+
exports.fetchRetrier = exports.FetchRetrierUnreachableError = exports.FetchRetrierNetworkError = exports.FetchRetrierHttpError = exports.FetchRetrierAlreadyAbortedError = exports.FetchRetrierAbortError = void 0;
|
|
4
|
+
/** Error thrown when a request is cancelled by timeout or an external {@link AbortSignal}. */
|
|
5
|
+
class FetchRetrierAbortError extends Error {
|
|
6
|
+
/**
|
|
7
|
+
* @param message - Human-readable reason (default: `'Aborted'`)
|
|
8
|
+
*/
|
|
9
|
+
constructor(message = 'Aborted') {
|
|
10
|
+
super(message);
|
|
11
|
+
this.name = 'FetchRetrierAbortError';
|
|
12
|
+
Object.setPrototypeOf(this, FetchRetrierAbortError.prototype);
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
exports.FetchRetrierAbortError = FetchRetrierAbortError;
|
|
16
|
+
/**
|
|
17
|
+
* Error thrown when {@link RequestOptions.signal} is already aborted before an attempt starts.
|
|
18
|
+
*/
|
|
19
|
+
class FetchRetrierAlreadyAbortedError extends FetchRetrierAbortError {
|
|
20
|
+
/**
|
|
21
|
+
* @param message - Human-readable reason (default: `'Signal was already aborted'`)
|
|
22
|
+
*/
|
|
23
|
+
constructor(message = 'Signal was already aborted') {
|
|
24
|
+
super(message);
|
|
25
|
+
this.name = 'FetchRetrierAlreadyAbortedError';
|
|
26
|
+
Object.setPrototypeOf(this, FetchRetrierAlreadyAbortedError.prototype);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
exports.FetchRetrierAlreadyAbortedError = FetchRetrierAlreadyAbortedError;
|
|
30
|
+
/** Error thrown when the server returns a non-OK HTTP status and no further retry is performed. */
|
|
31
|
+
class FetchRetrierHttpError extends Error {
|
|
32
|
+
/**
|
|
33
|
+
* @param message - Error description
|
|
34
|
+
* @param status - HTTP status code from the response
|
|
35
|
+
*/
|
|
36
|
+
constructor(message, status) {
|
|
37
|
+
super(message);
|
|
38
|
+
this.status = status;
|
|
39
|
+
this.name = 'FetchRetrierHttpError';
|
|
40
|
+
Object.setPrototypeOf(this, FetchRetrierHttpError.prototype);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
exports.FetchRetrierHttpError = FetchRetrierHttpError;
|
|
44
|
+
/**
|
|
45
|
+
* Error thrown when a fetch fails with a network-level error (e.g. DNS failure, connection refused).
|
|
46
|
+
*/
|
|
47
|
+
class FetchRetrierNetworkError extends Error {
|
|
48
|
+
/**
|
|
49
|
+
* @param message - Human-readable reason (default: `'Network error'`)
|
|
50
|
+
* @param cause - Original error, if any
|
|
51
|
+
*/
|
|
52
|
+
constructor(message = 'Network error', cause) {
|
|
53
|
+
super(message);
|
|
54
|
+
this.cause = cause;
|
|
55
|
+
this.name = 'FetchRetrierNetworkError';
|
|
56
|
+
Object.setPrototypeOf(this, FetchRetrierNetworkError.prototype);
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
exports.FetchRetrierNetworkError = FetchRetrierNetworkError;
|
|
60
|
+
/** Error thrown when an internal invariant fails (should not happen in normal use). */
|
|
61
|
+
class FetchRetrierUnreachableError extends Error {
|
|
62
|
+
/**
|
|
63
|
+
* @param message - Human-readable reason (default: `'Unreachable'`)
|
|
64
|
+
*/
|
|
65
|
+
constructor(message = 'Unreachable') {
|
|
66
|
+
super(message);
|
|
67
|
+
this.name = 'FetchRetrierUnreachableError';
|
|
68
|
+
Object.setPrototypeOf(this, FetchRetrierUnreachableError.prototype);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
exports.FetchRetrierUnreachableError = FetchRetrierUnreachableError;
|
|
72
|
+
/**
|
|
73
|
+
* Default {@link RequestOptions.shouldRetry} implementation: retry on HTTP 429, 500, 502, 503, 504.
|
|
74
|
+
*/
|
|
4
75
|
const defaultShouldRetry = (res) => {
|
|
5
76
|
return [429, 500, 502, 503, 504].includes(res.status);
|
|
6
77
|
};
|
|
7
78
|
/**
|
|
8
|
-
*
|
|
9
|
-
* @
|
|
10
|
-
*
|
|
11
|
-
*
|
|
79
|
+
* Performs `fetch` with retries, a per-attempt timeout, exponential backoff with full jitter,
|
|
80
|
+
* optional {@link RequestOptions.signal} cancellation, and a configurable retry predicate for
|
|
81
|
+
* non-OK responses.
|
|
82
|
+
*
|
|
83
|
+
* @param url - Request URL
|
|
84
|
+
* @param options - Retries, backoff, timeout, optional abort signal, and optional retry predicate
|
|
85
|
+
* @returns The first successful (OK) {@link Response}
|
|
86
|
+
* @throws {FetchRetrierAlreadyAbortedError} If `options.signal` is already aborted before an attempt
|
|
87
|
+
* @throws {FetchRetrierHttpError} On a non-OK response that is not retried or after the last attempt
|
|
88
|
+
* @throws {FetchRetrierNetworkError} On a network error on the final attempt
|
|
89
|
+
* @throws {FetchRetrierAbortError} On timeout or external abort on the final attempt
|
|
90
|
+
* @throws {FetchRetrierUnreachableError} If the retry loop exits without returning (internal bug)
|
|
12
91
|
*/
|
|
13
92
|
const fetchRetrier = async (url, options) => {
|
|
14
|
-
const { headers, retries, timeoutMs, baseBackoffMs, shouldRetry = defaultShouldRetry } = options;
|
|
93
|
+
const { headers, retries, timeoutMs, baseBackoffMs, signal: externalSignal, shouldRetry = defaultShouldRetry } = options;
|
|
15
94
|
for (let attempt = 1; attempt <= retries; attempt++) {
|
|
95
|
+
if (externalSignal?.aborted) {
|
|
96
|
+
throw new FetchRetrierAlreadyAbortedError();
|
|
97
|
+
}
|
|
16
98
|
const controller = new AbortController();
|
|
17
99
|
const timer = setTimeout(() => controller.abort(), timeoutMs);
|
|
100
|
+
const onExternalAbort = () => {
|
|
101
|
+
clearTimeout(timer);
|
|
102
|
+
controller.abort();
|
|
103
|
+
};
|
|
104
|
+
if (externalSignal) {
|
|
105
|
+
externalSignal.addEventListener('abort', onExternalAbort);
|
|
106
|
+
}
|
|
18
107
|
try {
|
|
19
108
|
const res = await fetch(url, {
|
|
20
109
|
headers,
|
|
21
110
|
signal: controller.signal,
|
|
22
111
|
});
|
|
23
112
|
clearTimeout(timer);
|
|
113
|
+
externalSignal?.removeEventListener('abort', onExternalAbort);
|
|
24
114
|
if (res.ok) {
|
|
25
115
|
return res;
|
|
26
116
|
}
|
|
@@ -28,45 +118,51 @@ const fetchRetrier = async (url, options) => {
|
|
|
28
118
|
const isContinue = shouldRetry(res, text);
|
|
29
119
|
if (isContinue) {
|
|
30
120
|
if (attempt === retries) {
|
|
31
|
-
throw new
|
|
121
|
+
throw new FetchRetrierHttpError(`HTTP ${res.status}`, res.status);
|
|
32
122
|
}
|
|
33
123
|
await wait(fullJitter(baseBackoffMs, attempt));
|
|
34
124
|
}
|
|
35
125
|
else {
|
|
36
|
-
throw new
|
|
126
|
+
throw new FetchRetrierHttpError(`Non-retriable HTTP error: ${res.status}`, res.status);
|
|
37
127
|
}
|
|
38
128
|
}
|
|
39
129
|
catch (err) {
|
|
40
130
|
clearTimeout(timer);
|
|
131
|
+
externalSignal?.removeEventListener('abort', onExternalAbort);
|
|
41
132
|
if (err instanceof Error && err.name === 'AbortError') {
|
|
42
133
|
if (attempt === retries)
|
|
43
|
-
throw err;
|
|
134
|
+
throw err instanceof FetchRetrierAbortError ? err : new FetchRetrierAbortError();
|
|
44
135
|
await wait(fullJitter(baseBackoffMs, attempt));
|
|
45
136
|
continue;
|
|
46
137
|
}
|
|
47
138
|
if (err instanceof TypeError) {
|
|
48
139
|
if (attempt === retries)
|
|
49
|
-
throw err;
|
|
140
|
+
throw new FetchRetrierNetworkError('Network error', err);
|
|
50
141
|
await wait(fullJitter(baseBackoffMs, attempt));
|
|
51
142
|
continue;
|
|
52
143
|
}
|
|
53
144
|
throw err;
|
|
54
145
|
}
|
|
55
146
|
}
|
|
56
|
-
throw new
|
|
147
|
+
throw new FetchRetrierUnreachableError();
|
|
57
148
|
};
|
|
58
149
|
exports.fetchRetrier = fetchRetrier;
|
|
150
|
+
/**
|
|
151
|
+
* @param ms - Delay in milliseconds
|
|
152
|
+
* @returns A promise that resolves after `ms`
|
|
153
|
+
*/
|
|
59
154
|
const wait = (ms) => {
|
|
60
155
|
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
61
156
|
};
|
|
62
157
|
/**
|
|
63
|
-
* AWS
|
|
64
|
-
*
|
|
65
|
-
* @param
|
|
66
|
-
* @
|
|
158
|
+
* Full jitter backoff: random delay in `[0, base * 2^attempt)` ms (AWS-recommended pattern).
|
|
159
|
+
*
|
|
160
|
+
* @param base - Base backoff in milliseconds
|
|
161
|
+
* @param attempt - 1-based attempt index (first retry uses `attempt === 1`)
|
|
162
|
+
* @returns Wait duration in milliseconds before the next attempt
|
|
67
163
|
*/
|
|
68
164
|
const fullJitter = (base, attempt) => {
|
|
69
165
|
const cap = base * Math.pow(2, attempt);
|
|
70
166
|
return Math.floor(Math.random() * cap);
|
|
71
167
|
};
|
|
72
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBZUEsTUFBTSxrQkFBa0IsR0FBRyxDQUFDLEdBQWEsRUFBVyxFQUFFO0lBQ3BELE9BQU8sQ0FBQyxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxDQUFDLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQztBQUN4RCxDQUFDLENBQUM7QUFFRjs7Ozs7R0FLRztBQUNJLE1BQU0sWUFBWSxHQUFHLEtBQUssRUFBRSxHQUFXLEVBQUUsT0FBdUIsRUFBcUIsRUFBRTtJQUM1RixNQUFNLEVBQUUsT0FBTyxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsYUFBYSxFQUFFLFdBQVcsR0FBRyxrQkFBa0IsRUFBRSxHQUFHLE9BQU8sQ0FBQztJQUVqRyxLQUFLLElBQUksT0FBTyxHQUFHLENBQUMsRUFBRSxPQUFPLElBQUksT0FBTyxFQUFFLE9BQU8sRUFBRSxFQUFFLENBQUM7UUFDcEQsTUFBTSxVQUFVLEdBQUcsSUFBSSxlQUFlLEVBQUUsQ0FBQztRQUN6QyxNQUFNLEtBQUssR0FBRyxVQUFVLENBQUMsR0FBRyxFQUFFLENBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxFQUFFLFNBQVMsQ0FBQyxDQUFDO1FBRTlELElBQUksQ0FBQztZQUNILE1BQU0sR0FBRyxHQUFHLE1BQU0sS0FBSyxDQUFDLEdBQUcsRUFBRTtnQkFDM0IsT0FBTztnQkFDUCxNQUFNLEVBQUUsVUFBVSxDQUFDLE1BQU07YUFDMUIsQ0FBQyxDQUFDO1lBRUgsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBRXBCLElBQUksR0FBRyxDQUFDLEVBQUUsRUFBRSxDQUFDO2dCQUNYLE9BQU8sR0FBRyxDQUFDO1lBQ2IsQ0FBQztZQUVELE1BQU0sSUFBSSxHQUFHLE1BQU0sR0FBRyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQzlCLE1BQU0sVUFBVSxHQUFHLFdBQVcsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7WUFFMUMsSUFBSSxVQUFVLEVBQUUsQ0FBQztnQkFDZixJQUFJLE9BQU8sS0FBSyxPQUFPLEVBQUUsQ0FBQztvQkFDeEIsTUFBTSxJQUFJLEtBQUssQ0FBQyxRQUFRLEdBQUcsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO2dCQUN4QyxDQUFDO2dCQUNELE1BQU0sSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLEVBQUUsT0FBTyxDQUFDLENBQUMsQ0FBQztZQUNqRCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sTUFBTSxJQUFJLEtBQUssQ0FBQyw2QkFBNkIsR0FBRyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUM7WUFDN0QsQ0FBQztRQUNILENBQUM7UUFBQyxPQUFPLEdBQVksRUFBRSxDQUFDO1lBQ3RCLFlBQVksQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUVwQixJQUFJLEdBQUcsWUFBWSxLQUFLLElBQUksR0FBRyxDQUFDLElBQUksS0FBSyxZQUFZLEVBQUUsQ0FBQztnQkFDdEQsSUFBSSxPQUFPLEtBQUssT0FBTztvQkFBRSxNQUFNLEdBQUcsQ0FBQztnQkFDbkMsTUFBTSxJQUFJLENBQUMsVUFBVSxDQUFDLGFBQWEsRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUFDO2dCQUMvQyxTQUFTO1lBQ1gsQ0FBQztZQUVELElBQUksR0FBRyxZQUFZLFNBQVMsRUFBRSxDQUFDO2dCQUM3QixJQUFJLE9BQU8sS0FBSyxPQUFPO29CQUFFLE1BQU0sR0FBRyxDQUFDO2dCQUNuQyxNQUFNLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7Z0JBQy9DLFNBQVM7WUFDWCxDQUFDO1lBRUQsTUFBTSxHQUFHLENBQUM7UUFDWixDQUFDO0lBQ0gsQ0FBQztJQUVELE1BQU0sSUFBSSxLQUFLLENBQUMsYUFBYSxDQUFDLENBQUM7QUFDakMsQ0FBQyxDQUFDO0FBbERXLFFBQUEsWUFBWSxnQkFrRHZCO0FBRUYsTUFBTSxJQUFJLEdBQUcsQ0FBQyxFQUFVLEVBQWlCLEVBQUU7SUFDekMsT0FBTyxJQUFJLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsVUFBVSxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDO0FBQzNELENBQUMsQ0FBQztBQUVGOzs7OztHQUtHO0FBQ0gsTUFBTSxVQUFVLEdBQUcsQ0FBQyxJQUFZLEVBQUUsT0FBZSxFQUFVLEVBQUU7SUFDM0QsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0lBQ3hDLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsR0FBRyxDQUFDLENBQUM7QUFDekMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiByZXF1ZXN0IE9wdGlvbnNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSZXF1ZXN0T3B0aW9ucyB7XG4gIGhlYWRlcnM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICByZXRyaWVzOiBudW1iZXI7XG4gIHRpbWVvdXRNczogbnVtYmVyO1xuICBiYXNlQmFja29mZk1zOiBudW1iZXI7XG4gIC8qKlxuICAgKiBDdXN0b20gcHJlZGljYXRlOiByZXR1cm4gdHJ1ZSB0byByZXRyeSBvbiB0aGlzIHJlc3BvbnNlLlxuICAgKiBEZWZhdWx0OiByZXRyeSBvbiA0MjksIDUwMCwgNTAyLCA1MDMsIDUwNFxuICAgKi9cbiAgc2hvdWxkUmV0cnk/OiAocmVzcG9uc2U6IFJlc3BvbnNlLCBib2R5OiBzdHJpbmcpID0+IGJvb2xlYW47XG59XG5cbmNvbnN0IGRlZmF1bHRTaG91bGRSZXRyeSA9IChyZXM6IFJlc3BvbnNlKTogYm9vbGVhbiA9PiB7XG4gIHJldHVybiBbNDI5LCA1MDAsIDUwMiwgNTAzLCA1MDRdLmluY2x1ZGVzKHJlcy5zdGF0dXMpO1xufTtcblxuLyoqXG4gKiByZXRyeSArIHRpbWVvdXQgKyBGdWxsIEppdHRlclxuICogQHBhcmFtIHVybCAtIFRoZSBVUkwgdG8gZmV0Y2hcbiAqIEBwYXJhbSBvcHRpb25zIC0gVGhlIG9wdGlvbnMgZm9yIHRoZSBmZXRjaFxuICogQHJldHVybnMgVGhlIHJlc3BvbnNlXG4gKi9cbmV4cG9ydCBjb25zdCBmZXRjaFJldHJpZXIgPSBhc3luYyAodXJsOiBzdHJpbmcsIG9wdGlvbnM6IFJlcXVlc3RPcHRpb25zKTogUHJvbWlzZTxSZXNwb25zZT4gPT4ge1xuICBjb25zdCB7IGhlYWRlcnMsIHJldHJpZXMsIHRpbWVvdXRNcywgYmFzZUJhY2tvZmZNcywgc2hvdWxkUmV0cnkgPSBkZWZhdWx0U2hvdWxkUmV0cnkgfSA9IG9wdGlvbnM7XG5cbiAgZm9yIChsZXQgYXR0ZW1wdCA9IDE7IGF0dGVtcHQgPD0gcmV0cmllczsgYXR0ZW1wdCsrKSB7XG4gICAgY29uc3QgY29udHJvbGxlciA9IG5ldyBBYm9ydENvbnRyb2xsZXIoKTtcbiAgICBjb25zdCB0aW1lciA9IHNldFRpbWVvdXQoKCkgPT4gY29udHJvbGxlci5hYm9ydCgpLCB0aW1lb3V0TXMpO1xuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlcyA9IGF3YWl0IGZldGNoKHVybCwge1xuICAgICAgICBoZWFkZXJzLFxuICAgICAgICBzaWduYWw6IGNvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgfSk7XG5cbiAgICAgIGNsZWFyVGltZW91dCh0aW1lcik7XG5cbiAgICAgIGlmIChyZXMub2spIHtcbiAgICAgICAgcmV0dXJuIHJlcztcbiAgICAgIH1cblxuICAgICAgY29uc3QgdGV4dCA9IGF3YWl0IHJlcy50ZXh0KCk7XG4gICAgICBjb25zdCBpc0NvbnRpbnVlID0gc2hvdWxkUmV0cnkocmVzLCB0ZXh0KTtcblxuICAgICAgaWYgKGlzQ29udGludWUpIHtcbiAgICAgICAgaWYgKGF0dGVtcHQgPT09IHJldHJpZXMpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEhUVFAgJHtyZXMuc3RhdHVzfWApO1xuICAgICAgICB9XG4gICAgICAgIGF3YWl0IHdhaXQoZnVsbEppdHRlcihiYXNlQmFja29mZk1zLCBhdHRlbXB0KSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYE5vbi1yZXRyaWFibGUgSFRUUCBlcnJvcjogJHtyZXMuc3RhdHVzfWApO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycjogdW5rbm93bikge1xuICAgICAgY2xlYXJUaW1lb3V0KHRpbWVyKTtcblxuICAgICAgaWYgKGVyciBpbnN0YW5jZW9mIEVycm9yICYmIGVyci5uYW1lID09PSAnQWJvcnRFcnJvcicpIHtcbiAgICAgICAgaWYgKGF0dGVtcHQgPT09IHJldHJpZXMpIHRocm93IGVycjtcbiAgICAgICAgYXdhaXQgd2FpdChmdWxsSml0dGVyKGJhc2VCYWNrb2ZmTXMsIGF0dGVtcHQpKTtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGlmIChlcnIgaW5zdGFuY2VvZiBUeXBlRXJyb3IpIHtcbiAgICAgICAgaWYgKGF0dGVtcHQgPT09IHJldHJpZXMpIHRocm93IGVycjtcbiAgICAgICAgYXdhaXQgd2FpdChmdWxsSml0dGVyKGJhc2VCYWNrb2ZmTXMsIGF0dGVtcHQpKTtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIHRocm93IGVycjtcbiAgICB9XG4gIH1cblxuICB0aHJvdyBuZXcgRXJyb3IoJ1VucmVhY2hhYmxlJyk7XG59O1xuXG5jb25zdCB3YWl0ID0gKG1zOiBudW1iZXIpOiBQcm9taXNlPHZvaWQ+ID0+IHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KHJlc29sdmUsIG1zKSk7XG59O1xuXG4vKipcbiAqIEFXUyByZWNvbW1lbmRlZCBGdWxsIEppdHRlclxuICogQHBhcmFtIGJhc2UgLSBUaGUgYmFzZSB0aW1lIGluIG1pbGxpc2Vjb25kc1xuICogQHBhcmFtIGF0dGVtcHQgLSBUaGUgYXR0ZW1wdCBudW1iZXJcbiAqIEByZXR1cm5zIFRoZSB0aW1lIHRvIHdhaXQgaW4gbWlsbGlzZWNvbmRzXG4gKi9cbmNvbnN0IGZ1bGxKaXR0ZXIgPSAoYmFzZTogbnVtYmVyLCBhdHRlbXB0OiBudW1iZXIpOiBudW1iZXIgPT4ge1xuICBjb25zdCBjYXAgPSBiYXNlICogTWF0aC5wb3coMiwgYXR0ZW1wdCk7XG4gIHJldHVybiBNYXRoLmZsb29yKE1hdGgucmFuZG9tKCkgKiBjYXApO1xufTtcbiJdfQ==
|
|
168
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Configuration for {@link fetchRetrier}.\n */\nexport interface RequestOptions {\n  /** Optional HTTP headers sent with each attempt. */\n  headers?: Record<string, string>;\n  /** Maximum number of attempts, including the first. */\n  retries: number;\n  /** Per-attempt timeout in milliseconds; uses an internal {@link AbortController} when exceeded. */\n  timeoutMs: number;\n  /**\n   * Base backoff in milliseconds for full jitter. The cap for attempt `n` is `baseBackoffMs * 2^n`.\n   */\n  baseBackoffMs: number;\n  /**\n   * Optional external {@link AbortSignal}. When aborted, the in-flight request is aborted; on the\n   * final attempt, cancellation surfaces as {@link FetchRetrierAbortError}.\n   */\n  signal?: AbortSignal;\n  /**\n   * Return `true` to schedule another attempt for this non-OK response.\n   * Default: retry on status 429, 500, 502, 503, or 504.\n   */\n  shouldRetry?: (response: Response, body: string) => boolean;\n}\n\n/** Error thrown when a request is cancelled by timeout or an external {@link AbortSignal}. */\nexport class FetchRetrierAbortError extends Error {\n  override readonly name: string = 'FetchRetrierAbortError';\n  /**\n   * @param message - Human-readable reason (default: `'Aborted'`)\n   */\n  constructor(message = 'Aborted') {\n    super(message);\n    Object.setPrototypeOf(this, FetchRetrierAbortError.prototype);\n  }\n}\n\n/**\n * Error thrown when {@link RequestOptions.signal} is already aborted before an attempt starts.\n */\nexport class FetchRetrierAlreadyAbortedError extends FetchRetrierAbortError {\n  override readonly name: string = 'FetchRetrierAlreadyAbortedError';\n  /**\n   * @param message - Human-readable reason (default: `'Signal was already aborted'`)\n   */\n  constructor(message = 'Signal was already aborted') {\n    super(message);\n    Object.setPrototypeOf(this, FetchRetrierAlreadyAbortedError.prototype);\n  }\n}\n\n/** Error thrown when the server returns a non-OK HTTP status and no further retry is performed. */\nexport class FetchRetrierHttpError extends Error {\n  override readonly name: string = 'FetchRetrierHttpError';\n  /**\n   * @param message - Error description\n   * @param status - HTTP status code from the response\n   */\n  constructor(\n    message: string,\n    public readonly status: number,\n  ) {\n    super(message);\n    Object.setPrototypeOf(this, FetchRetrierHttpError.prototype);\n  }\n}\n\n/**\n * Error thrown when a fetch fails with a network-level error (e.g. DNS failure, connection refused).\n */\nexport class FetchRetrierNetworkError extends Error {\n  override readonly name: string = 'FetchRetrierNetworkError';\n  /**\n   * @param message - Human-readable reason (default: `'Network error'`)\n   * @param cause - Original error, if any\n   */\n  constructor(message = 'Network error', public readonly cause?: unknown) {\n    super(message);\n    Object.setPrototypeOf(this, FetchRetrierNetworkError.prototype);\n  }\n}\n\n/** Error thrown when an internal invariant fails (should not happen in normal use). */\nexport class FetchRetrierUnreachableError extends Error {\n  override readonly name: string = 'FetchRetrierUnreachableError';\n  /**\n   * @param message - Human-readable reason (default: `'Unreachable'`)\n   */\n  constructor(message = 'Unreachable') {\n    super(message);\n    Object.setPrototypeOf(this, FetchRetrierUnreachableError.prototype);\n  }\n}\n\n/**\n * Default {@link RequestOptions.shouldRetry} implementation: retry on HTTP 429, 500, 502, 503, 504.\n */\nconst defaultShouldRetry = (res: Response): boolean => {\n  return [429, 500, 502, 503, 504].includes(res.status);\n};\n\n/**\n * Performs `fetch` with retries, a per-attempt timeout, exponential backoff with full jitter,\n * optional {@link RequestOptions.signal} cancellation, and a configurable retry predicate for\n * non-OK responses.\n *\n * @param url - Request URL\n * @param options - Retries, backoff, timeout, optional abort signal, and optional retry predicate\n * @returns The first successful (OK) {@link Response}\n * @throws {FetchRetrierAlreadyAbortedError} If `options.signal` is already aborted before an attempt\n * @throws {FetchRetrierHttpError} On a non-OK response that is not retried or after the last attempt\n * @throws {FetchRetrierNetworkError} On a network error on the final attempt\n * @throws {FetchRetrierAbortError} On timeout or external abort on the final attempt\n * @throws {FetchRetrierUnreachableError} If the retry loop exits without returning (internal bug)\n */\nexport const fetchRetrier = async (url: string, options: RequestOptions): Promise<Response> => {\n  const { headers, retries, timeoutMs, baseBackoffMs, signal: externalSignal, shouldRetry = defaultShouldRetry } = options;\n\n  for (let attempt = 1; attempt <= retries; attempt++) {\n    if (externalSignal?.aborted) {\n      throw new FetchRetrierAlreadyAbortedError();\n    }\n\n    const controller = new AbortController();\n    const timer = setTimeout(() => controller.abort(), timeoutMs);\n\n    const onExternalAbort = (): void => {\n      clearTimeout(timer);\n      controller.abort();\n    };\n\n    if (externalSignal) {\n      externalSignal.addEventListener('abort', onExternalAbort);\n    }\n\n    try {\n      const res = await fetch(url, {\n        headers,\n        signal: controller.signal,\n      });\n\n      clearTimeout(timer);\n      externalSignal?.removeEventListener('abort', onExternalAbort);\n\n      if (res.ok) {\n        return res;\n      }\n\n      const text = await res.text();\n      const isContinue = shouldRetry(res, text);\n\n      if (isContinue) {\n        if (attempt === retries) {\n          throw new FetchRetrierHttpError(`HTTP ${res.status}`, res.status);\n        }\n        await wait(fullJitter(baseBackoffMs, attempt));\n      } else {\n        throw new FetchRetrierHttpError(`Non-retriable HTTP error: ${res.status}`, res.status);\n      }\n    } catch (err: unknown) {\n      clearTimeout(timer);\n      externalSignal?.removeEventListener('abort', onExternalAbort);\n\n      if (err instanceof Error && err.name === 'AbortError') {\n        if (attempt === retries) throw err instanceof FetchRetrierAbortError ? err : new FetchRetrierAbortError();\n        await wait(fullJitter(baseBackoffMs, attempt));\n        continue;\n      }\n\n      if (err instanceof TypeError) {\n        if (attempt === retries) throw new FetchRetrierNetworkError('Network error', err);\n        await wait(fullJitter(baseBackoffMs, attempt));\n        continue;\n      }\n\n      throw err;\n    }\n  }\n\n  throw new FetchRetrierUnreachableError();\n};\n\n/**\n * @param ms - Delay in milliseconds\n * @returns A promise that resolves after `ms`\n */\nconst wait = (ms: number): Promise<void> => {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n};\n\n/**\n * Full jitter backoff: random delay in `[0, base * 2^attempt)` ms (AWS-recommended pattern).\n *\n * @param base - Base backoff in milliseconds\n * @param attempt - 1-based attempt index (first retry uses `attempt === 1`)\n * @returns Wait duration in milliseconds before the next attempt\n */\nconst fullJitter = (base: number, attempt: number): number => {\n  const cap = base * Math.pow(2, attempt);\n  return Math.floor(Math.random() * cap);\n};\n"]}
|
package/package.json
CHANGED
|
@@ -42,7 +42,7 @@
|
|
|
42
42
|
"eslint-plugin-import": "^2.32.0",
|
|
43
43
|
"jest": "^30.3.0",
|
|
44
44
|
"jest-junit": "^16",
|
|
45
|
-
"projen": "^0.99.
|
|
45
|
+
"projen": "^0.99.21",
|
|
46
46
|
"ts-jest": "^29.4.6",
|
|
47
47
|
"ts-node": "^10.9.2",
|
|
48
48
|
"typescript": "5.9.x"
|
|
@@ -55,7 +55,7 @@
|
|
|
55
55
|
"publishConfig": {
|
|
56
56
|
"access": "public"
|
|
57
57
|
},
|
|
58
|
-
"version": "0.
|
|
58
|
+
"version": "0.2.0",
|
|
59
59
|
"jest": {
|
|
60
60
|
"coverageProvider": "v8",
|
|
61
61
|
"testMatch": [
|