FIRFinalizePasskeySignInResponseTests.m 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. /*
  2. * Copyright 2023 Google LLC
  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/LICENSE-2.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 <XCTest/XCTest.h>
  17. #if TARGET_OS_IOS || TARGET_OS_TV || TARGET_OS_OSX || TARGET_OS_MACCATALYST
  18. #import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRAuthErrors.h"
  19. #import "FirebaseAuth/Sources/Backend/FIRAuthBackend.h"
  20. #import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeySignInRequest.h"
  21. #import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeySignInResponse.h"
  22. #import "FirebaseAuth/Sources/Utilities/FIRAuthInternalErrors.h"
  23. #import "FirebaseAuth/Tests/Unit/FIRFakeBackendRPCIssuer.h"
  24. /**
  25. @var kTestAPIKey
  26. @brief Fake API key used for testing.
  27. */
  28. static NSString *const kTestAPIKey = @"APIKey";
  29. /**
  30. @var kTestFirebaseAppID
  31. @brief Fake Firebase app ID used for testing.
  32. */
  33. static NSString *const kTestFirebaseAppID = @"appID";
  34. /**
  35. @var kIDToken
  36. @brief Token representing the user's identity.
  37. */
  38. static NSString *const kIDToken = @"idToken";
  39. /**
  40. @var kRefreshToken
  41. @brief Refresh Token
  42. */
  43. static NSString *const kRefreshToken = @"refreshToken";
  44. /**
  45. @var kCredentialID
  46. @brief credential ID.
  47. */
  48. static NSString *const kCredentialID = @"testCredentialID";
  49. /**
  50. @var kRawClientDataJSON
  51. @brief CollectedClientData object from the authenticator.
  52. */
  53. static NSString *const kRawClientDataJSON = @"testRawClientDataJSON";
  54. /**
  55. @var kAuthenticatorData
  56. @brief The authenticatorData from the authenticator.
  57. */
  58. static NSString *const kAuthenticatorData = @"TestAuthenticatorData";
  59. /**
  60. @var kSignature
  61. @brief The signature from the authenticator
  62. */
  63. static NSString *const kSignature = @"testSignature";
  64. /**
  65. @var kUserHandle
  66. @brief The key for the user handle.
  67. */
  68. static NSString *const kUserHandle = @"testUserHandle";
  69. /**
  70. @class FIRFinalizePasskeySignInResponseTests
  71. @brief Tests for @c FIRFinalizePasskeySignInResponse.
  72. */
  73. @interface FIRFinalizePasskeySignInResponseTests : XCTestCase
  74. @end
  75. @implementation FIRFinalizePasskeySignInResponseTests {
  76. /**
  77. @brief This backend RPC issuer is used to fake network responses for each test in the suite.
  78. In the @c setUp method we initialize this and set @c FIRAuthBackend's RPC issuer to it.
  79. */
  80. FIRFakeBackendRPCIssuer *_RPCIssuer;
  81. /**
  82. @brief This is the request configuration used for testing.
  83. */
  84. FIRAuthRequestConfiguration *_requestConfiguration;
  85. }
  86. - (void)setUp {
  87. [super setUp];
  88. FIRFakeBackendRPCIssuer *RPCIssuer = [[FIRFakeBackendRPCIssuer alloc] init];
  89. [FIRAuthBackend setDefaultBackendImplementationWithRPCIssuer:RPCIssuer];
  90. _RPCIssuer = RPCIssuer;
  91. _requestConfiguration = [[FIRAuthRequestConfiguration alloc] initWithAPIKey:kTestAPIKey
  92. appID:kTestFirebaseAppID];
  93. }
  94. - (void)tearDown {
  95. _RPCIssuer = nil;
  96. _requestConfiguration = nil;
  97. [FIRAuthBackend setDefaultBackendImplementationWithRPCIssuer:nil];
  98. [super tearDown];
  99. }
  100. /** @fn testSuccessfulFinalizePasskeySignInResponse
  101. @brief This test simulates a successful @c FinalizePasskeySignin flow.
  102. */
  103. - (void)testSuccessfulFinalizePasskeySignInResponse {
  104. if (@available(iOS 15.0, *)) {
  105. FIRFinalizePasskeySignInRequest *request =
  106. [[FIRFinalizePasskeySignInRequest alloc] initWithCredentialID:kCredentialID
  107. clientDataJson:kRawClientDataJSON
  108. authenticatorData:kAuthenticatorData
  109. signature:kSignature
  110. userID:kUserHandle
  111. requestConfiguration:_requestConfiguration];
  112. __block BOOL callbackInvoked;
  113. __block FIRFinalizePasskeySignInResponse *RPCResponse;
  114. __block NSError *RPCError;
  115. [FIRAuthBackend finalizePasskeySignIn:request
  116. callback:^(FIRFinalizePasskeySignInResponse *_Nullable response,
  117. NSError *_Nullable error) {
  118. callbackInvoked = YES;
  119. RPCResponse = response;
  120. RPCError = error;
  121. }];
  122. [_RPCIssuer respondWithJSON:@{
  123. @"idToken" : kIDToken,
  124. @"refreshToken" : kRefreshToken,
  125. }];
  126. XCTAssert(callbackInvoked);
  127. XCTAssertNil(RPCError);
  128. XCTAssertNotNil(RPCResponse);
  129. XCTAssertEqualObjects(RPCResponse.idToken, kIDToken);
  130. XCTAssertEqualObjects(RPCResponse.refreshToken, kRefreshToken);
  131. }
  132. }
  133. /** @fn testFinalizePasskeySignInResponseMissingIDTokenError
  134. @brief This test simulates an unexpected response returned from server in @c
  135. FinalizePasskeySignIn flow.
  136. */
  137. - (void)testFinalizePasskeySignInResponseMissingIDTokenError {
  138. if (@available(iOS 15.0, *)) {
  139. FIRFinalizePasskeySignInRequest *request =
  140. [[FIRFinalizePasskeySignInRequest alloc] initWithCredentialID:kCredentialID
  141. clientDataJson:kRawClientDataJSON
  142. authenticatorData:kAuthenticatorData
  143. signature:kSignature
  144. userID:kUserHandle
  145. requestConfiguration:_requestConfiguration];
  146. __block BOOL callbackInvoked;
  147. __block FIRFinalizePasskeySignInResponse *RPCResponse;
  148. __block NSError *RPCError;
  149. [FIRAuthBackend finalizePasskeySignIn:request
  150. callback:^(FIRFinalizePasskeySignInResponse *_Nullable response,
  151. NSError *_Nullable error) {
  152. callbackInvoked = YES;
  153. RPCResponse = response;
  154. RPCError = error;
  155. }];
  156. [_RPCIssuer respondWithJSON:@{
  157. @"wrongkey" : @{},
  158. @"refreshToken" : kRefreshToken,
  159. }];
  160. [self errorValidationHelperWithCallbackInvoked:callbackInvoked
  161. rpcError:RPCError
  162. rpcResponse:RPCResponse];
  163. }
  164. }
  165. /** @fn testFinalizePasskeySignInResponseMissingRefreshTokenError
  166. @brief This test simulates an unexpected response returned from server in @c
  167. FinalizePasskeySignIn flow.
  168. */
  169. - (void)testFinalizePasskeySignInResponseMissingRefreshTokenError {
  170. if (@available(iOS 15.0, *)) {
  171. FIRFinalizePasskeySignInRequest *request =
  172. [[FIRFinalizePasskeySignInRequest alloc] initWithCredentialID:kCredentialID
  173. clientDataJson:kRawClientDataJSON
  174. authenticatorData:kAuthenticatorData
  175. signature:kSignature
  176. userID:kUserHandle
  177. requestConfiguration:_requestConfiguration];
  178. __block BOOL callbackInvoked;
  179. __block FIRFinalizePasskeySignInResponse *RPCResponse;
  180. __block NSError *RPCError;
  181. [FIRAuthBackend finalizePasskeySignIn:request
  182. callback:^(FIRFinalizePasskeySignInResponse *_Nullable response,
  183. NSError *_Nullable error) {
  184. callbackInvoked = YES;
  185. RPCResponse = response;
  186. RPCError = error;
  187. }];
  188. [_RPCIssuer respondWithJSON:@{
  189. @"wrongkey" : @{},
  190. @"idToken" : kIDToken,
  191. }];
  192. [self errorValidationHelperWithCallbackInvoked:callbackInvoked
  193. rpcError:RPCError
  194. rpcResponse:RPCResponse];
  195. }
  196. }
  197. /** @fn errorValidationHelperWithCallbackInvoked:rpcError:rpcResponse:
  198. @brief Helper function to validate the unexpected response returned from server in @c
  199. FinalizePasskeySignIn flow.
  200. */
  201. - (void)errorValidationHelperWithCallbackInvoked:(BOOL)callbackInvoked
  202. rpcError:(NSError *)RPCError
  203. rpcResponse:(FIRFinalizePasskeySignInResponse *)RPCResponse {
  204. XCTAssert(callbackInvoked);
  205. XCTAssertNotNil(RPCError);
  206. XCTAssertEqualObjects(RPCError.domain, FIRAuthErrorDomain);
  207. XCTAssertEqual(RPCError.code, FIRAuthErrorCodeInternalError);
  208. XCTAssertNotNil(RPCError.userInfo[NSUnderlyingErrorKey]);
  209. NSError *underlyingError = RPCError.userInfo[NSUnderlyingErrorKey];
  210. XCTAssertNotNil(underlyingError);
  211. XCTAssertNotNil(underlyingError.userInfo[FIRAuthErrorUserInfoDeserializedResponseKey]);
  212. XCTAssertNil(RPCResponse);
  213. }
  214. @end
  215. #endif