2013-06-24 13:41:48 +02:00
|
|
|
|
/* 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 XenAdmin.Network;
|
|
|
|
|
using XenAPI;
|
|
|
|
|
using XenAdmin.Diagnostics.Problems;
|
|
|
|
|
using XenAdmin.Core;
|
|
|
|
|
using XenAdmin.Diagnostics.Problems.VMProblem;
|
|
|
|
|
using XenAdmin.Diagnostics.Problems.HostProblem;
|
|
|
|
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
namespace XenAdmin.Diagnostics.Checks
|
|
|
|
|
{
|
|
|
|
|
public class AssertCanEvacuateCheck : Check
|
|
|
|
|
{
|
|
|
|
|
private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
|
|
|
|
|
|
|
|
|
|
public AssertCanEvacuateCheck(Host host)
|
|
|
|
|
: base(host)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected Problem CheckHost()
|
|
|
|
|
{
|
|
|
|
|
var residentVMs = Host.Connection.ResolveAll(Host.resident_VMs);
|
|
|
|
|
foreach (var residentVM in residentVMs)
|
|
|
|
|
{
|
|
|
|
|
if (residentVM.AutoPowerOn)
|
|
|
|
|
return new AutoStartEnabled(this, residentVM);
|
|
|
|
|
|
|
|
|
|
SR sr = residentVM.FindVMCDROMSR();
|
|
|
|
|
if (sr != null && sr.IsToolsSR)
|
|
|
|
|
return new ToolsCD(this, residentVM);
|
|
|
|
|
if (sr != null && sr.content_type == SR.Content_Type_ISO)
|
|
|
|
|
return new LocalCD(this, residentVM);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Session session = Host.Connection.DuplicateSession();
|
|
|
|
|
Dictionary<XenRef<VM>, String[]> vms =
|
|
|
|
|
Host.get_vms_which_prevent_evacuation(session, Host.opaque_ref);
|
|
|
|
|
|
|
|
|
|
foreach (KeyValuePair<XenRef<VM>, String[]> kvp in vms)
|
|
|
|
|
{
|
|
|
|
|
String[] exception = kvp.Value;
|
|
|
|
|
XenRef<VM> vmRef = kvp.Key;
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
// We can actually ignore some errors, indicated by null
|
|
|
|
|
return GetProblem(Host.Connection, vmRef, exception);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
log.Debug("Didn't recognise reason", e);
|
|
|
|
|
log.Debug(exception);
|
|
|
|
|
log.Debug(e, e);
|
|
|
|
|
|
|
|
|
|
VM vm = Host.Connection.Resolve(kvp.Key);
|
|
|
|
|
|
|
|
|
|
if (vm != null)
|
|
|
|
|
return new CannotMigrateVM(this, vm);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Problem GetProblem(IXenConnection connection, XenRef<VM> vmRef, string[] exception)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
System.Diagnostics.Trace.Assert(exception.Length > 0);
|
|
|
|
|
|
|
|
|
|
VM vm;
|
|
|
|
|
|
|
|
|
|
switch (exception[0])
|
|
|
|
|
{
|
|
|
|
|
case Failure.VM_REQUIRES_SR:
|
|
|
|
|
vm = connection.Resolve<VM>(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException(Failure.VM_REQUIRES_SR);
|
|
|
|
|
|
|
|
|
|
XenRef<SR> srRef = new XenRef<SR>(exception[2]);
|
|
|
|
|
SR sr = connection.Resolve<SR>(srRef);
|
|
|
|
|
|
|
|
|
|
if (sr == null)
|
|
|
|
|
throw new NullReferenceException(Failure.VM_REQUIRES_SR);
|
|
|
|
|
|
|
|
|
|
if (sr.content_type == SR.Content_Type_ISO)
|
|
|
|
|
{
|
|
|
|
|
return new LocalCD(this, vm);
|
|
|
|
|
}
|
|
|
|
|
else if (!sr.shared)
|
|
|
|
|
{
|
|
|
|
|
// Only show the problem if it is really local storage
|
|
|
|
|
// As the pbd-plug checks will pick up broken storage.
|
|
|
|
|
return new LocalStorage(this, vm);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
case Failure.VM_MISSING_PV_DRIVERS:
|
|
|
|
|
vm = connection.Resolve<VM>(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException(Failure.VM_MISSING_PV_DRIVERS);
|
|
|
|
|
|
|
|
|
|
return new NoPVDrivers(this, vm);
|
|
|
|
|
|
|
|
|
|
case "VM_OLD_PV_DRIVERS":
|
|
|
|
|
vm = connection.Resolve<VM>(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException("VM_OLD_PV_DRIVERS");
|
|
|
|
|
|
|
|
|
|
return new PVDriversOutOfDate(this, vm);
|
|
|
|
|
|
|
|
|
|
case Failure.NO_HOSTS_AVAILABLE:
|
|
|
|
|
//CA-63531: Boston server will come here in case of single host pool or standalone host
|
|
|
|
|
vm = connection.Resolve<VM>(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException(Failure.NO_HOSTS_AVAILABLE);
|
|
|
|
|
|
|
|
|
|
return new NoHosts(this, vm);
|
|
|
|
|
|
|
|
|
|
case Failure.HOST_NOT_ENOUGH_FREE_MEMORY:
|
|
|
|
|
vm = connection.Resolve<VM>(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException(Failure.HOST_NOT_ENOUGH_FREE_MEMORY);
|
|
|
|
|
|
|
|
|
|
Pool pool = Helpers.GetPool(vm.Connection);
|
|
|
|
|
|
|
|
|
|
if (pool == null || pool.Connection.Cache.HostCount == 1)
|
|
|
|
|
{
|
|
|
|
|
//CA-63531: Cowley server will come here in case of single host pool or standalone host
|
|
|
|
|
return new NoHosts(this, vm);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Host host = vm.Connection.Resolve(vm.resident_on);
|
|
|
|
|
return new NotEnoughMem(this, host);
|
|
|
|
|
|
|
|
|
|
case Failure.VM_REQUIRES_NETWORK:
|
|
|
|
|
vm = connection.Resolve(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException(Failure.VM_REQUIRES_NETWORK);
|
|
|
|
|
|
|
|
|
|
XenRef<XenAPI.Network> netRef = new XenRef<XenAPI.Network>(exception[2]);
|
|
|
|
|
XenAPI.Network network = connection.Resolve(netRef);
|
|
|
|
|
|
|
|
|
|
if (network == null)
|
|
|
|
|
throw new NullReferenceException(Failure.VM_REQUIRES_NETWORK);
|
|
|
|
|
|
|
|
|
|
return new VMCannotSeeNetwork(this, vm, network);
|
|
|
|
|
|
2013-12-11 11:41:54 +01:00
|
|
|
|
case Failure.VM_HAS_VGPU:
|
|
|
|
|
vm = connection.Resolve(vmRef);
|
|
|
|
|
|
|
|
|
|
if (vm == null)
|
|
|
|
|
throw new NullReferenceException(Failure.VM_HAS_VGPU);
|
|
|
|
|
|
|
|
|
|
return new VmHasVgpu(this, vm);
|
|
|
|
|
|
2013-06-24 13:41:48 +02:00
|
|
|
|
default:
|
|
|
|
|
throw new NullReferenceException(exception[0]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
log.Debug("Exception parsing exception", e);
|
|
|
|
|
log.Debug(e, e);
|
|
|
|
|
|
|
|
|
|
throw new Failure(new List<String>(exception));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override Problem RunCheck()
|
|
|
|
|
{
|
|
|
|
|
if (!Host.IsLive)
|
|
|
|
|
return new HostNotLive(this, Host);
|
|
|
|
|
return CheckHost();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string Description
|
|
|
|
|
{
|
|
|
|
|
get { return Messages.ASSERT_CAN_EVACUATE_CHECK_DESCRIPTION; }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|