/* Copyright (c) Cloud Software Group, Inc. * * 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.Net; using System.Reflection; using System.Threading; using XenAdmin.Network; namespace XenAPI { partial class Task { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); public delegate task_status_type TaskStatusOp(Session session, string task); public delegate double TaskProgressOp(Session session, string task); public delegate Task TaskGetRecordOp(Session session, string task); /// /// Try and run the delegate. /// If it fails with a web exception or invalid session, try again. /// Only retry 60 times. /// public static Object DoWithSessionRetry(IXenConnection connection, ref Session session, Delegate f, params object[] p) { int retries = 60; while (true) { try { object[] ps = new object[p.Length + 1]; ps[0] = session; for (int i = 0; i < p.Length; i++) { ps[i + 1] = p[i]; } try { return f.DynamicInvoke(ps); } catch (TargetInvocationException exn) { if (exn.InnerException != null) throw exn.InnerException; throw; } } catch (WebException we) { log.Error($"WebException in DoWithSessionRetry, retry {retries}: ", we); if (retries <= 0) throw; } catch (Failure failure) { log.Error($"Failure in DoWithSessionRetry, retry {retries}", failure); if (retries <= 0) throw; if (failure.ErrorDescription.Count < 1 || failure.ErrorDescription[0] != Failure.SESSION_INVALID) throw; } Session newSession; try { // try to create a new TCP stream to use, as the other one has failed us newSession = connection.DuplicateSession(); session = newSession; } catch { // ignored } retries--; Thread.Sleep(connection.ExpectDisruption ? 500 : 100); } } /// /// A list of OpaqueRefs of objects to which the current task applies. This is set /// by one XenCenter instance, and picked up by other ones. /// public List AppliesTo() { string s = Get(other_config, "applies_to"); return s == null ? null : new List(s.Split(',')); } public static void SetAppliesTo(Session session, string task, List applies_to) { try { remove_from_other_config(session, task, "applies_to"); add_to_other_config(session, task, "applies_to", string.Join(",", applies_to.ToArray())); } catch (Failure f) { if (f.ErrorDescription[0] == Failure.RBAC_PERMISSION_DENIED) { // Read only user without task.other_config rights - just ignore this request return; } throw; } } public string GetXenCenterUUID() { return Get(other_config, "XenCenterUUID"); } public static void SetXenCenterUUID(Session session, string task, string uuid) { try { remove_from_other_config(session, task, "XenCenterUUID"); add_to_other_config(session, task, "XenCenterUUID", uuid); } catch (Failure f) { if (f.ErrorDescription.Count > 0 && f.ErrorDescription[0] == Failure.RBAC_PERMISSION_DENIED) { // Read only user without task.other_config rights - just ignore this request return; } throw; } } public override string Name() { return name_label.Replace("Async.", ""); } public override string Description() { return name_description; } public bool IgnoreInCacheUpdate() { switch (name_label) { case "SR.scan": return true; default: return false; } } } }