Files
openclaw/src/plugins/web-search-providers.runtime.ts
Tak Hoffman 0f6dbb4390 Fix configure cache ownership for plugin resolvers (#52018)
* fix configure cache ownership

* address web-search cache review

* rekey provider wizard cache

* honor plugin cache opt-outs

* fix cache invalidation gaps

* align plugin snapshot ttl

* refresh snapshot cache keys
2026-03-21 23:41:45 -05:00

188 lines
6.1 KiB
TypeScript

import type { OpenClawConfig } from "../config/config.js";
import { createSubsystemLogger } from "../logging/subsystem.js";
import { loadOpenClawPlugins } from "./loader.js";
import type { PluginLoadOptions } from "./loader.js";
import { createPluginLoaderLogger } from "./logger.js";
import { getActivePluginRegistry } from "./runtime.js";
import type { PluginWebSearchProviderEntry } from "./types.js";
import {
resolveBundledWebSearchResolutionConfig,
sortWebSearchProviders,
} from "./web-search-providers.shared.js";
const log = createSubsystemLogger("plugins");
type WebSearchProviderSnapshotCacheEntry = {
expiresAt: number;
providers: PluginWebSearchProviderEntry[];
};
const webSearchProviderSnapshotCache = new WeakMap<
OpenClawConfig,
WeakMap<NodeJS.ProcessEnv, Map<string, WebSearchProviderSnapshotCacheEntry>>
>();
const DEFAULT_DISCOVERY_CACHE_MS = 1000;
const DEFAULT_MANIFEST_CACHE_MS = 1000;
function shouldUseWebSearchProviderSnapshotCache(env: NodeJS.ProcessEnv): boolean {
if (env.OPENCLAW_DISABLE_PLUGIN_DISCOVERY_CACHE?.trim()) {
return false;
}
if (env.OPENCLAW_DISABLE_PLUGIN_MANIFEST_CACHE?.trim()) {
return false;
}
const discoveryCacheMs = env.OPENCLAW_PLUGIN_DISCOVERY_CACHE_MS?.trim();
if (discoveryCacheMs === "0") {
return false;
}
const manifestCacheMs = env.OPENCLAW_PLUGIN_MANIFEST_CACHE_MS?.trim();
if (manifestCacheMs === "0") {
return false;
}
return true;
}
function resolveWebSearchProviderSnapshotCacheTtlMs(env: NodeJS.ProcessEnv): number {
const discoveryCacheMs = resolveCacheMs(
env.OPENCLAW_PLUGIN_DISCOVERY_CACHE_MS,
DEFAULT_DISCOVERY_CACHE_MS,
);
const manifestCacheMs = resolveCacheMs(
env.OPENCLAW_PLUGIN_MANIFEST_CACHE_MS,
DEFAULT_MANIFEST_CACHE_MS,
);
return Math.min(discoveryCacheMs, manifestCacheMs);
}
function resolveCacheMs(rawValue: string | undefined, defaultMs: number): number {
const raw = rawValue?.trim();
if (raw === "" || raw === "0") {
return 0;
}
if (!raw) {
return defaultMs;
}
const parsed = Number.parseInt(raw, 10);
if (!Number.isFinite(parsed)) {
return defaultMs;
}
return Math.max(0, parsed);
}
function buildWebSearchSnapshotCacheKey(params: {
config?: OpenClawConfig;
workspaceDir?: string;
bundledAllowlistCompat?: boolean;
env: NodeJS.ProcessEnv;
}): string {
const effectiveVitest = params.env.VITEST ?? process.env.VITEST ?? "";
return JSON.stringify({
workspaceDir: params.workspaceDir ?? "",
bundledAllowlistCompat: params.bundledAllowlistCompat === true,
config: params.config ?? null,
env: {
OPENCLAW_BUNDLED_PLUGINS_DIR: params.env.OPENCLAW_BUNDLED_PLUGINS_DIR ?? "",
OPENCLAW_DISABLE_PLUGIN_DISCOVERY_CACHE:
params.env.OPENCLAW_DISABLE_PLUGIN_DISCOVERY_CACHE ?? "",
OPENCLAW_DISABLE_PLUGIN_MANIFEST_CACHE:
params.env.OPENCLAW_DISABLE_PLUGIN_MANIFEST_CACHE ?? "",
OPENCLAW_PLUGIN_DISCOVERY_CACHE_MS: params.env.OPENCLAW_PLUGIN_DISCOVERY_CACHE_MS ?? "",
OPENCLAW_PLUGIN_MANIFEST_CACHE_MS: params.env.OPENCLAW_PLUGIN_MANIFEST_CACHE_MS ?? "",
OPENCLAW_HOME: params.env.OPENCLAW_HOME ?? "",
OPENCLAW_STATE_DIR: params.env.OPENCLAW_STATE_DIR ?? "",
CLAWDBOT_STATE_DIR: params.env.CLAWDBOT_STATE_DIR ?? "",
OPENCLAW_CONFIG_PATH: params.env.OPENCLAW_CONFIG_PATH ?? "",
HOME: params.env.HOME ?? "",
USERPROFILE: params.env.USERPROFILE ?? "",
VITEST: effectiveVitest,
},
});
}
export function resolvePluginWebSearchProviders(params: {
config?: PluginLoadOptions["config"];
workspaceDir?: string;
env?: PluginLoadOptions["env"];
bundledAllowlistCompat?: boolean;
activate?: boolean;
cache?: boolean;
}): PluginWebSearchProviderEntry[] {
const env = params.env ?? process.env;
const cacheOwnerConfig = params.config;
const shouldMemoizeSnapshot =
params.activate !== true &&
params.cache !== true &&
shouldUseWebSearchProviderSnapshotCache(env);
const cacheKey = buildWebSearchSnapshotCacheKey({
config: cacheOwnerConfig,
workspaceDir: params.workspaceDir,
bundledAllowlistCompat: params.bundledAllowlistCompat,
env,
});
if (cacheOwnerConfig && shouldMemoizeSnapshot) {
const configCache = webSearchProviderSnapshotCache.get(cacheOwnerConfig);
const envCache = configCache?.get(env);
const cached = envCache?.get(cacheKey);
if (cached && cached.expiresAt > Date.now()) {
return cached.providers;
}
}
const { config } = resolveBundledWebSearchResolutionConfig({
...params,
env,
});
const registry = loadOpenClawPlugins({
config,
workspaceDir: params.workspaceDir,
env,
cache: params.cache ?? false,
activate: params.activate ?? false,
logger: createPluginLoaderLogger(log),
});
const resolved = sortWebSearchProviders(
registry.webSearchProviders.map((entry) => ({
...entry.provider,
pluginId: entry.pluginId,
})),
);
if (cacheOwnerConfig && shouldMemoizeSnapshot) {
const ttlMs = resolveWebSearchProviderSnapshotCacheTtlMs(env);
let configCache = webSearchProviderSnapshotCache.get(cacheOwnerConfig);
if (!configCache) {
configCache = new WeakMap<
NodeJS.ProcessEnv,
Map<string, WebSearchProviderSnapshotCacheEntry>
>();
webSearchProviderSnapshotCache.set(cacheOwnerConfig, configCache);
}
let envCache = configCache.get(env);
if (!envCache) {
envCache = new Map<string, WebSearchProviderSnapshotCacheEntry>();
configCache.set(env, envCache);
}
envCache.set(cacheKey, {
expiresAt: Date.now() + ttlMs,
providers: resolved,
});
}
return resolved;
}
export function resolveRuntimeWebSearchProviders(params: {
config?: PluginLoadOptions["config"];
workspaceDir?: string;
env?: PluginLoadOptions["env"];
bundledAllowlistCompat?: boolean;
}): PluginWebSearchProviderEntry[] {
const runtimeProviders = getActivePluginRegistry()?.webSearchProviders ?? [];
if (runtimeProviders.length > 0) {
return sortWebSearchProviders(
runtimeProviders.map((entry) => ({
...entry.provider,
pluginId: entry.pluginId,
})),
);
}
return resolvePluginWebSearchProviders(params);
}