@azure/communication-common 2.4.2-alpha.20260211.1 → 2.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (90) hide show
  1. package/README.md +1 -1
  2. package/dist/browser/credential/communicationAccessKeyCredentialPolicy.d.ts.map +1 -1
  3. package/dist/browser/credential/communicationAccessKeyCredentialPolicy.js +6 -10
  4. package/dist/browser/credential/communicationAccessKeyCredentialPolicy.js.map +1 -1
  5. package/dist/browser/entraTokenCredential.d.ts.map +1 -1
  6. package/dist/browser/entraTokenCredential.js +12 -7
  7. package/dist/browser/entraTokenCredential.js.map +1 -1
  8. package/dist/browser/index.d.ts +3 -3
  9. package/dist/browser/index.d.ts.map +1 -1
  10. package/dist/browser/index.js.map +1 -1
  11. package/dist/commonjs/autoRefreshTokenCredential.js +110 -95
  12. package/dist/commonjs/autoRefreshTokenCredential.js.map +7 -1
  13. package/dist/commonjs/azureCommunicationTokenCredential.js +64 -47
  14. package/dist/commonjs/azureCommunicationTokenCredential.js.map +7 -1
  15. package/dist/commonjs/communicationTokenCredential.js +16 -5
  16. package/dist/commonjs/communicationTokenCredential.js.map +7 -1
  17. package/dist/commonjs/credential/clientArguments.js +46 -40
  18. package/dist/commonjs/credential/clientArguments.js.map +7 -1
  19. package/dist/commonjs/credential/communicationAccessKeyCredentialPolicy.d.ts.map +1 -1
  20. package/dist/commonjs/credential/communicationAccessKeyCredentialPolicy.js +53 -42
  21. package/dist/commonjs/credential/communicationAccessKeyCredentialPolicy.js.map +7 -1
  22. package/dist/commonjs/credential/communicationAuthPolicy.js +39 -26
  23. package/dist/commonjs/credential/communicationAuthPolicy.js.map +7 -1
  24. package/dist/commonjs/credential/connectionString.js +39 -28
  25. package/dist/commonjs/credential/connectionString.js.map +7 -1
  26. package/dist/commonjs/credential/index.js +28 -10
  27. package/dist/commonjs/credential/index.js.map +7 -1
  28. package/dist/commonjs/entraTokenCredential.d.ts.map +1 -1
  29. package/dist/commonjs/entraTokenCredential.js +137 -113
  30. package/dist/commonjs/entraTokenCredential.js.map +7 -1
  31. package/dist/commonjs/identifierModelSerializer.js +172 -159
  32. package/dist/commonjs/identifierModelSerializer.js.map +7 -1
  33. package/dist/commonjs/identifierModels.js +183 -205
  34. package/dist/commonjs/identifierModels.js.map +7 -1
  35. package/dist/commonjs/index.d.ts +3 -3
  36. package/dist/commonjs/index.d.ts.map +1 -1
  37. package/dist/commonjs/index.js +35 -12
  38. package/dist/commonjs/index.js.map +7 -1
  39. package/dist/commonjs/staticTokenCredential.js +36 -20
  40. package/dist/commonjs/staticTokenCredential.js.map +7 -1
  41. package/dist/commonjs/tokenParser.js +33 -13
  42. package/dist/commonjs/tokenParser.js.map +7 -1
  43. package/dist/commonjs/tsdoc-metadata.json +1 -1
  44. package/dist/esm/credential/communicationAccessKeyCredentialPolicy.d.ts.map +1 -1
  45. package/dist/esm/credential/communicationAccessKeyCredentialPolicy.js +6 -10
  46. package/dist/esm/credential/communicationAccessKeyCredentialPolicy.js.map +1 -1
  47. package/dist/esm/entraTokenCredential.d.ts.map +1 -1
  48. package/dist/esm/entraTokenCredential.js +12 -7
  49. package/dist/esm/entraTokenCredential.js.map +1 -1
  50. package/dist/esm/index.d.ts +3 -3
  51. package/dist/esm/index.d.ts.map +1 -1
  52. package/dist/esm/index.js.map +1 -1
  53. package/dist/react-native/credential/communicationAccessKeyCredentialPolicy.d.ts.map +1 -1
  54. package/dist/react-native/credential/communicationAccessKeyCredentialPolicy.js +6 -10
  55. package/dist/react-native/credential/communicationAccessKeyCredentialPolicy.js.map +1 -1
  56. package/dist/react-native/entraTokenCredential.d.ts.map +1 -1
  57. package/dist/react-native/entraTokenCredential.js +12 -7
  58. package/dist/react-native/entraTokenCredential.js.map +1 -1
  59. package/dist/react-native/index.d.ts +3 -3
  60. package/dist/react-native/index.d.ts.map +1 -1
  61. package/dist/react-native/index.js.map +1 -1
  62. package/package.json +28 -35
  63. package/dist/browser/credential/cryptoUtils-browser.d.mts.map +0 -1
  64. package/dist/browser/credential/cryptoUtils-browser.mjs.map +0 -1
  65. package/dist/browser/credential/cryptoUtils.d.ts +0 -3
  66. package/dist/browser/credential/cryptoUtils.js +0 -19
  67. package/dist/browser/credential/encodeUtils-browser.d.mts +0 -4
  68. package/dist/browser/credential/encodeUtils-browser.d.mts.map +0 -1
  69. package/dist/browser/credential/encodeUtils-browser.mjs +0 -19
  70. package/dist/browser/credential/encodeUtils-browser.mjs.map +0 -1
  71. package/dist/commonjs/credential/cryptoUtils.d.ts +0 -3
  72. package/dist/commonjs/credential/cryptoUtils.d.ts.map +0 -1
  73. package/dist/commonjs/credential/cryptoUtils.js +0 -14
  74. package/dist/commonjs/credential/cryptoUtils.js.map +0 -1
  75. package/dist/esm/credential/cryptoUtils.d.ts +0 -3
  76. package/dist/esm/credential/cryptoUtils.d.ts.map +0 -1
  77. package/dist/esm/credential/cryptoUtils.js +0 -9
  78. package/dist/esm/credential/cryptoUtils.js.map +0 -1
  79. package/dist/esm/credential/encodeUtils-browser.d.mts +0 -4
  80. package/dist/esm/credential/encodeUtils-browser.d.mts.map +0 -1
  81. package/dist/esm/credential/encodeUtils-browser.mjs +0 -19
  82. package/dist/esm/credential/encodeUtils-browser.mjs.map +0 -1
  83. package/dist/react-native/credential/cryptoUtils.d.ts +0 -3
  84. package/dist/react-native/credential/cryptoUtils.d.ts.map +0 -1
  85. package/dist/react-native/credential/cryptoUtils.js +0 -9
  86. package/dist/react-native/credential/cryptoUtils.js.map +0 -1
  87. package/dist/react-native/credential/encodeUtils-browser.d.mts +0 -4
  88. package/dist/react-native/credential/encodeUtils-browser.d.mts.map +0 -1
  89. package/dist/react-native/credential/encodeUtils-browser.mjs +0 -19
  90. package/dist/react-native/credential/encodeUtils-browser.mjs.map +0 -1
