/* 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.Collections.Generic; using System.Linq; using System.Windows.Forms; using XenAPI; using XenAdmin.Actions; using XenAdmin.Core; using XenAdmin.Dialogs; using XenAdmin.Network; using XenAdmin.Wizards; namespace XenAdmin.Commands { /// /// IF HA is not enabled, it launches the HA wizard, otherwise the HA config dialog /// internal class HAConfigureCommand : HACommand { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); private RbacMethodList HA_PERMISSION_CHECKS = new RbacMethodList( "pool.set_ha_host_failures_to_tolerate", "pool.sync_database", "vm.set_ha_restart_priority", "pool.ha_compute_hypothetical_max_host_failures_to_tolerate" ); /// /// Initializes a new instance of this Command. The parameter-less constructor is required if /// this Command is to be attached to a ToolStrip menu item or button. It should not be used in any other scenario. /// public HAConfigureCommand() { } public HAConfigureCommand(IMainWindow mainWindow, IEnumerable selection) : base(mainWindow, selection) { } public HAConfigureCommand(IMainWindow mainWindow, IXenConnection connection) : base(mainWindow, Helpers.GetPoolOfOne(connection)) { } public HAConfigureCommand(IMainWindow mainWindow, Pool pool) : base(mainWindow, pool) { } private void Execute(IXenConnection connection) { if (connection == null) return; Pool pool = Helpers.GetPool(connection); if (pool == null) return; if (Helpers.FeatureForbidden(pool, Host.RestrictHA)) { UpsellDialog.ShowUpsellDialog(Messages.UPSELL_BLURB_HA, Parent); } else if (pool.ha_enabled) { if (pool.ha_statefiles.All(sf => pool.Connection.Resolve(new XenRef(sf)) == null))//empty gives true, which is correct { log.ErrorFormat("Cannot resolve HA statefile VDI (pool {0} has {1} statefiles).", pool.Name(), pool.ha_statefiles.Length); using (var dlg = new ErrorDialog(string.Format(Messages.HA_CONFIGURE_NO_STATEFILE, Helpers.GetName(pool).Ellipsise(30)), ThreeButtonDialog.ButtonOK) { WindowTitle = Messages.CONFIGURE_HA }) { dlg.ShowDialog(Program.MainWindow); } } else if (!Role.CanPerform(HA_PERMISSION_CHECKS, pool.Connection)) { var msg = string.Format(Messages.RBAC_HA_CONFIGURE_WARNING, Role.FriendlyCSVRoleList(Role.ValidRoleList(HA_PERMISSION_CHECKS, pool.Connection)), Role.FriendlyCSVRoleList(pool.Connection.Session.Roles)); using (var dlg = new ErrorDialog(msg)) dlg.ShowDialog(Parent); } else { MainWindowCommandInterface.ShowPerConnectionWizard(connection, new EditVmHaPrioritiesDialog(pool)); } } else { MainWindowCommandInterface.ShowPerConnectionWizard(connection, new HAWizard(pool)); } } protected override void ExecuteCore(SelectedItemCollection selection) { Execute(selection[0].Connection); } protected override bool CanExecuteCore(SelectedItemCollection selection) { return CanExecuteHACommand(selection); } protected override string GetCantExecuteReasonCore(IXenObject item) { var reason = base.GetCantExecuteReasonCore(item); if (!string.IsNullOrEmpty(reason) && reason != Messages.UNKNOWN) return reason; Pool pool = item == null ? null : Helpers.GetPool(item.Connection); if (pool != null && !pool.Connection.Cache.Hosts.Any(Host.RestrictPoolSecretRotation) && pool.is_psr_pending) return Messages.ROTATE_POOL_SECRET_PENDING_HA; return Messages.UNKNOWN; } protected override bool CanExecute(Pool pool) { return pool.Connection.Cache.Hosts.Any(Host.RestrictPoolSecretRotation) || !pool.is_psr_pending; } public override string MenuText => Messages.CONFIGURE_HA_ELLIPSIS; } internal class HADisableCommand : HACommand { /// /// Initializes a new instance of this Command. The parameter-less constructor is required if /// this Command is to be attached to a ToolStrip menu item or button. It should not be used in any other scenario. /// public HADisableCommand() { } public HADisableCommand(IMainWindow mainWindow, IEnumerable selection) : base(mainWindow, selection) { } public HADisableCommand(IMainWindow mainWindow, IXenConnection connection) : base(mainWindow, Helpers.GetPoolOfOne(connection)) { } protected override void ExecuteCore(SelectedItemCollection selection) { Pool pool = selection.Count == 1 ? selection[0].PoolAncestor : null; if (pool == null || !pool.ha_enabled) return; if (pool.ha_statefiles.All(sf => pool.Connection.Resolve(new XenRef(sf)) == null)) //empty gives true, which is correct { using (var dlg = new ErrorDialog(string.Format(Messages.HA_DISABLE_NO_STATEFILE, Helpers.GetName(pool).Ellipsise(30)), ThreeButtonDialog.ButtonOK) { WindowTitle = Messages.DISABLE_HA }) { dlg.ShowDialog(Parent); return; } } // Confirm the user wants to disable HA using (var dlg = new NoIconDialog(string.Format(Messages.HA_DISABLE_QUERY, Helpers.GetName(pool).Ellipsise(30)), ThreeButtonDialog.ButtonYes, ThreeButtonDialog.ButtonNo) { HelpNameSetter = "HADisable", WindowTitle = Messages.DISABLE_HA }) { if (dlg.ShowDialog(Parent) != DialogResult.Yes) return; } var action = new DisableHAAction(pool); // We will need to re-enable buttons when the action completes action.Completed += Program.MainWindow.action_Completed; action.RunAsync(); } protected override bool CanExecuteCore(SelectedItemCollection selection) { return CanExecuteHACommand(selection); } protected override bool CanExecute(Pool pool) { return pool.ha_enabled; } public override string MenuText => Messages.DISABLE_HA_HOTKEY; } internal class HACommand : Command { public HACommand() { } public HACommand(IMainWindow mainWindow, IEnumerable selection) : base(mainWindow, selection) { } public HACommand(IMainWindow mainWindow, IXenConnection connection) : base(mainWindow, Helpers.GetPoolOfOne(connection)) { } public HACommand(IMainWindow mainWindow, Pool pool) : base(mainWindow, pool) { } protected override bool CanExecuteCore(SelectedItemCollection selection) { return new HAConfigureCommand(MainWindowCommandInterface, selection).CanExecute() || new HADisableCommand(MainWindowCommandInterface, selection).CanExecute(); } protected override string GetCantExecuteReasonCore(IXenObject item) { Pool pool = item == null ? null : Helpers.GetPoolOfOne(item.Connection); if (pool == null) return string.Format(Messages.POOL_GONE, BrandManager.BrandConsole); if (!pool.IsVisible()) return Messages.HA_STANDALONE_SERVER; Host master = Helpers.GetMaster(pool.Connection); if (master == null) return string.Format(Messages.POOL_MASTER_GONE, BrandManager.BrandConsole); if (pool.Locked) return Messages.POOL_EDIT_IN_PROGRESS; var action = HelpersGUI.FindActiveHaAction(pool.Connection); if (action != null) return string.Format(action is EnableHAAction ? Messages.HA_PAGE_ENABLING : Messages.HA_PAGE_DISABLING, Helpers.GetName(pool.Connection)); return Messages.UNKNOWN; } protected bool CanExecuteHACommand(SelectedItemCollection selection) { if (selection.Count != 1) return false; Pool pool = selection[0].PoolAncestor; if (pool == null || pool.Locked || pool.Connection == null || !pool.Connection.IsConnected || Helpers.GetMaster(pool.Connection) == null || HelpersGUI.FindActiveHaAction(pool.Connection) != null) return false; return CanExecute(pool); } protected virtual bool CanExecute(Pool pool) { return true; } public override string MenuText => Messages.MAINWINDOW_HIGH_AVAILABILITY; } }