feat: KVM虚拟化管理平台初始版本
This commit is contained in:
@@ -0,0 +1,265 @@
|
||||
<template>
|
||||
<div class="vm-list">
|
||||
<!-- 操作栏 -->
|
||||
<div class="toolbar">
|
||||
<el-button type="primary" @click="showCreateDialog = true">
|
||||
<el-icon><Plus /></el-icon> 创建虚拟机
|
||||
</el-button>
|
||||
<el-button @click="loadData">
|
||||
<el-icon><Refresh /></el-icon> 刷新
|
||||
</el-button>
|
||||
</div>
|
||||
|
||||
<!-- 虚拟机列表 -->
|
||||
<el-table :data="vms" stripe style="width: 100%" v-loading="loading">
|
||||
<el-table-column type="expand">
|
||||
<template #default="{ row }">
|
||||
<div class="expand-content">
|
||||
<p><strong>UUID:</strong> {{ row.uuid }}</p>
|
||||
<p><strong>CPU模式:</strong> {{ row.cpu_mode }}</p>
|
||||
<p><strong>OS类型:</strong> {{ row.os_type }}</p>
|
||||
<p v-if="row.disks?.length">
|
||||
<strong>磁盘:</strong>
|
||||
<span v-for="d in row.disks" :key="d.dev" style="margin-right: 12px;">
|
||||
{{ d.dev }} - {{ d.file }} ({{ d.format }})
|
||||
</span>
|
||||
</p>
|
||||
<p v-if="row.interfaces?.length">
|
||||
<strong>网络:</strong>
|
||||
<span v-for="i in row.interfaces" :key="i.mac || i.dev" style="margin-right: 12px;">
|
||||
{{ i.type }} {{ i.network }} {{ i.mac }}
|
||||
<el-tag v-if="i.ip" type="success" size="small">{{ i.ip }}</el-tag>
|
||||
</span>
|
||||
</p>
|
||||
</div>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column prop="name" label="名称" min-width="150">
|
||||
<template #default="{ row }">
|
||||
<el-link type="primary" @click="$router.push(`/vm/${row.name}`)">{{ row.name }}</el-link>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column label="状态" width="100" align="center">
|
||||
<template #default="{ row }">
|
||||
<el-tag :type="stateType(row.state)" size="small" effect="dark">
|
||||
{{ stateLabel(row.state) }}
|
||||
</el-tag>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column prop="vcpus" label="CPU" width="80" align="center" />
|
||||
<el-table-column label="内存" width="110" align="center">
|
||||
<template #default="{ row }">{{ formatMem(row.memory_mb) }}</template>
|
||||
</el-table-column>
|
||||
<el-table-column label="自动启动" width="90" align="center">
|
||||
<template #default="{ row }">
|
||||
<el-tag :type="row.autostart ? 'success' : 'info'" size="small">
|
||||
{{ row.autostart ? '是' : '否' }}
|
||||
</el-tag>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column label="VNC" width="80" align="center">
|
||||
<template #default="{ row }">
|
||||
<span v-if="row.vnc_port > 0">{{ row.vnc_port }}</span>
|
||||
<span v-else>-</span>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column label="操作" width="280" align="center" fixed="right">
|
||||
<template #default="{ row }">
|
||||
<el-button-group>
|
||||
<el-button v-if="row.state === 'shutoff'" type="success" size="small"
|
||||
@click="doAction(row.name, 'start')">启动</el-button>
|
||||
<el-button v-if="row.state === 'running'" type="warning" size="small"
|
||||
@click="doAction(row.name, 'stop')">关机</el-button>
|
||||
<el-button v-if="row.state === 'running'" type="info" size="small"
|
||||
@click="doAction(row.name, 'pause')">暂停</el-button>
|
||||
<el-button v-if="row.state === 'paused'" type="success" size="small"
|
||||
@click="doAction(row.name, 'resume')">恢复</el-button>
|
||||
<el-button v-if="row.state === 'running'" type="danger" size="small"
|
||||
@click="doAction(row.name, 'force_stop')">强制关</el-button>
|
||||
<el-button type="primary" size="small"
|
||||
@click="$router.push(`/vm/${row.name}`)">详情</el-button>
|
||||
<el-button type="danger" size="small"
|
||||
@click="deleteVM(row)">删除</el-button>
|
||||
</el-button-group>
|
||||
</template>
|
||||
</el-table-column>
|
||||
</el-table>
|
||||
|
||||
<!-- 创建虚拟机对话框 -->
|
||||
<el-dialog v-model="showCreateDialog" title="创建虚拟机" width="600px" :close-on-click-modal="false">
|
||||
<el-form :model="createForm" label-width="100px">
|
||||
<el-form-item label="名称">
|
||||
<el-input v-model="createForm.name" placeholder="虚拟机名称" />
|
||||
</el-form-item>
|
||||
<el-row :gutter="16">
|
||||
<el-col :span="12">
|
||||
<el-form-item label="CPU核心">
|
||||
<el-input-number v-model="createForm.vcpus" :min="1" :max="64" />
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
<el-col :span="12">
|
||||
<el-form-item label="内存(MB)">
|
||||
<el-input-number v-model="createForm.memory_mb" :min="512" :step="512" />
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
</el-row>
|
||||
<el-form-item label="磁盘大小">
|
||||
<el-input-number v-model="createForm.disk_gb" :min="5" :max="2000" />
|
||||
<span style="margin-left: 8px; color: #7a8fa3;">GB</span>
|
||||
</el-form-item>
|
||||
<el-form-item label="存储池">
|
||||
<el-select v-model="createForm.pool_name">
|
||||
<el-option v-for="p in poolOptions" :key="p" :label="p" :value="p" />
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item label="网络">
|
||||
<el-select v-model="createForm.network">
|
||||
<el-option v-for="n in networkOptions" :key="n" :label="n" :value="n" />
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item label="ISO镜像">
|
||||
<el-select v-model="createForm.iso_path" clearable placeholder="可选,用于安装系统">
|
||||
<el-option v-for="iso in isoOptions" :key="iso.path" :label="`${iso.name} (${iso.size_gb}GB)`" :value="iso.path" />
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
<template #footer>
|
||||
<el-button @click="showCreateDialog = false">取消</el-button>
|
||||
<el-button type="primary" @click="createVM" :loading="creating">创建</el-button>
|
||||
</template>
|
||||
</el-dialog>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, onMounted } from 'vue'
|
||||
import { ElMessage, ElMessageBox } from 'element-plus'
|
||||
import { Plus, Refresh } from '@element-plus/icons-vue'
|
||||
import api from '../api'
|
||||
|
||||
const loading = ref(false)
|
||||
const creating = ref(false)
|
||||
const vms = ref([])
|
||||
const showCreateDialog = ref(false)
|
||||
const poolOptions = ref([])
|
||||
const networkOptions = ref([])
|
||||
const isoOptions = ref([])
|
||||
|
||||
const createForm = ref({
|
||||
name: '',
|
||||
vcpus: 2,
|
||||
memory_mb: 2048,
|
||||
disk_gb: 20,
|
||||
pool_name: 'default',
|
||||
network: 'default',
|
||||
iso_path: null,
|
||||
})
|
||||
|
||||
function stateType(s) {
|
||||
const map = { running: 'success', shutoff: 'info', paused: 'warning', crashed: 'danger' }
|
||||
return map[s] || 'info'
|
||||
}
|
||||
|
||||
function stateLabel(s) {
|
||||
const map = { running: '运行中', shutoff: '已关闭', paused: '已暂停', blocked: '阻塞', crashed: '崩溃' }
|
||||
return map[s] || s
|
||||
}
|
||||
|
||||
function formatMem(mb) {
|
||||
if (mb >= 1024) return (mb / 1024).toFixed(1) + ' GB'
|
||||
return mb + ' MB'
|
||||
}
|
||||
|
||||
async function loadData() {
|
||||
loading.value = true
|
||||
try {
|
||||
const data = await api.get('/vm/list')
|
||||
vms.value = data.vms || []
|
||||
} catch (e) {}
|
||||
loading.value = false
|
||||
}
|
||||
|
||||
async function loadOptions() {
|
||||
try {
|
||||
const [pools, nets, isos] = await Promise.all([
|
||||
api.get('/storage/pools'),
|
||||
api.get('/network/list'),
|
||||
api.get('/storage/isos'),
|
||||
])
|
||||
poolOptions.value = (pools.pools || []).map(p => p.name)
|
||||
networkOptions.value = (nets.networks || []).map(n => n.name)
|
||||
isoOptions.value = isos.isos || []
|
||||
} catch (e) {}
|
||||
}
|
||||
|
||||
async function doAction(name, action) {
|
||||
const labels = { start: '启动', stop: '关机', force_stop: '强制关机', pause: '暂停', resume: '恢复' }
|
||||
try {
|
||||
await ElMessageBox.confirm(`确定要${labels[action]}虚拟机 ${name} 吗?`, '确认', { type: 'info' })
|
||||
await api.post(`/vm/action/${name}`, { action })
|
||||
ElMessage.success(`${labels[action]}操作已发送`)
|
||||
setTimeout(loadData, 2000)
|
||||
} catch (e) {
|
||||
if (e !== 'cancel') ElMessage.error(e.response?.data?.detail || '操作失败')
|
||||
}
|
||||
}
|
||||
|
||||
async function createVM() {
|
||||
if (!createForm.value.name) {
|
||||
ElMessage.warning('请输入虚拟机名称')
|
||||
return
|
||||
}
|
||||
creating.value = true
|
||||
try {
|
||||
await api.post('/vm/create', createForm.value)
|
||||
ElMessage.success('虚拟机创建成功')
|
||||
showCreateDialog.value = false
|
||||
loadData()
|
||||
} catch (e) {
|
||||
ElMessage.error(e.response?.data?.detail || '创建失败')
|
||||
}
|
||||
creating.value = false
|
||||
}
|
||||
|
||||
async function deleteVM(row) {
|
||||
try {
|
||||
await ElMessageBox.confirm(
|
||||
`确定要删除虚拟机 ${row.name} 吗?此操作不可恢复!`,
|
||||
'危险操作',
|
||||
{ type: 'error', confirmButtonText: '确定删除', confirmButtonClass: 'el-button--danger' }
|
||||
)
|
||||
await api.delete(`/vm/delete/${row.name}`, { params: { force: row.state === 'running' } })
|
||||
ElMessage.success('虚拟机已删除')
|
||||
loadData()
|
||||
} catch (e) {
|
||||
if (e !== 'cancel') ElMessage.error(e.response?.data?.detail || '删除失败')
|
||||
}
|
||||
}
|
||||
|
||||
onMounted(() => {
|
||||
loadData()
|
||||
loadOptions()
|
||||
})
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.toolbar {
|
||||
margin-bottom: 16px;
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.expand-content {
|
||||
padding: 12px 20px;
|
||||
color: #7a8fa3;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.expand-content p {
|
||||
margin: 4px 0;
|
||||
}
|
||||
|
||||
.expand-content strong {
|
||||
color: #c0ccda;
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user