@@ -1,127 +1,151 @@
1
- "use strict";
2
- // Copyright (c) Microsoft Corporation.
3
- // Licensed under the MIT License.
4
- Object.defineProperty(exports, "__esModule", { value: true });
5
- exports.EntraTokenCredential = void 0;
6
- const core_client_1 = require("@azure-rest/core-client");
7
- const core_rest_pipeline_1 = require("@azure/core-rest-pipeline");
8
- const TeamsExtensionScopePrefix = "https://auth.msft.communication.azure.com/";
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
+ var __getOwnPropNames = Object.getOwnPropertyNames;
4
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
5
+ var __export = (target, all) => {
6
+ for (var name in all)
7
+ __defProp(target, name, { get: all[name], enumerable: true });
8
+ };
9
+ var __copyProps = (to, from, except, desc) => {
10
+ if (from && typeof from === "object" || typeof from === "function") {
11
+ for (let key of __getOwnPropNames(from))
12
+ if (!__hasOwnProp.call(to, key) && key !== except)
13
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
14
+ }
15
+ return to;
16
+ };
17
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
+ var entraTokenCredential_exports = {};
19
+ __export(entraTokenCredential_exports, {
20
+ EntraTokenCredential: () => EntraTokenCredential
21
+ });
22
+ module.exports = __toCommonJS(entraTokenCredential_exports);
23
+ var import_core_client = require("@azure-rest/core-client");
24
+ var import_core_rest_pipeline = require("@azure/core-rest-pipeline");
25
+ const TeamsExtensionScopePrefixes = [
26
+ "https://auth.msft.communication.azure.com/",
27
+ "https://auth.msft.communication.azure.us/"
28
+ ];
9
29
  const CommunicationClientsScopePrefix = "https://communication.azure.com/clients/";
