// 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 ValueTask SequenceEqualAsync(this IAsyncEnumerable first, IAsyncEnumerable second, CancellationToken cancellationToken = default) => SequenceEqualAsync(first, second, comparer: null, cancellationToken); public static ValueTask SequenceEqualAsync(this IAsyncEnumerable first, IAsyncEnumerable second, IEqualityComparer? comparer, CancellationToken cancellationToken = default) { if (first == null) throw Error.ArgumentNull(nameof(first)); if (second == null) throw Error.ArgumentNull(nameof(second)); comparer ??= EqualityComparer.Default; if (first is ICollection firstCol && second is ICollection secondCol) { if (firstCol.Count != secondCol.Count) { return new ValueTask(false); } if (firstCol is IList firstList && secondCol is IList secondList) { var count = firstCol.Count; for (var i = 0; i < count; i++) { if (!comparer.Equals(firstList[i], secondList[i])) { return new ValueTask(false); } } return new ValueTask(true); } } return Core(first, second, comparer, cancellationToken); static async ValueTask Core(IAsyncEnumerable first, IAsyncEnumerable second, IEqualityComparer comparer, CancellationToken cancellationToken) { await using (var e1 = first.GetConfiguredAsyncEnumerator(cancellationToken, false)) { await using (var e2 = second.GetConfiguredAsyncEnumerator(cancellationToken, false)) { while (await e1.MoveNextAsync()) { if (!(await e2.MoveNextAsync() && comparer.Equals(e1.Current, e2.Current))) { return false; } } return !await e2.MoveNextAsync(); } } } } } }