UNPKG

@azure/ms-rest-nodeauth

Version:

Azure Authentication library in node.js with type definitions.

154 lines (141 loc) 6.32 kB
// Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. See License.txt in the project root for license information. import { Constants, WebResource, HttpClient, DefaultHttpClient } from "@azure/ms-rest-js"; import { TokenClientCredentials, TokenResponse } from "./tokenClientCredentials"; import { AuthConstants } from "../util/authConstants"; /** * Defines the optional parameters for authentication with MSI. */ export interface MSIOptions { /** * @prop {string} [resource] - The resource uri or token audience for which the token is needed. * For example: * - Defaults to Azure Resource Manager from environment: AzureCloud. "https://management.azure.com/" (default) * - For ServiceManagement (ASM): "https://management.core.windows.net/" * - For Azure KeyVault: "https://vault.azure.net" * - For Azure Batch: "https://batch.core.windows.net" * - For Azure Active Directory Graph: "https://graph.windows.net" */ resource?: string; /** * The client responsible for sending HTTP requests. * By default it is Axios-based {@link DefaultHttpClient}. */ httpClient?: HttpClient; } /** * Describes the MSITokenResponse. */ export interface MSITokenResponse extends TokenResponse { /** * Placeholder for unknown properties. */ readonly [x: string]: any; } /** * Provides information about managed service identity token credentials. * This object can only be used to acquire token on a virtual machine provisioned in Azure with managed service identity. */ export abstract class MSITokenCredentials implements TokenClientCredentials { /** * Azure resource endpoints. * - Defaults to Azure Resource Manager from environment: AzureCloud. "https://management.azure.com/" * - For ServiceManagement (ASM): "https://management.core.windows.net/" * - For Azure KeyVault: "https://vault.azure.net" * - For Azure Batch: "https://batch.core.windows.net" * - For Azure Active Directory Graph: "https://graph.windows.net" */ resource: string; protected _httpClient: HttpClient; /** * Creates an instance of MSITokenCredentials. * @param options - Optional parameters * @param options.resource - The resource uri or token audience for which the token is needed. * For e.g. it can be: * - resource management endpoint "https://management.azure.com/"(default) * - management endpoint "https://management.core.windows.net/" */ constructor(options: MSIOptions) { if (!options) options = {}; if (!options.resource) { options.resource = AuthConstants.RESOURCE_MANAGER_ENDPOINT; } else if (typeof options.resource.valueOf() !== "string") { throw new Error("resource must be a uri."); } this.resource = options.resource; this._httpClient = options.httpClient || new DefaultHttpClient(); } /** * Parses a tokenResponse json string into a object, and converts properties on the first level to camelCase. * This method tries to standardize the tokenResponse * @param body - A json string * @returns The tokenResponse (tokenType and accessToken are the two important properties). */ parseTokenResponse(body: string): TokenResponse { // Docs show different examples of possible MSI responses for different services. https://docs.microsoft.com/en-us/azure/active-directory/managed-service-identity/overview // expires_on - is a Date like string in this doc // - https://docs.microsoft.com/en-us/azure/app-service/app-service-managed-service-identity#rest-protocol-examples // In other doc it is stringified number. // - https://docs.microsoft.com/en-us/azure/active-directory/managed-service-identity/tutorial-linux-vm-access-arm#get-an-access-token-using-the-vms-identity-and-use-it-to-call-resource-manager const parsedBody = JSON.parse(body); parsedBody.accessToken = parsedBody["access_token"]; delete parsedBody["access_token"]; parsedBody.tokenType = parsedBody["token_type"]; delete parsedBody["token_type"]; if (parsedBody["refresh_token"]) { parsedBody.refreshToken = parsedBody["refresh_token"]; delete parsedBody["refresh_token"]; } if (parsedBody["expires_in"]) { parsedBody.expiresIn = parsedBody["expires_in"]; if (typeof parsedBody["expires_in"] === "string") { // normal number as a string '1504130527' parsedBody.expiresIn = parseInt(parsedBody["expires_in"], 10); } delete parsedBody["expires_in"]; } if (parsedBody["not_before"]) { parsedBody.notBefore = parsedBody["not_before"]; if (typeof parsedBody["not_before"] === "string") { // normal number as a string '1504130527' parsedBody.notBefore = parseInt(parsedBody["not_before"], 10); } delete parsedBody["not_before"]; } if (parsedBody["expires_on"]) { parsedBody.expiresOn = parsedBody["expires_on"]; if (typeof parsedBody["expires_on"] === "string") { // possibly a Date string '09/14/2017 00:00:00 PM +00:00' if (parsedBody["expires_on"].includes(":") || parsedBody["expires_on"].includes("/")) { parsedBody.expiresOn = new Date(parseInt(parsedBody["expires_on"], 10)); } else { // normal number as a string '1504130527' parsedBody.expiresOn = new Date(parseInt(parsedBody["expires_on"], 10)); } } delete parsedBody["expires_on"]; } return parsedBody; } /** * Prepares and sends a POST request to a service endpoint hosted on the Azure VM, which responds with the access token. * @param callback - The callback in the form (err, result) * @returns Promise with the token response. */ abstract async getToken(): Promise<MSITokenResponse>; protected abstract prepareRequestOptions(): WebResource; /** * Signs a request with the Authentication header. * * @param webResource - The WebResource to be signed. * @returns Promise with signed WebResource. */ public async signRequest(webResource: WebResource): Promise<WebResource> { const tokenResponse = await this.getToken(); webResource.headers.set( Constants.HeaderConstants.AUTHORIZATION, `${tokenResponse.tokenType} ${tokenResponse.accessToken}` ); return webResource; } }