2017-01-16 20:59:50 +01:00
|
|
|
/* Copyright (c) Citrix Systems, Inc.
|
2013-06-24 13:41:48 +02:00
|
|
|
* 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.ComponentModel;
|
|
|
|
using System.Linq;
|
2019-11-28 01:44:19 +01:00
|
|
|
using System.Threading;
|
2019-10-27 16:52:15 +01:00
|
|
|
using XenAdmin.Network;
|
2013-06-24 13:41:48 +02:00
|
|
|
using XenAPI;
|
|
|
|
|
|
|
|
namespace XenAdmin.Dialogs
|
|
|
|
{
|
|
|
|
public interface ILicenseStatus : IDisposable
|
|
|
|
{
|
|
|
|
LicenseStatus.HostState CurrentState { get; }
|
|
|
|
Host.Edition LicenseEdition { get; }
|
|
|
|
TimeSpan LicenseExpiresIn { get; }
|
|
|
|
TimeSpan LicenseExpiresExactlyIn { get; }
|
|
|
|
DateTime? ExpiryDate { get; }
|
2019-11-28 01:44:19 +01:00
|
|
|
event Action ItemUpdated;
|
2013-06-24 13:41:48 +02:00
|
|
|
bool Updated { get; }
|
|
|
|
void BeginUpdate();
|
2019-11-28 01:44:19 +01:00
|
|
|
Host LicensedHost { get; }
|
2014-10-28 18:02:55 +01:00
|
|
|
string LicenseEntitlements { get; }
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public class LicenseStatus : ILicenseStatus
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
|
|
|
|
|
2013-06-24 13:41:48 +02:00
|
|
|
public enum HostState
|
|
|
|
{
|
|
|
|
Unknown,
|
|
|
|
Expired,
|
|
|
|
ExpiresSoon,
|
|
|
|
RegularGrace,
|
|
|
|
UpgradeGrace,
|
|
|
|
Licensed,
|
|
|
|
PartiallyLicensed,
|
|
|
|
Free,
|
|
|
|
Unavailable
|
|
|
|
}
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
private readonly EventHandlerList _events = new EventHandlerList();
|
|
|
|
|
2013-06-24 13:41:48 +02:00
|
|
|
private const string StatusUpdatedEventKey = "LicenseStatusStatusUpdatedEventKey";
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
public Host LicensedHost { get; private set; }
|
2013-06-24 13:41:48 +02:00
|
|
|
|
2016-08-03 22:07:04 +02:00
|
|
|
public static bool IsInfinite(TimeSpan span)
|
|
|
|
{
|
|
|
|
return span.TotalDays >= 3653;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static bool IsGraceLicence(TimeSpan span)
|
|
|
|
{
|
|
|
|
return span.TotalDays < 30;
|
|
|
|
}
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
private IXenObject XenObject { get; }
|
2013-06-24 13:41:48 +02:00
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
public bool Updated { get; private set; }
|
2013-06-24 13:41:48 +02:00
|
|
|
|
|
|
|
public LicenseStatus(IXenObject xo)
|
|
|
|
{
|
|
|
|
SetDefaultOptions();
|
|
|
|
XenObject = xo;
|
|
|
|
|
2019-01-31 00:00:56 +01:00
|
|
|
if (XenObject is Host host)
|
2019-11-28 01:44:19 +01:00
|
|
|
LicensedHost = host;
|
2019-01-31 00:00:56 +01:00
|
|
|
if (XenObject is Pool pool)
|
2013-06-24 13:41:48 +02:00
|
|
|
SetMinimumLicenseValueHost(pool);
|
2017-02-10 16:55:09 +01:00
|
|
|
|
|
|
|
if (XenObject != null)
|
|
|
|
{
|
|
|
|
XenObject.Connection.ConnectionStateChanged -= Connection_ConnectionStateChanged;
|
|
|
|
XenObject.Connection.ConnectionStateChanged += Connection_ConnectionStateChanged;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-27 16:52:15 +01:00
|
|
|
private void Connection_ConnectionStateChanged(IXenConnection conn)
|
2017-02-10 16:55:09 +01:00
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
if (LicensedHost != null)
|
2017-02-10 16:55:09 +01:00
|
|
|
{
|
|
|
|
TriggerStatusUpdatedEvent();
|
|
|
|
}
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private void SetMinimumLicenseValueHost(Pool pool)
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
LicensedHost = pool.Connection.Resolve(pool.master);
|
2013-06-24 13:41:48 +02:00
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
if(LicensedHost == null)
|
2013-06-24 13:41:48 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
foreach (Host host in pool.Connection.Cache.Hosts)
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
if(host.LicenseExpiryUTC() < LicensedHost.LicenseExpiryUTC())
|
|
|
|
LicensedHost = host;
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void SetDefaultOptions()
|
|
|
|
{
|
|
|
|
CurrentState = HostState.Unknown;
|
|
|
|
Updated = false;
|
|
|
|
LicenseExpiresExactlyIn = new TimeSpan();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void BeginUpdate()
|
|
|
|
{
|
|
|
|
SetDefaultOptions();
|
2019-11-28 01:44:19 +01:00
|
|
|
ThreadPool.QueueUserWorkItem(GetServerTime, LicensedHost);
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
private void GetServerTime(object state)
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
Host host = state as Host;
|
|
|
|
if (host?.Connection?.Session == null)
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
log.Error("Will not fetch server time: host or connection could not be resolved");
|
|
|
|
return;
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
try
|
|
|
|
{
|
|
|
|
//Note we're using the get_servertime call which returns the UTC time
|
|
|
|
var serverTime = Host.get_servertime(host.Connection.Session, host.opaque_ref);
|
|
|
|
|
|
|
|
if (LicensedHost != null)
|
|
|
|
{
|
|
|
|
//ServerTime is UTC
|
|
|
|
DateTime currentRefTime = serverTime;
|
|
|
|
LicenseExpiresExactlyIn = LicensedHost.LicenseExpiryUTC().Subtract(currentRefTime);
|
|
|
|
|
|
|
|
CurrentState = CalculateCurrentState();
|
|
|
|
Updated = true;
|
|
|
|
|
|
|
|
TriggerStatusUpdatedEvent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
|
|
|
log.Error($"Failed to fetch server time for host {host.name_label}: ", e);
|
|
|
|
}
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private void TriggerStatusUpdatedEvent()
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
if (_events[StatusUpdatedEventKey] is Action handler)
|
|
|
|
handler.Invoke();
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private bool InRegularGrace
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
return LicensedHost.license_params != null && LicensedHost.license_params.ContainsKey("grace") && LicenseExpiresIn.Ticks > 0 && LicensedHost.license_params["grace"] == "regular grace";
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private bool InUpgradeGrace
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
return LicensedHost.license_params != null && LicensedHost.license_params.ContainsKey("grace") && LicenseExpiresIn.Ticks > 0 && LicensedHost.license_params["grace"] == "upgrade grace";
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-09 18:18:15 +02:00
|
|
|
internal static bool PoolIsMixedFreeAndExpiring(IXenObject xenObject)
|
|
|
|
{
|
|
|
|
if (xenObject is Pool)
|
|
|
|
{
|
|
|
|
if (xenObject.Connection.Cache.Hosts.Length == 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
int freeCount = xenObject.Connection.Cache.Hosts.Count(h => Host.GetEdition(h.edition) == Host.Edition.Free);
|
|
|
|
if (freeCount == 0 || freeCount < xenObject.Connection.Cache.Hosts.Length)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
var expiryGroups = from Host h in xenObject.Connection.Cache.Hosts
|
2017-09-03 04:33:29 +02:00
|
|
|
let exp = h.LicenseExpiryUTC()
|
2013-08-09 18:18:15 +02:00
|
|
|
group h by exp
|
|
|
|
into g
|
|
|
|
select new { ExpiryDate = g.Key, Hosts = g };
|
|
|
|
|
|
|
|
if(expiryGroups.Count() > 1)
|
|
|
|
{
|
|
|
|
expiryGroups.OrderBy(g => g.ExpiryDate);
|
|
|
|
if ((expiryGroups.ElementAt(1).ExpiryDate - expiryGroups.ElementAt(0).ExpiryDate).TotalDays > 30)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-08-09 18:17:24 +02:00
|
|
|
internal static bool PoolIsPartiallyLicensed(IXenObject xenObject)
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
2013-08-09 18:17:24 +02:00
|
|
|
if (xenObject is Pool)
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
2013-08-09 18:17:24 +02:00
|
|
|
if (xenObject.Connection.Cache.Hosts.Length == 1)
|
|
|
|
return false;
|
2013-06-24 13:41:48 +02:00
|
|
|
|
2013-08-09 18:17:24 +02:00
|
|
|
int freeCount = xenObject.Connection.Cache.Hosts.Count(h => Host.GetEdition(h.edition) == Host.Edition.Free);
|
|
|
|
return freeCount > 0 && freeCount < xenObject.Connection.Cache.Hosts.Length;
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
2013-08-09 18:17:24 +02:00
|
|
|
return false;
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
2013-08-09 18:17:24 +02:00
|
|
|
internal static bool PoolHasMixedLicenses(IXenObject xenObject)
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
if (xenObject is Pool pool)
|
2013-08-09 18:17:24 +02:00
|
|
|
{
|
|
|
|
if (xenObject.Connection.Cache.Hosts.Length == 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (xenObject.Connection.Cache.Hosts.Any(h => Host.GetEdition(h.edition) == Host.Edition.Free))
|
|
|
|
return false;
|
2013-06-24 13:41:48 +02:00
|
|
|
|
2013-08-09 18:17:24 +02:00
|
|
|
var licenseGroups = from Host h in xenObject.Connection.Cache.Hosts
|
|
|
|
let ed = Host.GetEdition(h.edition)
|
|
|
|
group h by ed;
|
|
|
|
|
|
|
|
return licenseGroups.Count() > 1;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
private HostState CalculateCurrentState()
|
|
|
|
{
|
2013-06-24 13:41:48 +02:00
|
|
|
if (ExpiryDate.HasValue && ExpiryDate.Value.Day == 1 && ExpiryDate.Value.Month == 1 && ExpiryDate.Value.Year == 1970)
|
|
|
|
{
|
|
|
|
return HostState.Unavailable;
|
|
|
|
}
|
|
|
|
|
2013-08-09 18:17:24 +02:00
|
|
|
if (PoolIsPartiallyLicensed(XenObject))
|
2013-06-24 13:41:48 +02:00
|
|
|
return HostState.PartiallyLicensed;
|
|
|
|
|
2017-05-05 18:03:45 +02:00
|
|
|
if (LicenseEdition == Host.Edition.Free)
|
|
|
|
return HostState.Free;
|
2016-08-03 22:07:04 +02:00
|
|
|
|
2017-05-05 18:03:45 +02:00
|
|
|
if (!IsGraceLicence(LicenseExpiresIn))
|
|
|
|
return HostState.Licensed;
|
2013-06-24 13:41:48 +02:00
|
|
|
|
2016-08-03 22:07:04 +02:00
|
|
|
if (IsInfinite(LicenseExpiresIn))
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
|
|
|
return HostState.Licensed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (LicenseExpiresIn.Ticks <= 0)
|
|
|
|
{
|
|
|
|
return HostState.Expired;
|
|
|
|
}
|
|
|
|
|
2016-08-03 22:07:04 +02:00
|
|
|
if (IsGraceLicence(LicenseExpiresIn))
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
|
|
|
if (InRegularGrace)
|
|
|
|
return HostState.RegularGrace;
|
|
|
|
if (InUpgradeGrace)
|
|
|
|
return HostState.UpgradeGrace;
|
|
|
|
|
|
|
|
return HostState.ExpiresSoon;
|
|
|
|
}
|
|
|
|
|
2013-06-27 15:40:11 +02:00
|
|
|
return LicenseEdition == Host.Edition.Free ? HostState.Free : HostState.Licensed;
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#region ILicenseStatus Members
|
2019-11-28 01:44:19 +01:00
|
|
|
public event Action ItemUpdated
|
2013-06-24 13:41:48 +02:00
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
add => _events.AddHandler(StatusUpdatedEventKey, value);
|
|
|
|
remove => _events.RemoveHandler(StatusUpdatedEventKey, value);
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
public Host.Edition LicenseEdition => Host.GetEdition(LicensedHost.edition);
|
2013-06-24 13:41:48 +02:00
|
|
|
|
|
|
|
public HostState CurrentState { get; private set; }
|
|
|
|
|
|
|
|
public TimeSpan LicenseExpiresExactlyIn { get; private set; }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// License expiry, just days, hrs, mins
|
|
|
|
/// </summary>
|
|
|
|
public TimeSpan LicenseExpiresIn
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return new TimeSpan(LicenseExpiresExactlyIn.Days, LicenseExpiresExactlyIn.Hours, LicenseExpiresExactlyIn.Minutes, 0, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public DateTime? ExpiryDate
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2019-11-28 01:44:19 +01:00
|
|
|
if (LicensedHost.license_params != null && LicensedHost.license_params.ContainsKey("expiry"))
|
|
|
|
return LicensedHost.LicenseExpiryUTC().ToLocalTime();
|
2013-06-24 13:41:48 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-28 18:02:55 +01:00
|
|
|
public string LicenseEntitlements
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2019-01-31 00:00:56 +01:00
|
|
|
if (CurrentState == HostState.Licensed)
|
2014-10-28 18:02:55 +01:00
|
|
|
{
|
2017-09-03 04:33:29 +02:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.EnterpriseFeaturesEnabled()))
|
2014-10-28 18:02:55 +01:00
|
|
|
return Messages.LICENSE_SUPPORT_AND_ENTERPRISE_FEATURES_ENABLED;
|
2017-09-03 04:33:29 +02:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.DesktopPlusFeaturesEnabled()))
|
2014-12-08 15:29:54 +01:00
|
|
|
return Messages.LICENSE_SUPPORT_AND_DESKTOP_PLUS_FEATURES_ENABLED;
|
2017-09-03 04:33:29 +02:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.DesktopFeaturesEnabled()))
|
2014-11-14 18:29:01 +01:00
|
|
|
return Messages.LICENSE_SUPPORT_AND_DESKTOP_FEATURES_ENABLED;
|
2017-12-18 14:22:17 +01:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.DesktopCloudFeaturesEnabled()))
|
|
|
|
return Messages.LICENSE_SUPPORT_AND_DESKTOP_CLOUD_FEATURES_ENABLED;
|
2017-09-03 04:33:29 +02:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.PremiumFeaturesEnabled()))
|
2016-02-08 11:10:01 +01:00
|
|
|
return Messages.LICENSE_SUPPORT_AND_PREMIUM_FEATURES_ENABLED;
|
2017-09-03 04:33:29 +02:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.StandardFeaturesEnabled()))
|
2016-03-09 17:14:05 +01:00
|
|
|
return Messages.LICENSE_SUPPORT_AND_STANDARD_FEATURES_ENABLED;
|
2017-09-03 04:33:29 +02:00
|
|
|
if (XenObject.Connection.Cache.Hosts.All(h => h.EligibleForSupport()))
|
2014-10-28 18:02:55 +01:00
|
|
|
return Messages.LICENSE_SUPPORT_AND_STANDARD_FEATURES_ENABLED;
|
|
|
|
return Messages.LICENSE_NOT_ELIGIBLE_FOR_SUPPORT;
|
|
|
|
}
|
|
|
|
|
2017-05-05 18:03:45 +02:00
|
|
|
if (CurrentState == HostState.Free)
|
2014-10-28 18:02:55 +01:00
|
|
|
{
|
|
|
|
return Messages.LICENSE_NOT_ELIGIBLE_FOR_SUPPORT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Messages.UNKNOWN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-28 17:42:27 +01:00
|
|
|
#endregion
|
|
|
|
|
2013-06-24 13:41:48 +02:00
|
|
|
#region IDisposable Members
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
private bool disposed;
|
|
|
|
public void Dispose(bool disposing)
|
|
|
|
{
|
|
|
|
if(!disposed)
|
|
|
|
{
|
|
|
|
if(disposing)
|
|
|
|
{
|
2017-02-10 16:55:09 +01:00
|
|
|
if (XenObject != null && XenObject.Connection != null)
|
|
|
|
XenObject.Connection.ConnectionStateChanged -= Connection_ConnectionStateChanged;
|
|
|
|
|
2019-11-28 01:44:19 +01:00
|
|
|
_events.Dispose();
|
2013-06-24 13:41:48 +02:00
|
|
|
}
|
|
|
|
disposed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
}
|
|
|
|
}
|