fetch-assets.ts 2.0 KB

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