/* 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.ComponentModel; using System.Drawing; using System.Linq; using System.Threading; using System.Windows.Forms; using XenAdmin.Actions; using XenAdmin.Controls; using XenAdmin.Core; using XenAdmin.Diagnostics.Checks; using XenAdmin.Diagnostics.Checks.DR; using XenAdmin.Diagnostics.Problems; using XenAdmin.Dialogs; using XenAPI; using XenAdmin.Actions.DR; namespace XenAdmin.Wizards.DRWizards { public partial class DRFailoverWizardPrecheckPage : XenTabPage { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); private readonly BackgroundWorker _worker; private int _numberChecks; private readonly object _lock = new object(); private bool _enableNext; public event Action NewDrTaskIntroduced; public event Action> SrIntroduced; private Pool _pool; public Pool Pool { set { _pool = value; } } private Dictionary, PoolMetadata> selectedPoolMetadata; public Dictionary, PoolMetadata> SelectedPoolMetadata {set { selectedPoolMetadata = value; }} public List RevertActions = new List(); private string _nextButtonText = Messages.WIZARD_BUTTON_NEXT; public override string NextText(bool isLastPage) { return _nextButtonText; } public DRFailoverWizardPrecheckPage() { InitializeComponent(); dataGridView1.BackgroundColor = dataGridView1.DefaultCellStyle.BackColor; _worker = new BackgroundWorker { WorkerReportsProgress = true, WorkerSupportsCancellation = true }; _worker.DoWork += worker_DoWork; _worker.ProgressChanged += worker_ProgressChanged; _worker.RunWorkerCompleted += _worker_RunWorkerCompleted; panelErrorsFound.Visible = false; labelPrecheckStatus.Visible = false; checkBoxHideSuccessful.Checked = true; } public override string PageTitle { get { switch (WizardType) { case DRWizardType.Failback: return Messages.DR_WIZARD_PRECHECKPAGE_TITLE_FAILBACK; default: return Messages.DR_WIZARD_PRECHECKPAGE_TITLE_FAILOVER; } } } public override string Text => Messages.DR_WIZARD_PRECHECKPAGE_TEXT; public override string HelpID { get { switch (WizardType) { case DRWizardType.Failback: return "Failback_Prechecks"; case DRWizardType.Dryrun: return "Dryrun_Prechecks"; default: return "Failover_Prechecks"; } } } public override bool EnablePrevious() { return _worker != null && !_worker.IsBusy; } private int PercentageSelectedObjects(int i) { int count = 0; foreach (var poolMetadata in selectedPoolMetadata.Values) { count += (poolMetadata.VmAppliances.Count + poolMetadata.Vms.Count); } return (int)(((float)i / (float)(count)) * 100); } protected override void PageLoadedCore(PageLoadedDirection direction) { SetupLabels(); try { if (direction == PageLoadedDirection.Back) return; RefreshRechecks(); } catch (Exception e) { log.Error(e, e); throw;//better throw an exception rather than closing the wizard suddenly and silently } } public DRWizardType WizardType { private get; set; } private void SetupLabels() { switch (WizardType) { case DRWizardType.Failback: labelContinue.Text = Messages.DR_WIZARD_PRECHECKPAGE_CONTINUE_FAILBACK; _nextButtonText = Messages.DR_WIZARD_PRECHECKPAGE_NEXT_FAILBACK; break; case DRWizardType.Dryrun: labelContinue.Text = Messages.DR_WIZARD_PRECHECKPAGE_CONTINUE_DRYRUN; _nextButtonText = Messages.DR_WIZARD_PRECHECKPAGE_NEXT_FAILOVER; break; default: labelContinue.Text = Messages.DR_WIZARD_PRECHECKPAGE_CONTINUE_FAILOVER; _nextButtonText = Messages.DR_WIZARD_PRECHECKPAGE_NEXT_FAILOVER; break; } } private void RefreshRechecks() { buttonResolveAll.Enabled = buttonReCheckProblems.Enabled = false; panelErrorsFound.Visible = false; labelPrecheckStatus.Visible = true; _worker.RunWorkerAsync(); OnPageUpdated(); } private void _worker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { if (e.Cancelled) { OnPageUpdated(); return; } progressBar1.Value = 100; bool problemsFound = false; bool allFixable = true; foreach (PreCheckGridRow row in dataGridView1.Rows) { if (row is PreCheckItemRow preCheckRow && preCheckRow.Problem != null) { problemsFound = true; if (!preCheckRow.IsFixable) { allFixable = false; break; } } } labelPrecheckStatus.Visible = false; buttonReCheckProblems.Enabled = true; panelErrorsFound.Visible = problemsFound; buttonResolveAll.Enabled = problemsFound && allFixable; _enableNext = !problemsFound; OnPageUpdated(); } private void worker_ProgressChanged(object sender, ProgressChangedEventArgs e) { try { if (e.UserState is DataGridViewRow row && !dataGridView1.Rows.Contains(row)) { if (row is PreCheckHeaderRow headerRow) headerRow.Visible = true; else if (row is PreCheckItemRow preCheckRow && preCheckRow.IsProblem) preCheckRow.Visible = true; else row.Visible = !checkBoxHideSuccessful.Checked; dataGridView1.Rows.Add(row); } int step = (int)(1.0 / ((float)_numberChecks) * e.ProgressPercentage); progressBar1.Value += step + progressBar1.Value > 100 ? 0 : step; } catch (Exception) { // ignored } } private DataGridViewRow RunCheck(Check check) { var problems = check.RunAllChecks(); if (problems.Count != 0) { // None of the checks used in the DR wizard returns more than one problem, so we just take the first. // (Even if they did, we would only suffer the usability problem described in CA-77990). return new PreCheckItemRow(problems[0]); } return new PreCheckItemRow(check); } private void worker_DoWork(object sender, DoWorkEventArgs e) { lock (_lock) { Program.Invoke(this, () => { dataGridView1.Rows.Clear(); progressBar1.Value = 0; }); RevertActions.Clear(); List>> checks = GenerateChecks(); _numberChecks = checks.Count; for (int i = 0; i < checks.Count; i++) { List checkGroup = checks[i].Value; DataGridViewRow row = new PreCheckHeaderRow(string.Format(Messages.DR_WIZARD_PRECHECKPAGE_HEADER, i + 1, checks.Count, checks[i].Key)); _worker.ReportProgress(5, row); Session metadataSession = null; // run checks for (int j = 0; j < checkGroup.Count; j++) { if (_worker.CancellationPending) { e.Cancel = true; return; } Check check = checkGroup[j]; // special case - AssertCanBeRecoveredCheck - we need to open the metadata database if (check is AssertCanBeRecoveredCheck) { AssertCanBeRecoveredCheck thisCheck = check as AssertCanBeRecoveredCheck; AssertCanBeRecoveredCheck prevCheck = j > 0 ? checkGroup[j - 1] as AssertCanBeRecoveredCheck : null; if (prevCheck == null || prevCheck.Vdi.uuid != thisCheck.Vdi.uuid) { // close previous metadata session if (metadataSession != null) metadataSession.logout(); // open metadata database VdiOpenDatabaseAction action = new VdiOpenDatabaseAction(Connection, ((AssertCanBeRecoveredCheck )checkGroup[0]).Vdi); action.RunSync(action.Session); if (action.Succeeded && action.MetadataSession != null) metadataSession = action.MetadataSession; } // run check if (metadataSession != null) { thisCheck.MetadataSession = metadataSession; row = RunCheck(thisCheck); _worker.ReportProgress(PercentageSelectedObjects(j + 1), row); // close metadata session if this is the last check if (j == checkGroup.Count - 1) metadataSession.logout(); } } else { row = RunCheck(check); _worker.ReportProgress(PercentageSelectedObjects(j + 1), row); } } } } } private List>> GenerateChecks() { List>> checks = new List>>(); List checkGroup; //HA checks if (WizardType == DRWizardType.Dryrun) { checks.Add(new KeyValuePair>(Messages.DR_WIZARD_CHECKING_HA_STATUS, new List())); checkGroup = checks[checks.Count - 1].Value; checkGroup.Add(new DrHAEnabledCheck(_pool)); } // check if selected VMs and appliances are still running in the source pool if (WizardType != DRWizardType.Dryrun) { checkGroup = new List(); foreach (PoolMetadata poolMetadata in selectedPoolMetadata.Values) { // is source pool connected? PoolMetadata metadata = poolMetadata; Pool sourcePool = ConnectionsManager.XenConnectionsCopy.Select (connection => Helpers.GetPoolOfOne(connection)).FirstOrDefault (pool => pool != null && pool.uuid == metadata.Pool.uuid); if (sourcePool != null) { foreach (VM_appliance vmAppliance in poolMetadata.VmAppliances.Values) { checkGroup.Add(new RunningVmApplianceCheck(vmAppliance, sourcePool)); } foreach (VM vm in poolMetadata.Vms.Values) { if (vm.IsAssignedToVapp()) { //VM included in an appliance continue; } checkGroup.Add(new RunningVmCheck(vm, sourcePool)); } } } if (checkGroup.Count > 0) checks.Add( new KeyValuePair>( Messages.DR_WIZARD_CHECKING_POWER_STATE_IN_SOURCE_POOLS, checkGroup)); } //Existing VMs and appliances checks checkGroup = new List(); foreach (PoolMetadata poolMetadata in selectedPoolMetadata.Values) { foreach (VM_appliance vmAppliance in poolMetadata.VmAppliances.Values) { PoolMetadata metadata = poolMetadata; List vms = (from vmRef in vmAppliance.VMs where metadata.Vms.ContainsKey(vmRef) select metadata.Vms[vmRef]).ToList(); checkGroup.Add(new ExistingVmApplianceCheck(vmAppliance, vms, _pool)); } foreach (VM vm in poolMetadata.Vms.Values) { if (vm.IsAssignedToVapp()) { //VM included in an appliance continue; } checkGroup.Add(new ExistingVmCheck(vm, _pool)); } } if (checkGroup.Count > 0) checks.Add(new KeyValuePair>(String.Format(Messages.DR_WIZARD_CHECKING_EXISTING_APPLIANCES_AND_VMS, _pool.Name()), checkGroup)); //VM and appliance can be recovered checks checks.Add(new KeyValuePair>(String.Format(Messages.DR_WIZARD_CHECKING_VMS_CAN_BE_RECOVERED, _pool.Name()), new List())); checkGroup = checks[checks.Count - 1].Value; foreach (PoolMetadata poolMetadata in selectedPoolMetadata.Values) { foreach (VM_appliance vmAppliance in poolMetadata.VmAppliances.Values) { checkGroup.Add(new AssertCanBeRecoveredCheck(vmAppliance, _pool, poolMetadata.Vdi)); } foreach (VM vm in poolMetadata.Vms.Values) { if (vm.IsAssignedToVapp()) { //VM included in an appliance continue; } checkGroup.Add(new AssertCanBeRecoveredCheck(vm, _pool, poolMetadata.Vdi)); } } return checks; } protected override void PageLeaveCore(PageLoadedDirection direction, ref bool cancel) { if (direction == PageLoadedDirection.Back) _worker.CancelAsync(); } public override bool EnableNext() { return _enableNext; } public override void PageCancelled(ref bool cancel) { _worker.CancelAsync(); } internal List GetWarnings() { var warnings = new List(); foreach (DataGridViewRow row in dataGridView1.Rows) { var preCheckRow = row as PreCheckItemRow; if (preCheckRow != null && preCheckRow.IsWarning) { warnings.Add(preCheckRow.Description); } } return warnings; } private void RunSolution(PreCheckItemRow preCheckRow) { bool cancelled; AsyncAction action = preCheckRow.Problem.GetSolutionAction(out cancelled); if (action != null) { action.Completed += action_Completed; using (var progressDialog = new ActionProgressDialog(action, ProgressBarStyle.Blocks)) progressDialog.ShowDialog(this); if (action.Succeeded) { var revertAction = preCheckRow.Problem.CreateUnwindChangesAction(); if (revertAction != null) RevertActions.Add(revertAction); } } } private void action_Completed(ActionBase sender) { Thread.Sleep(1000); Program.Invoke(Program.MainWindow, RefreshRechecks); var drTaskCreateAction = sender as DrTaskCreateAction; if (drTaskCreateAction != null && drTaskCreateAction.Succeeded) { if (NewDrTaskIntroduced != null) NewDrTaskIntroduced(drTaskCreateAction.Result); return; } var srIntroduceAction = sender as SrIntroduceAction; if (srIntroduceAction != null && srIntroduceAction.Succeeded) { if (SrIntroduced != null) SrIntroduced(new XenRef(srIntroduceAction.Result)); } } private void buttonReCheckProblems_Click(object sender, EventArgs e) { RefreshRechecks(); } private void buttonResolveAll_Click(object sender, EventArgs e) { Dictionary actions = new Dictionary(); foreach (DataGridViewRow row in dataGridView1.Rows) { PreCheckItemRow preCheckRow = row as PreCheckItemRow; if (preCheckRow != null && preCheckRow.Problem != null) { bool cancelled; AsyncAction action = preCheckRow.Problem.GetSolutionAction(out cancelled); if (action != null) { actions.Add(action, preCheckRow.Problem.CreateUnwindChangesAction()); } } } foreach (var asyncAction in actions.Keys) { using (var progressDialog = new ActionProgressDialog(asyncAction, ProgressBarStyle.Blocks)) progressDialog.ShowDialog(this); if (asyncAction.Succeeded && actions[asyncAction] != null) RevertActions.Add(actions[asyncAction]); Program.Invoke(Program.MainWindow, RefreshRechecks); } Program.Invoke(Program.MainWindow, RefreshRechecks); } private void checkBoxViewPrecheckFailuresOnly_CheckedChanged(object sender, EventArgs e) { foreach (DataGridViewRow row in dataGridView1.Rows) { if (row is PreCheckHeaderRow headerRow) headerRow.Visible = true; else if (row is PreCheckItemRow preCheckRow && preCheckRow.IsProblem) preCheckRow.Visible = true; else row.Visible = !checkBoxHideSuccessful.Checked; } } private void dataGridView1_CellContentClick(object sender, DataGridViewCellEventArgs e) { if (dataGridView1.Rows[e.RowIndex] is PreCheckItemRow preCheckRow && e.ColumnIndex == 2) RunSolution(preCheckRow); } #region PreCheckGridRow private abstract class PreCheckGridRow : DataGridViewRow { protected DataGridViewImageCell _iconCell = new DataGridViewImageCell(); protected DataGridViewTextBoxCell _descriptionCell = new DataGridViewTextBoxCell(); protected DataGridViewCell _solutionCell = new DataGridViewLinkCell(); protected PreCheckGridRow() { Cells.Add(_iconCell); Cells.Add(_descriptionCell); Cells.Add(_solutionCell); } } private class PreCheckHeaderRow : PreCheckGridRow { public PreCheckHeaderRow(string text) { _iconCell.Value = new Bitmap(1, 1); _descriptionCell.Value = text; _descriptionCell.Style.Font = new Font(Program.DefaultFont, FontStyle.Bold); } } private class PreCheckItemRow : PreCheckGridRow { private readonly Check _check; public PreCheckItemRow(Problem problem) { Problem = problem; _check = problem.Check; UpdateRowFields(); } public PreCheckItemRow(Check check) { _check = check; UpdateRowFields(); } private void UpdateRowFields() { _iconCell.Value = Problem == null ? Images.GetImage16For(Icons.Ok) : Problem.Image; if (Problem != null) _descriptionCell.Value = String.Format(Messages.STRING_COLON_SPACE_STRING, _check.Description, Problem.Description); else if (_check != null) { _descriptionCell.Value = String.Format(Messages.DR_WIZARD_PRECHECKPAGE_OK, _check.Description); } _solutionCell.Value = Problem == null ? string.Empty : Problem.HelpMessage; } public Problem Problem { get; } public string Description { get { return _descriptionCell.Value as string; } } public bool IsProblem => Problem != null && !(Problem is Warning); public bool IsFixable => Problem != null && Problem.IsFixable && !string.IsNullOrEmpty(Problem.HelpMessage); public bool IsWarning => Problem != null && Problem is Warning; public override bool Equals(object obj) { if (obj is PreCheckItemRow other && other.Problem != null && Problem != null) return Problem.CompareTo(other.Problem) == 0; return false; } public override int GetHashCode() { return Problem != null ? Problem.GetHashCode() : 0; } } #endregion } }