fix: 虚拟机网络管理改为解绑逻辑,不再删除底层网络
Build and Deploy Vue3 / build (push) Successful in 1m33s
Build and Deploy Vue3 / deploy (push) Successful in 44s

将 user-vm/UserVmDetail 和 virtualization/VmDetail 中的网络删除操作改为通过 updateVm 接口解绑,与绑定网络逻辑对称,避免误删底层物理网络。

Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
shiran
2026-06-30 10:26:01 +08:00
parent cae5551107
commit 0155715278
3 changed files with 48 additions and 19 deletions
+18 -12
View File
@@ -344,7 +344,7 @@
<template #default="{ row }">
<el-button v-if="!row.is_primary" link type="warning" size="small" @click="handleSetVmNetworkPrimary(row)">设为主IP</el-button>
<el-button link type="danger" size="small" :loading="deletingNetworkId === row.id" @click="handleDeleteVmNetwork(row)">
<el-icon :size="14"><Delete /></el-icon>删除
<el-icon :size="14"><Delete /></el-icon>解绑
</el-button>
</template>
</el-table-column>
@@ -1807,21 +1807,27 @@ const resolveNetworkServiceHost = async (row) => {
}
const handleDeleteVmNetwork = (row) => {
ElMessageBox.confirm(
`将删除底层网络「${row.name}」(ID:${row.id}),该操作会影响所有绑定该网络的虚拟机,是否继续`,
'删除网络',
{ confirmButtonText: '确定删除', cancelButtonText: '取消', type: 'warning' }
`确定从该虚拟机解绑网络「${row.name}」(ID:${row.id})`,
'解绑网络',
{ confirmButtonText: '确定', cancelButtonText: '取消', type: 'warning' }
).then(async () => {
deletingNetworkId.value = row.id
try {
const { serviceId, hostId } = await resolveNetworkServiceHost(row)
if (!serviceId) { ElMessage.error('无法获取该网络所属服务ID,删除失败'); return }
const params = { service_id: serviceId, network_id: row.id }
if (hostId) params.host_id = hostId
const res = await deletePointNetwork(params)
if (res?.data?.code === 200) { ElMessage.success('删除成功'); loadDetail() }
else ElMessage.error(extractApiError(res?.data, '删除失败'))
const payload = { user_goods_id: userGoodsId.value }
const remainingNetworks = vmNetworks.value.filter(n => n.id !== row.id)
const bridgeIds = remainingNetworks.filter(n => n.type === 'bridge').map(n => n.id)
const natNet = remainingNetworks.find(n => n.type === 'nat')
payload.network_ids = bridgeIds
if (natNet) payload.internet_network_id = natNet.id
if (vm.value?.rx_bandwidth) payload.rx_bandwidth = vm.value.rx_bandwidth
if (vm.value?.tx_bandwidth) payload.tx_bandwidth = vm.value.tx_bandwidth
if (vm.value?.ssh_port) payload.ssh_port = vm.value.ssh_port
if (inPortGroup.value?.id) payload.port_group_id = inPortGroup.value.id
const res = await updateUserVm(payload)
if (res?.data?.code === 200) { ElMessage.success('解绑网络成功'); loadDetail() }
else ElMessage.error(extractApiError(res?.data, '解绑网络失败'))
} catch (e) {
ElMessage.error(extractApiError(e?.response?.data, '删除失败'))
ElMessage.error(extractApiError(e?.response?.data, '解绑网络失败'))
} finally {
deletingNetworkId.value = 0
}
+10 -1
View File
@@ -257,7 +257,11 @@
</el-tab-pane>
<el-tab-pane label="已购商品" name="5">
<el-table :data="userGoodsList" v-loading="goodsListLoading" stripe style="width: 100%" table-layout="auto">
<el-table-column prop="id" label="ID" width="70" />
<el-table-column prop="id" label="ID" width="70">
<template #default="{row}">
<el-link type="primary" :underline="false" @click="router.push(`/user-goods/detail/${row.id}`)">{{ row.id }}</el-link>
</template>
</el-table-column>
<el-table-column label="商品名称" min-width="120" show-overflow-tooltip>
<template #default="{row}">{{ row.good?.name || '-' }}</template>
</el-table-column>
@@ -280,6 +284,11 @@
<el-table-column label="购买时间" min-width="140">
<template #default="{row}">{{ formatDate(row.CreatedAt) }}</template>
</el-table-column>
<el-table-column label="操作" width="80" fixed="right">
<template #default="{row}">
<el-button type="primary" link size="small" @click="router.push(`/user-goods/detail/${row.id}`)">详情</el-button>
</template>
</el-table-column>
</el-table>
<div class="pagination-wrapper" v-if="goodsListTotal > 0">
<el-pagination
+20 -6
View File
@@ -302,7 +302,7 @@
<el-button link type="primary" size="small" @click="handleNetDetail(row)">详情</el-button>
<el-button link type="primary" size="small" @click="handleNetEdit(row)">编辑</el-button>
<el-button v-if="!row.is_primary" link type="warning" size="small" @click="handleSetPrimary(row)">设为主IP</el-button>
<el-button link type="danger" size="small" @click="handleNetDelete(row)">删除</el-button>
<el-button link type="danger" size="small" @click="handleNetDelete(row)">解绑</el-button>
</template>
</el-table-column>
</el-table>
@@ -2962,13 +2962,27 @@ const submitNetForm = () => {
}
const handleNetDetail = (row) => { netDetailData.value = row; netDetailVisible.value = true }
const handleNetDelete = (row) => {
ElMessageBox.confirm(`确定要删除网络「${row.name}」(ID: ${row.id}) 吗?`, '删除网络', { confirmButtonText: '确定', cancelButtonText: '取消', type: 'warning' })
ElMessageBox.confirm(`确定从该虚拟机解绑网络「${row.name}」(ID: ${row.id}) 吗?`, '解绑网络', { confirmButtonText: '确定', cancelButtonText: '取消', type: 'warning' })
.then(async () => {
actionLoading.value = true
try {
const res = await deleteNetwork({ service_id: serviceId.value, network_id: row.id, host_id: row.host_id })
if (res?.data?.code === 200) { ElMessage.success('删除成功'); loadDetail() }
else ElMessage.error(extractApiError(res?.data, '删除失败'))
} catch (e) { ElMessage.error(extractApiError(e?.response?.data, '删除失败')) }
const fd = new FormData()
fd.append('service_id', serviceId.value)
fd.append('vm_id', vmId.value)
const remainingNetworks = vmNetworks.value.filter(n => n.id !== row.id)
const bridgeIds = remainingNetworks.filter(n => n.type === 'bridge').map(n => n.id)
const natNet = remainingNetworks.find(n => n.type === 'nat')
bridgeIds.forEach(id => fd.append('network_ids', id))
if (natNet) fd.append('internet_network_id', natNet.id)
if (detail.value?.rx_bandwidth) fd.append('rx_bandwidth', detail.value.rx_bandwidth)
if (detail.value?.tx_bandwidth) fd.append('tx_bandwidth', detail.value.tx_bandwidth)
if (detail.value?.ssh_port) fd.append('ssh_port', detail.value.ssh_port)
if (vmPortGroup.value?.id) fd.append('port_group_id', vmPortGroup.value.id)
const res = await updateVm(fd)
if (res?.data?.code === 200) { ElMessage.success('解绑网络成功'); loadDetail() }
else ElMessage.error(extractApiError(res?.data, '解绑网络失败'))
} catch (e) { ElMessage.error(extractApiError(e?.response?.data, '解绑网络失败')) }
finally { actionLoading.value = false }
}).catch(() => {})
}