/* 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.IO; using System.Reflection; using System.Threading; using log4net; using XenAdmin.Controls; using XenAdmin.Diagnostics.Problems; using XenAdmin.Dialogs; using XenAdmin.Wizards.PatchingWizard.PlanActions; using XenAPI; using XenAdmin.Actions; namespace XenAdmin.Wizards.PatchingWizard { public partial class PatchingWizard_PatchingPage : XenTabPage { protected static readonly ILog log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private AsyncAction actionManualMode = null; private bool _thisPageHasBeenCompleted = false; private BackgroundWorker actionsWorker = null; public PatchingWizard_PatchingPage() { InitializeComponent(); } #region Accessors public List SelectedMasters { private get; set; } public List SelectedServers { private get; set; } public List SelectedPools { private get; set; } public UpdateType SelectedUpdateType { private get; set; } public Pool_patch Patch { private get; set; } public string ManualTextInstructions { private get; set; } public bool IsAutomaticMode { private get; set; } public bool RemoveUpdateFile { private get; set; } public string SelectedNewPatch { private get; set; } public List ProblemsResolvedPreCheck { private get; set; } public Dictionary SuppPackVdis { private get; set; } #endregion public override string Text { get { return Messages.PATCHINGWIZARD_PATCHINGPAGE_TEXT; } } public override string PageTitle { get { return Messages.PATCHINGWIZARD_PATCHINGPAGE_TITLE; } } public override string HelpID { get { return "InstallUpdate"; } } public override bool EnablePrevious() { return false; } private bool _nextEnabled; public override bool EnableNext() { return _nextEnabled; } private bool _cancelEnabled; public override bool EnableCancel() { return _cancelEnabled; } public override void PageLoaded(PageLoadedDirection direction) { base.PageLoaded(direction); if (_thisPageHasBeenCompleted) { actionManualMode = null; actionsWorker = null; oemWorker = null; return; } if (SelectedUpdateType == UpdateType.NewOem) { InstallOEMUpdates(); return; } //if we reach here it's either UpdateType.NewRetail, UpdateType.Existing or UpdateType.NewSuppPack if (!IsAutomaticMode) { if (SelectedUpdateType != UpdateType.NewSuppPack) actionManualMode = new ApplyPatchAction(new List { Patch }, SelectedServers); else actionManualMode = new InstallSupplementalPackAction(SuppPackVdis, false); actionManualMode.Changed += action_Changed; actionManualMode.Completed += action_Completed; textBoxLog.Text = ManualTextInstructions; actionManualMode.RunAsync(); return; } _nextEnabled = false; OnPageUpdated(); List planActions = new List(); foreach (Pool pool in SelectedPools) { Pool_patch poolPatch = null; if (SelectedUpdateType != UpdateType.NewSuppPack) { List poolPatches = new List(pool.Connection.Cache.Pool_patches); poolPatch = poolPatches.Find(delegate(Pool_patch otherPatch) { if (Patch != null) return otherPatch.uuid == Patch.uuid; return false; }); } List poolHosts = new List(pool.Connection.Cache.Hosts); Host master = SelectedServers.Find(host => host.IsMaster() && poolHosts.Contains(host)); if (master != null) planActions.AddRange(CompileActionList(master, poolPatch)); foreach (Host server in SelectedServers) { if (poolHosts.Contains(server)) { if (!server.IsMaster()) { // check patch isn't already applied here if (poolPatch == null || poolPatch.AppliedOn(server) == DateTime.MaxValue) planActions.AddRange(CompileActionList(server, poolPatch)); } } } if (RemoveUpdateFile) { planActions.Add(new RemoveUpdateFile(pool, poolPatch)); } } planActions.Add(new UnwindProblemsAction(ProblemsResolvedPreCheck)); actionsWorker = new BackgroundWorker(); actionsWorker.DoWork += new DoWorkEventHandler(PatchingWizardAutomaticPatchWork); actionsWorker.WorkerReportsProgress = true; actionsWorker.ProgressChanged += new ProgressChangedEventHandler(actionsWorker_ProgressChanged); actionsWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(actionsWorker_RunWorkerCompleted); actionsWorker.WorkerSupportsCancellation = true; actionsWorker.RunWorkerAsync(planActions); } #region oem private BackgroundWorker oemWorker = null; private void InstallOEMUpdates() { List actions = new List(); progressBar.Maximum = SelectedMasters.Count * 100; //Upload the patches using patch-upload to the master foreach (Host selectedServer in SelectedMasters) { AsyncAction action = new UploadPatchAction(selectedServer.Connection, SelectedNewPatch); actions.Add(action); } oemWorker = new BackgroundWorker(); oemWorker.WorkerReportsProgress = true; oemWorker.DoWork += new DoWorkEventHandler(oemWorker_DoWork); oemWorker.ProgressChanged += new ProgressChangedEventHandler(oemWorker_ProgressChanged); oemWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(oemWorker_RunWorkerCompleted); oemWorker.RunWorkerAsync(actions); } private void oemWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { Exception exception = e.Result as Exception; if (exception == null) { labelTitle.Text = string.Format(Messages.UPDATE_WAS_SUCCESSFULLY_INSTALLED, Messages.OEM); pictureBox1.Image = null; labelError.Text = Messages.CLOSE_WIZARD_CLICK_FINISH; } else { labelTitle.Text = string.Format(Messages.UPDATE_WAS_NOT_COMPLETED, Messages.OEM); string errorMessage = string.Format("\r\n{0}", exception.Message); textBoxLog.Text += errorMessage; labelError.Text = errorMessage; pictureBox1.Image = SystemIcons.Error.ToBitmap(); panel1.Visible = true; } _nextEnabled = true; _cancelEnabled = false; _thisPageHasBeenCompleted = true; OnPageUpdated(); } private void oemWorker_ProgressChanged(object sender, ProgressChangedEventArgs e) { if (e.ProgressPercentage == 0) { AsyncAction action = (AsyncAction)e.UserState; textBoxLog.Text += string.Format(Messages.UPDATES_WIZARD_APPLYING_UPDATE, Messages.OEM, action.Host.Name); } else if (e.ProgressPercentage == 1) textBoxLog.Text += string.Format("{0}\r\n", Messages.DONE); } private AutoResetEvent autoResetEvent = new AutoResetEvent(false); private int oemActionsDone = 0; private void oemWorker_DoWork(object sender, DoWorkEventArgs e) { List actions = (List)e.Argument; for (int i = 0; i < actions.Count; i++) { AsyncAction currentAction = actions[i]; oemWorker.ReportProgress(0, currentAction); currentAction.Changed += PatchingWizard_PatchingPage_Changed; currentAction.Completed += PatchingWizard_PatchingPage_Completed; currentAction.RunAsync(); autoResetEvent.WaitOne(); try { string result = currentAction.Result; oemWorker.ReportProgress(1, currentAction); } catch (Exception except) { e.Result = except; return; } oemActionsDone++; } } private void PatchingWizard_PatchingPage_Completed(ActionBase sender) { autoResetEvent.Set(); } private void PatchingWizard_PatchingPage_Changed(ActionBase sender) { AsyncAction action = (AsyncAction)sender; Program.Invoke(Program.MainWindow, delegate { progressBar.Value = oemActionsDone * 100 + action.PercentComplete; }); } #endregion #region manual_mode private void action_Completed(ActionBase sender) { _nextEnabled = true; _cancelEnabled = false; try { if (sender.Exception != null) { Program.Invoke(Program.MainWindow, () => FinishedWithErrors(new Exception(sender.Title, sender.Exception))); } else Program.Invoke(Program.MainWindow, FinishedSuccessfully); } catch (Exception except) { Program.Invoke(Program.MainWindow, () => FinishedWithErrors(except)); } Program.Invoke(Program.MainWindow, OnPageUpdated); _thisPageHasBeenCompleted = true; } private void action_Changed(ActionBase sender) { AsyncAction action = (AsyncAction)sender; Program.Invoke(Program.MainWindow, delegate { progressBar.Value = action.PercentComplete; }); } #endregion #region automatic_mode private void actionsWorker_ProgressChanged(object sender, ProgressChangedEventArgs e) { if (!actionsWorker.CancellationPending) { PlanAction action = (PlanAction)e.UserState; if (e.ProgressPercentage == 0) textBoxLog.Text += action; else { textBoxLog.Text += string.Format("{0}\r\n", Messages.DONE); progressBar.Value += e.ProgressPercentage; } } } private void PatchingWizardAutomaticPatchWork(object obj, DoWorkEventArgs doWorkEventArgs) { List actionList = (List)doWorkEventArgs.Argument; foreach (PlanAction action in actionList) { try { if (actionsWorker.CancellationPending) { doWorkEventArgs.Cancel = true; return; } this.actionsWorker.ReportProgress(0, action); action.Run(); Thread.Sleep(1000); this.actionsWorker.ReportProgress((int)((1.0 / (double)actionList.Count) * 100), action); } catch (Exception e) { log.Error("Failed to carry out plan.", e); log.Debug(actionList); doWorkEventArgs.Result = new Exception(action.Title, e); break; } } } private void actionsWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { if (!e.Cancelled) { Exception exception = e.Result as Exception; if (exception != null) { FinishedWithErrors(exception); } else { FinishedSuccessfully(); } progressBar.Value = 100; _cancelEnabled = false; _nextEnabled = true; } OnPageUpdated(); _thisPageHasBeenCompleted = true; } private List CompileActionList(Host host, Pool_patch patch) { if (SelectedUpdateType == UpdateType.NewSuppPack) return CompileSuppPackActionList(host); List actions = new List(); if (patch == null) return actions; List> runningVMs = RunningVMs(host, patch); if (!host.isOEM) actions.Add(new ApplyPatchPlanAction(host, patch)); if (patch.after_apply_guidance.Contains(after_apply_guidance.restartHost)) { actions.Add(new EvacuateHostPlanAction(host)); actions.Add(new RebootHostPlanAction(host)); actions.Add(new BringBabiesBackAction(runningVMs, host, false)); } if (patch.after_apply_guidance.Contains(after_apply_guidance.restartXAPI)) { actions.Add(new RestartAgentPlanAction(host)); } if (patch.after_apply_guidance.Contains(after_apply_guidance.restartHVM)) { actions.Add(new RebootVMsPlanAction(host, RunningHvmVMs(host))); } if (patch.after_apply_guidance.Contains(after_apply_guidance.restartPV)) { actions.Add(new RebootVMsPlanAction(host, RunningPvVMs(host))); } return actions; } private static List> RunningVMs(Host host, Pool_patch patch) { List> vms = new List>(); foreach (VM vm in patch.Connection.ResolveAll(host.resident_VMs)) { if (!vm.is_a_real_vm) continue; vms.Add(new XenRef(vm.opaque_ref)); } return vms; } private static List> RunningHvmVMs(Host host) { List> vms = new List>(); foreach (VM vm in host.Connection.ResolveAll(host.resident_VMs)) { if (!vm.IsHVM || !vm.is_a_real_vm) continue; vms.Add(new XenRef(vm.opaque_ref)); } return vms; } private static List> RunningPvVMs(Host host) { List> vms = new List>(); foreach (VM vm in host.Connection.ResolveAll(host.resident_VMs)) { if (vm.IsHVM || !vm.is_a_real_vm) continue; vms.Add(new XenRef(vm.opaque_ref)); } return vms; } private static List> RunningVMs(Host host) { List> vms = new List>(); foreach (VM vm in host.Connection.ResolveAll(host.resident_VMs)) { if (!vm.is_a_real_vm) continue; vms.Add(new XenRef(vm.opaque_ref)); } return vms; } private List CompileSuppPackActionList(Host host) { List actions = new List(); if (SelectedUpdateType != UpdateType.NewSuppPack || SuppPackVdis == null || !SuppPackVdis.ContainsKey(host)) return actions; List> runningVMs = RunningVMs(host); actions.Add(new InstallSupplementalPackPlanAction(host, SuppPackVdis[host])); // after_apply_guidance is restartHost actions.Add(new EvacuateHostPlanAction(host)); actions.Add(new RebootHostPlanAction(host)); actions.Add(new BringBabiesBackAction(runningVMs, host, false)); return actions; } #endregion private void FinishedWithErrors(Exception exception) { labelTitle.Text = string.Format(Messages.UPDATE_WAS_NOT_COMPLETED, GetUpdateName()); string errorMessage = string.Format(Messages.PATCHING_WIZARD_ERROR, exception.Message); labelError.Text = errorMessage; pictureBox1.Image = SystemIcons.Error.ToBitmap(); if (exception.InnerException is SupplementalPackInstallFailedException) { errorLinkLabel.Visible = true; errorLinkLabel.Tag = exception.InnerException; } panel1.Visible = true; } private void FinishedSuccessfully() { labelTitle.Text = string.Format(Messages.UPDATE_WAS_SUCCESSFULLY_INSTALLED, GetUpdateName()); pictureBox1.Image = null; labelError.Text = Messages.CLOSE_WIZARD_CLICK_FINISH; } private string GetUpdateName() { if (Patch != null) return Patch.Name; try { return new FileInfo(SelectedNewPatch).Name; } catch (Exception) { return SelectedNewPatch; } } private void errorLinkLabel_LinkClicked(object sender, System.Windows.Forms.LinkLabelLinkClickedEventArgs e) { if (errorLinkLabel.Tag is Exception) ShowMoreInfo(errorLinkLabel.Tag as Exception); } private void ShowMoreInfo(Exception exception) { if (!(exception is SupplementalPackInstallFailedException)) return; var msg = string.Format(Messages.SUPP_PACK_INSTALL_FAILED_MORE_INFO, exception.Message); new ThreeButtonDialog( new ThreeButtonDialog.Details(SystemIcons.Error, msg)) .ShowDialog(this); } } }