Skip to content

Commit

Permalink
[identity] Prep for release (#29981)
Browse files Browse the repository at this point in the history
  • Loading branch information
maorleger authored Jun 10, 2024
1 parent 3caf203 commit 9a2afdf
Show file tree
Hide file tree
Showing 8 changed files with 280 additions and 165 deletions.
154 changes: 84 additions & 70 deletions common/config/rush/pnpm-lock.yaml

Large diffs are not rendered by default.

6 changes: 6 additions & 0 deletions sdk/identity/identity/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
# Release History

## 4.2.1 (2024-06-10)

### Bugs Fixed

- Managed identity bug fixes

## 4.2.0 (2024-04-30)

### Features Added
Expand Down
4 changes: 2 additions & 2 deletions sdk/identity/identity/package.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"name": "@azure/identity",
"sdk-type": "client",
"version": "4.2.0",
"version": "4.2.1",
"description": "Provides credential implementations for Azure SDK libraries that can authenticate with Microsoft Entra ID",
"main": "dist/index.js",
"module": "dist-esm/src/index.js",
Expand Down Expand Up @@ -115,7 +115,7 @@
"@azure/core-util": "^1.3.0",
"@azure/logger": "^1.0.0",
"@azure/msal-browser": "^3.11.1",
"@azure/msal-node": "^2.6.6",
"@azure/msal-node": "^2.9.2",
"events": "^3.0.0",
"jws": "^4.0.0",
"open": "^8.0.0",
Expand Down
2 changes: 1 addition & 1 deletion sdk/identity/identity/src/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
/**
* Current version of the `@azure/identity` package.
*/
export const SDK_VERSION = `4.2.0`;
export const SDK_VERSION = `4.2.1`;

/**
* The default client ID for authentication
Expand Down
Original file line number Diff line number Diff line change
@@ -1,19 +1,20 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT license.

import { MSI, MSIConfiguration, MSIToken } from "./models";
import {
PipelineRequestOptions,
createHttpHeaders,
createPipelineRequest,
} from "@azure/core-rest-pipeline";
import { GetTokenOptions } from "@azure/core-auth";
import { readFile } from "fs";

import { AuthenticationError } from "../../errors";
import { credentialLogger } from "../../util/logging";
import { GetTokenOptions } from "@azure/core-auth";
import { IdentityClient } from "../../client/identityClient";
import { mapScopesToResource } from "./utils";
import { MSI, MSIConfiguration, MSIToken } from "./models";
import { azureArcAPIVersion } from "./constants";
import { credentialLogger } from "../../util/logging";
import fs from "node:fs";
import { mapScopesToResource } from "./utils";

const msiName = "ManagedIdentityCredential - Azure Arc MSI";
const logger = credentialLogger(msiName);
Expand Down Expand Up @@ -60,21 +61,6 @@ function prepareRequestOptions(
});
}

/**
* Retrieves the file contents at the given path using promises.
* Useful since `fs`'s readFileSync locks the thread, and to avoid extra dependencies.
*/
function readFileAsync(path: string, options: { encoding: BufferEncoding }): Promise<string> {
return new Promise((resolve, reject) =>
readFile(path, options, (err, data) => {
if (err) {
reject(err);
}
resolve(data);
}),
);
}

/**
* Does a request to the authentication provider that results in a file path.
*/
Expand Down Expand Up @@ -103,6 +89,50 @@ async function filePathRequest(
}
}

export function platformToFilePath(): string {
switch (process.platform) {
case "win32":
if (!process.env.PROGRAMDATA) {
throw new Error(`${msiName}: PROGRAMDATA environment variable has no value.`);
}
return `${process.env.PROGRAMDATA}\\AzureConnectedMachineAgent\\Tokens`;
case "linux":
return "/var/opt/azcmagent/tokens";
default:
throw new Error(`${msiName}: Unsupported platform ${process.platform}.`);
}
}

/**
* Validates that a given Azure Arc MSI file path is valid for use.
*
* A valid file will:
* 1. Be in the expected path for the current platform.
* 2. Have a `.key` extension.
* 3. Be at most 4096 bytes in size.
*/
export function validateKeyFile(filePath?: string): asserts filePath is string {
if (!filePath) {
throw new Error(`${msiName}: Failed to find the token file.`);
}

if (!filePath.endsWith(".key")) {
throw new Error(`${msiName}: unexpected file path from HIMDS service: ${filePath}.`);
}

const expectedPath = platformToFilePath();
if (!filePath.startsWith(expectedPath)) {
throw new Error(`${msiName}: unexpected file path from HIMDS service: ${filePath}.`);
}

const stats = fs.statSync(filePath);
if (stats.size > 4096) {
throw new Error(
`${msiName}: The file at ${filePath} is larger than expected at ${stats.size} bytes.`,
);
}
}

/**
* Defines how to determine whether the Azure Arc MSI is available, and also how to retrieve a token from the Azure Arc MSI.
*/
Expand Down Expand Up @@ -150,12 +180,9 @@ export const arcMsi: MSI = {
};

const filePath = await filePathRequest(identityClient, requestOptions);
validateKeyFile(filePath);

if (!filePath) {
throw new Error(`${msiName}: Failed to find the token file.`);
}

const key = await readFileAsync(filePath, { encoding: "utf-8" });
const key = await fs.promises.readFile(filePath, { encoding: "utf-8" });
requestOptions.headers?.set("Authorization", `Basic ${key}`);

const request = createPipelineRequest({
Expand Down
Loading

0 comments on commit 9a2afdf

Please sign in to comment.