mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2025-01-20 23:39:51 +01:00
a8efbbfba2
the action was failing to create a metadata session; the latter was due to accessing directly properties of the xmlrpc proxy which is null since we use the jsonrpc backend. Also fixed some more areas where the same might occur. Signed-off-by: Konstantina Chremmou <konstantina.chremmou@citrix.com>
298 lines
10 KiB
C#
298 lines
10 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.Linq;
|
|
using XenAdmin.Core;
|
|
using XenAdmin.Model;
|
|
|
|
namespace XenAPI
|
|
{
|
|
public partial class Pool : IComparable<Pool>, IEquatable<Pool>
|
|
{
|
|
public override string ToString()
|
|
{
|
|
return Name();
|
|
}
|
|
|
|
public override string Description()
|
|
{
|
|
return name_description;
|
|
}
|
|
|
|
public override string Name()
|
|
{
|
|
if (name_label != "")
|
|
return name_label;
|
|
|
|
if (Connection == null)
|
|
return string.Empty;
|
|
|
|
Host master = Connection.Resolve(this.master);
|
|
return master == null ? "" : master.Name();
|
|
}
|
|
|
|
internal override string LocationString()
|
|
{
|
|
return string.Empty;
|
|
}
|
|
|
|
public List<SR> GetHAHeartbeatSRs()
|
|
{
|
|
List<SR> result = new List<SR>();
|
|
|
|
foreach (string opaqueRef in ha_statefiles)
|
|
{
|
|
XenRef<VDI> vdiRef = new XenRef<VDI>(opaqueRef);
|
|
VDI vdi = Connection.Resolve(vdiRef);
|
|
if (vdi == null)
|
|
continue;
|
|
SR sr = vdi.Connection.Resolve(vdi.SR);
|
|
if (sr != null)
|
|
result.Add(sr);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public bool IsMasterUpgraded()
|
|
{
|
|
Host master = Helpers.GetMaster(this);
|
|
foreach (var host in this.Connection.Cache.Hosts)
|
|
{
|
|
if (host.LongProductVersion() != master.LongProductVersion())
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public bool IsFreeLicenseOrExpired
|
|
{
|
|
get
|
|
{
|
|
return Connection.Cache.Hosts.Any(h => h.IsFreeLicenseOrExpired());
|
|
}
|
|
}
|
|
|
|
public string LicenseString()
|
|
{
|
|
var hosts = new List<Host>(Connection.Cache.Hosts);
|
|
foreach (Host.Edition edition in Enum.GetValues(typeof(Host.Edition)))
|
|
{
|
|
Host.Edition edition1 = edition;
|
|
Host host = hosts.Find(h => Host.GetEdition(h.edition) == edition1);
|
|
|
|
if (host != null)
|
|
{
|
|
return Helpers.GetFriendlyLicenseName(host);
|
|
}
|
|
}
|
|
return PropertyManager.GetFriendlyName("Label-host.edition-free");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determine whether the given pool is a visible pool, i.e. not a pool-of-one.
|
|
/// </summary>
|
|
public bool IsVisible()
|
|
{
|
|
return Connection != null && (name_label != "" || Connection.Cache.HostCount > 1);
|
|
}
|
|
|
|
private const string ROLLING_UPGRADE_IN_PROGRESS = "rolling_upgrade_in_progress";
|
|
private const string FORBID_RPU_FOR_HCI = "hci-forbid-rpu";
|
|
private const string FAULT_TOLERANCE_LIMIT_FOR_HCI = "hci-limit-fault-tolerance";
|
|
private const string FORBID_UPDATE_AUTO_RESTARTS = "hci-forbid-update-auto-restart";
|
|
|
|
public bool RollingUpgrade()
|
|
{
|
|
return other_config != null && other_config.ContainsKey(ROLLING_UPGRADE_IN_PROGRESS);
|
|
}
|
|
|
|
public bool IsUpgradeForbidden()
|
|
{
|
|
return other_config != null && other_config.ContainsKey(FORBID_RPU_FOR_HCI);
|
|
}
|
|
|
|
public bool IsAutoUpdateRestartsForbidden()
|
|
{
|
|
return other_config != null && other_config.ContainsKey(FORBID_UPDATE_AUTO_RESTARTS);
|
|
}
|
|
|
|
public static long GetMaximumTolerableHostFailures(Session session, Dictionary<XenRef<VM>, string> config)
|
|
{
|
|
long max = ha_compute_hypothetical_max_host_failures_to_tolerate(session, config);
|
|
long result;
|
|
Pool p = Helpers.GetPoolOfOne(session.Connection);
|
|
|
|
if (p!= null && p.other_config != null
|
|
&& p.other_config.ContainsKey(FAULT_TOLERANCE_LIMIT_FOR_HCI)
|
|
&& long.TryParse(p.other_config[FAULT_TOLERANCE_LIMIT_FOR_HCI], out result)
|
|
&& result <= max)//extra check in case the number set is accidentally wrong
|
|
{
|
|
return result;
|
|
}
|
|
|
|
return max;
|
|
}
|
|
|
|
public string GetXCPluginSecret(string plugin_name, IXenObject obj)
|
|
{
|
|
return Get(gui_config, XCPluginSecretName(plugin_name, obj));
|
|
}
|
|
|
|
public void RemoveXCPluginSecret(Session session, string plugin_name, IXenObject obj)
|
|
{
|
|
Pool.remove_from_gui_config(session, opaque_ref, XCPluginSecretName(plugin_name, obj));
|
|
}
|
|
|
|
public void SetXCPluginSecret(Session session, string plugin_name, IXenObject obj, string secret_uuid)
|
|
{
|
|
string n = XCPluginSecretName(plugin_name, obj);
|
|
Pool.remove_from_gui_config(session, opaque_ref, n);
|
|
Pool.add_to_gui_config(session, opaque_ref, n, secret_uuid);
|
|
}
|
|
|
|
private string XCPluginSecretName(string plugin_name, IXenObject obj)
|
|
{
|
|
return string.Format("XC_PLUGIN_SECRET_{0}_{1}_{2}", obj.Connection.Username, plugin_name, Helpers.GetUuid(obj));
|
|
}
|
|
|
|
// Whether the vSwitch Controller appears to be configured.
|
|
// (Note that we can't tell whether it's actually working properly through the API).
|
|
public bool vSwitchController()
|
|
{
|
|
if (string.IsNullOrEmpty(vswitch_controller))
|
|
return false;
|
|
|
|
foreach (Host h in Connection.Cache.Hosts)
|
|
{
|
|
if (Host.RestrictVSwitchController(h) ||
|
|
!h.software_version.ContainsKey("network_backend") ||
|
|
h.software_version["network_backend"] != "openvswitch")
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public bool HasSriovNic()
|
|
{
|
|
return Connection.Cache.PIFs.Any(pif => pif.SriovCapable());
|
|
}
|
|
|
|
public List<XenAPI.Host> HostsToUpgrade()
|
|
{
|
|
//First one to upgrade has to be the master
|
|
var master = Helpers.GetMaster(Connection);
|
|
|
|
List<XenAPI.Host> result = IsMasterUpgraded()
|
|
? Connection.Cache.Hosts.Where(host => host.LongProductVersion() != master.LongProductVersion()).ToList()
|
|
: Connection.Cache.Hosts.ToList();
|
|
result.Sort();
|
|
|
|
return result;
|
|
}
|
|
|
|
public bool IsPoolFullyUpgraded()
|
|
{
|
|
Host master = Helpers.GetMaster(this);
|
|
|
|
foreach (var host in this.Connection.Cache.Hosts)
|
|
{
|
|
if (host.LongProductVersion() != master.LongProductVersion())
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
public Host SmallerVersionHost()
|
|
{
|
|
Host hostWithSmallerVersion = null;
|
|
foreach (var host in this.Connection.Cache.Hosts)
|
|
{
|
|
if (hostWithSmallerVersion == null)
|
|
hostWithSmallerVersion = host;
|
|
else if (Helpers.productVersionCompare(hostWithSmallerVersion.ProductVersion(), host.ProductVersion()) > 0)
|
|
hostWithSmallerVersion = host;
|
|
}
|
|
return hostWithSmallerVersion;
|
|
}
|
|
|
|
public virtual int CpuSockets()
|
|
{
|
|
return Connection.Cache.Hosts.Sum(h => h.CpuSockets());
|
|
}
|
|
|
|
public bool HasGpu()
|
|
{
|
|
return Connection.Cache.PGPUs.Length > 0;
|
|
}
|
|
|
|
public bool HasVGpu()
|
|
{
|
|
return HasGpu() && Connection.Cache.PGPUs.Any(pGpu => pGpu.HasVGpu());
|
|
}
|
|
|
|
/// <summary>
|
|
/// ssl_legacy is true if any host in the pool is in legacy mode
|
|
/// </summary>
|
|
public bool ssl_legacy()
|
|
{
|
|
return Connection.Cache.Hosts.Any(h => h.ssl_legacy);
|
|
}
|
|
|
|
#region Health Check settings
|
|
|
|
public HealthCheckSettings HealthCheckSettings()
|
|
{
|
|
return new HealthCheckSettings(health_check_config);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IEquatable<Pool> Members
|
|
|
|
/// <summary>
|
|
/// Indicates whether the current object is equal to the specified object. This calls the implementation from XenObject.
|
|
/// This implementation is required for ToStringWrapper.
|
|
/// </summary>
|
|
public bool Equals(Pool other)
|
|
{
|
|
return base.Equals(other);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|