Files
deer-flow/frontend/src/core/messages/utils.ts

245 lines
6.6 KiB
TypeScript
Raw Normal View History

2026-01-15 23:40:21 +08:00
import type { Message } from "@langchain/langgraph-sdk";
2026-01-18 11:25:46 +08:00
interface GenericMessageGroup<T = string> {
type: T;
id: string | undefined;
messages: Message[];
}
interface HumanMessageGroup extends GenericMessageGroup<"human"> {}
interface AssistantProcessingGroup extends GenericMessageGroup<"assistant:processing"> {}
interface AssistantMessageGroup extends GenericMessageGroup<"assistant"> {}
interface AssistantPresentFilesGroup extends GenericMessageGroup<"assistant:present-files"> {}
interface AssistantClarificationGroup extends GenericMessageGroup<"assistant:clarification"> {}
2026-01-18 11:25:46 +08:00
type MessageGroup =
| HumanMessageGroup
| AssistantProcessingGroup
| AssistantMessageGroup
| AssistantPresentFilesGroup
| AssistantClarificationGroup;
2026-01-18 11:25:46 +08:00
2026-01-15 23:40:21 +08:00
export function groupMessages<T>(
messages: Message[],
2026-01-18 11:25:46 +08:00
mapper: (group: MessageGroup) => T,
2026-01-15 23:40:21 +08:00
): T[] {
if (messages.length === 0) {
return [];
}
2026-01-18 11:25:46 +08:00
const groups: MessageGroup[] = [];
2026-01-15 23:40:21 +08:00
for (const message of messages) {
2026-01-18 11:25:46 +08:00
const lastGroup = groups[groups.length - 1];
2026-01-15 23:40:21 +08:00
if (message.type === "human") {
2026-01-18 11:25:46 +08:00
groups.push({
id: message.id,
type: "human",
messages: [message],
});
2026-01-15 23:40:21 +08:00
} else if (message.type === "tool") {
// Check if this is a clarification tool message
if (isClarificationToolMessage(message)) {
// Add to processing group if available (to maintain tool call association)
if (
lastGroup &&
lastGroup.type !== "human" &&
lastGroup.type !== "assistant" &&
lastGroup.type !== "assistant:clarification"
) {
lastGroup.messages.push(message);
}
// Also create a separate clarification group for prominent display
groups.push({
id: message.id,
type: "assistant:clarification",
messages: [message],
});
} else if (
2026-01-18 11:25:46 +08:00
lastGroup &&
lastGroup.type !== "human" &&
lastGroup.type !== "assistant" &&
lastGroup.type !== "assistant:clarification"
2026-01-15 23:40:21 +08:00
) {
2026-01-18 11:25:46 +08:00
lastGroup.messages.push(message);
} else {
throw new Error(
"Tool message must be matched with a previous assistant message with tool calls",
);
}
} else if (message.type === "ai") {
if (hasReasoning(message) || hasToolCalls(message)) {
if (hasPresentFiles(message)) {
groups.push({
id: message.id,
type: "assistant:present-files",
messages: [message],
});
2026-01-16 22:35:20 +08:00
} else {
2026-01-18 11:25:46 +08:00
if (lastGroup?.type !== "assistant:processing") {
groups.push({
id: message.id,
type: "assistant:processing",
messages: [],
});
}
const currentGroup = groups[groups.length - 1];
if (currentGroup?.type === "assistant:processing") {
currentGroup.messages.push(message);
} else {
throw new Error(
"Assistant message with reasoning or tool calls must be preceded by a processing group",
);
}
2026-01-16 22:35:20 +08:00
}
2026-01-18 11:25:46 +08:00
}
if (hasContent(message) && !hasToolCalls(message)) {
groups.push({
id: message.id,
type: "assistant",
messages: [message],
});
}
}
}
const resultsOfGroups: T[] = [];
for (const group of groups) {
const resultOfGroup = mapper(group);
if (resultOfGroup !== undefined && resultOfGroup !== null) {
resultsOfGroups.push(resultOfGroup);
}
}
2026-01-15 23:40:21 +08:00
return resultsOfGroups;
}
export function extractTextFromMessage(message: Message) {
if (typeof message.content === "string") {
return message.content.trim();
}
if (Array.isArray(message.content)) {
return message.content
.map((content) => (content.type === "text" ? content.text : ""))
.join("\n")
.trim();
}
return "";
}
export function extractContentFromMessage(message: Message) {
if (typeof message.content === "string") {
return message.content.trim();
}
if (Array.isArray(message.content)) {
return message.content
.map((content) => {
switch (content.type) {
case "text":
return content.text;
case "image_url":
const imageURL = extractURLFromImageURLContent(content.image_url);
return `![image](${imageURL})`;
default:
return "";
}
})
.join("\n")
.trim();
}
return "";
}
export function extractReasoningContentFromMessage(message: Message) {
if (message.type !== "ai" || !message.additional_kwargs) {
return null;
}
if ("reasoning_content" in message.additional_kwargs) {
return message.additional_kwargs.reasoning_content as string | null;
}
return null;
}
2026-01-18 13:07:56 +08:00
export function removeReasoningContentFromMessage(message: Message) {
if (message.type !== "ai" || !message.additional_kwargs) {
return;
}
delete message.additional_kwargs.reasoning_content;
}
2026-01-15 23:40:21 +08:00
export function extractURLFromImageURLContent(
content:
| string
| {
url: string;
},
) {
if (typeof content === "string") {
return content;
}
return content.url;
}
export function hasContent(message: Message) {
if (typeof message.content === "string") {
return message.content.trim().length > 0;
}
if (Array.isArray(message.content)) {
return message.content.length > 0;
}
return false;
}
export function hasReasoning(message: Message) {
return (
message.type === "ai" &&
typeof message.additional_kwargs?.reasoning_content === "string"
);
}
export function hasToolCalls(message: Message) {
return (
message.type === "ai" && message.tool_calls && message.tool_calls.length > 0
);
}
2026-01-16 22:35:20 +08:00
export function hasPresentFiles(message: Message) {
return (
message.type === "ai" && message.tool_calls?.[0]?.name === "present_files"
);
}
export function isClarificationToolMessage(message: Message) {
return message.type === "tool" && message.name === "ask_clarification";
}
2026-01-16 22:35:20 +08:00
export function extractPresentFilesFromMessage(message: Message) {
if (message.type !== "ai" || !hasPresentFiles(message)) {
return [];
}
2026-01-16 23:15:53 +08:00
const files: string[] = [];
2026-01-16 22:35:20 +08:00
for (const toolCall of message.tool_calls ?? []) {
2026-01-16 23:15:53 +08:00
if (
toolCall.name === "present_files" &&
Array.isArray(toolCall.args.filepaths)
) {
2026-01-16 22:35:20 +08:00
files.push(...(toolCall.args.filepaths as string[]));
}
}
return files;
}
2026-01-15 23:40:21 +08:00
export function findToolCallResult(toolCallId: string, messages: Message[]) {
for (const message of messages) {
if (message.type === "tool" && message.tool_call_id === toolCallId) {
const content = extractTextFromMessage(message);
if (content) {
return content;
}
}
}
return undefined;
}