fetch-assets.ts 1.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. import picocolors from 'picocolors';
  2. import { defaultRequestInit, fetchWithRetry } from './fetch-retry';
  3. class CustomAbortError extends Error {
  4. public readonly name = 'AbortError';
  5. public readonly digest = 'AbortError';
  6. }
  7. const sleepWithAbort = (ms: number, signal: AbortSignal) => new Promise<void>((resolve, reject) => {
  8. if (signal.aborted) {
  9. reject(signal.reason);
  10. return;
  11. }
  12. function stop(this: AbortSignal) { reject(this.reason); }
  13. signal.addEventListener('abort', stop, { once: true });
  14. Bun.sleep(ms).then(resolve).catch(reject).finally(() => signal.removeEventListener('abort', stop));
  15. });
  16. export async function fetchAssets(url: string, fallbackUrls: string[] | readonly string[]) {
  17. const controller = new AbortController();
  18. const fetchMainPromise = fetchWithRetry(url, { signal: controller.signal, ...defaultRequestInit })
  19. .then(r => r.text())
  20. .then(text => {
  21. controller.abort();
  22. return text;
  23. });
  24. const createFetchFallbackPromise = async (url: string, index: number) => {
  25. // Most assets can be downloaded within 250ms. To avoid wasting bandwidth, we will wait for 500ms before downloading from the fallback URL.
  26. try {
  27. await sleepWithAbort(500 + (index + 1) * 20, controller.signal);
  28. } catch {
  29. console.log(picocolors.gray('[fetch cancelled early]'), picocolors.gray(url));
  30. throw new CustomAbortError();
  31. }
  32. if (controller.signal.aborted) {
  33. console.log(picocolors.gray('[fetch cancelled]'), picocolors.gray(url));
  34. throw new CustomAbortError();
  35. }
  36. const res = await fetchWithRetry(url, { signal: controller.signal, ...defaultRequestInit });
  37. const text = await res.text();
  38. controller.abort();
  39. return text;
  40. };
  41. return Promise.any([
  42. fetchMainPromise,
  43. ...fallbackUrls.map(createFetchFallbackPromise)
  44. ]).catch(e => {
  45. console.log(`Download Rule for [${url}] failed`);
  46. throw e;
  47. });
  48. }