mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-24 06:19:30 +00:00
138 lines
4.6 KiB
TypeScript
138 lines
4.6 KiB
TypeScript
/**
|
|
* Shared provider/model reference normalization for static catalogs,
|
|
* allowlists, and display paths. Manifest policies are optional so tests can
|
|
* isolate built-in normalization behavior.
|
|
*/
|
|
import { normalizeProviderId } from "@openclaw/model-catalog-core/provider-id";
|
|
import {
|
|
collectManifestModelIdNormalizationPolicies,
|
|
normalizeBuiltInProviderModelId,
|
|
normalizeConfiguredProviderCatalogModelRef,
|
|
normalizeConfiguredProviderCatalogModelId as normalizeConfiguredProviderCatalogModelIdShared,
|
|
normalizeStaticProviderModelIdWithPolicies,
|
|
} from "@openclaw/model-catalog-core/provider-model-id-normalization";
|
|
import { normalizeLowercaseStringOrEmpty } from "@openclaw/normalization-core/string-coerce";
|
|
import { normalizeProviderModelIdWithManifest } from "../plugins/manifest-model-id-normalization.js";
|
|
import { modelKey } from "../shared/model-key.js";
|
|
export { modelKey } from "../shared/model-key.js";
|
|
|
|
type StaticModelRef = {
|
|
provider: string;
|
|
model: string;
|
|
};
|
|
|
|
export type ProviderModelIdNormalizationOptions = {
|
|
allowManifestNormalization?: boolean;
|
|
manifestPlugins?: readonly ManifestModelIdNormalizationRecord[];
|
|
};
|
|
|
|
type ManifestModelIdNormalizationProvider = {
|
|
aliases?: Record<string, string>;
|
|
stripPrefixes?: string[];
|
|
prefixWhenBare?: string;
|
|
prefixWhenBareAfterAliasStartsWith?: {
|
|
modelPrefix: string;
|
|
prefix: string;
|
|
}[];
|
|
};
|
|
|
|
type ManifestModelIdNormalizationRecord = {
|
|
modelIdNormalization?: {
|
|
providers?: Record<string, ManifestModelIdNormalizationProvider>;
|
|
};
|
|
};
|
|
|
|
/** Normalize a static provider model ID with built-in and optional manifest policy. */
|
|
export function normalizeStaticProviderModelId(
|
|
provider: string,
|
|
model: string,
|
|
options: ProviderModelIdNormalizationOptions = {},
|
|
): string {
|
|
const normalizedProvider = normalizeProviderId(provider);
|
|
if (options.allowManifestNormalization === false) {
|
|
return normalizeBuiltInProviderModelId(normalizedProvider, model);
|
|
}
|
|
if (options.manifestPlugins) {
|
|
return normalizeStaticProviderModelIdWithPolicies(
|
|
normalizedProvider,
|
|
model,
|
|
collectManifestModelIdNormalizationPolicies(options.manifestPlugins),
|
|
);
|
|
}
|
|
const manifestModelId =
|
|
normalizeProviderModelIdWithManifest({
|
|
provider: normalizedProvider,
|
|
context: {
|
|
provider: normalizedProvider,
|
|
modelId: model,
|
|
},
|
|
}) ?? model;
|
|
return normalizeBuiltInProviderModelId(normalizedProvider, manifestModelId);
|
|
}
|
|
|
|
/** Normalize a configured catalog model ID for comparisons against provider catalogs. */
|
|
export function normalizeConfiguredProviderCatalogModelId(
|
|
provider: string,
|
|
model: string,
|
|
options: ProviderModelIdNormalizationOptions = {},
|
|
): string {
|
|
if (options.allowManifestNormalization === false) {
|
|
return normalizeConfiguredProviderCatalogModelIdShared(provider, model, new Map());
|
|
}
|
|
if (options.manifestPlugins) {
|
|
return normalizeConfiguredProviderCatalogModelIdShared(
|
|
provider,
|
|
model,
|
|
collectManifestModelIdNormalizationPolicies(options.manifestPlugins),
|
|
);
|
|
}
|
|
return normalizeConfiguredProviderCatalogModelRef(
|
|
normalizeStaticProviderModelId(provider, model, options),
|
|
);
|
|
}
|
|
|
|
function parseStaticModelRef(raw: string, defaultProvider: string): StaticModelRef | null {
|
|
const trimmed = raw.trim();
|
|
if (!trimmed) {
|
|
return null;
|
|
}
|
|
const slash = trimmed.indexOf("/");
|
|
const providerRaw = slash === -1 ? defaultProvider : trimmed.slice(0, slash).trim();
|
|
const modelRaw = slash === -1 ? trimmed : trimmed.slice(slash + 1).trim();
|
|
if (!providerRaw || !modelRaw) {
|
|
return null;
|
|
}
|
|
const provider = normalizeProviderId(providerRaw);
|
|
return {
|
|
provider,
|
|
model: normalizeStaticProviderModelId(provider, modelRaw),
|
|
};
|
|
}
|
|
|
|
/** Resolve an allowlist entry to a canonical provider/model key. */
|
|
export function resolveStaticAllowlistModelKey(
|
|
raw: string,
|
|
defaultProvider: string,
|
|
): string | null {
|
|
const parsed = parseStaticModelRef(raw, defaultProvider);
|
|
if (!parsed) {
|
|
return null;
|
|
}
|
|
return modelKey(parsed.provider, parsed.model);
|
|
}
|
|
|
|
/** Preserve literal provider/model refs that already include a provider prefix twice. */
|
|
export function formatLiteralProviderPrefixedModelRef(provider: string, modelRef: string): string {
|
|
const providerId = normalizeProviderId(provider);
|
|
const trimmedRef = modelRef.trim();
|
|
if (!providerId || !trimmedRef) {
|
|
return trimmedRef;
|
|
}
|
|
const normalizedRef = normalizeLowercaseStringOrEmpty(trimmedRef);
|
|
const literalPrefix = `${providerId}/${providerId}/`;
|
|
if (normalizedRef.startsWith(literalPrefix)) {
|
|
return trimmedRef;
|
|
}
|
|
return normalizedRef.startsWith(`${providerId}/`) ? `${providerId}/${trimmedRef}` : trimmedRef;
|
|
}
|