FIRMultiFactor.m 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. /*
  2. * Copyright 2019 Google
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import <TargetConditionals.h>
  17. #if TARGET_OS_IOS
  18. #import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRMultiFactor.h"
  19. #import "FirebaseAuth/Sources/Auth/FIRAuthDataResult_Internal.h"
  20. #import "FirebaseAuth/Sources/Auth/FIRAuth_Internal.h"
  21. #import "FirebaseAuth/Sources/Backend/FIRAuthBackend+MultiFactor.h"
  22. #import "FirebaseAuth/Sources/Backend/RPC/MultiFactor/Enroll/FIRStartMFAEnrollmentRequest.h"
  23. #import "FirebaseAuth/Sources/MultiFactor/FIRMultiFactor+Internal.h"
  24. #import "FirebaseAuth/Sources/MultiFactor/FIRMultiFactorInfo+Internal.h"
  25. #import "FirebaseAuth/Sources/MultiFactor/FIRMultiFactorSession+Internal.h"
  26. #import "FirebaseAuth/Sources/User/FIRUser_Internal.h"
  27. #if TARGET_OS_IOS
  28. #import "FirebaseAuth/Sources/AuthProvider/Phone/FIRPhoneAuthCredential_Internal.h"
  29. #import "FirebaseAuth/Sources/MultiFactor/Phone/FIRPhoneMultiFactorAssertion+Internal.h"
  30. #import "FirebaseAuth/Sources/MultiFactor/Phone/FIRPhoneMultiFactorInfo+Internal.h"
  31. #import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRPhoneMultiFactorAssertion.h"
  32. #import "FirebaseAuth/Sources/MultiFactor/TOTP/FIRTOTPMultiFactorAssertion+Internal.h"
  33. #import "FirebaseAuth/Sources/MultiFactor/TOTP/FIRTOTPMultiFactorInfo.h"
  34. #import "FirebaseAuth/Sources/MultiFactor/TOTP/FIRTOTPSecret+Internal.h"
  35. #import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRTOTPMultiFactorAssertion.h"
  36. #import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRTOTPMultiFactorGenerator.h"
  37. #import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRTOTPSecret.h"
  38. #endif
  39. NS_ASSUME_NONNULL_BEGIN
  40. static NSString *kEnrolledFactorsCodingKey = @"enrolledFactors";
  41. static NSString *kUserCodingKey = @"user";
  42. @implementation FIRMultiFactor
  43. - (void)getSessionWithCompletion:(nullable FIRMultiFactorSessionCallback)completion {
  44. FIRMultiFactorSession *session = [FIRMultiFactorSession sessionForCurrentUser];
  45. if (completion) {
  46. completion(session, nil);
  47. }
  48. }
  49. - (void)enrollWithAssertion:(FIRMultiFactorAssertion *)assertion
  50. displayName:(nullable NSString *)displayName
  51. completion:(nullable FIRAuthVoidErrorCallback)completion {
  52. #if TARGET_OS_IOS
  53. FIRFinalizeMFAEnrollmentRequest *request = nil;
  54. if ([assertion.factorID isEqualToString:FIRPhoneMultiFactorID]) {
  55. FIRPhoneMultiFactorAssertion *phoneAssertion = (FIRPhoneMultiFactorAssertion *)assertion;
  56. FIRAuthProtoFinalizeMFAPhoneRequestInfo *finalizeMFAPhoneRequestInfo =
  57. [[FIRAuthProtoFinalizeMFAPhoneRequestInfo alloc]
  58. initWithSessionInfo:phoneAssertion.authCredential.verificationID
  59. verificationCode:phoneAssertion.authCredential.verificationCode];
  60. request =
  61. [[FIRFinalizeMFAEnrollmentRequest alloc] initWithIDToken:self.user.rawAccessToken
  62. displayName:displayName
  63. phoneVerificationInfo:finalizeMFAPhoneRequestInfo
  64. requestConfiguration:self.user.requestConfiguration];
  65. } else if ([assertion.factorID isEqualToString:FIRTOTPMultiFactorID]) {
  66. FIRTOTPMultiFactorAssertion *TOTPAssertion = (FIRTOTPMultiFactorAssertion *)assertion;
  67. FIRAuthProtoFinalizeMFATOTPEnrollmentRequestInfo *finalizeMFATOTPRequestInfo =
  68. [[FIRAuthProtoFinalizeMFATOTPEnrollmentRequestInfo alloc]
  69. initWithSessionInfo:TOTPAssertion.secret.sessionInfo
  70. verificationCode:TOTPAssertion.oneTimePassword];
  71. request =
  72. [[FIRFinalizeMFAEnrollmentRequest alloc] initWithIDToken:self.user.rawAccessToken
  73. displayName:displayName
  74. TOTPVerificationInfo:finalizeMFATOTPRequestInfo
  75. requestConfiguration:self.user.requestConfiguration];
  76. }
  77. if (request == nil) {
  78. return;
  79. }
  80. [FIRAuthBackend
  81. finalizeMultiFactorEnrollment:request
  82. callback:^(FIRFinalizeMFAEnrollmentResponse *_Nullable response,
  83. NSError *_Nullable error) {
  84. if (error) {
  85. if (completion) {
  86. completion(error);
  87. }
  88. } else {
  89. [self.user.auth
  90. completeSignInWithAccessToken:response.IDToken
  91. accessTokenExpirationDate:nil
  92. refreshToken:response.refreshToken
  93. anonymous:NO
  94. callback:^(FIRUser *_Nullable user,
  95. NSError *_Nullable error) {
  96. FIRAuthDataResult *result =
  97. [[FIRAuthDataResult alloc]
  98. initWithUser:user
  99. additionalUserInfo:nil];
  100. FIRAuthDataResultCallback
  101. decoratedCallback = [self.user.auth
  102. signInFlowAuthDataResultCallbackByDecoratingCallback:
  103. ^(FIRAuthDataResult
  104. *_Nullable authResult,
  105. NSError *_Nullable error) {
  106. if (completion) {
  107. completion(error);
  108. }
  109. }];
  110. decoratedCallback(result, error);
  111. }];
  112. }
  113. }];
  114. #endif
  115. }
  116. - (void)unenrollWithInfo:(FIRMultiFactorInfo *)factorInfo
  117. completion:(nullable FIRAuthVoidErrorCallback)completion {
  118. [self unenrollWithFactorUID:factorInfo.UID completion:completion];
  119. }
  120. - (void)unenrollWithFactorUID:(NSString *)factorUID
  121. completion:(nullable FIRAuthVoidErrorCallback)completion {
  122. FIRWithdrawMFARequest *request =
  123. [[FIRWithdrawMFARequest alloc] initWithIDToken:self.user.rawAccessToken
  124. MFAEnrollmentID:factorUID
  125. requestConfiguration:self.user.requestConfiguration];
  126. [FIRAuthBackend
  127. withdrawMultiFactor:request
  128. callback:^(FIRWithdrawMFAResponse *_Nullable response, NSError *_Nullable error) {
  129. if (error) {
  130. if (completion) {
  131. completion(error);
  132. }
  133. } else {
  134. [self.user.auth
  135. completeSignInWithAccessToken:response.IDToken
  136. accessTokenExpirationDate:nil
  137. refreshToken:response.refreshToken
  138. anonymous:NO
  139. callback:^(FIRUser *_Nullable user,
  140. NSError *_Nullable error) {
  141. FIRAuthDataResult *result =
  142. [[FIRAuthDataResult alloc] initWithUser:user
  143. additionalUserInfo:nil];
  144. FIRAuthDataResultCallback decoratedCallback = [FIRAuth
  145. .auth
  146. signInFlowAuthDataResultCallbackByDecoratingCallback:
  147. ^(FIRAuthDataResult *_Nullable authResult,
  148. NSError *_Nullable error) {
  149. if (error) {
  150. [[FIRAuth auth] signOut:NULL];
  151. }
  152. if (completion) {
  153. completion(error);
  154. }
  155. }];
  156. decoratedCallback(result, error);
  157. }];
  158. }
  159. }];
  160. }
  161. #pragma mark - Internal
  162. - (instancetype)initWithMFAEnrollments:(NSArray<FIRAuthProtoMFAEnrollment *> *)MFAEnrollments {
  163. self = [super init];
  164. if (self) {
  165. NSMutableArray<FIRMultiFactorInfo *> *multiFactorInfoArray = [[NSMutableArray alloc] init];
  166. for (FIRAuthProtoMFAEnrollment *MFAEnrollment in MFAEnrollments) {
  167. if (MFAEnrollment.phoneInfo) {
  168. FIRMultiFactorInfo *multiFactorInfo =
  169. [[FIRPhoneMultiFactorInfo alloc] initWithProto:MFAEnrollment];
  170. [multiFactorInfoArray addObject:multiFactorInfo];
  171. }
  172. if (MFAEnrollment.TOTPInfo) {
  173. FIRMultiFactorInfo *multiFactorInfo =
  174. [[FIRTOTPMultiFactorInfo alloc] initWithProto:MFAEnrollment];
  175. [multiFactorInfoArray addObject:multiFactorInfo];
  176. }
  177. }
  178. _enrolledFactors = [multiFactorInfoArray copy];
  179. }
  180. return self;
  181. }
  182. #pragma mark - NSSecureCoding
  183. + (BOOL)supportsSecureCoding {
  184. return YES;
  185. }
  186. - (nullable instancetype)initWithCoder:(NSCoder *)aDecoder {
  187. self = [self init];
  188. if (self) {
  189. NSSet *enrolledFactorsClasses = [NSSet setWithArray:@[
  190. [NSArray class], [FIRMultiFactorInfo class], [FIRPhoneMultiFactorInfo class],
  191. [FIRTOTPMultiFactorInfo class]
  192. ]];
  193. NSArray<FIRMultiFactorInfo *> *enrolledFactors =
  194. [aDecoder decodeObjectOfClasses:enrolledFactorsClasses forKey:kEnrolledFactorsCodingKey];
  195. _enrolledFactors = enrolledFactors;
  196. // Do not decode `user` weak property.
  197. }
  198. return self;
  199. }
  200. - (void)encodeWithCoder:(NSCoder *)aCoder {
  201. [aCoder encodeObject:_enrolledFactors forKey:kEnrolledFactorsCodingKey];
  202. // Do not encode `user` weak property.
  203. }
  204. @end
  205. NS_ASSUME_NONNULL_END
  206. #endif