Files
Mofox-Core/docs/plugins/action-components.md
2025-06-19 23:21:31 +08:00

12 KiB
Raw Blame History

Action组件详解

📖 什么是Action

Action是给麦麦在回复之外提供额外功能的智能组件由麦麦的决策系统自主选择是否使用具有随机性和拟人化的调用特点。Action不是直接响应用户命令而是让麦麦根据聊天情境智能地选择合适的动作使其行为更加自然和真实。

🎯 Action的特点

  • 🧠 智能激活:麦麦根据多种条件智能判断是否使用
  • 🎲 随机性:增加行为的不可预测性,更接近真人交流
  • 🤖 拟人化:让麦麦的回应更自然、更有个性
  • 🔄 情境感知:基于聊天上下文做出合适的反应

🎯 两层决策机制

Action采用两层决策机制来优化性能和决策质量:

第一层激活控制Activation Control

激活决定麦麦是否"知道"这个Action的存在即这个Action是否进入决策候选池。不被激活的Action麦麦永远不会选择

🎯 设计目的在加载许多插件的时候降低LLM决策压力避免让麦麦在过多的选项中纠结。

激活类型说明

激活类型 说明 使用场景
NEVER 从不激活Action对麦麦不可见 临时禁用某个Action
ALWAYS 永远激活Action总是在麦麦的候选池中 核心功能,如回复、不回复
LLM_JUDGE 通过LLM智能判断当前情境是否需要激活此Action 需要智能判断的复杂场景
RANDOM 基于随机概率决定是否激活 增加行为随机性的功能
KEYWORD 当检测到特定关键词时激活 明确触发条件的功能

聊天模式控制

模式 说明
ChatMode.FOCUS 仅在专注聊天模式下可激活
ChatMode.NORMAL 仅在普通聊天模式下可激活
ChatMode.ALL 所有模式下都可激活

第二层使用决策Usage Decision

在Action被激活后使用条件决定麦麦什么时候会"选择"使用这个Action

这一层由以下因素综合决定:

  • action_require使用场景描述帮助LLM判断何时选择
  • action_parameters所需参数影响Action的可执行性
  • 当前聊天上下文和麦麦的决策逻辑

🎬 决策流程示例

假设有一个"发送表情"Action

class EmojiAction(BaseAction):
    # 第一层:激活控制
    focus_activation_type = ActionActivationType.RANDOM  # 专注模式下随机激活
    normal_activation_type = ActionActivationType.KEYWORD  # 普通模式下关键词激活
    activation_keywords = ["表情", "emoji", "😊"]
  
    # 第二层:使用决策
    action_require = [
        "表达情绪时可以选择使用",
        "增加聊天趣味性",
        "不要连续发送多个表情"
    ]

决策流程

  1. 第一层激活判断

    • 普通模式:只有当用户消息包含"表情"、"emoji"或"😊"时,麦麦才"知道"可以使用这个Action
    • 专注模式:随机激活,有概率让麦麦"看到"这个Action
  2. 第二层使用决策

    • 即使Action被激活麦麦还会根据 action_require中的条件判断是否真正选择使用
    • 例如:如果刚刚已经发过表情,根据"不要连续发送多个表情"的要求麦麦可能不会选择这个Action

📋 Action必须项清单

每个Action类都必须包含以下属性:

1. 激活控制必须项

# 专注模式下的激活类型
focus_activation_type = ActionActivationType.LLM_JUDGE

# 普通模式下的激活类型
normal_activation_type = ActionActivationType.KEYWORD

# 启用的聊天模式
mode_enable = ChatMode.ALL

# 是否允许与其他Action并行执行
parallel_action = False

2. 基本信息必须项

# Action的唯一标识名称
action_name = "my_action"

# Action的功能描述
action_description = "描述这个Action的具体功能和用途"

3. 功能定义必须项

# Action参数定义 - 告诉LLM执行时需要什么参数
action_parameters = {
    "param1": "参数1的说明",
    "param2": "参数2的说明"
}

# Action使用场景描述 - 帮助LLM判断何时"选择"使用
action_require = [
    "使用场景描述1",
    "使用场景描述2"
]

# 关联的消息类型 - 说明Action能处理什么类型的内容
associated_types = ["text", "emoji", "image"]

4. 新API导入必须项

使用新插件系统时必须导入所需的API模块

# 导入新API模块
from src.plugin_system.apis import generator_api, send_api, emoji_api

# 如果需要使用其他API
from src.plugin_system.apis import llm_api, database_api, message_api

5. 动作记录必须项

每个 Action 在执行完成后,必须使用 store_action_info 记录动作信息:

async def execute(self) -> Tuple[bool, str]:
    # ... 执行动作的代码 ...
  
    if success:
        # 存储动作信息 - 使用新API格式
        await self.store_action_info(
            action_build_into_prompt=True,  # 让麦麦知道这个动作
            action_prompt_display=f"执行了xxx动作参数{param}",  # 动作描述
            action_done=True,  # 动作是否完成
        )
        return True, "动作执行成功"

⚠️ 重要提示新API格式中不再需要手动传递 thinking_id 等参数BaseAction会自动处理。

🚀 新API使用指南

📨 消息发送API

新的消息发送API更加简洁自动处理群聊/私聊逻辑:

