expo-updates
Version:
Fetches and manages remotely-hosted assets and updates to your app's JS bundle.
120 lines (103 loc) • 3.72 kB
text/typescript
import { RCTDeviceEventEmitter, CodedError, UnavailabilityError } from '@unimodules/core';
import { EventEmitter, EventSubscription } from 'fbemitter';
import ExpoUpdates from './ExpoUpdates';
import {
Listener,
LocalAssets,
Manifest,
UpdateCheckResult,
UpdateEvent,
UpdateFetchResult,
} from './Updates.types';
export * from './Updates.types';
export const updateId: string | null =
ExpoUpdates.updateId && typeof ExpoUpdates.updateId === 'string'
? ExpoUpdates.updateId.toLowerCase()
: null;
export const releaseChannel: string = ExpoUpdates.releaseChannel ?? 'default';
export const localAssets: LocalAssets = ExpoUpdates.localAssets ?? {};
export const isEmergencyLaunch: boolean = ExpoUpdates.isEmergencyLaunch || false;
export const isUsingEmbeddedAssets: boolean = ExpoUpdates.isUsingEmbeddedAssets || false;
let _manifest = ExpoUpdates.manifest;
if (ExpoUpdates.manifestString) {
_manifest = JSON.parse(ExpoUpdates.manifestString);
}
export const manifest: Manifest | object = _manifest ?? {};
export async function reloadAsync(): Promise<void> {
if (!ExpoUpdates.reload) {
throw new UnavailabilityError('Updates', 'reloadAsync');
}
if (__DEV__) {
throw new CodedError(
'ERR_UPDATES_DISABLED',
'You cannot use the Updates module in development mode. To test manual updates, make a ' +
'release build with `npm run ios --configuration Release` or ' +
'`npm run android --variant Release`.'
);
}
await ExpoUpdates.reload();
}
export async function checkForUpdateAsync(): Promise<UpdateCheckResult> {
if (!ExpoUpdates.checkForUpdateAsync) {
throw new UnavailabilityError('Updates', 'checkForUpdateAsync');
}
if (__DEV__) {
throw new CodedError(
'ERR_UPDATES_DISABLED',
'You cannot check for updates in development mode. To test manual updates, make a ' +
'release build with `npm run ios --configuration Release` or ' +
'`npm run android --variant Release`.'
);
}
const result = await ExpoUpdates.checkForUpdateAsync();
if (result.manifestString) {
result.manifest = JSON.parse(result.manifestString);
delete result.manifestString;
}
return result;
}
export async function fetchUpdateAsync(): Promise<UpdateFetchResult> {
if (!ExpoUpdates.fetchUpdateAsync) {
throw new UnavailabilityError('Updates', 'fetchUpdateAsync');
}
if (__DEV__) {
throw new CodedError(
'ERR_UPDATES_DISABLED',
'You cannot fetch updates in development mode. To test manual updates, make a ' +
'release build with `npm run ios --configuration Release` or ' +
'`npm run android --variant Release`.'
);
}
const result = await ExpoUpdates.fetchUpdateAsync();
if (result.manifestString) {
result.manifest = JSON.parse(result.manifestString);
delete result.manifestString;
}
return result;
}
let _emitter: EventEmitter | null;
function _getEmitter(): EventEmitter {
if (!_emitter) {
_emitter = new EventEmitter();
RCTDeviceEventEmitter.addListener('Expo.nativeUpdatesEvent', _emitEvent);
}
return _emitter;
}
function _emitEvent(params): void {
let newParams = params;
if (typeof params === 'string') {
newParams = JSON.parse(params);
}
if (newParams.manifestString) {
newParams.manifest = JSON.parse(newParams.manifestString);
delete newParams.manifestString;
}
if (!_emitter) {
throw new Error(`EventEmitter must be initialized to use from its listener`);
}
_emitter.emit('Expo.updatesEvent', newParams);
}
export function addListener(listener: Listener<UpdateEvent>): EventSubscription {
const emitter = _getEmitter();
return emitter.addListener('Expo.updatesEvent', listener);
}