2017-01-16 20:59:50 +01:00
/ * Copyright ( c ) Citrix Systems , Inc .
2013-06-24 13:41:48 +02:00
* 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.Windows.Forms ;
using XenAPI ;
using XenAdmin.Actions ;
using XenAdmin.Network ;
using XenAdmin.Core ;
using System.Threading ;
using XenAdmin.Controls ;
using XenAdmin.Dialogs ;
using System.Drawing ;
2013-10-10 13:22:58 +02:00
using System.Linq ;
2017-12-01 16:27:42 +01:00
using System.Web.Script.Serialization ;
2013-10-10 16:33:29 +02:00
using XenAdmin.Utils ;
2017-11-17 02:04:45 +01:00
using XenCenterLib ;
2013-06-24 13:41:48 +02:00
namespace XenAdmin.Wizards.NewSRWizard_Pages.Frontends
{
public partial class LVMoISCSI : XenTabPage
{
private static readonly log4net . ILog log = log4net . LogManager . GetLogger ( System . Reflection . MethodBase . GetCurrentMethod ( ) . DeclaringType ) ;
/// <summary>
/// If an SR already exists on this LUN, this will point to the SR info which indicates that
/// the SR should be reattached.
///
/// If this is null, indicates the LUN should be formatted into a new SR.
/// </summary>
private SR . SRInfo _srToIntroduce ;
private ISCSIPopulateLunsAction IscsiPopulateLunsAction ;
private ISCSIPopulateIQNsAction IscsiPopulateIqnsAction ;
private readonly Dictionary < String , ISCSIInfo > LunMap = new Dictionary < String , ISCSIInfo > ( ) ;
private readonly ToolTip TargetIqnToolTip = new ToolTip ( ) ;
private const string TARGET = "target" ;
private const string PORT = "port" ;
private const string TARGETIQN = "targetIQN" ;
private const string LUNSERIAL = "LUNSerial" ;
private const string SCSIID = "SCSIid" ;
private const string CHAPUSER = "chapuser" ;
private const string CHAPPASSWORD = "chappassword" ;
2013-10-10 13:22:58 +02:00
private IEnumerable < Control > ErrorIcons
{
get { return new Control [ ] { errorIconAtCHAPPassword , errorIconAtHostOrIP , errorIconAtTargetLUN } ; }
}
private IEnumerable < Control > ErrorLabels
{
get { return new Control [ ] { errorLabelAtHostname , errorLabelAtCHAPPassword , errorLabelAtTargetLUN } ; }
}
private IEnumerable < Control > SpinnerControls
{
get { return new Control [ ] { spinnerIconAtScanTargetHostButton , spinnerIconAtTargetIqn , spinnerIconAtTargetLun } ; }
}
private IEnumerable < Control > UserInputControls
{
get
{
return new Control [ ]
{
textBoxIscsiHost , textBoxIscsiPort , IscsiUseChapCheckBox , IScsiChapUserTextBox ,
IScsiChapSecretTextBox , scanTargetHostButton , comboBoxIscsiIqns , comboBoxIscsiLuns
} ;
}
}
private readonly TemporaryDisablerForControls controlDisabler = new TemporaryDisablerForControls ( ) ;
2013-06-24 13:41:48 +02:00
public LVMoISCSI ( )
{
InitializeComponent ( ) ;
2018-03-23 12:56:54 +01:00
SrType = SR . SRTypes . lvmoiscsi ;
2013-06-24 13:41:48 +02:00
}
#region XentabPage overrides
public override string PageTitle { get { return Messages . NEWSR_PATH_ISCSI ; } }
public override string Text { get { return Messages . NEWSR_LOCATION ; } }
public override string HelpID { get { return "Location_ISCSI" ; } }
2018-03-09 01:31:46 +01:00
protected override void PageLoadedCore ( PageLoadedDirection direction )
2013-06-24 13:41:48 +02:00
{
HelpersGUI . PerformIQNCheck ( ) ;
if ( direction = = PageLoadedDirection . Forward )
2020-02-10 16:45:16 +01:00
textBoxIscsiHost . Focus ( ) ;
2013-06-24 13:41:48 +02:00
}
2018-03-09 00:55:50 +01:00
protected override void PageLeaveCore ( PageLoadedDirection direction , ref bool cancel )
2013-06-24 13:41:48 +02:00
{
if ( direction = = PageLoadedDirection . Back )
return ;
// For Miami hosts we need to ensure an SR.probe()
// has been performed, and that the user has made a decision. Show the iSCSI choices dialog until
// they click something other than 'Cancel'. For earlier host versions, warn that data loss may
// occur.
Host master = Helpers . GetMaster ( Connection ) ;
if ( master = = null )
{
cancel = true ;
return ;
}
2018-03-29 15:32:39 +02:00
// Start probe
2013-06-24 13:41:48 +02:00
2018-03-29 11:04:22 +02:00
List < SR . SRInfo > srs ;
2018-03-29 15:32:39 +02:00
var currentSrType = SrType ;
2018-03-29 11:04:22 +02:00
2018-03-29 15:32:39 +02:00
if ( ! RunProbe ( master , currentSrType , out srs ) )
2016-06-20 14:17:42 +02:00
{
2018-03-29 11:04:22 +02:00
cancel = iscsiProbeError = true ;
return ;
2016-06-20 14:17:42 +02:00
}
2018-03-29 11:04:22 +02:00
var performSecondProbe = Helpers . KolkataOrGreater ( Connection ) & &
! Helpers . FeatureForbidden ( Connection , Host . CorosyncDisabled ) ;
if ( performSecondProbe & & srs . Count = = 0 )
{
// Start second probe
2018-03-29 15:32:39 +02:00
currentSrType = SrType = = SR . SRTypes . gfs2 ? SR . SRTypes . lvmoiscsi : SR . SRTypes . gfs2 ;
2018-03-29 11:04:22 +02:00
2018-03-29 15:32:39 +02:00
if ( ! RunProbe ( master , currentSrType , out srs ) )
2018-03-29 11:04:22 +02:00
{
cancel = iscsiProbeError = true ;
return ;
}
}
2013-06-24 13:41:48 +02:00
// Probe has been performed. Now ask the user if they want to Reattach/Format/Cancel.
// Will return false on cancel
2018-03-29 15:32:39 +02:00
cancel = iscsiProbeError = ! ExamineIscsiProbeResults ( currentSrType , srs ) ;
2018-03-29 11:04:22 +02:00
}
2018-03-29 15:32:39 +02:00
private bool RunProbe ( Host master , SR . SRTypes srType , out List < SR . SRInfo > srs )
2018-03-29 11:04:22 +02:00
{
2018-03-29 15:32:39 +02:00
srs = null ;
Dictionary < String , String > dconf = GetDeviceConfig ( srType ) ;
if ( dconf = = null )
return false ;
var action = new SrProbeAction ( Connection , master , srType , dconf ) ;
2018-03-29 11:04:22 +02:00
using ( var dialog = new ActionProgressDialog ( action , ProgressBarStyle . Marquee ) )
{
dialog . ShowCancel = true ;
dialog . ShowDialog ( this ) ;
}
_srToIntroduce = null ;
if ( action . Succeeded )
{
try
{
srs = action . ProbeExtResult ! = null ? SR . ParseSRList ( action . ProbeExtResult ) : SR . ParseSRListXML ( action . Result ) ;
return true ;
}
catch ( Exception )
{
return false ;
}
}
Exception exn = action . Exception ;
log . Warn ( exn , exn ) ;
2018-03-29 15:32:39 +02:00
2018-03-29 11:04:22 +02:00
Failure failure = exn as Failure ;
2018-03-29 15:32:39 +02:00
if ( failure ! = null )
2018-03-29 11:04:22 +02:00
{
errorIconAtHostOrIP . Visible = true ;
errorLabelAtHostname . Visible = true ;
2018-03-29 15:32:39 +02:00
errorLabelAtHostname . Text = failure . ErrorDescription [ 0 ] = = "SR_BACKEND_FAILURE_140"
? Messages . INVALID_HOST
: ( failure . ErrorDescription . Count > 2
? failure . ErrorDescription [ 2 ]
: failure . ErrorDescription [ 0 ] ) ;
2018-03-29 11:04:22 +02:00
textBoxIscsiHost . Focus ( ) ;
}
2018-03-29 15:32:39 +02:00
return false ;
2013-06-24 13:41:48 +02:00
}
2015-09-01 18:25:44 +02:00
bool iscsiProbeError = false ;
2013-06-24 13:41:48 +02:00
public override bool EnableNext ( )
{
UInt16 i ;
bool portValid = UInt16 . TryParse ( textBoxIscsiPort . Text , out i ) ;
return ! String . IsNullOrEmpty ( getIscsiHost ( ) )
& & portValid
& & ! ( IscsiUseChapCheckBox . Checked & & String . IsNullOrEmpty ( IScsiChapUserTextBox . Text ) )
2013-10-10 13:22:58 +02:00
& & comboBoxIscsiLuns . SelectedItem ! = null & & comboBoxIscsiLuns . SelectedItem as string ! = Messages . SELECT_TARGET_LUN
2015-09-01 18:25:44 +02:00
& & ! iscsiProbeError & & ! IsLunInUse ( ) ;
2013-06-24 13:41:48 +02:00
}
public override bool EnablePrevious ( )
{
if ( SrWizardType . DisasterRecoveryTask & & SrWizardType . SrToReattach = = null )
return false ;
return true ;
}
public override void PopulatePage ( )
{
2013-10-10 13:22:58 +02:00
HideAllErrorIconsAndLabels ( ) ;
2013-06-24 13:41:48 +02:00
// Enable IQN scanning
comboBoxIscsiIqns . Visible = true ;
// IQN's can be very long, so we will show the value as a mouse over tooltip.
// Initialize the tooltip here.
TargetIqnToolTip . Active = true ;
TargetIqnToolTip . AutomaticDelay = 0 ;
TargetIqnToolTip . AutoPopDelay = 50000 ;
TargetIqnToolTip . InitialDelay = 50 ;
TargetIqnToolTip . ReshowDelay = 50 ;
TargetIqnToolTip . ShowAlways = true ;
}
#endregion
private void UpdateButtons ( )
{
UInt16 i ;
bool portValid = UInt16 . TryParse ( textBoxIscsiPort . Text , out i ) ;
2013-10-10 13:22:58 +02:00
scanTargetHostButton . Enabled =
2013-06-24 13:41:48 +02:00
! String . IsNullOrEmpty ( getIscsiHost ( ) )
& & portValid ;
// Cause wizards next etc to update
OnPageUpdated ( ) ;
}
private void textBoxIscsiHost_TextChanged ( object sender , EventArgs e )
{
2013-10-10 13:22:58 +02:00
HideAllErrorIconsAndLabels ( ) ;
HideAllSpinnerIcons ( ) ;
2013-06-24 13:41:48 +02:00
IScsiParams_TextChanged ( null , null ) ;
}
/// <summary>
/// Called when any of the iSCSI filer params change: resets the IQNs/LUNs.
/// Must be called on the event thread.
/// </summary>
private void IScsiParams_TextChanged ( object sender , EventArgs e )
{
Program . AssertOnEventThread ( ) ;
2013-10-10 13:22:58 +02:00
spinnerIconAtScanTargetHostButton . Visible = false ;
2013-06-24 13:41:48 +02:00
// User has changed filer hostname/username/password - clear IQN/LUN boxes
comboBoxIscsiIqns . Items . Clear ( ) ;
comboBoxIscsiIqns . Enabled = false ;
2013-10-10 13:22:58 +02:00
labelIscsiIQN . Enabled = false ;
iSCSITargetGroupBox . Enabled = false ;
2013-06-24 13:41:48 +02:00
// Cancel pending IQN/LUN scans
if ( IscsiPopulateIqnsAction ! = null )
{
IscsiPopulateIqnsAction . Cancel ( ) ;
}
ChapSettings_Changed ( null , null ) ;
}
2013-10-10 13:22:58 +02:00
private bool IsLunInUse ( )
2013-06-24 13:41:48 +02:00
{
SR sr = UniquenessCheck ( ) ;
// LUN is not in use iff sr != null
if ( sr = = null )
{
2013-10-10 13:22:58 +02:00
HideAllErrorIconsAndLabels ( ) ;
2013-06-24 13:41:48 +02:00
return false ;
}
2013-10-10 13:22:58 +02:00
spinnerIconAtTargetLun . Visible = false ;
2013-06-24 13:41:48 +02:00
Pool pool = Helpers . GetPool ( sr . Connection ) ;
if ( pool ! = null )
{
2013-10-10 13:22:58 +02:00
errorIconAtTargetLUN . Visible = true ;
errorLabelAtTargetLUN . Visible = true ;
2017-09-03 04:33:29 +02:00
errorLabelAtTargetLUN . Text = String . Format ( Messages . NEWSR_LUN_IN_USE_ON_POOL , sr . Name ( ) , pool . Name ( ) ) ;
2013-06-24 13:41:48 +02:00
return true ;
}
Host master = Helpers . GetMaster ( sr . Connection ) ;
if ( master ! = null )
{
2013-10-10 13:22:58 +02:00
errorIconAtTargetLUN . Visible = true ;
errorLabelAtTargetLUN . Visible = true ;
2017-09-03 04:33:29 +02:00
errorLabelAtTargetLUN . Text = String . Format ( Messages . NEWSR_LUN_IN_USE_ON_SERVER , sr . Name ( ) , master . Name ( ) ) ;
2013-06-24 13:41:48 +02:00
return true ;
}
2013-10-10 13:22:58 +02:00
errorIconAtTargetLUN . Visible = true ;
errorLabelAtTargetLUN . Visible = true ;
errorLabelAtTargetLUN . Text = Messages . NEWSR_LUN_IN_USE ;
2013-06-24 13:41:48 +02:00
return true ;
}
private void comboBoxIscsiLuns_SelectedIndexChanged ( object sender , EventArgs e )
{
2015-09-01 18:25:44 +02:00
iscsiProbeError = false ;
2013-10-10 13:22:58 +02:00
if ( comboBoxIscsiLuns . SelectedItem as string ! = Messages . SELECT_TARGET_LUN )
{
2020-03-11 10:17:50 +01:00
spinnerIconAtTargetLun . ShowSuccessImage ( ) ;
2013-10-10 13:22:58 +02:00
}
else
{
spinnerIconAtTargetLun . Visible = false ;
HideAllErrorIconsAndLabels ( ) ;
}
2013-06-24 13:41:48 +02:00
UpdateButtons ( ) ;
}
private void ChapSettings_Changed ( object sender , EventArgs e )
{
comboBoxIscsiLuns . Items . Clear ( ) ;
comboBoxIscsiLuns . Text = "" ;
comboBoxIscsiLuns . Enabled = false ;
2013-10-10 13:22:58 +02:00
targetLunLabel . Enabled = false ;
2013-06-24 13:41:48 +02:00
if ( IscsiPopulateLunsAction ! = null )
{
IscsiPopulateLunsAction . Cancel ( ) ;
}
UpdateButtons ( ) ;
}
/// <summary>
/// Check the current config of the iSCSI sr in the wizard is unique across
/// all active connections.
/// </summary>
/// <returns>SR that uses this config if not unique, null if unique</returns>
private SR UniquenessCheck ( )
{
// Check currently selected lun is unique amongst other connected hosts.
foreach ( IXenConnection connection in ConnectionsManager . XenConnectionsCopy )
{
foreach ( SR sr in connection . Cache . SRs )
{
2018-03-27 15:17:34 +02:00
if ( sr . GetSRType ( false ) ! = SR . SRTypes . lvmoiscsi & & sr . GetSRType ( false ) ! = SR . SRTypes . gfs2 )
2013-06-24 13:41:48 +02:00
continue ;
if ( sr . PBDs . Count < 1 )
continue ;
PBD pbd = connection . Resolve ( sr . PBDs [ 0 ] ) ;
if ( pbd = = null )
continue ;
if ( UniquenessCheckMiami ( connection , pbd ) )
return sr ;
}
}
return null ;
}
/// <summary>
/// Check currently LUN against miami host
/// </summary>
/// <param name="connection"></param>
/// <param name="pbd"></param>
/// <returns></returns>
private bool UniquenessCheckMiami ( IXenConnection connection , PBD pbd )
{
2018-03-27 15:17:34 +02:00
String scsiID ;
if ( pbd . device_config . ContainsKey ( SCSIID ) )
scsiID = pbd . device_config [ SCSIID ] ;
else
2013-06-24 13:41:48 +02:00
return false ;
String myLUN = getIscsiLUN ( ) ;
if ( ! LunMap . ContainsKey ( myLUN ) )
return false ;
ISCSIInfo info = LunMap [ myLUN ] ;
return info . ScsiID = = scsiID ;
}
private String getIscsiHost ( )
{
// If the user has selected an IQN, use the host from that IQN (due to multi-homing,
// this may differ from the host they first entered). Otherwise use the host
// they first entered,
ToStringWrapper < IScsiIqnInfo > wrapper = comboBoxIscsiIqns . SelectedItem as ToStringWrapper < IScsiIqnInfo > ;
if ( wrapper ! = null )
return wrapper . item . IpAddress ;
return textBoxIscsiHost . Text . Trim ( ) ;
}
private UInt16 getIscsiPort ( )
{
ToStringWrapper < IScsiIqnInfo > wrapper = comboBoxIscsiIqns . SelectedItem as ToStringWrapper < IScsiIqnInfo > ;
if ( wrapper ! = null )
return wrapper . item . Port ;
// No combobox item was selected
UInt16 port ;
if ( UInt16 . TryParse ( textBoxIscsiPort . Text , out port ) )
{
return port ;
}
else
{
return Util . DEFAULT_ISCSI_PORT ;
}
}
private String getIscsiIQN ( )
{
ToStringWrapper < IScsiIqnInfo > wrapper = comboBoxIscsiIqns . SelectedItem as ToStringWrapper < IScsiIqnInfo > ;
if ( wrapper = = null )
return "" ;
else
return wrapper . item . TargetIQN ;
}
private void IScsiTargetIqnComboBox_SelectedIndexChanged ( object sender , EventArgs e )
{
ToStringWrapper < IScsiIqnInfo > wrapper = comboBoxIscsiIqns . SelectedItem as ToStringWrapper < IScsiIqnInfo > ;
2013-10-10 13:22:58 +02:00
ClearLunMapAndCombo ( ) ;
HideAllErrorIconsAndLabels ( ) ;
2013-06-24 13:41:48 +02:00
if ( wrapper ! = null )
{
TargetIqnToolTip . SetToolTip ( comboBoxIscsiIqns , wrapper . ToString ( ) ) ;
2013-10-10 13:22:58 +02:00
IscsiPopulateLUNs ( ) ;
2013-06-24 13:41:48 +02:00
}
else
{
2013-10-10 13:22:58 +02:00
TargetIqnToolTip . SetToolTip ( comboBoxIscsiIqns , Messages . SELECT_TARGET_IQN ) ;
2013-06-24 13:41:48 +02:00
}
}
private String getIscsiLUN ( )
{
return comboBoxIscsiLuns . Text ;
}
private void IscsiUseChapCheckBox_CheckedChanged ( object sender , EventArgs e )
{
2013-10-10 13:22:58 +02:00
bool enabled = IscsiUseChapCheckBox . Checked ;
IScsiChapUserTextBox . Enabled = enabled ;
IScsiChapSecretTextBox . Enabled = enabled ;
labelCHAPuser . Enabled = enabled ;
IScsiChapSecretLabel . Enabled = enabled ;
HideAllErrorIconsAndLabels ( ) ;
2013-06-24 13:41:48 +02:00
ChapSettings_Changed ( null , null ) ;
}
2013-10-10 13:22:58 +02:00
private void scanTargetHostButton_Click ( object sender , EventArgs e )
2013-06-24 13:41:48 +02:00
{
2013-10-10 13:22:58 +02:00
HideAllErrorIconsAndLabels ( ) ;
spinnerIconAtTargetIqn . Visible = false ;
spinnerIconAtTargetLun . Visible = false ;
spinnerIconAtScanTargetHostButton . StartSpinning ( ) ;
scanTargetHostButton . Enabled = false ;
2013-06-24 13:41:48 +02:00
// For this button to be enabled, we must be Miami or newer
comboBoxIscsiIqns . Items . Clear ( ) ;
// Clear LUNs as they may no longer be valid
ClearLunMapAndCombo ( ) ;
// Cancel any LUN scan in progress, as it is no longer meaningful
if ( IscsiPopulateLunsAction ! = null )
{
IscsiPopulateLunsAction . Cancel ( ) ;
}
2013-10-10 13:22:58 +02:00
UpdateButtons ( ) ;
2018-03-28 17:17:08 +02:00
var chapUser = IscsiUseChapCheckBox . Checked ? IScsiChapUserTextBox . Text : null ;
var chapPwd = IscsiUseChapCheckBox . Checked ? IScsiChapSecretTextBox . Text : null ;
2013-06-24 13:41:48 +02:00
2018-03-28 17:17:08 +02:00
IscsiPopulateIqnsAction = SrType = = SR . SRTypes . gfs2
? new Gfs2PopulateIQNsAction ( Connection , getIscsiHost ( ) , getIscsiPort ( ) , chapUser , chapPwd )
: new ISCSIPopulateIQNsAction ( Connection , getIscsiHost ( ) , getIscsiPort ( ) , chapUser , chapPwd ) ;
2013-06-24 13:41:48 +02:00
IscsiPopulateIqnsAction . Completed + = IscsiPopulateIqnsAction_Completed ;
2013-10-10 13:22:58 +02:00
controlDisabler . Reset ( ) ;
controlDisabler . SaveOrUpdateEnabledStates ( UserInputControls ) ;
controlDisabler . DisableAllControls ( ) ;
scanTargetHostButton . Enabled = false ;
IscsiPopulateIqnsAction . RunAsync ( ) ;
2013-06-24 13:41:48 +02:00
}
private void ClearLunMapAndCombo ( )
{
// Clear LUNs as they may no longer be valid
comboBoxIscsiLuns . Items . Clear ( ) ;
comboBoxIscsiLuns . Text = "" ;
comboBoxIscsiLuns . Enabled = false ;
2013-10-10 13:22:58 +02:00
targetLunLabel . Enabled = false ;
2013-06-24 13:41:48 +02:00
LunMap . Clear ( ) ;
2013-10-10 13:22:58 +02:00
spinnerIconAtTargetIqn . Visible = false ;
spinnerIconAtTargetLun . Visible = false ;
2013-06-24 13:41:48 +02:00
UpdateButtons ( ) ;
}
2013-08-05 15:28:21 +02:00
private void IscsiPopulateIqnsAction_Completed ( ActionBase sender )
2013-06-24 13:41:48 +02:00
{
2015-04-16 15:17:02 +02:00
Program . Invoke ( this , ( System . Threading . WaitCallback ) IscsiPopulateIqnsAction_Completed_ , sender ) ;
2013-06-24 13:41:48 +02:00
}
private void IscsiPopulateIqnsAction_Completed_ ( object o )
{
Program . AssertOnEventThread ( ) ;
ISCSIPopulateIQNsAction action = ( ISCSIPopulateIQNsAction ) o ;
2013-10-10 13:22:58 +02:00
controlDisabler . RestoreEnabledOnAllControls ( ) ;
2013-06-24 13:41:48 +02:00
if ( action . Succeeded )
{
if ( action . IQNs . Length = = 0 )
{
// Do nothing: ActionProgressDialog will show Messages.NEWSR_NO_IQNS_FOUND
}
else
{
int width = comboBoxIscsiIqns . Width ;
2013-10-10 13:22:58 +02:00
comboBoxIscsiIqns . Items . Add ( Messages . SELECT_TARGET_IQN ) ;
2013-06-24 13:41:48 +02:00
foreach ( Actions . IScsiIqnInfo iqnInfo in action . IQNs )
{
if ( ! String . IsNullOrEmpty ( iqnInfo . TargetIQN ) )
{
String toString = String . Format ( "{0} ({1}:{2})" , iqnInfo . TargetIQN , iqnInfo . IpAddress , iqnInfo . Port ) ;
comboBoxIscsiIqns . Items . Add ( new ToStringWrapper < IScsiIqnInfo > ( iqnInfo , toString ) ) ;
width = Math . Max ( width , Drawing . MeasureText ( toString , comboBoxIscsiIqns . Font ) . Width ) ;
}
}
// Set the combo box dropdown width to accommodate the widest item (within reason)
comboBoxIscsiIqns . DropDownWidth = Math . Min ( width , Int16 . MaxValue ) ;
if ( comboBoxIscsiIqns . Items . Count > 0 )
{
2013-10-10 13:22:58 +02:00
comboBoxIscsiIqns . SelectedItem = Messages . SELECT_TARGET_IQN ;
2013-06-24 13:41:48 +02:00
comboBoxIscsiIqns . Enabled = true ;
2013-10-10 13:22:58 +02:00
labelIscsiIQN . Enabled = true ;
iSCSITargetGroupBox . Enabled = true ;
2013-06-24 13:41:48 +02:00
}
2013-10-10 13:22:58 +02:00
2020-03-11 10:17:50 +01:00
spinnerIconAtScanTargetHostButton . ShowSuccessImage ( ) ;
2013-10-10 13:22:58 +02:00
2013-06-24 13:41:48 +02:00
comboBoxIscsiIqns . Focus ( ) ;
}
}
else
{
2013-10-10 13:22:58 +02:00
spinnerIconAtScanTargetHostButton . Visible = false ;
2013-06-24 13:41:48 +02:00
Failure failure = action . Exception as Failure ;
if ( failure ! = null & & failure . ErrorDescription [ 0 ] = = "SR_BACKEND_FAILURE_140" )
{
2013-10-10 13:22:58 +02:00
errorIconAtHostOrIP . Visible = true ;
errorLabelAtHostname . Text = Messages . INVALID_HOST ;
errorLabelAtHostname . Visible = true ;
2013-06-24 13:41:48 +02:00
textBoxIscsiHost . Focus ( ) ;
}
2013-10-10 13:22:58 +02:00
else if ( failure ! = null & & failure . ErrorDescription [ 0 ] = = "SR_BACKEND_FAILURE_141" )
{
errorIconAtHostOrIP . Visible = true ;
errorLabelAtHostname . Text = Messages . SR_UNABLE_TO_CONNECT_TO_SCSI_TARGET ;
errorLabelAtHostname . Visible = true ;
textBoxIscsiHost . Focus ( ) ;
}
2015-03-30 15:46:30 +02:00
else if ( failure ! = null & & failure . ErrorDescription [ 0 ] = = "SR_BACKEND_FAILURE_68" )
{
errorIconAtHostOrIP . Visible = true ;
2015-03-30 17:06:49 +02:00
errorLabelAtHostname . Text = Messages . LOGGING_IN_TO_THE_ISCSI_TARGET_FAILED ;
2015-03-30 15:46:30 +02:00
errorLabelAtHostname . Visible = true ;
textBoxIscsiHost . Focus ( ) ;
}
2013-10-10 13:22:58 +02:00
else
{
errorIconAtHostOrIP . Visible = true ;
errorLabelAtHostname . Text = failure . ErrorDescription . Count > 2 ? failure . ErrorDescription [ 2 ] : failure . ErrorDescription [ 0 ] ;
errorLabelAtHostname . Visible = true ;
textBoxIscsiHost . Focus ( ) ;
}
2013-06-24 13:41:48 +02:00
}
2013-10-10 13:22:58 +02:00
scanTargetHostButton . Enabled = true ;
2013-06-24 13:41:48 +02:00
}
2013-10-10 13:22:58 +02:00
private void IscsiPopulateLUNs ( )
2013-06-24 13:41:48 +02:00
{
2013-10-10 13:22:58 +02:00
spinnerIconAtTargetIqn . StartSpinning ( ) ;
2013-06-24 13:41:48 +02:00
comboBoxIscsiLuns . Items . Clear ( ) ;
LunMap . Clear ( ) ;
2018-03-28 17:17:08 +02:00
var chapUser = IscsiUseChapCheckBox . Checked ? IScsiChapUserTextBox . Text : null ;
var chapPwd = IscsiUseChapCheckBox . Checked ? IScsiChapSecretTextBox . Text : null ;
2013-06-24 13:41:48 +02:00
2018-03-28 17:17:08 +02:00
IscsiPopulateLunsAction = SrType = = SR . SRTypes . gfs2
? new Gfs2PopulateLunsAction ( Connection , getIscsiHost ( ) , getIscsiPort ( ) , getIscsiIQN ( ) , chapUser , chapPwd )
: new ISCSIPopulateLunsAction ( Connection , getIscsiHost ( ) , getIscsiPort ( ) , getIscsiIQN ( ) , chapUser , chapPwd ) ;
2013-06-24 13:41:48 +02:00
IscsiPopulateLunsAction . Completed + = IscsiPopulateLunsAction_Completed ;
2013-10-10 13:22:58 +02:00
controlDisabler . Reset ( ) ;
controlDisabler . SaveOrUpdateEnabledStates ( UserInputControls ) ;
controlDisabler . DisableAllControls ( ) ;
IscsiPopulateLunsAction . RunAsync ( ) ;
2013-06-24 13:41:48 +02:00
}
2013-08-05 15:28:21 +02:00
private void IscsiPopulateLunsAction_Completed ( ActionBase sender )
2013-06-24 13:41:48 +02:00
{
2015-04-16 15:17:02 +02:00
Program . Invoke ( this , ( WaitCallback ) IscsiPopulateLunsAction_Completed_ , sender ) ;
2013-06-24 13:41:48 +02:00
}
private void IscsiPopulateLunsAction_Completed_ ( object o )
{
Program . AssertOnEventThread ( ) ;
2013-10-10 13:22:58 +02:00
controlDisabler . RestoreEnabledOnAllControls ( ) ;
ISCSIPopulateLunsAction action = ( ISCSIPopulateLunsAction ) o ;
2013-06-24 13:41:48 +02:00
if ( ! action . Succeeded )
{
2013-10-10 13:22:58 +02:00
spinnerIconAtTargetIqn . Visible = false ;
2013-06-24 13:41:48 +02:00
Failure failure = action . Exception as Failure ;
2013-10-10 13:22:58 +02:00
if ( failure ! = null & & failure . ErrorDescription ! = null & & failure . ErrorDescription . Count > 0 )
2013-06-24 13:41:48 +02:00
{
2013-10-10 13:22:58 +02:00
if ( failure . ErrorDescription [ 0 ] = = "SR_BACKEND_FAILURE_140" )
{
errorIconAtHostOrIP . Visible = true ;
errorLabelAtHostname . Text = Messages . INVALID_HOST ;
errorLabelAtHostname . Visible = true ;
textBoxIscsiHost . Focus ( ) ;
}
else if ( failure . ErrorDescription [ 0 ] = = "SR_BACKEND_FAILURE_68" )
{
errorIconAtCHAPPassword . Visible = true ;
errorLabelAtCHAPPassword . Text = Messages . LOGGING_IN_TO_THE_ISCSI_TARGET_FAILED ;
errorLabelAtCHAPPassword . Visible = true ;
IScsiChapUserTextBox . Focus ( ) ;
}
else
{
errorIconAtTargetLUN . Visible = true ;
errorIconAtTargetLUN . Text = failure . ErrorDescription . Count > 2 ? failure . ErrorDescription [ 2 ] : failure . ErrorDescription [ 0 ] ;
errorIconAtTargetLUN . Visible = true ;
textBoxIscsiHost . Focus ( ) ;
}
2013-06-24 13:41:48 +02:00
}
return ;
}
if ( action . LUNs . Length = = 0 )
{
// Do nothing: ActionProgressDialog will show Messages.NEWSR_NO_LUNS_FOUND
}
else
{
2013-10-10 13:22:58 +02:00
comboBoxIscsiLuns . Items . Add ( Messages . SELECT_TARGET_LUN ) ;
2013-06-24 13:41:48 +02:00
foreach ( Actions . ISCSIInfo i in action . LUNs )
{
String label = "LUN" ;
if ( i . LunID ! = - 1 )
label + = String . Format ( " {0}" , i . LunID ) ;
if ( i . Serial ! = "" )
label + = String . Format ( ": {0}" , i . Serial ) ;
if ( i . Size > = 0 )
label + = String . Format ( ": {0}" , Util . DiskSizeString ( i . Size ) ) ;
if ( i . Vendor ! = "" )
label + = String . Format ( " ({0})" , i . Vendor ) ;
comboBoxIscsiLuns . Items . Add ( label ) ;
LunMap . Add ( label , i ) ;
}
2013-10-16 12:12:00 +02:00
comboBoxIscsiLuns . SelectedItem = comboBoxIscsiLuns . Items . Count = = 2 ? comboBoxIscsiLuns . Items [ 1 ] : Messages . SELECT_TARGET_LUN ; //if there is only one choice, select that one by default
2013-06-24 13:41:48 +02:00
comboBoxIscsiLuns . Enabled = true ;
2013-10-10 13:22:58 +02:00
targetLunLabel . Enabled = true ;
2013-06-24 13:41:48 +02:00
comboBoxIscsiLuns . Focus ( ) ;
2013-10-10 13:22:58 +02:00
2020-03-11 10:17:50 +01:00
spinnerIconAtTargetIqn . ShowSuccessImage ( ) ;
2013-06-24 13:41:48 +02:00
}
2013-10-10 13:22:58 +02:00
comboBoxIscsiLuns . Enabled = true ;
comboBoxIscsiIqns . Enabled = true ;
2013-06-24 13:41:48 +02:00
UpdateButtons ( ) ;
}
/// <summary>
/// Called with the results of an iSCSI SR.probe(), either immediately after the scan, or after the
/// user has performed a scan, clicked 'cancel' on a dialog, and then clicked 'next' again (this
/// avoids duplicate probing if none of the settings have changed).
/// </summary>
/// <returns>
/// Whether to continue or not - wheter to format or not is stored in
/// iScsiFormatLUN.
/// </returns>
2018-03-29 15:32:39 +02:00
private bool ExamineIscsiProbeResults ( SR . SRTypes currentSrType , List < SR . SRInfo > srs )
2013-06-24 13:41:48 +02:00
{
_srToIntroduce = null ;
2018-03-29 15:32:39 +02:00
if ( srs = = null )
2013-06-24 13:41:48 +02:00
return false ;
2018-03-29 11:04:22 +02:00
2013-06-24 13:41:48 +02:00
try
{
if ( ! String . IsNullOrEmpty ( SrWizardType . UUID ) )
{
// Check LUN contains correct SR
2018-03-29 11:04:22 +02:00
if ( srs . Count = = 1 & & srs [ 0 ] . UUID = = SrWizardType . UUID )
2013-06-24 13:41:48 +02:00
{
2018-03-29 11:04:22 +02:00
_srToIntroduce = srs [ 0 ] ;
2018-03-29 15:32:39 +02:00
SrType = currentSrType ; // the type of the existing SR
2013-06-24 13:41:48 +02:00
return true ;
}
2013-10-10 13:22:58 +02:00
errorIconAtTargetLUN . Visible = true ;
errorLabelAtTargetLUN . Visible = true ;
errorLabelAtTargetLUN . Text = String . Format ( Messages . INCORRECT_LUN_FOR_SR , SrWizardType . SrName ) ;
2013-06-24 13:41:48 +02:00
return false ;
}
2018-03-29 11:04:22 +02:00
else if ( srs . Count = = 0 )
2013-06-24 13:41:48 +02:00
{
// No existing SRs were found on this LUN. If allowed to create new SR, ask the user if they want to proceed and format.
if ( ! SrWizardType . AllowToCreateNewSr )
{
2020-04-22 15:47:03 +02:00
using ( var dlg = new ErrorDialog ( Messages . NEWSR_LUN_HAS_NO_SRS ) )
2016-06-20 11:49:12 +02:00
dlg . ShowDialog ( this ) ;
2013-06-24 13:41:48 +02:00
return false ;
}
2016-06-20 11:49:12 +02:00
DialogResult result = DialogResult . Yes ;
if ( ! Program . RunInAutomatedTestMode )
{
2020-04-22 15:47:03 +02:00
using ( var dlg = new WarningDialog ( Messages . NEWSR_ISCSI_FORMAT_WARNING ,
2013-06-24 13:41:48 +02:00
ThreeButtonDialog . ButtonYes ,
2020-04-22 15:47:03 +02:00
new ThreeButtonDialog . TBDButton ( Messages . NO_BUTTON_CAPTION , DialogResult . No , selected : true ) )
{ WindowTitle = Text } )
2016-06-20 11:49:12 +02:00
{
result = dlg . ShowDialog ( this ) ;
}
}
2013-06-24 13:41:48 +02:00
return result = = DialogResult . Yes ;
}
else
{
// There should be 0 or 1 SRs on the LUN
2018-03-29 11:04:22 +02:00
System . Diagnostics . Trace . Assert ( srs . Count = = 1 ) ;
2013-06-24 13:41:48 +02:00
// CA-17230
// Check this isn't a detached SR
2018-03-29 11:04:22 +02:00
SR . SRInfo info = srs [ 0 ] ;
2013-06-24 13:41:48 +02:00
SR sr = SrWizardHelpers . SrInUse ( info . UUID ) ;
if ( sr ! = null )
{
2016-06-20 11:49:12 +02:00
DialogResult res ;
2020-04-22 15:47:03 +02:00
using ( var d = new NoIconDialog ( string . Format ( Messages . DETACHED_ISCI_DETECTED , Helpers . GetName ( sr . Connection ) ) ,
2013-06-24 13:41:48 +02:00
new ThreeButtonDialog . TBDButton ( Messages . ATTACH_SR , DialogResult . OK ) ,
2016-06-20 11:49:12 +02:00
ThreeButtonDialog . ButtonCancel ) )
{
res = d . ShowDialog ( Program . MainWindow ) ;
}
2013-06-24 13:41:48 +02:00
if ( res = = DialogResult . Cancel )
return false ;
_srToIntroduce = info ;
2018-03-29 15:32:39 +02:00
SrType = currentSrType ; // the type of the existing SR
2013-06-24 13:41:48 +02:00
return true ;
}
// An SR exists on this LUN. Ask the user if they want to attach it, format it and
// create a new SR, or cancel.
DialogResult result = Program . RunInAutomatedTestMode ? DialogResult . Yes :
2018-03-29 15:32:39 +02:00
new IscsiChoicesDialog ( Connection , info , currentSrType , SrType ) . ShowDialog ( this ) ;
2013-06-24 13:41:48 +02:00
switch ( result )
{
case DialogResult . Yes :
// Reattach
2018-03-29 11:04:22 +02:00
_srToIntroduce = srs [ 0 ] ;
2018-03-29 15:32:39 +02:00
SrType = currentSrType ; // the type of the existing SR
2013-06-24 13:41:48 +02:00
return true ;
case DialogResult . No :
// Format - SrToIntroduce is already null
return true ;
default :
return false ;
}
}
}
catch
{
// We really want to prevent the user getting to the next step if there is any kind of
// exception here, since clicking 'finish' might destroy data: require another probe.
return false ;
}
}
2013-10-10 13:22:58 +02:00
private void HideAllErrorIconsAndLabels ( )
{
foreach ( var c in ErrorIcons . Concat ( ErrorLabels ) )
c . Visible = false ;
}
private void HideAllSpinnerIcons ( )
{
foreach ( var c in SpinnerControls )
c . Visible = false ;
}
2016-04-28 13:10:24 +02:00
private Dictionary < String , String > GetDeviceConfig ( SR . SRTypes srType )
{
Dictionary < String , String > dconf = new Dictionary < String , String > ( ) ;
ToStringWrapper < IScsiIqnInfo > iqn = comboBoxIscsiIqns . SelectedItem as ToStringWrapper < IScsiIqnInfo > ;
if ( iqn = = null | | ! LunMap . ContainsKey ( getIscsiLUN ( ) ) )
return null ;
if ( srType = = SR . SRTypes . gfs2 )
{
2018-07-12 12:28:03 +02:00
if ( _srToIntroduce ! = null & & _srToIntroduce . Configuration ! = null )
dconf = _srToIntroduce . Configuration ;
2018-03-27 15:17:34 +02:00
dconf [ "provider" ] = "iscsi" ;
2018-05-16 18:08:58 +02:00
dconf [ "target" ] = iqn . item . IpAddress ;
2018-03-27 15:17:34 +02:00
dconf [ "port" ] = iqn . item . Port . ToString ( ) ;
2018-05-16 18:08:58 +02:00
dconf [ "targetIQN" ] = getIscsiIQN ( ) ;
dconf [ "SCSIid" ] = LunMap [ getIscsiLUN ( ) ] . ScsiID ;
2018-03-27 15:17:34 +02:00
dconf [ "chapuser" ] = IScsiChapUserTextBox . Text ;
dconf [ "chappassword" ] = IScsiChapSecretTextBox . Text ;
2016-04-28 13:10:24 +02:00
return dconf ;
}
// Reset target IP address to home address specified in IQN scan.
// Allows multi-homing - see CA-11607
dconf [ TARGET ] = iqn . item . IpAddress ;
dconf [ PORT ] = iqn . item . Port . ToString ( ) ;
dconf [ TARGETIQN ] = getIscsiIQN ( ) ;
ISCSIInfo info = LunMap [ getIscsiLUN ( ) ] ;
if ( info . LunID = = - 1 )
{
dconf [ LUNSERIAL ] = info . Serial ;
}
else
{
dconf [ SCSIID ] = info . ScsiID ;
}
if ( IscsiUseChapCheckBox . Checked )
{
dconf [ CHAPUSER ] = IScsiChapUserTextBox . Text ;
dconf [ CHAPPASSWORD ] = IScsiChapSecretTextBox . Text ;
}
return dconf ;
}
2013-06-24 13:41:48 +02:00
#region Accessors
public SrWizardType SrWizardType { private get ; set ; }
public string UUID { get { return _srToIntroduce = = null ? null : _srToIntroduce . UUID ; } }
2015-08-26 18:14:01 +02:00
2015-08-26 18:41:59 +02:00
public long SRSize
2015-08-26 18:14:01 +02:00
{
get
{
ISCSIInfo info = LunMap [ getIscsiLUN ( ) ] ;
return info . Size ;
}
}
2013-06-24 13:41:48 +02:00
public Dictionary < String , String > DeviceConfig
{
get
{
2017-11-28 11:51:44 +01:00
return GetDeviceConfig ( SrType ) ;
2013-06-24 13:41:48 +02:00
}
}
public string SrDescription
{
get
{
ToStringWrapper < IScsiIqnInfo > iqn = comboBoxIscsiIqns . SelectedItem as ToStringWrapper < IScsiIqnInfo > ;
return iqn = = null ? null : string . Format ( Messages . NEWSR_ISCSI_DESCRIPTION , iqn . item . IpAddress , getIscsiIQN ( ) , getIscsiLUN ( ) ) ;
}
}
2017-11-28 11:51:44 +01:00
public SR . SRTypes SrType { get ; set ; }
2013-06-24 13:41:48 +02:00
#endregion
}
}