Join.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  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.Tasks;
  8. using Xunit;
  9. namespace Tests
  10. {
  11. public class Join : AsyncEnumerableTests
  12. {
  13. [Fact]
  14. public void Join_Null()
  15. {
  16. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(default, Return42, x => x, x => x, (x, y) => x));
  17. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, default, x => x, x => x, (x, y) => x));
  18. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, Return42, default, x => x, (x, y) => x));
  19. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, Return42, x => x, default, (x, y) => x));
  20. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, Return42, x => x, x => x, default));
  21. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(default, Return42, x => x, x => x, (x, y) => x, EqualityComparer<int>.Default));
  22. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, default, x => x, x => x, (x, y) => x, EqualityComparer<int>.Default));
  23. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, Return42, default, x => x, (x, y) => x, EqualityComparer<int>.Default));
  24. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, Return42, x => x, default, (x, y) => x, EqualityComparer<int>.Default));
  25. AssertThrows<ArgumentNullException>(() => AsyncEnumerable.Join<int, int, int, int>(Return42, Return42, x => x, x => x, default, EqualityComparer<int>.Default));
  26. }
  27. [Fact]
  28. public void Join1()
  29. {
  30. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  31. var ys = new[] { 3, 6, 4 }.ToAsyncEnumerable();
  32. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  33. var e = res.GetAsyncEnumerator();
  34. HasNext(e, 0 + 3);
  35. HasNext(e, 0 + 6);
  36. HasNext(e, 1 + 4);
  37. NoNext(e);
  38. }
  39. [Fact]
  40. public void Join2()
  41. {
  42. var xs = new[] { 3, 6, 4 }.ToAsyncEnumerable();
  43. var ys = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  44. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  45. var e = res.GetAsyncEnumerator();
  46. HasNext(e, 3 + 0);
  47. HasNext(e, 6 + 0);
  48. HasNext(e, 4 + 1);
  49. NoNext(e);
  50. }
  51. [Fact]
  52. public void Join3()
  53. {
  54. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  55. var ys = new[] { 3, 6 }.ToAsyncEnumerable();
  56. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  57. var e = res.GetAsyncEnumerator();
  58. HasNext(e, 0 + 3);
  59. HasNext(e, 0 + 6);
  60. NoNext(e);
  61. }
  62. [Fact]
  63. public void Join4()
  64. {
  65. var xs = new[] { 3, 6 }.ToAsyncEnumerable();
  66. var ys = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  67. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  68. var e = res.GetAsyncEnumerator();
  69. HasNext(e, 3 + 0);
  70. HasNext(e, 6 + 0);
  71. NoNext(e);
  72. }
  73. [Fact]
  74. public void Join5()
  75. {
  76. var ex = new Exception("Bang!");
  77. var xs = Throw<int>(ex);
  78. var ys = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  79. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  80. var e = res.GetAsyncEnumerator();
  81. AssertThrows(() => e.MoveNextAsync().Wait(WaitTimeoutMs), SingleInnerExceptionMatches(ex));
  82. }
  83. [Fact]
  84. public void Join6()
  85. {
  86. var ex = new Exception("Bang!");
  87. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  88. var ys = Throw<int>(ex);
  89. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  90. var e = res.GetAsyncEnumerator();
  91. AssertThrows(() => e.MoveNextAsync().Wait(WaitTimeoutMs), SingleInnerExceptionMatches(ex));
  92. }
  93. [Fact]
  94. public void Join7()
  95. {
  96. var ex = new Exception("Bang!");
  97. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  98. var ys = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  99. var res = xs.Join(ys, x => { throw ex; }, y => y, (x, y) => x + y);
  100. var e = res.GetAsyncEnumerator();
  101. AssertThrows(() => e.MoveNextAsync().Wait(WaitTimeoutMs), SingleInnerExceptionMatches(ex));
  102. }
  103. [Fact]
  104. public void Join8()
  105. {
  106. var ex = new Exception("Bang!");
  107. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  108. var ys = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  109. var res = xs.Join(ys, x => x, y => { throw ex; }, (x, y) => x + y);
  110. var e = res.GetAsyncEnumerator();
  111. AssertThrows(() => e.MoveNextAsync().Wait(WaitTimeoutMs), SingleInnerExceptionMatches(ex));
  112. }
  113. [Fact]
  114. public void Join9()
  115. {
  116. var ex = new Exception("Bang!");
  117. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  118. var ys = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  119. var res = xs.Join<int, int, int, int>(ys, x => x, y => y, (x, y) => { throw ex; });
  120. var e = res.GetAsyncEnumerator();
  121. AssertThrows(() => e.MoveNextAsync().Wait(WaitTimeoutMs), SingleInnerExceptionMatches(ex));
  122. }
  123. [Fact]
  124. public async Task Join10()
  125. {
  126. var xs = new[] { 0, 1, 2 }.ToAsyncEnumerable();
  127. var ys = new[] { 3, 6, 4 }.ToAsyncEnumerable();
  128. var res = xs.Join(ys, x => x % 3, y => y % 3, (x, y) => x + y);
  129. await SequenceIdentity(res);
  130. }
  131. [Fact]
  132. public void Join11()
  133. {
  134. var customers = new List<Customer>
  135. {
  136. new Customer { CustomerId = "ALFKI" },
  137. new Customer { CustomerId = "ANANT" },
  138. new Customer { CustomerId = "FISSA" },
  139. };
  140. var orders = new List<Order>
  141. {
  142. new Order { OrderId = 1, CustomerId = "ALFKI"},
  143. new Order { OrderId = 2, CustomerId = "ALFKI"},
  144. new Order { OrderId = 3, CustomerId = "ALFKI"},
  145. new Order { OrderId = 4, CustomerId = "FISSA"},
  146. new Order { OrderId = 5, CustomerId = "FISSA"},
  147. new Order { OrderId = 6, CustomerId = "FISSA"},
  148. };
  149. var asyncResult = customers.ToAsyncEnumerable()
  150. .Join(orders.ToAsyncEnumerable(), c => c.CustomerId, o => o.CustomerId,
  151. (c, o) => new CustomerOrder { CustomerId = c.CustomerId, OrderId = o.OrderId });
  152. var e = asyncResult.GetAsyncEnumerator();
  153. HasNext(e, new CustomerOrder { CustomerId = "ALFKI", OrderId = 1 });
  154. HasNext(e, new CustomerOrder { CustomerId = "ALFKI", OrderId = 2 });
  155. HasNext(e, new CustomerOrder { CustomerId = "ALFKI", OrderId = 3 });
  156. HasNext(e, new CustomerOrder { CustomerId = "FISSA", OrderId = 4 });
  157. HasNext(e, new CustomerOrder { CustomerId = "FISSA", OrderId = 5 });
  158. HasNext(e, new CustomerOrder { CustomerId = "FISSA", OrderId = 6 });
  159. NoNext(e);
  160. }
  161. [Fact]
  162. public void Join12()
  163. {
  164. var customers = new List<Customer>
  165. {
  166. new Customer {CustomerId = "ANANT"},
  167. new Customer {CustomerId = "ALFKI"},
  168. new Customer {CustomerId = "FISSA"}
  169. };
  170. var orders = new List<Order>
  171. {
  172. new Order { OrderId = 1, CustomerId = "ALFKI"},
  173. new Order { OrderId = 2, CustomerId = "ALFKI"},
  174. new Order { OrderId = 3, CustomerId = "ALFKI"},
  175. new Order { OrderId = 4, CustomerId = "FISSA"},
  176. new Order { OrderId = 5, CustomerId = "FISSA"},
  177. new Order { OrderId = 6, CustomerId = "FISSA"},
  178. };
  179. var asyncResult = customers.ToAsyncEnumerable()
  180. .Join(orders.ToAsyncEnumerable(), c => c.CustomerId, o => o.CustomerId,
  181. (c, o) => new CustomerOrder { CustomerId = c.CustomerId, OrderId = o.OrderId });
  182. var e = asyncResult.GetAsyncEnumerator();
  183. HasNext(e, new CustomerOrder { CustomerId = "ALFKI", OrderId = 1 });
  184. HasNext(e, new CustomerOrder { CustomerId = "ALFKI", OrderId = 2 });
  185. HasNext(e, new CustomerOrder { CustomerId = "ALFKI", OrderId = 3 });
  186. HasNext(e, new CustomerOrder { CustomerId = "FISSA", OrderId = 4 });
  187. HasNext(e, new CustomerOrder { CustomerId = "FISSA", OrderId = 5 });
  188. HasNext(e, new CustomerOrder { CustomerId = "FISSA", OrderId = 6 });
  189. NoNext(e);
  190. }
  191. public class Customer
  192. {
  193. public string CustomerId { get; set; }
  194. }
  195. public class Order
  196. {
  197. public int OrderId { get; set; }
  198. public string CustomerId { get; set; }
  199. }
  200. public class CustomerOrder : IEquatable<CustomerOrder>
  201. {
  202. public int OrderId { get; set; }
  203. public string CustomerId { get; set; }
  204. public bool Equals(CustomerOrder other)
  205. {
  206. if (other is null) return false;
  207. if (ReferenceEquals(this, other)) return true;
  208. return OrderId == other.OrderId && string.Equals(CustomerId, other.CustomerId);
  209. }
  210. public override bool Equals(object obj)
  211. {
  212. if (obj is null) return false;
  213. if (ReferenceEquals(this, obj)) return true;
  214. if (obj.GetType() != GetType()) return false;
  215. return Equals((CustomerOrder)obj);
  216. }
  217. public override int GetHashCode()
  218. {
  219. unchecked
  220. {
  221. return (OrderId * 397) ^ (CustomerId != null ? CustomerId.GetHashCode() : 0);
  222. }
  223. }
  224. public static bool operator ==(CustomerOrder left, CustomerOrder right)
  225. {
  226. return Equals(left, right);
  227. }
  228. public static bool operator !=(CustomerOrder left, CustomerOrder right)
  229. {
  230. return !Equals(left, right);
  231. }
  232. }
  233. }
  234. }