feat: add Azure OpenAI as a dedicated provider

Azure OpenAI uses a different URL scheme (deployments-based) and api-key
header auth instead of Bearer tokens. This adds a dedicated AzureExecutor
that constructs the correct URL and headers, plus dashboard UI fields for
endpoint, deployment, API version, and organization.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
kundeng
2026-04-20 01:06:45 -04:00
parent befb2bc3d5
commit 6ac3f4a97a
5 changed files with 133 additions and 2 deletions

View File

@@ -338,4 +338,11 @@ export const PROVIDERS = {
headers: { "x-opencode-client": "desktop" },
noAuth: true
},
// Azure OpenAI - requires azureEndpoint, apiVersion, deployment in providerSpecificData
// Uses api-key header for authentication instead of Bearer token
azure: {
baseUrl: "", // Constructed dynamically by AzureExecutor based on credentials
format: "openai",
headers: {}
},
};

View File

@@ -0,0 +1,57 @@
import { DefaultExecutor } from "./default.js";
export class AzureExecutor extends DefaultExecutor {
constructor() {
super("azure");
}
buildUrl(model, stream, urlIndex = 0, credentials = null) {
const azureEndpoint = credentials?.providerSpecificData?.azureEndpoint
|| process.env.AZURE_ENDPOINT
|| "https://api.openai.com";
const apiVersion = credentials?.providerSpecificData?.apiVersion
|| process.env.AZURE_API_VERSION
|| "2024-10-01-preview";
const deployment = credentials?.providerSpecificData?.deployment
|| model
|| process.env.AZURE_DEPLOYMENT
|| "gpt-4";
const endpoint = azureEndpoint.replace(/\/$/, "");
return `${endpoint}/openai/deployments/${deployment}/chat/completions?api-version=${apiVersion}`;
}
buildHeaders(credentials, stream = true) {
const headers = {
"Content-Type": "application/json",
...this.config.headers
};
const apiKey = credentials?.apiKey
|| credentials?.accessToken
|| process.env.OPENAI_API_KEY;
if (apiKey) {
headers["api-key"] = apiKey;
}
const organization = credentials?.providerSpecificData?.organization
|| process.env.AZURE_ORGANIZATION;
if (organization) {
headers["OpenAI-Organization"] = organization;
}
if (stream) {
headers["Accept"] = "text/event-stream";
}
return headers;
}
transformRequest(model, body, stream, credentials) {
return body;
}
}

View File

