// 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
{
///
/// Returns the element at a specified index in a sequence.
///
/// The type of the elements in the source sequence.
/// async-enumerable sequence to return the element from.
/// The zero-based index of the element to retrieve.
/// The optional cancellation token to be used for cancelling the sequence at any time.
/// An async-enumerable sequence that produces the element at the specified position in the source sequence.
/// is null.
/// is less than zero.
/// (Asynchronous) is greater than or equal to the number of elements in the source sequence.
public static ValueTask ElementAtAsync(this IAsyncEnumerable source, int index, CancellationToken cancellationToken = default)
{
if (source == null)
throw Error.ArgumentNull(nameof(source));
return Core(source, index, cancellationToken);
static async ValueTask Core(IAsyncEnumerable source, int index, CancellationToken cancellationToken)
{
if (source is IAsyncPartition p)
{
var first = await p.TryGetElementAtAsync(index, cancellationToken).ConfigureAwait(false);
if (first.HasValue)
{
return first.Value;
}
}
else
{
if (source is IList list)
{
return list[index];
}
if (index >= 0)
{
await foreach (var item in source.WithCancellation(cancellationToken).ConfigureAwait(false))
{
if (index == 0)
{
return item;
}
index--;
}
}
}
// NB: Even though index is captured, no closure is created.
// The nameof expression is lowered to a literal prior to creating closures.
throw Error.ArgumentOutOfRange(nameof(index));
}
}
}
}