fetch-assets.ts 2.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465
  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. signal.addEventListener('abort', stop);
  13. Bun.sleep(ms).then(done).catch(doReject);
  14. function done() {
  15. signal.removeEventListener('abort', stop);
  16. resolve();
  17. }
  18. function stop(this: AbortSignal) {
  19. reject(this.reason);
  20. }
  21. function doReject(reason: unknown) {
  22. signal.removeEventListener('abort', stop);
  23. reject(reason);
  24. }
  25. });
  26. export async function fetchAssets(url: string, fallbackUrls: string[] | readonly string[]) {
  27. const controller = new AbortController();
  28. const fetchMainPromise = fetchWithRetry(url, { signal: controller.signal, ...defaultRequestInit })
  29. .then(r => r.text())
  30. .then(text => {
  31. controller.abort();
  32. return text;
  33. });
  34. const createFetchFallbackPromise = async (url: string, index: number) => {
  35. // Most assets can be downloaded within 250ms. To avoid wasting bandwidth, we will wait for 500ms before downloading from the fallback URL.
  36. try {
  37. await sleepWithAbort(500 + (index + 1) * 20, controller.signal);
  38. } catch {
  39. console.log(picocolors.gray('[fetch cancelled early]'), picocolors.gray(url));
  40. throw new CustomAbortError();
  41. }
  42. if (controller.signal.aborted) {
  43. console.log(picocolors.gray('[fetch cancelled]'), picocolors.gray(url));
  44. throw new CustomAbortError();
  45. }
  46. const res = await fetchWithRetry(url, { signal: controller.signal, ...defaultRequestInit });
  47. const text = await res.text();
  48. controller.abort();
  49. return text;
  50. };
  51. return Promise.any([
  52. fetchMainPromise,
  53. ...fallbackUrls.map(createFetchFallbackPromise)
  54. ]).catch(e => {
  55. console.log(`Download Rule for [${url}] failed`);
  56. throw e;
  57. });
  58. }