// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT 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 AsyncEnumerableEx
    {
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    if (!await e.MoveNextAsync())
                    {
                        throw Error.NoElements();
                    }
                    long sum = selector(e.Current);
                    long count = 1;
                    checked
                    {
                        while (await e.MoveNextAsync())
                        {
                            sum += selector(e.Current);
                            ++count;
                        }
                    }
                    return (double)sum / count;
                }
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    if (!await e.MoveNextAsync())
                    {
                        throw Error.NoElements();
                    }
                    long sum = selector(e.Current);
                    long count = 1;
                    checked
                    {
                        while (await e.MoveNextAsync())
                        {
                            sum += selector(e.Current);
                            ++count;
                        }
                    }
                    return (double)sum / count;
                }
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    if (!await e.MoveNextAsync())
                    {
                        throw Error.NoElements();
                    }
                    double sum = selector(e.Current);
                    long count = 1;
                    checked
                    {
                        while (await e.MoveNextAsync())
                        {
                            sum += selector(e.Current);
                            ++count;
                        }
                    }
                    return (float)(sum / count);
                }
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    if (!await e.MoveNextAsync())
                    {
                        throw Error.NoElements();
                    }
                    double sum = selector(e.Current);
                    long count = 1;
                    checked
                    {
                        while (await e.MoveNextAsync())
                        {
                            sum += selector(e.Current);
                            ++count;
                        }
                    }
                    return sum / count;
                }
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    if (!await e.MoveNextAsync())
                    {
                        throw Error.NoElements();
                    }
                    decimal sum = selector(e.Current);
                    long count = 1;
                    checked
                    {
                        while (await e.MoveNextAsync())
                        {
                            sum += selector(e.Current);
                            ++count;
                        }
                    }
                    return sum / count;
                }
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var v = selector(e.Current);
                        if (v.HasValue)
                        {
                            long sum = v.GetValueOrDefault();
                            long count = 1;
                            checked
                            {
                                while (await e.MoveNextAsync())
                                {
                                    v = selector(e.Current);
                                    if (v.HasValue)
                                    {
                                        sum += v.GetValueOrDefault();
                                        ++count;
                                    }
                                }
                            }
                            return (double)sum / count;
                        }
                    }
                }
                return null;
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var v = selector(e.Current);
                        if (v.HasValue)
                        {
                            long sum = v.GetValueOrDefault();
                            long count = 1;
                            checked
                            {
                                while (await e.MoveNextAsync())
                                {
                                    v = selector(e.Current);
                                    if (v.HasValue)
                                    {
                                        sum += v.GetValueOrDefault();
                                        ++count;
                                    }
                                }
                            }
                            return (double)sum / count;
                        }
                    }
                }
                return null;
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var v = selector(e.Current);
                        if (v.HasValue)
                        {
                            double sum = v.GetValueOrDefault();
                            long count = 1;
                            checked
                            {
                                while (await e.MoveNextAsync())
                                {
                                    v = selector(e.Current);
                                    if (v.HasValue)
                                    {
                                        sum += v.GetValueOrDefault();
                                        ++count;
                                    }
                                }
                            }
                            return (float)(sum / count);
                        }
                    }
                }
                return null;
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var v = selector(e.Current);
                        if (v.HasValue)
                        {
                            double sum = v.GetValueOrDefault();
                            long count = 1;
                            checked
                            {
                                while (await e.MoveNextAsync())
                                {
                                    v = selector(e.Current);
                                    if (v.HasValue)
                                    {
                                        sum += v.GetValueOrDefault();
                                        ++count;
                                    }
                                }
                            }
                            return sum / count;
                        }
                    }
                }
                return null;
            }
        }
        /// 
        /// Computes the average of an async-enumerable sequence of  values that are obtained by invoking a transform function on each element of the input sequence.
        /// 
        /// The type of the elements in the source sequence.
        /// A sequence of values to calculate the average of.
        /// A transform function to apply to each element.
        /// The optional cancellation token to be used for cancelling the sequence at any time.
        /// An async-enumerable sequence containing a single element with the average of the sequence of values, or null if the source sequence is empty or contains only values that are null.
        ///  or  is null.
        /// (Asynchronous) The source sequence is empty.
        /// The return type of this operator differs from the corresponding operator on IEnumerable in order to retain asynchronous behavior.
        public static ValueTask AverageAsync(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken = default)
        {
            if (source == null)
                throw Error.ArgumentNull(nameof(source));
            if (selector == null)
                throw Error.ArgumentNull(nameof(selector));
            return Core(source, selector, cancellationToken);
            static async ValueTask Core(IAsyncEnumerable source, Func selector, CancellationToken cancellationToken)
            {
                await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
                {
                    while (await e.MoveNextAsync())
                    {
                        var v = selector(e.Current);
                        if (v.HasValue)
                        {
                            decimal sum = v.GetValueOrDefault();
                            long count = 1;
                            checked
                            {
                                while (await e.MoveNextAsync())
                                {
                                    v = selector(e.Current);
                                    if (v.HasValue)
                                    {
                                        sum += v.GetValueOrDefault();
                                        ++count;
                                    }
                                }
                            }
                            return sum / count;
                        }
                    }
                }
                return null;
            }
        }
    }
}