1
0
Code Issues Pull Requests Packages Projects Releases Wiki Activity GitHub Gitee
react-native-pushy/lib/main.ts

384 lines
9.2 KiB
TypeScript
Raw Normal View History

2021-10-09 13:12:09 +08:00
import {
tryBackupEndpoints,
getCheckUrl,
setCustomEndpoints,
} from './endpoint';
import {
NativeEventEmitter,
NativeModules,
Platform,
PermissionsAndroid,
} from 'react-native';
2023-08-31 19:24:00 +08:00
import {
EventType,
ProgressData,
UpdateAvailableResult,
UpdateEventsListener,
} from './type';
2021-10-09 13:12:09 +08:00
export { setCustomEndpoints };
const {
version: v,
} = require('react-native/Libraries/Core/ReactNativeVersion');
const RNVersion = `${v.major}.${v.minor}.${v.patch}`;
2023-02-19 18:20:21 +08:00
const isTurboModuleEnabled = global.__turboModuleProxy != null;
2021-10-09 13:12:09 +08:00
2023-02-20 13:03:12 +08:00
export const PushyModule = isTurboModuleEnabled
? require('./NativeUpdate').default
: NativeModules.Pushy;
if (!PushyModule) {
2021-10-09 13:12:09 +08:00
throw new Error('react-native-update模块无法加载请对照安装文档检查配置。');
}
2023-02-20 13:03:12 +08:00
const PushyConstants = isTurboModuleEnabled
? PushyModule.getConstants()
: PushyModule;
2021-10-09 13:12:09 +08:00
2023-02-19 21:14:03 +08:00
export const downloadRootDir = PushyConstants.downloadRootDir;
export const packageVersion = PushyConstants.packageVersion;
export const currentVersion = PushyConstants.currentVersion;
export const isFirstTime = PushyConstants.isFirstTime;
const rolledBackVersion = PushyConstants.rolledBackVersion;
2021-10-21 09:29:37 +08:00
export const isRolledBack = typeof rolledBackVersion === 'string';
2023-02-19 21:14:03 +08:00
export const buildTime = PushyConstants.buildTime;
let blockUpdate = PushyConstants.blockUpdate;
let uuid = PushyConstants.uuid;
2021-10-09 13:12:09 +08:00
2023-09-02 22:35:45 +08:00
if (!PushyConstants.isUsingBundleUrl) {
2021-10-09 13:12:09 +08:00
throw new Error(
'react-native-update模块无法加载请对照文档检查Bundle URL的配置',
);
}
2023-08-31 19:24:00 +08:00
function setLocalHashInfo(hash: string, info: Record<string, any>) {
2023-02-20 13:03:12 +08:00
PushyModule.setLocalHashInfo(hash, JSON.stringify(info));
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
async function getLocalHashInfo(hash: string) {
2023-02-20 13:03:12 +08:00
return JSON.parse(await PushyModule.getLocalHashInfo(hash));
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
export async function getCurrentVersionInfo(): Promise<{
name?: string;
description?: string;
metaInfo?: string;
}> {
2021-10-21 09:29:37 +08:00
return currentVersion ? (await getLocalHashInfo(currentVersion)) || {} : {};
2021-10-09 13:12:09 +08:00
}
2023-02-20 13:03:12 +08:00
const eventEmitter = new NativeEventEmitter(PushyModule);
2021-10-09 13:12:09 +08:00
if (!uuid) {
uuid = require('nanoid/non-secure').nanoid();
2023-02-20 13:03:12 +08:00
PushyModule.setUuid(uuid);
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
function logger(...args: string[]) {
console.log('Pushy: ', ...args);
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
const noop = () => {};
let reporter: UpdateEventsListener = noop;
export function onEvents(customReporter: UpdateEventsListener) {
reporter = customReporter;
if (isRolledBack) {
report({
type: 'rollback',
data: {
rolledBackVersion,
},
});
}
}
function report({
type,
message = '',
data = {},
}: {
type: EventType;
message?: string;
data?: Record<string, string | number>;
}) {
logger(type + ' ' + message);
reporter({
type,
data: {
currentVersion,
2021-10-21 09:29:37 +08:00
cInfo,
packageVersion,
buildTime,
2023-08-31 19:24:00 +08:00
message,
...data,
},
});
2021-10-21 09:29:37 +08:00
}
2021-10-09 13:12:09 +08:00
logger('uuid: ' + uuid);
export const cInfo = {
pushy: require('../package.json').version,
rn: RNVersion,
os: Platform.OS + ' ' + Platform.Version,
uuid,
};
function assertRelease() {
2023-08-31 19:24:00 +08:00
// @ts-expect-error
2021-10-09 13:12:09 +08:00
if (__DEV__) {
2022-04-27 21:52:38 +08:00
throw new Error('react-native-update 只能在 RELEASE 版本中运行.');
2021-10-09 13:12:09 +08:00
}
}
let checkingThrottling = false;
2023-08-31 19:24:00 +08:00
export async function checkUpdate(APPKEY: string, isRetry?: boolean) {
2021-10-09 13:12:09 +08:00
assertRelease();
if (checkingThrottling) {
logger('repeated checking, ignored');
return;
}
checkingThrottling = true;
setTimeout(() => {
checkingThrottling = false;
}, 3000);
if (blockUpdate && blockUpdate.until > Date.now() / 1000) {
2023-08-31 19:24:00 +08:00
return report({
type: 'errorChecking',
message: `热更新已暂停,原因:${blockUpdate.reason}。请在"${new Date(
2021-10-09 13:12:09 +08:00
blockUpdate.until * 1000,
).toLocaleString()}"`,
2023-08-31 19:24:00 +08:00
});
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
report({ type: 'checking' });
2021-10-09 13:12:09 +08:00
let resp;
try {
resp = await fetch(getCheckUrl(APPKEY), {
method: 'POST',
headers: {
Accept: 'application/json',
'Content-Type': 'application/json',
},
body: JSON.stringify({
packageVersion,
hash: currentVersion,
buildTime,
cInfo,
}),
});
} catch (e) {
if (isRetry) {
2023-08-31 19:24:00 +08:00
return report({
type: 'errorChecking',
message: '无法连接更新服务器,请检查网络连接后重试',
});
2021-10-09 13:12:09 +08:00
}
2022-11-23 18:23:48 +08:00
await tryBackupEndpoints();
2021-10-09 13:12:09 +08:00
return checkUpdate(APPKEY, true);
}
const result = await resp.json();
checkOperation(result.op);
if (resp.status !== 200) {
2023-08-31 19:24:00 +08:00
return report({ type: 'errorChecking', message: result.message });
2021-10-09 13:12:09 +08:00
}
return result;
}
2023-08-31 19:24:00 +08:00
function checkOperation(
op: { type: string; reason: string; duration: number }[],
) {
2021-10-09 13:12:09 +08:00
if (!Array.isArray(op)) {
return;
}
op.forEach((action) => {
if (action.type === 'block') {
blockUpdate = {
reason: action.reason,
until: Math.round((Date.now() + action.duration) / 1000),
};
2023-02-20 13:03:12 +08:00
PushyModule.setBlockUpdate(blockUpdate);
2021-10-09 13:12:09 +08:00
}
});
}
let downloadingThrottling = false;
2023-08-31 19:24:00 +08:00
let downloadedHash: string;
export async function downloadUpdate(
options: UpdateAvailableResult,
eventListeners?: {
onDownloadProgress?: (data: ProgressData) => void;
},
) {
2021-10-09 13:12:09 +08:00
assertRelease();
if (!options.update) {
return;
}
2021-10-30 14:56:23 +08:00
if (rolledBackVersion === options.hash) {
logger(`rolledback hash ${rolledBackVersion}, ignored`);
2021-10-21 09:29:37 +08:00
return;
}
2021-10-09 13:12:09 +08:00
if (downloadedHash === options.hash) {
logger(`duplicated downloaded hash ${downloadedHash}, ignored`);
return downloadedHash;
2021-10-09 13:12:09 +08:00
}
if (downloadingThrottling) {
logger('repeated downloading, ignored');
return;
}
downloadingThrottling = true;
setTimeout(() => {
downloadingThrottling = false;
}, 3000);
let progressHandler;
if (eventListeners) {
if (eventListeners.onDownloadProgress) {
const downloadCallback = eventListeners.onDownloadProgress;
progressHandler = eventEmitter.addListener(
'RCTPushyDownloadProgress',
(progressData) => {
if (progressData.hash === options.hash) {
downloadCallback(progressData);
}
},
);
}
}
2022-04-27 21:52:38 +08:00
let succeeded = false;
2023-08-31 19:24:00 +08:00
report({ type: 'downloading' });
2021-10-09 13:12:09 +08:00
if (options.diffUrl) {
logger('downloading diff');
try {
2023-02-20 13:03:12 +08:00
await PushyModule.downloadPatchFromPpk({
2021-10-09 13:12:09 +08:00
updateUrl: options.diffUrl,
hash: options.hash,
originHash: currentVersion,
});
2022-04-27 21:52:38 +08:00
succeeded = true;
2021-10-09 13:12:09 +08:00
} catch (e) {
2021-10-24 16:19:17 +08:00
logger(`diff error: ${e.message}, try pdiff`);
}
2022-04-27 21:52:38 +08:00
}
if (!succeeded && options.pdiffUrl) {
2021-10-24 16:19:17 +08:00
logger('downloading pdiff');
try {
2023-02-20 13:03:12 +08:00
await PushyModule.downloadPatchFromPackage({
2021-10-09 13:12:09 +08:00
updateUrl: options.pdiffUrl,
hash: options.hash,
});
2022-04-27 21:52:38 +08:00
succeeded = true;
} catch (e) {
logger(`pdiff error: ${e.message}, try full patch`);
}
}
if (!succeeded && options.updateUrl) {
logger('downloading full patch');
try {
2023-02-20 13:03:12 +08:00
await PushyModule.downloadFullUpdate({
2022-04-27 21:52:38 +08:00
updateUrl: options.updateUrl,
hash: options.hash,
});
succeeded = true;
2021-10-24 16:19:17 +08:00
} catch (e) {
2022-04-27 21:52:38 +08:00
logger(`full patch error: ${e.message}`);
2021-10-09 13:12:09 +08:00
}
}
2022-04-27 21:52:38 +08:00
progressHandler && progressHandler.remove();
if (!succeeded) {
2023-08-31 19:24:00 +08:00
return report({ type: 'errorUpdate', data: { newVersion: options.hash } });
2022-04-27 21:52:38 +08:00
}
2021-10-09 13:12:09 +08:00
setLocalHashInfo(options.hash, {
name: options.name,
description: options.description,
metaInfo: options.metaInfo,
});
downloadedHash = options.hash;
return options.hash;
}
2023-08-31 19:24:00 +08:00
function assertHash(hash: string) {
2021-10-09 13:12:09 +08:00
if (!downloadedHash) {
logger(`no downloaded hash`);
return;
}
if (hash !== downloadedHash) {
logger(`use downloaded hash ${downloadedHash} first`);
return;
}
2021-10-29 13:02:49 +08:00
return true;
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
export function switchVersion(hash: string) {
2021-10-09 13:12:09 +08:00
assertRelease();
2021-10-29 13:02:49 +08:00
if (assertHash(hash)) {
logger('switchVersion: ' + hash);
2023-02-20 13:03:12 +08:00
PushyModule.reloadUpdate({ hash });
2021-10-29 13:02:49 +08:00
}
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
export function switchVersionLater(hash: string) {
2021-10-09 13:12:09 +08:00
assertRelease();
2021-10-29 13:02:49 +08:00
if (assertHash(hash)) {
logger('switchVersionLater: ' + hash);
2023-02-20 13:03:12 +08:00
PushyModule.setNeedUpdate({ hash });
2021-10-29 13:02:49 +08:00
}
2021-10-09 13:12:09 +08:00
}
let marked = false;
export function markSuccess() {
assertRelease();
if (marked) {
logger('repeated markSuccess, ignored');
return;
}
marked = true;
2023-02-20 13:03:12 +08:00
PushyModule.markSuccess();
2023-08-31 19:24:00 +08:00
report({ type: 'markSuccess' });
2021-10-09 13:12:09 +08:00
}
2023-08-31 19:24:00 +08:00
export async function downloadAndInstallApk({
url,
onDownloadProgress,
}: {
url: string;
onDownloadProgress?: (data: ProgressData) => void;
}) {
if (Platform.OS !== 'android') {
return;
}
report({ type: 'downloadingApk' });
if (Platform.Version <= 23) {
2021-10-09 13:12:09 +08:00
try {
const granted = await PermissionsAndroid.request(
PermissionsAndroid.PERMISSIONS.WRITE_EXTERNAL_STORAGE,
);
if (granted !== PermissionsAndroid.RESULTS.GRANTED) {
2023-08-31 19:24:00 +08:00
return report({ type: 'rejectStoragePermission' });
2021-10-09 13:12:09 +08:00
}
} catch (err) {
2023-08-31 19:24:00 +08:00
return report({ type: 'errorStoragePermission' });
2021-10-09 13:12:09 +08:00
}
}
let hash = Date.now().toString();
let progressHandler;
if (onDownloadProgress) {
progressHandler = eventEmitter.addListener(
'RCTPushyDownloadProgress',
2023-08-31 19:24:00 +08:00
(progressData: ProgressData) => {
2021-10-09 13:12:09 +08:00
if (progressData.hash === hash) {
onDownloadProgress(progressData);
}
},
);
}
2023-02-20 13:03:12 +08:00
await PushyModule.downloadAndInstallApk({
2021-10-09 13:12:09 +08:00
url,
target: 'update.apk',
hash,
2023-08-31 19:24:00 +08:00
}).catch(() => {
report({ type: 'errowDownloadAndInstallApk' });
2021-10-09 13:12:09 +08:00
});
progressHandler && progressHandler.remove();
}