First.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the Apache 2.0 License.
  3. // See the LICENSE file in the project root for more information.
  4. using System;
  5. using System.Collections.Generic;
  6. using System.Linq;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. namespace System.Linq
  10. {
  11. public static partial class AsyncEnumerable
  12. {
  13. public static Task<TSource> First<TSource>(this IAsyncEnumerable<TSource> source)
  14. {
  15. if (source == null)
  16. throw new ArgumentNullException(nameof(source));
  17. return First(source, CancellationToken.None);
  18. }
  19. public static Task<TSource> First<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, bool> predicate)
  20. {
  21. if (source == null)
  22. throw new ArgumentNullException(nameof(source));
  23. if (predicate == null)
  24. throw new ArgumentNullException(nameof(predicate));
  25. return First(source, predicate, CancellationToken.None);
  26. }
  27. public static Task<TSource> First<TSource>(this IAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
  28. {
  29. if (source == null)
  30. throw new ArgumentNullException(nameof(source));
  31. return First_(source, cancellationToken);
  32. }
  33. public static Task<TSource> First<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, bool> predicate, CancellationToken cancellationToken)
  34. {
  35. if (source == null)
  36. throw new ArgumentNullException(nameof(source));
  37. if (predicate == null)
  38. throw new ArgumentNullException(nameof(predicate));
  39. return source.Where(predicate)
  40. .First(cancellationToken);
  41. }
  42. public static Task<TSource> FirstOrDefault<TSource>(this IAsyncEnumerable<TSource> source)
  43. {
  44. if (source == null)
  45. throw new ArgumentNullException(nameof(source));
  46. return FirstOrDefault(source, CancellationToken.None);
  47. }
  48. public static Task<TSource> FirstOrDefault<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, bool> predicate)
  49. {
  50. if (source == null)
  51. throw new ArgumentNullException(nameof(source));
  52. if (predicate == null)
  53. throw new ArgumentNullException(nameof(predicate));
  54. return FirstOrDefault(source, predicate, CancellationToken.None);
  55. }
  56. public static Task<TSource> FirstOrDefault<TSource>(this IAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
  57. {
  58. if (source == null)
  59. throw new ArgumentNullException(nameof(source));
  60. return FirstOrDefault_(source, cancellationToken);
  61. }
  62. public static Task<TSource> FirstOrDefault<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, bool> predicate, CancellationToken cancellationToken)
  63. {
  64. if (source == null)
  65. throw new ArgumentNullException(nameof(source));
  66. if (predicate == null)
  67. throw new ArgumentNullException(nameof(predicate));
  68. return source.Where(predicate)
  69. .FirstOrDefault(cancellationToken);
  70. }
  71. private static async Task<TSource> First_<TSource>(IAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
  72. {
  73. var list = source as IList<TSource>;
  74. if (list?.Count > 0)
  75. {
  76. return list[0];
  77. }
  78. var e = source.GetAsyncEnumerator();
  79. try
  80. {
  81. if (await e.MoveNextAsync(cancellationToken)
  82. .ConfigureAwait(false))
  83. {
  84. return e.Current;
  85. }
  86. }
  87. finally
  88. {
  89. await e.DisposeAsync().ConfigureAwait(false);
  90. }
  91. throw new InvalidOperationException(Strings.NO_ELEMENTS);
  92. }
  93. private static async Task<TSource> FirstOrDefault_<TSource>(IAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
  94. {
  95. var list = source as IList<TSource>;
  96. if (list?.Count > 0)
  97. {
  98. return list[0];
  99. }
  100. var e = source.GetAsyncEnumerator();
  101. try
  102. {
  103. if (await e.MoveNextAsync(cancellationToken)
  104. .ConfigureAwait(false))
  105. {
  106. return e.Current;
  107. }
  108. }
  109. finally
  110. {
  111. await e.DisposeAsync().ConfigureAwait(false);
  112. }
  113. return default(TSource);
  114. }
  115. }
  116. }