mirror of
https://gitcode.com/gh_mirrors/re/react-native-pushy.git
synced 2025-10-29 04:23:10 +08:00
125 lines
3.0 KiB
TypeScript
125 lines
3.0 KiB
TypeScript
import { Platform } from 'react-native';
|
|
|
|
export function log(...args: any[]) {
|
|
console.log('react-native-update: ', ...args);
|
|
}
|
|
|
|
export function promiseAny<T>(promises: Promise<T>[]) {
|
|
return new Promise<T>((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<typeof fetch>[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);
|
|
});
|
|
};
|