Files
Mofox-Core/docs/plugins/quick-start.md
2025-12-17 09:44:51 +08:00

15 KiB
Raw Blame History

🚀 快速开始指南

本指南将带你从零开始创建一个功能完整的MoFox_Bot插件。

📖 概述

这个指南将带你快速创建你的第一个MoFox_Bot插件。我们将创建一个简单的问候插件展示插件系统的基本概念。

以下代码都在我们的plugins/hello_world_plugin/目录下。

一个方便的小设计

在开发中,我们在__init__.py中定义了一个__all__变量,包含了所有需要导出的类和函数。 这样在其他地方导入时,可以直接使用 from src.plugin_system import * 来导入所有插件相关的类和函数。 或者你可以直接使用 from src.plugin_system import BasePlugin, register_plugin, ComponentInfo 之类的方式来导入你需要的部分。

📂 准备工作

确保你已经:

  1. 克隆了MoFox_Bot项目
  2. 安装了Python依赖
  3. 了解基本的Python语法

🏗️ 创建插件

1. 创建插件目录

在项目根目录的 plugins/ 文件夹下创建你的插件目录

这里我们创建一个名为 hello_world_plugin 的目录

2. 创建_manifest.json文件

在插件目录下面创建一个 _manifest.json 文件,内容如下:

{
  "manifest_version": 1,
  "name": "Hello World 插件",
  "version": "1.0.0",
  "description": "一个简单的 Hello World 插件",
  "author": {
    "name": "你的名字"
  }
}

有关 _manifest.json 的详细说明,请参考 Manifest文件指南

3. 创建最简单的插件

让我们从最基础的开始!创建 plugin.py 文件:

from typing import List, Tuple, Type
from src.plugin_system import BasePlugin, register_plugin, ComponentInfo

@register_plugin # 注册插件
class HelloWorldPlugin(BasePlugin):
    """Hello World插件 - 你的第一个MoFox_Bot插件"""

    # 以下是插件基本信息和方法(必须填写)
    plugin_name = "hello_world_plugin"
    enable_plugin = True  # 启用插件
    dependencies = []  # 插件依赖列表(目前为空)
    python_dependencies = []  # Python依赖列表目前为空
    config_file_name = "config.toml"  # 配置文件名
    config_schema = {}  # 配置文件模式(目前为空)

    def get_plugin_components(self) -> List[Tuple[ComponentInfo, Type]]: # 获取插件组件
        """返回插件包含的组件列表(目前是空的)"""
        return []

🎉 恭喜你刚刚创建了一个最简单但完整的MoFox_Bot插件

解释一下这些代码:

  • 首先,我们在plugin.py中定义了一个HelloWorldPlugin插件类继承自 BasePlugin ,提供基本功能。
  • 通过给类加上,@register_plugin 装饰器,我们告诉系统"这是一个插件"
  • plugin_name 等是插件的基本信息,必须填写
  • get_plugin_components() 返回插件的功能组件,现在我们没有定义任何 Action, Command 或者 EventHandler所以返回空列表。

4. 测试基础插件

现在就可以测试这个插件了启动MoFox_Bot

直接通过启动器运行MoFox_Bot或者 python bot.py

在日志中你应该能看到插件被加载的信息。虽然插件还没有任何功能,但它已经成功运行了!

1750326700269

5. 添加第一个功能问候Action

现在我们要给插件加入一个有用的功能我们从最好玩的Action做起

Action是一类可以让MoFox_Bot根据自身意愿选择使用的“动作”在MoFox_Bot中不论是“回复”还是“不回复”或者“发送表情”以及“禁言”等等都是通过Action实现的。

你可以通过编写动作来拓展MoFox_Bot的能力包括发送语音截图甚至操作文件编写代码......

现在让我们给插件添加第一个简单的功能。这个Action可以对用户发送一句问候语。

plugin.py 文件中添加Action组件完整代码如下

from typing import List, Tuple, Type
from src.plugin_system import (
    BasePlugin, register_plugin, BaseAction, 
    ComponentInfo, ActionActivationType, ChatMode
)

# ===== Action组件 =====

class HelloAction(BaseAction):
    """问候Action - 简单的问候动作"""

    # === 基本信息(必须填写)===
    action_name = "hello_greeting"
    action_description = "向用户发送问候消息"
    activation_type = ActionActivationType.ALWAYS  # 始终激活

    # === 功能描述(必须填写)===
    action_parameters = {"greeting_message": "要发送的问候消息"}
    action_require = ["需要发送友好问候时使用", "当有人向你问好时使用", "当你遇见没有见过的人时使用"]
    associated_types = ["text"]

    async def execute(self) -> Tuple[bool, str]:
        """执行问候动作 - 这是核心功能"""
        # 发送问候消息
        greeting_message = self.action_data.get("greeting_message", "")
        base_message = self.get_config("greeting.message", "嗨!很开心见到你!😊")
        message = base_message + greeting_message
        await self.send_text(message)

        return True, "发送了问候消息"

