From bded8d43a6eee5a1ca8e7689dca42787ab1a4172 Mon Sep 17 00:00:00 2001 From: pragatimodi <110490169+pragatimodi@users.noreply.github.com> Date: Wed, 18 Oct 2023 20:06:35 +0000 Subject: [PATCH] update passkeys changes --- src/auth/auth-api-request.ts | 24 ++++----- src/auth/passkey-config-manager.ts | 33 +++++++++++- src/auth/passkey-config.ts | 74 +++++++++++++++++++++++---- test/unit/auth/passkey-config.spec.ts | 2 +- 4 files changed, 110 insertions(+), 23 deletions(-) diff --git a/src/auth/auth-api-request.ts b/src/auth/auth-api-request.ts index f469a8cd46..7f36461fc4 100644 --- a/src/auth/auth-api-request.ts +++ b/src/auth/auth-api-request.ts @@ -2071,51 +2071,51 @@ const CREATE_TENANT = new ApiSettings('/tenants', 'POST') } }); -/** Instantiates the getPasskeyConfig endpoint settings. */ +/** Instantiates the GET_PASSKEY_CONFIG endpoint settings. */ const GET_PASSKEY_CONFIG = new ApiSettings('/passkeyConfig', 'GET') .setResponseValidator((response: any) => { - // Response should always contain at least the config name. + // Validate the response for GET_PASSKEY_CONFIG. if (!validator.isNonEmptyString(response.name)) { throw new FirebaseAuthError( AuthClientErrorCode.INTERNAL_ERROR, - 'INTERNAL ASSERT FAILED: Unable to get project config', + 'INTERNAL ASSERT FAILED: Unable to get passkey config', ); } }); -/** Instantiates the getPasskeyConfig endpoint settings. */ +/** Instantiates the GET_TENANT_PASSKEY_CONFIG endpoint settings. */ const GET_TENANT_PASSKEY_CONFIG = new ApiSettings('/tenants/{tenantId}/passkeyConfig', 'GET') .setResponseValidator((response: any) => { - // Response should always contain at least the config name. + // Validate the response for GET_TENANT_PASSKEY_CONFIG. if (!validator.isNonEmptyString(response.name)) { throw new FirebaseAuthError( AuthClientErrorCode.INTERNAL_ERROR, - 'INTERNAL ASSERT FAILED: Unable to get project config', + 'INTERNAL ASSERT FAILED: Unable to get tenant passkey config', ); } }); -/** Instantiates the getPasskeyConfig endpoint settings. */ +/** Instantiates the UPDATE_PASSKEY_CONFIG endpoint settings. */ const UPDATE_PASSKEY_CONFIG = new ApiSettings('/passkeyConfig?updateMask={updateMask}', 'PATCH') .setResponseValidator((response: any) => { - // Response should always contain at least the config name. + // Validate the response for UPDATE_PASSKEY_CONFIG. if (!validator.isNonEmptyString(response.name)) { throw new FirebaseAuthError( AuthClientErrorCode.INTERNAL_ERROR, - 'INTERNAL ASSERT FAILED: Unable to get project config', + 'INTERNAL ASSERT FAILED: Unable to update passkey config', ); } }); -/** Instantiates the getPasskeyConfig endpoint settings. */ +/** Instantiates the UPDATE_TENANT_PASSKEY_CONFIG endpoint settings. */ const UPDATE_TENANT_PASSKEY_CONFIG = new ApiSettings( '/tenant/{tenantId}/passkeyConfig?updateMask={updateMask}', 'PATCH') .setResponseValidator((response: any) => { - // Response should always contain at least the config name. + // Validate the response for UPDATE_TENANT_PASSKEY_CONFIG. if (!validator.isNonEmptyString(response.name)) { throw new FirebaseAuthError( AuthClientErrorCode.INTERNAL_ERROR, - 'INTERNAL ASSERT FAILED: Unable to get project config', + 'INTERNAL ASSERT FAILED: Unable to update tenant passkey config', ); } }); diff --git a/src/auth/passkey-config-manager.ts b/src/auth/passkey-config-manager.ts index a0793d1d23..100ba2df89 100644 --- a/src/auth/passkey-config-manager.ts +++ b/src/auth/passkey-config-manager.ts @@ -24,14 +24,30 @@ import { PasskeyConfigServerResponse } from './passkey-config'; - +/** + * Manages Passkey Configuration for a Firebase app. + */ export class PasskeyConfigManager { private readonly authRequestHandler: AuthRequestHandler; + /** + * Initializes a PasskeyConfigManager instance for a specified FirebaseApp. + * + * @param app - The Firebase app associated with this PasskeyConfigManager instance. + * + * @constructor + * @internal + */ constructor(app: App) { this.authRequestHandler = new AuthRequestHandler(app); } + /** + * Retrieves the Passkey Configuration. + * + * @param tenantId - (optional) The tenant ID if querying passkeys on a specific tenant. + * @returns A promise fulfilled with the passkey configuration. + */ public getPasskeyConfig(tenantId?: string): Promise { return this.authRequestHandler.getPasskeyConfig(tenantId) .then((response: PasskeyConfigServerResponse) => { @@ -39,6 +55,14 @@ export class PasskeyConfigManager { }); } + /** + * Creates a new passkey configuration. + * + * @param rpId - The relying party ID. + * @param passkeyConfigRequest - Configuration details for the passkey. + * @param tenantId - (optional) The tenant ID for which the passkey config is created. + * @returns A promise fulfilled with the newly created passkey configuration. + */ public createPasskeyConfig(rpId: string, passkeyConfigRequest: PasskeyConfigRequest, tenantId?: string): Promise { return this.authRequestHandler.updatePasskeyConfig(true, tenantId, passkeyConfigRequest, rpId) @@ -47,6 +71,13 @@ export class PasskeyConfigManager { }); } + /** + * Updates an existing passkey configuration. + * + * @param passkeyConfigRequest - Updated configuration details for the passkey. + * @param tenantId - (optional) The tenant ID for which the passkey config is updated. + * @returns A promise fulfilled with the updated passkey configuration. + */ public updatePasskeyConfig(passkeyConfigRequest: PasskeyConfigRequest, tenantId?: string): Promise { return this.authRequestHandler.updatePasskeyConfig(false, tenantId, passkeyConfigRequest) .then((response: PasskeyConfigClientRequest) => { diff --git a/src/auth/passkey-config.ts b/src/auth/passkey-config.ts index 043189eb99..68e1b66237 100644 --- a/src/auth/passkey-config.ts +++ b/src/auth/passkey-config.ts @@ -17,50 +17,87 @@ import * as validator from '../utils/validator'; import { AuthClientErrorCode, FirebaseAuthError } from '../utils/error'; import { deepCopy } from '../utils/deep-copy'; +/** + * Interface representing the properties to update in the provided passkey config. + */ export interface PasskeyConfigRequest { + /** + * An array of website or app origins associated with the customer's sites or apps. + * Only challenges signed from these origins will be allowed for signing in with passkeys. + */ expectedOrigins?: string[]; } +/** + * Response received from the server when retrieving, creating, or updating the passkey config. + */ export interface PasskeyConfigServerResponse { name?: string; rpId?: string; expectedOrigins?: string[]; } +/** + * Request for creating or updating the passkey config on the server. + */ export interface PasskeyConfigClientRequest { rpId?: string; expectedOrigins?: string[]; } - +/** + * Configuration for signing in users using passkeys. + */ export class PasskeyConfig { + /** + * The name of the PasskeyConfig resource. + */ public readonly name?: string; + /** + * The relying party ID for passkey verifications. + * This cannot be changed once created. + */ public readonly rpId?: string; + /** + * The website or app origins associated with the customer's sites or apps. + * Only challenges signed from these origins will be allowed for signing in with passkeys. + */ public readonly expectedOrigins?: string[]; + /** + * Validates a passkey config request object and throws an error on failure. + * @param isCreateRequest - A boolean indicating if it's a create request or not. + * @param passkeyConfigRequest - Passkey config to be set. + * @param rpId - (optional) Relying party ID if it's a create request. + * @throws FirebaseAuthError - If validation fails. + * + * @internal + */ private static validate(isCreateRequest: boolean, passkeyConfigRequest?: PasskeyConfigRequest, rpId?: string): void { + // Validation for creating a new PasskeyConfig. if (isCreateRequest && !validator.isNonEmptyString(rpId)) { throw new FirebaseAuthError( AuthClientErrorCode.INVALID_ARGUMENT, - '\'rpId\' must be a valid non-empty string\'', + "'rpId' must be a valid non-empty string.", ); } + // Validation for updating an existing PasskeyConfig. if (!isCreateRequest && typeof rpId !== 'undefined') { throw new FirebaseAuthError( AuthClientErrorCode.INVALID_ARGUMENT, - '\'rpId\' cannot be changed once created.\'', + "'rpId' cannot be changed once created.", ); } if (!validator.isNonNullObject(passkeyConfigRequest)) { throw new FirebaseAuthError( AuthClientErrorCode.INVALID_ARGUMENT, - '\'passkeyConfigRequest\' must be a valid non-empty object.\'', + "'passkeyConfigRequest' must be a valid non-empty object.", ); } const validKeys = { expectedOrigins: true, }; - // Check for unsupported top level attributes. + // Check for unsupported top-level attributes. for (const key in passkeyConfigRequest) { if (!(key in validKeys)) { throw new FirebaseAuthError( @@ -72,19 +109,29 @@ export class PasskeyConfig { if (!validator.isNonEmptyArray(passkeyConfigRequest.expectedOrigins)) { throw new FirebaseAuthError( AuthClientErrorCode.INVALID_ARGUMENT, - '\'passkeyConfigRequest.expectedOrigins\' must be a valid non-empty array of strings.\'', + "'passkeyConfigRequest.expectedOrigins' must be a valid non-empty array of strings.", ); } for (const origin of passkeyConfigRequest.expectedOrigins) { if (!validator.isNonEmptyString(origin)) { throw new FirebaseAuthError( AuthClientErrorCode.INVALID_ARGUMENT, - '\'passkeyConfigRequest.expectedOrigins\' must be a valid non-empty array of strings.\'', + "'passkeyConfigRequest.expectedOrigins' must be a valid non-empty array of strings.", ); } } } + /** + * Build the corresponding server request for a Passkey Config object. + * @param isCreateRequest - A boolean stating if it's a create request. + * @param passkeyConfigRequest - Passkey config to be updated. + * @param rpId - (optional) Relying party ID for the request if it's a create request. + * @returns The equivalent server request. + * @throws FirebaseAuthError - If validation fails. + * + * @internal + */ public static buildServerRequest(isCreateRequest: boolean, passkeyConfigRequest?: PasskeyConfigRequest, rpId?: string): PasskeyConfigClientRequest { PasskeyConfig.validate(isCreateRequest, passkeyConfigRequest, rpId); @@ -98,6 +145,13 @@ export class PasskeyConfig { return request; } + /** + * The Passkey Config object constructor. + * @param response - The server-side response used to initialize the Passkey Config object. + * @constructor + * + * @internal + */ constructor(response: PasskeyConfigServerResponse) { if (typeof response.name !== 'undefined') { this.name = response.name; @@ -110,6 +164,10 @@ export class PasskeyConfig { } } + /** + * Returns a JSON-serializable representation of this object. + * @returns A JSON-serializable representation of this object. + */ public toJSON(): object { const json = { name: deepCopy(this.name), @@ -127,6 +185,4 @@ export class PasskeyConfig { } return json; } - } - diff --git a/test/unit/auth/passkey-config.spec.ts b/test/unit/auth/passkey-config.spec.ts index 298615af79..1d61bf6ec0 100644 --- a/test/unit/auth/passkey-config.spec.ts +++ b/test/unit/auth/passkey-config.spec.ts @@ -140,4 +140,4 @@ describe('PasskeyConfig', () => { }); }); }); -}); \ No newline at end of file +});