diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md
index f3389c25..71ef28fc 100644
--- a/.github/pull_request_template.md
+++ b/.github/pull_request_template.md
@@ -9,7 +9,7 @@
*请在方括号间写`x`以打勾 / Please tick the box with `x`*
-- [ ] 阅读仓库[贡献指引](https://github.com/RockChinQ/LangBot/blob/master/CONTRIBUTING.md)了吗? / Have you read the [contribution guide](https://github.com/RockChinQ/LangBot/blob/master/CONTRIBUTING.md)?
+- [ ] 阅读仓库[贡献指引](https://github.com/langbot-app/LangBot/blob/master/CONTRIBUTING.md)了吗? / Have you read the [contribution guide](https://github.com/langbot-app/LangBot/blob/master/CONTRIBUTING.md)?
- [ ] 与项目所有者沟通过了吗? / Have you communicated with the project maintainer?
- [ ] 我确定已自行测试所作的更改,确保功能符合预期。 / I have tested the changes and ensured they work as expected.
diff --git a/README.md b/README.md
index 6e0fa350..87c99a27 100644
--- a/README.md
+++ b/README.md
@@ -1,32 +1,25 @@
-
+
-

+简体中文 / [English](README_EN.md) / [日本語](README_JP.md) / (PR for your language)
+
+[](https://discord.gg/wdNEHETs87)
+[](https://qm.qq.com/q/JLi38whHum)
+[](https://deepwiki.com/langbot-app/LangBot)
+[](https://github.com/langbot-app/LangBot/releases/latest)
+

+[](https://gitcode.com/langbot-app/LangBot)
项目主页 |
部署文档 |
插件介绍 |
-
提交插件
+
提交插件
-
-😎高稳定、🧩支持扩展、🦄多模态 - 大模型原生即时通信机器人平台🤖
-
-
-
-
-[](https://discord.gg/wdNEHETs87)
-[](https://qm.qq.com/q/JLi38whHum)
-[](https://deepwiki.com/RockChinQ/LangBot)
-[](https://github.com/RockChinQ/LangBot/releases/latest)
-

-[](https://gitcode.com/RockChinQ/LangBot)
-
-简体中文 / [English](README_EN.md) / [日本語](README_JP.md) / (PR for your language)
@@ -44,7 +37,7 @@
#### Docker Compose 部署
```bash
-git clone https://github.com/RockChinQ/LangBot
+git clone https://github.com/langbot-app/LangBot
cd LangBot
docker compose up -d
```
@@ -149,10 +142,10 @@ docker compose up -d
## 😘 社区贡献
-感谢以下[代码贡献者](https://github.com/RockChinQ/LangBot/graphs/contributors)和社区里其他成员对 LangBot 的贡献:
+感谢以下[代码贡献者](https://github.com/langbot-app/LangBot/graphs/contributors)和社区里其他成员对 LangBot 的贡献:
-
-
+
+
## 😎 保持更新
diff --git a/README_EN.md b/README_EN.md
index 07667f84..114bc8c4 100644
--- a/README_EN.md
+++ b/README_EN.md
@@ -1,30 +1,21 @@
-
+
-

+[简体中文](README.md) / English / [日本語](README_JP.md) / (PR for your language)
+
+[](https://discord.gg/wdNEHETs87)
+[](https://deepwiki.com/langbot-app/LangBot)
+[](https://github.com/langbot-app/LangBot/releases/latest)
+
Home |
Deployment |
Plugin |
-
Submit Plugin
-
-
-😎High Stability, 🧩Extension Supported, 🦄Multi-modal - LLM Native Instant Messaging Bot Platform🤖
-
-
-
-
-
-[](https://discord.gg/wdNEHETs87)
-[](https://deepwiki.com/RockChinQ/LangBot)
-[](https://github.com/RockChinQ/LangBot/releases/latest)
-

-
-[简体中文](README.md) / English / [日本語](README_JP.md) / (PR for your language)
+
Submit Plugin
@@ -42,7 +33,7 @@
#### Docker Compose Deployment
```bash
-git clone https://github.com/RockChinQ/LangBot
+git clone https://github.com/langbot-app/LangBot
cd LangBot
docker compose up -d
```
@@ -132,10 +123,10 @@ Directly use the released version to run, see the [Manual Deployment](https://do
## 🤝 Community Contribution
-Thank you for the following [code contributors](https://github.com/RockChinQ/LangBot/graphs/contributors) and other members in the community for their contributions to LangBot:
+Thank you for the following [code contributors](https://github.com/langbot-app/LangBot/graphs/contributors) and other members in the community for their contributions to LangBot:
-
-
+
+
## 😎 Stay Ahead
diff --git a/README_JP.md b/README_JP.md
index c54ce51b..d4a372ec 100644
--- a/README_JP.md
+++ b/README_JP.md
@@ -1,29 +1,21 @@
-
+
-

+[简体中文](README.md) / [English](README_EN.md) / 日本語 / (PR for your language)
+
+[](https://discord.gg/wdNEHETs87)
+[](https://deepwiki.com/langbot-app/LangBot)
+[](https://github.com/langbot-app/LangBot/releases/latest)
+
ホーム |
デプロイ |
プラグイン |
-
プラグインの提出
-
-
-😎高い安定性、🧩拡張サポート、🦄マルチモーダル - LLMネイティブインスタントメッセージングボットプラットフォーム🤖
-
-
-
-
-[](https://discord.gg/wdNEHETs87)
-[](https://deepwiki.com/RockChinQ/LangBot)
-[](https://github.com/RockChinQ/LangBot/releases/latest)
-

-
-[简体中文](README_CN.md) / [English](README.md) / [日本語](README_JP.md) / (PR for your language)
+
プラグインの提出
@@ -41,7 +33,7 @@
#### Docker Compose デプロイ
```bash
-git clone https://github.com/RockChinQ/LangBot
+git clone https://github.com/langbot-app/LangBot
cd LangBot
docker compose up -d
```
@@ -131,10 +123,10 @@ LangBotはBTPanelにリストされています。BTPanelをインストール
## 🤝 コミュニティ貢献
-LangBot への貢献に対して、以下の [コード貢献者](https://github.com/RockChinQ/LangBot/graphs/contributors) とコミュニティの他のメンバーに感謝します。
+LangBot への貢献に対して、以下の [コード貢献者](https://github.com/langbot-app/LangBot/graphs/contributors) とコミュニティの他のメンバーに感謝します。
-
-
+
+
## 😎 最新情報を入手
diff --git a/main.py b/main.py
index 19cb32d6..1909e343 100644
--- a/main.py
+++ b/main.py
@@ -11,7 +11,7 @@ asciiart = r"""
|____\__,_|_||_\__, |___/\___/\__|
|___/
-⭐️ Open Source 开源地址: https://github.com/RockChinQ/LangBot
+⭐️ Open Source 开源地址: https://github.com/langbot-app/LangBot
📖 Documentation 文档地址: https://docs.langbot.app
"""
diff --git a/pkg/api/http/controller/group.py b/pkg/api/http/controller/group.py
index 2088ecc1..16fa1df1 100644
--- a/pkg/api/http/controller/group.py
+++ b/pkg/api/http/controller/group.py
@@ -11,7 +11,7 @@ from ....core import app
preregistered_groups: list[type[RouterGroup]] = []
-"""RouterGroup 的预注册列表"""
+"""Pre-registered list of RouterGroup"""
def group_class(name: str, path: str) -> typing.Callable[[typing.Type[RouterGroup]], typing.Type[RouterGroup]]:
@@ -27,7 +27,7 @@ def group_class(name: str, path: str) -> typing.Callable[[typing.Type[RouterGrou
class AuthType(enum.Enum):
- """认证类型"""
+ """Authentication type"""
NONE = 'none'
USER_TOKEN = 'user-token'
@@ -56,7 +56,7 @@ class RouterGroup(abc.ABC):
auth_type: AuthType = AuthType.USER_TOKEN,
**options: typing.Any,
) -> typing.Callable[[RouteCallable], RouteCallable]: # decorator
- """注册一个路由"""
+ """Register a route"""
def decorator(f: RouteCallable) -> RouteCallable:
nonlocal rule
@@ -64,11 +64,11 @@ class RouterGroup(abc.ABC):
async def handler_error(*args, **kwargs):
if auth_type == AuthType.USER_TOKEN:
- # 从Authorization头中获取token
+ # get token from Authorization header
token = quart.request.headers.get('Authorization', '').replace('Bearer ', '')
if not token:
- return self.http_status(401, -1, '未提供有效的用户令牌')
+ return self.http_status(401, -1, 'No valid user token provided')
try:
user_email = await self.ap.user_service.verify_jwt_token(token)
@@ -76,9 +76,9 @@ class RouterGroup(abc.ABC):
# check if this account exists
user = await self.ap.user_service.get_user_by_email(user_email)
if not user:
- return self.http_status(401, -1, '用户不存在')
+ return self.http_status(401, -1, 'User not found')
- # 检查f是否接受user_email参数
+ # check if f accepts user_email parameter
if 'user_email' in f.__code__.co_varnames:
kwargs['user_email'] = user_email
except Exception as e:
@@ -86,6 +86,7 @@ class RouterGroup(abc.ABC):
try:
return await f(*args, **kwargs)
+
except Exception as e: # 自动 500
traceback.print_exc()
# return self.http_status(500, -2, str(e))
@@ -101,7 +102,7 @@ class RouterGroup(abc.ABC):
return decorator
def success(self, data: typing.Any = None) -> quart.Response:
- """返回一个 200 响应"""
+ """Return a 200 response"""
return quart.jsonify(
{
'code': 0,
@@ -111,7 +112,7 @@ class RouterGroup(abc.ABC):
)
def fail(self, code: int, msg: str) -> quart.Response:
- """返回一个异常响应"""
+ """Return an error response"""
return quart.jsonify(
{
@@ -122,4 +123,4 @@ class RouterGroup(abc.ABC):
def http_status(self, status: int, code: int, msg: str) -> typing.Tuple[quart.Response, int]:
"""返回一个指定状态码的响应"""
- return (self.fail(code, msg), status)
+ return (self.fail(code, msg), status)
\ No newline at end of file
diff --git a/pkg/api/http/controller/groups/pipelines/webchat.py b/pkg/api/http/controller/groups/pipelines/webchat.py
index 005738db..c8c8db54 100644
--- a/pkg/api/http/controller/groups/pipelines/webchat.py
+++ b/pkg/api/http/controller/groups/pipelines/webchat.py
@@ -8,7 +8,7 @@ class WebChatDebugRouterGroup(group.RouterGroup):
async def initialize(self) -> None:
@self.route('/send', methods=['POST'])
async def send_message(pipeline_uuid: str) -> str:
- """发送调试消息到流水线"""
+ """Send a message to the pipeline for debugging"""
try:
data = await quart.request.get_json()
session_type = data.get('session_type', 'person')
@@ -38,7 +38,7 @@ class WebChatDebugRouterGroup(group.RouterGroup):
@self.route('/messages/', methods=['GET'])
async def get_messages(pipeline_uuid: str, session_type: str) -> str:
- """获取调试消息历史"""
+ """Get the message history of the pipeline for debugging"""
try:
if session_type not in ['person', 'group']:
return self.http_status(400, -1, 'session_type must be person or group')
@@ -57,7 +57,7 @@ class WebChatDebugRouterGroup(group.RouterGroup):
@self.route('/reset/', methods=['POST'])
async def reset_session(session_type: str) -> str:
- """重置调试会话"""
+ """Reset the debug session"""
try:
if session_type not in ['person', 'group']:
return self.http_status(400, -1, 'session_type must be person or group')
diff --git a/pkg/api/http/controller/groups/plugins.py b/pkg/api/http/controller/groups/plugins.py
index daf6ea7d..b7e0a5e9 100644
--- a/pkg/api/http/controller/groups/plugins.py
+++ b/pkg/api/http/controller/groups/plugins.py
@@ -40,7 +40,7 @@ class PluginsRouterGroup(group.RouterGroup):
self.ap.plugin_mgr.update_plugin(plugin_name, task_context=ctx),
kind='plugin-operation',
name=f'plugin-update-{plugin_name}',
- label=f'更新插件 {plugin_name}',
+ label=f'Updating plugin {plugin_name}',
context=ctx,
)
return self.success(data={'task_id': wrapper.id})
@@ -62,7 +62,7 @@ class PluginsRouterGroup(group.RouterGroup):
self.ap.plugin_mgr.uninstall_plugin(plugin_name, task_context=ctx),
kind='plugin-operation',
name=f'plugin-remove-{plugin_name}',
- label=f'删除插件 {plugin_name}',
+ label=f'Removing plugin {plugin_name}',
context=ctx,
)
@@ -102,7 +102,7 @@ class PluginsRouterGroup(group.RouterGroup):
self.ap.plugin_mgr.install_plugin(data['source'], task_context=ctx),
kind='plugin-operation',
name='plugin-install-github',
- label=f'安装插件 ...{short_source_str}',
+ label=f'Installing plugin ...{short_source_str}',
context=ctx,
)
diff --git a/pkg/api/http/controller/groups/user.py b/pkg/api/http/controller/groups/user.py
index 3ad1335b..d8024107 100644
--- a/pkg/api/http/controller/groups/user.py
+++ b/pkg/api/http/controller/groups/user.py
@@ -14,7 +14,7 @@ class UserRouterGroup(group.RouterGroup):
return self.success(data={'initialized': await self.ap.user_service.is_initialized()})
if await self.ap.user_service.is_initialized():
- return self.fail(1, '系统已初始化')
+ return self.fail(1, 'System already initialized')
json_data = await quart.request.json
@@ -32,7 +32,7 @@ class UserRouterGroup(group.RouterGroup):
try:
token = await self.ap.user_service.authenticate(json_data['user'], json_data['password'])
except argon2.exceptions.VerifyMismatchError:
- return self.fail(1, '用户名或密码错误')
+ return self.fail(1, 'Invalid username or password')
return self.success(data={'token': token})
@@ -54,15 +54,15 @@ class UserRouterGroup(group.RouterGroup):
await asyncio.sleep(3)
if not await self.ap.user_service.is_initialized():
- return self.http_status(400, -1, 'system not initialized')
+ return self.http_status(400, -1, 'System not initialized')
user_obj = await self.ap.user_service.get_user_by_email(user_email)
if user_obj is None:
- return self.http_status(400, -1, 'user not found')
+ return self.http_status(400, -1, 'User not found')
if recovery_key != self.ap.instance_config.data['system']['recovery_key']:
- return self.http_status(403, -1, 'invalid recovery key')
+ return self.http_status(403, -1, 'Invalid recovery key')
await self.ap.user_service.reset_password(user_email, new_password)
diff --git a/pkg/api/http/controller/main.py b/pkg/api/http/controller/main.py
index 4eec4e1d..e45b461d 100644
--- a/pkg/api/http/controller/main.py
+++ b/pkg/api/http/controller/main.py
@@ -47,7 +47,7 @@ class HTTPController:
try:
await self.quart_app.run_task(*args, **kwargs)
except Exception as e:
- self.ap.logger.error(f'启动 HTTP 服务失败: {e}')
+ self.ap.logger.error(f'Failed to start HTTP service: {e}')
self.ap.task_mgr.create_task(
exception_handler(
diff --git a/pkg/api/http/service/bot.py b/pkg/api/http/service/bot.py
index e5010007..adf19d03 100644
--- a/pkg/api/http/service/bot.py
+++ b/pkg/api/http/service/bot.py
@@ -10,7 +10,7 @@ from ....entity.persistence import pipeline as persistence_pipeline
class BotService:
- """机器人服务"""
+ """Bot service"""
ap: app.Application
@@ -18,7 +18,7 @@ class BotService:
self.ap = ap
async def get_bots(self) -> list[dict]:
- """获取所有机器人"""
+ """Get all bots"""
result = await self.ap.persistence_mgr.execute_async(sqlalchemy.select(persistence_bot.Bot))
bots = result.all()
@@ -26,7 +26,7 @@ class BotService:
return [self.ap.persistence_mgr.serialize_model(persistence_bot.Bot, bot) for bot in bots]
async def get_bot(self, bot_uuid: str) -> dict | None:
- """获取机器人"""
+ """Get bot"""
result = await self.ap.persistence_mgr.execute_async(
sqlalchemy.select(persistence_bot.Bot).where(persistence_bot.Bot.uuid == bot_uuid)
)
@@ -39,7 +39,7 @@ class BotService:
return self.ap.persistence_mgr.serialize_model(persistence_bot.Bot, bot)
async def create_bot(self, bot_data: dict) -> str:
- """创建机器人"""
+ """Create bot"""
# TODO: 检查配置信息格式
bot_data['uuid'] = str(uuid.uuid4())
@@ -63,7 +63,7 @@ class BotService:
return bot_data['uuid']
async def update_bot(self, bot_uuid: str, bot_data: dict) -> None:
- """更新机器人"""
+ """Update bot"""
if 'uuid' in bot_data:
del bot_data['uuid']
@@ -99,7 +99,7 @@ class BotService:
session.using_conversation = None
async def delete_bot(self, bot_uuid: str) -> None:
- """删除机器人"""
+ """Delete bot"""
await self.ap.platform_mgr.remove_bot(bot_uuid)
await self.ap.persistence_mgr.execute_async(
sqlalchemy.delete(persistence_bot.Bot).where(persistence_bot.Bot.uuid == bot_uuid)
diff --git a/pkg/config/impls/json.py b/pkg/config/impls/json.py
index 07fc533c..44b4843c 100644
--- a/pkg/config/impls/json.py
+++ b/pkg/config/impls/json.py
@@ -6,7 +6,7 @@ from .. import model as file_model
class JSONConfigFile(file_model.ConfigFile):
- """JSON配置文件"""
+ """JSON config file"""
def __init__(
self,
@@ -42,7 +42,7 @@ class JSONConfigFile(file_model.ConfigFile):
try:
cfg = json.load(f)
except json.JSONDecodeError as e:
- raise Exception(f'配置文件 {self.config_file_name} 语法错误: {e}')
+ raise Exception(f'Syntax error in config file {self.config_file_name}: {e}')
if completion:
for key in self.template_data:
diff --git a/pkg/config/impls/pymodule.py b/pkg/config/impls/pymodule.py
index 2311992e..c3d04bc8 100644
--- a/pkg/config/impls/pymodule.py
+++ b/pkg/config/impls/pymodule.py
@@ -7,13 +7,13 @@ from .. import model as file_model
class PythonModuleConfigFile(file_model.ConfigFile):
- """Python模块配置文件"""
+ """Python module config file"""
config_file_name: str = None
- """配置文件名"""
+ """Config file name"""
template_file_name: str = None
- """模板文件名"""
+ """Template file name"""
def __init__(self, config_file_name: str, template_file_name: str) -> None:
self.config_file_name = config_file_name
@@ -42,7 +42,7 @@ class PythonModuleConfigFile(file_model.ConfigFile):
cfg[key] = getattr(module, key)
- # 从模板模块文件中进行补全
+ # complete from template module file
if completion:
module_name = os.path.splitext(os.path.basename(self.template_file_name))[0]
module = importlib.import_module(module_name)
@@ -60,7 +60,7 @@ class PythonModuleConfigFile(file_model.ConfigFile):
return cfg
async def save(self, data: dict):
- logging.warning('Python模块配置文件不支持保存')
+ logging.warning('Python module config file does not support saving')
def save_sync(self, data: dict):
- logging.warning('Python模块配置文件不支持保存')
+ logging.warning('Python module config file does not support saving')
diff --git a/pkg/config/impls/yaml.py b/pkg/config/impls/yaml.py
index 55045186..0d69ef9e 100644
--- a/pkg/config/impls/yaml.py
+++ b/pkg/config/impls/yaml.py
@@ -6,7 +6,7 @@ from .. import model as file_model
class YAMLConfigFile(file_model.ConfigFile):
- """YAML配置文件"""
+ """YAML config file"""
def __init__(
self,
@@ -42,7 +42,7 @@ class YAMLConfigFile(file_model.ConfigFile):
try:
cfg = yaml.load(f, Loader=yaml.FullLoader)
except yaml.YAMLError as e:
- raise Exception(f'配置文件 {self.config_file_name} 语法错误: {e}')
+ raise Exception(f'Syntax error in config file {self.config_file_name}: {e}')
if completion:
for key in self.template_data:
diff --git a/pkg/config/manager.py b/pkg/config/manager.py
index c2e6bdf4..d552b038 100644
--- a/pkg/config/manager.py
+++ b/pkg/config/manager.py
@@ -5,27 +5,27 @@ from .impls import pymodule, json as json_file, yaml as yaml_file
class ConfigManager:
- """配置文件管理器"""
+ """Config file manager"""
name: str = None
- """配置管理器名"""
+ """Config manager name"""
description: str = None
- """配置管理器描述"""
+ """Config manager description"""
schema: dict = None
- """配置文件 schema
- 需要符合 JSON Schema Draft 7 规范
+ """Config file schema
+ Must conform to JSON Schema Draft 7 specification
"""
file: file_model.ConfigFile = None
- """配置文件实例"""
+ """Config file instance"""
data: dict = None
- """配置数据"""
+ """Config data"""
doc_link: str = None
- """配置文件文档链接"""
+ """Config file documentation link"""
def __init__(self, cfg_file: file_model.ConfigFile) -> None:
self.file = cfg_file
@@ -42,15 +42,15 @@ class ConfigManager:
async def load_python_module_config(config_name: str, template_name: str, completion: bool = True) -> ConfigManager:
- """加载Python模块配置文件
+ """Load Python module config file
Args:
- config_name (str): 配置文件名
- template_name (str): 模板文件名
- completion (bool): 是否自动补全内存中的配置文件
+ config_name (str): Config file name
+ template_name (str): Template file name
+ completion (bool): Whether to automatically complete the config file in memory
Returns:
- ConfigManager: 配置文件管理器
+ ConfigManager: Config file manager
"""
cfg_inst = pymodule.PythonModuleConfigFile(config_name, template_name)
@@ -66,13 +66,13 @@ async def load_json_config(
template_data: dict = None,
completion: bool = True,
) -> ConfigManager:
- """加载JSON配置文件
+ """Load JSON config file
Args:
- config_name (str): 配置文件名
- template_name (str): 模板文件名
- template_data (dict): 模板数据
- completion (bool): 是否自动补全内存中的配置文件
+ config_name (str): Config file name
+ template_name (str): Template file name
+ template_data (dict): Template data
+ completion (bool): Whether to automatically complete the config file in memory
"""
cfg_inst = json_file.JSONConfigFile(config_name, template_name, template_data)
@@ -88,16 +88,16 @@ async def load_yaml_config(
template_data: dict = None,
completion: bool = True,
) -> ConfigManager:
- """加载YAML配置文件
+ """Load YAML config file
Args:
- config_name (str): 配置文件名
- template_name (str): 模板文件名
- template_data (dict): 模板数据
- completion (bool): 是否自动补全内存中的配置文件
+ config_name (str): Config file name
+ template_name (str): Template file name
+ template_data (dict): Template data
+ completion (bool): Whether to automatically complete the config file in memory
Returns:
- ConfigManager: 配置文件管理器
+ ConfigManager: Config file manager
"""
cfg_inst = yaml_file.YAMLConfigFile(config_name, template_name, template_data)
diff --git a/pkg/config/model.py b/pkg/config/model.py
index f3536804..8b040f05 100644
--- a/pkg/config/model.py
+++ b/pkg/config/model.py
@@ -2,16 +2,16 @@ import abc
class ConfigFile(metaclass=abc.ABCMeta):
- """配置文件抽象类"""
+ """Config file abstract class"""
config_file_name: str = None
- """配置文件名"""
+ """Config file name"""
template_file_name: str = None
- """模板文件名"""
+ """Template file name"""
template_data: dict = None
- """模板数据"""
+ """Template data"""
@abc.abstractmethod
def exists(self) -> bool:
diff --git a/pkg/core/app.py b/pkg/core/app.py
index f0c30aee..21816cfc 100644
--- a/pkg/core/app.py
+++ b/pkg/core/app.py
@@ -33,7 +33,7 @@ from ..vector import mgr as vectordb_mgr
class Application:
- """运行时应用对象和上下文"""
+ """Runtime application object and context"""
event_loop: asyncio.AbstractEventLoop = None
@@ -52,10 +52,10 @@ class Application:
rag_mgr: rag_mgr.RAGManager = None
- # TODO 移动到 pipeline 里
+ # TODO move to pipeline
tool_mgr: llm_tool_mgr.ToolManager = None
- # ======= 配置管理器 =======
+ # ======= Config manager =======
command_cfg: config_mgr.ConfigManager = None # deprecated
@@ -69,7 +69,7 @@ class Application:
instance_config: config_mgr.ConfigManager = None
- # ======= 元数据配置管理器 =======
+ # ======= Metadata config manager =======
sensitive_meta: config_mgr.ConfigManager = None
@@ -166,11 +166,11 @@ class Application:
except asyncio.CancelledError:
pass
except Exception as e:
- self.logger.error(f'应用运行致命异常: {e}')
+ self.logger.error(f'Application runtime fatal exception: {e}')
self.logger.debug(f'Traceback: {traceback.format_exc()}')
async def print_web_access_info(self):
- """打印访问 webui 的提示"""
+ """Print access webui tips"""
if not os.path.exists(os.path.join('.', 'web/out')):
self.logger.warning('WebUI 文件缺失,请根据文档部署:https://docs.langbot.app/zh')
@@ -202,7 +202,7 @@ class Application:
):
match scope:
case core_entities.LifecycleControlScope.PLATFORM.value:
- self.logger.info('执行热重载 scope=' + scope)
+ self.logger.info('Hot reload scope=' + scope)
await self.platform_mgr.shutdown()
self.platform_mgr = im_mgr.PlatformManager(self)
@@ -218,7 +218,7 @@ class Application:
],
)
case core_entities.LifecycleControlScope.PLUGIN.value:
- self.logger.info('执行热重载 scope=' + scope)
+ self.logger.info('Hot reload scope=' + scope)
await self.plugin_mgr.destroy_plugins()
# 删除 sys.module 中所有的 plugins/* 下的模块
@@ -234,7 +234,7 @@ class Application:
await self.plugin_mgr.load_plugins()
await self.plugin_mgr.initialize_plugins()
case core_entities.LifecycleControlScope.PROVIDER.value:
- self.logger.info('执行热重载 scope=' + scope)
+ self.logger.info('Hot reload scope=' + scope)
await self.tool_mgr.shutdown()
diff --git a/pkg/core/boot.py b/pkg/core/boot.py
index aff117e6..b8243d4a 100644
--- a/pkg/core/boot.py
+++ b/pkg/core/boot.py
@@ -8,7 +8,7 @@ from . import app
from . import stage
from ..utils import constants, importutil
-# 引入启动阶段实现以便注册
+# Import startup stage implementation to register
from . import stages
importutil.import_modules_in_pkg(stages)
@@ -25,7 +25,7 @@ stage_order = [
async def make_app(loop: asyncio.AbstractEventLoop) -> app.Application:
- # 确定是否为调试模式
+ # Determine if it is debug mode
if 'DEBUG' in os.environ and os.environ['DEBUG'] in ['true', '1']:
constants.debug_mode = True
@@ -33,7 +33,7 @@ async def make_app(loop: asyncio.AbstractEventLoop) -> app.Application:
ap.event_loop = loop
- # 执行启动阶段
+ # Execute startup stage
for stage_name in stage_order:
stage_cls = stage.preregistered_stages[stage_name]
stage_inst = stage_cls()
@@ -47,11 +47,11 @@ async def make_app(loop: asyncio.AbstractEventLoop) -> app.Application:
async def main(loop: asyncio.AbstractEventLoop):
try:
- # 挂系统信号处理
+ # Hang system signal processing
import signal
def signal_handler(sig, frame):
- print('[Signal] 程序退出.')
+ print('[Signal] Program exit.')
# ap.shutdown()
os._exit(0)
diff --git a/pkg/core/bootutils/deps.py b/pkg/core/bootutils/deps.py
index b403bf8d..1a439af8 100644
--- a/pkg/core/bootutils/deps.py
+++ b/pkg/core/bootutils/deps.py
@@ -2,8 +2,8 @@ import pip
import os
from ...utils import pkgmgr
-# 检查依赖,防止用户未安装
-# 左边为引入名称,右边为依赖名称
+# Check dependencies to prevent users from not installing
+# Left is the import name, right is the dependency name
required_deps = {
'requests': 'requests',
'openai': 'openai',
@@ -65,7 +65,7 @@ async def install_deps(deps: list[str]):
async def precheck_plugin_deps():
print('[Startup] Prechecking plugin dependencies...')
- # 只有在plugins目录存在时才执行插件依赖安装
+ # Only execute plugin dependency installation when the plugins directory exists
if os.path.exists('plugins'):
for dir in os.listdir('plugins'):
subdir = os.path.join('plugins', dir)
diff --git a/pkg/core/bootutils/log.py b/pkg/core/bootutils/log.py
index eb6806fa..631b05e2 100644
--- a/pkg/core/bootutils/log.py
+++ b/pkg/core/bootutils/log.py
@@ -17,7 +17,7 @@ log_colors_config = {
async def init_logging(extra_handlers: list[logging.Handler] = None) -> logging.Logger:
- # 删除所有现有的logger
+ # Remove all existing loggers
for handler in logging.root.handlers[:]:
logging.root.removeHandler(handler)
@@ -54,13 +54,13 @@ async def init_logging(extra_handlers: list[logging.Handler] = None) -> logging.
handler.setFormatter(color_formatter)
qcg_logger.addHandler(handler)
- qcg_logger.debug('日志初始化完成,日志级别:%s' % level)
+ qcg_logger.debug('Logging initialized, log level: %s' % level)
logging.basicConfig(
- level=logging.CRITICAL, # 设置日志输出格式
+ level=logging.CRITICAL, # Set log output format
format='[DEPR][%(asctime)s.%(msecs)03d] %(pathname)s (%(lineno)d) - [%(levelname)s] :\n%(message)s',
- # 日志输出的格式
- # -8表示占位符,让输出左对齐,输出长度都为8位
- datefmt='%Y-%m-%d %H:%M:%S', # 时间输出的格式
+ # Log output format
+ # -8 is a placeholder, left-align the output, and output length is 8
+ datefmt='%Y-%m-%d %H:%M:%S', # Time output format
handlers=[logging.NullHandler()],
)
diff --git a/pkg/core/migration.py b/pkg/core/migration.py
index e97c0cf3..a921e6c7 100644
--- a/pkg/core/migration.py
+++ b/pkg/core/migration.py
@@ -7,11 +7,11 @@ from . import app
preregistered_migrations: list[typing.Type[Migration]] = []
-"""当前阶段暂不支持扩展"""
+"""Currently not supported for extension"""
def migration_class(name: str, number: int):
- """注册一个迁移"""
+ """Register a migration"""
def decorator(cls: typing.Type[Migration]) -> typing.Type[Migration]:
cls.name = name
@@ -23,7 +23,7 @@ def migration_class(name: str, number: int):
class Migration(abc.ABC):
- """一个版本的迁移"""
+ """A version migration"""
name: str
@@ -36,10 +36,10 @@ class Migration(abc.ABC):
@abc.abstractmethod
async def need_migrate(self) -> bool:
- """判断当前环境是否需要运行此迁移"""
+ """Determine if the current environment needs to run this migration"""
pass
@abc.abstractmethod
async def run(self):
- """执行迁移"""
+ """Run migration"""
pass
diff --git a/pkg/core/note.py b/pkg/core/note.py
index 07171581..b4c37ce1 100644
--- a/pkg/core/note.py
+++ b/pkg/core/note.py
@@ -9,7 +9,7 @@ preregistered_notes: list[typing.Type[LaunchNote]] = []
def note_class(name: str, number: int):
- """注册一个启动信息"""
+ """Register a launch information"""
def decorator(cls: typing.Type[LaunchNote]) -> typing.Type[LaunchNote]:
cls.name = name
@@ -21,7 +21,7 @@ def note_class(name: str, number: int):
class LaunchNote(abc.ABC):
- """启动信息"""
+ """Launch information"""
name: str
@@ -34,10 +34,10 @@ class LaunchNote(abc.ABC):
@abc.abstractmethod
async def need_show(self) -> bool:
- """判断当前环境是否需要显示此启动信息"""
+ """Determine if the current environment needs to display this launch information"""
pass
@abc.abstractmethod
async def yield_note(self) -> typing.AsyncGenerator[typing.Tuple[str, int], None]:
- """生成启动信息"""
+ """Generate launch information"""
pass
diff --git a/pkg/core/notes/n001_classic_msgs.py b/pkg/core/notes/n001_classic_msgs.py
index 3f3bd8e0..265ddbe9 100644
--- a/pkg/core/notes/n001_classic_msgs.py
+++ b/pkg/core/notes/n001_classic_msgs.py
@@ -7,7 +7,7 @@ from .. import note
@note.note_class('ClassicNotes', 1)
class ClassicNotes(note.LaunchNote):
- """经典启动信息"""
+ """Classic launch information"""
async def need_show(self) -> bool:
return True
diff --git a/pkg/core/notes/n002_selection_mode_on_windows.py b/pkg/core/notes/n002_selection_mode_on_windows.py
index 23bff24a..16028de1 100644
--- a/pkg/core/notes/n002_selection_mode_on_windows.py
+++ b/pkg/core/notes/n002_selection_mode_on_windows.py
@@ -9,7 +9,7 @@ from .. import note
@note.note_class('SelectionModeOnWindows', 2)
class SelectionModeOnWindows(note.LaunchNote):
- """Windows 上的选择模式提示信息"""
+ """Selection mode prompt information on Windows"""
async def need_show(self) -> bool:
return os.name == 'nt'
@@ -19,3 +19,8 @@ class SelectionModeOnWindows(note.LaunchNote):
"""您正在使用 Windows 系统,若窗口左上角显示处于”选择“模式,程序将被暂停运行,此时请右键窗口中空白区域退出选择模式。""",
logging.INFO,
)
+
+ yield (
+ """You are using Windows system, if the top left corner of the window displays "Selection" mode, the program will be paused running, please right-click on the blank area in the window to exit the selection mode.""",
+ logging.INFO,
+ )
diff --git a/pkg/core/stage.py b/pkg/core/stage.py
index 220c474d..1483e23a 100644
--- a/pkg/core/stage.py
+++ b/pkg/core/stage.py
@@ -7,9 +7,9 @@ from . import app
preregistered_stages: dict[str, typing.Type[BootingStage]] = {}
-"""预注册的请求处理阶段。在初始化时,所有请求处理阶段类会被注册到此字典中。
+"""Pre-registered request processing stages. All request processing stage classes are registered in this dictionary during initialization.
-当前阶段暂不支持扩展
+Currently not supported for extension
"""
@@ -22,11 +22,11 @@ def stage_class(name: str):
class BootingStage(abc.ABC):
- """启动阶段"""
+ """Booting stage"""
name: str = None
@abc.abstractmethod
async def run(self, ap: app.Application):
- """启动"""
+ """Run"""
pass
diff --git a/pkg/core/stages/build_app.py b/pkg/core/stages/build_app.py
index e48fdd99..0f28f0c8 100644
--- a/pkg/core/stages/build_app.py
+++ b/pkg/core/stages/build_app.py
@@ -27,10 +27,10 @@ from .. import taskmgr
@stage.stage_class('BuildAppStage')
class BuildAppStage(stage.BootingStage):
- """构建应用阶段"""
+ """Build LangBot application"""
async def run(self, ap: app.Application):
- """构建app对象的各个组件对象并初始化"""
+ """Build LangBot application"""
ap.task_mgr = taskmgr.AsyncTaskManager(ap)
discover = discover_engine.ComponentDiscoveryEngine(ap)
@@ -45,7 +45,7 @@ class BuildAppStage(stage.BootingStage):
await ver_mgr.initialize()
ap.ver_mgr = ver_mgr
- # 发送公告
+ # Send announcement
ann_mgr = announce.AnnouncementManager(ap)
ap.ann_mgr = ann_mgr
diff --git a/pkg/core/stages/genkeys.py b/pkg/core/stages/genkeys.py
index 50e7cf7b..f0412b9d 100644
--- a/pkg/core/stages/genkeys.py
+++ b/pkg/core/stages/genkeys.py
@@ -7,10 +7,10 @@ from .. import stage, app
@stage.stage_class('GenKeysStage')
class GenKeysStage(stage.BootingStage):
- """生成密钥阶段"""
+ """Generate keys stage"""
async def run(self, ap: app.Application):
- """启动"""
+ """Generate keys"""
if not ap.instance_config.data['system']['jwt']['secret']:
ap.instance_config.data['system']['jwt']['secret'] = secrets.token_hex(16)
diff --git a/pkg/core/stages/load_config.py b/pkg/core/stages/load_config.py
index ef5f611b..0474b33a 100644
--- a/pkg/core/stages/load_config.py
+++ b/pkg/core/stages/load_config.py
@@ -8,10 +8,10 @@ from ..bootutils import config
@stage.stage_class('LoadConfigStage')
class LoadConfigStage(stage.BootingStage):
- """加载配置文件阶段"""
+ """Load config file stage"""
async def run(self, ap: app.Application):
- """启动"""
+ """Load config file"""
# ======= deprecated =======
if os.path.exists('data/config/command.json'):
diff --git a/pkg/core/stages/migrate.py b/pkg/core/stages/migrate.py
index 02b03256..229e0060 100644
--- a/pkg/core/stages/migrate.py
+++ b/pkg/core/stages/migrate.py
@@ -11,10 +11,13 @@ importutil.import_modules_in_pkg(migrations)
@stage.stage_class('MigrationStage')
class MigrationStage(stage.BootingStage):
- """迁移阶段"""
+ """Migration stage
+
+ These migrations are legacy, only performed in version 3.x
+ """
async def run(self, ap: app.Application):
- """启动"""
+ """Run migration"""
if any(
[
@@ -29,7 +32,7 @@ class MigrationStage(stage.BootingStage):
migrations = migration.preregistered_migrations
- # 按照迁移号排序
+ # Sort by migration number
migrations.sort(key=lambda x: x.number)
for migration_cls in migrations:
@@ -37,4 +40,4 @@ class MigrationStage(stage.BootingStage):
if await migration_instance.need_migrate():
await migration_instance.run()
- print(f'已执行迁移 {migration_instance.name}')
+ print(f'Migration {migration_instance.name} executed')
diff --git a/pkg/core/stages/setup_logger.py b/pkg/core/stages/setup_logger.py
index 0c630175..1f7c81ac 100644
--- a/pkg/core/stages/setup_logger.py
+++ b/pkg/core/stages/setup_logger.py
@@ -8,7 +8,7 @@ from ..bootutils import log
class PersistenceHandler(logging.Handler, object):
"""
- 保存日志到数据库
+ Save logs to database
"""
ap: app.Application
@@ -19,9 +19,9 @@ class PersistenceHandler(logging.Handler, object):
def emit(self, record):
"""
- emit函数为自定义handler类时必重写的函数,这里可以根据需要对日志消息做一些处理,比如发送日志到服务器
+ emit function is a required function for custom handler classes, here you can process the log messages as needed, such as sending logs to the server
- 发出记录(Emit a record)
+ Emit a record
"""
try:
msg = self.format(record)
@@ -34,10 +34,10 @@ class PersistenceHandler(logging.Handler, object):
@stage.stage_class('SetupLoggerStage')
class SetupLoggerStage(stage.BootingStage):
- """设置日志器阶段"""
+ """Setup logger stage"""
async def run(self, ap: app.Application):
- """启动"""
+ """Setup logger"""
persistence_handler = PersistenceHandler('LoggerHandler', ap)
extra_handlers = []
diff --git a/pkg/core/stages/show_notes.py b/pkg/core/stages/show_notes.py
index 5fa7ff08..d0f861ba 100644
--- a/pkg/core/stages/show_notes.py
+++ b/pkg/core/stages/show_notes.py
@@ -12,10 +12,10 @@ importutil.import_modules_in_pkg(notes)
@stage.stage_class('ShowNotesStage')
class ShowNotesStage(stage.BootingStage):
- """显示启动信息阶段"""
+ """Show notes stage"""
async def run(self, ap: app.Application):
- # 排序
+ # Sort
note.preregistered_notes.sort(key=lambda x: x.number)
for note_cls in note.preregistered_notes:
diff --git a/pkg/core/taskmgr.py b/pkg/core/taskmgr.py
index 0f756118..ca6eb029 100644
--- a/pkg/core/taskmgr.py
+++ b/pkg/core/taskmgr.py
@@ -9,13 +9,13 @@ from . import entities as core_entities
class TaskContext:
- """任务跟踪上下文"""
+ """Task tracking context"""
current_action: str
- """当前正在执行的动作"""
+ """Current action being executed"""
log: str
- """记录日志"""
+ """Log"""
def __init__(self):
self.current_action = 'default'
@@ -58,40 +58,40 @@ placeholder_context: TaskContext | None = None
class TaskWrapper:
- """任务包装器"""
+ """Task wrapper"""
_id_index: int = 0
- """任务ID索引"""
+ """Task ID index"""
id: int
- """任务ID"""
+ """Task ID"""
- task_type: str = 'system' # 任务类型: system 或 user
- """任务类型"""
+ task_type: str = 'system' # Task type: system or user
+ """Task type"""
- kind: str = 'system_task' # 由发起者确定任务种类,通常同质化的任务种类相同
- """任务种类"""
+ kind: str = 'system_task' # Task type determined by the initiator, usually the same task type
+ """Task type"""
name: str = ''
- """任务唯一名称"""
+ """Task unique name"""
label: str = ''
- """任务显示名称"""
+ """Task display name"""
task_context: TaskContext
- """任务上下文"""
+ """Task context"""
task: asyncio.Task
- """任务"""
+ """Task"""
task_stack: list = None
- """任务堆栈"""
+ """Task stack"""
ap: app.Application
- """应用实例"""
+ """Application instance"""
scopes: list[core_entities.LifecycleControlScope]
- """任务所属生命周期控制范围"""
+ """Task scope"""
def __init__(
self,
@@ -165,13 +165,13 @@ class TaskWrapper:
class AsyncTaskManager:
- """保存app中的所有异步任务
- 包含系统级的和用户级(插件安装、更新等由用户直接发起的)的"""
+ """Save all asynchronous tasks in the app
+ Include system-level and user-level (plugin installation, update, etc. initiated by users directly)"""
ap: app.Application
tasks: list[TaskWrapper]
- """所有任务"""
+ """All tasks"""
def __init__(self, ap: app.Application):
self.ap = ap
diff --git a/pkg/entity/persistence/bot.py b/pkg/entity/persistence/bot.py
index 3c08f4ec..08eda478 100644
--- a/pkg/entity/persistence/bot.py
+++ b/pkg/entity/persistence/bot.py
@@ -4,7 +4,7 @@ from .base import Base
class Bot(Base):
- """机器人"""
+ """Bot"""
__tablename__ = 'bots'
diff --git a/pkg/entity/persistence/metadata.py b/pkg/entity/persistence/metadata.py
index d9e03663..4db732b9 100644
--- a/pkg/entity/persistence/metadata.py
+++ b/pkg/entity/persistence/metadata.py
@@ -12,7 +12,7 @@ initial_metadata = [
class Metadata(Base):
- """数据库元数据"""
+ """Database metadata"""
__tablename__ = 'metadata'
diff --git a/pkg/entity/persistence/model.py b/pkg/entity/persistence/model.py
index 418cab70..e9a104c4 100644
--- a/pkg/entity/persistence/model.py
+++ b/pkg/entity/persistence/model.py
@@ -4,7 +4,7 @@ from .base import Base
class LLMModel(Base):
- """LLM 模型"""
+ """LLM model"""
__tablename__ = 'llm_models'
diff --git a/pkg/entity/persistence/pipeline.py b/pkg/entity/persistence/pipeline.py
index 70e76dab..3a21dbf2 100644
--- a/pkg/entity/persistence/pipeline.py
+++ b/pkg/entity/persistence/pipeline.py
@@ -4,7 +4,7 @@ from .base import Base
class LegacyPipeline(Base):
- """旧版流水线"""
+ """Legacy pipeline"""
__tablename__ = 'legacy_pipelines'
@@ -25,7 +25,7 @@ class LegacyPipeline(Base):
class PipelineRunRecord(Base):
- """流水线运行记录"""
+ """Pipeline run record"""
__tablename__ = 'pipeline_run_records'
diff --git a/pkg/entity/persistence/plugin.py b/pkg/entity/persistence/plugin.py
index 30db6bd6..e777441f 100644
--- a/pkg/entity/persistence/plugin.py
+++ b/pkg/entity/persistence/plugin.py
@@ -4,7 +4,7 @@ from .base import Base
class PluginSetting(Base):
- """插件配置"""
+ """Plugin setting"""
__tablename__ = 'plugin_settings'
diff --git a/pkg/persistence/database.py b/pkg/persistence/database.py
index 528c6a34..4debb03d 100644
--- a/pkg/persistence/database.py
+++ b/pkg/persistence/database.py
@@ -11,7 +11,7 @@ preregistered_managers: list[type[BaseDatabaseManager]] = []
def manager_class(name: str) -> None:
- """注册一个数据库管理类"""
+ """Register a database manager class"""
def decorator(cls: type[BaseDatabaseManager]) -> type[BaseDatabaseManager]:
cls.name = name
@@ -22,7 +22,7 @@ def manager_class(name: str) -> None:
class BaseDatabaseManager(abc.ABC):
- """基础数据库管理类"""
+ """Base database manager class"""
name: str
diff --git a/pkg/persistence/databases/sqlite.py b/pkg/persistence/databases/sqlite.py
index 7b095e61..c1337459 100644
--- a/pkg/persistence/databases/sqlite.py
+++ b/pkg/persistence/databases/sqlite.py
@@ -7,7 +7,7 @@ from .. import database
@database.manager_class('sqlite')
class SQLiteDatabaseManager(database.BaseDatabaseManager):
- """SQLite 数据库管理类"""
+ """SQLite database manager"""
async def initialize(self) -> None:
sqlite_path = 'data/langbot.db'
diff --git a/pkg/persistence/mgr.py b/pkg/persistence/mgr.py
index 606aa9fd..9d2bab7b 100644
--- a/pkg/persistence/mgr.py
+++ b/pkg/persistence/mgr.py
@@ -22,12 +22,12 @@ importutil.import_modules_in_pkg(persistence)
class PersistenceManager:
- """持久化模块管理器"""
+ """Persistence module manager"""
ap: app.Application
db: database.BaseDatabaseManager
- """数据库管理器"""
+ """Database manager"""
meta: sqlalchemy.MetaData
@@ -79,7 +79,7 @@ class PersistenceManager:
'stages': pipeline_service.default_stage_order,
'is_default': True,
'name': 'ChatPipeline',
- 'description': '默认提供的流水线,您配置的机器人、第一个模型将自动绑定到此流水线',
+ 'description': 'Default pipeline provided, your new bots will be automatically bound to this pipeline | 默认提供的流水线,您配置的机器人将自动绑定到此流水线',
'config': pipeline_config,
}
diff --git a/pkg/persistence/migration.py b/pkg/persistence/migration.py
index c191b686..294e30ca 100644
--- a/pkg/persistence/migration.py
+++ b/pkg/persistence/migration.py
@@ -10,7 +10,7 @@ preregistered_db_migrations: list[typing.Type[DBMigration]] = []
def migration_class(number: int):
- """迁移类装饰器"""
+ """Migration class decorator"""
def wrapper(cls: typing.Type[DBMigration]) -> typing.Type[DBMigration]:
cls.number = number
@@ -21,20 +21,20 @@ def migration_class(number: int):
class DBMigration(abc.ABC):
- """数据库迁移"""
+ """Database migration"""
number: int
- """迁移号"""
+ """Migration number"""
def __init__(self, ap: app.Application):
self.ap = ap
@abc.abstractmethod
async def upgrade(self):
- """升级"""
+ """Upgrade"""
pass
@abc.abstractmethod
async def downgrade(self):
- """降级"""
+ """Downgrade"""
pass
diff --git a/pkg/persistence/migrations/dbm001_migrate_v3_config.py b/pkg/persistence/migrations/dbm001_migrate_v3_config.py
index a1145527..58f05e04 100644
--- a/pkg/persistence/migrations/dbm001_migrate_v3_config.py
+++ b/pkg/persistence/migrations/dbm001_migrate_v3_config.py
@@ -15,21 +15,21 @@ from ...entity.persistence import (
@migration.migration_class(1)
class DBMigrateV3Config(migration.DBMigration):
- """从 v3 的配置迁移到 v4 的数据库"""
+ """Migrate v3 config to v4 database"""
async def upgrade(self):
- """升级"""
+ """Upgrade"""
"""
- 将 data/config 下的所有配置文件进行迁移。
- 迁移后,之前的配置文件都保存到 data/legacy/config 下。
- 迁移后,data/metadata/ 下的所有配置文件都保存到 data/legacy/metadata 下。
+ Migrate all config files under data/config.
+ After migration, all previous config files are saved under data/legacy/config.
+ After migration, all config files under data/metadata/ are saved under data/legacy/metadata.
"""
if self.ap.provider_cfg is None:
return
- # ======= 迁移模型 =======
- # 只迁移当前选中的模型
+ # ======= Migrate model =======
+ # Only migrate the currently selected model
model_name = self.ap.provider_cfg.data.get('model', 'gpt-4o')
model_requester = 'openai-chat-completions'
@@ -91,8 +91,8 @@ class DBMigrateV3Config(migration.DBMigration):
sqlalchemy.insert(persistence_model.LLMModel).values(**llm_model_data)
)
- # ======= 迁移流水线配置 =======
- # 修改到默认流水线
+ # ======= Migrate pipeline config =======
+ # Modify to default pipeline
default_pipeline = [
self.ap.persistence_mgr.serialize_model(persistence_pipeline.LegacyPipeline, pipeline)
for pipeline in (
@@ -184,8 +184,8 @@ class DBMigrateV3Config(migration.DBMigration):
.where(persistence_pipeline.LegacyPipeline.uuid == default_pipeline['uuid'])
)
- # ======= 迁移机器人 =======
- # 只迁移启用的机器人
+ # ======= Migrate bot =======
+ # Only migrate enabled bots
for adapter in self.ap.platform_cfg.data.get('platform-adapters', []):
if not adapter.get('enable'):
continue
@@ -207,7 +207,7 @@ class DBMigrateV3Config(migration.DBMigration):
await self.ap.persistence_mgr.execute_async(sqlalchemy.insert(persistence_bot.Bot).values(**bot_data))
- # ======= 迁移系统设置 =======
+ # ======= Migrate system settings =======
self.ap.instance_config.data['admins'] = self.ap.system_cfg.data['admin-sessions']
self.ap.instance_config.data['api']['port'] = self.ap.system_cfg.data['http-api']['port']
self.ap.instance_config.data['command'] = {
@@ -223,7 +223,7 @@ class DBMigrateV3Config(migration.DBMigration):
await self.ap.instance_config.dump_config()
# ======= move files =======
- # 迁移 data/config 下的所有配置文件
+ # Migrate all config files under data/config
all_legacy_dir_name = [
'config',
# 'metadata',
@@ -246,4 +246,4 @@ class DBMigrateV3Config(migration.DBMigration):
move_legacy_files(dir_name)
async def downgrade(self):
- """降级"""
+ """Downgrade"""
diff --git a/pkg/persistence/migrations/dbm002_combine_quote_msg_config.py b/pkg/persistence/migrations/dbm002_combine_quote_msg_config.py
index cebf403b..349bb0c2 100644
--- a/pkg/persistence/migrations/dbm002_combine_quote_msg_config.py
+++ b/pkg/persistence/migrations/dbm002_combine_quote_msg_config.py
@@ -7,10 +7,10 @@ from ...entity.persistence import pipeline as persistence_pipeline
@migration.migration_class(2)
class DBMigrateCombineQuoteMsgConfig(migration.DBMigration):
- """引用消息合并配置"""
+ """Combine quote message config"""
async def upgrade(self):
- """升级"""
+ """Upgrade"""
# read all pipelines
pipelines = await self.ap.persistence_mgr.execute_async(sqlalchemy.select(persistence_pipeline.LegacyPipeline))
@@ -37,5 +37,5 @@ class DBMigrateCombineQuoteMsgConfig(migration.DBMigration):
)
async def downgrade(self):
- """降级"""
+ """Downgrade"""
pass
diff --git a/pkg/persistence/migrations/dbm003_n8n_config.py b/pkg/persistence/migrations/dbm003_n8n_config.py
index 8705040b..15484f22 100644
--- a/pkg/persistence/migrations/dbm003_n8n_config.py
+++ b/pkg/persistence/migrations/dbm003_n8n_config.py
@@ -7,10 +7,10 @@ from ...entity.persistence import pipeline as persistence_pipeline
@migration.migration_class(3)
class DBMigrateN8nConfig(migration.DBMigration):
- """N8n配置"""
+ """N8n config"""
async def upgrade(self):
- """升级"""
+ """Upgrade"""
# read all pipelines
pipelines = await self.ap.persistence_mgr.execute_async(sqlalchemy.select(persistence_pipeline.LegacyPipeline))
@@ -45,5 +45,5 @@ class DBMigrateN8nConfig(migration.DBMigration):
)
async def downgrade(self):
- """降级"""
+ """Downgrade"""
pass
diff --git a/pkg/pipeline/bansess/bansess.py b/pkg/pipeline/bansess/bansess.py
index 3b927a55..c88a1aa2 100644
--- a/pkg/pipeline/bansess/bansess.py
+++ b/pkg/pipeline/bansess/bansess.py
@@ -6,9 +6,9 @@ from ...core import entities as core_entities
@stage.stage_class('BanSessionCheckStage')
class BanSessionCheckStage(stage.PipelineStage):
- """访问控制处理阶段
+ """Access control processing stage
- 仅检查query中群号或个人号是否在访问控制列表中。
+ Only check if the group or personal number in the query is in the access control list.
"""
async def initialize(self, pipeline_config: dict):
@@ -41,5 +41,7 @@ class BanSessionCheckStage(stage.PipelineStage):
return entities.StageProcessResult(
result_type=entities.ResultType.CONTINUE if ctn else entities.ResultType.INTERRUPT,
new_query=query,
- console_notice=f'根据访问控制忽略消息: {query.launcher_type.value}_{query.launcher_id}' if not ctn else '',
+ console_notice=f'Ignore message according to access control: {query.launcher_type.value}_{query.launcher_id}'
+ if not ctn
+ else '',
)
diff --git a/pkg/pipeline/cntfilter/filter.py b/pkg/pipeline/cntfilter/filter.py
index 0a3ceaae..36d8a7f4 100644
--- a/pkg/pipeline/cntfilter/filter.py
+++ b/pkg/pipeline/cntfilter/filter.py
@@ -13,13 +13,13 @@ preregistered_filters: list[typing.Type[ContentFilter]] = []
def filter_class(
name: str,
) -> typing.Callable[[typing.Type[ContentFilter]], typing.Type[ContentFilter]]:
- """内容过滤器类装饰器
+ """Content filter class decorator
Args:
- name (str): 过滤器名称
+ name (str): Filter name
Returns:
- typing.Callable[[typing.Type[ContentFilter]], typing.Type[ContentFilter]]: 装饰器
+ typing.Callable[[typing.Type[ContentFilter]], typing.Type[ContentFilter]]: Decorator
"""
def decorator(cls: typing.Type[ContentFilter]) -> typing.Type[ContentFilter]:
@@ -35,7 +35,7 @@ def filter_class(
class ContentFilter(metaclass=abc.ABCMeta):
- """内容过滤器抽象类"""
+ """Content filter abstract class"""
name: str
@@ -46,31 +46,31 @@ class ContentFilter(metaclass=abc.ABCMeta):
@property
def enable_stages(self):
- """启用的阶段
+ """Enabled stages
- 默认为消息请求AI前后的两个阶段。
+ Default is the two stages before and after the message request to AI.
- entity.EnableStage.PRE: 消息请求AI前,此时需要检查的内容是用户的输入消息。
- entity.EnableStage.POST: 消息请求AI后,此时需要检查的内容是AI的回复消息。
+ entity.EnableStage.PRE: Before message request to AI, the content to check is the user's input message.
+ entity.EnableStage.POST: After message request to AI, the content to check is the AI's reply message.
"""
return [entities.EnableStage.PRE, entities.EnableStage.POST]
async def initialize(self):
- """初始化过滤器"""
+ """Initialize filter"""
pass
@abc.abstractmethod
async def process(self, query: core_entities.Query, message: str = None, image_url=None) -> entities.FilterResult:
- """处理消息
+ """Process message
- 分为前后阶段,具体取决于 enable_stages 的值。
- 对于内容过滤器来说,不需要考虑消息所处的阶段,只需要检查消息内容即可。
+ It is divided into two stages, depending on the value of enable_stages.
+ For content filters, you do not need to consider the stage of the message, you only need to check the message content.
Args:
- message (str): 需要检查的内容
- image_url (str): 要检查的图片的 URL
+ message (str): Content to check
+ image_url (str): URL of the image to check
Returns:
- entities.FilterResult: 过滤结果,具体内容请查看 entities.FilterResult 类的文档
+ entities.FilterResult: Filter result, please refer to the documentation of entities.FilterResult class
"""
raise NotImplementedError
diff --git a/pkg/pipeline/cntfilter/filters/banwords.py b/pkg/pipeline/cntfilter/filters/banwords.py
index 916a1bc1..b03e79a9 100644
--- a/pkg/pipeline/cntfilter/filters/banwords.py
+++ b/pkg/pipeline/cntfilter/filters/banwords.py
@@ -8,7 +8,7 @@ from ....core import entities as core_entities
@filter_model.filter_class('ban-word-filter')
class BanWordFilter(filter_model.ContentFilter):
- """根据内容过滤"""
+ """Filter content"""
async def initialize(self):
pass
diff --git a/pkg/pipeline/cntfilter/filters/cntignore.py b/pkg/pipeline/cntfilter/filters/cntignore.py
index 5e410e31..b80d90eb 100644
--- a/pkg/pipeline/cntfilter/filters/cntignore.py
+++ b/pkg/pipeline/cntfilter/filters/cntignore.py
@@ -8,7 +8,7 @@ from ....core import entities as core_entities
@filter_model.filter_class('content-ignore')
class ContentIgnore(filter_model.ContentFilter):
- """根据内容忽略消息"""
+ """Ignore message according to content"""
@property
def enable_stages(self):
@@ -24,7 +24,7 @@ class ContentIgnore(filter_model.ContentFilter):
level=entities.ResultLevel.BLOCK,
replacement='',
user_notice='',
- console_notice='根据 ignore_rules 中的 prefix 规则,忽略消息',
+ console_notice='Ignore message according to prefix rule in ignore_rules',
)
if 'regexp' in query.pipeline_config['trigger']['ignore-rules']:
@@ -34,7 +34,7 @@ class ContentIgnore(filter_model.ContentFilter):
level=entities.ResultLevel.BLOCK,
replacement='',
user_notice='',
- console_notice='根据 ignore_rules 中的 regexp 规则,忽略消息',
+ console_notice='Ignore message according to regexp rule in ignore_rules',
)
return entities.FilterResult(
diff --git a/pkg/pipeline/longtext/longtext.py b/pkg/pipeline/longtext/longtext.py
index 5be20650..03457212 100644
--- a/pkg/pipeline/longtext/longtext.py
+++ b/pkg/pipeline/longtext/longtext.py
@@ -16,9 +16,9 @@ importutil.import_modules_in_pkg(strategies)
@stage.stage_class('LongTextProcessStage')
class LongTextProcessStage(stage.PipelineStage):
- """长消息处理阶段
+ """Long message processing stage
- 改写:
+ Rewrite:
- resp_message_chain
"""
@@ -36,22 +36,22 @@ class LongTextProcessStage(stage.PipelineStage):
use_font = 'C:/Windows/Fonts/msyh.ttc'
if not os.path.exists(use_font):
self.ap.logger.warn(
- '未找到字体文件,且无法使用Windows自带字体,更换为转发消息组件以发送长消息,您可以在配置文件中调整相关设置。'
+ 'Font file not found, and Windows system font cannot be used, switch to forward message component to send long messages, you can adjust the related settings in the configuration file.'
)
config['blob_message_strategy'] = 'forward'
else:
- self.ap.logger.info('使用Windows自带字体:' + use_font)
+ self.ap.logger.info('Using Windows system font: ' + use_font)
config['font-path'] = use_font
else:
self.ap.logger.warn(
- '未找到字体文件,且无法使用系统自带字体,更换为转发消息组件以发送长消息,您可以在配置文件中调整相关设置。'
+ 'Font file not found, and system font cannot be used, switch to forward message component to send long messages, you can adjust the related settings in the configuration file.'
)
pipeline_config['output']['long-text-processing']['strategy'] = 'forward'
except Exception:
traceback.print_exc()
self.ap.logger.error(
- '加载字体文件失败({}),更换为转发消息组件以发送长消息,您可以在配置文件中调整相关设置。'.format(
+ 'Failed to load font file ({}), switch to forward message component to send long messages, you can adjust the related settings in the configuration file.'.format(
use_font
)
)
@@ -63,12 +63,12 @@ class LongTextProcessStage(stage.PipelineStage):
self.strategy_impl = strategy_cls(self.ap)
break
else:
- raise ValueError(f'未找到名为 {config["strategy"]} 的长消息处理策略')
+ raise ValueError(f'Long message processing strategy not found: {config["strategy"]}')
await self.strategy_impl.initialize()
async def process(self, query: core_entities.Query, stage_inst_name: str) -> entities.StageProcessResult:
- # 检查是否包含非 Plain 组件
+ # Check if it contains non-Plain components
contains_non_plain = False
for msg in query.resp_message_chain[-1]:
@@ -77,7 +77,7 @@ class LongTextProcessStage(stage.PipelineStage):
break
if contains_non_plain:
- self.ap.logger.debug('消息中包含非 Plain 组件,跳过长消息处理。')
+ self.ap.logger.debug('Message contains non-Plain components, skip long message processing.')
elif (
len(str(query.resp_message_chain[-1]))
> query.pipeline_config['output']['long-text-processing']['threshold']
diff --git a/pkg/pipeline/longtext/strategies/forward.py b/pkg/pipeline/longtext/strategies/forward.py
index 6228d580..cb772339 100644
--- a/pkg/pipeline/longtext/strategies/forward.py
+++ b/pkg/pipeline/longtext/strategies/forward.py
@@ -15,17 +15,17 @@ Forward = platform_message.Forward
class ForwardComponentStrategy(strategy_model.LongTextStrategy):
async def process(self, message: str, query: core_entities.Query) -> list[platform_message.MessageComponent]:
display = ForwardMessageDiaplay(
- title='群聊的聊天记录',
- brief='[聊天记录]',
- source='聊天记录',
- preview=['QQ用户: ' + message],
- summary='查看1条转发消息',
+ title='Group chat history',
+ brief='[Chat history]',
+ source='Chat history',
+ preview=['User: ' + message],
+ summary='View 1 forwarded message',
)
node_list = [
platform_message.ForwardMessageNode(
sender_id=query.adapter.bot_account_id,
- sender_name='QQ用户',
+ sender_name='User',
message_chain=platform_message.MessageChain([message]),
)
]
diff --git a/pkg/pipeline/longtext/strategy.py b/pkg/pipeline/longtext/strategy.py
index 0ddec0c6..5b521067 100644
--- a/pkg/pipeline/longtext/strategy.py
+++ b/pkg/pipeline/longtext/strategy.py
@@ -14,13 +14,13 @@ preregistered_strategies: list[typing.Type[LongTextStrategy]] = []
def strategy_class(
name: str,
) -> typing.Callable[[typing.Type[LongTextStrategy]], typing.Type[LongTextStrategy]]:
- """长文本处理策略类装饰器
+ """Long text processing strategy class decorator
Args:
- name (str): 策略名称
+ name (str): Strategy name
Returns:
- typing.Callable[[typing.Type[LongTextStrategy]], typing.Type[LongTextStrategy]]: 装饰器
+ typing.Callable[[typing.Type[LongTextStrategy]], typing.Type[LongTextStrategy]]: Decorator
"""
def decorator(cls: typing.Type[LongTextStrategy]) -> typing.Type[LongTextStrategy]:
@@ -36,7 +36,7 @@ def strategy_class(
class LongTextStrategy(metaclass=abc.ABCMeta):
- """长文本处理策略抽象类"""
+ """Long text processing strategy abstract class"""
name: str
@@ -50,15 +50,15 @@ class LongTextStrategy(metaclass=abc.ABCMeta):
@abc.abstractmethod
async def process(self, message: str, query: core_entities.Query) -> list[platform_message.MessageComponent]:
- """处理长文本
+ """Process long text
- 在 platform.json 中配置 long-text-process 字段,只要 文本长度超过了 threshold 就会调用此方法
+ If the text length exceeds the threshold, this method will be called.
Args:
- message (str): 消息
- query (core_entities.Query): 此次请求的上下文对象
+ message (str): Message
+ query (core_entities.Query): Query object
Returns:
- list[platform_message.MessageComponent]: 转换后的 平台 消息组件列表
+ list[platform_message.MessageComponent]: Converted platform message components
"""
return []
diff --git a/pkg/pipeline/msgtrun/msgtrun.py b/pkg/pipeline/msgtrun/msgtrun.py
index c64f67fc..1c5ee17d 100644
--- a/pkg/pipeline/msgtrun/msgtrun.py
+++ b/pkg/pipeline/msgtrun/msgtrun.py
@@ -12,9 +12,9 @@ importutil.import_modules_in_pkg(truncators)
@stage.stage_class('ConversationMessageTruncator')
class ConversationMessageTruncator(stage.PipelineStage):
- """会话消息截断器
+ """Conversation message truncator
- 用于截断会话消息链,以适应平台消息长度限制。
+ Used to truncate the conversation message chain to adapt to the LLM message length limit.
"""
trun: truncator.Truncator
@@ -27,10 +27,10 @@ class ConversationMessageTruncator(stage.PipelineStage):
self.trun = trun(self.ap)
break
else:
- raise ValueError(f'未知的截断器: {use_method}')
+ raise ValueError(f'Unknown truncator: {use_method}')
async def process(self, query: core_entities.Query, stage_inst_name: str) -> entities.StageProcessResult:
- """处理"""
+ """Process"""
query = await self.trun.truncate(query)
return entities.StageProcessResult(result_type=entities.ResultType.CONTINUE, new_query=query)
diff --git a/pkg/pipeline/msgtrun/truncators/round.py b/pkg/pipeline/msgtrun/truncators/round.py
index fa72a0e1..2acb1d8c 100644
--- a/pkg/pipeline/msgtrun/truncators/round.py
+++ b/pkg/pipeline/msgtrun/truncators/round.py
@@ -6,17 +6,17 @@ from ....core import entities as core_entities
@truncator.truncator_class('round')
class RoundTruncator(truncator.Truncator):
- """前文回合数阶段器"""
+ """Truncate the conversation message chain to adapt to the LLM message length limit."""
async def truncate(self, query: core_entities.Query) -> core_entities.Query:
- """截断"""
+ """Truncate"""
max_round = query.pipeline_config['ai']['local-agent']['max-round']
temp_messages = []
current_round = 0
- # 从后往前遍历
+ # Traverse from back to front
for msg in query.messages[::-1]:
if current_round < max_round:
temp_messages.append(msg)
diff --git a/pkg/pipeline/preproc/preproc.py b/pkg/pipeline/preproc/preproc.py
index fd4c0bb6..1aada6b3 100644
--- a/pkg/pipeline/preproc/preproc.py
+++ b/pkg/pipeline/preproc/preproc.py
@@ -11,11 +11,11 @@ from ...platform.types import message as platform_message
@stage.stage_class('PreProcessor')
class PreProcessor(stage.PipelineStage):
- """请求预处理阶段
+ """Request pre-processing stage
- 签出会话、prompt、上文、模型、内容函数。
+ Check out session, prompt, context, model, and content functions.
- 改写:
+ Rewrite:
- session
- prompt
- messages
@@ -29,12 +29,12 @@ class PreProcessor(stage.PipelineStage):
query: core_entities.Query,
stage_inst_name: str,
) -> entities.StageProcessResult:
- """处理"""
+ """Process"""
selected_runner = query.pipeline_config['ai']['runner']['runner']
session = await self.ap.sess_mgr.get_session(query)
- # 非 local-agent 时,llm_model 为 None
+ # When not local-agent, llm_model is None
llm_model = (
await self.ap.model_mgr.get_model_by_uuid(query.pipeline_config['ai']['local-agent']['model'])
if selected_runner == 'local-agent'
@@ -51,7 +51,7 @@ class PreProcessor(stage.PipelineStage):
conversation.use_llm_model = llm_model
- # 设置query
+ # Set query
query.session = session
query.prompt = conversation.prompt.copy()
query.messages = conversation.messages.copy()
@@ -112,7 +112,7 @@ class PreProcessor(stage.PipelineStage):
query.variables['user_message_text'] = plain_text
query.user_message = llm_entities.Message(role='user', content=content_list)
- # =========== 触发事件 PromptPreProcessing
+ # =========== Trigger event PromptPreProcessing
event_ctx = await self.ap.plugin_mgr.emit_event(
event=events.PromptPreProcessing(
diff --git a/pkg/pipeline/process/handler.py b/pkg/pipeline/process/handler.py
index 8a32bcfb..837b72e2 100644
--- a/pkg/pipeline/process/handler.py
+++ b/pkg/pipeline/process/handler.py
@@ -25,7 +25,7 @@ class MessageHandler(metaclass=abc.ABCMeta):
def cut_str(self, s: str) -> str:
"""
- 取字符串第一行,最多20个字符,若有多行,或超过20个字符,则加省略号
+ Take the first line of the string, up to 20 characters, if there are multiple lines, or more than 20 characters, add an ellipsis
"""
s0 = s.split('\n')[0]
if len(s0) > 20 or '\n' in s:
diff --git a/pkg/pipeline/process/handlers/chat.py b/pkg/pipeline/process/handlers/chat.py
index 35fa1611..2aa08e17 100644
--- a/pkg/pipeline/process/handlers/chat.py
+++ b/pkg/pipeline/process/handlers/chat.py
@@ -22,11 +22,11 @@ class ChatMessageHandler(handler.MessageHandler):
self,
query: core_entities.Query,
) -> typing.AsyncGenerator[entities.StageProcessResult, None]:
- """处理"""
- # 调API
- # 生成器
+ """Process"""
+ # Call API
+ # generator
- # 触发插件事件
+ # Trigger plugin event
event_class = (
events.PersonNormalMessageReceived
if query.launcher_type == core_entities.LauncherTypes.PERSON
@@ -54,7 +54,7 @@ class ChatMessageHandler(handler.MessageHandler):
yield entities.StageProcessResult(result_type=entities.ResultType.INTERRUPT, new_query=query)
else:
if event_ctx.event.alter is not None:
- # if isinstance(event_ctx.event, str): # 现在暂时不考虑多模态alter
+ # if isinstance(event_ctx.event, str): # Currently not considering multi-modal alter
query.user_message.content = event_ctx.event.alter
text_length = 0
@@ -65,12 +65,12 @@ class ChatMessageHandler(handler.MessageHandler):
runner = r(self.ap, query.pipeline_config)
break
else:
- raise ValueError(f'未找到请求运行器: {query.pipeline_config["ai"]["runner"]["runner"]}')
+ raise ValueError(f'Request runner not found: {query.pipeline_config["ai"]["runner"]["runner"]}')
async for result in runner.run(query):
query.resp_messages.append(result)
- self.ap.logger.info(f'对话({query.query_id})响应: {self.cut_str(result.readable_str())}')
+ self.ap.logger.info(f'Response({query.query_id}): {self.cut_str(result.readable_str())}')
if result.content is not None:
text_length += len(result.content)
@@ -80,7 +80,7 @@ class ChatMessageHandler(handler.MessageHandler):
query.session.using_conversation.messages.append(query.user_message)
query.session.using_conversation.messages.extend(query.resp_messages)
except Exception as e:
- self.ap.logger.error(f'对话({query.query_id})请求失败: {type(e).__name__} {str(e)}')
+ self.ap.logger.error(f'Request failed({query.query_id}): {type(e).__name__} {str(e)}')
hide_exception_info = query.pipeline_config['output']['misc']['hide-exception']
diff --git a/pkg/pipeline/process/handlers/command.py b/pkg/pipeline/process/handlers/command.py
index cc0e9314..7348d6b8 100644
--- a/pkg/pipeline/process/handlers/command.py
+++ b/pkg/pipeline/process/handlers/command.py
@@ -15,7 +15,7 @@ class CommandHandler(handler.MessageHandler):
self,
query: core_entities.Query,
) -> typing.AsyncGenerator[entities.StageProcessResult, None]:
- """处理"""
+ """Process"""
command_text = str(query.message_chain).strip()[1:]
@@ -70,7 +70,7 @@ class CommandHandler(handler.MessageHandler):
)
)
- self.ap.logger.info(f'命令({query.query_id})报错: {self.cut_str(str(ret.error))}')
+ self.ap.logger.info(f'Command({query.query_id}) error: {self.cut_str(str(ret.error))}')
yield entities.StageProcessResult(result_type=entities.ResultType.CONTINUE, new_query=query)
elif ret.text is not None or ret.image_url is not None:
@@ -89,7 +89,7 @@ class CommandHandler(handler.MessageHandler):
)
)
- self.ap.logger.info(f'命令返回: {self.cut_str(str(content[0]))}')
+ self.ap.logger.info(f'Command returned: {self.cut_str(str(content[0]))}')
yield entities.StageProcessResult(result_type=entities.ResultType.CONTINUE, new_query=query)
else:
diff --git a/pkg/pipeline/process/process.py b/pkg/pipeline/process/process.py
index 64903552..db66135c 100644
--- a/pkg/pipeline/process/process.py
+++ b/pkg/pipeline/process/process.py
@@ -33,11 +33,11 @@ class Processor(stage.PipelineStage):
query: core_entities.Query,
stage_inst_name: str,
) -> entities.StageProcessResult:
- """处理"""
+ """Process"""
message_text = str(query.message_chain).strip()
self.ap.logger.info(
- f'处理 {query.launcher_type.value}_{query.launcher_id} 的请求({query.query_id}): {message_text}'
+ f'Processing request from {query.launcher_type.value}_{query.launcher_id} ({query.query_id}): {message_text}'
)
async def generator():
diff --git a/pkg/platform/sources/aiocqhttp.py b/pkg/platform/sources/aiocqhttp.py
index 2730874f..357eb48a 100644
--- a/pkg/platform/sources/aiocqhttp.py
+++ b/pkg/platform/sources/aiocqhttp.py
@@ -76,8 +76,7 @@ class AiocqhttpMessageConverter(adapter.MessageConverter):
return msg_list, msg_id, msg_time
@staticmethod
- async def target2yiri(message: str, message_id: int = -1, bot=None):
- print(message)
+ async def target2yiri(message: str, message_id: int = -1, bot: aiocqhttp.CQHttp = None):
message = aiocqhttp.Message(message)
def get_face_name(face_id):
@@ -271,16 +270,16 @@ class AiocqhttpMessageConverter(adapter.MessageConverter):
)
yiri_msg_list.append(reply_msg)
- elif msg.type == 'file':
- # file_name = msg.data['file']
- file_id = msg.data['file_id']
- file_data = await bot.get_file(file_id=file_id)
- file_name = file_data.get('file_name')
- file_path = file_data.get('file')
- _ = file_path
- file_url = file_data.get('file_url')
- file_size = file_data.get('file_size')
- yiri_msg_list.append(platform_message.File(id=file_id, name=file_name, url=file_url, size=file_size))
+ # 这里下载所有文件会导致下载文件过多,暂时不下载
+ # elif msg.type == 'file':
+ # # file_name = msg.data['file']
+ # file_id = msg.data['file_id']
+ # file_data = await bot.get_file(file_id=file_id)
+ # file_name = file_data.get('file_name')
+ # file_path = file_data.get('file')
+ # file_url = file_data.get('file_url')
+ # file_size = file_data.get('file_size')
+ # yiri_msg_list.append(platform_message.File(id=file_id, name=file_name,url=file_url,size=file_size))
elif msg.type == 'face':
face_id = msg.data['id']
face_name = msg.data['raw']['faceText']
diff --git a/pkg/platform/sources/dingtalk.py b/pkg/platform/sources/dingtalk.py
index 3147c984..a40b0f9b 100644
--- a/pkg/platform/sources/dingtalk.py
+++ b/pkg/platform/sources/dingtalk.py
@@ -116,6 +116,15 @@ class DingTalkAdapter(adapter.MessagePlatformAdapter):
self.bot_account_id = self.config['robot_name']
+ self.bot = DingTalkClient(
+ client_id=config['client_id'],
+ client_secret=config['client_secret'],
+ robot_name=config['robot_name'],
+ robot_code=config['robot_code'],
+ markdown_card=config['markdown_card'],
+ logger=self.logger,
+ )
+
async def reply_message(
self,
message_source: platform_events.MessageEvent,
@@ -157,15 +166,6 @@ class DingTalkAdapter(adapter.MessagePlatformAdapter):
self.bot.on_message('GroupMessage')(on_message)
async def run_async(self):
- config = self.config
- self.bot = DingTalkClient(
- client_id=config['client_id'],
- client_secret=config['client_secret'],
- robot_name=config['robot_name'],
- robot_code=config['robot_code'],
- markdown_card=config['markdown_card'],
- logger=self.logger,
- )
await self.bot.start()
async def kill(self) -> bool:
diff --git a/pkg/platform/sources/discord.py b/pkg/platform/sources/discord.py
index 6cc09a72..c279e714 100644
--- a/pkg/platform/sources/discord.py
+++ b/pkg/platform/sources/discord.py
@@ -8,15 +8,592 @@ import base64
import uuid
import os
import datetime
+import io
+import asyncio
+from enum import Enum
import aiohttp
from .. import adapter
from ...core import app
+from ..logger import EventLogger
from ..types import message as platform_message
from ..types import events as platform_events
from ..types import entities as platform_entities
-from ..logger import EventLogger
+
+# 语音功能相关异常定义
+class VoiceConnectionError(Exception):
+ """语音连接基础异常"""
+ def __init__(self, message: str, error_code: str = None, guild_id: int = None):
+ super().__init__(message)
+ self.error_code = error_code
+ self.guild_id = guild_id
+ self.timestamp = datetime.datetime.now()
+
+
+class VoicePermissionError(VoiceConnectionError):
+ """语音权限异常"""
+ def __init__(self, message: str, missing_permissions: list = None, user_id: int = None, channel_id: int = None):
+ super().__init__(message, "PERMISSION_ERROR")
+ self.missing_permissions = missing_permissions or []
+ self.user_id = user_id
+ self.channel_id = channel_id
+
+
+class VoiceNetworkError(VoiceConnectionError):
+ """语音网络异常"""
+ def __init__(self, message: str, retry_count: int = 0):
+ super().__init__(message, "NETWORK_ERROR")
+ self.retry_count = retry_count
+ self.last_attempt = datetime.datetime.now()
+
+
+class VoiceConnectionStatus(Enum):
+ """语音连接状态枚举"""
+ IDLE = "idle"
+ CONNECTING = "connecting"
+ CONNECTED = "connected"
+ PLAYING = "playing"
+ RECONNECTING = "reconnecting"
+ FAILED = "failed"
+
+
+class VoiceConnectionInfo:
+ """
+ 语音连接信息类
+
+ 用于存储和管理单个语音连接的详细信息,包括连接状态、时间戳、
+ 频道信息等。提供连接信息的标准化数据结构。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+ """
+
+ def __init__(self, guild_id: int, channel_id: int, channel_name: str = None):
+ """
+ 初始化语音连接信息
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+ channel_id (int): 语音频道ID
+ channel_name (str, optional): 语音频道名称
+ """
+ self.guild_id = guild_id
+ self.channel_id = channel_id
+ self.channel_name = channel_name or f"Channel-{channel_id}"
+ self.connected = False
+ self.connection_time: datetime.datetime = None
+ self.last_activity = datetime.datetime.now()
+ self.status = VoiceConnectionStatus.IDLE
+ self.user_count = 0
+ self.latency = 0.0
+ self.connection_health = "unknown"
+ self.voice_client = None
+
+ def update_status(self, status: VoiceConnectionStatus):
+ """
+ 更新连接状态
+
+ @author: @ydzat
+
+ Args:
+ status (VoiceConnectionStatus): 新的连接状态
+ """
+ self.status = status
+ self.last_activity = datetime.datetime.now()
+
+ if status == VoiceConnectionStatus.CONNECTED:
+ self.connected = True
+ if self.connection_time is None:
+ self.connection_time = datetime.datetime.now()
+ elif status in [VoiceConnectionStatus.IDLE, VoiceConnectionStatus.FAILED]:
+ self.connected = False
+ self.connection_time = None
+ self.voice_client = None
+
+ def to_dict(self) -> dict:
+ """
+ 转换为字典格式
+
+ @author: @ydzat
+
+ Returns:
+ dict: 连接信息的字典表示
+ """
+ return {
+ "guild_id": self.guild_id,
+ "channel_id": self.channel_id,
+ "channel_name": self.channel_name,
+ "connected": self.connected,
+ "connection_time": self.connection_time.isoformat() if self.connection_time else None,
+ "last_activity": self.last_activity.isoformat(),
+ "status": self.status.value,
+ "user_count": self.user_count,
+ "latency": self.latency,
+ "connection_health": self.connection_health
+ }
+
+
+class VoiceConnectionManager:
+ """
+ 语音连接管理器
+
+ 负责管理多个服务器的语音连接,提供连接建立、断开、状态查询等功能。
+ 采用单例模式确保全局只有一个连接管理器实例。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+ """
+
+ def __init__(self, bot: discord.Client, logger: EventLogger):
+ """
+ 初始化语音连接管理器
+
+ @author: @ydzat
+
+ Args:
+ bot (discord.Client): Discord 客户端实例
+ logger (EventLogger): 事件日志记录器
+ """
+ self.bot = bot
+ self.logger = logger
+ self.connections: typing.Dict[int, VoiceConnectionInfo] = {}
+ self._connection_lock = asyncio.Lock()
+ self._cleanup_task = None
+ self._monitoring_enabled = True
+
+ async def join_voice_channel(self, guild_id: int, channel_id: int,
+ user_id: int = None) -> discord.VoiceClient:
+ """
+ 加入语音频道
+
+ 验证用户权限和频道状态后,建立到指定语音频道的连接。
+ 支持连接复用和自动重连机制。
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+ channel_id (int): 语音频道ID
+ user_id (int, optional): 请求用户ID,用于权限验证
+
+ Returns:
+ discord.VoiceClient: 语音客户端实例
+
+ Raises:
+ VoicePermissionError: 权限不足时抛出
+ VoiceNetworkError: 网络连接失败时抛出
+ VoiceConnectionError: 其他连接错误时抛出
+ """
+ async with self._connection_lock:
+ try:
+ # 获取服务器和频道对象
+ guild = self.bot.get_guild(guild_id)
+ if not guild:
+ raise VoiceConnectionError(
+ f"无法找到服务器 {guild_id}",
+ "GUILD_NOT_FOUND",
+ guild_id
+ )
+
+ channel = guild.get_channel(channel_id)
+ if not channel or not isinstance(channel, discord.VoiceChannel):
+ raise VoiceConnectionError(
+ f"无法找到语音频道 {channel_id}",
+ "CHANNEL_NOT_FOUND",
+ guild_id
+ )
+
+ # 验证用户是否在语音频道中(如果提供了用户ID)
+ if user_id:
+ await self._validate_user_in_channel(guild, channel, user_id)
+
+ # 验证机器人权限
+ await self._validate_bot_permissions(channel)
+
+ # 检查是否已有连接
+ if guild_id in self.connections:
+ existing_conn = self.connections[guild_id]
+ if existing_conn.connected and existing_conn.voice_client:
+ if existing_conn.channel_id == channel_id:
+ # 已连接到相同频道,返回现有连接
+ await self.logger.info(f"复用现有语音连接: {guild.name} -> {channel.name}")
+ return existing_conn.voice_client
+ else:
+ # 连接到不同频道,先断开旧连接
+ await self._disconnect_internal(guild_id)
+
+ # 建立新连接
+ voice_client = await channel.connect()
+
+ # 更新连接信息
+ conn_info = VoiceConnectionInfo(guild_id, channel_id, channel.name)
+ conn_info.voice_client = voice_client
+ conn_info.update_status(VoiceConnectionStatus.CONNECTED)
+ conn_info.user_count = len(channel.members)
+ self.connections[guild_id] = conn_info
+
+ await self.logger.info(f"成功连接到语音频道: {guild.name} -> {channel.name}")
+ return voice_client
+
+ except discord.ClientException as e:
+ raise VoiceNetworkError(f"Discord 客户端错误: {str(e)}")
+ except discord.opus.OpusNotLoaded as e:
+ raise VoiceConnectionError(f"Opus 编码器未加载: {str(e)}", "OPUS_NOT_LOADED", guild_id)
+ except Exception as e:
+ await self.logger.error(f"连接语音频道时发生未知错误: {str(e)}")
+ raise VoiceConnectionError(f"连接失败: {str(e)}", "UNKNOWN_ERROR", guild_id)
+
+ async def leave_voice_channel(self, guild_id: int) -> bool:
+ """
+ 离开语音频道
+
+ 断开指定服务器的语音连接,清理相关资源和状态信息。
+ 确保音频播放停止后再断开连接。
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+
+ Returns:
+ bool: 断开是否成功
+ """
+ async with self._connection_lock:
+ return await self._disconnect_internal(guild_id)
+
+ async def _disconnect_internal(self, guild_id: int) -> bool:
+ """
+ 内部断开连接方法
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+
+ Returns:
+ bool: 断开是否成功
+ """
+ if guild_id not in self.connections:
+ return True
+
+ conn_info = self.connections[guild_id]
+
+ try:
+ if conn_info.voice_client and conn_info.voice_client.is_connected():
+ # 停止当前播放
+ if conn_info.voice_client.is_playing():
+ conn_info.voice_client.stop()
+
+ # 等待播放完全停止
+ await asyncio.sleep(0.1)
+
+ # 断开连接
+ await conn_info.voice_client.disconnect()
+
+ conn_info.update_status(VoiceConnectionStatus.IDLE)
+ del self.connections[guild_id]
+
+ await self.logger.info(f"已断开语音连接: Guild {guild_id}")
+ return True
+
+ except Exception as e:
+ await self.logger.error(f"断开语音连接时发生错误: {str(e)}")
+ # 即使出错也要清理连接记录
+ conn_info.update_status(VoiceConnectionStatus.FAILED)
+ if guild_id in self.connections:
+ del self.connections[guild_id]
+ return False
+
+ async def get_voice_client(self, guild_id: int) -> typing.Optional[discord.VoiceClient]:
+ """
+ 获取语音客户端
+
+ 返回指定服务器的语音客户端实例,如果未连接则返回 None。
+ 会验证连接的有效性,自动清理无效连接。
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+
+ Returns:
+ Optional[discord.VoiceClient]: 语音客户端实例或 None
+ """
+ if guild_id not in self.connections:
+ return None
+
+ conn_info = self.connections[guild_id]
+
+ # 验证连接是否仍然有效
+ if conn_info.voice_client and not conn_info.voice_client.is_connected():
+ # 连接已失效,清理状态
+ await self._disconnect_internal(guild_id)
+ return None
+
+ return conn_info.voice_client if conn_info.connected else None
+
+ async def is_connected_to_voice(self, guild_id: int) -> bool:
+ """
+ 检查是否连接到语音频道
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+
+ Returns:
+ bool: 是否已连接
+ """
+ if guild_id not in self.connections:
+ return False
+
+ conn_info = self.connections[guild_id]
+
+ # 检查实际连接状态
+ if conn_info.voice_client and not conn_info.voice_client.is_connected():
+ # 连接已失效,清理状态
+ await self._disconnect_internal(guild_id)
+ return False
+
+ return conn_info.connected
+
+ async def get_connection_status(self, guild_id: int) -> typing.Optional[dict]:
+ """
+ 获取连接状态信息
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+
+ Returns:
+ Optional[dict]: 连接状态信息字典或 None
+ """
+ if guild_id not in self.connections:
+ return None
+
+ conn_info = self.connections[guild_id]
+
+ # 更新实时信息
+ if conn_info.voice_client and conn_info.voice_client.is_connected():
+ conn_info.latency = conn_info.voice_client.latency * 1000 # 转换为毫秒
+ conn_info.connection_health = "good" if conn_info.latency < 100 else "poor"
+
+ # 更新频道用户数
+ guild = self.bot.get_guild(guild_id)
+ if guild:
+ channel = guild.get_channel(conn_info.channel_id)
+ if channel and isinstance(channel, discord.VoiceChannel):
+ conn_info.user_count = len(channel.members)
+
+ return conn_info.to_dict()
+
+ async def list_active_connections(self) -> typing.List[dict]:
+ """
+ 列出所有活跃连接
+
+ @author: @ydzat
+
+ Returns:
+ List[dict]: 活跃连接列表
+ """
+ active_connections = []
+
+ for guild_id, conn_info in self.connections.items():
+ if conn_info.connected:
+ status = await self.get_connection_status(guild_id)
+ if status:
+ active_connections.append(status)
+
+ return active_connections
+
+ async def get_voice_channel_info(self, guild_id: int, channel_id: int) -> typing.Optional[dict]:
+ """
+ 获取语音频道信息
+
+ @author: @ydzat
+
+ Args:
+ guild_id (int): 服务器ID
+ channel_id (int): 频道ID
+
+ Returns:
+ Optional[dict]: 频道信息字典或 None
+ """
+ guild = self.bot.get_guild(guild_id)
+ if not guild:
+ return None
+
+ channel = guild.get_channel(channel_id)
+ if not channel or not isinstance(channel, discord.VoiceChannel):
+ return None
+
+ # 获取用户信息
+ users = []
+ for member in channel.members:
+ users.append({
+ "id": member.id,
+ "name": member.display_name,
+ "status": str(member.status),
+ "is_bot": member.bot
+ })
+
+ # 获取权限信息
+ bot_member = guild.me
+ permissions = channel.permissions_for(bot_member)
+
+ return {
+ "channel_id": channel_id,
+ "channel_name": channel.name,
+ "guild_id": guild_id,
+ "guild_name": guild.name,
+ "user_limit": channel.user_limit,
+ "current_users": users,
+ "user_count": len(users),
+ "bitrate": channel.bitrate,
+ "permissions": {
+ "connect": permissions.connect,
+ "speak": permissions.speak,
+ "use_voice_activation": permissions.use_voice_activation,
+ "priority_speaker": permissions.priority_speaker
+ }
+ }
+
+ async def _validate_user_in_channel(self, guild: discord.Guild,
+ channel: discord.VoiceChannel, user_id: int):
+ """
+ 验证用户是否在语音频道中
+
+ @author: @ydzat
+
+ Args:
+ guild: Discord 服务器对象
+ channel: 语音频道对象
+ user_id: 用户ID
+
+ Raises:
+ VoicePermissionError: 用户不在频道中时抛出
+ """
+ member = guild.get_member(user_id)
+ if not member:
+ raise VoicePermissionError(
+ f"无法找到用户 {user_id}",
+ ["member_not_found"],
+ user_id,
+ channel.id
+ )
+
+ if not member.voice or member.voice.channel != channel:
+ raise VoicePermissionError(
+ f"用户 {member.display_name} 不在语音频道 {channel.name} 中",
+ ["user_not_in_channel"],
+ user_id,
+ channel.id
+ )
+
+ async def _validate_bot_permissions(self, channel: discord.VoiceChannel):
+ """
+ 验证机器人权限
+
+ @author: @ydzat
+
+ Args:
+ channel: 语音频道对象
+
+ Raises:
+ VoicePermissionError: 权限不足时抛出
+ """
+ bot_member = channel.guild.me
+ permissions = channel.permissions_for(bot_member)
+
+ missing_permissions = []
+
+ if not permissions.connect:
+ missing_permissions.append("connect")
+ if not permissions.speak:
+ missing_permissions.append("speak")
+
+ if missing_permissions:
+ raise VoicePermissionError(
+ f"机器人在频道 {channel.name} 中缺少权限: {', '.join(missing_permissions)}",
+ missing_permissions,
+ channel_id=channel.id
+ )
+
+ async def cleanup_inactive_connections(self):
+ """
+ 清理无效连接
+
+ 定期检查并清理已断开或无效的语音连接,释放资源。
+
+ @author: @ydzat
+ """
+ cleanup_guilds = []
+
+ for guild_id, conn_info in self.connections.items():
+ if not conn_info.voice_client or not conn_info.voice_client.is_connected():
+ cleanup_guilds.append(guild_id)
+
+ for guild_id in cleanup_guilds:
+ await self._disconnect_internal(guild_id)
+
+ if cleanup_guilds:
+ await self.logger.info(f"清理了 {len(cleanup_guilds)} 个无效的语音连接")
+
+ async def start_monitoring(self):
+ """
+ 开始连接监控
+
+ @author: @ydzat
+ """
+ if self._cleanup_task is None and self._monitoring_enabled:
+ self._cleanup_task = asyncio.create_task(self._monitoring_loop())
+
+ async def stop_monitoring(self):
+ """
+ 停止连接监控
+
+ @author: @ydzat
+ """
+ self._monitoring_enabled = False
+ if self._cleanup_task:
+ self._cleanup_task.cancel()
+ try:
+ await self._cleanup_task
+ except asyncio.CancelledError:
+ pass
+ self._cleanup_task = None
+
+ async def _monitoring_loop(self):
+ """
+ 监控循环
+
+ @author: @ydzat
+ """
+ try:
+ while self._monitoring_enabled:
+ await asyncio.sleep(60) # 每分钟检查一次
+ await self.cleanup_inactive_connections()
+ except asyncio.CancelledError:
+ pass
+
+ async def disconnect_all(self):
+ """
+ 断开所有连接
+
+ @author: @ydzat
+ """
+ async with self._connection_lock:
+ guild_ids = list(self.connections.keys())
+ for guild_id in guild_ids:
+ await self._disconnect_internal(guild_id)
+
+ await self.stop_monitoring()
class DiscordMessageConverter(adapter.MessageConverter):
@@ -238,6 +815,9 @@ class DiscordAdapter(adapter.MessagePlatformAdapter):
self.logger = logger
self.bot_account_id = self.config['client_id']
+
+ # 初始化语音连接管理器
+ self.voice_manager: VoiceConnectionManager = None
adapter_self = self
@@ -258,6 +838,169 @@ class DiscordAdapter(adapter.MessagePlatformAdapter):
args['proxy'] = os.getenv('http_proxy')
self.bot = MyClient(intents=intents, **args)
+
+ # Voice functionality methods
+ async def join_voice_channel(self, guild_id: int, channel_id: int,
+ user_id: int = None) -> discord.VoiceClient:
+ """
+ 加入语音频道
+
+ 为指定服务器的语音频道建立连接,支持用户权限验证和连接复用。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Args:
+ guild_id (int): Discord 服务器ID
+ channel_id (int): 语音频道ID
+ user_id (int, optional): 请求用户ID,用于权限验证
+
+ Returns:
+ discord.VoiceClient: 语音客户端实例
+
+ Raises:
+ VoicePermissionError: 权限不足
+ VoiceNetworkError: 网络连接失败
+ VoiceConnectionError: 其他连接错误
+ """
+ if not self.voice_manager:
+ raise VoiceConnectionError("语音管理器未初始化", "MANAGER_NOT_READY")
+
+ return await self.voice_manager.join_voice_channel(guild_id, channel_id, user_id)
+
+ async def leave_voice_channel(self, guild_id: int) -> bool:
+ """
+ 离开语音频道
+
+ 断开指定服务器的语音连接,清理相关资源。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Args:
+ guild_id (int): Discord 服务器ID
+
+ Returns:
+ bool: 是否成功断开连接
+ """
+ if not self.voice_manager:
+ return False
+
+ return await self.voice_manager.leave_voice_channel(guild_id)
+
+ async def get_voice_client(self, guild_id: int) -> typing.Optional[discord.VoiceClient]:
+ """
+ 获取语音客户端
+
+ 返回指定服务器的语音客户端实例,用于音频播放控制。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Args:
+ guild_id (int): Discord 服务器ID
+
+ Returns:
+ Optional[discord.VoiceClient]: 语音客户端实例或 None
+ """
+ if not self.voice_manager:
+ return None
+
+ return await self.voice_manager.get_voice_client(guild_id)
+
+ async def is_connected_to_voice(self, guild_id: int) -> bool:
+ """
+ 检查语音连接状态
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Args:
+ guild_id (int): Discord 服务器ID
+
+ Returns:
+ bool: 是否已连接到语音频道
+ """
+ if not self.voice_manager:
+ return False
+
+ return await self.voice_manager.is_connected_to_voice(guild_id)
+
+ async def get_voice_connection_status(self, guild_id: int) -> typing.Optional[dict]:
+ """
+ 获取语音连接详细状态
+
+ 返回包含连接时间、延迟、用户数等详细信息的状态字典。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Args:
+ guild_id (int): Discord 服务器ID
+
+ Returns:
+ Optional[dict]: 连接状态信息或 None
+ """
+ if not self.voice_manager:
+ return None
+
+ return await self.voice_manager.get_connection_status(guild_id)
+
+ async def list_active_voice_connections(self) -> typing.List[dict]:
+ """
+ 列出所有活跃的语音连接
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Returns:
+ List[dict]: 活跃语音连接列表
+ """
+ if not self.voice_manager:
+ return []
+
+ return await self.voice_manager.list_active_connections()
+
+ async def get_voice_channel_info(self, guild_id: int, channel_id: int) -> typing.Optional[dict]:
+ """
+ 获取语音频道详细信息
+
+ 包括频道名称、用户列表、权限信息等。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+
+ Args:
+ guild_id (int): Discord 服务器ID
+ channel_id (int): 语音频道ID
+
+ Returns:
+ Optional[dict]: 频道信息字典或 None
+ """
+ if not self.voice_manager:
+ return None
+
+ return await self.voice_manager.get_voice_channel_info(guild_id, channel_id)
+
+ async def cleanup_voice_connections(self):
+ """
+ 清理无效的语音连接
+
+ 手动触发语音连接清理,移除已断开或无效的连接。
+
+ @author: @ydzat
+ @version: 1.0
+ @since: 2025-07-04
+ """
+ if self.voice_manager:
+ await self.voice_manager.cleanup_inactive_connections()
async def send_message(self, target_type: str, target_id: str, message: platform_message.MessageChain):
msg_to_send, image_files = await self.message_converter.yiri2target(message)
@@ -324,9 +1067,32 @@ class DiscordAdapter(adapter.MessagePlatformAdapter):
self.listeners.pop(event_type)
async def run_async(self):
+ """
+ 启动 Discord 适配器
+
+ 初始化语音管理器并启动 Discord 客户端连接。
+
+ @author: @ydzat (修改)
+ """
async with self.bot:
+ # 初始化语音管理器
+ self.voice_manager = VoiceConnectionManager(self.bot, self.logger)
+ await self.voice_manager.start_monitoring()
+
+ await self.logger.info("Discord 适配器语音功能已启用")
await self.bot.start(self.config['token'], reconnect=True)
async def kill(self) -> bool:
+ """
+ 关闭 Discord 适配器
+
+ 清理语音连接并关闭 Discord 客户端。
+
+ @author: @ydzat (修改)
+ """
+ if self.voice_manager:
+ await self.voice_manager.disconnect_all()
+
await self.bot.close()
return True
+
diff --git a/pkg/platform/sources/wechatpad.py b/pkg/platform/sources/wechatpad.py
index 75cad727..120daa9b 100644
--- a/pkg/platform/sources/wechatpad.py
+++ b/pkg/platform/sources/wechatpad.py
@@ -29,10 +29,11 @@ import logging
class WeChatPadMessageConverter(adapter.MessageConverter):
- def __init__(self, config: dict):
+
+ def __init__(self, config: dict, logger: logging.Logger):
self.config = config
- self.bot = WeChatPadClient(self.config['wechatpad_url'], self.config['token'])
- self.logger = logging.getLogger('WeChatPadMessageConverter')
+ self.bot = WeChatPadClient(self.config["wechatpad_url"],self.config["token"])
+ self.logger = logger
@staticmethod
async def yiri2target(message_chain: platform_message.MessageChain) -> list[dict]:
@@ -73,20 +74,34 @@ class WeChatPadMessageConverter(adapter.MessageConverter):
return content_list
- async def target2yiri(self, message: dict, bot_account_id: str) -> platform_message.MessageChain:
+ async def target2yiri(
+ self,
+ message: dict,
+ bot_account_id: str,
+ ) -> platform_message.MessageChain:
"""外部消息转平台消息"""
# 数据预处理
message_list = []
+ bot_wxid = self.config['wxid']
ats_bot = False # 是否被@
content = message['content']['str']
content_no_preifx = content # 群消息则去掉前缀
is_group_message = self._is_group_message(message)
if is_group_message:
ats_bot = self._ats_bot(message, bot_account_id)
- if '@所有人' in content:
+
+ self.logger.info(f"ats_bot: {ats_bot}; bot_account_id: {bot_account_id}; bot_wxid: {bot_wxid}")
+ if "@所有人" in content:
message_list.append(platform_message.AtAll())
elif ats_bot:
message_list.append(platform_message.At(target=bot_account_id))
+
+ # 解析@信息并生成At组件
+ at_targets = self._extract_at_targets(message)
+ for target_id in at_targets:
+ if target_id != bot_wxid: # 避免重复添加机器人的At
+ message_list.append(platform_message.At(target=target_id))
+
content_no_preifx, _ = self._extract_content_and_sender(content)
msg_type = message['msg_type']
@@ -395,6 +410,23 @@ class WeChatPadMessageConverter(adapter.MessageConverter):
finally:
return ats_bot
+ # 提取一下at的wxid列表
+ def _extract_at_targets(self, message: dict) -> list[str]:
+ """从消息中提取被@用户的ID列表"""
+ at_targets = []
+ try:
+ # 从msg_source中解析atuserlist
+ msg_source = message.get('msg_source', '') or ''
+ if len(msg_source) > 0:
+ msg_source_data = ET.fromstring(msg_source)
+ at_user_list = msg_source_data.findtext("atuserlist") or ""
+ if at_user_list:
+ # atuserlist格式通常是逗号分隔的用户ID列表
+ at_targets = [user_id.strip() for user_id in at_user_list.split(',') if user_id.strip()]
+ except Exception as e:
+ self.logger.error(f"_extract_at_targets got except: {e}")
+ return at_targets
+
# 提取一下content前面的sender_id, 和去掉前缀的内容
def _extract_content_and_sender(self, raw_content: str) -> Tuple[str, Optional[str]]:
try:
@@ -418,16 +450,22 @@ class WeChatPadMessageConverter(adapter.MessageConverter):
class WeChatPadEventConverter(adapter.EventConverter):
- def __init__(self, config: dict):
- self.config = config
- self.message_converter = WeChatPadMessageConverter(config)
- self.logger = logging.getLogger('WeChatPadEventConverter')
+ def __init__(self, config: dict, logger: logging.Logger):
+ self.config = config
+ self.message_converter = WeChatPadMessageConverter(config, logger)
+ self.logger = logger
+
@staticmethod
async def yiri2target(event: platform_events.MessageEvent) -> dict:
pass
- async def target2yiri(self, event: dict, bot_account_id: str) -> platform_events.MessageEvent:
+ async def target2yiri(
+ self,
+ event: dict,
+ bot_account_id: str,
+ ) -> platform_events.MessageEvent:
+
# 排除公众号以及微信团队消息
if (
event['from_user_name']['str'].startswith('gh_')
@@ -503,8 +541,8 @@ class WeChatPadAdapter(adapter.MessagePlatformAdapter):
self.logger = logger
self.quart_app = quart.Quart(__name__)
- self.message_converter = WeChatPadMessageConverter(config)
- self.event_converter = WeChatPadEventConverter(config)
+ self.message_converter = WeChatPadMessageConverter(config, ap.logger)
+ self.event_converter = WeChatPadEventConverter(config, ap.logger)
async def ws_message(self, data):
"""处理接收到的消息"""
diff --git a/pkg/provider/modelmgr/requesters/302aichatcmpl.py b/pkg/provider/modelmgr/requesters/302aichatcmpl.py
index bd9aaccd..40a41718 100644
--- a/pkg/provider/modelmgr/requesters/302aichatcmpl.py
+++ b/pkg/provider/modelmgr/requesters/302aichatcmpl.py
@@ -7,7 +7,7 @@ from . import chatcmpl
class AI302ChatCompletions(chatcmpl.OpenAIChatCompletions):
- """302 AI ChatCompletion API 请求器"""
+ """302.AI ChatCompletion API 请求器"""
client: openai.AsyncClient
diff --git a/pkg/utils/announce.py b/pkg/utils/announce.py
index 7108a08c..a6b8539a 100644
--- a/pkg/utils/announce.py
+++ b/pkg/utils/announce.py
@@ -46,7 +46,7 @@ class AnnouncementManager:
async def fetch_all(self) -> list[Announcement]:
"""获取所有公告"""
resp = requests.get(
- url='https://api.github.com/repos/RockChinQ/LangBot/contents/res/announcement.json',
+ url='https://api.github.com/repos/langbot-app/LangBot/contents/res/announcement.json',
proxies=self.ap.proxy_mgr.get_forward_proxies(),
timeout=5,
)
diff --git a/pkg/utils/constants.py b/pkg/utils/constants.py
index 711ebf5d..4886d186 100644
--- a/pkg/utils/constants.py
+++ b/pkg/utils/constants.py
@@ -1,4 +1,4 @@
-semantic_version = 'v4.0.8'
+semantic_version = 'v4.0.9'
required_database_version = 4
"""Tag the version of the database schema, used to check if the database needs to be migrated"""
diff --git a/pkg/utils/version.py b/pkg/utils/version.py
index ec0683c3..b26b1e33 100644
--- a/pkg/utils/version.py
+++ b/pkg/utils/version.py
@@ -29,7 +29,7 @@ class VersionManager:
async def get_release_list(self) -> list:
"""获取发行列表"""
rls_list_resp = requests.get(
- url='https://api.github.com/repos/RockChinQ/LangBot/releases',
+ url='https://api.github.com/repos/langbot-app/LangBot/releases',
proxies=self.ap.proxy_mgr.get_forward_proxies(),
timeout=5,
)
diff --git a/pyproject.toml b/pyproject.toml
index 27a03a92..3bbfe4d3 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -1,6 +1,6 @@
[project]
name = "langbot"
-version = "4.0.7"
+version = "4.0.9"
description = "高稳定、支持扩展、多模态 - 大模型原生即时通信机器人平台"
readme = "README.md"
requires-python = ">=3.10.1"
@@ -19,6 +19,7 @@ dependencies = [
"dashscope>=1.23.2",
"dingtalk-stream>=0.24.0",
"discord-py>=2.5.2",
+ "pynacl>=1.5.0", # Required for Discord voice support
"gewechat-client>=0.1.5",
"lark-oapi>=1.4.15",
"mcp>=1.8.1",
@@ -90,11 +91,13 @@ classifiers = [
[project.urls]
Homepage = "https://langbot.app"
Documentation = "https://docs.langbot.app"
-Repository = "https://github.com/RockChinQ/langbot"
+Repository = "https://github.com/langbot-app/LangBot"
[dependency-groups]
dev = [
"pre-commit>=4.2.0",
+ "pytest>=8.4.1",
+ "pytest-asyncio>=1.0.0",
"ruff>=0.11.9",
]
diff --git a/web/package.json b/web/package.json
index 8cd6b8dc..3ece8e3a 100644
--- a/web/package.json
+++ b/web/package.json
@@ -5,6 +5,7 @@
"scripts": {
"dev": "next dev --turbopack",
"dev:local": "NEXT_PUBLIC_API_BASE_URL=http://localhost:5300 next dev --turbopack",
+ "dev:local:win": "set NEXT_PUBLIC_API_BASE_URL=http://localhost:5300 && next dev --turbopack",
"build": "next build",
"start": "next start",
"lint": "next lint",