/* 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 System.Linq;
using XenAPI;
using System.Windows.Forms;
using XenAdmin.Actions;
using XenAdmin.Core;
using XenAdmin.Commands.Controls;
using XenAdmin.Dialogs;
namespace XenAdmin.Commands
{
class DeleteVirtualDiskCommand : Command
{
///
/// Allows auto unplug and delete on running VMs
///
public bool AllowRunningVMDelete = false;
///
/// Allows deletion of the VDI when multiple VMs are using this VDI. If the VMs are running you also need to set AllowRunningVMDelete
///
public bool AllowMultipleVBDDelete = false;
public DeleteVirtualDiskCommand(IMainWindow mainWindow, IEnumerable selection)
: base(mainWindow, selection)
{
}
public override string ContextMenuText
{
get
{
var selection = GetSelection();
if (selection.Count > 1)
return Messages.MAINWINDOW_DELETE_OBJECTS;
return selection.AsXenObjects().All(v => v.is_a_snapshot)
? Messages.DELETE_SNAPSHOT_MENU_ITEM
: Messages.DELETE_VIRTUAL_DISK;
}
}
protected override bool CanExecuteCore(SelectedItemCollection selection)
{
return selection.AllItemsAre(CanExecute);
}
protected override bool ConfirmationRequired
{
get
{
// We always do some sort of confirmation for this delete
return true;
}
}
protected override string ConfirmationDialogText
{
get
{
if (!NeedMultipleWarnings)
{
// In the single warning type case we use the in built confirmation dialog to show a single warning
// if there are mixed vdi types then we will use the multiple warning dialog in confirm() override
SelectedItemCollection selectedItems = GetSelection();
VDI vdi = selectedItems[0].XenObject as VDI;
SR sr = vdi.Connection.Resolve(vdi.SR);
bool single = selectedItems.Count == 1;
var typ = vdi.VDIType();
switch (typ)
{
case VDI.FriendlyType.SNAPSHOT:
return single ? Messages.MESSAGEBOX_DELETE_SNAPSHOT : Messages.MESSAGEBOX_DELETE_SNAPSHOT_MULTIPLE;
case VDI.FriendlyType.ISO:
return single ? string.Format(Messages.MESSAGEBOX_DELETE_ISO, Helpers.GetName(vdi)) : Messages.MESSAGEBOX_DELETE_ISO_MULTIPLE;
case VDI.FriendlyType.SYSTEM_DISK:
return single ? Messages.MESSAGEBOX_DELETE_SYS_DISK : Messages.MESSAGEBOX_DELETE_SYS_DISK_MULTIPLE;
case VDI.FriendlyType.VIRTUAL_DISK:
return single ? Messages.MESSAGEBOX_DELETE_VD : Messages.MESSAGEBOX_DELETE_VD_MULTIPLE;
case VDI.FriendlyType.NONE:
return "";
}
}
return base.ConfirmationDialogText;
}
}
protected override string ConfirmationDialogTitle
{
get
{
if (!NeedMultipleWarnings)
{
// In the single warning type case we use the in built confirmation dialog to show a single warning
// if there are mixed vdi types then we will use the multiple warning dialog in confirm() override
SelectedItemCollection selectedItems = GetSelection();
VDI vdi = selectedItems[0].XenObject as VDI;
bool single = selectedItems.Count == 1;
var typ = vdi.VDIType();
switch (typ)
{
case VDI.FriendlyType.SNAPSHOT:
return single ? Messages.MESSAGEBOX_DELETE_SNAPSHOT_TITLE : Messages.MESSAGEBOX_DELETE_SNAPSHOTS_TITLE_MULTIPLE;
case VDI.FriendlyType.ISO:
return single ? Messages.MESSAGEBOX_DELETE_ISO_TITLE : Messages.MESSAGEBOX_DELETE_ISO_TITLE_MULTIPLE;
case VDI.FriendlyType.SYSTEM_DISK:
return single ? Messages.MESSAGEBOX_DELETE_SYS_DISK_TITLE : Messages.MESSAGEBOX_DELETE_SYS_DISK_TITLE_MULTIPLE;
case VDI.FriendlyType.VIRTUAL_DISK:
return single ? Messages.MESSAGEBOX_DELETE_VD_TITLE : Messages.MESSAGEBOX_DELETE_VD_TITLE_MUTLIPLE;
case VDI.FriendlyType.NONE:
return "";
}
}
return base.ConfirmationDialogTitle;
}
}
private bool? _needMultipleWarnings = null;
private bool NeedMultipleWarnings
{
get
{
// While just O(n) it does cast an entire list, so we cache the first result.
if (_needMultipleWarnings.HasValue)
return _needMultipleWarnings.Value;
_needMultipleWarnings = false;
SelectedItemCollection selection = GetSelection();
VDI.FriendlyType current = VDI.FriendlyType.NONE;
VDI.FriendlyType previous = VDI.FriendlyType.NONE;
if (selection.Count > 1)
{
for (int i = 0; i < selection.Count; i++)
{
VDI v = selection[i].XenObject as VDI;
if (v == null)
current = VDI.FriendlyType.NONE;
else
current = v.VDIType();
if (i > 0 && current != previous)
{
_needMultipleWarnings = true;
break;
}
previous = current;
}
}
return _needMultipleWarnings.Value;
}
}
protected override bool Confirm()
{
if (Program.RunInAutomatedTestMode)
return true;
if (NeedMultipleWarnings)
{
MultipleWarningDialog warningDialog = new MultipleWarningDialog(
Messages.MESSAGEBOX_DELETE_VD_TITLE_MUTLIPLE,
Messages.MULTI_VDI_DELETE_WARNING,
Messages.DELETE_ALL_BUTTON_LABEL);
SelectedItemCollection selectedItems = GetSelection();
List snapshots = new List();
List isos = new List();
List systemVDisks = new List();
List virtualDisks = new List();
foreach (VDI vdi in selectedItems.AsXenObjects())
{
var typ = vdi.VDIType();
switch (typ)
{
case VDI.FriendlyType.SNAPSHOT:
snapshots.Add(vdi);
break;
case VDI.FriendlyType.ISO:
isos.Add(vdi);
break;
case VDI.FriendlyType.SYSTEM_DISK:
systemVDisks.Add(vdi);
break;
case VDI.FriendlyType.VIRTUAL_DISK:
virtualDisks.Add(vdi);
break;
case VDI.FriendlyType.NONE:
break;
}
}
if (snapshots.Count == 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_SNAPSHOT_TITLE,
Messages.WARNING_DELETE_SNAPSHOT,
snapshots.ConvertAll(delegate(VDI v){return (IXenObject)v;}));
}
else if (snapshots.Count > 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_SNAPSHOTS_TITLE_MULTIPLE,
Messages.WARNING_DELETE_SNAPSHOT_MULTIPLE,
snapshots.ConvertAll(delegate(VDI v){return (IXenObject)v;}));
}
if (isos.Count == 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_ISO_TITLE,
Messages.WARNING_DELETE_ISO,
isos.ConvertAll(delegate(VDI v) { return (IXenObject)v; }));
}
else if (isos.Count > 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_ISO_TITLE_MULTIPLE,
Messages.WARNING_DELETE_ISO_MULTIPLE,
isos.ConvertAll(delegate(VDI v) { return (IXenObject)v; }));
}
if (systemVDisks.Count == 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_SYS_DISK_TITLE,
Messages.WARNING_DELETE_SYS_DISK,
systemVDisks.ConvertAll(delegate(VDI v) { return (IXenObject)v; }));
}
else if (systemVDisks.Count > 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_SYS_DISK_TITLE_MULTIPLE,
Messages.WARNING_DELETE_SYS_DISK_MULTIPLE,
systemVDisks.ConvertAll(delegate(VDI v) { return (IXenObject)v; }));
}
if (virtualDisks.Count == 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_VD_TITLE,
Messages.WARNING_DELETE_VD,
virtualDisks.ConvertAll(delegate(VDI v) { return (IXenObject)v; }));
}
else if (virtualDisks.Count > 1)
{
warningDialog.AddWarningMessage(
Messages.MESSAGEBOX_DELETE_VD_TITLE_MUTLIPLE,
Messages.WARNING_DELETE_VD_MULTIPLE,
virtualDisks.ConvertAll(delegate(VDI v) { return (IXenObject)v; }));
}
return warningDialog.ShowDialog(Parent) == DialogResult.Yes;
}
else
{
return base.Confirm();
}
}
protected bool CanExecute(VDI vdi)
{
if (vdi == null)
return false;
SR sr = vdi.Connection.Resolve(vdi.SR);
if (sr == null)
return false;
if (vdi.Locked)
return false;
if (sr.Physical())
return false;
if (sr.IsToolsSR())
return false;
if (vdi.IsUsedByHA())
{
return false;
}
List vbds = vdi.Connection.ResolveAll(vdi.VBDs);
if (vbds.Count > 1 && !AllowMultipleVBDDelete)
return false;
foreach (VBD vbd in vbds)
{
VM vm = vdi.Connection.Resolve(vbd.VM);
if (vdi.type == vdi_type.system)
{
if (vm.power_state == vm_power_state.Running)
return false;
}
if (vbd.Locked)
return false;
if (vbd.currently_attached)
{
//Check if we can unplug
DeactivateVBDCommand cmd = new DeactivateVBDCommand(Program.MainWindow, vbd);
if (!AllowRunningVMDelete || !cmd.CanExecute())
return false;
}
}
if (sr.HBALunPerVDI())
return true;
if (!vdi.allowed_operations.Contains(vdi_operations.destroy))
{
if (AllowRunningVMDelete)
{
// We deliberately DONT call allowed operations because we assume we know better :)
// Xapi will think we can't delete because VBDs are plugged. We are going to unplug them.
// Known risks of this method that will make us fail because we are disrespecting xapi:
// - someone else is calling a delete on this vdi already, altering the allowed ops
// - the storage manager cannot perform a delete on the SR due to drivers
return true;
}
return false;
}
return true;
}
protected override string GetCantExecuteReasonCore(IXenObject item)
{
VDI vdi = item as VDI;
if (vdi == null)
return base.GetCantExecuteReasonCore(item);
SR sr = vdi.Connection.Resolve(vdi.SR);
if (sr == null)
return Messages.SR_COULD_NOT_BE_CONTACTED;
VDI.FriendlyType vdiType = vdi.VDIType();
if (vdi.Locked)
return vdiType == VDI.FriendlyType.SNAPSHOT ? Messages.CANNOT_DELETE_SNAPSHOT_IN_USE
: vdiType == VDI.FriendlyType.ISO ? Messages.CANNOT_DELETE_ISO_IN_USE
: Messages.CANNOT_DELETE_VD_IN_USE;
if (sr.Physical())
return FriendlyErrorNames.VDI_IS_A_PHYSICAL_DEVICE;
if (sr.IsToolsSR())
return Messages.CANNOT_DELETE_TOOLS_SR;
if (vdi.IsUsedByHA())
return Messages.CANNOT_DELETE_HA_VD;
if (vdi.IsMetadataForDR())
return Messages.CANNOT_DELETE_DR_VD;
List vbds = vdi.Connection.ResolveAll(vdi.VBDs);
if (vbds.Count > 1 && !AllowMultipleVBDDelete)
return Messages.CANNOT_DELETE_VDI_MULTIPLE_VBDS;
foreach (VBD vbd in vbds)
{
VM vm = vdi.Connection.Resolve(vbd.VM);
if (vdiType == VDI.FriendlyType.SYSTEM_DISK)
{
if (vm.power_state == vm_power_state.Running)
return string.Format(
Messages.CANNOT_DELETE_IN_USE_SYS_VD,
Helpers.GetName(vm).Ellipsise(20));
}
if (vbd.Locked)
return vdiType == VDI.FriendlyType.SNAPSHOT ? Messages.CANNOT_DELETE_SNAPSHOT_IN_USE
: vdiType == VDI.FriendlyType.ISO ? Messages.CANNOT_DELETE_ISO_IN_USE
: Messages.CANNOT_DELETE_VD_IN_USE;
if (vbd.currently_attached)
{
if (!AllowRunningVMDelete)
{
return string.Format(Messages.CANNOT_DELETE_VDI_ACTIVE_ON,
Helpers.GetName(vm).Ellipsise(20));
}
DeactivateVBDCommand cmd = new DeactivateVBDCommand(Program.MainWindow, vbd);
if (!cmd.CanExecute())
{
var reasons = cmd.GetCantExecuteReasons();
return reasons.Count > 0
? string.Format(Messages.CANNOT_DELETE_CANNOT_DEACTIVATE_REASON,
Helpers.GetName(vm).Ellipsise(20), reasons.ElementAt(0).Value)
: Messages.UNKNOWN;
}
}
}
// This is a necessary final check, there are other blocking reasons non covered in this method
// Known examples:
// - someone else is calling a delete on this vdi already, altering the allowed ops
// - the storage manager cannot perform a delete on the SR due to drivers
if (!vdi.allowed_operations.Contains(vdi_operations.destroy))
return vdiType == VDI.FriendlyType.SNAPSHOT ? Messages.CANNOT_DELETE_SNAPSHOT_GENERIC
: vdiType == VDI.FriendlyType.ISO ? Messages.CANNOT_DELETE_ISO_GENERIC
: Messages.CANNOT_DELETE_VD_GENERIC;
return base.GetCantExecuteReasonCore(item);
}
protected override CommandErrorDialog GetErrorDialogCore(IDictionary cantExecuteReasons)
{
return new CommandErrorDialog(Messages.ERROR_DESTROYING_STORAGE_ITEMS_TITLE, Messages.ERROR_DESTROYING_STORAGE_ITEMS_MESSAGE, cantExecuteReasons);
}
protected override void ExecuteCore(SelectedItemCollection selection)
{
List actionsToComplete = new List();
List deletedVMSnapshots = new List();
foreach (VDI vdi in selection.AsXenObjects())
{
if (vdi.Locked || !vdi.Show(XenAdmin.Properties.Settings.Default.ShowHiddenVMs))
continue;
actionsToComplete.AddRange(getDestroyVDIAction(vdi, deletedVMSnapshots));
}
if (actionsToComplete.Count == 0)
return;
if (actionsToComplete.Count > 1)
RunMultipleActions(actionsToComplete, Messages.ACTION_DELETING_MULTIPLE_STORAGE_ITEMS_TITLE, Messages.ACTION_DELETING_MULTIPLE_STORAGE_ITEMS_STATUS, Messages.COMPLETED, true);
else
actionsToComplete[0].RunAsync();
}
private List getDestroyVDIAction(VDI vdi, List deletedVMSnapshots)
{
List actions = new List();
// Destroy the entire snapshot if it exists. Else destroy disk
if (vdi.is_a_snapshot && vdi.GetVMs().Count >= 1)
{
foreach (VM vm in vdi.GetVMs())
{
if (!vm.is_a_snapshot || deletedVMSnapshots.Contains(vm))
continue;
AsyncAction action = new VMSnapshotDeleteAction(vm);
actions.Add(action);
deletedVMSnapshots.Add(vm);
}
}
else
{
SR sr = vdi.Connection.Resolve(vdi.SR);
if (sr == null)
{
// Nothing we can do here, but this should have been caught in the getcantexecutereason method and prompted
return actions;
}
DestroyDiskAction a = new DestroyDiskAction(vdi);
a.AllowRunningVMDelete = AllowRunningVMDelete;
actions.Add(a);
}
return actions;
}
}
}