Skip to content

Commit 17ac5f3

Browse files
openai[patch]: add AZURE_OPENAI_ENDPOINT supoprt (langchain-ai#6868)
Co-authored-by: jacoblee93 <[email protected]>
1 parent 5dfebf3 commit 17ac5f3

File tree

5 files changed

+122
-4
lines changed

5 files changed

+122
-4
lines changed

libs/langchain-openai/src/azure/chat_models.ts

+44
Original file line numberDiff line numberDiff line change
@@ -434,6 +434,11 @@ export class AzureChatOpenAI extends ChatOpenAI {
434434
openAIApiKey: "openai_api_key",
435435
openAIApiVersion: "openai_api_version",
436436
openAIBasePath: "openai_api_base",
437+
deploymentName: "deployment_name",
438+
azureOpenAIEndpoint: "azure_endpoint",
439+
azureOpenAIApiVersion: "openai_api_version",
440+
azureOpenAIBasePath: "openai_api_base",
441+
azureOpenAIApiDeploymentName: "deployment_name",
437442
};
438443
}
439444

@@ -477,6 +482,7 @@ export class AzureChatOpenAI extends ChatOpenAI {
477482
azureOpenAIBasePath: this.azureOpenAIBasePath,
478483
azureADTokenProvider: this.azureADTokenProvider,
479484
baseURL: this.clientConfig.baseURL,
485+
azureOpenAIEndpoint: this.azureOpenAIEndpoint,
480486
};
481487

482488
const endpoint = getEndpoint(openAIEndpointConfig);
@@ -541,6 +547,44 @@ export class AzureChatOpenAI extends ChatOpenAI {
541547
delete json.kwargs.azure_openai_api_key;
542548
delete json.kwargs.azure_openai_api_version;
543549
delete json.kwargs.azure_open_ai_base_path;
550+
551+
if (!json.kwargs.azure_endpoint && this.azureOpenAIEndpoint) {
552+
json.kwargs.azure_endpoint = this.azureOpenAIEndpoint;
553+
}
554+
if (!json.kwargs.azure_endpoint && this.azureOpenAIBasePath) {
555+
const parts = this.azureOpenAIBasePath.split("/openai/deployments/");
556+
if (parts.length === 2 && parts[0].startsWith("http")) {
557+
const [endpoint] = parts;
558+
json.kwargs.azure_endpoint = endpoint;
559+
}
560+
}
561+
if (!json.kwargs.azure_endpoint && this.azureOpenAIApiInstanceName) {
562+
json.kwargs.azure_endpoint = `https://${this.azureOpenAIApiInstanceName}.openai.azure.com/`;
563+
}
564+
if (!json.kwargs.deployment_name && this.azureOpenAIApiDeploymentName) {
565+
json.kwargs.deployment_name = this.azureOpenAIApiDeploymentName;
566+
}
567+
if (!json.kwargs.deployment_name && this.azureOpenAIBasePath) {
568+
const parts = this.azureOpenAIBasePath.split("/openai/deployments/");
569+
if (parts.length === 2) {
570+
const [, deployment] = parts;
571+
json.kwargs.deployment_name = deployment;
572+
}
573+
}
574+
575+
if (
576+
json.kwargs.azure_endpoint &&
577+
json.kwargs.deployment_name &&
578+
json.kwargs.openai_api_base
579+
) {
580+
delete json.kwargs.openai_api_base;
581+
}
582+
if (
583+
json.kwargs.azure_openai_api_instance_name &&
584+
json.kwargs.azure_endpoint
585+
) {
586+
delete json.kwargs.azure_openai_api_instance_name;
587+
}
544588
}
545589

546590
return json;

libs/langchain-openai/src/chat_models.ts

+20-1
Original file line numberDiff line numberDiff line change
@@ -933,6 +933,8 @@ export class ChatOpenAI<
933933

934934
azureOpenAIBasePath?: string;
935935

936+
azureOpenAIEndpoint?: string;
937+
936938
organization?: string;
937939

938940
__includeRawResponse?: boolean;
@@ -993,6 +995,10 @@ export class ChatOpenAI<
993995
fields?.configuration?.organization ??
994996
getEnvironmentVariable("OPENAI_ORGANIZATION");
995997

998+
this.azureOpenAIEndpoint =
999+
fields?.azureOpenAIEndpoint ??
1000+
getEnvironmentVariable("AZURE_OPENAI_ENDPOINT");
1001+
9961002
this.modelName = fields?.model ?? fields?.modelName ?? this.model;
9971003
this.model = this.modelName;
9981004
this.modelKwargs = fields?.modelKwargs ?? {};
@@ -1013,9 +1019,21 @@ export class ChatOpenAI<
10131019
this.__includeRawResponse = fields?.__includeRawResponse;
10141020

