Files
openclaw/src/media-understanding/scope.ts
2026-02-18 01:34:35 +00:00

65 lines
1.9 KiB
TypeScript

import { normalizeChatType } from "../channels/chat-type.js";
import type { MediaUnderstandingScopeConfig } from "../config/types.tools.js";
export type MediaUnderstandingScopeDecision = "allow" | "deny";
function normalizeDecision(value?: string | null): MediaUnderstandingScopeDecision | undefined {
const normalized = value?.trim().toLowerCase();
if (normalized === "allow") {
return "allow";
}
if (normalized === "deny") {
return "deny";
}
return undefined;
}
function normalizeMatch(value?: string | null): string | undefined {
const normalized = value?.trim().toLowerCase();
return normalized || undefined;
}
export function normalizeMediaUnderstandingChatType(raw?: string | null): string | undefined {
return normalizeChatType(raw ?? undefined);
}
export function resolveMediaUnderstandingScope(params: {
scope?: MediaUnderstandingScopeConfig;
sessionKey?: string;
channel?: string;
chatType?: string;
}): MediaUnderstandingScopeDecision {
const scope = params.scope;
if (!scope) {
return "allow";
}
const channel = normalizeMatch(params.channel);
const chatType = normalizeMediaUnderstandingChatType(params.chatType);
const sessionKey = normalizeMatch(params.sessionKey) ?? "";
for (const rule of scope.rules ?? []) {
if (!rule) {
continue;
}
const action = normalizeDecision(rule.action) ?? "allow";
const match = rule.match ?? {};
const matchChannel = normalizeMatch(match.channel);
const matchChatType = normalizeMediaUnderstandingChatType(match.chatType);
const matchPrefix = normalizeMatch(match.keyPrefix);
if (matchChannel && matchChannel !== channel) {
continue;
}
if (matchChatType && matchChatType !== chatType) {
continue;
}
if (matchPrefix && !sessionKey.startsWith(matchPrefix)) {
continue;
}
return action;
}
return normalizeDecision(scope.default) ?? "allow";
}