@alwatr/fetch 4.1.2 → 4.1.4
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/CHANGELOG.md +10 -0
- package/dist/main.cjs +246 -2
- package/dist/main.cjs.LEGAL.txt +0 -0
- package/dist/main.cjs.map +2 -2
- package/dist/main.mjs +218 -2
- package/dist/main.mjs.LEGAL.txt +0 -0
- package/dist/main.mjs.map +2 -2
- package/package.json +13 -12
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,16 @@
|
|
|
3
3
|
All notable changes to this project will be documented in this file.
|
|
4
4
|
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
|
|
5
5
|
|
|
6
|
+
## [4.1.4](https://github.com/Alwatr/nanolib/compare/@alwatr/fetch@4.1.3...@alwatr/fetch@4.1.4) (2024-10-11)
|
|
7
|
+
|
|
8
|
+
### Miscellaneous Chores
|
|
9
|
+
|
|
10
|
+
* include LICENSE and LEGAL files to publish ([09f366f](https://github.com/Alwatr/nanolib/commit/09f366f680bfa9fb26acb2cd1ccbc68c5a9e9ad8)) by @AliMD
|
|
11
|
+
|
|
12
|
+
## [4.1.3](https://github.com/Alwatr/nanolib/compare/@alwatr/fetch@4.1.2...@alwatr/fetch@4.1.3) (2024-10-11)
|
|
13
|
+
|
|
14
|
+
**Note:** Version bump only for package @alwatr/fetch
|
|
15
|
+
|
|
6
16
|
## [4.1.2](https://github.com/Alwatr/nanolib/compare/@alwatr/fetch@4.1.1...@alwatr/fetch@4.1.2) (2024-10-10)
|
|
7
17
|
|
|
8
18
|
### Dependencies update
|
package/dist/main.cjs
CHANGED
|
@@ -1,3 +1,247 @@
|
|
|
1
|
-
/* @alwatr/fetch v4.1.
|
|
2
|
-
"use strict";
|
|
1
|
+
/* @alwatr/fetch v4.1.4 */
|
|
2
|
+
"use strict";
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
7
|
+
var __export = (target, all) => {
|
|
8
|
+
for (var name in all)
|
|
9
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
10
|
+
};
|
|
11
|
+
var __copyProps = (to, from, except, desc) => {
|
|
12
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
13
|
+
for (let key of __getOwnPropNames(from))
|
|
14
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
15
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
16
|
+
}
|
|
17
|
+
return to;
|
|
18
|
+
};
|
|
19
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
20
|
+
|
|
21
|
+
// src/main.ts
|
|
22
|
+
var main_exports = {};
|
|
23
|
+
__export(main_exports, {
|
|
24
|
+
cacheSupported: () => cacheSupported,
|
|
25
|
+
fetch: () => fetch,
|
|
26
|
+
fetchJson: () => fetchJson
|
|
27
|
+
});
|
|
28
|
+
module.exports = __toCommonJS(main_exports);
|
|
29
|
+
|
|
30
|
+
// src/core.ts
|
|
31
|
+
var import_delay = require("@alwatr/delay");
|
|
32
|
+
var import_global_scope = require("@alwatr/global-scope");
|
|
33
|
+
var import_logger = require("@alwatr/logger");
|
|
34
|
+
var import_package_tracer = require("@alwatr/package-tracer");
|
|
35
|
+
var import_parse_duration = require("@alwatr/parse-duration");
|
|
36
|
+
import_package_tracer.packageTracer.add("@alwatr/fetch", "4.1.4");
|
|
37
|
+
var logger_ = (0, import_logger.createLogger)("@alwatr/fetch");
|
|
38
|
+
var cacheStorage_;
|
|
39
|
+
var cacheSupported = Object.hasOwn(import_global_scope.globalScope, "caches");
|
|
40
|
+
var duplicateRequestStorage_ = {};
|
|
41
|
+
function processOptions_(options) {
|
|
42
|
+
options.method ?? (options.method = "GET");
|
|
43
|
+
options.window ?? (options.window = null);
|
|
44
|
+
options.timeout ?? (options.timeout = 8e3);
|
|
45
|
+
options.retry ?? (options.retry = 3);
|
|
46
|
+
options.retryDelay ?? (options.retryDelay = 1e3);
|
|
47
|
+
options.cacheStrategy ?? (options.cacheStrategy = "network_only");
|
|
48
|
+
options.removeDuplicate ?? (options.removeDuplicate = "never");
|
|
49
|
+
options.headers ?? (options.headers = {});
|
|
50
|
+
if (options.cacheStrategy !== "network_only" && cacheSupported !== true) {
|
|
51
|
+
logger_.incident?.("fetch", "fetch_cache_strategy_unsupported", {
|
|
52
|
+
cacheSupported
|
|
53
|
+
});
|
|
54
|
+
options.cacheStrategy = "network_only";
|
|
55
|
+
}
|
|
56
|
+
if (options.removeDuplicate === "auto") {
|
|
57
|
+
options.removeDuplicate = cacheSupported ? "until_load" : "always";
|
|
58
|
+
}
|
|
59
|
+
if (options.url.lastIndexOf("?") === -1 && options.queryParams != null) {
|
|
60
|
+
const queryParams = options.queryParams;
|
|
61
|
+
const queryArray = Object.keys(queryParams).map((key) => `${key}=${String(queryParams[key])}`);
|
|
62
|
+
if (queryArray.length > 0) {
|
|
63
|
+
options.url += "?" + queryArray.join("&");
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
if (options.bodyJson !== void 0) {
|
|
67
|
+
options.body = JSON.stringify(options.bodyJson);
|
|
68
|
+
options.headers["Content-Type"] = "application/json";
|
|
69
|
+
}
|
|
70
|
+
if (options.bearerToken !== void 0) {
|
|
71
|
+
options.headers.Authorization = `Bearer ${options.bearerToken}`;
|
|
72
|
+
} else if (options.alwatrAuth !== void 0) {
|
|
73
|
+
options.headers.Authorization = `Alwatr ${options.alwatrAuth.userId}:${options.alwatrAuth.userToken}`;
|
|
74
|
+
}
|
|
75
|
+
return options;
|
|
76
|
+
}
|
|
77
|
+
async function handleCacheStrategy_(options) {
|
|
78
|
+
if (options.cacheStrategy === "network_only") {
|
|
79
|
+
return handleRemoveDuplicate_(options);
|
|
80
|
+
}
|
|
81
|
+
logger_.logMethod?.("_handleCacheStrategy");
|
|
82
|
+
if (cacheStorage_ == null && options.cacheStorageName == null) {
|
|
83
|
+
cacheStorage_ = await caches.open("fetch_cache");
|
|
84
|
+
}
|
|
85
|
+
const cacheStorage = options.cacheStorageName != null ? await caches.open(options.cacheStorageName) : cacheStorage_;
|
|
86
|
+
const request = new Request(options.url, options);
|
|
87
|
+
switch (options.cacheStrategy) {
|
|
88
|
+
case "cache_first": {
|
|
89
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
90
|
+
if (cachedResponse != null) {
|
|
91
|
+
return cachedResponse;
|
|
92
|
+
}
|
|
93
|
+
const response = await handleRemoveDuplicate_(options);
|
|
94
|
+
if (response.ok) {
|
|
95
|
+
cacheStorage.put(request, response.clone());
|
|
96
|
+
}
|
|
97
|
+
return response;
|
|
98
|
+
}
|
|
99
|
+
case "cache_only": {
|
|
100
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
101
|
+
if (cachedResponse == null) {
|
|
102
|
+
logger_.accident("_handleCacheStrategy", "fetch_cache_not_found", { url: request.url });
|
|
103
|
+
throw new Error("fetch_cache_not_found");
|
|
104
|
+
}
|
|
105
|
+
return cachedResponse;
|
|
106
|
+
}
|
|
107
|
+
case "network_first": {
|
|
108
|
+
try {
|
|
109
|
+
const networkResponse = await handleRemoveDuplicate_(options);
|
|
110
|
+
if (networkResponse.ok) {
|
|
111
|
+
cacheStorage.put(request, networkResponse.clone());
|
|
112
|
+
}
|
|
113
|
+
return networkResponse;
|
|
114
|
+
} catch (err) {
|
|
115
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
116
|
+
if (cachedResponse != null) {
|
|
117
|
+
return cachedResponse;
|
|
118
|
+
}
|
|
119
|
+
throw err;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
case "update_cache": {
|
|
123
|
+
const networkResponse = await handleRemoveDuplicate_(options);
|
|
124
|
+
if (networkResponse.ok) {
|
|
125
|
+
cacheStorage.put(request, networkResponse.clone());
|
|
126
|
+
}
|
|
127
|
+
return networkResponse;
|
|
128
|
+
}
|
|
129
|
+
case "stale_while_revalidate": {
|
|
130
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
131
|
+
const fetchedResponsePromise = handleRemoveDuplicate_(options).then((networkResponse) => {
|
|
132
|
+
if (networkResponse.ok) {
|
|
133
|
+
cacheStorage.put(request, networkResponse.clone());
|
|
134
|
+
if (typeof options.revalidateCallback === "function") {
|
|
135
|
+
setTimeout(options.revalidateCallback, 0, networkResponse.clone());
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
return networkResponse;
|
|
139
|
+
});
|
|
140
|
+
return cachedResponse ?? fetchedResponsePromise;
|
|
141
|
+
}
|
|
142
|
+
default: {
|
|
143
|
+
return handleRemoveDuplicate_(options);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
async function handleRemoveDuplicate_(options) {
|
|
148
|
+
if (options.removeDuplicate === "never") return handleRetryPattern_(options);
|
|
149
|
+
logger_.logMethod?.("handleRemoveDuplicate_");
|
|
150
|
+
const cacheKey = options.method + " " + options.url;
|
|
151
|
+
duplicateRequestStorage_[cacheKey] ?? (duplicateRequestStorage_[cacheKey] = handleRetryPattern_(options));
|
|
152
|
+
try {
|
|
153
|
+
const response = await duplicateRequestStorage_[cacheKey];
|
|
154
|
+
if (duplicateRequestStorage_[cacheKey] != null) {
|
|
155
|
+
if (response.ok !== true || options.removeDuplicate === "until_load") {
|
|
156
|
+
delete duplicateRequestStorage_[cacheKey];
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
return response.clone();
|
|
160
|
+
} catch (err) {
|
|
161
|
+
delete duplicateRequestStorage_[cacheKey];
|
|
162
|
+
throw err;
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
async function handleRetryPattern_(options) {
|
|
166
|
+
if (!(options.retry > 1)) return handleTimeout_(options);
|
|
167
|
+
logger_.logMethod?.("_handleRetryPattern");
|
|
168
|
+
options.retry--;
|
|
169
|
+
const externalAbortSignal = options.signal;
|
|
170
|
+
try {
|
|
171
|
+
const response = await handleTimeout_(options);
|
|
172
|
+
if (response.status < 500) {
|
|
173
|
+
return response;
|
|
174
|
+
}
|
|
175
|
+
throw new Error("fetch_server_error");
|
|
176
|
+
} catch (err) {
|
|
177
|
+
logger_.accident("fetch", "fetch_failed_retry", err);
|
|
178
|
+
if (import_global_scope.globalScope.navigator?.onLine === false) {
|
|
179
|
+
logger_.accident("handleRetryPattern_", "offline", "Skip retry because offline");
|
|
180
|
+
throw err;
|
|
181
|
+
}
|
|
182
|
+
await import_delay.delay.by(options.retryDelay);
|
|
183
|
+
options.signal = externalAbortSignal;
|
|
184
|
+
return handleRetryPattern_(options);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
function handleTimeout_(options) {
|
|
188
|
+
if (options.timeout === 0) {
|
|
189
|
+
return import_global_scope.globalScope.fetch(options.url, options);
|
|
190
|
+
}
|
|
191
|
+
logger_.logMethod?.("handleTimeout_");
|
|
192
|
+
return new Promise((resolved, reject) => {
|
|
193
|
+
const abortController = typeof AbortController === "function" ? new AbortController() : null;
|
|
194
|
+
const externalAbortSignal = options.signal;
|
|
195
|
+
options.signal = abortController?.signal;
|
|
196
|
+
if (abortController !== null && externalAbortSignal != null) {
|
|
197
|
+
externalAbortSignal.addEventListener("abort", () => abortController.abort(), { once: true });
|
|
198
|
+
}
|
|
199
|
+
const timeoutId = setTimeout(() => {
|
|
200
|
+
reject(new Error("fetch_timeout"));
|
|
201
|
+
abortController?.abort("fetch_timeout");
|
|
202
|
+
}, (0, import_parse_duration.parseDuration)(options.timeout));
|
|
203
|
+
import_global_scope.globalScope.fetch(options.url, options).then((response) => resolved(response)).catch((reason) => reject(reason)).finally(() => {
|
|
204
|
+
delete options.signal;
|
|
205
|
+
clearTimeout(timeoutId);
|
|
206
|
+
});
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
// src/main.ts
|
|
211
|
+
async function fetchJson(options) {
|
|
212
|
+
let response;
|
|
213
|
+
let responseText;
|
|
214
|
+
let responseJson;
|
|
215
|
+
try {
|
|
216
|
+
response = await fetch(options);
|
|
217
|
+
responseText = await response.text();
|
|
218
|
+
responseJson = JSON.parse(responseText);
|
|
219
|
+
responseJson.ok = true;
|
|
220
|
+
responseJson.statusCode = response.status;
|
|
221
|
+
return responseJson;
|
|
222
|
+
} catch (error) {
|
|
223
|
+
const responseError = {
|
|
224
|
+
ok: false,
|
|
225
|
+
statusCode: response?.status ?? 500,
|
|
226
|
+
errorCode: responseJson?.errorCode ?? error.message,
|
|
227
|
+
errorMessage: responseJson?.errorMessage ?? error.message,
|
|
228
|
+
responseText,
|
|
229
|
+
meta: responseJson?.meta
|
|
230
|
+
};
|
|
231
|
+
logger_.accident("fetchJson", "fetch_json_failed", { responseError, error });
|
|
232
|
+
return responseError;
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
function fetch(options) {
|
|
236
|
+
options = processOptions_(options);
|
|
237
|
+
logger_.logMethodArgs?.("fetch", { options });
|
|
238
|
+
return handleCacheStrategy_(options);
|
|
239
|
+
}
|
|
240
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
241
|
+
0 && (module.exports = {
|
|
242
|
+
cacheSupported,
|
|
243
|
+
fetch,
|
|
244
|
+
fetchJson
|
|
245
|
+
});
|
|
246
|
+
/*! For license information please see main.cjs.LEGAL.txt */
|
|
3
247
|
//# sourceMappingURL=main.cjs.map
|
|
File without changes
|
package/dist/main.cjs.map
CHANGED
|
@@ -2,6 +2,6 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../src/main.ts", "../src/core.ts"],
|
|
4
4
|
"sourcesContent": ["import {handleCacheStrategy_, logger_, processOptions_, cacheSupported} from './core.js';\n\nimport type {FetchOptions, ResponseError, ResponseSuccess} from './type.js';\n\nexport {cacheSupported};\nexport type * from './type.js';\n\n/**\n * It's a wrapper around the browser's `fetch` function that adds retry pattern, timeout, cacheStrategy,\n * remove duplicates, etc.\n *\n * @see {@link FetchOptions}\n * @see {@link ResponseSuccess}\n * @see {@link ResponseError}\n *\n * @param options Fetch options.\n *\n * @returns A success or error response.\n *\n * @example\n * ```typescript\n * const responseJson = await fetchJson({\n * url: '/api/products',\n * queryParameters: {limit: 10},\n * timeout: 8_000,\n * retry: 3,\n * cacheStrategy: 'stale_while_revalidate',\n * cacheDuplicate: 'auto',\n * });\n * ```\n */\nexport async function fetchJson<T extends JsonObject>(options: FetchOptions): Promise<ResponseSuccess<T> | ResponseError> {\n let response;\n let responseText;\n let responseJson;\n\n try {\n response = await fetch(options);\n responseText = await response.text();\n responseJson = JSON.parse(responseText) as ResponseSuccess<T>;\n responseJson.ok = true;\n responseJson.statusCode = response.status;\n return responseJson;\n }\n catch (error) {\n const responseError: ResponseError = {\n ok: false,\n statusCode: response?.status ?? 500,\n errorCode: (responseJson?.errorCode as string) ?? (error as Error).message,\n errorMessage: (responseJson?.errorMessage as string) ?? (error as Error).message,\n responseText,\n meta: responseJson?.meta as JsonObject,\n };\n\n logger_.accident('fetchJson', 'fetch_json_failed', {responseError, error});\n return responseError;\n }\n}\n\n/**\n * It's a wrapper around the browser's `fetch` function that adds retry pattern, timeout, cacheStrategy,\n * remove duplicates, etc.\n *\n * @see {@link FetchOptions}\n *\n * @param options Fetch options.\n *\n * @returns A promise that resolves to the Response to that request, whether it is successful or not.\n *\n * @example\n * ```typescript\n * const response = await fetch({\n * url: '/api/products',\n * queryParameters: {limit: 10},\n * timeout: 8_000,\n * retry: 3,\n * cacheStrategy: 'stale_while_revalidate',\n * cacheDuplicate: 'auto',\n * });\n * ```\n */\nexport function fetch(options: FetchOptions): Promise<Response> {\n options = processOptions_(options);\n logger_.logMethodArgs?.('fetch', {options});\n return handleCacheStrategy_(options as Required<FetchOptions>);\n}\n", "import {delay} from '@alwatr/delay';\nimport {globalScope} from '@alwatr/global-scope';\nimport {createLogger} from '@alwatr/logger';\nimport {packageTracer} from '@alwatr/package-tracer';\nimport {parseDuration} from '@alwatr/parse-duration';\n\nimport type {FetchOptions} from './type.js';\n\npackageTracer.add(__package_name__, __package_version__);\n\nexport const logger_ = createLogger('@alwatr/fetch');\n\nlet cacheStorage_: Cache;\nexport const cacheSupported = Object.hasOwn(globalScope, 'caches');\n\nconst duplicateRequestStorage_: Record<string, Promise<Response>> = {};\n\n/**\n * Process fetch options and set defaults, etc.\n *\n * @param options Fetch options.\n *\n * @returns Required fetch options.\n */\nexport function processOptions_(options: FetchOptions): Required<FetchOptions> {\n options.method ??= 'GET';\n options.window ??= null;\n\n options.timeout ??= 8_000;\n options.retry ??= 3;\n options.retryDelay ??= 1_000;\n options.cacheStrategy ??= 'network_only';\n options.removeDuplicate ??= 'never';\n options.headers ??= {};\n\n if (options.cacheStrategy !== 'network_only' && cacheSupported !== true) {\n logger_.incident?.('fetch', 'fetch_cache_strategy_unsupported', {\n cacheSupported,\n });\n options.cacheStrategy = 'network_only';\n }\n\n if (options.removeDuplicate === 'auto') {\n options.removeDuplicate = cacheSupported ? 'until_load' : 'always';\n }\n\n if (options.url.lastIndexOf('?') === -1 && options.queryParams != null) {\n const queryParams = options.queryParams;\n // prettier-ignore\n const queryArray = Object\n .keys(queryParams)\n .map((key) => `${key}=${String(queryParams[key])}`);\n\n if (queryArray.length > 0) {\n options.url += '?' + queryArray.join('&');\n }\n }\n\n if (options.bodyJson !== undefined) {\n options.body = JSON.stringify(options.bodyJson);\n options.headers['Content-Type'] = 'application/json';\n }\n\n if (options.bearerToken !== undefined) {\n options.headers.Authorization = `Bearer ${options.bearerToken}`;\n }\n else if (options.alwatrAuth !== undefined) {\n options.headers.Authorization = `Alwatr ${options.alwatrAuth.userId}:${options.alwatrAuth.userToken}`;\n }\n\n return options as Required<FetchOptions>;\n}\n\n/**\n * Handle Cache Strategy over `handleRemoveDuplicate_`.\n */\nexport async function handleCacheStrategy_(options: Required<FetchOptions>): Promise<Response> {\n if (options.cacheStrategy === 'network_only') {\n return handleRemoveDuplicate_(options);\n }\n // else handle cache strategies!\n logger_.logMethod?.('_handleCacheStrategy');\n\n if (cacheStorage_ == null && options.cacheStorageName == null) {\n cacheStorage_ = await caches.open('fetch_cache');\n }\n\n const cacheStorage = options.cacheStorageName != null ? await caches.open(options.cacheStorageName) : cacheStorage_;\n\n const request = new Request(options.url, options);\n\n switch (options.cacheStrategy) {\n case 'cache_first': {\n const cachedResponse = await cacheStorage.match(request);\n if (cachedResponse != null) {\n return cachedResponse;\n }\n // else\n const response = await handleRemoveDuplicate_(options);\n if (response.ok) {\n cacheStorage.put(request, response.clone());\n }\n return response;\n }\n\n case 'cache_only': {\n const cachedResponse = await cacheStorage.match(request);\n if (cachedResponse == null) {\n logger_.accident('_handleCacheStrategy', 'fetch_cache_not_found', {url: request.url});\n throw new Error('fetch_cache_not_found');\n }\n // else\n return cachedResponse;\n }\n\n case 'network_first': {\n try {\n const networkResponse = await handleRemoveDuplicate_(options);\n if (networkResponse.ok) {\n cacheStorage.put(request, networkResponse.clone());\n }\n return networkResponse;\n }\n catch (err) {\n const cachedResponse = await cacheStorage.match(request);\n if (cachedResponse != null) {\n return cachedResponse;\n }\n // else\n throw err;\n }\n }\n\n case 'update_cache': {\n const networkResponse = await handleRemoveDuplicate_(options);\n if (networkResponse.ok) {\n cacheStorage.put(request, networkResponse.clone());\n }\n return networkResponse;\n }\n\n case 'stale_while_revalidate': {\n const cachedResponse = await cacheStorage.match(request);\n const fetchedResponsePromise = handleRemoveDuplicate_(options).then((networkResponse) => {\n if (networkResponse.ok) {\n cacheStorage.put(request, networkResponse.clone());\n if (typeof options.revalidateCallback === 'function') {\n setTimeout(options.revalidateCallback, 0, networkResponse.clone());\n }\n }\n return networkResponse;\n });\n\n return cachedResponse ?? fetchedResponsePromise;\n }\n\n default: {\n return handleRemoveDuplicate_(options);\n }\n }\n}\n\n/**\n * Handle Remove Duplicates over `_handleRetryPattern`.\n */\nexport async function handleRemoveDuplicate_(options: Required<FetchOptions>): Promise<Response> {\n if (options.removeDuplicate === 'never') return handleRetryPattern_(options);\n\n logger_.logMethod?.('handleRemoveDuplicate_');\n\n const cacheKey = options.method + ' ' + options.url;\n\n // We must cache fetch promise without await for handle other parallel requests.\n duplicateRequestStorage_[cacheKey] ??= handleRetryPattern_(options);\n\n try {\n // For all requests need to await for clone responses.\n const response = await duplicateRequestStorage_[cacheKey];\n\n if (duplicateRequestStorage_[cacheKey] != null) {\n if (response.ok !== true || options.removeDuplicate === 'until_load') {\n delete duplicateRequestStorage_[cacheKey];\n }\n }\n\n return response.clone();\n }\n catch (err) {\n // clean cache on any error.\n delete duplicateRequestStorage_[cacheKey];\n throw err;\n }\n}\n\n/**\n * Handle retry pattern over `handleTimeout_`.\n */\nexport async function handleRetryPattern_(options: Required<FetchOptions>): Promise<Response> {\n if (!(options.retry > 1)) return handleTimeout_(options);\n\n logger_.logMethod?.('_handleRetryPattern');\n options.retry--;\n\n const externalAbortSignal = options.signal;\n\n try {\n const response = await handleTimeout_(options);\n\n if (response.status < 500) {\n return response;\n }\n // else\n throw new Error('fetch_server_error');\n }\n catch (err) {\n logger_.accident('fetch', 'fetch_failed_retry', err);\n\n if (globalScope.navigator?.onLine === false) {\n logger_.accident('handleRetryPattern_', 'offline', 'Skip retry because offline');\n throw err;\n }\n\n await delay.by(options.retryDelay);\n\n options.signal = externalAbortSignal;\n return handleRetryPattern_(options);\n }\n}\n\n/**\n * It's a wrapper around the browser's `fetch` with timeout.\n */\nexport function handleTimeout_(options: FetchOptions): Promise<Response> {\n if (options.timeout === 0) {\n return globalScope.fetch(options.url, options);\n }\n // else\n logger_.logMethod?.('handleTimeout_');\n return new Promise((resolved, reject) => {\n const abortController = typeof AbortController === 'function' ? new AbortController() : null;\n const externalAbortSignal = options.signal;\n options.signal = abortController?.signal;\n\n if (abortController !== null && externalAbortSignal != null) {\n // Respect external abort signal\n externalAbortSignal.addEventListener('abort', () => abortController.abort(), {once: true});\n }\n\n const timeoutId = setTimeout(() => {\n reject(new Error('fetch_timeout'));\n abortController?.abort('fetch_timeout');\n }, parseDuration(options.timeout!));\n\n // abortController.signal.addEventListener('abort', () => {\n // logger.incident('fetch', 'fetch_abort_signal', {\n // reason: abortController.signal.reason,\n // });\n // });\n\n globalScope\n .fetch(options.url, options)\n .then((response) => resolved(response))\n .catch((reason) => reject(reason))\n .finally(() => {\n delete options.signal; // try to avoid memory leak in nodejs!\n clearTimeout(timeoutId);\n });\n });\n}\n"],
|
|
5
|
-
"mappings": ";
|
|
6
|
-
"names": [
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
7
|
}
|
package/dist/main.mjs
CHANGED
|
@@ -1,3 +1,219 @@
|
|
|
1
|
-
/* @alwatr/fetch v4.1.
|
|
2
|
-
|
|
1
|
+
/* @alwatr/fetch v4.1.4 */
|
|
2
|
+
|
|
3
|
+
// src/core.ts
|
|
4
|
+
import { delay } from "@alwatr/delay";
|
|
5
|
+
import { globalScope } from "@alwatr/global-scope";
|
|
6
|
+
import { createLogger } from "@alwatr/logger";
|
|
7
|
+
import { packageTracer } from "@alwatr/package-tracer";
|
|
8
|
+
import { parseDuration } from "@alwatr/parse-duration";
|
|
9
|
+
packageTracer.add("@alwatr/fetch", "4.1.4");
|
|
10
|
+
var logger_ = createLogger("@alwatr/fetch");
|
|
11
|
+
var cacheStorage_;
|
|
12
|
+
var cacheSupported = Object.hasOwn(globalScope, "caches");
|
|
13
|
+
var duplicateRequestStorage_ = {};
|
|
14
|
+
function processOptions_(options) {
|
|
15
|
+
options.method ?? (options.method = "GET");
|
|
16
|
+
options.window ?? (options.window = null);
|
|
17
|
+
options.timeout ?? (options.timeout = 8e3);
|
|
18
|
+
options.retry ?? (options.retry = 3);
|
|
19
|
+
options.retryDelay ?? (options.retryDelay = 1e3);
|
|
20
|
+
options.cacheStrategy ?? (options.cacheStrategy = "network_only");
|
|
21
|
+
options.removeDuplicate ?? (options.removeDuplicate = "never");
|
|
22
|
+
options.headers ?? (options.headers = {});
|
|
23
|
+
if (options.cacheStrategy !== "network_only" && cacheSupported !== true) {
|
|
24
|
+
logger_.incident?.("fetch", "fetch_cache_strategy_unsupported", {
|
|
25
|
+
cacheSupported
|
|
26
|
+
});
|
|
27
|
+
options.cacheStrategy = "network_only";
|
|
28
|
+
}
|
|
29
|
+
if (options.removeDuplicate === "auto") {
|
|
30
|
+
options.removeDuplicate = cacheSupported ? "until_load" : "always";
|
|
31
|
+
}
|
|
32
|
+
if (options.url.lastIndexOf("?") === -1 && options.queryParams != null) {
|
|
33
|
+
const queryParams = options.queryParams;
|
|
34
|
+
const queryArray = Object.keys(queryParams).map((key) => `${key}=${String(queryParams[key])}`);
|
|
35
|
+
if (queryArray.length > 0) {
|
|
36
|
+
options.url += "?" + queryArray.join("&");
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
if (options.bodyJson !== void 0) {
|
|
40
|
+
options.body = JSON.stringify(options.bodyJson);
|
|
41
|
+
options.headers["Content-Type"] = "application/json";
|
|
42
|
+
}
|
|
43
|
+
if (options.bearerToken !== void 0) {
|
|
44
|
+
options.headers.Authorization = `Bearer ${options.bearerToken}`;
|
|
45
|
+
} else if (options.alwatrAuth !== void 0) {
|
|
46
|
+
options.headers.Authorization = `Alwatr ${options.alwatrAuth.userId}:${options.alwatrAuth.userToken}`;
|
|
47
|
+
}
|
|
48
|
+
return options;
|
|
49
|
+
}
|
|
50
|
+
async function handleCacheStrategy_(options) {
|
|
51
|
+
if (options.cacheStrategy === "network_only") {
|
|
52
|
+
return handleRemoveDuplicate_(options);
|
|
53
|
+
}
|
|
54
|
+
logger_.logMethod?.("_handleCacheStrategy");
|
|
55
|
+
if (cacheStorage_ == null && options.cacheStorageName == null) {
|
|
56
|
+
cacheStorage_ = await caches.open("fetch_cache");
|
|
57
|
+
}
|
|
58
|
+
const cacheStorage = options.cacheStorageName != null ? await caches.open(options.cacheStorageName) : cacheStorage_;
|
|
59
|
+
const request = new Request(options.url, options);
|
|
60
|
+
switch (options.cacheStrategy) {
|
|
61
|
+
case "cache_first": {
|
|
62
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
63
|
+
if (cachedResponse != null) {
|
|
64
|
+
return cachedResponse;
|
|
65
|
+
}
|
|
66
|
+
const response = await handleRemoveDuplicate_(options);
|
|
67
|
+
if (response.ok) {
|
|
68
|
+
cacheStorage.put(request, response.clone());
|
|
69
|
+
}
|
|
70
|
+
return response;
|
|
71
|
+
}
|
|
72
|
+
case "cache_only": {
|
|
73
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
74
|
+
if (cachedResponse == null) {
|
|
75
|
+
logger_.accident("_handleCacheStrategy", "fetch_cache_not_found", { url: request.url });
|
|
76
|
+
throw new Error("fetch_cache_not_found");
|
|
77
|
+
}
|
|
78
|
+
return cachedResponse;
|
|
79
|
+
}
|
|
80
|
+
case "network_first": {
|
|
81
|
+
try {
|
|
82
|
+
const networkResponse = await handleRemoveDuplicate_(options);
|
|
83
|
+
if (networkResponse.ok) {
|
|
84
|
+
cacheStorage.put(request, networkResponse.clone());
|
|
85
|
+
}
|
|
86
|
+
return networkResponse;
|
|
87
|
+
} catch (err) {
|
|
88
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
89
|
+
if (cachedResponse != null) {
|
|
90
|
+
return cachedResponse;
|
|
91
|
+
}
|
|
92
|
+
throw err;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
case "update_cache": {
|
|
96
|
+
const networkResponse = await handleRemoveDuplicate_(options);
|
|
97
|
+
if (networkResponse.ok) {
|
|
98
|
+
cacheStorage.put(request, networkResponse.clone());
|
|
99
|
+
}
|
|
100
|
+
return networkResponse;
|
|
101
|
+
}
|
|
102
|
+
case "stale_while_revalidate": {
|
|
103
|
+
const cachedResponse = await cacheStorage.match(request);
|
|
104
|
+
const fetchedResponsePromise = handleRemoveDuplicate_(options).then((networkResponse) => {
|
|
105
|
+
if (networkResponse.ok) {
|
|
106
|
+
cacheStorage.put(request, networkResponse.clone());
|
|
107
|
+
if (typeof options.revalidateCallback === "function") {
|
|
108
|
+
setTimeout(options.revalidateCallback, 0, networkResponse.clone());
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
return networkResponse;
|
|
112
|
+
});
|
|
113
|
+
return cachedResponse ?? fetchedResponsePromise;
|
|
114
|
+
}
|
|
115
|
+
default: {
|
|
116
|
+
return handleRemoveDuplicate_(options);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
async function handleRemoveDuplicate_(options) {
|
|
121
|
+
if (options.removeDuplicate === "never") return handleRetryPattern_(options);
|
|
122
|
+
logger_.logMethod?.("handleRemoveDuplicate_");
|
|
123
|
+
const cacheKey = options.method + " " + options.url;
|
|
124
|
+
duplicateRequestStorage_[cacheKey] ?? (duplicateRequestStorage_[cacheKey] = handleRetryPattern_(options));
|
|
125
|
+
try {
|
|
126
|
+
const response = await duplicateRequestStorage_[cacheKey];
|
|
127
|
+
if (duplicateRequestStorage_[cacheKey] != null) {
|
|
128
|
+
if (response.ok !== true || options.removeDuplicate === "until_load") {
|
|
129
|
+
delete duplicateRequestStorage_[cacheKey];
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
return response.clone();
|
|
133
|
+
} catch (err) {
|
|
134
|
+
delete duplicateRequestStorage_[cacheKey];
|
|
135
|
+
throw err;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
async function handleRetryPattern_(options) {
|
|
139
|
+
if (!(options.retry > 1)) return handleTimeout_(options);
|
|
140
|
+
logger_.logMethod?.("_handleRetryPattern");
|
|
141
|
+
options.retry--;
|
|
142
|
+
const externalAbortSignal = options.signal;
|
|
143
|
+
try {
|
|
144
|
+
const response = await handleTimeout_(options);
|
|
145
|
+
if (response.status < 500) {
|
|
146
|
+
return response;
|
|
147
|
+
}
|
|
148
|
+
throw new Error("fetch_server_error");
|
|
149
|
+
} catch (err) {
|
|
150
|
+
logger_.accident("fetch", "fetch_failed_retry", err);
|
|
151
|
+
if (globalScope.navigator?.onLine === false) {
|
|
152
|
+
logger_.accident("handleRetryPattern_", "offline", "Skip retry because offline");
|
|
153
|
+
throw err;
|
|
154
|
+
}
|
|
155
|
+
await delay.by(options.retryDelay);
|
|
156
|
+
options.signal = externalAbortSignal;
|
|
157
|
+
return handleRetryPattern_(options);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
function handleTimeout_(options) {
|
|
161
|
+
if (options.timeout === 0) {
|
|
162
|
+
return globalScope.fetch(options.url, options);
|
|
163
|
+
}
|
|
164
|
+
logger_.logMethod?.("handleTimeout_");
|
|
165
|
+
return new Promise((resolved, reject) => {
|
|
166
|
+
const abortController = typeof AbortController === "function" ? new AbortController() : null;
|
|
167
|
+
const externalAbortSignal = options.signal;
|
|
168
|
+
options.signal = abortController?.signal;
|
|
169
|
+
if (abortController !== null && externalAbortSignal != null) {
|
|
170
|
+
externalAbortSignal.addEventListener("abort", () => abortController.abort(), { once: true });
|
|
171
|
+
}
|
|
172
|
+
const timeoutId = setTimeout(() => {
|
|
173
|
+
reject(new Error("fetch_timeout"));
|
|
174
|
+
abortController?.abort("fetch_timeout");
|
|
175
|
+
}, parseDuration(options.timeout));
|
|
176
|
+
globalScope.fetch(options.url, options).then((response) => resolved(response)).catch((reason) => reject(reason)).finally(() => {
|
|
177
|
+
delete options.signal;
|
|
178
|
+
clearTimeout(timeoutId);
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// src/main.ts
|
|
184
|
+
async function fetchJson(options) {
|
|
185
|
+
let response;
|
|
186
|
+
let responseText;
|
|
187
|
+
let responseJson;
|
|
188
|
+
try {
|
|
189
|
+
response = await fetch(options);
|
|
190
|
+
responseText = await response.text();
|
|
191
|
+
responseJson = JSON.parse(responseText);
|
|
192
|
+
responseJson.ok = true;
|
|
193
|
+
responseJson.statusCode = response.status;
|
|
194
|
+
return responseJson;
|
|
195
|
+
} catch (error) {
|
|
196
|
+
const responseError = {
|
|
197
|
+
ok: false,
|
|
198
|
+
statusCode: response?.status ?? 500,
|
|
199
|
+
errorCode: responseJson?.errorCode ?? error.message,
|
|
200
|
+
errorMessage: responseJson?.errorMessage ?? error.message,
|
|
201
|
+
responseText,
|
|
202
|
+
meta: responseJson?.meta
|
|
203
|
+
};
|
|
204
|
+
logger_.accident("fetchJson", "fetch_json_failed", { responseError, error });
|
|
205
|
+
return responseError;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
function fetch(options) {
|
|
209
|
+
options = processOptions_(options);
|
|
210
|
+
logger_.logMethodArgs?.("fetch", { options });
|
|
211
|
+
return handleCacheStrategy_(options);
|
|
212
|
+
}
|
|
213
|
+
export {
|
|
214
|
+
cacheSupported,
|
|
215
|
+
fetch,
|
|
216
|
+
fetchJson
|
|
217
|
+
};
|
|
218
|
+
/*! For license information please see main.mjs.LEGAL.txt */
|
|
3
219
|
//# sourceMappingURL=main.mjs.map
|
|
File without changes
|
package/dist/main.mjs.map
CHANGED
|
@@ -2,6 +2,6 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../src/core.ts", "../src/main.ts"],
|
|
4
4
|
"sourcesContent": ["import {delay} from '@alwatr/delay';\nimport {globalScope} from '@alwatr/global-scope';\nimport {createLogger} from '@alwatr/logger';\nimport {packageTracer} from '@alwatr/package-tracer';\nimport {parseDuration} from '@alwatr/parse-duration';\n\nimport type {FetchOptions} from './type.js';\n\npackageTracer.add(__package_name__, __package_version__);\n\nexport const logger_ = createLogger('@alwatr/fetch');\n\nlet cacheStorage_: Cache;\nexport const cacheSupported = Object.hasOwn(globalScope, 'caches');\n\nconst duplicateRequestStorage_: Record<string, Promise<Response>> = {};\n\n/**\n * Process fetch options and set defaults, etc.\n *\n * @param options Fetch options.\n *\n * @returns Required fetch options.\n */\nexport function processOptions_(options: FetchOptions): Required<FetchOptions> {\n options.method ??= 'GET';\n options.window ??= null;\n\n options.timeout ??= 8_000;\n options.retry ??= 3;\n options.retryDelay ??= 1_000;\n options.cacheStrategy ??= 'network_only';\n options.removeDuplicate ??= 'never';\n options.headers ??= {};\n\n if (options.cacheStrategy !== 'network_only' && cacheSupported !== true) {\n logger_.incident?.('fetch', 'fetch_cache_strategy_unsupported', {\n cacheSupported,\n });\n options.cacheStrategy = 'network_only';\n }\n\n if (options.removeDuplicate === 'auto') {\n options.removeDuplicate = cacheSupported ? 'until_load' : 'always';\n }\n\n if (options.url.lastIndexOf('?') === -1 && options.queryParams != null) {\n const queryParams = options.queryParams;\n // prettier-ignore\n const queryArray = Object\n .keys(queryParams)\n .map((key) => `${key}=${String(queryParams[key])}`);\n\n if (queryArray.length > 0) {\n options.url += '?' + queryArray.join('&');\n }\n }\n\n if (options.bodyJson !== undefined) {\n options.body = JSON.stringify(options.bodyJson);\n options.headers['Content-Type'] = 'application/json';\n }\n\n if (options.bearerToken !== undefined) {\n options.headers.Authorization = `Bearer ${options.bearerToken}`;\n }\n else if (options.alwatrAuth !== undefined) {\n options.headers.Authorization = `Alwatr ${options.alwatrAuth.userId}:${options.alwatrAuth.userToken}`;\n }\n\n return options as Required<FetchOptions>;\n}\n\n/**\n * Handle Cache Strategy over `handleRemoveDuplicate_`.\n */\nexport async function handleCacheStrategy_(options: Required<FetchOptions>): Promise<Response> {\n if (options.cacheStrategy === 'network_only') {\n return handleRemoveDuplicate_(options);\n }\n // else handle cache strategies!\n logger_.logMethod?.('_handleCacheStrategy');\n\n if (cacheStorage_ == null && options.cacheStorageName == null) {\n cacheStorage_ = await caches.open('fetch_cache');\n }\n\n const cacheStorage = options.cacheStorageName != null ? await caches.open(options.cacheStorageName) : cacheStorage_;\n\n const request = new Request(options.url, options);\n\n switch (options.cacheStrategy) {\n case 'cache_first': {\n const cachedResponse = await cacheStorage.match(request);\n if (cachedResponse != null) {\n return cachedResponse;\n }\n // else\n const response = await handleRemoveDuplicate_(options);\n if (response.ok) {\n cacheStorage.put(request, response.clone());\n }\n return response;\n }\n\n case 'cache_only': {\n const cachedResponse = await cacheStorage.match(request);\n if (cachedResponse == null) {\n logger_.accident('_handleCacheStrategy', 'fetch_cache_not_found', {url: request.url});\n throw new Error('fetch_cache_not_found');\n }\n // else\n return cachedResponse;\n }\n\n case 'network_first': {\n try {\n const networkResponse = await handleRemoveDuplicate_(options);\n if (networkResponse.ok) {\n cacheStorage.put(request, networkResponse.clone());\n }\n return networkResponse;\n }\n catch (err) {\n const cachedResponse = await cacheStorage.match(request);\n if (cachedResponse != null) {\n return cachedResponse;\n }\n // else\n throw err;\n }\n }\n\n case 'update_cache': {\n const networkResponse = await handleRemoveDuplicate_(options);\n if (networkResponse.ok) {\n cacheStorage.put(request, networkResponse.clone());\n }\n return networkResponse;\n }\n\n case 'stale_while_revalidate': {\n const cachedResponse = await cacheStorage.match(request);\n const fetchedResponsePromise = handleRemoveDuplicate_(options).then((networkResponse) => {\n if (networkResponse.ok) {\n cacheStorage.put(request, networkResponse.clone());\n if (typeof options.revalidateCallback === 'function') {\n setTimeout(options.revalidateCallback, 0, networkResponse.clone());\n }\n }\n return networkResponse;\n });\n\n return cachedResponse ?? fetchedResponsePromise;\n }\n\n default: {\n return handleRemoveDuplicate_(options);\n }\n }\n}\n\n/**\n * Handle Remove Duplicates over `_handleRetryPattern`.\n */\nexport async function handleRemoveDuplicate_(options: Required<FetchOptions>): Promise<Response> {\n if (options.removeDuplicate === 'never') return handleRetryPattern_(options);\n\n logger_.logMethod?.('handleRemoveDuplicate_');\n\n const cacheKey = options.method + ' ' + options.url;\n\n // We must cache fetch promise without await for handle other parallel requests.\n duplicateRequestStorage_[cacheKey] ??= handleRetryPattern_(options);\n\n try {\n // For all requests need to await for clone responses.\n const response = await duplicateRequestStorage_[cacheKey];\n\n if (duplicateRequestStorage_[cacheKey] != null) {\n if (response.ok !== true || options.removeDuplicate === 'until_load') {\n delete duplicateRequestStorage_[cacheKey];\n }\n }\n\n return response.clone();\n }\n catch (err) {\n // clean cache on any error.\n delete duplicateRequestStorage_[cacheKey];\n throw err;\n }\n}\n\n/**\n * Handle retry pattern over `handleTimeout_`.\n */\nexport async function handleRetryPattern_(options: Required<FetchOptions>): Promise<Response> {\n if (!(options.retry > 1)) return handleTimeout_(options);\n\n logger_.logMethod?.('_handleRetryPattern');\n options.retry--;\n\n const externalAbortSignal = options.signal;\n\n try {\n const response = await handleTimeout_(options);\n\n if (response.status < 500) {\n return response;\n }\n // else\n throw new Error('fetch_server_error');\n }\n catch (err) {\n logger_.accident('fetch', 'fetch_failed_retry', err);\n\n if (globalScope.navigator?.onLine === false) {\n logger_.accident('handleRetryPattern_', 'offline', 'Skip retry because offline');\n throw err;\n }\n\n await delay.by(options.retryDelay);\n\n options.signal = externalAbortSignal;\n return handleRetryPattern_(options);\n }\n}\n\n/**\n * It's a wrapper around the browser's `fetch` with timeout.\n */\nexport function handleTimeout_(options: FetchOptions): Promise<Response> {\n if (options.timeout === 0) {\n return globalScope.fetch(options.url, options);\n }\n // else\n logger_.logMethod?.('handleTimeout_');\n return new Promise((resolved, reject) => {\n const abortController = typeof AbortController === 'function' ? new AbortController() : null;\n const externalAbortSignal = options.signal;\n options.signal = abortController?.signal;\n\n if (abortController !== null && externalAbortSignal != null) {\n // Respect external abort signal\n externalAbortSignal.addEventListener('abort', () => abortController.abort(), {once: true});\n }\n\n const timeoutId = setTimeout(() => {\n reject(new Error('fetch_timeout'));\n abortController?.abort('fetch_timeout');\n }, parseDuration(options.timeout!));\n\n // abortController.signal.addEventListener('abort', () => {\n // logger.incident('fetch', 'fetch_abort_signal', {\n // reason: abortController.signal.reason,\n // });\n // });\n\n globalScope\n .fetch(options.url, options)\n .then((response) => resolved(response))\n .catch((reason) => reject(reason))\n .finally(() => {\n delete options.signal; // try to avoid memory leak in nodejs!\n clearTimeout(timeoutId);\n });\n });\n}\n", "import {handleCacheStrategy_, logger_, processOptions_, cacheSupported} from './core.js';\n\nimport type {FetchOptions, ResponseError, ResponseSuccess} from './type.js';\n\nexport {cacheSupported};\nexport type * from './type.js';\n\n/**\n * It's a wrapper around the browser's `fetch` function that adds retry pattern, timeout, cacheStrategy,\n * remove duplicates, etc.\n *\n * @see {@link FetchOptions}\n * @see {@link ResponseSuccess}\n * @see {@link ResponseError}\n *\n * @param options Fetch options.\n *\n * @returns A success or error response.\n *\n * @example\n * ```typescript\n * const responseJson = await fetchJson({\n * url: '/api/products',\n * queryParameters: {limit: 10},\n * timeout: 8_000,\n * retry: 3,\n * cacheStrategy: 'stale_while_revalidate',\n * cacheDuplicate: 'auto',\n * });\n * ```\n */\nexport async function fetchJson<T extends JsonObject>(options: FetchOptions): Promise<ResponseSuccess<T> | ResponseError> {\n let response;\n let responseText;\n let responseJson;\n\n try {\n response = await fetch(options);\n responseText = await response.text();\n responseJson = JSON.parse(responseText) as ResponseSuccess<T>;\n responseJson.ok = true;\n responseJson.statusCode = response.status;\n return responseJson;\n }\n catch (error) {\n const responseError: ResponseError = {\n ok: false,\n statusCode: response?.status ?? 500,\n errorCode: (responseJson?.errorCode as string) ?? (error as Error).message,\n errorMessage: (responseJson?.errorMessage as string) ?? (error as Error).message,\n responseText,\n meta: responseJson?.meta as JsonObject,\n };\n\n logger_.accident('fetchJson', 'fetch_json_failed', {responseError, error});\n return responseError;\n }\n}\n\n/**\n * It's a wrapper around the browser's `fetch` function that adds retry pattern, timeout, cacheStrategy,\n * remove duplicates, etc.\n *\n * @see {@link FetchOptions}\n *\n * @param options Fetch options.\n *\n * @returns A promise that resolves to the Response to that request, whether it is successful or not.\n *\n * @example\n * ```typescript\n * const response = await fetch({\n * url: '/api/products',\n * queryParameters: {limit: 10},\n * timeout: 8_000,\n * retry: 3,\n * cacheStrategy: 'stale_while_revalidate',\n * cacheDuplicate: 'auto',\n * });\n * ```\n */\nexport function fetch(options: FetchOptions): Promise<Response> {\n options = processOptions_(options);\n logger_.logMethodArgs?.('fetch', {options});\n return handleCacheStrategy_(options as Required<FetchOptions>);\n}\n"],
|
|
5
|
-
"mappings": "
|
|
6
|
-
"names": [
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
7
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@alwatr/fetch",
|
|
3
|
-
"version": "4.1.
|
|
3
|
+
"version": "4.1.4",
|
|
4
4
|
"description": "Enhanced fetch API with cache strategy, retry pattern, timeout, helper methods and enhanced types.",
|
|
5
5
|
"author": "S. Ali Mihandoost <ali.mihandoost@gmail.com>",
|
|
6
6
|
"keywords": [
|
|
@@ -38,7 +38,8 @@
|
|
|
38
38
|
},
|
|
39
39
|
"license": "AGPL-3.0-only",
|
|
40
40
|
"files": [
|
|
41
|
-
"**/*.{js,mjs,cjs,map,d.ts,html,md}",
|
|
41
|
+
"**/*.{js,mjs,cjs,map,d.ts,html,md,LEGAL.txt}",
|
|
42
|
+
"LICENSE",
|
|
42
43
|
"!demo/**/*"
|
|
43
44
|
],
|
|
44
45
|
"publishConfig": {
|
|
@@ -71,19 +72,19 @@
|
|
|
71
72
|
"clean": "rm -rfv dist *.tsbuildinfo"
|
|
72
73
|
},
|
|
73
74
|
"dependencies": {
|
|
74
|
-
"@alwatr/delay": "^1.0.
|
|
75
|
-
"@alwatr/global-scope": "^1.1.
|
|
76
|
-
"@alwatr/logger": "^4.0.
|
|
77
|
-
"@alwatr/package-tracer": "^1.0.
|
|
78
|
-
"@alwatr/parse-duration": "^1.1.
|
|
75
|
+
"@alwatr/delay": "^1.0.4",
|
|
76
|
+
"@alwatr/global-scope": "^1.1.26",
|
|
77
|
+
"@alwatr/logger": "^4.0.4",
|
|
78
|
+
"@alwatr/package-tracer": "^1.0.4",
|
|
79
|
+
"@alwatr/parse-duration": "^1.1.4"
|
|
79
80
|
},
|
|
80
81
|
"devDependencies": {
|
|
81
|
-
"@alwatr/nano-build": "^
|
|
82
|
-
"@alwatr/prettier-config": "^1.0.
|
|
83
|
-
"@alwatr/tsconfig-base": "^1.3.
|
|
84
|
-
"@alwatr/type-helper": "^2.0.
|
|
82
|
+
"@alwatr/nano-build": "^2.0.1",
|
|
83
|
+
"@alwatr/prettier-config": "^1.0.6",
|
|
84
|
+
"@alwatr/tsconfig-base": "^1.3.2",
|
|
85
|
+
"@alwatr/type-helper": "^2.0.2",
|
|
85
86
|
"jest": "^29.7.0",
|
|
86
87
|
"typescript": "^5.6.3"
|
|
87
88
|
},
|
|
88
|
-
"gitHead": "
|
|
89
|
+
"gitHead": "2a35f99b0347aacdccf3b6f28b30ca902f02a2f1"
|
|
89
90
|
}
|