2013-03-18 01:06:52 +04:00
#
# Common code for all guests
#
2013-07-09 16:53:22 +04:00
# Copyright 2006-2009, 2013 Red Hat, Inc.
2013-03-18 01:06:52 +04:00
# Jeremy Katz <katzj@redhat.com>
#
# 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
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# 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 os
import time
import logging
import signal
import urlgrabber . progress as progress
import libvirt
import libxml2
2013-07-17 15:53:47 +04:00
import virtinst
2013-04-11 03:48:07 +04:00
from virtinst import util
from virtinst import support
2013-07-17 15:53:47 +04:00
from virtinst . osxml import OSXML
2013-07-14 02:56:09 +04:00
from virtinst . xmlbuilder import XMLBuilder , XMLProperty
2013-04-11 03:48:07 +04:00
from virtinst . VirtualDevice import VirtualDevice
from virtinst . VirtualDisk import VirtualDisk
from virtinst . VirtualInputDevice import VirtualInputDevice
2013-07-16 05:52:18 +04:00
from virtinst . VirtualController import VirtualController
2013-04-11 03:48:07 +04:00
from virtinst . Clock import Clock
from virtinst . Seclabel import Seclabel
from virtinst . CPU import CPU
from virtinst . DomainNumatune import DomainNumatune
from virtinst . DomainFeatures import DomainFeatures
from virtinst import osdict
2013-03-18 01:06:52 +04:00
2013-07-14 02:56:09 +04:00
class Guest ( XMLBuilder ) :
2013-03-18 01:06:52 +04:00
# OS Dictionary static variables and methods
_DEFAULTS = osdict . DEFAULTS
_OS_TYPES = osdict . OS_TYPES
@staticmethod
def pretty_os_list ( ) :
"""
Return a strip representation of OS list for printing
"""
ret = " "
for t in Guest . list_os_types ( ) :
for v in Guest . list_os_variants ( t ) :
ret + = " %-20s : %s \n " % ( v , Guest . get_os_variant_label ( t , v ) )
return ret
@staticmethod
def list_os_types ( supported = False , filtervars = None ) :
"""
@param filtervars : List of only variants we want to show by default
"""
vals = osdict . sort_helper ( Guest . _OS_TYPES )
for t in vals [ : ] :
if not Guest . list_os_variants ( t , supported = supported ,
filtervars = filtervars ) :
vals . remove ( t )
return vals
@staticmethod
2013-04-12 16:26:21 +04:00
def list_os_variants ( typ , sortpref = None , supported = False , filtervars = None ) :
2013-03-18 01:06:52 +04:00
"""
Return a list of sorted os variants for the passed distro type
@param sortpref : An option list of osdict ' distro ' tags to
prioritize in the returned list , e . g . passing [ " fedora " ] will make
the sorted list have all fedora distros first
@param filtervars : List of only variants we want to show by default
"""
2013-04-12 16:26:21 +04:00
vals = osdict . sort_helper ( Guest . _OS_TYPES [ typ ] [ " variants " ] ,
2013-03-18 01:06:52 +04:00
sortpref )
ret = [ ]
for v in vals :
if filtervars :
if v not in filtervars :
continue
elif supported :
if not osdict . lookup_osdict_key ( None , None ,
2013-04-12 16:26:21 +04:00
typ , v , " supported " ) :
2013-03-18 01:06:52 +04:00
continue
ret . append ( v )
return ret
@staticmethod
2013-04-12 16:26:21 +04:00
def get_os_type_label ( typ ) :
return Guest . _OS_TYPES [ typ ] [ " label " ]
2013-03-18 01:06:52 +04:00
@staticmethod
2013-04-12 16:26:21 +04:00
def get_os_variant_label ( typ , variant ) :
return Guest . _OS_TYPES [ typ ] [ " variants " ] [ variant ] [ " label " ]
2013-03-18 01:06:52 +04:00
@staticmethod
def cpuset_str_to_tuple ( conn , cpuset ) :
return DomainNumatune . cpuset_str_to_tuple ( conn , cpuset )
@staticmethod
def generate_cpuset ( conn , mem ) :
"""
Generates a cpu pinning string based on host NUMA configuration .
If host doesn ' t have a suitable NUMA configuration, a RuntimeError
is thrown .
"""
2013-07-06 22:12:13 +04:00
if conn . caps . host . topology is None :
2013-03-18 01:06:52 +04:00
raise RuntimeError ( _ ( " No topology section in capabilities xml. " ) )
2013-07-06 22:12:13 +04:00
cells = conn . caps . host . topology . cells
2013-03-18 01:06:52 +04:00
if len ( cells ) < = 1 :
raise RuntimeError ( _ ( " Capabilities only show <= 1 cell. "
" Not NUMA capable " ) )
# Capabilities tells us about the available memory 'cells' on the
# system. Each 'cell' has associated 'cpu's.
#
# Use getCellsFreeMemory to determine which 'cell' has the smallest
# amount of memory which fits the requested VM memory amount, then
# pin the VM to that 'cell's associated 'cpu's
cell_mem = conn . getCellsFreeMemory ( 0 , len ( cells ) )
cell_id = - 1
for i in range ( len ( cells ) ) :
if cell_mem [ i ] < mem :
# Cell doesn't have enough mem to fit, skip it
continue
if len ( cells [ i ] . cpus ) == 0 :
# No cpus to use for the cell
continue
# Find smallest cell that fits
if cell_id < 0 or cell_mem [ i ] < cell_mem [ cell_id ] :
cell_id = i
if cell_id < 0 :
raise RuntimeError ( _ ( " Could not find any usable NUMA "
" cell/cpu combinations. " ) )
# Build cpuset string
cpustr = " "
for cpu in cells [ cell_id ] . cpus :
if cpustr != " " :
cpustr + = " , "
cpustr + = str ( cpu . id )
return cpustr
2013-07-17 00:47:08 +04:00
def __init__ ( self , conn , parsexml = None , parsexmlnode = None ) :
2013-03-18 01:06:52 +04:00
self . _name = None
self . _uuid = None
self . _memory = None
self . _maxmemory = None
self . _hugepage = None
self . _vcpus = 1
self . _maxvcpus = 1
self . _cpuset = None
self . _autostart = False
self . _clock = None
self . _seclabel = None
self . _description = None
self . _features = None
self . _replace = None
self . _emulator = None
2013-07-17 04:39:24 +04:00
self . _type = None
2013-03-18 01:06:52 +04:00
self . _os_type = None
self . _os_variant = None
self . _os_autodetect = False
2013-07-17 00:47:08 +04:00
self . installer = None
2013-03-18 01:06:52 +04:00
# General device list. Only access through API calls (even internally)
self . _devices = [ ]
# Device list to use/alter during install process. Don't access
# directly, use internal APIs
self . _install_devices = [ ]
# The libvirt virDomain object we 'Create'
self . domain = None
self . _consolechild = None
2013-07-14 02:56:09 +04:00
XMLBuilder . __init__ ( self , conn , parsexml )
2013-03-18 01:06:52 +04:00
if self . _is_parse ( ) :
return
2013-07-17 00:47:08 +04:00
self . installer = virtinst . DistroInstaller ( conn )
2013-03-18 01:06:52 +04:00
2013-07-17 04:39:24 +04:00
self . _type = " xen "
2013-03-18 01:06:52 +04:00
# Need to do this after all parameter init
2013-07-17 15:53:47 +04:00
self . os = OSXML ( self . conn , parsexml , parsexmlnode )
self . features = DomainFeatures ( self . conn )
self . clock = Clock ( self . conn )
self . seclabel = Seclabel ( self . conn )
self . cpu = CPU ( self . conn )
self . numatune = DomainNumatune ( self . conn )
2013-03-18 01:06:52 +04:00
######################
# Property accessors #
######################
# Domain name of the guest
def get_name ( self ) :
return self . _name
def set_name ( self , val ) :
2013-04-11 18:27:02 +04:00
util . validate_name ( _ ( " Guest " ) , val , lencheck = True )
2013-03-18 01:06:52 +04:00
do_fail = False
2013-04-13 22:34:52 +04:00
if self . replace is not True :
2013-03-18 01:06:52 +04:00
try :
self . conn . lookupByName ( val )
do_fail = True
except :
# Name not found
pass
if do_fail :
raise ValueError ( _ ( " Guest name ' %s ' is already in use. " ) % val )
self . _name = val
2013-07-14 02:56:09 +04:00
name = XMLProperty ( get_name , set_name ,
2013-03-18 01:06:52 +04:00
xpath = " ./name " )
# Memory allocated to the guest. Should be given in MB
def get_memory ( self ) :
return self . _memory
def set_memory ( self , val ) :
self . _memory = val
if self . maxmemory is None or self . maxmemory < val :
self . maxmemory = val
2013-07-14 07:07:01 +04:00
memory = XMLProperty ( get_memory , set_memory , is_int = True ,
xpath = " ./currentMemory " )
2013-03-18 01:06:52 +04:00
# Memory allocated to the guest. Should be given in MB
def get_maxmemory ( self ) :
return self . _maxmemory
def set_maxmemory ( self , val ) :
self . _maxmemory = val
2013-07-14 07:07:01 +04:00
maxmemory = XMLProperty ( get_maxmemory , set_maxmemory , is_int = True ,
xpath = " ./memory " )
2013-03-18 01:06:52 +04:00
def get_hugepage ( self ) :
return self . _hugepage
def set_hugepage ( self , val ) :
if val is None :
return val
self . _hugepage = bool ( val )
2013-07-14 02:56:09 +04:00
hugepage = XMLProperty ( get_hugepage , set_hugepage ,
2013-03-18 01:06:52 +04:00
xpath = " ./memoryBacking/hugepages " , is_bool = True )
# UUID for the guest
def get_uuid ( self ) :
return self . _uuid
def set_uuid ( self , val ) :
2013-04-11 18:27:02 +04:00
val = util . validate_uuid ( val )
2013-03-18 01:06:52 +04:00
self . _uuid = val
2013-07-14 02:56:09 +04:00
uuid = XMLProperty ( get_uuid , set_uuid ,
2013-03-18 01:06:52 +04:00
xpath = " ./uuid " )
def __validate_cpus ( self , val ) :
val = int ( val )
if val < 1 :
raise ValueError ( _ ( " Number of vcpus must be a positive integer. " ) )
return val
# number of vcpus for the guest
def get_vcpus ( self ) :
return self . _vcpus
def set_vcpus ( self , val ) :
val = self . __validate_cpus ( val )
self . _vcpus = val
# Don't force set maxvcpus unless already specified
if self . maxvcpus is not None and self . maxvcpus < val :
self . maxvcpus = val
def _vcpus_get_converter ( self , val ) :
# If no current VCPUs, return maxvcpus
if not val :
val = self . maxvcpus
return int ( val )
2013-07-14 02:56:09 +04:00
vcpus = XMLProperty ( get_vcpus , set_vcpus ,
2013-03-18 01:06:52 +04:00
xpath = " ./vcpu/@current " ,
get_converter = _vcpus_get_converter )
def _get_maxvcpus ( self ) :
return self . _maxvcpus
def _set_maxvcpus ( self , val ) :
val = self . __validate_cpus ( val )
self . _maxvcpus = val
2013-07-14 02:56:09 +04:00
maxvcpus = XMLProperty ( _get_maxvcpus , _set_maxvcpus ,
2013-07-14 05:49:32 +04:00
xpath = " ./vcpu " , is_int = True )
2013-03-18 01:06:52 +04:00
# set phy-cpus for the guest
def get_cpuset ( self ) :
return self . _cpuset
def set_cpuset ( self , val ) :
if val is None or val == " " :
self . _cpuset = None
return
DomainNumatune . validate_cpuset ( self . conn , val )
self . _cpuset = val
2013-07-14 02:56:09 +04:00
cpuset = XMLProperty ( get_cpuset , set_cpuset ,
2013-03-18 01:06:52 +04:00
xpath = " ./vcpu/@cpuset " )
# GAH! - installer.os_type = "hvm" or "xen" (aka xen paravirt)
# guest.os_type = "Solaris", "Windows", "Linux"
# FIXME: We should really rename this property to something else,
# change it throughout the codebase for readability sake, but
# maintain back compat.
def get_os_type ( self ) :
return self . _os_type
def set_os_type ( self , val ) :
if type ( val ) is not str :
raise ValueError ( _ ( " OS type must be a string. " ) )
val = val . lower ( )
if val in self . _OS_TYPES :
if self . _os_type != val :
# Invalidate variant, since it may not apply to the new os type
self . _os_type = val
self . _os_variant = None
else :
raise ValueError ( _ ( " OS type ' %s ' does not exist in our "
" dictionary " ) % val )
os_type = property ( get_os_type , set_os_type )
def get_os_variant ( self ) :
return self . _os_variant
def set_os_variant ( self , val ) :
if type ( val ) is not str :
raise ValueError ( _ ( " OS variant must be a string. " ) )
val = val . lower ( )
if self . os_type :
if val in self . _OS_TYPES [ self . os_type ] [ " variants " ] :
self . _os_variant = val
else :
raise ValueError ( _ ( " OS variant ' %(var)s ' does not exist in "
" our dictionary for OS type ' %(ty)s ' " ) %
{ ' var ' : val , ' ty ' : self . _os_type } )
else :
found = False
for ostype in self . list_os_types ( ) :
if ( val in self . _OS_TYPES [ ostype ] [ " variants " ] and
not self . _OS_TYPES [ ostype ] [ " variants " ] [ val ] . get ( " skip " ) ) :
logging . debug ( " Setting os type to ' %s ' for variant ' %s ' " ,
ostype , val )
self . os_type = ostype
self . _os_variant = val
found = True
if not found :
raise ValueError ( _ ( " Unknown OS variant ' %s ' " % val ) )
os_variant = property ( get_os_variant , set_os_variant )
def set_os_autodetect ( self , val ) :
self . _os_autodetect = bool ( val )
def get_os_autodetect ( self ) :
return self . _os_autodetect
os_autodetect = property ( get_os_autodetect , set_os_autodetect )
# Get the current variants 'distro' tag: 'rhel', 'fedora', etc.
def get_os_distro ( self ) :
return self . _lookup_osdict_key ( " distro " )
os_distro = property ( get_os_distro )
def get_autostart ( self ) :
return self . _autostart
def set_autostart ( self , val ) :
self . _autostart = bool ( val )
autostart = property ( get_autostart , set_autostart ,
doc = " Have domain autostart when the host boots. " )
def _get_description ( self ) :
return self . _description
def _set_description ( self , val ) :
self . _description = val
2013-07-14 02:56:09 +04:00
description = XMLProperty ( _get_description , _set_description ,
2013-03-18 01:06:52 +04:00
xpath = " ./description " )
def _get_emulator ( self ) :
return self . _emulator
def _set_emulator ( self , val ) :
self . _emulator = val
2013-07-14 02:56:09 +04:00
emulator = XMLProperty ( _get_emulator , _set_emulator ,
2013-03-18 01:06:52 +04:00
xpath = " ./devices/emulator " )
def _get_replace ( self ) :
return self . _replace
def _set_replace ( self , val ) :
self . _replace = bool ( val )
replace = property ( _get_replace , _set_replace ,
doc = _ ( " Whether we should overwrite an existing guest "
" with the same name. " ) )
########################################
# Device Add/Remove Public API methods #
########################################
def _dev_build_list ( self , devtype , devlist = None ) :
2013-07-17 21:06:01 +04:00
if devlist is None :
2013-03-18 01:06:52 +04:00
devlist = self . _devices
newlist = [ ]
for i in devlist :
2013-07-14 04:22:19 +04:00
if devtype == " all " or i . virtual_device_type == devtype :
2013-03-18 01:06:52 +04:00
newlist . append ( i )
return newlist
2013-07-09 16:53:22 +04:00
def add_device ( self , dev , set_defaults = False ) :
2013-03-18 01:06:52 +04:00
"""
Add the passed device to the guest ' s device list.
@param dev : VirtualDevice instance to attach to guest
2013-07-09 16:53:22 +04:00
@param set_defaults : Whether to set defaults for the device
2013-03-18 01:06:52 +04:00
"""
if not isinstance ( dev , VirtualDevice ) :
raise ValueError ( _ ( " Must pass a VirtualDevice instance. " ) )
if self . _is_parse ( ) :
xml = dev . get_xml_config ( )
node = libxml2 . parseDoc ( xml ) . children
dev . set_xml_node ( node )
self . _add_child_node ( " ./devices " , node )
2013-07-16 23:59:27 +04:00
self . _track_device ( dev )
2013-07-09 16:53:22 +04:00
if set_defaults :
def list_one_dev ( devtype ) :
if dev . virtual_device_type == devtype :
return [ dev ] [ : ]
else :
return [ ]
2013-07-17 21:06:01 +04:00
origdev = self . _devices
try :
self . _devices = [ dev ]
self . _set_defaults ( self . features )
except :
self . _devices = origdev
2013-07-16 23:59:27 +04:00
def _track_device ( self , dev ) :
2013-04-12 17:39:34 +04:00
self . _devices . append ( dev )
2013-03-18 01:06:52 +04:00
def get_devices ( self , devtype ) :
"""
Return a list of devices of type ' devtype ' that will installed on
the guest .
@param devtype : Device type to search for ( one of
VirtualDevice . virtual_device_types )
"""
2013-04-12 17:39:34 +04:00
devlist = self . _dev_build_list ( devtype )
2013-03-18 01:06:52 +04:00
devlist . extend ( self . _install_devices )
return self . _dev_build_list ( devtype , devlist )
def get_all_devices ( self ) :
"""
Return a list of all devices being installed with the guest
"""
retlist = [ ]
for devtype in VirtualDevice . virtual_device_types :
retlist . extend ( self . get_devices ( devtype ) )
return retlist
def remove_device ( self , dev ) :
"""
Remove the passed device from the guest ' s device list
@param dev : VirtualDevice instance
"""
found = False
2013-04-12 17:39:34 +04:00
for devlist in [ self . _devices , self . _install_devices ] :
2013-03-18 01:06:52 +04:00
if found :
break
if dev in devlist :
devlist . remove ( dev )
found = True
break
if not found :
raise ValueError ( _ ( " Did not find device %s " ) % str ( dev ) )
if self . _is_parse ( ) :
xpath = dev . get_xml_node_path ( )
if xpath :
self . _remove_child_xpath ( xpath )
2013-07-17 21:06:01 +04:00
bootloader = XMLProperty ( xpath = " ./bootloader " )
type = XMLProperty ( xpath = " ./@type " , default_cb = lambda s : " xen " )
def _cleanup_xml ( self , xml ) :
if not xml . endswith ( " \n " ) :
xml + = " \n "
return xml
2013-03-18 01:06:52 +04:00
################################
# Private xml building methods #
################################
def _parsexml ( self , xml , node ) :
2013-07-14 02:56:09 +04:00
XMLBuilder . _parsexml ( self , xml , node )
2013-03-18 01:06:52 +04:00
device_mappings = {
" disk " : virtinst . VirtualDisk ,
" interface " : virtinst . VirtualNetworkInterface ,
" sound " : virtinst . VirtualAudio ,
" hostdev " : virtinst . VirtualHostDevice ,
" input " : virtinst . VirtualInputDevice ,
2013-07-16 17:14:37 +04:00
" serial " : virtinst . VirtualSerialDevice ,
" parallel " : virtinst . VirtualParallelDevice ,
" console " : virtinst . VirtualConsoleDevice ,
" channel " : virtinst . VirtualChannelDevice ,
2013-03-18 01:06:52 +04:00
" graphics " : virtinst . VirtualGraphics ,
" video " : virtinst . VirtualVideoDevice ,
" watchdog " : virtinst . VirtualWatchdog ,
" controller " : virtinst . VirtualController ,
" filesystem " : virtinst . VirtualFilesystem ,
" smartcard " : virtinst . VirtualSmartCardDevice ,
" redirdev " : virtinst . VirtualRedirDevice ,
" memballoon " : virtinst . VirtualMemballoon ,
2013-06-26 05:45:06 +04:00
" tpm " : virtinst . VirtualTPMDevice ,
2013-04-13 22:34:52 +04:00
}
2013-03-18 01:06:52 +04:00
# Hand off all child element parsing to relevant classes
for node in self . _xml_node . children :
if node . name != " devices " :
continue
2013-04-12 00:32:00 +04:00
children = [ x for x in node . children if
( x . name in device_mappings and
x . parent == node ) ]
2013-03-18 01:06:52 +04:00
for devnode in children :
2013-07-10 16:05:08 +04:00
devnode . virtinst_root_doc = self . _xml_root_doc
2013-03-18 01:06:52 +04:00
objclass = device_mappings . get ( devnode . name )
2013-07-16 17:14:37 +04:00
dev = objclass ( self . conn , parsexmlnode = devnode )
2013-07-16 23:59:27 +04:00
self . _track_device ( dev )
2013-03-18 01:06:52 +04:00
2013-07-10 16:05:08 +04:00
self . _xml_node . virtinst_root_doc = self . _xml_root_doc
2013-07-17 15:53:47 +04:00
self . os = OSXML ( self . conn , parsexmlnode = self . _xml_node )
self . features = DomainFeatures ( self . conn ,
parsexmlnode = self . _xml_node )
self . clock = Clock ( self . conn , parsexmlnode = self . _xml_node )
self . seclabel = Seclabel ( self . conn , parsexmlnode = self . _xml_node )
self . cpu = CPU ( self . conn , parsexmlnode = self . _xml_node )
self . numatune = DomainNumatune ( self . conn ,
parsexmlnode = self . _xml_node )
2013-03-18 01:06:52 +04:00
2013-07-16 23:59:27 +04:00
def add_default_input_device ( self ) :
2013-07-17 15:53:47 +04:00
if self . os . is_container ( ) :
2013-07-16 23:59:27 +04:00
return
self . add_device ( VirtualInputDevice ( self . conn ) )
2013-03-18 01:06:52 +04:00
2013-07-16 23:59:27 +04:00
def add_default_console_device ( self ) :
2013-07-17 15:53:47 +04:00
if self . os . is_xenpv ( ) :
2013-07-16 23:59:27 +04:00
return
2013-07-16 17:14:37 +04:00
dev = virtinst . VirtualConsoleDevice ( self . conn )
dev . type = dev . TYPE_PTY
2013-07-16 23:59:27 +04:00
self . add_device ( dev )
2013-03-18 01:06:52 +04:00
def _get_device_xml ( self , devs , install = True ) :
def do_remove_media ( d ) :
# Keep cdrom around, but with no media attached,
# But only if we are a distro that doesn't have a multi
# stage install (aka not Windows)
return ( d . virtual_device_type == VirtualDevice . VIRTUAL_DEV_DISK and
d . device == VirtualDisk . DEVICE_CDROM
and d . transient
and not install and
not self . get_continue_inst ( ) )
def do_skip_disk ( d ) :
# Skip transient labeled non-media disks
return ( d . virtual_device_type == VirtualDevice . VIRTUAL_DEV_DISK and
d . device == VirtualDisk . DEVICE_DISK
and d . transient
and not install )
# Wrapper for building disk XML, handling transient CDROMs
def get_dev_xml ( dev ) :
origpath = None
try :
if do_skip_disk ( dev ) :
return " "
if do_remove_media ( dev ) :
origpath = dev . path
dev . path = None
return dev . get_xml_config ( )
finally :
if origpath :
dev . path = origpath
def get_vscsi_ctrl_xml ( ) :
2013-07-16 05:52:18 +04:00
ctrl = virtinst . VirtualController ( self . conn )
ctrl . type = " scsi "
2013-07-17 00:05:22 +04:00
ctrl . address . set_addrstr ( " spapr-vio " )
2013-03-18 01:06:52 +04:00
return ctrl . get_xml_config ( )
xml = self . _get_emulator_xml ( )
# Build XML
for dev in devs :
2013-04-11 18:27:02 +04:00
xml = util . xml_append ( xml , get_dev_xml ( dev ) )
2013-03-18 01:06:52 +04:00
if ( dev . address . type == " spapr-vio " and
2013-04-11 03:48:07 +04:00
dev . virtual_device_type == VirtualDevice . VIRTUAL_DEV_DISK ) :
2013-04-11 18:27:02 +04:00
xml = util . xml_append ( xml , get_vscsi_ctrl_xml ( ) )
2013-03-18 01:06:52 +04:00
return xml
def _get_emulator_xml ( self ) :
emulator = self . emulator
2013-07-17 15:53:47 +04:00
if self . os . is_xenpv ( ) :
2013-03-18 01:06:52 +04:00
return " "
if ( not self . emulator and
2013-07-17 15:53:47 +04:00
self . os . is_hvm ( ) and
2013-03-18 01:06:52 +04:00
self . type == " xen " ) :
2013-07-06 22:12:13 +04:00
if self . conn . caps . host . arch in ( " x86_64 " ) :
2013-03-18 01:06:52 +04:00
emulator = " /usr/lib64/xen/bin/qemu-dm "
else :
emulator = " /usr/lib/xen/bin/qemu-dm "
emu_xml = " "
if emulator is not None :
emu_xml = " <emulator> %s </emulator> " % emulator
return emu_xml
def _get_features_xml ( self , features ) :
"""
Return features ( pae , acpi , apic ) xml
"""
2013-07-17 15:53:47 +04:00
if self . os . is_container ( ) :
2013-03-18 01:06:52 +04:00
return " "
return features . get_xml_config ( )
def _get_cpu_xml ( self ) :
"""
Return < cpu > XML
"""
self . cpu . set_topology_defaults ( self . vcpus )
return self . cpu . get_xml_config ( )
def _get_clock_xml ( self ) :
"""
Return < clock / > xml
"""
return self . clock . get_xml_config ( )
def _get_seclabel_xml ( self ) :
"""
Return < seclabel > XML
"""
xml = " "
if self . seclabel :
xml = self . seclabel . get_xml_config ( )
return xml
2013-07-17 21:06:01 +04:00
def _get_default_init ( self ) :
for fs in self . get_devices ( " filesystem " ) :
if fs . target == " / " :
return " /sbin/init "
return " /bin/sh "
2013-03-18 01:06:52 +04:00
def _get_osblob ( self , install ) :
"""
Return os , features , and clock xml ( Implemented in subclass )
"""
2013-07-17 15:53:47 +04:00
oscopy = self . os . copy ( )
self . installer . alter_bootconfig ( self , install , oscopy )
2013-07-17 21:06:01 +04:00
if oscopy . is_container ( ) and not oscopy . init :
oscopy . init = self . _get_default_init ( )
if not oscopy . loader and oscopy . is_hvm ( ) and self . type == " xen " :
oscopy . loader = " /usr/lib/xen/boot/hvmloader "
if oscopy . os_type == " xen " and self . type == " xen " :
# Use older libvirt 'linux' value for back compat
oscopy . os_type = " linux "
if oscopy . kernel or oscopy . init :
oscopy . bootorder = [ ]
return oscopy . get_xml_config ( ) or None
def _get_osblob_helper ( self , guest , isinstall ,
bootconfig , endbootconfig ) :
return self . get_xml_config ( )
2013-03-18 01:06:52 +04:00
def _get_vcpu_xml ( self ) :
2013-07-06 19:20:28 +04:00
curvcpus_supported = self . conn . check_conn_support (
self . conn . SUPPORT_CONN_MAXVCPUS_XML )
2013-03-18 01:06:52 +04:00
cpuset = " "
if self . cpuset is not None :
cpuset = " cpuset= ' %s ' " % self . cpuset
maxv = self . maxvcpus
curv = self . vcpus
curxml = " "
if maxv != curv and curvcpus_supported :
curxml = " current= ' %s ' " % curv
else :
maxv = curv
return " <vcpu %s %s > %s </vcpu> " % ( cpuset , curxml , maxv )
############################
# Install Helper functions #
############################
def _prepare_install ( self , meter , dry = False ) :
self . _install_devices = [ ]
ignore = dry
# Fetch install media, prepare installer devices
2013-07-17 15:53:47 +04:00
self . installer . prepare ( self , meter ,
util . make_scratchdir ( self . conn , self . type ) )
2013-03-18 01:06:52 +04:00
# Initialize install device list
2013-07-17 00:47:08 +04:00
for dev in self . installer . install_devices :
2013-03-18 01:06:52 +04:00
self . _install_devices . append ( dev )
def _cleanup_install ( self ) :
2013-07-17 00:47:08 +04:00
self . installer . cleanup ( )
2013-03-18 01:06:52 +04:00
def _create_devices ( self , progresscb ) :
"""
Ensure that devices are setup
"""
for dev in self . get_all_devices ( ) :
2013-07-06 04:14:57 +04:00
dev . setup ( progresscb )
2013-03-18 01:06:52 +04:00
##############
# Public API #
##############
2013-07-17 21:06:01 +04:00
def _get_xml_config ( self , * args , * * kwargs ) :
# We do a shallow copy of the device list here, and set the defaults.
# This way, default changes aren't persistent, and we don't need
# to worry about when to call set_defaults
origdevs = self . _devices
try :
self . _devices = [ dev . copy ( ) for dev in self . _devices ]
return self . _do_get_xml_config ( * args , * * kwargs )
finally :
self . _devices = origdevs
def _do_get_xml_config ( self , install = True , disk_boot = False ) :
2013-03-18 01:06:52 +04:00
"""
Return the full Guest xml configuration .
@param install : Whether we want the ' OS install ' configuration or
the ' post-install ' configuration . ( Some Installers ,
like the LiveCDInstaller may not have an ' install '
config . )
@type install : C { bool }
@param disk_boot : Whether we should boot off the harddisk , regardless
of our position in the install process ( this is
used for 2 stage installs , where the second stage
boots off the disk . You probably don ' t need to touch
this . )
@type disk_boot : C { bool }
"""
2013-04-12 17:51:26 +04:00
# pylint: disable=W0221
# Argument number differs from overridden method
2013-03-18 01:06:52 +04:00
tmpfeat = self . features . copy ( )
2013-07-17 21:06:01 +04:00
self . _set_defaults ( tmpfeat )
2013-03-18 01:06:52 +04:00
2013-07-17 21:06:01 +04:00
action = install and " destroy " or " restart "
osblob_install = install and not disk_boot
2013-03-18 01:06:52 +04:00
if osblob_install and not self . installer . has_install_phase ( ) :
return None
2013-07-17 21:06:01 +04:00
if ( not install and
self . os . is_xenpv ( ) and
not self . os . kernel ) :
osblob = " <bootloader>/usr/bin/pygrub</bootloader> "
else :
osblob = self . _get_osblob ( osblob_install )
2013-03-18 01:06:52 +04:00
desc_xml = " "
if self . description is not None :
desc = str ( self . description )
desc_xml = ( " <description> %s </description> " %
2013-04-11 18:27:02 +04:00
util . xml_escape ( desc ) )
2013-03-18 01:06:52 +04:00
xml = " "
2013-04-11 18:27:02 +04:00
add = lambda x : util . xml_append ( xml , x )
2013-03-18 01:06:52 +04:00
xml = add ( " <domain type= ' %s ' > " % self . type )
xml = add ( " <name> %s </name> " % self . name )
xml = add ( " <uuid> %s </uuid> " % self . uuid )
xml = add ( desc_xml )
2013-07-14 07:07:01 +04:00
xml = add ( " <memory> %s </memory> " % self . maxmemory )
xml = add ( " <currentMemory> %s </currentMemory> " % self . memory )
2013-03-18 01:06:52 +04:00
# <blkiotune>
# <memtune>
if self . hugepage is True :
xml = add ( " <memoryBacking> " )
xml = add ( " <hugepages/> " )
xml = add ( " </memoryBacking> " )
xml = add ( self . _get_vcpu_xml ( ) )
# <cputune>
xml = add ( self . numatune . get_xml_config ( ) )
# <sysinfo>
2013-07-17 21:06:01 +04:00
xml = add ( osblob )
2013-03-18 01:06:52 +04:00
xml = add ( self . _get_features_xml ( tmpfeat ) )
xml = add ( self . _get_cpu_xml ( ) )
xml = add ( self . _get_clock_xml ( ) )
xml = add ( " <on_poweroff>destroy</on_poweroff> " )
xml = add ( " <on_reboot> %s </on_reboot> " % action )
xml = add ( " <on_crash> %s </on_crash> " % action )
xml = add ( " <devices> " )
2013-07-17 21:06:01 +04:00
xml = add ( self . _get_device_xml ( self . get_all_devices ( ) , install ) )
2013-03-18 01:06:52 +04:00
xml = add ( " </devices> " )
xml = add ( self . _get_seclabel_xml ( ) )
xml = add ( " </domain> \n " )
2013-07-12 22:19:54 +04:00
def cb ( doc , ctx ) :
ignore = ctx
return doc . serialize ( )
xml = util . xml_parse_wrapper ( xml , cb )
return " \n " . join ( xml . splitlines ( ) [ 1 : ] ) + " \n "
2013-03-18 01:06:52 +04:00
def get_continue_inst ( self ) :
"""
Return True if this guest requires a call to ' continue_install ' ,
which means the OS requires a 2 stage install ( windows )
"""
# If we are doing an 'import' or 'liveCD' install, there is
# no true install process, so continue install has no meaning
if not self . installer . has_install_phase ( ) :
return False
return self . _lookup_osdict_key ( " continue " )
def connect_console ( self , consolecb , wait = True ) :
"""
Launched the passed console callback for the already defined
domain . If domain isn ' t running, return an error.
"""
( self . domain ,
self . _consolechild ) = self . _wait_and_connect_console ( consolecb )
# If we connected the console, wait for it to finish
self . _waitpid_console ( self . _consolechild , wait )
def terminate_console ( self ) :
"""
Kill guest console if it is open ( and actually exists ) , otherwise
do nothing
"""
if self . _consolechild :
try :
os . kill ( self . _consolechild , signal . SIGKILL )
except :
pass
def domain_is_shutdown ( self ) :
"""
Return True if the created domain object is shutdown
"""
dom = self . domain
if not dom :
return False
dominfo = dom . info ( )
state = dominfo [ 0 ]
cpu_time = dominfo [ 4 ]
if state == libvirt . VIR_DOMAIN_SHUTOFF :
return True
# If 'wait' was specified, the dom object we have was looked up
# before initially shutting down, which seems to bogus up the
# info data (all 0's). So, if it is bogus, assume the domain is
# shutdown. We will catch the error later.
return state == libvirt . VIR_DOMAIN_NOSTATE and cpu_time == 0
def domain_is_crashed ( self ) :
"""
Return True if the created domain object is in a crashed state
"""
if not self . domain :
return False
dominfo = self . domain . info ( )
state = dominfo [ 0 ]
return state == libvirt . VIR_DOMAIN_CRASHED
##########################
# Actual install methods #
##########################
def remove_original_vm ( self , force = None ) :
"""
Remove the existing VM with the same name if requested , or error
if there is a collision .
"""
2013-04-13 22:34:52 +04:00
if force is None :
2013-03-18 01:06:52 +04:00
force = self . replace
vm = None
try :
vm = self . conn . lookupByName ( self . name )
except libvirt . libvirtError :
pass
if vm is None :
return
if not force :
raise RuntimeError ( _ ( " Domain named %s already exists! " ) %
self . name )
try :
logging . debug ( " Explicitly replacing guest ' %s ' " , self . name )
if vm . ID ( ) != - 1 :
logging . info ( " Destroying guest ' %s ' " , self . name )
vm . destroy ( )
logging . info ( " Undefining guest ' %s ' " , self . name )
vm . undefine ( )
except libvirt . libvirtError , e :
raise RuntimeError ( _ ( " Could not remove old vm ' %s ' : %s " ) %
( self . name , str ( e ) ) )
def start_install ( self , consolecb = None , meter = None , removeOld = None ,
wait = True , dry = False , return_xml = False , noboot = False ) :
"""
Begin the guest install ( stage1 ) .
@param return_xml : Don ' t create the guest, just return generated XML
"""
2013-07-14 03:43:52 +04:00
if self . domain is not None :
raise RuntimeError ( _ ( " Domain has already been started! " ) )
2013-03-18 01:06:52 +04:00
2013-07-14 03:43:52 +04:00
is_initial = True
2013-03-18 01:06:52 +04:00
self . _consolechild = None
self . _prepare_install ( meter , dry )
try :
# Create devices if required (disk images, etc.)
if not dry :
self . _create_devices ( meter )
start_xml , final_xml = self . _build_xml ( is_initial )
if return_xml :
return ( start_xml , final_xml )
if dry :
return
# Remove existing VM if requested
self . remove_original_vm ( removeOld )
self . domain = self . _create_guest ( consolecb , meter , wait ,
start_xml , final_xml , is_initial ,
noboot )
# Set domain autostart flag if requested
self . _flag_autostart ( )
return self . domain
finally :
self . _cleanup_install ( )
def continue_install ( self , consolecb = None , meter = None , wait = True ,
dry = False , return_xml = False ) :
"""
Continue with stage 2 of a guest install . Only required for
guests which have the ' continue ' flag set ( accessed via
get_continue_inst )
"""
is_initial = False
start_xml , final_xml = self . _build_xml ( is_initial )
if return_xml :
return ( start_xml , final_xml )
if dry :
return
return self . _create_guest ( consolecb , meter , wait ,
start_xml , final_xml , is_initial , False )
def _build_meter ( self , meter , is_initial ) :
if is_initial :
meter_label = _ ( " Creating domain... " )
else :
meter_label = _ ( " Starting domain... " )
2013-04-13 22:34:52 +04:00
if meter is None :
2013-03-18 01:06:52 +04:00
meter = progress . BaseMeter ( )
meter . start ( size = None , text = meter_label )
return meter
def _build_xml ( self , is_initial ) :
log_label = is_initial and " install " or " continue "
disk_boot = not is_initial
start_xml = self . get_xml_config ( install = True , disk_boot = disk_boot )
final_xml = self . get_xml_config ( install = False )
logging . debug ( " Generated %s XML: %s " ,
log_label ,
( start_xml and ( " \n " + start_xml ) or " None required " ) )
logging . debug ( " Generated boot XML: \n %s " , final_xml )
return start_xml , final_xml
def _create_guest ( self , consolecb , meter , wait ,
start_xml , final_xml , is_initial , noboot ) :
"""
Actually do the XML logging , guest defining / creating , console
launching and waiting
@param is_initial : If running initial guest creation , else we
are continuing the install
@param noboot : Don ' t boot guest if no install phase
"""
meter = self . _build_meter ( meter , is_initial )
doboot = not noboot or self . installer . has_install_phase ( )
if not doboot :
consolecb = None
if is_initial and doboot :
dom = self . conn . createLinux ( start_xml or final_xml , 0 )
else :
dom = self . conn . defineXML ( start_xml or final_xml )
if doboot :
dom . create ( )
self . domain = dom
meter . end ( 0 )
if doboot :
logging . debug ( " Started guest, connecting to console if requested " )
( self . domain ,
self . _consolechild ) = self . _wait_and_connect_console ( consolecb )
self . domain = self . conn . defineXML ( final_xml )
if is_initial :
try :
logging . debug ( " XML fetched from libvirt object: \n %s " ,
dom . XMLDesc ( 0 ) )
except Exception , e :
logging . debug ( " Error fetching XML from libvirt object: %s " , e )
# if we connected the console, wait for it to finish
self . _waitpid_console ( self . _consolechild , wait )
return self . conn . lookupByName ( self . name )
def _wait_and_connect_console ( self , consolecb ) :
"""
Wait for domain to appear and be running , then connect to
the console if necessary
"""
child = None
dom = _wait_for_domain ( self . conn , self . name )
if dom is None :
raise RuntimeError ( _ ( " Domain has not existed. You should be "
" able to find more information in the logs " ) )
elif dom . ID ( ) == - 1 :
raise RuntimeError ( _ ( " Domain has not run yet. You should be "
" able to find more information in the logs " ) )
if consolecb :
child = consolecb ( dom )
return dom , child
def _waitpid_console ( self , console_child , do_wait ) :
"""
Wait for console to close if it was launched
"""
if not console_child or not do_wait :
return
try :
os . waitpid ( console_child , 0 )
except OSError , ( err_no , msg ) :
logging . debug ( " waitpid: %s : %s " , err_no , msg )
# ensure there's time for the domain to finish destroying if the
# install has finished or the guest crashed
time . sleep ( 1 )
def _flag_autostart ( self ) :
"""
Set the autostart flag for self . domain if the user requested it
"""
if not self . autostart :
return
try :
self . domain . setAutostart ( True )
except libvirt . libvirtError , e :
2013-07-06 19:20:28 +04:00
if util . is_error_nosupport ( e ) :
2013-03-18 01:06:52 +04:00
logging . warn ( " Could not set autostart flag: libvirt "
" connection does not support autostart. " )
else :
raise e
###################
# Device defaults #
###################
def set_defaults ( self ) :
"""
Public function to set guest defaults . Things like preferred
disk bus ( unless one is specified ) . These changes are persistent .
The install process will call a non - persistent version , so calling
this manually isn ' t required.
"""
2013-07-17 21:06:01 +04:00
self . _set_defaults ( self . features )
2013-03-18 01:06:52 +04:00
2013-07-17 21:06:01 +04:00
def _set_hvm_defaults ( self , features ) :
2013-03-18 01:06:52 +04:00
disktype = VirtualDevice . VIRTUAL_DEV_DISK
nettype = VirtualDevice . VIRTUAL_DEV_NET
disk_bus = self . _lookup_device_param ( disktype , " bus " )
net_model = self . _lookup_device_param ( nettype , " model " )
# Only overwrite params if they weren't already specified
2013-07-17 21:06:01 +04:00
for net in self . get_devices ( nettype ) :
2013-03-18 01:06:52 +04:00
if net_model and not net . model :
net . model = net_model
2013-07-17 21:06:01 +04:00
for disk in self . get_devices ( disktype ) :
2013-03-18 01:06:52 +04:00
if ( disk_bus and not disk . bus and
disk . device == VirtualDisk . DEVICE_DISK ) :
disk . bus = disk_bus
2013-04-13 22:34:52 +04:00
if self . clock . offset is None :
2013-03-18 01:06:52 +04:00
self . clock . offset = self . _lookup_osdict_key ( " clock " )
if features [ " acpi " ] is None :
features [ " acpi " ] = self . _lookup_osdict_key ( " acpi " )
if features [ " apic " ] is None :
features [ " apic " ] = self . _lookup_osdict_key ( " apic " )
2013-07-06 22:12:13 +04:00
if features [ " pae " ] is None :
features [ " pae " ] = self . conn . caps . support_pae ( )
2013-03-18 01:06:52 +04:00
2013-07-17 15:53:47 +04:00
if ( self . os . machine is None and
2013-07-06 22:12:13 +04:00
self . conn . caps . host . arch == " ppc64 " ) :
2013-07-17 15:53:47 +04:00
self . os . machine = " pseries "
2013-03-18 01:06:52 +04:00
2013-07-17 21:06:01 +04:00
def _set_pv_defaults ( self ) :
2013-03-18 01:06:52 +04:00
# Default file backed PV guests to tap driver
2013-07-17 21:06:01 +04:00
for d in self . get_devices ( VirtualDevice . VIRTUAL_DEV_DISK ) :
2013-03-18 01:06:52 +04:00
if ( d . type == VirtualDisk . TYPE_FILE
2013-07-06 20:44:53 +04:00
and d . driver_name is None
and util . is_blktap_capable ( self . conn ) ) :
2013-03-18 01:06:52 +04:00
d . driver_name = VirtualDisk . DRIVER_TAP
2013-07-17 21:06:01 +04:00
for d in self . get_devices ( VirtualDevice . VIRTUAL_DEV_INPUT ) :
2013-07-16 00:55:49 +04:00
if d . type == d . TYPE_DEFAULT :
d . type = d . TYPE_MOUSE
if d . bus == d . BUS_DEFAULT :
d . bus = d . BUS_XEN
2013-03-18 01:06:52 +04:00
def add_usb_ich9_controllers ( self ) :
2013-07-16 05:52:18 +04:00
ctrl = VirtualController ( self . conn )
ctrl . type = " usb "
ctrl . model = " ich9-ehci1 "
2013-03-18 01:06:52 +04:00
self . add_device ( ctrl )
2013-07-16 05:52:18 +04:00
ctrl = VirtualController ( self . conn )
ctrl . type = " usb "
ctrl . model = " ich9-uhci1 "
ctrl . master_startport = 0
2013-03-18 01:06:52 +04:00
self . add_device ( ctrl )
2013-07-16 05:52:18 +04:00
ctrl = VirtualController ( self . conn )
ctrl . type = " usb "
ctrl . model = " ich9-uhci2 "
ctrl . master_startport = 2
2013-03-18 01:06:52 +04:00
self . add_device ( ctrl )
2013-07-16 05:52:18 +04:00
ctrl = VirtualController ( self . conn )
ctrl . type = " usb "
ctrl . model = " ich9-uhci3 "
ctrl . master_startport = 4
2013-03-18 01:06:52 +04:00
self . add_device ( ctrl )
2013-07-17 21:06:01 +04:00
def _set_defaults ( self , features ) :
for dev in self . get_devices ( " all " ) :
2013-07-14 04:22:19 +04:00
dev . set_defaults ( )
2013-07-17 15:53:47 +04:00
if self . os . is_hvm ( ) :
2013-07-17 21:06:01 +04:00
self . _set_hvm_defaults ( features )
2013-07-17 15:53:47 +04:00
if self . os . is_xenpv ( ) :
2013-07-17 21:06:01 +04:00
self . _set_pv_defaults ( )
2013-03-18 01:06:52 +04:00
soundtype = VirtualDevice . VIRTUAL_DEV_AUDIO
videotype = VirtualDevice . VIRTUAL_DEV_VIDEO
inputtype = VirtualDevice . VIRTUAL_DEV_INPUT
gfxtype = VirtualDevice . VIRTUAL_DEV_GRAPHICS
channeltype = VirtualDevice . VIRTUAL_DEV_CHANNEL
# Set default input values
input_type = self . _lookup_device_param ( inputtype , " type " )
input_bus = self . _lookup_device_param ( inputtype , " bus " )
2013-07-17 21:06:01 +04:00
for inp in self . get_devices ( inputtype ) :
2013-07-16 00:55:49 +04:00
if ( inp . type == inp . TYPE_DEFAULT and
inp . bus == inp . BUS_DEFAULT ) :
2013-03-18 01:06:52 +04:00
inp . type = input_type
inp . bus = input_bus
# Generate disk targets, and set preferred disk bus
used_targets = [ ]
2013-07-17 21:06:01 +04:00
for disk in self . get_devices ( VirtualDevice . VIRTUAL_DEV_DISK ) :
2013-03-18 01:06:52 +04:00
if not disk . bus :
if disk . device == disk . DEVICE_FLOPPY :
disk . bus = " fdc "
else :
2013-07-17 15:53:47 +04:00
if self . os . is_hvm ( ) :
2013-07-17 21:06:01 +04:00
if ( self . type == " kvm " and
2013-07-17 15:53:47 +04:00
self . os . machine == " pseries " ) :
2013-03-18 01:06:52 +04:00
disk . bus = " scsi "
else :
disk . bus = " ide "
2013-07-17 15:53:47 +04:00
elif self . os . is_xenpv ( ) :
2013-03-18 01:06:52 +04:00
disk . bus = " xen "
if disk . target :
used_targets . append ( disk . target )
else :
used_targets . append ( disk . generate_target ( used_targets ) )
# Set sound device model
sound_model = self . _lookup_device_param ( soundtype , " model " )
2013-07-17 21:06:01 +04:00
for sound in self . get_devices ( soundtype ) :
2013-03-18 01:06:52 +04:00
if sound . model == sound . MODEL_DEFAULT :
sound . model = sound_model
# Set video device model
# QXL device (only if we use spice) - safe even if guest is VGA only
def has_spice ( ) :
2013-07-17 21:06:01 +04:00
for gfx in self . get_devices ( gfxtype ) :
2013-03-18 01:06:52 +04:00
if gfx . type == gfx . TYPE_SPICE :
return True
if has_spice ( ) :
video_model = " qxl "
else :
2013-07-15 19:07:40 +04:00
video_model = self . _lookup_device_param ( videotype , " model " )
2013-03-18 01:06:52 +04:00
2013-07-17 21:06:01 +04:00
for video in self . get_devices ( videotype ) :
2013-07-15 19:07:40 +04:00
if video . model == video . MODEL_DEFAULT :
video . model = video_model
2013-03-18 01:06:52 +04:00
# Spice agent channel (only if we use spice)
def has_spice_agent ( ) :
2013-07-17 21:06:01 +04:00
for chn in self . get_devices ( channeltype ) :
2013-07-16 17:14:37 +04:00
if chn . type == chn . TYPE_SPICEVMC :
2013-03-18 01:06:52 +04:00
return True
if ( has_spice ( ) and
not has_spice_agent ( ) and
2013-07-06 19:20:28 +04:00
self . conn . check_conn_support (
self . conn . SUPPORT_CONN_HV_CHAR_SPICEVMC ) ) :
2013-07-16 17:14:37 +04:00
agentdev = virtinst . VirtualChannelDevice ( self . conn )
agentdev . type = agentdev . TYPE_SPICEVMC
2013-03-18 01:06:52 +04:00
self . add_device ( agentdev )
# Generate UUID
if self . uuid is None :
2013-04-11 18:27:02 +04:00
self . uuid = util . generate_uuid ( self . conn )
2013-03-18 01:06:52 +04:00
###################################
# Guest Dictionary Helper methods #
###################################
def _is_rhel6 ( self ) :
emulator = self . emulator or " "
return ( self . type in [ " qemu " , " kvm " ] and
emulator . startswith ( " /usr/libexec/qemu " ) )
def _lookup_osdict_key ( self , key ) :
"""
Using self . os_type and self . os_variant to find key in OSTYPES
@returns : dict value , or None if os_type / variant wasn ' t set
"""
return osdict . lookup_osdict_key ( self . conn , self . type ,
self . os_type , self . os_variant ,
key )
def _lookup_device_param ( self , device_key , param ) :
"""
Check the OS dictionary for the prefered device setting for passed
device type and param ( bus , model , etc . )
"""
try :
2013-04-12 16:26:21 +04:00
support . set_rhel6 ( self . _is_rhel6 ( ) )
2013-03-18 01:06:52 +04:00
return osdict . lookup_device_param ( self . conn , self . type ,
self . os_type , self . os_variant ,
device_key , param )
finally :
2013-04-12 16:26:21 +04:00
support . set_rhel6 ( False )
2013-03-18 01:06:52 +04:00
def _wait_for_domain ( conn , name ) :
# sleep in .25 second increments until either a) we get running
# domain ID or b) it's been 5 seconds. this is so that
# we can try to gracefully handle domain restarting failures
dom = None
2013-04-13 22:34:52 +04:00
for ignore in range ( 1 , int ( 5 / .25 ) ) : # 5 seconds, .25 second sleeps
2013-03-18 01:06:52 +04:00
try :
dom = conn . lookupByName ( name )
if dom and dom . ID ( ) != - 1 :
break
except libvirt . libvirtError , e :
logging . debug ( " No guest running yet: " + str ( e ) )
dom = None
time . sleep ( 0.25 )
return dom