mirror of
https://github.com/zhayujie/chatgpt-on-wechat.git
synced 2026-01-19 01:21:01 +08:00
183 lines
7.0 KiB
Python
183 lines
7.0 KiB
Python
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 *
|
||
|
||
|
||
@plugins.register(
|
||
name="tool",
|
||
desc="Arming your ChatGPT bot with various tools",
|
||
version="0.3",
|
||
author="goldfishh",
|
||
desire_priority=0,
|
||
)
|
||
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", ""),
|
||
"request_timeout": conf().get("request_timeout", 60),
|
||
# 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", ""),
|
||
# for morning-news tool
|
||
"zaobao_api_key": kwargs.get("zaobao_api_key", ""),
|
||
# for visual_dl tool
|
||
"cuda_device": kwargs.get("cuda_device", "cpu"),
|
||
# for browser tool
|
||
"phantomjs_exec_path": kwargs.get("phantomjs_exec_path", ""),
|
||
}
|
||
|
||
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", {})),
|
||
)
|