mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-23 12:48:10 +00:00
150 lines
5.3 KiB
TypeScript
150 lines
5.3 KiB
TypeScript
// Microsoft Foundry plugin module implements runtime behavior.
|
|
import type {
|
|
ProviderPreparedRuntimeAuth,
|
|
ProviderPrepareRuntimeAuthContext,
|
|
} from "openclaw/plugin-sdk/core";
|
|
import { formatErrorMessage } from "openclaw/plugin-sdk/error-runtime";
|
|
import {
|
|
asDateTimestampMs,
|
|
resolveDateTimestampMs,
|
|
resolveExpiresAtMsFromDurationMs,
|
|
} from "openclaw/plugin-sdk/number-runtime";
|
|
import { ensureAuthProfileStore } from "openclaw/plugin-sdk/provider-auth";
|
|
import { normalizeOptionalString } from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
import { getAccessTokenResultAsync } from "./cli.js";
|
|
import {
|
|
ANTHROPIC_MESSAGES_API,
|
|
type CachedTokenEntry,
|
|
FOUNDRY_ANTHROPIC_SCOPE,
|
|
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 }>>();
|
|
const FOUNDRY_TOKEN_FALLBACK_LIFETIME_MS = 55 * 60 * 1000;
|
|
|
|
export function resetFoundryRuntimeAuthCaches(): void {
|
|
cachedTokens.clear();
|
|
refreshPromises.clear();
|
|
}
|
|
|
|
async function refreshEntraToken(params?: {
|
|
scope?: string;
|
|
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 now = resolveDateTimestampMs(Date.now());
|
|
const expiresAt =
|
|
asDateTimestampMs(rawExpiry) ??
|
|
resolveExpiresAtMsFromDurationMs(FOUNDRY_TOKEN_FALLBACK_LIFETIME_MS, { nowMs: now }) ??
|
|
now;
|
|
cachedTokens.set(getFoundryTokenCacheKey(params), {
|
|
token: result.accessToken,
|
|
expiresAt,
|
|
});
|
|
return { apiKey: result.accessToken, expiresAt };
|
|
}
|
|
|
|
export async function prepareFoundryRuntimeAuth(
|
|
ctx: ProviderPrepareRuntimeAuthContext,
|
|
): Promise<ProviderPreparedRuntimeAuth> {
|
|
if (ctx.apiKey !== "__entra_id_dynamic__") {
|
|
return {
|
|
apiKey: ctx.apiKey,
|
|
request: {
|
|
auth: {
|
|
mode: "header" as const,
|
|
headerName: ctx.model.api === ANTHROPIC_MESSAGES_API ? "x-api-key" : "api-key",
|
|
value: ctx.apiKey,
|
|
},
|
|
},
|
|
};
|
|
}
|
|
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 =
|
|
normalizeOptionalString(ctx.modelId) ??
|
|
normalizeOptionalString(metadata?.modelId) ??
|
|
ctx.modelId;
|
|
const requestedModelId = normalizeOptionalString(ctx.modelId);
|
|
const metadataModelId = normalizeOptionalString(metadata?.modelId);
|
|
const activeModelUsesMetadata = !requestedModelId || requestedModelId === metadataModelId;
|
|
const activeModelNameHint = activeModelUsesMetadata ? metadata?.modelName : undefined;
|
|
const modelNameHint = resolveConfiguredModelNameHint(
|
|
modelId,
|
|
ctx.model.name ?? activeModelNameHint,
|
|
);
|
|
const configuredApi = isFoundryProviderApi(ctx.model.api)
|
|
? ctx.model.api
|
|
: activeModelUsesMetadata &&
|
|
typeof metadata?.api === "string" &&
|
|
isFoundryProviderApi(metadata.api)
|
|
? metadata.api
|
|
: undefined;
|
|
const endpoint =
|
|
extractFoundryEndpoint(ctx.model.baseUrl ?? "") ??
|
|
normalizeOptionalString(metadata?.endpoint);
|
|
const tokenScope =
|
|
configuredApi === ANTHROPIC_MESSAGES_API ? FOUNDRY_ANTHROPIC_SCOPE : undefined;
|
|
const baseUrl = endpoint
|
|
? buildFoundryProviderBaseUrl(endpoint, modelId, modelNameHint, configuredApi)
|
|
: undefined;
|
|
const cacheKey = getFoundryTokenCacheKey({
|
|
scope: tokenScope,
|
|
subscriptionId: metadata?.subscriptionId,
|
|
tenantId: metadata?.tenantId,
|
|
});
|
|
const cachedToken = cachedTokens.get(cacheKey);
|
|
const rawNow = Date.now();
|
|
const hasValidClock = asDateTimestampMs(rawNow) !== undefined;
|
|
const now = resolveDateTimestampMs(rawNow);
|
|
const refreshAfterMs =
|
|
resolveExpiresAtMsFromDurationMs(TOKEN_REFRESH_MARGIN_MS, { nowMs: now }) ?? now;
|
|
if (cachedToken && hasValidClock && cachedToken.expiresAt > refreshAfterMs) {
|
|
return {
|
|
apiKey: cachedToken.token,
|
|
expiresAt: cachedToken.expiresAt,
|
|
...(baseUrl ? { baseUrl } : {}),
|
|
request: {
|
|
auth: { mode: "authorization-bearer" as const, token: cachedToken.token },
|
|
},
|
|
};
|
|
}
|
|
let refreshPromise = refreshPromises.get(cacheKey);
|
|
if (!refreshPromise) {
|
|
refreshPromise = refreshEntraToken({
|
|
scope: tokenScope,
|
|
subscriptionId: metadata?.subscriptionId,
|
|
tenantId: metadata?.tenantId,
|
|
}).finally(() => {
|
|
refreshPromises.delete(cacheKey);
|
|
});
|
|
refreshPromises.set(cacheKey, refreshPromise);
|
|
}
|
|
const token = await refreshPromise;
|
|
return {
|
|
...token,
|
|
...(baseUrl ? { baseUrl } : {}),
|
|
request: {
|
|
auth: { mode: "authorization-bearer" as const, token: token.apiKey },
|
|
},
|
|
};
|
|
} catch (err) {
|
|
const details = formatErrorMessage(err);
|
|
throw new Error(`Failed to refresh Azure Entra ID token via az CLI: ${details}`, {
|
|
cause: err,
|
|
});
|
|
}
|
|
}
|