FIRMultiFactor.m 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  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/MultiFactor/FIRMultiFactor+Internal.h"
  23. #import "FirebaseAuth/Sources/User/FIRUser_Internal.h"
  24. #import "FirebaseAuth-Swift.h"
  25. NS_ASSUME_NONNULL_BEGIN
  26. static NSString *kEnrolledFactorsCodingKey = @"enrolledFactors";
  27. static NSString *kUserCodingKey = @"user";
  28. @implementation FIRMultiFactor
  29. - (void)getSessionWithCompletion:(nullable FIRMultiFactorSessionCallback)completion {
  30. FIRMultiFactorSession *session = [FIRMultiFactorSession sessionForCurrentUser];
  31. if (completion) {
  32. completion(session, nil);
  33. }
  34. }
  35. - (void)enrollWithAssertion:(FIRMultiFactorAssertion *)assertion
  36. displayName:(nullable NSString *)displayName
  37. completion:(nullable FIRAuthVoidErrorCallback)completion {
  38. #if TARGET_OS_IOS
  39. FIRFinalizeMFAEnrollmentRequest *request = nil;
  40. if ([assertion.factorID isEqualToString:FIRPhoneMultiFactorID]) {
  41. FIRPhoneMultiFactorAssertion *phoneAssertion = (FIRPhoneMultiFactorAssertion *)assertion;
  42. FIRAuthProtoFinalizeMFAPhoneRequestInfo *finalizeMFAPhoneRequestInfo =
  43. [[FIRAuthProtoFinalizeMFAPhoneRequestInfo alloc]
  44. initWithSessionInfo:phoneAssertion.authCredential.verificationID
  45. verificationCode:phoneAssertion.authCredential.verificationCode];
  46. request =
  47. [[FIRFinalizeMFAEnrollmentRequest alloc] initWithIDToken:self.user.rawAccessToken
  48. displayName:displayName
  49. phoneVerificationInfo:finalizeMFAPhoneRequestInfo
  50. requestConfiguration:self.user.requestConfiguration];
  51. } else if ([assertion.factorID isEqualToString:FIRTOTPMultiFactorID]) {
  52. FIRTOTPMultiFactorAssertion *TOTPAssertion = (FIRTOTPMultiFactorAssertion *)assertion;
  53. FIRAuthProtoFinalizeMFATOTPEnrollmentRequestInfo *finalizeMFATOTPRequestInfo =
  54. [[FIRAuthProtoFinalizeMFATOTPEnrollmentRequestInfo alloc]
  55. initWithSessionInfo:TOTPAssertion.secret.sessionInfo
  56. verificationCode:TOTPAssertion.oneTimePassword];
  57. request =
  58. [[FIRFinalizeMFAEnrollmentRequest alloc] initWithIDToken:self.user.rawAccessToken
  59. displayName:displayName
  60. TOTPVerificationInfo:finalizeMFATOTPRequestInfo
  61. requestConfiguration:self.user.requestConfiguration];
  62. }
  63. if (request == nil) {
  64. return;
  65. }
  66. [FIRAuthBackend
  67. finalizeMultiFactorEnrollment:request
  68. callback:^(FIRFinalizeMFAEnrollmentResponse *_Nullable response,
  69. NSError *_Nullable error) {
  70. if (error) {
  71. if (completion) {
  72. completion(error);
  73. }
  74. } else {
  75. [FIRAuth.auth
  76. completeSignInWithAccessToken:response.IDToken
  77. accessTokenExpirationDate:nil
  78. refreshToken:response.refreshToken
  79. anonymous:NO
  80. callback:^(FIRUser *_Nullable user,
  81. NSError *_Nullable error) {
  82. FIRAuthDataResult *result =
  83. [[FIRAuthDataResult alloc]
  84. initWithUser:user
  85. additionalUserInfo:nil];
  86. FIRAuthDataResultCallback
  87. decoratedCallback = [FIRAuth.auth
  88. signInFlowAuthDataResultCallbackByDecoratingCallback:
  89. ^(FIRAuthDataResult
  90. *_Nullable authResult,
  91. NSError *_Nullable error) {
  92. if (completion) {
  93. completion(error);
  94. }
  95. }];
  96. decoratedCallback(result, error);
  97. }];
  98. }
  99. }];
  100. #endif
  101. }
  102. - (void)unenrollWithInfo:(FIRMultiFactorInfo *)factorInfo
  103. completion:(nullable FIRAuthVoidErrorCallback)completion {
  104. [self unenrollWithFactorUID:factorInfo.UID completion:completion];
  105. }
  106. - (void)unenrollWithFactorUID:(NSString *)factorUID
  107. completion:(nullable FIRAuthVoidErrorCallback)completion {
  108. FIRWithdrawMFARequest *request =
  109. [[FIRWithdrawMFARequest alloc] initWithIDToken:self.user.rawAccessToken
  110. MFAEnrollmentID:factorUID
  111. requestConfiguration:self.user.requestConfiguration];
  112. [FIRAuthBackend
  113. withdrawMultiFactor:request
  114. callback:^(FIRWithdrawMFAResponse *_Nullable response, NSError *_Nullable error) {
  115. if (error) {
  116. if (completion) {
  117. completion(error);
  118. }
  119. } else {
  120. [FIRAuth.auth
  121. completeSignInWithAccessToken:response.IDToken
  122. accessTokenExpirationDate:nil
  123. refreshToken:response.refreshToken
  124. anonymous:NO
  125. callback:^(FIRUser *_Nullable user,
  126. NSError *_Nullable error) {
  127. FIRAuthDataResult *result =
  128. [[FIRAuthDataResult alloc] initWithUser:user
  129. additionalUserInfo:nil];
  130. FIRAuthDataResultCallback decoratedCallback = [FIRAuth
  131. .auth
  132. signInFlowAuthDataResultCallbackByDecoratingCallback:
  133. ^(FIRAuthDataResult *_Nullable authResult,
  134. NSError *_Nullable error) {
  135. if (error) {
  136. [[FIRAuth auth] signOut:NULL];
  137. }
  138. if (completion) {
  139. completion(error);
  140. }
  141. }];
  142. decoratedCallback(result, error);
  143. }];
  144. }
  145. }];
  146. }
  147. #pragma mark - Internal
  148. - (instancetype)initWithMFAEnrollments:(NSArray<FIRAuthProtoMFAEnrollment *> *)MFAEnrollments {
  149. self = [super init];
  150. if (self) {
  151. NSMutableArray<FIRMultiFactorInfo *> *multiFactorInfoArray = [[NSMutableArray alloc] init];
  152. for (FIRAuthProtoMFAEnrollment *MFAEnrollment in MFAEnrollments) {
  153. if (MFAEnrollment.phoneInfo) {
  154. FIRMultiFactorInfo *multiFactorInfo =
  155. [[FIRPhoneMultiFactorInfo alloc] initWithProto:MFAEnrollment];
  156. [multiFactorInfoArray addObject:multiFactorInfo];
  157. }
  158. if (MFAEnrollment.TOTPInfo) {
  159. FIRMultiFactorInfo *multiFactorInfo =
  160. [[FIRTOTPMultiFactorInfo alloc] initWithProto:MFAEnrollment];
  161. [multiFactorInfoArray addObject:multiFactorInfo];
  162. }
  163. }
  164. _enrolledFactors = [multiFactorInfoArray copy];
  165. }
  166. return self;
  167. }
  168. #pragma mark - NSSecureCoding
  169. + (BOOL)supportsSecureCoding {
  170. return YES;
  171. }
  172. - (nullable instancetype)initWithCoder:(NSCoder *)aDecoder {
  173. self = [self init];
  174. if (self) {
  175. NSSet *enrolledFactorsClasses = [NSSet setWithArray:@[
  176. [NSArray class], [FIRMultiFactorInfo class], [FIRPhoneMultiFactorInfo class],
  177. [FIRTOTPMultiFactorInfo class]
  178. ]];
  179. NSArray<FIRMultiFactorInfo *> *enrolledFactors =
  180. [aDecoder decodeObjectOfClasses:enrolledFactorsClasses forKey:kEnrolledFactorsCodingKey];
  181. _enrolledFactors = enrolledFactors;
  182. // Do not decode `user` weak property.
  183. }
  184. return self;
  185. }
  186. - (void)encodeWithCoder:(NSCoder *)aCoder {
  187. [aCoder encodeObject:_enrolledFactors forKey:kEnrolledFactorsCodingKey];
  188. // Do not encode `user` weak property.
  189. }
  190. @end
  191. NS_ASSUME_NONNULL_END
  192. #endif