/* 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, String[]> vms = Host.get_vms_which_prevent_evacuation(session, Host.opaque_ref); foreach (KeyValuePair, String[]> kvp in vms) { String[] exception = kvp.Value; XenRef 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 vmRef, string[] exception) { try { System.Diagnostics.Trace.Assert(exception.Length > 0); VM vm; switch (exception[0]) { case Failure.VM_REQUIRES_SR: vm = connection.Resolve(vmRef); if (vm == null) throw new NullReferenceException(Failure.VM_REQUIRES_SR); XenRef srRef = new XenRef(exception[2]); SR sr = connection.Resolve(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(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(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(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(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 netRef = new XenRef(exception[2]); XenAPI.Network network = connection.Resolve(netRef); if (network == null) throw new NullReferenceException(Failure.VM_REQUIRES_NETWORK); return new VMCannotSeeNetwork(this, vm, network); default: throw new NullReferenceException(exception[0]); } } catch (Exception e) { log.Debug("Exception parsing exception", e); log.Debug(e, e); throw new Failure(new List(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; } } } }