class MessageAction(BaseAction):
    async def execute(self) -> Tuple[bool, str]:
        # 发送文本消息 - 自动判断群聊/私聊
        await self.send_text("Hello World!")
  
        # 发送表情包
        emoji_base64 = await emoji_api.get_by_description("开心")
        if emoji_base64:
            await self.send_emoji(emoji_base64)
  
        # 发送图片
        await self.send_image(image_base64)
  
        # 发送自定义类型消息
        await self.send_custom("video", video_data, typing=True)
  
        return True, "消息发送完成"

🤖 智能生成API (replyer_1)

使用replyer_1生成个性化内容

class SmartReplyAction(BaseAction):
    async def execute(self) -> Tuple[bool, str]:
        # 构建生成参数
        reply_data = {
            "text": "请生成一个友好的回复",
            "style": "casual",
            "topic": "日常聊天",
            "replyer_name": "replyer_1"  # 指定使用replyer_1
        }
  
        # 使用generator_api生成回复
        success, reply_set = await generator_api.generate_reply(
            chat_stream=self.chat_stream,
            action_data=reply_data,
            platform=self.platform,
            chat_id=self.chat_id,
            is_group=self.is_group
        )
  
        if success and reply_set:
            # 提取并发送文本回复
            for reply_type, reply_content in reply_set:
                if reply_type == "text":
                    await self.send_text(reply_content)
                elif reply_type == "emoji":
                    await self.send_emoji(reply_content)
    
            # 记录动作
            await self.store_action_info(
                action_build_into_prompt=True,
                action_prompt_display=f"使用replyer_1生成了智能回复",
                action_done=True
            )
    
            return True, "智能回复生成成功"
        else:
            return False, "回复生成失败"

⚙️ 配置访问API

使用便捷的配置访问方法:

class ConfigurableAction(BaseAction):
    async def execute(self) -> Tuple[bool, str]:
        # 获取插件配置 - 支持嵌套键访问
        enable_feature = self.get_config("features.enable_smart_mode", False)
        max_length = self.get_config("limits.max_text_length", 200)
        style = self.get_config("behavior.response_style", "friendly")
  
        if enable_feature:
            # 启用高级功能
            pass
  
        return True, "配置获取成功"

📊 数据库API

使用新的数据库API存储和查询数据

class DataAction(BaseAction):
    async def execute(self) -> Tuple[bool, str]:
        # 使用database_api
        from src.plugin_system.apis import database_api
  
        # 存储数据
        await database_api.store_action_info(
            chat_stream=self.chat_stream,
            action_name=self.action_name,
            action_data=self.action_data,
            # ... 其他参数
        )
  
        return True, "数据存储完成"

🔧 激活类型详解

KEYWORD激活

当检测到特定关键词时激活Action

class GreetingAction(BaseAction):
    focus_activation_type = ActionActivationType.KEYWORD
    normal_activation_type = ActionActivationType.KEYWORD
  
    # 关键词配置
    activation_keywords = ["你好", "hello", "hi", "嗨"]
    keyword_case_sensitive = False  # 不区分大小写
  
    async def execute(self) -> Tuple[bool, str]:
        # 可选使用replyer_1生成个性化问候
        if self.get_config("greeting.use_smart_reply", False):
            greeting_data = {
                "text": "生成一个友好的问候语",
                "replyer_name": "replyer_1"
            }
    
            success, reply_set = await generator_api.generate_reply(
                chat_stream=self.chat_stream,
                action_data=greeting_data
            )
    
            if success:
                for reply_type, content in reply_set:
                    if reply_type == "text":
                        await self.send_text(content)
                        break
                return True, "发送智能问候"
  
        # 传统问候方式
        await self.send_text("你好!很高兴见到你!")
        return True, "发送问候"

LLM_JUDGE激活

通过LLM智能判断是否激活

class HelpAction(BaseAction):
    focus_activation_type = ActionActivationType.LLM_JUDGE
    normal_activation_type = ActionActivationType.LLM_JUDGE
  
    # LLM判断提示词
    llm_judge_prompt = """
    判定是否需要使用帮助动作的条件:
    1. 用户表达了困惑或需要帮助
    2. 用户提出了问题但没有得到满意答案
    3. 对话中出现了技术术语或复杂概念
  
    请回答"是"或"否"。
    """
  
    async def execute(self) -> Tuple[bool, str]:
        # 使用replyer_1生成帮助内容
        help_data = {
            "text": "用户需要帮助,请提供适当的帮助信息",
            "help_type": self.action_data.get("help_type", "general"),
            "replyer_name": "replyer_1"
        }
  
        success, reply_set = await generator_api.generate_reply(
            chat_stream=self.chat_stream,
            action_data=help_data
        )
  
        if success:
            for reply_type, content in reply_set:
                if reply_type == "text":
                    await self.send_text(content)
            return True, "提供了帮助"
        else:
            await self.send_text("我来帮助你!有什么问题吗?")
            return True, "提供了默认帮助"

RANDOM激活

基于随机概率激活:

class SurpriseAction(BaseAction):
    focus_activation_type = ActionActivationType.RANDOM
    normal_activation_type = ActionActivationType.RANDOM
  
    # 随机激活概率
    random_activation_probability = 0.1  # 10%概率激活
  
    async def execute(self) -> Tuple[bool, str]:
        import random
  
        surprises = ["🎉", "✨", "🌟", "💝", "🎈"]
        selected = random.choice(surprises)
  
        await self.send_emoji(selected)
        return True, f"发送了惊喜表情: {selected}"