ForEachAsyncTest.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599
  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.Reactive.Concurrency;
  7. using System.Reactive.Disposables;
  8. using System.Reactive.Linq;
  9. using System.Threading;
  10. using System.Threading.Tasks;
  11. using Microsoft.Reactive.Testing;
  12. using Microsoft.VisualStudio.TestTools.UnitTesting;
  13. using Assert = Xunit.Assert;
  14. namespace ReactiveTests.Tests
  15. {
  16. [TestClass]
  17. public class ForEachAsyncTest : ReactiveTest
  18. {
  19. [TestMethod]
  20. public void ForEachAsync_ArgumentChecking()
  21. {
  22. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), x => { }));
  23. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int>)));
  24. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), x => { }, CancellationToken.None));
  25. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int>), CancellationToken.None));
  26. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), (x, i) => { }));
  27. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int, int>)));
  28. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), (x, i) => { }, CancellationToken.None));
  29. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int, int>), CancellationToken.None));
  30. }
  31. [TestMethod]
  32. public void ForEachAsync_Never()
  33. {
  34. var scheduler = new TestScheduler();
  35. var xs = scheduler.CreateHotObservable(
  36. OnNext(100, 1),
  37. OnNext(200, 2),
  38. OnNext(300, 3),
  39. OnNext(400, 4),
  40. OnNext(500, 5)
  41. );
  42. var task = default(Task);
  43. var cts = new CancellationTokenSource();
  44. var list = new List<Recorded<int>>();
  45. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  46. scheduler.Start();
  47. xs.Subscriptions.AssertEqual(
  48. Subscribe(150)
  49. );
  50. list.AssertEqual(
  51. new Recorded<int>(200, 2),
  52. new Recorded<int>(300, 3),
  53. new Recorded<int>(400, 4),
  54. new Recorded<int>(500, 5)
  55. );
  56. Assert.Equal(TaskStatus.WaitingForActivation, task.Status);
  57. }
  58. [TestMethod]
  59. public void ForEachAsync_Completed()
  60. {
  61. var scheduler = new TestScheduler();
  62. var xs = scheduler.CreateHotObservable(
  63. OnNext(100, 1),
  64. OnNext(200, 2),
  65. OnNext(300, 3),
  66. OnNext(400, 4),
  67. OnNext(500, 5),
  68. OnCompleted<int>(600)
  69. );
  70. var task = default(Task);
  71. var cts = new CancellationTokenSource();
  72. var list = new List<Recorded<int>>();
  73. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  74. scheduler.Start();
  75. xs.Subscriptions.AssertEqual(
  76. Subscribe(150, 600)
  77. );
  78. list.AssertEqual(
  79. new Recorded<int>(200, 2),
  80. new Recorded<int>(300, 3),
  81. new Recorded<int>(400, 4),
  82. new Recorded<int>(500, 5)
  83. );
  84. Assert.Equal(TaskStatus.RanToCompletion, task.Status);
  85. }
  86. [TestMethod]
  87. public void ForEachAsync_Error()
  88. {
  89. var scheduler = new TestScheduler();
  90. var exception = new Exception();
  91. var xs = scheduler.CreateHotObservable(
  92. OnNext(100, 1),
  93. OnNext(200, 2),
  94. OnNext(300, 3),
  95. OnNext(400, 4),
  96. OnNext(500, 5),
  97. OnError<int>(600, exception)
  98. );
  99. var task = default(Task);
  100. var cts = new CancellationTokenSource();
  101. var list = new List<Recorded<int>>();
  102. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  103. scheduler.Start();
  104. xs.Subscriptions.AssertEqual(
  105. Subscribe(150, 600)
  106. );
  107. list.AssertEqual(
  108. new Recorded<int>(200, 2),
  109. new Recorded<int>(300, 3),
  110. new Recorded<int>(400, 4),
  111. new Recorded<int>(500, 5)
  112. );
  113. Assert.Equal(TaskStatus.Faulted, task.Status);
  114. Assert.Same(exception, task.Exception.InnerException);
  115. }
  116. [TestMethod]
  117. public void ForEachAsync_Throw()
  118. {
  119. var scheduler = new TestScheduler();
  120. var exception = new Exception();
  121. var xs = scheduler.CreateHotObservable(
  122. OnNext(100, 1),
  123. OnNext(200, 2),
  124. OnNext(300, 3),
  125. OnNext(400, 4),
  126. OnNext(500, 5),
  127. OnCompleted<int>(600)
  128. );
  129. var task = default(Task);
  130. var cts = new CancellationTokenSource();
  131. var list = new List<Recorded<int>>();
  132. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x =>
  133. {
  134. if (scheduler.Clock > 400)
  135. {
  136. throw exception;
  137. }
  138. list.Add(new Recorded<int>(scheduler.Clock, x));
  139. }, cts.Token));
  140. scheduler.Start();
  141. xs.Subscriptions.AssertEqual(
  142. Subscribe(150, 500)
  143. );
  144. list.AssertEqual(
  145. new Recorded<int>(200, 2),
  146. new Recorded<int>(300, 3),
  147. new Recorded<int>(400, 4)
  148. );
  149. Assert.Equal(TaskStatus.Faulted, task.Status);
  150. Assert.Same(exception, task.Exception.InnerException);
  151. }
  152. [TestMethod]
  153. public void ForEachAsync_CancelDuring()
  154. {
  155. var scheduler = new TestScheduler();
  156. var xs = scheduler.CreateHotObservable(
  157. OnNext(100, 1),
  158. OnNext(200, 2),
  159. OnNext(300, 3),
  160. OnNext(400, 4),
  161. OnNext(500, 5),
  162. OnCompleted<int>(600)
  163. );
  164. var task = default(Task);
  165. var cts = new CancellationTokenSource();
  166. var list = new List<Recorded<int>>();
  167. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  168. scheduler.ScheduleAbsolute(350, () => cts.Cancel());
  169. scheduler.Start();
  170. xs.Subscriptions.AssertEqual(
  171. Subscribe(150, 350)
  172. );
  173. list.AssertEqual(
  174. new Recorded<int>(200, 2),
  175. new Recorded<int>(300, 3)
  176. );
  177. Assert.Equal(TaskStatus.Canceled, task.Status);
  178. }
  179. [TestMethod]
  180. public void ForEachAsync_CancelBefore()
  181. {
  182. var scheduler = new TestScheduler();
  183. var xs = scheduler.CreateHotObservable(
  184. OnNext(100, 1),
  185. OnNext(200, 2),
  186. OnNext(300, 3),
  187. OnNext(400, 4),
  188. OnNext(500, 5),
  189. OnCompleted<int>(600)
  190. );
  191. var task = default(Task);
  192. var cts = new CancellationTokenSource();
  193. var list = new List<Recorded<int>>();
  194. cts.Cancel();
  195. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  196. scheduler.Start();
  197. xs.Subscriptions.AssertEqual(
  198. );
  199. list.AssertEqual(
  200. );
  201. Assert.Equal(TaskStatus.Canceled, task.Status);
  202. }
  203. [TestMethod]
  204. public void ForEachAsync_CancelAfter()
  205. {
  206. var scheduler = new TestScheduler();
  207. var xs = scheduler.CreateHotObservable(
  208. OnNext(100, 1),
  209. OnNext(200, 2),
  210. OnNext(300, 3),
  211. OnNext(400, 4),
  212. OnNext(500, 5),
  213. OnCompleted<int>(600)
  214. );
  215. var task = default(Task);
  216. var cts = new CancellationTokenSource();
  217. var list = new List<Recorded<int>>();
  218. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  219. scheduler.ScheduleAbsolute(700, () => cts.Cancel());
  220. scheduler.Start();
  221. xs.Subscriptions.AssertEqual(
  222. Subscribe(150, 600)
  223. );
  224. list.AssertEqual(
  225. new Recorded<int>(200, 2),
  226. new Recorded<int>(300, 3),
  227. new Recorded<int>(400, 4),
  228. new Recorded<int>(500, 5)
  229. );
  230. Assert.Equal(TaskStatus.RanToCompletion, task.Status);
  231. }
  232. [TestMethod]
  233. public void ForEachAsync_Default()
  234. {
  235. var list = new List<int>();
  236. Observable.Range(1, 10).ForEachAsync(list.Add).Wait();
  237. list.AssertEqual(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
  238. }
  239. [TestMethod]
  240. public void ForEachAsync_Index()
  241. {
  242. var list = new List<int>();
  243. Observable.Range(3, 5).ForEachAsync((x, i) => list.Add(x * i)).Wait();
  244. list.AssertEqual(3 * 0, 4 * 1, 5 * 2, 6 * 3, 7 * 4);
  245. }
  246. [TestMethod]
  247. public void ForEachAsync_Default_Cancel()
  248. {
  249. var N = 10;
  250. for (var n = 0; n < N; n++)
  251. {
  252. var cts = new CancellationTokenSource();
  253. var done = false;
  254. var xs = Observable.Create<int>(observer =>
  255. {
  256. return new CompositeDisposable(
  257. Observable.Repeat(42, Scheduler.Default).Subscribe(observer),
  258. Disposable.Create(() => done = true)
  259. );
  260. });
  261. var lst = new List<int>();
  262. var t = xs.ForEachAsync(
  263. x =>
  264. {
  265. lock (lst)
  266. {
  267. lst.Add(x);
  268. }
  269. },
  270. cts.Token
  271. );
  272. while (true)
  273. {
  274. lock (lst)
  275. {
  276. if (lst.Count >= 10)
  277. {
  278. break;
  279. }
  280. }
  281. }
  282. cts.Cancel();
  283. while (!t.IsCompleted)
  284. {
  285. ;
  286. }
  287. for (var i = 0; i < 10; i++)
  288. {
  289. Assert.Equal(42, lst[i]);
  290. }
  291. Assert.True(done);
  292. Assert.True(t.IsCanceled);
  293. }
  294. }
  295. [TestMethod]
  296. public void ForEachAsync_Index_Cancel()
  297. {
  298. var N = 10;
  299. for (var n = 0; n < N; n++)
  300. {
  301. var cts = new CancellationTokenSource();
  302. var done = false;
  303. var xs = Observable.Create<int>(observer =>
  304. {
  305. return new CompositeDisposable(
  306. Observable.Repeat(42, Scheduler.Default).Subscribe(observer),
  307. Disposable.Create(() => done = true)
  308. );
  309. });
  310. var lst = new List<int>();
  311. var t = xs.ForEachAsync(
  312. (x, i) =>
  313. {
  314. lock (lst)
  315. {
  316. lst.Add(x * i);
  317. }
  318. },
  319. cts.Token
  320. );
  321. while (true)
  322. {
  323. lock (lst)
  324. {
  325. if (lst.Count >= 10)
  326. {
  327. break;
  328. }
  329. }
  330. }
  331. cts.Cancel();
  332. while (!t.IsCompleted)
  333. {
  334. ;
  335. }
  336. for (var i = 0; i < 10; i++)
  337. {
  338. Assert.Equal(i * 42, lst[i]);
  339. }
  340. Assert.True(done);
  341. Assert.True(t.IsCanceled);
  342. }
  343. }
  344. [TestMethod]
  345. public void ForEachAsync_DisposeThrows1()
  346. {
  347. var cts = new CancellationTokenSource();
  348. var ex = new Exception();
  349. var xs = Observable.Create<int>(observer =>
  350. {
  351. return new CompositeDisposable(
  352. Observable.Range(0, 10, Scheduler.CurrentThread).Subscribe(observer),
  353. Disposable.Create(() => { throw ex; })
  354. );
  355. });
  356. var lst = new List<int>();
  357. var t = xs.ForEachAsync(lst.Add, cts.Token);
  358. //
  359. // Unfortunately, this doesn't throw for CurrentThread scheduling. The
  360. // subscription completes prior to assignment of the disposable, so we
  361. // succeed calling the TrySetResult method for the OnCompleted handler
  362. // prior to observing the exception of the Dispose operation, which is
  363. // surfacing upon assignment to the SingleAssignmentDisposable. As a
  364. // result, the exception evaporates.
  365. //
  366. // It'd be a breaking change at this point to rethrow the exception in
  367. // that case, so we're merely asserting regressions here.
  368. //
  369. try
  370. {
  371. t.Wait();
  372. }
  373. catch
  374. {
  375. Assert.True(false);
  376. }
  377. }
  378. [TestMethod]
  379. public void ForEachAsync_DisposeThrows2()
  380. {
  381. var cts = new CancellationTokenSource();
  382. var ex = new Exception();
  383. var xs = Observable.Create<int>(observer =>
  384. {
  385. return new CompositeDisposable(
  386. Observable.Range(0, 10, Scheduler.CurrentThread).Subscribe(observer),
  387. Disposable.Create(() => { throw ex; })
  388. );
  389. });
  390. var lst = new List<int>();
  391. var t = default(Task);
  392. Scheduler.CurrentThread.Schedule(() =>
  393. {
  394. t = xs.ForEachAsync(lst.Add, cts.Token);
  395. });
  396. //
  397. // If the trampoline of the CurrentThread has been installed higher
  398. // up the stack, the assignment of the subscription's disposable to
  399. // the SingleAssignmentDisposable can complete prior to the Dispose
  400. // method being called from the OnCompleted handler. In this case,
  401. // the OnCompleted handler's invocation of Dispose will cause the
  402. // exception to occur, and it bubbles out through TrySetException.
  403. //
  404. try
  405. {
  406. t.Wait();
  407. }
  408. catch (AggregateException err)
  409. {
  410. Assert.Equal(1, err.InnerExceptions.Count);
  411. Assert.Same(ex, err.InnerExceptions[0]);
  412. }
  413. }
  414. #if !NO_THREAD
  415. [TestMethod]
  416. [TestCategory("SkipCI")]
  417. public void ForEachAsync_DisposeThrows()
  418. {
  419. //
  420. // Unfortunately, this test is non-deterministic due to the race
  421. // conditions described above in the tests using the CurrentThread
  422. // scheduler. The exception can come out through the OnCompleted
  423. // handler but can equally well get swallowed if the main thread
  424. // hasn't reached the assignment of the disposable yet, causing
  425. // the OnCompleted handler to win the race. The user can deal with
  426. // this by hooking an exception handler to the scheduler, so we
  427. // assert this behavior here.
  428. //
  429. // It'd be a breaking change at this point to change rethrowing
  430. // behavior, so we're merely asserting regressions here.
  431. //
  432. var hasCaughtEscapingException = 0;
  433. var cts = new CancellationTokenSource();
  434. var ex = new Exception();
  435. var s = Scheduler.Default.Catch<Exception>(err =>
  436. {
  437. Volatile.Write(ref hasCaughtEscapingException, 1);
  438. return ex == err;
  439. });
  440. while (Volatile.Read(ref hasCaughtEscapingException) == 0)
  441. {
  442. var xs = Observable.Create<int>(observer =>
  443. {
  444. return new CompositeDisposable(
  445. Observable.Range(0, 10, s).Subscribe(observer),
  446. Disposable.Create(() => { throw ex; })
  447. );
  448. });
  449. var lst = new List<int>();
  450. var t = xs.ForEachAsync(lst.Add, cts.Token);
  451. try
  452. {
  453. t.Wait();
  454. }
  455. catch (AggregateException err)
  456. {
  457. Assert.Equal(1, err.InnerExceptions.Count);
  458. Assert.Same(ex, err.InnerExceptions[0]);
  459. }
  460. }
  461. }
  462. [TestMethod]
  463. public void ForEachAsync_SubscribeThrows()
  464. {
  465. var ex = new Exception();
  466. var x = 42;
  467. var xs = Observable.Create<int>(observer =>
  468. {
  469. if (x == 42)
  470. {
  471. throw ex;
  472. }
  473. return Disposable.Empty;
  474. });
  475. var t = xs.ForEachAsync(_ => { });
  476. try
  477. {
  478. t.Wait();
  479. Assert.True(false);
  480. }
  481. catch (AggregateException err)
  482. {
  483. Assert.Equal(1, err.InnerExceptions.Count);
  484. Assert.Same(ex, err.InnerExceptions[0]);
  485. }
  486. }
  487. #endif
  488. }
  489. }