// 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 SingleOrDefaultAsync(this IAsyncEnumerable source) { if (source == null) throw Error.ArgumentNull(nameof(source)); return SingleOrDefaultCore(source, CancellationToken.None); } public static Task SingleOrDefaultAsync(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); return SingleOrDefaultCore(source, cancellationToken); } public static Task SingleOrDefaultAsync(this IAsyncEnumerable source, Func predicate) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return SingleOrDefaultCore(source, predicate, CancellationToken.None); } public static Task SingleOrDefaultAsync(this IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return SingleOrDefaultCore(source, predicate, cancellationToken); } public static Task SingleOrDefaultAsync(this IAsyncEnumerable source, Func> predicate) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return SingleOrDefaultCore(source, predicate, CancellationToken.None); } public static Task SingleOrDefaultAsync(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return SingleOrDefaultCore(source, predicate, cancellationToken); } #if !NO_DEEP_CANCELLATION public static Task SingleOrDefaultAsync(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return SingleOrDefaultCore(source, predicate, cancellationToken); } #endif private static async Task SingleOrDefaultCore(IAsyncEnumerable source, CancellationToken cancellationToken) { if (source is IList list) { switch (list.Count) { case 0: return default; case 1: return list[0]; } throw Error.MoreThanOneElement(); } IAsyncEnumerator e = source.GetAsyncEnumerator(cancellationToken); try { if (!await e.MoveNextAsync().ConfigureAwait(false)) { return default; } TSource result = e.Current; if (!await e.MoveNextAsync().ConfigureAwait(false)) { return result; } } finally { await e.DisposeAsync().ConfigureAwait(false); } throw Error.MoreThanOneElement(); } private static async Task SingleOrDefaultCore(IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { IAsyncEnumerator e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { TSource result = e.Current; if (predicate(result)) { while (await e.MoveNextAsync().ConfigureAwait(false)) { if (predicate(e.Current)) { throw Error.MoreThanOneElement(); } } return result; } } return default; } finally { await e.DisposeAsync().ConfigureAwait(false); } } private static async Task SingleOrDefaultCore(IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { IAsyncEnumerator e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { TSource result = e.Current; if (await predicate(result).ConfigureAwait(false)) { while (await e.MoveNextAsync().ConfigureAwait(false)) { if (await predicate(e.Current).ConfigureAwait(false)) { throw Error.MoreThanOneElement(); } } return result; } } return default; } finally { await e.DisposeAsync().ConfigureAwait(false); } } #if !NO_DEEP_CANCELLATION private static async Task SingleOrDefaultCore(IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { IAsyncEnumerator e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { TSource result = e.Current; if (await predicate(result, cancellationToken).ConfigureAwait(false)) { while (await e.MoveNextAsync().ConfigureAwait(false)) { if (await predicate(e.Current, cancellationToken).ConfigureAwait(false)) { throw Error.MoreThanOneElement(); } } return result; } } return default; } finally { await e.DisposeAsync().ConfigureAwait(false); } } #endif } }