// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. #nullable enable using System; using System.Threading; using System.Threading.Tasks; using osu.Framework.Extensions.ExceptionExtensions; using osu.Framework.Logging; namespace osu.Game.Extensions { public static class TaskExtensions { /// /// Denote a task which is to be run without local error handling logic, where failure is not catastrophic. /// Avoids unobserved exceptions from being fired. /// /// The task. /// /// Whether errors should be logged as errors visible to users, or as debug messages. /// Logging as debug will essentially silence the errors on non-release builds. /// public static Task CatchUnobservedExceptions(this Task task, bool logAsError = false) { return task.ContinueWith(t => { Exception? exception = t.Exception?.AsSingular(); if (logAsError) Logger.Error(exception, $"Error running task: {exception?.Message ?? "(unknown)"}", LoggingTarget.Runtime, true); else Logger.Log($"Error running task: {exception}", LoggingTarget.Runtime, LogLevel.Debug); }, TaskContinuationOptions.NotOnRanToCompletion); } /// /// Add a continuation to be performed only after the attached task has completed. /// /// The previous task to be awaited on. /// The action to run. /// An optional cancellation token. Will only cancel the provided action, not the sequence. /// A task representing the provided action. public static Task ContinueWithSequential(this Task task, Action action, CancellationToken cancellationToken = default) => task.ContinueWithSequential(() => Task.Run(action, cancellationToken), cancellationToken); /// /// Add a continuation to be performed only after the attached task has completed. /// /// The previous task to be awaited on. /// The continuation to run. Generally should be an async function. /// An optional cancellation token. Will only cancel the provided action, not the sequence. /// A task representing the provided action. public static Task ContinueWithSequential(this Task task, Func continuationFunction, CancellationToken cancellationToken = default) { var tcs = new TaskCompletionSource(); task.ContinueWith(t => { // the previous task has finished execution or been cancelled, so we can run the provided continuation. if (cancellationToken.IsCancellationRequested) { tcs.SetCanceled(); } else { continuationFunction().ContinueWith(continuationTask => { if (cancellationToken.IsCancellationRequested || continuationTask.IsCanceled) { tcs.TrySetCanceled(); } else if (continuationTask.IsFaulted) { tcs.TrySetException(continuationTask.Exception); } else { tcs.TrySetResult(true); } }, cancellationToken: default); } }, cancellationToken: default); // importantly, we are not returning the continuation itself but rather a task which represents its status in sequential execution order. // this will not be cancelled or completed until the previous task has also. return tcs.Task; } } }