react-native-firebase/lib/modules/auth/index.js

527 lines
15 KiB
JavaScript
Raw Normal View History

2017-11-17 14:22:46 +00:00
/**
* @flow
* Auth representation wrapper
*/
import User from './User';
import ModuleBase from '../../utils/ModuleBase';
import { getAppEventName, SharedEventEmitter } from '../../utils/events';
import { getLogger } from '../../utils/log';
import { getNativeModule } from '../../utils/native';
import INTERNALS from '../../utils/internals';
2017-08-25 11:16:23 +00:00
import ConfirmationResult from './ConfirmationResult';
// providers
import EmailAuthProvider from './providers/EmailAuthProvider';
2017-09-24 12:57:59 +00:00
import PhoneAuthProvider from './providers/PhoneAuthProvider';
import GoogleAuthProvider from './providers/GoogleAuthProvider';
import GithubAuthProvider from './providers/GithubAuthProvider';
2018-01-24 15:46:18 +00:00
import OAuthProvider from './providers/OAuthProvider';
import TwitterAuthProvider from './providers/TwitterAuthProvider';
import FacebookAuthProvider from './providers/FacebookAuthProvider';
2017-09-24 12:57:59 +00:00
import PhoneAuthListener from './PhoneAuthListener';
2017-02-14 11:41:27 +00:00
2018-01-25 18:25:39 +00:00
import type {
ActionCodeInfo,
2018-01-25 18:25:39 +00:00
ActionCodeSettings,
AuthCredential,
NativeUser,
NativeUserCredential,
UserCredential,
} from './types';
import type App from '../core/app';
2017-11-17 14:22:46 +00:00
2018-01-24 09:46:39 +00:00
type AuthState = {
user?: NativeUser,
};
2017-11-23 17:29:40 +00:00
const NATIVE_EVENTS = [
'auth_state_changed',
'auth_id_token_changed',
'phone_auth_state_changed',
];
export const MODULE_NAME = 'RNFirebaseAuth';
export const NAMESPACE = 'auth';
export default class Auth extends ModuleBase {
2018-01-24 09:46:39 +00:00
_authResult: boolean;
_languageCode: string;
_user: User | null;
2017-02-14 11:41:27 +00:00
constructor(app: App) {
super(app, {
events: NATIVE_EVENTS,
moduleName: MODULE_NAME,
multiApp: true,
namespace: NAMESPACE,
});
2017-02-14 11:41:27 +00:00
this._user = null;
2018-01-24 09:46:39 +00:00
this._authResult = false;
2018-01-25 18:25:39 +00:00
this._languageCode =
getNativeModule(this).APP_LANGUAGE[app._name] ||
getNativeModule(this).APP_LANGUAGE['[DEFAULT]'];
SharedEventEmitter.addListener(
// sub to internal native event - this fans out to
2017-09-21 15:48:54 +00:00
// public event name: onAuthStateChanged
getAppEventName(this, 'auth_state_changed'),
(state: AuthState) => {
this._setUser(state.user);
2018-01-25 18:25:39 +00:00
SharedEventEmitter.emit(
getAppEventName(this, 'onAuthStateChanged'),
this._user
);
}
);
SharedEventEmitter.addListener(
// sub to internal native event - this fans out to
// public events based on event.type
getAppEventName(this, 'phone_auth_state_changed'),
(event: Object) => {
const eventKey = `phone:auth:${event.requestKey}:${event.type}`;
SharedEventEmitter.emit(eventKey, event.state);
2018-01-25 18:25:39 +00:00
}
);
SharedEventEmitter.addListener(
2017-09-21 15:48:54 +00:00
// sub to internal native event - this fans out to
// public event name: onIdTokenChanged
getAppEventName(this, 'auth_id_token_changed'),
(auth: AuthState) => {
this._setUser(auth.user);
2018-01-25 18:25:39 +00:00
SharedEventEmitter.emit(
getAppEventName(this, 'onIdTokenChanged'),
this._user
);
}
2017-09-21 15:48:54 +00:00
);
getNativeModule(this).addAuthStateListener();
getNativeModule(this).addIdTokenListener();
2017-02-14 11:41:27 +00:00
}
2018-01-24 09:46:39 +00:00
_setUser(user: ?NativeUser): ?User {
this._authResult = true;
this._user = user ? new User(this, user) : null;
SharedEventEmitter.emit(getAppEventName(this, 'onUserChanged'), this._user);
2018-01-24 09:46:39 +00:00
return this._user;
}
_setUserCredential(userCredential: NativeUserCredential): UserCredential {
const user = new User(this, userCredential.user);
this._authResult = true;
this._user = user;
SharedEventEmitter.emit(getAppEventName(this, 'onUserChanged'), this._user);
return {
additionalUserInfo: userCredential.additionalUserInfo,
user,
};
}
2017-02-14 11:41:27 +00:00
/*
* WEB API
*/
/**
* Listen for auth changes.
* @param listener
*/
onAuthStateChanged(listener: Function) {
getLogger(this).info('Creating onAuthStateChanged listener');
2018-01-25 18:25:39 +00:00
SharedEventEmitter.addListener(
getAppEventName(this, 'onAuthStateChanged'),
listener
);
2017-06-13 01:09:12 +00:00
if (this._authResult) listener(this._user || null);
2017-02-14 11:41:27 +00:00
return () => {
getLogger(this).info('Removing onAuthStateChanged listener');
2018-01-25 18:25:39 +00:00
SharedEventEmitter.removeListener(
getAppEventName(this, 'onAuthStateChanged'),
listener
);
};
}
2017-09-21 15:48:54 +00:00
/**
* Listen for id token changes.
* @param listener
*/
onIdTokenChanged(listener: Function) {
getLogger(this).info('Creating onIdTokenChanged listener');
2018-01-25 18:25:39 +00:00
SharedEventEmitter.addListener(
getAppEventName(this, 'onIdTokenChanged'),
listener
);
2017-09-21 15:48:54 +00:00
if (this._authResult) listener(this._user || null);
return () => {
getLogger(this).info('Removing onIdTokenChanged listener');
2018-01-25 18:25:39 +00:00
SharedEventEmitter.removeListener(
getAppEventName(this, 'onIdTokenChanged'),
listener
);
};
}
/**
* Listen for user changes.
* @param listener
*/
onUserChanged(listener: Function) {
getLogger(this).info('Creating onUserChanged listener');
2018-01-25 18:25:39 +00:00
SharedEventEmitter.addListener(
getAppEventName(this, 'onUserChanged'),
listener
);
if (this._authResult) listener(this._user || null);
return () => {
getLogger(this).info('Removing onUserChanged listener');
2018-01-25 18:25:39 +00:00
SharedEventEmitter.removeListener(
getAppEventName(this, 'onUserChanged'),
listener
);
};
}
2017-02-14 11:41:27 +00:00
/**
* Sign the current user out
* @return {Promise}
2017-02-14 11:41:27 +00:00
*/
2017-11-17 14:22:46 +00:00
signOut(): Promise<void> {
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signOut()
.then(() => {
this._setUser();
});
2017-02-14 11:41:27 +00:00
}
/**
* Sign a user in anonymously
* @deprecated Deprecated signInAnonymously in favor of signInAnonymouslyAndRetrieveData.
* @return {Promise} A promise resolved upon completion
*/
2017-11-17 14:22:46 +00:00
signInAnonymously(): Promise<User> {
2018-01-25 18:25:39 +00:00
console.warn(
'Deprecated firebase.User.prototype.signInAnonymously in favor of firebase.User.prototype.signInAnonymouslyAndRetrieveData.'
);
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signInAnonymously()
.then(user => this._setUser(user));
}
/**
* Sign a user in anonymously
* @return {Promise} A promise resolved upon completion
*/
signInAnonymouslyAndRetrieveData(): Promise<UserCredential> {
return getNativeModule(this)
.signInAnonymouslyAndRetrieveData()
.then(userCredential => this._setUserCredential(userCredential));
}
2017-02-14 11:41:27 +00:00
/**
* Create a user with the email/password functionality
* @deprecated Deprecated createUserWithEmailAndPassword in favor of createUserAndRetrieveDataWithEmailAndPassword.
2017-02-14 11:41:27 +00:00
* @param {string} email The user's email
* @param {string} password The user's password
* @return {Promise} A promise indicating the completion
*/
2018-01-25 18:25:39 +00:00
createUserWithEmailAndPassword(
email: string,
password: string
): Promise<User> {
console.warn(
'Deprecated firebase.User.prototype.createUserWithEmailAndPassword in favor of firebase.User.prototype.createUserAndRetrieveDataWithEmailAndPassword.'
);
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.createUserWithEmailAndPassword(email, password)
.then(user => this._setUser(user));
}
/**
* 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
*/
2018-01-25 18:25:39 +00:00
createUserAndRetrieveDataWithEmailAndPassword(
email: string,
password: string
): Promise<User> {
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.createUserAndRetrieveDataWithEmailAndPassword(email, password)
.then(userCredential => this._setUserCredential(userCredential));
2017-02-14 11:41:27 +00:00
}
/**
* Sign a user in with email/password
* @deprecated Deprecated signInWithEmailAndPassword in favor of signInAndRetrieveDataWithEmailAndPassword
2017-02-14 11:41:27 +00:00
* @param {string} email The user's email
* @param {string} password The user's password
* @return {Promise} A promise that is resolved upon completion
*/
2017-11-17 14:22:46 +00:00
signInWithEmailAndPassword(email: string, password: string): Promise<User> {
2018-01-25 18:25:39 +00:00
console.warn(
'Deprecated firebase.User.prototype.signInWithEmailAndPassword in favor of firebase.User.prototype.signInAndRetrieveDataWithEmailAndPassword.'
);
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signInWithEmailAndPassword(email, password)
.then(user => this._setUser(user));
}
/**
* 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
*/
2018-01-25 18:25:39 +00:00
signInAndRetrieveDataWithEmailAndPassword(
email: string,
password: string
): Promise<UserCredential> {
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signInAndRetrieveDataWithEmailAndPassword(email, password)
.then(userCredential => this._setUserCredential(userCredential));
2017-02-14 11:41:27 +00:00
}
/**
* Sign the user in with a custom auth token
* @deprecated Deprecated signInWithCustomToken in favor of signInAndRetrieveDataWithCustomToken
2017-02-14 11:41:27 +00:00
* @param {string} customToken A self-signed custom auth token.
* @return {Promise} A promise resolved upon completion
*/
2017-11-17 14:22:46 +00:00
signInWithCustomToken(customToken: string): Promise<User> {
2018-01-25 18:25:39 +00:00
console.warn(
'Deprecated firebase.User.prototype.signInWithCustomToken in favor of firebase.User.prototype.signInAndRetrieveDataWithCustomToken.'
);
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signInWithCustomToken(customToken)
.then(user => this._setUser(user));
}
/**
* 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
*/
2018-01-25 18:25:39 +00:00
signInAndRetrieveDataWithCustomToken(
customToken: string
): Promise<UserCredential> {
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signInAndRetrieveDataWithCustomToken(customToken)
.then(userCredential => this._setUserCredential(userCredential));
2017-02-14 11:41:27 +00:00
}
/**
* Sign the user in with a third-party authentication provider
* @deprecated Deprecated signInWithCredential in favor of signInAndRetrieveDataWithCredential.
2017-02-14 11:41:27 +00:00
* @return {Promise} A promise resolved upon completion
*/
2017-11-17 14:22:46 +00:00
signInWithCredential(credential: AuthCredential): Promise<User> {
2018-01-25 18:25:39 +00:00
console.warn(
'Deprecated firebase.User.prototype.signInWithCredential in favor of firebase.User.prototype.signInAndRetrieveDataWithCredential.'
);
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
2018-01-25 18:25:39 +00:00
.signInWithCredential(
credential.providerId,
credential.token,
credential.secret
)
2018-01-24 09:46:39 +00:00
.then(user => this._setUser(user));
}
/**
* Sign the user in with a third-party authentication provider
* @return {Promise} A promise resolved upon completion
*/
2018-01-25 18:25:39 +00:00
signInAndRetrieveDataWithCredential(
credential: AuthCredential
): Promise<UserCredential> {
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
2018-01-25 18:25:39 +00:00
.signInAndRetrieveDataWithCredential(
credential.providerId,
credential.token,
credential.secret
)
2018-01-24 09:46:39 +00:00
.then(userCredential => this._setUserCredential(userCredential));
2017-02-14 11:41:27 +00:00
}
2017-08-25 11:16:23 +00:00
/**
* Asynchronously signs in using a phone number.
*
*/
2017-11-17 14:22:46 +00:00
signInWithPhoneNumber(phoneNumber: string): Promise<ConfirmationResult> {
2018-01-24 09:46:39 +00:00
return getNativeModule(this)
.signInWithPhoneNumber(phoneNumber)
2018-01-25 18:25:39 +00:00
.then(result => new ConfirmationResult(this, result.verificationId));
2017-08-25 11:16:23 +00:00
}
2017-09-24 12:57:59 +00:00
/**
* Returns a PhoneAuthListener to listen to phone verification events,
* on the final completion event a PhoneAuthCredential can be generated for
* authentication purposes.
*
* @param phoneNumber
* @param autoVerifyTimeout Android Only
* @returns {PhoneAuthListener}
*/
2018-01-25 18:25:39 +00:00
verifyPhoneNumber(
phoneNumber: string,
autoVerifyTimeout?: number
): PhoneAuthListener {
2017-09-24 12:57:59 +00:00
return new PhoneAuthListener(this, phoneNumber, autoVerifyTimeout);
}
2017-02-14 11:41:27 +00:00
/**
* Send reset password instructions via email
* @param {string} email The email to send password reset instructions
*/
2018-01-25 18:25:39 +00:00
sendPasswordResetEmail(
email: string,
actionCodeSettings?: ActionCodeSettings
): Promise<void> {
return getNativeModule(this).sendPasswordResetEmail(
email,
actionCodeSettings
);
2017-02-14 11:41:27 +00: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>}
*/
2017-11-17 14:22:46 +00:00
confirmPasswordReset(code: string, newPassword: string): Promise<void> {
return getNativeModule(this).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.<Null>}
*/
2017-11-17 14:22:46 +00:00
applyActionCode(code: string): Promise<void> {
return getNativeModule(this).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
2017-09-25 20:29:40 +00:00
* @return {Promise.<any>|Promise<ActionCodeInfo>}
*/
2018-01-18 09:49:11 +00:00
checkActionCode(code: string): Promise<ActionCodeInfo> {
return getNativeModule(this).checkActionCode(code);
}
/**
* Returns a list of authentication providers that can be used to sign in a given user (identified by its main email address).
* @return {Promise}
*/
2018-01-24 12:20:06 +00:00
fetchProvidersForEmail(email: string): Promise<string[]> {
return getNativeModule(this).fetchProvidersForEmail(email);
}
2018-01-24 12:20:06 +00:00
verifyPasswordResetCode(code: string): Promise<string> {
return getNativeModule(this).verifyPasswordResetCode(code);
}
/**
* Sets the language for the auth module
* @param code
* @returns {*}
*/
2017-11-30 10:56:01 +00:00
set languageCode(code: string) {
this._languageCode = code;
getNativeModule(this).setLanguageCode(code);
}
2017-02-14 11:41:27 +00:00
/**
* Get the currently signed in user
* @return {Promise}
*/
get currentUser(): User | null {
2017-02-14 11:41:27 +00:00
return this._user;
}
get languageCode(): string {
return this._languageCode;
}
/**
* KNOWN UNSUPPORTED METHODS
*/
getRedirectResult() {
2018-01-25 18:25:39 +00:00
throw new Error(
INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(
'auth',
'getRedirectResult'
)
);
}
setPersistence() {
2018-01-25 18:25:39 +00:00
throw new Error(
INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(
'auth',
'setPersistence'
)
);
}
signInWithPopup() {
2018-01-25 18:25:39 +00:00
throw new Error(
INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(
'auth',
'signInWithPopup'
)
);
}
signInWithRedirect() {
2018-01-25 18:25:39 +00:00
throw new Error(
INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(
'auth',
'signInWithRedirect'
)
);
}
// firebase issue - https://github.com/invertase/react-native-firebase/pull/655#issuecomment-349904680
useDeviceLanguage() {
2018-01-25 18:25:39 +00:00
throw new Error(
INTERNALS.STRINGS.ERROR_UNSUPPORTED_MODULE_METHOD(
'auth',
'useDeviceLanguage'
)
);
}
2017-02-14 11:41:27 +00:00
}
2017-03-27 18:11:26 +00:00
export const statics = {
EmailAuthProvider,
PhoneAuthProvider,
GoogleAuthProvider,
2017-03-27 18:11:26 +00:00
GithubAuthProvider,
TwitterAuthProvider,
FacebookAuthProvider,
OAuthProvider,
2017-09-24 12:57:59 +00:00
PhoneAuthState: {
CODE_SENT: 'sent',
AUTO_VERIFY_TIMEOUT: 'timeout',
2017-09-24 12:57:59 +00:00
AUTO_VERIFIED: 'verified',
ERROR: 'error',
},
2017-03-27 18:11:26 +00:00
};