// 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)); #if HAS_ASYNC_ENUMERABLE_CANCELLATION return Core(source, finallyAction); static async IAsyncEnumerable Core(IAsyncEnumerable source, Action finallyAction, [System.Runtime.CompilerServices.EnumeratorCancellation]CancellationToken cancellationToken = default) #else return AsyncEnumerable.Create(Core); async IAsyncEnumerator Core(CancellationToken cancellationToken) #endif { 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)); #if HAS_ASYNC_ENUMERABLE_CANCELLATION return Core(source, finallyAction); static async IAsyncEnumerable Core(IAsyncEnumerable source, Func finallyAction, [System.Runtime.CompilerServices.EnumeratorCancellation]CancellationToken cancellationToken = default) #else return AsyncEnumerable.Create(Core); async IAsyncEnumerator Core(CancellationToken cancellationToken) #endif { 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); } } } } }