SequenceEqual.cs 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  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.Reactive.Disposables;
  6. using System.Threading;
  7. namespace System.Reactive.Linq
  8. {
  9. partial class AsyncObservable
  10. {
  11. // TODO: Add SequenceEqual<T>(IAsyncObservable<T>, IAsyncEnumerable<T>).
  12. public static IAsyncObservable<bool> SequenceEqual<TSource>(this IAsyncObservable<TSource> first, IAsyncObservable<TSource> second)
  13. {
  14. if (first == null)
  15. throw new ArgumentNullException(nameof(first));
  16. if (second == null)
  17. throw new ArgumentNullException(nameof(second));
  18. return Create<bool>(async observer =>
  19. {
  20. var (firstObserver, secondObserver) = AsyncObserver.SequenceEqual<TSource>(observer);
  21. var firstTask = first.SubscribeSafeAsync(firstObserver);
  22. var secondTask = second.SubscribeSafeAsync(secondObserver);
  23. var d1 = await firstTask.ConfigureAwait(false);
  24. var d2 = await secondTask.ConfigureAwait(false);
  25. return StableCompositeAsyncDisposable.Create(d1, d2);
  26. });
  27. }
  28. public static IAsyncObservable<bool> SequenceEqual<TSource>(this IAsyncObservable<TSource> first, IAsyncObservable<TSource> second, IEqualityComparer<TSource> comparer)
  29. {
  30. if (first == null)
  31. throw new ArgumentNullException(nameof(first));
  32. if (second == null)
  33. throw new ArgumentNullException(nameof(second));
  34. if (comparer == null)
  35. throw new ArgumentNullException(nameof(comparer));
  36. return Create<bool>(async observer =>
  37. {
  38. var (firstObserver, secondObserver) = AsyncObserver.SequenceEqual<TSource>(observer, comparer);
  39. var firstTask = first.SubscribeSafeAsync(firstObserver);
  40. var secondTask = second.SubscribeSafeAsync(secondObserver);
  41. var d1 = await firstTask.ConfigureAwait(false);
  42. var d2 = await secondTask.ConfigureAwait(false);
  43. return StableCompositeAsyncDisposable.Create(d1, d2);
  44. });
  45. }
  46. }
  47. partial class AsyncObserver
  48. {
  49. public static (IAsyncObserver<TSource>, IAsyncObserver<TSource>) SequenceEqual<TSource>(IAsyncObserver<bool> observer)
  50. {
  51. if (observer == null)
  52. throw new ArgumentNullException(nameof(observer));
  53. return SequenceEqual(observer, EqualityComparer<TSource>.Default);
  54. }
  55. public static (IAsyncObserver<TSource>, IAsyncObserver<TSource>) SequenceEqual<TSource>(IAsyncObserver<bool> observer, IEqualityComparer<TSource> comparer)
  56. {
  57. if (observer == null)
  58. throw new ArgumentNullException(nameof(observer));
  59. if (comparer == null)
  60. throw new ArgumentNullException(nameof(comparer));
  61. var gate = new AsyncLock();
  62. var queueLeft = new Queue<TSource>();
  63. var queueRight = new Queue<TSource>();
  64. var doneLeft = false;
  65. var doneRight = false;
  66. return
  67. (
  68. Create<TSource>(
  69. async x =>
  70. {
  71. using (await gate.LockAsync().ConfigureAwait(false))
  72. {
  73. if (queueRight.Count > 0)
  74. {
  75. var v = queueRight.Dequeue();
  76. var equal = false;
  77. try
  78. {
  79. equal = comparer.Equals(x, v);
  80. }
  81. catch (Exception ex)
  82. {
  83. await observer.OnErrorAsync(ex).ConfigureAwait(false);
  84. return;
  85. }
  86. if (!equal)
  87. {
  88. await observer.OnNextAsync(false).ConfigureAwait(false);
  89. await observer.OnCompletedAsync().ConfigureAwait(false);
  90. }
  91. }
  92. else if (doneRight)
  93. {
  94. await observer.OnNextAsync(false).ConfigureAwait(false);
  95. await observer.OnCompletedAsync().ConfigureAwait(false);
  96. }
  97. else
  98. {
  99. queueLeft.Enqueue(x);
  100. }
  101. }
  102. },
  103. async ex =>
  104. {
  105. using (await gate.LockAsync().ConfigureAwait(false))
  106. {
  107. await observer.OnErrorAsync(ex).ConfigureAwait(false);
  108. }
  109. },
  110. async () =>
  111. {
  112. using (await gate.LockAsync().ConfigureAwait(false))
  113. {
  114. doneLeft = true;
  115. if (queueLeft.Count == 0)
  116. {
  117. if (queueRight.Count > 0)
  118. {
  119. await observer.OnNextAsync(false).ConfigureAwait(false);
  120. await observer.OnCompletedAsync().ConfigureAwait(false);
  121. }
  122. else if (doneRight)
  123. {
  124. await observer.OnNextAsync(true).ConfigureAwait(false);
  125. await observer.OnCompletedAsync().ConfigureAwait(false);
  126. }
  127. }
  128. }
  129. }
  130. ),
  131. Create<TSource>(
  132. async x =>
  133. {
  134. using (await gate.LockAsync().ConfigureAwait(false))
  135. {
  136. if (queueLeft.Count > 0)
  137. {
  138. var v = queueLeft.Dequeue();
  139. var equal = false;
  140. try
  141. {
  142. equal = comparer.Equals(v, x);
  143. }
  144. catch (Exception ex)
  145. {
  146. await observer.OnErrorAsync(ex).ConfigureAwait(false);
  147. return;
  148. }
  149. if (!equal)
  150. {
  151. await observer.OnNextAsync(false).ConfigureAwait(false);
  152. await observer.OnCompletedAsync().ConfigureAwait(false);
  153. }
  154. }
  155. else if (doneLeft)
  156. {
  157. await observer.OnNextAsync(false).ConfigureAwait(false);
  158. await observer.OnCompletedAsync().ConfigureAwait(false);
  159. }
  160. else
  161. {
  162. queueRight.Enqueue(x);
  163. }
  164. }
  165. },
  166. async ex =>
  167. {
  168. using (await gate.LockAsync().ConfigureAwait(false))
  169. {
  170. await observer.OnErrorAsync(ex).ConfigureAwait(false);
  171. }
  172. },
  173. async () =>
  174. {
  175. using (await gate.LockAsync().ConfigureAwait(false))
  176. {
  177. doneRight = true;
  178. if (queueRight.Count == 0)
  179. {
  180. if (queueLeft.Count > 0)
  181. {
  182. await observer.OnNextAsync(false).ConfigureAwait(false);
  183. await observer.OnCompletedAsync().ConfigureAwait(false);
  184. }
  185. else if (doneLeft)
  186. {
  187. await observer.OnNextAsync(true).ConfigureAwait(false);
  188. await observer.OnCompletedAsync().ConfigureAwait(false);
  189. }
  190. }
  191. }
  192. }
  193. )
  194. );
  195. }
  196. }
  197. }