0edb79243a
Fixes #144
192 lines
5.7 KiB
JavaScript
192 lines
5.7 KiB
JavaScript
// @flow
|
|
import { NativeModules, NativeEventEmitter } from 'react-native';
|
|
|
|
import User from './user';
|
|
import { Base } from './../base';
|
|
import { nativeSDKMissing } from './../../utils';
|
|
|
|
// providers
|
|
import EmailAuthProvider from './providers/Email';
|
|
import GoogleAuthProvider from './providers/Google';
|
|
import FacebookAuthProvider from './providers/Facebook';
|
|
import TwitterAuthProvider from './providers/Twitter';
|
|
import GithubAuthProvider from './providers/Github';
|
|
|
|
const FirebaseAuth = NativeModules.RNFirebaseAuth;
|
|
const FirebaseAuthEvt = FirebaseAuth && new NativeEventEmitter(FirebaseAuth);
|
|
|
|
export default class Auth extends Base {
|
|
_user: User | null;
|
|
_authResult: AuthResultType | null;
|
|
authenticated: boolean;
|
|
|
|
constructor(firebase: Object, options: Object = {}) {
|
|
super(firebase, options);
|
|
if (!FirebaseAuth) {
|
|
return nativeSDKMissing('auth');
|
|
}
|
|
|
|
this._user = null;
|
|
this._authResult = null;
|
|
this.authenticated = false;
|
|
|
|
// start listening immediately for auth changes
|
|
FirebaseAuthEvt.addListener('onAuthStateChanged', this._onAuthStateChanged.bind(this));
|
|
FirebaseAuth.addAuthStateListener();
|
|
}
|
|
|
|
/**
|
|
* 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('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('onAuthStateChanged', listener);
|
|
}
|
|
|
|
/**
|
|
* Intercept all user actions and send their results to
|
|
* auth state change before resolving
|
|
* @param promise
|
|
* @returns {Promise.<TResult>|*}
|
|
* @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('onAuthStateChanged', listener);
|
|
if (this._authResult) listener(this._user || null);
|
|
return this._offAuthStateChanged.bind(this, listener);
|
|
}
|
|
|
|
/**
|
|
* Sign the current user out
|
|
* @return {Promise}
|
|
*/
|
|
signOut(): Promise<null> {
|
|
return this._interceptUserValue(FirebaseAuth.signOut());
|
|
}
|
|
|
|
/**
|
|
* Sign a user in anonymously
|
|
* @return {Promise} A promise resolved upon completion
|
|
*/
|
|
signInAnonymously(): Promise<Object> {
|
|
return this._interceptUserValue(FirebaseAuth.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<Object> {
|
|
return this._interceptUserValue(FirebaseAuth.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<Object> {
|
|
return this._interceptUserValue(FirebaseAuth.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<Object> {
|
|
return this._interceptUserValue(FirebaseAuth.signInWithCustomToken(customToken));
|
|
}
|
|
|
|
/**
|
|
* Sign the user in with a third-party authentication provider
|
|
* @return {Promise} A promise resolved upon completion
|
|
*/
|
|
signInWithCredential(credential: CredentialType): Promise<Object> {
|
|
return this._interceptUserValue(FirebaseAuth.signInWithCredential(credential.provider, credential.token, credential.secret));
|
|
}
|
|
|
|
/**
|
|
* Send reset password instructions via email
|
|
* @param {string} email The email to send password reset instructions
|
|
*/
|
|
sendPasswordResetEmail(email: string): Promise<Object> {
|
|
return FirebaseAuth.sendPasswordResetEmail(email);
|
|
}
|
|
|
|
/**
|
|
* Get the currently signed in user
|
|
* @return {Promise}
|
|
*/
|
|
getCurrentUser(): Promise<Object> {
|
|
return this._interceptUserValue(FirebaseAuth.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<Array<String>> {
|
|
return FirebaseAuth.fetchProvidersForEmail(email);
|
|
}
|
|
|
|
/**
|
|
* Get the currently signed in user
|
|
* @return {Promise}
|
|
*/
|
|
get currentUser(): User | null {
|
|
return this._user;
|
|
}
|
|
|
|
get namespace(): string {
|
|
return 'firebase:auth';
|
|
}
|
|
}
|
|
|
|
export const statics = {
|
|
GoogleAuthProvider,
|
|
EmailAuthProvider,
|
|
FacebookAuthProvider,
|
|
TwitterAuthProvider,
|
|
GithubAuthProvider,
|
|
};
|