Do.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  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.Collections.Generic;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. namespace System.Linq
  8. {
  9. public static partial class AsyncEnumerableEx
  10. {
  11. // REVIEW: Should we convert Task-based overloads to ValueTask?
  12. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Action<TSource> onNext)
  13. {
  14. if (source == null)
  15. throw Error.ArgumentNull(nameof(source));
  16. if (onNext == null)
  17. throw Error.ArgumentNull(nameof(onNext));
  18. return DoCore(source, onNext: onNext, onError: null, onCompleted: null);
  19. }
  20. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Action<TSource> onNext, Action onCompleted)
  21. {
  22. if (source == null)
  23. throw Error.ArgumentNull(nameof(source));
  24. if (onNext == null)
  25. throw Error.ArgumentNull(nameof(onNext));
  26. if (onCompleted == null)
  27. throw Error.ArgumentNull(nameof(onCompleted));
  28. return DoCore(source, onNext: onNext, onError: null, onCompleted: onCompleted);
  29. }
  30. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Action<TSource> onNext, Action<Exception> onError)
  31. {
  32. if (source == null)
  33. throw Error.ArgumentNull(nameof(source));
  34. if (onNext == null)
  35. throw Error.ArgumentNull(nameof(onNext));
  36. if (onError == null)
  37. throw Error.ArgumentNull(nameof(onError));
  38. return DoCore(source, onNext: onNext, onError: onError, onCompleted: null);
  39. }
  40. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Action<TSource> onNext, Action<Exception> onError, Action onCompleted)
  41. {
  42. if (source == null)
  43. throw Error.ArgumentNull(nameof(source));
  44. if (onNext == null)
  45. throw Error.ArgumentNull(nameof(onNext));
  46. if (onError == null)
  47. throw Error.ArgumentNull(nameof(onError));
  48. if (onCompleted == null)
  49. throw Error.ArgumentNull(nameof(onCompleted));
  50. return DoCore(source, onNext, onError, onCompleted);
  51. }
  52. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, Task> onNext)
  53. {
  54. if (source == null)
  55. throw Error.ArgumentNull(nameof(source));
  56. if (onNext == null)
  57. throw Error.ArgumentNull(nameof(onNext));
  58. return DoCore(source, onNext: onNext, onError: null, onCompleted: null);
  59. }
  60. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, Task> onNext, Func<Task> onCompleted)
  61. {
  62. if (source == null)
  63. throw Error.ArgumentNull(nameof(source));
  64. if (onNext == null)
  65. throw Error.ArgumentNull(nameof(onNext));
  66. if (onCompleted == null)
  67. throw Error.ArgumentNull(nameof(onCompleted));
  68. return DoCore(source, onNext: onNext, onError: null, onCompleted: onCompleted);
  69. }
  70. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, Task> onNext, Func<Exception, Task> onError)
  71. {
  72. if (source == null)
  73. throw Error.ArgumentNull(nameof(source));
  74. if (onNext == null)
  75. throw Error.ArgumentNull(nameof(onNext));
  76. if (onError == null)
  77. throw Error.ArgumentNull(nameof(onError));
  78. return DoCore(source, onNext: onNext, onError: onError, onCompleted: null);
  79. }
  80. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, Task> onNext, Func<Exception, Task> onError, Func<Task> onCompleted)
  81. {
  82. if (source == null)
  83. throw Error.ArgumentNull(nameof(source));
  84. if (onNext == null)
  85. throw Error.ArgumentNull(nameof(onNext));
  86. if (onError == null)
  87. throw Error.ArgumentNull(nameof(onError));
  88. if (onCompleted == null)
  89. throw Error.ArgumentNull(nameof(onCompleted));
  90. return DoCore(source, onNext, onError, onCompleted);
  91. }
  92. #if !NO_DEEP_CANCELLATION
  93. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, Task> onNext)
  94. {
  95. if (source == null)
  96. throw Error.ArgumentNull(nameof(source));
  97. if (onNext == null)
  98. throw Error.ArgumentNull(nameof(onNext));
  99. return DoCore(source, onNext: onNext, onError: null, onCompleted: null);
  100. }
  101. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, Task> onNext, Func<CancellationToken, Task> onCompleted)
  102. {
  103. if (source == null)
  104. throw Error.ArgumentNull(nameof(source));
  105. if (onNext == null)
  106. throw Error.ArgumentNull(nameof(onNext));
  107. if (onCompleted == null)
  108. throw Error.ArgumentNull(nameof(onCompleted));
  109. return DoCore(source, onNext: onNext, onError: null, onCompleted: onCompleted);
  110. }
  111. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, Task> onNext, Func<Exception, CancellationToken, Task> onError)
  112. {
  113. if (source == null)
  114. throw Error.ArgumentNull(nameof(source));
  115. if (onNext == null)
  116. throw Error.ArgumentNull(nameof(onNext));
  117. if (onError == null)
  118. throw Error.ArgumentNull(nameof(onError));
  119. return DoCore(source, onNext: onNext, onError: onError, onCompleted: null);
  120. }
  121. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, Task> onNext, Func<Exception, CancellationToken, Task> onError, Func<CancellationToken, Task> onCompleted)
  122. {
  123. if (source == null)
  124. throw Error.ArgumentNull(nameof(source));
  125. if (onNext == null)
  126. throw Error.ArgumentNull(nameof(onNext));
  127. if (onError == null)
  128. throw Error.ArgumentNull(nameof(onError));
  129. if (onCompleted == null)
  130. throw Error.ArgumentNull(nameof(onCompleted));
  131. return DoCore(source, onNext, onError, onCompleted);
  132. }
  133. #endif
  134. public static IAsyncEnumerable<TSource> Do<TSource>(this IAsyncEnumerable<TSource> source, IObserver<TSource> observer)
  135. {
  136. if (source == null)
  137. throw Error.ArgumentNull(nameof(source));
  138. if (observer == null)
  139. throw Error.ArgumentNull(nameof(observer));
  140. return DoCore(source, new Action<TSource>(observer.OnNext), new Action<Exception>(observer.OnError), new Action(observer.OnCompleted));
  141. }
  142. private static IAsyncEnumerable<TSource> DoCore<TSource>(IAsyncEnumerable<TSource> source, Action<TSource> onNext, Action<Exception>? onError, Action? onCompleted)
  143. {
  144. return AsyncEnumerable.Create(Core);
  145. async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
  146. {
  147. await using var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false);
  148. while (true)
  149. {
  150. TSource item;
  151. try
  152. {
  153. if (!await e.MoveNextAsync())
  154. {
  155. break;
  156. }
  157. item = e.Current;
  158. onNext(item);
  159. }
  160. catch (OperationCanceledException)
  161. {
  162. throw;
  163. }
  164. catch (Exception ex) when (onError != null)
  165. {
  166. onError(ex);
  167. throw;
  168. }
  169. yield return item;
  170. }
  171. onCompleted?.Invoke();
  172. }
  173. }
  174. private static IAsyncEnumerable<TSource> DoCore<TSource>(IAsyncEnumerable<TSource> source, Func<TSource, Task> onNext, Func<Exception, Task>? onError, Func<Task>? onCompleted)
  175. {
  176. return AsyncEnumerable.Create(Core);
  177. async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
  178. {
  179. await using var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false);
  180. while (true)
  181. {
  182. TSource item;
  183. try
  184. {
  185. if (!await e.MoveNextAsync())
  186. {
  187. break;
  188. }
  189. item = e.Current;
  190. await onNext(item).ConfigureAwait(false);
  191. }
  192. catch (OperationCanceledException)
  193. {
  194. throw;
  195. }
  196. catch (Exception ex) when (onError != null)
  197. {
  198. await onError(ex).ConfigureAwait(false);
  199. throw;
  200. }
  201. yield return item;
  202. }
  203. if (onCompleted != null)
  204. {
  205. await onCompleted().ConfigureAwait(false);
  206. }
  207. }
  208. }
  209. #if !NO_DEEP_CANCELLATION
  210. private static IAsyncEnumerable<TSource> DoCore<TSource>(IAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, Task> onNext, Func<Exception, CancellationToken, Task>? onError, Func<CancellationToken, Task>? onCompleted)
  211. {
  212. return AsyncEnumerable.Create(Core);
  213. async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
  214. {
  215. await using var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false);
  216. while (true)
  217. {
  218. TSource item;
  219. try
  220. {
  221. if (!await e.MoveNextAsync())
  222. {
  223. break;
  224. }
  225. item = e.Current;
  226. await onNext(item, cancellationToken).ConfigureAwait(false);
  227. }
  228. catch (OperationCanceledException)
  229. {
  230. throw;
  231. }
  232. catch (Exception ex) when (onError != null)
  233. {
  234. await onError(ex, cancellationToken).ConfigureAwait(false);
  235. throw;
  236. }
  237. yield return item;
  238. }
  239. if (onCompleted != null)
  240. {
  241. await onCompleted(cancellationToken).ConfigureAwait(false);
  242. }
  243. }
  244. }
  245. #endif
  246. }
  247. }