@register_plugin
class HelloWorldPlugin(BasePlugin):
    """Hello World插件 - 你的第一个MoFox_Bot插件"""

    # 插件基本信息
    plugin_name = "hello_world_plugin"
    enable_plugin = True
    dependencies = []
    python_dependencies = []
    config_file_name = "config.toml"
    config_schema = {}

    def get_plugin_components(self) -> List[Tuple[ComponentInfo, Type]]:
        """返回插件包含的组件列表"""
        return [
            # 添加我们的问候Action
            (HelloAction.get_action_info(), HelloAction),
        ]

解释一下这些代码:

  • HelloAction 是我们定义的问候动作类,继承自 BaseAction,并实现了核心功能。
  • HelloWorldPlugin 中,我们通过 get_plugin_components() 方法,通过调用get_action_info()这个内置方法将 HelloAction 注册为插件的一个组件。
  • 这样一来当插件被加载时问候动作也会被一并加载并可以在MoFox_Bot中使用。
  • execute() 函数是Action的核心定义了当Action被MoFox_Bot选择后具体要做什么
  • self.send_text() 是发送文本消息的便捷方法

Action 组件中有关activation_typeaction_parametersaction_requireassociated_types 等的详细说明请参考 Action组件指南

6. 测试问候Action

重启MoFox_Bot然后在聊天中发送任意消息比如

你好

MoFox_Bot可能会选择使用你的问候Action发送回复

嗨!很开心见到你!😊

1750332508760

💡 小提示MoFox_Bot会智能地决定什么时候使用它。如果没有立即看到效果多试几次不同的消息。

🎉 太棒了!你的插件已经有实际功能了!

7. 添加第二个功能时间查询Command

现在让我们添加一个Command组件。Command和Action不同它是直接响应用户命令的

Command是最简单最直接的响应不由LLM判断选择使用

# 在现有代码基础上添加Command组件
import datetime
from src.plugin_system import PlusCommand, CommandArgs
# 导入增强命令基类 - 推荐使用!

class TimeCommand(PlusCommand):
    """时间查询Command - 响应/time命令"""

    command_name = "time"
    command_description = "查询当前时间"

    # 注意:使用 PlusCommand 不需要 command_pattern会自动生成

    async def execute(self, args: CommandArgs) -> Tuple[bool, Optional[str], bool]:
        """执行时间查询
        
        Args:
            args: 命令参数(本例中不使用)
            
        Returns:
            (成功标志, 日志描述, 是否拦截消息)
        """
        # 获取当前时间
        time_format: str = "%Y-%m-%d %H:%M:%S"
        now = datetime.datetime.now()
        time_str = now.strftime(time_format)

        # 发送时间信息给用户
        await self.send_text(f"⏰ 当前时间:{time_str}")

        # 返回:成功、日志描述、拦截消息
        return True, f"显示了当前时间: {time_str}", True

@register_plugin
class HelloWorldPlugin(BasePlugin):
    """Hello World插件 - 你的第一个MoFox_Bot插件"""

    # 插件基本信息
    plugin_name = "hello_world_plugin"
    enable_plugin = True
    dependencies = []
    python_dependencies = []
    config_file_name = "config.toml"
    config_schema = {}

    def get_plugin_components(self) -> List[Tuple[ComponentInfo, Type]]:
        return [
            (HelloAction.get_action_info(), HelloAction),
            (TimeCommand.get_command_info(), TimeCommand),
        ]

同样的,我们通过 get_plugin_components() 方法,通过调用get_command_info()这个内置方法将 TimeCommand 注册为插件的一个组件。

Command组件解释

⚠️ 重要:请使用 PlusCommand 而不是 BaseCommand

  • PlusCommand:推荐使用,自动处理参数解析,无需编写正则表达式
  • BaseCommand:仅供框架内部使用,插件开发者不应直接使用

