Add domain api for invalidating and updating cached xml.

Use this be a bit more intelligent with Details hardware list: only update
list if domain xml has changed. Don't even check for new xml if the details
window isn't present.
This commit is contained in:
Cole Robinson 2009-02-28 19:27:20 -05:00
parent 05a0c51217
commit 782406b222
2 changed files with 128 additions and 93 deletions

View File

@ -331,6 +331,7 @@ class vmmDetails(gobject.GObject):
self.vm.connect("status-changed", self.update_widget_states) self.vm.connect("status-changed", self.update_widget_states)
self.vm.connect("resources-sampled", self.refresh_resources) self.vm.connect("resources-sampled", self.refresh_resources)
self.vm.connect("config-changed", self.refresh_vm_info)
self.window.get_widget("hw-list").get_selection().connect("changed", self.hw_selected) self.window.get_widget("hw-list").get_selection().connect("changed", self.hw_selected)
self.update_widget_states(self.vm, self.vm.status()) self.update_widget_states(self.vm, self.vm.status())
@ -339,7 +340,7 @@ class vmmDetails(gobject.GObject):
self.pixbuf_memory = gtk.gdk.pixbuf_new_from_file(config.get_icon_dir() + "/icon_cpu.png") self.pixbuf_memory = gtk.gdk.pixbuf_new_from_file(config.get_icon_dir() + "/icon_cpu.png")
self.prepare_hw_list() self.prepare_hw_list()
self.hw_selected() self.hw_selected()
self.refresh_resources(self.vm) self.refresh_vm_info()
# Black magic todo with scrolled windows. Basically the behaviour we want # Black magic todo with scrolled windows. Basically the behaviour we want
@ -560,8 +561,10 @@ class vmmDetails(gobject.GObject):
return return
dialog.show_all() dialog.show_all()
dialog.present() dialog.present()
self.engine.increment_window_counter() self.engine.increment_window_counter()
self.update_widget_states(self.vm, self.vm.status()) self.update_widget_states(self.vm, self.vm.status())
self.vm.update_xml()
def show_help(self, src): def show_help(self, src):
# From the Details window, show the help document from the Details page # From the Details window, show the help document from the Details page
@ -643,15 +646,16 @@ class vmmDetails(gobject.GObject):
def hw_selected(self, src=None): def hw_selected(self, src=None):
vmlist = self.window.get_widget("hw-list") pagetype = self.get_hw_selection(HW_LIST_COL_TYPE)
selection = vmlist.get_selection() if not pagetype:
active = selection.get_selected() self.window.get_widget("hw-panel").set_sensitive(True)
if active[1] != None: self.window.get_widget("hw-list").get_selection().select_path(0)
pagetype = active[0].get_value(active[1], HW_LIST_COL_TYPE) self.window.get_widget("hw-panel").set_current_page(0)
return
self.window.get_widget("hw-panel").set_sensitive(True) self.window.get_widget("hw-panel").set_sensitive(True)
self.window.get_widget("hw-panel").show_all() self.window.get_widget("hw-panel").show_all()
pagenum = pagetype
if pagetype == HW_LIST_TYPE_GENERAL: if pagetype == HW_LIST_TYPE_GENERAL:
pass pass
elif pagetype == HW_LIST_TYPE_STATS: elif pagetype == HW_LIST_TYPE_STATS:
@ -680,13 +684,9 @@ class vmmDetails(gobject.GObject):
elif pagetype == HW_LIST_TYPE_HOSTDEV: elif pagetype == HW_LIST_TYPE_HOSTDEV:
self.refresh_hostdev_page() self.refresh_hostdev_page()
else: else:
pagenum = -1 pagetype = -1
self.window.get_widget("hw-panel").set_current_page(pagenum) self.window.get_widget("hw-panel").set_current_page(pagetype)
else:
self.window.get_widget("hw-panel").set_sensitive(False)
selection.select_path(0)
self.window.get_widget("hw-panel").set_current_page(0)
def control_vm_pause(self, src): def control_vm_pause(self, src):
if self.ignorePause: if self.ignorePause:
@ -843,21 +843,35 @@ class vmmDetails(gobject.GObject):
def switch_page(self, ignore1=None, ignore2=None, newpage=None): def switch_page(self, ignore1=None, ignore2=None, newpage=None):
self.page_refresh(newpage) self.page_refresh(newpage)
def refresh_resources(self, ignore=None): def refresh_resources(self, ignore):
details = self.window.get_widget("details-pages")
page = details.get_current_page()
if self.is_visible():
# Force an xml update, so we check for changed xml on every tick
self.vm.update_xml()
if (page == PAGE_DETAILS and
self.get_hw_selection(HW_LIST_COL_TYPE) == HW_LIST_TYPE_STATS):
self.refresh_stats_page()
def refresh_vm_info(self, ignore=None):
details = self.window.get_widget("details-pages") details = self.window.get_widget("details-pages")
self.page_refresh(details.get_current_page()) self.page_refresh(details.get_current_page())
def page_refresh(self, page): def page_refresh(self, page):
if page == PAGE_DETAILS: if page != PAGE_DETAILS:
return
# This function should only be called when the VM xml actually
# changes (not everytime it is refreshed). This saves us from blindly
# parsing the xml every tick
pagetype = self.get_hw_selection(HW_LIST_COL_TYPE)
# Add / remove new devices # Add / remove new devices
self.repopulate_hw_list() self.repopulate_hw_list()
# Now refresh desired page
hw_list = self.window.get_widget("hw-list")
selection = hw_list.get_selection()
active = selection.get_selected()
if active[1] != None:
pagetype = active[0].get_value(active[1], HW_LIST_COL_TYPE)
if pagetype == HW_LIST_TYPE_GENERAL: if pagetype == HW_LIST_TYPE_GENERAL:
# Nothing to refresh at this point # Nothing to refresh at this point
pass pass
@ -987,7 +1001,7 @@ class vmmDetails(gobject.GObject):
else: else:
perms = "Read/Write" perms = "Read/Write"
if diskinfo[7] == True: if diskinfo[7] == True:
perms += ", Sharable" perms += ", Shareable"
self.window.get_widget("disk-permissions").set_text(perms) self.window.get_widget("disk-permissions").set_text(perms)
bus = diskinfo[8] or _("Unknown") bus = diskinfo[8] or _("Unknown")
self.window.get_widget("disk-bus").set_text(bus) self.window.get_widget("disk-bus").set_text(bus)
@ -1531,7 +1545,7 @@ class vmmDetails(gobject.GObject):
return return
self.remove_device(info[0], info[1]) self.remove_device(info[0], info[1])
self.refresh_resources() self.vm.update_xml()
def prepare_hw_list(self): def prepare_hw_list(self):
hw_list_model = gtk.ListStore(str, str, int, gtk.gdk.Pixbuf, int, gobject.TYPE_PYOBJECT) hw_list_model = gtk.ListStore(str, str, int, gtk.gdk.Pixbuf, int, gobject.TYPE_PYOBJECT)
@ -1770,7 +1784,7 @@ class vmmDetails(gobject.GObject):
self.addhw.show() self.addhw.show()
def add_hardware_done(self, ignore=None): def add_hardware_done(self, ignore=None):
self.refresh_resources() self.vm.update_xml()
def toggle_cdrom(self, src): def toggle_cdrom(self, src):
info = self.get_hw_selection(HW_LIST_COL_DEVICE) info = self.get_hw_selection(HW_LIST_COL_DEVICE)

