If you've upgraded to Firebase Authentication with Identity Platform , you can add SMS multi-factor authentication to your web app.
Multi-factor authentication increases the security of your app. While attackers often compromise passwords and social accounts, intercepting a text message is more difficult.
Before you begin
-  Enable at least one provider that supports multi-factor authentication. Every provider supports MFA, exceptphone auth, anonymous auth, and Apple Game Center. 
-  Enable the regions where you plan to use SMS authentication. Firebase uses a fully blocking SMS region policy, which helps to create your projects in a more secure state by default. 
-  Ensure your app is verifying user emails. MFA requires email verification. This prevents malicious actors from registering for a service with an email they don't own, and then locking out the real owner by adding a second factor. 
Using multi-tenancy
If you are enabling multi-factor authentication for use in a multi-tenant environment, make sure to complete the following steps (in addition to the rest of the instructions in this document):
-  In the Google Cloud console, select the tenant you want to work with. 
-  In your code, set the tenantIdfield on theAuthinstance to your tenant's ID. For example:Webimport { getAuth } from "firebase/auth" ; const auth = getAuth ( app ); auth . tenantId = "myTenantId1" ;Webfirebase.auth().tenantId = 'myTenantId1';
Enabling multi-factor authentication
-  Open the Authentication > Sign-in method page of the Firebase console. 
-  In the Advancedsection, enable SMS Multi-factor Authentication. You should also enter the phone numbers you'll be testing your app with. While optional, registering test phone numbers is strongly recommended to avoid throttling during development. 
-  If you haven't already authorized your app's domain, add it to the allow list on the Authentication > Settings page of the Firebase console. 
Choosing an enrollment pattern
You can choose whether your app requires multi-factor authentication, and how and when to enroll your users. Some common patterns include:
-  Enroll the user's second factor as part of registration. Use this method if your app requires multi-factor authentication for all users. 
-  Offer a skippable option to enroll a second factor during registration. Apps that want to encourage, but not require, multi-factor authentication may prefer this approach. 
-  Provide the ability to add a second factor from the user's account or profile management page, instead of the sign up screen. This minimizes friction during the registration process, while still making multi-factor authentication available for security-sensitive users. 
-  Require adding a second factor incrementally when the user wants to access features with increased security requirements. 
Setting up the reCAPTCHA verifier
Before you can send SMS codes, you need to configure a reCAPTCHA verifier. Firebase uses reCAPTCHA to prevent abuse by ensuring that phone number verification requests come from one of your app's allowed domains.
You don't need to manually set up a reCAPTCHA client; the client SDK's RecaptchaVerifier 
object automatically creates and initializes any necessary
client keys and secrets.
Using invisible reCAPTCHA
The RecaptchaVerifier 
object supports invisible reCAPTCHA 
,
which can often verify the user without requiring any interaction. To use an
invisible reCAPTCHA, create a RecaptchaVerifier 
with the size 
parameter set
to invisible 
, and specify the ID of the UI element that starts multi-factor
enrollment:
Web
  import 
  
 { 
  
 RecaptchaVerifier 
 , 
  
 getAuth 
  
 } 
  
 from 
  
 "firebase/auth" 
 ; 
 const 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 RecaptchaVerifier 
 ( 
 getAuth 
 (), 
  
 "sign-in-button" 
 , 
  
 { 
  
 "size" 
 : 
  
 "invisible" 
 , 
  
 "callback" 
 : 
  
 function 
 ( 
 response 
 ) 
  
 { 
  
 // reCAPTCHA solved, you can proceed with 
  
 // phoneAuthProvider.verifyPhoneNumber(...). 
  
 onSolvedRecaptcha 
 (); 
  
 } 
 }); 
 
 
Web
  var 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 firebase 
 . 
 auth 
 . 
 RecaptchaVerifier 
 ( 
 'sign-in-button' 
 , 
  
 { 
 'size' 
 : 
  
 'invisible' 
 , 
 'callback' 
 : 
  
 function 
 ( 
 response 
 ) 
  
 { 
  
 // 
  
 reCAPTCHA 
  
 solved 
 , 
  
 you 
  
 can 
  
 proceed 
  
 with 
  
 phoneAuthProvider 
 . 
 verifyPhoneNumber 
 ( 
 ... 
 ) 
 . 
  
 onSolvedRecaptcha 
 (); 
 } 
 }); 
 
 
