UNPKG

firebase-admin

Version:
239 lines (238 loc) 9.43 kB
/*! firebase-admin v14.0.0 */ "use strict"; /*! * @license * Copyright 2021 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. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.AppCheckApiClient = void 0; const api_request_1 = require("../utils/api-request"); const error_1 = require("../utils/error"); const error_2 = require("./error"); const utils = require("../utils/index"); const validator = require("../utils/validator"); // App Check backend constants const FIREBASE_APP_CHECK_V1_API_URL_FORMAT = 'https://firebaseappcheck.googleapis.com/v1/projects/{projectId}/apps/{appId}:exchangeCustomToken'; const ONE_TIME_USE_TOKEN_VERIFICATION_URL_FORMAT = 'https://firebaseappcheck.googleapis.com/v1beta/projects/{projectId}:verifyAppCheckToken'; const FIREBASE_APP_CHECK_CONFIG_HEADERS = { 'X-Firebase-Client': `fire-admin-node/${utils.getSdkVersion()}` }; /** * Class that facilitates sending requests to the Firebase App Check backend API. * * @internal */ class AppCheckApiClient { constructor(app) { this.app = app; if (!validator.isNonNullObject(app) || !('options' in app)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: 'First argument passed to admin.appCheck() must be a valid Firebase app instance.' }); } this.httpClient = new api_request_1.AuthorizedHttpClient(app); } /** * Exchange a signed custom token to App Check token * * @param customToken - The custom token to be exchanged. * @param appId - The mobile App ID. * @returns A promise that fulfills with a `AppCheckToken`. */ exchangeToken(customToken, appId, options) { if (!validator.isNonEmptyString(appId)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`appId` must be a non-empty string.' }); } if (!validator.isNonEmptyString(customToken)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`customToken` must be a non-empty string.' }); } if (typeof options?.limitedUse !== 'undefined' && !validator.isBoolean(options.limitedUse)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`limitedUse` must be a boolean value.' }); } if (typeof options?.jti !== 'undefined') { if (!validator.isString(options.jti)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`jti` must be a string value.' }); } if (!options.limitedUse) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`jti` cannot be specified without setting `limitedUse` to `true`.' }); } } return this.getUrl(appId) .then((url) => { const request = { method: 'POST', url, headers: FIREBASE_APP_CHECK_CONFIG_HEADERS, data: { customToken, ...(options?.limitedUse !== undefined && { limitedUse: options.limitedUse }), ...(options?.jti !== undefined && { jti: options.jti }), } }; return this.httpClient.send(request); }) .then((resp) => { return this.toAppCheckToken(resp); }) .catch((err) => { throw this.toFirebaseError(err); }); } verifyReplayProtection(token) { if (!validator.isNonEmptyString(token)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`token` must be a non-empty string.' }); } return this.getVerifyTokenUrl() .then((url) => { const request = { method: 'POST', url, headers: FIREBASE_APP_CHECK_CONFIG_HEADERS, data: { app_check_token: token } }; return this.httpClient.send(request); }) .then((resp) => { if (typeof resp.data.alreadyConsumed !== 'undefined' && !validator.isBoolean(resp.data?.alreadyConsumed)) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`alreadyConsumed` must be a boolean value.', httpResponse: (0, error_1.toHttpResponse)(resp) }); } return resp.data.alreadyConsumed || false; }) .catch((err) => { throw this.toFirebaseError(err); }); } getUrl(appId) { return this.getProjectId() .then((projectId) => { const urlParams = { projectId, appId, }; const baseUrl = utils.formatString(FIREBASE_APP_CHECK_V1_API_URL_FORMAT, urlParams); return utils.formatString(baseUrl); }); } getVerifyTokenUrl() { return this.getProjectId() .then((projectId) => { const urlParams = { projectId }; const baseUrl = utils.formatString(ONE_TIME_USE_TOKEN_VERIFICATION_URL_FORMAT, urlParams); return utils.formatString(baseUrl); }); } getProjectId() { if (this.projectId) { return Promise.resolve(this.projectId); } return utils.findProjectId(this.app) .then((projectId) => { if (!validator.isNonEmptyString(projectId)) { throw new error_2.FirebaseAppCheckError({ code: 'unknown-error', message: 'Failed to determine project ID. Initialize the ' + 'SDK with service account credentials or set project ID as an app option. ' + 'Alternatively, set the GOOGLE_CLOUD_PROJECT environment variable.' }); } this.projectId = projectId; return projectId; }); } toFirebaseError(err) { if (err instanceof error_1.FirebaseError) { return err; } const response = err.response; if (!response.isJson()) { return new error_2.FirebaseAppCheckError({ code: 'unknown-error', message: `Unexpected response with status: ${response.status} and body: ${response.text}`, httpResponse: (0, error_1.toHttpResponse)(response), cause: err }); } const error = response.data.error || {}; let code = 'unknown-error'; if (error.status && error.status in error_2.APP_CHECK_ERROR_CODE_MAPPING) { code = error_2.APP_CHECK_ERROR_CODE_MAPPING[error.status]; } const message = error.message || 'Unknown server error'; return new error_2.FirebaseAppCheckError({ code, message, httpResponse: (0, error_1.toHttpResponse)(response), cause: err }); } /** * Creates an AppCheckToken from the API response. * * @param resp - API response object. * @returns An AppCheckToken instance. */ toAppCheckToken(resp) { const token = resp.data.token; // `ttl` is a string with the suffix "s" preceded by the number of seconds, // with nanoseconds expressed as fractional seconds. const ttlMillis = this.stringToMilliseconds(resp.data.ttl); return { token, ttlMillis }; } /** * Converts a duration string with the suffix `s` to milliseconds. * * @param duration - The duration as a string with the suffix "s" preceded by the * number of seconds, with fractional seconds. For example, 3 seconds with 0 nanoseconds * is expressed as "3s", while 3 seconds and 1 nanosecond is expressed as "3.000000001s", * and 3 seconds and 1 microsecond is expressed as "3.000001s". * * @returns The duration in milliseconds. */ stringToMilliseconds(duration) { if (!validator.isNonEmptyString(duration) || !duration.endsWith('s')) { throw new error_2.FirebaseAppCheckError({ code: 'invalid-argument', message: '`ttl` must be a valid duration string with the suffix `s`.' }); } const seconds = duration.slice(0, -1); return Math.floor(Number(seconds) * 1000); } } exports.AppCheckApiClient = AppCheckApiClient;