namespace Utils { public static class Time { public static void Sleep(TimeSpan span) { Thread.Sleep(span); } public static T Wait(Task task) { task.Wait(); return task.Result; } public static string FormatDuration(TimeSpan d) { var result = ""; if (d.Days > 0) result += $"{d.Days} days, "; if (d.Hours > 0) result += $"{d.Hours} hours, "; if (d.Minutes > 0) result += $"{d.Minutes} mins, "; result += $"{d.Seconds} secs"; return result; } public static void WaitUntil(Func predicate) { WaitUntil(predicate, TimeSpan.FromMinutes(1), TimeSpan.FromSeconds(1)); } public static void WaitUntil(Func predicate, TimeSpan timeout, TimeSpan retryDelay) { var start = DateTime.UtcNow; var state = predicate(); while (!state) { if (DateTime.UtcNow - start > timeout) { throw new TimeoutException("Operation timed out."); } Sleep(retryDelay); state = predicate(); } } public static void Retry(Action action, string description) { Retry(action, 1, description); } public static T Retry(Func action, string description) { return Retry(action, 1, description); } public static void Retry(Action action, int maxRetries, string description) { Retry(action, maxRetries, TimeSpan.FromSeconds(1), description); } public static T Retry(Func action, int maxRetries, string description) { return Retry(action, maxRetries, TimeSpan.FromSeconds(1), description); } public static void Retry(Action action, int maxRetries, TimeSpan retryTime, string description) { var start = DateTime.UtcNow; var retries = 0; var exceptions = new List(); while (true) { if (retries > maxRetries) { var duration = DateTime.UtcNow - start; throw new TimeoutException($"Retry '{description}' timed out after {maxRetries} tries over {Time.FormatDuration(duration)}.", new AggregateException(exceptions)); } try { action(); return; } catch (Exception ex) { exceptions.Add(ex); retries++; } Sleep(retryTime); } } public static T Retry(Func action, int maxRetries, TimeSpan retryTime, string description) { var start = DateTime.UtcNow; var retries = 0; var exceptions = new List(); while (true) { if (retries > maxRetries) { var duration = DateTime.UtcNow - start; throw new TimeoutException($"Retry '{description}' timed out after {maxRetries} tries over {Time.FormatDuration(duration)}.", new AggregateException(exceptions)); } try { return action(); } catch (Exception ex) { exceptions.Add(ex); retries++; } Sleep(retryTime); } } } }