Spaces:
Runtime error
Runtime error
import json | |
import os | |
from chatgpt_tool_hub.apps import load_app | |
from chatgpt_tool_hub.apps.app import App | |
from chatgpt_tool_hub.tools.all_tool_list import get_all_tool_names | |
import plugins | |
from bridge.bridge import Bridge | |
from bridge.context import ContextType | |
from bridge.reply import Reply, ReplyType | |
from common import const | |
from common.log import logger | |
from config import conf | |
from plugins import * | |
class Tool(Plugin): | |
def __init__(self): | |
super().__init__() | |
self.handlers[Event.ON_HANDLE_CONTEXT] = self.on_handle_context | |
self.app = self._reset_app() | |
logger.info("[tool] inited") | |
def get_help_text(self, verbose=False, **kwargs): | |
help_text = "这是一个能让chatgpt联网,搜索,数字运算的插件,将赋予强大且丰富的扩展能力。" | |
if not verbose: | |
return help_text | |
trigger_prefix = conf().get('plugin_trigger_prefix', "$") | |
help_text += "使用说明:\n" | |
help_text += f"{trigger_prefix}tool "+"命令: 根据给出的{命令}使用一些可用工具尽力为你得到结果。\n" | |
help_text += f"{trigger_prefix}tool reset: 重置工具。\n" | |
return help_text | |
def on_handle_context(self, e_context: EventContext): | |
if e_context['context'].type != ContextType.TEXT: | |
return | |
# 暂时不支持未来扩展的bot | |
if Bridge().get_bot_type("chat") not in (const.CHATGPT, const.OPEN_AI, const.CHATGPTONAZURE): | |
return | |
content = e_context['context'].content | |
content_list = e_context['context'].content.split(maxsplit=1) | |
if not content or len(content_list) < 1: | |
e_context.action = EventAction.CONTINUE | |
return | |
logger.debug("[tool] on_handle_context. content: %s" % content) | |
reply = Reply() | |
reply.type = ReplyType.TEXT | |
trigger_prefix = conf().get('plugin_trigger_prefix', "$") | |
# todo: 有些工具必须要api-key,需要修改config文件,所以这里没有实现query增删tool的功能 | |
if content.startswith(f"{trigger_prefix}tool"): | |
if len(content_list) == 1: | |
logger.debug("[tool]: get help") | |
reply.content = self.get_help_text() | |
e_context['reply'] = reply | |
e_context.action = EventAction.BREAK_PASS | |
return | |
elif len(content_list) > 1: | |
if content_list[1].strip() == "reset": | |
logger.debug("[tool]: reset config") | |
self.app = self._reset_app() | |
reply.content = "重置工具成功" | |
e_context['reply'] = reply | |
e_context.action = EventAction.BREAK_PASS | |
return | |
elif content_list[1].startswith("reset"): | |
logger.debug("[tool]: remind") | |
e_context['context'].content = "请你随机用一种聊天风格,提醒用户:如果想重置tool插件,reset之后不要加任何字符" | |
e_context.action = EventAction.BREAK | |
return | |
query = content_list[1].strip() | |
# Don't modify bot name | |
all_sessions = Bridge().get_bot("chat").sessions | |
user_session = all_sessions.session_query(query, e_context['context']['session_id']).messages | |
# chatgpt-tool-hub will reply you with many tools | |
logger.debug("[tool]: just-go") | |
try: | |
_reply = self.app.ask(query, user_session) | |
e_context.action = EventAction.BREAK_PASS | |
all_sessions.session_reply(_reply, e_context['context']['session_id']) | |
except Exception as e: | |
logger.exception(e) | |
logger.error(str(e)) | |
e_context['context'].content = "请你随机用一种聊天风格,提醒用户:这个问题tool插件暂时无法处理" | |
reply.type = ReplyType.ERROR | |
e_context.action = EventAction.BREAK | |
return | |
reply.content = _reply | |
e_context['reply'] = reply | |
return | |
def _read_json(self) -> dict: | |
curdir = os.path.dirname(__file__) | |
config_path = os.path.join(curdir, "config.json") | |
tool_config = { | |
"tools": [], | |
"kwargs": {} | |
} | |
if not os.path.exists(config_path): | |
return tool_config | |
else: | |
with open(config_path, "r") as f: | |
tool_config = json.load(f) | |
return tool_config | |
def _build_tool_kwargs(self, kwargs: dict): | |
tool_model_name = kwargs.get("model_name") | |
return { | |
"openai_api_key": conf().get("open_ai_api_key", ""), | |
"proxy": conf().get("proxy", ""), | |
# note: 目前tool暂未对其他模型测试,但这里仍对配置来源做了优先级区分,一般插件配置可覆盖全局配置 | |
"model_name": tool_model_name if tool_model_name else conf().get("model", "gpt-3.5-turbo"), | |
"no_default": kwargs.get("no_default", False), | |
"top_k_results": kwargs.get("top_k_results", 2), | |
# for news tool | |
"news_api_key": kwargs.get("news_api_key", ""), | |
# for bing-search tool | |
"bing_subscription_key": kwargs.get("bing_subscription_key", ""), | |
# for google-search tool | |
"google_api_key": kwargs.get("google_api_key", ""), | |
"google_cse_id": kwargs.get("google_cse_id", ""), | |
# for searxng-search tool | |
"searx_host": kwargs.get("searx_host", ""), | |
# for wolfram-alpha tool | |
"wolfram_alpha_appid": kwargs.get("wolfram_alpha_appid", ""), | |
} | |
def _filter_tool_list(self, tool_list: list): | |
valid_list = [] | |
for tool in tool_list: | |
if tool in get_all_tool_names(): | |
valid_list.append(tool) | |
else: | |
logger.warning("[tool] filter invalid tool: " + repr(tool)) | |
return valid_list | |
def _reset_app(self) -> App: | |
tool_config = self._read_json() | |
# filter not support tool | |
tool_list = self._filter_tool_list(tool_config.get("tools", [])) | |
return load_app(tools_list=tool_list, **self._build_tool_kwargs(tool_config.get("kwargs", {}))) | |