Files
openclaw/src/utils/transcript-tools.ts
Peter Steinberger 00d8d7ead0 refactor: extract normalization core package
Extract shared normalization/coercion helpers into private @openclaw/normalization-core workspace package while preserving existing plugin SDK helper subpaths.\n\nAlso keeps direct normalization-core imports internal, wires UI/build/loader resolution, and replaces the slow PR network CodeQL lane with a fast added-line boundary scan while retaining full CodeQL for scheduled/manual runs.\n\nVerification: local moved tests, plugin SDK boundary tests, extension loader tests, agents-support shard, UI build/test, build artifacts, lint, workflow guards, autoreview, and GitHub CI passed on PR head 963d893715.
2026-05-31 01:33:00 +01:00

78 lines
2.0 KiB
TypeScript

import {
normalizeOptionalLowercaseString,
normalizeOptionalString,
} from "@openclaw/normalization-core/string-coerce";
type ToolResultCounts = {
total: number;
errors: number;
};
const TOOL_CALL_TYPES = new Set(["tool_use", "toolcall", "tool_call"]);
const TOOL_RESULT_TYPES = new Set(["tool_result", "tool_result_error"]);
const normalizeType = (value: unknown): string => {
return typeof value === "string" ? (normalizeOptionalLowercaseString(value) ?? "") : "";
};
export const extractToolCallNames = (message: Record<string, unknown>): string[] => {
const names = new Set<string>();
const toolNameRaw = message.toolName ?? message.tool_name;
const toolName =
typeof toolNameRaw === "string" ? normalizeOptionalString(toolNameRaw) : undefined;
if (toolName) {
names.add(toolName);
}
const content = message.content;
if (!Array.isArray(content)) {
return Array.from(names);
}
for (const entry of content) {
if (!entry || typeof entry !== "object") {
continue;
}
const block = entry as Record<string, unknown>;
const type = normalizeType(block.type);
if (!TOOL_CALL_TYPES.has(type)) {
continue;
}
const name = typeof block.name === "string" ? normalizeOptionalString(block.name) : undefined;
if (name) {
names.add(name);
}
}
return Array.from(names);
};
export const hasToolCall = (message: Record<string, unknown>): boolean =>
extractToolCallNames(message).length > 0;
export const countToolResults = (message: Record<string, unknown>): ToolResultCounts => {
const content = message.content;
if (!Array.isArray(content)) {
return { total: 0, errors: 0 };
}
let total = 0;
let errors = 0;
for (const entry of content) {
if (!entry || typeof entry !== "object") {
continue;
}
const block = entry as Record<string, unknown>;
const type = normalizeType(block.type);
if (!TOOL_RESULT_TYPES.has(type)) {
continue;
}
total += 1;
if (block.is_error === true) {
errors += 1;
}
}
return { total, errors };
};