UNPKG

quodolores

Version:

Monorepo for the Firebase JavaScript SDK

246 lines (230 loc) 7.48 kB
/** * @license * Copyright 2019 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 { AuthProvider, UserCredential } from '../../model/public_types'; import { _assert } from '../util/assert'; import { AuthErrorCode } from '../errors'; import { OAuthCredential, OAuthCredentialParams } from '../credentials/oauth'; import { UserCredentialInternal } from '../../model/user'; import { FirebaseError } from '@firebase/util'; import { TaggedWithTokenResponse } from '../../model/id_token'; import { SignInWithIdpResponse } from '../../../internal'; import { FederatedAuthProvider } from './federated'; /** * Defines the options for initializing an {@link OAuthCredential}. * * @remarks * For ID tokens with nonce claim, the raw nonce has to also be provided. * * @public */ export interface OAuthCredentialOptions { /** * The OAuth ID token used to initialize the {@link OAuthCredential}. */ idToken?: string; /** * The OAuth access token used to initialize the {@link OAuthCredential}. */ accessToken?: string; /** * The raw nonce associated with the ID token. * * @remarks * It is required when an ID token with a nonce field is provided. The SHA-256 hash of the * raw nonce must match the nonce field in the ID token. */ rawNonce?: string; } /** * Common code to all OAuth providers. This is separate from the * {@link OAuthProvider} so that child providers (like * {@link GoogleAuthProvider}) don't inherit the `credential` instance method. * Instead, they rely on a static `credential` method. */ export abstract class BaseOAuthProvider extends FederatedAuthProvider implements AuthProvider { /** @internal */ private scopes: string[] = []; /** * Add an OAuth scope to the credential. * * @param scope - Provider OAuth scope to add. */ addScope(scope: string): AuthProvider { // If not already added, add scope to list. if (!this.scopes.includes(scope)) { this.scopes.push(scope); } return this; } /** * Retrieve the current list of OAuth scopes. */ getScopes(): string[] { return [...this.scopes]; } } /** * Provider for generating generic {@link OAuthCredential}. * * @example * ```javascript * // Sign in using a redirect. * const provider = new OAuthProvider('google.com'); * // Start a sign in process for an unauthenticated user. * provider.addScope('profile'); * provider.addScope('email'); * await signInWithRedirect(auth, provider); * // This will trigger a full page redirect away from your app * * // After returning from the redirect when your app initializes you can obtain the result * const result = await getRedirectResult(auth); * if (result) { * // This is the signed-in user * const user = result.user; * // This gives you a OAuth Access Token for the provider. * const credential = provider.credentialFromResult(auth, result); * const token = credential.accessToken; * } * ``` * * @example * ```javascript * // Sign in using a popup. * const provider = new OAuthProvider('google.com'); * provider.addScope('profile'); * provider.addScope('email'); * const result = await signInWithPopup(auth, provider); * * // The signed-in user info. * const user = result.user; * // This gives you a OAuth Access Token for the provider. * const credential = provider.credentialFromResult(auth, result); * const token = credential.accessToken; * ``` * @public */ export class OAuthProvider extends BaseOAuthProvider { /** * Creates an {@link OAuthCredential} from a JSON string or a plain object. * @param json A plain object or a JSON string */ static credentialFromJSON(json: object | string): OAuthCredential { const obj = typeof json === 'string' ? JSON.parse(json) : json; _assert( 'providerId' in obj && 'signInMethod' in obj, AuthErrorCode.ARGUMENT_ERROR ); return OAuthCredential._fromParams(obj); } /** * Creates a {@link OAuthCredential} from a generic OAuth provider's access token or ID token. * * @remarks * The raw nonce is required when an ID token with a nonce field is provided. The SHA-256 hash of * the raw nonce must match the nonce field in the ID token. * * @example * ```javascript * // `googleUser` from the onsuccess Google Sign In callback. * // Initialize a generate OAuth provider with a `google.com` providerId. * const provider = new OAuthProvider('google.com'); * const credential = provider.credential({ * idToken: googleUser.getAuthResponse().id_token, * }); * const result = await signInWithCredential(credential); * ``` * * @param params - Either the options object containing the ID token, access token and raw nonce * or the ID token string. */ credential(params: OAuthCredentialOptions): OAuthCredential { return this._credential(params); } /** An internal credential method that accepts more permissive options */ private _credential( params: OAuthCredentialOptions | OAuthCredentialParams ): OAuthCredential { _assert(params.idToken || params.accessToken, AuthErrorCode.ARGUMENT_ERROR); // For OAuthCredential, sign in method is same as providerId. return OAuthCredential._fromParams({ ...params, providerId: this.providerId, signInMethod: this.providerId }); } /** * Used to extract the underlying {@link OAuthCredential} from a {@link UserCredential}. * * @param userCredential - The user credential. */ static credentialFromResult( userCredential: UserCredential ): OAuthCredential | null { return OAuthProvider.oauthCredentialFromTaggedObject( userCredential as UserCredentialInternal ); } /** * Used to extract the underlying {@link OAuthCredential} from a {@link AuthError} which was * thrown during a sign-in, link, or reauthenticate operation. * * @param userCredential - The user credential. */ static credentialFromError(error: FirebaseError): OAuthCredential | null { return OAuthProvider.oauthCredentialFromTaggedObject( (error.customData || {}) as TaggedWithTokenResponse ); } private static oauthCredentialFromTaggedObject({ _tokenResponse: tokenResponse }: TaggedWithTokenResponse): OAuthCredential | null { if (!tokenResponse) { return null; } const { oauthIdToken, oauthAccessToken, oauthTokenSecret, pendingToken, nonce, providerId } = tokenResponse as SignInWithIdpResponse; if ( !oauthAccessToken && !oauthTokenSecret && !oauthIdToken && !pendingToken ) { return null; } if (!providerId) { return null; } try { return new OAuthProvider(providerId)._credential({ idToken: oauthIdToken, accessToken: oauthAccessToken, rawNonce: nonce, pendingToken }); } catch (e) { return null; } } }