添加插件管理功能,显示插件的相关信息。
This commit is contained in:
158
admin/dashboard/blueprints/plugin_routes.py
Normal file
158
admin/dashboard/blueprints/plugin_routes.py
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
import logging
|
||||||
|
from flask import Blueprint, request, jsonify, render_template
|
||||||
|
|
||||||
|
from admin.dashboard.blueprints.auth import login_required
|
||||||
|
from plugin_common.plugin_manager import PluginManager
|
||||||
|
from plugin_common.plugin_registry import PluginRegistry
|
||||||
|
|
||||||
|
# 创建蓝图
|
||||||
|
plugin_routes = Blueprint('plugin_routes', __name__)
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
# 机器人管理页面
|
||||||
|
@plugin_routes.route('/plugins')
|
||||||
|
@login_required
|
||||||
|
def robot_management():
|
||||||
|
return render_template('plugins_manage.html')
|
||||||
|
|
||||||
|
@plugin_routes.route('/api/plugins', methods=['GET'])
|
||||||
|
@login_required
|
||||||
|
def get_plugins():
|
||||||
|
"""获取所有插件列表"""
|
||||||
|
try:
|
||||||
|
# 获取插件注册表
|
||||||
|
plugin_registry = PluginRegistry()
|
||||||
|
plugins = 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/info', methods=['GET'])
|
||||||
|
@login_required
|
||||||
|
def get_plugin_info():
|
||||||
|
"""获取插件详细信息"""
|
||||||
|
try:
|
||||||
|
plugin_name = request.args.get('plugin_name')
|
||||||
|
if not plugin_name:
|
||||||
|
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||||||
|
|
||||||
|
# 获取插件管理器
|
||||||
|
plugin_manager = PluginManager().get_instance()
|
||||||
|
display_name, plugin = 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:
|
||||||
|
data = request.get_json()
|
||||||
|
plugin_name = data.get('plugin_name')
|
||||||
|
if not plugin_name:
|
||||||
|
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||||||
|
|
||||||
|
# 获取插件管理器
|
||||||
|
plugin_manager = PluginManager().get_instance()
|
||||||
|
|
||||||
|
# 启用插件
|
||||||
|
if 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:
|
||||||
|
data = request.get_json()
|
||||||
|
plugin_name = data.get('plugin_name')
|
||||||
|
if not plugin_name:
|
||||||
|
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||||||
|
|
||||||
|
# 获取插件管理器
|
||||||
|
plugin_manager = PluginManager().get_instance()
|
||||||
|
|
||||||
|
# 禁用插件
|
||||||
|
if 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:
|
||||||
|
data = request.get_json()
|
||||||
|
plugin_name = data.get('plugin_name')
|
||||||
|
if not plugin_name:
|
||||||
|
return jsonify({"success": False, "message": "缺少插件名称参数"})
|
||||||
|
|
||||||
|
# 获取插件管理器
|
||||||
|
plugin_manager = PluginManager().get_instance()
|
||||||
|
|
||||||
|
# 重载插件
|
||||||
|
reloaded_plugin = 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)}"})
|
||||||
@@ -148,6 +148,11 @@
|
|||||||
<i class="el-icon-notebook-1"></i>
|
<i class="el-icon-notebook-1"></i>
|
||||||
<span slot="title">通讯录管理</span>
|
<span slot="title">通讯录管理</span>
|
||||||
</el-menu-item>
|
</el-menu-item>
|
||||||
|
<!-- 在导航菜单中添加插件管理选项 -->
|
||||||
|
<el-menu-item index="11">
|
||||||
|
<i class="el-icon-s-tools"></i>
|
||||||
|
<span slot="title">插件管理</span>
|
||||||
|
</el-menu-item>
|
||||||
</el-menu>
|
</el-menu>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
200
admin/dashboard/templates/plugins_manage.html
Normal file
200
admin/dashboard/templates/plugins_manage.html
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
{% extends "base.html" %}
|
||||||
|
|
||||||
|
{% block title %}插件管理 - 机器人管理后台{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<!-- 插件管理 -->
|
||||||
|
<div>
|
||||||
|
<el-row {% raw %}:gutter="20"{% endraw %}>
|
||||||
|
<el-col {% raw %}:span="24"{% endraw %}>
|
||||||
|
<el-card shadow="hover">
|
||||||
|
<div slot="header">
|
||||||
|
<span>插件管理</span>
|
||||||
|
<el-button style="float: right; padding: 3px 0" type="text" {% raw %}@click="refreshPlugins"{% endraw %}>
|
||||||
|
<i class="el-icon-refresh"></i> 刷新
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
<el-table {% raw %}:data="plugins"{% endraw %} style="width: 100%" border>
|
||||||
|
<el-table-column prop="name" label="插件名称"></el-table-column>
|
||||||
|
<el-table-column prop="module_name" label="模块名称"></el-table-column>
|
||||||
|
<el-table-column prop="version" label="版本"></el-table-column>
|
||||||
|
<el-table-column prop="author" label="作者"></el-table-column>
|
||||||
|
<el-table-column prop="description" label="描述" show-overflow-tooltip></el-table-column>
|
||||||
|
<el-table-column label="状态">
|
||||||
|
<template slot-scope="scope">
|
||||||
|
<el-tag {% raw %}:type="scope.row.status === 'RUNNING' ? 'success' : 'danger'"{% endraw %}>
|
||||||
|
{% raw %}{{ scope.row.status === 'RUNNING' ? '已启用' : '已禁用' }}{% endraw %}
|
||||||
|
</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="操作" width="280">
|
||||||
|
<template slot-scope="scope">
|
||||||
|
<el-button
|
||||||
|
size="mini"
|
||||||
|
{% raw %}:type="scope.row.status === 'RUNNING' ? 'danger' : 'success'"{% endraw %}
|
||||||
|
{% raw %}@click="togglePluginStatus(scope.row)"{% endraw %}>
|
||||||
|
{% raw %}{{ scope.row.status === 'RUNNING' ? '禁用' : '启用' }}{% endraw %}
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
size="mini"
|
||||||
|
type="primary"
|
||||||
|
{% raw %}@click="reloadPlugin(scope.row)"{% endraw %}>
|
||||||
|
重载
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
size="mini"
|
||||||
|
type="info"
|
||||||
|
{% raw %}@click="showPluginInfo(scope.row)"{% endraw %}>
|
||||||
|
详情
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-card>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<!-- 插件详情对话框 -->
|
||||||
|
<el-dialog title="插件详情" {% raw %}:visible.sync="pluginInfoVisible"{% endraw %} width="50%">
|
||||||
|
<div v-if="selectedPlugin">
|
||||||
|
<el-descriptions border direction="vertical" :column="1">
|
||||||
|
<el-descriptions-item label="插件名称">{% raw %}{{ selectedPlugin.name }}{% endraw %}</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="模块名称">{% raw %}{{ selectedPlugin.module_name }}{% endraw %}</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="版本">{% raw %}{{ selectedPlugin.version }}{% endraw %}</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="作者">{% raw %}{{ selectedPlugin.author }}{% endraw %}</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="描述">{% raw %}{{ selectedPlugin.description }}{% endraw %}</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="状态">
|
||||||
|
<el-tag {% raw %}:type="selectedPlugin.status === 'RUNNING' ? 'success' : 'danger'"{% endraw %}>
|
||||||
|
{% raw %}{{ selectedPlugin.status === 'RUNNING' ? '已启用' : '已禁用' }}{% endraw %}
|
||||||
|
</el-tag>
|
||||||
|
</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="命令前缀" v-if="selectedPlugin.command_prefix !== undefined">
|
||||||
|
{% raw %}{{ selectedPlugin.command_prefix || '无' }}{% endraw %}
|
||||||
|
</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="命令列表" v-if="selectedPlugin.commands && selectedPlugin.commands.length > 0">
|
||||||
|
<el-tag v-for="cmd in selectedPlugin.commands" :key="cmd" style="margin-right: 5px; margin-bottom: 5px;">
|
||||||
|
{% raw %}{{ cmd }}{% endraw %}
|
||||||
|
</el-tag>
|
||||||
|
</el-descriptions-item>
|
||||||
|
<el-descriptions-item label="配置信息" v-if="selectedPlugin.config">
|
||||||
|
<pre style="max-height: 300px; overflow-y: auto;">{% raw %}{{ JSON.stringify(selectedPlugin.config, null, 2) }}{% endraw %}</pre>
|
||||||
|
</el-descriptions-item>
|
||||||
|
</el-descriptions>
|
||||||
|
</div>
|
||||||
|
</el-dialog>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block scripts %}
|
||||||
|
<script>
|
||||||
|
new Vue({
|
||||||
|
el: '#app',
|
||||||
|
mixins: [baseApp],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
plugins: [],
|
||||||
|
selectedPlugin: null,
|
||||||
|
pluginInfoVisible: false,
|
||||||
|
loading: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.currentView = '11'; // 设置当前菜单项
|
||||||
|
this.loadPlugins();
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
loadPlugins() {
|
||||||
|
this.loading = true;
|
||||||
|
axios.get('/api/plugins')
|
||||||
|
.then(response => {
|
||||||
|
if (response.data.success) {
|
||||||
|
this.plugins = response.data.data || [];
|
||||||
|
} else {
|
||||||
|
this.$message.error(response.data.message || '加载插件列表失败');
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('加载插件列表出错:', error);
|
||||||
|
this.$message.error('加载插件列表出错');
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
this.loading = false;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
refreshPlugins() {
|
||||||
|
this.loadPlugins();
|
||||||
|
this.$message.success('插件列表已刷新');
|
||||||
|
},
|
||||||
|
togglePluginStatus(plugin) {
|
||||||
|
const action = plugin.status === 'RUNNING' ? 'disable' : 'enable';
|
||||||
|
const actionText = plugin.status === 'RUNNING' ? '禁用' : '启用';
|
||||||
|
|
||||||
|
this.$confirm(`确定要${actionText}插件 "${plugin.name}" 吗?`, '提示', {
|
||||||
|
confirmButtonText: '确定',
|
||||||
|
cancelButtonText: '取消',
|
||||||
|
type: 'warning'
|
||||||
|
}).then(() => {
|
||||||
|
axios.post(`/api/plugins/${action}`, {
|
||||||
|
plugin_name: plugin.module_name
|
||||||
|
})
|
||||||
|
.then(response => {
|
||||||
|
if (response.data.success) {
|
||||||
|
this.$message.success(`${actionText}插件成功`);
|
||||||
|
this.loadPlugins(); // 重新加载插件列表
|
||||||
|
} else {
|
||||||
|
this.$message.error(response.data.message || `${actionText}插件失败`);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error(`${actionText}插件出错:`, error);
|
||||||
|
this.$message.error(`${actionText}插件出错`);
|
||||||
|
});
|
||||||
|
}).catch(() => {
|
||||||
|
this.$message.info('已取消操作');
|
||||||
|
});
|
||||||
|
},
|
||||||
|
reloadPlugin(plugin) {
|
||||||
|
this.$confirm(`确定要重载插件 "${plugin.name}" 吗?`, '提示', {
|
||||||
|
confirmButtonText: '确定',
|
||||||
|
cancelButtonText: '取消',
|
||||||
|
type: 'warning'
|
||||||
|
}).then(() => {
|
||||||
|
axios.post('/api/plugins/reload', {
|
||||||
|
plugin_name: plugin.module_name
|
||||||
|
})
|
||||||
|
.then(response => {
|
||||||
|
if (response.data.success) {
|
||||||
|
this.$message.success('重载插件成功');
|
||||||
|
this.loadPlugins(); // 重新加载插件列表
|
||||||
|
} else {
|
||||||
|
this.$message.error(response.data.message || '重载插件失败');
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('重载插件出错:', error);
|
||||||
|
this.$message.error('重载插件出错');
|
||||||
|
});
|
||||||
|
}).catch(() => {
|
||||||
|
this.$message.info('已取消操作');
|
||||||
|
});
|
||||||
|
},
|
||||||
|
showPluginInfo(plugin) {
|
||||||
|
// 获取插件详细信息
|
||||||
|
axios.get(`/api/plugins/info?plugin_name=${plugin.module_name}`)
|
||||||
|
.then(response => {
|
||||||
|
if (response.data.success) {
|
||||||
|
this.selectedPlugin = response.data.data;
|
||||||
|
this.pluginInfoVisible = true;
|
||||||
|
} else {
|
||||||
|
this.$message.error(response.data.message || '获取插件详情失败');
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('获取插件详情出错:', error);
|
||||||
|
this.$message.error('获取插件详情出错');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
{% endblock %}
|
||||||
@@ -22,6 +22,7 @@ class PermissionStatus(Enum):
|
|||||||
DISABLED = "disabled"
|
DISABLED = "disabled"
|
||||||
|
|
||||||
|
|
||||||
|
# 在Feature枚举类中添加新的功能权限
|
||||||
class Feature(Enum):
|
class Feature(Enum):
|
||||||
"""功能权限枚举,带序号"""
|
"""功能权限枚举,带序号"""
|
||||||
ROBOT = 1, "群机器人"
|
ROBOT = 1, "群机器人"
|
||||||
@@ -42,6 +43,7 @@ class Feature(Enum):
|
|||||||
GROUP_ADD = 16, "加群提醒功能"
|
GROUP_ADD = 16, "加群提醒功能"
|
||||||
DOUYIN_PARSER = 17, "抖音链接转视频功能"
|
DOUYIN_PARSER = 17, "抖音链接转视频功能"
|
||||||
GROUP_MEMBER_CHANGE = 18, "群成员变更提醒功能"
|
GROUP_MEMBER_CHANGE = 18, "群成员变更提醒功能"
|
||||||
|
KID_PHOTO_EXTRACT =19, "儿童照片提取转发功能" # 小朋友照片提取功能
|
||||||
|
|
||||||
def __new__(cls, value, description):
|
def __new__(cls, value, description):
|
||||||
obj = object.__new__(cls)
|
obj = object.__new__(cls)
|
||||||
|
|||||||
Reference in New Issue
Block a user