// 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 only element of an async-enumerable sequence, and reports an exception if there is not exactly one element in the async-enumerable sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// Source async-enumerable sequence.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// ValueTask containing the single element in the async-enumerable sequence.
        ///  is null.
        /// (Asynchronous) The source sequence contains more than one element. -or- The source sequence is empty.
        public static ValueTask SingleAsync(this IAsyncEnumerable source, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            return Core(source, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, CancellationToken cancellationToken)
            {
                if (source is IList list)
                {
                    return list.Count switch
                    {
                        0 => throw Error.NoElements(),
                        1 => list[0],
                        _ => throw Error.MoreThanOneElement(),
                    };
                }
                await using var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false);
                if (!await e.MoveNextAsync())
                {
                    throw Error.NoElements();
                }
                var result = e.Current;
                if (await e.MoveNextAsync())
                {
                    throw Error.MoreThanOneElement();
                }
                return result;
            }
        }
        /// 
        /// Returns the only element of an async-enumerable sequence that satisfies the condition in the predicate, and reports an exception if there is not exactly one element in the async-enumerable sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// Source async-enumerable sequence.
        /// A predicate function to evaluate for elements in the source sequence.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// ValueTask containing the single element in the async-enumerable sequence that satisfies the condition in the predicate.
        ///  or  is null.
        /// (Asynchronous) No element satisfies the condition in the predicate. -or- More than one element satisfies the condition in the predicate. -or- The source sequence is empty.
        public static ValueTask SingleAsync(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 ValueTask Core(IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var result = e.Current;
                        if (predicate(result))
                        {
                            while (await e.MoveNextAsync())
                            {
                                if (predicate(e.Current))
                                {
                                    throw Error.MoreThanOneElement();
                                }
                            }
                            return result;
                        }
                    }
                }
                throw Error.NoElements();
            }
        }
        internal static ValueTask SingleAwaitAsyncCore(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 ValueTask Core(IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var result = e.Current;
                        if (await predicate(result).ConfigureAwait(false))
                        {
                            while (await e.MoveNextAsync())
                            {
                                if (await predicate(e.Current).ConfigureAwait(false))
                                {
                                    throw Error.MoreThanOneElement();
                                }
                            }
                            return result;
                        }
                    }
                }
                throw Error.NoElements();
            }
        }
#if !NO_DEEP_CANCELLATION
        internal static ValueTask SingleAwaitWithCancellationAsyncCore(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 ValueTask Core(IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var result = e.Current;
                        if (await predicate(result, cancellationToken).ConfigureAwait(false))
                        {
                            while (await e.MoveNextAsync())
                            {
                                if (await predicate(e.Current, cancellationToken).ConfigureAwait(false))
                                {
                                    throw Error.MoreThanOneElement();
                                }
                            }
                            return result;
                        }
                    }
                }
                throw Error.NoElements();
            }
        }
#endif
    }
}