@module-federation/retry-plugin 2.0.1 → 2.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 +44 -60
- package/dist/CHANGELOG.md +17 -1
- package/dist/README.md +44 -60
- package/dist/esm/index.js +282 -341
- package/dist/index.d.ts +641 -20
- package/dist/index.js +288 -371
- package/dist/package.json +8 -1
- package/package.json +10 -3
- package/dist/index.d.mts +0 -110
package/dist/index.js
CHANGED
|
@@ -1,390 +1,307 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
-
var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
|
|
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
|
-
// packages/retry-plugin/src/index.ts
|
|
22
|
-
var src_exports = {};
|
|
23
|
-
__export(src_exports, {
|
|
24
|
-
RetryPlugin: () => RetryPlugin,
|
|
25
|
-
appendRetryCountQuery: () => appendRetryCountQuery,
|
|
26
|
-
combineUrlDomainWithPathQuery: () => combineUrlDomainWithPathQuery,
|
|
27
|
-
getRetryUrl: () => getRetryUrl,
|
|
28
|
-
rewriteWithNextDomain: () => rewriteWithNextDomain
|
|
29
|
-
});
|
|
30
|
-
module.exports = __toCommonJS(src_exports);
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
|
2
|
+
let _module_federation_sdk = require("@module-federation/sdk");
|
|
31
3
|
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
4
|
+
//#region src/constant.ts
|
|
5
|
+
const defaultRetries = 3;
|
|
6
|
+
const defaultRetryDelay = 1e3;
|
|
7
|
+
const PLUGIN_IDENTIFIER = "[ Module Federation RetryPlugin ]";
|
|
8
|
+
const ERROR_ABANDONED = "The request failed and has now been abandoned";
|
|
9
|
+
const RUNTIME_008 = "RUNTIME-008";
|
|
38
10
|
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
var logger_default = logger;
|
|
11
|
+
//#endregion
|
|
12
|
+
//#region src/logger.ts
|
|
13
|
+
const logger = (0, _module_federation_sdk.createLogger)(PLUGIN_IDENTIFIER);
|
|
43
14
|
|
|
44
|
-
|
|
15
|
+
//#endregion
|
|
16
|
+
//#region src/utils.ts
|
|
45
17
|
function rewriteWithNextDomain(currentUrl, domains) {
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
u.port = "";
|
|
90
|
-
}
|
|
91
|
-
u.protocol = candidate.protocol || u.protocol;
|
|
92
|
-
return u.toString();
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
return null;
|
|
96
|
-
} catch {
|
|
97
|
-
return null;
|
|
98
|
-
}
|
|
18
|
+
if (!domains || domains.length === 0) return null;
|
|
19
|
+
try {
|
|
20
|
+
const u = new URL(currentUrl);
|
|
21
|
+
const currentHostname = u.hostname;
|
|
22
|
+
const currentPort = u.port;
|
|
23
|
+
const currentHost = `${currentHostname}${currentPort ? `:${currentPort}` : ""}`;
|
|
24
|
+
const normalized = domains.map((d) => {
|
|
25
|
+
try {
|
|
26
|
+
const du = new URL(d.startsWith("http") ? d : `https://${d}`);
|
|
27
|
+
return {
|
|
28
|
+
hostname: du.hostname,
|
|
29
|
+
port: du.port,
|
|
30
|
+
protocol: du.protocol
|
|
31
|
+
};
|
|
32
|
+
} catch {
|
|
33
|
+
return {
|
|
34
|
+
hostname: d,
|
|
35
|
+
port: "",
|
|
36
|
+
protocol: u.protocol
|
|
37
|
+
};
|
|
38
|
+
}
|
|
39
|
+
}).filter((d) => !!d.hostname);
|
|
40
|
+
if (normalized.length === 0) return null;
|
|
41
|
+
let idx = -1;
|
|
42
|
+
for (let i = normalized.length - 1; i >= 0; i--) if (`${normalized[i].hostname}${normalized[i].port ? `:${normalized[i].port}` : ""}` === currentHost) {
|
|
43
|
+
idx = i;
|
|
44
|
+
break;
|
|
45
|
+
}
|
|
46
|
+
const total = normalized.length;
|
|
47
|
+
for (let step = 1; step <= total; step++) {
|
|
48
|
+
const candidate = normalized[((idx >= 0 ? idx : -1) + step) % total];
|
|
49
|
+
if (`${candidate.hostname}${candidate.port ? `:${candidate.port}` : ""}` !== currentHost) {
|
|
50
|
+
u.hostname = candidate.hostname;
|
|
51
|
+
if (candidate.port !== void 0 && candidate.port !== null && candidate.port !== "") u.port = candidate.port;
|
|
52
|
+
else u.port = "";
|
|
53
|
+
u.protocol = candidate.protocol || u.protocol;
|
|
54
|
+
return u.toString();
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
return null;
|
|
58
|
+
} catch {
|
|
59
|
+
return null;
|
|
60
|
+
}
|
|
99
61
|
}
|
|
100
|
-
__name(rewriteWithNextDomain, "rewriteWithNextDomain");
|
|
101
62
|
function appendRetryCountQuery(url, retryIndex, key = "retryCount") {
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
63
|
+
try {
|
|
64
|
+
const u = new URL(url);
|
|
65
|
+
u.searchParams.delete(key);
|
|
66
|
+
u.searchParams.set(key, String(retryIndex));
|
|
67
|
+
return u.toString();
|
|
68
|
+
} catch {
|
|
69
|
+
return url;
|
|
70
|
+
}
|
|
110
71
|
}
|
|
111
|
-
__name(appendRetryCountQuery, "appendRetryCountQuery");
|
|
112
72
|
function getRetryUrl(baseUrl, opts = {}) {
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
nextUrl = appendRetryCountQuery(nextUrl, retryIndex, queryKey);
|
|
143
|
-
}
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
return nextUrl;
|
|
73
|
+
const { domains, addQuery, retryIndex = 0, queryKey = "retryCount" } = opts;
|
|
74
|
+
let cleanBaseUrl = baseUrl;
|
|
75
|
+
try {
|
|
76
|
+
const urlObj = new URL(baseUrl);
|
|
77
|
+
urlObj.searchParams.delete(queryKey);
|
|
78
|
+
cleanBaseUrl = urlObj.toString();
|
|
79
|
+
} catch {}
|
|
80
|
+
let nextUrl = rewriteWithNextDomain(cleanBaseUrl, domains) ?? cleanBaseUrl;
|
|
81
|
+
if (retryIndex > 0 && addQuery) try {
|
|
82
|
+
const u = new URL(nextUrl);
|
|
83
|
+
const originalUrl = new URL(baseUrl);
|
|
84
|
+
originalUrl.searchParams.delete(queryKey);
|
|
85
|
+
const originalQuery = originalUrl.search.startsWith("?") ? originalUrl.search.slice(1) : originalUrl.search;
|
|
86
|
+
if (typeof addQuery === "function") {
|
|
87
|
+
const newQuery = addQuery({
|
|
88
|
+
times: retryIndex,
|
|
89
|
+
originalQuery
|
|
90
|
+
});
|
|
91
|
+
u.search = newQuery ? `?${newQuery.replace(/^\?/, "")}` : "";
|
|
92
|
+
nextUrl = u.toString();
|
|
93
|
+
} else if (addQuery === true) {
|
|
94
|
+
u.searchParams.delete(queryKey);
|
|
95
|
+
u.searchParams.set(queryKey, String(retryIndex));
|
|
96
|
+
nextUrl = u.toString();
|
|
97
|
+
}
|
|
98
|
+
} catch {
|
|
99
|
+
if (addQuery === true) nextUrl = appendRetryCountQuery(nextUrl, retryIndex, queryKey);
|
|
100
|
+
}
|
|
101
|
+
return nextUrl;
|
|
147
102
|
}
|
|
148
|
-
|
|
103
|
+
/**
|
|
104
|
+
* Extract domain/host info from a URL and combine it with path/query from another URL
|
|
105
|
+
* This is useful for domain rotation while preserving original path and query parameters
|
|
106
|
+
* @param domainUrl - URL containing the target domain/host
|
|
107
|
+
* @param pathQueryUrl - URL containing the target path and query parameters
|
|
108
|
+
* @returns Combined URL with domain from domainUrl and path/query from pathQueryUrl
|
|
109
|
+
*/
|
|
149
110
|
function combineUrlDomainWithPathQuery(domainUrl, pathQueryUrl) {
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
111
|
+
try {
|
|
112
|
+
const domainUrlObj = new URL(domainUrl);
|
|
113
|
+
const pathQueryUrlObj = new URL(pathQueryUrl);
|
|
114
|
+
domainUrlObj.pathname = pathQueryUrlObj.pathname;
|
|
115
|
+
domainUrlObj.search = pathQueryUrlObj.search;
|
|
116
|
+
return domainUrlObj.toString();
|
|
117
|
+
} catch {
|
|
118
|
+
return pathQueryUrl;
|
|
119
|
+
}
|
|
159
120
|
}
|
|
160
|
-
__name(combineUrlDomainWithPathQuery, "combineUrlDomainWithPathQuery");
|
|
161
121
|
|
|
162
|
-
|
|
122
|
+
//#endregion
|
|
123
|
+
//#region src/fetch-retry.ts
|
|
163
124
|
function autoParseResponse(url, response) {
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
return response.json();
|
|
172
|
-
}
|
|
125
|
+
try {
|
|
126
|
+
const parsed = new URL(url);
|
|
127
|
+
if (parsed.pathname.endsWith(".js") || parsed.pathname.endsWith(".cjs") || parsed.pathname.endsWith(".mjs")) return response.text();
|
|
128
|
+
return response.json();
|
|
129
|
+
} catch (error) {
|
|
130
|
+
return response.json();
|
|
131
|
+
}
|
|
173
132
|
}
|
|
174
|
-
__name(autoParseResponse, "autoParseResponse");
|
|
175
133
|
async function fetchRetry(params, lastRequestUrl, originalTotal) {
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
logger_default.log(`${PLUGIN_IDENTIFIER}: retry failed, no retries left for url: ${requestUrl}`);
|
|
237
|
-
}
|
|
238
|
-
throw new Error(`${RUNTIME_008}: ${PLUGIN_IDENTIFIER}: ${ERROR_ABANDONED} | url: ${requestUrl}`);
|
|
239
|
-
} else {
|
|
240
|
-
const nextIndex = total - retryTimes + 1;
|
|
241
|
-
const predictedBaseUrl = combineUrlDomainWithPathQuery(requestUrl, url);
|
|
242
|
-
const predictedNextUrl = getRetryUrl(predictedBaseUrl, {
|
|
243
|
-
domains,
|
|
244
|
-
addQuery,
|
|
245
|
-
retryIndex: nextIndex,
|
|
246
|
-
queryKey: "retryCount"
|
|
247
|
-
});
|
|
248
|
-
onRetry && onRetry({
|
|
249
|
-
times: nextIndex,
|
|
250
|
-
domains,
|
|
251
|
-
url: predictedNextUrl,
|
|
252
|
-
tagName: "fetch"
|
|
253
|
-
});
|
|
254
|
-
logger_default.log(`${PLUGIN_IDENTIFIER}: Trying again. Number of retries left: ${retryTimes - 1}`);
|
|
255
|
-
return await fetchRetry({
|
|
256
|
-
...params,
|
|
257
|
-
retryTimes: retryTimes - 1
|
|
258
|
-
}, requestUrl, total);
|
|
259
|
-
}
|
|
260
|
-
}
|
|
134
|
+
const { url, fetchOptions = {}, retryTimes = defaultRetries, retryDelay = defaultRetryDelay, domains, addQuery, onRetry, onSuccess, onError } = params;
|
|
135
|
+
if (!url) throw new Error(`${PLUGIN_IDENTIFIER}: url is required in fetchWithRetry`);
|
|
136
|
+
const total = originalTotal ?? params.retryTimes ?? defaultRetries;
|
|
137
|
+
const isFirstAttempt = !lastRequestUrl;
|
|
138
|
+
let baseUrl = url;
|
|
139
|
+
if (!isFirstAttempt && lastRequestUrl) baseUrl = combineUrlDomainWithPathQuery(lastRequestUrl, url);
|
|
140
|
+
let requestUrl = baseUrl;
|
|
141
|
+
if (!isFirstAttempt) requestUrl = getRetryUrl(baseUrl, {
|
|
142
|
+
domains,
|
|
143
|
+
addQuery,
|
|
144
|
+
retryIndex: total - retryTimes,
|
|
145
|
+
queryKey: "retryCount"
|
|
146
|
+
});
|
|
147
|
+
try {
|
|
148
|
+
if (!isFirstAttempt && retryDelay > 0) await new Promise((resolve) => setTimeout(resolve, retryDelay));
|
|
149
|
+
const response = await fetch(requestUrl, fetchOptions);
|
|
150
|
+
const responseClone = response.clone();
|
|
151
|
+
if (!response.ok) throw new Error(`${PLUGIN_IDENTIFIER}: Request failed: ${response.status} ${response.statusText || ""} | url: ${requestUrl}`);
|
|
152
|
+
await autoParseResponse(requestUrl, responseClone).catch((error) => {
|
|
153
|
+
throw new Error(`${PLUGIN_IDENTIFIER}: JSON parse failed: ${error?.message || String(error)} | url: ${requestUrl}`);
|
|
154
|
+
});
|
|
155
|
+
if (!isFirstAttempt) onSuccess && requestUrl && onSuccess({
|
|
156
|
+
domains,
|
|
157
|
+
url: requestUrl,
|
|
158
|
+
tagName: "fetch"
|
|
159
|
+
});
|
|
160
|
+
return response;
|
|
161
|
+
} catch (error) {
|
|
162
|
+
if (retryTimes <= 0) {
|
|
163
|
+
const attemptedRetries = total - retryTimes;
|
|
164
|
+
if (!isFirstAttempt && attemptedRetries > 0) {
|
|
165
|
+
onError && onError({
|
|
166
|
+
domains,
|
|
167
|
+
url: requestUrl,
|
|
168
|
+
tagName: "fetch"
|
|
169
|
+
});
|
|
170
|
+
logger.log(`${PLUGIN_IDENTIFIER}: retry failed, no retries left for url: ${requestUrl}`);
|
|
171
|
+
}
|
|
172
|
+
throw new Error(`${RUNTIME_008}: ${PLUGIN_IDENTIFIER}: ${ERROR_ABANDONED} | url: ${requestUrl}`);
|
|
173
|
+
} else {
|
|
174
|
+
const nextIndex = total - retryTimes + 1;
|
|
175
|
+
const predictedNextUrl = getRetryUrl(combineUrlDomainWithPathQuery(requestUrl, url), {
|
|
176
|
+
domains,
|
|
177
|
+
addQuery,
|
|
178
|
+
retryIndex: nextIndex,
|
|
179
|
+
queryKey: "retryCount"
|
|
180
|
+
});
|
|
181
|
+
onRetry && onRetry({
|
|
182
|
+
times: nextIndex,
|
|
183
|
+
domains,
|
|
184
|
+
url: predictedNextUrl,
|
|
185
|
+
tagName: "fetch"
|
|
186
|
+
});
|
|
187
|
+
logger.log(`${PLUGIN_IDENTIFIER}: Trying again. Number of retries left: ${retryTimes - 1}`);
|
|
188
|
+
return await fetchRetry({
|
|
189
|
+
...params,
|
|
190
|
+
retryTimes: retryTimes - 1
|
|
191
|
+
}, requestUrl, total);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
261
194
|
}
|
|
262
|
-
__name(fetchRetry, "fetchRetry");
|
|
263
195
|
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
domains,
|
|
320
|
-
url: lastRequestUrl,
|
|
321
|
-
tagName: "script"
|
|
322
|
-
});
|
|
323
|
-
throw new Error(`${PLUGIN_IDENTIFIER}: ${ERROR_ABANDONED} | url: ${lastRequestUrl || "unknown"}`);
|
|
324
|
-
}
|
|
325
|
-
}
|
|
326
|
-
}
|
|
327
|
-
return retryWrapper;
|
|
328
|
-
};
|
|
196
|
+
//#endregion
|
|
197
|
+
//#region src/script-retry.ts
|
|
198
|
+
function scriptRetry({ retryOptions, retryFn, beforeExecuteRetry = () => {} }) {
|
|
199
|
+
return async function(params) {
|
|
200
|
+
let retryWrapper;
|
|
201
|
+
let lastRequestUrl;
|
|
202
|
+
let originalUrl;
|
|
203
|
+
const { retryTimes = defaultRetries, retryDelay = defaultRetryDelay, domains, addQuery, onRetry, onSuccess, onError } = retryOptions || {};
|
|
204
|
+
let attempts = 0;
|
|
205
|
+
const maxAttempts = retryTimes;
|
|
206
|
+
while (attempts < maxAttempts) try {
|
|
207
|
+
beforeExecuteRetry();
|
|
208
|
+
if (retryDelay > 0 && attempts > 0) await new Promise((resolve) => setTimeout(resolve, retryDelay));
|
|
209
|
+
const retryIndex = attempts + 1;
|
|
210
|
+
retryWrapper = await retryFn({
|
|
211
|
+
...params,
|
|
212
|
+
getEntryUrl: (url) => {
|
|
213
|
+
if (!originalUrl) originalUrl = url;
|
|
214
|
+
let baseUrl = originalUrl;
|
|
215
|
+
if (lastRequestUrl) baseUrl = combineUrlDomainWithPathQuery(lastRequestUrl, originalUrl);
|
|
216
|
+
const next = getRetryUrl(baseUrl, {
|
|
217
|
+
domains,
|
|
218
|
+
addQuery,
|
|
219
|
+
retryIndex,
|
|
220
|
+
queryKey: "retryCount"
|
|
221
|
+
});
|
|
222
|
+
onRetry && onRetry({
|
|
223
|
+
times: retryIndex,
|
|
224
|
+
domains,
|
|
225
|
+
url: next,
|
|
226
|
+
tagName: "script"
|
|
227
|
+
});
|
|
228
|
+
lastRequestUrl = next;
|
|
229
|
+
return next;
|
|
230
|
+
}
|
|
231
|
+
});
|
|
232
|
+
onSuccess && lastRequestUrl && onSuccess({
|
|
233
|
+
domains,
|
|
234
|
+
url: lastRequestUrl,
|
|
235
|
+
tagName: "script"
|
|
236
|
+
});
|
|
237
|
+
break;
|
|
238
|
+
} catch (error) {
|
|
239
|
+
attempts++;
|
|
240
|
+
if (attempts >= maxAttempts) {
|
|
241
|
+
onError && lastRequestUrl && onError({
|
|
242
|
+
domains,
|
|
243
|
+
url: lastRequestUrl,
|
|
244
|
+
tagName: "script"
|
|
245
|
+
});
|
|
246
|
+
throw new Error(`${PLUGIN_IDENTIFIER}: ${ERROR_ABANDONED} | url: ${lastRequestUrl || "unknown"}`);
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
return retryWrapper;
|
|
250
|
+
};
|
|
329
251
|
}
|
|
330
|
-
__name(scriptRetry, "scriptRetry");
|
|
331
252
|
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
combineUrlDomainWithPathQuery,
|
|
388
|
-
getRetryUrl,
|
|
389
|
-
rewriteWithNextDomain
|
|
390
|
-
});
|
|
253
|
+
//#endregion
|
|
254
|
+
//#region src/index.ts
|
|
255
|
+
const RetryPlugin = (params) => {
|
|
256
|
+
if (params?.fetch || params?.script) logger.warn(`${PLUGIN_IDENTIFIER}: params is ${params}, fetch or script config is deprecated, please use the new config style. See docs: https://module-federation.io/plugin/plugins/retry-plugin.html`);
|
|
257
|
+
const { fetchOptions = {}, retryTimes = defaultRetries, successTimes = 0, retryDelay = defaultRetryDelay, domains = [], manifestDomains = [], addQuery, onRetry, onSuccess, onError } = params || {};
|
|
258
|
+
return {
|
|
259
|
+
name: "retry-plugin",
|
|
260
|
+
async fetch(manifestUrl, options) {
|
|
261
|
+
return fetchRetry({
|
|
262
|
+
url: manifestUrl,
|
|
263
|
+
fetchOptions: {
|
|
264
|
+
...options,
|
|
265
|
+
...fetchOptions
|
|
266
|
+
},
|
|
267
|
+
domains: manifestDomains || domains,
|
|
268
|
+
addQuery,
|
|
269
|
+
onRetry,
|
|
270
|
+
onSuccess,
|
|
271
|
+
onError,
|
|
272
|
+
retryTimes,
|
|
273
|
+
successTimes,
|
|
274
|
+
retryDelay
|
|
275
|
+
});
|
|
276
|
+
},
|
|
277
|
+
async loadEntryError({ getRemoteEntry, origin, remoteInfo, remoteEntryExports, globalLoading, uniqueKey }) {
|
|
278
|
+
const beforeExecuteRetry = () => {
|
|
279
|
+
delete globalLoading[uniqueKey];
|
|
280
|
+
};
|
|
281
|
+
return await scriptRetry({
|
|
282
|
+
retryOptions: {
|
|
283
|
+
retryTimes,
|
|
284
|
+
retryDelay,
|
|
285
|
+
domains,
|
|
286
|
+
addQuery,
|
|
287
|
+
onRetry,
|
|
288
|
+
onSuccess,
|
|
289
|
+
onError
|
|
290
|
+
},
|
|
291
|
+
retryFn: getRemoteEntry,
|
|
292
|
+
beforeExecuteRetry
|
|
293
|
+
})({
|
|
294
|
+
origin,
|
|
295
|
+
remoteInfo,
|
|
296
|
+
remoteEntryExports
|
|
297
|
+
});
|
|
298
|
+
}
|
|
299
|
+
};
|
|
300
|
+
};
|
|
301
|
+
|
|
302
|
+
//#endregion
|
|
303
|
+
exports.RetryPlugin = RetryPlugin;
|
|
304
|
+
exports.appendRetryCountQuery = appendRetryCountQuery;
|
|
305
|
+
exports.combineUrlDomainWithPathQuery = combineUrlDomainWithPathQuery;
|
|
306
|
+
exports.getRetryUrl = getRetryUrl;
|
|
307
|
+
exports.rewriteWithNextDomain = rewriteWithNextDomain;
|