* refactor: LLM response handling with reasoning content - Added a `show_reasoning` parameter to `run_agent` to control the display of reasoning content. - Updated `LLMResponse` to include a `reasoning_content` field for storing reasoning text. - Modified `WebChatMessageEvent` to handle and send reasoning content in streaming responses. - Implemented reasoning extraction in various provider sources (e.g., OpenAI, Gemini). - Updated the chat interface to display reasoning content in a collapsible format. - Removed the deprecated `thinking_filter` package and its associated logic. - Updated localization files to include new reasoning-related strings. * feat: add Groq chat completion provider and associated configurations * Update astrbot/core/provider/sources/gemini_source.py Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com> --------- Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com>
362 lines
14 KiB
Python
362 lines
14 KiB
Python
import traceback
|
|
|
|
from astrbot.api import star
|
|
from astrbot.api.event import AstrMessageEvent, filter
|
|
from astrbot.api.message_components import Image, Plain
|
|
from astrbot.api.provider import LLMResponse, ProviderRequest
|
|
from astrbot.core import logger
|
|
from astrbot.core.provider.sources.dify_source import ProviderDify
|
|
|
|
from .commands import (
|
|
AdminCommands,
|
|
AlterCmdCommands,
|
|
ConversationCommands,
|
|
HelpCommand,
|
|
LLMCommands,
|
|
PersonaCommands,
|
|
PluginCommands,
|
|
ProviderCommands,
|
|
SetUnsetCommands,
|
|
SIDCommand,
|
|
T2ICommand,
|
|
ToolCommands,
|
|
TTSCommand,
|
|
)
|
|
from .long_term_memory import LongTermMemory
|
|
from .process_llm_request import ProcessLLMRequest
|
|
|
|
|
|
class Main(star.Star):
|
|
def __init__(self, context: star.Context) -> None:
|
|
self.context = context
|
|
self.ltm = None
|
|
try:
|
|
self.ltm = LongTermMemory(self.context.astrbot_config_mgr, self.context)
|
|
except BaseException as e:
|
|
logger.error(f"聊天增强 err: {e}")
|
|
|
|
self.help_c = HelpCommand(self.context)
|
|
self.llm_c = LLMCommands(self.context)
|
|
self.tool_c = ToolCommands(self.context)
|
|
self.plugin_c = PluginCommands(self.context)
|
|
self.admin_c = AdminCommands(self.context)
|
|
self.conversation_c = ConversationCommands(self.context, self.ltm)
|
|
self.provider_c = ProviderCommands(self.context)
|
|
self.persona_c = PersonaCommands(self.context)
|
|
self.alter_cmd_c = AlterCmdCommands(self.context)
|
|
self.setunset_c = SetUnsetCommands(self.context)
|
|
self.t2i_c = T2ICommand(self.context)
|
|
self.tts_c = TTSCommand(self.context)
|
|
self.sid_c = SIDCommand(self.context)
|
|
self.proc_llm_req = ProcessLLMRequest(self.context)
|
|
|
|
def ltm_enabled(self, event: AstrMessageEvent):
|
|
ltmse = self.context.get_config(umo=event.unified_msg_origin)[
|
|
"provider_ltm_settings"
|
|
]
|
|
return ltmse["group_icl_enable"] or ltmse["active_reply"]["enable"]
|
|
|
|
@filter.command("help")
|
|
async def help(self, event: AstrMessageEvent):
|
|
"""查看帮助"""
|
|
await self.help_c.help(event)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("llm")
|
|
async def llm(self, event: AstrMessageEvent):
|
|
"""开启/关闭 LLM"""
|
|
await self.llm_c.llm(event)
|
|
|
|
@filter.command_group("tool")
|
|
def tool(self):
|
|
pass
|
|
|
|
@tool.command("ls")
|
|
async def tool_ls(self, event: AstrMessageEvent):
|
|
"""查看函数工具列表"""
|
|
await self.tool_c.tool_ls(event)
|
|
|
|
@tool.command("on")
|
|
async def tool_on(self, event: AstrMessageEvent, tool_name: str):
|
|
"""启用一个函数工具"""
|
|
await self.tool_c.tool_on(event, tool_name)
|
|
|
|
@tool.command("off")
|
|
async def tool_off(self, event: AstrMessageEvent, tool_name: str):
|
|
"""停用一个函数工具"""
|
|
await self.tool_c.tool_off(event, tool_name)
|
|
|
|
@tool.command("off_all")
|
|
async def tool_all_off(self, event: AstrMessageEvent):
|
|
"""停用所有函数工具"""
|
|
await self.tool_c.tool_all_off(event)
|
|
|
|
@filter.command_group("plugin")
|
|
def plugin(self):
|
|
pass
|
|
|
|
@plugin.command("ls")
|
|
async def plugin_ls(self, event: AstrMessageEvent):
|
|
"""获取已经安装的插件列表。"""
|
|
await self.plugin_c.plugin_ls(event)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@plugin.command("off")
|
|
async def plugin_off(self, event: AstrMessageEvent, plugin_name: str = ""):
|
|
"""禁用插件"""
|
|
await self.plugin_c.plugin_off(event, plugin_name)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@plugin.command("on")
|
|
async def plugin_on(self, event: AstrMessageEvent, plugin_name: str = ""):
|
|
"""启用插件"""
|
|
await self.plugin_c.plugin_on(event, plugin_name)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@plugin.command("get")
|
|
async def plugin_get(self, event: AstrMessageEvent, plugin_repo: str = ""):
|
|
"""安装插件"""
|
|
await self.plugin_c.plugin_get(event, plugin_repo)
|
|
|
|
@plugin.command("help")
|
|
async def plugin_help(self, event: AstrMessageEvent, plugin_name: str = ""):
|
|
"""获取插件帮助"""
|
|
await self.plugin_c.plugin_help(event, plugin_name)
|
|
|
|
@filter.command("t2i")
|
|
async def t2i(self, event: AstrMessageEvent):
|
|
"""开关文本转图片"""
|
|
await self.t2i_c.t2i(event)
|
|
|
|
@filter.command("tts")
|
|
async def tts(self, event: AstrMessageEvent):
|
|
"""开关文本转语音(会话级别)"""
|
|
await self.tts_c.tts(event)
|
|
|
|
@filter.command("sid")
|
|
async def sid(self, event: AstrMessageEvent):
|
|
"""获取会话 ID 和 管理员 ID"""
|
|
await self.sid_c.sid(event)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("op")
|
|
async def op(self, event: AstrMessageEvent, admin_id: str = ""):
|
|
"""授权管理员。op <admin_id>"""
|
|
await self.admin_c.op(event, admin_id)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("deop")
|
|
async def deop(self, event: AstrMessageEvent, admin_id: str):
|
|
"""取消授权管理员。deop <admin_id>"""
|
|
await self.admin_c.deop(event, admin_id)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("wl")
|
|
async def wl(self, event: AstrMessageEvent, sid: str = ""):
|
|
"""添加白名单。wl <sid>"""
|
|
await self.admin_c.wl(event, sid)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("dwl")
|
|
async def dwl(self, event: AstrMessageEvent, sid: str):
|
|
"""删除白名单。dwl <sid>"""
|
|
await self.admin_c.dwl(event, sid)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("provider")
|
|
async def provider(
|
|
self,
|
|
event: AstrMessageEvent,
|
|
idx: str | int | None = None,
|
|
idx2: int | None = None,
|
|
):
|
|
"""查看或者切换 LLM Provider"""
|
|
await self.provider_c.provider(event, idx, idx2)
|
|
|
|
@filter.command("reset")
|
|
async def reset(self, message: AstrMessageEvent):
|
|
"""重置 LLM 会话"""
|
|
await self.conversation_c.reset(message)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("model")
|
|
async def model_ls(
|
|
self,
|
|
message: AstrMessageEvent,
|
|
idx_or_name: int | str | None = None,
|
|
):
|
|
"""查看或者切换模型"""
|
|
await self.provider_c.model_ls(message, idx_or_name)
|
|
|
|
@filter.command("history")
|
|
async def his(self, message: AstrMessageEvent, page: int = 1):
|
|
"""查看对话记录"""
|
|
await self.conversation_c.his(message, page)
|
|
|
|
@filter.command("ls")
|
|
async def convs(self, message: AstrMessageEvent, page: int = 1):
|
|
"""查看对话列表"""
|
|
await self.conversation_c.convs(message, page)
|
|
|
|
@filter.command("new")
|
|
async def new_conv(self, message: AstrMessageEvent):
|
|
"""创建新对话"""
|
|
await self.conversation_c.new_conv(message)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("groupnew")
|
|
async def groupnew_conv(self, message: AstrMessageEvent, sid: str):
|
|
"""创建新群聊对话"""
|
|
await self.conversation_c.groupnew_conv(message, sid)
|
|
|
|
@filter.command("switch")
|
|
async def switch_conv(self, message: AstrMessageEvent, index: int | None = None):
|
|
"""通过 /ls 前面的序号切换对话"""
|
|
await self.conversation_c.switch_conv(message, index)
|
|
|
|
@filter.command("rename")
|
|
async def rename_conv(self, message: AstrMessageEvent, new_name: str):
|
|
"""重命名对话"""
|
|
await self.conversation_c.rename_conv(message, new_name)
|
|
|
|
@filter.command("del")
|
|
async def del_conv(self, message: AstrMessageEvent):
|
|
"""删除当前对话"""
|
|
await self.conversation_c.del_conv(message)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("key")
|
|
async def key(self, message: AstrMessageEvent, index: int | None = None):
|
|
"""查看或者切换 Key"""
|
|
await self.provider_c.key(message, index)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("persona")
|
|
async def persona(self, message: AstrMessageEvent):
|
|
"""查看或者切换 Persona"""
|
|
await self.persona_c.persona(message)
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("dashboard_update")
|
|
async def update_dashboard(self, event: AstrMessageEvent):
|
|
await self.admin_c.update_dashboard(event)
|
|
|
|
@filter.command("set")
|
|
async def set_variable(self, event: AstrMessageEvent, key: str, value: str):
|
|
await self.setunset_c.set_variable(event, key, value)
|
|
|
|
@filter.command("unset")
|
|
async def unset_variable(self, event: AstrMessageEvent, key: str):
|
|
await self.setunset_c.unset_variable(event, key)
|
|
|
|
@filter.platform_adapter_type(filter.PlatformAdapterType.ALL)
|
|
async def on_message(self, event: AstrMessageEvent):
|
|
"""群聊记忆增强"""
|
|
has_image_or_plain = False
|
|
for comp in event.message_obj.message:
|
|
if isinstance(comp, Plain) or isinstance(comp, Image):
|
|
has_image_or_plain = True
|
|
break
|
|
|
|
if self.ltm_enabled(event) and self.ltm and has_image_or_plain:
|
|
need_active = await self.ltm.need_active_reply(event)
|
|
|
|
group_icl_enable = self.context.get_config()["provider_ltm_settings"][
|
|
"group_icl_enable"
|
|
]
|
|
if group_icl_enable:
|
|
"""记录对话"""
|
|
try:
|
|
await self.ltm.handle_message(event)
|
|
except BaseException as e:
|
|
logger.error(e)
|
|
|
|
if need_active:
|
|
"""主动回复"""
|
|
provider = self.context.get_using_provider(event.unified_msg_origin)
|
|
if not provider:
|
|
logger.error("未找到任何 LLM 提供商。请先配置。无法主动回复")
|
|
return
|
|
try:
|
|
conv = None
|
|
if provider.meta().type != "dify":
|
|
session_curr_cid = await self.context.conversation_manager.get_curr_conversation_id(
|
|
event.unified_msg_origin,
|
|
)
|
|
|
|
if not session_curr_cid:
|
|
logger.error(
|
|
"当前未处于对话状态,无法主动回复,请确保 平台设置->会话隔离(unique_session) 未开启,并使用 /switch 序号 切换或者 /new 创建一个会话。",
|
|
)
|
|
return
|
|
|
|
conv = await self.context.conversation_manager.get_conversation(
|
|
event.unified_msg_origin,
|
|
session_curr_cid,
|
|
)
|
|
else:
|
|
# Dify 自己有维护对话,不需要 bot 端维护。
|
|
assert isinstance(provider, ProviderDify)
|
|
cid = provider.conversation_ids.get(
|
|
event.unified_msg_origin,
|
|
None,
|
|
)
|
|
if cid is None:
|
|
logger.error(
|
|
"[Dify] 当前未处于对话状态,无法主动回复,请确保 平台设置->会话隔离(unique_session) 未开启,并使用 /switch 序号 切换或者 /new 创建一个会话。",
|
|
)
|
|
return
|
|
|
|
prompt = event.message_str
|
|
|
|
if not conv:
|
|
logger.error("未找到对话,无法主动回复")
|
|
return
|
|
|
|
yield event.request_llm(
|
|
prompt=prompt,
|
|
func_tool_manager=self.context.get_llm_tool_manager(),
|
|
session_id=event.session_id,
|
|
conversation=conv,
|
|
)
|
|
except BaseException as e:
|
|
logger.error(traceback.format_exc())
|
|
logger.error(f"主动回复失败: {e}")
|
|
|
|
@filter.on_llm_request()
|
|
async def decorate_llm_req(self, event: AstrMessageEvent, req: ProviderRequest):
|
|
"""在请求 LLM 前注入人格信息、Identifier、时间、回复内容等 System Prompt"""
|
|
await self.proc_llm_req.process_llm_request(event, req)
|
|
|
|
if self.ltm and self.ltm_enabled(event):
|
|
try:
|
|
await self.ltm.on_req_llm(event, req)
|
|
except BaseException as e:
|
|
logger.error(f"ltm: {e}")
|
|
|
|
@filter.on_llm_response()
|
|
async def inject_reasoning(self, event: AstrMessageEvent, resp: LLMResponse):
|
|
"""在 LLM 响应后基于配置注入思考过程文本"""
|
|
umo = event.unified_msg_origin
|
|
cfg = self.context.get_config(umo).get("provider_settings", {})
|
|
show_reasoning = cfg.get("display_reasoning_text", False)
|
|
if show_reasoning and resp.reasoning_content:
|
|
resp.completion_text = (
|
|
f"🤔 思考: {resp.reasoning_content}\n\n{resp.completion_text}"
|
|
)
|
|
|
|
@filter.after_message_sent()
|
|
async def after_llm_req(self, event: AstrMessageEvent):
|
|
"""在 LLM 请求后记录对话"""
|
|
if self.ltm and self.ltm_enabled(event):
|
|
try:
|
|
await self.ltm.after_req_llm(event)
|
|
except Exception as e:
|
|
logger.error(f"ltm: {e}")
|
|
|
|
@filter.permission_type(filter.PermissionType.ADMIN)
|
|
@filter.command("alter_cmd", alias={"alter"})
|
|
async def alter_cmd(self, event: AstrMessageEvent):
|
|
"""修改命令权限"""
|
|
await self.alter_cmd_c.alter_cmd(event)
|