2017-02-14 11:41:27 +00:00
|
|
|
// @flow
|
|
|
|
import User from './user';
|
2017-06-29 17:24:34 +01:00
|
|
|
import ModuleBase from './../../utils/ModuleBase';
|
2017-08-25 12:16:23 +01:00
|
|
|
import ConfirmationResult from './ConfirmationResult';
|
2017-03-17 23:51:26 +00:00
|
|
|
|
|
|
|
// providers
|
2017-03-17 23:40:12 +00:00
|
|
|
import EmailAuthProvider from './providers/Email';
|
2017-03-17 23:51:26 +00:00
|
|
|
import GoogleAuthProvider from './providers/Google';
|
|
|
|
import GithubAuthProvider from './providers/Github';
|
2017-08-12 19:07:51 +01:00
|
|
|
import TwitterAuthProvider from './providers/Twitter';
|
|
|
|
import FacebookAuthProvider from './providers/Facebook';
|
2017-08-25 12:16:23 +01:00
|
|
|
import PhoneAuthProvider from './providers/Phone';
|
2017-02-14 11:41:27 +00:00
|
|
|
|
2017-06-29 17:24:34 +01:00
|
|
|
export default class Auth extends ModuleBase {
|
2017-08-17 17:58:28 +01:00
|
|
|
static _NAMESPACE = 'auth';
|
|
|
|
static _NATIVE_MODULE = 'RNFirebaseAuth';
|
|
|
|
|
2017-05-25 23:39:06 +01:00
|
|
|
_user: User | null;
|
2017-02-14 11:41:27 +00:00
|
|
|
_authResult: AuthResultType | null;
|
|
|
|
authenticated: boolean;
|
|
|
|
|
2017-06-29 17:24:34 +01:00
|
|
|
constructor(firebaseApp: Object, options: Object = {}) {
|
2017-08-17 17:58:28 +01:00
|
|
|
super(firebaseApp, options, true);
|
2017-02-14 11:41:27 +00:00
|
|
|
this._user = null;
|
|
|
|
this._authResult = null;
|
|
|
|
this.authenticated = false;
|
2017-07-12 15:49:33 +01:00
|
|
|
this.addListener(
|
|
|
|
// sub to internal native event - this fans out to
|
|
|
|
// public event name: onAuthStateChangedPublic
|
|
|
|
this._getAppEventName('onAuthStateChanged'),
|
|
|
|
this._onAuthStateChanged.bind(this),
|
|
|
|
);
|
2017-07-17 20:56:08 +01:00
|
|
|
|
|
|
|
this._native.addAuthStateListener();
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Internal auth changed listener
|
|
|
|
* @param auth
|
2017-03-13 20:02:44 +00:00
|
|
|
* @param emit
|
2017-02-14 11:41:27 +00:00
|
|
|
* @private
|
|
|
|
*/
|
2017-03-17 23:40:12 +00:00
|
|
|
_onAuthStateChanged(auth: AuthResultType, emit: boolean = true) {
|
2017-02-14 11:41:27 +00:00
|
|
|
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);
|
2017-07-12 15:49:33 +01:00
|
|
|
if (emit) this.emit(this._getAppEventName('onAuthStateChangedPublic'), this._user);
|
2017-03-13 20:02:44 +00:00
|
|
|
return auth ? this._user : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove auth change listener
|
|
|
|
* @param listener
|
|
|
|
*/
|
|
|
|
_offAuthStateChanged(listener: Function) {
|
|
|
|
this.log.info('Removing onAuthStateChanged listener');
|
2017-07-12 15:49:33 +01:00
|
|
|
this.removeListener(this._getAppEventName('onAuthStateChangedPublic'), listener);
|
2017-03-13 20:02:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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);
|
2017-03-17 23:40:12 +00:00
|
|
|
if (result.uid) return this._onAuthStateChanged({ authenticated: true, user: result }, false);
|
2017-03-13 20:02:44 +00:00
|
|
|
return result;
|
|
|
|
});
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* WEB API
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Listen for auth changes.
|
|
|
|
* @param listener
|
|
|
|
*/
|
|
|
|
onAuthStateChanged(listener: Function) {
|
|
|
|
this.log.info('Creating onAuthStateChanged listener');
|
2017-07-12 15:49:33 +01:00
|
|
|
this.on(this._getAppEventName('onAuthStateChangedPublic'), listener);
|
2017-06-13 02:09:12 +01:00
|
|
|
if (this._authResult) listener(this._user || null);
|
2017-02-14 11:41:27 +00:00
|
|
|
return this._offAuthStateChanged.bind(this, listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-03-13 20:02:44 +00:00
|
|
|
* Sign the current user out
|
|
|
|
* @return {Promise}
|
2017-02-14 11:41:27 +00:00
|
|
|
*/
|
2017-03-13 20:02:44 +00:00
|
|
|
signOut(): Promise<null> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._interceptUserValue(this._native.signOut());
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
2017-03-13 17:23:41 +00:00
|
|
|
/**
|
|
|
|
* Sign a user in anonymously
|
2017-03-13 20:02:44 +00:00
|
|
|
* @return {Promise} A promise resolved upon completion
|
2017-03-13 17:23:41 +00:00
|
|
|
*/
|
|
|
|
signInAnonymously(): Promise<Object> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._interceptUserValue(this._native.signInAnonymously());
|
2017-03-13 17:23:41 +00:00
|
|
|
}
|
|
|
|
|
2017-02-14 11:41:27 +00:00
|
|
|
/**
|
|
|
|
* 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> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._interceptUserValue(this._native.createUserWithEmailAndPassword(email, password));
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._interceptUserValue(this._native.signInWithEmailAndPassword(email, password));
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._interceptUserValue(this._native.signInWithCustomToken(customToken));
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sign the user in with a third-party authentication provider
|
|
|
|
* @return {Promise} A promise resolved upon completion
|
|
|
|
*/
|
|
|
|
signInWithCredential(credential: CredentialType): Promise<Object> {
|
2017-08-19 05:22:07 +01:00
|
|
|
return this._interceptUserValue(
|
|
|
|
this._native.signInWithCredential(
|
|
|
|
credential.provider, credential.token, credential.secret,
|
|
|
|
),
|
|
|
|
);
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
2017-08-25 12:16:23 +01:00
|
|
|
/**
|
|
|
|
* Asynchronously signs in using a phone number.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
signInWithPhoneNumber(phoneNumber: string): Promise<Object> {
|
|
|
|
return this._native.signInWithPhoneNumber(phoneNumber).then((result) => {
|
|
|
|
return new ConfirmationResult(this, result.verificationId);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-02-14 11:41:27 +00:00
|
|
|
/**
|
|
|
|
* Send reset password instructions via email
|
|
|
|
* @param {string} email The email to send password reset instructions
|
|
|
|
*/
|
|
|
|
sendPasswordResetEmail(email: string): Promise<Object> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._native.sendPasswordResetEmail(email);
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
2017-07-04 18:22:18 +01:00
|
|
|
/**
|
|
|
|
* 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.<Null>}
|
|
|
|
*/
|
|
|
|
confirmPasswordReset(code: string, newPassword: string): Promise<Null> {
|
|
|
|
return FirebaseAuth.confirmPasswordReset(code, newPassword);
|
|
|
|
}
|
|
|
|
|
2017-07-05 14:16:35 +01:00
|
|
|
/**
|
|
|
|
* 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.<Null>}
|
|
|
|
*/
|
2017-07-05 14:56:18 +01:00
|
|
|
applyActionCode(code: string): Promise<Any> {
|
2017-07-05 14:16:35 +01:00
|
|
|
return FirebaseAuth.applyActionCode(code);
|
|
|
|
}
|
|
|
|
|
2017-07-05 14:56:18 +01:00
|
|
|
/**
|
|
|
|
* 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.<Any>|Promise<ActionCodeInfo>}
|
|
|
|
*/
|
|
|
|
checkActionCode(code: string): Promise<Any> {
|
|
|
|
return FirebaseAuth.checkActionCode(code);
|
|
|
|
}
|
2017-07-04 18:22:18 +01:00
|
|
|
|
2017-02-14 11:41:27 +00:00
|
|
|
/**
|
|
|
|
* Get the currently signed in user
|
|
|
|
* @return {Promise}
|
|
|
|
*/
|
|
|
|
getCurrentUser(): Promise<Object> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._interceptUserValue(this._native.getCurrentUser());
|
2017-02-14 11:41:27 +00:00
|
|
|
}
|
|
|
|
|
2017-05-25 14:45:03 +01:00
|
|
|
/**
|
|
|
|
* Returns a list of authentication providers that can be used to sign in a given user (identified by its main email address).
|
|
|
|
* @return {Promise}
|
|
|
|
*/
|
2017-05-25 15:30:39 +01:00
|
|
|
fetchProvidersForEmail(email: string): Promise<Array<String>> {
|
2017-06-30 17:23:32 +01:00
|
|
|
return this._native.fetchProvidersForEmail(email);
|
2017-05-25 14:45:03 +01:00
|
|
|
}
|
|
|
|
|
2017-02-14 11:41:27 +00:00
|
|
|
/**
|
|
|
|
* Get the currently signed in user
|
|
|
|
* @return {Promise}
|
|
|
|
*/
|
2017-05-25 23:39:06 +01:00
|
|
|
get currentUser(): User | null {
|
2017-02-14 11:41:27 +00:00
|
|
|
return this._user;
|
|
|
|
}
|
|
|
|
|
|
|
|
get namespace(): string {
|
|
|
|
return 'firebase:auth';
|
|
|
|
}
|
|
|
|
}
|
2017-03-27 19:11:26 +01:00
|
|
|
|
|
|
|
export const statics = {
|
|
|
|
EmailAuthProvider,
|
2017-08-12 19:07:51 +01:00
|
|
|
PhoneAuthProvider,
|
|
|
|
GoogleAuthProvider,
|
2017-03-27 19:11:26 +01:00
|
|
|
GithubAuthProvider,
|
2017-08-12 19:07:51 +01:00
|
|
|
TwitterAuthProvider,
|
|
|
|
FacebookAuthProvider,
|
2017-03-27 19:11:26 +01:00
|
|
|
};
|