// @flow import User from './user'; import ModuleBase from './../../utils/ModuleBase'; import INTERNALS from './../../internals'; import ConfirmationResult from './ConfirmationResult'; // providers import EmailAuthProvider from './providers/EmailAuthProvider'; import GoogleAuthProvider from './providers/GoogleAuthProvider'; import GithubAuthProvider from './providers/GithubAuthProvider'; import TwitterAuthProvider from './providers/TwitterAuthProvider'; import FacebookAuthProvider from './providers/FacebookAuthProvider'; import PhoneAuthProvider from './providers/PhoneAuthProvider'; export default class Auth extends ModuleBase { static _NAMESPACE = 'auth'; static _NATIVE_MODULE = 'RNFirebaseAuth'; _user: User | null; _authResult: AuthResultType | null; authenticated: boolean; constructor(firebaseApp: Object, options: Object = {}) { super(firebaseApp, options, true); this._user = null; this._authResult = null; this.authenticated = false; this.addListener( // sub to internal native event - this fans out to // public event name: onAuthStateChanged this._getAppEventName('auth_state_changed'), this._onAuthStateChanged.bind(this), ); this.addListener( // sub to internal native event - this fans out to // public event name: onIdTokenChanged this._getAppEventName('auth_id_token_changed'), this._onIdTokenChanged.bind(this), ); this._native.addAuthStateListener(); this._native.addIdTokenListener(); } /** * Internal auth changed listener * @param auth * @param emit * @private */ _onAuthStateChanged(auth: AuthResultType, emit: boolean = true) { this._authResult = auth; this.authenticated = auth ? auth.authenticated || false : false; if (auth && auth.user && !this._user) this._user = new User(this, auth); else if ((!auth || !auth.user) && this._user) this._user = null; else if (this._user) this._user._updateValues(auth); if (emit) this.emit(this._getAppEventName('onAuthStateChanged'), this._user); return auth ? this._user : null; } /** * Remove auth change listener * @param listener */ _offAuthStateChanged(listener: Function) { this.log.info('Removing onAuthStateChanged listener'); this.removeListener(this._getAppEventName('onAuthStateChanged'), listener); } /** * Internal auth changed listener * @param auth * @param emit * @private */ _onIdTokenChanged(auth: AuthResultType, emit: boolean = true) { this._authResult = auth; this.authenticated = auth ? auth.authenticated || false : false; if (auth && auth.user && !this._user) this._user = new User(this, auth); else if ((!auth || !auth.user) && this._user) this._user = null; else if (this._user) this._user._updateValues(auth); if (emit) this.emit(this._getAppEventName('onIdTokenChanged'), this._user); return auth ? this._user : null; } /** * Remove id token change listener * @param listener */ _offIdTokenChanged(listener: Function) { this.log.info('Removing onIdTokenChanged listener'); this.removeListener(this._getAppEventName('onIdTokenChanged'), listener); } /** * Intercept all user actions and send their results to * auth state change before resolving * @param promise * @returns {Promise.|*} * @private */ _interceptUserValue(promise) { return promise.then((result) => { if (!result) return this._onAuthStateChanged(null, false); if (result.user) return this._onAuthStateChanged(result, false); if (result.uid) return this._onAuthStateChanged({ authenticated: true, user: result }, false); return result; }); } /* * WEB API */ /** * Listen for auth changes. * @param listener */ onAuthStateChanged(listener: Function) { this.log.info('Creating onAuthStateChanged listener'); this.on(this._getAppEventName('onAuthStateChanged'), listener); if (this._authResult) listener(this._user || null); return this._offAuthStateChanged.bind(this, listener); } /** * Listen for id token changes. * @param listener */ onIdTokenChanged(listener: Function) { this.log.info('Creating onIdTokenChanged listener'); this.on(this._getAppEventName('onIdTokenChanged'), listener); if (this._authResult) listener(this._user || null); return this._offIdTokenChanged.bind(this, listener); } /** * Sign the current user out * @return {Promise} */ signOut(): Promise { return this._interceptUserValue(this._native.signOut()); } /** * Sign a user in anonymously * @return {Promise} A promise resolved upon completion */ signInAnonymously(): Promise { return this._interceptUserValue(this._native.signInAnonymously()); } /** * Create a user with the email/password functionality * @param {string} email The user's email * @param {string} password The user's password * @return {Promise} A promise indicating the completion */ createUserWithEmailAndPassword(email: string, password: string): Promise { return this._interceptUserValue(this._native.createUserWithEmailAndPassword(email, password)); } /** * Sign a user in with email/password * @param {string} email The user's email * @param {string} password The user's password * @return {Promise} A promise that is resolved upon completion */ signInWithEmailAndPassword(email: string, password: string): Promise { return this._interceptUserValue(this._native.signInWithEmailAndPassword(email, password)); } /** * Sign the user in with a custom auth token * @param {string} customToken A self-signed custom auth token. * @return {Promise} A promise resolved upon completion */ signInWithCustomToken(customToken: string): Promise { return this._interceptUserValue(this._native.signInWithCustomToken(customToken)); } /** * Sign the user in with a third-party authentication provider * @return {Promise} A promise resolved upon completion */ signInWithCredential(credential: CredentialType): Promise { return this._interceptUserValue( this._native.signInWithCredential( credential.providerId, credential.token, credential.secret, ), ); } /** * Asynchronously signs in using a phone number. * */ signInWithPhoneNumber(phoneNumber: string): Promise { return this._native.signInWithPhoneNumber(phoneNumber).then((result) => { return new ConfirmationResult(this, result.verificationId); }); } /** * Send reset password instructions via email * @param {string} email The email to send password reset instructions */ sendPasswordResetEmail(email: string): Promise { return this._native.sendPasswordResetEmail(email); } /** * Completes the password reset process, given a confirmation code and new password. * * @link https://firebase.google.com/docs/reference/js/firebase.auth.Auth#confirmPasswordReset * @param code * @param newPassword * @return {Promise.} */ confirmPasswordReset(code: string, newPassword: string): Promise { return FirebaseAuth.confirmPasswordReset(code, newPassword); } /** * Applies a verification code sent to the user by email or other out-of-band mechanism. * * @link https://firebase.google.com/docs/reference/js/firebase.auth.Auth#applyActionCode * @param code * @return {Promise.} */ applyActionCode(code: string): Promise { return FirebaseAuth.applyActionCode(code); } /** * Checks a verification code sent to the user by email or other out-of-band mechanism. * * @link https://firebase.google.com/docs/reference/js/firebase.auth.Auth#checkActionCode * @param code * @return {Promise.|Promise} */ checkActionCode(code: string): Promise { return FirebaseAuth.checkActionCode(code); } /** * Get the currently signed in user * @return {Promise} */ getCurrentUser(): Promise { return this._interceptUserValue(this._native.getCurrentUser()); } /** * Returns a list of authentication providers that can be used to sign in a given user (identified by its main email address). * @return {Promise} */ fetchProvidersForEmail(email: string): Promise> { return this._native.fetchProvidersForEmail(email); } /** * Get the currently signed in user * @return {Promise} */ get currentUser(): User | null { return this._user; } get namespace(): string { return 'firebase:auth'; } /** * KNOWN UNSUPPORTED METHODS */ getRedirectResult() { throw new Error(INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(Auth, 'getRedirectResult')); } setPersistence() { throw new Error(INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(Auth, 'setPersistence')); } signInAndRetrieveDataWithCredential() { throw new Error(INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(Auth, 'signInAndRetrieveDataWithCredential')); } signInWithPopup() { throw new Error(INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(Auth, 'signInWithPopup')); } signInWithRedirect() { throw new Error(INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(Auth, 'signInWithRedirect')); } } export const statics = { EmailAuthProvider, PhoneAuthProvider, GoogleAuthProvider, GithubAuthProvider, TwitterAuthProvider, FacebookAuthProvider, };