@@ -1,4 +1,5 @@
import { AntigravityExecutor } from "./antigravity.js";
import { AzureExecutor } from "./azure.js";
import { GeminiCLIExecutor } from "./gemini-cli.js";
import { GithubExecutor } from "./github.js";
import { IFlowExecutor } from "./iflow.js";
@@ -13,6 +14,7 @@ import { DefaultExecutor } from "./default.js";
const executors = {
antigravity: new AntigravityExecutor(),
azure: new AzureExecutor(),
"gemini-cli": new GeminiCLIExecutor(),
github: new GithubExecutor(),
iflow: new IFlowExecutor(),
@@ -41,6 +43,7 @@ export function hasSpecializedExecutor(provider) {
export { BaseExecutor } from "./base.js";
export { AntigravityExecutor } from "./antigravity.js";
export { AzureExecutor } from "./azure.js";
export { GeminiCLIExecutor } from "./gemini-cli.js";
export { GithubExecutor } from "./github.js";
export { IFlowExecutor } from "./iflow.js";

View File

@@ -14,6 +14,12 @@ export default function EditConnectionModal({ isOpen, connection, proxyPools, on
priority: 1,
apiKey: "",
});
const [azureData, setAzureData] = useState({
azureEndpoint: "",
apiVersion: "2024-10-01-preview",
deployment: "",
organization: "",
});
const [testing, setTesting] = useState(false);
const [testResult, setTestResult] = useState(null);
const [validating, setValidating] = useState(false);
@@ -27,12 +33,22 @@ export default function EditConnectionModal({ isOpen, connection, proxyPools, on
priority: connection.priority || 1,
apiKey: "",
});
// Load Azure-specific data if present
if (connection.provider === "azure" && connection.providerSpecificData) {
setAzureData({
azureEndpoint: connection.providerSpecificData.azureEndpoint || "",
apiVersion: connection.providerSpecificData.apiVersion || "2024-10-01-preview",
deployment: connection.providerSpecificData.deployment || "",
organization: connection.providerSpecificData.organization || "",
});
}
setTestResult(null);
setValidationResult(null);
}
}, [connection]);
const isOAuth = connection?.authType === "oauth";
const isAzure = connection?.provider === "azure";
const isCompatible = connection
? (isOpenAICompatibleProvider(connection.provider) || isAnthropicCompatibleProvider(connection.provider))
: false;
@@ -106,6 +122,17 @@ export default function EditConnectionModal({ isOpen, connection, proxyPools, on
updates.lastErrorAt = null;
}
}
// Add Azure-specific data if this is an Azure connection
if (isAzure) {
updates.providerSpecificData = {
azureEndpoint: azureData.azureEndpoint,
apiVersion: azureData.apiVersion,
deployment: azureData.deployment,
organization: azureData.organization,
};
}
await onSave(updates);
} finally {
setSaving(false);
@@ -162,7 +189,43 @@ export default function EditConnectionModal({ isOpen, connection, proxyPools, on
</>
)}
{!isCompatible && (
{isAzure && (
<div className="bg-sidebar/50 p-4 rounded-lg border border-accent/20">
<h3 className="font-semibold mb-3 text-sm">Azure OpenAI Configuration</h3>
<div className="flex flex-col gap-3">
<Input
label="Azure Endpoint"
value={azureData.azureEndpoint}
onChange={(e) => setAzureData({ ...azureData, azureEndpoint: e.target.value })}
placeholder="https://your-resource.openai.azure.com"
hint="Your Azure OpenAI resource endpoint URL"
/>
<Input
label="Deployment Name"
value={azureData.deployment}
onChange={(e) => setAzureData({ ...azureData, deployment: e.target.value })}
placeholder="gpt-4"
hint="The deployment name in your Azure resource"
/>
<Input
label="API Version"
value={azureData.apiVersion}
onChange={(e) => setAzureData({ ...azureData, apiVersion: e.target.value })}
placeholder="2024-10-01-preview"
hint="Azure OpenAI API version to use"
/>
<Input
label="Organization (Optional)"
value={azureData.organization}
onChange={(e) => setAzureData({ ...azureData, organization: e.target.value })}
placeholder="Organization ID"
hint="Optional: Your Azure organization ID"
/>
</div>
</div>
)}
{!isCompatible && !isAzure && (
<div className="flex items-center gap-3">
<Button onClick={handleTest} variant="secondary" disabled={testing}>
{testing ? "Testing..." : "Test Connection"}
@@ -202,3 +265,4 @@ EditConnectionModal.propTypes = {
onSave: PropTypes.func.isRequired,
onClose: PropTypes.func.isRequired,
};

View File

@@ -61,7 +61,7 @@ export const APIKEY_PROVIDERS = {
"alicode-intl": { id: "alicode-intl", alias: "alicode-intl", name: "Alibaba Intl", icon: "cloud", color: "#FF6A00", textIcon: "ALi" },
openai: { id: "openai", alias: "openai", name: "OpenAI", icon: "auto_awesome", color: "#10A37F", textIcon: "OA", website: "https://platform.openai.com", serviceKinds: ["llm", "embedding", "tts", "image", "imageToText", "webSearch"], thinkingConfig: THINKING_CONFIG.effort },
anthropic: { id: "anthropic", alias: "anthropic", name: "Anthropic", icon: "smart_toy", color: "#D97757", textIcon: "AN", website: "https://console.anthropic.com", serviceKinds: ["llm", "imageToText"] },
azure: { id: "azure", alias: "azure", name: "Azure OpenAI", icon: "cloud", color: "#0078D4", textIcon: "AZ", website: "https://azure.microsoft.com/en-us/products/ai-services/openai-service", hasProviderSpecificData: true },
deepseek: { id: "deepseek", alias: "ds", name: "DeepSeek", icon: "bolt", color: "#4D6BFE", textIcon: "DS", website: "https://deepseek.com" },
groq: { id: "groq", alias: "groq", name: "Groq", icon: "speed", color: "#F55036", textIcon: "GQ", website: "https://groq.com", serviceKinds: ["llm", "imageToText"] },