Using the reCAPTCHA widget
To use a visible reCAPTCHA widget, create an HTML element to contain
the widget, then create a RecaptchaVerifier 
object with the ID of the UI
container. You can also optionally set callbacks that are invoked when the
reCAPTCHA is solved or expires:
Web
  import 
  
 { 
  
 RecaptchaVerifier 
 , 
  
 getAuth 
  
 } 
  
 from 
  
 "firebase/auth" 
 ; 
 const 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 RecaptchaVerifier 
 ( 
  
 getAuth 
 (), 
  
 "recaptcha-container" 
 , 
  
 // Optional reCAPTCHA parameters. 
  
 { 
  
 "size" 
 : 
  
 "normal" 
 , 
  
 "callback" 
 : 
  
 function 
 ( 
 response 
 ) 
  
 { 
  
 // reCAPTCHA solved, you can proceed with 
  
 // phoneAuthProvider.verifyPhoneNumber(...). 
  
 onSolvedRecaptcha 
 (); 
  
 }, 
  
 "expired-callback" 
 : 
  
 function 
 () 
  
 { 
  
 // Response expired. Ask user to solve reCAPTCHA again. 
  
 // ... 
  
 } 
  
 } 
 ); 
 
 
Web
  var 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 firebase 
 . 
 auth 
 . 
 RecaptchaVerifier 
 ( 
  
 'recaptcha-container' 
 , 
  
 // 
  
 Optional 
  
 reCAPTCHA 
  
 parameters 
 . 
  
 { 
  
 'size' 
 : 
  
 'normal' 
 , 
  
 'callback' 
 : 
  
 function 
 ( 
 response 
 ) 
  
 { 
  
 // 
  
 reCAPTCHA 
  
 solved 
 , 
  
 you 
  
 can 
  
 proceed 
  
 with 
  
 phoneAuthProvider 
 . 
 verifyPhoneNumber 
 ( 
 ... 
 ) 
 . 
  
 // 
  
 ... 
  
 onSolvedRecaptcha 
 (); 
  
 }, 
  
 'expired-callback' 
 : 
  
 function 
 () 
  
 { 
  
 // 
  
 Response 
  
 expired 
 . 
  
 Ask 
  
 user 
  
 to 
  
 solve 
  
 reCAPTCHA 
  
 again 
 . 
  
 // 
  
 ... 
  
 } 
  
 }); 
 
 
Pre-rendering the reCAPTCHA
Optionally, you can pre-render the reCAPTCHA before starting two-factor enrollment:
Web
  recaptchaVerifier 
 . 
 render 
 () 
  
 . 
 then 
 ( 
 function 
  
 ( 
 widgetId 
 ) 
  
 { 
  
 window 
 . 
 recaptchaWidgetId 
  
 = 
  
 widgetId 
 ; 
  
 }); 
 
 
Web
 recaptchaVerifier.render()
  .then(function(widgetId) {
    window.recaptchaWidgetId = widgetId;
  }); 
 
After render() 
resolves, you get the reCAPTCHA's widget ID, which you can use
to make calls to the reCAPTCHA API 
:
  var 
  
 recaptchaResponse 
  
 = 
  
 grecaptcha 
 . 
 getResponse 
 ( 
 window 
 . 
 recaptchaWidgetId 
 ); 
 
 
