239 lines
9.2 KiB
JavaScript
239 lines
9.2 KiB
JavaScript
//#region ../node_modules/.pnpm/is-network-error@1.3.0/node_modules/is-network-error/index.js
|
|
const objectToString = Object.prototype.toString;
|
|
const isError = (value) => objectToString.call(value) === "[object Error]";
|
|
const errorMessages = new Set([
|
|
"network error",
|
|
"Failed to fetch",
|
|
"NetworkError when attempting to fetch resource.",
|
|
"The Internet connection appears to be offline.",
|
|
"Network request failed",
|
|
"fetch failed",
|
|
"terminated",
|
|
" A network error occurred.",
|
|
"Network connection lost"
|
|
]);
|
|
function isNetworkError(error) {
|
|
if (!(error && isError(error) && error.name === "TypeError" && typeof error.message === "string")) return false;
|
|
const { message, stack } = error;
|
|
if (message === "Load failed") return stack === void 0 || "__sentry_captured__" in error;
|
|
if (message.startsWith("error sending request for url")) return true;
|
|
return errorMessages.has(message);
|
|
}
|
|
|
|
//#endregion
|
|
//#region ../node_modules/.pnpm/p-retry@7.1.1/node_modules/p-retry/index.js
|
|
function validateRetries(retries) {
|
|
if (typeof retries === "number") {
|
|
if (retries < 0) throw new TypeError("Expected `retries` to be a non-negative number.");
|
|
if (Number.isNaN(retries)) throw new TypeError("Expected `retries` to be a valid number or Infinity, got NaN.");
|
|
} else if (retries !== void 0) throw new TypeError("Expected `retries` to be a number or Infinity.");
|
|
}
|
|
function validateNumberOption(name, value, { min = 0, allowInfinity = false } = {}) {
|
|
if (value === void 0) return;
|
|
if (typeof value !== "number" || Number.isNaN(value)) throw new TypeError(`Expected \`${name}\` to be a number${allowInfinity ? " or Infinity" : ""}.`);
|
|
if (!allowInfinity && !Number.isFinite(value)) throw new TypeError(`Expected \`${name}\` to be a finite number.`);
|
|
if (value < min) throw new TypeError(`Expected \`${name}\` to be \u2265 ${min}.`);
|
|
}
|
|
var AbortError = class extends Error {
|
|
constructor(message) {
|
|
super();
|
|
if (message instanceof Error) {
|
|
this.originalError = message;
|
|
({message} = message);
|
|
} else {
|
|
this.originalError = new Error(message);
|
|
this.originalError.stack = this.stack;
|
|
}
|
|
this.name = "AbortError";
|
|
this.message = message;
|
|
}
|
|
};
|
|
function calculateDelay(retriesConsumed, options) {
|
|
const attempt = Math.max(1, retriesConsumed + 1);
|
|
const random = options.randomize ? Math.random() + 1 : 1;
|
|
let timeout = Math.round(random * options.minTimeout * options.factor ** (attempt - 1));
|
|
timeout = Math.min(timeout, options.maxTimeout);
|
|
return timeout;
|
|
}
|
|
function calculateRemainingTime(start, max) {
|
|
if (!Number.isFinite(max)) return max;
|
|
return max - (performance.now() - start);
|
|
}
|
|
async function onAttemptFailure({ error, attemptNumber, retriesConsumed, startTime, options }) {
|
|
const normalizedError = error instanceof Error ? error : /* @__PURE__ */ new TypeError(`Non-error was thrown: "${error}". You should only throw errors.`);
|
|
if (normalizedError instanceof AbortError) throw normalizedError.originalError;
|
|
const retriesLeft = Number.isFinite(options.retries) ? Math.max(0, options.retries - retriesConsumed) : options.retries;
|
|
const maxRetryTime = options.maxRetryTime ?? Number.POSITIVE_INFINITY;
|
|
const context = Object.freeze({
|
|
error: normalizedError,
|
|
attemptNumber,
|
|
retriesLeft,
|
|
retriesConsumed
|
|
});
|
|
await options.onFailedAttempt(context);
|
|
if (calculateRemainingTime(startTime, maxRetryTime) <= 0) throw normalizedError;
|
|
const consumeRetry = await options.shouldConsumeRetry(context);
|
|
const remainingTime = calculateRemainingTime(startTime, maxRetryTime);
|
|
if (remainingTime <= 0 || retriesLeft <= 0) throw normalizedError;
|
|
if (normalizedError instanceof TypeError && !isNetworkError(normalizedError)) {
|
|
if (consumeRetry) throw normalizedError;
|
|
options.signal?.throwIfAborted();
|
|
return false;
|
|
}
|
|
if (!await options.shouldRetry(context)) throw normalizedError;
|
|
if (!consumeRetry) {
|
|
options.signal?.throwIfAborted();
|
|
return false;
|
|
}
|
|
const delayTime = calculateDelay(retriesConsumed, options);
|
|
const finalDelay = Math.min(delayTime, remainingTime);
|
|
options.signal?.throwIfAborted();
|
|
if (finalDelay > 0) await new Promise((resolve, reject) => {
|
|
const onAbort = () => {
|
|
clearTimeout(timeoutToken);
|
|
options.signal?.removeEventListener("abort", onAbort);
|
|
reject(options.signal.reason);
|
|
};
|
|
const timeoutToken = setTimeout(() => {
|
|
options.signal?.removeEventListener("abort", onAbort);
|
|
resolve();
|
|
}, finalDelay);
|
|
if (options.unref) timeoutToken.unref?.();
|
|
options.signal?.addEventListener("abort", onAbort, { once: true });
|
|
});
|
|
options.signal?.throwIfAborted();
|
|
return true;
|
|
}
|
|
async function pRetry(input, options = {}) {
|
|
options = { ...options };
|
|
validateRetries(options.retries);
|
|
if (Object.hasOwn(options, "forever")) throw new Error("The `forever` option is no longer supported. For many use-cases, you can set `retries: Infinity` instead.");
|
|
options.retries ??= 10;
|
|
options.factor ??= 2;
|
|
options.minTimeout ??= 1e3;
|
|
options.maxTimeout ??= Number.POSITIVE_INFINITY;
|
|
options.maxRetryTime ??= Number.POSITIVE_INFINITY;
|
|
options.randomize ??= false;
|
|
options.onFailedAttempt ??= () => {};
|
|
options.shouldRetry ??= () => true;
|
|
options.shouldConsumeRetry ??= () => true;
|
|
validateNumberOption("factor", options.factor, {
|
|
min: 0,
|
|
allowInfinity: false
|
|
});
|
|
validateNumberOption("minTimeout", options.minTimeout, {
|
|
min: 0,
|
|
allowInfinity: false
|
|
});
|
|
validateNumberOption("maxTimeout", options.maxTimeout, {
|
|
min: 0,
|
|
allowInfinity: true
|
|
});
|
|
validateNumberOption("maxRetryTime", options.maxRetryTime, {
|
|
min: 0,
|
|
allowInfinity: true
|
|
});
|
|
if (!(options.factor > 0)) options.factor = 1;
|
|
options.signal?.throwIfAborted();
|
|
let attemptNumber = 0;
|
|
let retriesConsumed = 0;
|
|
const startTime = performance.now();
|
|
while (Number.isFinite(options.retries) ? retriesConsumed <= options.retries : true) {
|
|
attemptNumber++;
|
|
try {
|
|
options.signal?.throwIfAborted();
|
|
const result = await input(attemptNumber);
|
|
options.signal?.throwIfAborted();
|
|
return result;
|
|
} catch (error) {
|
|
if (await onAttemptFailure({
|
|
error,
|
|
attemptNumber,
|
|
retriesConsumed,
|
|
startTime,
|
|
options
|
|
})) retriesConsumed++;
|
|
}
|
|
}
|
|
throw new Error("Retry attempts exhausted without throwing an error.");
|
|
}
|
|
|
|
//#endregion
|
|
//#region src/api.ts
|
|
const defaultRetryOptions = {
|
|
retries: 5,
|
|
factor: 2,
|
|
minTimeout: 1e3,
|
|
maxTimeout: Infinity,
|
|
randomize: false
|
|
};
|
|
const defaultOptions = { apiEndpoint: "https://npm.antfu.dev/" };
|
|
async function getLatestVersionBatch(packages, options = {}) {
|
|
const { apiEndpoint = defaultOptions.apiEndpoint, fetch: fetchApi = fetch, throw: throwError = true, retry = defaultRetryOptions } = options;
|
|
let query = [
|
|
options.force ? "force=true" : "",
|
|
options.metadata ? "metadata=true" : "",
|
|
throwError ? "" : "throw=false"
|
|
].filter(Boolean).join("&");
|
|
if (query) query = `?${query}`;
|
|
const fetchFn = () => fetchApi(new URL(packages.join("+") + query, apiEndpoint)).then((r) => r.json());
|
|
const retryOptions = typeof retry === "number" ? {
|
|
...defaultRetryOptions,
|
|
retries: retry
|
|
} : retry;
|
|
const list = toArray(await (retryOptions === false ? fetchFn() : pRetry(fetchFn, retryOptions)));
|
|
return throwError ? throwErrorObject(list) : list;
|
|
}
|
|
async function getLatestVersion(name, options = {}) {
|
|
const [data] = await getLatestVersionBatch([name], options);
|
|
return data;
|
|
}
|
|
async function getVersionsBatch(packages, options = {}) {
|
|
const { apiEndpoint = defaultOptions.apiEndpoint, fetch: fetchApi = fetch, throw: throwError = true, retry = defaultRetryOptions } = options;
|
|
let query = [
|
|
options.force ? "force=true" : "",
|
|
options.loose ? "loose=true" : "",
|
|
options.metadata ? "metadata=true" : "",
|
|
options.after ? `after=${encodeURIComponent(options.after)}` : "",
|
|
throwError ? "" : "throw=false"
|
|
].filter(Boolean).join("&");
|
|
if (query) query = `?${query}`;
|
|
const fetchFn = () => fetchApi(new URL(`/versions/${packages.join("+")}${query}`, apiEndpoint)).then((r) => r.json());
|
|
const list = toArray(await (retry === false ? fetchFn() : pRetry(fetchFn, typeof retry === "number" ? {
|
|
...defaultRetryOptions,
|
|
retries: retry
|
|
} : retry)));
|
|
return throwError ? throwErrorObject(list) : list;
|
|
}
|
|
async function getVersions(name, options = {}) {
|
|
const [data] = await getVersionsBatch([name], options);
|
|
return data;
|
|
}
|
|
function throwErrorObject(data) {
|
|
for (const item of toArray(data)) if (item && "error" in item) throw new Error(item.message || item.error);
|
|
return data;
|
|
}
|
|
function toArray(data) {
|
|
if (Array.isArray(data)) return data;
|
|
return [data];
|
|
}
|
|
|
|
//#endregion
|
|
//#region src/helpers.ts
|
|
const NPM_REGISTRY = "https://registry.npmjs.org/";
|
|
/**
|
|
* Lightweight replacement of `npm-registry-fetch` function `pickRegistry`'
|
|
*
|
|
* @param scope - scope of package, get from 'npm-package-arg'
|
|
* @param npmConfigs - npm configs, read from `.npmrc` file
|
|
* @param defaultRegistry - default registry, default to 'https://registry.npmjs.org/'
|
|
*/
|
|
function pickRegistry(scope, npmConfigs, defaultRegistry = NPM_REGISTRY) {
|
|
let registry = scope ? npmConfigs[`${scope.replace(/^@?/, "@")}:registry`] : void 0;
|
|
if (!registry && typeof npmConfigs.scope === "string") registry = npmConfigs[`${npmConfigs.scope.replace(/^@?/, "@")}:registry`];
|
|
if (!registry) registry = npmConfigs.registry || defaultRegistry;
|
|
return registry;
|
|
}
|
|
|
|
//#endregion
|
|
export { NPM_REGISTRY, defaultOptions, getLatestVersion, getLatestVersionBatch, getVersions, getVersionsBatch, pickRegistry }; |