Files
openclaw/src/utils/usage-format.ts
Peter Steinberger 0b8aabe864 docs: document auth profile failure policy contract (#89613)
* docs: document markdown marker renderer

* docs: document rendered markdown chunking

* docs: document markdown text chunking

* docs: document shared text chunking

* docs: document plugin text chunking exports

* docs: document avatar policy constants

* docs: document node match candidates

* docs: document scoped expiring id cache

* docs: document runtime import normalization

* docs: document string sample summaries

* docs: document session usage timeseries types

* docs: document session usage response types

* docs: document manifest frontmatter shapes

* docs: document channel route input metadata

* docs: document pair loop guard settings

* docs: document migration config patch helpers

* docs: document api provider registry

* docs: document tool call repair payloads

* docs: document plugin tool payload helpers

* docs: document lazy promise loader

* docs: document store writer queue state

* docs: document thread binding lifecycle

* docs: document concurrency helper contract

* docs: document gateway client info contract

* docs: document delivery context contracts

* docs: document secret ref defaults contract

* docs: document command gating contract

* docs: document avatar policy contract

* docs: document node match policy

* docs: document message channel normalization

* docs: document boolean parsing contract

* docs: document zod parse helpers

* docs: document direct dm guard policy

* docs: document fixed window limiter contract

* docs: document node presence event contract

* docs: document secret normalization contract

* docs: document progress draft line removal

* docs: document usage formatting contracts

* docs: document agent run status contract

* docs: document runtime import helpers

* docs: document provider utility ownership

* docs: document invalid config helpers

* docs: document json compat parser

* docs: document channel config metadata ownership

* docs: document channel logging helpers

* docs: document sender identity validation ownership

* docs: document string sampling helper

* docs: document global singleton helpers

* docs: document transcript tool helpers

* docs: document exec safe-bin normalization

* docs: document reaction level resolver

* docs: document account snapshot redaction boundary

* docs: document messaging target helpers

* docs: document thread binding messages

* docs: document conversation binding context

* docs: document conversation resolution helper

* docs: document owner display secret retention

* docs: document provider request config types

* docs: document skills config types

* docs: document memory config types

* docs: document imessage config types

* docs: document crestodian config types

* docs: document tools config policies

* docs: document shared config base types

* docs: document channel config contracts

* docs: document openclaw config state types

* docs: document model config contracts

* docs: document shared agent config types

* docs: document agent defaults config types

* docs: document secret input contracts

* docs: document auth config contracts

* docs: document gateway config contracts

* docs: document tool call stream repair contracts

* docs: document memory host facades

* docs: document llm core contracts

* docs: document markdown core contracts

* docs: document gateway connect error contracts

* docs: document gateway protocol primitives

* docs: document gateway frame schemas

* docs: document gateway device schemas

* docs: document gateway environment schemas

* docs: document gateway push schemas

* docs: document gateway plugin schemas

* docs: document gateway artifact schemas

* docs: document gateway command schemas

* docs: document gateway task schemas

* docs: document gateway exec approval schemas

* docs: document gateway secret schemas

* docs: document gateway config schemas

* docs: document gateway snapshot schemas

* docs: document gateway chat schemas

* docs: document gateway wizard schemas

* docs: document gateway node schemas

* docs: document gateway plugin approval schemas

* docs: document gateway talk schemas

* docs: document gateway agent schemas

* docs: document gateway session schemas

* docs: document gateway cron schemas

* docs: document gateway agent model skill schemas

* docs: document gateway skill proposal tool schemas

* docs: document gateway protocol registry

* docs: document gateway channel status schemas

* docs: document gateway schema regression tests

* docs: document gateway schema barrel

* docs: document gateway validator tests

* docs: document gateway primitive push tests

* docs: document gateway contract tests

* docs: document native protocol guard

* docs: document channel schema tests

* docs: document gateway protocol smoke tests

* docs: document gateway protocol entrypoint

* docs: document gateway protocol type exports

* docs: document gateway error codes

* docs: document protocol schema registry

* docs: document talk audio codec

* docs: document talk activation names

* docs: document talk consult questions

* docs: document talk consult tool

* docs: document talk run control contracts

* docs: document talk run control adapter

* docs: document talkback consult queue

* docs: document talk consult transcript guard

* docs: document talk fast context runtime

* docs: document forced talk consult coordinator

* docs: document talk output activity tracker

* docs: document talk event metrics

* docs: document talk diagnostics

* docs: document talk observability hook

* docs: document talk provider resolver

* docs: document talk provider registry

* docs: document talk runtime primitives

* docs: document talk consult controller logs

* docs: document channel identity helpers

* docs: document channel account allowlist helpers

* docs: document channel metadata draft controls

* docs: document channel ingress policy

* docs: document channel sender access gates

* docs: document channel catalog message contracts

* docs: document channel account plugin helpers

* docs: document configured binding helpers

* docs: document channel acp approval config helpers

* docs: document channel bundled config write helpers

* docs: document channel plugin utility contracts

* docs: document channel config access helpers

* docs: document channel message action helpers

* docs: document channel outbound runtime helpers

* docs: document channel pairing promotion helpers

* docs: document channel registry helpers

* docs: document channel setup wizard helpers

* docs: document channel lifecycle status helpers

* docs: document channel target thread helpers

* docs: document channel session binding helpers

* docs: document channel package module probes

* docs: document channel setup wizard contracts

* docs: document channel plugin API barrels

* docs: document channel contract test helpers

* docs: document channel core helpers

* docs: document small core facades

* docs: document provider runtime helpers

* docs: document persistence and realtime helpers

* docs: document mcp and state helpers

* docs: document tool planner contracts

* docs: document music generation runtime

* docs: document crestodian command flow

* docs: document utility helpers

* docs: document node host helpers

* docs: document transcript contracts

* docs: document trajectory export contracts

* docs: document image generation contracts

* docs: document routing helper contracts

* docs: document session helper contracts

* docs: document video generation contracts

* docs: document model catalog contracts

* docs: document proxy capture contracts

* docs: document status rendering contracts

* docs: document test helper contracts

* docs: document wizard setup contracts

* docs: document process contracts

* docs: document memory host sdk contracts

* docs: document tts contracts

* docs: document secrets runtime contracts

* docs: document shared helper contracts

* docs: document hook runtime contracts

* docs: document security audit contracts

* docs: document flow contracts

* docs: document media understanding contracts

* docs: document tui contracts

* docs: document logging contracts

* docs: document llm contracts

* docs: document cron contracts

* docs: document daemon contracts

* docs: document task contracts

* docs: document acp contracts

* docs: document test utility contracts

* docs: document skill contracts

* docs: document config contracts

* docs: document outbound infra contracts

* docs: document command analysis contracts

* docs: document provider usage infra contracts

* docs: document file safety infra contracts

* docs: document exec approval infra contracts

* docs: document gateway runtime infra contracts

* docs: document infra utility contracts

* docs: document infra queue storage contracts

* docs: document heartbeat infra contracts

* docs: document remaining infra contracts

* docs: document gateway auth contracts

* docs: document gateway display helpers

* docs: document gateway http helpers

* docs: document gateway node helpers

* docs: document gateway mcp helpers

* docs: document gateway support helpers

* docs: document gateway server runtime helpers

* docs: document gateway runtime bootstrap helpers

* docs: document gateway session events

* docs: document gateway utility helpers

* docs: document gateway talk helpers

* docs: document gateway helper contracts

* docs: document gateway server method helpers

* docs: document gateway server auth helpers

* docs: document gateway server tests

* docs: document gateway test helpers

* docs: document gateway node tests

* docs: document gateway channel tests

* docs: document gateway session tests

* docs: document gateway server startup tests

* docs: document gateway tool test helpers

* docs: document gateway server test helpers

* docs: document gateway server method tests

* docs: document remaining gateway tests

* docs: document plugin sdk public subpaths

* docs: document plugin sdk runtime helpers

* docs: document plugin sdk memory provider helpers

* docs: document plugin sdk runtime facades

* docs: document plugin sdk command approval helpers

* docs: document plugin sdk runtime types

* docs: document plugin sdk browser account helpers

* docs: document plugin sdk media memory helpers

* docs: document plugin sdk core tests

* docs: document plugin sdk contract helpers

* docs: document plugin sdk test helpers

* docs: document remaining plugin sdk tests

* docs: document cli utility helpers

* docs: document cli runtime helpers

* docs: document cli command registration helpers

* docs: document node cli helpers

* docs: document cli program registration

* docs: document message cli registration

* docs: document daemon cli helpers

* docs: document cli route parsers
2026-06-03 15:20:39 -07:00

766 lines
25 KiB
TypeScript

/**
* Shared token/cost formatting and pricing lookup helpers for CLI, TUI, gateway, and status output.
* Keep this module synchronous; request paths call it while rendering usage summaries.
*/
import path from "node:path";
import { normalizeOptionalString } from "@openclaw/normalization-core/string-coerce";
import { resolveDefaultAgentDir } from "../agents/agent-scope-config.js";
import { modelKey, normalizeModelRef, normalizeProviderId } from "../agents/model-selection.js";
import type { NormalizedUsage } from "../agents/usage.js";
import type { ModelProviderConfig } from "../config/types.models.js";
import type { OpenClawConfig } from "../config/types.openclaw.js";
import { getGatewayModelPricingCacheFingerprint } from "../gateway/model-pricing-cache-state.js";
import { getCachedGatewayModelPricing } from "../gateway/model-pricing-cache.js";
import { tryReadJsonSync } from "../infra/json-files.js";
/**
* A single tier in a tiered-pricing schedule. Prices are expressed as
* USD per-million tokens, just like the flat `ModelCostConfig` fields.
*
* `range` is a half-open interval `[start, end)` expressed in *input*
* token counts. The tiers MUST be sorted in ascending `range[0]` order
* with no gaps.
*/
export type PricingTier = {
input: number;
output: number;
cacheRead: number;
cacheWrite: number;
/** [startTokens, endTokens) — half-open interval on the input token axis. */
range: [number, number];
};
type RawPricingTier = {
input: number;
output: number;
cacheRead: number;
cacheWrite: number;
range: [number, number] | [number];
};
/** Per-million-token model pricing used by usage summaries and cost estimates. */
export type ModelCostConfig = {
input: number;
output: number;
cacheRead: number;
cacheWrite: number;
/** Optional tiered pricing tiers. When present, `estimateUsageCost`
* uses them instead of the flat rates above. The flat rates still
* serve as the "default / first-tier" fallback for callers that are
* unaware of tiered pricing. */
tieredPricing?: PricingTier[];
};
export type UsageTotals = {
input?: number;
output?: number;
cacheRead?: number;
cacheWrite?: number;
total?: number;
};
type ModelsJsonCostCache = {
path: string;
providers: Record<string, ModelProviderConfig> | undefined;
normalizedEntries: Map<string, ModelCostConfig> | null;
rawEntries: Map<string, ModelCostConfig> | null;
};
type ProviderCostIndexCacheEntry = {
normalizedEntries?: ProviderCostIndex;
rawEntries?: ProviderCostIndex;
};
type ProviderCostIndexSource = {
fingerprint: string;
model: NonNullable<ModelProviderConfig["models"]>[number];
providerKey: string;
rawCost: RawModelCostConfig;
};
type ProviderCostIndex = {
entries: Map<string, ModelCostConfig>;
sources: Map<string, ProviderCostIndexSource>;
structureFingerprint: string;
};
type RawModelCostConfig = Omit<ModelCostConfig, "tieredPricing"> & {
tieredPricing?: RawPricingTier[];
};
const EMPTY_PROVIDER_COST_INDEX = new Map<string, ModelCostConfig>();
const MODEL_KEY_CACHE_LIMIT = 4096;
let modelsJsonCostCache: ModelsJsonCostCache | null = null;
let providerCostIndexByConfig = new WeakMap<
Record<string, ModelProviderConfig>,
ProviderCostIndexCacheEntry
>();
let modelKeyCache = new Map<string, string | null>();
let sortedPricingTiersByInput = new WeakMap<PricingTier[], PricingTier[]>();
/** Formats a token count for compact human-facing status text. */
export function formatTokenCount(value?: number): string {
if (value === undefined || !Number.isFinite(value)) {
return "0";
}
const safe = Math.max(0, value);
if (safe >= 1_000_000) {
return `${(safe / 1_000_000).toFixed(1)}m`;
}
if (safe >= 1_000) {
const precision = safe >= 10_000 ? 0 : 1;
const formattedThousands = (safe / 1_000).toFixed(precision);
if (Number(formattedThousands) >= 1_000) {
return `${(safe / 1_000_000).toFixed(1)}m`;
}
return `${formattedThousands}k`;
}
return String(Math.round(safe));
}
/** Formats a USD amount for usage summaries, keeping tiny costs visible. */
export function formatUsd(value?: number): string | undefined {
if (value === undefined || !Number.isFinite(value)) {
return undefined;
}
if (value >= 1) {
return `$${value.toFixed(2)}`;
}
if (value >= 0.01) {
return `$${value.toFixed(2)}`;
}
return `$${value.toFixed(4)}`;
}
function toResolvedModelKey(params: {
provider?: string;
model?: string;
allowPluginNormalization?: boolean;
}): string | null {
const cacheKey = [
"resolved",
params.allowPluginNormalization === false ? "raw" : "default",
params.provider ?? "",
params.model ?? "",
].join("\0");
if (modelKeyCache.has(cacheKey)) {
return modelKeyCache.get(cacheKey) ?? null;
}
const provider = normalizeOptionalString(params.provider);
const model = normalizeOptionalString(params.model);
if (!provider || !model) {
cacheModelKey(cacheKey, null);
return null;
}
const normalized = normalizeModelRef(provider, model, {
allowManifestNormalization: params.allowPluginNormalization === false ? false : undefined,
allowPluginNormalization: params.allowPluginNormalization,
});
const key = modelKey(normalized.provider, normalized.model);
cacheModelKey(cacheKey, key);
return key;
}
function toDirectModelKey(params: { provider?: string; model?: string }): string | null {
const cacheKey = ["direct", params.provider ?? "", params.model ?? ""].join("\0");
if (modelKeyCache.has(cacheKey)) {
return modelKeyCache.get(cacheKey) ?? null;
}
const provider = normalizeProviderId(normalizeOptionalString(params.provider) ?? "");
const model = normalizeOptionalString(params.model);
if (!provider || !model) {
cacheModelKey(cacheKey, null);
return null;
}
const key = modelKey(provider, model);
cacheModelKey(cacheKey, key);
return key;
}
function cacheModelKey(cacheKey: string, key: string | null): void {
if (modelKeyCache.size >= MODEL_KEY_CACHE_LIMIT) {
modelKeyCache.clear();
}
modelKeyCache.set(cacheKey, key);
}
function shouldUseNormalizedCostLookup(params: { provider?: string; model?: string }): boolean {
const provider = normalizeProviderId(normalizeOptionalString(params.provider) ?? "");
const model = normalizeOptionalString(params.model) ?? "";
if (!provider || !model) {
return false;
}
return provider === "anthropic" || provider === "openrouter" || provider === "vercel-ai-gateway";
}
/**
* Normalize a raw tieredPricing array from models.json / config.
* Supports open-ended ranges such as `[128000]` or `[128000, -1]`,
* which are converted to `[128000, Infinity]`.
*/
function normalizeTieredPricing(raw: RawPricingTier[] | undefined): PricingTier[] | undefined {
if (!raw || raw.length === 0) {
return undefined;
}
const result: PricingTier[] = [];
for (const tier of raw) {
const range = tier.range;
if (!Array.isArray(range) || range.length < 1) {
continue;
}
const start = typeof range[0] === "number" ? range[0] : Number.NaN;
if (!Number.isFinite(start)) {
continue;
}
const rawEnd = range.length >= 2 ? range[1] : null;
const end =
typeof rawEnd === "number" && Number.isFinite(rawEnd) && rawEnd > start ? rawEnd : Infinity;
if (
!Number.isFinite(tier.input) ||
!Number.isFinite(tier.output) ||
!Number.isFinite(tier.cacheRead) ||
!Number.isFinite(tier.cacheWrite)
) {
continue;
}
result.push({
input: tier.input,
output: tier.output,
cacheRead: tier.cacheRead,
cacheWrite: tier.cacheWrite,
range: [start, end],
});
}
return result.length > 0 ? result.toSorted((a, b) => a.range[0] - b.range[0]) : undefined;
}
function normalizeModelCostConfig(cost: RawModelCostConfig): ModelCostConfig {
const normalizedTiers = normalizeTieredPricing(cost.tieredPricing);
return {
input: cost.input,
output: cost.output,
cacheRead: cost.cacheRead,
cacheWrite: cost.cacheWrite,
...(normalizedTiers ? { tieredPricing: normalizedTiers } : {}),
};
}
function isRawModelCostConfig(value: unknown): value is RawModelCostConfig {
return value !== null && typeof value === "object";
}
function buildProviderCostStructureFingerprint(
providers: Record<string, ModelProviderConfig> | undefined,
): string {
if (!providers) {
return "";
}
return Object.entries(providers)
.toSorted(([a], [b]) => a.localeCompare(b))
.flatMap(([providerKey, providerConfig]) =>
(providerConfig?.models ?? []).map(
(model) =>
`${providerKey}\0${model.id}\0${isRawModelCostConfig(model.cost) ? "cost" : "metadata"}`,
),
)
.join("\0");
}
function buildProviderCostIndexBundle(
providers: Record<string, ModelProviderConfig> | undefined,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): ProviderCostIndex {
const entries = new Map<string, ModelCostConfig>();
const sources = new Map<string, ProviderCostIndexSource>();
const structureFingerprint = buildProviderCostStructureFingerprint(providers);
if (!providers) {
return { entries, sources, structureFingerprint };
}
for (const [providerKey, providerConfig] of Object.entries(providers)) {
const normalizedProvider = normalizeProviderId(providerKey);
for (const model of providerConfig?.models ?? []) {
const normalized = normalizeModelRef(normalizedProvider, model.id, {
allowManifestNormalization:
options?.allowManifestNormalization ??
(options?.allowPluginNormalization === false ? false : undefined),
allowPluginNormalization: options?.allowPluginNormalization,
});
const key = modelKey(normalized.provider, normalized.model);
if (!isRawModelCostConfig(model.cost)) {
continue;
}
const rawCost = model.cost;
entries.set(key, normalizeModelCostConfig(rawCost));
sources.set(key, {
fingerprint: buildModelCostFingerprint(rawCost),
model,
providerKey,
rawCost,
});
}
}
return { entries, sources, structureFingerprint };
}
function buildProviderCostIndex(
providers: Record<string, ModelProviderConfig> | undefined,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): Map<string, ModelCostConfig> {
return buildProviderCostIndexBundle(providers, options).entries;
}
function getProviderCostIndex(
providers: Record<string, ModelProviderConfig> | undefined,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): Map<string, ModelCostConfig> {
if (!providers) {
return EMPTY_PROVIDER_COST_INDEX;
}
const isRawLookup =
options?.allowPluginNormalization === false &&
(options.allowManifestNormalization === false ||
options.allowManifestNormalization === undefined);
const isDefaultNormalizedLookup =
options?.allowPluginNormalization !== false &&
options?.allowManifestNormalization === undefined;
if (!isRawLookup && !isDefaultNormalizedLookup) {
return buildProviderCostIndex(providers, options);
}
let cache = providerCostIndexByConfig.get(providers);
if (!cache) {
cache = {};
providerCostIndexByConfig.set(providers, cache);
}
if (isRawLookup) {
cache.rawEntries ??= buildProviderCostIndexBundle(providers, {
allowManifestNormalization: false,
allowPluginNormalization: false,
});
const rawOptions = {
allowManifestNormalization: false,
allowPluginNormalization: false,
};
if (refreshProviderCostIndexMutations(cache.rawEntries, providers, rawOptions) === "rebuild") {
cache.rawEntries = buildProviderCostIndexBundle(providers, rawOptions);
}
if (
cache.rawEntries.structureFingerprint !== buildProviderCostStructureFingerprint(providers)
) {
cache.rawEntries = buildProviderCostIndexBundle(providers, rawOptions);
}
return cache.rawEntries.entries;
}
cache.normalizedEntries ??= buildProviderCostIndexBundle(providers);
if (refreshProviderCostIndexMutations(cache.normalizedEntries, providers) === "rebuild") {
cache.normalizedEntries = buildProviderCostIndexBundle(providers);
}
if (
cache.normalizedEntries.structureFingerprint !==
buildProviderCostStructureFingerprint(providers)
) {
cache.normalizedEntries = buildProviderCostIndexBundle(providers);
}
return cache.normalizedEntries.entries;
}
function loadModelsJsonCostIndex(options?: {
allowPluginNormalization?: boolean;
}): Map<string, ModelCostConfig> {
const useRawEntries = options?.allowPluginNormalization === false;
const modelsPath = path.join(resolveDefaultAgentDir({}), "models.json");
try {
if (!modelsJsonCostCache || modelsJsonCostCache.path !== modelsPath) {
const parsed = tryReadJsonSync<{
providers?: Record<string, ModelProviderConfig>;
}>(modelsPath);
if (!parsed) {
return EMPTY_PROVIDER_COST_INDEX;
}
modelsJsonCostCache = {
path: modelsPath,
providers: parsed?.providers,
normalizedEntries: null,
rawEntries: null,
};
}
if (useRawEntries) {
modelsJsonCostCache.rawEntries ??= getProviderCostIndex(modelsJsonCostCache.providers, {
allowPluginNormalization: false,
});
return modelsJsonCostCache.rawEntries;
}
modelsJsonCostCache.normalizedEntries ??= getProviderCostIndex(modelsJsonCostCache.providers);
return modelsJsonCostCache.normalizedEntries;
} catch {
return EMPTY_PROVIDER_COST_INDEX;
}
}
function findConfiguredProviderCost(params: {
provider?: string;
model?: string;
config?: OpenClawConfig;
allowPluginNormalization?: boolean;
}): ModelCostConfig | undefined {
const key = toResolvedModelKey(params);
if (!key) {
return undefined;
}
return getProviderCostFromIndex(params.config?.models?.providers, key, {
allowPluginNormalization: params.allowPluginNormalization,
});
}
function stableCostFingerprintValue(value: unknown): string {
if (typeof value === "number") {
return Number.isFinite(value) ? JSON.stringify(value) : JSON.stringify(String(value));
}
if (value === null || typeof value !== "object") {
return JSON.stringify(value);
}
if (Array.isArray(value)) {
return `[${value.map((entry) => stableCostFingerprintValue(entry)).join(",")}]`;
}
const record = value as Record<string, unknown>;
return `{${Object.keys(record)
.filter((key) => record[key] !== undefined)
.toSorted()
.map((key) => `${JSON.stringify(key)}:${stableCostFingerprintValue(record[key])}`)
.join(",")}}`;
}
function buildModelCostFingerprint(cost: RawModelCostConfig): string {
const tierFingerprint = Array.isArray(cost.tieredPricing)
? cost.tieredPricing.flatMap((tier) => {
const range = Array.isArray(tier.range) ? tier.range : [];
return [tier.input, tier.output, tier.cacheRead, tier.cacheWrite, ...range];
})
: [];
return [cost.input, cost.output, cost.cacheRead, cost.cacheWrite, ...tierFingerprint].join("|");
}
function isProviderCostSourceCurrent(
providers: Record<string, ModelProviderConfig>,
source: ProviderCostIndexSource,
key: string,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): boolean {
const providerConfig = providers[source.providerKey];
if (!providerConfig?.models?.includes(source.model)) {
return false;
}
const normalized = normalizeModelRef(normalizeProviderId(source.providerKey), source.model.id, {
allowManifestNormalization:
options?.allowManifestNormalization ??
(options?.allowPluginNormalization === false ? false : undefined),
allowPluginNormalization: options?.allowPluginNormalization,
});
return modelKey(normalized.provider, normalized.model) === key;
}
function refreshProviderCostIndexEntry(
index: ProviderCostIndex,
key: string,
providers?: Record<string, ModelProviderConfig>,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): "current" | "rebuild" {
const source = index.sources.get(key);
if (!source) {
return "current";
}
if (providers && !isProviderCostSourceCurrent(providers, source, key, options)) {
return "rebuild";
}
if (!isRawModelCostConfig(source.model.cost)) {
return "rebuild";
}
if (source.model.cost !== source.rawCost) {
source.rawCost = source.model.cost;
}
const fingerprint = buildModelCostFingerprint(source.rawCost);
if (source.fingerprint === fingerprint) {
return "current";
}
source.fingerprint = fingerprint;
index.entries.set(key, normalizeModelCostConfig(source.rawCost));
return "current";
}
function refreshProviderCostIndexMutations(
index: ProviderCostIndex,
providers?: Record<string, ModelProviderConfig>,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): "current" | "rebuild" {
for (const key of index.sources.keys()) {
if (refreshProviderCostIndexEntry(index, key, providers, options) === "rebuild") {
return "rebuild";
}
}
return "current";
}
function hasProviderCostSourceForKey(
providers: Record<string, ModelProviderConfig>,
key: string,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): boolean {
for (const [providerKey, providerConfig] of Object.entries(providers)) {
const normalizedProvider = normalizeProviderId(providerKey);
for (const model of providerConfig?.models ?? []) {
if (!isRawModelCostConfig(model.cost)) {
continue;
}
const normalized = normalizeModelRef(normalizedProvider, model.id, {
allowManifestNormalization:
options?.allowManifestNormalization ??
(options?.allowPluginNormalization === false ? false : undefined),
allowPluginNormalization: options?.allowPluginNormalization,
});
if (modelKey(normalized.provider, normalized.model) === key) {
return true;
}
}
}
return false;
}
function getProviderCostFromIndex(
providers: Record<string, ModelProviderConfig> | undefined,
key: string,
options?: { allowManifestNormalization?: boolean; allowPluginNormalization?: boolean },
): ModelCostConfig | undefined {
if (!providers) {
return undefined;
}
const isRawLookup =
options?.allowPluginNormalization === false &&
(options.allowManifestNormalization === false ||
options.allowManifestNormalization === undefined);
const isDefaultNormalizedLookup =
options?.allowPluginNormalization !== false &&
options?.allowManifestNormalization === undefined;
if (!isRawLookup && !isDefaultNormalizedLookup) {
return buildProviderCostIndex(providers, options).get(key);
}
let cache = providerCostIndexByConfig.get(providers);
if (!cache) {
cache = {};
providerCostIndexByConfig.set(providers, cache);
}
const index = isRawLookup
? (cache.rawEntries ??= buildProviderCostIndexBundle(providers, {
allowManifestNormalization: false,
allowPluginNormalization: false,
}))
: (cache.normalizedEntries ??= buildProviderCostIndexBundle(providers));
const sourceMissingWithStructuralChange =
!index.sources.has(key) &&
index.structureFingerprint !== buildProviderCostStructureFingerprint(providers);
const sourceMissingWithNewCost =
!index.sources.has(key) && hasProviderCostSourceForKey(providers, key, options);
if (
refreshProviderCostIndexEntry(index, key, providers, options) === "rebuild" ||
sourceMissingWithStructuralChange ||
sourceMissingWithNewCost
) {
const rebuilt = buildProviderCostIndexBundle(
providers,
isRawLookup
? {
allowManifestNormalization: false,
allowPluginNormalization: false,
}
: undefined,
);
if (isRawLookup) {
cache.rawEntries = rebuilt;
} else {
cache.normalizedEntries = rebuilt;
}
return rebuilt.entries.get(key);
}
return index.entries.get(key);
}
function serializeCostIndex(
entries: Map<string, ModelCostConfig>,
): Array<[string, ModelCostConfig]> {
return Array.from(entries.entries()).toSorted(([a], [b]) => a.localeCompare(b));
}
/**
* Fingerprints all model-pricing sources that can affect usage cost estimates.
* Consumers cache this value to know when resolved cost entries need recomputation.
*/
export function resolveModelCostConfigFingerprint(config?: OpenClawConfig): string {
return stableCostFingerprintValue({
configuredRaw: serializeCostIndex(
getProviderCostIndex(config?.models?.providers, { allowPluginNormalization: false }),
),
configuredNormalized: serializeCostIndex(getProviderCostIndex(config?.models?.providers)),
modelsJsonRaw: serializeCostIndex(loadModelsJsonCostIndex({ allowPluginNormalization: false })),
modelsJsonNormalized: serializeCostIndex(loadModelsJsonCostIndex()),
gatewayPricing: getGatewayModelPricingCacheFingerprint(),
});
}
/**
* Resolves pricing for a provider/model pair from local models.json, configured models, then gateway cache.
* Direct keys win before plugin normalization so configured pricing does not trigger provider discovery.
*/
export function resolveModelCostConfig(params: {
provider?: string;
model?: string;
config?: OpenClawConfig;
allowPluginNormalization?: boolean;
}): ModelCostConfig | undefined {
const rawKey = toDirectModelKey(params);
if (!rawKey) {
return undefined;
}
// Favor direct configured keys first so local pricing/status lookups stay
// synchronous and do not drag plugin/provider discovery into the hot path.
const rawModelsJsonCost = loadModelsJsonCostIndex({
allowPluginNormalization: false,
}).get(rawKey);
if (rawModelsJsonCost) {
return rawModelsJsonCost;
}
const rawConfiguredCost = findConfiguredProviderCost({
...params,
allowPluginNormalization: false,
});
if (rawConfiguredCost) {
return rawConfiguredCost;
}
if (params.allowPluginNormalization === false) {
return undefined;
}
if (shouldUseNormalizedCostLookup(params)) {
const key = toResolvedModelKey(params);
if (key && key !== rawKey) {
const modelsJsonCost = loadModelsJsonCostIndex().get(key);
if (modelsJsonCost) {
return modelsJsonCost;
}
const configuredCost = findConfiguredProviderCost(params);
if (configuredCost) {
return configuredCost;
}
}
}
return getCachedGatewayModelPricing(params);
}
const toNumber = (value: number | undefined): number =>
typeof value === "number" && Number.isFinite(value) ? value : 0;
function selectPricingTier(tiers: PricingTier[], input: number): PricingTier | undefined {
const sortedTiers = getSortedPricingTiers(tiers);
if (sortedTiers.length === 0) {
return undefined;
}
if (input <= 0) {
return sortedTiers[0];
}
for (const tier of sortedTiers) {
const [start, end] = tier.range;
if (input >= start && input < end) {
return tier;
}
}
for (let index = sortedTiers.length - 1; index >= 0; index -= 1) {
const tier = sortedTiers[index];
if (input >= tier.range[0]) {
return tier;
}
}
return sortedTiers[0];
}
function getSortedPricingTiers(tiers: PricingTier[]): PricingTier[] {
const cached = sortedPricingTiersByInput.get(tiers);
if (cached) {
return cached;
}
const sorted = tiers.toSorted((a, b) => a.range[0] - b.range[0]);
sortedPricingTiersByInput.set(tiers, sorted);
return sorted;
}
function computeTieredCost(
tiers: PricingTier[],
input: number,
output: number,
cacheRead: number,
cacheWrite: number,
): number {
const tier = selectPricingTier(tiers, input);
if (!tier) {
return 0;
}
return (
input * tier.input +
output * tier.output +
cacheRead * tier.cacheRead +
cacheWrite * tier.cacheWrite
);
}
/**
* Estimates USD usage cost from normalized token totals.
* Tiered pricing selects one whole-request tier by input size; it does not blend tiers.
*/
export function estimateUsageCost(params: {
usage?: NormalizedUsage | UsageTotals | null;
cost?: ModelCostConfig;
}): number | undefined {
const usage = params.usage;
const cost = params.cost;
if (!usage || !cost) {
return undefined;
}
const input = toNumber(usage.input);
const output = toNumber(usage.output);
const cacheRead = toNumber(usage.cacheRead);
const cacheWrite = toNumber(usage.cacheWrite);
let total: number;
if (cost.tieredPricing && cost.tieredPricing.length > 0) {
total = computeTieredCost(cost.tieredPricing, input, output, cacheRead, cacheWrite);
} else {
total =
input * cost.input +
output * cost.output +
cacheRead * cost.cacheRead +
cacheWrite * cost.cacheWrite;
}
if (!Number.isFinite(total)) {
return undefined;
}
return total / 1_000_000;
}
export function resetUsageFormatCachesForTest(): void {
modelsJsonCostCache = null;
providerCostIndexByConfig = new WeakMap();
modelKeyCache = new Map();
sortedPricingTiersByInput = new WeakMap();
}