RecaptchaVerifier abstracts this logic away with the verify 
method, so you need not handle the grecaptcha 
variable directly.
Enrolling a second factor
To enroll a new secondary factor for a user:
-  Re-authenticate the user. 
-  Ask the user enter their phone number. 
-  Initialize the reCAPTCHA verifier as illustrated in the previous section. Skip this step if a RecaptchaVerifier instance is already configured: Webimport { RecaptchaVerifier , getAuth } from "firebase/auth" ; const recaptchaVerifier = new RecaptchaVerifier ( getAuth (), 'recaptcha-container-id' , undefined );Webvar recaptchaVerifier = new firebase . auth . RecaptchaVerifier ( 'recaptcha-container-id' );
-  Get a multi-factor session for the user: Webimport { multiFactor } from "firebase/auth" ; multiFactor ( user ). getSession (). then ( function ( multiFactorSession ) { // ... });Webuser.multiFactor.getSession().then(function(multiFactorSession) { // ... })
-  Initialize a PhoneInfoOptionsobject with the user's phone number and the multi-factor session:Web// Specify the phone number and pass the MFA session. const phoneInfoOptions = { phoneNumber : phoneNumber , session : multiFactorSession };Web// Specify the phone number and pass the MFA session . var phoneInfoOptions = { phoneNumber : phoneNumber , session : multiFactorSession };
-  Send a verification message to the user's phone: Webimport { PhoneAuthProvider } from "firebase/auth" ; const phoneAuthProvider = new PhoneAuthProvider ( auth ); phoneAuthProvider . verifyPhoneNumber ( phoneInfoOptions , recaptchaVerifier ) . then ( function ( verificationId ) { // verificationId will be needed to complete enrollment. });Webvar phoneAuthProvider = new firebase . auth . PhoneAuthProvider (); // Send SMS verification code . return phoneAuthProvider . verifyPhoneNumber ( phoneInfoOptions , recaptchaVerifier ) . then ( function ( verificationId ) { // verificationId will be needed for enrollment completion . })While not required, it's a best practice to inform users beforehand that they will receive an SMS message, and that standard rates apply. 
-  If the request fails, reset the reCAPTCHA, then repeat the previous step so the user can try again. Note that verifyPhoneNumber()will automatically reset the reCAPTCHA when it throws an error, as reCAPTCHA tokens are one-time use only.WebrecaptchaVerifier . clear ();WebrecaptchaVerifier.clear();
-  Once the SMS code is sent, ask the user to verify the code: Web// Ask user for the verification code. Then: const cred = PhoneAuthProvider . credential ( verificationId , verificationCode );Web// Ask user for the verification code . Then : var cred = firebase . auth . PhoneAuthProvider . credential ( verificationId , verificationCode );
-  Initialize a MultiFactorAssertionobject with thePhoneAuthCredential:Webimport { PhoneMultiFactorGenerator } from "firebase/auth" ; const multiFactorAssertion = PhoneMultiFactorGenerator . assertion ( cred );Webvar multiFactorAssertion = firebase . auth . PhoneMultiFactorGenerator . assertion ( cred );
-  Complete the enrollment. Optionally, you can specify a display name for the second factor. This is useful for users with multiple second factors, since the phone number is masked during the authentication flow (for example, +1******1234). Web// Complete enrollment. This will update the underlying tokens // and trigger ID token change listener. multiFactor ( user ). enroll ( multiFactorAssertion , "My personal phone number" );Web// Complete enrollment. This will update the underlying tokens // and trigger ID token change listener. user . multiFactor . enroll ( multiFactorAssertion , 'My personal phone number' );
The code below shows a complete example of enrolling a second factor:
Web
  import 
  
 { 
  
 multiFactor 
 , 
  
 PhoneAuthProvider 
 , 
  
 PhoneMultiFactorGenerator 
 , 
  
 RecaptchaVerifier 
 , 
  
 getAuth 
 } 
  
 from 
  
 "firebase/auth" 
 ; 
 const 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 RecaptchaVerifier 
 ( 
 getAuth 
 (), 
  
 'recaptcha-container-id' 
 , 
  
 undefined 
 ); 
 multiFactor 
 ( 
 user 
 ). 
 getSession 
 () 
  
 . 
 then 
 ( 
 function 
  
 ( 
 multiFactorSession 
 ) 
  
 { 
  
 // Specify the phone number and pass the MFA session. 
  
 const 
  
 phoneInfoOptions 
  
 = 
  
 { 
  
 phoneNumber 
 : 
  
 phoneNumber 
 , 
  
 session 
 : 
  
 multiFactorSession 
  
 }; 
  
 const 
  
 phoneAuthProvider 
  
 = 
  
 new 
  
 PhoneAuthProvider 
 ( 
 auth 
 ); 
  
 // Send SMS verification code. 
  
 return 
  
 phoneAuthProvider 
 . 
 verifyPhoneNumber 
 ( 
 phoneInfoOptions 
 , 
  
 recaptchaVerifier 
 ); 
  
 }). 
 then 
 ( 
 function 
  
 ( 
 verificationId 
 ) 
  
 { 
  
 // Ask user for the verification code. Then: 
  
 const 
  
 cred 
  
 = 
  
 PhoneAuthProvider 
 . 
 credential 
 ( 
 verificationId 
 , 
  
 verificationCode 
 ); 
  
 const 
  
 multiFactorAssertion 
  
 = 
  
 PhoneMultiFactorGenerator 
 . 
 assertion 
 ( 
 cred 
 ); 
  
 // Complete enrollment. 
  
 return 
  
 multiFactor 
 ( 
 user 
 ). 
 enroll 
 ( 
 multiFactorAssertion 
 , 
  
 mfaDisplayName 
 ); 
  
 }); 
 
 
Web
  var 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 firebase 
 . 
 auth 
 . 
 RecaptchaVerifier 
 ( 
 'recaptcha-container-id' 
 ); 
 user 
 . 
 multiFactor 
 . 
 getSession 
 () 
 . 
 then 
 ( 
 function 
 ( 
 multiFactorSession 
 ) 
  
 { 
  
 // 
  
 Specify 
  
 the 
  
 phone 
  
 number 
  
 and 
  
 pass 
  
 the 
  
 MFA 
  
 session 
 . 
  
 var 
  
 phoneInfoOptions 
  
 = 
  
 { 
  
 phoneNumber 
 : 
  
 phoneNumber 
 , 
  
 session 
 : 
  
 multiFactorSession 
  
 }; 
  
 var 
  
 phoneAuthProvider 
  
 = 
  
 new 
  
 firebase 
 . 
 auth 
 . 
 PhoneAuthProvider 
 (); 
  
 // 
  
 Send 
  
 SMS 
  
 verification 
  
 code 
 . 
  
 return 
  
 phoneAuthProvider 
 . 
 verifyPhoneNumber 
 ( 
  
 phoneInfoOptions 
 , 
  
 recaptchaVerifier 
 ); 
 }) 
 . 
 then 
 ( 
 function 
 ( 
 verificationId 
 ) 
  
 { 
  
 // 
  
 Ask 
  
 user 
  
 for 
  
 the 
  
 verification 
  
 code 
 . 
  
 var 
  
 cred 
  
 = 
  
 firebase 
 . 
 auth 
 . 
 PhoneAuthProvider 
 . 
 credential 
 ( 
 verificationId 
 , 
  
 verificationCode 
 ); 
  
 var 
  
 multiFactorAssertion 
  
 = 
  
 firebase 
 . 
 auth 
 . 
 PhoneMultiFactorGenerator 
 . 
 assertion 
 ( 
 cred 
 ); 
  
 // 
  
 Complete 
  
 enrollment 
 . 
  
 return 
  
 user 
 . 
 multiFactor 
 . 
 enroll 
 ( 
 multiFactorAssertion 
 , 
  
 mfaDisplayName 
 ); 
 }); 
 
 
Congratulations! You successfully registered a second authentication factor for a user.
Signing users in with a second factor
To sign in a user with two-factor SMS verification:
-  Sign the user in with their first factor, then catch the auth/multi-factor-auth-requirederror. This error contains a resolver, hints on the enrolled second factors, and an underlying session proving the user successfully authenticated with the first factor.For example, if the user's first factor was an email and password: Webimport { getAuth , signInWithEmailAndPassword , getMultiFactorResolver } from "firebase/auth" ; const auth = getAuth (); signInWithEmailAndPassword ( auth , email , password ) . then ( function ( userCredential ) { // User successfully signed in and is not enrolled with a second factor. }) . catch ( function ( error ) { if ( error . code == 'auth/multi-factor-auth-required' ) { // The user is a multi-factor user. Second factor challenge is required. resolver = getMultiFactorResolver ( auth , error ); // ... } else if ( error . code == 'auth/wrong-password' ) { // Handle other errors such as wrong password. } });Webfirebase.auth().signInWithEmailAndPassword(email, password) .then(function(userCredential) { // User successfully signed in and is not enrolled with a second factor. }) .catch(function(error) { if (error.code == 'auth/multi-factor-auth-required') { // The user is a multi-factor user. Second factor challenge is required. resolver = error.resolver; // ... } else if (error.code == 'auth/wrong-password') { // Handle other errors such as wrong password. } ... });If the user's first factor is a federated provider, such as OAuth, SAML, or OIDC, catch the error after calling signInWithPopup()orsignInWithRedirect().
-  If the user has multiple secondary factors enrolled, ask them which one to use: Web// Ask user which second factor to use. // You can get the masked phone number via resolver.hints[selectedIndex].phoneNumber // You can get the display name via resolver.hints[selectedIndex].displayName if ( resolver . hints [ selectedIndex ]. factorId === PhoneMultiFactorGenerator . FACTOR_ID ) { // User selected a phone second factor. // ... } else if ( resolver . hints [ selectedIndex ]. factorId === TotpMultiFactorGenerator . FACTOR_ID ) { // User selected a TOTP second factor. // ... } else { // Unsupported second factor. }Web// Ask user which second factor to use . // You can get the masked phone number via resolver . hints [ selectedIndex ] . phoneNumber // You can get the display name via resolver . hints [ selectedIndex ] . displayName if ( resolver . hints [ selectedIndex ] . factorId === firebase . auth . PhoneMultiFactorGenerator . FACTOR_ID ) { // User selected a phone second factor . // ... } else if ( resolver . hints [ selectedIndex ] . factorId === firebase . auth . TotpMultiFactorGenerator . FACTOR_ID ) { // User selected a TOTP second factor . // ... } else { // Unsupported second factor . }
-  Initialize the reCAPTCHA verifier as illustrated in the previous section. Skip this step if a RecaptchaVerifier instance is already configured: Webimport { RecaptchaVerifier , getAuth } from "firebase/auth" ; recaptchaVerifier = new RecaptchaVerifier ( getAuth (), 'recaptcha-container-id' , undefined );Webvar recaptchaVerifier = new firebase . auth . RecaptchaVerifier ( 'recaptcha-container-id' );
-  Initialize a PhoneInfoOptionsobject with the user's phone number and the multi-factor session. These values are contained in theresolverobject passed to theauth/multi-factor-auth-requirederror:Webconst phoneInfoOptions = { multiFactorHint : resolver . hints [ selectedIndex ], session : resolver . session };Webvar phoneInfoOptions = { multiFactorHint : resolver . hints [ selectedIndex ] , session : resolver . session } ;
-  Send a verification message to the user's phone: Web// Send SMS verification code. const phoneAuthProvider = new PhoneAuthProvider ( auth ); phoneAuthProvider . verifyPhoneNumber ( phoneInfoOptions , recaptchaVerifier ) . then ( function ( verificationId ) { // verificationId will be needed for sign-in completion. });Webvar phoneAuthProvider = new firebase . auth . PhoneAuthProvider (); // Send SMS verification code . return phoneAuthProvider . verifyPhoneNumber ( phoneInfoOptions , recaptchaVerifier ) . then ( function ( verificationId ) { // verificationId will be needed for sign - in completion . })
-  If the request fails, reset the reCAPTCHA, then repeat the previous step so the user can try again: WebrecaptchaVerifier . clear ();WebrecaptchaVerifier.clear();
-  Once the SMS code is sent, ask the user to verify the code: Webconst cred = PhoneAuthProvider . credential ( verificationId , verificationCode );Web// Ask user for the verification code . Then : var cred = firebase . auth . PhoneAuthProvider . credential ( verificationId , verificationCode );
-  Initialize a MultiFactorAssertionobject with thePhoneAuthCredential:Webconst multiFactorAssertion = PhoneMultiFactorGenerator . assertion ( cred );Webvar multiFactorAssertion = firebase . auth . PhoneMultiFactorGenerator . assertion ( cred );
-  Call resolver.resolveSignIn()to complete secondary authentication. You can then access the original sign-in result, which includes the standard provider-specific data and authentication credentials:Web// Complete sign-in. This will also trigger the Auth state listeners. resolver . resolveSignIn ( multiFactorAssertion ) . then ( function ( userCredential ) { // userCredential will also contain the user, additionalUserInfo, optional // credential (null for email/password) associated with the first factor sign-in. // For example, if the user signed in with Google as a first factor, // userCredential.additionalUserInfo will contain data related to Google // provider that the user signed in with. // - user.credential contains the Google OAuth credential. // - user.credential.accessToken contains the Google OAuth access token. // - user.credential.idToken contains the Google OAuth ID token. });Web// Complete sign-in. This will also trigger the Auth state listeners. resolver . resolveSignIn ( multiFactorAssertion ) . then ( function ( userCredential ) { // userCredential will also contain the user, additionalUserInfo, optional // credential (null for email/password) associated with the first factor sign-in. // For example, if the user signed in with Google as a first factor, // userCredential.additionalUserInfo will contain data related to Google provider that // the user signed in with. // user.credential contains the Google OAuth credential. // user.credential.accessToken contains the Google OAuth access token. // user.credential.idToken contains the Google OAuth ID token. });
The code below shows a complete example of signing in a multi-factor user:
Web
  import 
  
 { 
  
 getAuth 
 , 
  
 getMultiFactorResolver 
 , 
  
 PhoneAuthProvider 
 , 
  
 PhoneMultiFactorGenerator 
 , 
  
 RecaptchaVerifier 
 , 
  
 signInWithEmailAndPassword 
 } 
  
 from 
  
 "firebase/auth" 
 ; 
 const 
  
 recaptchaVerifier 
  
 = 
  
 new 
  
 RecaptchaVerifier 
 ( 
 getAuth 
 (), 
  
 'recaptcha-container-id' 
 , 
  
 undefined 
 ); 
 const 
  
 auth 
  
 = 
  
 getAuth 
 (); 
 signInWithEmailAndPassword 
 ( 
 auth 
 , 
  
 email 
 , 
  
 password 
 ) 
  
 . 
 then 
 ( 
 function 
  
 ( 
 userCredential 
 ) 
  
 { 
  
 // User is not enrolled with a second factor and is successfully 
  
 // signed in. 
  
 // ... 
  
 }) 
  
 . 
 catch 
 ( 
 function 
  
 ( 
 error 
 ) 
  
 { 
  
 if 
  
 ( 
 error 
 . 
 code 
  
 == 
  
 'auth/multi-factor-auth-required' 
 ) 
  
 { 
  
 const 
  
 resolver 
  
 = 
  
 getMultiFactorResolver 
 ( 
 auth 
 , 
  
 error 
 ); 
  
 // Ask user which second factor to use. 
  
 if 
  
 ( 
 resolver 
 . 
 hints 
 [ 
 selectedIndex 
 ]. 
 factorId 
  
 === 
  
 PhoneMultiFactorGenerator 
 . 
 FACTOR_ID 
 ) 
  
 { 
  
 const 
  
 phoneInfoOptions 
  
 = 
  
 { 
  
 multiFactorHint 
 : 
  
 resolver 
 . 
 hints 
 [ 
 selectedIndex 
 ], 
  
 session 
 : 
  
 resolver 
 . 
 session 
  
 }; 
  
 const 
  
 phoneAuthProvider 
  
 = 
  
 new 
  
 PhoneAuthProvider 
 ( 
 auth 
 ); 
  
 // Send SMS verification code 
  
 return 
  
 phoneAuthProvider 
 . 
 verifyPhoneNumber 
 ( 
 phoneInfoOptions 
 , 
  
 recaptchaVerifier 
 ) 
  
 . 
 then 
 ( 
 function 
  
 ( 
 verificationId 
 ) 
  
 { 
  
 // Ask user for the SMS verification code. Then: 
  
 const 
  
 cred 
  
 = 
  
 PhoneAuthProvider 
 . 
 credential 
 ( 
  
 verificationId 
 , 
  
 verificationCode 
 ); 
  
 const 
  
 multiFactorAssertion 
  
 = 
  
 PhoneMultiFactorGenerator 
 . 
 assertion 
 ( 
 cred 
 ); 
  
 // Complete sign-in. 
  
 return 
  
 resolver 
 . 
 resolveSignIn 
 ( 
 multiFactorAssertion 
 ) 
  
 }) 
  
 . 
 then 
 ( 
 function 
  
 ( 
 userCredential 
 ) 
  
 { 
  
 // User successfully signed in with the second factor phone number. 
  
 }); 
  
 } 
  
 else 
  
 if 
  
 ( 
 resolver 
 . 
 hints 
 [ 
 selectedIndex 
 ]. 
 factorId 
  
 === 
  
 TotpMultiFactorGenerator 
 . 
 FACTOR_ID 
 ) 
  
 { 
  
 // Handle TOTP MFA. 
  
 // ... 
  
 } 
  
 else 
  
 { 
  
 // Unsupported second factor. 
  
 } 
  
 } 
  
 else 
  
 if 
  
 ( 
 error 
 . 
 code 
  
 == 
  
 'auth/wrong-password' 
 ) 
  
 { 
  
 // Handle other errors such as wrong password. 
  
 } 
  
 }); 
 
 
Web
  var 
  
 resolver 
 ; 
 firebase 
 . 
 auth 
 (). 
 signInWithEmailAndPassword 
 ( 
 email 
 , 
  
 password 
 ) 
  
 . 
 then 
 ( 
 function 
 ( 
 userCredential 
 ) 
  
 { 
  
 // 
  
 User 
  
 is 
  
 not 
  
 enrolled 
  
 with 
  
 a 
  
 second 
  
 factor 
  
 and 
  
 is 
  
 successfully 
  
 signed 
  
 in 
 . 
  
 // 
  
 ... 
  
 } 
 ) 
  
 . 
 catch 
 ( 
 function 
 ( 
 error 
 ) 
  
 { 
  
 if 
  
 ( 
 error 
 . 
 code 
  
 == 
  
 'auth/multi-factor-auth-required' 
 ) 
  
 { 
  
 resolver 
  
 = 
  
 error 
 . 
 resolver 
 ; 
  
 // 
  
 Ask 
  
 user 
  
 which 
  
 second 
  
 factor 
  
 to 
  
 use 
 . 
  
 if 
  
 ( 
 resolver 
 . 
 hints 
 [ 
 selectedIndex 
 ] 
 . 
 factorId 
  
 === 
  
 firebase 
 . 
 auth 
 . 
 PhoneMultiFactorGenerator 
 . 
 FACTOR_ID 
 ) 
  
 { 
  
 var 
  
 phoneInfoOptions 
  
 = 
  
 { 
  
 multiFactorHint 
 : 
  
 resolver 
 . 
 hints 
 [ 
 selectedIndex 
 ] 
 , 
  
 session 
 : 
  
 resolver 
 . 
 session 
  
 } 
 ; 
  
 var 
  
 phoneAuthProvider 
  
 = 
  
 new 
  
 firebase 
 . 
 auth 
 . 
 PhoneAuthProvider 
 (); 
  
 // 
  
 Send 
  
 SMS 
  
 verification 
  
 code 
  
 return 
  
 phoneAuthProvider 
 . 
 verifyPhoneNumber 
 ( 
 phoneInfoOptions 
 , 
  
 recaptchaVerifier 
 ) 
  
 . 
 then 
 ( 
 function 
 ( 
 verificationId 
 ) 
  
 { 
  
 // 
  
 Ask 
  
 user 
  
 for 
  
 the 
  
 SMS 
  
 verification 
  
 code 
 . 
  
 var 
  
 cred 
  
 = 
  
 firebase 
 . 
 auth 
 . 
 PhoneAuthProvider 
 . 
 credential 
 ( 
  
 verificationId 
 , 
  
 verificationCode 
 ); 
  
 var 
  
 multiFactorAssertion 
  
 = 
  
 firebase 
 . 
 auth 
 . 
 PhoneMultiFactorGenerator 
 . 
 assertion 
 ( 
 cred 
 ); 
  
 // 
  
 Complete 
  
 sign 
 - 
 in 
 . 
  
 return 
  
 resolver 
 . 
 resolveSignIn 
 ( 
 multiFactorAssertion 
 ) 
  
 } 
 ) 
  
 . 
 then 
 ( 
 function 
 ( 
 userCredential 
 ) 
  
 { 
  
 // 
  
 User 
  
 successfully 
  
 signed 
  
 in 
  
 with 
  
 the 
  
 second 
  
 factor 
  
 phone 
  
 number 
 . 
  
 } 
 ); 
  
 } 
  
 else 
  
 if 
  
 ( 
 resolver 
 . 
 hints 
 [ 
 selectedIndex 
 ] 
 . 
 factorId 
  
 === 
  
 firebase 
 . 
 auth 
 . 
 TotpMultiFactorGenerator 
 . 
 FACTOR_ID 
 ) 
  
 { 
  
 // 
  
 Handle 
  
 TOTP 
  
 MFA 
 . 
  
 // 
  
 ... 
  
 } 
  
 else 
  
 { 
  
 // 
  
 Unsupported 
  
 second 
  
 factor 
 . 
  
 } 
  
 } 
  
 else 
  
 if 
  
 ( 
 error 
 . 
 code 
  
 == 
  
 'auth/wrong-password' 
 ) 
  
 { 
  
 // 
  
 Handle 
  
 other 
  
 errors 
  
 such 
  
 as 
  
 wrong 
  
 password 
 . 
  
 } 
  
 ... 
  
 } 
 ); 
 
 
Congratulations! You successfully signed in a user using multi-factor authentication.
What's next
- Manage multi-factor users programmatically with the Admin SDK .

