xenadmin/XenAdmin/Commands/Controls/WlbRecommendations.cs

230 lines
8.5 KiB
C#
Raw Normal View History

/* 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.Text;
using XenAPI;
using System.Collections.ObjectModel;
using XenAdmin.Core;
namespace XenAdmin.Commands
{
/// <summary>
/// Performs the WLB recommendations for the Start-On, Resume-On and Migrate menu items.
/// </summary>
internal class WlbRecommendations
{
private readonly static log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
private readonly ReadOnlyCollection<VM> _vms;
private readonly Session _session;
private bool _initialized;
private readonly Dictionary<VM, Dictionary<XenRef<Host>, string[]>> _recommendations = new Dictionary<VM, Dictionary<XenRef<Host>, string[]>>();
private bool _isError;
/// <summary>
/// Initializes a new instance of the <see cref="WlbRecommendations"/> class.
/// </summary>
/// <param name="vms">The VMs that the recommendations are required for.</param>
/// <param name="session">The session.</param>
public WlbRecommendations(IEnumerable<VM> vms, Session session)
{
Util.ThrowIfEnumerableParameterNullOrEmpty(vms, "vms");
Util.ThrowIfParameterNull(session, "session");
_vms = new ReadOnlyCollection<VM>(new List<VM>(vms));
_session = session;
}
/// <summary>
/// Calls VM.retrieve_wlb_recommendations for each of the VMs specified in the constructor.
/// </summary>
public void Initialize()
{
if (_initialized)
{
throw new InvalidOperationException("Already initialized");
}
_initialized = true;
if (Helpers.WlbEnabled(_vms[0].Connection))
{
try
{
foreach (VM vm in _vms)
{
_recommendations[vm] = VM.retrieve_wlb_recommendations(_session, vm.opaque_ref);
}
}
catch (Exception e)
{
log.Error("Error getting WLB recommendations", e);
_isError = true;
}
}
else
{
_isError = true;
}
}
/// <summary>
/// Gets a value indicating whether an exception was thrown when calling VM.retrieve_wlb_recommendations.
/// </summary>
/// <value><c>true</c> if an exception was thrown; otherwise, <c>false</c>.</value>
public bool IsError
{
get
{
return _isError;
}
}
private void Verify()
{
if (_isError)
{
throw new InvalidOperationException("There was an error getting the WLB recommendations.");
}
if (!_initialized)
{
throw new InvalidOperationException("Initialize() has not been called.");
}
}
public Host GetOptimalServer(VM vm)
{
Verify();
double highStars = 0;
Host recHost = null;
foreach (KeyValuePair<XenRef<Host>, string[]> rec in _recommendations[vm])
{
if (rec.Value.Length > 0 && rec.Value[0].Trim().ToLower() == "wlb")
{
// WLB: stars equal to highStars handles when there are only two hosts and vm resides on a host with high stars
double stars = Helpers.ParseStringToDouble(rec.Value[1].Trim(), 0);
if ((stars >= highStars) && (rec.Key.opaque_ref != vm.resident_on.opaque_ref))
{
highStars = stars;
recHost = vm.Connection.Resolve(rec.Key);
}
}
}
return recHost;
}
public WlbRecommendation GetStarRating(Host host)
{
Verify();
List<double> starRatings = new List<double>();
Dictionary<VM, bool> canExecutes = new Dictionary<VM, bool>();
Dictionary<VM, string> cantExecuteReasons = new Dictionary<VM, string>();
foreach (VM vm in _vms)
{
Host residentHost = vm.Connection.Resolve(vm.resident_on);
string[] rec = new string[0];
if (residentHost != null && residentHost.opaque_ref == host.opaque_ref)
{
cantExecuteReasons[vm] = Messages.HOST_MENU_CURRENT_SERVER;
canExecutes[vm] = false;
}
else if (_recommendations[vm].TryGetValue(new XenRef<Host>(host.opaque_ref), out rec))
{
if (rec.Length > 0 && rec[0].Trim().ToLower() == "wlb")
{
double stars = 0;
ParseStarRating(rec, out stars);
canExecutes[vm] = true;
starRatings.Add(stars);
}
else
{
cantExecuteReasons[vm] = new Failure(rec).ShortMessage;
canExecutes[vm] = false;
}
}
else
{
cantExecuteReasons[vm] = FriendlyErrorNames.HOST_NOT_LIVE_SHORT;
canExecutes[vm] = false;
}
}
double averageStarRating = 0.0;
foreach (double s in starRatings)
{
averageStarRating += s;
}
averageStarRating /= starRatings.Count;
return new WlbRecommendation(canExecutes, averageStarRating, cantExecuteReasons);
}
private static bool ParseStarRating(string[] rec, out double starRating)
{
// recommedation string[] format examples:
// WLB; 0.0; zero_score_reason
// WLB; 3.4
// xapiError(such as the assert can boot failsthe);detail;detail
starRating = 0;
if (rec != null && rec.Length > 1 && rec[0].Trim().ToLower() == "wlb")
{
starRating = Helpers.ParseStringToDouble(rec[1], 0);
return true;
}
return false;
}
internal class WlbRecommendation
{
public readonly Dictionary<VM, bool> CanExecuteByVM;
public readonly double StarRating;
public readonly Dictionary<VM, string> CantExecuteReasons;
public WlbRecommendation(Dictionary<VM, bool> canExecuteByVM, double starRating, Dictionary<VM, string> cantExecuteReasons)
{
CanExecuteByVM = canExecuteByVM;
StarRating = starRating;
CantExecuteReasons = cantExecuteReasons;
}
}
}
}