CreateAsyncTest.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750
  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.Reactive;
  8. using System.Reactive.Concurrency;
  9. using System.Reactive.Disposables;
  10. using System.Reactive.Linq;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. using Microsoft.Reactive.Testing;
  14. using Xunit;
  15. namespace ReactiveTests.Tests
  16. {
  17. public class CreateAsyncTest : ReactiveTest
  18. {
  19. [Fact]
  20. public void CreateAsync_ArgumentChecking()
  21. {
  22. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Create(default(Func<IObserver<int>, Task>)));
  23. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Create(default(Func<IObserver<int>, CancellationToken, Task>)));
  24. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Create(default(Func<IObserver<int>, Task<IDisposable>>)));
  25. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Create(default(Func<IObserver<int>, CancellationToken, Task<IDisposable>>)));
  26. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Create(default(Func<IObserver<int>, Task<Action>>)));
  27. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Create(default(Func<IObserver<int>, CancellationToken, Task<Action>>)));
  28. }
  29. [Fact]
  30. public void CreateAsync_NullCoalescingAction1()
  31. {
  32. var xs = Observable.Create<int>(o =>
  33. {
  34. o.OnNext(42);
  35. return Task.Factory.StartNew(() => default(Action));
  36. });
  37. var lst = new List<int>();
  38. var d = xs.Subscribe(lst.Add);
  39. d.Dispose();
  40. Assert.True(lst.SequenceEqual(new[] { 42 }));
  41. }
  42. [Fact]
  43. public void CreateAsync_NullCoalescingAction2()
  44. {
  45. var xs = Observable.Create<int>((o, ct) =>
  46. {
  47. o.OnNext(42);
  48. return Task.Factory.StartNew(() => default(Action));
  49. });
  50. var lst = new List<int>();
  51. var d = xs.Subscribe(lst.Add);
  52. d.Dispose();
  53. Assert.True(lst.SequenceEqual(new[] { 42 }));
  54. }
  55. [Fact]
  56. public void CreateAsync_NullCoalescingDisposable1()
  57. {
  58. var xs = Observable.Create<int>(o =>
  59. {
  60. o.OnNext(42);
  61. return Task.Factory.StartNew(() => default(IDisposable));
  62. });
  63. var lst = new List<int>();
  64. var d = xs.Subscribe(lst.Add);
  65. d.Dispose();
  66. Assert.True(lst.SequenceEqual(new[] { 42 }));
  67. }
  68. [Fact]
  69. public void CreateAsync_NullCoalescingDisposable2()
  70. {
  71. var xs = Observable.Create<int>((o, ct) =>
  72. {
  73. o.OnNext(42);
  74. return Task.Factory.StartNew(() => default(IDisposable));
  75. });
  76. var lst = new List<int>();
  77. var d = xs.Subscribe(lst.Add);
  78. d.Dispose();
  79. Assert.True(lst.SequenceEqual(new[] { 42 }));
  80. }
  81. private Task Producer1(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  82. {
  83. var tcs = new TaskCompletionSource<object>();
  84. var x = 0;
  85. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  86. {
  87. results.OnNext(++x);
  88. self(TimeSpan.FromTicks(100));
  89. });
  90. token.Register(d.Dispose);
  91. return tcs.Task;
  92. }
  93. [Fact]
  94. public void CreateAsync_Never()
  95. {
  96. RunSynchronously(() =>
  97. {
  98. var scheduler = new TestScheduler();
  99. var res = scheduler.Start(() =>
  100. Observable.Create<int>((observer, token) => Producer1(observer, token, scheduler))
  101. );
  102. res.Messages.AssertEqual(
  103. OnNext(300, 1),
  104. OnNext(400, 2),
  105. OnNext(500, 3),
  106. OnNext(600, 4),
  107. OnNext(700, 5),
  108. OnNext(800, 6),
  109. OnNext(900, 7)
  110. );
  111. });
  112. }
  113. private Task Producer2(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  114. {
  115. var tcs = new TaskCompletionSource<object>();
  116. var x = 0;
  117. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  118. {
  119. if (x == 4)
  120. {
  121. tcs.SetResult(null);
  122. }
  123. results.OnNext(++x);
  124. self(TimeSpan.FromTicks(100));
  125. });
  126. token.Register(d.Dispose);
  127. return tcs.Task;
  128. }
  129. [Fact]
  130. public void CreateAsync_Completed1()
  131. {
  132. RunSynchronously(() =>
  133. {
  134. var scheduler = new TestScheduler();
  135. var res = scheduler.Start(() =>
  136. Observable.Create<int>((observer, token) => Producer2(observer, token, scheduler))
  137. );
  138. res.Messages.AssertEqual(
  139. OnNext(300, 1),
  140. OnNext(400, 2),
  141. OnNext(500, 3),
  142. OnNext(600, 4),
  143. OnCompleted<int>(700)
  144. );
  145. });
  146. }
  147. private Task Producer3(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  148. {
  149. var tcs = new TaskCompletionSource<object>();
  150. var x = 0;
  151. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  152. {
  153. if (x == 4)
  154. {
  155. results.OnCompleted();
  156. }
  157. results.OnNext(++x);
  158. self(TimeSpan.FromTicks(100));
  159. });
  160. token.Register(d.Dispose);
  161. return tcs.Task;
  162. }
  163. [Fact]
  164. public void CreateAsync_Completed2()
  165. {
  166. RunSynchronously(() =>
  167. {
  168. var scheduler = new TestScheduler();
  169. var res = scheduler.Start(() =>
  170. Observable.Create<int>((observer, token) => Producer3(observer, token, scheduler))
  171. );
  172. res.Messages.AssertEqual(
  173. OnNext(300, 1),
  174. OnNext(400, 2),
  175. OnNext(500, 3),
  176. OnNext(600, 4),
  177. OnCompleted<int>(700)
  178. );
  179. });
  180. }
  181. private Task Producer4(IObserver<int> results, CancellationToken token, IScheduler scheduler, Exception exception)
  182. {
  183. var tcs = new TaskCompletionSource<object>();
  184. var x = 0;
  185. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  186. {
  187. if (x == 4)
  188. {
  189. results.OnError(exception);
  190. }
  191. results.OnNext(++x);
  192. self(TimeSpan.FromTicks(100));
  193. });
  194. token.Register(d.Dispose);
  195. return tcs.Task;
  196. }
  197. [Fact]
  198. public void CreateAsync_Error1()
  199. {
  200. RunSynchronously(() =>
  201. {
  202. var scheduler = new TestScheduler();
  203. var exception = new Exception();
  204. var res = scheduler.Start(() =>
  205. Observable.Create<int>((observer, token) => Producer4(observer, token, scheduler, exception))
  206. );
  207. res.Messages.AssertEqual(
  208. OnNext(300, 1),
  209. OnNext(400, 2),
  210. OnNext(500, 3),
  211. OnNext(600, 4),
  212. OnError<int>(700, exception)
  213. );
  214. });
  215. }
  216. private Task Producer5(IObserver<int> results, CancellationToken token, IScheduler scheduler, Exception exception)
  217. {
  218. var tcs = new TaskCompletionSource<object>();
  219. var x = 0;
  220. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  221. {
  222. if (x == 4)
  223. {
  224. tcs.SetException(exception);
  225. }
  226. results.OnNext(++x);
  227. self(TimeSpan.FromTicks(100));
  228. });
  229. token.Register(d.Dispose);
  230. return tcs.Task;
  231. }
  232. [Fact]
  233. public void CreateAsync_Error2()
  234. {
  235. RunSynchronously(() =>
  236. {
  237. var scheduler = new TestScheduler();
  238. var exception = new Exception();
  239. var res = scheduler.Start(() =>
  240. Observable.Create<int>((observer, token) => Producer5(observer, token, scheduler, exception))
  241. );
  242. res.Messages.AssertEqual(
  243. OnNext(300, 1),
  244. OnNext(400, 2),
  245. OnNext(500, 3),
  246. OnNext(600, 4),
  247. OnError<int>(700, exception)
  248. );
  249. });
  250. }
  251. private Task Producer6(IObserver<int> results, CancellationToken token, Exception exception)
  252. {
  253. throw exception;
  254. }
  255. [Fact]
  256. public void CreateAsync_Error3()
  257. {
  258. RunSynchronously(() =>
  259. {
  260. var scheduler = new TestScheduler();
  261. var exception = new InvalidOperationException();
  262. var res = scheduler.Start(() =>
  263. Observable.Create<int>((observer, token) => Producer6(observer, token, exception))
  264. );
  265. res.Messages.AssertEqual(
  266. OnError<int>(200, exception)
  267. );
  268. });
  269. }
  270. private Task Producer7(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  271. {
  272. var tcs = new TaskCompletionSource<object>();
  273. var x = 0;
  274. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  275. {
  276. if (x == 4)
  277. {
  278. tcs.SetResult(null);
  279. }
  280. results.OnNext(++x);
  281. self(TimeSpan.FromTicks(100));
  282. });
  283. token.Register(d.Dispose);
  284. return tcs.Task;
  285. }
  286. [Fact]
  287. public void CreateAsync_Cancel1()
  288. {
  289. RunSynchronously(() =>
  290. {
  291. var scheduler = new TestScheduler();
  292. var res = scheduler.Start(() =>
  293. Observable.Create<int>((observer, token) => Producer7(observer, token, scheduler)),
  294. 650
  295. );
  296. res.Messages.AssertEqual(
  297. OnNext(300, 1),
  298. OnNext(400, 2),
  299. OnNext(500, 3),
  300. OnNext(600, 4)
  301. );
  302. });
  303. }
  304. private Task Producer8(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  305. {
  306. var tcs = new TaskCompletionSource<object>();
  307. var x = 0;
  308. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  309. {
  310. if (x == 4)
  311. {
  312. results.OnCompleted();
  313. }
  314. results.OnNext(++x);
  315. self(TimeSpan.FromTicks(100));
  316. });
  317. token.Register(d.Dispose);
  318. return tcs.Task;
  319. }
  320. [Fact]
  321. public void CreateAsync_Cancel2()
  322. {
  323. RunSynchronously(() =>
  324. {
  325. var scheduler = new TestScheduler();
  326. var res = scheduler.Start(() =>
  327. Observable.Create<int>((observer, token) => Producer8(observer, token, scheduler)),
  328. 650
  329. );
  330. res.Messages.AssertEqual(
  331. OnNext(300, 1),
  332. OnNext(400, 2),
  333. OnNext(500, 3),
  334. OnNext(600, 4)
  335. );
  336. });
  337. }
  338. private Task Producer9(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  339. {
  340. var tcs = new TaskCompletionSource<object>();
  341. var x = 0;
  342. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  343. {
  344. if (x == 4)
  345. {
  346. results.OnCompleted();
  347. }
  348. results.OnNext(++x);
  349. self(TimeSpan.FromTicks(100));
  350. });
  351. token.Register(d.Dispose);
  352. return tcs.Task;
  353. }
  354. [Fact]
  355. public void CreateAsync_Cancel3()
  356. {
  357. RunSynchronously(() =>
  358. {
  359. var scheduler = new TestScheduler();
  360. var res = scheduler.Start(() =>
  361. Observable.Create<int>((observer, token) => Producer9(observer, token, scheduler)),
  362. 750
  363. );
  364. res.Messages.AssertEqual(
  365. OnNext(300, 1),
  366. OnNext(400, 2),
  367. OnNext(500, 3),
  368. OnNext(600, 4),
  369. OnCompleted<int>(700)
  370. );
  371. });
  372. }
  373. private Task Producer10(IObserver<int> results, CancellationToken token, IScheduler scheduler)
  374. {
  375. var tcs = new TaskCompletionSource<object>();
  376. var x = 0;
  377. var d = scheduler.Schedule(TimeSpan.FromTicks(100), self =>
  378. {
  379. if (x == 4)
  380. {
  381. tcs.SetCanceled();
  382. }
  383. results.OnNext(++x);
  384. self(TimeSpan.FromTicks(100));
  385. });
  386. token.Register(d.Dispose);
  387. return tcs.Task;
  388. }
  389. [Fact]
  390. public void CreateAsync_Cancel4()
  391. {
  392. RunSynchronously(() =>
  393. {
  394. var scheduler = new TestScheduler();
  395. var res = scheduler.Start(() =>
  396. Observable.Create<int>((observer, token) => Producer10(observer, token, scheduler))
  397. );
  398. res.Messages.Take(4).AssertEqual(
  399. OnNext(300, 1),
  400. OnNext(400, 2),
  401. OnNext(500, 3),
  402. OnNext(600, 4)
  403. );
  404. Assert.Equal(5, res.Messages.Count);
  405. Assert.Equal(700, res.Messages[4].Time);
  406. Assert.Equal(NotificationKind.OnError, res.Messages[4].Value.Kind);
  407. Assert.True(res.Messages[4].Value.Exception is OperationCanceledException);
  408. });
  409. }
  410. private void RunSynchronously(Action action)
  411. {
  412. var t = new Task(action);
  413. t.RunSynchronously(new SynchronousScheduler());
  414. t.Wait();
  415. }
  416. private class SynchronousScheduler : TaskScheduler
  417. {
  418. protected override IEnumerable<Task> GetScheduledTasks()
  419. {
  420. throw new NotImplementedException();
  421. }
  422. protected override void QueueTask(Task task)
  423. {
  424. TryExecuteTask(task);
  425. }
  426. protected override bool TryExecuteTaskInline(Task task, bool taskWasPreviouslyQueued)
  427. {
  428. return TryExecuteTask(task);
  429. }
  430. }
  431. [Fact]
  432. public void CreateAsync_Task_Simple()
  433. {
  434. var xs = Observable.Create<int>(observer =>
  435. {
  436. return Task.Factory.StartNew(() =>
  437. {
  438. observer.OnNext(42);
  439. observer.OnCompleted();
  440. });
  441. });
  442. var lst = new List<int>();
  443. xs.ForEach(lst.Add);
  444. Assert.True(new[] { 42 }.SequenceEqual(lst));
  445. }
  446. [Fact]
  447. public void CreateAsync_Task_Token()
  448. {
  449. var e = new ManualResetEvent(false);
  450. var xs = Observable.Create<int>((observer, ct) =>
  451. {
  452. return Task.Factory.StartNew(() =>
  453. {
  454. var i = 0;
  455. while (!ct.IsCancellationRequested)
  456. {
  457. if (i++ == 10)
  458. {
  459. e.Set();
  460. }
  461. observer.OnNext(42);
  462. }
  463. });
  464. });
  465. var lst = new List<int>();
  466. var d = xs.Subscribe(lst.Add);
  467. e.WaitOne();
  468. d.Dispose();
  469. Assert.True(lst.Take(10).SequenceEqual(Enumerable.Repeat(42, 10)));
  470. }
  471. [Fact]
  472. public void CreateAsync_IDisposable_Simple()
  473. {
  474. var stopped = new ManualResetEvent(false);
  475. var s = Disposable.Create(() => stopped.Set());
  476. var xs = Observable.Create<int>(observer =>
  477. {
  478. return Task.Factory.StartNew(() =>
  479. {
  480. observer.OnNext(42);
  481. observer.OnCompleted();
  482. return s;
  483. });
  484. });
  485. var lst = new List<int>();
  486. xs.ForEach(lst.Add);
  487. stopped.WaitOne();
  488. Assert.True(new[] { 42 }.SequenceEqual(lst));
  489. }
  490. [Fact]
  491. public void CreateAsync_IDisposable_Token()
  492. {
  493. var stopped = new ManualResetEvent(false);
  494. var s = Disposable.Create(() => stopped.Set());
  495. var e = new ManualResetEvent(false);
  496. var xs = Observable.Create<int>((observer, ct) =>
  497. {
  498. return Task.Factory.StartNew(() =>
  499. {
  500. var i = 0;
  501. while (!ct.IsCancellationRequested)
  502. {
  503. if (i++ == 10)
  504. {
  505. e.Set();
  506. }
  507. observer.OnNext(42);
  508. }
  509. return s;
  510. });
  511. });
  512. var lst = new List<int>();
  513. var d = xs.Subscribe(lst.Add);
  514. e.WaitOne();
  515. d.Dispose();
  516. stopped.WaitOne();
  517. Assert.True(lst.Take(10).SequenceEqual(Enumerable.Repeat(42, 10)));
  518. }
  519. [Fact]
  520. public void CreateAsync_Action_Simple()
  521. {
  522. var stopped = new ManualResetEvent(false);
  523. var s = new Action(() => stopped.Set());
  524. var xs = Observable.Create<int>(observer =>
  525. {
  526. return Task.Factory.StartNew(() =>
  527. {
  528. observer.OnNext(42);
  529. observer.OnCompleted();
  530. return s;
  531. });
  532. });
  533. var lst = new List<int>();
  534. xs.ForEach(lst.Add);
  535. stopped.WaitOne();
  536. Assert.True(new[] { 42 }.SequenceEqual(lst));
  537. }
  538. [Fact]
  539. public void CreateAsync_Action_Token()
  540. {
  541. var stopped = new ManualResetEvent(false);
  542. var s = new Action(() => stopped.Set());
  543. var e = new ManualResetEvent(false);
  544. var xs = Observable.Create<int>((observer, ct) =>
  545. {
  546. return Task.Factory.StartNew(() =>
  547. {
  548. var i = 0;
  549. while (!ct.IsCancellationRequested)
  550. {
  551. if (i++ == 10)
  552. {
  553. e.Set();
  554. }
  555. observer.OnNext(42);
  556. }
  557. return s;
  558. });
  559. });
  560. var lst = new List<int>();
  561. var d = xs.Subscribe(lst.Add);
  562. e.WaitOne();
  563. d.Dispose();
  564. stopped.WaitOne();
  565. Assert.True(lst.Take(10).SequenceEqual(Enumerable.Repeat(42, 10)));
  566. }
  567. [Fact]
  568. public void CreateWithTaskDisposable_NoPrematureTermination()
  569. {
  570. var obs = Observable.Create<int>(async o =>
  571. {
  572. // avoid warning on async o due to no await
  573. await Task.CompletedTask;
  574. var inner = Observable.Range(1, 3);
  575. return inner.Subscribe(x =>
  576. {
  577. o.OnNext(x);
  578. });
  579. });
  580. var result = obs.Take(1).Wait();
  581. }
  582. [Fact]
  583. public void CreateWithTaskAction_NoPrematureTermination()
  584. {
  585. var obs = Observable.Create<int>(async o =>
  586. {
  587. // avoid warning on async o due to no await
  588. await Task.CompletedTask;
  589. var inner = Observable.Range(1, 3);
  590. var d = inner.Subscribe(x =>
  591. {
  592. o.OnNext(x);
  593. });
  594. Action a = () => d.Dispose();
  595. return a;
  596. });
  597. var result = obs.Take(1).Wait();
  598. }
  599. }
  600. }