@react-native-firebase/remote-config
Version:
React Native Firebase - React Native Firebase provides native integration with Remote Config, allowing you to change the appearance and/or functionality of your app without requiring an app update.
347 lines (295 loc) • 10.3 kB
JavaScript
/*
* Copyright (c) 2016-present Invertase Limited & Contributors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this library except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*/
import {
hasOwnProperty,
isNumber,
isObject,
isString,
isUndefined,
isIOS,
} from '@react-native-firebase/app/lib/common';
import Value from './RemoteConfigValue';
import {
createModuleNamespace,
FirebaseModule,
getFirebaseRoot,
} from '@react-native-firebase/app/lib/internal';
import { setReactNativeModule } from '@react-native-firebase/app/lib/internal/nativeModule';
import fallBackModule from './web/RNFBConfigModule';
import version from './version';
import { LastFetchStatus, ValueSource } from './statics';
const statics = {
LastFetchStatus,
ValueSource,
};
const namespace = 'remoteConfig';
const nativeModuleName = 'RNFBConfigModule';
class FirebaseConfigModule extends FirebaseModule {
constructor(...args) {
super(...args);
this._settings = {
// defaults to 1 minute.
fetchTimeMillis: 60000,
// defaults to 12 hours.
minimumFetchIntervalMillis: 43200000,
};
this._lastFetchTime = -1;
this._values = {};
this._configUpdateListenerCount = 0;
}
get defaultConfig() {
const updatedDefaultConfig = {};
Object.keys(this._values).forEach(key => {
// Need to make it an object with key and literal value. Not `Value` instance.
updatedDefaultConfig[key] = this._values[key].value;
});
return updatedDefaultConfig;
}
set defaultConfig(defaults) {
if (!isObject(defaults)) {
throw new Error("firebase.remoteConfig().defaultConfig: 'defaults' must be an object.");
}
// To make Firebase web v9 API compatible, we update the config first so it immediately
// updates defaults on the instance. We then pass to underlying SDK to update. We do this because
// there is no way to "await" a setter.
this._updateFromConstants(defaults);
this.setDefaults.call(this, defaults, true);
}
get settings() {
return this._settings;
}
set settings(settings) {
// To make Firebase web v9 API compatible, we update the settings first so it immediately
// updates settings on the instance. We then pass to underlying SDK to update. We do this because
// there is no way to "await" a setter. We can't delegate to `setConfigSettings()` as it is setup
// for native.
this._updateFromConstants(settings);
this.setConfigSettings.call(this, settings, true);
}
getValue(key) {
if (!isString(key)) {
throw new Error("firebase.remoteConfig().getValue(): 'key' must be a string value.");
}
if (typeof this._values === 'undefined' || !hasOwnProperty(this._values, key)) {
return new Value({
value: '',
source: 'static',
});
}
return new Value({ value: `${this._values[key].value}`, source: this._values[key].source });
}
getBoolean(key) {
return this.getValue(key).asBoolean();
}
getNumber(key) {
return this.getValue(key).asNumber();
}
getString(key) {
return this.getValue(key).asString();
}
getAll() {
const values = {};
Object.keys(this._values).forEach(key => (values[key] = this.getValue(key)));
return values;
}
get fetchTimeMillis() {
// android returns -1 if no fetch yet and iOS returns 0
return this._lastFetchTime;
}
get lastFetchStatus() {
return this._lastFetchStatus;
}
/**
* Deletes all activated, fetched and defaults configs and resets all Firebase Remote Config settings.
* @returns {Promise<null>}
*/
reset() {
if (isIOS) {
return Promise.resolve(null);
}
return this._promiseWithConstants(this.native.reset());
}
setConfigSettings(settings) {
const updatedSettings = {};
updatedSettings.fetchTimeout = this._settings.fetchTimeMillis / 1000;
updatedSettings.minimumFetchInterval = this._settings.minimumFetchIntervalMillis / 1000;
const apiCalled = arguments[1] == true ? 'settings' : 'setConfigSettings';
if (!isObject(settings)) {
throw new Error(`firebase.remoteConfig().${apiCalled}(*): settings must set an object.`);
}
if (hasOwnProperty(settings, 'minimumFetchIntervalMillis')) {
if (!isNumber(settings.minimumFetchIntervalMillis)) {
throw new Error(
`firebase.remoteConfig().${apiCalled}(): 'settings.minimumFetchIntervalMillis' must be a number type in milliseconds.`,
);
} else {
updatedSettings.minimumFetchInterval = settings.minimumFetchIntervalMillis / 1000;
}
}
if (hasOwnProperty(settings, 'fetchTimeMillis')) {
if (!isNumber(settings.fetchTimeMillis)) {
throw new Error(
`firebase.remoteConfig().${apiCalled}(): 'settings.fetchTimeMillis' must be a number type in milliseconds.`,
);
} else {
updatedSettings.fetchTimeout = settings.fetchTimeMillis / 1000;
}
}
return this._promiseWithConstants(this.native.setConfigSettings(updatedSettings));
}
/**
* Activates the Fetched RemoteConfig, so that the fetched key-values take effect.
* @returns {Promise<boolean>}
*/
activate() {
return this._promiseWithConstants(this.native.activate());
}
/**
* Fetches parameter values for your app.
* @param {number} expirationDurationSeconds
* @returns {Promise}
*/
fetch(expirationDurationSeconds) {
if (!isUndefined(expirationDurationSeconds) && !isNumber(expirationDurationSeconds)) {
throw new Error(
"firebase.remoteConfig().fetch(): 'expirationDurationSeconds' must be a number value.",
);
}
return this._promiseWithConstants(
this.native.fetch(expirationDurationSeconds !== undefined ? expirationDurationSeconds : -1),
);
}
fetchAndActivate() {
return this._promiseWithConstants(this.native.fetchAndActivate());
}
ensureInitialized() {
return this._promiseWithConstants(this.native.ensureInitialized());
}
/**
* Sets defaults.
*
* @param {object} defaults
*/
setDefaults(defaults) {
const apiCalled = arguments[1] === true ? 'defaultConfig' : 'setDefaults';
if (!isObject(defaults)) {
throw new Error(`firebase.remoteConfig().${apiCalled}(): 'defaults' must be an object.`);
}
return this._promiseWithConstants(this.native.setDefaults(defaults));
}
/**
* Sets defaults based on resource.
* @param {string} resourceName
*/
setDefaultsFromResource(resourceName) {
if (!isString(resourceName)) {
throw new Error(
"firebase.remoteConfig().setDefaultsFromResource(): 'resourceName' must be a string value.",
);
}
return this._promiseWithConstants(this.native.setDefaultsFromResource(resourceName));
}
/**
* Registers a listener to changes in the configuration.
*
* @param listenerOrObserver - function called on config change
* @returns {function} unsubscribe listener
*/
onConfigUpdated(listenerOrObserver) {
const listener = this._parseListener(listenerOrObserver);
let unsubscribed = false;
const subscription = this.emitter.addListener(
this.eventNameForApp('on_config_updated'),
event => {
const { resultType } = event;
if (resultType === 'success') {
listener({ updatedKeys: event.updatedKeys }, undefined);
return;
}
listener(undefined, {
code: event.code,
message: event.message,
nativeErrorMessage: event.nativeErrorMessage,
});
},
);
if (this._configUpdateListenerCount === 0) {
this.native.onConfigUpdated();
}
this._configUpdateListenerCount++;
return () => {
if (unsubscribed) {
// there is no harm in calling this multiple times to unsubscribe,
// but anything after the first call is a no-op
return;
} else {
unsubscribed = true;
}
subscription.remove();
this._configUpdateListenerCount--;
if (this._configUpdateListenerCount === 0) {
this.native.removeConfigUpdateRegistration();
}
};
}
_parseListener(listenerOrObserver) {
return typeof listenerOrObserver === 'object'
? listenerOrObserver.next.bind(listenerOrObserver)
: listenerOrObserver;
}
_updateFromConstants(constants) {
// Wrapped this as we update using sync getters initially for `defaultConfig` & `settings`
if (constants.lastFetchTime) {
this._lastFetchTime = constants.lastFetchTime;
}
// Wrapped this as we update using sync getters initially for `defaultConfig` & `settings`
if (constants.lastFetchStatus) {
this._lastFetchStatus = constants.lastFetchStatus;
}
this._settings = {
fetchTimeMillis: constants.fetchTimeout * 1000,
minimumFetchIntervalMillis: constants.minimumFetchInterval * 1000,
};
this._values = Object.freeze(constants.values);
}
_promiseWithConstants(promise) {
return promise.then(({ result, constants }) => {
this._updateFromConstants(constants);
return result;
});
}
}
// import { SDK_VERSION } from '@react-native-firebase/remote-config';
export const SDK_VERSION = version;
// import config from '@react-native-firebase/remote-config';
// config().X(...);
export default createModuleNamespace({
statics,
version,
namespace,
nativeModuleName,
nativeEvents: ['on_config_updated'],
hasMultiAppSupport: true,
hasCustomUrlOrRegionSupport: false,
ModuleClass: FirebaseConfigModule,
});
export * from './modular';
// import config, { firebase } from '@react-native-firebase/remote-config';
// config().X(...);
// firebase.remoteConfig().X(...);
export const firebase = getFirebaseRoot();
setReactNativeModule(nativeModuleName, fallBackModule);