// 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);
}
}
}
}
}