|
@@ -1,7 +1,8 @@
|
|
|
/**
|
|
/**
|
|
|
- * Parses thinking content from a message that may contain <think> tags
|
|
|
|
|
|
|
+ * Parses thinking content from a message that may contain <think> tags or [THINK] tags
|
|
|
* Returns an object with thinking content and cleaned message content
|
|
* Returns an object with thinking content and cleaned message content
|
|
|
- * Handles both complete <think>...</think> blocks and incomplete <think> blocks (streaming)
|
|
|
|
|
|
|
+ * Handles both complete blocks and incomplete blocks (streaming)
|
|
|
|
|
+ * Supports formats: <think>...</think> and [THINK]...[/THINK]
|
|
|
* @param content - The message content to parse
|
|
* @param content - The message content to parse
|
|
|
* @returns An object containing the extracted thinking content and the cleaned message content
|
|
* @returns An object containing the extracted thinking content and the cleaned message content
|
|
|
*/
|
|
*/
|
|
@@ -9,12 +10,11 @@ export function parseThinkingContent(content: string): {
|
|
|
thinking: string | null;
|
|
thinking: string | null;
|
|
|
cleanContent: string;
|
|
cleanContent: string;
|
|
|
} {
|
|
} {
|
|
|
- const incompleteMatch = content.includes('<think>') && !content.includes('</think>');
|
|
|
|
|
|
|
+ const incompleteThinkMatch = content.includes('<think>') && !content.includes('</think>');
|
|
|
|
|
+ const incompleteThinkBracketMatch = content.includes('[THINK]') && !content.includes('[/THINK]');
|
|
|
|
|
|
|
|
- if (incompleteMatch) {
|
|
|
|
|
- // Remove the entire <think>... part from clean content
|
|
|
|
|
|
|
+ if (incompleteThinkMatch) {
|
|
|
const cleanContent = content.split('</think>')?.[1]?.trim();
|
|
const cleanContent = content.split('</think>')?.[1]?.trim();
|
|
|
- // Extract everything after <think> as thinking content
|
|
|
|
|
const thinkingContent = content.split('<think>')?.[1]?.trim();
|
|
const thinkingContent = content.split('<think>')?.[1]?.trim();
|
|
|
|
|
|
|
|
return {
|
|
return {
|
|
@@ -23,12 +23,40 @@ export function parseThinkingContent(content: string): {
|
|
|
};
|
|
};
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- const completeMatch = content.includes('</think>');
|
|
|
|
|
|
|
+ if (incompleteThinkBracketMatch) {
|
|
|
|
|
+ const cleanContent = content.split('[/THINK]')?.[1]?.trim();
|
|
|
|
|
+ const thinkingContent = content.split('[THINK]')?.[1]?.trim();
|
|
|
|
|
|
|
|
- if (completeMatch) {
|
|
|
|
|
return {
|
|
return {
|
|
|
- thinking: content.split('</think>')?.[0]?.trim(),
|
|
|
|
|
- cleanContent: content.split('</think>')?.[1]?.trim()
|
|
|
|
|
|
|
+ cleanContent,
|
|
|
|
|
+ thinking: thinkingContent
|
|
|
|
|
+ };
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ const completeThinkMatch = content.match(/<think>([\s\S]*?)<\/think>/);
|
|
|
|
|
+ const completeThinkBracketMatch = content.match(/\[THINK\]([\s\S]*?)\[\/THINK\]/);
|
|
|
|
|
+
|
|
|
|
|
+ if (completeThinkMatch) {
|
|
|
|
|
+ const thinkingContent = completeThinkMatch[1]?.trim() ?? '';
|
|
|
|
|
+ const cleanContent = `${content.slice(0, completeThinkMatch.index ?? 0)}${content.slice(
|
|
|
|
|
+ (completeThinkMatch.index ?? 0) + completeThinkMatch[0].length
|
|
|
|
|
+ )}`.trim();
|
|
|
|
|
+
|
|
|
|
|
+ return {
|
|
|
|
|
+ thinking: thinkingContent,
|
|
|
|
|
+ cleanContent
|
|
|
|
|
+ };
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ if (completeThinkBracketMatch) {
|
|
|
|
|
+ const thinkingContent = completeThinkBracketMatch[1]?.trim() ?? '';
|
|
|
|
|
+ const cleanContent = `${content.slice(0, completeThinkBracketMatch.index ?? 0)}${content.slice(
|
|
|
|
|
+ (completeThinkBracketMatch.index ?? 0) + completeThinkBracketMatch[0].length
|
|
|
|
|
+ )}`.trim();
|
|
|
|
|
+
|
|
|
|
|
+ return {
|
|
|
|
|
+ thinking: thinkingContent,
|
|
|
|
|
+ cleanContent
|
|
|
};
|
|
};
|
|
|
}
|
|
}
|
|
|
|
|
|
|
@@ -39,26 +67,33 @@ export function parseThinkingContent(content: string): {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
/**
|
|
|
- * Checks if content contains an opening <think> tag (for streaming)
|
|
|
|
|
|
|
+ * Checks if content contains an opening thinking tag (for streaming)
|
|
|
|
|
+ * Supports both <think> and [THINK] formats
|
|
|
* @param content - The message content to check
|
|
* @param content - The message content to check
|
|
|
- * @returns True if the content contains an opening <think> tag
|
|
|
|
|
|
|
+ * @returns True if the content contains an opening thinking tag
|
|
|
*/
|
|
*/
|
|
|
export function hasThinkingStart(content: string): boolean {
|
|
export function hasThinkingStart(content: string): boolean {
|
|
|
- return content.includes('<think>') || content.includes('<|channel|>analysis');
|
|
|
|
|
|
|
+ return (
|
|
|
|
|
+ content.includes('<think>') ||
|
|
|
|
|
+ content.includes('[THINK]') ||
|
|
|
|
|
+ content.includes('<|channel|>analysis')
|
|
|
|
|
+ );
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
/**
|
|
|
- * Checks if content contains a closing </think> tag (for streaming)
|
|
|
|
|
|
|
+ * Checks if content contains a closing thinking tag (for streaming)
|
|
|
|
|
+ * Supports both </think> and [/THINK] formats
|
|
|
* @param content - The message content to check
|
|
* @param content - The message content to check
|
|
|
- * @returns True if the content contains a closing </think> tag
|
|
|
|
|
|
|
+ * @returns True if the content contains a closing thinking tag
|
|
|
*/
|
|
*/
|
|
|
export function hasThinkingEnd(content: string): boolean {
|
|
export function hasThinkingEnd(content: string): boolean {
|
|
|
- return content.includes('</think>');
|
|
|
|
|
|
|
+ return content.includes('</think>') || content.includes('[/THINK]');
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
/**
|
|
|
* Extracts partial thinking content during streaming
|
|
* Extracts partial thinking content during streaming
|
|
|
- * Used when we have <think> but not yet </think>
|
|
|
|
|
|
|
+ * Supports both <think> and [THINK] formats
|
|
|
|
|
+ * Used when we have opening tag but not yet closing tag
|
|
|
* @param content - The message content to extract partial thinking from
|
|
* @param content - The message content to extract partial thinking from
|
|
|
* @returns An object containing the extracted partial thinking content and the remaining content
|
|
* @returns An object containing the extracted partial thinking content and the remaining content
|
|
|
*/
|
|
*/
|
|
@@ -66,23 +101,41 @@ export function extractPartialThinking(content: string): {
|
|
|
thinking: string | null;
|
|
thinking: string | null;
|
|
|
remainingContent: string;
|
|
remainingContent: string;
|
|
|
} {
|
|
} {
|
|
|
- const startIndex = content.indexOf('<think>');
|
|
|
|
|
- if (startIndex === -1) {
|
|
|
|
|
- return { thinking: null, remainingContent: content };
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ const thinkStartIndex = content.indexOf('<think>');
|
|
|
|
|
+ const thinkEndIndex = content.indexOf('</think>');
|
|
|
|
|
|
|
|
- const endIndex = content.indexOf('</think>');
|
|
|
|
|
- if (endIndex === -1) {
|
|
|
|
|
- // Still streaming thinking content
|
|
|
|
|
- const thinkingStart = startIndex + '<think>'.length;
|
|
|
|
|
- return {
|
|
|
|
|
- thinking: content.substring(thinkingStart),
|
|
|
|
|
- remainingContent: content.substring(0, startIndex)
|
|
|
|
|
- };
|
|
|
|
|
|
|
+ const bracketStartIndex = content.indexOf('[THINK]');
|
|
|
|
|
+ const bracketEndIndex = content.indexOf('[/THINK]');
|
|
|
|
|
+
|
|
|
|
|
+ const useThinkFormat =
|
|
|
|
|
+ thinkStartIndex !== -1 && (bracketStartIndex === -1 || thinkStartIndex < bracketStartIndex);
|
|
|
|
|
+ const useBracketFormat =
|
|
|
|
|
+ bracketStartIndex !== -1 && (thinkStartIndex === -1 || bracketStartIndex < thinkStartIndex);
|
|
|
|
|
+
|
|
|
|
|
+ if (useThinkFormat) {
|
|
|
|
|
+ if (thinkEndIndex === -1) {
|
|
|
|
|
+ const thinkingStart = thinkStartIndex + '<think>'.length;
|
|
|
|
|
+
|
|
|
|
|
+ return {
|
|
|
|
|
+ thinking: content.substring(thinkingStart),
|
|
|
|
|
+ remainingContent: content.substring(0, thinkStartIndex)
|
|
|
|
|
+ };
|
|
|
|
|
+ }
|
|
|
|
|
+ } else if (useBracketFormat) {
|
|
|
|
|
+ if (bracketEndIndex === -1) {
|
|
|
|
|
+ const thinkingStart = bracketStartIndex + '[THINK]'.length;
|
|
|
|
|
+
|
|
|
|
|
+ return {
|
|
|
|
|
+ thinking: content.substring(thinkingStart),
|
|
|
|
|
+ remainingContent: content.substring(0, bracketStartIndex)
|
|
|
|
|
+ };
|
|
|
|
|
+ }
|
|
|
|
|
+ } else {
|
|
|
|
|
+ return { thinking: null, remainingContent: content };
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- // Complete thinking block found
|
|
|
|
|
const parsed = parseThinkingContent(content);
|
|
const parsed = parseThinkingContent(content);
|
|
|
|
|
+
|
|
|
return {
|
|
return {
|
|
|
thinking: parsed.thinking,
|
|
thinking: parsed.thinking,
|
|
|
remainingContent: parsed.cleanContent
|
|
remainingContent: parsed.cleanContent
|