mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2025-01-04 13:52:07 +01:00
a4fe507adf
In some cases calling Control.Invoke() from a background thread causes that thread to go in a "sleep, wait, or join" mode, while waiting for Invoke to happen, although the UI thread is running normally. If the Control is the MainWindow, it works as expected, but we've seen it happening while connecting or disconnecting from a large pool, on calling Invoke on controls like NavigationView, AlertSummaryPage, HistoryPage, etc. To fix this, we call the Invoke on the MainWindow in all the places where we've seen the issue. With this changes, the previous fix for CA-148245 (call RequestRefreshTreeView on CacheClearing event) is not needed anymore, so I removed that call. Signed-off-by: Mihaela Stoica <mihaela.stoica@citrix.com>
406 lines
15 KiB
C#
406 lines
15 KiB
C#
/* Copyright (c) Citrix Systems Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms,
|
|
* with or without modification, are permitted provided
|
|
* that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above
|
|
* copyright notice, this list of conditions and the
|
|
* following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the
|
|
* following disclaimer in the documentation and/or other
|
|
* materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
|
|
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Windows.Forms;
|
|
using XenAdmin.Core;
|
|
using XenAdmin.Network;
|
|
using XenAPI;
|
|
|
|
namespace XenAdmin.Controls
|
|
{
|
|
class FilterLocationToolStripDropDownButton : ToolStripDropDownButton
|
|
{
|
|
[Browsable(true)]
|
|
public event Action FilterChanged;
|
|
|
|
/// <summary>
|
|
/// Maintain a list of all the objects we currently have events on for clearing out on rebuild
|
|
/// </summary>
|
|
private List<IXenConnection> connectionsWithEvents = new List<IXenConnection>();
|
|
private List<Pool> poolsWithEvents = new List<Pool>();
|
|
private List<Host> hostsWithEvents = new List<Host>();
|
|
/// <summary>
|
|
/// Store only host check states because the pools can be in an
|
|
/// indeterminate state.
|
|
/// </summary>
|
|
private Dictionary<string, bool> HostCheckStates = new Dictionary<string, bool>();
|
|
private readonly CollectionChangeEventHandler m_hostCollectionChangedWithInvoke;
|
|
private bool inFilterListUpdate;
|
|
private bool retryFilterListUpdate;
|
|
|
|
private ToolStripMenuItem toolStripMenuItemAll;
|
|
|
|
public FilterLocationToolStripDropDownButton()
|
|
{
|
|
ConnectionsManager.XenConnections.CollectionChanged += XenConnections_CollectionChanged;
|
|
m_hostCollectionChangedWithInvoke = Program.ProgramInvokeHandler(Host_CollectionChanged);
|
|
}
|
|
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
base.Dispose(disposing);
|
|
DeregisterEvents();
|
|
ConnectionsManager.XenConnections.CollectionChanged -= XenConnections_CollectionChanged;
|
|
}
|
|
|
|
private void OnFilterChanged()
|
|
{
|
|
if (FilterChanged != null)
|
|
FilterChanged();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Build the list of hosts to filter by for the first time and set all
|
|
/// of them to be checked
|
|
/// </summary>
|
|
public void InitializeHostList()
|
|
{
|
|
foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy)
|
|
{
|
|
foreach (Host h in c.Cache.Hosts)
|
|
HostCheckStates[h.uuid] = true;
|
|
}
|
|
}
|
|
|
|
public void RefreshLists()
|
|
{
|
|
BuildFilterList();
|
|
OnFilterChanged();
|
|
}
|
|
|
|
public void BuildFilterList()
|
|
{
|
|
Program.AssertOnEventThread();
|
|
|
|
if (inFilterListUpdate)
|
|
{
|
|
// queue up an update after the current one has finished,
|
|
// in case the update has missed the relevant change
|
|
retryFilterListUpdate = true;
|
|
return;
|
|
}
|
|
|
|
inFilterListUpdate = true;
|
|
|
|
try
|
|
{
|
|
DropDownItems.Clear();
|
|
DeregisterEvents();
|
|
RegisterEvents();
|
|
|
|
foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy)
|
|
{
|
|
Pool p = Helpers.GetPool(c);
|
|
|
|
if (p == null)// Stand alone host
|
|
{
|
|
foreach (Host h in c.Cache.Hosts)
|
|
{
|
|
var item = GenerateFilterItem(h, h.uuid);
|
|
item.Checked = HostCheckStates.ContainsKey(h.uuid);
|
|
DropDownItems.Add(item);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
DropDownItems.Add(GeneratePoolFilterItem(p));
|
|
}
|
|
|
|
if (DropDownItems.Count > 0)
|
|
{
|
|
toolStripMenuItemAll = new ToolStripMenuItem
|
|
{
|
|
Text = Messages.FILTER_SHOW_ALL,
|
|
Enabled = FilterIsOn
|
|
};
|
|
|
|
DropDownItems.AddRange(new ToolStripItem[]
|
|
{
|
|
new ToolStripSeparator(),
|
|
toolStripMenuItemAll
|
|
});
|
|
}
|
|
|
|
Enabled = DropDownItems.Count > 0;
|
|
}
|
|
finally
|
|
{
|
|
inFilterListUpdate = false;
|
|
if (retryFilterListUpdate)
|
|
{
|
|
// there was a request to update while we were building,
|
|
// rebuild in case we missed something
|
|
retryFilterListUpdate = false;
|
|
BuildFilterList();
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool HideByLocation(string hostUuid)
|
|
{
|
|
if (hostUuid == null)
|
|
return false;
|
|
|
|
if (HostCheckStates.ContainsKey(hostUuid) && !HostCheckStates[hostUuid])
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
public bool HideByLocation(List<string> hostUuids)
|
|
{
|
|
if (hostUuids.Count == 0)
|
|
return false;
|
|
|
|
return hostUuids.TrueForAll(uuid => HostCheckStates.ContainsKey(uuid) && !HostCheckStates[uuid]);
|
|
}
|
|
|
|
public bool FilterIsOn
|
|
{
|
|
get { return HostCheckStates.ContainsValue(false); }
|
|
}
|
|
|
|
private ToolStripMenuItem GeneratePoolFilterItem(Pool p)
|
|
{
|
|
List<ToolStripMenuItem> subItems = new List<ToolStripMenuItem>();
|
|
|
|
foreach (Host h in p.Connection.Cache.Hosts)
|
|
{
|
|
var hostItem = GenerateFilterItem(h, h.uuid);
|
|
hostItem.Checked = HostCheckStates.ContainsKey(h.uuid);
|
|
subItems.Add(hostItem);
|
|
}
|
|
|
|
var poolItem = GenerateFilterItem(p, p.uuid);
|
|
poolItem.DropDownItems.AddRange(subItems.ToArray());
|
|
poolItem.CheckState = subItems.TrueForAll(item => item.Checked)
|
|
? CheckState.Checked
|
|
: subItems.TrueForAll(item => !item.Checked)
|
|
? CheckState.Unchecked
|
|
: CheckState.Indeterminate;
|
|
poolItem.DropDownItemClicked += poolItem_DropDownItemClicked;
|
|
return poolItem;
|
|
}
|
|
|
|
private ToolStripMenuItem GenerateFilterItem(IXenObject xenObject, string xenObjectUuid)
|
|
{
|
|
var item = new ToolStripMenuItem
|
|
{
|
|
Text = Helpers.GetName(xenObject),
|
|
Tag = xenObjectUuid
|
|
};
|
|
return item;
|
|
}
|
|
|
|
private void RegisterEvents()
|
|
{
|
|
foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy)
|
|
{
|
|
c.ConnectionStateChanged += connection_ConnectionStateChanged;
|
|
c.Cache.RegisterCollectionChanged<Host>(m_hostCollectionChangedWithInvoke);
|
|
c.CachePopulated += connection_CachePopulated;
|
|
connectionsWithEvents.Add(c);
|
|
|
|
foreach (var pool in c.Cache.Pools)
|
|
{
|
|
pool.PropertyChanged += pool_PropertyChanged;
|
|
poolsWithEvents.Add(pool);
|
|
}
|
|
|
|
foreach (Host host in c.Cache.Hosts)
|
|
{
|
|
RegisterHostEvents(host);
|
|
hostsWithEvents.Add(host);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void DeregisterEvents()
|
|
{
|
|
foreach (IXenConnection c in connectionsWithEvents)
|
|
{
|
|
c.ConnectionStateChanged -= connection_ConnectionStateChanged;
|
|
c.Cache.DeregisterCollectionChanged<Host>(m_hostCollectionChangedWithInvoke);
|
|
c.CachePopulated -= connection_CachePopulated;
|
|
}
|
|
|
|
foreach (var pool in poolsWithEvents)
|
|
pool.PropertyChanged -= pool_PropertyChanged;
|
|
|
|
foreach (Host h in hostsWithEvents)
|
|
DeregisterHostEvents(h);
|
|
|
|
connectionsWithEvents.Clear();
|
|
poolsWithEvents.Clear();
|
|
hostsWithEvents.Clear();
|
|
}
|
|
|
|
private void RegisterHostEvents(Host host)
|
|
{
|
|
Host_metrics metrics = host.Connection.Resolve(host.metrics);
|
|
if (metrics != null)
|
|
metrics.PropertyChanged += hostMetrics_PropertyChanged;
|
|
host.PropertyChanged += host_PropertyChanged;
|
|
}
|
|
|
|
private void DeregisterHostEvents(Host host)
|
|
{
|
|
Host_metrics metrics = host.Connection.Resolve(host.metrics);
|
|
if (metrics != null)
|
|
metrics.PropertyChanged -= hostMetrics_PropertyChanged;
|
|
host.PropertyChanged -= host_PropertyChanged;
|
|
}
|
|
|
|
private void connection_ConnectionStateChanged(object sender, EventArgs e)
|
|
{
|
|
Program.Invoke(Program.MainWindow, RefreshLists);
|
|
}
|
|
|
|
private void connection_CachePopulated(object sender, EventArgs e)
|
|
{
|
|
Program.Invoke(Program.MainWindow, RefreshLists);
|
|
}
|
|
|
|
private void XenConnections_CollectionChanged(object sender, CollectionChangeEventArgs e)
|
|
{
|
|
if (e.Action == CollectionChangeAction.Add)
|
|
{
|
|
IXenConnection connection = e.Element as IXenConnection;
|
|
|
|
foreach (Host host in connection.Cache.Hosts)
|
|
HostCheckStates[host.uuid] = true;
|
|
}
|
|
}
|
|
|
|
private void Host_CollectionChanged(object sender, CollectionChangeEventArgs e)
|
|
{
|
|
if (e.Action == CollectionChangeAction.Add)
|
|
HostCheckStates[((Host)e.Element).uuid] = true;
|
|
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
private void pool_PropertyChanged(object obj, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "other_config" || e.PropertyName == "name_label")
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
private void hostMetrics_PropertyChanged(object sender, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "live")
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
private void host_PropertyChanged(object sender, PropertyChangedEventArgs e)
|
|
{
|
|
if (e.PropertyName == "name_label" || e.PropertyName == "metrics")
|
|
Program.Invoke(Parent, RefreshLists);
|
|
}
|
|
|
|
protected override void OnDropDownItemClicked(ToolStripItemClickedEventArgs e)
|
|
{
|
|
//this method pertains to pool or stand alone host items
|
|
base.OnDropDownItemClicked(e);
|
|
HandleItemClicked(e.ClickedItem as ToolStripMenuItem);
|
|
}
|
|
|
|
private void poolItem_DropDownItemClicked(object sender, ToolStripItemClickedEventArgs e)
|
|
{
|
|
HandleItemClicked(e.ClickedItem as ToolStripMenuItem);
|
|
}
|
|
|
|
private void HandleItemClicked(ToolStripMenuItem item)
|
|
{
|
|
if (item == null)
|
|
return;
|
|
|
|
string uuid = (string)item.Tag;
|
|
|
|
if (item.HasDropDownItems)
|
|
{
|
|
//this is a pool node
|
|
|
|
item.CheckState = item.CheckState == CheckState.Checked
|
|
? CheckState.Unchecked
|
|
: CheckState.Checked;
|
|
|
|
foreach (ToolStripMenuItem child in item.DropDownItems)
|
|
{
|
|
child.Checked = item.Checked;
|
|
string hostUuid = (string)child.Tag;
|
|
HostCheckStates[hostUuid] = child.Checked;
|
|
}
|
|
|
|
toolStripMenuItemAll.Enabled = FilterIsOn;
|
|
}
|
|
else if (item == toolStripMenuItemAll)
|
|
{
|
|
toolStripMenuItemAll.Enabled = false;
|
|
InitializeHostList();
|
|
BuildFilterList();
|
|
}
|
|
else
|
|
{
|
|
//this is a host node
|
|
|
|
item.Checked = !item.Checked;
|
|
HostCheckStates[uuid] = item.Checked;
|
|
|
|
ToolStripMenuItem poolItem = item.OwnerItem as ToolStripMenuItem;
|
|
|
|
if (poolItem != null)
|
|
{
|
|
//this is not a standalone host; change the parent pool's check state
|
|
|
|
var itemArray = new ToolStripMenuItem[poolItem.DropDownItems.Count];
|
|
poolItem.DropDownItems.CopyTo(itemArray, 0);
|
|
|
|
poolItem.CheckState = Array.TrueForAll(itemArray, i => i.Checked)
|
|
? CheckState.Checked
|
|
: Array.TrueForAll(itemArray, i => !i.Checked)
|
|
? CheckState.Unchecked
|
|
: CheckState.Indeterminate;
|
|
}
|
|
|
|
toolStripMenuItemAll.Enabled = FilterIsOn;
|
|
}
|
|
|
|
OnFilterChanged();
|
|
}
|
|
}
|
|
}
|