From 9eefbcb6f2790397fb7df9c9925d993c0cea15c6 Mon Sep 17 00:00:00 2001 From: wanjiaju <邮箱> Date: Sat, 8 Feb 2025 10:27:19 +0800 Subject: [PATCH 01/13] =?UTF-8?q?=E9=98=BF=E9=87=8C=E4=BA=91=E7=99=BE?= =?UTF-8?q?=E7=82=BC=E9=80=82=E9=85=8D?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 新增阿里云请求器配置、阿里云模型配置、阿里云令牌配置 新增硅基模型配置 --- pkg/provider/modelmgr/modelmgr.py | 2 +- .../modelmgr/requesters/aliyunchatcmpl.py | 21 ++++++++++++++++ templates/metadata/llm-models.json | 24 +++++++++++++++++++ templates/provider.json | 2 +- 4 files changed, 47 insertions(+), 2 deletions(-) create mode 100644 pkg/provider/modelmgr/requesters/aliyunchatcmpl.py diff --git a/pkg/provider/modelmgr/modelmgr.py b/pkg/provider/modelmgr/modelmgr.py index 33a65ff3..6748291c 100644 --- a/pkg/provider/modelmgr/modelmgr.py +++ b/pkg/provider/modelmgr/modelmgr.py @@ -6,7 +6,7 @@ from . import entities, requester from ...core import app from . import token -from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl +from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl, aliyunchatcmpl FETCH_MODEL_LIST_URL = "https://api.qchatgpt.rockchin.top/api/v2/fetch/model_list" diff --git a/pkg/provider/modelmgr/requesters/aliyunchatcmpl.py b/pkg/provider/modelmgr/requesters/aliyunchatcmpl.py new file mode 100644 index 00000000..4f4900d7 --- /dev/null +++ b/pkg/provider/modelmgr/requesters/aliyunchatcmpl.py @@ -0,0 +1,21 @@ +from __future__ import annotations + +import openai + +from . import chatcmpl +from .. import requester +from ....core import app + + +@requester.requester_class("aliyun-chat-completions") +class AliyunChatCompletions(chatcmpl.OpenAIChatCompletions): + """Aliyun ChatCompletion API 请求器""" + + client: openai.AsyncClient + + requester_cfg: dict + + def __init__(self, ap: app.Application): + self.ap = ap + + self.requester_cfg = self.ap.provider_cfg.data['requester']['aliyun-chat-completions'] diff --git a/templates/metadata/llm-models.json b/templates/metadata/llm-models.json index b5c29cf3..110c18da 100644 --- a/templates/metadata/llm-models.json +++ b/templates/metadata/llm-models.json @@ -211,6 +211,30 @@ "requester": "zhipuai-chat-completions", "token_mgr": "zhipuai", "vision_supported": true + }, + { + "name": "siliconflow-r1", + "model_name": "deepseek-ai/DeepSeek-R1", + "requester": "siliconflow-chat-completions", + "token_mgr": "siliconflow" + }, + { + "name": "siliconflow-v3", + "model_name": "deepseek-ai/DeepSeek-V3", + "requester": "siliconflow-chat-completions", + "token_mgr": "siliconflow" + }, + { + "name": "aliyun-r1", + "model_name": "deepseek-r1", + "requester": "aliyun-chat-completions", + "token_mgr": "aliyun" + }, + { + "name": "aliyun-v3", + "model_name": "deepseek-v3", + "requester": "aliyun-chat-completions", + "token_mgr": "aliyun" } ] } \ No newline at end of file diff --git a/templates/provider.json b/templates/provider.json index 5ab5cd64..0d6e6ea0 100644 --- a/templates/provider.json +++ b/templates/provider.json @@ -84,7 +84,7 @@ "model": "gpt-4o", "prompt-mode": "normal", "prompt": { - "default": "" + "default": "" }, "runner": "local-agent", "dify-service-api": { From 8e9f43885a46c44dcff942acf4b2e7b923540990 Mon Sep 17 00:00:00 2001 From: wanjiaju <邮箱> Date: Sat, 8 Feb 2025 10:30:19 +0800 Subject: [PATCH 02/13] =?UTF-8?q?=E9=98=BF=E9=87=8C=E4=BA=91=E7=99=BE?= =?UTF-8?q?=E7=82=BC=E9=80=82=E9=85=8D?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 新增阿里云请求器配置、阿里云模型配置、阿里云令牌配置 新增硅基模型配置 --- templates/provider.json | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/templates/provider.json b/templates/provider.json index 0d6e6ea0..c22ba42c 100644 --- a/templates/provider.json +++ b/templates/provider.json @@ -25,6 +25,9 @@ ], "siliconflow": [ "xxxxxxx" + ], + "aliyun": [ + "sk-aliyun_token" ] }, "requester": { @@ -79,6 +82,11 @@ "base-url": "https://api.siliconflow.cn/v1", "args": {}, "timeout": 120 + }, + "aliyun-chat-completions": { + "args": {}, + "base-url": "https://dashscope.aliyuncs.com/compatible-mode/v1", + "timeout": 120 } }, "model": "gpt-4o", From 52eb37d13dbca0e175aa23bde56080ff8ab9b7ea Mon Sep 17 00:00:00 2001 From: Civic_Crab <1308770968@qq.com> Date: Sun, 9 Feb 2025 06:32:49 +0800 Subject: [PATCH 03/13] =?UTF-8?q?=E6=94=AF=E6=8C=81=E9=98=BF=E9=87=8C?= =?UTF-8?q?=E4=BA=91=E7=99=BE=E7=82=BC=E7=9A=84=E9=80=9A=E7=94=A8=E6=A8=A1?= =?UTF-8?q?=E5=9E=8B=E5=92=8C=E8=87=AA=E5=AE=9A=E4=B9=89=E5=A4=A7=E6=A8=A1?= =?UTF-8?q?=E5=9E=8B=E5=BA=94=E7=94=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pkg/provider/modelmgr/modelmgr.py | 2 +- .../modelmgr/requesters/dashscopecmpl.py | 167 ++++++++++++++++++ .../modelmgr/requesters/qwenchatcmpl.py | 21 +++ templates/metadata/llm-models.json | 10 ++ templates/provider.json | 16 ++ 5 files changed, 215 insertions(+), 1 deletion(-) create mode 100644 pkg/provider/modelmgr/requesters/dashscopecmpl.py create mode 100644 pkg/provider/modelmgr/requesters/qwenchatcmpl.py diff --git a/pkg/provider/modelmgr/modelmgr.py b/pkg/provider/modelmgr/modelmgr.py index 33a65ff3..737d3476 100644 --- a/pkg/provider/modelmgr/modelmgr.py +++ b/pkg/provider/modelmgr/modelmgr.py @@ -6,7 +6,7 @@ from . import entities, requester from ...core import app from . import token -from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl +from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl, dashscopecmpl, qwenchatcmpl FETCH_MODEL_LIST_URL = "https://api.qchatgpt.rockchin.top/api/v2/fetch/model_list" diff --git a/pkg/provider/modelmgr/requesters/dashscopecmpl.py b/pkg/provider/modelmgr/requesters/dashscopecmpl.py new file mode 100644 index 00000000..d2f2b2d2 --- /dev/null +++ b/pkg/provider/modelmgr/requesters/dashscopecmpl.py @@ -0,0 +1,167 @@ +from __future__ import annotations + +import re +import asyncio +import typing +import dashscope + +from .. import entities, errors, requester +from ....core import entities as core_entities, app +from ... import entities as llm_entities +from ...tools import entities as tools_entities + +#阿里云百炼平台的自定义应用支持资料引用,此函数可以将引用标签替换为参考资料 +def replace_references(text, references_dict): + # 修正正则表达式,匹配 [index_id] 形式的字符串 + pattern = re.compile(r'\[(.*?)\]') + + def replacement(match): + ref_key = match.group(1) # 获取引用编号 + if ref_key in references_dict: + return f"(参考资料来自:{references_dict[ref_key]})" + else: + return match.group(0) # 如果没有对应的参考资料,保留原样 + + # 使用 re.sub() 进行替换 + return pattern.sub(replacement, text) + + +@requester.requester_class("dashscope-chat-applications") +class DashscopeChatApplication(requester.LLMAPIRequester): + """Dashscope ChatApplications API 请求器""" + + requester_cfg: dict + + def __init__(self, ap: app.Application): + self.requester_cfg = ap.provider_cfg.data['requester']['dashscope-chat-applications'] + self.ap = ap + + async def initialize(self): + dashscope.api_key = self.ap.provider_cfg.data['keys']['dashscope'][0] + + async def _req(self, args: dict): + + #print("args:", args) + + #局部变量 + chunk = None + pending_content = "" + output = { + "role": "assistant", + "content": "", + "tool_calls": [], + "tool_call_id": None # Dashscope暂时不支持工具调用 + } #由于Dashscope的content的键值是text,所以需要定义一个新格式的字典适配llm_entities.Message + + references_dict = {} # 用于存储引用编号和对应的参考资料 + + #调用API + response = dashscope.Application.call( + api_key=dashscope.api_key, + app_id=args["model"], + prompt=args["messages"], + stream=True, # 设置流式输出 + tools=args.get("tools", None), + incremental_output = True, + ) + + #处理API返回的流式输出 + for chunk in response: + #print(chunk) + if not chunk: + continue + + #获取流式传输的output + stream_output = chunk.get("output", {}) + if stream_output.get("text") is not None: + pending_content += stream_output.get("text") + + + #获取模型传出的参考资料列表 + references_dict_list = stream_output.get("doc_references", []) + + #从模型传出的参考资料信息中提取用于替换的字典 + if references_dict_list is not None: + for doc in references_dict_list: + if doc.get("index_id") is not None: + references_dict[doc.get("index_id")] = doc.get("doc_name") + + #将参考资料替换到文本中 + pending_content = replace_references(pending_content, references_dict) + + #将流式传输的内容整合到output中 + output["content"] = pending_content + + return output if chunk else None + + async def _make_msg( + self, + chat_completion: dict, + ) -> llm_entities.Message: + chatcmpl_message = chat_completion + + # 确保 role 字段存在且不为 None + if 'role' not in chatcmpl_message or chatcmpl_message['role'] is None: + chatcmpl_message['role'] = 'assistant' + + message = llm_entities.Message(**chatcmpl_message) + #print("message:", message) + return message + + async def _closure( + self, + query: core_entities.Query, + req_messages: list[dict], + use_model: entities.LLMModelInfo, + use_funcs: list[tools_entities.LLMFunction] = None, + ) -> llm_entities.Message: + + args = self.requester_cfg['args'].copy() + args["model"] = use_model.name if use_model.model_name is None else use_model.model_name + + # 设置此次请求中的messages + messages = req_messages.copy() + + # 检查vision + for msg in messages: + if 'content' in msg and isinstance(msg["content"], list): + for me in msg["content"]: + if me["type"] == "image_base64": + me["image_url"] = { + "url": me["image_base64"] + } + me["type"] = "image_url" + del me["image_base64"] + + args["messages"] = messages + + # 发送请求 + resp = await self._req(args) + + # 处理请求结果 + message = await self._make_msg(resp) + + return message + + async def call( + self, + query: core_entities.Query, + model: entities.LLMModelInfo, + messages: typing.List[llm_entities.Message], + funcs: typing.List[tools_entities.LLMFunction] = None, + ) -> llm_entities.Message: + req_messages = [] # req_messages 仅用于类内,外部同步由 query.messages 进行 + for m in messages: + msg_dict = m.dict(exclude_none=True) + content = msg_dict.get("content") + if isinstance(content, list): + # 检查 content 列表中是否每个部分都是文本 + if all(isinstance(part, dict) and part.get("type") == "text" for part in content): + # 将所有文本部分合并为一个字符串 + msg_dict["content"] = "\n".join(part["text"] for part in content) + req_messages.append(msg_dict) + + try: + return await self._closure(query=query, req_messages=req_messages, use_model=model, use_funcs=funcs) + except asyncio.TimeoutError: + raise errors.RequesterError('请求超时') diff --git a/pkg/provider/modelmgr/requesters/qwenchatcmpl.py b/pkg/provider/modelmgr/requesters/qwenchatcmpl.py new file mode 100644 index 00000000..8dfbae8e --- /dev/null +++ b/pkg/provider/modelmgr/requesters/qwenchatcmpl.py @@ -0,0 +1,21 @@ +from __future__ import annotations + +import openai + +from . import chatcmpl +from .. import requester +from ....core import app + + +@requester.requester_class("siliconflow-chat-completions") +class QwenChatCompletions(chatcmpl.OpenAIChatCompletions): + """Qwen ChatCompletion API 请求器""" + + client: openai.AsyncClient + + requester_cfg: dict + + def __init__(self, ap: app.Application): + self.ap = ap + + self.requester_cfg = self.ap.provider_cfg.data['requester']['qwen-chat-completions'] diff --git a/templates/metadata/llm-models.json b/templates/metadata/llm-models.json index b5c29cf3..0dfb2a43 100644 --- a/templates/metadata/llm-models.json +++ b/templates/metadata/llm-models.json @@ -211,6 +211,16 @@ "requester": "zhipuai-chat-completions", "token_mgr": "zhipuai", "vision_supported": true + }, + { + "name": "your-dashscope-app-id", + "requester": "dashscope-chat-applications", + "token_mgr": "dashscope", + }, + { + "name": "qwen-plus", + "requester": "qwen-chat-completions", + "token_mgr": "qwen", } ] } \ No newline at end of file diff --git a/templates/provider.json b/templates/provider.json index 5ab5cd64..9db3c020 100644 --- a/templates/provider.json +++ b/templates/provider.json @@ -25,6 +25,12 @@ ], "siliconflow": [ "xxxxxxx" + ], + "dashscope": [ + "sk-1234567890" + ], + "qwen": [ + "sk-1234567890", ] }, "requester": { @@ -40,6 +46,16 @@ }, "timeout": 120 }, + "dashscope-chat-applications": { + "args": {}, + "base-url": "https://dashscope.aliyuncs.com/api/v1", + "timeout": 120 + }, + "qwen-chat-completions": { + "base-url": "https://dashscope.aliyuncs.com/compatible-mode/v1", + "args": {}, + "timeout": 120 + }, "moonshot-chat-completions": { "base-url": "https://api.moonshot.cn/v1", "args": {}, From cbec2f6d02480694b13333ff37e7849a62c69472 Mon Sep 17 00:00:00 2001 From: Civic_Crab <1308770968@qq.com> Date: Sun, 9 Feb 2025 06:37:55 +0800 Subject: [PATCH 04/13] =?UTF-8?q?=E6=96=B0=E5=A2=9Edashscope=E4=BE=9D?= =?UTF-8?q?=E8=B5=96?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- requirements.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/requirements.txt b/requirements.txt index 27349f64..dbee8256 100644 --- a/requirements.txt +++ b/requirements.txt @@ -29,6 +29,7 @@ lark-oapi discord.py cryptography gewechat-client +dashscope # indirect taskgroup==0.0.0a4 \ No newline at end of file From 9e718a2e8ad85db83db6d961b9f97d7546401e61 Mon Sep 17 00:00:00 2001 From: Civic_Crab <1308770968@qq.com> Date: Sun, 9 Feb 2025 06:39:39 +0800 Subject: [PATCH 05/13] =?UTF-8?q?=E6=96=B0=E5=A2=9Edashscope=E4=BE=9D?= =?UTF-8?q?=E8=B5=96?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pkg/core/bootutils/deps.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/pkg/core/bootutils/deps.py b/pkg/core/bootutils/deps.py index b4a67f35..83a9b2a7 100644 --- a/pkg/core/bootutils/deps.py +++ b/pkg/core/bootutils/deps.py @@ -29,7 +29,8 @@ required_deps = { "lark_oapi": "lark-oapi", "discord": "discord.py", "cryptography": "cryptography", - "gewechat_client": "gewechat-client" + "gewechat_client": "gewechat-client", + "dashscope": "dashscope", } From 4c344e063681e60d1bd51512c5277722cadbdf55 Mon Sep 17 00:00:00 2001 From: Civic_Crab <1308770968@qq.com> Date: Tue, 11 Feb 2025 18:48:21 +0800 Subject: [PATCH 06/13] =?UTF-8?q?=E9=98=BF=E9=87=8C=E4=BA=91=E7=99=BE?= =?UTF-8?q?=E7=82=BC=E5=B9=B3=E5=8F=B0=E5=BA=94=E7=94=A8API=E6=94=AF?= =?UTF-8?q?=E6=8C=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pkg/provider/modelmgr/modelmgr.py | 2 +- .../modelmgr/requesters/dashscopecmpl.py | 167 ------------- pkg/provider/runnermgr.py | 1 + pkg/provider/runners/dashscopeapi.py | 236 ++++++++++++++++++ templates/provider.json | 33 +-- 5 files changed, 255 insertions(+), 184 deletions(-) delete mode 100644 pkg/provider/modelmgr/requesters/dashscopecmpl.py create mode 100644 pkg/provider/runners/dashscopeapi.py diff --git a/pkg/provider/modelmgr/modelmgr.py b/pkg/provider/modelmgr/modelmgr.py index 737d3476..33a65ff3 100644 --- a/pkg/provider/modelmgr/modelmgr.py +++ b/pkg/provider/modelmgr/modelmgr.py @@ -6,7 +6,7 @@ from . import entities, requester from ...core import app from . import token -from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl, dashscopecmpl, qwenchatcmpl +from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl FETCH_MODEL_LIST_URL = "https://api.qchatgpt.rockchin.top/api/v2/fetch/model_list" diff --git a/pkg/provider/modelmgr/requesters/dashscopecmpl.py b/pkg/provider/modelmgr/requesters/dashscopecmpl.py deleted file mode 100644 index d2f2b2d2..00000000 --- a/pkg/provider/modelmgr/requesters/dashscopecmpl.py +++ /dev/null @@ -1,167 +0,0 @@ -from __future__ import annotations - -import re -import asyncio -import typing -import dashscope - -from .. import entities, errors, requester -from ....core import entities as core_entities, app -from ... import entities as llm_entities -from ...tools import entities as tools_entities - -#阿里云百炼平台的自定义应用支持资料引用,此函数可以将引用标签替换为参考资料 -def replace_references(text, references_dict): - # 修正正则表达式,匹配 [index_id] 形式的字符串 - pattern = re.compile(r'\[(.*?)\]') - - def replacement(match): - ref_key = match.group(1) # 获取引用编号 - if ref_key in references_dict: - return f"(参考资料来自:{references_dict[ref_key]})" - else: - return match.group(0) # 如果没有对应的参考资料,保留原样 - - # 使用 re.sub() 进行替换 - return pattern.sub(replacement, text) - - -@requester.requester_class("dashscope-chat-applications") -class DashscopeChatApplication(requester.LLMAPIRequester): - """Dashscope ChatApplications API 请求器""" - - requester_cfg: dict - - def __init__(self, ap: app.Application): - self.requester_cfg = ap.provider_cfg.data['requester']['dashscope-chat-applications'] - self.ap = ap - - async def initialize(self): - dashscope.api_key = self.ap.provider_cfg.data['keys']['dashscope'][0] - - async def _req(self, args: dict): - - #print("args:", args) - - #局部变量 - chunk = None - pending_content = "" - output = { - "role": "assistant", - "content": "", - "tool_calls": [], - "tool_call_id": None # Dashscope暂时不支持工具调用 - } #由于Dashscope的content的键值是text,所以需要定义一个新格式的字典适配llm_entities.Message - - references_dict = {} # 用于存储引用编号和对应的参考资料 - - #调用API - response = dashscope.Application.call( - api_key=dashscope.api_key, - app_id=args["model"], - prompt=args["messages"], - stream=True, # 设置流式输出 - tools=args.get("tools", None), - incremental_output = True, - ) - - #处理API返回的流式输出 - for chunk in response: - #print(chunk) - if not chunk: - continue - - #获取流式传输的output - stream_output = chunk.get("output", {}) - if stream_output.get("text") is not None: - pending_content += stream_output.get("text") - - - #获取模型传出的参考资料列表 - references_dict_list = stream_output.get("doc_references", []) - - #从模型传出的参考资料信息中提取用于替换的字典 - if references_dict_list is not None: - for doc in references_dict_list: - if doc.get("index_id") is not None: - references_dict[doc.get("index_id")] = doc.get("doc_name") - - #将参考资料替换到文本中 - pending_content = replace_references(pending_content, references_dict) - - #将流式传输的内容整合到output中 - output["content"] = pending_content - - return output if chunk else None - - async def _make_msg( - self, - chat_completion: dict, - ) -> llm_entities.Message: - chatcmpl_message = chat_completion - - # 确保 role 字段存在且不为 None - if 'role' not in chatcmpl_message or chatcmpl_message['role'] is None: - chatcmpl_message['role'] = 'assistant' - - message = llm_entities.Message(**chatcmpl_message) - #print("message:", message) - return message - - async def _closure( - self, - query: core_entities.Query, - req_messages: list[dict], - use_model: entities.LLMModelInfo, - use_funcs: list[tools_entities.LLMFunction] = None, - ) -> llm_entities.Message: - - args = self.requester_cfg['args'].copy() - args["model"] = use_model.name if use_model.model_name is None else use_model.model_name - - # 设置此次请求中的messages - messages = req_messages.copy() - - # 检查vision - for msg in messages: - if 'content' in msg and isinstance(msg["content"], list): - for me in msg["content"]: - if me["type"] == "image_base64": - me["image_url"] = { - "url": me["image_base64"] - } - me["type"] = "image_url" - del me["image_base64"] - - args["messages"] = messages - - # 发送请求 - resp = await self._req(args) - - # 处理请求结果 - message = await self._make_msg(resp) - - return message - - async def call( - self, - query: core_entities.Query, - model: entities.LLMModelInfo, - messages: typing.List[llm_entities.Message], - funcs: typing.List[tools_entities.LLMFunction] = None, - ) -> llm_entities.Message: - req_messages = [] # req_messages 仅用于类内,外部同步由 query.messages 进行 - for m in messages: - msg_dict = m.dict(exclude_none=True) - content = msg_dict.get("content") - if isinstance(content, list): - # 检查 content 列表中是否每个部分都是文本 - if all(isinstance(part, dict) and part.get("type") == "text" for part in content): - # 将所有文本部分合并为一个字符串 - msg_dict["content"] = "\n".join(part["text"] for part in content) - req_messages.append(msg_dict) - - try: - return await self._closure(query=query, req_messages=req_messages, use_model=model, use_funcs=funcs) - except asyncio.TimeoutError: - raise errors.RequesterError('请求超时') diff --git a/pkg/provider/runnermgr.py b/pkg/provider/runnermgr.py index f45abfa4..4cd38c04 100644 --- a/pkg/provider/runnermgr.py +++ b/pkg/provider/runnermgr.py @@ -5,6 +5,7 @@ from ..core import app from .runners import localagent from .runners import difysvapi +from .runners import dashscopeapi class RunnerManager: diff --git a/pkg/provider/runners/dashscopeapi.py b/pkg/provider/runners/dashscopeapi.py new file mode 100644 index 00000000..8a6aef9d --- /dev/null +++ b/pkg/provider/runners/dashscopeapi.py @@ -0,0 +1,236 @@ +from __future__ import annotations + +import typing +import json +import base64 +import re + +import dashscope + +from .. import runner +from ...core import entities as core_entities +from .. import entities as llm_entities +from ...utils import image + +class DashscopeAPIError(Exception): + """Dashscope API 请求失败""" + + def __init__(self, message: str): + self.message = message + super().__init__(self.message) + + +@runner.runner_class("dashscope-service-api") +class DashScopeAPIRunner(runner.RequestRunner): + "阿里云百炼DashsscopeAPI对话请求器" + + # 运行器内部使用的配置 + app_type: str # 应用类型 + app_id: str # 应用ID + api_key: str # API Key + references_quote: str # 引用资料提示(当展示回答来源功能开启时,这个变量会作为引用资料名前的提示,可在provider.json中配置) + biz_params: dict = {} # 工作流应用参数(仅在工作流应用中生效) + + async def initialize(self): + """初始化""" + valid_app_types = ["agent", "workflow"] + self.app_type = self.ap.provider_cfg.data["dashscope-service-api"]["app-type"] + #检查配置文件中使用的应用类型是否支持 + if (self.app_type not in valid_app_types): + raise DashscopeAPIError( + f"不支持的 Dashscope 应用类型: {self.app_type}" + ) + + #初始化Dashscope 参数配置 + self.app_id = self.ap.provider_cfg.data["dashscope-service-api"][self.app_type]["app-id"] + self.api_key = self.ap.provider_cfg.data["dashscope-service-api"][self.app_type]["api-key"] + self.references_quote = self.ap.provider_cfg.data["dashscope-service-api"][self.app_type]["references_quote"] + self.biz_params = self.ap.provider_cfg.data["dashscope-service-api"]["workflow"]["biz_params"] + + def _replace_references(self, text, references_dict): + """阿里云百炼平台的自定义应用支持资料引用,此函数可以将引用标签替换为参考资料""" + + # 匹配 [index_id] 形式的字符串 + pattern = re.compile(r'\[(.*?)\]') + + def replacement(match): + # 获取引用编号 + ref_key = match.group(1) + if ref_key in references_dict: + # 如果有对应的参考资料按照provider.json中的reference_quote返回提示,来自哪个参考资料文件 + return f"({self.references_quote} {references_dict[ref_key]})" + else: + # 如果没有对应的参考资料,保留原样 + return match.group(0) + + # 使用 re.sub() 进行替换 + return pattern.sub(replacement, text) + + async def _preprocess_user_message( + self, query: core_entities.Query + ) -> tuple[str, list[str]]: + """预处理用户消息,提取纯文本,阿里云提供的上传文件方法过于复杂,暂不支持上传文件(包括图片)""" + plain_text = "" + image_ids = [] + if isinstance(query.user_message.content, list): + for ce in query.user_message.content: + if ce.type == "text": + plain_text += ce.text + # 暂时不支持上传图片,保留代码以便后续扩展 + # elif ce.type == "image_base64": + # image_b64, image_format = await image.extract_b64_and_format(ce.image_base64) + # file_bytes = base64.b64decode(image_b64) + # file = ("img.png", file_bytes, f"image/{image_format}") + # file_upload_resp = await self.dify_client.upload_file( + # file, + # f"{query.session.launcher_type.value}_{query.session.launcher_id}", + # ) + # image_id = file_upload_resp["id"] + # image_ids.append(image_id) + elif isinstance(query.user_message.content, str): + plain_text = query.user_message.content + + return plain_text, image_ids + + + async def _agent_messages( + self, query: core_entities.Query + ) -> typing.AsyncGenerator[llm_entities.Message, None]: + """Dashscope 智能体对话请求""" + + #局部变量 + chunk = None # 流式传输的块 + pending_content = "" # 待处理的Agent输出内容 + references_dict = {} # 用于存储引用编号和对应的参考资料 + plain_text = "" # 用户输入的纯文本信息 + image_ids = [] # 用户输入的图片ID列表 (暂不支持) + + plain_text, image_ids = await self._preprocess_user_message(query) + + #发送对话请求 + response = dashscope.Application.call( + api_key=self.api_key, # 智能体应用的API Key + app_id=self.app_id, # 智能体应用的ID + prompt=plain_text, # 用户输入的文本信息 + stream=True, # 流式输出 + incremental_output=True, # 增量输出,使用流式输出需要开启增量输出 + session_id=query.session.using_conversation.uuid, # 会话ID用于,多轮对话 + # rag_options={ # 主要用于文件交互,暂不支持 + # "session_file_ids": ["FILE_ID1"], # FILE_ID1 替换为实际的临时文件ID,逗号隔开多个 + # } + ) + + for chunk in response: + if chunk.get("status_code") != 200: + raise DashscopeAPIError( + f"Dashscope API 请求失败: status_code={chunk.get('status_code')} message={chunk.get('message')} request_id={chunk.get('request_id')} " + ) + if not chunk: + continue + + #获取流式传输的output + stream_output = chunk.get("output", {}) + if stream_output.get("text") is not None: + pending_content += stream_output.get("text") + + #保存当前会话的session_id用于下次对话的语境 + query.session.using_conversation.uuid = stream_output.get("session_id") + + #获取模型传出的参考资料列表 + references_dict_list = stream_output.get("doc_references", []) + + #从模型传出的参考资料信息中提取用于替换的字典 + if references_dict_list is not None: + for doc in references_dict_list: + if doc.get("index_id") is not None: + references_dict[doc.get("index_id")] = doc.get("doc_name") + + #将参考资料替换到文本中 + pending_content = self._replace_references(pending_content, references_dict) + + yield llm_entities.Message( + role="assistant", + content=pending_content, + ) + + + async def _workflow_messages( + self, query: core_entities.Query + ) -> typing.AsyncGenerator[llm_entities.Message, None]: + """Dashscope 工作流对话请求""" + + #局部变量 + chunk = None # 流式传输的块 + pending_content = "" # 待处理的Agent输出内容 + references_dict = {} # 用于存储引用编号和对应的参考资料 + plain_text = "" # 用户输入的纯文本信息 + image_ids = [] # 用户输入的图片ID列表 (暂不支持) + + plain_text, image_ids = await self._preprocess_user_message(query) + + #发送对话请求 + response = dashscope.Application.call( + api_key=self.api_key, # 智能体应用的API Key + app_id=self.app_id, # 智能体应用的ID + prompt=plain_text, # 用户输入的文本信息 + stream=True, # 流式输出 + incremental_output=True, # 增量输出,使用流式输出需要开启增量输出 + session_id=query.session.using_conversation.uuid, # 会话ID用于,多轮对话 + biz_params=self.biz_params # 工作流应用的自定义输入参数传递 + # rag_options={ # 主要用于文件交互,暂不支持 + # "session_file_ids": ["FILE_ID1"], # FILE_ID1 替换为实际的临时文件ID,逗号隔开多个 + # } + ) + + #处理API返回的流式输出 + for chunk in response: + if chunk.get("status_code") != 200: + raise DashscopeAPIError( + f"Dashscope API 请求失败: status_code={chunk.get('status_code')} message={chunk.get('message')} request_id={chunk.get('request_id')} " + ) + if not chunk: + continue + + #获取流式传输的output + stream_output = chunk.get("output", {}) + if stream_output.get("text") is not None: + pending_content += stream_output.get("text") + + #保存当前会话的session_id用于下次对话的语境 + query.session.using_conversation.uuid = stream_output.get("session_id") + + #获取模型传出的参考资料列表 + references_dict_list = stream_output.get("doc_references", []) + + #从模型传出的参考资料信息中提取用于替换的字典 + if references_dict_list is not None: + for doc in references_dict_list: + if doc.get("index_id") is not None: + references_dict[doc.get("index_id")] = doc.get("doc_name") + + #将参考资料替换到文本中 + pending_content = self._replace_references(pending_content, references_dict) + + yield llm_entities.Message( + role="assistant", + content=pending_content, + ) + + + + async def run( + self, query: core_entities.Query + ) -> typing.AsyncGenerator[llm_entities.Message, None]: + """运行""" + if self.ap.provider_cfg.data["dashscope-service-api"]["app-type"] == "agent": + async for msg in self._agent_messages(query): + yield msg + elif self.ap.provider_cfg.data["dashscope-service-api"]["app-type"] == "workflow": + async for msg in self._workflow_messages(query): + yield msg + else: + raise DashscopeAPIError( + f"不支持的 Dashscope 应用类型: {self.ap.provider_cfg.data['dashscope-service-api']['app-type']}" + ) + + diff --git a/templates/provider.json b/templates/provider.json index 9db3c020..13044f1e 100644 --- a/templates/provider.json +++ b/templates/provider.json @@ -25,12 +25,6 @@ ], "siliconflow": [ "xxxxxxx" - ], - "dashscope": [ - "sk-1234567890" - ], - "qwen": [ - "sk-1234567890", ] }, "requester": { @@ -46,16 +40,6 @@ }, "timeout": 120 }, - "dashscope-chat-applications": { - "args": {}, - "base-url": "https://dashscope.aliyuncs.com/api/v1", - "timeout": 120 - }, - "qwen-chat-completions": { - "base-url": "https://dashscope.aliyuncs.com/compatible-mode/v1", - "args": {}, - "timeout": 120 - }, "moonshot-chat-completions": { "base-url": "https://api.moonshot.cn/v1", "args": {}, @@ -119,5 +103,22 @@ "output-key": "summary", "timeout": 120 } + }, + "dashscope-service-api": { + "agent": { + "api-key": "sk-1234567890", + "app-id": "Your_app_id", + "references_quote": "参考资料来自:" + }, + "app-type": "agent", + "workflow": { + "api-key": "sk-1234567890", + "app-id": "Your_app_id", + "references_quote": "参考资料来自:", + "biz_params": { + "city": "北京", + "date": "2023-08-10" + } + } } } \ No newline at end of file From e51950aa751b2e3ee7aebff5245f360bd24456de Mon Sep 17 00:00:00 2001 From: Civic_Crab <1308770968@qq.com> Date: Tue, 11 Feb 2025 18:50:56 +0800 Subject: [PATCH 07/13] =?UTF-8?q?=E4=BF=AE=E6=94=B9llm-model.json=EF=BC=8C?= =?UTF-8?q?=E5=8E=BB=E9=99=A4=E8=88=8D=E5=BC=83=E7=9A=84qwen=E5=8A=9F?= =?UTF-8?q?=E8=83=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- templates/metadata/llm-models.json | 5 ----- 1 file changed, 5 deletions(-) diff --git a/templates/metadata/llm-models.json b/templates/metadata/llm-models.json index 0dfb2a43..88757e66 100644 --- a/templates/metadata/llm-models.json +++ b/templates/metadata/llm-models.json @@ -216,11 +216,6 @@ "name": "your-dashscope-app-id", "requester": "dashscope-chat-applications", "token_mgr": "dashscope", - }, - { - "name": "qwen-plus", - "requester": "qwen-chat-completions", - "token_mgr": "qwen", } ] } \ No newline at end of file From 51cca31f04e43df93a930bb14a1e5a1d354d714e Mon Sep 17 00:00:00 2001 From: Civic_Crab <1308770968@qq.com> Date: Tue, 11 Feb 2025 18:52:27 +0800 Subject: [PATCH 08/13] =?UTF-8?q?=E5=8E=BB=E9=99=A4qwen=E8=AF=B7=E6=B1=82?= =?UTF-8?q?=E5=99=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../modelmgr/requesters/qwenchatcmpl.py | 21 ------------------- 1 file changed, 21 deletions(-) delete mode 100644 pkg/provider/modelmgr/requesters/qwenchatcmpl.py diff --git a/pkg/provider/modelmgr/requesters/qwenchatcmpl.py b/pkg/provider/modelmgr/requesters/qwenchatcmpl.py deleted file mode 100644 index 8dfbae8e..00000000 --- a/pkg/provider/modelmgr/requesters/qwenchatcmpl.py +++ /dev/null @@ -1,21 +0,0 @@ -from __future__ import annotations - -import openai - -from . import chatcmpl -from .. import requester -from ....core import app - - -@requester.requester_class("siliconflow-chat-completions") -class QwenChatCompletions(chatcmpl.OpenAIChatCompletions): - """Qwen ChatCompletion API 请求器""" - - client: openai.AsyncClient - - requester_cfg: dict - - def __init__(self, ap: app.Application): - self.ap = ap - - self.requester_cfg = self.ap.provider_cfg.data['requester']['qwen-chat-completions'] From e17da4e2ee2074e156080b5004cdec965724ba24 Mon Sep 17 00:00:00 2001 From: Junyan Qin Date: Wed, 12 Feb 2025 11:11:07 +0800 Subject: [PATCH 09/13] chore: remove models of MaaS from `llm-models.json` --- templates/metadata/llm-models.json | 24 ------------------------ 1 file changed, 24 deletions(-) diff --git a/templates/metadata/llm-models.json b/templates/metadata/llm-models.json index 110c18da..b5c29cf3 100644 --- a/templates/metadata/llm-models.json +++ b/templates/metadata/llm-models.json @@ -211,30 +211,6 @@ "requester": "zhipuai-chat-completions", "token_mgr": "zhipuai", "vision_supported": true - }, - { - "name": "siliconflow-r1", - "model_name": "deepseek-ai/DeepSeek-R1", - "requester": "siliconflow-chat-completions", - "token_mgr": "siliconflow" - }, - { - "name": "siliconflow-v3", - "model_name": "deepseek-ai/DeepSeek-V3", - "requester": "siliconflow-chat-completions", - "token_mgr": "siliconflow" - }, - { - "name": "aliyun-r1", - "model_name": "deepseek-r1", - "requester": "aliyun-chat-completions", - "token_mgr": "aliyun" - }, - { - "name": "aliyun-v3", - "model_name": "deepseek-v3", - "requester": "aliyun-chat-completions", - "token_mgr": "aliyun" } ] } \ No newline at end of file From 191f8866ae11a3896271229e47782a7298155eae Mon Sep 17 00:00:00 2001 From: Junyan Qin Date: Wed, 12 Feb 2025 11:25:28 +0800 Subject: [PATCH 10/13] chore(bailian): related configuration --- .../m028_aliyun_requester_config.py | 27 ++++++++++++++++++ pkg/core/stages/migrate.py | 2 +- pkg/provider/modelmgr/modelmgr.py | 2 +- .../{aliyunchatcmpl.py => bailianchatcmpl.py} | 8 +++--- templates/provider.json | 6 ++-- templates/schema/provider.json | 28 +++++++++++++++++++ 6 files changed, 64 insertions(+), 9 deletions(-) create mode 100644 pkg/core/migrations/m028_aliyun_requester_config.py rename pkg/provider/modelmgr/requesters/{aliyunchatcmpl.py => bailianchatcmpl.py} (58%) diff --git a/pkg/core/migrations/m028_aliyun_requester_config.py b/pkg/core/migrations/m028_aliyun_requester_config.py new file mode 100644 index 00000000..f28bc04f --- /dev/null +++ b/pkg/core/migrations/m028_aliyun_requester_config.py @@ -0,0 +1,27 @@ +from __future__ import annotations + +from .. import migration + + +@migration.migration_class("bailian-requester-config", 28) +class BailianRequesterConfigMigration(migration.Migration): + """迁移""" + + async def need_migrate(self) -> bool: + """判断当前环境是否需要运行此迁移""" + + return 'bailian-chat-completions' not in self.ap.provider_cfg.data['requester'] + + async def run(self): + """执行迁移""" + self.ap.provider_cfg.data['keys']['bailian'] = [ + "sk-xxxxxxx" + ] + + self.ap.provider_cfg.data['requester']['bailian-chat-completions'] = { + "base-url": "https://dashscope.aliyuncs.com/compatible-mode/v1", + "args": {}, + "timeout": 120 + } + + await self.ap.provider_cfg.dump_config() diff --git a/pkg/core/stages/migrate.py b/pkg/core/stages/migrate.py index 16faa53a..e9b0b4d7 100644 --- a/pkg/core/stages/migrate.py +++ b/pkg/core/stages/migrate.py @@ -9,7 +9,7 @@ from ..migrations import m005_deepseek_cfg_completion, m006_vision_config, m007_ from ..migrations import m010_ollama_requester_config, m011_command_prefix_config, m012_runner_config, m013_http_api_config, m014_force_delay_config from ..migrations import m015_gitee_ai_config, m016_dify_service_api, m017_dify_api_timeout_params, m018_xai_config, m019_zhipuai_config from ..migrations import m020_wecom_config, m021_lark_config, m022_lmstudio_config, m023_siliconflow_config, m024_discord_config, m025_gewechat_config -from ..migrations import m026_qqofficial_config +from ..migrations import m026_qqofficial_config, m028_aliyun_requester_config @stage.stage_class("MigrationStage") class MigrationStage(stage.BootingStage): diff --git a/pkg/provider/modelmgr/modelmgr.py b/pkg/provider/modelmgr/modelmgr.py index 6748291c..489a322f 100644 --- a/pkg/provider/modelmgr/modelmgr.py +++ b/pkg/provider/modelmgr/modelmgr.py @@ -6,7 +6,7 @@ from . import entities, requester from ...core import app from . import token -from .requesters import chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl, aliyunchatcmpl +from .requesters import bailianchatcmpl, chatcmpl, anthropicmsgs, moonshotchatcmpl, deepseekchatcmpl, ollamachat, giteeaichatcmpl, xaichatcmpl, zhipuaichatcmpl, lmstudiochatcmpl, siliconflowchatcmpl FETCH_MODEL_LIST_URL = "https://api.qchatgpt.rockchin.top/api/v2/fetch/model_list" diff --git a/pkg/provider/modelmgr/requesters/aliyunchatcmpl.py b/pkg/provider/modelmgr/requesters/bailianchatcmpl.py similarity index 58% rename from pkg/provider/modelmgr/requesters/aliyunchatcmpl.py rename to pkg/provider/modelmgr/requesters/bailianchatcmpl.py index 4f4900d7..cce003bd 100644 --- a/pkg/provider/modelmgr/requesters/aliyunchatcmpl.py +++ b/pkg/provider/modelmgr/requesters/bailianchatcmpl.py @@ -7,9 +7,9 @@ from .. import requester from ....core import app -@requester.requester_class("aliyun-chat-completions") -class AliyunChatCompletions(chatcmpl.OpenAIChatCompletions): - """Aliyun ChatCompletion API 请求器""" +@requester.requester_class("bailian-chat-completions") +class BailianChatCompletions(chatcmpl.OpenAIChatCompletions): + """阿里云百炼大模型平台 ChatCompletion API 请求器""" client: openai.AsyncClient @@ -18,4 +18,4 @@ class AliyunChatCompletions(chatcmpl.OpenAIChatCompletions): def __init__(self, ap: app.Application): self.ap = ap - self.requester_cfg = self.ap.provider_cfg.data['requester']['aliyun-chat-completions'] + self.requester_cfg = self.ap.provider_cfg.data['requester']['bailian-chat-completions'] diff --git a/templates/provider.json b/templates/provider.json index c22ba42c..551c7d88 100644 --- a/templates/provider.json +++ b/templates/provider.json @@ -26,8 +26,8 @@ "siliconflow": [ "xxxxxxx" ], - "aliyun": [ - "sk-aliyun_token" + "bailian": [ + "sk-xxxxxxx" ] }, "requester": { @@ -83,7 +83,7 @@ "args": {}, "timeout": 120 }, - "aliyun-chat-completions": { + "bailian-chat-completions": { "args": {}, "base-url": "https://dashscope.aliyuncs.com/compatible-mode/v1", "timeout": 120 diff --git a/templates/schema/provider.json b/templates/schema/provider.json index bd2084b8..e6f806c5 100644 --- a/templates/schema/provider.json +++ b/templates/schema/provider.json @@ -82,6 +82,14 @@ "type": "string" }, "default": [] + }, + "bailian": { + "type": "array", + "title": "阿里云百炼大模型平台 API 密钥", + "items": { + "type": "string" + }, + "default": [] } } }, @@ -288,6 +296,26 @@ "default": 120 } } + }, + "bailian-chat-completions": { + "type": "object", + "title": "阿里云百炼大模型平台 API 请求配置", + "description": "仅可编辑 URL 和 超时时间,额外请求参数不支持可视化编辑,请到编辑器编辑", + "properties": { + "base-url": { + "type": "string", + "title": "API URL" + }, + "args": { + "type": "object", + "default": {} + }, + "timeout": { + "type": "number", + "title": "API 请求超时时间", + "default": 120 + } + } } } }, From a90f996b249f80c0eed231870021201b35c28caa Mon Sep 17 00:00:00 2001 From: Junyan Qin Date: Wed, 12 Feb 2025 13:33:07 +0800 Subject: [PATCH 11/13] chore: related configuration of dashscope runner --- .../m029_dashscope_app_api_config.py | 33 ++++++++++++ pkg/core/stages/migrate.py | 2 + pkg/provider/runnermgr.py | 2 + pkg/provider/runners/dashscopeapi.py | 18 +++---- templates/metadata/llm-models.json | 5 -- templates/provider.json | 13 +++-- templates/schema/provider.json | 53 +++++++++++++++++++ 7 files changed, 105 insertions(+), 21 deletions(-) create mode 100644 pkg/core/migrations/m029_dashscope_app_api_config.py diff --git a/pkg/core/migrations/m029_dashscope_app_api_config.py b/pkg/core/migrations/m029_dashscope_app_api_config.py new file mode 100644 index 00000000..3a069bac --- /dev/null +++ b/pkg/core/migrations/m029_dashscope_app_api_config.py @@ -0,0 +1,33 @@ +from __future__ import annotations + +from .. import migration + + +@migration.migration_class("dashscope-app-api-config", 29) +class DashscopeAppAPICfgMigration(migration.Migration): + """迁移""" + + async def need_migrate(self) -> bool: + """判断当前环境是否需要运行此迁移""" + return 'dashscope-app-api' not in self.ap.provider_cfg.data + + async def run(self): + """执行迁移""" + self.ap.provider_cfg.data['dashscope-app-api'] = { + "app-type": "agent", + "api-key": "sk-1234567890", + "agent": { + "app-id": "Your_app_id", + "references_quote": "参考资料来自:" + }, + "workflow": { + "app-id": "Your_app_id", + "references_quote": "参考资料来自:", + "biz_params": { + "city": "北京", + "date": "2023-08-10" + } + } + } + + await self.ap.provider_cfg.dump_config() diff --git a/pkg/core/stages/migrate.py b/pkg/core/stages/migrate.py index a1983f0b..7be46b4c 100644 --- a/pkg/core/stages/migrate.py +++ b/pkg/core/stages/migrate.py @@ -11,6 +11,8 @@ from ..migrations import m015_gitee_ai_config, m016_dify_service_api, m017_dify_ from ..migrations import m020_wecom_config, m021_lark_config, m022_lmstudio_config, m023_siliconflow_config, m024_discord_config, m025_gewechat_config from ..migrations import m026_qqofficial_config, m027_wx_official_account_config +from ..migrations import m029_dashscope_app_api_config + @stage.stage_class("MigrationStage") class MigrationStage(stage.BootingStage): """迁移阶段 diff --git a/pkg/provider/runnermgr.py b/pkg/provider/runnermgr.py index 4cd38c04..52e1d8d2 100644 --- a/pkg/provider/runnermgr.py +++ b/pkg/provider/runnermgr.py @@ -23,6 +23,8 @@ class RunnerManager: self.using_runner = r(self.ap) await self.using_runner.initialize() break + else: + raise ValueError(f"未找到请求运行器: {self.ap.provider_cfg.data['runner']}") def get_runner(self) -> runner.RequestRunner: return self.using_runner diff --git a/pkg/provider/runners/dashscopeapi.py b/pkg/provider/runners/dashscopeapi.py index 8a6aef9d..0201f35d 100644 --- a/pkg/provider/runners/dashscopeapi.py +++ b/pkg/provider/runners/dashscopeapi.py @@ -20,7 +20,7 @@ class DashscopeAPIError(Exception): super().__init__(self.message) -@runner.runner_class("dashscope-service-api") +@runner.runner_class("dashscope-app-api") class DashScopeAPIRunner(runner.RequestRunner): "阿里云百炼DashsscopeAPI对话请求器" @@ -34,7 +34,7 @@ class DashScopeAPIRunner(runner.RequestRunner): async def initialize(self): """初始化""" valid_app_types = ["agent", "workflow"] - self.app_type = self.ap.provider_cfg.data["dashscope-service-api"]["app-type"] + self.app_type = self.ap.provider_cfg.data["dashscope-app-api"]["app-type"] #检查配置文件中使用的应用类型是否支持 if (self.app_type not in valid_app_types): raise DashscopeAPIError( @@ -42,10 +42,10 @@ class DashScopeAPIRunner(runner.RequestRunner): ) #初始化Dashscope 参数配置 - self.app_id = self.ap.provider_cfg.data["dashscope-service-api"][self.app_type]["app-id"] - self.api_key = self.ap.provider_cfg.data["dashscope-service-api"][self.app_type]["api-key"] - self.references_quote = self.ap.provider_cfg.data["dashscope-service-api"][self.app_type]["references_quote"] - self.biz_params = self.ap.provider_cfg.data["dashscope-service-api"]["workflow"]["biz_params"] + self.app_id = self.ap.provider_cfg.data["dashscope-app-api"][self.app_type]["app-id"] + self.api_key = self.ap.provider_cfg.data["dashscope-app-api"]["api-key"] + self.references_quote = self.ap.provider_cfg.data["dashscope-app-api"][self.app_type]["references_quote"] + self.biz_params = self.ap.provider_cfg.data["dashscope-app-api"]["workflow"]["biz_params"] def _replace_references(self, text, references_dict): """阿里云百炼平台的自定义应用支持资料引用,此函数可以将引用标签替换为参考资料""" @@ -222,15 +222,15 @@ class DashScopeAPIRunner(runner.RequestRunner): self, query: core_entities.Query ) -> typing.AsyncGenerator[llm_entities.Message, None]: """运行""" - if self.ap.provider_cfg.data["dashscope-service-api"]["app-type"] == "agent": + if self.ap.provider_cfg.data["dashscope-app-api"]["app-type"] == "agent": async for msg in self._agent_messages(query): yield msg - elif self.ap.provider_cfg.data["dashscope-service-api"]["app-type"] == "workflow": + elif self.ap.provider_cfg.data["dashscope-app-api"]["app-type"] == "workflow": async for msg in self._workflow_messages(query): yield msg else: raise DashscopeAPIError( - f"不支持的 Dashscope 应用类型: {self.ap.provider_cfg.data['dashscope-service-api']['app-type']}" + f"不支持的 Dashscope 应用类型: {self.ap.provider_cfg.data['dashscope-app-api']['app-type']}" ) diff --git a/templates/metadata/llm-models.json b/templates/metadata/llm-models.json index 88757e66..b5c29cf3 100644 --- a/templates/metadata/llm-models.json +++ b/templates/metadata/llm-models.json @@ -211,11 +211,6 @@ "requester": "zhipuai-chat-completions", "token_mgr": "zhipuai", "vision_supported": true - }, - { - "name": "your-dashscope-app-id", - "requester": "dashscope-chat-applications", - "token_mgr": "dashscope", } ] } \ No newline at end of file diff --git a/templates/provider.json b/templates/provider.json index 13044f1e..e1eb2e9c 100644 --- a/templates/provider.json +++ b/templates/provider.json @@ -104,21 +104,20 @@ "timeout": 120 } }, - "dashscope-service-api": { + "dashscope-app-api": { + "app-type": "agent", + "api-key": "sk-1234567890", "agent": { - "api-key": "sk-1234567890", "app-id": "Your_app_id", "references_quote": "参考资料来自:" }, - "app-type": "agent", "workflow": { - "api-key": "sk-1234567890", "app-id": "Your_app_id", "references_quote": "参考资料来自:", "biz_params": { - "city": "北京", - "date": "2023-08-10" - } + "city": "北京", + "date": "2023-08-10" + } } } } \ No newline at end of file diff --git a/templates/schema/provider.json b/templates/schema/provider.json index bd2084b8..b4a72253 100644 --- a/templates/schema/provider.json +++ b/templates/schema/provider.json @@ -397,6 +397,59 @@ } } } + }, + "dashscope-app-api": { + "type": "object", + "title": "阿里百炼平台自建应用 API 配置", + "properties": { + "app-type": { + "type": "string", + "title": "应用类型", + "description": "支持 workflow 和 agent,workflow:智能体编排;agent:普通智能体;请填写下方对应的应用类型 API 参数", + "enum": ["workflow", "agent"], + "default": "agent" + }, + "api-key": { + "type": "string", + "title": "API 密钥" + }, + "agent": { + "type": "object", + "title": "Agent API 参数", + "properties": { + "app-id": { + "type": "string", + "title": "应用 ID" + }, + "references_quote": { + "type": "string", + "title": "参考资料引用", + "description": "设置参考资料引用,用于从 Dashscope App API 结束节点返回的 JSON 数据中提取引用内容", + "default": "参考资料来自:" + } + } + }, + "workflow": { + "type": "object", + "title": "工作流 API 参数", + "properties": { + "app-id": { + "type": "string", + "title": "应用 ID" + }, + "references_quote": { + "type": "string", + "title": "参考资料引用", + "default": "参考资料来自:" + }, + "biz_params": { + "type": "object", + "title": "传入参数", + "default": {} + } + } + } + } } } } \ No newline at end of file From 35721c134051361c765f1ce4d3c767733097ad9f Mon Sep 17 00:00:00 2001 From: Junyan Qin Date: Wed, 12 Feb 2025 13:47:01 +0800 Subject: [PATCH 12/13] doc(README): update comment of aliyun bailian --- README.md | 2 +- README_EN.md | 2 +- README_JP.md | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 4cf10013..8fa57c20 100644 --- a/README.md +++ b/README.md @@ -111,7 +111,7 @@ | [LMStudio](https://lmstudio.ai/) | ✅ | 本地大模型运行平台 | | [GiteeAI](https://ai.gitee.com/) | ✅ | 大模型接口聚合平台 | | [SiliconFlow](https://siliconflow.cn/) | ✅ | 大模型聚合平台 | -| [阿里云百炼](https://bailian.console.aliyun.com/) | ✅ | 大模型聚合平台 | +| [阿里云百炼](https://bailian.console.aliyun.com/) | ✅ | 大模型聚合平台, LLMOps 平台 | ## 😘 社区贡献 diff --git a/README_EN.md b/README_EN.md index 691774ef..6684a9ae 100644 --- a/README_EN.md +++ b/README_EN.md @@ -110,7 +110,7 @@ Directly use the released version to run, see the [Manual Deployment](https://do | [LMStudio](https://lmstudio.ai/) | ✅ | Local LLM running platform | | [GiteeAI](https://ai.gitee.com/) | ✅ | LLM interface gateway(MaaS) | | [SiliconFlow](https://siliconflow.cn/) | ✅ | LLM gateway(MaaS) | -| [Aliyun Bailian](https://bailian.console.aliyun.com/) | ✅ | LLM gateway(MaaS) | +| [Aliyun Bailian](https://bailian.console.aliyun.com/) | ✅ | LLM gateway(MaaS), LLMOps platform | ## 🤝 Community Contribution diff --git a/README_JP.md b/README_JP.md index d2303fba..f328349c 100644 --- a/README_JP.md +++ b/README_JP.md @@ -108,7 +108,7 @@ LangBotはBTPanelにリストされています。BTPanelをインストール | [LMStudio](https://lmstudio.ai/) | ✅ | ローカルLLM実行プラットフォーム | | [GiteeAI](https://ai.gitee.com/) | ✅ | LLMインターフェースゲートウェイ(MaaS) | | [SiliconFlow](https://siliconflow.cn/) | ✅ | LLMゲートウェイ(MaaS) | -| [Aliyun Bailian](https://bailian.console.aliyun.com/) | ✅ | LLMゲートウェイ(MaaS) | +| [Aliyun Bailian](https://bailian.console.aliyun.com/) | ✅ | LLMゲートウェイ(MaaS), LLMOpsプラットフォーム | ## 🤝 コミュニティ貢献 From 5311e787766b100584166113a23a69344beb5e09 Mon Sep 17 00:00:00 2001 From: Junyan Qin Date: Wed, 12 Feb 2025 15:16:02 +0800 Subject: [PATCH 13/13] chore: release v3.4.7.1 --- pkg/utils/constants.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/utils/constants.py b/pkg/utils/constants.py index 7e582b0f..0b9966da 100644 --- a/pkg/utils/constants.py +++ b/pkg/utils/constants.py @@ -1,4 +1,4 @@ -semantic_version = "v3.4.7" +semantic_version = "v3.4.7.1" debug_mode = False