import type { ProviderNormalizeResolvedModelContext } from "openclaw/plugin-sdk/core" ;
import type {
ModelProviderConfig,
ProviderPlugin,
} from "openclaw/plugin-sdk/provider-model-shared" ;
import { apiKeyAuthMethod, entraIdAuthMethod } from "./auth.js" ;
import { prepareFoundryRuntimeAuth } from "./runtime.js" ;
import {
PROVIDER_ID,
applyFoundryProfileBinding,
applyFoundryProviderConfig,
buildFoundryProviderBaseUrl,
extractFoundryEndpoint,
isFoundryProviderApi,
normalizeFoundryEndpoint,
resolveFoundryModelCapabilities,
resolveFoundryTargetProfileId,
} from "./shared.js" ;
export function buildMicrosoftFoundryProvider(): ProviderPlugin {
return {
id: PROVIDER_ID,
label: "Microsoft Foundry" ,
docsPath: "/providers/models" ,
envVars: ["AZURE_OPENAI_API_KEY" , "AZURE_OPENAI_ENDPOINT" ],
auth: [entraIdAuthMethod, apiKeyAuthMethod],
onModelSelected: async (ctx) => {
const providerConfig = ctx.config.models?.providers?.[PROVIDER_ID];
if (!providerConfig || !ctx.model.startsWith(`${PROVIDER_ID}/`)) {
return ;
}
const selectedModelId = ctx.model.slice(`${PROVIDER_ID}/`.length);
const existingModel = providerConfig.models.find(
(model: { id: string }) => model.id === selectedModelId,
);
const selectedModelCapabilities = resolveFoundryModelCapabilities(
selectedModelId,
existingModel?.name,
isFoundryProviderApi(existingModel?.api) ? existingModel.api : providerConfig.api,
existingModel?.input,
);
const providerEndpoint = normalizeFoundryEndpoint(providerConfig.baseUrl ?? "" );
// Prefer the persisted per-model API choice from onboarding/discovery so arbitrary
// deployment aliases (for example prod-primary) do not fall back to name heuristics.
const selectedModelApi = isFoundryProviderApi(existingModel?.api)
? existingModel.api
: providerConfig.api;
const nextModels = providerConfig.models.map((model) =>
model.id === selectedModelId
? {
...model,
name: selectedModelCapabilities.modelName,
api: selectedModelCapabilities.api,
input: selectedModelCapabilities.input,
...(selectedModelCapabilities.compat
? { compat: selectedModelCapabilities.compat }
: {}),
}
: model,
);
if (!nextModels.some((model) => model.id === selectedModelId)) {
nextModels.push({
id: selectedModelId,
name: selectedModelCapabilities.modelName,
api: selectedModelCapabilities.api,
reasoning: false ,
input: selectedModelCapabilities.input,
cost: { input: 0 , output: 0 , cacheRead: 0 , cacheWrite: 0 },
contextWindow: 128 _000 ,
maxTokens: 16 _384 ,
...(selectedModelCapabilities.compat ? { compat: selectedModelCapabilities.compat } : {}),
});
}
const nextProviderConfig: ModelProviderConfig = {
...providerConfig,
baseUrl: buildFoundryProviderBaseUrl(
providerEndpoint,
selectedModelId,
selectedModelCapabilities.modelName,
selectedModelApi,
),
api: selectedModelCapabilities.api,
models: nextModels,
};
const targetProfileId = resolveFoundryTargetProfileId(ctx.config);
if (targetProfileId) {
applyFoundryProfileBinding(ctx.config, targetProfileId);
}
applyFoundryProviderConfig(ctx.config, nextProviderConfig);
},
normalizeResolvedModel: ({ modelId, model }: ProviderNormalizeResolvedModelContext) => {
const endpoint = extractFoundryEndpoint(model.baseUrl ?? "" );
if (!endpoint) {
return model;
}
const capabilities = resolveFoundryModelCapabilities(
modelId,
model.name,
isFoundryProviderApi(model.api) ? model.api : undefined,
model.input,
);
return {
...model,
name: capabilities.modelName,
api: capabilities.api,
input: capabilities.input,
baseUrl: buildFoundryProviderBaseUrl(
endpoint,
modelId,
capabilities.modelName,
capabilities.api,
),
...(capabilities.compat ? { compat: capabilities.compat } : {}),
};
},
prepareRuntimeAuth: prepareFoundryRuntimeAuth,
};
}
Messung V0.5 in Prozent C=99 H=100 G=99
¤ Dauer der Verarbeitung: 0.8 Sekunden
(vorverarbeitet am 2026-06-10)
¤
*© Formatika GbR, Deutschland