Do.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  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. {
  149. while (true)
  150. {
  151. TSource item;
  152. try
  153. {
  154. if (!await e.MoveNextAsync())
  155. {
  156. break;
  157. }
  158. item = e.Current;
  159. onNext(item);
  160. }
  161. catch (OperationCanceledException)
  162. {
  163. throw;
  164. }
  165. catch (Exception ex) when (onError != null)
  166. {
  167. onError(ex);
  168. throw;
  169. }
  170. yield return item;
  171. }
  172. onCompleted?.Invoke();
  173. }
  174. }
  175. }
  176. private static IAsyncEnumerable<TSource> DoCore<TSource>(IAsyncEnumerable<TSource> source, Func<TSource, Task> onNext, Func<Exception, Task> onError, Func<Task> onCompleted)
  177. {
  178. return AsyncEnumerable.Create(Core);
  179. async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
  180. {
  181. await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
  182. {
  183. while (true)
  184. {
  185. TSource item;
  186. try
  187. {
  188. if (!await e.MoveNextAsync())
  189. {
  190. break;
  191. }
  192. item = e.Current;
  193. await onNext(item).ConfigureAwait(false);
  194. }
  195. catch (OperationCanceledException)
  196. {
  197. throw;
  198. }
  199. catch (Exception ex) when (onError != null)
  200. {
  201. await onError(ex).ConfigureAwait(false);
  202. throw;
  203. }
  204. yield return item;
  205. }
  206. if (onCompleted != null)
  207. {
  208. await onCompleted().ConfigureAwait(false);
  209. }
  210. }
  211. }
  212. }
  213. #if !NO_DEEP_CANCELLATION
  214. private static IAsyncEnumerable<TSource> DoCore<TSource>(IAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, Task> onNext, Func<Exception, CancellationToken, Task> onError, Func<CancellationToken, Task> onCompleted)
  215. {
  216. return AsyncEnumerable.Create(Core);
  217. async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
  218. {
  219. await using (var e = source.GetConfiguredAsyncEnumerator(cancellationToken, false))
  220. {
  221. while (true)
  222. {
  223. TSource item;
  224. try
  225. {
  226. if (!await e.MoveNextAsync())
  227. {
  228. break;
  229. }
  230. item = e.Current;
  231. await onNext(item, cancellationToken).ConfigureAwait(false);
  232. }
  233. catch (OperationCanceledException)
  234. {
  235. throw;
  236. }
  237. catch (Exception ex) when (onError != null)
  238. {
  239. await onError(ex, cancellationToken).ConfigureAwait(false);
  240. throw;
  241. }
  242. yield return item;
  243. }
  244. if (onCompleted != null)
  245. {
  246. await onCompleted(cancellationToken).ConfigureAwait(false);
  247. }
  248. }
  249. }
  250. }
  251. #endif
  252. }
  253. }