-
Notifications
You must be signed in to change notification settings - Fork 168
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
virttools: add new test type and virt-install test
1. /virttools: Add new test type for tools from virt-manager repository; by using avocado-vt we get access to many useful test functions ready to use for qemu/libvirt testing. The test type assumes the same default setup as for tp-libvirt/libvirt and builds on top of that. During testing, switching from a bootstrapped libvirt environment to a bootstrapped virttools environment worked and the test case passed. 2. hostdev_mdev: Add a new test case. a. 'run': This virt-install test case is currently ccw device specific but other types on other platforms can be implemented by creating a subclass for the MdevHandler for the desired mdev_type. b. 'clean_up/sleep': The introduction of 1sec sleep is necessary to make sure setting the device offline will work. Using the usual 'wait_for' here seemed to overcomplicate the code. c. 'virt_install_with_hostdev': The function hardcodes several vm installation parameters. This is to keep the code simple at this point but can be changed in the future. 3. spell.ignore: Add command and device names used in ccw.py and hostdev_mdev.py. 4. ccw.py: Add a new function to set devices offline to reset them to their previous state. Test cases with real ccw devices usually assume the test environment is configured in a way that the first unused ccw device will be used for the test case, possibly in a destructive way. Some functions have been added return value 'True' on success to allow for usage in 'wait_for' constructions. 5. README.md: Add readme file to give a bit more context and intention for the new test type. Signed-off-by: Sebastian Mitterle <[email protected]>
- Loading branch information
Showing
5 changed files
with
225 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
The test type 'virttools' is meant to cover test cases for the tools in the | ||
virt-manager repository, e.g. virt-xml, virt-clone, virt-install. | ||
|
||
By using avocado-vt we obtain access to many existing test functions for libvirt | ||
and qemu. Please, note the related commit in avocado-vt to allow for the new | ||
test type. | ||
|
||
'virttools' uses the same basic setup as the tp-libvirt/libvirt test type assuming | ||
most tests will suppose there's at least one existing vm 'avocado-vt-vm1' with | ||
image in the default location (avocado-vt/.../images/jeos-27-s390x.qcow2), e.g. | ||
for 'virt-xml avocado-vt-vm1 --add-device...', 'virt-clone avocado-vt-vm1'. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
- virt_install.hostdev.mdev: | ||
type = hostdev_mdev | ||
variants: | ||
- check_present_inside_guest: | ||
only s390-virtio | ||
mdev_type = vfio_ccw-io |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,180 @@ | ||
import logging | ||
|
||
from time import sleep | ||
from uuid import uuid4 | ||
from avocado.core.exceptions import TestError | ||
from avocado.core.exceptions import TestFail | ||
from provider.vfio import ccw | ||
from virttest.libvirt_xml.vm_xml import VMXML | ||
from virttest.utils_misc import cmd_status_output | ||
from virttest import virsh | ||
|
||
|
||
class MdevHandler(object): | ||
""" Base class for mdev type specific implementations """ | ||
|
||
def create_nodedev(self): | ||
""" Creates the mdev and returns its name """ | ||
raise NotImplementedError() | ||
|
||
def get_target_address(self): | ||
""" Returns a target address to use for hostdev """ | ||
raise NotImplementedError() | ||
|
||
def check_device_present_inside_guest(self, session): | ||
""" | ||
Checks if the host device is present inside the guest | ||
:param session: guest session | ||
""" | ||
raise NotImplementedError() | ||
|
||
def clean_up(self): | ||
""" Stops the mediated device and returns resources to the host """ | ||
raise NotImplementedError() | ||
|
||
@staticmethod | ||
def from_type(mdev_type): | ||
""" | ||
Creates implementing instance for mdev_type | ||
:param mdev_type: The mediated device type as by nodedev API | ||
""" | ||
if mdev_type == "vfio_ccw-io": | ||
return CcwMdevHandler() | ||
else: | ||
raise TestError("Test doesn't know how to handle %s." % mdev_type) | ||
|
||
|
||
class CcwMdevHandler(MdevHandler): | ||
""" Class implementing test methods for vfio_ccw-io """ | ||
|
||
def __init__(self): | ||
self.uuid = None | ||
self.chpids = None | ||
self.schid = None | ||
self.target_address = None | ||
self.expected_device_address = None | ||
self.device_id = None | ||
|
||
def create_nodedev(self): | ||
""" | ||
Creates a mediated device of a specific type | ||
and returns its name from libvirt. | ||
""" | ||
self.schid, self.chpids = ccw.get_device_info() | ||
self.device_id, _ = ccw.get_first_device_identifiers(self.chpids, None) | ||
ccw.set_override(self.schid) | ||
self.uuid = str(uuid4()) | ||
ccw.start_device(self.uuid, self.schid) | ||
|
||
return first_mdev_nodedev_name() | ||
|
||
def get_target_address(self): | ||
""" | ||
Returns a valid target device address | ||
:param address_type: guest device address type | ||
""" | ||
self.target_address = "address.type=ccw,address.cssid=0xfe,address.ssid=0x0,address.devno=0x1111" | ||
self.expected_device_address = "0.0.1111" | ||
return self.target_address | ||
|
||
def check_device_present_inside_guest(self, session): | ||
""" | ||
Fails the test if the device can't be found inside the guest. | ||
:param session: guest session | ||
:raises: TestFail if device not found | ||
""" | ||
device, _ = ccw.get_first_device_identifiers(self.chpids, session) | ||
if not device == self.expected_device_address: | ||
raise TestFail("Couldn't find device inside guest." | ||
"Expected address %s, found %s." % | ||
(self.expected_device_address, device)) | ||
logging.debug("Device was found inside guest with" | ||
" expected id %s." % device) | ||
|
||
def clean_up(self): | ||
""" | ||
Returns the mdev resources to the host. | ||
""" | ||
if self.uuid: | ||
ccw.stop_device(self.uuid) | ||
if self.schid: | ||
ccw.unset_override(self.schid) | ||
# need to sleep to avoid issue with setting device offline | ||
# adding a wait_for would likely be more complicated | ||
sleep(1) | ||
if self.device_id: | ||
ccw.set_device_offline(self.device_id) | ||
|
||
|
||
def disk_for_import(vmxml): | ||
""" | ||
Returns the absolute path to a disk image for import. | ||
Assume the boot image is the first disk and an image file. | ||
:param vmxml: VMXML instance | ||
""" | ||
disks = vmxml.get_disk_all() | ||
disk_list = list(disks.values()) | ||
first_disk = disk_list[0] | ||
return first_disk.find('source').get('file') | ||
|
||
|
||
def first_mdev_nodedev_name(): | ||
""" Returns the first nodedev of type mdev known to libvirt """ | ||
result = virsh.nodedev_list(cap="mdev", debug=True) | ||
device_names = result.stdout.strip().splitlines() | ||
if result.exit_status or len(device_names) == 0: | ||
raise TestError("Couldn't create nodedev. %s. %s." % | ||
(result.stderr, result.stdout)) | ||
return device_names[0] | ||
|
||
|
||
def virt_install_with_hostdev(vm_name, mdev_nodedev, target_address, disk_path): | ||
""" Runs virt-install with hostdev""" | ||
cmd = ("virt-install --import --name %s" | ||
" --hostdev %s,%s" | ||
" --disk %s" | ||
" --vcpus 2 --memory 2048" | ||
" --nographics --noautoconsole" % | ||
(vm_name, mdev_nodedev, target_address, disk_path)) | ||
err, out = cmd_status_output(cmd, shell=True, verbose=True) | ||
if err: | ||
raise TestError("Couldn't install vm with hostdev: %s" % out) | ||
|
||
|
||
def run(test, params, env): | ||
""" | ||
Confirm that a mediated device can be used by virt-install. | ||
For this we import a disk we know will boot and check the | ||
result inside the guest. | ||
The mediated device is created by the test and assumed | ||
to be the only mediated device in the test environment. | ||
""" | ||
|
||
vm_name = params.get("main_vm") | ||
vm = env.get_vm(vm_name) | ||
vmxml = VMXML.new_from_inactive_dumpxml(vm_name) | ||
mdev_type = params.get("mdev_type", "vfio_ccw-io") | ||
handler = None | ||
|
||
try: | ||
|
||
vm.undefine() | ||
handler = MdevHandler.from_type(mdev_type) | ||
disk = disk_for_import(vmxml) | ||
mdev_nodedev = handler.create_nodedev() | ||
target_address = handler.get_target_address() | ||
|
||
virt_install_with_hostdev(vm_name, mdev_nodedev, target_address, disk) | ||
|
||
session = vm.wait_for_login() | ||
handler.check_device_present_inside_guest(session) | ||
|
||
finally: | ||
vmxml.sync() | ||
if handler: | ||
handler.clean_up() |