// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the Apache 2.0 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 AsyncEnumerable { public static Task LastOrDefaultAsync(this IAsyncEnumerable source, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); return Core(source, cancellationToken); static async Task Core(IAsyncEnumerable _source, CancellationToken _cancellationToken) { var last = await TryGetLast(_source, _cancellationToken).ConfigureAwait(false); return last.HasValue ? last.Value : default; } } public static Task LastOrDefaultAsync(this IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return Core(source, predicate, cancellationToken); static async Task Core(IAsyncEnumerable _source, Func _predicate, CancellationToken _cancellationToken) { var last = await TryGetLast(_source, _predicate, _cancellationToken).ConfigureAwait(false); return last.HasValue ? last.Value : default; } } public static Task LastOrDefaultAsync(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return Core(source, predicate, cancellationToken); static async Task Core(IAsyncEnumerable _source, Func> _predicate, CancellationToken _cancellationToken) { var last = await TryGetLast(_source, _predicate, _cancellationToken).ConfigureAwait(false); return last.HasValue ? last.Value : default; } } #if !NO_DEEP_CANCELLATION public static Task LastOrDefaultAsync(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return Core(source, predicate, cancellationToken); static async Task Core(IAsyncEnumerable _source, Func> _predicate, CancellationToken _cancellationToken) { var last = await TryGetLast(_source, _predicate, _cancellationToken).ConfigureAwait(false); return last.HasValue ? last.Value : default; } } #endif private static ValueTask> TryGetLast(IAsyncEnumerable source, CancellationToken cancellationToken) { if (source is IList list) { var count = list.Count; if (count > 0) { return new ValueTask>(new Maybe(list[count - 1])); } } else if (source is IAsyncPartition p) { return p.TryGetLastAsync(cancellationToken); } else { return Core(source, cancellationToken); static async ValueTask> Core(IAsyncEnumerable _source, CancellationToken _cancellationToken) { var last = default(TSource); var hasLast = false; #if USE_AWAIT_FOREACH await foreach (TSource item in _source.WithCancellation(_cancellationToken).ConfigureAwait(false)) { hasLast = true; last = item; } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { hasLast = true; last = e.Current; } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return hasLast ? new Maybe(last) : new Maybe(); } } return new ValueTask>(new Maybe()); } private static async Task> TryGetLast(IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { var last = default(TSource); var hasLast = false; #if USE_AWAIT_FOREACH await foreach (TSource item in source.WithCancellation(cancellationToken).ConfigureAwait(false)) { if (predicate(item)) { hasLast = true; last = item; } } #else var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var value = e.Current; if (predicate(value)) { hasLast = true; last = value; } } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return hasLast ? new Maybe(last) : new Maybe(); } private static async Task> TryGetLast(IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { var last = default(TSource); var hasLast = false; #if USE_AWAIT_FOREACH await foreach (TSource item in source.WithCancellation(cancellationToken).ConfigureAwait(false)) { if (await predicate(item).ConfigureAwait(false)) { hasLast = true; last = item; } } #else var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var value = e.Current; if (await predicate(value).ConfigureAwait(false)) { hasLast = true; last = value; } } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return hasLast ? new Maybe(last) : new Maybe(); } #if !NO_DEEP_CANCELLATION private static async Task> TryGetLast(IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { var last = default(TSource); var hasLast = false; #if USE_AWAIT_FOREACH await foreach (TSource item in source.WithCancellation(cancellationToken).ConfigureAwait(false)) { if (await predicate(item, cancellationToken).ConfigureAwait(false)) { hasLast = true; last = item; } } #else var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var value = e.Current; if (await predicate(value, cancellationToken).ConfigureAwait(false)) { hasLast = true; last = value; } } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return hasLast ? new Maybe(last) : new Maybe(); } #endif } }