VerifyPhoneNumberTests.swift 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. // Copyright 2023 Google LLC
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. import Foundation
  15. import XCTest
  16. @testable import FirebaseAuth
  17. #if os(iOS)
  18. @available(iOS 13, tvOS 13, macOS 10.15, macCatalyst 13, watchOS 7, *)
  19. class VerifyPhoneNumberTests: RPCBaseTests {
  20. private let kVerificationCode = "12345678"
  21. private let kVerificationID = "55432"
  22. private let kPhoneNumber = "4155551234"
  23. private let kTemporaryProof = "12345658"
  24. private let kVerificationCodeKey = "code"
  25. private let kVerificationIDKey = "sessionInfo"
  26. private let kIDTokenKey = "idToken"
  27. private let kOperationKey = "operation"
  28. private let kTestAccessToken = "accessToken"
  29. private let kTemporaryProofKey = "temporaryProof"
  30. private let kPhoneNumberKey = "phoneNumber"
  31. private let kExpectedAPIURL =
  32. "https://www.googleapis.com/identitytoolkit/v3/relyingparty/verifyPhoneNumber?key=APIKey"
  33. /** @fn testVerifyPhoneNumberRequest
  34. @brief Tests the verifyPhoneNumber request.
  35. */
  36. func testVerifyPhoneNumberRequest() async throws {
  37. let request = makeVerifyPhoneNumberRequest()
  38. request.accessToken = kTestAccessToken
  39. try await checkRequest(
  40. request: request,
  41. expected: kExpectedAPIURL,
  42. key: kVerificationIDKey,
  43. value: kVerificationID
  44. )
  45. let requestDictionary = try XCTUnwrap(rpcIssuer.decodedRequest as? [String: AnyHashable])
  46. XCTAssertEqual(requestDictionary[kVerificationCodeKey], kVerificationCode)
  47. XCTAssertEqual(requestDictionary[kIDTokenKey], kTestAccessToken)
  48. XCTAssertEqual(
  49. requestDictionary[kOperationKey],
  50. AuthOperationType.signUpOrSignIn.operationString
  51. )
  52. }
  53. /** @fn testVerifyPhoneNumberRequestWithTemporaryProof
  54. @brief Tests the verifyPhoneNumber request when created using a temporary proof.
  55. */
  56. func testVerifyPhoneNumberRequestWithTemporaryProof() async throws {
  57. let request = makeVerifyPhoneNumberRequestWithTemporaryProof()
  58. request.accessToken = kTestAccessToken
  59. try await checkRequest(
  60. request: request,
  61. expected: kExpectedAPIURL,
  62. key: kTemporaryProofKey,
  63. value: kTemporaryProof
  64. )
  65. let requestDictionary = try XCTUnwrap(rpcIssuer.decodedRequest as? [String: AnyHashable])
  66. XCTAssertEqual(requestDictionary[kPhoneNumberKey], kPhoneNumber)
  67. XCTAssertEqual(requestDictionary[kIDTokenKey], kTestAccessToken)
  68. XCTAssertEqual(
  69. requestDictionary[kOperationKey],
  70. AuthOperationType.signUpOrSignIn.operationString
  71. )
  72. }
  73. func testVerifyPhoneNumberRequestErrors() async throws {
  74. let kInvalidVerificationCodeErrorMessage = "INVALID_CODE"
  75. let kInvalidSessionInfoErrorMessage = "INVALID_SESSION_INFO"
  76. let kSessionExpiredErrorMessage = "SESSION_EXPIRED"
  77. try await checkBackendError(
  78. request: makeVerifyPhoneNumberRequest(),
  79. message: kInvalidVerificationCodeErrorMessage,
  80. errorCode: AuthErrorCode.invalidVerificationCode
  81. )
  82. try await checkBackendError(
  83. request: makeVerifyPhoneNumberRequest(),
  84. message: kInvalidSessionInfoErrorMessage,
  85. errorCode: AuthErrorCode.invalidVerificationID
  86. )
  87. try await checkBackendError(
  88. request: makeVerifyPhoneNumberRequest(),
  89. message: kSessionExpiredErrorMessage,
  90. errorCode: AuthErrorCode.sessionExpired
  91. )
  92. }
  93. /** @fn testSuccessfulVerifyPhoneNumberResponse
  94. @brief Tests a successful to verify phone number flow.
  95. */
  96. func testSuccessfulVerifyPhoneNumberResponse() async throws {
  97. let kTestLocalID = "testLocalId"
  98. let kTestIDToken = "ID_TOKEN"
  99. let kTestExpiresIn = "12345"
  100. let kTestRefreshToken = "REFRESH_TOKEN"
  101. rpcIssuer.respondBlock = {
  102. try self.rpcIssuer?.respond(withJSON: [
  103. "idToken": kTestIDToken,
  104. "refreshToken": kTestRefreshToken,
  105. "localId": kTestLocalID,
  106. "expiresIn": kTestExpiresIn,
  107. "isNewUser": true,
  108. ])
  109. }
  110. let rpcResponse = try await AuthBackend.call(with: makeVerifyPhoneNumberRequest())
  111. XCTAssertEqual(rpcResponse.localID, kTestLocalID)
  112. XCTAssertEqual(rpcResponse.idToken, kTestIDToken)
  113. let expiresIn = try XCTUnwrap(rpcResponse.approximateExpirationDate?.timeIntervalSinceNow)
  114. XCTAssertEqual(expiresIn, 12345, accuracy: 0.1)
  115. XCTAssertEqual(rpcResponse.refreshToken, kTestRefreshToken)
  116. }
  117. /** @fn testSuccessfulVerifyPhoneNumberResponseWithTemporaryProof
  118. @brief Tests a successful to verify phone number flow with temporary proof response.
  119. */
  120. func testSuccessfulVerifyPhoneNumberResponseWithTemporaryProof() async throws {
  121. rpcIssuer.respondBlock = {
  122. try self.rpcIssuer?.respond(withJSON: [
  123. "temporaryProof": self.kTemporaryProof,
  124. "phoneNumber": self.kPhoneNumber,
  125. ])
  126. }
  127. do {
  128. let _ = try await AuthBackend.call(with: makeVerifyPhoneNumberRequestWithTemporaryProof())
  129. XCTFail("Expected to throw")
  130. } catch {
  131. let rpcError = error as NSError
  132. let credential = try XCTUnwrap(rpcError
  133. .userInfo[AuthErrors.userInfoUpdatedCredentialKey] as? PhoneAuthCredential)
  134. switch credential.credentialKind {
  135. case let .phoneNumber(phoneNumber, temporaryProof):
  136. XCTAssertEqual(temporaryProof, kTemporaryProof)
  137. XCTAssertEqual(phoneNumber, kPhoneNumber)
  138. case .verification: XCTFail("Should be phoneNumber case")
  139. }
  140. }
  141. }
  142. private func makeVerifyPhoneNumberRequest() -> VerifyPhoneNumberRequest {
  143. return VerifyPhoneNumberRequest(verificationID: kVerificationID,
  144. verificationCode: kVerificationCode,
  145. operation: AuthOperationType.signUpOrSignIn,
  146. requestConfiguration: makeRequestConfiguration())
  147. }
  148. private func makeVerifyPhoneNumberRequestWithTemporaryProof() -> VerifyPhoneNumberRequest {
  149. return VerifyPhoneNumberRequest(temporaryProof: kTemporaryProof,
  150. phoneNumber: kPhoneNumber,
  151. operation: AuthOperationType.signUpOrSignIn,
  152. requestConfiguration: makeRequestConfiguration())
  153. }
  154. }
  155. #endif