// 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 SequenceEqualAsync(this IAsyncEnumerable first, IAsyncEnumerable second, CancellationToken cancellationToken = default) { if (first == null) throw Error.ArgumentNull(nameof(first)); if (second == null) throw Error.ArgumentNull(nameof(second)); return SequenceEqualCore(first, second, comparer: null, cancellationToken); } public static Task 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)); return SequenceEqualCore(first, second, comparer, cancellationToken); } private static Task SequenceEqualCore(IAsyncEnumerable first, IAsyncEnumerable second, IEqualityComparer comparer, CancellationToken cancellationToken) { if (comparer == null) { comparer = EqualityComparer.Default; } if (first is ICollection firstCol && second is ICollection secondCol) { if (firstCol.Count != secondCol.Count) { return Task.FromResult(false); } if (firstCol is IList firstList && secondCol is IList secondList) { int count = firstCol.Count; for (int i = 0; i < count; i++) { if (!comparer.Equals(firstList[i], secondList[i])) { return Task.FromResult(false); } } return Task.FromResult(true); } } return Core(first, second, comparer, cancellationToken); async Task Core(IAsyncEnumerable _first, IAsyncEnumerable _second, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var e1 = _first.GetAsyncEnumerator(_cancellationToken); try { var e2 = _second.GetAsyncEnumerator(_cancellationToken); try { while (await e1.MoveNextAsync().ConfigureAwait(false)) { if (!(await e2.MoveNextAsync().ConfigureAwait(false) && _comparer.Equals(e1.Current, e2.Current))) { return false; } } return !await e2.MoveNextAsync().ConfigureAwait(false); } finally { await e2.DisposeAsync().ConfigureAwait(false); } } finally { await e1.DisposeAsync().ConfigureAwait(false); } } } } }