// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT License. // See the LICENSE file in the project root for more information. using System.Collections.Generic; using System.Threading; using System.Threading.Tasks; namespace System.Linq { public static partial class AsyncEnumerableEx { /// /// Invokes a specified action after the source async-enumerable sequence terminates gracefully or exceptionally. /// /// The type of the elements in the source sequence. /// Source sequence. /// Action to invoke after the source async-enumerable sequence terminates. /// Source sequence with the action-invoking termination behavior applied. /// or is null. public static IAsyncEnumerable Finally(this IAsyncEnumerable source, Action finallyAction) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (finallyAction == null) throw Error.ArgumentNull(nameof(finallyAction)); return Core(source, finallyAction); static async IAsyncEnumerable Core(IAsyncEnumerable source, Action finallyAction, [System.Runtime.CompilerServices.EnumeratorCancellation] CancellationToken cancellationToken = default) { try { await foreach (var item in source.WithCancellation(cancellationToken).ConfigureAwait(false)) { yield return item; } } finally { finallyAction(); } } } /// /// Invokes a specified asynchronous action after the source async-enumerable sequence terminates gracefully or exceptionally. /// /// The type of the elements in the source sequence. /// Source sequence. /// Action to invoke and await asynchronously after the source async-enumerable sequence terminates. /// Source sequence with the action-invoking termination behavior applied. /// or is null. public static IAsyncEnumerable Finally(this IAsyncEnumerable source, Func finallyAction) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (finallyAction == null) throw Error.ArgumentNull(nameof(finallyAction)); return Core(source, finallyAction); static async IAsyncEnumerable Core(IAsyncEnumerable source, Func finallyAction, [System.Runtime.CompilerServices.EnumeratorCancellation] CancellationToken cancellationToken = default) { try { await foreach (var item in source.WithCancellation(cancellationToken).ConfigureAwait(false)) { yield return item; } } finally { // REVIEW: No cancellation support for finally action. await finallyAction().ConfigureAwait(false); } } } } }