react-native-firebase-compiled
Version:
A well tested, feature rich Firebase implementation for React Native, supporting iOS & Android. Individual module support for Admob, Analytics, Auth, Crash Reporting, Cloud Firestore, Database, Dynamic Links, Functions, Messaging (FCM), Remote Config, Sto
214 lines (180 loc) • 6.26 kB
JavaScript
/**
*
* Firestore representation wrapper
*/
import { NativeModules } from 'react-native';
import { getAppEventName, SharedEventEmitter } from '../../utils/events';
import ModuleBase from '../../utils/ModuleBase';
import CollectionReference from './CollectionReference';
import DocumentReference from './DocumentReference';
import FieldPath from './FieldPath';
import FieldValue from './FieldValue';
import GeoPoint from './GeoPoint';
import Blob from './Blob';
import Path from './Path';
import WriteBatch from './WriteBatch';
import TransactionHandler from './TransactionHandler';
import Transaction from './Transaction';
import { isBoolean, isObject, isString, hop } from '../../utils';
import { getNativeModule } from '../../utils/native';
const NATIVE_EVENTS = ['firestore_transaction_event', 'firestore_document_sync_event', 'firestore_collection_sync_event'];
const LogLevels = ['debug', 'error', 'silent'];
export const MODULE_NAME = 'RNFirebaseFirestore';
export const NAMESPACE = 'firestore';
/**
* @class Firestore
*/
export default class Firestore extends ModuleBase {
constructor(app) {
super(app, {
events: NATIVE_EVENTS,
moduleName: MODULE_NAME,
hasMultiAppSupport: true,
hasCustomUrlSupport: false,
namespace: NAMESPACE
});
this._referencePath = new Path([]);
this._transactionHandler = new TransactionHandler(this);
SharedEventEmitter.addListener( // sub to internal native event - this fans out to
// public event name: onCollectionSnapshot
getAppEventName(this, 'firestore_collection_sync_event'), this._onCollectionSyncEvent.bind(this));
SharedEventEmitter.addListener( // sub to internal native event - this fans out to
// public event name: onDocumentSnapshot
getAppEventName(this, 'firestore_document_sync_event'), this._onDocumentSyncEvent.bind(this));
}
/**
* -------------
* PUBLIC API
* -------------
*/
/**
* Creates a write batch, used for performing multiple writes as a single atomic operation.
*
* @returns {WriteBatch}
*/
batch() {
return new WriteBatch(this);
}
/**
* Gets a CollectionReference instance that refers to the collection at the specified path.
*
* @param collectionPath
* @returns {CollectionReference}
*/
collection(collectionPath) {
const path = this._referencePath.child(collectionPath);
if (!path.isCollection) {
throw new Error('Argument "collectionPath" must point to a collection.');
}
return new CollectionReference(this, path);
}
disableNetwork() {
return getNativeModule(this).disableNetwork();
}
/**
* Gets a DocumentReference instance that refers to the document at the specified path.
*
* @param documentPath
* @returns {DocumentReference}
*/
doc(documentPath) {
const path = this._referencePath.child(documentPath);
if (!path.isDocument) {
throw new Error('Argument "documentPath" must point to a document.');
}
return new DocumentReference(this, path);
}
enableNetwork() {
return getNativeModule(this).enableNetwork();
}
/**
* Executes the given updateFunction and then attempts to commit the
* changes applied within the transaction. If any document read within
* the transaction has changed, Cloud Firestore retries the updateFunction.
*
* If it fails to commit after 5 attempts, the transaction fails.
*
* @param updateFunction
* @returns {void|Promise<any>}
*/
runTransaction(updateFunction) {
return this._transactionHandler._add(updateFunction);
}
settings(settings) {
if (!isObject(settings)) {
return Promise.reject(new Error('Firestore.settings failed: settings must be an object.'));
}
if (hop(settings, 'host') && !isString(settings.host)) {
return Promise.reject(new Error('Firestore.settings failed: settings.host must be a string.'));
}
if (hop(settings, 'persistence') && !isBoolean(settings.persistence)) {
return Promise.reject(new Error('Firestore.settings failed: settings.persistence must be boolean.'));
}
if (hop(settings, 'ssl') && !isBoolean(settings.ssl)) {
return Promise.reject(new Error('Firestore.settings failed: settings.ssl must be boolean.'));
}
if (hop(settings, 'timestampsInSnapshots') && !isBoolean(settings.timestampsInSnapshots)) {
return Promise.reject(new Error('Firestore.settings failed: settings.timestampsInSnapshots must be boolean.'));
}
return getNativeModule(this).settings(settings);
}
/**
* -------------
* UNSUPPORTED
* -------------
*/
enablePersistence() {
console.warn('Due to restrictions in the native SDK, persistence must be configured in firebase.firestore().settings()');
return Promise.resolve();
}
/**
* -------------
* INTERNALS
* -------------
*/
/**
* Internal collection sync listener
*
* @param event
* @private
*/
_onCollectionSyncEvent(event) {
if (event.error) {
SharedEventEmitter.emit(getAppEventName(this, `onQuerySnapshotError:${event.listenerId}`), event.error);
} else {
SharedEventEmitter.emit(getAppEventName(this, `onQuerySnapshot:${event.listenerId}`), event.querySnapshot);
}
}
/**
* Internal document sync listener
*
* @param event
* @private
*/
_onDocumentSyncEvent(event) {
if (event.error) {
SharedEventEmitter.emit(getAppEventName(this, `onDocumentSnapshotError:${event.listenerId}`), event.error);
} else {
SharedEventEmitter.emit(getAppEventName(this, `onDocumentSnapshot:${event.listenerId}`), event.documentSnapshot);
}
}
}
export const statics = {
Blob,
FieldPath,
FieldValue,
GeoPoint,
enableLogging(enabled) {
// DEPRECATED: Remove method in v4.1.0
console.warn('firebase.firestore.enableLogging is deprecated, use firebase.firestore().setLogLevel instead.');
this.setLogLevel(enabled ? 'debug' : 'silent');
},
setLogLevel(logLevel) {
if (LogLevels.indexOf(logLevel) === -1) {
throw new Error('Argument `logLevel` must be one of: `debug`, `error`, `silent`');
}
if (NativeModules[MODULE_NAME]) {
NativeModules[MODULE_NAME].setLogLevel(logLevel);
}
}
};