From 5bf50ce4ca823f7be7c8dcceff3d9c0a36c9c793 Mon Sep 17 00:00:00 2001 From: Liubin Jiang Date: Mon, 2 Oct 2023 14:49:19 -0700 Subject: [PATCH 1/4] 1. Implemented finalizePasskeyEnrollment rpc call 2. Defined finalizePasskeyEnrollment Request and Response object 3. Added unit tests --- FirebaseAuth/Sources/Backend/FIRAuthBackend.h | 29 +++ FirebaseAuth/Sources/Backend/FIRAuthBackend.m | 22 ++ .../Sources/Backend/FIRAuthRPCResponse.h | 2 +- .../RPC/FIRFinalizePasskeyEnrollmentRequest.h | 68 ++++++ .../RPC/FIRFinalizePasskeyEnrollmentRequest.m | 127 ++++++++++ .../FIRFinalizePasskeyEnrollmentResponse.h | 41 ++++ .../FIRFinalizePasskeyEnrollmentResponse.m | 48 ++++ ...FIRFinalizePasskeyEnrollmentRequestTests.m | 172 ++++++++++++++ ...IRFinalizePasskeyEnrollmentResponseTests.m | 221 ++++++++++++++++++ 9 files changed, 729 insertions(+), 1 deletion(-) create mode 100644 FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h create mode 100644 FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m create mode 100644 FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h create mode 100644 FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m create mode 100644 FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m create mode 100644 FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m diff --git a/FirebaseAuth/Sources/Backend/FIRAuthBackend.h b/FirebaseAuth/Sources/Backend/FIRAuthBackend.h index 52dca974389..407a332eba4 100644 --- a/FirebaseAuth/Sources/Backend/FIRAuthBackend.h +++ b/FirebaseAuth/Sources/Backend/FIRAuthBackend.h @@ -60,6 +60,8 @@ @class FIRGetRecaptchaConfigResponse; @class FIRStartPasskeyEnrollmentRequest; @class FIRStartPasskeyEnrollmentResponse; +@class FIRFinalizePasskeyEnrollmentRequest; +@class FIRFinalizePasskeyEnrollmentResponse; @protocol FIRAuthBackendImplementation; @protocol FIRAuthBackendRPCIssuer; @@ -266,6 +268,17 @@ endpoint. typedef void (^FIRStartPasskeyEnrollmentResponseCallback)( FIRStartPasskeyEnrollmentResponse *_Nullable response, NSError *_Nullable error); +/** + @typedef FIRFinalizePasskeyEnrollmentResponseCallback + @brief The type of block used to return the result of a call to the startPasskeyEnrollment +endpoint. + @param response The received response, if any. + @param error The error which occurred, if any. + @remarks One of response or error will be non-nil. + */ +typedef void (^FIRFinalizePasskeyEnrollmentResponseCallback)( + FIRFinalizePasskeyEnrollmentResponse *_Nullable response, NSError *_Nullable error); + /** @class FIRAuthBackend @brief Simple static class with methods representing the backend RPCs. @remarks All callback blocks passed as method parameters are invoked asynchronously on the @@ -471,6 +484,14 @@ typedef void (^FIRStartPasskeyEnrollmentResponseCallback)( */ + (void)startPasskeyEnrollment:(FIRStartPasskeyEnrollmentRequest *)request callback:(FIRStartPasskeyEnrollmentResponseCallback)callback; + +/** @fn finalizePasskeyEnrollment:callback: + @brief Sends the platform created public info to the finalizePasskeyEnrollment endpoint. + @param request The request parameters. + @param callback The callback. + */ ++ (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request + callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback; #endif /** @fn revokeToken:callback: @@ -656,6 +677,14 @@ typedef void (^FIRStartPasskeyEnrollmentResponseCallback)( */ - (void)startPasskeyEnrollment:(FIRStartPasskeyEnrollmentRequest *)request callback:(FIRStartPasskeyEnrollmentResponseCallback)callback; + +/** @fn finalizePasskeyEnrollment:callback: + @brief Calls the finalizePasskeyEnrollment endpoint, which is responsible for sending the platform credential details to GCIP backend to exchange the access token and refresh token. + @param request The request parameters. + @param callback The callback. + */ +- (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request + callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback; #endif /** @fn revokeToken:callback: diff --git a/FirebaseAuth/Sources/Backend/FIRAuthBackend.m b/FirebaseAuth/Sources/Backend/FIRAuthBackend.m index 53c98e00c0a..f657f1f5cc5 100644 --- a/FirebaseAuth/Sources/Backend/FIRAuthBackend.m +++ b/FirebaseAuth/Sources/Backend/FIRAuthBackend.m @@ -64,12 +64,16 @@ #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyAssertionResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyClientRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyClientResponse.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRStartPasskeyEnrollmentRequest.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRStartPasskeyEnrollmentResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyCustomTokenRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyCustomTokenResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPasswordRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPasswordResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPhoneNumberRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPhoneNumberResponse.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h" #import "FirebaseAuth/Sources/Utilities/FIRAuthErrorUtils.h" #import "FirebaseCore/Extension/FirebaseCoreInternal.h" @@ -680,6 +684,10 @@ + (void)startPasskeyEnrollment:(FIRStartPasskeyEnrollmentRequest *)request callback:(FIRStartPasskeyEnrollmentResponseCallback)callback { [[self implementation] startPasskeyEnrollment:request callback:callback]; } + ++ (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback { + [[self implementation] finalizePasskeyEnrollment:request callback:callback]; +} #endif + (void)revokeToken:(FIRRevokeTokenRequest *)request @@ -1122,6 +1130,20 @@ - (void)startPasskeyEnrollment:(FIRStartPasskeyEnrollmentRequest *)request callback(response, nil); }]; } + +- (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback { + FIRFinalizePasskeyEnrollmentResponse *response = [[FIRFinalizePasskeyEnrollmentResponse alloc] init]; + [self callWithRequest:request + response:response + callback:^(NSError *error) { + if (error) { + callback(nil, error); + return; + } + callback(response, nil); + }]; +} + #endif - (void)revokeToken:(FIRRevokeTokenRequest *)request diff --git a/FirebaseAuth/Sources/Backend/FIRAuthRPCResponse.h b/FirebaseAuth/Sources/Backend/FIRAuthRPCResponse.h index 0fe981a6eae..ee1a71fa182 100644 --- a/FirebaseAuth/Sources/Backend/FIRAuthRPCResponse.h +++ b/FirebaseAuth/Sources/Backend/FIRAuthRPCResponse.h @@ -23,7 +23,7 @@ NS_ASSUME_NONNULL_BEGIN */ @protocol FIRAuthRPCResponse -/** @fn setFieldsWithDictionary:error: +/** @fn setWithDictionary:error: @brief Sets the response instance from the decoded JSON response. @param dictionary The dictionary decoded from HTTP JSON response. @param error An out field for an error which occurred constructing the request. diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h new file mode 100644 index 00000000000..f8558b14278 --- /dev/null +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h @@ -0,0 +1,68 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#import "FirebaseAuth/Sources/Backend/FIRAuthRPCRequest.h" +#import "FirebaseAuth/Sources/Backend/FIRIdentityToolkitRequest.h" +#import + +NS_ASSUME_NONNULL_BEGIN + +/** @class FIRFinalizePasskeyEnrollmentRequest + @brief Represents the parameters for the finalizePasskeyEnrollment endpoint. + */ +@interface FIRFinalizePasskeyEnrollmentRequest : FIRIdentityToolkitRequest + +/** + @property IDToken + @brief The raw user access token. + */ +@property(nonatomic, copy, readonly) NSString *IDToken; + +/** + @property name + @brief The passkey name. + */ +@property(nonatomic, copy, readonly) NSString *name; + +/** + @property credentialID + @brief The credential ID. + */ +@property(nonatomic, copy, readonly) NSString *credentialID; + +/** + @property clientDataJson + @brief The CollectedClientData object from the authenticator. + */ +@property(nonatomic, copy, readonly) NSString *clientDataJson; + +/** + + @property attestationObject + @brief The attestation object from the authenticator. + */ +@property(nonatomic, copy, readonly) NSString *attestationObject; + +- (nullable instancetype)initWithIDToken:(NSString *)IDToken + name:(NSString *)name + credentialID:(NSString *)credentialID + clientDataJson:(NSString *)clientDataJson + attestationObject:(NSString *)attestationObject + requestConfiguration:(FIRAuthRequestConfiguration *)requestConfiguration; + +@end + +NS_ASSUME_NONNULL_END diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m new file mode 100644 index 00000000000..2cdad6d193a --- /dev/null +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m @@ -0,0 +1,127 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h" +NS_ASSUME_NONNULL_BEGIN + +/** + @var kFinalizePasskeyEnrollmentEndPoint + @brief GCIP endpoint for finalizePasskeyEnrollment rpc + */ +static NSString *const kFinalizePasskeyEnrollmentEndPoint = @"accounts/passkeyEnrollment:finalize"; + +/** + @var kTenantIDKey + @brief The key for the tenant id value in the request. + */ +static NSString *const kTenantIDKey = @"tenantId"; + +/** + @var kIDTokenKey + @brief The key for idToken value in the request. + */ +static NSString *const kIDTokenKey = @"idToken"; + +/** + @var kAuthRegistrationRespKey + @brief The key for registration object from the authenticator. + */ +static NSString *const kAuthRegistrationRespKey = @"authenticatorRegistrationResponse"; + +/** + @var kNameKey + @brief The key of passkey name. + */ +static NSString *const kNameKey = @"name"; + +/** + @var kCredentialIDKey + @brief The key for registered credential identifier. + */ +static NSString *const kCredentialIDKey = @"credentialId"; + +/** + @var kAuthAttestationRespKey + @brief The key for attestation response from a FIDO authenticator. + */ +static NSString *const kAuthAttestationRespKey = @"authenticatorAttestationResponse"; + +/** + @var kClientDataJsonKey + @brief The key for CollectedClientData object from the authenticator. + */ +static NSString *const kClientDataJsonKey = @"clientDataJson"; + +/** + @var kAttestationObject + @brief The key for the attestation object from the authenticator. + */ +static NSString *const kAttestationObject = @"attestationObject"; + + +@implementation FIRFinalizePasskeyEnrollmentRequest + +- (nullable instancetype)initWithIDToken:(NSString *)IDToken + name:(NSString *)name + credentialID:(NSString *)credentialID + clientDataJson:(NSString *)clientDataJson + attestationObject:(NSString *)attestationObject requestConfiguration:(FIRAuthRequestConfiguration *)requestConfiguration { + self = [super initWithEndpoint:kFinalizePasskeyEnrollmentEndPoint + requestConfiguration:requestConfiguration]; + if (self) { + self.useIdentityPlatform = YES; + _IDToken = IDToken; + _name = name; + _credentialID = credentialID; + _clientDataJson = clientDataJson; + _attestationObject = attestationObject; + } + return self; +} + +- (nullable id)unencodedHTTPRequestBodyWithError:(NSError *__autoreleasing _Nullable *)error { + NSMutableDictionary *postBody = [NSMutableDictionary dictionary]; + NSMutableDictionary *authRegistrationResponse = [NSMutableDictionary dictionary]; + NSMutableDictionary *authAttestationResponse = [NSMutableDictionary dictionary]; + + if (_IDToken) { + postBody[kIDTokenKey] = _IDToken; + } + if (_name) { + postBody[kNameKey] = _name; + } + if (_credentialID) { + authRegistrationResponse[kCredentialIDKey] = _credentialID; + } + if (_clientDataJson) { + authAttestationResponse[kClientDataJsonKey] = _clientDataJson; + } + if (_attestationObject) { + authAttestationResponse[kAttestationObject] = _attestationObject; + } + if (self.tenantID) { + postBody[kTenantIDKey] = self.tenantID; + } + + authRegistrationResponse[kAuthAttestationRespKey] = authAttestationResponse; + postBody[kAuthRegistrationRespKey] = authRegistrationResponse; + + return [postBody copy]; +} + +@end + +NS_ASSUME_NONNULL_END diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h new file mode 100644 index 00000000000..21c553bf9a8 --- /dev/null +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h @@ -0,0 +1,41 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#import "FirebaseAuth/Sources/Backend/FIRAuthRPCResponse.h" + +NS_ASSUME_NONNULL_BEGIN + +/** + @class FIRFinalizePasskeyEnrollmentResponse + @brief Represents the response from the startPasskeyEnrollment endpoint. + */ +@interface FIRFinalizePasskeyEnrollmentResponse : NSObject + +/** + @property idToken + @brief The user raw access token. + */ +@property(nonatomic, readonly, copy) NSString *idToken; + +/** + @property refershToken + @brief Refresh token for the authenticated user. + */ +@property(nonatomic, copy, readonly) NSData *refreshToken; + +@end + +NS_ASSUME_NONNULL_END diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m new file mode 100644 index 00000000000..603560c9930 --- /dev/null +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m @@ -0,0 +1,48 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h" + +/** + @var kIDTokenKey + @brief The name of the field in the response JSON for id token. + */ +static const NSString *kIdTokenKey = @"idToken"; + +/** + @var kRefreshTokenKey + @brief The name of the field in the response JSON for refresh token. + */ +static const NSString *kRefreshTokenKey = @"refreshToken"; + + +@implementation FIRFinalizePasskeyEnrollmentResponse + +- (BOOL)setWithDictionary:(nonnull NSDictionary *)dictionary + error:(NSError *__autoreleasing _Nullable *_Nullable)error { + if (dictionary[kIdTokenKey] == nil) { + return NO; + } + if (dictionary[kRefreshTokenKey] == nil) { + return NO; + } + + _idToken = dictionary[kIdTokenKey]; + _refreshToken = dictionary[kRefreshTokenKey]; + return YES; +} + +@end diff --git a/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m new file mode 100644 index 00000000000..cfe984ab53e --- /dev/null +++ b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m @@ -0,0 +1,172 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#import +#if TARGET_OS_IOS || TARGET_OS_TV || TARGET_OS_OSX || TARGET_OS_MACCATALYST + +#import + +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h" +#import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRAuthErrors.h" +#import "FirebaseAuth/Tests/Unit/FIRFakeBackendRPCIssuer.h" + +/** + @var kTestAPIKey + @brief Fake API key used for testing. + */ +static NSString *const kTestAPIKey = @"APIKey"; + +/** + @var kTestFirebaseAppID + @brief Fake Firebase app ID used for testing. + */ +static NSString *const kTestFirebaseAppID = @"appID"; + +/** + @var kExpectedAPIURL + @brief The expected URL for the test calls. + */ +static NSString *const kExpectedAPIURL = + @"https://identitytoolkit.googleapis.com/v2/accounts/passkeyEnrollment:finalize?key=APIKey"; + +/** + @var kIDToken + @brief Token representing the user's identity. + */ +static NSString *const kIDToken = @"testIDToken"; + +/** + @var kIDTokenKey + @brief ID Token field. + */ +static NSString *const kIDTokenKey = @"idToken"; + +/** + @var kName + @brief Passkey name. + */ +static NSString *const kName = @"testName"; + +/** + @var kNameKey + @brief Passkey name field + */ +static NSString *const kNameKey = @"name"; + +/** + @var kCredentialID + @brief credential ID. + */ +static NSString *const kCredentialID = @"testCredentialID"; + +/** + @var kCredentialIDKey + @brief credential ID field. + */ +static NSString *const kCredentialIDKey = @"credentialId"; + +/** + @var kRawAttestationObject + @brief Passkey attestation object. + */ +static NSString *const kRawAttestationObject = @"testRawAttestationObject"; + +/** + @var kRawAttestationObjectKey + @brief The key for the attestation object from the authenticator. + */ +static NSString *const kRawAttestationObjectKey = @"attestationObject"; + +/** + @var kRawClientDataJSON + @brief CollectedClientData object from the authenticator. + */ +static NSString *const kRawClientDataJSON = @"testRawClientDataJSON"; + +/** + @var kRawClientDataJSONKey + @brief The key for the attestation object from the authenticator. + */ +static NSString *const kRawClientDataJSONKey = @"clientDataJson"; + +/** + @var kAuthRegistrationRespKey + @brief The registration object from the authenticator. + */ +static NSString *const kAuthRegistrationRespKey = @"authenticatorRegistrationResponse"; + +/** + @var kAuthAttestationRespKey + @brief The key for attestation response from a FIDO authenticator. + */ +static NSString *const kAuthAttestationRespKey = @"authenticatorAttestationResponse"; + +/** + @class FIRFinalizePasskeyEnrollmentRequestTests + @brief Tests for @c FIRFinalizePasskeyEnrollmentRequest. + */ +@interface FIRFinalizePasskeyEnrollmentRequestTests : XCTestCase +@end + +@implementation FIRFinalizePasskeyEnrollmentRequestTests { + /** + @brief This backend RPC issuer is used to fake network responses for each test in the suite. + In the @c setUp method we initialize this and set @c FIRAuthBackend's RPC issuer to it. + */ + FIRFakeBackendRPCIssuer *_RPCIssuer; + + /** + @brief This is the request configuration used for testing. + */ + FIRAuthRequestConfiguration *_requestConfiguration; +} + +- (void)setUp { + [super setUp]; + FIRFakeBackendRPCIssuer *RPCIssuer = [[FIRFakeBackendRPCIssuer alloc] init]; + [FIRAuthBackend setDefaultBackendImplementationWithRPCIssuer:RPCIssuer]; + _RPCIssuer = RPCIssuer; + _requestConfiguration = [[FIRAuthRequestConfiguration alloc] initWithAPIKey:kTestAPIKey + appID:kTestFirebaseAppID]; +} + +- (void)tearDown { + _RPCIssuer = nil; + _requestConfiguration = nil; + [FIRAuthBackend setDefaultBackendImplementationWithRPCIssuer:nil]; + [super tearDown]; +} + +- (void)testFinalizePasskeyEnrollmentRequest { + if (@available(iOS 15.0, *)) { + FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; + + [FIRAuthBackend finalizePasskeyEnrollment:request callback:^(FIRFinalizePasskeyEnrollmentResponse * _Nullable response, NSError * _Nullable error) { + }]; + XCTAssertEqualObjects(_RPCIssuer.requestURL.absoluteString, kExpectedAPIURL); + XCTAssertNotNil(_RPCIssuer.decodedRequest); + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kIDTokenKey], kIDToken); + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kNameKey], kName); + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kAuthAttestationRespKey][kRawClientDataJSONKey], kRawClientDataJSON); + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kAuthAttestationRespKey][kRawAttestationObjectKey], kRawAttestationObject); + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kCredentialIDKey], kCredentialID); + + } +} + +@end +#endif diff --git a/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m new file mode 100644 index 00000000000..0b461da3927 --- /dev/null +++ b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m @@ -0,0 +1,221 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#import + +#if TARGET_OS_IOS || TARGET_OS_TV || TARGET_OS_OSX || TARGET_OS_MACCATALYST +#import "FirebaseAuth/Sources/Public/FirebaseAuth/FIRAuthErrors.h" + +#import "FirebaseAuth/Sources/Backend/FIRAuthBackend.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h" +#import "FirebaseAuth/Sources/Utilities/FIRAuthInternalErrors.h" +#import "FirebaseAuth/Tests/Unit/FIRFakeBackendRPCIssuer.h" +/** + @var kTestAPIKey + @brief Fake API key used for testing. + */ +static NSString *const kTestAPIKey = @"APIKey"; + +/** + @var kTestFirebaseAppID + @brief Fake Firebase app ID used for testing. + */ +static NSString *const kTestFirebaseAppID = @"appID"; + +/** + @var kIDToken + @brief Token representing the user's identity. + */ +static NSString *const kIDToken = @"idToken"; + +/** + @var kRefreshToken + @brief Refresh Token + */ +static NSString *const kRefreshToken = @"refreshToken"; + +/** + @var kName + @brief Passkey name. + */ +static NSString *const kName = @"testName"; + +/** + @var kCredentialID + @brief credential ID. + */ +static NSString *const kCredentialID = @"testCredentialID"; + +/** + @var kRawAttestationObject + @brief Passkey attestation object. + */ +static NSString *const kRawAttestationObject = @"testRawAttestationObject"; + +/** + @var kRawClientDataJSON + @brief Passkey client data json. + */ +static NSString *const kRawClientDataJSON = @"testRawClientDataJSON"; + +/** + @class FIRFinalizePasskeyEnrollmentResponseTests + @brief Tests for @c FIRFinalizePasskeyEnrollmentResponse. + */ +@interface FIRFinalizePasskeyEnrollmentResponseTests : XCTestCase +@end +@implementation FIRFinalizePasskeyEnrollmentResponseTests { + /** + @brief This backend RPC issuer is used to fake network responses for each test in the suite. + In the @c setUp method we initialize this and set @c FIRAuthBackend's RPC issuer to it. + */ + FIRFakeBackendRPCIssuer *_RPCIssuer; + + /** + @brief This is the request configuration used for testing. + */ + FIRAuthRequestConfiguration *_requestConfiguration; +} + +- (void)setUp { + [super setUp]; + FIRFakeBackendRPCIssuer *RPCIssuer = [[FIRFakeBackendRPCIssuer alloc] init]; + [FIRAuthBackend setDefaultBackendImplementationWithRPCIssuer:RPCIssuer]; + _RPCIssuer = RPCIssuer; + _requestConfiguration = [[FIRAuthRequestConfiguration alloc] initWithAPIKey:kTestAPIKey + appID:kTestFirebaseAppID]; +} + +- (void)tearDown { + _RPCIssuer = nil; + _requestConfiguration = nil; + [FIRAuthBackend setDefaultBackendImplementationWithRPCIssuer:nil]; + [super tearDown]; +} + +/** @fn testSuccessfulFinalizePasskeyEnrollmentResponse + @brief This test simulates a successful @c FinalizePasskeyEnrollment flow. + */ +- (void)testSuccessfulFinalizePasskeyEnrollmentResponse { + if (@available(iOS 15.0, *)) { + FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; + + __block BOOL callbackInvoked; + __block FIRFinalizePasskeyEnrollmentResponse *RPCResponse; + __block NSError *RPCError; + + [FIRAuthBackend finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error) { + callbackInvoked = YES; + RPCResponse = response; + RPCError = error; + }]; + + [_RPCIssuer respondWithJSON:@{ + @"idToken" : kIDToken, + @"refreshToken": kRefreshToken, + }]; + + XCTAssert(callbackInvoked); + XCTAssertNil(RPCError); + XCTAssertNotNil(RPCResponse); + XCTAssertEqualObjects(RPCResponse.idToken, kIDToken); + XCTAssertEqualObjects(RPCResponse.refreshToken, kRefreshToken); + } +} + +/** @fn testFinalizePasskeyEnrollmentResponseMissingIDTokenError + @brief This test simulates an unexpected response returned from server in @c + FinalizePasskeyEnrollment flow. + */ +- (void)testFinalizePasskeyEnrollmentResponseMissingIDTokenError { + if (@available(iOS 15.0, *)) { + FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; + + __block BOOL callbackInvoked; + __block FIRFinalizePasskeyEnrollmentResponse *RPCResponse; + __block NSError *RPCError; + + [FIRAuthBackend finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error) { + callbackInvoked = YES; + RPCResponse = response; + RPCError = error; + }]; + + [_RPCIssuer respondWithJSON:@{ + @"wrongkey" : @{}, + @"refreshToken" : kRefreshToken, + }]; + [self errorValidationHelperWithCallbackInvoked:callbackInvoked + rpcError:RPCError + rpcResponse:RPCResponse]; + } +} +/** @fn testFinalizePasskeyEnrollmentResponseMissingRefreshTokenError + @brief This test simulates an unexpected response returned from server in @c + FinalizePasskeyEnrollment flow. + */ +- (void)testFinalizePasskeyEnrollmentResponseMissingRefreshTokenError { + if (@available(iOS 15.0, *)) { + FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc]initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; + + __block BOOL callbackInvoked; + __block FIRFinalizePasskeyEnrollmentResponse *RPCResponse; + __block NSError *RPCError; + + [FIRAuthBackend finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error) { + callbackInvoked = YES; + RPCResponse = response; + RPCError = error; + }]; + + [_RPCIssuer respondWithJSON:@{ + @"wrongkey" : @{}, + @"idToken" : kIDToken, + }]; + [self errorValidationHelperWithCallbackInvoked:callbackInvoked + rpcError:RPCError + rpcResponse:RPCResponse]; + } +} + +/** @fn errorValidationHelperWithCallbackInvoked:rpcError:rpcResponse: + @brief Helper function to validate the unexpected response returned from server in @c + FinalizePasskeyEnrollment flow. + */ +- (void)errorValidationHelperWithCallbackInvoked:(BOOL)callbackInvoked + rpcError:(NSError *)RPCError + rpcResponse:(FIRFinalizePasskeyEnrollmentResponse *)RPCResponse { + XCTAssert(callbackInvoked); + XCTAssertNotNil(RPCError); + XCTAssertEqualObjects(RPCError.domain, FIRAuthErrorDomain); + XCTAssertEqual(RPCError.code, FIRAuthErrorCodeInternalError); + XCTAssertNotNil(RPCError.userInfo[NSUnderlyingErrorKey]); + NSError *underlyingError = RPCError.userInfo[NSUnderlyingErrorKey]; + XCTAssertNotNil(underlyingError); + XCTAssertNotNil(underlyingError.userInfo[FIRAuthErrorUserInfoDeserializedResponseKey]); + XCTAssertNil(RPCResponse); +} + +@end +#endif + From 127a6e999381befaebac3ccca80334a8f505a68a Mon Sep 17 00:00:00 2001 From: Liubin Jiang Date: Mon, 2 Oct 2023 14:53:39 -0700 Subject: [PATCH 2/4] ran check.sh --- FirebaseAuth/Sources/Backend/FIRAuthBackend.h | 3 +- FirebaseAuth/Sources/Backend/FIRAuthBackend.m | 17 ++-- .../RPC/FIRFinalizePasskeyEnrollmentRequest.h | 4 +- .../RPC/FIRFinalizePasskeyEnrollmentRequest.m | 10 +- .../FIRFinalizePasskeyEnrollmentResponse.m | 3 +- ...FIRFinalizePasskeyEnrollmentRequestTests.m | 30 ++++-- ...IRFinalizePasskeyEnrollmentResponseTests.m | 91 ++++++++++++------- 7 files changed, 97 insertions(+), 61 deletions(-) diff --git a/FirebaseAuth/Sources/Backend/FIRAuthBackend.h b/FirebaseAuth/Sources/Backend/FIRAuthBackend.h index 407a332eba4..2f59827a01a 100644 --- a/FirebaseAuth/Sources/Backend/FIRAuthBackend.h +++ b/FirebaseAuth/Sources/Backend/FIRAuthBackend.h @@ -679,7 +679,8 @@ typedef void (^FIRFinalizePasskeyEnrollmentResponseCallback)( callback:(FIRStartPasskeyEnrollmentResponseCallback)callback; /** @fn finalizePasskeyEnrollment:callback: - @brief Calls the finalizePasskeyEnrollment endpoint, which is responsible for sending the platform credential details to GCIP backend to exchange the access token and refresh token. + @brief Calls the finalizePasskeyEnrollment endpoint, which is responsible for sending the + platform credential details to GCIP backend to exchange the access token and refresh token. @param request The request parameters. @param callback The callback. */ diff --git a/FirebaseAuth/Sources/Backend/FIRAuthBackend.m b/FirebaseAuth/Sources/Backend/FIRAuthBackend.m index f657f1f5cc5..4002be4d75f 100644 --- a/FirebaseAuth/Sources/Backend/FIRAuthBackend.m +++ b/FirebaseAuth/Sources/Backend/FIRAuthBackend.m @@ -36,6 +36,8 @@ #import "FirebaseAuth/Sources/Backend/RPC/FIRDeleteAccountResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIREmailLinkSignInRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIREmailLinkSignInResponse.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h" +#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRGetAccountInfoRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRGetAccountInfoResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRGetOOBConfirmationCodeRequest.h" @@ -64,16 +66,12 @@ #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyAssertionResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyClientRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyClientResponse.h" -#import "FirebaseAuth/Sources/Backend/RPC/FIRStartPasskeyEnrollmentRequest.h" -#import "FirebaseAuth/Sources/Backend/RPC/FIRStartPasskeyEnrollmentResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyCustomTokenRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyCustomTokenResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPasswordRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPasswordResponse.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPhoneNumberRequest.h" #import "FirebaseAuth/Sources/Backend/RPC/FIRVerifyPhoneNumberResponse.h" -#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h" -#import "FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.h" #import "FirebaseAuth/Sources/Utilities/FIRAuthErrorUtils.h" #import "FirebaseCore/Extension/FirebaseCoreInternal.h" @@ -684,8 +682,9 @@ + (void)startPasskeyEnrollment:(FIRStartPasskeyEnrollmentRequest *)request callback:(FIRStartPasskeyEnrollmentResponseCallback)callback { [[self implementation] startPasskeyEnrollment:request callback:callback]; } - -+ (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback { + ++ (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request + callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback { [[self implementation] finalizePasskeyEnrollment:request callback:callback]; } #endif @@ -1131,8 +1130,10 @@ - (void)startPasskeyEnrollment:(FIRStartPasskeyEnrollmentRequest *)request }]; } -- (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback { - FIRFinalizePasskeyEnrollmentResponse *response = [[FIRFinalizePasskeyEnrollmentResponse alloc] init]; +- (void)finalizePasskeyEnrollment:(FIRFinalizePasskeyEnrollmentRequest *)request + callback:(FIRFinalizePasskeyEnrollmentResponseCallback)callback { + FIRFinalizePasskeyEnrollmentResponse *response = + [[FIRFinalizePasskeyEnrollmentResponse alloc] init]; [self callWithRequest:request response:response callback:^(NSError *error) { diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h index f8558b14278..6530558b13d 100644 --- a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h @@ -14,9 +14,9 @@ * limitations under the License. */ +#import #import "FirebaseAuth/Sources/Backend/FIRAuthRPCRequest.h" #import "FirebaseAuth/Sources/Backend/FIRIdentityToolkitRequest.h" -#import NS_ASSUME_NONNULL_BEGIN @@ -50,7 +50,7 @@ NS_ASSUME_NONNULL_BEGIN @property(nonatomic, copy, readonly) NSString *clientDataJson; /** - + @property attestationObject @brief The attestation object from the authenticator. */ diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m index 2cdad6d193a..3d5cb794310 100644 --- a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.m @@ -71,14 +71,14 @@ */ static NSString *const kAttestationObject = @"attestationObject"; - @implementation FIRFinalizePasskeyEnrollmentRequest - (nullable instancetype)initWithIDToken:(NSString *)IDToken name:(NSString *)name credentialID:(NSString *)credentialID clientDataJson:(NSString *)clientDataJson - attestationObject:(NSString *)attestationObject requestConfiguration:(FIRAuthRequestConfiguration *)requestConfiguration { + attestationObject:(NSString *)attestationObject + requestConfiguration:(FIRAuthRequestConfiguration *)requestConfiguration { self = [super initWithEndpoint:kFinalizePasskeyEnrollmentEndPoint requestConfiguration:requestConfiguration]; if (self) { @@ -96,7 +96,7 @@ - (nullable id)unencodedHTTPRequestBodyWithError:(NSError *__autoreleasing _Null NSMutableDictionary *postBody = [NSMutableDictionary dictionary]; NSMutableDictionary *authRegistrationResponse = [NSMutableDictionary dictionary]; NSMutableDictionary *authAttestationResponse = [NSMutableDictionary dictionary]; - + if (_IDToken) { postBody[kIDTokenKey] = _IDToken; } @@ -115,10 +115,10 @@ - (nullable id)unencodedHTTPRequestBodyWithError:(NSError *__autoreleasing _Null if (self.tenantID) { postBody[kTenantIDKey] = self.tenantID; } - + authRegistrationResponse[kAuthAttestationRespKey] = authAttestationResponse; postBody[kAuthRegistrationRespKey] = authRegistrationResponse; - + return [postBody copy]; } diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m index 603560c9930..93f4334871b 100644 --- a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentResponse.m @@ -28,7 +28,6 @@ */ static const NSString *kRefreshTokenKey = @"refreshToken"; - @implementation FIRFinalizePasskeyEnrollmentResponse - (BOOL)setWithDictionary:(nonnull NSDictionary *)dictionary @@ -39,7 +38,7 @@ - (BOOL)setWithDictionary:(nonnull NSDictionary *)dictionary if (dictionary[kRefreshTokenKey] == nil) { return NO; } - + _idToken = dictionary[kIdTokenKey]; _refreshToken = dictionary[kRefreshTokenKey]; return YES; diff --git a/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m index cfe984ab53e..3b3b78f0dd8 100644 --- a/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m +++ b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentRequestTests.m @@ -153,18 +153,32 @@ - (void)tearDown { - (void)testFinalizePasskeyEnrollmentRequest { if (@available(iOS 15.0, *)) { - FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; - - [FIRAuthBackend finalizePasskeyEnrollment:request callback:^(FIRFinalizePasskeyEnrollmentResponse * _Nullable response, NSError * _Nullable error) { - }]; + FIRFinalizePasskeyEnrollmentRequest *request = + [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken + name:kName + credentialID:kCredentialID + clientDataJson:kRawClientDataJSON + attestationObject:kRawAttestationObject + requestConfiguration:_requestConfiguration]; + + [FIRAuthBackend + finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error){ + }]; XCTAssertEqualObjects(_RPCIssuer.requestURL.absoluteString, kExpectedAPIURL); XCTAssertNotNil(_RPCIssuer.decodedRequest); XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kIDTokenKey], kIDToken); XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kNameKey], kName); - XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kAuthAttestationRespKey][kRawClientDataJSONKey], kRawClientDataJSON); - XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kAuthAttestationRespKey][kRawAttestationObjectKey], kRawAttestationObject); - XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kCredentialIDKey], kCredentialID); - + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey] + [kAuthAttestationRespKey][kRawClientDataJSONKey], + kRawClientDataJSON); + XCTAssertEqualObjects( + _RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kAuthAttestationRespKey] + [kRawAttestationObjectKey], + kRawAttestationObject); + XCTAssertEqualObjects(_RPCIssuer.decodedRequest[kAuthRegistrationRespKey][kCredentialIDKey], + kCredentialID); } } diff --git a/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m index 0b461da3927..cafdabd7990 100644 --- a/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m +++ b/FirebaseAuth/Tests/Unit/FIRFinalizePasskeyEnrollmentResponseTests.m @@ -112,25 +112,32 @@ - (void)tearDown { */ - (void)testSuccessfulFinalizePasskeyEnrollmentResponse { if (@available(iOS 15.0, *)) { - FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; + FIRFinalizePasskeyEnrollmentRequest *request = + [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken + name:kName + credentialID:kCredentialID + clientDataJson:kRawClientDataJSON + attestationObject:kRawAttestationObject + requestConfiguration:_requestConfiguration]; __block BOOL callbackInvoked; __block FIRFinalizePasskeyEnrollmentResponse *RPCResponse; __block NSError *RPCError; - - [FIRAuthBackend finalizePasskeyEnrollment:request - callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, - NSError *_Nullable error) { - callbackInvoked = YES; - RPCResponse = response; - RPCError = error; - }]; - + + [FIRAuthBackend + finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error) { + callbackInvoked = YES; + RPCResponse = response; + RPCError = error; + }]; + [_RPCIssuer respondWithJSON:@{ @"idToken" : kIDToken, - @"refreshToken": kRefreshToken, + @"refreshToken" : kRefreshToken, }]; - + XCTAssert(callbackInvoked); XCTAssertNil(RPCError); XCTAssertNotNil(RPCResponse); @@ -145,20 +152,27 @@ - (void)testSuccessfulFinalizePasskeyEnrollmentResponse { */ - (void)testFinalizePasskeyEnrollmentResponseMissingIDTokenError { if (@available(iOS 15.0, *)) { - FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; - + FIRFinalizePasskeyEnrollmentRequest *request = + [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken + name:kName + credentialID:kCredentialID + clientDataJson:kRawClientDataJSON + attestationObject:kRawAttestationObject + requestConfiguration:_requestConfiguration]; + __block BOOL callbackInvoked; __block FIRFinalizePasskeyEnrollmentResponse *RPCResponse; __block NSError *RPCError; - - [FIRAuthBackend finalizePasskeyEnrollment:request - callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, - NSError *_Nullable error) { - callbackInvoked = YES; - RPCResponse = response; - RPCError = error; - }]; - + + [FIRAuthBackend + finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error) { + callbackInvoked = YES; + RPCResponse = response; + RPCError = error; + }]; + [_RPCIssuer respondWithJSON:@{ @"wrongkey" : @{}, @"refreshToken" : kRefreshToken, @@ -174,20 +188,27 @@ - (void)testFinalizePasskeyEnrollmentResponseMissingIDTokenError { */ - (void)testFinalizePasskeyEnrollmentResponseMissingRefreshTokenError { if (@available(iOS 15.0, *)) { - FIRFinalizePasskeyEnrollmentRequest *request = [[FIRFinalizePasskeyEnrollmentRequest alloc]initWithIDToken:kIDToken name:kName credentialID:kCredentialID clientDataJson:kRawClientDataJSON attestationObject:kRawAttestationObject requestConfiguration:_requestConfiguration]; + FIRFinalizePasskeyEnrollmentRequest *request = + [[FIRFinalizePasskeyEnrollmentRequest alloc] initWithIDToken:kIDToken + name:kName + credentialID:kCredentialID + clientDataJson:kRawClientDataJSON + attestationObject:kRawAttestationObject + requestConfiguration:_requestConfiguration]; __block BOOL callbackInvoked; __block FIRFinalizePasskeyEnrollmentResponse *RPCResponse; __block NSError *RPCError; - - [FIRAuthBackend finalizePasskeyEnrollment:request - callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, - NSError *_Nullable error) { - callbackInvoked = YES; - RPCResponse = response; - RPCError = error; - }]; - + + [FIRAuthBackend + finalizePasskeyEnrollment:request + callback:^(FIRFinalizePasskeyEnrollmentResponse *_Nullable response, + NSError *_Nullable error) { + callbackInvoked = YES; + RPCResponse = response; + RPCError = error; + }]; + [_RPCIssuer respondWithJSON:@{ @"wrongkey" : @{}, @"idToken" : kIDToken, @@ -204,7 +225,8 @@ - (void)testFinalizePasskeyEnrollmentResponseMissingRefreshTokenError { */ - (void)errorValidationHelperWithCallbackInvoked:(BOOL)callbackInvoked rpcError:(NSError *)RPCError - rpcResponse:(FIRFinalizePasskeyEnrollmentResponse *)RPCResponse { + rpcResponse: + (FIRFinalizePasskeyEnrollmentResponse *)RPCResponse { XCTAssert(callbackInvoked); XCTAssertNotNil(RPCError); XCTAssertEqualObjects(RPCError.domain, FIRAuthErrorDomain); @@ -218,4 +240,3 @@ - (void)errorValidationHelperWithCallbackInvoked:(BOOL)callbackInvoked @end #endif - From b78ce9ef34b4ca73539e7d7f833451d1a73e3252 Mon Sep 17 00:00:00 2001 From: Liubin Jiang Date: Mon, 2 Oct 2023 15:26:32 -0700 Subject: [PATCH 3/4] exclude import from watchOS --- .../Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h | 3 +++ 1 file changed, 3 insertions(+) diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h index 6530558b13d..3a091a12b76 100644 --- a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h @@ -14,7 +14,10 @@ * limitations under the License. */ +#if TARGET_OS_IOS || TARGET_OS_TV || TARGET_OS_OSX || TARGET_OS_MACCATALYST #import +#endif + #import "FirebaseAuth/Sources/Backend/FIRAuthRPCRequest.h" #import "FirebaseAuth/Sources/Backend/FIRIdentityToolkitRequest.h" From 02eb013154784f8a0e39d7b7a5daa242fc062faf Mon Sep 17 00:00:00 2001 From: Liubin Jiang Date: Tue, 3 Oct 2023 10:01:40 -0700 Subject: [PATCH 4/4] fix CI imports for watch OS --- .../Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h | 4 ---- 1 file changed, 4 deletions(-) diff --git a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h index 3a091a12b76..53836f50048 100644 --- a/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h +++ b/FirebaseAuth/Sources/Backend/RPC/FIRFinalizePasskeyEnrollmentRequest.h @@ -14,10 +14,6 @@ * limitations under the License. */ -#if TARGET_OS_IOS || TARGET_OS_TV || TARGET_OS_OSX || TARGET_OS_MACCATALYST -#import -#endif - #import "FirebaseAuth/Sources/Backend/FIRAuthRPCRequest.h" #import "FirebaseAuth/Sources/Backend/FIRIdentityToolkitRequest.h"