mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-28 18:33:37 +00:00
* Microsoft Foundry: add native provider * Microsoft Foundry: tighten review fixes * Microsoft Foundry: enable by default * Microsoft Foundry: stabilize API routing
102 lines
3.7 KiB
TypeScript
102 lines
3.7 KiB
TypeScript
import type { ProviderPrepareRuntimeAuthContext } from "openclaw/plugin-sdk/core";
|
|
import { ensureAuthProfileStore } from "openclaw/plugin-sdk/provider-auth";
|
|
import { getAccessTokenResultAsync } from "./cli.js";
|
|
import {
|
|
type CachedTokenEntry,
|
|
TOKEN_REFRESH_MARGIN_MS,
|
|
buildFoundryProviderBaseUrl,
|
|
extractFoundryEndpoint,
|
|
getFoundryTokenCacheKey,
|
|
isFoundryProviderApi,
|
|
resolveConfiguredModelNameHint,
|
|
} from "./shared-runtime.js";
|
|
|
|
const cachedTokens = new Map<string, CachedTokenEntry>();
|
|
const refreshPromises = new Map<string, Promise<{ apiKey: string; expiresAt: number }>>();
|
|
|
|
export function resetFoundryRuntimeAuthCaches(): void {
|
|
cachedTokens.clear();
|
|
refreshPromises.clear();
|
|
}
|
|
|
|
async function refreshEntraToken(params?: {
|
|
subscriptionId?: string;
|
|
tenantId?: string;
|
|
}): Promise<{ apiKey: string; expiresAt: number }> {
|
|
const result = await getAccessTokenResultAsync(params);
|
|
const rawExpiry = result.expiresOn ? new Date(result.expiresOn).getTime() : Number.NaN;
|
|
const expiresAt = Number.isFinite(rawExpiry) ? rawExpiry : Date.now() + 55 * 60 * 1000;
|
|
cachedTokens.set(getFoundryTokenCacheKey(params), {
|
|
token: result.accessToken,
|
|
expiresAt,
|
|
});
|
|
return { apiKey: result.accessToken, expiresAt };
|
|
}
|
|
|
|
export async function prepareFoundryRuntimeAuth(ctx: ProviderPrepareRuntimeAuthContext) {
|
|
if (ctx.apiKey !== "__entra_id_dynamic__") {
|
|
return null;
|
|
}
|
|
try {
|
|
const authStore = ensureAuthProfileStore(ctx.agentDir, {
|
|
allowKeychainPrompt: false,
|
|
});
|
|
const credential = ctx.profileId ? authStore.profiles[ctx.profileId] : undefined;
|
|
const metadata = credential?.type === "api_key" ? credential.metadata : undefined;
|
|
const modelId =
|
|
typeof ctx.modelId === "string" && ctx.modelId.trim().length > 0
|
|
? ctx.modelId.trim()
|
|
: typeof metadata?.modelId === "string" && metadata.modelId.trim().length > 0
|
|
? metadata.modelId.trim()
|
|
: ctx.modelId;
|
|
const activeModelNameHint = ctx.modelId === metadata?.modelId ? metadata?.modelName : undefined;
|
|
const modelNameHint = resolveConfiguredModelNameHint(
|
|
modelId,
|
|
ctx.model.name ?? activeModelNameHint,
|
|
);
|
|
const configuredApi =
|
|
typeof metadata?.api === "string" && isFoundryProviderApi(metadata.api)
|
|
? metadata.api
|
|
: isFoundryProviderApi(ctx.model.api)
|
|
? ctx.model.api
|
|
: undefined;
|
|
const endpoint =
|
|
typeof metadata?.endpoint === "string" && metadata.endpoint.trim().length > 0
|
|
? metadata.endpoint.trim()
|
|
: extractFoundryEndpoint(ctx.model.baseUrl ?? "");
|
|
const baseUrl = endpoint
|
|
? buildFoundryProviderBaseUrl(endpoint, modelId, modelNameHint, configuredApi)
|
|
: undefined;
|
|
const cacheKey = getFoundryTokenCacheKey({
|
|
subscriptionId: metadata?.subscriptionId,
|
|
tenantId: metadata?.tenantId,
|
|
});
|
|
const cachedToken = cachedTokens.get(cacheKey);
|
|
if (cachedToken && cachedToken.expiresAt > Date.now() + TOKEN_REFRESH_MARGIN_MS) {
|
|
return {
|
|
apiKey: cachedToken.token,
|
|
expiresAt: cachedToken.expiresAt,
|
|
...(baseUrl ? { baseUrl } : {}),
|
|
};
|
|
}
|
|
let refreshPromise = refreshPromises.get(cacheKey);
|
|
if (!refreshPromise) {
|
|
refreshPromise = refreshEntraToken({
|
|
subscriptionId: metadata?.subscriptionId,
|
|
tenantId: metadata?.tenantId,
|
|
}).finally(() => {
|
|
refreshPromises.delete(cacheKey);
|
|
});
|
|
refreshPromises.set(cacheKey, refreshPromise);
|
|
}
|
|
const token = await refreshPromise;
|
|
return {
|
|
...token,
|
|
...(baseUrl ? { baseUrl } : {}),
|
|
};
|
|
} catch (err) {
|
|
const details = err instanceof Error ? err.message : String(err);
|
|
throw new Error(`Failed to refresh Azure Entra ID token via az CLI: ${details}`);
|
|
}
|
|
}
|