ForkJoinTest.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631
  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;
  5. using System.Collections.Generic;
  6. using System.Linq;
  7. using System.Reactive.Linq;
  8. using Microsoft.Reactive.Testing;
  9. using ReactiveTests.Dummies;
  10. using Microsoft.VisualStudio.TestTools.UnitTesting;
  11. namespace ReactiveTests.Tests
  12. {
  13. [TestClass]
  14. public class ForkJoinTest : ReactiveTest
  15. {
  16. [TestMethod]
  17. public void ForkJoin_ArgumentChecking()
  18. {
  19. var someObservable = DummyObservable<int>.Instance;
  20. ReactiveAssert.Throws<ArgumentNullException>(() => ObservableEx.ForkJoin(someObservable, someObservable, (Func<int, int, int>)null));
  21. ReactiveAssert.Throws<ArgumentNullException>(() => ObservableEx.ForkJoin(someObservable, (IObservable<int>)null, (_, __) => _ + __));
  22. ReactiveAssert.Throws<ArgumentNullException>(() => ObservableEx.ForkJoin((IObservable<int>)null, someObservable, (_, __) => _ + __));
  23. ReactiveAssert.Throws<ArgumentNullException>(() => ObservableEx.ForkJoin((IObservable<int>[])null));
  24. ReactiveAssert.Throws<ArgumentNullException>(() => ObservableEx.ForkJoin((IEnumerable<IObservable<int>>)null));
  25. }
  26. [TestMethod]
  27. public void ForkJoin_EmptyEmpty()
  28. {
  29. var scheduler = new TestScheduler();
  30. var msgs1 = new[] {
  31. OnNext(150, 1),
  32. OnCompleted<int>(230)
  33. };
  34. var msgs2 = new[] {
  35. OnNext(150, 1),
  36. OnCompleted<int>(250)
  37. };
  38. var o = scheduler.CreateHotObservable(msgs1);
  39. var e = scheduler.CreateHotObservable(msgs2);
  40. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  41. res.Messages.AssertEqual(
  42. OnCompleted<int>(250)
  43. );
  44. }
  45. [TestMethod]
  46. public void ForkJoin_None()
  47. {
  48. var scheduler = new TestScheduler();
  49. var res = scheduler.Start(() => ObservableEx.ForkJoin<int>());
  50. res.Messages.AssertEqual(
  51. OnCompleted<int[]>(200)
  52. );
  53. }
  54. [TestMethod]
  55. public void ForkJoin_EmptyReturn()
  56. {
  57. var scheduler = new TestScheduler();
  58. var msgs1 = new[] {
  59. OnNext(150, 1),
  60. OnCompleted<int>(230)
  61. };
  62. var msgs2 = new[] {
  63. OnNext(150, 1),
  64. OnNext(210, 2),
  65. OnCompleted<int>(250)
  66. };
  67. var o = scheduler.CreateHotObservable(msgs1);
  68. var e = scheduler.CreateHotObservable(msgs2);
  69. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  70. res.Messages.AssertEqual(
  71. OnCompleted<int>(250)
  72. );
  73. }
  74. [TestMethod]
  75. public void ForkJoin_ReturnEmpty()
  76. {
  77. var scheduler = new TestScheduler();
  78. var msgs1 = new[] {
  79. OnNext(150, 1),
  80. OnNext(210, 2),
  81. OnCompleted<int>(230)
  82. };
  83. var msgs2 = new[] {
  84. OnNext(150, 1),
  85. OnCompleted<int>(250)
  86. };
  87. var o = scheduler.CreateHotObservable(msgs1);
  88. var e = scheduler.CreateHotObservable(msgs2);
  89. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  90. res.Messages.AssertEqual(
  91. OnCompleted<int>(250)
  92. );
  93. }
  94. [TestMethod]
  95. public void ForkJoin_ReturnReturn()
  96. {
  97. var scheduler = new TestScheduler();
  98. var msgs1 = new[] {
  99. OnNext(150, 1),
  100. OnNext(210, 2),
  101. OnCompleted<int>(230)
  102. };
  103. var msgs2 = new[] {
  104. OnNext(150, 1),
  105. OnNext(220, 3),
  106. OnCompleted<int>(250)
  107. };
  108. var o = scheduler.CreateHotObservable(msgs1);
  109. var e = scheduler.CreateHotObservable(msgs2);
  110. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  111. res.Messages.AssertEqual(
  112. OnNext(250, 2 + 3),
  113. OnCompleted<int>(250)
  114. );
  115. }
  116. [TestMethod]
  117. public void ForkJoin_EmptyThrow()
  118. {
  119. var ex = new Exception();
  120. var scheduler = new TestScheduler();
  121. var msgs1 = new[] {
  122. OnNext(150, 1),
  123. OnCompleted<int>(230)
  124. };
  125. var msgs2 = new[] {
  126. OnNext(150, 1),
  127. OnError<int>(210, ex),
  128. OnCompleted<int>(250)
  129. };
  130. var o = scheduler.CreateHotObservable(msgs1);
  131. var e = scheduler.CreateHotObservable(msgs2);
  132. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  133. res.Messages.AssertEqual(
  134. OnError<int>(210, ex)
  135. );
  136. }
  137. [TestMethod]
  138. public void ForkJoin_ThrowEmpty()
  139. {
  140. var ex = new Exception();
  141. var scheduler = new TestScheduler();
  142. var msgs1 = new[] {
  143. OnNext(150, 1),
  144. OnError<int>(210, ex),
  145. OnCompleted<int>(230)
  146. };
  147. var msgs2 = new[] {
  148. OnNext(150, 1),
  149. OnCompleted<int>(250)
  150. };
  151. var o = scheduler.CreateHotObservable(msgs1);
  152. var e = scheduler.CreateHotObservable(msgs2);
  153. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  154. res.Messages.AssertEqual(
  155. OnError<int>(210, ex)
  156. );
  157. }
  158. [TestMethod]
  159. public void ForkJoin_ReturnThrow()
  160. {
  161. var ex = new Exception();
  162. var scheduler = new TestScheduler();
  163. var msgs1 = new[] {
  164. OnNext(150, 1),
  165. OnNext(210, 2),
  166. OnCompleted<int>(230)
  167. };
  168. var msgs2 = new[] {
  169. OnNext(150, 1),
  170. OnError<int>(220, ex),
  171. OnCompleted<int>(250)
  172. };
  173. var o = scheduler.CreateHotObservable(msgs1);
  174. var e = scheduler.CreateHotObservable(msgs2);
  175. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  176. res.Messages.AssertEqual(
  177. OnError<int>(220, ex)
  178. );
  179. }
  180. [TestMethod]
  181. public void ForkJoin_ThrowReturn()
  182. {
  183. var ex = new Exception();
  184. var scheduler = new TestScheduler();
  185. var msgs1 = new[] {
  186. OnNext(150, 1),
  187. OnError<int>(220, ex),
  188. OnCompleted<int>(230)
  189. };
  190. var msgs2 = new[] {
  191. OnNext(150, 1),
  192. OnNext(210, 2),
  193. OnCompleted<int>(250)
  194. };
  195. var o = scheduler.CreateHotObservable(msgs1);
  196. var e = scheduler.CreateHotObservable(msgs2);
  197. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  198. res.Messages.AssertEqual(
  199. OnError<int>(220, ex)
  200. );
  201. }
  202. [TestMethod]
  203. public void ForkJoin_Binary()
  204. {
  205. var scheduler = new TestScheduler();
  206. var msgs1 = new[] {
  207. OnNext(150, 1),
  208. OnNext(215, 2),
  209. OnNext(225, 4),
  210. OnCompleted<int>(230)
  211. };
  212. var msgs2 = new[] {
  213. OnNext(150, 1),
  214. OnNext(235, 6),
  215. OnNext(240, 7),
  216. OnCompleted<int>(250)
  217. };
  218. var o = scheduler.CreateHotObservable(msgs1);
  219. var e = scheduler.CreateHotObservable(msgs2);
  220. var res = scheduler.Start(() => e.ForkJoin(o, (x, y) => x + y));
  221. res.Messages.AssertEqual(
  222. OnNext(250, 4 + 7), // TODO: fix ForkJoin behavior
  223. OnCompleted<int>(250)
  224. );
  225. }
  226. [TestMethod]
  227. public void ForkJoin_NaryParams()
  228. {
  229. var scheduler = new TestScheduler();
  230. var msgs1 = new[] {
  231. OnNext(150, 1),
  232. OnNext(215, 2),
  233. OnNext(225, 4),
  234. OnCompleted<int>(230)
  235. };
  236. var msgs2 = new[] {
  237. OnNext(150, 1),
  238. OnNext(235, 6),
  239. OnNext(240, 7),
  240. OnCompleted<int>(250)
  241. };
  242. var msgs3 = new[] {
  243. OnNext(150, 1),
  244. OnNext(230, 3),
  245. OnNext(245, 5),
  246. OnCompleted<int>(270)
  247. };
  248. var o1 = scheduler.CreateHotObservable(msgs1);
  249. var o2 = scheduler.CreateHotObservable(msgs2);
  250. var o3 = scheduler.CreateHotObservable(msgs3);
  251. var res = scheduler.Start(() => ObservableEx.ForkJoin(o1, o2, o3));
  252. res.Messages.AssertEqual(
  253. OnNext<int[]>(270, l => l.SequenceEqual([4, 7, 5])), // TODO: fix ForkJoin behavior
  254. OnCompleted<int[]>(270)
  255. );
  256. }
  257. [TestMethod]
  258. public void ForkJoin_NaryParamsEmpty()
  259. {
  260. var scheduler = new TestScheduler();
  261. var msgs1 = new[] {
  262. OnNext(150, 1),
  263. OnNext(215, 2),
  264. OnNext(225, 4),
  265. OnCompleted<int>(230)
  266. };
  267. var msgs2 = new[] {
  268. OnNext(150, 1),
  269. OnNext(235, 6),
  270. OnNext(240, 7),
  271. OnCompleted<int>(250)
  272. };
  273. var msgs3 = new[] {
  274. OnCompleted<int>(270)
  275. };
  276. var o1 = scheduler.CreateHotObservable(msgs1);
  277. var o2 = scheduler.CreateHotObservable(msgs2);
  278. var o3 = scheduler.CreateHotObservable(msgs3);
  279. var res = scheduler.Start(() => ObservableEx.ForkJoin(o1, o2, o3));
  280. res.Messages.AssertEqual(
  281. OnCompleted<int[]>(270)
  282. );
  283. }
  284. [TestMethod]
  285. public void ForkJoin_NaryParamsEmptyBeforeEnd()
  286. {
  287. var scheduler = new TestScheduler();
  288. var msgs1 = new[] {
  289. OnNext(150, 1),
  290. OnNext(215, 2),
  291. OnNext(225, 4),
  292. OnCompleted<int>(230)
  293. };
  294. var msgs2 = new[] {
  295. OnCompleted<int>(235)
  296. };
  297. var msgs3 = new[] {
  298. OnNext(150, 1),
  299. OnNext(230, 3),
  300. OnNext(245, 5),
  301. OnCompleted<int>(270)
  302. };
  303. var o1 = scheduler.CreateHotObservable(msgs1);
  304. var o2 = scheduler.CreateHotObservable(msgs2);
  305. var o3 = scheduler.CreateHotObservable(msgs3);
  306. var res = scheduler.Start(() => ObservableEx.ForkJoin(o1, o2, o3));
  307. res.Messages.AssertEqual(
  308. OnCompleted<int[]>(235)
  309. );
  310. }
  311. [TestMethod]
  312. public void ForkJoin_Nary_Immediate()
  313. {
  314. ObservableEx.ForkJoin(Observable.Return(1), Observable.Return(2)).First().AssertEqual([1, 2]);
  315. }
  316. [TestMethod]
  317. public void ForkJoin_Nary_Virtual_And_Immediate()
  318. {
  319. var scheduler = new TestScheduler();
  320. var msgs1 = new[] {
  321. OnNext(150, 1),
  322. OnNext(215, 2),
  323. OnNext(225, 4),
  324. OnCompleted<int>(230)
  325. };
  326. var msgs2 = new[] {
  327. OnNext(150, 1),
  328. OnNext(235, 6),
  329. OnNext(240, 7),
  330. OnCompleted<int>(250)
  331. };
  332. var msgs3 = new[] {
  333. OnNext(150, 1),
  334. OnNext(230, 3),
  335. OnNext(245, 5),
  336. OnCompleted<int>(270)
  337. };
  338. var o1 = scheduler.CreateHotObservable(msgs1);
  339. var o2 = scheduler.CreateHotObservable(msgs2);
  340. var o3 = scheduler.CreateHotObservable(msgs3);
  341. var res = scheduler.Start(() => ObservableEx.ForkJoin(new List<IObservable<int>> { o1, o2, o3, Observable.Return(20) }));
  342. res.Messages.AssertEqual(
  343. OnNext<int[]>(270, l => l.SequenceEqual([4, 7, 5, 20])),
  344. OnCompleted<int[]>(270)
  345. );
  346. }
  347. [TestMethod]
  348. public void ForkJoin_Nary_Immediate_And_Virtual()
  349. {
  350. var scheduler = new TestScheduler();
  351. var msgs1 = new[] {
  352. OnNext(150, 1),
  353. OnNext(215, 2),
  354. OnNext(225, 4),
  355. OnCompleted<int>(230)
  356. };
  357. var msgs2 = new[] {
  358. OnNext(150, 1),
  359. OnNext(235, 6),
  360. OnNext(240, 7),
  361. OnCompleted<int>(250)
  362. };
  363. var msgs3 = new[] {
  364. OnNext(150, 1),
  365. OnNext(230, 3),
  366. OnNext(245, 5),
  367. OnCompleted<int>(270)
  368. };
  369. var o1 = scheduler.CreateHotObservable(msgs1);
  370. var o2 = scheduler.CreateHotObservable(msgs2);
  371. var o3 = scheduler.CreateHotObservable(msgs3);
  372. var res = scheduler.Start(() => ObservableEx.ForkJoin(new List<IObservable<int>> { Observable.Return(20), o1, o2, o3 }));
  373. res.Messages.AssertEqual(
  374. OnNext<int[]>(270, l => l.SequenceEqual([20, 4, 7, 5])),
  375. OnCompleted<int[]>(270)
  376. );
  377. }
  378. [TestMethod]
  379. public void ForkJoin_Nary()
  380. {
  381. var scheduler = new TestScheduler();
  382. var msgs1 = new[] {
  383. OnNext(150, 1),
  384. OnNext(215, 2),
  385. OnNext(225, 4),
  386. OnCompleted<int>(230)
  387. };
  388. var msgs2 = new[] {
  389. OnNext(150, 1),
  390. OnNext(235, 6),
  391. OnNext(240, 7),
  392. OnCompleted<int>(250)
  393. };
  394. var msgs3 = new[] {
  395. OnNext(150, 1),
  396. OnNext(230, 3),
  397. OnNext(245, 5),
  398. OnCompleted<int>(270)
  399. };
  400. var o1 = scheduler.CreateHotObservable(msgs1);
  401. var o2 = scheduler.CreateHotObservable(msgs2);
  402. var o3 = scheduler.CreateHotObservable(msgs3);
  403. var res = scheduler.Start(() => ObservableEx.ForkJoin(new List<IObservable<int>> { o1, o2, o3 }));
  404. res.Messages.AssertEqual(
  405. OnNext<int[]>(270, l => l.SequenceEqual([4, 7, 5])), // TODO: fix ForkJoin behavior
  406. OnCompleted<int[]>(270)
  407. );
  408. }
  409. [TestMethod]
  410. public void Bug_1302_SelectorThrows_LeftLast()
  411. {
  412. var scheduler = new TestScheduler();
  413. var xs = scheduler.CreateHotObservable(
  414. OnNext(210, 1),
  415. OnCompleted<int>(220)
  416. );
  417. var ys = scheduler.CreateHotObservable(
  418. OnNext(215, 2),
  419. OnCompleted<int>(217)
  420. );
  421. var ex = new Exception();
  422. var results = scheduler.Start(() => xs.ForkJoin<int, int, int>(ys, (x, y) => { throw ex; }));
  423. results.Messages.AssertEqual(
  424. OnError<int>(220, ex)
  425. );
  426. xs.Subscriptions.AssertEqual(
  427. Subscribe(200, 220)
  428. );
  429. ys.Subscriptions.AssertEqual(
  430. Subscribe(200, 217)
  431. );
  432. }
  433. [TestMethod]
  434. public void Bug_1302_SelectorThrows_RightLast()
  435. {
  436. var scheduler = new TestScheduler();
  437. var xs = scheduler.CreateHotObservable(
  438. OnNext(210, 1),
  439. OnCompleted<int>(217)
  440. );
  441. var ys = scheduler.CreateHotObservable(
  442. OnNext(215, 2),
  443. OnCompleted<int>(220)
  444. );
  445. var ex = new Exception();
  446. var results = scheduler.Start(() => xs.ForkJoin<int, int, int>(ys, (x, y) => { throw ex; }));
  447. results.Messages.AssertEqual(
  448. OnError<int>(220, ex)
  449. );
  450. xs.Subscriptions.AssertEqual(
  451. Subscribe(200, 217)
  452. );
  453. ys.Subscriptions.AssertEqual(
  454. Subscribe(200, 220)
  455. );
  456. }
  457. [TestMethod]
  458. public void Bug_1302_RightLast_NoLeft()
  459. {
  460. var scheduler = new TestScheduler();
  461. var xs = scheduler.CreateHotObservable(
  462. OnCompleted<int>(217)
  463. );
  464. var ys = scheduler.CreateHotObservable(
  465. OnNext(215, 2),
  466. OnCompleted<int>(220)
  467. );
  468. var results = scheduler.Start(() => xs.ForkJoin(ys, (x, y) => x + y));
  469. results.Messages.AssertEqual(
  470. OnCompleted<int>(220)
  471. );
  472. xs.Subscriptions.AssertEqual(
  473. Subscribe(200, 217)
  474. );
  475. ys.Subscriptions.AssertEqual(
  476. Subscribe(200, 220)
  477. );
  478. }
  479. [TestMethod]
  480. public void Bug_1302_RightLast_NoRight()
  481. {
  482. var scheduler = new TestScheduler();
  483. var xs = scheduler.CreateHotObservable(
  484. OnNext(215, 2),
  485. OnCompleted<int>(217)
  486. );
  487. var ys = scheduler.CreateHotObservable(
  488. OnCompleted<int>(220)
  489. );
  490. var results = scheduler.Start(() => xs.ForkJoin(ys, (x, y) => x + y));
  491. results.Messages.AssertEqual(
  492. OnCompleted<int>(220)
  493. );
  494. xs.Subscriptions.AssertEqual(
  495. Subscribe(200, 217)
  496. );
  497. ys.Subscriptions.AssertEqual(
  498. Subscribe(200, 220)
  499. );
  500. }
  501. }
  502. }