tests: improve VMs cleanup wrt custom templates
Cleanup VMs in template reverse topological order, not network one. Network can be set to None to break dependency, but template can't. For netvm to be changed, kill VMs first (kill doesn't check network dependency), so netvm change will not trigger side effects (runtime change, which could fail). This fixes cleanup for tests creating custom templates - previously order was undefined and if template was tried removed before its child VMs, it fails. All the relevant files were removed later anyway, but it lead to python objects leaks.
This commit is contained in:
parent
4e762788a9
commit
08ddeee9fb
@ -789,13 +789,6 @@ class SystemTestCase(QubesTestCase):
|
|||||||
vmname = vm.name
|
vmname = vm.name
|
||||||
app = vm.app
|
app = vm.app
|
||||||
|
|
||||||
try:
|
|
||||||
# XXX .is_running() may throw libvirtError if undefined
|
|
||||||
if vm.is_running():
|
|
||||||
self.loop.run_until_complete(vm.kill())
|
|
||||||
except: # pylint: disable=bare-except
|
|
||||||
pass
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.loop.run_until_complete(vm.remove_from_disk())
|
self.loop.run_until_complete(vm.remove_from_disk())
|
||||||
except: # pylint: disable=bare-except
|
except: # pylint: disable=bare-except
|
||||||
@ -876,18 +869,36 @@ class SystemTestCase(QubesTestCase):
|
|||||||
vms = list(vms)
|
vms = list(vms)
|
||||||
if not vms:
|
if not vms:
|
||||||
return
|
return
|
||||||
|
# first kill all the domains, to avoid side effects of changing netvm
|
||||||
|
for vm in vms:
|
||||||
|
try:
|
||||||
|
# XXX .is_running() may throw libvirtError if undefined
|
||||||
|
if vm.is_running():
|
||||||
|
self.loop.run_until_complete(vm.kill())
|
||||||
|
except: # pylint: disable=bare-except
|
||||||
|
pass
|
||||||
# break dependencies
|
# break dependencies
|
||||||
for vm in vms:
|
for vm in vms:
|
||||||
vm.default_dispvm = None
|
vm.default_dispvm = None
|
||||||
# then remove in reverse topological order (wrt netvm), using naive
|
vm.netvm = None
|
||||||
|
# take app instance from any VM to be removed
|
||||||
|
app = vms[0].app
|
||||||
|
if app.default_dispvm in vms:
|
||||||
|
app.default_dispvm = None
|
||||||
|
if app.default_netvm in vms:
|
||||||
|
app.default_netvm = None
|
||||||
|
del app
|
||||||
|
# then remove in reverse topological order (wrt template), using naive
|
||||||
# algorithm
|
# algorithm
|
||||||
# this heavily depends on lack of netvm loops
|
# this heavily depends on lack of template loops, but those are
|
||||||
|
# impossible
|
||||||
while vms:
|
while vms:
|
||||||
vm = vms.pop(0)
|
vm = vms.pop(0)
|
||||||
# make sure that all connected VMs are going to be removed,
|
# make sure that all connected VMs are going to be removed,
|
||||||
# otherwise this will loop forever
|
# otherwise this will loop forever
|
||||||
assert all(x in vms for x in vm.connected_vms)
|
child_vms = list(getattr(vm, 'appvms', []))
|
||||||
if list(vm.connected_vms):
|
assert all(x in vms for x in child_vms)
|
||||||
|
if child_vms:
|
||||||
# if still something use this VM, put it at the end of queue
|
# if still something use this VM, put it at the end of queue
|
||||||
# and try next one
|
# and try next one
|
||||||
vms.append(vm)
|
vms.append(vm)
|
||||||
|
Loading…
Reference in New Issue
Block a user