/* 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.Threading; using XenAPI; using XenAdmin.Core; namespace XenAdmin.Actions.GUIActions { /// /// A "meddling" Action is one being performed by someone else; in other words, /// they are ones that we've inferred by the presence of task instances on the pool. /// public class MeddlingAction : CancellingAction { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); /// /// Heuristic to determine whether a new task was created by a client /// aware of our task.AppliesTo scheme, or by some other client. /// private static readonly TimeSpan awareClientHeuristic = TimeSpan.FromSeconds(5); private static readonly string XapiExportTaskPrefix = "Export of VM: "; private static readonly string XapiImportTaskName = "VM import"; private static readonly List RecognisedVmOperations = new List { vm_operations.clean_reboot, vm_operations.clean_shutdown, vm_operations.clone, vm_operations.hard_reboot, vm_operations.hard_shutdown, vm_operations.migrate_send, vm_operations.pool_migrate, vm_operations.resume, vm_operations.resume_on, vm_operations.start, vm_operations.start_on, vm_operations.suspend, vm_operations.checkpoint, vm_operations.snapshot, vm_operations.export, vm_operations.import }; private vm_operations vmOperation; public MeddlingAction(Task task) : base(task.Name(), task.Description(), false, false) { RelatedTask = new XenRef(task.opaque_ref); Host = task.Connection.Resolve(task.resident_on) ?? Helpers.GetCoordinator(task.Connection); Started = (task.created + task.Connection.ServerTimeOffset).ToLocalTime(); SetAppliesToData(task); Connection = task.Connection; VM = GetVm(task); vmOperation = GetVmOperation(task); UpdateActionTitleAndDescription(task); Update(task, false); } public void Update(Task task, bool deleting) { ThreadPool.QueueUserWorkItem(delegate { if (!deleting) { RecomputeCanCancel(); } if (task == null || IsCompleted) return; log.DebugFormat("Updating task {0} : {1} - {2}", task.opaque_ref, task.created, task.finished); int percentComplete = task.progress < 0 ? 0 : (int) (100.0*task.progress); if (PercentComplete < percentComplete) PercentComplete = percentComplete; SetFatalErrorData(task); DetermineIfTaskIsComplete(task, deleting); if (IsCompleted) Description = Messages.COMPLETED; DestroyUnwantedOperations(task); if (deleting) LogoutCancelSession(); }); } private void DetermineIfTaskIsComplete(Task task, bool deleting) { if (task.finished.Year > 1970) { DateTime t = task.finished + task.Connection.ServerTimeOffset; Finished = t.ToLocalTime(); IsCompleted = true; } else if (deleting) { Finished = DateTime.Now; IsCompleted = true; } else { StartedRunning = true; } } private void SetFatalErrorData(Task task) { string[] err = task.error_info; if (err != null && err.Length > 0) Exception = new Failure(err); else if (task.status == task_status_type.cancelled) Exception = new CancelledException(); } private void SetAppliesToData(Task task) { List applies_to = task.AppliesTo(); if (applies_to != null) { AppliesTo.AddRange(applies_to); } else { // A non-aware client has created this task. We'll create a new action for this, and place it under // the task.resident_on host, or if that doesn't resolve, the pool coordinator. Host host = task.Connection.Resolve(task.resident_on) ?? Helpers.GetCoordinator(task.Connection); if (host != null) AppliesTo.Add(host.opaque_ref); } } private VM GetVm(Task task) { // try to find the VM in AppliesTo foreach (string r in AppliesTo) { VM vm = task.Connection.Resolve(new XenRef(r)); if (vm != null) return vm; } // try to find a VM in the cache which has this task in its current_operations return task.Connection.Cache.VMs.FirstOrDefault(vm => vm.current_operations.Keys.Contains(task.opaque_ref)); } private static vm_operations GetVmOperation(Task task) { string nl = task.name_label.Replace("Async.", ""); if (nl.StartsWith("VM.")) { nl = nl.Replace("VM.", ""); if (Enum.TryParse(nl, out vm_operations vmOperation) && RecognisedVmOperations.Contains(vmOperation)) { return vmOperation; } } else { // other tasks, e.g. export, import if (task.name_label == XapiImportTaskName) return vm_operations.import; if (task.name_label.StartsWith(XapiExportTaskPrefix)) return vm_operations.export; } return vm_operations.unknown; } private void UpdateActionTitleAndDescription(Task task) { Host host1 = null; Host host2 = null; var appliesTo = task.AppliesTo(); if (appliesTo != null) { foreach (string r in appliesTo) { var host = Connection.Resolve(new XenRef(r)); if (host == null) continue; if (host1 == null) host1 = host; else host2 = host; } } else { host1 = task.Connection.Resolve(task.resident_on) ?? Helpers.GetCoordinator(task.Connection); } List names = new List(); if (VM != null) names.Add(VM.name_label); if (host1 != null) names.Add(host1.name_label); if (host2 != null) names.Add(host2.name_label); string titleFormat; switch (vmOperation) { case vm_operations.clean_reboot: case vm_operations.hard_reboot: titleFormat = names.Count > 1 ? Messages.ACTION_VM_REBOOTING_ON_TITLE : Messages.ACTION_VM_REBOOTING_TITLE; Description = Messages.ACTION_VM_REBOOTING; break; case vm_operations.clean_shutdown: case vm_operations.hard_shutdown: titleFormat = names.Count > 1 ? Messages.ACTION_VM_SHUTTING_DOWN_ON_TITLE : Messages.ACTION_VM_SHUTTING_DOWN_TITLE; Description = Messages.ACTION_VM_SHUTTING_DOWN; break; case vm_operations.clone: titleFormat = Messages.ACTION_VM_COPYING_TITLE_MEDDLING; Description = Messages.ACTION_VM_COPYING; break; case vm_operations.migrate_send: case vm_operations.pool_migrate: titleFormat = names.Count > 2 ? Messages.ACTION_VM_MIGRATING_RESIDENT : Messages.ACTION_VM_MIGRATING_TITLE; Description = Messages.ACTION_VM_MIGRATING; break; case vm_operations.resume: case vm_operations.resume_on: titleFormat = names.Count > 1 ? Messages.ACTION_VM_RESUMING_ON_TITLE : Messages.ACTION_VM_RESUMING_TITLE; Description = Messages.ACTION_VM_RESUMING; break; case vm_operations.start: case vm_operations.start_on: titleFormat = names.Count > 1 ? Messages.ACTION_VM_STARTING_ON_TITLE : Messages.ACTION_VM_STARTING_TITLE; Description = Messages.ACTION_VM_STARTING; break; case vm_operations.suspend: titleFormat =Messages.ACTION_VM_SUSPENDING_TITLE; Description = Messages.ACTION_VM_SUSPENDING; break; case vm_operations.checkpoint: case vm_operations.snapshot: titleFormat = Messages.ACTION_VM_SNAPSHOT_TITLE; Description = Messages.SNAPSHOTTING; break; case vm_operations.export: titleFormat = Messages.ACTION_EXPORT_TASK_NAME; Description = Messages.ACTION_EXPORT_DESCRIPTION_IN_PROGRESS; break; case vm_operations.import: titleFormat = VM == null ? Messages.IMPORTING : names.Count > 1 ? Messages.ACTION_IMPORT_VM_TO_HOST_TITLE : Messages.ACTION_IMPORT_VM_TITLE; Description = Messages.IMPORTING; break; default: titleFormat = task.name_label; Description = task.name_description; break; } try { Title = string.Format(titleFormat, names.ToArray()); } catch (Exception e) { log.Error(e.Message); Title = Description ?? task.name_label; } } /// /// This is one of our tasks, or it's a sub-task of something else, or it corresponds /// to an operation we don't care to recognize. We're going to do no more with it. /// public static bool IsTaskUnwanted(Task task) { return task.GetXenCenterUUID() == Program.XenCenterUUID || task.Connection.Resolve(task.subtask_of) != null || GetVmOperation(task) == vm_operations.unknown; } /// /// Decides whether a MeddlingAction can be created for a given task. /// If AppliesTo is set, then the client that created this task knows about our scheme for passing /// info between clients. /// Otherwise, we give the client a window (awareClientHeuristic) to set this field before we decide /// that it's a non-aware client. /// public static bool IsTaskSuitable(Task task) { return task.AppliesTo() != null || task.created + task.Connection.ServerTimeOffset < DateTime.UtcNow - awareClientHeuristic; } private void DestroyUnwantedOperations(Task task) { string[] err = task.error_info; if (task.Name() == "SR.create" && err != null && err.Length > 0 && err[0] == Failure.SR_BACKEND_FAILURE_107) { // This isn't an SR create at all, it is a scan for LUNs. Hide it, since the 'error' info contains loads of XML, // and is not useful. We don't know this until the error occurs though. Destroy the MeddlingAction. task.PropertyChanged -= MeddlingActionManager.Task_PropertyChanged; ConnectionsManager.History.Remove(this); } } } }