PlusCommand 的优势:

  • 无需编写 command_pattern 正则表达式
  • 自动解析命令参数(通过 CommandArgs
  • 支持命令别名(command_aliases
  • 更简单的 API更容易上手

execute() 方法说明:

  • 参数:args: CommandArgs - 包含解析后的命令参数
  • 返回值:(bool, str, bool) 三元组
    • bool:命令是否执行成功
    • str:日志描述(不是发给用户的消息
    • bool:是否拦截消息,阻止后续处理

有关增强命令的详细信息,请参考 增强命令指南

8. 测试时间查询Command

重启MoFox_Bot发送命令

/time

你应该会收到回复:

⏰ 当前时间2024-01-01 12:00:00

🎉 太棒了!现在你已经了解了基本的 Action 和 Command 组件的使用方法。你可以根据自己的需求,继续扩展插件的功能,添加更多的 Action 和 Command 组件,让你的插件更加丰富和强大!


进阶教程

如果你想让插件更加灵活和强大,可以参考接下来的进阶教程。

1. 添加配置文件

想要为插件添加配置文件吗?让我们一起来配置config_schema属性!

🚨 重要不要手动创建config.toml文件

我们需要在插件代码中定义配置Schema让系统自动生成配置文件。

首先在插件类中定义配置Schema

from src.plugin_system import ConfigField

@register_plugin
class HelloWorldPlugin(BasePlugin):
    """Hello World插件 - 你的第一个MaiCore插件"""

    # 插件基本信息
    plugin_name: str = "hello_world_plugin"  # 内部标识符
    enable_plugin: bool = True
    dependencies: List[str] = []  # 插件依赖列表
    python_dependencies: List[str] = []  # Python包依赖列表
    config_file_name: str = "config.toml"  # 配置文件名

    # 配置Schema定义
    config_schema: dict = {
        "plugin": {
            "name": ConfigField(type=str, default="hello_world_plugin", description="插件名称"),
            "version": ConfigField(type=str, default="1.0.0", description="插件版本"),
            "enabled": ConfigField(type=bool, default=False, description="是否启用插件"),
        },
        "greeting": {
            "message": ConfigField(type=str, default="嗨!很开心见到你!😊", description="默认问候消息"),
            "enable_emoji": ConfigField(type=bool, default=True, description="是否启用表情符号"),
        },
        "time": {"format": ConfigField(type=str, default="%Y-%m-%d %H:%M:%S", description="时间显示格式")},
    }

    def get_plugin_components(self) -> List[Tuple[ComponentInfo, Type]]:
        return [
            (HelloAction.get_action_info(), HelloAction),
            (TimeCommand.get_command_info(), TimeCommand),
        ]

这会生成一个如下的 config.toml 文件:

# hello_world_plugin - 自动生成的配置文件
# 我的第一个MoFox_Bot插件包含问候功能和时间查询等基础示例

# 插件基本信息
[plugin]

# 插件名称
name = "hello_world_plugin"

# 插件版本
version = "1.0.0"

# 是否启用插件
enabled = false


# 问候功能配置
[greeting]

# 默认问候消息
message = "嗨!很开心见到你!😊"

# 是否启用表情符号
enable_emoji = true


# 时间查询配置
[time]

# 时间显示格式
format = "%Y-%m-%d %H:%M:%S"

然后修改Action和Command代码通过 get_config() 方法让它们读取配置(配置的键是命名空间式的):

class HelloAction(BaseAction):
    """问候Action - 简单的问候动作"""

    # === 基本信息(必须填写)===
    action_name = "hello_greeting"
    action_description = "向用户发送问候消息"
    activation_type = ActionActivationType.ALWAYS  # 始终激活

    # === 功能描述(必须填写)===
    action_parameters = {"greeting_message": "要发送的问候消息"}
    action_require = ["需要发送友好问候时使用", "当有人向你问好时使用", "当你遇见没有见过的人时使用"]
    associated_types = ["text"]

    async def execute(self) -> Tuple[bool, str]:
        """执行问候动作 - 这是核心功能"""
        # 发送问候消息
        greeting_message = self.action_data.get("greeting_message", "")
        base_message = self.get_config("greeting.message", "嗨!很开心见到你!😊")
        message = base_message + greeting_message
        await self.send_text(message)

        return True, "发送了问候消息"

class TimeCommand(PlusCommand):
    """时间查询Command - 响应/time命令"""

    command_name = "time"
    command_description = "查询当前时间"

    # 注意PlusCommand 不需要 command_pattern

    async def execute(self, args: CommandArgs) -> Tuple[bool, str, bool]:
        """执行时间查询
        
        Args:
            args: 命令参数对象
        """
        import datetime

        # 从配置获取时间格式
        time_format: str = self.get_config("time.format", "%Y-%m-%d %H:%M:%S")  # type: ignore
        now = datetime.datetime.now()
        time_str = now.strftime(time_format)

        # 发送时间信息给用户
        await self.send_text(f"⏰ 当前时间:{time_str}")

        # 返回:成功、日志描述、拦截消息
        return True, f"显示了当前时间: {time_str}", True

配置系统工作流程:

  1. 定义Schema: 在插件代码中定义配置结构
  2. 自动生成: 启动插件时,系统会自动生成 config.toml 文件
  3. 用户修改: 用户可以修改生成的配置文件
  4. 代码读取: 使用 self.get_config() 读取配置值

绝对不要手动创建 config.toml 文件!

更详细的配置系统介绍请参考 配置指南

2. 创建说明文档

你可以创建一个 README.md 文件,描述插件的功能和使用方法。


🎉 恭喜你!你已经成功的创建了自己的插件了!