// 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 First(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return First(source, CancellationToken.None); } public static Task First(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return First_(source, cancellationToken); } public static Task First(this IAsyncEnumerable source, Func predicate) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return First(source, predicate, CancellationToken.None); } public static Task First(this IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return source.Where(predicate).First(cancellationToken); } public static Task First(this IAsyncEnumerable source, Func> predicate) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return First(source, predicate, CancellationToken.None); } public static Task First(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return source.Where(predicate).First(cancellationToken); } public static Task FirstOrDefault(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return FirstOrDefault(source, CancellationToken.None); } public static Task FirstOrDefault(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return FirstOrDefault_(source, cancellationToken); } public static Task FirstOrDefault(this IAsyncEnumerable source, Func predicate) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return FirstOrDefault(source, predicate, CancellationToken.None); } public static Task FirstOrDefault(this IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return source.Where(predicate).FirstOrDefault(cancellationToken); } public static Task FirstOrDefault(this IAsyncEnumerable source, Func> predicate) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return FirstOrDefault(source, predicate, CancellationToken.None); } public static Task FirstOrDefault(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (predicate == null) throw new ArgumentNullException(nameof(predicate)); return source.Where(predicate).FirstOrDefault(cancellationToken); } private static async Task First_(IAsyncEnumerable source, CancellationToken cancellationToken) { var list = source as IList; if (list?.Count > 0) { return list[0]; } var e = source.GetAsyncEnumerator(); try { if (await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) { return e.Current; } } finally { await e.DisposeAsync().ConfigureAwait(false); } throw new InvalidOperationException(Strings.NO_ELEMENTS); } private static async Task FirstOrDefault_(IAsyncEnumerable source, CancellationToken cancellationToken) { var list = source as IList; if (list?.Count > 0) { return list[0]; } var e = source.GetAsyncEnumerator(); try { if (await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) { return e.Current; } } finally { await e.DisposeAsync().ConfigureAwait(false); } return default(TSource); } } }