GroupBy.cs 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the MIT License.
  3. // See the LICENSE file in the project root for more information.
  4. using System.Collections.Generic;
  5. using System.Reactive.Disposables;
  6. using System.Reactive.Subjects;
  7. using System.Threading.Tasks;
  8. namespace System.Reactive.Linq
  9. {
  10. partial class AsyncObservable
  11. {
  12. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector)
  13. {
  14. if (source == null)
  15. throw new ArgumentNullException(nameof(source));
  16. if (keySelector == null)
  17. throw new ArgumentNullException(nameof(keySelector));
  18. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector)));
  19. }
  20. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  21. {
  22. if (source == null)
  23. throw new ArgumentNullException(nameof(source));
  24. if (keySelector == null)
  25. throw new ArgumentNullException(nameof(keySelector));
  26. if (comparer == null)
  27. throw new ArgumentNullException(nameof(comparer));
  28. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, comparer)));
  29. }
  30. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, int capacity)
  31. {
  32. if (source == null)
  33. throw new ArgumentNullException(nameof(source));
  34. if (keySelector == null)
  35. throw new ArgumentNullException(nameof(keySelector));
  36. if (capacity < 0)
  37. throw new ArgumentOutOfRangeException(nameof(capacity));
  38. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, capacity)));
  39. }
  40. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, int capacity, IEqualityComparer<TKey> comparer)
  41. {
  42. if (source == null)
  43. throw new ArgumentNullException(nameof(source));
  44. if (keySelector == null)
  45. throw new ArgumentNullException(nameof(keySelector));
  46. if (capacity < 0)
  47. throw new ArgumentOutOfRangeException(nameof(capacity));
  48. if (comparer == null)
  49. throw new ArgumentNullException(nameof(comparer));
  50. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, capacity, comparer)));
  51. }
  52. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector)
  53. {
  54. if (source == null)
  55. throw new ArgumentNullException(nameof(source));
  56. if (keySelector == null)
  57. throw new ArgumentNullException(nameof(keySelector));
  58. if (elementSelector == null)
  59. throw new ArgumentNullException(nameof(elementSelector));
  60. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector)));
  61. }
  62. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer)
  63. {
  64. if (source == null)
  65. throw new ArgumentNullException(nameof(source));
  66. if (keySelector == null)
  67. throw new ArgumentNullException(nameof(keySelector));
  68. if (elementSelector == null)
  69. throw new ArgumentNullException(nameof(elementSelector));
  70. if (comparer == null)
  71. throw new ArgumentNullException(nameof(comparer));
  72. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector, comparer)));
  73. }
  74. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, int capacity)
  75. {
  76. if (source == null)
  77. throw new ArgumentNullException(nameof(source));
  78. if (keySelector == null)
  79. throw new ArgumentNullException(nameof(keySelector));
  80. if (elementSelector == null)
  81. throw new ArgumentNullException(nameof(elementSelector));
  82. if (capacity < 0)
  83. throw new ArgumentOutOfRangeException(nameof(capacity));
  84. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector, capacity)));
  85. }
  86. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, int capacity, IEqualityComparer<TKey> comparer)
  87. {
  88. if (source == null)
  89. throw new ArgumentNullException(nameof(source));
  90. if (keySelector == null)
  91. throw new ArgumentNullException(nameof(keySelector));
  92. if (elementSelector == null)
  93. throw new ArgumentNullException(nameof(elementSelector));
  94. if (capacity < 0)
  95. throw new ArgumentOutOfRangeException(nameof(capacity));
  96. if (comparer == null)
  97. throw new ArgumentNullException(nameof(comparer));
  98. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector, capacity, comparer)));
  99. }
  100. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector)
  101. {
  102. if (source == null)
  103. throw new ArgumentNullException(nameof(source));
  104. if (keySelector == null)
  105. throw new ArgumentNullException(nameof(keySelector));
  106. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector)));
  107. }
  108. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, IEqualityComparer<TKey> comparer)
  109. {
  110. if (source == null)
  111. throw new ArgumentNullException(nameof(source));
  112. if (keySelector == null)
  113. throw new ArgumentNullException(nameof(keySelector));
  114. if (comparer == null)
  115. throw new ArgumentNullException(nameof(comparer));
  116. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, comparer)));
  117. }
  118. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, int capacity)
  119. {
  120. if (source == null)
  121. throw new ArgumentNullException(nameof(source));
  122. if (keySelector == null)
  123. throw new ArgumentNullException(nameof(keySelector));
  124. if (capacity < 0)
  125. throw new ArgumentOutOfRangeException(nameof(capacity));
  126. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, capacity)));
  127. }
  128. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, int capacity, IEqualityComparer<TKey> comparer)
  129. {
  130. if (source == null)
  131. throw new ArgumentNullException(nameof(source));
  132. if (keySelector == null)
  133. throw new ArgumentNullException(nameof(keySelector));
  134. if (capacity < 0)
  135. throw new ArgumentOutOfRangeException(nameof(capacity));
  136. if (comparer == null)
  137. throw new ArgumentNullException(nameof(comparer));
  138. return Create<IGroupedAsyncObservable<TKey, TSource>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, capacity, comparer)));
  139. }
  140. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector)
  141. {
  142. if (source == null)
  143. throw new ArgumentNullException(nameof(source));
  144. if (keySelector == null)
  145. throw new ArgumentNullException(nameof(keySelector));
  146. if (elementSelector == null)
  147. throw new ArgumentNullException(nameof(elementSelector));
  148. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector)));
  149. }
  150. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector, IEqualityComparer<TKey> comparer)
  151. {
  152. if (source == null)
  153. throw new ArgumentNullException(nameof(source));
  154. if (keySelector == null)
  155. throw new ArgumentNullException(nameof(keySelector));
  156. if (elementSelector == null)
  157. throw new ArgumentNullException(nameof(elementSelector));
  158. if (comparer == null)
  159. throw new ArgumentNullException(nameof(comparer));
  160. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector, comparer)));
  161. }
  162. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector, int capacity)
  163. {
  164. if (source == null)
  165. throw new ArgumentNullException(nameof(source));
  166. if (keySelector == null)
  167. throw new ArgumentNullException(nameof(keySelector));
  168. if (elementSelector == null)
  169. throw new ArgumentNullException(nameof(elementSelector));
  170. if (capacity < 0)
  171. throw new ArgumentOutOfRangeException(nameof(capacity));
  172. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector, capacity)));
  173. }
  174. public static IAsyncObservable<IGroupedAsyncObservable<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncObservable<TSource> source, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector, int capacity, IEqualityComparer<TKey> comparer)
  175. {
  176. if (source == null)
  177. throw new ArgumentNullException(nameof(source));
  178. if (keySelector == null)
  179. throw new ArgumentNullException(nameof(keySelector));
  180. if (elementSelector == null)
  181. throw new ArgumentNullException(nameof(elementSelector));
  182. if (capacity < 0)
  183. throw new ArgumentOutOfRangeException(nameof(capacity));
  184. if (comparer == null)
  185. throw new ArgumentNullException(nameof(comparer));
  186. return Create<IGroupedAsyncObservable<TKey, TElement>>(observer => GroupByCore(source, observer, (o, d) => AsyncObserver.GroupBy(o, d, keySelector, elementSelector, capacity, comparer)));
  187. }
  188. private static async Task<IAsyncDisposable> GroupByCore<TSource, TKey, TElement>(IAsyncObservable<TSource> source, IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, Func<IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>>, IAsyncDisposable, (IAsyncObserver<TSource>, IAsyncDisposable)> createObserver)
  189. {
  190. var d = new SingleAssignmentAsyncDisposable();
  191. var (sink, subscription) = createObserver(observer, d);
  192. var inner = await source.SubscribeSafeAsync(sink).ConfigureAwait(false);
  193. await d.AssignAsync(inner).ConfigureAwait(false);
  194. return subscription;
  195. }
  196. }
  197. partial class AsyncObserver
  198. {
  199. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector)
  200. {
  201. if (observer == null)
  202. throw new ArgumentNullException(nameof(observer));
  203. if (subscription == null)
  204. throw new ArgumentNullException(nameof(subscription));
  205. if (keySelector == null)
  206. throw new ArgumentNullException(nameof(keySelector));
  207. return GroupBy(observer, subscription, keySelector, int.MaxValue, EqualityComparer<TKey>.Default);
  208. }
  209. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  210. {
  211. if (observer == null)
  212. throw new ArgumentNullException(nameof(observer));
  213. if (subscription == null)
  214. throw new ArgumentNullException(nameof(subscription));
  215. if (keySelector == null)
  216. throw new ArgumentNullException(nameof(keySelector));
  217. if (comparer == null)
  218. throw new ArgumentNullException(nameof(comparer));
  219. return GroupBy(observer, subscription, keySelector, int.MaxValue, comparer);
  220. }
  221. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, int capacity)
  222. {
  223. if (observer == null)
  224. throw new ArgumentNullException(nameof(observer));
  225. if (subscription == null)
  226. throw new ArgumentNullException(nameof(subscription));
  227. if (keySelector == null)
  228. throw new ArgumentNullException(nameof(keySelector));
  229. if (capacity < 0)
  230. throw new ArgumentOutOfRangeException(nameof(capacity));
  231. return GroupBy(observer, subscription, keySelector, capacity, EqualityComparer<TKey>.Default);
  232. }
  233. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, int capacity, IEqualityComparer<TKey> comparer)
  234. {
  235. if (observer == null)
  236. throw new ArgumentNullException(nameof(observer));
  237. if (subscription == null)
  238. throw new ArgumentNullException(nameof(subscription));
  239. if (keySelector == null)
  240. throw new ArgumentNullException(nameof(keySelector));
  241. if (capacity < 0)
  242. throw new ArgumentOutOfRangeException(nameof(capacity));
  243. if (comparer == null)
  244. throw new ArgumentNullException(nameof(comparer));
  245. return GroupBy(observer, subscription, x => Task.FromResult(keySelector(x)), capacity, comparer);
  246. }
  247. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector)
  248. {
  249. if (observer == null)
  250. throw new ArgumentNullException(nameof(observer));
  251. if (subscription == null)
  252. throw new ArgumentNullException(nameof(subscription));
  253. if (keySelector == null)
  254. throw new ArgumentNullException(nameof(keySelector));
  255. if (elementSelector == null)
  256. throw new ArgumentNullException(nameof(elementSelector));
  257. return GroupBy(observer, subscription, keySelector, elementSelector, int.MaxValue, EqualityComparer<TKey>.Default);
  258. }
  259. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer)
  260. {
  261. if (observer == null)
  262. throw new ArgumentNullException(nameof(observer));
  263. if (subscription == null)
  264. throw new ArgumentNullException(nameof(subscription));
  265. if (keySelector == null)
  266. throw new ArgumentNullException(nameof(keySelector));
  267. if (elementSelector == null)
  268. throw new ArgumentNullException(nameof(elementSelector));
  269. if (comparer == null)
  270. throw new ArgumentNullException(nameof(comparer));
  271. return GroupBy(observer, subscription, keySelector, elementSelector, int.MaxValue, comparer);
  272. }
  273. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, int capacity)
  274. {
  275. if (observer == null)
  276. throw new ArgumentNullException(nameof(observer));
  277. if (subscription == null)
  278. throw new ArgumentNullException(nameof(subscription));
  279. if (keySelector == null)
  280. throw new ArgumentNullException(nameof(keySelector));
  281. if (elementSelector == null)
  282. throw new ArgumentNullException(nameof(elementSelector));
  283. if (capacity < 0)
  284. throw new ArgumentOutOfRangeException(nameof(capacity));
  285. return GroupBy(observer, subscription, keySelector, elementSelector, capacity, EqualityComparer<TKey>.Default);
  286. }
  287. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, int capacity, IEqualityComparer<TKey> comparer)
  288. {
  289. if (observer == null)
  290. throw new ArgumentNullException(nameof(observer));
  291. if (subscription == null)
  292. throw new ArgumentNullException(nameof(subscription));
  293. if (keySelector == null)
  294. throw new ArgumentNullException(nameof(keySelector));
  295. if (elementSelector == null)
  296. throw new ArgumentNullException(nameof(elementSelector));
  297. if (capacity < 0)
  298. throw new ArgumentOutOfRangeException(nameof(capacity));
  299. if (comparer == null)
  300. throw new ArgumentNullException(nameof(comparer));
  301. return GroupBy<TSource, TKey, TElement>(observer, subscription, x => Task.FromResult(keySelector(x)), x => Task.FromResult(elementSelector(x)), capacity, comparer);
  302. }
  303. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector)
  304. {
  305. if (observer == null)
  306. throw new ArgumentNullException(nameof(observer));
  307. if (subscription == null)
  308. throw new ArgumentNullException(nameof(subscription));
  309. if (keySelector == null)
  310. throw new ArgumentNullException(nameof(keySelector));
  311. return GroupBy(observer, subscription, keySelector, int.MaxValue, EqualityComparer<TKey>.Default);
  312. }
  313. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, IEqualityComparer<TKey> comparer)
  314. {
  315. if (observer == null)
  316. throw new ArgumentNullException(nameof(observer));
  317. if (subscription == null)
  318. throw new ArgumentNullException(nameof(subscription));
  319. if (keySelector == null)
  320. throw new ArgumentNullException(nameof(keySelector));
  321. if (comparer == null)
  322. throw new ArgumentNullException(nameof(comparer));
  323. return GroupBy(observer, subscription, keySelector, int.MaxValue, comparer);
  324. }
  325. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, int capacity)
  326. {
  327. if (observer == null)
  328. throw new ArgumentNullException(nameof(observer));
  329. if (subscription == null)
  330. throw new ArgumentNullException(nameof(subscription));
  331. if (keySelector == null)
  332. throw new ArgumentNullException(nameof(keySelector));
  333. if (capacity < 0)
  334. throw new ArgumentOutOfRangeException(nameof(capacity));
  335. return GroupBy(observer, subscription, keySelector, capacity, EqualityComparer<TKey>.Default);
  336. }
  337. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey>(IAsyncObserver<IGroupedAsyncObservable<TKey, TSource>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, int capacity, IEqualityComparer<TKey> comparer)
  338. {
  339. if (observer == null)
  340. throw new ArgumentNullException(nameof(observer));
  341. if (subscription == null)
  342. throw new ArgumentNullException(nameof(subscription));
  343. if (keySelector == null)
  344. throw new ArgumentNullException(nameof(keySelector));
  345. if (capacity < 0)
  346. throw new ArgumentOutOfRangeException(nameof(capacity));
  347. if (comparer == null)
  348. throw new ArgumentNullException(nameof(comparer));
  349. return GroupBy(observer, subscription, keySelector, x => Task.FromResult(x), capacity, comparer);
  350. }
  351. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector)
  352. {
  353. if (observer == null)
  354. throw new ArgumentNullException(nameof(observer));
  355. if (subscription == null)
  356. throw new ArgumentNullException(nameof(subscription));
  357. if (keySelector == null)
  358. throw new ArgumentNullException(nameof(keySelector));
  359. if (elementSelector == null)
  360. throw new ArgumentNullException(nameof(elementSelector));
  361. return GroupBy(observer, subscription, keySelector, elementSelector, int.MaxValue, EqualityComparer<TKey>.Default);
  362. }
  363. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector, IEqualityComparer<TKey> comparer)
  364. {
  365. if (observer == null)
  366. throw new ArgumentNullException(nameof(observer));
  367. if (subscription == null)
  368. throw new ArgumentNullException(nameof(subscription));
  369. if (keySelector == null)
  370. throw new ArgumentNullException(nameof(keySelector));
  371. if (elementSelector == null)
  372. throw new ArgumentNullException(nameof(elementSelector));
  373. if (comparer == null)
  374. throw new ArgumentNullException(nameof(comparer));
  375. return GroupBy(observer, subscription, keySelector, elementSelector, int.MaxValue, comparer);
  376. }
  377. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector, int capacity)
  378. {
  379. if (observer == null)
  380. throw new ArgumentNullException(nameof(observer));
  381. if (subscription == null)
  382. throw new ArgumentNullException(nameof(subscription));
  383. if (keySelector == null)
  384. throw new ArgumentNullException(nameof(keySelector));
  385. if (elementSelector == null)
  386. throw new ArgumentNullException(nameof(elementSelector));
  387. if (capacity < 0)
  388. throw new ArgumentOutOfRangeException(nameof(capacity));
  389. return GroupBy(observer, subscription, keySelector, elementSelector, capacity, EqualityComparer<TKey>.Default);
  390. }
  391. public static (IAsyncObserver<TSource>, IAsyncDisposable) GroupBy<TSource, TKey, TElement>(IAsyncObserver<IGroupedAsyncObservable<TKey, TElement>> observer, IAsyncDisposable subscription, Func<TSource, Task<TKey>> keySelector, Func<TSource, Task<TElement>> elementSelector, int capacity, IEqualityComparer<TKey> comparer)
  392. {
  393. if (observer == null)
  394. throw new ArgumentNullException(nameof(observer));
  395. if (subscription == null)
  396. throw new ArgumentNullException(nameof(subscription));
  397. if (keySelector == null)
  398. throw new ArgumentNullException(nameof(keySelector));
  399. if (elementSelector == null)
  400. throw new ArgumentNullException(nameof(elementSelector));
  401. if (capacity < 0)
  402. throw new ArgumentOutOfRangeException(nameof(capacity));
  403. if (comparer == null)
  404. throw new ArgumentNullException(nameof(comparer));
  405. var refCount = new RefCountAsyncDisposable(subscription);
  406. var groups = default(Dictionary<TKey, SequentialSimpleAsyncSubject<TElement>>);
  407. if (capacity == int.MaxValue)
  408. {
  409. groups = new Dictionary<TKey, SequentialSimpleAsyncSubject<TElement>>(comparer);
  410. }
  411. else
  412. {
  413. groups = new Dictionary<TKey, SequentialSimpleAsyncSubject<TElement>>(capacity, comparer);
  414. }
  415. var nullGroup = default(SequentialSimpleAsyncSubject<TElement>);
  416. async Task OnErrorAsync(Exception ex)
  417. {
  418. if (nullGroup != null)
  419. {
  420. await nullGroup.OnErrorAsync(ex).ConfigureAwait(false);
  421. }
  422. foreach (var group in groups.Values)
  423. {
  424. await group.OnErrorAsync(ex).ConfigureAwait(false);
  425. }
  426. await observer.OnErrorAsync(ex).ConfigureAwait(false);
  427. }
  428. return
  429. (
  430. Create<TSource>
  431. (
  432. async x =>
  433. {
  434. var key = default(TKey);
  435. try
  436. {
  437. key = await keySelector(x).ConfigureAwait(false);
  438. }
  439. catch (Exception ex)
  440. {
  441. await OnErrorAsync(ex).ConfigureAwait(false);
  442. return;
  443. }
  444. var shouldEmit = false;
  445. var group = default(SequentialSimpleAsyncSubject<TElement>);
  446. if (key == null)
  447. {
  448. if (nullGroup == null)
  449. {
  450. nullGroup = new SequentialSimpleAsyncSubject<TElement>();
  451. shouldEmit = true;
  452. }
  453. group = nullGroup;
  454. }
  455. else
  456. {
  457. try
  458. {
  459. if (!groups.TryGetValue(key, out group))
  460. {
  461. group = new SequentialSimpleAsyncSubject<TElement>();
  462. groups.Add(key, group);
  463. shouldEmit = true;
  464. }
  465. }
  466. catch (Exception ex)
  467. {
  468. await OnErrorAsync(ex).ConfigureAwait(false);
  469. return;
  470. }
  471. }
  472. if (shouldEmit)
  473. {
  474. var g = new GroupedAsyncObservable<TKey, TElement>(key, group, refCount);
  475. await observer.OnNextAsync(g).ConfigureAwait(false);
  476. }
  477. var element = default(TElement);
  478. try
  479. {
  480. element = await elementSelector(x).ConfigureAwait(false);
  481. }
  482. catch (Exception ex)
  483. {
  484. await OnErrorAsync(ex).ConfigureAwait(false);
  485. return;
  486. }
  487. await group.OnNextAsync(element).ConfigureAwait(false);
  488. },
  489. OnErrorAsync,
  490. async () =>
  491. {
  492. if (nullGroup != null)
  493. {
  494. await nullGroup.OnCompletedAsync().ConfigureAwait(false);
  495. }
  496. foreach (var group in groups.Values)
  497. {
  498. await group.OnCompletedAsync().ConfigureAwait(false);
  499. }
  500. await observer.OnCompletedAsync().ConfigureAwait(false);
  501. }
  502. ),
  503. refCount
  504. );
  505. }
  506. }
  507. }