mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-22 06:00:56 +02:00

* fix: handling of top_k and top_p parameters for Claude-3.7 models (allowed without reasoning) * feat: bump @librechat/agents for Anthropic Reasoning support * fix: update reasoning handling for OpenRouter integration * fix: enhance agent token spending logic to include cache creation and read details * fix: update logic for thinking status in ContentParts component * refactor: improve agent title handling * chore: bump @librechat/agents to version 2.1.7 for parallel tool calling for Google models
152 lines
4.6 KiB
TypeScript
152 lines
4.6 KiB
TypeScript
import { memo, useMemo, useState } from 'react';
|
|
import { useRecoilValue, useRecoilState } from 'recoil';
|
|
import { ContentTypes } from 'librechat-data-provider';
|
|
import type { TMessageContentParts, TAttachment, Agents } from 'librechat-data-provider';
|
|
import { ThinkingButton } from '~/components/Artifacts/Thinking';
|
|
import EditTextPart from './Parts/EditTextPart';
|
|
import useLocalize from '~/hooks/useLocalize';
|
|
import { mapAttachments } from '~/utils/map';
|
|
import { MessageContext } from '~/Providers';
|
|
import store from '~/store';
|
|
import Part from './Part';
|
|
|
|
type ContentPartsProps = {
|
|
content: Array<TMessageContentParts | undefined> | undefined;
|
|
messageId: string;
|
|
conversationId?: string | null;
|
|
attachments?: TAttachment[];
|
|
isCreatedByUser: boolean;
|
|
isLast: boolean;
|
|
isSubmitting: boolean;
|
|
edit?: boolean;
|
|
enterEdit?: (cancel?: boolean) => void | null | undefined;
|
|
siblingIdx?: number;
|
|
setSiblingIdx?:
|
|
| ((value: number) => void | React.Dispatch<React.SetStateAction<number>>)
|
|
| null
|
|
| undefined;
|
|
};
|
|
|
|
const ContentParts = memo(
|
|
({
|
|
content,
|
|
messageId,
|
|
conversationId,
|
|
attachments,
|
|
isCreatedByUser,
|
|
isLast,
|
|
isSubmitting,
|
|
edit,
|
|
enterEdit,
|
|
siblingIdx,
|
|
setSiblingIdx,
|
|
}: ContentPartsProps) => {
|
|
const localize = useLocalize();
|
|
const [showThinking, setShowThinking] = useRecoilState<boolean>(store.showThinking);
|
|
const [isExpanded, setIsExpanded] = useState(showThinking);
|
|
const messageAttachmentsMap = useRecoilValue(store.messageAttachmentsMap);
|
|
const attachmentMap = useMemo(
|
|
() => mapAttachments(attachments ?? messageAttachmentsMap[messageId] ?? []),
|
|
[attachments, messageAttachmentsMap, messageId],
|
|
);
|
|
|
|
const hasReasoningParts = useMemo(() => {
|
|
const hasThinkPart = content?.some((part) => part?.type === ContentTypes.THINK) ?? false;
|
|
const allThinkPartsHaveContent =
|
|
content?.every((part) => {
|
|
if (part?.type !== ContentTypes.THINK) {
|
|
return true;
|
|
}
|
|
|
|
if (typeof part.think === 'string') {
|
|
const cleanedContent = part.think.replace(/<\/?think>/g, '').trim();
|
|
return cleanedContent.length > 0;
|
|
}
|
|
|
|
return false;
|
|
}) ?? false;
|
|
|
|
return hasThinkPart && allThinkPartsHaveContent;
|
|
}, [content]);
|
|
if (!content) {
|
|
return null;
|
|
}
|
|
if (edit === true && enterEdit && setSiblingIdx) {
|
|
return (
|
|
<>
|
|
{content.map((part, idx) => {
|
|
if (part?.type !== ContentTypes.TEXT || typeof part.text !== 'string') {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<EditTextPart
|
|
index={idx}
|
|
text={part.text}
|
|
messageId={messageId}
|
|
isSubmitting={isSubmitting}
|
|
enterEdit={enterEdit}
|
|
siblingIdx={siblingIdx ?? null}
|
|
setSiblingIdx={setSiblingIdx}
|
|
key={`edit-${messageId}-${idx}`}
|
|
/>
|
|
);
|
|
})}
|
|
</>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{hasReasoningParts && (
|
|
<div className="mb-5">
|
|
<ThinkingButton
|
|
isExpanded={isExpanded}
|
|
onClick={() =>
|
|
setIsExpanded((prev) => {
|
|
const val = !prev;
|
|
setShowThinking(val);
|
|
return val;
|
|
})
|
|
}
|
|
label={
|
|
isSubmitting && isLast ? localize('com_ui_thinking') : localize('com_ui_thoughts')
|
|
}
|
|
/>
|
|
</div>
|
|
)}
|
|
{content
|
|
.filter((part) => part)
|
|
.map((part, idx) => {
|
|
const toolCallId =
|
|
(part?.[ContentTypes.TOOL_CALL] as Agents.ToolCall | undefined)?.id ?? '';
|
|
const attachments = attachmentMap[toolCallId];
|
|
|
|
return (
|
|
<MessageContext.Provider
|
|
key={`provider-${messageId}-${idx}`}
|
|
value={{
|
|
messageId,
|
|
conversationId,
|
|
partIndex: idx,
|
|
isExpanded,
|
|
nextType: content[idx + 1]?.type,
|
|
}}
|
|
>
|
|
<Part
|
|
part={part}
|
|
attachments={attachments}
|
|
isSubmitting={isSubmitting}
|
|
key={`part-${messageId}-${idx}`}
|
|
isCreatedByUser={isCreatedByUser}
|
|
showCursor={idx === content.length - 1 && isLast}
|
|
/>
|
|
</MessageContext.Provider>
|
|
);
|
|
})}
|
|
</>
|
|
);
|
|
},
|
|
);
|
|
|
|
export default ContentParts;
|