mirror of
https://github.com/GNS3/gns3-server.git
synced 2025-01-18 15:33:49 +02:00
Prevent corruption of VM in VirtualBox when using linked clone
Fix https://github.com/GNS3/gns3-gui/issues/1821
This commit is contained in:
parent
27a1089806
commit
f0ff035c0b
@ -161,7 +161,7 @@ class VirtualBox(BaseManager):
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def list_vms(self):
|
def list_vms(self, allow_clone=False):
|
||||||
"""
|
"""
|
||||||
Gets VirtualBox VM list.
|
Gets VirtualBox VM list.
|
||||||
"""
|
"""
|
||||||
@ -176,7 +176,7 @@ class VirtualBox(BaseManager):
|
|||||||
if vmname == "<inaccessible>":
|
if vmname == "<inaccessible>":
|
||||||
continue # ignore inaccessible VMs
|
continue # ignore inaccessible VMs
|
||||||
extra_data = yield from self.execute("getextradata", [vmname, "GNS3/Clone"])
|
extra_data = yield from self.execute("getextradata", [vmname, "GNS3/Clone"])
|
||||||
if len(extra_data) == 0 or not extra_data[0].strip() == "Value: yes":
|
if allow_clone or len(extra_data) == 0 or not extra_data[0].strip() == "Value: yes":
|
||||||
# get the amount of RAM
|
# get the amount of RAM
|
||||||
info_results = yield from self.execute("showvminfo", [vmname, "--machinereadable"])
|
info_results = yield from self.execute("showvminfo", [vmname, "--machinereadable"])
|
||||||
ram = 0
|
ram = 0
|
||||||
|
@ -603,9 +603,16 @@ class VirtualBoxVM(BaseNode):
|
|||||||
:param vmname: VirtualBox VM name
|
:param vmname: VirtualBox VM name
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
if vmname == self._vmname:
|
||||||
|
return
|
||||||
|
|
||||||
if self.linked_clone:
|
if self.linked_clone:
|
||||||
if self.status == "started":
|
if self.status == "started":
|
||||||
raise VirtualBoxError("You can't change the name of running VM {}".format(self._name))
|
raise VirtualBoxError("You can't change the name of running VM {}".format(self._name))
|
||||||
|
# We can't rename a VM to name that already exists
|
||||||
|
vms = yield from self.manager.list_vms(allow_clone=True)
|
||||||
|
if vmname in [vm["vmname"] for vm in vms]:
|
||||||
|
raise VirtualBoxError("You can't change the name to {} it's already use in VirtualBox".format(vmname))
|
||||||
yield from self._modify_vm('--name "{}"'.format(vmname))
|
yield from self._modify_vm('--name "{}"'.format(vmname))
|
||||||
|
|
||||||
log.info("VirtualBox VM '{name}' [{id}] has set the VM name to '{vmname}'".format(name=self.name, id=self.id, vmname=vmname))
|
log.info("VirtualBox VM '{name}' [{id}] has set the VM name to '{vmname}'".format(name=self.name, id=self.id, vmname=vmname))
|
||||||
|
@ -112,10 +112,13 @@ class VirtualBoxHandler:
|
|||||||
vbox_manager = VirtualBox.instance()
|
vbox_manager = VirtualBox.instance()
|
||||||
vm = vbox_manager.get_node(request.match_info["node_id"], project_id=request.match_info["project_id"])
|
vm = vbox_manager.get_node(request.match_info["node_id"], project_id=request.match_info["project_id"])
|
||||||
|
|
||||||
if "vmname" in request.json:
|
if "name" in request.json:
|
||||||
vmname = request.json.pop("vmname")
|
name = request.json.pop("name")
|
||||||
if vmname != vm.vmname:
|
vmname = request.json.pop("vmname", None)
|
||||||
yield from vm.set_vmname(vmname)
|
if name != vm.name:
|
||||||
|
vm.name = name
|
||||||
|
if vm.linked_clone:
|
||||||
|
yield from vm.set_vmname(vm.name)
|
||||||
|
|
||||||
if "adapters" in request.json:
|
if "adapters" in request.json:
|
||||||
adapters = int(request.json.pop("adapters"))
|
adapters = int(request.json.pop("adapters"))
|
||||||
|
@ -18,7 +18,7 @@
|
|||||||
import os
|
import os
|
||||||
import pytest
|
import pytest
|
||||||
import asyncio
|
import asyncio
|
||||||
from tests.utils import asyncio_patch
|
from tests.utils import asyncio_patch, AsyncioMagicMock
|
||||||
|
|
||||||
from gns3server.compute.virtualbox.virtualbox_vm import VirtualBoxVM
|
from gns3server.compute.virtualbox.virtualbox_vm import VirtualBoxVM
|
||||||
from gns3server.compute.virtualbox.virtualbox_error import VirtualBoxError
|
from gns3server.compute.virtualbox.virtualbox_error import VirtualBoxError
|
||||||
@ -46,13 +46,30 @@ def test_vm(project, manager):
|
|||||||
|
|
||||||
def test_rename_vmname(project, manager, async_run):
|
def test_rename_vmname(project, manager, async_run):
|
||||||
"""
|
"""
|
||||||
Rename a VM is not allowed when using linked clone
|
Rename a VM is not allowed when using a running linked clone
|
||||||
|
or if the vm already exists in Vbox
|
||||||
"""
|
"""
|
||||||
vm = VirtualBoxVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager, "test", False)
|
vm = VirtualBoxVM("test", "00010203-0405-0607-0809-0a0b0c0d0e0f", project, manager, "test", False)
|
||||||
vm._node_status = "started"
|
vm.manager.list_vms = AsyncioMagicMock(return_value=[{"vmname": "Debian"}])
|
||||||
vm._linked_clone = True
|
vm._linked_clone = True
|
||||||
|
vm._modify_vm = AsyncioMagicMock()
|
||||||
|
|
||||||
|
# Vm is running
|
||||||
|
vm._node_status = "started"
|
||||||
with pytest.raises(VirtualBoxError):
|
with pytest.raises(VirtualBoxError):
|
||||||
async_run(vm.set_vmname("toto"))
|
async_run(vm.set_vmname("Arch"))
|
||||||
|
assert not vm._modify_vm.called
|
||||||
|
|
||||||
|
vm._node_status = "stopped"
|
||||||
|
|
||||||
|
# Name already use
|
||||||
|
with pytest.raises(VirtualBoxError):
|
||||||
|
async_run(vm.set_vmname("Debian"))
|
||||||
|
assert not vm._modify_vm.called
|
||||||
|
|
||||||
|
# Work
|
||||||
|
async_run(vm.set_vmname("Arch"))
|
||||||
|
assert vm._modify_vm.called
|
||||||
|
|
||||||
|
|
||||||
def test_vm_valid_virtualbox_api_version(loop, project, manager):
|
def test_vm_valid_virtualbox_api_version(loop, project, manager):
|
||||||
|
Loading…
Reference in New Issue
Block a user