import { Platform } from 'react-native'; export function log(...args: any[]) { console.log('react-native-update: ', ...args); } export function promiseAny(promises: Promise[]) { return new Promise((resolve, reject) => { let count = 0; promises.forEach(promise => { Promise.resolve(promise) .then(resolve) .catch(() => { count++; if (count === promises.length) { reject(new Error('All promises were rejected')); } }); }); }); } export const emptyObj = {}; export const noop = () => {}; class EmptyModule { constructor() { return new Proxy(this, { get() { return noop; }, }); } } export const emptyModule = new EmptyModule(); const ping = Platform.OS === 'web' ? Promise.resolve : async (url: string) => { let pingFinished = false; return Promise.race([ enhancedFetch(url, { method: 'HEAD', }) .then(({ status, statusText, url: finalUrl }) => { pingFinished = true; if (status === 200) { return finalUrl; } log('ping failed', url, status, statusText); throw new Error('Ping failed'); }) .catch(e => { pingFinished = true; log('ping error', url, e); throw e; }), new Promise((_, reject) => setTimeout(() => { reject(new Error('Ping timeout')); if (!pingFinished) { log('ping timeout', url); } }, 5000), ), ]); }; export function joinUrls(paths: string[], fileName?: string) { if (fileName) { return paths.map(path => `https://${path}/${fileName}`); } } export const testUrls = async (urls?: string[]) => { if (!urls?.length) { return null; } try { const ret = await promiseAny(urls.map(ping)); if (ret) { log('ping success, use url:', ret); return ret; } } catch {} log('all ping failed, use first url:', urls[0]); return urls[0]; }; export const assertWeb = () => { if (Platform.OS === 'web') { console.warn( 'react-native-update does not support the Web platform and will not perform any operations', ); return false; } return true; }; // export const isAndroid70AndBelow = () => { // // android 7.0 and below devices do not support letsencrypt cert // // https://letsencrypt.org/2023/07/10/cross-sign-expiration/ // return Platform.OS === 'android' && Platform.Version <= 24; // }; export const enhancedFetch = async ( url: string, params: Parameters[1], ) => { return fetch(url, params) .then(r => { if (r.ok) { return r; } throw new Error(`${r.status} ${r.statusText}`); }) .catch(e => { log('fetch error', url, e); log('trying fallback to http'); return fetch(url.replace('https', 'http'), params); }); };