View File

@ -33,6 +33,9 @@ class vmmDomain(gobject.GObject):
"resources-sampled": (gobject.SIGNAL_RUN_FIRST, "resources-sampled": (gobject.SIGNAL_RUN_FIRST,
gobject.TYPE_NONE, gobject.TYPE_NONE,
[]), []),
"config-changed": (gobject.SIGNAL_RUN_FIRST,
gobject.TYPE_NONE,
[]),
} }
def __init__(self, config, connection, vm, uuid): def __init__(self, config, connection, vm, uuid):
@ -49,14 +52,16 @@ class vmmDomain(gobject.GObject):
"netRxRate" : 10.0, "netRxRate" : 10.0,
} }
self._update_status() self._xml = None
self.xml = None self._valid_xml = False
self._mem_stats = None self._mem_stats = None
self._cpu_stats = None self._cpu_stats = None
self._network_traffic = None self._network_traffic = None
self._disk_io = None self._disk_io = None
self._update_status()
self.config.on_stats_enable_mem_poll_changed(self.toggle_sample_mem_stats) self.config.on_stats_enable_mem_poll_changed(self.toggle_sample_mem_stats)
self.config.on_stats_enable_cpu_poll_changed(self.toggle_sample_cpu_stats) self.config.on_stats_enable_cpu_poll_changed(self.toggle_sample_cpu_stats)
self.config.on_stats_enable_net_poll_changed(self.toggle_sample_network_traffic) self.config.on_stats_enable_net_poll_changed(self.toggle_sample_network_traffic)
@ -67,14 +72,26 @@ class vmmDomain(gobject.GObject):
self.toggle_sample_network_traffic() self.toggle_sample_network_traffic()
self.toggle_sample_disk_io() self.toggle_sample_disk_io()
def get_xml(self): def get_xml(self):
if self.xml is None: # Get domain xml. If cached xml is invalid, update.
if self._xml is None or not self._valid_xml:
self.update_xml() self.update_xml()
return self.xml return self._xml
def update_xml(self): def update_xml(self):
self.xml = self.vm.XMLDesc(libvirt.VIR_DOMAIN_XML_SECURE) # Force an xml update. Signal 'config-changed' if domain xml has
# changed since last refresh
origxml = self._xml
self._xml = self.vm.XMLDesc(libvirt.VIR_DOMAIN_XML_SECURE)
self._valid_xml = True
if origxml != self._xml:
self.emit("config-changed")
def invalidate_xml(self):
# Mark cached xml as invalid
self._valid_xml = False
def release_handle(self): def release_handle(self):
del(self.vm) del(self.vm)
@ -121,6 +138,7 @@ class vmmDomain(gobject.GObject):
def is_hvm(self): def is_hvm(self):
os_type = util.get_xml_path(self.get_xml(), "/domain/os/type") os_type = util.get_xml_path(self.get_xml(), "/domain/os/type")
# FIXME: This should be static, not parse xml everytime
# XXX libvirt bug - doesn't work for inactive guests # XXX libvirt bug - doesn't work for inactive guests
#os_type = self.vm.OSType() #os_type = self.vm.OSType()
logging.debug("OS Type: %s" % os_type) logging.debug("OS Type: %s" % os_type)
@ -129,6 +147,7 @@ class vmmDomain(gobject.GObject):
return False return False
def get_type(self): def get_type(self):
# FIXME: This should be static, not parse xml everytime
return util.get_xml_path(self.get_xml(), "/domain/@type") return util.get_xml_path(self.get_xml(), "/domain/@type")
def is_vcpu_hotplug_capable(self): def is_vcpu_hotplug_capable(self):
@ -261,8 +280,9 @@ class vmmDomain(gobject.GObject):
if self.connection.get_state() != self.connection.STATE_ACTIVE: if self.connection.get_state() != self.connection.STATE_ACTIVE:
return return
# Clear cached XML # Invalidate cached xml
self.xml = None self.invalidate_xml()
info = self.vm.info() info = self.vm.info()
expected = self.config.get_stats_history_length() expected = self.config.get_stats_history_length()
current = len(self.record) current = len(self.record)
@ -588,7 +608,8 @@ class vmmDomain(gobject.GObject):
return self._parse_device_xml(_parse_serial_consoles) return self._parse_device_xml(_parse_serial_consoles)
def get_graphics_console(self): def get_graphics_console(self):
self.xml = None self.update_xml()
typ = util.get_xml_path(self.get_xml(), typ = util.get_xml_path(self.get_xml(),
"/domain/devices/graphics/@type") "/domain/devices/graphics/@type")
port = None port = None
@ -1085,7 +1106,7 @@ class vmmDomain(gobject.GObject):
self.get_connection().define_domain(newxml) self.get_connection().define_domain(newxml)
# Invalidate cached XML # Invalidate cached XML
self.xml = None self.invalidate_xml()
def remove_device(self, dev_type, dev_id_info): def remove_device(self, dev_type, dev_id_info):
newxml = self._remove_xml_device(dev_type, dev_id_info) newxml = self._remove_xml_device(dev_type, dev_id_info)
@ -1094,7 +1115,7 @@ class vmmDomain(gobject.GObject):
self.get_connection().define_domain(newxml) self.get_connection().define_domain(newxml)
# Invalidate cached XML # Invalidate cached XML
self.xml = None self.invalidate_xml()
def _change_cdrom(self, newdev, dev_id_info): def _change_cdrom(self, newdev, dev_id_info):
# If vm is shutoff, remove device, and redefine with media # If vm is shutoff, remove device, and redefine with media
@ -1234,7 +1255,7 @@ class vmmDomain(gobject.GObject):
doc.freeDoc() doc.freeDoc()
# Invalidate cached xml # Invalidate cached xml
self.xml = None self.invalidate_xml()
def toggle_sample_cpu_stats(self, ignore1=None, ignore2=None, def toggle_sample_cpu_stats(self, ignore1=None, ignore2=None,
ignore3=None, ignore4=None): ignore3=None, ignore4=None):