mirror of
https://github.com/decolua/9router.git
synced 2026-05-08 12:01:28 +00:00
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:
@@ -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: {}
|
||||
},
|
||||
};
|
||||
|
||||
57
open-sse/executors/azure.js
Normal file
57
open-sse/executors/azure.js
Normal 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;
|
||||
}
|
||||
}
|
||||
@@ -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";
|
||||
|
||||
@@ -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,
|
||||
};
|
||||
|
||||
|
||||
@@ -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"] },
|
||||
|
||||
Reference in New Issue
Block a user