10151021
if (this.azureOpenAIApiKey || this.azureADTokenProvider) {
1016-
if (!this.azureOpenAIApiInstanceName && !this.azureOpenAIBasePath) {
1022+
if (
1023+
!this.azureOpenAIApiInstanceName &&
1024+
!this.azureOpenAIBasePath &&
1025+
!this.azureOpenAIEndpoint
1026+
) {
10171027
throw new Error("Azure OpenAI API instance name not found");
10181028
}
1029+
1030+
if (!this.azureOpenAIApiDeploymentName && this.azureOpenAIBasePath) {
1031+
const parts = this.azureOpenAIBasePath.split("/openai/deployments/");
1032+
if (parts.length === 2) {
1033+
const [, deployment] = parts;
1034+
this.azureOpenAIApiDeploymentName = deployment;
1035+
}
1036+
}
10191037
if (!this.azureOpenAIApiDeploymentName) {
10201038
throw new Error("Azure OpenAI API deployment name not found");
10211039
}
@@ -1654,6 +1672,7 @@ export class ChatOpenAI<
16541672
azureOpenAIApiKey: this.azureOpenAIApiKey,
16551673
azureOpenAIBasePath: this.azureOpenAIBasePath,
16561674
baseURL: this.clientConfig.baseURL,
1675+
azureOpenAIEndpoint: this.azureOpenAIEndpoint,
16571676
};
16581677

16591678
const endpoint = getEndpoint(openAIEndpointConfig);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
import { AzureChatOpenAI } from "../../azure/chat_models.js";
2+
3+
test("Test Azure OpenAI serialization from azure endpoint", async () => {
4+
const chat = new AzureChatOpenAI({
5+
azureOpenAIEndpoint: "https://foobar.openai.azure.com/",
6+
azureOpenAIApiDeploymentName: "gpt-4o",
7+
azureOpenAIApiVersion: "2024-08-01-preview",
8+
azureOpenAIApiKey: "foo",
9+
});
10+
expect(JSON.stringify(chat)).toEqual(
11+
`{"lc":1,"type":"constructor","id":["langchain","chat_models","azure_openai","AzureChatOpenAI"],"kwargs":{"azure_endpoint":"https://foobar.openai.azure.com/","openai_api_key":{"lc":1,"type":"secret","id":["OPENAI_API_KEY"]},"deployment_name":"gpt-4o"}}`
12+
);
13+
});
14+
15+
test("Test Azure OpenAI serialization from base path", async () => {
16+
const chat = new AzureChatOpenAI({
17+
azureOpenAIBasePath:
18+
"https://foobar.openai.azure.com/openai/deployments/gpt-4o",
19+
azureOpenAIApiVersion: "2024-08-01-preview",
20+
azureOpenAIApiKey: "foo",
21+
});
22+
expect(JSON.stringify(chat)).toEqual(
23+
`{"lc":1,"type":"constructor","id":["langchain","chat_models","azure_openai","AzureChatOpenAI"],"kwargs":{"openai_api_key":{"lc":1,"type":"secret","id":["OPENAI_API_KEY"]},"azure_endpoint":"https://foobar.openai.azure.com","deployment_name":"gpt-4o"}}`
24+
);
25+
});
26+
27+
test("Test Azure OpenAI serialization from instance name", async () => {
28+
const chat = new AzureChatOpenAI({
29+
azureOpenAIApiInstanceName: "foobar",
30+
azureOpenAIApiDeploymentName: "gpt-4o",
31+
azureOpenAIApiVersion: "2024-08-01-preview",
32+
azureOpenAIApiKey: "foo",
33+
});
34+
expect(JSON.stringify(chat)).toEqual(
35+
`{"lc":1,"type":"constructor","id":["langchain","chat_models","azure_openai","AzureChatOpenAI"],"kwargs":{"openai_api_key":{"lc":1,"type":"secret","id":["OPENAI_API_KEY"]},"azure_endpoint":"https://foobar.openai.azure.com/","deployment_name":"gpt-4o"}}`
36+
);
37+
});

libs/langchain-openai/src/types.ts

+8-1
Original file line numberDiff line numberDiff line change
@@ -216,12 +216,19 @@ export declare interface AzureOpenAIInput {
216216
azureOpenAIApiCompletionsDeploymentName?: string;
217217

218218
/**
219-
* Custom endpoint for Azure OpenAI API. This is useful in case you have a deployment in another region.
219+
* Custom base url for Azure OpenAI API. This is useful in case you have a deployment in another region.
220220
* e.g. setting this value to "https://westeurope.api.cognitive.microsoft.com/openai/deployments"
221221
* will be result in the endpoint URL: https://westeurope.api.cognitive.microsoft.com/openai/deployments/{DeploymentName}/
222222
*/
223223
azureOpenAIBasePath?: string;
224224

225+
/**
226+
* Custom endpoint for Azure OpenAI API. This is useful in case you have a deployment in another region.
227+
* e.g. setting this value to "https://westeurope.api.cognitive.microsoft.com/"
228+
* will be result in the endpoint URL: https://westeurope.api.cognitive.microsoft.com/openai/deployments/{DeploymentName}/
229+
*/
230+
azureOpenAIEndpoint?: string;
231+
225232
/**
226233
* A function that returns an access token for Microsoft Entra (formerly known as Azure Active Directory),
227234
* which will be invoked on every request.

libs/langchain-openai/src/utils/azure.ts

+13-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ export interface OpenAIEndpointConfig {
55
azureADTokenProvider?: () => Promise<string>;
66
azureOpenAIBasePath?: string;
77
baseURL?: string | null;
8+
azureOpenAIEndpoint?: string;
89
}
910

1011
/**
@@ -14,13 +15,15 @@ export interface OpenAIEndpointConfig {
1415
* @param {OpenAIEndpointConfig} config - The configuration object for the (Azure) endpoint.
1516
*
1617
* @property {string} config.azureOpenAIApiDeploymentName - The deployment name of Azure OpenAI.
17-
* @property {string} config.azureOpenAIApiInstanceName - The instance name of Azure OpenAI.
18+
* @property {string} config.azureOpenAIApiInstanceName - The instance name of Azure OpenAI, e.g. `example-resource`.
1819
* @property {string} config.azureOpenAIApiKey - The API Key for Azure OpenAI.
19-
* @property {string} config.azureOpenAIBasePath - The base path for Azure OpenAI.
20+
* @property {string} config.azureOpenAIBasePath - The base path for Azure OpenAI, e.g. `https://example-resource.azure.openai.com/openai/deployments/`.
2021
* @property {string} config.baseURL - Some other custom base path URL.
22+
* @property {string} config.azureOpenAIEndpoint - The endpoint for the Azure OpenAI instance, e.g. `https://example-resource.azure.openai.com/`.
2123
*
2224
* The function operates as follows:
2325
* - If both `azureOpenAIBasePath` and `azureOpenAIApiDeploymentName` (plus `azureOpenAIApiKey`) are provided, it returns an URL combining these two parameters (`${azureOpenAIBasePath}/${azureOpenAIApiDeploymentName}`).
26+
* - If both `azureOpenAIEndpoint` and `azureOpenAIApiDeploymentName` (plus `azureOpenAIApiKey`) are provided, it returns an URL combining these two parameters (`${azureOpenAIEndpoint}/openai/deployments/${azureOpenAIApiDeploymentName}`).
2427
* - If `azureOpenAIApiKey` is provided, it checks for `azureOpenAIApiInstanceName` and `azureOpenAIApiDeploymentName` and throws an error if any of these is missing. If both are provided, it generates an URL incorporating these parameters.
2528
* - If none of the above conditions are met, return any custom `baseURL`.
2629
* - The function returns the generated URL as a string, or undefined if no custom paths are specified.
@@ -37,6 +40,7 @@ export function getEndpoint(config: OpenAIEndpointConfig) {
3740
azureOpenAIBasePath,
3841
baseURL,
3942
azureADTokenProvider,
43+
azureOpenAIEndpoint,
4044
} = config;
4145

4246
if (
@@ -46,6 +50,13 @@ export function getEndpoint(config: OpenAIEndpointConfig) {
4650
) {
4751
return `${azureOpenAIBasePath}/${azureOpenAIApiDeploymentName}`;
4852
}
53+
if (
54+
(azureOpenAIApiKey || azureADTokenProvider) &&
55+
azureOpenAIEndpoint &&
56+
azureOpenAIApiDeploymentName
57+
) {
58+
return `${azureOpenAIEndpoint}/openai/deployments/${azureOpenAIApiDeploymentName}`;
59+
}
4960

5061
if (azureOpenAIApiKey || azureADTokenProvider) {
5162
if (!azureOpenAIApiInstanceName) {

0 commit comments

Comments
 (0)