// 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> ToDictionaryAsync(this IAsyncEnumerable source, Func keySelector, CancellationToken cancellationToken = default) => ToDictionaryAsync(source, keySelector, comparer: null, cancellationToken); public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func keySelector, IEqualityComparer comparer, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (keySelector == null) throw Error.ArgumentNull(nameof(keySelector)); return Core(source, keySelector, comparer, cancellationToken); async Task> Core(IAsyncEnumerable _source, Func _keySelector, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var d = new Dictionary(_comparer); #if CSHARP8 await foreach (TSource item in source.WithCancellation(cancellationToken).ConfigureAwait(false)) { var key = _keySelector(item); d.Add(key, item); } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var item = e.Current; var key = _keySelector(item); d.Add(key, item); } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return d; } } public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, CancellationToken cancellationToken = default) => ToDictionaryAsync(source, keySelector, comparer: null, cancellationToken); public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, IEqualityComparer comparer, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (keySelector == null) throw Error.ArgumentNull(nameof(keySelector)); return Core(source, keySelector, comparer, cancellationToken); async Task> Core(IAsyncEnumerable _source, Func> _keySelector, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var d = new Dictionary(_comparer); #if CSHARP8 await foreach (TSource item in _source.WithCancellation(_cancellationToken).ConfigureAwait(false)) { var key = await _keySelector(item).ConfigureAwait(false); d.Add(key, item); } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var item = e.Current; var key = await _keySelector(item).ConfigureAwait(false); d.Add(key, item); } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return d; } } #if !NO_DEEP_CANCELLATION public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, CancellationToken cancellationToken = default) => ToDictionaryAsync(source, keySelector, comparer: null, cancellationToken); public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, IEqualityComparer comparer, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (keySelector == null) throw Error.ArgumentNull(nameof(keySelector)); return Core(source, keySelector, comparer, cancellationToken); async Task> Core(IAsyncEnumerable _source, Func> _keySelector, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var d = new Dictionary(_comparer); #if CSHARP8 await foreach (TSource item in _source.WithCancellation(_cancellationToken).ConfigureAwait(false)) { var key = await _keySelector(item, _cancellationToken).ConfigureAwait(false); d.Add(key, item); } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var item = e.Current; var key = await _keySelector(item, _cancellationToken).ConfigureAwait(false); d.Add(key, item); } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return d; } } #endif public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func keySelector, Func elementSelector, CancellationToken cancellationToken = default) => ToDictionaryAsync(source, keySelector, elementSelector, comparer: null, cancellationToken); public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func keySelector, Func elementSelector, IEqualityComparer comparer, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (keySelector == null) throw Error.ArgumentNull(nameof(keySelector)); if (elementSelector == null) throw Error.ArgumentNull(nameof(elementSelector)); return Core(source, keySelector, elementSelector, comparer, cancellationToken); async Task> Core(IAsyncEnumerable _source, Func _keySelector, Func _elementSelector, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var d = new Dictionary(_comparer); #if CSHARP8 await foreach (TSource item in _source.WithCancellation(_cancellationToken).ConfigureAwait(false)) { var key = _keySelector(item); var value = _elementSelector(item); d.Add(key, value); } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var item = e.Current; var key = _keySelector(item); var value = _elementSelector(item); d.Add(key, value); } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return d; } } public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, Func> elementSelector, CancellationToken cancellationToken = default) => ToDictionaryAsync(source, keySelector, elementSelector, comparer: null, cancellationToken); public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, Func> elementSelector, IEqualityComparer comparer, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (keySelector == null) throw Error.ArgumentNull(nameof(keySelector)); if (elementSelector == null) throw Error.ArgumentNull(nameof(elementSelector)); return Core(source, keySelector, elementSelector, comparer, cancellationToken); async Task> Core(IAsyncEnumerable _source, Func> _keySelector, Func> _elementSelector, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var d = new Dictionary(_comparer); #if CSHARP8 await foreach (TSource item in _source.WithCancellation(_cancellationToken).ConfigureAwait(false)) { var key = await _keySelector(item).ConfigureAwait(false); var value = await _elementSelector(item).ConfigureAwait(false); d.Add(key, value); } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var item = e.Current; var key = await _keySelector(item).ConfigureAwait(false); var value = await _elementSelector(item).ConfigureAwait(false); d.Add(key, value); } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return d; } } #if !NO_DEEP_CANCELLATION public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, Func> elementSelector, CancellationToken cancellationToken = default) => ToDictionaryAsync(source, keySelector, elementSelector, comparer: null, cancellationToken); public static Task> ToDictionaryAsync(this IAsyncEnumerable source, Func> keySelector, Func> elementSelector, IEqualityComparer comparer, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (keySelector == null) throw Error.ArgumentNull(nameof(keySelector)); if (elementSelector == null) throw Error.ArgumentNull(nameof(elementSelector)); return Core(source, keySelector, elementSelector, comparer, cancellationToken); async Task> Core(IAsyncEnumerable _source, Func> _keySelector, Func> _elementSelector, IEqualityComparer _comparer, CancellationToken _cancellationToken) { var d = new Dictionary(_comparer); #if CSHARP8 await foreach (TSource item in _source.WithCancellation(_cancellationToken).ConfigureAwait(false)) { var key = await _keySelector(item, _cancellationToken).ConfigureAwait(false); var value = await _elementSelector(item, _cancellationToken).ConfigureAwait(false); d.Add(key, value); } #else var e = _source.GetAsyncEnumerator(_cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { var item = e.Current; var key = await _keySelector(item, _cancellationToken).ConfigureAwait(false); var value = await _elementSelector(item, _cancellationToken).ConfigureAwait(false); d.Add(key, value); } } finally { await e.DisposeAsync().ConfigureAwait(false); } #endif return d; } } #endif } }