2015-02-24 15:21:24 +03:00
# Copyright (C) 2006, 2013, 2014, 2015 Red Hat, Inc.
2006-06-28 23:50:17 +04:00
# Copyright (C) 2006 Daniel P. Berrange <berrange@redhat.com>
#
2018-04-04 16:35:41 +03:00
# This work is licensed under the GNU GPLv2 or later.
2018-03-20 22:00:02 +03:00
# See the COPYING file in the top-level directory.
2006-06-14 18:59:40 +04:00
2010-12-10 19:47:07 +03:00
import os
2015-04-02 02:10:16 +03:00
import threading
2011-04-29 01:11:04 +04:00
import time
2012-01-28 03:31:21 +04:00
import traceback
2010-01-06 21:09:33 +03:00
import libvirt
2016-06-07 18:33:21 +03:00
2008-02-22 19:48:29 +03:00
import virtinst
2019-06-17 04:12:39 +03:00
from virtinst import log
2013-07-07 21:53:37 +04:00
from virtinst import pollhelpers
2006-06-14 18:59:40 +04:00
2019-06-17 05:19:17 +03:00
from . lib import connectauth
2014-09-13 00:10:45 +04:00
from . baseclass import vmmGObject
2019-06-17 05:19:17 +03:00
from . lib . libvirtenummap import LibvirtEnumMap
2019-06-17 04:40:46 +03:00
from . object . domain import vmmDomain
from . object . network import vmmNetwork
from . object . nodedev import vmmNodeDevice
from . object . storagepool import vmmStoragePool
2019-06-17 05:19:17 +03:00
from . lib . statsmanager import vmmStatsManager
2006-06-14 18:59:40 +04:00
2012-01-28 03:31:21 +04:00
2015-04-10 17:33:04 +03:00
class _ObjectList ( vmmGObject ) :
"""
Class that wraps our internal list of libvirt objects
"""
2018-03-12 22:48:09 +03:00
# pylint: disable=not-context-manager
# pylint doesn't know that lock() has 'with' support
2017-09-08 10:36:58 +03:00
BLACKLIST_COUNT = 3
2015-04-10 17:33:04 +03:00
def __init__ ( self ) :
vmmGObject . __init__ ( self )
self . _objects = [ ]
2017-09-08 10:36:58 +03:00
self . _blacklist = { }
2015-04-10 17:33:04 +03:00
self . _lock = threading . Lock ( )
def _cleanup ( self ) :
2018-03-12 22:48:09 +03:00
self . _objects = [ ]
2015-04-10 17:33:04 +03:00
2015-09-17 22:48:42 +03:00
def _blacklist_key ( self , obj ) :
2020-09-01 19:35:26 +03:00
return str ( obj . __class__ ) + obj . get_name ( )
2015-09-17 22:48:42 +03:00
def add_blacklist ( self , obj ) :
"""
Add an object to the blacklist . Basically a list of objects we
choose not to poll , because they threw an error at init time
: param obj : vmmLibvirtObject to blacklist
2017-09-08 10:36:58 +03:00
: returns : number of added object to list
2015-09-17 22:48:42 +03:00
"""
2017-09-08 10:36:58 +03:00
key = self . _blacklist_key ( obj )
2020-09-02 18:22:02 +03:00
count = self . _blacklist . get ( key , 0 )
self . _blacklist [ key ] = count + 1
2017-09-08 10:36:58 +03:00
return self . _blacklist [ key ]
def remove_blacklist ( self , obj ) :
"""
: param obj : vmmLibvirtObject to remove from blacklist
: returns : True if object was blacklisted or False otherwise .
"""
2020-09-02 18:22:02 +03:00
key = self . _blacklist_key ( obj )
return bool ( self . _blacklist . pop ( key , 0 ) )
2015-09-17 22:48:42 +03:00
def in_blacklist ( self , obj ) :
"""
2017-09-08 10:36:58 +03:00
If an object is in list only once don ' t consider it blacklisted,
give it one more chance .
2015-09-17 22:48:42 +03:00
: param obj : vmmLibvirtObject to check
2017-09-08 10:36:58 +03:00
: returns : True if object is blacklisted
2015-09-17 22:48:42 +03:00
"""
2020-09-02 18:22:02 +03:00
key = self . _blacklist_key ( obj )
return self . _blacklist . get ( key , 0 ) > = _ObjectList . BLACKLIST_COUNT
2015-09-17 22:48:42 +03:00
2015-04-10 17:33:04 +03:00
def remove ( self , obj ) :
"""
Remove an object from the list .
: param obj : vmmLibvirtObject to remove
: returns : True if object removed , False if object was not found
"""
2018-03-12 22:48:09 +03:00
with self . _lock :
2015-04-10 17:33:04 +03:00
# Identity check is sufficient here, since we should never be
# asked to remove an object that wasn't at one point in the list.
if obj not in self . _objects :
2017-09-08 10:36:58 +03:00
return self . remove_blacklist ( obj )
2015-04-10 17:33:04 +03:00
self . _objects . remove ( obj )
return True
def add ( self , obj ) :
"""
Add an object to the list .
: param obj : vmmLibvirtObject to add
: returns : True if object added , False if object already in the list
"""
2018-03-12 22:48:09 +03:00
with self . _lock :
2015-04-10 17:33:04 +03:00
# We don't look up based on identity here, to prevent tick()
# races from adding the same domain twice
#
# We don't use lookup_object here since we need to hold the
# lock the whole time to prevent a 'time of check' issue
for checkobj in self . _objects :
if ( checkobj . __class__ == obj . __class__ and
2020-09-01 19:35:26 +03:00
checkobj . get_name ( ) == obj . get_name ( ) ) :
2015-04-10 17:33:04 +03:00
return False
if obj in self . _objects :
return False
self . _objects . append ( obj )
return True
def get_objects_for_class ( self , classobj ) :
"""
Return all objects over the passed vmmLibvirtObject class
"""
2018-03-12 22:48:09 +03:00
with self . _lock :
2015-04-10 17:33:04 +03:00
return [ o for o in self . _objects if o . __class__ is classobj ]
2020-09-01 19:35:26 +03:00
def lookup_object ( self , classobj , name ) :
2015-04-10 17:33:04 +03:00
"""
2020-09-01 19:35:26 +03:00
Lookup an object with the passed classobj + name
2015-04-10 17:33:04 +03:00
"""
# Doesn't require locking, since get_objects_for_class covers us
for obj in self . get_objects_for_class ( classobj ) :
2020-09-01 19:35:26 +03:00
if obj . get_name ( ) == name :
2015-04-10 17:33:04 +03:00
return obj
return None
2018-03-12 22:48:09 +03:00
def all_objects ( self ) :
with self . _lock :
return self . _objects [ : ]
2015-04-10 17:33:04 +03:00
2010-12-09 20:37:48 +03:00
class vmmConnection ( vmmGObject ) :
2012-05-14 17:24:56 +04:00
__gsignals__ = {
2020-09-01 18:54:10 +03:00
" vm-added " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" vm-removed " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" net-added " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" net-removed " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" pool-added " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" pool-removed " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" nodedev-added " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
" nodedev-removed " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
2018-03-15 15:10:09 +03:00
" resources-sampled " : ( vmmGObject . RUN_FIRST , None , [ ] ) ,
" state-changed " : ( vmmGObject . RUN_FIRST , None , [ ] ) ,
2018-03-16 04:22:14 +03:00
" open-completed " : ( vmmGObject . RUN_FIRST , None , [ object ] ) ,
2012-05-14 17:24:56 +04:00
}
2006-06-14 21:52:46 +04:00
2014-09-12 02:16:21 +04:00
( _STATE_DISCONNECTED ,
_STATE_CONNECTING ,
_STATE_ACTIVE ) = range ( 1 , 4 )
2007-09-10 06:57:24 +04:00
2013-07-03 21:56:43 +04:00
def __init__ ( self , uri ) :
2011-05-03 21:11:33 +04:00
self . _uri = uri
2018-03-15 14:43:56 +03:00
vmmGObject . __init__ ( self )
2014-09-12 02:16:21 +04:00
self . _state = self . _STATE_DISCONNECTED
2018-03-20 19:18:35 +03:00
self . _backend = virtinst . VirtinstConnection ( self . _uri )
2014-04-16 18:32:52 +04:00
self . _closing = False
2009-11-20 01:38:43 +03:00
2018-03-16 04:22:14 +03:00
# Error strings are stored here if open() fails
self . connect_error = None
2015-04-10 21:08:25 +03:00
self . _init_object_count = None
self . _init_object_event = None
2014-02-11 21:07:13 +04:00
self . using_domain_events = False
2014-03-31 20:43:49 +04:00
self . _domain_cb_ids = [ ]
2014-02-11 22:19:15 +04:00
self . using_network_events = False
2014-03-31 20:43:49 +04:00
self . _network_cb_ids = [ ]
2016-06-15 00:52:40 +03:00
self . using_storage_pool_events = False
self . _storage_pool_cb_ids = [ ]
2016-08-11 17:10:09 +03:00
self . using_node_device_events = False
self . _node_device_cb_ids = [ ]
2010-02-26 03:35:01 +03:00
self . _xml_flags = { }
2006-06-14 18:59:40 +04:00
2015-04-10 17:33:04 +03:00
self . _objects = _ObjectList ( )
2018-10-05 17:54:25 +03:00
self . statsmanager = vmmStatsManager ( )
2015-04-10 17:33:04 +03:00
2015-05-05 01:16:09 +03:00
self . _stats = [ ]
self . _hostinfo = None
2006-06-14 18:59:40 +04:00
2015-04-11 20:39:25 +03:00
self . add_gsettings_handle (
self . _on_config_pretty_name_changed (
self . _config_pretty_name_changed_cb ) )
2013-07-08 02:54:08 +04:00
self . _init_virtconn ( )
2013-07-05 16:59:58 +04:00
2014-01-27 02:42:24 +04:00
@staticmethod
def pretty_hv ( gtype , domtype ) :
"""
Convert XML < domain type = ' foo ' > and < os > < type > bar < / type >
into a more human relevant string .
"""
gtype = gtype . lower ( )
domtype = domtype . lower ( )
label = domtype
2015-04-03 23:25:57 +03:00
if domtype == " xen " :
2014-01-27 02:42:24 +04:00
if gtype == " xen " :
label = " xen (paravirt) "
elif gtype == " hvm " :
label = " xen (fullvirt) "
elif domtype == " test " :
if gtype == " xen " :
label = " test (xen) "
elif gtype == " hvm " :
label = " test (hvm) "
2015-09-15 01:31:56 +03:00
elif domtype == " qemu " :
label = " QEMU TCG "
elif domtype == " kvm " :
label = " KVM "
2014-01-27 02:42:24 +04:00
return label
2018-03-15 14:43:56 +03:00
def __repr__ ( self ) :
2020-07-05 01:02:00 +03:00
# pylint: disable=arguments-differ
2018-03-15 14:43:56 +03:00
return " < %s uri= %s id= %s > " % (
self . __class__ . __name__ , self . get_uri ( ) , hex ( id ( self ) ) )
2009-11-26 01:07:12 +03:00
#################
# Init routines #
#################
2017-07-17 23:21:13 +03:00
def _wait_for_condition ( self , compare_cb , timeout = 3 ) :
"""
Wait for this object to emit the specified signal . Will not
block the mainloop .
"""
from gi . repository import Gtk
is_main_thread = ( threading . current_thread ( ) . name == " MainThread " )
start_time = time . time ( )
while True :
cur_time = time . time ( )
if compare_cb ( ) :
return
if ( cur_time - start_time ) > = timeout :
return
if is_main_thread :
if Gtk . events_pending ( ) :
Gtk . main_iteration_do ( False )
continue
time . sleep ( .1 )
2013-07-08 02:54:08 +04:00
def _init_virtconn ( self ) :
2018-08-31 22:20:50 +03:00
self . _backend . cb_fetch_all_domains = (
2013-09-29 04:05:13 +04:00
lambda : [ obj . get_xmlobj ( refresh_if_nec = False )
2015-04-10 17:33:04 +03:00
for obj in self . list_vms ( ) ] )
2013-09-29 04:05:13 +04:00
self . _backend . cb_fetch_all_pools = (
lambda : [ obj . get_xmlobj ( refresh_if_nec = False )
2015-04-10 17:33:04 +03:00
for obj in self . list_pools ( ) ] )
2014-09-20 19:30:24 +04:00
self . _backend . cb_fetch_all_nodedevs = (
lambda : [ obj . get_xmlobj ( refresh_if_nec = False )
2015-04-10 17:33:04 +03:00
for obj in self . list_nodedevs ( ) ] )
2014-02-27 22:16:21 +04:00
def fetch_all_vols ( ) :
ret = [ ]
2015-04-10 17:33:04 +03:00
for pool in self . list_pools ( ) :
2015-04-10 19:52:42 +03:00
for vol in pool . get_volumes ( ) :
2014-02-27 22:16:21 +04:00
try :
ret . append ( vol . get_xmlobj ( refresh_if_nec = False ) )
2020-08-20 20:34:01 +03:00
except Exception as e : # pragma: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Fetching volume XML failed: %s " , e )
2014-02-27 22:16:21 +04:00
return ret
self . _backend . cb_fetch_all_vols = fetch_all_vols
2013-09-29 04:05:13 +04:00
2017-07-19 23:11:17 +03:00
def cache_new_pool ( obj ) :
2017-07-17 23:21:13 +03:00
if not self . is_active ( ) :
return
2017-07-19 23:11:17 +03:00
name = obj . name ( )
2017-07-17 23:21:13 +03:00
self . schedule_priority_tick ( pollpool = True )
def compare_cb ( ) :
2020-09-01 19:35:26 +03:00
return bool ( self . get_pool_by_name ( name ) )
2017-07-17 23:21:13 +03:00
self . _wait_for_condition ( compare_cb )
2017-07-19 22:56:43 +03:00
self . _backend . cb_cache_new_pool = cache_new_pool
2017-05-19 15:26:49 +03:00
2013-04-16 03:34:10 +04:00
2009-11-26 01:07:12 +03:00
########################
# General data getters #
########################
2009-11-26 00:12:03 +03:00
2009-11-26 00:13:46 +03:00
def get_uri ( self ) :
2011-05-03 21:11:33 +04:00
return self . _uri
2013-07-05 16:59:58 +04:00
def get_backend ( self ) :
return self . _backend
2009-03-09 23:16:45 +03:00
2010-12-08 20:17:14 +03:00
def invalidate_caps ( self ) :
2013-07-06 22:12:13 +04:00
return self . _backend . invalidate_caps ( )
caps = property ( lambda self : getattr ( self , " _backend " ) . caps )
2009-11-26 00:13:46 +03:00
def host_memory_size ( self ) :
2018-09-26 12:00:57 +03:00
if not self . _backend . is_open ( ) or self . _hostinfo is None :
2009-11-26 00:13:46 +03:00
return 0
2015-05-05 01:16:09 +03:00
return self . _hostinfo [ 1 ] * 1024
2009-11-26 00:13:46 +03:00
def host_active_processor_count ( self ) :
2018-09-26 12:00:57 +03:00
if not self . _backend . is_open ( ) or self . _hostinfo is None :
2009-11-26 00:13:46 +03:00
return 0
2015-05-05 01:16:09 +03:00
return self . _hostinfo [ 2 ]
2009-11-26 00:13:46 +03:00
2015-04-07 20:56:48 +03:00
2009-11-26 00:13:46 +03:00
##########################
# URI + hostname helpers #
##########################
2009-11-15 23:17:03 +03:00
2015-04-16 18:21:34 +03:00
def libvirt_gethostname ( self ) :
return self . _backend . getHostname ( )
2006-11-06 18:34:54 +03:00
2013-07-06 04:36:28 +04:00
get_uri_hostname = property ( lambda s :
2015-04-11 19:08:57 +03:00
getattr ( s , " _backend " ) . get_uri_hostname )
2015-04-11 19:57:32 +03:00
get_uri_username = property ( lambda s :
getattr ( s , " _backend " ) . get_uri_username )
2015-04-11 19:08:57 +03:00
get_uri_transport = property ( lambda s :
getattr ( s , " _backend " ) . get_uri_transport )
get_uri_port = property ( lambda s : getattr ( s , " _backend " ) . get_uri_port )
2013-07-06 04:36:28 +04:00
get_driver = property ( lambda s : getattr ( s , " _backend " ) . get_uri_driver )
2020-08-22 20:22:51 +03:00
is_container_only = property (
lambda s : getattr ( s , " _backend " ) . is_container_only )
2013-07-06 04:36:28 +04:00
is_lxc = property ( lambda s : getattr ( s , " _backend " ) . is_lxc )
2017-03-13 15:01:51 +03:00
is_vz = property ( lambda s : getattr ( s , " _backend " ) . is_vz )
2013-07-06 04:36:28 +04:00
is_xen = property ( lambda s : getattr ( s , " _backend " ) . is_xen )
is_remote = property ( lambda s : getattr ( s , " _backend " ) . is_remote )
is_qemu = property ( lambda s : getattr ( s , " _backend " ) . is_qemu )
2020-02-03 03:43:06 +03:00
is_qemu_privileged = property ( lambda s : getattr ( s , " _backend " ) . is_qemu_privileged )
is_qemu_unprivileged = property ( lambda s :
getattr ( s , " _backend " ) . is_qemu_unprivileged )
2017-06-27 21:13:36 +03:00
is_test = property ( lambda s : getattr ( s , " _backend " ) . is_test )
2020-02-03 03:43:06 +03:00
is_unprivileged = property ( lambda s : getattr ( s , " _backend " ) . is_unprivileged )
2009-03-09 23:16:45 +03:00
2010-12-11 02:03:51 +03:00
2013-10-01 05:41:10 +04:00
def get_cache_dir ( self ) :
uri = self . get_uri ( ) . replace ( " / " , " _ " )
2019-06-08 00:50:58 +03:00
ret = os . path . join ( self . _backend . get_app_cache_dir ( ) , uri )
2013-10-01 05:41:10 +04:00
if not os . path . exists ( ret ) :
2017-05-05 20:05:30 +03:00
os . makedirs ( ret , 0o755 )
2013-10-01 05:41:10 +04:00
return ret
2013-10-02 23:17:15 +04:00
def get_default_storage_format ( self ) :
raw = self . config . get_default_storage_format ( raw = True )
if raw != " default " :
return raw
fmt = self . config . get_default_storage_format ( )
if fmt != " qcow2 " :
2014-02-11 03:47:58 +04:00
return fmt
2013-10-02 23:17:15 +04:00
2019-06-07 23:06:52 +03:00
if self . support . conn_default_qcow2 ( ) :
2013-10-02 23:17:15 +04:00
return fmt
return None
2010-02-11 04:26:40 +03:00
2014-02-11 03:47:58 +04:00
2013-07-06 22:12:13 +04:00
####################################
# Connection pretty print routines #
####################################
2010-02-11 04:26:40 +03:00
2015-04-11 19:57:32 +03:00
def get_pretty_desc ( self ) :
2014-07-08 01:57:50 +04:00
"""
2015-04-11 19:57:32 +03:00
Return a pretty label for use in the manager view , and various
connection lists .
2014-07-08 01:57:50 +04:00
"""
2015-04-11 20:39:25 +03:00
if self . _get_config_pretty_name ( ) :
return self . _get_config_pretty_name ( )
2009-07-29 03:21:56 +04:00
2011-06-08 03:41:02 +04:00
pretty_map = {
2017-08-05 09:39:32 +03:00
" esx " : " ESX " ,
" lxc " : " LXC " ,
" openvz " : " OpenVZ " ,
" qemu " : " QEMU/KVM " ,
2020-08-29 18:53:19 +03:00
" vbox " : " Virtualbox " ,
2017-08-05 09:39:32 +03:00
" vmware " : " VMWare " ,
" xen " : " Xen " ,
2011-06-08 03:41:02 +04:00
}
2015-04-11 19:57:32 +03:00
hv = pretty_map . get ( self . get_driver ( ) , self . get_driver ( ) )
hostname = self . get_uri_hostname ( )
path = self . get_backend ( ) . get_uri_path ( )
2014-07-08 01:57:50 +04:00
2015-04-11 19:57:32 +03:00
ret = hv
2010-12-12 07:51:20 +03:00
2020-02-03 03:43:06 +03:00
if path == " /session " :
2016-02-05 18:18:16 +03:00
ret + = " " + _ ( " User session " )
2015-09-16 02:30:09 +03:00
elif ( path and path != " /system " and os . path . basename ( path ) ) :
2015-04-11 19:57:32 +03:00
# Used by test URIs to report what XML file they are using
ret + = " %s " % os . path . basename ( path )
2009-07-20 22:47:50 +04:00
2015-04-11 19:57:32 +03:00
if hostname :
2015-05-03 02:20:18 +03:00
ret + = " : %s " % hostname
2013-07-17 01:15:51 +04:00
2015-04-11 19:57:32 +03:00
return ret
2009-07-20 22:47:50 +04:00
2009-11-26 00:13:46 +03:00
#######################
# API support helpers #
#######################
2019-06-07 23:06:52 +03:00
@property
def support ( self ) :
return self . _backend . support
2013-07-06 19:20:28 +04:00
2010-02-26 03:35:01 +03:00
def _get_flags_helper ( self , obj , key , check_func ) :
2010-12-09 19:22:35 +03:00
ignore = obj
2010-02-26 03:35:01 +03:00
flags_dict = self . _xml_flags . get ( key )
2009-11-26 00:13:46 +03:00
2012-11-08 17:15:02 +04:00
if flags_dict is None :
2010-02-26 03:35:01 +03:00
# Flags already set
inact , act = check_func ( )
flags_dict = { }
flags_dict [ " active " ] = act
flags_dict [ " inactive " ] = inact
self . _xml_flags [ key ] = flags_dict
active_flags = flags_dict [ " active " ]
inactive_flags = flags_dict [ " inactive " ]
return ( inactive_flags , active_flags )
def get_dom_flags ( self , vm ) :
key = " domain "
def check_func ( ) :
act = 0
inact = 0
2019-06-07 23:06:52 +03:00
if self . support . domain_xml_inactive ( vm ) :
2010-02-26 03:35:01 +03:00
inact = libvirt . VIR_DOMAIN_XML_INACTIVE
2020-08-20 20:34:01 +03:00
else : # pragma: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Domain XML inactive flag not supported. " )
2010-02-26 03:35:01 +03:00
2019-06-07 23:06:52 +03:00
if self . support . domain_xml_secure ( vm ) :
2010-02-26 03:35:01 +03:00
inact | = libvirt . VIR_DOMAIN_XML_SECURE
act = libvirt . VIR_DOMAIN_XML_SECURE
2020-08-20 20:34:01 +03:00
else : # pragma: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Domain XML secure flag not supported. " )
2010-02-26 03:35:01 +03:00
return inact , act
return self . _get_flags_helper ( vm , key , check_func )
2009-11-26 00:13:46 +03:00
2014-06-03 01:17:47 +04:00
def get_default_pool ( self ) :
2019-07-03 02:44:10 +03:00
poolxml = virtinst . StoragePool . lookup_default_pool ( self . get_backend ( ) )
2019-06-11 15:38:20 +03:00
if poolxml :
for p in self . list_pools ( ) :
if p . get_name ( ) == poolxml . name :
return p
2014-06-03 01:17:47 +04:00
return None
def get_vol_by_path ( self , path ) :
2015-04-10 17:33:04 +03:00
for pool in self . list_pools ( ) :
2015-04-10 19:52:42 +03:00
for vol in pool . get_volumes ( ) :
2014-12-05 04:38:55 +03:00
try :
if vol . get_target_path ( ) == path :
return vol
2020-08-20 20:34:01 +03:00
except Exception as e : # pragma: no cover
2014-12-05 04:38:55 +03:00
# Errors can happen if the volume disappeared, bug 1092739
2019-06-17 04:12:39 +03:00
log . debug ( " Error looking up volume from path= %s : %s " ,
2014-12-05 04:38:55 +03:00
path , e )
2014-06-03 01:17:47 +04:00
return None
2013-07-06 19:20:28 +04:00
2009-11-26 00:13:46 +03:00
###################################
# Connection state getter/setters #
###################################
def _change_state ( self , newstate ) :
2014-09-12 02:16:21 +04:00
if self . _state != newstate :
self . _state = newstate
2019-06-17 04:12:39 +03:00
log . debug ( " conn= %s changed to state= %s " ,
2014-09-12 02:48:04 +04:00
self . get_uri ( ) , self . get_state_text ( ) )
2009-11-26 00:13:46 +03:00
self . emit ( " state-changed " )
2014-09-12 02:16:21 +04:00
def is_active ( self ) :
return self . _state == self . _STATE_ACTIVE
def is_disconnected ( self ) :
return self . _state == self . _STATE_DISCONNECTED
def is_connecting ( self ) :
return self . _state == self . _STATE_CONNECTING
2009-11-26 00:13:46 +03:00
def get_state_text ( self ) :
2014-09-12 02:16:21 +04:00
if self . is_disconnected ( ) :
2009-11-26 00:13:46 +03:00
return _ ( " Disconnected " )
2014-09-12 02:16:21 +04:00
elif self . is_connecting ( ) :
2009-11-26 00:13:46 +03:00
return _ ( " Connecting " )
2020-08-20 20:34:01 +03:00
return _ ( " Active " )
2009-11-26 00:13:46 +03:00
2014-06-03 01:17:47 +04:00
2009-11-26 00:13:46 +03:00
#################################
# Libvirt object lookup methods #
#################################
2006-06-14 22:36:26 +04:00
2020-09-01 19:35:26 +03:00
def get_vm_by_name ( self , name ) :
return self . _objects . lookup_object ( vmmDomain , name )
2014-06-03 01:17:47 +04:00
def list_vms ( self ) :
2015-04-10 17:33:04 +03:00
return self . _objects . get_objects_for_class ( vmmDomain )
2014-06-03 01:17:47 +04:00
2020-09-01 19:35:26 +03:00
def get_net_by_name ( self , name ) :
return self . _objects . lookup_object ( vmmNetwork , name )
2014-06-03 01:17:47 +04:00
def list_nets ( self ) :
2015-04-10 17:33:04 +03:00
return self . _objects . get_objects_for_class ( vmmNetwork )
2014-06-03 01:17:47 +04:00
2020-09-01 19:35:26 +03:00
def get_pool_by_name ( self , name ) :
return self . _objects . lookup_object ( vmmStoragePool , name )
2014-06-03 01:17:47 +04:00
def list_pools ( self ) :
2015-04-10 17:33:04 +03:00
return self . _objects . get_objects_for_class ( vmmStoragePool )
2014-06-03 01:17:47 +04:00
2020-09-01 19:35:26 +03:00
def get_nodedev_by_name ( self , name ) :
return self . _objects . lookup_object ( vmmNodeDevice , name )
2015-04-10 17:33:04 +03:00
def list_nodedevs ( self ) :
return self . _objects . get_objects_for_class ( vmmNodeDevice )
2015-04-10 16:37:03 +03:00
############################
# nodedev helper functions #
############################
2019-06-05 21:45:48 +03:00
def filter_nodedevs ( self , devtype ) :
2009-03-09 23:20:23 +03:00
retdevs = [ ]
2015-04-10 17:33:04 +03:00
for dev in self . list_nodedevs ( ) :
2015-06-06 21:20:21 +03:00
try :
xmlobj = dev . get_xmlobj ( )
2020-08-20 20:34:01 +03:00
except libvirt . libvirtError as e : # pragma: no cover
2015-06-09 18:41:39 +03:00
# Libvirt nodedev XML fetching can be busted
# https://bugzilla.redhat.com/show_bug.cgi?id=1225771
if e . get_error_code ( ) != libvirt . VIR_ERR_NO_NODE_DEVICE :
2019-06-17 04:12:39 +03:00
log . debug ( " Error fetching nodedev XML " , exc_info = True )
2015-06-09 18:41:39 +03:00
continue
2015-06-06 21:20:21 +03:00
2013-10-05 22:40:38 +04:00
if devtype and xmlobj . device_type != devtype :
2009-03-09 23:20:23 +03:00
continue
2015-04-07 21:12:00 +03:00
retdevs . append ( dev )
2009-03-09 23:20:23 +03:00
return retdevs
2009-11-26 00:13:46 +03:00
###################################
# Libvirt object creation methods #
###################################
2013-09-29 20:14:00 +04:00
def define_domain ( self , xml ) :
return self . _backend . defineXML ( xml )
def define_network ( self , xml ) :
return self . _backend . networkDefineXML ( xml )
def define_pool ( self , xml ) :
return self . _backend . storagePoolDefineXML ( xml , 0 )
2009-12-11 21:45:53 +03:00
2020-09-01 19:35:26 +03:00
def rename_object ( self , obj , origxml , newxml ) :
2018-10-14 00:21:22 +03:00
if obj . is_domain ( ) :
2014-09-21 00:55:11 +04:00
define_cb = self . define_domain
2018-10-14 00:21:22 +03:00
elif obj . is_pool ( ) :
2014-09-21 00:55:11 +04:00
define_cb = self . define_pool
2018-10-14 00:21:22 +03:00
elif obj . is_network ( ) :
2014-09-21 00:55:11 +04:00
define_cb = self . define_network
else :
2020-07-18 01:58:00 +03:00
raise virtinst . xmlutil . DevError ( " rename_object "
2014-09-21 00:55:11 +04:00
" helper doesn ' t support object class %s " % obj . __class__ )
2013-09-29 20:14:00 +04:00
# Undefine the original object
2013-09-30 23:23:14 +04:00
obj . delete ( force = False )
2011-03-23 23:56:12 +03:00
newobj = None
try :
2014-09-21 00:55:11 +04:00
# Redefine new domain
newobj = define_cb ( newxml )
2017-05-05 19:47:21 +03:00
except Exception as renameerr :
2011-03-23 23:56:12 +03:00
try :
2019-06-17 04:12:39 +03:00
log . debug ( " Error defining new name %s XML " ,
2015-04-10 16:15:44 +03:00
obj . class_name ( ) , exc_info = True )
2014-09-21 00:55:11 +04:00
newobj = define_cb ( origxml )
2017-05-05 19:47:21 +03:00
except Exception as fixerr :
2019-06-17 04:12:39 +03:00
log . debug ( " Failed to redefine original %s ! " ,
2015-04-10 16:15:44 +03:00
obj . class_name ( ) , exc_info = True )
2020-07-14 10:41:49 +03:00
msg = _ ( " %(object)s rename failed. Attempting to recover also "
" failed. \n "
" \n "
" Original error: %(origerror)s \n "
" \n "
2020-07-15 15:48:56 +03:00
" Recover error: %(recovererror)s " ) % {
2020-07-14 10:41:49 +03:00
" object " : obj . class_name ( ) ,
" origerror " : str ( renameerr ) ,
" recovererror " : str ( fixerr ) ,
}
2020-07-12 00:31:40 +03:00
raise RuntimeError ( msg )
2014-09-21 00:55:11 +04:00
raise
2011-03-23 23:56:12 +03:00
finally :
if newobj :
2013-09-29 20:14:00 +04:00
# Reinsert handle into new obj
obj . change_name_backend ( newobj )
2009-11-26 00:13:46 +03:00
2014-02-11 21:07:13 +04:00
#########################
# Domain event handling #
#########################
# Our strategy here isn't the most efficient: since we need to keep the
# poll helpers around for compat with old libvirt, switching to a fully
# event driven setup is hard, so we end up doing more polling than
# necessary on most events.
2014-03-31 20:43:49 +04:00
def _domain_xml_misc_event ( self , conn , domain , * args ) :
# Just trigger a domain XML refresh for hotplug type events
ignore = conn
2018-02-27 19:56:18 +03:00
args = list ( args )
eventstr = args . pop ( - 1 )
2014-03-31 20:43:49 +04:00
2015-04-14 01:02:16 +03:00
name = domain . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " domain xmlmisc event: domain= %s event= %s args= %s " ,
2018-02-27 19:56:18 +03:00
name , eventstr , args )
2020-09-01 19:35:26 +03:00
obj = self . get_vm_by_name ( name )
2014-03-31 20:43:49 +04:00
if not obj :
return
2015-04-10 01:02:42 +03:00
2016-06-23 21:48:04 +03:00
self . idle_add ( obj . recache_from_event_loop )
2014-03-31 20:43:49 +04:00
2018-02-27 19:57:29 +03:00
def _domain_lifecycle_event ( self , conn , domain , state , reason , userdata ) :
2014-02-11 21:07:13 +04:00
ignore = conn
ignore = userdata
2015-04-14 01:02:16 +03:00
name = domain . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " domain lifecycle event: domain= %s %s " , name ,
2018-02-27 19:57:29 +03:00
LibvirtEnumMap . domain_lifecycle_str ( state , reason ) )
2020-09-01 19:35:26 +03:00
obj = self . get_vm_by_name ( name )
2014-02-11 21:07:13 +04:00
2014-02-11 22:19:15 +04:00
if obj :
2016-06-23 21:48:04 +03:00
self . idle_add ( obj . recache_from_event_loop )
2014-02-11 21:07:13 +04:00
else :
self . schedule_priority_tick ( pollvm = True , force = True )
2018-07-04 11:10:37 +03:00
def _domain_agent_lifecycle_event ( self , conn , domain , state , reason , userdata ) :
ignore = conn
ignore = userdata
name = domain . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " domain agent lifecycle event: domain= %s %s " , name ,
2018-07-04 11:10:37 +03:00
LibvirtEnumMap . domain_agent_lifecycle_str ( state , reason ) )
2020-09-01 19:35:26 +03:00
obj = self . get_vm_by_name ( name )
2018-07-04 11:10:38 +03:00
if obj :
self . idle_add ( obj . recache_from_event_loop )
else :
self . schedule_priority_tick ( pollvm = True , force = True )
2018-02-27 19:57:29 +03:00
def _network_lifecycle_event ( self , conn , network , state , reason , userdata ) :
2014-02-11 22:19:15 +04:00
ignore = conn
ignore = userdata
2015-04-14 01:02:16 +03:00
name = network . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " network lifecycle event: network= %s %s " ,
2018-02-27 19:57:29 +03:00
name , LibvirtEnumMap . network_lifecycle_str ( state , reason ) )
2020-09-01 19:35:26 +03:00
obj = self . get_net_by_name ( name )
2014-02-11 22:19:15 +04:00
if obj :
2016-06-23 21:48:04 +03:00
self . idle_add ( obj . recache_from_event_loop )
2014-02-11 22:19:15 +04:00
else :
self . schedule_priority_tick ( pollnet = True , force = True )
2014-02-11 21:07:13 +04:00
2016-06-15 00:52:40 +03:00
def _storage_pool_lifecycle_event ( self , conn , pool ,
2018-02-27 19:57:29 +03:00
state , reason , userdata ) :
2016-06-15 00:52:40 +03:00
ignore = conn
ignore = userdata
name = pool . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " storage pool lifecycle event: pool= %s %s " ,
2018-02-27 19:57:29 +03:00
name , LibvirtEnumMap . storage_lifecycle_str ( state , reason ) )
2016-06-20 23:40:33 +03:00
2020-09-01 19:35:26 +03:00
obj = self . get_pool_by_name ( name )
2016-06-15 00:52:40 +03:00
if obj :
2016-08-18 18:26:02 +03:00
self . idle_add ( obj . recache_from_event_loop )
2016-06-15 00:52:40 +03:00
else :
self . schedule_priority_tick ( pollpool = True , force = True )
2016-08-18 18:26:02 +03:00
def _storage_pool_refresh_event ( self , conn , pool , userdata ) :
ignore = conn
ignore = userdata
name = pool . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " storage pool refresh event: pool= %s " , name )
2016-08-18 18:26:02 +03:00
2020-09-01 19:35:26 +03:00
obj = self . get_pool_by_name ( name )
2016-08-18 18:26:02 +03:00
if not obj :
return
self . idle_add ( obj . refresh_pool_cache_from_event_loop )
2016-08-11 17:10:09 +03:00
def _node_device_lifecycle_event ( self , conn , dev ,
2018-02-27 19:57:29 +03:00
state , reason , userdata ) :
2016-08-11 17:10:09 +03:00
ignore = conn
ignore = userdata
name = dev . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " node device lifecycle event: nodedev= %s %s " ,
2018-02-27 19:57:29 +03:00
name , LibvirtEnumMap . nodedev_lifecycle_str ( state , reason ) )
2016-08-11 17:10:09 +03:00
self . schedule_priority_tick ( pollnodedev = True , force = True )
2016-08-17 11:32:36 +03:00
def _node_device_update_event ( self , conn , dev , userdata ) :
ignore = conn
ignore = userdata
name = dev . name ( )
2019-06-17 04:12:39 +03:00
log . debug ( " node device update event: nodedev= %s " , name )
2016-08-17 11:32:36 +03:00
2020-09-01 19:35:26 +03:00
obj = self . get_nodedev_by_name ( name )
2016-08-17 11:32:36 +03:00
if obj :
self . idle_add ( obj . recache_from_event_loop )
2014-02-11 22:19:15 +04:00
def _add_conn_events ( self ) :
2019-06-07 23:06:52 +03:00
if not self . support . conn_working_xen_events ( ) :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2015-02-25 18:25:21 +03:00
2019-06-05 20:19:44 +03:00
def _check_events_disabled ( ) :
if self . config . CLITestOptions . no_events :
raise RuntimeError ( " events disabled via cli " )
2014-02-11 21:07:13 +04:00
try :
2019-06-05 20:19:44 +03:00
_check_events_disabled ( )
2014-06-02 20:00:07 +04:00
2014-03-31 20:43:49 +04:00
self . _domain_cb_ids . append (
self . get_backend ( ) . domainEventRegisterAny (
2014-02-11 22:27:59 +04:00
None , libvirt . VIR_DOMAIN_EVENT_ID_LIFECYCLE ,
2014-03-31 20:43:49 +04:00
self . _domain_lifecycle_event , None ) )
2014-02-11 21:07:13 +04:00
self . using_domain_events = True
2019-06-17 04:12:39 +03:00
log . debug ( " Using domain events " )
2017-05-05 19:47:21 +03:00
except Exception as e :
2014-02-11 21:07:13 +04:00
self . using_domain_events = False
2019-06-17 04:12:39 +03:00
log . debug ( " Error registering domain events: %s " , e )
2014-02-11 21:07:13 +04:00
2018-07-04 11:10:35 +03:00
def _add_domain_xml_event ( eventname , eventval , cb = None ) :
2014-03-31 20:43:49 +04:00
if not self . using_domain_events :
return
2018-07-04 11:10:35 +03:00
if not cb :
cb = self . _domain_xml_misc_event
2014-03-31 20:43:49 +04:00
try :
2018-02-27 19:56:18 +03:00
eventid = getattr ( libvirt , eventname , eventval )
2014-03-31 20:43:49 +04:00
self . _domain_cb_ids . append (
self . get_backend ( ) . domainEventRegisterAny (
2018-07-04 11:10:35 +03:00
None , eventid , cb , eventname ) )
2020-08-20 20:34:01 +03:00
except Exception as e : # pragma: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Error registering %s event: %s " ,
2018-02-27 19:56:18 +03:00
eventname , e )
_add_domain_xml_event ( " VIR_DOMAIN_EVENT_ID_BALLOON_CHANGE " , 13 )
_add_domain_xml_event ( " VIR_DOMAIN_EVENT_ID_TRAY_CHANGE " , 10 )
_add_domain_xml_event ( " VIR_DOMAIN_EVENT_ID_DEVICE_REMOVED " , 15 )
_add_domain_xml_event ( " VIR_DOMAIN_EVENT_ID_DEVICE_ADDED " , 19 )
2018-07-04 11:10:37 +03:00
_add_domain_xml_event ( " VIR_DOMAIN_EVENT_ID_AGENT_LIFECYCLE " , 18 ,
self . _domain_agent_lifecycle_event )
2014-03-31 20:43:49 +04:00
2014-02-11 22:19:15 +04:00
try :
2019-06-05 20:19:44 +03:00
_check_events_disabled ( )
2014-06-02 20:00:07 +04:00
2014-03-22 19:21:19 +04:00
eventid = getattr ( libvirt , " VIR_NETWORK_EVENT_ID_LIFECYCLE " , 0 )
2014-03-31 20:43:49 +04:00
self . _network_cb_ids . append (
self . get_backend ( ) . networkEventRegisterAny (
None , eventid , self . _network_lifecycle_event , None ) )
2014-02-11 22:19:15 +04:00
self . using_network_events = True
2019-06-17 04:12:39 +03:00
log . debug ( " Using network events " )
2017-05-05 19:47:21 +03:00
except Exception as e :
2014-02-11 22:19:15 +04:00
self . using_network_events = False
2019-06-17 04:12:39 +03:00
log . debug ( " Error registering network events: %s " , e )
2014-02-11 22:19:15 +04:00
2016-06-15 00:52:40 +03:00
try :
2019-06-05 20:19:44 +03:00
_check_events_disabled ( )
2016-06-15 00:52:40 +03:00
eventid = getattr ( libvirt ,
" VIR_STORAGE_POOL_EVENT_ID_LIFECYCLE " , 0 )
2016-08-18 18:26:02 +03:00
refreshid = getattr ( libvirt ,
" VIR_STORAGE_POOL_EVENT_ID_REFRESH " , 1 )
2016-06-15 00:52:40 +03:00
self . _storage_pool_cb_ids . append (
self . get_backend ( ) . storagePoolEventRegisterAny (
None , eventid , self . _storage_pool_lifecycle_event , None ) )
2016-08-18 18:26:02 +03:00
self . _storage_pool_cb_ids . append (
self . get_backend ( ) . storagePoolEventRegisterAny (
None , refreshid , self . _storage_pool_refresh_event , None ) )
2016-06-15 00:52:40 +03:00
self . using_storage_pool_events = True
2019-06-17 04:12:39 +03:00
log . debug ( " Using storage pool events " )
2017-05-05 19:47:21 +03:00
except Exception as e :
2016-06-15 00:52:40 +03:00
self . using_storage_pool_events = False
2019-06-17 04:12:39 +03:00
log . debug ( " Error registering storage pool events: %s " , e )
2016-06-15 00:52:40 +03:00
2016-08-11 17:10:09 +03:00
try :
2019-06-05 20:19:44 +03:00
_check_events_disabled ( )
2016-08-11 17:10:09 +03:00
eventid = getattr ( libvirt , " VIR_NODE_DEVICE_EVENT_ID_LIFECYCLE " , 0 )
2016-08-17 11:32:36 +03:00
updateid = getattr ( libvirt , " VIR_NODE_DEVICE_EVENT_ID_UPDATE " , 1 )
2016-08-11 17:10:09 +03:00
self . _node_device_cb_ids . append (
self . get_backend ( ) . nodeDeviceEventRegisterAny (
None , eventid , self . _node_device_lifecycle_event , None ) )
2016-08-17 11:32:36 +03:00
self . _node_device_cb_ids . append (
self . get_backend ( ) . nodeDeviceEventRegisterAny (
None , updateid , self . _node_device_update_event , None ) )
2016-08-11 17:10:09 +03:00
self . using_node_device_events = True
2019-06-17 04:12:39 +03:00
log . debug ( " Using node device events " )
2017-05-05 19:47:21 +03:00
except Exception as e :
2016-08-11 17:10:09 +03:00
self . using_network_events = False
2019-06-17 04:12:39 +03:00
log . debug ( " Error registering node device events: %s " , e )
2016-08-11 17:10:09 +03:00
2013-07-07 05:31:50 +04:00
2009-11-26 00:13:46 +03:00
######################################
# Connection closing/opening methods #
######################################
2014-09-12 02:48:04 +04:00
def _schedule_close ( self ) :
self . _closing = True
self . idle_add ( self . close )
2009-11-26 00:13:46 +03:00
def close ( self ) :
2014-09-12 02:16:21 +04:00
if not self . is_disconnected ( ) :
2019-06-17 04:12:39 +03:00
log . debug ( " conn.close() uri= %s " , self . get_uri ( ) )
2014-04-16 18:32:52 +04:00
self . _closing = True
2014-03-10 17:33:04 +04:00
try :
2020-01-28 19:42:33 +03:00
if self . _backend . is_open ( ) :
2014-03-31 20:43:49 +04:00
for eid in self . _domain_cb_ids :
self . _backend . domainEventDeregisterAny ( eid )
for eid in self . _network_cb_ids :
self . _backend . networkEventDeregisterAny ( eid )
2016-06-15 00:52:40 +03:00
for eid in self . _storage_pool_cb_ids :
self . _backend . storagePoolEventDeregisterAny ( eid )
2016-08-11 17:10:09 +03:00
for eid in self . _node_device_cb_ids :
self . _backend . nodeDeviceEventDeregisterAny ( eid )
2020-08-20 20:34:01 +03:00
except Exception : # pragma: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Failed to deregister events in conn cleanup " ,
2014-03-10 17:33:04 +04:00
exc_info = True )
2014-04-16 20:23:57 +04:00
finally :
self . _domain_cb_ids = [ ]
self . _network_cb_ids = [ ]
2016-06-15 00:52:40 +03:00
self . _storage_pool_cb_ids = [ ]
2016-08-11 17:10:09 +03:00
self . _node_device_cb_ids = [ ]
2014-02-11 22:27:59 +04:00
2015-05-05 01:16:09 +03:00
self . _stats = [ ]
2011-04-11 19:00:57 +04:00
2015-04-10 21:08:25 +03:00
if self . _init_object_event :
self . _init_object_event . clear ( )
2018-03-12 22:48:09 +03:00
for obj in self . _objects . all_objects ( ) :
self . _objects . remove ( obj )
try :
self . _remove_object_signal ( obj )
obj . cleanup ( )
2020-08-20 20:34:01 +03:00
except Exception as e : # pramga: no cover
2019-06-17 04:12:39 +03:00
log . debug ( " Failed to cleanup %s : %s " , obj , e )
2015-04-10 17:33:04 +03:00
self . _objects . cleanup ( )
self . _objects = _ObjectList ( )
2011-03-23 17:07:23 +03:00
2018-03-12 23:31:13 +03:00
closeret = self . _backend . close ( )
2019-06-05 20:19:44 +03:00
if closeret == 1 and self . config . CLITestOptions . leak_debug :
2020-08-20 20:34:01 +03:00
log . debug ( # pragma: no cover
" LEAK: conn close() returned 1, "
2018-03-12 23:31:13 +03:00
" meaning refs may have leaked. " )
2014-09-12 02:16:21 +04:00
self . _change_state ( self . _STATE_DISCONNECTED )
2014-04-16 18:32:52 +04:00
self . _closing = False
2009-11-26 00:13:46 +03:00
2011-07-24 05:16:54 +04:00
def _cleanup ( self ) :
2011-04-13 17:27:02 +04:00
self . close ( )
2015-09-20 21:20:50 +03:00
self . _objects = None
2018-08-31 22:20:50 +03:00
self . _backend . cb_fetch_all_domains = None
2015-09-20 21:20:50 +03:00
self . _backend . cb_fetch_all_pools = None
self . _backend . cb_fetch_all_nodedevs = None
self . _backend . cb_fetch_all_vols = None
2017-07-19 22:56:43 +03:00
self . _backend . cb_cache_new_pool = None
2015-09-20 21:20:50 +03:00
2015-04-10 20:26:54 +03:00
def open ( self ) :
2014-09-12 02:16:21 +04:00
if not self . is_disconnected ( ) :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2007-09-10 06:57:24 +04:00
2014-09-12 02:16:21 +04:00
self . _change_state ( self . _STATE_CONNECTING )
2007-09-11 04:56:01 +04:00
2019-06-17 04:12:39 +03:00
log . debug ( " Scheduling background open thread for %s " ,
2018-03-02 11:01:23 +03:00
self . get_uri ( ) )
2015-04-10 20:26:54 +03:00
self . _start_thread ( self . _open_thread , " Connect %s " % self . get_uri ( ) )
2007-09-11 04:56:01 +04:00
2018-10-08 16:27:22 +03:00
def _do_open ( self ) :
2014-09-12 02:48:04 +04:00
warnconsole = False
libvirt_error_code = None
libvirt_error_message = None
2017-12-20 22:15:56 +03:00
exc = None
2009-03-09 23:21:32 +03:00
2014-09-12 02:48:04 +04:00
try :
2018-05-03 02:55:07 +03:00
self . _backend . open ( connectauth . creds_dialog , self )
2014-09-12 19:30:09 +04:00
return True , None
2017-12-20 22:15:56 +03:00
except Exception as e :
exc = e
2014-09-12 02:48:04 +04:00
tb = " " . join ( traceback . format_exc ( ) )
2016-04-18 23:42:12 +03:00
if isinstance ( exc , libvirt . libvirtError ) :
# pylint: disable=no-member
2014-09-12 02:48:04 +04:00
libvirt_error_code = exc . get_error_code ( )
libvirt_error_message = exc . get_error_message ( )
if ( libvirt_error_code ==
getattr ( libvirt , " VIR_ERR_AUTH_CANCELLED " , None ) ) :
2019-06-17 04:12:39 +03:00
log . debug ( " User cancelled auth, not raising any error. " )
2014-09-12 19:30:09 +04:00
return False , None
2007-09-11 04:56:01 +04:00
2014-09-12 02:48:04 +04:00
if ( libvirt_error_code == libvirt . VIR_ERR_AUTH_FAILED and
" not authorized " in libvirt_error_message . lower ( ) ) :
2019-06-17 04:12:39 +03:00
log . debug ( " Looks like we might have failed policykit "
2014-09-12 02:48:04 +04:00
" auth. Checking to see if we have a valid "
" console session " )
if ( not self . is_remote ( ) and
not connectauth . do_we_have_session ( ) ) :
warnconsole = True
2018-03-16 04:22:14 +03:00
ConnectError = connectauth . connect_error (
self , str ( exc ) , tb , warnconsole )
return False , ConnectError
2014-09-12 02:48:04 +04:00
def _populate_initial_state ( self ) :
2020-01-24 20:09:23 +03:00
log . debug ( " libvirt version= %s " , self . _backend . local_libvirt_version ( ) )
log . debug ( " daemon version= %s " , self . _backend . daemon_version ( ) )
2019-06-17 04:12:39 +03:00
log . debug ( " conn version= %s " , self . _backend . conn_version ( ) )
2020-01-24 20:09:23 +03:00
log . debug ( " %s capabilities: \n %s " , self . get_uri ( ) , self . caps . get_xml ( ) )
2020-08-20 20:34:01 +03:00
if not self . support . conn_domain ( ) : # pragma: no cover
2020-01-24 23:16:19 +03:00
raise RuntimeError ( " Connection does not support required "
" domain listing APIs " )
2020-08-20 20:34:01 +03:00
if not self . support . conn_storage ( ) : # pragma: no cover
2020-01-24 20:09:23 +03:00
log . debug ( " Connection doesn ' t seem to support storage APIs. " )
2020-08-20 20:34:01 +03:00
if not self . support . conn_network ( ) : # pragma: no cover
2020-01-24 20:09:23 +03:00
log . debug ( " Connection doesn ' t seem to support network APIs. " )
2020-08-20 20:34:01 +03:00
if not self . support . conn_nodedev ( ) : # pragma: no cover
2020-01-24 20:09:23 +03:00
log . debug ( " Connection doesn ' t seem to support nodedev APIs. " )
2016-06-21 00:00:12 +03:00
2014-09-12 02:48:04 +04:00
self . _add_conn_events ( )
2007-09-11 04:56:01 +04:00
2014-09-12 02:48:04 +04:00
try :
self . _backend . setKeepAlive ( 20 , 1 )
2017-05-05 19:47:21 +03:00
except Exception as e :
2020-01-27 12:49:46 +03:00
log . debug ( " Failed to setKeepAlive: %s " , str ( e ) )
2009-07-12 05:23:16 +04:00
2015-04-10 21:08:25 +03:00
# The initial tick will set up a threading event that will only
# trigger after all the polled libvirt objects are fully initialized.
# That way we only report the connection is open when everything is
# nicely setup for the rest of the app.
self . _init_object_event = threading . Event ( )
self . _init_object_count = 0
self . schedule_priority_tick ( stats_update = True ,
pollvm = True , pollnet = True ,
2020-01-26 04:28:43 +03:00
pollpool = True , pollnodedev = True ,
force = True , initial_poll = True )
2015-04-10 21:08:25 +03:00
self . _init_object_event . wait ( )
self . _init_object_event = None
self . _init_object_count = None
2019-10-25 22:27:18 +03:00
# Try to create the default storage pool
# We need this after events setup so we can determine if the default
# pool already exists
try :
virtinst . StoragePool . build_default_pool ( self . get_backend ( ) )
2020-08-20 20:34:01 +03:00
except Exception as e : # pragma: no cover
2019-10-25 22:27:18 +03:00
log . debug ( " Building default pool failed: %s " , str ( e ) )
2014-09-12 02:48:04 +04:00
def _open_thread ( self ) :
2018-03-16 04:22:14 +03:00
ConnectError = None
2014-07-05 01:37:42 +04:00
try :
2018-03-16 04:22:14 +03:00
is_active , ConnectError = self . _do_open ( )
2014-09-12 02:48:04 +04:00
if is_active :
self . _populate_initial_state ( )
2014-07-05 01:43:24 +04:00
2015-04-10 21:08:25 +03:00
self . idle_add ( self . _change_state , is_active and
self . _STATE_ACTIVE or self . _STATE_DISCONNECTED )
2017-05-05 19:47:21 +03:00
except Exception as e :
2014-09-12 02:48:04 +04:00
is_active = False
self . _schedule_close ( )
2018-03-16 04:22:14 +03:00
ConnectError = connectauth . connect_error ( self , str ( e ) ,
" " . join ( traceback . format_exc ( ) ) , False )
2010-11-24 22:01:51 +03:00
2018-03-16 04:22:14 +03:00
self . idle_emit ( " open-completed " , ConnectError )
2007-09-11 04:56:01 +04:00
2006-10-11 01:24:59 +04:00
2009-11-26 00:13:46 +03:00
#######################
# Tick/Update methods #
#######################
2006-06-14 18:59:40 +04:00
2018-03-12 22:48:09 +03:00
def _remove_object_signal ( self , obj ) :
2018-10-14 00:21:22 +03:00
if obj . is_domain ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " vm-removed " , obj )
2018-10-14 00:21:22 +03:00
elif obj . is_network ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " net-removed " , obj )
2018-10-14 00:21:22 +03:00
elif obj . is_pool ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " pool-removed " , obj )
2018-10-14 00:21:22 +03:00
elif obj . is_nodedev ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " nodedev-removed " , obj )
2018-03-12 22:48:09 +03:00
2015-04-10 21:08:25 +03:00
def _gone_object_signals ( self , gone_objects ) :
2015-04-10 17:33:04 +03:00
"""
Responsible for signaling the UI for any updates . All possible UI
updates need to go here to enable threading that doesn ' t block the
app with long tick operations .
"""
if not self . _backend . is_open ( ) :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2015-04-10 17:33:04 +03:00
2015-04-10 16:15:44 +03:00
for obj in gone_objects :
2015-04-10 21:08:25 +03:00
class_name = obj . class_name ( )
2015-05-07 19:30:28 +03:00
try :
name = obj . get_name ( )
2017-07-24 11:26:48 +03:00
except Exception :
2015-05-07 19:30:28 +03:00
name = str ( obj )
2015-04-10 21:08:25 +03:00
2015-04-10 17:33:04 +03:00
if not self . _objects . remove ( obj ) :
2019-06-17 04:12:39 +03:00
log . debug ( " Requested removal of %s = %s , but it ' s "
2015-05-07 19:30:28 +03:00
" not in our object list. " , class_name , name )
2015-04-10 17:33:04 +03:00
continue
2019-06-17 04:12:39 +03:00
log . debug ( " %s = %s removed " , class_name , name )
2018-03-12 22:48:09 +03:00
self . _remove_object_signal ( obj )
2015-04-10 16:15:44 +03:00
obj . cleanup ( )
2018-03-12 22:57:36 +03:00
def _new_object_cb ( self , obj , initialize_failed ) :
2015-04-10 21:08:25 +03:00
if not self . _backend . is_open ( ) :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2015-04-10 17:33:04 +03:00
2015-04-10 21:08:25 +03:00
try :
2015-04-10 16:15:44 +03:00
class_name = obj . class_name ( )
2015-04-10 21:08:25 +03:00
2015-09-17 22:48:42 +03:00
if initialize_failed :
2019-06-17 04:12:39 +03:00
log . debug ( " Blacklisting %s = %s " , class_name , obj . get_name ( ) )
2017-09-08 10:36:58 +03:00
count = self . _objects . add_blacklist ( obj )
2020-09-02 18:22:02 +03:00
log . debug ( " Object added in blacklist, count= %d " , count )
2015-09-17 22:48:42 +03:00
return
2020-09-02 18:22:02 +03:00
self . _objects . remove_blacklist ( obj )
2015-04-10 21:08:25 +03:00
if not self . _objects . add ( obj ) :
2019-06-17 04:12:39 +03:00
log . debug ( " New %s = %s requested, but it ' s already tracked. " ,
2015-04-10 21:08:25 +03:00
class_name , obj . get_name ( ) )
return
2018-10-14 00:21:22 +03:00
if not obj . is_nodedev ( ) :
2015-04-10 16:15:44 +03:00
# Skip nodedev logging since it's noisy and not interesting
2019-06-17 04:12:39 +03:00
log . debug ( " %s = %s status= %s added " , class_name ,
2015-04-10 16:15:44 +03:00
obj . get_name ( ) , obj . run_status ( ) )
2018-10-14 00:21:22 +03:00
if obj . is_domain ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " vm-added " , obj )
2018-10-14 00:21:22 +03:00
elif obj . is_network ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " net-added " , obj )
2018-10-14 00:21:22 +03:00
elif obj . is_pool ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " pool-added " , obj )
2018-10-14 00:21:22 +03:00
elif obj . is_nodedev ( ) :
2020-09-01 18:54:10 +03:00
self . emit ( " nodedev-added " , obj )
2015-04-10 21:08:25 +03:00
finally :
2018-03-12 22:57:36 +03:00
if self . _init_object_event :
2015-04-10 21:08:25 +03:00
self . _init_object_count - = 1
if self . _init_object_count < = 0 :
self . _init_object_event . set ( )
2014-09-12 19:55:34 +04:00
2015-04-10 21:08:25 +03:00
def _poll ( self , initial_poll ,
2020-01-26 04:28:43 +03:00
pollvm , pollnet , pollpool , pollnodedev ) :
2015-04-10 19:52:42 +03:00
"""
Helper called from tick ( ) to do necessary polling and return
the relevant object lists
"""
gone_objects = [ ]
preexisting_objects = [ ]
2020-02-02 04:51:10 +03:00
def _process_objects ( ptype ) :
if ptype == " nets " :
dopoll = pollnet
objs = self . list_nets ( )
cls = vmmNetwork
pollcb = pollhelpers . fetch_nets
elif ptype == " pools " :
dopoll = pollpool
objs = self . list_pools ( )
cls = vmmStoragePool
pollcb = pollhelpers . fetch_pools
elif ptype == " nodedevs " :
dopoll = pollnodedev
objs = self . list_nodedevs ( )
cls = vmmNodeDevice
pollcb = pollhelpers . fetch_nodedevs
else :
dopoll = pollvm
objs = self . list_vms ( )
cls = vmmDomain
pollcb = pollhelpers . fetch_vms
2020-01-24 23:46:02 +03:00
2020-09-01 19:35:26 +03:00
keymap = dict ( ( o . get_name ( ) , o ) for o in objs )
def cb ( obj , name ) :
return cls ( self , obj , name )
2020-02-02 04:51:10 +03:00
if dopoll :
gone , new , master = pollcb ( self . _backend , keymap , cb )
else :
gone , new , master = [ ] , [ ] , list ( keymap . values ( ) )
2015-04-10 21:08:25 +03:00
if initial_poll :
self . _init_object_count + = len ( new )
2015-04-10 19:52:42 +03:00
gone_objects . extend ( gone )
preexisting_objects . extend ( [ o for o in master if o not in new ] )
2015-09-17 22:48:42 +03:00
new = [ n for n in new if not self . _objects . in_blacklist ( n ) ]
2015-04-10 19:52:42 +03:00
return new
2020-02-02 04:51:10 +03:00
new_vms = _process_objects ( " vms " )
new_nets = _process_objects ( " nets " )
new_pools = _process_objects ( " pools " )
new_nodedevs = _process_objects ( " nodedevs " )
2015-04-10 21:08:25 +03:00
# Kick off one thread per object type to handle the initial
# XML fetching. Going any more fine grained then this probably
# won't be that useful due to libvirt's locking structure.
#
# Would prefer to start refreshing some objects before all polling
# is complete, but we need init_object_count to be fully accurate
# before we start initializing objects
2018-03-09 00:30:34 +03:00
if initial_poll and self . _init_object_count == 0 :
# If the connection doesn't have any objects, new_object_cb
# is never called and the event is never set, so let's do it here
self . _init_object_event . set ( )
2020-01-26 04:28:43 +03:00
for newlist in [ new_vms , new_nets , new_pools , new_nodedevs ] :
2015-04-10 21:08:25 +03:00
if not newlist :
continue
def cb ( lst ) :
for obj in lst :
obj . connect_once ( " initialized " , self . _new_object_cb )
obj . init_libvirt_state ( )
2015-04-10 19:52:42 +03:00
2015-04-10 21:08:25 +03:00
self . _start_thread ( cb ,
" refreshing xml for new %s " % newlist [ 0 ] . class_name ( ) ,
args = ( newlist , ) )
2015-04-10 19:52:42 +03:00
2015-04-10 21:08:25 +03:00
return gone_objects , preexisting_objects
2015-04-10 19:52:42 +03:00
2015-04-13 23:56:46 +03:00
def _tick ( self , stats_update = False ,
2013-07-07 19:06:15 +04:00
pollvm = False , pollnet = False ,
2020-01-26 04:28:43 +03:00
pollpool = False , pollnodedev = False ,
2015-04-10 21:08:25 +03:00
force = False , initial_poll = False ) :
2014-02-11 21:07:13 +04:00
"""
main update function : polls for new objects , updates stats , . . .
2015-04-10 19:52:42 +03:00
: param force : Perform the requested polling even if async events
2015-04-10 21:08:25 +03:00
are in use .
2014-02-11 21:07:13 +04:00
"""
2014-09-12 03:12:43 +04:00
if self . _closing :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2014-09-12 03:12:43 +04:00
if self . is_disconnected ( ) :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2015-04-10 21:08:25 +03:00
if self . is_connecting ( ) and not force :
return
2006-10-09 21:28:13 +04:00
2015-04-10 19:52:42 +03:00
# We need to set this before the event check, since stats polling
# is independent of events
2013-07-07 19:06:15 +04:00
if not pollvm :
stats_update = False
2014-02-11 22:19:15 +04:00
2014-02-11 21:07:13 +04:00
if self . using_domain_events and not force :
pollvm = False
2014-02-11 22:19:15 +04:00
if self . using_network_events and not force :
pollnet = False
2016-06-15 00:52:40 +03:00
if self . using_storage_pool_events and not force :
pollpool = False
2016-08-11 17:10:09 +03:00
if self . using_node_device_events and not force :
pollnodedev = False
2013-07-07 19:06:15 +04:00
2015-05-05 01:16:09 +03:00
self . _hostinfo = self . _backend . getInfo ( )
2018-10-10 23:59:55 +03:00
if stats_update :
self . statsmanager . cache_all_stats ( self )
2008-10-06 21:21:06 +04:00
2015-04-10 21:08:25 +03:00
gone_objects , preexisting_objects = self . _poll (
2020-01-26 04:28:43 +03:00
initial_poll , pollvm , pollnet , pollpool , pollnodedev )
2015-04-10 21:08:25 +03:00
self . idle_add ( self . _gone_object_signals , gone_objects )
2008-08-08 01:37:16 +04:00
2015-04-10 19:52:42 +03:00
# Only tick() pre-existing objects, since new objects will be
# initialized asynchronously and tick() would be redundant
for obj in preexisting_objects :
2009-09-17 00:02:19 +04:00
try :
2015-04-10 19:52:42 +03:00
if obj . reports_stats ( ) and stats_update :
pass
2018-10-14 00:21:22 +03:00
elif obj . is_domain ( ) and not pollvm :
2015-04-10 19:52:42 +03:00
continue
2018-10-14 00:21:22 +03:00
elif obj . is_network ( ) and not pollnet :
2015-04-10 19:52:42 +03:00
continue
2018-10-14 00:21:22 +03:00
elif obj . is_pool ( ) and not pollpool :
2015-04-10 19:52:42 +03:00
continue
2018-10-14 00:21:22 +03:00
elif obj . is_nodedev ( ) and not pollnodedev :
2015-04-10 19:52:42 +03:00
continue
2020-09-02 19:54:53 +03:00
if self . config . CLITestOptions . conn_crash :
self . _backend . close ( )
e = libvirt . libvirtError ( " fake error " )
e . err = [ libvirt . VIR_ERR_SYSTEM_ERROR ]
raise e
2015-04-10 16:15:44 +03:00
obj . tick ( stats_update = stats_update )
2017-05-05 19:47:21 +03:00
except Exception as e :
2019-06-17 04:12:39 +03:00
log . exception ( " Tick for %s failed " , obj )
2012-01-29 20:26:24 +04:00
if ( isinstance ( e , libvirt . libvirtError ) and
2012-02-01 04:07:32 +04:00
( getattr ( e , " get_error_code " ) ( ) ==
libvirt . VIR_ERR_SYSTEM_ERROR ) ) :
2012-01-29 20:26:24 +04:00
# Try a simple getInfo call to see if conn was dropped
2013-07-05 16:59:58 +04:00
self . _backend . getInfo ( )
2019-06-17 04:12:39 +03:00
log . debug ( " vm tick raised system error but "
2012-01-29 20:26:24 +04:00
" connection doesn ' t seem to have dropped. "
" Ignoring. " )
2007-03-28 03:52:00 +04:00
2013-07-07 17:42:21 +04:00
if stats_update :
2015-04-10 19:52:42 +03:00
self . _recalculate_stats (
[ o for o in preexisting_objects if o . reports_stats ( ) ] )
2011-04-18 19:12:36 +04:00
self . idle_emit ( " resources-sampled " )
2009-07-12 05:23:16 +04:00
2013-07-07 16:12:15 +04:00
def _recalculate_stats ( self , vms ) :
2013-07-05 16:59:58 +04:00
if not self . _backend . is_open ( ) :
2020-08-20 20:34:01 +03:00
return # pragma: no cover
2010-02-11 23:25:41 +03:00
2013-07-07 16:12:15 +04:00
now = time . time ( )
2007-03-28 03:52:00 +04:00
expected = self . config . get_stats_history_length ( )
2015-05-05 01:16:09 +03:00
current = len ( self . _stats )
2007-03-28 03:52:00 +04:00
if current > expected :
2015-05-05 01:16:09 +03:00
del self . _stats [ expected : current ]
2007-03-28 03:52:00 +04:00
mem = 0
cpuTime = 0
2008-10-18 23:21:33 +04:00
rdRate = 0
wrRate = 0
rxRate = 0
txRate = 0
2011-07-25 23:09:42 +04:00
diskMaxRate = self . disk_io_max_rate ( ) or 10.0
netMaxRate = self . network_traffic_max_rate ( ) or 10.0
2007-03-28 03:52:00 +04:00
2013-07-07 05:42:41 +04:00
for vm in vms :
2011-04-10 06:40:22 +04:00
if not vm . is_active ( ) :
continue
cpuTime + = vm . cpu_time ( )
2011-04-10 08:02:39 +04:00
mem + = vm . stats_memory ( )
2011-04-10 06:40:22 +04:00
rdRate + = vm . disk_read_rate ( )
wrRate + = vm . disk_write_rate ( )
rxRate + = vm . network_rx_rate ( )
txRate + = vm . network_tx_rate ( )
2007-03-28 03:52:00 +04:00
2011-07-25 23:09:42 +04:00
netMaxRate = max ( netMaxRate , vm . network_traffic_max_rate ( ) )
diskMaxRate = max ( diskMaxRate , vm . disk_io_max_rate ( ) )
2011-07-12 05:22:50 +04:00
pcentHostCpu = 0
pcentMem = mem * 100.0 / self . host_memory_size ( )
2015-05-05 01:16:09 +03:00
if len ( self . _stats ) > 0 :
prevTimestamp = self . _stats [ 0 ] [ " timestamp " ]
2010-12-10 19:47:07 +03:00
host_cpus = self . host_active_processor_count ( )
2011-07-12 05:22:50 +04:00
pcentHostCpu = ( ( cpuTime ) * 100.0 /
2010-12-10 19:47:07 +03:00
( ( now - prevTimestamp ) *
1000.0 * 1000.0 * 1000.0 * host_cpus ) )
2007-03-28 03:52:00 +04:00
2011-07-12 05:22:50 +04:00
pcentHostCpu = max ( 0.0 , min ( 100.0 , pcentHostCpu ) )
pcentMem = max ( 0.0 , min ( 100.0 , pcentMem ) )
2007-03-28 03:52:00 +04:00
newStats = {
" timestamp " : now ,
" memory " : mem ,
" memoryPercent " : pcentMem ,
" cpuTime " : cpuTime ,
2011-07-12 05:22:50 +04:00
" cpuHostPercent " : pcentHostCpu ,
2017-08-05 09:39:32 +03:00
" diskRdRate " : rdRate ,
" diskWrRate " : wrRate ,
" netRxRate " : rxRate ,
" netTxRate " : txRate ,
" diskMaxRate " : diskMaxRate ,
" netMaxRate " : netMaxRate ,
2007-03-28 03:52:00 +04:00
}
2015-05-05 01:16:09 +03:00
self . _stats . insert ( 0 , newStats )
2007-03-28 03:52:00 +04:00
2009-11-26 00:13:46 +03:00
2015-04-10 19:52:42 +03:00
def schedule_priority_tick ( self , * * kwargs ) :
2018-03-16 04:22:14 +03:00
from . engine import vmmEngine
vmmEngine . get_instance ( ) . schedule_priority_tick ( self , kwargs )
2015-04-10 19:52:42 +03:00
2015-04-13 23:56:46 +03:00
def tick_from_engine ( self , * args , * * kwargs ) :
2015-04-10 19:52:42 +03:00
try :
self . _tick ( * args , * * kwargs )
2020-09-02 19:53:25 +03:00
except Exception :
self . _schedule_close ( )
raise
2015-04-10 19:52:42 +03:00
2009-11-26 00:13:46 +03:00
########################
# Stats getter methods #
########################
2012-05-14 17:24:56 +04:00
2015-05-04 23:33:56 +03:00
def _get_record_helper ( self , record_name ) :
2015-05-05 01:16:09 +03:00
if len ( self . _stats ) == 0 :
2015-05-04 23:33:56 +03:00
return 0
2015-05-05 01:16:09 +03:00
return self . _stats [ 0 ] [ record_name ]
2015-05-04 23:33:56 +03:00
def _vector_helper ( self , record_name , limit , ceil = 100.0 ) :
2007-03-28 03:52:00 +04:00
vector = [ ]
2015-05-04 23:33:56 +03:00
statslen = self . config . get_stats_history_length ( ) + 1
if limit is not None :
statslen = min ( statslen , limit )
for i in range ( statslen ) :
2015-05-05 01:16:09 +03:00
if i < len ( self . _stats ) :
vector . append ( self . _stats [ i ] [ record_name ] / ceil )
2007-03-28 03:52:00 +04:00
else :
vector . append ( 0 )
2013-04-12 16:26:21 +04:00
2015-05-04 23:33:56 +03:00
return vector
2011-04-14 20:20:02 +04:00
2015-05-04 23:33:56 +03:00
def stats_memory_vector ( self , limit = None ) :
return self . _vector_helper ( " memoryPercent " , limit )
def host_cpu_time_vector ( self , limit = None ) :
return self . _vector_helper ( " cpuHostPercent " , limit )
2011-04-14 20:20:02 +04:00
def stats_memory ( self ) :
return self . _get_record_helper ( " memory " )
2011-07-12 05:22:50 +04:00
def host_cpu_time_percentage ( self ) :
return self . _get_record_helper ( " cpuHostPercent " )
2015-05-04 23:33:56 +03:00
def guest_cpu_time_percentage ( self ) :
return self . host_cpu_time_percentage ( )
2008-10-18 23:21:33 +04:00
def network_traffic_rate ( self ) :
2015-05-04 23:33:56 +03:00
return ( self . _get_record_helper ( " netRxRate " ) +
self . _get_record_helper ( " netTxRate " ) )
def disk_io_rate ( self ) :
return ( self . _get_record_helper ( " diskRdRate " ) +
self . _get_record_helper ( " diskWrRate " ) )
2011-07-25 23:09:42 +04:00
def network_traffic_max_rate ( self ) :
return self . _get_record_helper ( " netMaxRate " )
def disk_io_max_rate ( self ) :
return self . _get_record_helper ( " diskMaxRate " )
2015-04-11 20:39:25 +03:00
###########################
# Per-conn config helpers #
###########################
def get_autoconnect ( self ) :
return self . config . get_conn_autoconnect ( self . get_uri ( ) )
def set_autoconnect ( self , val ) :
self . config . set_conn_autoconnect ( self . get_uri ( ) , val )
def set_config_pretty_name ( self , value ) :
2015-09-16 02:35:30 +03:00
cfgname = self . _get_config_pretty_name ( )
if value == cfgname :
return
if not cfgname and value == self . get_pretty_desc ( ) :
# Don't encode the default connection value into gconf right
# away, require the user to edit it first
return
self . config . set_perconn ( self . get_uri ( ) , " /pretty-name " , value )
2015-04-11 20:39:25 +03:00
def _get_config_pretty_name ( self ) :
return self . config . get_perconn ( self . get_uri ( ) , " /pretty-name " )
def _on_config_pretty_name_changed ( self , * args , * * kwargs ) :
return self . config . listen_perconn ( self . get_uri ( ) , " /pretty-name " ,
* args , * * kwargs )
def _config_pretty_name_changed_cb ( self ) :
self . emit ( " state-changed " )
2019-04-13 21:52:17 +03:00
def set_details_window_size ( self , w , h ) :
self . config . set_perconn ( self . get_uri ( ) , " /window-size " , ( w , h ) )
def get_details_window_size ( self ) :
ret = self . config . get_perconn ( self . get_uri ( ) , " /window-size " )
return ret