mirror of
				https://github.com/labring/FastGPT.git
				synced 2025-10-20 18:54:09 +00:00 
			
		
		
		
	 6b7b03c245
			
		
	
	6b7b03c245
	
	
	
		
			
			* fix: plugin update * feat: get current time plugin * fix: ts * perf: select app ux * fix: ts * perf: max w * move code * perf: inform tip * fix: inform * doc * fix: tool handle * perf: tmp file store * doc * fix: message file selector * feat: doc * perf: switch trigger * doc * fix: openapi import * rount the number * parse openapi schema * fix empty line after variables (#64) * doc image * image size * doc * doc * catch error --------- Co-authored-by: heheer <71265218+newfish-cmyk@users.noreply.github.com>
		
			
				
	
	
		
			177 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { replaceVariable } from '@fastgpt/global/common/string/tools';
 | ||
| import { getAIApi } from '../config';
 | ||
| import { ChatItemType } from '@fastgpt/global/core/chat/type';
 | ||
| import { countGptMessagesTokens } from '@fastgpt/global/common/string/tiktoken';
 | ||
| import { ChatCompletionMessageParam } from '@fastgpt/global/core/ai/type';
 | ||
| 
 | ||
| /* 
 | ||
|     query extension - 问题扩展
 | ||
|     可以根据上下文,消除指代性问题以及扩展问题,利于检索。
 | ||
| */
 | ||
| 
 | ||
| const defaultPrompt = `作为一个向量检索助手,你的任务是结合历史记录,从不同角度,为“原问题”生成个不同版本的“检索词”,从而提高向量检索的语义丰富度,提高向量检索的精度。生成的问题要求指向对象清晰明确,并与“原问题语言相同”。例如:
 | ||
| 历史记录: 
 | ||
| """
 | ||
| """
 | ||
| 原问题: 介绍下剧情。
 | ||
| 检索词: ["介绍下故事的背景和主要人物。","故事的主题是什么?","剧情是是如何发展的?"]
 | ||
| ----------------
 | ||
| 历史记录: 
 | ||
| """
 | ||
| Q: 对话背景。
 | ||
| A: 当前对话是关于 Nginx 的介绍和使用等。
 | ||
| """
 | ||
| 原问题: 怎么下载
 | ||
| 检索词: ["Nginx 如何下载?","下载 Nginx 需要什么条件?","有哪些渠道可以下载 Nginx?"]
 | ||
| ----------------
 | ||
| 历史记录: 
 | ||
| """
 | ||
| Q: 对话背景。
 | ||
| A: 当前对话是关于 Nginx 的介绍和使用等。
 | ||
| Q: 报错 "no connection"
 | ||
| A: 报错"no connection"可能是因为……
 | ||
| """
 | ||
| 原问题: 怎么解决
 | ||
| 检索词: ["Nginx报错"no connection"如何解决?","造成'no connection'报错的原因。","Nginx提示'no connection',要怎么办?"]
 | ||
| ----------------
 | ||
| 历史记录: 
 | ||
| """
 | ||
| Q: 护产假多少天?
 | ||
| A: 护产假的天数根据员工所在的城市而定。请提供您所在的城市,以便我回答您的问题。
 | ||
| """
 | ||
| 原问题: 沈阳
 | ||
| 检索词: ["沈阳的护产假多少天?"]
 | ||
| ----------------
 | ||
| 历史记录: 
 | ||
| """
 | ||
| Q: 作者是谁?
 | ||
| A: FastGPT 的作者是 labring。
 | ||
| """
 | ||
| 原问题: Tell me about him
 | ||
| 检索词: ["Introduce labring, the author of FastGPT." ," Background information on author labring." "," Why does labring do FastGPT?"]
 | ||
| ----------------
 | ||
| 历史记录:
 | ||
| """
 | ||
| Q: 对话背景。
 | ||
| A: 关于 FatGPT 的介绍和使用等问题。
 | ||
| """
 | ||
| 原问题: 你好。
 | ||
| 检索词: ["你好"]
 | ||
| ----------------
 | ||
| 历史记录:
 | ||
| """
 | ||
| Q: FastGPT 如何收费?
 | ||
| A: FastGPT 收费可以参考……
 | ||
| """
 | ||
| 原问题: 你知道 laf 么?
 | ||
| 检索词: ["laf是什么?","如何使用laf?","laf的介绍。"]
 | ||
| ----------------
 | ||
| 历史记录:
 | ||
| """
 | ||
| Q: FastGPT 的优势
 | ||
| A: 1. 开源
 | ||
|    2. 简便
 | ||
|    3. 扩展性强
 | ||
| """
 | ||
| 原问题: 介绍下第2点。
 | ||
| 检索词: ["介绍下 FastGPT 简便的优势", "FastGPT 为什么使用起来简便?","FastGPT的有哪些简便的功能?"]。
 | ||
| ----------------
 | ||
| 历史记录:
 | ||
| """
 | ||
| Q: 什么是 FastGPT?
 | ||
| A: FastGPT 是一个 RAG 平台。
 | ||
| Q: 什么是 Laf?
 | ||
| A: Laf 是一个云函数开发平台。
 | ||
| """
 | ||
| 原问题: 它们有什么关系?
 | ||
| 检索词: ["FastGPT和Laf有什么关系?","FastGPT的RAG是用Laf实现的么?"]
 | ||
| ----------------
 | ||
| 历史记录:
 | ||
| """
 | ||
| {{histories}}
 | ||
| """
 | ||
| 原问题: {{query}}
 | ||
| 检索词: `;
 | ||
| 
 | ||
| export const queryExtension = async ({
 | ||
|   chatBg,
 | ||
|   query,
 | ||
|   histories = [],
 | ||
|   model
 | ||
| }: {
 | ||
|   chatBg?: string;
 | ||
|   query: string;
 | ||
|   histories: ChatItemType[];
 | ||
|   model: string;
 | ||
| }): Promise<{
 | ||
|   rawQuery: string;
 | ||
|   extensionQueries: string[];
 | ||
|   model: string;
 | ||
|   tokens: number;
 | ||
| }> => {
 | ||
|   const systemFewShot = chatBg
 | ||
|     ? `Q: 对话背景。
 | ||
| A: ${chatBg}
 | ||
| `
 | ||
|     : '';
 | ||
|   const historyFewShot = histories
 | ||
|     .map((item) => {
 | ||
|       const role = item.obj === 'Human' ? 'Q' : 'A';
 | ||
|       return `${role}: ${item.value}`;
 | ||
|     })
 | ||
|     .join('\n');
 | ||
|   const concatFewShot = `${systemFewShot}${historyFewShot}`.trim();
 | ||
| 
 | ||
|   const ai = getAIApi({
 | ||
|     timeout: 480000
 | ||
|   });
 | ||
| 
 | ||
|   const messages = [
 | ||
|     {
 | ||
|       role: 'user',
 | ||
|       content: replaceVariable(defaultPrompt, {
 | ||
|         query: `${query}`,
 | ||
|         histories: concatFewShot
 | ||
|       })
 | ||
|     }
 | ||
|   ] as ChatCompletionMessageParam[];
 | ||
|   const result = await ai.chat.completions.create({
 | ||
|     model: model,
 | ||
|     temperature: 0.01,
 | ||
|     // @ts-ignore
 | ||
|     messages,
 | ||
|     stream: false
 | ||
|   });
 | ||
| 
 | ||
|   let answer = result.choices?.[0]?.message?.content || '';
 | ||
|   if (!answer) {
 | ||
|     return {
 | ||
|       rawQuery: query,
 | ||
|       extensionQueries: [],
 | ||
|       model,
 | ||
|       tokens: 0
 | ||
|     };
 | ||
|   }
 | ||
| 
 | ||
|   answer = answer.replace(/\\"/g, '"');
 | ||
| 
 | ||
|   try {
 | ||
|     const queries = JSON.parse(answer) as string[];
 | ||
| 
 | ||
|     return {
 | ||
|       rawQuery: query,
 | ||
|       extensionQueries: Array.isArray(queries) ? queries : [],
 | ||
|       model,
 | ||
|       tokens: countGptMessagesTokens(messages)
 | ||
|     };
 | ||
|   } catch (error) {
 | ||
|     console.log(error);
 | ||
|     return {
 | ||
|       rawQuery: query,
 | ||
|       extensionQueries: [],
 | ||
|       model,
 | ||
|       tokens: 0
 | ||
|     };
 | ||
|   }
 | ||
| };
 |