2013-07-05 16:59:58 +04:00
#
2015-02-24 15:21:24 +03:00
# Copyright 2013, 2014, 2015 Red Hat, Inc.
2013-07-05 16:59:58 +04: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 00:59:47 +04:00
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
2013-07-05 16:59:58 +04: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 logging
import re
2013-07-10 03:50:49 +04:00
import weakref
2013-07-05 16:59:58 +04:00
import libvirt
2014-09-12 23:59:22 +04:00
from . import pollhelpers
from . import support
from . import util
2014-12-09 16:22:51 +03:00
from . import URISplit
2015-04-03 19:40:16 +03:00
from . import Capabilities
2014-09-12 23:59:22 +04:00
from . cli import VirtOptionString
from . guest import Guest
2014-09-20 19:30:24 +04:00
from . nodedev import NodeDevice
2014-09-12 23:59:22 +04:00
from . storage import StoragePool , StorageVolume
2015-04-06 22:42:40 +03:00
from virtcli import CLIConfig
2013-07-05 16:59:58 +04:00
_virtinst_uri_magic = " __virtinst_test__ "
2013-07-11 03:42:28 +04:00
def _sanitize_xml ( xml ) :
import difflib
orig = xml
2013-10-06 20:54:59 +04:00
xml = re . sub ( " arch= \" .* \" " , " arch= \" i686 \" " , xml )
xml = re . sub ( " domain type= \" .* \" " , " domain type= \" test \" " , xml )
2013-07-12 22:19:54 +04:00
xml = re . sub ( " machine type= \" .* \" " , " " , xml )
2013-07-11 03:42:28 +04:00
xml = re . sub ( " >exe< " , " >hvm< " , xml )
2013-10-06 20:54:59 +04:00
diff = " \n " . join ( difflib . unified_diff ( orig . split ( " \n " ) ,
xml . split ( " \n " ) ) )
if diff :
logging . debug ( " virtinst test sanitizing diff \n : %s " , diff )
2013-07-11 03:42:28 +04:00
return xml
2013-07-05 16:59:58 +04:00
class VirtualConnection ( object ) :
"""
Wrapper for libvirt connection that provides various bits like
- caching static data
- lookup for API feature support
- simplified API wrappers that handle new and old ways of doing things
"""
def __init__ ( self , uri ) :
2014-02-06 04:09:26 +04:00
self . _initial_uri = uri or " "
2013-07-06 04:36:28 +04:00
2013-07-17 01:15:51 +04:00
self . _fake_pretty_name = None
self . _fake_libvirt_version = None
self . _fake_conn_version = None
self . _daemon_version = None
self . _conn_version = None
2014-02-06 04:09:26 +04:00
if self . _initial_uri . startswith ( _virtinst_uri_magic ) :
2013-07-17 01:15:51 +04:00
# virtinst unit test URI handling
2014-02-06 04:09:26 +04:00
uri = self . _initial_uri . replace ( _virtinst_uri_magic , " " )
2013-07-06 04:36:28 +04:00
ret = uri . split ( " , " , 1 )
self . _open_uri = ret [ 0 ]
2014-01-19 22:56:06 +04:00
self . _test_opts = VirtOptionString (
2014-01-26 00:44:14 +04:00
len ( ret ) > 1 and ret [ 1 ] or " " , [ ] , None ) . opts
2013-07-17 01:15:51 +04:00
self . _early_virtinst_test_uri ( )
2013-07-06 04:36:28 +04:00
self . _uri = self . _virtinst_uri_make_fake ( )
else :
2014-02-06 04:09:26 +04:00
self . _open_uri = self . _initial_uri
self . _uri = self . _initial_uri
2013-07-06 04:36:28 +04:00
self . _test_opts = { }
2013-07-05 16:59:58 +04:00
2013-07-06 04:36:28 +04:00
self . _libvirtconn = None
2014-12-09 16:22:51 +03:00
self . _urisplits = URISplit ( self . _uri )
2013-07-06 23:53:35 +04:00
self . _caps = None
2013-07-06 23:39:00 +04:00
2013-07-06 23:53:35 +04:00
self . _support_cache = { }
2013-07-08 00:38:11 +04:00
self . _fetch_cache = { }
# Setting this means we only do fetch_all* once and just carry
# the result. For the virt-* CLI tools this ensures any revalidation
# isn't hammering the connection over and over
self . cache_object_fetch = False
2013-07-06 23:53:35 +04:00
2013-07-08 02:54:08 +04:00
# These let virt-manager register a callback which provides its
# own cached object lists, rather than doing fresh calls
self . cb_fetch_all_guests = None
self . cb_fetch_all_pools = None
2013-09-29 04:05:13 +04:00
self . cb_fetch_all_vols = None
2014-09-20 19:30:24 +04:00
self . cb_fetch_all_nodedevs = None
2014-01-18 23:57:39 +04:00
self . cb_clear_cache = None
2013-07-08 02:54:08 +04:00
2013-07-06 22:12:13 +04:00
2013-07-06 23:53:35 +04:00
##############
# Properties #
##############
2013-07-05 16:59:58 +04:00
def __getattr__ ( self , attr ) :
if attr in self . __dict__ :
return self . __dict__ [ attr ]
2014-03-20 21:34:34 +04:00
# Proxy virConnect API calls
2013-07-05 16:59:58 +04:00
libvirtconn = self . __dict__ . get ( " _libvirtconn " )
return getattr ( libvirtconn , attr )
2013-07-06 04:36:28 +04:00
def _get_uri ( self ) :
return self . _uri or self . _open_uri
uri = property ( _get_uri )
2013-07-06 22:12:13 +04:00
def _get_caps ( self ) :
if not self . _caps :
2015-04-03 19:40:16 +03:00
self . _caps = Capabilities ( self ,
self . _libvirtconn . getCapabilities ( ) )
2013-07-06 22:12:13 +04:00
return self . _caps
caps = property ( _get_caps )
2014-03-20 21:34:34 +04:00
def get_conn_for_api_arg ( self ) :
return self . _libvirtconn
2013-07-05 16:59:58 +04:00
##############
# Public API #
##############
2014-02-12 01:26:15 +04:00
def is_closed ( self ) :
return not bool ( self . _libvirtconn )
2013-07-05 16:59:58 +04:00
def close ( self ) :
self . _libvirtconn = None
2013-07-06 04:36:28 +04:00
self . _uri = None
2013-07-08 00:38:11 +04:00
self . _fetch_cache = { }
2013-07-05 16:59:58 +04:00
2013-07-06 22:12:13 +04:00
def invalidate_caps ( self ) :
self . _caps = None
2013-07-05 16:59:58 +04:00
def is_open ( self ) :
return bool ( self . _libvirtconn )
def open ( self , passwordcb ) :
open_flags = 0
valid_auth_options = [ libvirt . VIR_CRED_AUTHNAME ,
libvirt . VIR_CRED_PASSPHRASE ]
authcb = self . _auth_cb
authcb_data = passwordcb
2013-07-06 04:36:28 +04:00
conn = libvirt . openAuth ( self . _open_uri ,
2013-07-05 16:59:58 +04:00
[ valid_auth_options , authcb ,
( authcb_data , valid_auth_options ) ] ,
open_flags )
2013-07-06 04:36:28 +04:00
self . _fixup_virtinst_test_uri ( conn )
2013-07-05 16:59:58 +04:00
self . _libvirtconn = conn
2013-07-14 00:31:05 +04:00
if not self . _open_uri :
self . _uri = self . _libvirtconn . getURI ( )
2014-12-09 16:22:51 +03:00
self . _urisplits = URISplit ( self . _uri )
2013-07-05 16:59:58 +04:00
2014-09-20 19:30:24 +04:00
def set_keep_alive ( self , interval , count ) :
if hasattr ( self . _libvirtconn , " setKeepAlive " ) :
self . _libvirtconn . setKeepAlive ( interval , count )
####################
# Polling routines #
####################
2014-01-18 23:57:39 +04:00
_FETCH_KEY_GUESTS = " vms "
_FETCH_KEY_POOLS = " pools "
_FETCH_KEY_VOLS = " vols "
2014-09-20 19:30:24 +04:00
_FETCH_KEY_NODEDEVS = " nodedevs "
def clear_cache ( self , pools = False ) :
if self . cb_clear_cache :
self . cb_clear_cache ( pools = pools ) # pylint: disable=not-callable
return
if pools :
self . _fetch_cache . pop ( self . _FETCH_KEY_POOLS , None )
2014-01-18 23:57:39 +04:00
2013-09-29 04:05:13 +04:00
def _fetch_all_guests_cached ( self ) :
2014-01-18 23:57:39 +04:00
key = self . _FETCH_KEY_GUESTS
2013-07-08 00:38:11 +04:00
if key in self . _fetch_cache :
return self . _fetch_cache [ key ]
2015-04-10 19:52:42 +03:00
ignore , ignore , ret = pollhelpers . fetch_vms (
self , { } , lambda obj , ignore : obj )
2013-07-10 03:50:49 +04:00
ret = [ Guest ( weakref . ref ( self ) , parsexml = obj . XMLDesc ( 0 ) )
2015-04-10 19:52:42 +03:00
for obj in ret ]
2013-07-08 00:38:11 +04:00
if self . cache_object_fetch :
self . _fetch_cache [ key ] = ret
return ret
2013-07-07 22:54:48 +04:00
2013-09-29 04:05:13 +04:00
def fetch_all_guests ( self ) :
"""
Returns a list of Guest ( ) objects
"""
if self . cb_fetch_all_guests :
2014-04-03 02:39:43 +04:00
return self . cb_fetch_all_guests ( ) # pylint: disable=not-callable
2013-09-29 04:05:13 +04:00
return self . _fetch_all_guests_cached ( )
2013-07-08 02:54:08 +04:00
2013-09-29 04:05:13 +04:00
def _fetch_all_pools_cached ( self ) :
2014-01-18 23:57:39 +04:00
key = self . _FETCH_KEY_POOLS
2013-07-08 00:38:11 +04:00
if key in self . _fetch_cache :
return self . _fetch_cache [ key ]
2015-04-10 19:52:42 +03:00
ignore , ignore , ret = pollhelpers . fetch_pools (
self , { } , lambda obj , ignore : obj )
2013-09-29 04:05:13 +04:00
ret = [ StoragePool ( weakref . ref ( self ) , parsexml = obj . XMLDesc ( 0 ) )
2015-04-10 19:52:42 +03:00
for obj in ret ]
2013-09-29 04:05:13 +04:00
if self . cache_object_fetch :
self . _fetch_cache [ key ] = ret
return ret
def fetch_all_pools ( self ) :
"""
Returns a list of StoragePool objects
"""
if self . cb_fetch_all_pools :
2014-04-03 02:39:43 +04:00
return self . cb_fetch_all_pools ( ) # pylint: disable=not-callable
2013-09-29 04:05:13 +04:00
return self . _fetch_all_pools_cached ( )
def _fetch_all_vols_cached ( self ) :
2014-01-18 23:57:39 +04:00
key = self . _FETCH_KEY_VOLS
2013-09-29 04:05:13 +04:00
if key in self . _fetch_cache :
return self . _fetch_cache [ key ]
ret = [ ]
for xmlobj in self . fetch_all_pools ( ) :
pool = self . _libvirtconn . storagePoolLookupByName ( xmlobj . name )
2014-12-05 03:25:15 +03:00
if pool . info ( ) [ 0 ] != libvirt . VIR_STORAGE_POOL_RUNNING :
continue
2013-09-29 17:31:39 +04:00
ignore , ignore , vols = pollhelpers . fetch_volumes (
self , pool , { } , lambda obj , ignore : obj )
2014-02-12 22:57:44 +04:00
2015-04-10 19:52:42 +03:00
for vol in vols :
2014-02-12 22:57:44 +04:00
try :
xml = vol . XMLDesc ( 0 )
ret . append ( StorageVolume ( weakref . ref ( self ) , parsexml = xml ) )
2014-03-06 21:04:08 +04:00
except Exception , e :
2014-02-12 22:57:44 +04:00
logging . debug ( " Fetching volume XML failed: %s " , e )
2013-09-29 04:05:13 +04:00
2013-07-08 00:38:11 +04:00
if self . cache_object_fetch :
self . _fetch_cache [ key ] = ret
return ret
2013-07-07 22:54:48 +04:00
2013-09-29 04:05:13 +04:00
def fetch_all_vols ( self ) :
"""
Returns a list of StorageVolume objects
"""
if self . cb_fetch_all_vols :
2014-04-03 02:39:43 +04:00
return self . cb_fetch_all_vols ( ) # pylint: disable=not-callable
2013-09-29 04:05:13 +04:00
return self . _fetch_all_vols_cached ( )
2014-09-20 19:30:24 +04:00
def _fetch_all_nodedevs_cached ( self ) :
key = self . _FETCH_KEY_NODEDEVS
if key in self . _fetch_cache :
return self . _fetch_cache [ key ]
2014-01-18 23:57:39 +04:00
2014-09-20 19:30:24 +04:00
ignore , ignore , ret = pollhelpers . fetch_nodedevs (
self , { } , lambda obj , ignore : obj )
ret = [ NodeDevice . parse ( weakref . ref ( self ) , obj . XMLDesc ( 0 ) )
2015-04-10 19:52:42 +03:00
for obj in ret ]
2014-09-20 19:30:24 +04:00
if self . cache_object_fetch :
self . _fetch_cache [ key ] = ret
return ret
def fetch_all_nodedevs ( self ) :
"""
Returns a list of NodeDevice ( ) objects
"""
if self . cb_fetch_all_nodedevs :
return self . cb_fetch_all_nodedevs ( ) # pylint: disable=not-callable
return self . _fetch_all_nodedevs_cached ( )
2013-07-26 00:21:30 +04:00
2013-07-05 16:59:58 +04:00
2013-07-11 03:42:28 +04:00
#########################
# Libvirt API overrides #
#########################
def getURI ( self ) :
return self . _uri
2013-07-06 23:39:00 +04:00
#########################
# Public version checks #
#########################
def local_libvirt_version ( self ) :
if self . _fake_libvirt_version is not None :
return self . _fake_libvirt_version
# This handles caching for us
return util . local_libvirt_version ( )
def daemon_version ( self ) :
if self . _fake_libvirt_version is not None :
return self . _fake_libvirt_version
if not self . is_remote ( ) :
return self . local_libvirt_version ( )
if not self . _daemon_version :
2013-10-06 18:08:04 +04:00
if not self . check_support ( support . SUPPORT_CONN_LIBVERSION ) :
2013-07-06 23:39:00 +04:00
self . _daemon_version = 0
else :
2014-03-20 21:34:34 +04:00
self . _daemon_version = self . _libvirtconn . getLibVersion ( )
2013-07-06 23:39:00 +04:00
return self . _daemon_version
def conn_version ( self ) :
if self . _fake_conn_version is not None :
return self . _fake_conn_version
if not self . _conn_version :
2013-10-06 18:08:04 +04:00
if not self . check_support ( support . SUPPORT_CONN_GETVERSION ) :
2013-07-06 23:39:00 +04:00
self . _conn_version = 0
else :
2014-03-20 21:34:34 +04:00
self . _conn_version = self . _libvirtconn . getVersion ( )
2013-07-06 23:39:00 +04:00
return self . _conn_version
2015-03-25 16:04:51 +03:00
def stable_defaults ( self , emulator = None , force = False ) :
"""
: param force : Just check if we are running on RHEL , regardless of
whether stable defaults are requested by the build . This is needed
to ensure we don ' t enable VM devices that are compiled out on
RHEL , like vmvga
"""
2015-04-06 22:42:40 +03:00
if not CLIConfig . stable_defaults and not force :
2015-03-23 23:48:43 +03:00
return False
2015-02-24 15:21:24 +03:00
if not self . is_qemu_system ( ) :
return False
2015-03-23 23:48:43 +03:00
2015-02-24 15:21:24 +03:00
if emulator :
2015-03-23 23:48:43 +03:00
return str ( emulator ) . startswith ( " /usr/libexec " )
for guest in self . caps . guests :
for dom in guest . domains :
if dom . emulator . startswith ( " /usr/libexec " ) :
return True
return False
2013-07-06 23:39:00 +04:00
2013-07-06 04:36:28 +04:00
###################
# Public URI bits #
###################
2013-07-17 01:15:51 +04:00
def fake_name ( self ) :
return self . _fake_pretty_name
2013-07-06 04:36:28 +04:00
def is_remote ( self ) :
2014-02-02 03:59:10 +04:00
return ( hasattr ( self , " _virtinst__fake_conn_remote " ) or
2014-12-09 16:22:51 +03:00
self . _urisplits . hostname )
2013-07-06 04:36:28 +04:00
def get_uri_hostname ( self ) :
2015-04-11 19:57:32 +03:00
return self . _urisplits . hostname
2015-04-11 19:08:57 +03:00
def get_uri_port ( self ) :
return self . _urisplits . port
def get_uri_username ( self ) :
return self . _urisplits . username
2013-07-06 04:36:28 +04:00
def get_uri_transport ( self ) :
2015-04-11 19:08:57 +03:00
return self . _urisplits . transport
2015-04-11 19:57:32 +03:00
def get_uri_path ( self ) :
return self . _urisplits . path
2013-07-06 04:36:28 +04:00
def get_uri_driver ( self ) :
2014-12-09 16:22:51 +03:00
return self . _urisplits . scheme
2013-07-06 04:36:28 +04:00
def is_session_uri ( self ) :
2015-04-11 19:57:32 +03:00
return self . get_uri_path ( ) == " /session "
2013-07-06 04:36:28 +04:00
def is_qemu ( self ) :
2014-12-09 16:22:51 +03:00
return self . _urisplits . scheme . startswith ( " qemu " )
2013-07-06 04:36:28 +04:00
def is_qemu_system ( self ) :
2014-12-09 16:22:51 +03:00
return ( self . is_qemu ( ) and self . _urisplits . path == " /system " )
2013-07-06 04:36:28 +04:00
def is_qemu_session ( self ) :
return ( self . is_qemu ( ) and self . is_session_uri ( ) )
2014-12-09 20:36:09 +03:00
def is_really_test ( self ) :
return URISplit ( self . _open_uri ) . scheme . startswith ( " test " )
2013-07-06 04:36:28 +04:00
def is_test ( self ) :
2014-12-09 16:22:51 +03:00
return self . _urisplits . scheme . startswith ( " test " )
2013-07-06 04:36:28 +04:00
def is_xen ( self ) :
2014-12-09 16:22:51 +03:00
return ( self . _urisplits . scheme . startswith ( " xen " ) or
self . _urisplits . scheme . startswith ( " libxl " ) )
2013-07-06 04:36:28 +04:00
def is_lxc ( self ) :
2014-12-09 16:22:51 +03:00
return self . _urisplits . scheme . startswith ( " lxc " )
2013-07-06 04:36:28 +04:00
def is_openvz ( self ) :
2014-12-09 16:22:51 +03:00
return self . _urisplits . scheme . startswith ( " openvz " )
2013-07-06 04:36:28 +04:00
def is_container ( self ) :
return self . is_lxc ( ) or self . is_openvz ( )
2013-07-05 16:59:58 +04:00
2013-07-06 19:20:28 +04:00
#########################
# Support check helpers #
#########################
for _supportname in [ _supportname for _supportname in dir ( support ) if
_supportname . startswith ( " SUPPORT_ " ) ] :
locals ( ) [ _supportname ] = getattr ( support , _supportname )
2013-10-06 18:08:04 +04:00
def check_support ( self , feature , data = None ) :
2013-07-06 23:53:35 +04:00
key = feature
2013-10-06 18:08:04 +04:00
data = data or self
2013-07-06 23:53:35 +04:00
if key not in self . _support_cache :
2013-09-29 17:26:03 +04:00
self . _support_cache [ key ] = support . check_support (
self , feature , data )
2013-07-06 23:53:35 +04:00
return self . _support_cache [ key ]
2013-09-29 17:26:03 +04:00
2013-10-06 18:08:04 +04:00
def support_remote_url_install ( self ) :
if hasattr ( self , " _virtinst__fake_conn " ) :
return False
return ( self . check_support ( self . SUPPORT_CONN_STREAM ) and
self . check_support ( self . SUPPORT_STREAM_UPLOAD ) )
2013-07-06 19:20:28 +04:00
2013-07-05 16:59:58 +04:00
###################
# Private helpers #
###################
def _auth_cb ( self , creds , ( passwordcb , passwordcreds ) ) :
for cred in creds :
if cred [ 0 ] not in passwordcreds :
raise RuntimeError ( " Unknown cred type ' %s ' , expected only "
" %s " % ( cred [ 0 ] , passwordcreds ) )
return passwordcb ( creds )
2013-07-06 04:36:28 +04:00
def _virtinst_uri_make_fake ( self ) :
if " qemu " in self . _test_opts :
return " qemu+abc:///system "
elif " xen " in self . _test_opts :
return " xen+abc:/// "
elif " lxc " in self . _test_opts :
return " lxc+abc:/// "
return self . _open_uri
2013-07-17 01:15:51 +04:00
def _early_virtinst_test_uri ( self ) :
# Need tmpfile names to be deterministic
2013-07-06 04:36:28 +04:00
if not self . _test_opts :
return
2013-07-17 01:15:51 +04:00
opts = self . _test_opts
2013-07-06 04:36:28 +04:00
2013-07-05 16:59:58 +04:00
if " predictable " in opts :
opts . pop ( " predictable " )
setattr ( self , " _virtinst__fake_conn_predictable " , True )
# Fake remote status
if " remote " in opts :
opts . pop ( " remote " )
setattr ( self , " _virtinst__fake_conn_remote " , True )
2013-07-17 01:15:51 +04:00
if " prettyname " in opts :
self . _fake_pretty_name = opts . pop ( " prettyname " )
def _fixup_virtinst_test_uri ( self , conn ) :
"""
This hack allows us to fake various drivers via passing a magic
URI string to virt - * . Helps with testing
"""
if not self . _test_opts :
return
opts = self . _test_opts . copy ( )
2013-07-05 16:59:58 +04:00
# Fake capabilities
if " caps " in opts :
capsxml = file ( opts . pop ( " caps " ) ) . read ( )
conn . getCapabilities = lambda : capsxml
2015-02-18 22:49:53 +03:00
# Fake domcapabilities. This is insufficient since output should
# vary per type/arch/emulator combo, but it can be expanded later
# if needed
if " domcaps " in opts :
domcapsxml = file ( opts . pop ( " domcaps " ) ) . read ( )
2015-02-22 19:02:55 +03:00
def fake_domcaps ( emulator , arch , machine , virttype , flags = 0 ) :
ignore = emulator
ignore = flags
ignore = machine
ignore = virttype
ret = domcapsxml
if arch :
ret = re . sub ( " arch>.+</arch " , " arch> %s </arch " % arch , ret )
return ret
2015-02-18 22:49:53 +03:00
conn . getDomainCapabilities = fake_domcaps
2013-07-05 16:59:58 +04:00
if ( " qemu " in opts ) or ( " xen " in opts ) or ( " lxc " in opts ) :
2013-07-06 04:36:28 +04:00
opts . pop ( " qemu " , None )
opts . pop ( " xen " , None )
opts . pop ( " lxc " , None )
2013-07-06 23:39:00 +04:00
self . _fake_conn_version = 10000000000
2013-07-05 16:59:58 +04:00
origcreate = conn . createLinux
origdefine = conn . defineXML
def newcreate ( xml , flags ) :
2013-07-11 03:42:28 +04:00
xml = _sanitize_xml ( xml )
2013-07-05 16:59:58 +04:00
return origcreate ( xml , flags )
def newdefine ( xml ) :
2013-07-11 03:42:28 +04:00
xml = _sanitize_xml ( xml )
2013-07-05 16:59:58 +04:00
return origdefine ( xml )
conn . createLinux = newcreate
conn . defineXML = newdefine
# These need to come after the HV setter, since that sets a default
# conn version
if " connver " in opts :
2013-07-06 23:39:00 +04:00
self . _fake_conn_version = int ( opts . pop ( " connver " ) )
2013-07-05 16:59:58 +04:00
if " libver " in opts :
2013-07-06 23:39:00 +04:00
self . _fake_libvirt_version = int ( opts . pop ( " libver " ) )
2013-07-05 16:59:58 +04:00
if opts :
raise RuntimeError ( " Unhandled virtinst test uri options %s " % opts )
setattr ( self , " _virtinst__fake_conn " , True )