VerifyPhoneNumberTests.swift 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  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() throws {
  37. let request = makeVerifyPhoneNumberRequest()
  38. request.accessToken = kTestAccessToken
  39. let issuer = try checkRequest(
  40. request: request,
  41. expected: kExpectedAPIURL,
  42. key: kVerificationIDKey,
  43. value: kVerificationID
  44. )
  45. let requestDictionary = try XCTUnwrap(issuer.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() throws {
  57. let request = makeVerifyPhoneNumberRequestWithTemporaryProof()
  58. request.accessToken = kTestAccessToken
  59. let issuer = try checkRequest(
  60. request: request,
  61. expected: kExpectedAPIURL,
  62. key: kTemporaryProofKey,
  63. value: kTemporaryProof
  64. )
  65. let requestDictionary = try XCTUnwrap(issuer.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() throws {
  74. let kInvalidVerificationCodeErrorMessage = "INVALID_CODE"
  75. let kInvalidSessionInfoErrorMessage = "INVALID_SESSION_INFO"
  76. let kSessionExpiredErrorMessage = "SESSION_EXPIRED"
  77. try checkBackendError(
  78. request: makeVerifyPhoneNumberRequest(),
  79. message: kInvalidVerificationCodeErrorMessage,
  80. errorCode: AuthErrorCode.invalidVerificationCode
  81. )
  82. try checkBackendError(
  83. request: makeVerifyPhoneNumberRequest(),
  84. message: kInvalidSessionInfoErrorMessage,
  85. errorCode: AuthErrorCode.invalidVerificationID
  86. )
  87. try 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() throws {
  97. let kTestLocalID = "testLocalId"
  98. let kTestIDToken = "ID_TOKEN"
  99. let kTestExpiresIn = "12345"
  100. let kTestRefreshToken = "REFRESH_TOKEN"
  101. var callbackInvoked = false
  102. var rpcResponse: VerifyPhoneNumberResponse?
  103. var rpcError: NSError?
  104. AuthBackend.post(with: makeVerifyPhoneNumberRequest()) { response, error in
  105. callbackInvoked = true
  106. rpcResponse = response
  107. rpcError = error as? NSError
  108. }
  109. _ = try rpcIssuer?.respond(withJSON: [
  110. "idToken": kTestIDToken,
  111. "refreshToken": kTestRefreshToken,
  112. "localId": kTestLocalID,
  113. "expiresIn": kTestExpiresIn,
  114. "isNewUser": true,
  115. ])
  116. XCTAssert(callbackInvoked)
  117. XCTAssertNil(rpcError)
  118. XCTAssertEqual(rpcResponse?.localID, kTestLocalID)
  119. XCTAssertEqual(rpcResponse?.idToken, kTestIDToken)
  120. let expiresIn = try XCTUnwrap(rpcResponse?.approximateExpirationDate?.timeIntervalSinceNow)
  121. XCTAssertEqual(expiresIn, 12345, accuracy: 0.1)
  122. XCTAssertEqual(rpcResponse?.refreshToken, kTestRefreshToken)
  123. }
  124. /** @fn testSuccessfulVerifyPhoneNumberResponseWithTemporaryProof
  125. @brief Tests a successful to verify phone number flow with temporary proof response.
  126. */
  127. func testSuccessfulVerifyPhoneNumberResponseWithTemporaryProof() throws {
  128. var callbackInvoked = false
  129. var rpcResponse: VerifyPhoneNumberResponse?
  130. var rpcError: NSError?
  131. AuthBackend
  132. .post(with: makeVerifyPhoneNumberRequestWithTemporaryProof()) { response, error in
  133. callbackInvoked = true
  134. rpcResponse = response
  135. rpcError = error as? NSError
  136. }
  137. _ = try rpcIssuer?.respond(withJSON: [
  138. "temporaryProof": kTemporaryProof,
  139. "phoneNumber": kPhoneNumber,
  140. ])
  141. XCTAssert(callbackInvoked)
  142. XCTAssertNil(rpcResponse)
  143. let credential = try XCTUnwrap(rpcError?
  144. .userInfo[AuthErrors.userInfoUpdatedCredentialKey] as? PhoneAuthCredential)
  145. switch credential.credentialKind {
  146. case let .phoneNumber(phoneNumber, temporaryProof):
  147. XCTAssertEqual(temporaryProof, kTemporaryProof)
  148. XCTAssertEqual(phoneNumber, kPhoneNumber)
  149. case .verification: XCTFail("Should be phoneNumber case")
  150. }
  151. }
  152. private func makeVerifyPhoneNumberRequest() -> VerifyPhoneNumberRequest {
  153. return VerifyPhoneNumberRequest(verificationID: kVerificationID,
  154. verificationCode: kVerificationCode,
  155. operation: AuthOperationType.signUpOrSignIn,
  156. requestConfiguration: makeRequestConfiguration())
  157. }
  158. private func makeVerifyPhoneNumberRequestWithTemporaryProof() -> VerifyPhoneNumberRequest {
  159. return VerifyPhoneNumberRequest(temporaryProof: kTemporaryProof,
  160. phoneNumber: kPhoneNumber,
  161. operation: AuthOperationType.signUpOrSignIn,
  162. requestConfiguration: makeRequestConfiguration())
  163. }
  164. }
  165. #endif