/** * @flow * DocumentReference representation wrapper */ import CollectionReference from './CollectionReference'; import DocumentSnapshot from './DocumentSnapshot'; import { parseUpdateArgs } from './utils'; import { buildNativeMap } from './utils/serialize'; import { getAppEventName, SharedEventEmitter } from '../../utils/events'; import { getLogger } from '../../utils/log'; import { firestoreAutoId, isFunction, isObject } from '../../utils'; import { getNativeModule } from '../../utils/native'; import type Firestore from './'; import type { GetOptions, MetadataChanges, NativeDocumentSnapshot, SetOptions, } from './types'; import type Path from './Path'; type ObserverOnError = Object => void; type ObserverOnNext = DocumentSnapshot => void; type Observer = { error?: ObserverOnError, next: ObserverOnNext, }; /** * @class DocumentReference */ export default class DocumentReference { _documentPath: Path; _firestore: Firestore; constructor(firestore: Firestore, documentPath: Path) { this._documentPath = documentPath; this._firestore = firestore; } get firestore(): Firestore { return this._firestore; } get id(): string | null { return this._documentPath.id; } get parent(): CollectionReference { const parentPath = this._documentPath.parent(); // $FlowExpectedError: parentPath can never be null return new CollectionReference(this._firestore, parentPath); } get path(): string { return this._documentPath.relativeName; } collection(collectionPath: string): CollectionReference { const path = this._documentPath.child(collectionPath); if (!path.isCollection) { throw new Error('Argument "collectionPath" must point to a collection.'); } return new CollectionReference(this._firestore, path); } delete(): Promise { return getNativeModule(this._firestore).documentDelete(this.path); } get(options?: GetOptions): Promise { if (options) { if (!isObject(options)) { return Promise.reject( new Error( 'DocumentReference.get failed: First argument must be an object.' ) ); } else if ( options.source && (options.source !== 'default' && options.source !== 'server' && options.source !== 'cache') ) { return Promise.reject( new Error( 'DocumentReference.get failed: GetOptions.source must be one of `default`, `server` or `cache`.' ) ); } } return getNativeModule(this._firestore) .documentGet(this.path, options) .then(result => new DocumentSnapshot(this._firestore, result)); } onSnapshot( optionsOrObserverOrOnNext: MetadataChanges | Observer | ObserverOnNext, observerOrOnNextOrOnError?: Observer | ObserverOnNext | ObserverOnError, onError?: ObserverOnError ) { let observer: Observer; let docListenOptions = {}; // Called with: onNext, ?onError if (isFunction(optionsOrObserverOrOnNext)) { if (observerOrOnNextOrOnError && !isFunction(observerOrOnNextOrOnError)) { throw new Error( 'DocumentReference.onSnapshot failed: Second argument must be a valid function.' ); } // $FlowExpectedError: Not coping with the overloaded method signature observer = { next: optionsOrObserverOrOnNext, error: observerOrOnNextOrOnError, }; } else if ( optionsOrObserverOrOnNext && isObject(optionsOrObserverOrOnNext) ) { // Called with: Observer if (optionsOrObserverOrOnNext.next) { if (isFunction(optionsOrObserverOrOnNext.next)) { if ( optionsOrObserverOrOnNext.error && !isFunction(optionsOrObserverOrOnNext.error) ) { throw new Error( 'DocumentReference.onSnapshot failed: Observer.error must be a valid function.' ); } // $FlowExpectedError: Not coping with the overloaded method signature observer = { next: optionsOrObserverOrOnNext.next, error: optionsOrObserverOrOnNext.error, }; } else { throw new Error( 'DocumentReference.onSnapshot failed: Observer.next must be a valid function.' ); } } else if ( Object.prototype.hasOwnProperty.call( optionsOrObserverOrOnNext, 'includeMetadataChanges' ) ) { docListenOptions = optionsOrObserverOrOnNext; // Called with: Options, onNext, ?onError if (isFunction(observerOrOnNextOrOnError)) { if (onError && !isFunction(onError)) { throw new Error( 'DocumentReference.onSnapshot failed: Third argument must be a valid function.' ); } // $FlowExpectedError: Not coping with the overloaded method signature observer = { next: observerOrOnNextOrOnError, error: onError, }; // Called with Options, Observer } else if ( observerOrOnNextOrOnError && isObject(observerOrOnNextOrOnError) && observerOrOnNextOrOnError.next ) { if (isFunction(observerOrOnNextOrOnError.next)) { if ( observerOrOnNextOrOnError.error && !isFunction(observerOrOnNextOrOnError.error) ) { throw new Error( 'DocumentReference.onSnapshot failed: Observer.error must be a valid function.' ); } observer = { next: observerOrOnNextOrOnError.next, error: observerOrOnNextOrOnError.error, }; } else { throw new Error( 'DocumentReference.onSnapshot failed: Observer.next must be a valid function.' ); } } else { throw new Error( 'DocumentReference.onSnapshot failed: Second argument must be a function or observer.' ); } } else { throw new Error( 'DocumentReference.onSnapshot failed: First argument must be a function, observer or options.' ); } } else { throw new Error( 'DocumentReference.onSnapshot failed: Called with invalid arguments.' ); } const listenerId = firestoreAutoId(); const listener = (nativeDocumentSnapshot: NativeDocumentSnapshot) => { const documentSnapshot = new DocumentSnapshot( this.firestore, nativeDocumentSnapshot ); observer.next(documentSnapshot); }; // Listen to snapshot events SharedEventEmitter.addListener( getAppEventName(this._firestore, `onDocumentSnapshot:${listenerId}`), listener ); // Listen for snapshot error events if (observer.error) { SharedEventEmitter.addListener( getAppEventName( this._firestore, `onDocumentSnapshotError:${listenerId}` ), observer.error ); } // Add the native listener getNativeModule(this._firestore).documentOnSnapshot( this.path, listenerId, docListenOptions ); // Return an unsubscribe method return this._offDocumentSnapshot.bind(this, listenerId, listener); } set(data: Object, options?: SetOptions): Promise { const nativeData = buildNativeMap(data); return getNativeModule(this._firestore).documentSet( this.path, nativeData, options ); } update(...args: any[]): Promise { const data = parseUpdateArgs(args, 'DocumentReference.update'); const nativeData = buildNativeMap(data); return getNativeModule(this._firestore).documentUpdate( this.path, nativeData ); } /** * INTERNALS */ /** * Remove document snapshot listener * @param listener */ _offDocumentSnapshot(listenerId: string, listener: Function) { getLogger(this._firestore).info('Removing onDocumentSnapshot listener'); SharedEventEmitter.removeListener( getAppEventName(this._firestore, `onDocumentSnapshot:${listenerId}`), listener ); SharedEventEmitter.removeListener( getAppEventName(this._firestore, `onDocumentSnapshotError:${listenerId}`), listener ); getNativeModule(this._firestore).documentOffSnapshot(this.path, listenerId); } }