2014-02-06 19:12:12 +08:00
# Copyright (C) 2013, 2014 Red Hat, Inc.
2013-03-18 05:06:52 +08:00
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
2013-10-28 04:59:47 +08:00
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
2013-03-18 05:06:52 +08:00
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
# MA 02110-1301 USA.
import atexit
import logging
import os
import shlex
2013-09-29 05:23:23 +08:00
import shutil
2013-03-18 05:06:52 +08:00
import subprocess
import sys
import time
import traceback
import unittest
import StringIO
2013-08-18 20:19:58 +08:00
from virtinst import support
2013-03-18 05:06:52 +08:00
2014-09-07 06:05:43 +08:00
from tests import virtinstall , virtclone , virtconvert , virtxml
2013-04-11 07:48:07 +08:00
from tests import utils
2013-03-18 05:06:52 +08:00
os . environ [ " LANG " ] = " en_US.UTF-8 "
2013-09-29 02:42:37 +08:00
# Used to ensure consistent SDL xml output
os . environ [ " HOME " ] = " /tmp "
os . environ [ " DISPLAY " ] = " :3.4 "
2013-08-18 01:50:36 +08:00
_defaultconn = utils . open_testdefault ( )
2013-03-18 05:06:52 +08:00
# Location
image_prefix = " /tmp/__virtinst_cli_ "
xmldir = " tests/cli-test-xml "
treedir = " %s /faketree " % xmldir
vcdir = " %s /virtconv " % xmldir
compare_xmldir = " %s /compare " % xmldir
virtconv_out = " /tmp/__virtinst_tests__virtconv-outdir "
# Images that will be created by virt-install/virt-clone, and removed before
# each run
new_images = [
image_prefix + " new1.img " ,
image_prefix + " new2.img " ,
image_prefix + " new3.img " ,
image_prefix + " exist1-clone.img " ,
image_prefix + " exist2-clone.img " ,
]
# Images that are expected to exist before a command is run
exist_images = [
image_prefix + " exist1.img " ,
image_prefix + " exist2.img " ,
]
2014-03-07 11:09:08 +08:00
# Fake iso for --location iso mounting
fake_iso = [ " /tmp/fake.iso " ]
2014-09-07 06:05:43 +08:00
exist_files = exist_images + fake_iso
new_files = new_images
2014-12-09 20:13:27 +08:00
clean_files = ( new_images + exist_images + fake_iso )
2013-03-18 05:06:52 +08:00
promptlist = [ ]
test_files = {
2013-08-18 20:19:58 +08:00
' TESTURI ' : utils . testuri ,
' DEFAULTURI ' : utils . defaulturi ,
' REMOTEURI ' : utils . uriremote ,
' KVMURI ' : utils . urikvm ,
2015-02-22 06:44:21 +08:00
' KVMURI_NODOMCAPS ' : utils . urikvm_nodomcaps ,
2013-08-18 20:19:58 +08:00
' XENURI ' : utils . urixencaps ,
' XENIA64URI ' : utils . urixenia64 ,
' LXCURI ' : utils . urilxc ,
2013-03-18 05:06:52 +08:00
' CLONE_DISK_XML ' : " %s /clone-disk.xml " % xmldir ,
' CLONE_STORAGE_XML ' : " %s /clone-disk-managed.xml " % xmldir ,
' CLONE_NOEXIST_XML ' : " %s /clone-disk-noexist.xml " % xmldir ,
' IMAGE_XML ' : " %s /image.xml " % xmldir ,
' IMAGE_NOGFX_XML ' : " %s /image-nogfx.xml " % xmldir ,
2014-02-09 05:36:45 +08:00
' NEWIMG1 ' : " /dev/default-pool/new1.img " ,
' NEWIMG2 ' : " /dev/default-pool/new2.img " ,
' NEWCLONEIMG1 ' : new_images [ 0 ] ,
' NEWCLONEIMG2 ' : new_images [ 1 ] ,
' NEWCLONEIMG3 ' : new_images [ 2 ] ,
2014-02-09 08:05:26 +08:00
' AUTOMANAGEIMG ' : " /some/new/pool/dir/new " ,
' EXISTIMG1 ' : " /dev/default-pool/testvol1.img " ,
' EXISTIMG2 ' : " /dev/default-pool/testvol2.img " ,
' EXISTUPPER ' : " /dev/default-pool/UPPER " ,
2013-03-18 05:06:52 +08:00
' POOL ' : " default-pool " ,
' VOL ' : " testvol1.img " ,
2014-12-11 11:45:45 +08:00
' DIR ' : " /var " ,
2013-03-18 05:06:52 +08:00
' TREEDIR ' : treedir ,
2013-07-26 10:06:28 +08:00
' MANAGEDNEW1 ' : " /dev/default-pool/clonevol " ,
' MANAGEDDISKNEW1 ' : " /dev/disk-pool/newvol1.img " ,
' COLLIDE ' : " /dev/default-pool/collidevol1.img " ,
' SHARE ' : " /dev/default-pool/sharevol.img " ,
2013-03-18 05:06:52 +08:00
2014-02-06 08:09:26 +08:00
' OVF_IMG1 ' : " %s /tests/virtconv-files/ovf_input/test1.ovf " % os . getcwd ( ) ,
' VMX_IMG1 ' : " %s /tests/virtconv-files/vmx_input/test1.vmx " % os . getcwd ( ) ,
2013-03-18 05:06:52 +08:00
}
######################
# Test class helpers #
######################
class Command ( object ) :
"""
Instance of a single cli command to test
"""
def __init__ ( self , cmd ) :
self . cmdstr = cmd % test_files
self . check_success = True
self . compare_file = None
2014-01-26 07:16:16 +08:00
self . input_file = None
2013-03-18 05:06:52 +08:00
2014-01-30 21:50:52 +08:00
self . skip_check = None
self . compare_check = None
2013-03-18 05:06:52 +08:00
app , opts = self . cmdstr . split ( " " , 1 )
2014-01-19 06:01:43 +08:00
self . app = app
2013-03-18 05:06:52 +08:00
self . argv = [ os . path . abspath ( app ) ] + shlex . split ( opts )
2014-01-30 21:50:52 +08:00
def _launch_command ( self , conn ) :
2013-03-18 05:06:52 +08:00
logging . debug ( self . cmdstr )
app = self . argv [ 0 ]
oldstdout = sys . stdout
oldstderr = sys . stderr
2014-01-26 07:16:16 +08:00
oldstdin = sys . stdin
2013-03-18 05:06:52 +08:00
oldargv = sys . argv
try :
out = StringIO . StringIO ( )
sys . stdout = out
sys . stderr = out
sys . argv = self . argv
2014-01-26 07:16:16 +08:00
if self . input_file :
sys . stdin = file ( self . input_file )
2013-03-18 05:06:52 +08:00
2014-02-09 05:36:45 +08:00
exc = " "
2013-03-18 05:06:52 +08:00
try :
if app . count ( " virt-install " ) :
ret = virtinstall . main ( conn = conn )
elif app . count ( " virt-clone " ) :
ret = virtclone . main ( conn = conn )
elif app . count ( " virt-convert " ) :
2014-02-06 08:09:26 +08:00
ret = virtconvert . main ( conn = conn )
2014-01-19 23:37:14 +08:00
elif app . count ( " virt-xml " ) :
2014-02-06 08:09:26 +08:00
ret = virtxml . main ( conn = conn )
2013-03-18 05:06:52 +08:00
except SystemExit , sys_e :
ret = sys_e . code
2014-02-09 05:36:45 +08:00
except Exception :
ret = - 1
exc = " \n " + " " . join ( traceback . format_exc ( ) )
2013-03-18 05:06:52 +08:00
if ret != 0 :
ret = - 1
2014-02-09 05:36:45 +08:00
outt = out . getvalue ( ) + exc
2013-03-18 05:06:52 +08:00
if outt . endswith ( " \n " ) :
outt = outt [ : - 1 ]
return ( ret , outt )
finally :
sys . stdout = oldstdout
sys . stderr = oldstderr
2014-01-26 07:16:16 +08:00
sys . stdin = oldstdin
2013-03-18 05:06:52 +08:00
sys . argv = oldargv
2014-01-30 21:50:52 +08:00
def _get_output ( self , conn ) :
2013-03-18 05:06:52 +08:00
try :
for i in new_files :
2013-09-29 05:23:23 +08:00
if os . path . isdir ( i ) :
shutil . rmtree ( i )
elif os . path . exists ( i ) :
os . unlink ( i )
2013-03-18 05:06:52 +08:00
2014-01-30 21:50:52 +08:00
code , output = self . _launch_command ( conn )
2013-03-18 05:06:52 +08:00
logging . debug ( output + " \n " )
return code , output
except Exception , e :
return ( - 1 , " " . join ( traceback . format_exc ( ) ) + str ( e ) )
2013-08-18 20:59:19 +08:00
def _skip_msg ( self , conn ) :
2014-01-30 21:50:52 +08:00
if self . skip_check is None :
2013-08-18 01:50:36 +08:00
return
2013-08-18 20:59:19 +08:00
if conn is None :
2014-01-30 21:50:52 +08:00
raise RuntimeError ( " skip_check is not None, but conn is None " )
if conn . check_support ( self . skip_check ) :
2013-08-18 01:50:36 +08:00
return
return " skipped "
2013-08-18 20:59:19 +08:00
def run ( self , tests ) :
2013-03-18 05:06:52 +08:00
err = None
try :
2014-01-30 21:50:52 +08:00
conn = None
for idx in reversed ( range ( len ( self . argv ) ) ) :
if self . argv [ idx ] == " --connect " :
conn = utils . openconn ( self . argv [ idx + 1 ] )
break
2014-02-06 08:09:26 +08:00
if not conn :
2014-01-30 21:50:52 +08:00
raise RuntimeError ( " couldn ' t parse URI from command %s " %
self . argv )
skipmsg = self . _skip_msg ( conn )
if skipmsg is not None :
tests . skipTest ( skipmsg )
2013-08-18 20:59:19 +08:00
return
2013-03-18 05:06:52 +08:00
2014-01-30 21:50:52 +08:00
code , output = self . _get_output ( conn )
2013-03-18 05:06:52 +08:00
if bool ( code ) == self . check_success :
raise AssertionError (
2013-09-20 08:18:12 +08:00
( " Expected command to %s , but it didn ' t. \n " %
2013-03-18 05:06:52 +08:00
( self . check_success and " pass " or " fail " ) ) +
( " Command was: %s \n " % self . cmdstr ) +
( " Error code : %d \n " % code ) +
( " Output was: \n %s " % output ) )
2014-02-26 04:05:58 +08:00
if self . compare_file :
if ( self . compare_check and not
conn . check_support ( self . compare_check ) ) :
tests . skipTest (
" Skipping compare check due to lack of support " )
return
2013-03-18 05:06:52 +08:00
# Generate test files that don't exist yet
2014-02-26 04:05:58 +08:00
filename = self . compare_file
2014-02-01 02:44:50 +08:00
if utils . REGENERATE_OUTPUT or not os . path . exists ( filename ) :
2013-03-18 05:06:52 +08:00
file ( filename , " w " ) . write ( output )
2014-02-03 02:41:43 +08:00
if " --print-diff " in self . argv and output . count ( " \n " ) > 3 :
2014-12-07 05:32:10 +08:00
# 1) Strip header
# 2) Simplify context lines to reduce churn when
# libvirt or testdriver changes
newlines = [ ]
for line in output . splitlines ( ) [ 3 : ] :
if line . startswith ( " @@ " ) :
line = " @@ "
newlines . append ( line )
output = " \n " . join ( newlines )
2014-02-03 02:41:43 +08:00
2013-03-18 05:06:52 +08:00
utils . diff_compare ( output , filename )
except AssertionError , e :
err = self . cmdstr + " \n " + str ( e )
2013-08-18 20:59:19 +08:00
if err :
tests . fail ( err )
2013-03-18 05:06:52 +08:00
class PromptCheck ( object ) :
"""
Individual question / response pair for automated - - prompt tests
"""
2014-01-19 23:37:14 +08:00
def __init__ ( self , prompt , response = None , num_lines = 1 ) :
2013-03-18 05:06:52 +08:00
self . prompt = prompt
self . response = response
if self . response :
self . response = self . response % test_files
2014-01-19 23:37:14 +08:00
self . num_lines = num_lines
2013-03-18 05:06:52 +08:00
2014-01-19 23:37:14 +08:00
self . _output = None
2013-03-18 05:06:52 +08:00
2014-01-19 23:37:14 +08:00
def check ( self , proc ) :
timeout = 3
def _set_output ( ) :
self . _output = " "
for ignore in range ( self . num_lines ) :
self . _output + = proc . stdout . readline ( )
import threading
thread = threading . Thread ( target = _set_output )
thread . start ( )
thread . join ( timeout )
if thread . isAlive ( ) :
proc . terminate ( )
return False , self . _output + " \n Process hung on readline() "
if not self . _output . count ( self . prompt ) :
self . _output + = ( " \n Content didn ' t contain prompt ' %s ' " %
( self . prompt ) )
return False , self . _output
2013-03-18 05:06:52 +08:00
if self . response :
proc . stdin . write ( self . response + " \n " )
2014-01-19 23:37:14 +08:00
return True , self . _output
2013-03-18 05:06:52 +08:00
class PromptTest ( Command ) :
"""
Fully automated - - prompt test
"""
def __init__ ( self , cmdstr ) :
Command . __init__ ( self , cmdstr )
self . prompt_list = [ ]
def add ( self , * args , * * kwargs ) :
self . prompt_list . append ( PromptCheck ( * args , * * kwargs ) )
2014-01-30 21:50:52 +08:00
def _launch_command ( self , conn ) :
ignore = conn
2013-03-18 05:06:52 +08:00
proc = subprocess . Popen ( self . argv ,
stdin = subprocess . PIPE ,
stdout = subprocess . PIPE ,
stderr = subprocess . STDOUT )
out = " Running %s \n " % self . cmdstr
for p in self . prompt_list :
ret , content = p . check ( proc )
out + = content
if not ret :
# Since we didn't match output, process might be hung
proc . kill ( )
break
exited = False
for ignore in range ( 30 ) :
if proc . poll ( ) is not None :
exited = True
break
time . sleep ( .1 )
if not exited :
proc . kill ( )
out + = " \n Process was killed by test harness "
return proc . wait ( ) , out
2013-08-17 21:44:11 +08:00
class _CategoryProxy ( object ) :
2014-01-30 21:50:52 +08:00
def __init__ ( self , app , name , default_args , skip_check , compare_check ) :
2013-08-17 21:44:11 +08:00
self . _app = app
self . _name = name
2014-01-30 21:50:52 +08:00
self . default_args = default_args
self . skip_check = skip_check
self . compare_check = compare_check
2013-08-17 21:44:11 +08:00
def add_valid ( self , * args , * * kwargs ) :
return self . _app . add_valid ( self . _name , * args , * * kwargs )
def add_invalid ( self , * args , * * kwargs ) :
return self . _app . add_invalid ( self . _name , * args , * * kwargs )
def add_compare ( self , * args , * * kwargs ) :
return self . _app . add_compare ( self . _name , * args , * * kwargs )
2013-04-24 04:16:30 +08:00
class App ( object ) :
def __init__ ( self , appname ) :
self . appname = appname
self . categories = { }
self . cmds = [ ]
2014-05-12 07:19:00 +08:00
def _default_args ( self , cli , iscompare , auto_printarg ) :
2013-04-24 04:16:30 +08:00
args = " "
if not iscompare :
args = " --debug "
2014-02-06 08:09:26 +08:00
if " --connect " not in cli :
args + = " --connect %(TESTURI)s "
2013-04-24 04:16:30 +08:00
if self . appname in [ " virt-install " ] :
if " --name " not in cli :
args + = " --name foobar "
if " --ram " not in cli :
args + = " --ram 64 "
2014-05-12 07:19:00 +08:00
if iscompare and auto_printarg :
2013-04-24 04:16:30 +08:00
if self . appname == " virt-install " :
if ( not cli . count ( " --print-xml " ) and
not cli . count ( " --print-step " ) and
not cli . count ( " --quiet " ) ) :
args + = " --print-step all "
elif self . appname == " virt-clone " :
if not cli . count ( " --print-xml " ) :
args + = " --print-xml "
return args
2014-01-30 21:50:52 +08:00
def add_category ( self , catname , default_args ,
skip_check = None , compare_check = None ) :
obj = _CategoryProxy ( self , catname , default_args ,
skip_check , compare_check )
self . categories [ catname ] = obj
return obj
def _add ( self , catname , testargs , valid , compfile ,
2014-05-12 07:19:00 +08:00
skip_check = None , compare_check = None , input_file = None ,
auto_printarg = True ) :
2013-04-24 04:16:30 +08:00
2014-01-30 21:50:52 +08:00
category = self . categories [ catname ]
args = category . default_args + " " + testargs
2014-05-12 07:19:00 +08:00
args = ( self . _default_args ( args , bool ( compfile ) , auto_printarg ) +
" " + args )
2013-04-24 04:16:30 +08:00
cmdstr = " ./ %s %s " % ( self . appname , args )
cmd = Command ( cmdstr )
cmd . check_success = valid
if compfile :
2014-02-06 08:09:26 +08:00
compfile = os . path . basename ( self . appname ) + " - " + compfile
2013-04-24 04:16:30 +08:00
cmd . compare_file = " %s / %s .xml " % ( compare_xmldir , compfile )
2014-01-30 21:50:52 +08:00
cmd . skip_check = skip_check or category . skip_check
cmd . compare_check = compare_check or category . compare_check
2014-01-26 07:16:16 +08:00
cmd . input_file = input_file
2013-04-24 04:16:30 +08:00
self . cmds . append ( cmd )
2013-08-18 01:50:36 +08:00
def add_valid ( self , cat , args , * * kwargs ) :
self . _add ( cat , args , True , None , * * kwargs )
def add_invalid ( self , cat , args , * * kwargs ) :
self . _add ( cat , args , False , None , * * kwargs )
def add_compare ( self , cat , args , compfile , * * kwargs ) :
self . _add ( cat , args , not compfile . endswith ( " -fail " ) ,
compfile , * * kwargs )
2013-04-24 04:16:30 +08:00
#
# The test matrix
#
# add_valid: A test that should pass
# add_invalid: A test that should fail
# add_compare: Get the generated XML, and compare against the passed filename
# in tests/clitest-xml/compare/
#
2014-12-10 22:33:02 +08:00
######################
# virt-install tests #
######################
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
vinst = App ( " virt-install " )
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
#############################################
# virt-install verbose XML comparison tests #
#############################################
c = vinst . add_category ( " xml-comparsion " , " --connect %(KVMURI)s --noautoconsole --os-variant fedora20 " )
# Singleton element test #1, for simpler strings
c . add_compare ( """ \
- - memory 1024 \
- - vcpus 4 - - cpuset = 1 , 3 - 5 \
- - cpu host \
- - description \" foobar & baz \" \
2015-02-19 04:16:48 +08:00
- - boot uefi \
2014-12-10 22:33:02 +08:00
- - security type = dynamic \
- - numatune 1 , 2 , 3 , 5 - 7 , ^ 6 \
- - memorybacking hugepages = on \
- - features apic = off \
- - clock offset = localtime \
- - resource / virtualmachines / production \
- - events on_crash = restart \
\
- - disk none \
- - console none \
2015-01-09 23:52:59 +08:00
- - channel none \
2014-12-10 22:33:02 +08:00
- - network none \
- - controller usb2 \
- - graphics spice \
- - video vga \
- - sound none \
- - redirdev none \
- - memballoon none \
- - smartcard none \
- - watchdog default \
- - panic default \
- - tpm / dev / tpm0 \
- - rng / dev / random \
""" , " singleton-config-1 " )
# Singleton element test #2, for complex strings
c . add_compare ( """ --pxe \
- - memory 512 , maxmemory = 1024 \
- - vcpus 4 , cores = 2 , threads = 2 , sockets = 2 \
- - cpu foobar , + x2apic , + x2apicagain , - distest , forbid = foo , forbid = bar , disable = distest2 , optional = opttest , require = reqtest , match = strict , vendor = meee \
- - metadata title = my - title , description = my - description , uuid = 00000000 - 1111 - 2222 - 3333 - 444444444444 \
- - boot cdrom , fd , hd , network , menu = off , loader = / foo / bar \
- - idmap uid_start = 0 , uid_target = 1000 , uid_count = 10 , gid_start = 0 , gid_target = 1000 , gid_count = 10 \
- - security type = static , label = ' system_u:object_r:svirt_image_t:s0:c100,c200 ' , relabel = yes \
- - numatune 1 - 3 , 4 , mode = strict \
- - memtune hard_limit = 10 , soft_limit = 20 , swap_hard_limit = 30 , min_guarantee = 40 \
- - blkiotune weight = 100 , device_path = / home / test / 1. img , device_weight = 200 \
- - memorybacking size = 1 , unit = ' G ' , nodeset = ' 1,2-5 ' , nosharepages = yes , locked = yes \
- - features acpi = off , eoi = on , privnet = on , hyperv_spinlocks = on , hyperv_spinlocks_retries = 1234 \
- - clock offset = utc , hpet_present = no , rtc_tickpolicy = merge \
- - pm suspend_to_mem = yes , suspend_to_disk = no \
- - resource partition = / virtualmachines / production \
- - events on_poweroff = destroy , on_reboot = restart , on_crash = preserve \
\
- - controller usb3 \
- - controller virtio - scsi \
- - graphics vnc \
- - filesystem / foo / source , / bar / target \
- - memballoon virtio \
- - watchdog ib700 , action = pause \
- - tpm passthrough , model = tpm - tis , path = / dev / tpm0 \
- - rng egd , backend_host = 127.0 .0 .1 , backend_service = 8000 , backend_type = udp , backend_mode = bind , backend_connect_host = foo , backend_connect_service = 708 \
- - panic iobase = 0x506 \
""" , " singleton-config-2 " )
# Device testing #1
c . add_compare ( """ \
- - vcpus 4 , cores = 1 \
- - cpu none \
\
2014-02-10 02:36:12 +08:00
- - disk % ( EXISTUPPER ) s , cache = writeback , io = threads , perms = sh , serial = WD - WMAP9A966149 , boot_order = 2 \
2014-04-29 22:29:33 +08:00
- - disk % ( NEWIMG1 ) s , sparse = false , size = .001 , perms = ro , error_policy = enospace , discard = unmap \
2014-02-01 04:38:39 +08:00
- - disk device = cdrom , bus = sata , read_bytes_sec = 1 , read_iops_sec = 2 , total_bytes_sec = 10 , total_iops_sec = 20 , write_bytes_sec = 5 , write_iops_sec = 6 \
2014-02-11 07:47:58 +08:00
- - disk size = 1 \
2014-12-10 22:33:02 +08:00
- - disk / iscsi - pool / diskvol1 \
- - disk / dev / default - pool / iso - vol \
- - disk / dev / default - pool / iso - vol , format = qcow2 \
2014-12-11 01:31:56 +08:00
- - disk source_pool = rbd - ceph , source_volume = some - rbd - vol , size = .1 \
2014-12-11 01:55:08 +08:00
- - disk pool = rbd - ceph , size = .1 \
2014-12-10 22:33:02 +08:00
- - disk source_protocol = http , source_host_name = example . com , source_host_port = 8000 , source_name = / path / to / my / file \
2014-12-07 07:33:11 +08:00
- - disk source_protocol = nbd , source_host_transport = unix , source_host_socket = / tmp / socket , bus = scsi \
2015-01-27 21:10:12 +08:00
- - disk gluster : / / 192.168 .1 .100 / test - volume / some / dir / test - gluster . qcow2 \
2014-12-10 22:33:02 +08:00
- - disk qemu + nbd : / / / var / foo / bar / socket , bus = usb , removable = on \
- - disk path = http : / / [ 1 : 2 : 3 : 4 : 1 : 2 : 3 : 4 ] : 5522 / my / path ? query = foo \
- - disk vol = gluster - pool / test - gluster . raw , startup_policy = optional \
- - disk % ( DIR ) s , device = floppy \
\
2014-06-01 02:30:07 +08:00
- - network user , mac = 12 : 34 : 56 : 78 : 11 : 22 , portgroup = foo \
2014-02-01 20:48:04 +08:00
- - network bridge = foobar , model = virtio , driver_name = qemu , driver_queues = 3 \
2014-02-10 02:36:12 +08:00
- - network type = direct , source = eth5 , source_mode = vepa , target = mytap12 , virtualport_type = 802.1 Qbg , virtualport_managerid = 12 , virtualport_typeid = 1193046 , virtualport_typeidversion = 1 , virtualport_instanceid = 09 b11c53 - 8 b5c - 4 eeb - 8 f00 - d84eaa0aaa3b , boot_order = 1 \
2014-12-10 22:33:02 +08:00
\
- - graphics sdl \
- - graphics spice , keymap = none \
- - graphics vnc , port = 5950 , listen = 1.2 .3 .4 , keymap = ja , password = foo \
- - graphics spice , port = 5950 , tlsport = 5950 , listen = 1.2 .3 .4 , keymap = ja \
\
- - controller usb , model = ich9 - ehci1 , address = 0 : 0 : 4.7 , index = 0 \
- - controller usb , model = ich9 - uhci1 , address = 0 : 0 : 4.0 , index = 0 , master = 0 \
- - controller usb , model = ich9 - uhci2 , address = 0 : 0 : 4.1 , index = 0 , master = 2 \
- - controller usb , model = ich9 - uhci3 , address = 0 : 0 : 4.2 , index = 0 , master = 4 \
\
- - serial tcp , host = : 2222 , mode = bind , protocol = telnet \
- - parallel udp , host = 0.0 .0 .0 : 1234 , bind_host = 127.0 .0 .1 : 1234 \
- - parallel unix , path = / tmp / foo - socket \
- - channel pty , target_type = guestfwd , target_address = 127.0 .0 .1 : 10000 \
- - channel pty , target_type = virtio , name = org . linux - kvm . port1 \
- - console pty , target_type = virtio \
2013-09-28 21:36:11 +08:00
- - channel spicevmc \
2014-12-10 22:33:02 +08:00
\
2014-09-21 02:29:34 +08:00
- - hostdev net_00_1c_25_10_b1_e4 , boot_order = 4 , rom_bar = off \
2014-12-10 22:33:02 +08:00
- - host - device usb_device_781_5151_2004453082054CA1BEEE \
- - host - device 001.003 \
- - hostdev 15 : 0.1 \
- - host - device 2 : 15 : 0.2 \
- - hostdev 0 : 15 : 0.3 \
- - host - device 0x0781 : 0x5151 , driver_name = vfio \
- - host - device 04 b3 : 4485 \
\
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
- - filesystem / source , / target , mode = squash \
- - filesystem template_name , / , type = template \
\
- - soundhw default \
- - sound ac97 \
\
- - video cirrus \
- - video model = qxl \
\
- - smartcard passthrough , type = spicevmc \
- - smartcard type = host \
\
- - redirdev usb , type = spicevmc \
- - redirdev usb , type = tcp , server = localhost : 4000 \
- - redirdev usb , type = tcp , server = 127.0 .0 .1 : 4002 , boot_order = 3 \
\
- - rng egd , backend_host = 127.0 .0 .1 , backend_service = 8000 , backend_type = tcp \
\
- - panic iobase = 507 \
""" , " many-devices " )
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
####################################################
# CPU/RAM/numa and other singleton VM config tests #
####################################################
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
c = vinst . add_category ( " cpuram " , " --hvm --nographics --noautoconsole --nodisks --pxe " )
c . add_valid ( " --vcpus 4 --cpuset=1,3-5, " ) # Cpuset with trailing comma
c . add_valid ( " --vcpus 4 --cpuset=auto " ) # cpuset=auto but caps doesn't support it
c . add_valid ( " --ram 4000000 " ) # Ram overcommit
c . add_valid ( " --vcpus sockets=2,threads=2 " ) # Topology only
c . add_valid ( " --cpu somemodel " ) # Simple --cpu
c . add_valid ( " --security label=foobar.label,relabel=yes " ) # --security implicit static
c . add_valid ( " --security label=foobar.label,a1,z2,b3,type=static,relabel=no " ) # static with commas 1
c . add_valid ( " --security label=foobar.label,a1,z2,b3 " ) # --security static with commas 2
c . add_compare ( " --connect %(DEFAULTURI)s --cpuset auto --vcpus 2 " , " cpuset-auto " ) # --cpuset=auto actually works
c . add_invalid ( " --vcpus 32 --cpuset=969-1000 " ) # Bogus cpuset
c . add_invalid ( " --vcpus 32 --cpuset=autofoo " ) # Bogus cpuset
c . add_invalid ( " --clock foo_tickpolicy=merge " ) # Unknown timer
c . add_invalid ( " --security foobar " ) # Busted --security
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
########################
# Storage provisioning #
########################
2013-08-17 21:44:11 +08:00
c = vinst . add_category ( " storage " , " --pxe --nographics --noautoconsole --hvm " )
c . add_valid ( " --disk path= %(EXISTIMG1)s " ) # Existing disk, no extra options
c . add_valid ( " --disk pool= %(POOL)s ,size=.0001 --disk pool= %(POOL)s ,size=.0001 " ) # Create 2 volumes in a pool
c . add_valid ( " --disk vol= %(POOL)s / %(VOL)s " ) # Existing volume
c . add_valid ( " --disk path= %(EXISTIMG1)s --disk path= %(EXISTIMG1)s --disk path= %(EXISTIMG1)s --disk path= %(EXISTIMG1)s ,device=cdrom " ) # 3 IDE and CD
2014-12-10 22:33:02 +08:00
c . add_valid ( " --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi --disk path= %(EXISTIMG1)s ,bus=scsi " ) # > 16 scsi disks
2013-08-17 21:44:11 +08:00
c . add_valid ( " --disk path= %(NEWIMG1)s ,format=raw,size=.0000001 " ) # Unmanaged file using format 'raw'
c . add_valid ( " --disk path= %(MANAGEDNEW1)s ,format=raw,size=.0000001 " ) # Managed file using format raw
c . add_valid ( " --disk path= %(MANAGEDNEW1)s ,format=qcow2,size=.0000001 " ) # Managed file using format qcow2
c . add_valid ( " --disk %(EXISTIMG1)s " ) # Not specifying path=
c . add_valid ( " --disk %(NEWIMG1)s ,format=raw,size=.0000001 " ) # Not specifying path= but creating storage
c . add_valid ( " --disk %(COLLIDE)s --force " ) # Colliding storage with --force
c . add_valid ( " --disk %(SHARE)s ,perms=sh " ) # Colliding shareable storage
c . add_valid ( " --disk path= %(EXISTIMG1)s ,device=cdrom --disk path= %(EXISTIMG1)s ,device=cdrom " ) # Two IDE cds
c . add_valid ( " --disk %(EXISTIMG1)s ,driver_name=qemu,driver_type=qcow2 " ) # Driver name and type options
2014-12-05 08:36:26 +08:00
c . add_valid ( " --disk /dev/zero " ) # Referencing a local unmanaged /dev node
2013-08-17 21:44:11 +08:00
c . add_valid ( " --disk pool=default,size=.00001 " ) # Building 'default' pool
2014-02-09 05:36:45 +08:00
c . add_valid ( " --disk %(AUTOMANAGEIMG)s ,size=.1 " ) # autocreate the pool
2014-02-05 05:16:39 +08:00
c . add_invalid ( " --disk %(NEWIMG1)s ,sparse=true,size=100000000000 --force " ) # Don't warn about fully allocated file exceeding disk space
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --file %(NEWIMG1)s --file-size 100000 --nonsparse " ) # Nonexisting file, size too big
c . add_invalid ( " --file %(NEWIMG1)s --file-size 100000 " ) # Huge file, sparse, but no prompting
c . add_invalid ( " --file %(NEWIMG1)s " ) # Nonexisting file, no size
c . add_invalid ( " --file %(EXISTIMG1)s --file %(EXISTIMG1)s --file %(EXISTIMG1)s --file %(EXISTIMG1)s --file %(EXISTIMG1)s " ) # Too many IDE
c . add_invalid ( " --disk pool=foopool,size=.0001 " ) # Specify a nonexistent pool
c . add_invalid ( " --disk vol= %(POOL)s /foovol " ) # Specify a nonexistent volume
c . add_invalid ( " --disk pool= %(POOL)s " ) # Specify a pool with no size
c . add_invalid ( " --disk path= %(EXISTIMG1)s ,perms=ro,size=.0001,cache=FOOBAR " ) # Unknown cache type
2014-02-09 05:36:45 +08:00
c . add_invalid ( " --disk path=/dev/foo/bar/baz,format=qcow2,size=.0000001 " ) # Unmanaged file using non-raw format
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --disk path= %(MANAGEDDISKNEW1)s ,format=raw,size=.0000001 " ) # Managed disk using any format
c . add_invalid ( " --disk %(NEWIMG1)s " ) # Not specifying path= and non existent storage w/ no size
2014-02-05 05:16:39 +08:00
c . add_invalid ( " --disk %(NEWIMG1)s ,sparse=true,size=100000000000 " ) # Fail if fully allocated file would exceed disk space
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --disk %(COLLIDE)s " ) # Colliding storage without --force
2014-02-05 05:16:39 +08:00
c . add_invalid ( " --disk %(COLLIDE)s --prompt " ) # Colliding storage with --prompt should still fail
2014-01-15 06:44:14 +08:00
c . add_invalid ( " --disk /dev/default-pool/backingl3.img " ) # Colliding storage via backing store
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --disk %(DIR)s ,device=cdrom " ) # Dir without floppy
c . add_invalid ( " --disk %(EXISTIMG1)s ,driver_name=foobar,driver_type=foobaz " ) # Unknown driver name and type options (as of 1.0.0)
2014-12-07 05:22:00 +08:00
c . add_invalid ( " --disk source_pool=rbd-ceph,source_volume=vol1 " ) # Collision with existing VM, via source pool/volume
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
################################################
# Invalid devices that hit virtinst code paths #
################################################
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
c = vinst . add_category ( " invalid-devices " , " --noautoconsole --nodisks --pxe " )
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --host-device 1d6b:2 " ) # multiple USB devices with identical vendorId and productId
c . add_invalid ( " --host-device pci_8086_2850_scsi_host_scsi_host " ) # Unsupported hostdev type
c . add_invalid ( " --host-device foobarhostdev " ) # Unknown hostdev
c . add_invalid ( " --host-device 300:400 " ) # Parseable hostdev, but unknown digits
2014-12-10 22:33:02 +08:00
c . add_invalid ( " --graphics vnc,keymap=ZZZ " ) # Invalid keymap
c . add_invalid ( " --graphics vnc,port=-50 " ) # Invalid port
c . add_invalid ( " --graphics spice,tlsport=5 " ) # Invalid port
c . add_invalid ( " --serial unix " ) # Unix with no path
c . add_invalid ( " --serial null,path=/tmp/foo " ) # Path where it doesn't belong
c . add_invalid ( " --channel pty,target_type=guestfwd " ) # --channel guestfwd without target_address
2015-02-19 04:16:48 +08:00
c . add_invalid ( " --boot uefi " ) # URI doesn't support UEFI bits
c . add_invalid ( " --connect %(KVMURI)s --boot uefi,arch=ppc64 " ) # unsupported arch for UEFI
2015-02-22 06:44:21 +08:00
c . add_valid ( " --connect %(KVMURI_NODOMCAPS)s --arch aarch64 --nodisks " ) # attempt to default to aarch64 UEFI, but it fails, but should only print warnings
2013-08-17 21:44:11 +08:00
2014-12-10 22:33:02 +08:00
########################
# Install option tests #
########################
c = vinst . add_category ( " nodisk-install " , " --nographics --noautoconsole --nodisks " )
2013-08-17 21:44:11 +08:00
c . add_valid ( " --hvm --cdrom %(EXISTIMG1)s " ) # Simple cdrom install
2014-12-10 22:33:02 +08:00
c . add_valid ( " --wait 0 --os-variant winxp --cdrom %(EXISTIMG1)s " ) # Windows (2 stage) install
c . add_valid ( " --pxe --virt-type test " ) # Explicit virt-type
2013-08-17 21:44:11 +08:00
c . add_valid ( " --arch i686 --pxe " ) # Explicity fullvirt + arch
c . add_valid ( " --arch i486 --pxe " ) # Convert i*86 -> i686
2014-12-10 22:33:02 +08:00
c . add_valid ( " --location %(TREEDIR)s " ) # Directory tree URL install
c . add_valid ( " --location %(TREEDIR)s --initrd-inject virt-install --extra-args ks=file:/virt-install " ) # initrd-inject
2013-08-17 21:44:11 +08:00
c . add_valid ( " --hvm --location %(TREEDIR)s --extra-args console=ttyS0 " ) # Directory tree URL install with extra-args
c . add_valid ( " --hvm --cdrom %(TREEDIR)s " ) # Directory tree CDROM install
c . add_valid ( " --paravirt --location %(TREEDIR)s " ) # Paravirt location
c . add_valid ( " --paravirt --location %(TREEDIR)s --os-variant none " ) # Paravirt location with --os-variant none
2014-12-10 22:33:02 +08:00
c . add_valid ( " --location %(TREEDIR)s --os-variant fedora12 " ) # URL install with manual os-variant
c . add_valid ( " --cdrom %(EXISTIMG2)s --os-variant win2k3 --wait 0 " ) # HVM windows install with disk
c . add_valid ( " --cdrom %(EXISTIMG2)s --os-variant win2k3 --wait 0 --print-step 3 " ) # HVM windows install, print 3rd stage XML
c . add_valid ( " --pxe --autostart " ) # --autostart flag
c . add_compare ( " --pxe --print-step all " , " simple-pxe " ) # Diskless PXE install
c . add_invalid ( " --pxe --virt-type bogus " ) # Bogus virt-type
c . add_invalid ( " --pxe --arch bogus " ) # Bogus arch
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --paravirt --pxe " ) # PXE w/ paravirt
c . add_invalid ( " --import " ) # Import with no disks
c . add_invalid ( " --livecd " ) # LiveCD with no media
2014-12-10 22:33:02 +08:00
c . add_invalid ( " --pxe --print-xml " ) # 2 stage install with --print-xml
c . add_invalid ( " --pxe --os-variant farrrrrrrge " ) # Bogus --os-variant
c . add_invalid ( " --pxe --boot menu=foobar " )
c . add_invalid ( " --cdrom %(EXISTIMG1)s --extra-args console=ttyS0 " ) # cdrom fail w/ extra-args
2014-09-24 02:14:21 +08:00
c . add_invalid ( " --hvm --boot kernel= %(TREEDIR)s /pxeboot/vmlinuz,initrd= %(TREEDIR)s /pxeboot/initrd.img,kernel_args= ' foo bar ' --initrd-inject virt-install " ) # initrd-inject with manual kernel/initrd
2013-04-24 04:16:30 +08:00
2014-12-10 22:33:02 +08:00
c = vinst . add_category ( " single-disk-install " , " --nographics --noautoconsole --disk %(EXISTIMG1)s " )
c . add_valid ( " --hvm --import " ) # FV Import install
c . add_valid ( " --hvm --import --prompt --force " ) # Working scenario w/ prompt shouldn't ask anything
c . add_valid ( " --paravirt --import " ) # PV Import install
c . add_valid ( " --paravirt --print-xml " ) # print single XML, implied import install
c . add_compare ( " --cdrom %(EXISTIMG2)s --os-variant win2k3 --wait 0 --vcpus cores=4 --controller usb,model=none " , " w2k3-cdrom " ) # HVM windows install with disk
c . add_invalid ( " --paravirt --import --print-step 2 " ) # PV Import install, no second XML step
c = vinst . add_category ( " misc-install " , " --nographics --noautoconsole " )
c . add_valid ( " --disk path= %(EXISTIMG1)s ,device=cdrom " ) # Implied cdrom install
c . add_compare ( " " , " noargs-fail " , auto_printarg = False ) # No arguments
c . add_valid ( " --panic help --disk=? " ) # Make sure introspection doesn't blow up
c . add_invalid ( " --hvm --nodisks --pxe foobar " ) # Positional arguments error
c . add_invalid ( " --nodisks --pxe --name test " ) # Colliding name
#############################
# Remote URI specific tests #
#############################
c = vinst . add_category ( " remote " , " --connect %(REMOTEURI)s --nographics --noautoconsole " )
c . add_valid ( " --nodisks --pxe " ) # Simple pxe nodisks
c . add_valid ( " --pxe --disk /foo/bar/baz,size=.01 " ) # Creating any random path on the remote host
c . add_valid ( " --pxe --disk /dev/zde " ) # /dev file that we just pass through to the remote VM
c . add_invalid ( " --pxe --disk /foo/bar/baz " ) # File that doesn't exist after auto storage setup
c . add_invalid ( " --nodisks --location /tmp " ) # Use of --location
c . add_invalid ( " --file /foo/bar/baz --pxe " ) # Trying to use unmanaged storage without size argument
###########################
# QEMU/KVM specific tests #
###########################
c = vinst . add_category ( " kvm " , " --connect %(KVMURI)s --noautoconsole " )
c . add_compare ( " --os-variant fedora20 --file %(EXISTIMG1)s --location %(TREEDIR)s --extra-args console=ttyS0 --cpu host --channel none --console none --sound none --redirdev none " , " kvm-f14-url " ) # Fedora Directory tree URL install with extra-args
c . add_compare ( " --test-media-detection %(TREEDIR)s " , " test-url-detection " ) # --test-media-detection
c . add_compare ( " --os-variant fedora20 --disk %(NEWIMG1)s ,size=.01,format=vmdk --location %(TREEDIR)s --extra-args console=ttyS0 --quiet " , " quiet-url " ) # Quiet URL install should make no noise
c . add_compare ( " --cdrom %(EXISTIMG2)s --file %(EXISTIMG1)s --os-variant win2k3 --wait 0 --sound --controller usb " , " kvm-win2k3-cdrom " ) # HVM windows install with disk
c . add_compare ( " --os-variant ubuntusaucy --nodisks --boot cdrom --virt-type qemu --cpu Penryn " , " qemu-plain " ) # plain qemu
c . add_compare ( " --os-variant fedora20 --nodisks --boot network --nographics --arch i686 " , " qemu-32-on-64 " ) # 32 on 64
c . add_compare ( " --arch armv7l --machine vexpress-a9 --boot kernel=/f19-arm.kernel,initrd=/f19-arm.initrd,dtb=/f19-arm.dtb,extra_args= \" console=ttyAMA0 rw root=/dev/mmcblk0p3 \" --disk %(EXISTIMG1)s --nographics " , " arm-vexpress-plain " , skip_check = support . SUPPORT_CONN_DISK_SD )
c . add_compare ( " --arch armv7l --machine vexpress-a15 --boot kernel=/f19-arm.kernel,initrd=/f19-arm.initrd,dtb=/f19-arm.dtb,kernel_args= \" console=ttyAMA0,1234 rw root=/dev/vda3 \" ,extra_args=foo --disk %(EXISTIMG1)s --nographics --os-variant fedora19 " , " arm-vexpress-f19 " , skip_check = support . SUPPORT_CONN_VIRTIO_MMIO )
c . add_compare ( " --arch armv7l --machine virt --boot kernel=/f19-arm.kernel,initrd=/f19-arm.initrd,kernel_args= \" console=ttyAMA0,1234 rw root=/dev/vda3 \" ,extra_args=foo --disk %(EXISTIMG1)s --nographics --os-variant fedora20 " , " arm-virt-f20 " )
c . add_compare ( " --arch armv7l --boot kernel=/f19-arm.kernel,initrd=/f19-arm.initrd,kernel_args= \" console=ttyAMA0,1234 rw root=/dev/vda3 \" ,extra_args=foo --disk %(EXISTIMG1)s --os-variant fedora20 " , " arm-defaultmach-f20 " )
c . add_compare ( " --arch aarch64 --machine virt --boot kernel=/f19-arm.kernel,initrd=/f19-arm.initrd,kernel_args= \" console=ttyAMA0,1234 rw root=/dev/vda3 \" ,extra_args=foo --disk %(EXISTIMG1)s " , " aarch64-machvirt " )
c . add_compare ( " --arch aarch64 --boot kernel=/f19-arm.kernel,initrd=/f19-arm.initrd,kernel_args= \" console=ttyAMA0,1234 rw root=/dev/vda3 \" ,extra_args=foo --disk %(EXISTIMG1)s " , " aarch64-machdefault " )
2015-03-24 05:46:09 +08:00
c . add_compare ( " --arch aarch64 --cdrom %(EXISTIMG2)s --boot loader=CODE.fd,nvram_template=VARS.fd --disk %(EXISTIMG1)s --cpu none " , " aarch64-cdrom " )
2014-12-10 22:33:02 +08:00
c . add_compare ( " --arch ppc64 --machine pseries --boot network --disk %(EXISTIMG1)s --os-variant fedora20 --network none " , " ppc64-pseries-f20 " )
c . add_compare ( " --arch ppc64 --boot network --disk %(EXISTIMG1)s --os-variant fedora20 --network none " , " ppc64-machdefault-f20 " )
2015-02-22 06:44:21 +08:00
c . add_compare ( " --arch aarch64 --nodisks " , " aarch64-default-uefi " ) # ensure aarch64 defaults to UEFI
2014-12-10 22:33:02 +08:00
c . add_compare ( " --disk none --location /tmp/fake.iso --nonetworks " , " location-iso " ) # Using --location iso mounting
c . add_compare ( " --disk %(EXISTIMG1)s --pxe --os-variant rhel6.4 " , " kvm-rhel6 " ) # RHEL6 defaults
c . add_compare ( " --disk %(EXISTIMG1)s --pxe --os-variant rhel7.0 " , " kvm-rhel7 " ) # RHEL7 defaults
c . add_compare ( " --disk %(EXISTIMG1)s --pxe --os-variant centos7.0 " , " kvm-centos7 " ) # Centos 7 defaults
c . add_invalid ( " --disk none --boot network --machine foobar " ) # Unknown machine type
c . add_invalid ( " --nodisks --boot network --arch mips --virt-type kvm " ) # Invalid domain type for arch
c . add_invalid ( " --nodisks --boot network --paravirt --arch mips " ) # Invalid arch/virt combo
2015-01-28 03:41:43 +08:00
c . add_compare ( " --os-variant win7 --cdrom %(EXISTIMG2)s --boot loader_type=pflash,loader=CODE.fd,nvram_template=VARS.fd --disk %(EXISTIMG1)s " , " win7-uefi " ) # no HYPER-V
2014-12-10 22:33:02 +08:00
######################
# LXC specific tests #
######################
c = vinst . add_category ( " lxc " , " --connect %(LXCURI)s --noautoconsole --name foolxc --memory 64 " )
c . add_compare ( " " , " default " )
c . add_compare ( " --filesystem /source,/ " , " fs-default " )
c . add_compare ( " --init /usr/bin/httpd " , " manual-init " )
######################
# Xen specific tests #
######################
c = vinst . add_category ( " xen " , " --connect %(XENURI)s --noautoconsole " )
c . add_compare ( " --disk %(EXISTIMG1)s --import " , " xen-default " ) # Xen default
c . add_compare ( " --disk %(EXISTIMG1)s --location %(TREEDIR)s --paravirt " , " xen-pv " ) # Xen PV
c . add_compare ( " --disk %(EXISTIMG1)s --cdrom %(EXISTIMG1)s --livecd --hvm " , " xen-hvm " ) # Xen HVM
2013-04-24 04:16:30 +08:00
2014-12-10 22:33:02 +08:00
#####################################
# Device option back compat testing #
#####################################
c = vinst . add_category ( " device-back-compat " , " --nodisks --pxe --noautoconsole " )
c . add_valid ( " --sdl " ) # SDL
c . add_valid ( " --vnc --keymap ja --vncport 5950 --vnclisten 1.2.3.4 " ) # VNC w/ lots of options
c . add_valid ( " --sound " ) # --sound with no option back compat
c . add_valid ( " --mac 22:22:33:44:55:AF " ) # Just a macaddr
c . add_valid ( " --bridge mybr0 --mac 22:22:33:44:55:AF " ) # Old bridge w/ mac
c . add_valid ( " --network bridge:mybr0,model=e1000 " ) # --network bridge:
c . add_valid ( " --network network:default --mac RANDOM " ) # VirtualNetwork with a random macaddr
c . add_valid ( " --nonetworks " ) # no networks
c . add_valid ( " --vnc --keymap=local " ) # --keymap local
c . add_invalid ( " --graphics vnc --vnclisten 1.2.3.4 " ) # mixing old and new
c . add_invalid ( " --network=FOO " ) # Nonexistent network
c . add_invalid ( " --mac 1234 " ) # Invalid mac
c . add_invalid ( " --network user --bridge foo0 " ) # Mixing bridge and network
c . add_invalid ( " --mac 22:22:33:12:34:AB " ) # Colliding macaddr
c = vinst . add_category ( " storage-back-compat " , " --pxe --noautoconsole " )
c . add_valid ( " --file %(EXISTIMG1)s --nonsparse --file-size 4 " ) # Existing file, other opts
c . add_valid ( " --file %(EXISTIMG1)s " ) # Existing file, no opts
c . add_valid ( " --file %(EXISTIMG1)s --file virt-clone --file virt-clone " ) # Multiple existing files
c . add_valid ( " --file %(NEWIMG1)s --file-size .00001 --nonsparse " ) # Nonexistent file
##################
# virt-xml tests #
##################
2013-04-24 04:16:30 +08:00
2014-01-19 23:37:14 +08:00
vixml = App ( " virt-xml " )
c = vixml . add_category ( " misc " , " " )
c . add_valid ( " --help " ) # basic --help test
2014-02-06 01:32:53 +08:00
c . add_valid ( " --sound=? --tpm=? " ) # basic introspection test
2014-01-26 07:16:16 +08:00
c . add_invalid ( " test --edit --hostdev driver_name=vfio " ) # Guest has no hostdev to edit
c . add_invalid ( " test --edit --cpu host-passthrough --boot hd,network " ) # Specified more than 1 option
c . add_invalid ( " test --edit " ) # specified no edit option
c . add_invalid ( " test --edit 2 --cpu host-passthrough " ) # specifing --edit number where it doesn't make sense
c . add_invalid ( " test-many-devices --edit 5 --tpm /dev/tpm " ) # device edit out of range
2014-01-26 08:51:56 +08:00
c . add_invalid ( " test-many-devices --add-device --host-device 0x0781:0x5151 --update " ) # test driver doesn't support attachdevice...
c . add_invalid ( " test-many-devices --remove-device --host-device 1 --update " ) # test driver doesn't support detachdevice...
c . add_invalid ( " test-many-devices --edit --graphics password=foo --update " ) # test driver doesn't support updatdevice...
2014-01-26 09:14:42 +08:00
c . add_invalid ( " --build-xml --memory 10,maxmemory=20 " ) # building XML for option that doesn't support it
2014-02-06 08:09:26 +08:00
c . add_compare ( " test --print-xml --edit --vcpus 7 " , " print-xml " ) # test --print-xml
c . add_compare ( " --edit --cpu host-passthrough " , " stdin-edit " , input_file = ( xmldir + " /virtxml-stdin-edit.xml " ) ) # stdin test
c . add_compare ( " --build-xml --cpu pentium3,+x2apic " , " build-cpu " )
c . add_compare ( " --build-xml --tpm /dev/tpm " , " build-tpm " )
c . add_compare ( " --build-xml --blkiotune weight=100,device_path=/dev/sdf,device_weight=200 " , " build-blkiotune " )
2014-02-09 23:25:25 +08:00
c . add_compare ( " --build-xml --idmap uid_start=0,uid_target=1000,uid_count=10,gid_start=0,gid_target=1000,gid_count=10 " , " build-idmap " )
2014-02-10 02:36:12 +08:00
c . add_compare ( " test --edit --boot network,cdrom " , " edit-bootorder " )
2014-01-26 07:16:16 +08:00
2014-02-26 04:05:58 +08:00
c = vixml . add_category ( " simple edit diff " , " test-many-devices --edit --print-diff --define " , compare_check = support . SUPPORT_CONN_INPUT_KEYBOARD )
2014-01-25 09:03:30 +08:00
c . add_compare ( """ --metadata name=foo-my-new-name,uuid=12345678-12F4-1234-1234-123456789AFA,description= " hey this is my
new
2014-02-06 08:09:26 +08:00
very , very = new desc \\\' " ,title= " This is my,funky=new title " " " " , " edit-simple-metadata " )
2014-05-29 09:46:24 +08:00
c . add_compare ( " --events on_poweroff=destroy,on_reboot=restart,on_crash=preserve " , " edit-simple-events " )
2014-02-06 08:09:26 +08:00
c . add_compare ( " --memory 500,maxmemory=1000,hugepages=off " , " edit-simple-memory " )
c . add_compare ( " --vcpus 10,maxvcpus=20,cores=5,sockets=4,threads=1 " , " edit-simple-vcpus " )
c . add_compare ( " --cpu model=pentium2,+x2apic,forbid=pbe " , " edit-simple-cpu " )
c . add_compare ( " --numatune 1-5,7,mode=strict " , " edit-simple-numatune " )
c . add_compare ( " --blkiotune weight=500,device_path=/dev/sdf,device_weight=600 " , " edit-simple-blkiotune " )
2014-09-16 10:02:57 +08:00
c . add_compare ( " --idmap uid_start=0,uid_target=2000,uid_count=30,gid_start=0,gid_target=3000,gid_count=40 " , " edit-simple-idmap " , compare_check = support . SUPPORT_CONN_LOADER_ROM )
c . add_compare ( " --boot loader=foo.bar,useserial=on,init=/bin/bash " , " edit-simple-boot " , compare_check = support . SUPPORT_CONN_LOADER_ROM )
2014-02-06 08:09:26 +08:00
c . add_compare ( " --security label=foo,bar,baz,UNKNOWN=val,relabel=on " , " edit-simple-security " )
c . add_compare ( " --features eoi=on,hyperv_relaxed=off,acpi= " , " edit-simple-features " )
c . add_compare ( " --clock offset=localtime,hpet_present=yes,kvmclock_present=no,rtc_tickpolicy=merge " , " edit-simple-clock " )
c . add_compare ( " --pm suspend_to_mem=yes,suspend_to_disk=no " , " edit-simple-pm " )
c . add_compare ( " --disk /dev/zero,perms=ro,startup_policy=optional " , " edit-simple-disk " )
c . add_compare ( " --disk path= " , " edit-simple-disk-remove-path " )
c . add_compare ( " --network source=br0,type=bridge,model=virtio,mac= " , " edit-simple-network " )
c . add_compare ( " --graphics tlsport=5902,keymap=ja " , " edit-simple-graphics " )
c . add_compare ( " --controller index=2,model=lsilogic " , " edit-simple-controller " )
c . add_compare ( " --smartcard type=spicevmc " , " edit-simple-smartcard " )
c . add_compare ( " --redirdev type=spicevmc,server=example.com:12345 " , " edit-simple-redirdev " )
c . add_compare ( " --tpm path=/dev/tpm " , " edit-simple-tpm " )
c . add_compare ( " --rng rate_bytes=3333,rate_period=4444 " , " edit-simple-rng " )
c . add_compare ( " --watchdog action=reset " , " edit-simple-watchdog " )
c . add_compare ( " --memballoon model=none " , " edit-simple-memballoon " )
c . add_compare ( " --serial pty " , " edit-simple-serial " )
c . add_compare ( " --parallel unix,path=/some/other/log " , " edit-simple-parallel " )
c . add_compare ( " --channel null " , " edit-simple-channel " )
c . add_compare ( " --console target_type=serial " , " edit-simple-console " )
c . add_compare ( " --filesystem /1/2/3,/4/5/6,mode=mapped " , " edit-simple-filesystem " )
2014-12-05 02:04:02 +08:00
c . add_compare ( " --video cirrus " , " edit-simple-video " , compare_check = support . SUPPORT_CONN_VIDEO_NEW_RAM_OUTPUT )
2014-02-06 08:09:26 +08:00
c . add_compare ( " --sound pcspk " , " edit-simple-soundhw " )
c . add_compare ( " --host-device 0x0781:0x5151,driver_name=vfio " , " edit-simple-host-device " )
2014-01-19 23:37:14 +08:00
2014-02-26 04:05:58 +08:00
c = vixml . add_category ( " edit selection " , " test-many-devices --print-diff --define " , compare_check = support . SUPPORT_CONN_INPUT_KEYBOARD )
2014-01-19 23:37:14 +08:00
c . add_invalid ( " --edit target=vvv --disk /dev/null " ) # no match found
2014-12-05 02:04:02 +08:00
c . add_compare ( " --edit 3 --sound pcspk " , " edit-pos-num " , compare_check = support . SUPPORT_CONN_VIDEO_NEW_RAM_OUTPUT )
c . add_compare ( " --edit -1 --video qxl " , " edit-neg-num " , compare_check = support . SUPPORT_CONN_VIDEO_NEW_RAM_OUTPUT )
2014-02-06 08:09:26 +08:00
c . add_compare ( " --edit all --host-device driver_name=vfio " , " edit-all " )
2014-12-05 02:04:02 +08:00
c . add_compare ( " --edit ich6 --sound pcspk " , " edit-select-sound-model " , compare_check = support . SUPPORT_CONN_VIDEO_NEW_RAM_OUTPUT )
2014-02-06 08:09:26 +08:00
c . add_compare ( " --edit target=hda --disk /dev/null " , " edit-select-disk-target " )
c . add_compare ( " --edit /tmp/foobar2 --disk shareable=off,readonly=on " , " edit-select-disk-path " )
c . add_compare ( " --edit mac=00:11:7f:33:44:55 --network target=nic55 " , " edit-select-network-mac " )
2014-01-19 23:37:14 +08:00
2014-02-26 04:05:58 +08:00
c = vixml . add_category ( " edit clear " , " test-many-devices --print-diff --define " , compare_check = support . SUPPORT_CONN_INPUT_KEYBOARD )
2014-01-26 09:20:55 +08:00
c . add_invalid ( " --edit --memory 200,clearxml=yes " ) # clear isn't wired up for memory
2014-12-05 04:17:34 +08:00
c . add_invalid ( " --edit --disk /foo/bar,target=fda,bus=fdc,device=floppy,clearxml=yes " ) # clearxml isn't supported for devices
2014-02-06 08:09:26 +08:00
c . add_compare ( " --edit --cpu host-passthrough,clearxml=yes " , " edit-clear-cpu " )
c . add_compare ( " --edit --clock offset=utc,clearxml=yes " , " edit-clear-clock " )
2014-01-26 04:44:14 +08:00
2014-02-26 04:05:58 +08:00
c = vixml . add_category ( " add/rm devices " , " test-many-devices --print-diff --define " , compare_check = support . SUPPORT_CONN_INPUT_KEYBOARD )
2014-01-26 06:06:31 +08:00
c . add_invalid ( " --add-device --security foo " ) # --add-device without a device
c . add_invalid ( " --remove-device --clock utc " ) # --remove-device without a dev
2014-02-06 08:09:26 +08:00
c . add_compare ( " --add-device --host-device net_00_1c_25_10_b1_e4 " , " add-host-device " )
c . add_compare ( " --add-device --sound pcspk " , " add-sound " )
c . add_compare ( " --add-device --disk %(EXISTIMG1)s ,bus=virtio,target=vdf " , " add-disk-basic " )
c . add_compare ( " --add-device --disk %(EXISTIMG1)s " , " add-disk-notarget " ) # filling in acceptable target
c . add_compare ( " --add-device --disk %(NEWIMG1)s ,size=.01 " , " add-disk-create-storage " )
2014-12-05 02:04:02 +08:00
c . add_compare ( " --remove-device --sound ich6 " , " remove-sound-model " , compare_check = support . SUPPORT_CONN_VIDEO_NEW_RAM_OUTPUT )
2014-02-06 08:09:26 +08:00
c . add_compare ( " --remove-device --disk 6 " , " remove-disk-index " )
c . add_compare ( " --remove-device --disk /dev/null " , " remove-disk-path " )
2014-12-05 02:04:02 +08:00
c . add_compare ( " --remove-device --video all " , " remove-video-all " , compare_check = support . SUPPORT_CONN_VIDEO_NEW_RAM_OUTPUT )
2014-09-21 01:18:20 +08:00
c . add_compare ( " --remove-device --host-device 0x04b3:0x4485 " , " remove-hostdev-name " )
2014-01-26 04:44:14 +08:00
2014-01-19 23:37:14 +08:00
2013-04-24 04:16:30 +08:00
2014-12-10 22:33:02 +08:00
####################
# virt-clone tests #
####################
2013-04-24 04:16:30 +08:00
vclon = App ( " virt-clone " )
2013-08-17 21:44:11 +08:00
c = vclon . add_category ( " remote " , " --connect %(REMOTEURI)s " )
c . add_valid ( " -o test --auto-clone " ) # Auto flag, no storage
c . add_valid ( " --original-xml %(CLONE_STORAGE_XML)s --auto-clone " ) # Auto flag w/ managed storage,
c . add_invalid ( " --original-xml %(CLONE_DISK_XML)s --auto-clone " ) # Auto flag w/ storage,
c = vclon . add_category ( " misc " , " " )
2014-09-16 10:03:27 +08:00
c . add_compare ( " --connect %(KVMURI)s -o test-for-clone --auto-clone --clone-running " , " clone-auto1 " , compare_check = support . SUPPORT_CONN_LOADER_ROM )
c . add_compare ( " -o test-clone-simple --name newvm --auto-clone --clone-running " , " clone-auto2 " , compare_check = support . SUPPORT_CONN_LOADER_ROM )
2013-08-17 21:44:11 +08:00
c . add_valid ( " -o test --auto-clone " ) # Auto flag, no storage
c . add_valid ( " --original-xml %(CLONE_DISK_XML)s --auto-clone " ) # Auto flag w/ storage,
c . add_valid ( " --original-xml %(CLONE_STORAGE_XML)s --auto-clone " ) # Auto flag w/ managed storage,
c . add_valid ( " -o test-for-clone --auto-clone --clone-running " ) # Auto flag, actual VM, skip state check
c . add_valid ( " -o test-clone-simple -n newvm --preserve-data --file /dev/default-pool/default-vol --clone-running --force " ) # Preserve data shouldn't complain about existing volume
2014-12-10 22:33:02 +08:00
c . add_invalid ( " --auto-clone " ) # Just the auto flag
2013-08-17 21:44:11 +08:00
c . add_invalid ( " -o test-for-clone --auto-clone " )
c = vclon . add_category ( " general " , " -n clonetest " )
c . add_valid ( " -o test " ) # Nodisk guest
2014-02-09 05:36:45 +08:00
c . add_valid ( " -o test --file %(NEWCLONEIMG1)s --file %(NEWCLONEIMG2)s " ) # Nodisk, but with spurious files passed
c . add_valid ( " -o test --file %(NEWCLONEIMG1)s --file %(NEWCLONEIMG2)s --prompt " ) # Working scenario w/ prompt shouldn't ask anything
c . add_valid ( " --original-xml %(CLONE_DISK_XML)s --file %(NEWCLONEIMG1)s --file %(NEWCLONEIMG2)s " ) # XML File with 2 disks
2013-08-17 21:44:11 +08:00
c . add_valid ( " --original-xml %(CLONE_DISK_XML)s --file virt-install --file %(EXISTIMG1)s --preserve " ) # XML w/ disks, overwriting existing files with --preserve
2014-02-09 05:36:45 +08:00
c . add_valid ( " --original-xml %(CLONE_DISK_XML)s --file %(NEWCLONEIMG1)s --file %(NEWCLONEIMG2)s --file %(NEWCLONEIMG3)s --force-copy=hdc " ) # XML w/ disks, force copy a readonly target
c . add_valid ( " --original-xml %(CLONE_DISK_XML)s --file %(NEWCLONEIMG1)s --file %(NEWCLONEIMG2)s --force-copy=fda " ) # XML w/ disks, force copy a target with no media
2013-08-17 21:44:11 +08:00
c . add_valid ( " --original-xml %(CLONE_STORAGE_XML)s --file %(MANAGEDNEW1)s " ) # XML w/ managed storage, specify managed path
c . add_valid ( " --original-xml %(CLONE_NOEXIST_XML)s --file %(EXISTIMG1)s --preserve " ) # XML w/ managed storage, specify managed path across pools# Libvirt test driver doesn't support cloning across pools# XML w/ non-existent storage, with --preserve
c . add_valid ( " -o test -n test-many-devices --replace " ) # Overwriting existing VM
c . add_invalid ( " -o test foobar " ) # Positional arguments error
c . add_invalid ( " -o idontexist " ) # Non-existent vm name
c . add_invalid ( " -o idontexist --auto-clone " ) # Non-existent vm name with auto flag,
c . add_invalid ( " -o test -n test " ) # Colliding new name
c . add_invalid ( " --original-xml %(CLONE_DISK_XML)s " ) # XML file with several disks, but non specified
c . add_invalid ( " --original-xml %(CLONE_DISK_XML)s --file virt-install --file %(EXISTIMG1)s " ) # XML w/ disks, overwriting existing files with no --preserve
2014-02-09 05:36:45 +08:00
c . add_invalid ( " --original-xml %(CLONE_DISK_XML)s --file %(NEWCLONEIMG1)s --file %(NEWCLONEIMG2)s --force-copy=hdc " ) # XML w/ disks, force copy but not enough disks passed
2013-08-17 21:44:11 +08:00
c . add_invalid ( " --original-xml %(CLONE_STORAGE_XML)s --file /tmp/clonevol " ) # XML w/ managed storage, specify unmanaged path (should fail)
c . add_invalid ( " --original-xml %(CLONE_NOEXIST_XML)s --file %(EXISTIMG1)s " ) # XML w/ non-existent storage, WITHOUT --preserve
2013-04-24 04:16:30 +08:00
2014-12-10 22:33:02 +08:00
######################
# virt-convert tests #
######################
vconv = App ( " virt-convert " )
c = vconv . add_category ( " misc " , " --connect %(KVMURI)s --dry " )
c . add_invalid ( " %(VMX_IMG1)s --input-format foo " ) # invalid input format
c . add_invalid ( " %(EXISTIMG1)s " ) # invalid input file
c . add_compare ( " %(VMX_IMG1)s --disk-format qcow2 --print-xml " , " vmx-compare " )
c . add_compare ( " %(OVF_IMG1)s --disk-format none --destination /tmp --print-xml " , " ovf-compare " )
2013-04-24 04:16:30 +08:00
2013-03-18 05:06:52 +08:00
##########################
# Automated prompt tests #
##########################
2014-02-05 05:16:39 +08:00
_p = PromptTest ( " virt-xml --connect %(TESTURI)s --confirm test "
2014-01-19 23:37:14 +08:00
" --edit --cpu host-passthrough " )
2014-02-05 05:16:39 +08:00
_p . add ( " Define ' test ' with the changed XML " , " yes " , num_lines = 10 )
promptlist . append ( _p )
2013-03-18 05:06:52 +08:00
2013-04-24 04:16:30 +08:00
2013-03-18 05:06:52 +08:00
#########################
# Test runner functions #
#########################
newidx = 0
curtest = 0
2013-04-14 02:34:52 +08:00
2013-03-18 05:06:52 +08:00
def setup ( ) :
"""
Create initial test files / dirs
"""
for i in exist_files :
os . system ( " touch %s " % i )
def cleanup ( ) :
"""
Cleanup temporary files used for testing
"""
for i in clean_files :
os . system ( " chmod 777 %s > /dev/null 2>&1 " % i )
os . system ( " rm -rf %s > /dev/null 2>&1 " % i )
2013-04-14 02:34:52 +08:00
2013-03-18 05:06:52 +08:00
class CLITests ( unittest . TestCase ) :
def setUp ( self ) :
global curtest
curtest + = 1
# Only run this for first test
if curtest == 1 :
setup ( )
def tearDown ( self ) :
# Only run this on the last test
if curtest == newidx :
cleanup ( )
2013-04-14 02:34:52 +08:00
2013-03-18 05:06:52 +08:00
def maketest ( cmd ) :
2013-08-18 01:50:36 +08:00
def cmdtemplate ( self , _cmdobj ) :
2013-08-18 20:59:19 +08:00
_cmdobj . run ( self )
2013-03-18 05:06:52 +08:00
return lambda s : cmdtemplate ( s , cmd )
2014-02-02 03:17:03 +08:00
_cmdlist = promptlist [ : ]
2013-04-24 04:16:30 +08:00
_cmdlist + = vinst . cmds
_cmdlist + = vclon . cmds
_cmdlist + = vconv . cmds
2014-01-19 23:37:14 +08:00
_cmdlist + = vixml . cmds
2013-04-24 04:16:30 +08:00
2013-03-18 05:06:52 +08:00
for _cmd in _cmdlist :
newidx + = 1
2014-02-02 03:17:03 +08:00
_name = " testCLI "
if _cmd in promptlist :
_name + = " prompt "
_name + = " %s %.4d " % ( os . path . basename ( _cmd . app . replace ( " - " , " " ) ) , newidx )
setattr ( CLITests , _name , maketest ( _cmd ) )
2013-03-18 05:06:52 +08:00
atexit . register ( cleanup )