30
+ const ScopeValidationErrorMessage = `Scopes validation failed. Ensure all scopes start with one of ${[
31
+ ...TeamsExtensionScopePrefixes,
32
+ CommunicationClientsScopePrefix
33
+ ].join(", ")}.`;
10
34
  const TeamsExtensionEndpoint = "/access/teamsExtension/:exchangeAccessToken";
11
35
  const TeamsExtensionApiVersion = "2025-06-30";
12
36
  const CommunicationClientsEndpoint = "/access/entra/:exchangeAccessToken";
13
37
  const CommunicationClientsApiVersion = "2025-03-02-preview";
14
- /**
15
- * Represents a credential that exchanges an Entra token for an Azure Communication Services (ACS) token, enabling access to ACS resources.
16
- */
17
38
  class EntraTokenCredential {
18
- options;
19
- isPending;
20
- result = {
21
- entraToken: undefined,
22
- acsToken: { token: "", expiresOnTimestamp: 0 },
23
- };
24
- client;
25
- httpClient;
26
- constructor(options) {
27
- this.options = options;
28
- this.client = (0, core_client_1.getClient)(options.resourceEndpoint);
29
- this.httpClient = (0, core_rest_pipeline_1.createDefaultHttpClient)();
30
- this.options = options;
31
- this.options.scopes = this.options.scopes || [
32
- "https://communication.azure.com/clients/.default",
33
- ];
34
- // immediately fetch the token to pre-warm
35
- this.isPending = this.getToken();
39
+ constructor(options) {
40
+ this.options = options;
41
+ this.client = (0, import_core_client.getClient)(options.resourceEndpoint);
42
+ this.httpClient = (0, import_core_rest_pipeline.createDefaultHttpClient)();
43
+ this.options = options;
44
+ this.options.scopes = this.options.scopes || [
45
+ "https://communication.azure.com/clients/.default"
46
+ ];
47
+ this.isPending = this.getToken();
48
+ }
49
+ options;
50
+ isPending;
51
+ result = {
52
+ entraToken: void 0,
53
+ acsToken: { token: "", expiresOnTimestamp: 0 }
54
+ };
55
+ client;
56
+ httpClient;
57
+ async getToken(options) {
58
+ if (options?.abortSignal?.aborted) {
59
+ return { token: "", expiresOnTimestamp: 0 };
60
+ }
61
+ if (!this.isPending) {
62
+ this.isPending = this.getTokenInternal(options);
36
63
  }
37
- async getToken(options) {
38
- if (options?.abortSignal?.aborted) {
39
- return { token: "", expiresOnTimestamp: 0 };
40
- }
41
- // we're awaiting the token fetch, so we don't want to start another one
42
- // however, we're ignoring the new abortSignal, unfortunately
43
- if (!this.isPending) {
44
- this.isPending = this.getTokenInternal(options);
45
- }
46
- try {
47
- await this.isPending;
48
- }
49
- finally {
50
- this.isPending = null;
51
- }
52
- return this.result.acsToken;
64
+ try {
65
+ await this.isPending;
66
+ } finally {
67
+ this.isPending = null;
53
68
  }
54
- async getTokenInternal(options) {
55
- const getTokenOptions = options?.abortSignal ? { abortSignal: options.abortSignal } : undefined;
56
- const token = await this.options.tokenCredential.getToken(this.options.scopes
57
- ? this.options.scopes
58
- : ["https://communication.azure.com/clients/.default"], getTokenOptions);
59
- const currentDateTime = new Date();
60
- const tokenExpiresOn = new Date(this.result.acsToken.expiresOnTimestamp);
61
- if (token === null) {
62
- this.result = {
63
- entraToken: undefined,
64
- acsToken: { token: "", expiresOnTimestamp: 0 },
65
- };
66
- }
67
- else if (this.result.acsToken.token === "" ||
68
- token.token !== this.result.entraToken ||
69
- tokenExpiresOn < currentDateTime) {
70
- const acsToken = await this.exchangeEntraToken(this.options.resourceEndpoint, token.token, getTokenOptions);
71
- this.result = {
72
- entraToken: token.token,
73
- acsToken,
74
- };
75
- }
76
- return this.result.acsToken;
69
+ return this.result.acsToken;
70
+ }
71
+ async getTokenInternal(options) {
72
+ const getTokenOptions = options?.abortSignal ? { abortSignal: options.abortSignal } : void 0;
73
+ const token = await this.options.tokenCredential.getToken(
74
+ this.options.scopes ? this.options.scopes : ["https://communication.azure.com/clients/.default"],
75
+ getTokenOptions
76
+ );
77
+ const currentDateTime = /* @__PURE__ */ new Date();
78
+ const tokenExpiresOn = new Date(this.result.acsToken.expiresOnTimestamp);
79
+ if (token === null) {
80
+ this.result = {
81
+ entraToken: void 0,
82
+ acsToken: { token: "", expiresOnTimestamp: 0 }
83
+ };
84
+ } else if (this.result.acsToken.token === "" || token.token !== this.result.entraToken || tokenExpiresOn < currentDateTime) {
85
+ const acsToken = await this.exchangeEntraToken(
86
+ this.options.resourceEndpoint,
87
+ token.token,
88
+ getTokenOptions
89
+ );
90
+ this.result = {
91
+ entraToken: token.token,
92
+ acsToken
93
+ };
77
94
  }
78
- dispose() {
79
- this.result = {
80
- entraToken: undefined,
81
- acsToken: { token: "", expiresOnTimestamp: 0 },
82
- };
95
+ return this.result.acsToken;
96
+ }
97
+ dispose() {
98
+ this.result = {
99
+ entraToken: void 0,
100
+ acsToken: { token: "", expiresOnTimestamp: 0 }
101
+ };
102
+ }
103
+ async exchangeEntraToken(resourceEndpoint, entraToken, options) {
104
+ const request = (0, import_core_rest_pipeline.createPipelineRequest)({
105
+ url: this.createRequestUri(resourceEndpoint),
106
+ method: "POST",
107
+ headers: (0, import_core_rest_pipeline.createHttpHeaders)({
108
+ Authorization: `Bearer ${entraToken}`,
109
+ "Content-Type": "application/json",
110
+ Accept: "application/json"
111
+ }),
112
+ abortSignal: options?.abortSignal,
113
+ body: JSON.stringify({})
114
+ });
115
+ const response = await this.client.pipeline.sendRequest(this.httpClient, request);
116
+ if (response.status !== 200 || !response.bodyAsText) {
117
+ throw new Error(
118
+ `Service request failed. Status: ${response.status}, Body: ${response.bodyAsText}`
119
+ );
83
120
  }
84
- async exchangeEntraToken(resourceEndpoint, entraToken, options) {
85
- const request = (0, core_rest_pipeline_1.createPipelineRequest)({
86
- url: this.createRequestUri(resourceEndpoint),
87
- method: "POST",
88
- headers: (0, core_rest_pipeline_1.createHttpHeaders)({
89
- Authorization: `Bearer ${entraToken}`,
90
- "Content-Type": "application/json",
91
- Accept: "application/json",
92
- }),
93
- abortSignal: options?.abortSignal,
94
- body: JSON.stringify({}),
95
- });
96
- const response = await this.client.pipeline.sendRequest(this.httpClient, request);
97
- if (response.status !== 200 || !response.bodyAsText) {
98
- throw new Error(`Service request failed. Status: ${response.status}, Body: ${response.bodyAsText}`);
99
- }
100
- const json = JSON.parse(response.bodyAsText);
101
- return {
102
- token: json.accessToken.token,
103
- expiresOnTimestamp: Date.parse(json.accessToken.expiresOn),
104
- };
121
+ const json = JSON.parse(response.bodyAsText);
122
+ return {
123
+ token: json.accessToken.token,
124
+ expiresOnTimestamp: Date.parse(json.accessToken.expiresOn)
125
+ };
126
+ }
127
+ createRequestUri(resourceEndpoint) {
128
+ const [endpoint, apiVersion] = this.determineEndpointAndApiVersion();
129
+ const requestUri = `${resourceEndpoint}${endpoint}?api-version=${apiVersion}`;
130
+ return requestUri;
131
+ }
132
+ determineEndpointAndApiVersion() {
133
+ if (!this.options.scopes || this.options.scopes.length === 0) {
134
+ throw new Error(ScopeValidationErrorMessage);
105
135
  }
106
- createRequestUri(resourceEndpoint) {
107
- const [endpoint, apiVersion] = this.determineEndpointAndApiVersion();
108
- const requestUri = `${resourceEndpoint}${endpoint}?api-version=${apiVersion}`;
109
- return requestUri;
136
+ if (this.options.scopes.every(
137
+ (scope) => TeamsExtensionScopePrefixes.some((prefix) => scope.startsWith(prefix))
138
+ )) {
139
+ return [TeamsExtensionEndpoint, TeamsExtensionApiVersion];
110
140
  }
111
- determineEndpointAndApiVersion() {
112
- if (!this.options.scopes || this.options.scopes.length === 0) {
113
- throw new Error(`Scopes validation failed. Ensure all scopes start with either {TeamsExtensionScopePrefix} or {CommunicationClientsScopePrefix}.`);
114
- }
115
- else if (this.options.scopes.every((scope) => scope.startsWith(TeamsExtensionScopePrefix))) {
116
- return [TeamsExtensionEndpoint, TeamsExtensionApiVersion];
117
- }
118
- else if (this.options.scopes.every((scope) => scope.startsWith(CommunicationClientsScopePrefix))) {
119
- return [CommunicationClientsEndpoint, CommunicationClientsApiVersion];
120
- }
121
- else {
122
- throw new Error(`Scopes validation failed. Ensure all scopes start with either {TeamsExtensionScopePrefix} or {CommunicationClientsScopePrefix}.`);
123
- }
141
+ if (this.options.scopes.every((scope) => scope.startsWith(CommunicationClientsScopePrefix))) {
142
+ return [CommunicationClientsEndpoint, CommunicationClientsApiVersion];
124
143
  }
144
+ throw new Error(ScopeValidationErrorMessage);
145
+ }
125
146
  }
126
- exports.EntraTokenCredential = EntraTokenCredential;
127
- //# sourceMappingURL=entraTokenCredential.js.map
147
+ // Annotate the CommonJS export names for ESM import in node:
148
+ 0 && (module.exports = {
149
+ EntraTokenCredential
150
+ });
151
+ //# sourceMappingURL=entraTokenCredential.js.map
@@ -1 +1,7 @@
1
- {"version":3,"file":"entraTokenCredential.js","sourceRoot":"","sources":["../../src/entraTokenCredential.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) Microsoft Corporation.\n// Licensed under the MIT License.\n\nimport { type AccessToken, type TokenCredential } from \"@azure/core-auth\";\nimport {\n type TokenCredential as AcsTokenCredential,\n type CommunicationGetTokenOptions,\n} from \"./communicationTokenCredential.js\";\nimport { type AbortSignalLike } from \"@azure/abort-controller\";\nimport { type Client, getClient } from \"@azure-rest/core-client\";\nimport {\n type HttpClient,\n createDefaultHttpClient,\n createHttpHeaders,\n createPipelineRequest,\n} from \"@azure/core-rest-pipeline\";\n\nconst TeamsExtensionScopePrefix = \"https://auth.msft.communication.azure.com/\";\nconst CommunicationClientsScopePrefix = \"https://communication.azure.com/clients/\";\nconst TeamsExtensionEndpoint = \"/access/teamsExtension/:exchangeAccessToken\";\nconst TeamsExtensionApiVersion = \"2025-06-30\";\nconst CommunicationClientsEndpoint = \"/access/entra/:exchangeAccessToken\";\nconst CommunicationClientsApiVersion = \"2025-03-02-preview\";\n\nexport interface ExchangeTokenResponse {\n identity: string;\n accessToken: {\n token: string;\n expiresOn: string;\n };\n}\n\n/**\n * The Entra Communication Token Options.\n */\nexport interface EntraCommunicationTokenCredentialOptions {\n /**\n * The Azure Communication Service resource endpoint URL, e.g. https://myResource.communication.azure.com.\n */\n resourceEndpoint: string;\n /**\n * The Entra ID token credential.\n */\n tokenCredential: TokenCredential;\n /**\n * The scopes for retrieving the Entra ID access token.\n */\n scopes?: string[];\n}\n\n/**\n * Represents a credential that exchanges an Entra token for an Azure Communication Services (ACS) token, enabling access to ACS resources.\n */\nexport class EntraTokenCredential implements AcsTokenCredential {\n private isPending: Promise<AccessToken> | null;\n private result = {\n entraToken: undefined as string | undefined,\n acsToken: { token: \"\", expiresOnTimestamp: 0 },\n };\n private client: Client;\n private httpClient: HttpClient;\n\n constructor(private options: EntraCommunicationTokenCredentialOptions) {\n this.client = getClient(options.resourceEndpoint);\n this.httpClient = createDefaultHttpClient();\n this.options = options;\n this.options.scopes = this.options.scopes || [\n \"https://communication.azure.com/clients/.default\",\n ];\n\n // immediately fetch the token to pre-warm\n this.isPending = this.getToken();\n }\n\n public async getToken(options?: CommunicationGetTokenOptions): Promise<AccessToken> {\n if (options?.abortSignal?.aborted) {\n return { token: \"\", expiresOnTimestamp: 0 };\n }\n\n // we're awaiting the token fetch, so we don't want to start another one\n // however, we're ignoring the new abortSignal, unfortunately\n if (!this.isPending) {\n this.isPending = this.getTokenInternal(options);\n }\n\n try {\n await this.isPending;\n } finally {\n this.isPending = null;\n }\n\n return this.result.acsToken;\n }\n\n private async getTokenInternal(options?: CommunicationGetTokenOptions): Promise<AccessToken> {\n const getTokenOptions = options?.abortSignal ? { abortSignal: options.abortSignal } : undefined;\n const token = await this.options.tokenCredential.getToken(\n this.options.scopes\n ? this.options.scopes\n : [\"https://communication.azure.com/clients/.default\"],\n getTokenOptions,\n );\n const currentDateTime = new Date();\n const tokenExpiresOn = new Date(this.result.acsToken.expiresOnTimestamp);\n\n if (token === null) {\n this.result = {\n entraToken: undefined,\n acsToken: { token: \"\", expiresOnTimestamp: 0 },\n };\n } else if (\n this.result.acsToken.token === \"\" ||\n token.token !== this.result.entraToken ||\n tokenExpiresOn < currentDateTime\n ) {\n const acsToken = await this.exchangeEntraToken(\n this.options.resourceEndpoint,\n token.token,\n getTokenOptions,\n );\n this.result = {\n entraToken: token.token,\n acsToken,\n };\n }\n\n return this.result.acsToken;\n }\n\n public dispose(): void {\n this.result = {\n entraToken: undefined,\n acsToken: { token: \"\", expiresOnTimestamp: 0 },\n };\n }\n\n private async exchangeEntraToken(\n resourceEndpoint: string,\n entraToken: string,\n options?: { abortSignal: AbortSignalLike },\n ): Promise<AccessToken> {\n const request = createPipelineRequest({\n url: this.createRequestUri(resourceEndpoint),\n method: \"POST\",\n headers: createHttpHeaders({\n Authorization: `Bearer ${entraToken}`,\n \"Content-Type\": \"application/json\",\n Accept: \"application/json\",\n }),\n abortSignal: options?.abortSignal,\n body: JSON.stringify({}),\n });\n const response = await this.client.pipeline.sendRequest(this.httpClient, request);\n\n if (response.status !== 200 || !response.bodyAsText) {\n throw new Error(\n `Service request failed. Status: ${response.status}, Body: ${response.bodyAsText}`,\n );\n }\n const json = JSON.parse(response.bodyAsText) as ExchangeTokenResponse;\n return {\n token: json.accessToken.token,\n expiresOnTimestamp: Date.parse(json.accessToken.expiresOn),\n };\n }\n\n private createRequestUri(resourceEndpoint: string): string {\n const [endpoint, apiVersion] = this.determineEndpointAndApiVersion();\n const requestUri = `${resourceEndpoint}${endpoint}?api-version=${apiVersion}`;\n return requestUri;\n }\n\n private determineEndpointAndApiVersion(): [string, string] {\n if (!this.options.scopes || this.options.scopes.length === 0) {\n throw new Error(\n `Scopes validation failed. Ensure all scopes start with either {TeamsExtensionScopePrefix} or {CommunicationClientsScopePrefix}.`,\n );\n } else if (this.options.scopes.every((scope) => scope.startsWith(TeamsExtensionScopePrefix))) {\n return [TeamsExtensionEndpoint, TeamsExtensionApiVersion];\n } else if (\n this.options.scopes.every((scope) => scope.startsWith(CommunicationClientsScopePrefix))\n ) {\n return [CommunicationClientsEndpoint, CommunicationClientsApiVersion];\n } else {\n throw new Error(\n `Scopes validation failed. Ensure all scopes start with either {TeamsExtensionScopePrefix} or {CommunicationClientsScopePrefix}.`,\n );\n }\n }\n}\n"]}
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/entraTokenCredential.ts"],
4
+ "sourcesContent": ["// Copyright (c) Microsoft Corporation.\n// Licensed under the MIT License.\n\nimport { type AccessToken, type TokenCredential } from \"@azure/core-auth\";\nimport {\n type TokenCredential as AcsTokenCredential,\n type CommunicationGetTokenOptions,\n} from \"./communicationTokenCredential.js\";\nimport { type AbortSignalLike } from \"@azure/abort-controller\";\nimport { type Client, getClient } from \"@azure-rest/core-client\";\nimport {\n type HttpClient,\n createDefaultHttpClient,\n createHttpHeaders,\n createPipelineRequest,\n} from \"@azure/core-rest-pipeline\";\n\nconst TeamsExtensionScopePrefixes = [\n \"https://auth.msft.communication.azure.com/\",\n \"https://auth.msft.communication.azure.us/\",\n];\nconst CommunicationClientsScopePrefix = \"https://communication.azure.com/clients/\";\nconst ScopeValidationErrorMessage = `Scopes validation failed. Ensure all scopes start with one of ${[\n ...TeamsExtensionScopePrefixes,\n CommunicationClientsScopePrefix,\n].join(\", \")}.`;\nconst TeamsExtensionEndpoint = \"/access/teamsExtension/:exchangeAccessToken\";\nconst TeamsExtensionApiVersion = \"2025-06-30\";\nconst CommunicationClientsEndpoint = \"/access/entra/:exchangeAccessToken\";\nconst CommunicationClientsApiVersion = \"2025-03-02-preview\";\n\nexport interface ExchangeTokenResponse {\n identity: string;\n accessToken: {\n token: string;\n expiresOn: string;\n };\n}\n\n/**\n * The Entra Communication Token Options.\n */\nexport interface EntraCommunicationTokenCredentialOptions {\n /**\n * The Azure Communication Service resource endpoint URL, e.g. https://myResource.communication.azure.com.\n */\n resourceEndpoint: string;\n /**\n * The Entra ID token credential.\n */\n tokenCredential: TokenCredential;\n /**\n * The scopes for retrieving the Entra ID access token.\n */\n scopes?: string[];\n}\n\n/**\n * Represents a credential that exchanges an Entra token for an Azure Communication Services (ACS) token, enabling access to ACS resources.\n */\nexport class EntraTokenCredential implements AcsTokenCredential {\n private isPending: Promise<AccessToken> | null;\n private result = {\n entraToken: undefined as string | undefined,\n acsToken: { token: \"\", expiresOnTimestamp: 0 },\n };\n private client: Client;\n private httpClient: HttpClient;\n\n constructor(private options: EntraCommunicationTokenCredentialOptions) {\n this.client = getClient(options.resourceEndpoint);\n this.httpClient = createDefaultHttpClient();\n this.options = options;\n this.options.scopes = this.options.scopes || [\n \"https://communication.azure.com/clients/.default\",\n ];\n\n // immediately fetch the token to pre-warm\n this.isPending = this.getToken();\n }\n\n public async getToken(options?: CommunicationGetTokenOptions): Promise<AccessToken> {\n if (options?.abortSignal?.aborted) {\n return { token: \"\", expiresOnTimestamp: 0 };\n }\n\n // we're awaiting the token fetch, so we don't want to start another one\n // however, we're ignoring the new abortSignal, unfortunately\n if (!this.isPending) {\n this.isPending = this.getTokenInternal(options);\n }\n\n try {\n await this.isPending;\n } finally {\n this.isPending = null;\n }\n\n return this.result.acsToken;\n }\n\n private async getTokenInternal(options?: CommunicationGetTokenOptions): Promise<AccessToken> {\n const getTokenOptions = options?.abortSignal ? { abortSignal: options.abortSignal } : undefined;\n const token = await this.options.tokenCredential.getToken(\n this.options.scopes\n ? this.options.scopes\n : [\"https://communication.azure.com/clients/.default\"],\n getTokenOptions,\n );\n const currentDateTime = new Date();\n const tokenExpiresOn = new Date(this.result.acsToken.expiresOnTimestamp);\n\n if (token === null) {\n this.result = {\n entraToken: undefined,\n acsToken: { token: \"\", expiresOnTimestamp: 0 },\n };\n } else if (\n this.result.acsToken.token === \"\" ||\n token.token !== this.result.entraToken ||\n tokenExpiresOn < currentDateTime\n ) {\n const acsToken = await this.exchangeEntraToken(\n this.options.resourceEndpoint,\n token.token,\n getTokenOptions,\n );\n this.result = {\n entraToken: token.token,\n acsToken,\n };\n }\n\n return this.result.acsToken;\n }\n\n public dispose(): void {\n this.result = {\n entraToken: undefined,\n acsToken: { token: \"\", expiresOnTimestamp: 0 },\n };\n }\n\n private async exchangeEntraToken(\n resourceEndpoint: string,\n entraToken: string,\n options?: { abortSignal: AbortSignalLike },\n ): Promise<AccessToken> {\n const request = createPipelineRequest({\n url: this.createRequestUri(resourceEndpoint),\n method: \"POST\",\n headers: createHttpHeaders({\n Authorization: `Bearer ${entraToken}`,\n \"Content-Type\": \"application/json\",\n Accept: \"application/json\",\n }),\n abortSignal: options?.abortSignal,\n body: JSON.stringify({}),\n });\n const response = await this.client.pipeline.sendRequest(this.httpClient, request);\n\n if (response.status !== 200 || !response.bodyAsText) {\n throw new Error(\n `Service request failed. Status: ${response.status}, Body: ${response.bodyAsText}`,\n );\n }\n const json = JSON.parse(response.bodyAsText) as ExchangeTokenResponse;\n return {\n token: json.accessToken.token,\n expiresOnTimestamp: Date.parse(json.accessToken.expiresOn),\n };\n }\n\n private createRequestUri(resourceEndpoint: string): string {\n const [endpoint, apiVersion] = this.determineEndpointAndApiVersion();\n const requestUri = `${resourceEndpoint}${endpoint}?api-version=${apiVersion}`;\n return requestUri;\n }\n\n private determineEndpointAndApiVersion(): [string, string] {\n if (!this.options.scopes || this.options.scopes.length === 0) {\n throw new Error(ScopeValidationErrorMessage);\n }\n\n if (\n this.options.scopes.every((scope) =>\n TeamsExtensionScopePrefixes.some((prefix) => scope.startsWith(prefix)),\n )\n ) {\n return [TeamsExtensionEndpoint, TeamsExtensionApiVersion];\n }\n\n if (this.options.scopes.every((scope) => scope.startsWith(CommunicationClientsScopePrefix))) {\n return [CommunicationClientsEndpoint, CommunicationClientsApiVersion];\n }\n\n throw new Error(ScopeValidationErrorMessage);\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }