mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-30 00:48:41 +00:00
* refactor: share talk event metric extraction * refactor: reuse shared coercion helpers * refactor: reuse shared primitive guards * refactor: reuse shared record guard * refactor: reuse shared primitive helpers * refactor: reuse shared string guards * refactor: reuse shared non-empty string guard * refactor: share plugin primitive coercion helpers * refactor: reuse plugin coercion helpers * refactor: reuse plugin coercion helpers in more plugins * refactor: reuse channel coercion helpers * refactor: reuse monitor coercion helpers * refactor: reuse provider coercion helpers * refactor: reuse core coercion helpers * refactor: reuse runtime coercion helpers * refactor: reuse helper coercion in codex paths * refactor: reuse helper coercion in runtime paths * refactor: reuse codex app-server coercion helpers * refactor: reuse codex record helpers * refactor: reuse migration and qa record helpers * refactor: reuse feishu and core helper guards * refactor: reuse browser and policy coercion helpers * refactor: reuse memory wiki record helper * refactor: share boolean coercion helpers * refactor: reuse finite number coercion * refactor: reuse trimmed string list helpers * refactor: reuse string list normalization * refactor: reuse remaining string list helpers * refactor: reuse string entry normalizer * refactor: share sorted string helpers * refactor: share string list normalization * test: preserve command registry browser imports * refactor: reuse trimmed list helpers * refactor: reuse string dedupe helpers * refactor: reuse local dedupe helpers * refactor: reuse more string dedupe helpers * refactor: reuse command string dedupe helpers * refactor: dedupe memory path lists with helper * refactor: expose string dedupe helpers to plugins * refactor: reuse core string dedupe helpers * refactor: reuse shared unique value helpers * refactor: reuse unique helpers in agent utilities * refactor: reuse unique helpers in config plumbing * refactor: reuse unique helpers in extensions * refactor: reuse unique helpers in core utilities * refactor: reuse unique helpers in qa plugins * refactor: reuse unique helpers in memory plugins * refactor: reuse unique helpers in channel plugins * refactor: reuse unique helpers in core tails * refactor: reuse unique helper in comfy workflow * refactor: reuse unique helpers in test utilities * refactor: expose unique value helper to plugins * refactor: reuse unique helpers for numeric lists * refactor: replace index dedupe filters * refactor: reuse string entry normalization * refactor: reuse string normalization in plugin helpers * refactor: reuse string normalization in extension helpers * refactor: reuse string normalization in channel parsers * refactor: reuse string normalization in memory search * refactor: reuse string normalization in provider parsers * refactor: reuse string normalization in qa helpers * refactor: reuse string normalization in infra parsers * refactor: reuse string normalization in messaging parsers * refactor: reuse string normalization in core parsers * refactor: reuse string normalization in extension parsers * refactor: reuse string normalization in remaining parsers * refactor: reuse string normalization in final parser spots * refactor: reuse string normalization in qa media helpers * refactor: reuse normalization in provider and media lists * refactor: reuse normalization for remaining set filters * refactor: reuse normalization in policy allowlists * refactor: reuse normalization in session and owner lists * refactor: centralize primitive string lists * refactor: reuse lowercase entry helpers * refactor: reuse sorted string helpers * refactor: reuse unique trimmed helpers * refactor: reuse string normalization helpers * refactor: reuse catalog string helpers * refactor: reuse remaining string helpers * refactor: simplify remaining list normalization * refactor: reuse codex auth order normalization * chore: refresh plugin sdk api baseline * fix: make shared string sorting deterministic * chore: refresh plugin sdk api baseline * fix: align host env security ordering
206 lines
6.3 KiB
TypeScript
206 lines
6.3 KiB
TypeScript
import type {
|
|
ProviderDefaultThinkingPolicyContext,
|
|
ProviderThinkingProfile,
|
|
} from "openclaw/plugin-sdk/core";
|
|
import type { ModelProviderConfig } from "openclaw/plugin-sdk/provider-model-types";
|
|
import { normalizeOptionalString } from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
import { normalizeAntigravityModelId, normalizeGoogleModelId } from "./model-id.js";
|
|
import { isGoogleGemini3ProModel, isGoogleGemini3ThinkingLevelModel } from "./thinking-api.js";
|
|
|
|
type GoogleApiCarrier = {
|
|
api?: string | null;
|
|
};
|
|
|
|
type GoogleProviderConfigLike = GoogleApiCarrier & {
|
|
models?: ReadonlyArray<GoogleApiCarrier | null | undefined> | null;
|
|
};
|
|
|
|
export const DEFAULT_GOOGLE_API_BASE_URL = "https://generativelanguage.googleapis.com/v1beta";
|
|
const GOOGLE_MODEL_ID_PROVIDERS = new Set(["google", "google-gemini-cli", "google-vertex"]);
|
|
|
|
function trimTrailingSlashes(value: string): string {
|
|
return value.replace(/\/+$/, "");
|
|
}
|
|
|
|
function isCanonicalGoogleApiOriginShorthand(value: string): boolean {
|
|
return /^https:\/\/generativelanguage\.googleapis\.com\/?$/i.test(value);
|
|
}
|
|
|
|
function isGoogleGenerativeAiUrl(url: URL): boolean {
|
|
return (
|
|
url.protocol === "https:" && url.hostname.toLowerCase() === "generativelanguage.googleapis.com"
|
|
);
|
|
}
|
|
|
|
function stripUrlUserInfo(url: URL): void {
|
|
url.username = "";
|
|
url.password = "";
|
|
}
|
|
|
|
export function normalizeGoogleApiBaseUrl(baseUrl?: string): string {
|
|
const raw = trimTrailingSlashes(normalizeOptionalString(baseUrl) || DEFAULT_GOOGLE_API_BASE_URL);
|
|
try {
|
|
const url = new URL(raw);
|
|
url.hash = "";
|
|
url.search = "";
|
|
stripUrlUserInfo(url);
|
|
if (isGoogleGenerativeAiUrl(url)) {
|
|
const normalizedPath = trimTrailingSlashes(url.pathname || "");
|
|
url.pathname = normalizedPath || "/v1beta";
|
|
}
|
|
return trimTrailingSlashes(url.toString());
|
|
} catch {
|
|
if (isCanonicalGoogleApiOriginShorthand(raw)) {
|
|
return DEFAULT_GOOGLE_API_BASE_URL;
|
|
}
|
|
return raw;
|
|
}
|
|
}
|
|
|
|
export function isGoogleGenerativeAiApi(api?: string | null): boolean {
|
|
return api === "google-generative-ai";
|
|
}
|
|
|
|
export function normalizeGoogleGenerativeAiBaseUrl(baseUrl?: string): string | undefined {
|
|
if (!baseUrl) {
|
|
return baseUrl;
|
|
}
|
|
|
|
const normalized = normalizeGoogleApiBaseUrl(baseUrl);
|
|
try {
|
|
const url = new URL(normalized);
|
|
stripUrlUserInfo(url);
|
|
if (isGoogleGenerativeAiUrl(url)) {
|
|
url.pathname = trimTrailingSlashes(url.pathname || "").replace(/\/openai$/i, "") || "/v1beta";
|
|
return trimTrailingSlashes(url.toString());
|
|
}
|
|
} catch {
|
|
// `normalizeGoogleApiBaseUrl` already returned the best-effort input form.
|
|
}
|
|
|
|
return normalized;
|
|
}
|
|
|
|
export function resolveGoogleGenerativeAiTransport<TApi extends string | null | undefined>(params: {
|
|
api: TApi;
|
|
baseUrl?: string;
|
|
}): { api: TApi; baseUrl?: string } {
|
|
return {
|
|
api: params.api,
|
|
baseUrl: isGoogleGenerativeAiApi(params.api)
|
|
? normalizeGoogleGenerativeAiBaseUrl(params.baseUrl)
|
|
: params.baseUrl,
|
|
};
|
|
}
|
|
|
|
export function resolveGoogleGenerativeAiApiOrigin(baseUrl?: string): string {
|
|
return (
|
|
normalizeGoogleGenerativeAiBaseUrl(baseUrl) ?? normalizeGoogleApiBaseUrl(baseUrl)
|
|
).replace(/\/v1beta$/i, "");
|
|
}
|
|
|
|
export function shouldNormalizeGoogleGenerativeAiProviderConfig(
|
|
providerKey: string,
|
|
provider: GoogleProviderConfigLike,
|
|
): boolean {
|
|
if (isGoogleGenerativeAiApi(provider.api)) {
|
|
return true;
|
|
}
|
|
const hasGoogleGenerativeAiModelApi =
|
|
provider.models?.some((model) => isGoogleGenerativeAiApi(model?.api)) ?? false;
|
|
if (hasGoogleGenerativeAiModelApi) {
|
|
return true;
|
|
}
|
|
if (providerKey !== "google" && providerKey !== "google-vertex") {
|
|
return false;
|
|
}
|
|
const hasExplicitNonGoogleApi = normalizeOptionalString(provider.api) !== undefined;
|
|
return !hasExplicitNonGoogleApi;
|
|
}
|
|
|
|
export function shouldNormalizeGoogleProviderConfig(
|
|
providerKey: string,
|
|
provider: GoogleProviderConfigLike,
|
|
): boolean {
|
|
return (
|
|
providerKey === "google-antigravity" ||
|
|
shouldNormalizeGoogleGenerativeAiProviderConfig(providerKey, provider)
|
|
);
|
|
}
|
|
|
|
function normalizeProviderModels(
|
|
provider: ModelProviderConfig,
|
|
normalizeId: (id: string) => string,
|
|
): ModelProviderConfig {
|
|
const models = provider.models;
|
|
if (!Array.isArray(models) || models.length === 0) {
|
|
return provider;
|
|
}
|
|
|
|
let mutated = false;
|
|
const nextModels = models.map((model) => {
|
|
const nextId = normalizeId(model.id);
|
|
if (nextId === model.id) {
|
|
return model;
|
|
}
|
|
mutated = true;
|
|
return Object.assign({}, model, { id: nextId });
|
|
});
|
|
|
|
return mutated ? { ...provider, models: nextModels } : provider;
|
|
}
|
|
|
|
export function normalizeGoogleProviderConfig(
|
|
providerKey: string,
|
|
provider: ModelProviderConfig,
|
|
): ModelProviderConfig {
|
|
let nextProvider = provider;
|
|
const shouldNormalizeModelIds = GOOGLE_MODEL_ID_PROVIDERS.has(providerKey);
|
|
|
|
if (shouldNormalizeModelIds) {
|
|
const modelNormalized = normalizeProviderModels(nextProvider, normalizeGoogleModelId);
|
|
if (shouldNormalizeGoogleGenerativeAiProviderConfig(providerKey, modelNormalized)) {
|
|
const normalizedBaseUrl = normalizeGoogleGenerativeAiBaseUrl(modelNormalized.baseUrl);
|
|
nextProvider =
|
|
normalizedBaseUrl !== modelNormalized.baseUrl
|
|
? { ...modelNormalized, baseUrl: normalizedBaseUrl ?? modelNormalized.baseUrl }
|
|
: modelNormalized;
|
|
} else {
|
|
nextProvider = modelNormalized;
|
|
}
|
|
}
|
|
|
|
if (providerKey === "google-antigravity") {
|
|
nextProvider = normalizeProviderModels(nextProvider, normalizeAntigravityModelId);
|
|
}
|
|
|
|
return nextProvider;
|
|
}
|
|
|
|
export function resolveGoogleThinkingProfile({
|
|
modelId,
|
|
reasoning,
|
|
}: ProviderDefaultThinkingPolicyContext): ProviderThinkingProfile | undefined {
|
|
const normalizedModelId = normalizeGoogleModelId(modelId);
|
|
const isGemini3ThinkingModel = isGoogleGemini3ThinkingLevelModel(normalizedModelId);
|
|
if (reasoning === false && !isGemini3ThinkingModel) {
|
|
return undefined;
|
|
}
|
|
|
|
const levels: ProviderThinkingProfile["levels"] = isGoogleGemini3ProModel(normalizedModelId)
|
|
? [{ id: "off" }, { id: "low" }, { id: "adaptive" }, { id: "high" }]
|
|
: [
|
|
{ id: "off" },
|
|
{ id: "minimal" },
|
|
{ id: "low" },
|
|
{ id: "medium" },
|
|
{ id: "adaptive" },
|
|
{ id: "high" },
|
|
];
|
|
|
|
return {
|
|
levels,
|
|
...(isGemini3ThinkingModel ? { preserveWhenCatalogReasoningFalse: true } : {}),
|
|
};
|
|
}
|