1
0
mirror of https://github.com/xcp-ng/xenadmin.git synced 2025-01-12 05:30:22 +01:00
xenadmin/XenAdmin/Wizards/NewSRWizard_Pages/Frontends/LVMoHBA.cs

587 lines
23 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.Linq;
using System.Windows.Forms;
using XenAdmin.Actions;
using XenAdmin.Controls;
using XenAdmin.Core;
using XenAdmin.Dialogs;
using XenAdmin.Dialogs.WarningDialogs;
using XenAdmin.Network;
using XenAPI;
using System.Drawing;
namespace XenAdmin.Wizards.NewSRWizard_Pages.Frontends
{
public partial class LVMoHBA : XenTabPage
{
private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
private List<FibreChannelDevice> _selectedDevices = new List<FibreChannelDevice>();
public LVMoHBA()
{
InitializeComponent();
SrType = SR.SRTypes.lvmohba;
}
public SR.SRTypes SrType { get; set; }
public virtual bool ShowNicColumn { get { return false; } }
private FibreChannelDescriptor CreateSrDescriptor(FibreChannelDevice device)
{
return SrType == SR.SRTypes.gfs2 ? CreateGfs2Descriptor(device) : CreateLvmSrDescriptor(device);
}
protected virtual FibreChannelDescriptor CreateLvmSrDescriptor(FibreChannelDevice device)
{
return new LvmOhbaSrDescriptor(device);
}
protected virtual FibreChannelDescriptor CreateGfs2Descriptor(FibreChannelDevice device)
{
return new Gfs2HbaSrDescriptor(device);
}
#region XenTabPage overrides
public override string PageTitle { get { return Messages.NEWSR_SELECT_LUN; } }
public override string Text { get { return Messages.NEWSR_LOCATION; } }
public override string HelpID { get { return "Location_HBA"; } }
protected override void PageLeaveCore(PageLoadedDirection direction, ref bool cancel)
{
if (direction == PageLoadedDirection.Back)
return;
Host master = Helpers.GetMaster(Connection);
if (master == null)
{
cancel = true;
return;
}
SrDescriptors = new List<FibreChannelDescriptor>();
var existingSrDescriptors = new List<FibreChannelDescriptor>();
var formatDiskDescriptors = new List<FibreChannelDescriptor>();
var performSecondProbe = Helpers.KolkataOrGreater(Connection) && !Helpers.FeatureForbidden(Connection, Host.CorosyncDisabled)
&& SrType != SR.SRTypes.lvmofcoe; // gfs2 over fcoe is not supported yet
foreach (var device in _selectedDevices)
{
// Start probe
var formatDiskDescriptor = CreateSrDescriptor(device);
List<SR.SRInfo> srs;
var currentSrDescriptor = formatDiskDescriptor;
if (!RunProbe(master, currentSrDescriptor, out srs))
{
cancel = true;
return;
}
if (performSecondProbe && srs.Count == 0)
{
// Start second probe
currentSrDescriptor = SrType == SR.SRTypes.gfs2 ? CreateLvmSrDescriptor(device) : CreateGfs2Descriptor(device);
if (!RunProbe(master, currentSrDescriptor, out srs))
{
cancel = true;
return;
}
}
currentSrDescriptor.UUID = srs.Select(sr => sr.UUID).FirstOrDefault();
if (!string.IsNullOrEmpty(SrWizardType.UUID))
{
// Check LUN contains correct SR
if (currentSrDescriptor.UUID == SrWizardType.UUID)
{
SrDescriptors.Add(currentSrDescriptor);
continue;
}
using (var dlog = new ThreeButtonDialog(
new ThreeButtonDialog.Details(SystemIcons.Error,
String.Format(Messages.INCORRECT_LUN_FOR_SR, SrWizardType.SrName), Messages.XENCENTER)))
{
dlog.ShowDialog(this);
}
cancel = true;
return;
}
if (string.IsNullOrEmpty(currentSrDescriptor.UUID))
{
// No existing SRs were found on this LUN. If allowed to create
// a new SR, ask the user if they want to proceed and format.
if (!SrWizardType.AllowToCreateNewSr)
{
using (var dlog = new ThreeButtonDialog(
new ThreeButtonDialog.Details(SystemIcons.Error,
Messages.NEWSR_LUN_HAS_NO_SRS, Messages.XENCENTER)))
{
dlog.ShowDialog(this);
}
cancel = true;
return;
}
if (!Program.RunInAutomatedTestMode)
formatDiskDescriptors.Add(formatDiskDescriptor);
}
else
{
// Check this isn't an existing SR on the current pool
var existingSr = Connection.Cache.SRs.FirstOrDefault(sr => sr.uuid == currentSrDescriptor.UUID);
if (existingSr != null)
{
var pool = Helpers.GetPool(existingSr.Connection);
var errorText = pool != null
? string.Format(Messages.NEWSR_LUN_IN_USE_ON_SELECTED_POOL, device.SCSIid, existingSr.Name())
: string.Format(Messages.NEWSR_LUN_IN_USE_ON_SELECTED_SERVER, device.SCSIid, existingSr.Name());
using (var dlog = new ThreeButtonDialog(new ThreeButtonDialog.Details(SystemIcons.Error,
errorText, Messages.XENCENTER)))
{
dlog.ShowDialog(this);
}
cancel = true;
return;
}
// CA-17230: Check this isn't an existing SR on any of the known pools.
// If it is then just continue (i.e. do not ask the user if they want to format or reattach it, we will just reattach it)
existingSr = SrWizardHelpers.SrInUse(currentSrDescriptor.UUID);
if (existingSr != null)
{
SrDescriptors.Add(currentSrDescriptor);
continue;
}
// We found a SR on this LUN. Will ask user for choice later.
existingSrDescriptors.Add(currentSrDescriptor);
}
}
if (!cancel && existingSrDescriptors.Count > 0)
{
var launcher = new LVMoHBAWarningDialogLauncher(this, existingSrDescriptors, true, SrType);
launcher.ShowWarnings();
cancel = launcher.Cancelled;
if (!cancel && launcher.SrDescriptors.Count > 0)
SrDescriptors.AddRange(launcher.SrDescriptors);
}
if (!cancel && formatDiskDescriptors.Count > 0)
{
var launcher = new LVMoHBAWarningDialogLauncher(this, formatDiskDescriptors, false, SrType);
launcher.ShowWarnings();
cancel = launcher.Cancelled;
if (!cancel && launcher.SrDescriptors.Count > 0)
SrDescriptors.AddRange(launcher.SrDescriptors);
}
}
private bool RunProbe(Host master, FibreChannelDescriptor srDescriptor, out List<SR.SRInfo> srs)
{
srs = null;
var action = new SrProbeAction(Connection, master, srDescriptor.SrType, srDescriptor.DeviceConfig);
using (var dlg = new ActionProgressDialog(action, ProgressBarStyle.Marquee))
dlg.ShowDialog(this);
if (action.Succeeded)
{
try
{
srs = action.ProbeExtResult != null ? SR.ParseSRList(action.ProbeExtResult) : SR.ParseSRListXML(action.Result);
return true;
}
catch (Exception)
{
return false;
}
}
return false;
}
public override bool EnableNext()
{
UpdateSelectionButtons();
return _selectedDevices.Count > 0;
}
public override bool EnablePrevious()
{
if (SrWizardType.DisasterRecoveryTask && SrWizardType.SrToReattach == null)
return false;
return true;
}
protected override void PageLoadedCore(PageLoadedDirection direction)
{
if (direction == PageLoadedDirection.Back)
return;
colNic.Visible = ShowNicColumn;
dataGridView.Rows.Clear();
var vendorGroups = from device in FCDevices
group device by device.Vendor into g
orderby g.Key
select new { VendorName = g.Key, Devices = g.OrderBy(x => x.Serial) };
foreach (var vGroup in vendorGroups)
{
var vendorRow = new VendorRow(vGroup.VendorName);
dataGridView.Rows.Add(vendorRow);
using (var font = new Font(dataGridView.DefaultCellStyle.Font, FontStyle.Bold))
vendorRow.DefaultCellStyle = new DataGridViewCellStyle(dataGridView.DefaultCellStyle)
{
Font = font,
SelectionBackColor = dataGridView.DefaultCellStyle.BackColor,
SelectionForeColor = dataGridView.DefaultCellStyle.ForeColor
};
var deviceRows = from device in vGroup.Devices select new FCDeviceRow(device, ShowNicColumn);
dataGridView.Rows.AddRange(deviceRows.ToArray());
}
}
public override string NextText(bool isLastPage)
{
// for Dundee or greater connections, we have "Storage provisioning settings" page after this page, so the Next button should say "Next", not "Create"
return Helpers.DundeeOrGreater(Connection) ? Messages.WIZARD_BUTTON_NEXT : Messages.NEWSR_LVMOHBA_NEXT_TEXT;
}
public override void SelectDefaultControl()
{
dataGridView.Select();
}
#endregion
#region Event handlers
private void dataGridView_CellClick(object sender, DataGridViewCellEventArgs e)
{
if (_srWizardType.SrToReattach != null)
return;
if (e.ColumnIndex != colCheck.Index || e.RowIndex < 0 || e.RowIndex > dataGridView.RowCount - 1)
return;
var deviceRow = dataGridView.Rows[e.RowIndex] as FCDeviceRow;
if (deviceRow == null)
return;
deviceRow.Cells[colCheck.Index].Value = !(bool)deviceRow.Cells[colCheck.Index].Value;
UpdateSelectedDevices();
OnPageUpdated();
}
private void dataGridView_SelectionChanged(object sender, EventArgs e)
{
if (_srWizardType.SrToReattach == null)
return;
UpdateSelectedDevices();
OnPageUpdated();
}
private void buttonSelectAll_Click(object sender, EventArgs e)
{
foreach (var row in dataGridView.Rows)
{
var deviceRow = row as FCDeviceRow;
if (deviceRow != null && deviceRow.Cells.Count > 0)
deviceRow.Cells[colCheck.Index].Value = true;
}
UpdateSelectedDevices();
OnPageUpdated();
}
private void buttonClearAll_Click(object sender, EventArgs e)
{
foreach (var row in dataGridView.Rows)
{
var deviceRow = row as FCDeviceRow;
if (deviceRow != null && deviceRow.Cells.Count > 0)
deviceRow.Cells[colCheck.Index].Value = false;
}
UpdateSelectedDevices();
OnPageUpdated();
}
#endregion
private void UpdateSelectedDevices()
{
if (SrWizardType.SrToReattach == null)
{
//when creating a new SR the checkbox column is visible
_selectedDevices = (from DataGridViewRow row in dataGridView.Rows
let deviceRow = row as FCDeviceRow
where deviceRow != null
&& deviceRow.Cells.Count > 0
&& (bool)(deviceRow.Cells[colCheck.Index].Value)
select deviceRow.Device).ToList();
}
else
{
//when reattaching SR the checkbox column is hidden
_selectedDevices = (from DataGridViewRow row in dataGridView.Rows
let deviceRow = row as FCDeviceRow
where deviceRow != null && deviceRow.Selected
select deviceRow.Device).ToList();
}
}
private void UpdateSelectionButtons()
{
if (buttonSelectAll.Visible)
buttonSelectAll.Enabled = _selectedDevices.Count < FCDevices.Count;
if (buttonClearAll.Visible)
buttonClearAll.Enabled = _selectedDevices.Count > 0;
}
public bool FiberChannelScan(IWin32Window owner, IXenConnection connection, out List<FibreChannelDevice> devices)
{
devices = new List<FibreChannelDevice>();
Host master = Helpers.GetMaster(connection);
if (master == null)
return false;
var action = new FibreChannelProbeAction(master, SrType);
using (var dialog = new ActionProgressDialog(action, ProgressBarStyle.Marquee))
dialog.ShowDialog(owner); //Will block until dialog closes, action completed
if (!action.Succeeded)
return false;
devices = action.FibreChannelDevices;
if (devices != null && devices.Count > 0)
return true;
using (var dlg = new ThreeButtonDialog(
new ThreeButtonDialog.Details(SystemIcons.Warning, Messages.FIBRECHANNEL_NO_RESULTS, Messages.XENCENTER)))
{
dlg.ShowDialog();
}
return false;
}
#region Accessors
public List<FibreChannelDevice> FCDevices { private get; set; }
private SrWizardType _srWizardType;
public SrWizardType SrWizardType
{
private get
{
return _srWizardType;
}
set
{
_srWizardType = value;
bool creatingNew = _srWizardType.SrToReattach == null;
colCheck.Visible = creatingNew;
dataGridView.MultiSelect = creatingNew;
buttonSelectAll.Visible = creatingNew;
buttonClearAll.Visible = creatingNew;
labelCreate.Visible = creatingNew;
labelReattach.Visible = !creatingNew;
}
}
public List<FibreChannelDescriptor> SrDescriptors { get; private set; }
/// <summary>
/// min size
/// </summary>
public long SRSize
{
get
{
long size = long.MaxValue;
foreach (var srDescriptor in SrDescriptors)
{
if (srDescriptor.Device.Size < size)
size = srDescriptor.Device.Size;
}
return size;
}
}
#endregion
#region Nested classes
private class FCDeviceRow : DataGridViewRow
{
public FibreChannelDevice Device { get; private set; }
public FCDeviceRow(FibreChannelDevice device, bool showNicColumn)
{
Device = device;
string id = string.IsNullOrEmpty(device.SCSIid) ? device.Path : device.SCSIid;
string details = String.Format("{0}:{1}:{2}:{3}", device.adapter, device.channel, device.id, device.lun);
Cells.AddRange(new DataGridViewCheckBoxCell{ThreeState = false, Value = false},
new DataGridViewTextBoxCell { Value = Util.DiskSizeString(device.Size) },
new DataGridViewTextBoxCell { Value = device.Serial },
new DataGridViewTextBoxCell { Value = id },
new DataGridViewTextBoxCell { Value = details });
if (showNicColumn)
Cells.Add(new DataGridViewTextBoxCell {Value = device.eth});
}
}
private class VendorRow : DataGridViewRow
{
public VendorRow(string vendor)
{
Cells.AddRange(new DataGridViewCheckBoxCellVendor(),
new DataGridViewTextBoxCell { Value = vendor },
new DataGridViewTextBoxCell(),
new DataGridViewTextBoxCell(),
new DataGridViewTextBoxCell());
}
private class DataGridViewCheckBoxCellVendor : DataGridViewCheckBoxCell
{
protected override void Paint(Graphics graphics, Rectangle clipBounds, Rectangle cellBounds, int rowIndex, DataGridViewElementStates elementState, object value, object formattedValue, string errorText, DataGridViewCellStyle cellStyle, DataGridViewAdvancedBorderStyle advancedBorderStyle, DataGridViewPaintParts paintParts)
{
using (var normalBrush = new SolidBrush(OwningRow.DefaultCellStyle.BackColor))
using (var selectedBrush = new SolidBrush(OwningRow.DefaultCellStyle.SelectionBackColor))
{
graphics.FillRectangle(
(elementState & DataGridViewElementStates.Selected) != 0 ? selectedBrush : normalBrush,
cellBounds.X, cellBounds.Y, cellBounds.Width, cellBounds.Height);
}
}
}
}
public enum UserSelectedOption { Cancel, Reattach, Format, Skip }
private class LVMoHBAWarningDialogLauncher
{
private readonly List<FibreChannelDescriptor> inputSrDescriptors;
private readonly bool foundExistingSRs;
private readonly IWin32Window owner;
private readonly SR.SRTypes requestedSrType;
public List<FibreChannelDescriptor> SrDescriptors { get; private set; }
public bool Cancelled { get; private set; }
public LVMoHBAWarningDialogLauncher(IWin32Window owner, List<FibreChannelDescriptor> srDescriptors,
bool foundExistingSRs, SR.SRTypes requestedSrType)
{
this.owner = owner;
this.foundExistingSRs = foundExistingSRs;
inputSrDescriptors = srDescriptors;
this.requestedSrType = requestedSrType;
SrDescriptors = new List<FibreChannelDescriptor>();
}
private UserSelectedOption GetSelectedOption(FibreChannelDescriptor descriptor,
out bool repeatForRemainingLUNs)
{
int remainingCount = inputSrDescriptors.Count - 1 - inputSrDescriptors.IndexOf(descriptor);
using (var dialog = new LVMoHBAWarningDialog(descriptor.Device, remainingCount, foundExistingSRs, descriptor.SrType, requestedSrType))
{
dialog.ShowDialog(owner);
repeatForRemainingLUNs = dialog.RepeatForRemainingLUNs;
return dialog.SelectedOption;
}
}
public void ShowWarnings()
{
bool repeatForRemainingLUNs = false;
UserSelectedOption selectedOption = UserSelectedOption.Cancel;
foreach (var descriptor in inputSrDescriptors)
{
if (!repeatForRemainingLUNs)
{
selectedOption = GetSelectedOption(descriptor, out repeatForRemainingLUNs);
}
switch (selectedOption)
{
case UserSelectedOption.Format:
descriptor.UUID = null;
SrDescriptors.Add(descriptor);
break;
case UserSelectedOption.Reattach:
SrDescriptors.Add(descriptor);
break;
case UserSelectedOption.Cancel:
SrDescriptors.Clear();
Cancelled = true;
return;
}
}
}
}
#endregion
}
}