// 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 ElementAt(this IAsyncEnumerable source, int index, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (index < 0) throw new ArgumentOutOfRangeException(nameof(index)); return ElementAt_(source, index, cancellationToken); } public static Task ElementAt(this IAsyncEnumerable source, int index) { if (source == null) throw new ArgumentNullException(nameof(source)); return ElementAt(source, index, CancellationToken.None); } public static Task ElementAtOrDefault(this IAsyncEnumerable source, int index, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (index < 0) throw new ArgumentOutOfRangeException(nameof(index)); return ElementAtOrDefault_(source, index, cancellationToken); } public static Task ElementAtOrDefault(this IAsyncEnumerable source, int index) { if (source == null) throw new ArgumentNullException(nameof(source)); return ElementAtOrDefault(source, index, CancellationToken.None); } private static async Task ElementAt_(IAsyncEnumerable source, int index, CancellationToken cancellationToken) { if (source is IList list) { return list[index]; } if (index >= 0) { var e = source.GetAsyncEnumerator(); try { while (await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) { if (index == 0) { return e.Current; } index--; } } finally { await e.DisposeAsync().ConfigureAwait(false); } } throw new ArgumentOutOfRangeException(nameof(index)); } private static async Task ElementAtOrDefault_(IAsyncEnumerable source, int index, CancellationToken cancellationToken) { if (index >= 0) { var e = source.GetAsyncEnumerator(); try { while (await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) { if (index == 0) { return e.Current; } index--; } } finally { await e.DisposeAsync().ConfigureAwait(false); } } return default(TSource); } } }