389 lines
16 KiB
Python
389 lines
16 KiB
Python
import json
|
||
from loguru import logger
|
||
import os
|
||
|
||
import toml
|
||
from flask import Blueprint, request, jsonify, render_template, current_app
|
||
|
||
from admin.dashboard.blueprints.auth import login_required
|
||
from utils.robot_cmd.robot_command import GroupBotManager, PermissionStatus
|
||
|
||
# 创建蓝图
|
||
plugin_routes = Blueprint('plugin_routes', __name__)
|
||
LOG = logger
|
||
|
||
|
||
# 机器人管理页面
|
||
@plugin_routes.route('/plugins_manage')
|
||
@login_required
|
||
def robot_management():
|
||
return render_template('plugins_manage.html')
|
||
|
||
|
||
@plugin_routes.route('/api/plugins', methods=['GET'])
|
||
@login_required
|
||
def get_plugins():
|
||
"""获取所有插件列表"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
# 获取插件注册表
|
||
plugins = server.plugin_registry.get_all_plugins()
|
||
|
||
# 转换为前端需要的格式
|
||
plugin_list = []
|
||
for name, plugin in plugins.items():
|
||
# 获取插件模块名
|
||
try:
|
||
module_name = plugin.__class__.__module__.split('.')[-2]
|
||
except (IndexError, AttributeError):
|
||
module_name = "unknown"
|
||
|
||
plugin_info = {
|
||
"name": plugin.name,
|
||
"module_name": module_name,
|
||
"version": getattr(plugin, 'version', 'N/A'),
|
||
"author": getattr(plugin, 'author', 'N/A'),
|
||
"description": getattr(plugin, 'description', 'N/A'),
|
||
"status": plugin.status.name if hasattr(plugin, 'status') else 'UNKNOWN'
|
||
}
|
||
plugin_list.append(plugin_info)
|
||
|
||
return jsonify({"success": True, "data": plugin_list})
|
||
except Exception as e:
|
||
LOG.error(f"获取插件列表失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"获取插件列表失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/group_status', methods=['GET'])
|
||
@login_required
|
||
def get_plugin_group_status():
|
||
"""获取单个插件在各群的启用状态(已启用/未启用)"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
plugin_name = request.args.get('plugin_name')
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
|
||
# 通过现有插件管理器查找插件实例,兼容传入模块名或展示名两种形式。
|
||
display_name, plugin = server.plugin_manager.find_plugin_by_name(plugin_name)
|
||
if not plugin:
|
||
return jsonify({"success": False, "message": f"未找到插件: {plugin_name}"})
|
||
|
||
# 统一构建群列表:优先使用通讯录中的群(覆盖面更全),并补齐机器人管理列表中的群。
|
||
group_contacts = {}
|
||
try:
|
||
group_contacts = (server.contact_manager.get_group_contacts() or {})
|
||
except Exception:
|
||
# 这里降级为空字典,后续仍会使用 GroupBotManager 群列表,不中断主流程。
|
||
group_contacts = {}
|
||
|
||
# 从群机器人管理缓存中取群,确保“已托管但通讯录暂缺”的群也会展示出来。
|
||
managed_groups = set(GroupBotManager.get_group_list() or [])
|
||
contact_groups = set(group_contacts.keys())
|
||
all_group_ids = sorted(contact_groups | managed_groups)
|
||
|
||
# 再做一次兜底:极端情况下如果前两者都为空,尝试读取本地缓存中的群集合。
|
||
if not all_group_ids and hasattr(GroupBotManager, "local_cache"):
|
||
all_group_ids = sorted(GroupBotManager.local_cache.get("group_list", set()) or [])
|
||
|
||
# 插件是否支持“按群开关”依赖于插件是否注册了 feature(多数消息插件会注册)。
|
||
plugin_feature = getattr(plugin, "feature", None)
|
||
supports_group_switch = plugin_feature is not None
|
||
feature_key = getattr(plugin_feature, "name", "") if plugin_feature else ""
|
||
feature_description = getattr(plugin_feature, "description", "") if plugin_feature else ""
|
||
|
||
enabled_groups = []
|
||
disabled_groups = []
|
||
|
||
for group_id in all_group_ids:
|
||
# 群名优先从通讯录获取,拿不到时降级为群ID,保证页面总能显示。
|
||
group_name = group_contacts.get(group_id) or server.contact_manager.get_nickname(group_id) or group_id
|
||
group_item = {
|
||
"group_id": group_id,
|
||
"group_name": group_name,
|
||
}
|
||
|
||
# 支持群开关的插件:按 feature 的真实权限划分已开启/未开启。
|
||
if supports_group_switch:
|
||
permission_status = GroupBotManager.get_group_permission(group_id, plugin_feature)
|
||
if permission_status == PermissionStatus.ENABLED:
|
||
enabled_groups.append(group_item)
|
||
else:
|
||
disabled_groups.append(group_item)
|
||
else:
|
||
# 不支持群级开关的插件默认归入“未开启”列表,并通过 supports_group_switch 让前端给提示。
|
||
disabled_groups.append(group_item)
|
||
|
||
return jsonify({
|
||
"success": True,
|
||
"data": {
|
||
"plugin_name": plugin.name,
|
||
"module_name": plugin_name,
|
||
"supports_group_switch": supports_group_switch,
|
||
"feature_key": feature_key,
|
||
"feature_description": feature_description,
|
||
"enabled_count": len(enabled_groups),
|
||
"disabled_count": len(disabled_groups),
|
||
"total_group_count": len(all_group_ids),
|
||
"enabled_groups": enabled_groups,
|
||
"disabled_groups": disabled_groups,
|
||
}
|
||
})
|
||
except Exception as e:
|
||
LOG.error(f"获取插件群状态失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"获取插件群状态失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/group_status/toggle', methods=['POST'])
|
||
@login_required
|
||
def toggle_plugin_group_status():
|
||
"""切换插件在指定群的启用状态(开启/关闭)"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
data = request.get_json() or {}
|
||
plugin_name = (data.get('plugin_name') or '').strip()
|
||
group_id = (data.get('group_id') or '').strip()
|
||
status = (data.get('status') or '').strip().lower()
|
||
|
||
# 参数校验:保证前端传入完整且合法的切换目标。
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
if not group_id:
|
||
return jsonify({"success": False, "message": "缺少群ID参数"})
|
||
if status not in ('enabled', 'disabled'):
|
||
return jsonify({"success": False, "message": "状态参数非法,仅支持 enabled / disabled"})
|
||
|
||
# 查找插件实例,沿用现有查找逻辑,兼容模块名与展示名。
|
||
display_name, plugin = server.plugin_manager.find_plugin_by_name(plugin_name)
|
||
if not plugin:
|
||
return jsonify({"success": False, "message": f"未找到插件: {plugin_name}"})
|
||
|
||
# 插件必须注册了 feature 才能进行群级开关;否则无法映射到权限系统。
|
||
plugin_feature = getattr(plugin, "feature", None)
|
||
if plugin_feature is None:
|
||
return jsonify({"success": False, "message": "该插件未接入群级开关能力,无法直接切换"})
|
||
|
||
# 将字符串状态映射到统一权限枚举,然后写入群权限缓存与Redis。
|
||
target_status = PermissionStatus.ENABLED if status == 'enabled' else PermissionStatus.DISABLED
|
||
GroupBotManager.set_group_permission(group_id, plugin_feature, target_status)
|
||
|
||
# 返回简洁确认信息,方便前端即时提示与二次渲染。
|
||
return jsonify({
|
||
"success": True,
|
||
"message": f"已将群 {group_id} 的插件“{plugin.name}”设置为{'开启' if target_status == PermissionStatus.ENABLED else '关闭'}",
|
||
"data": {
|
||
"plugin_name": plugin.name,
|
||
"group_id": group_id,
|
||
"status": target_status.value,
|
||
"feature_key": getattr(plugin_feature, "name", ""),
|
||
}
|
||
})
|
||
except Exception as e:
|
||
LOG.error(f"切换插件群状态失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"切换插件群状态失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/info', methods=['GET'])
|
||
@login_required
|
||
def get_plugin_info():
|
||
"""获取插件详细信息"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
plugin_name = request.args.get('plugin_name')
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
|
||
# 获取插件管理器
|
||
display_name, plugin = server.plugin_manager.find_plugin_by_name(plugin_name)
|
||
|
||
if not plugin:
|
||
return jsonify({"success": False, "message": f"未找到插件: {plugin_name}"})
|
||
|
||
# 获取插件模块名
|
||
try:
|
||
module_name = plugin.__class__.__module__.split('.')[-2]
|
||
except (IndexError, AttributeError):
|
||
module_name = "unknown"
|
||
|
||
# 构建详细信息
|
||
plugin_info = {
|
||
"name": plugin.name,
|
||
"module_name": module_name,
|
||
"version": getattr(plugin, 'version', 'N/A'),
|
||
"author": getattr(plugin, 'author', 'N/A'),
|
||
"description": getattr(plugin, 'description', 'N/A'),
|
||
"status": plugin.status.name if hasattr(plugin, 'status') else 'UNKNOWN',
|
||
"command_prefix": getattr(plugin, 'command_prefix', ''),
|
||
"commands": getattr(plugin, 'commands', []),
|
||
"config": getattr(plugin, '_config', {})
|
||
}
|
||
|
||
return jsonify({"success": True, "data": plugin_info})
|
||
except Exception as e:
|
||
LOG.error(f"获取插件详情失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"获取插件详情失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/enable', methods=['POST'])
|
||
@login_required
|
||
def enable_plugin():
|
||
"""启用插件"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
data = request.get_json()
|
||
plugin_name = data.get('plugin_name')
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
|
||
# 获取插件管理器
|
||
# 启用插件
|
||
if server.plugin_manager.start_plugin(plugin_name):
|
||
return jsonify({"success": True, "message": f"插件 {plugin_name} 启用成功"})
|
||
else:
|
||
return jsonify({"success": False, "message": f"插件 {plugin_name} 启用失败"})
|
||
except Exception as e:
|
||
LOG.error(f"启用插件失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"启用插件失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/disable', methods=['POST'])
|
||
@login_required
|
||
def disable_plugin():
|
||
"""禁用插件"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
data = request.get_json()
|
||
plugin_name = data.get('plugin_name')
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
|
||
# 禁用插件
|
||
if server.plugin_manager.stop_plugin(plugin_name):
|
||
return jsonify({"success": True, "message": f"插件 {plugin_name} 禁用成功"})
|
||
else:
|
||
return jsonify({"success": False, "message": f"插件 {plugin_name} 禁用失败"})
|
||
except Exception as e:
|
||
LOG.error(f"禁用插件失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"禁用插件失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/reload', methods=['POST'])
|
||
@login_required
|
||
def reload_plugin():
|
||
"""重载插件"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
data = request.get_json()
|
||
plugin_name = data.get('plugin_name')
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
|
||
# 重载插件
|
||
reloaded_plugin = server.plugin_manager.reload_plugin(plugin_name)
|
||
|
||
if reloaded_plugin:
|
||
return jsonify({"success": True, "message": f"插件 {plugin_name} 重载成功"})
|
||
else:
|
||
return jsonify({"success": False, "message": f"插件 {plugin_name} 重载失败"})
|
||
except Exception as e:
|
||
LOG.error(f"重载插件失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"重载插件失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/config/raw', methods=['GET'])
|
||
@login_required
|
||
def get_raw_plugin_config():
|
||
"""获取插件原始配置文件内容"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
plugin_name = request.args.get('plugin_name')
|
||
if not plugin_name:
|
||
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||
|
||
# 获取插件管理器
|
||
|
||
# 查找插件
|
||
display_name, plugin = server.plugin_manager.find_plugin_by_name(plugin_name)
|
||
|
||
if not plugin:
|
||
return jsonify({"success": False, "message": f"未找到插件: {plugin_name}"})
|
||
|
||
# 获取配置文件路径
|
||
config_path = plugin.get_config_path()
|
||
|
||
if not os.path.exists(config_path):
|
||
return jsonify({"success": False, "message": f"配置文件不存在: {config_path}"})
|
||
|
||
# 读取配置文件内容
|
||
with open(config_path, 'r', encoding='utf-8') as f:
|
||
config_text = f.read()
|
||
|
||
# 确定配置文件格式
|
||
format_type = 'toml'
|
||
if config_path.endswith('.json'):
|
||
format_type = 'json'
|
||
elif config_path.endswith('.yaml') or config_path.endswith('.yml'):
|
||
format_type = 'yaml'
|
||
|
||
return jsonify({
|
||
"success": True,
|
||
"data": config_text,
|
||
"format": format_type
|
||
})
|
||
except Exception as e:
|
||
LOG.error(f"获取插件配置文件失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"获取配置文件失败: {str(e)}"})
|
||
|
||
|
||
@plugin_routes.route('/api/plugins/config/update', methods=['POST'])
|
||
@login_required
|
||
def update_plugin_config():
|
||
"""更新插件配置"""
|
||
try:
|
||
server = current_app.dashboard_server
|
||
data = request.json
|
||
plugin_name = data.get('plugin_name')
|
||
config_text = data.get('config_text')
|
||
format_type = data.get('format', 'toml')
|
||
|
||
if not plugin_name or config_text is None:
|
||
return jsonify({"success": False, "message": "缺少必要参数"})
|
||
|
||
# 查找插件
|
||
# 获取插件管理器
|
||
display_name, plugin = server.plugin_manager.find_plugin_by_name(plugin_name)
|
||
|
||
if not plugin:
|
||
return jsonify({"success": False, "message": f"未找到插件: {plugin_name}"})
|
||
|
||
# 获取配置文件路径
|
||
config_path = plugin.get_config_path()
|
||
|
||
# 确保配置目录存在
|
||
os.makedirs(os.path.dirname(config_path), exist_ok=True)
|
||
|
||
# 写入配置文件
|
||
with open(config_path, 'w', encoding='utf-8') as f:
|
||
f.write(config_text)
|
||
|
||
# 解析配置并更新插件内部配置
|
||
try:
|
||
if format_type == 'toml':
|
||
config_obj = toml.loads(config_text)
|
||
elif format_type == 'json':
|
||
config_obj = json.loads(config_text)
|
||
else:
|
||
return jsonify({"success": False, "message": f"不支持的配置格式: {format_type}"})
|
||
|
||
# 更新插件内部配置
|
||
plugin._config = config_obj
|
||
|
||
return jsonify({"success": True, "message": "配置已保存"})
|
||
except Exception as e:
|
||
LOG.error(f"解析配置失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"配置已保存,但解析失败: {str(e)}"})
|
||
|
||
except Exception as e:
|
||
LOG.error(f"更新插件配置失败: {str(e)}", exc_info=True)
|
||
return jsonify({"success": False, "message": f"更新配置失败: {str(e)}"})
|