插件管理新增群状态按钮与群开关明细弹窗,后端补充按插件查询群启用状态接口
This commit is contained in:
@@ -6,6 +6,7 @@ import toml
|
|||||||
from flask import Blueprint, request, jsonify, render_template, current_app
|
from flask import Blueprint, request, jsonify, render_template, current_app
|
||||||
|
|
||||||
from admin.dashboard.blueprints.auth import login_required
|
from admin.dashboard.blueprints.auth import login_required
|
||||||
|
from utils.robot_cmd.robot_command import GroupBotManager, PermissionStatus
|
||||||
|
|
||||||
# 创建蓝图
|
# 创建蓝图
|
||||||
plugin_routes = Blueprint('plugin_routes', __name__)
|
plugin_routes = Blueprint('plugin_routes', __name__)
|
||||||
@@ -53,6 +54,86 @@ def get_plugins():
|
|||||||
return jsonify({"success": False, "message": f"获取插件列表失败: {str(e)}"})
|
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/info', methods=['GET'])
|
@plugin_routes.route('/api/plugins/info', methods=['GET'])
|
||||||
@login_required
|
@login_required
|
||||||
def get_plugin_info():
|
def get_plugin_info():
|
||||||
|
|||||||
@@ -93,6 +93,9 @@
|
|||||||
<el-button size="mini" type="info" plain @click="showPluginInfo(scope.row)">
|
<el-button size="mini" type="info" plain @click="showPluginInfo(scope.row)">
|
||||||
详情
|
详情
|
||||||
</el-button>
|
</el-button>
|
||||||
|
<el-button size="mini" type="warning" plain @click="showPluginGroupStatus(scope.row)">
|
||||||
|
群状态
|
||||||
|
</el-button>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
@@ -148,6 +151,75 @@
|
|||||||
</el-descriptions>
|
</el-descriptions>
|
||||||
</div>
|
</div>
|
||||||
</el-dialog>
|
</el-dialog>
|
||||||
|
|
||||||
|
<el-dialog title="插件群状态" :visible.sync="pluginGroupStatusVisible" width="72%" top="5vh">
|
||||||
|
<div class="plugin-group-status-dialog" v-loading="groupStatusLoading">
|
||||||
|
<div v-if="pluginGroupStatusData" class="group-status-header">
|
||||||
|
<div class="group-status-title">
|
||||||
|
{% raw %}{{ pluginGroupStatusData.plugin_name || '未知插件' }}{% endraw %}
|
||||||
|
<span class="group-status-subtitle">
|
||||||
|
{% raw %}{{ `共 ${pluginGroupStatusData.total_group_count || 0} 个群` }}{% endraw %}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="group-status-summary">
|
||||||
|
<el-tag size="small" type="success">已开启 {% raw %}{{ pluginGroupStatusData.enabled_count || 0 }}{% endraw %}</el-tag>
|
||||||
|
<el-tag size="small" type="info">未开启 {% raw %}{{ pluginGroupStatusData.disabled_count || 0 }}{% endraw %}</el-tag>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<el-alert
|
||||||
|
v-if="pluginGroupStatusData && !pluginGroupStatusData.supports_group_switch"
|
||||||
|
title="该插件未接入群级开关能力,当前仅展示群列表,状态默认为未开启。"
|
||||||
|
type="warning"
|
||||||
|
:closable="false"
|
||||||
|
show-icon
|
||||||
|
class="group-status-alert">
|
||||||
|
</el-alert>
|
||||||
|
|
||||||
|
<el-row :gutter="16" v-if="pluginGroupStatusData">
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-card shadow="never" class="group-status-card">
|
||||||
|
<div slot="header" class="group-status-card-header">
|
||||||
|
<span>已开启群</span>
|
||||||
|
<el-tag size="mini" type="success">{% raw %}{{ pluginGroupStatusData.enabled_count || 0 }}{% endraw %}</el-tag>
|
||||||
|
</div>
|
||||||
|
<el-table
|
||||||
|
:data="pluginGroupStatusData.enabled_groups || []"
|
||||||
|
size="mini"
|
||||||
|
max-height="420"
|
||||||
|
empty-text="暂无已开启群">
|
||||||
|
<el-table-column label="群名称" min-width="170">
|
||||||
|
<template slot-scope="scope">
|
||||||
|
{% raw %}{{ scope.row.group_name || scope.row.group_id }}{% endraw %}
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="group_id" label="群ID" min-width="210" show-overflow-tooltip></el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-card>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-card shadow="never" class="group-status-card">
|
||||||
|
<div slot="header" class="group-status-card-header">
|
||||||
|
<span>未开启群</span>
|
||||||
|
<el-tag size="mini" type="info">{% raw %}{{ pluginGroupStatusData.disabled_count || 0 }}{% endraw %}</el-tag>
|
||||||
|
</div>
|
||||||
|
<el-table
|
||||||
|
:data="pluginGroupStatusData.disabled_groups || []"
|
||||||
|
size="mini"
|
||||||
|
max-height="420"
|
||||||
|
empty-text="暂无未开启群">
|
||||||
|
<el-table-column label="群名称" min-width="170">
|
||||||
|
<template slot-scope="scope">
|
||||||
|
{% raw %}{{ scope.row.group_name || scope.row.group_id }}{% endraw %}
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="group_id" label="群ID" min-width="210" show-overflow-tooltip></el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-card>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
</div>
|
||||||
|
</el-dialog>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
@@ -165,7 +237,10 @@
|
|||||||
isEditingConfig: false,
|
isEditingConfig: false,
|
||||||
editedConfig: '',
|
editedConfig: '',
|
||||||
configError: '',
|
configError: '',
|
||||||
configFormat: 'toml'
|
configFormat: 'toml',
|
||||||
|
pluginGroupStatusVisible: false,
|
||||||
|
groupStatusLoading: false,
|
||||||
|
pluginGroupStatusData: null
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -344,6 +419,43 @@
|
|||||||
console.error('获取插件详情出错:', error);
|
console.error('获取插件详情出错:', error);
|
||||||
this.$message.error('获取插件详情出错');
|
this.$message.error('获取插件详情出错');
|
||||||
});
|
});
|
||||||
|
},
|
||||||
|
showPluginGroupStatus(plugin) {
|
||||||
|
// 打开弹窗前先进入加载态,避免用户在慢接口场景下看到旧数据。
|
||||||
|
this.pluginGroupStatusVisible = true;
|
||||||
|
this.groupStatusLoading = true;
|
||||||
|
this.pluginGroupStatusData = null;
|
||||||
|
|
||||||
|
// 统一使用插件模块名查询,和启用/禁用/重载接口参数保持一致。
|
||||||
|
axios.get('/api/plugins/group_status', {
|
||||||
|
params: {
|
||||||
|
plugin_name: plugin.module_name
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(response => {
|
||||||
|
if (response.data.success) {
|
||||||
|
// 接口返回已按“已开启/未开启”拆分好,前端仅做展示。
|
||||||
|
this.pluginGroupStatusData = response.data.data || {
|
||||||
|
enabled_groups: [],
|
||||||
|
disabled_groups: [],
|
||||||
|
enabled_count: 0,
|
||||||
|
disabled_count: 0,
|
||||||
|
total_group_count: 0,
|
||||||
|
supports_group_switch: false
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this.$message.error(response.data.message || '获取插件群状态失败');
|
||||||
|
this.pluginGroupStatusVisible = false;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('获取插件群状态出错:', error);
|
||||||
|
this.$message.error('获取插件群状态出错');
|
||||||
|
this.pluginGroupStatusVisible = false;
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
this.groupStatusLoading = false;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -421,5 +533,26 @@
|
|||||||
.config-actions { margin-bottom: 10px; display: flex; gap: 10px; }
|
.config-actions { margin-bottom: 10px; display: flex; gap: 10px; }
|
||||||
.config-editor { font-family: monospace; font-size: 12px; }
|
.config-editor { font-family: monospace; font-size: 12px; }
|
||||||
.config-error { color: #ef4444; font-size: 12px; margin-top: 5px; }
|
.config-error { color: #ef4444; font-size: 12px; margin-top: 5px; }
|
||||||
|
.plugin-group-status-dialog { min-height: 240px; }
|
||||||
|
.group-status-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
margin-bottom: 14px;
|
||||||
|
gap: 10px;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
.group-status-title { font-size: 16px; font-weight: 600; color: #0f172a; }
|
||||||
|
.group-status-subtitle { margin-left: 8px; font-size: 12px; color: #64748b; font-weight: 500; }
|
||||||
|
.group-status-summary { display: flex; align-items: center; gap: 8px; }
|
||||||
|
.group-status-alert { margin-bottom: 12px; }
|
||||||
|
.group-status-card { border-radius: 12px; }
|
||||||
|
.group-status-card-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #334155;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|||||||
Reference in New Issue
Block a user