ObservableImperativeTest.cs 53 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855
  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.Concurrency;
  8. using System.Reactive.Disposables;
  9. using System.Reactive.Linq;
  10. using System.Threading;
  11. using Microsoft.Reactive.Testing;
  12. using Xunit;
  13. using ReactiveTests.Dummies;
  14. #if !NO_TPL
  15. using System.Threading.Tasks;
  16. #endif
  17. namespace ReactiveTests.Tests
  18. {
  19. public partial class ObservableImperativeTest : ReactiveTest
  20. {
  21. #region ForEachAsync
  22. #if !NO_TPL
  23. [Fact]
  24. public void ForEachAsync_ArgumentChecking()
  25. {
  26. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), x => { }));
  27. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int>)));
  28. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), x => { }, CancellationToken.None));
  29. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int>), CancellationToken.None));
  30. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), (x, i) => { }));
  31. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int, int>)));
  32. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(default(IObservable<int>), (x, i) => { }, CancellationToken.None));
  33. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ForEachAsync(Observable.Never<int>(), default(Action<int, int>), CancellationToken.None));
  34. }
  35. [Fact]
  36. public void ForEachAsync_Never()
  37. {
  38. var scheduler = new TestScheduler();
  39. var xs = scheduler.CreateHotObservable(
  40. OnNext(100, 1),
  41. OnNext(200, 2),
  42. OnNext(300, 3),
  43. OnNext(400, 4),
  44. OnNext(500, 5)
  45. );
  46. var task = default(Task);
  47. var cts = new CancellationTokenSource();
  48. var list = new List<Recorded<int>>();
  49. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  50. scheduler.Start();
  51. xs.Subscriptions.AssertEqual(
  52. Subscribe(150)
  53. );
  54. list.AssertEqual(
  55. new Recorded<int>(200, 2),
  56. new Recorded<int>(300, 3),
  57. new Recorded<int>(400, 4),
  58. new Recorded<int>(500, 5)
  59. );
  60. Assert.Equal(TaskStatus.WaitingForActivation, task.Status);
  61. }
  62. [Fact]
  63. public void ForEachAsync_Completed()
  64. {
  65. var scheduler = new TestScheduler();
  66. var xs = scheduler.CreateHotObservable(
  67. OnNext(100, 1),
  68. OnNext(200, 2),
  69. OnNext(300, 3),
  70. OnNext(400, 4),
  71. OnNext(500, 5),
  72. OnCompleted<int>(600)
  73. );
  74. var task = default(Task);
  75. var cts = new CancellationTokenSource();
  76. var list = new List<Recorded<int>>();
  77. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  78. scheduler.Start();
  79. xs.Subscriptions.AssertEqual(
  80. Subscribe(150, 600)
  81. );
  82. list.AssertEqual(
  83. new Recorded<int>(200, 2),
  84. new Recorded<int>(300, 3),
  85. new Recorded<int>(400, 4),
  86. new Recorded<int>(500, 5)
  87. );
  88. Assert.Equal(TaskStatus.RanToCompletion, task.Status);
  89. }
  90. [Fact]
  91. public void ForEachAsync_Error()
  92. {
  93. var scheduler = new TestScheduler();
  94. var exception = new Exception();
  95. var xs = scheduler.CreateHotObservable(
  96. OnNext(100, 1),
  97. OnNext(200, 2),
  98. OnNext(300, 3),
  99. OnNext(400, 4),
  100. OnNext(500, 5),
  101. OnError<int>(600, exception)
  102. );
  103. var task = default(Task);
  104. var cts = new CancellationTokenSource();
  105. var list = new List<Recorded<int>>();
  106. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  107. scheduler.Start();
  108. xs.Subscriptions.AssertEqual(
  109. Subscribe(150, 600)
  110. );
  111. list.AssertEqual(
  112. new Recorded<int>(200, 2),
  113. new Recorded<int>(300, 3),
  114. new Recorded<int>(400, 4),
  115. new Recorded<int>(500, 5)
  116. );
  117. Assert.Equal(TaskStatus.Faulted, task.Status);
  118. Assert.Same(exception, task.Exception.InnerException);
  119. }
  120. [Fact]
  121. public void ForEachAsync_Throw()
  122. {
  123. var scheduler = new TestScheduler();
  124. var exception = new Exception();
  125. var xs = scheduler.CreateHotObservable(
  126. OnNext(100, 1),
  127. OnNext(200, 2),
  128. OnNext(300, 3),
  129. OnNext(400, 4),
  130. OnNext(500, 5),
  131. OnCompleted<int>(600)
  132. );
  133. var task = default(Task);
  134. var cts = new CancellationTokenSource();
  135. var list = new List<Recorded<int>>();
  136. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x =>
  137. {
  138. if (scheduler.Clock > 400)
  139. throw exception;
  140. list.Add(new Recorded<int>(scheduler.Clock, x));
  141. }, cts.Token));
  142. scheduler.Start();
  143. xs.Subscriptions.AssertEqual(
  144. Subscribe(150, 500)
  145. );
  146. list.AssertEqual(
  147. new Recorded<int>(200, 2),
  148. new Recorded<int>(300, 3),
  149. new Recorded<int>(400, 4)
  150. );
  151. Assert.Equal(TaskStatus.Faulted, task.Status);
  152. Assert.Same(exception, task.Exception.InnerException);
  153. }
  154. [Fact]
  155. public void ForEachAsync_CancelDuring()
  156. {
  157. var scheduler = new TestScheduler();
  158. var xs = scheduler.CreateHotObservable(
  159. OnNext(100, 1),
  160. OnNext(200, 2),
  161. OnNext(300, 3),
  162. OnNext(400, 4),
  163. OnNext(500, 5),
  164. OnCompleted<int>(600)
  165. );
  166. var task = default(Task);
  167. var cts = new CancellationTokenSource();
  168. var list = new List<Recorded<int>>();
  169. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  170. scheduler.ScheduleAbsolute(350, () => cts.Cancel());
  171. scheduler.Start();
  172. xs.Subscriptions.AssertEqual(
  173. Subscribe(150, 350)
  174. );
  175. list.AssertEqual(
  176. new Recorded<int>(200, 2),
  177. new Recorded<int>(300, 3)
  178. );
  179. Assert.Equal(TaskStatus.Canceled, task.Status);
  180. }
  181. [Fact]
  182. public void ForEachAsync_CancelBefore()
  183. {
  184. var scheduler = new TestScheduler();
  185. var xs = scheduler.CreateHotObservable(
  186. OnNext(100, 1),
  187. OnNext(200, 2),
  188. OnNext(300, 3),
  189. OnNext(400, 4),
  190. OnNext(500, 5),
  191. OnCompleted<int>(600)
  192. );
  193. var task = default(Task);
  194. var cts = new CancellationTokenSource();
  195. var list = new List<Recorded<int>>();
  196. cts.Cancel();
  197. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  198. scheduler.Start();
  199. xs.Subscriptions.AssertEqual(
  200. );
  201. list.AssertEqual(
  202. );
  203. Assert.Equal(TaskStatus.Canceled, task.Status);
  204. }
  205. [Fact]
  206. public void ForEachAsync_CancelAfter()
  207. {
  208. var scheduler = new TestScheduler();
  209. var xs = scheduler.CreateHotObservable(
  210. OnNext(100, 1),
  211. OnNext(200, 2),
  212. OnNext(300, 3),
  213. OnNext(400, 4),
  214. OnNext(500, 5),
  215. OnCompleted<int>(600)
  216. );
  217. var task = default(Task);
  218. var cts = new CancellationTokenSource();
  219. var list = new List<Recorded<int>>();
  220. scheduler.ScheduleAbsolute(150, () => task = xs.ForEachAsync(x => list.Add(new Recorded<int>(scheduler.Clock, x)), cts.Token));
  221. scheduler.ScheduleAbsolute(700, () => cts.Cancel());
  222. scheduler.Start();
  223. xs.Subscriptions.AssertEqual(
  224. Subscribe(150, 600)
  225. );
  226. list.AssertEqual(
  227. new Recorded<int>(200, 2),
  228. new Recorded<int>(300, 3),
  229. new Recorded<int>(400, 4),
  230. new Recorded<int>(500, 5)
  231. );
  232. Assert.Equal(TaskStatus.RanToCompletion, task.Status);
  233. }
  234. [Fact]
  235. public void ForEachAsync_Default()
  236. {
  237. var list = new List<int>();
  238. Observable.Range(1, 10).ForEachAsync(list.Add).Wait();
  239. list.AssertEqual(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
  240. }
  241. [Fact]
  242. public void ForEachAsync_Index()
  243. {
  244. var list = new List<int>();
  245. Observable.Range(3, 5).ForEachAsync((x, i) => list.Add(x * i)).Wait();
  246. list.AssertEqual(3 * 0, 4 * 1, 5 * 2, 6 * 3, 7 * 4);
  247. }
  248. [Fact]
  249. public void ForEachAsync_Default_Cancel()
  250. {
  251. var N = 10;
  252. for (int n = 0; n < N; n++)
  253. {
  254. var cts = new CancellationTokenSource();
  255. var done = false;
  256. var xs = Observable.Create<int>(observer =>
  257. {
  258. return new CompositeDisposable(
  259. Observable.Repeat(42, Scheduler.Default).Subscribe(observer),
  260. Disposable.Create(() => done = true)
  261. );
  262. });
  263. var lst = new List<int>();
  264. var t = xs.ForEachAsync(
  265. x =>
  266. {
  267. lock (lst)
  268. lst.Add(x);
  269. },
  270. cts.Token
  271. );
  272. while (true)
  273. {
  274. lock (lst)
  275. if (lst.Count >= 10)
  276. break;
  277. }
  278. cts.Cancel();
  279. while (!t.IsCompleted)
  280. ;
  281. for (int i = 0; i < 10; i++)
  282. Assert.Equal(42, lst[i]);
  283. Assert.True(done);
  284. Assert.True(t.IsCanceled);
  285. }
  286. }
  287. [Fact]
  288. public void ForEachAsync_Index_Cancel()
  289. {
  290. var N = 10;
  291. for (int n = 0; n < N; n++)
  292. {
  293. var cts = new CancellationTokenSource();
  294. var done = false;
  295. var xs = Observable.Create<int>(observer =>
  296. {
  297. return new CompositeDisposable(
  298. Observable.Repeat(42, Scheduler.Default).Subscribe(observer),
  299. Disposable.Create(() => done = true)
  300. );
  301. });
  302. var lst = new List<int>();
  303. var t = xs.ForEachAsync(
  304. (x, i) =>
  305. {
  306. lock (lst)
  307. lst.Add(x * i);
  308. },
  309. cts.Token
  310. );
  311. while (true)
  312. {
  313. lock (lst)
  314. if (lst.Count >= 10)
  315. break;
  316. }
  317. cts.Cancel();
  318. while (!t.IsCompleted)
  319. ;
  320. for (int i = 0; i < 10; i++)
  321. Assert.Equal(i * 42, lst[i]);
  322. Assert.True(done);
  323. Assert.True(t.IsCanceled);
  324. }
  325. }
  326. [Fact]
  327. public void ForEachAsync_DisposeThrows1()
  328. {
  329. var cts = new CancellationTokenSource();
  330. var ex = new Exception();
  331. var xs = Observable.Create<int>(observer =>
  332. {
  333. return new CompositeDisposable(
  334. Observable.Range(0, 10, Scheduler.CurrentThread).Subscribe(observer),
  335. Disposable.Create(() => { throw ex; })
  336. );
  337. });
  338. var lst = new List<int>();
  339. var t = xs.ForEachAsync(lst.Add, cts.Token);
  340. //
  341. // Unfortunately, this doesn't throw for CurrentThread scheduling. The
  342. // subscription completes prior to assignment of the disposable, so we
  343. // succeed calling the TrySetResult method for the OnCompleted handler
  344. // prior to observing the exception of the Dispose operation, which is
  345. // surfacing upon assignment to the SingleAssignmentDisposable. As a
  346. // result, the exception evaporates.
  347. //
  348. // It'd be a breaking change at this point to rethrow the exception in
  349. // that case, so we're merely asserting regressions here.
  350. //
  351. try
  352. {
  353. t.Wait();
  354. }
  355. catch
  356. {
  357. Assert.True(false);
  358. }
  359. }
  360. [Fact]
  361. public void ForEachAsync_DisposeThrows2()
  362. {
  363. var cts = new CancellationTokenSource();
  364. var ex = new Exception();
  365. var xs = Observable.Create<int>(observer =>
  366. {
  367. return new CompositeDisposable(
  368. Observable.Range(0, 10, Scheduler.CurrentThread).Subscribe(observer),
  369. Disposable.Create(() => { throw ex; })
  370. );
  371. });
  372. var lst = new List<int>();
  373. var t = default(Task);
  374. Scheduler.CurrentThread.Schedule(() =>
  375. {
  376. t = xs.ForEachAsync(lst.Add, cts.Token);
  377. });
  378. //
  379. // If the trampoline of the CurrentThread has been installed higher
  380. // up the stack, the assignment of the subscription's disposable to
  381. // the SingleAssignmentDisposable can complete prior to the Dispose
  382. // method being called from the OnCompleted handler. In this case,
  383. // the OnCompleted handler's invocation of Dispose will cause the
  384. // exception to occur, and it bubbles out through TrySetException.
  385. //
  386. try
  387. {
  388. t.Wait();
  389. }
  390. catch (AggregateException err)
  391. {
  392. Assert.Equal(1, err.InnerExceptions.Count);
  393. Assert.Same(ex, err.InnerExceptions[0]);
  394. }
  395. }
  396. #if !NO_THREAD
  397. [Fact]
  398. [Trait("SkipCI", "true")]
  399. public void ForEachAsync_DisposeThrows()
  400. {
  401. //
  402. // Unfortunately, this test is non-deterministic due to the race
  403. // conditions described above in the tests using the CurrentThread
  404. // scheduler. The exception can come out through the OnCompleted
  405. // handler but can equally well get swallowed if the main thread
  406. // hasn't reached the assignment of the disposable yet, causing
  407. // the OnCompleted handler to win the race. The user can deal with
  408. // this by hooking an exception handler to the scheduler, so we
  409. // assert this behavior here.
  410. //
  411. // It'd be a breaking change at this point to change rethrowing
  412. // behavior, so we're merely asserting regressions here.
  413. //
  414. var hasCaughtEscapingException = 0;
  415. var cts = new CancellationTokenSource();
  416. var ex = new Exception();
  417. var s = Scheduler.Default.Catch<Exception>(err =>
  418. {
  419. Volatile.Write(ref hasCaughtEscapingException, 1);
  420. return ex == err;
  421. });
  422. while (Volatile.Read(ref hasCaughtEscapingException) == 0)
  423. {
  424. var xs = Observable.Create<int>(observer =>
  425. {
  426. return new CompositeDisposable(
  427. Observable.Range(0, 10, s).Subscribe(observer),
  428. Disposable.Create(() => { throw ex; })
  429. );
  430. });
  431. var lst = new List<int>();
  432. var t = xs.ForEachAsync(lst.Add, cts.Token);
  433. try
  434. {
  435. t.Wait();
  436. }
  437. catch (AggregateException err)
  438. {
  439. Assert.Equal(1, err.InnerExceptions.Count);
  440. Assert.Same(ex, err.InnerExceptions[0]);
  441. }
  442. }
  443. }
  444. #endif
  445. [Fact]
  446. public void ForEachAsync_SubscribeThrows()
  447. {
  448. var ex = new Exception();
  449. var x = 42;
  450. var xs = Observable.Create<int>(observer =>
  451. {
  452. if (x == 42)
  453. throw ex;
  454. return Disposable.Empty;
  455. });
  456. var t = xs.ForEachAsync(_ => { });
  457. try
  458. {
  459. t.Wait();
  460. Assert.True(false);
  461. }
  462. catch (AggregateException err)
  463. {
  464. Assert.Equal(1, err.InnerExceptions.Count);
  465. Assert.Same(ex, err.InnerExceptions[0]);
  466. }
  467. }
  468. #endif
  469. #endregion
  470. #region + Case +
  471. [Fact]
  472. public void Case_ArgumentChecking()
  473. {
  474. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case(null, new Dictionary<int, IObservable<int>>(), DummyObservable<int>.Instance));
  475. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case(DummyFunc<int>.Instance, null, DummyObservable<int>.Instance));
  476. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case(DummyFunc<int>.Instance, new Dictionary<int, IObservable<int>>(), default(IObservable<int>)));
  477. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case(null, new Dictionary<int, IObservable<int>>(), DummyScheduler.Instance));
  478. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case<int, int>(DummyFunc<int>.Instance, null, DummyScheduler.Instance));
  479. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case(DummyFunc<int>.Instance, new Dictionary<int, IObservable<int>>(), default(IScheduler)));
  480. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case(null, new Dictionary<int, IObservable<int>>()));
  481. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Case<int, int>(DummyFunc<int>.Instance, null));
  482. }
  483. [Fact]
  484. public void Case_One()
  485. {
  486. var scheduler = new TestScheduler();
  487. var xs = scheduler.CreateHotObservable(
  488. OnNext(210, 1),
  489. OnNext(240, 2),
  490. OnNext(270, 3),
  491. OnCompleted<int>(300)
  492. );
  493. var ys = scheduler.CreateHotObservable(
  494. OnNext(220, 11),
  495. OnNext(250, 12),
  496. OnNext(280, 13),
  497. OnCompleted<int>(310)
  498. );
  499. var zs = scheduler.CreateHotObservable(
  500. OnNext(230, 21),
  501. OnNext(240, 22),
  502. OnNext(290, 23),
  503. OnCompleted<int>(320)
  504. );
  505. var map = new Dictionary<int, IObservable<int>>
  506. {
  507. { 1, xs },
  508. { 2, ys }
  509. };
  510. var results = scheduler.Start(() => Observable.Case(() => 1, map, zs));
  511. results.Messages.AssertEqual(
  512. OnNext(210, 1),
  513. OnNext(240, 2),
  514. OnNext(270, 3),
  515. OnCompleted<int>(300)
  516. );
  517. xs.Subscriptions.AssertEqual(
  518. Subscribe(200, 300)
  519. );
  520. ys.Subscriptions.AssertEqual(
  521. );
  522. zs.Subscriptions.AssertEqual(
  523. );
  524. }
  525. [Fact]
  526. public void Case_Two()
  527. {
  528. var scheduler = new TestScheduler();
  529. var xs = scheduler.CreateHotObservable(
  530. OnNext(210, 1),
  531. OnNext(240, 2),
  532. OnNext(270, 3),
  533. OnCompleted<int>(300)
  534. );
  535. var ys = scheduler.CreateHotObservable(
  536. OnNext(220, 11),
  537. OnNext(250, 12),
  538. OnNext(280, 13),
  539. OnCompleted<int>(310)
  540. );
  541. var zs = scheduler.CreateHotObservable(
  542. OnNext(230, 21),
  543. OnNext(240, 22),
  544. OnNext(290, 23),
  545. OnCompleted<int>(320)
  546. );
  547. var map = new Dictionary<int, IObservable<int>>
  548. {
  549. { 1, xs },
  550. { 2, ys }
  551. };
  552. var results = scheduler.Start(() => Observable.Case(() => 2, map, zs));
  553. results.Messages.AssertEqual(
  554. OnNext(220, 11),
  555. OnNext(250, 12),
  556. OnNext(280, 13),
  557. OnCompleted<int>(310)
  558. );
  559. xs.Subscriptions.AssertEqual(
  560. );
  561. ys.Subscriptions.AssertEqual(
  562. Subscribe(200, 310)
  563. );
  564. zs.Subscriptions.AssertEqual(
  565. );
  566. }
  567. [Fact]
  568. public void Case_Three()
  569. {
  570. var scheduler = new TestScheduler();
  571. var xs = scheduler.CreateHotObservable(
  572. OnNext(210, 1),
  573. OnNext(240, 2),
  574. OnNext(270, 3),
  575. OnCompleted<int>(300)
  576. );
  577. var ys = scheduler.CreateHotObservable(
  578. OnNext(220, 11),
  579. OnNext(250, 12),
  580. OnNext(280, 13),
  581. OnCompleted<int>(310)
  582. );
  583. var zs = scheduler.CreateHotObservable(
  584. OnNext(230, 21),
  585. OnNext(240, 22),
  586. OnNext(290, 23),
  587. OnCompleted<int>(320)
  588. );
  589. var map = new Dictionary<int, IObservable<int>>
  590. {
  591. { 1, xs },
  592. { 2, ys }
  593. };
  594. var results = scheduler.Start(() => Observable.Case(() => 3, map, zs));
  595. results.Messages.AssertEqual(
  596. OnNext(230, 21),
  597. OnNext(240, 22),
  598. OnNext(290, 23),
  599. OnCompleted<int>(320)
  600. );
  601. xs.Subscriptions.AssertEqual(
  602. );
  603. ys.Subscriptions.AssertEqual(
  604. );
  605. zs.Subscriptions.AssertEqual(
  606. Subscribe(200, 320)
  607. );
  608. }
  609. [Fact]
  610. public void Case_Throw()
  611. {
  612. var scheduler = new TestScheduler();
  613. var xs = scheduler.CreateHotObservable(
  614. OnNext(210, 1),
  615. OnNext(240, 2),
  616. OnNext(270, 3),
  617. OnCompleted<int>(300)
  618. );
  619. var ys = scheduler.CreateHotObservable(
  620. OnNext(220, 11),
  621. OnNext(250, 12),
  622. OnNext(280, 13),
  623. OnCompleted<int>(310)
  624. );
  625. var zs = scheduler.CreateHotObservable(
  626. OnNext(230, 21),
  627. OnNext(240, 22),
  628. OnNext(290, 23),
  629. OnCompleted<int>(320)
  630. );
  631. var map = new Dictionary<int, IObservable<int>>
  632. {
  633. { 1, xs },
  634. { 2, ys }
  635. };
  636. var ex = new Exception();
  637. var results = scheduler.Start(() => Observable.Case(() => Throw<int>(ex), map, zs));
  638. results.Messages.AssertEqual(
  639. OnError<int>(200, ex)
  640. );
  641. xs.Subscriptions.AssertEqual(
  642. );
  643. ys.Subscriptions.AssertEqual(
  644. );
  645. zs.Subscriptions.AssertEqual(
  646. );
  647. }
  648. [Fact]
  649. public void CaseWithDefault_One()
  650. {
  651. var scheduler = new TestScheduler();
  652. var xs = scheduler.CreateHotObservable(
  653. OnNext(210, 1),
  654. OnNext(240, 2),
  655. OnNext(270, 3),
  656. OnCompleted<int>(300)
  657. );
  658. var ys = scheduler.CreateHotObservable(
  659. OnNext(220, 11),
  660. OnNext(250, 12),
  661. OnNext(280, 13),
  662. OnCompleted<int>(310)
  663. );
  664. var map = new Dictionary<int, IObservable<int>>
  665. {
  666. { 1, xs },
  667. { 2, ys }
  668. };
  669. var results = scheduler.Start(() => Observable.Case(() => 1, map, scheduler));
  670. results.Messages.AssertEqual(
  671. OnNext(210, 1),
  672. OnNext(240, 2),
  673. OnNext(270, 3),
  674. OnCompleted<int>(300)
  675. );
  676. xs.Subscriptions.AssertEqual(
  677. Subscribe(200, 300)
  678. );
  679. ys.Subscriptions.AssertEqual(
  680. );
  681. }
  682. [Fact]
  683. public void CaseWithDefault_Two()
  684. {
  685. var scheduler = new TestScheduler();
  686. var xs = scheduler.CreateHotObservable(
  687. OnNext(210, 1),
  688. OnNext(240, 2),
  689. OnNext(270, 3),
  690. OnCompleted<int>(300)
  691. );
  692. var ys = scheduler.CreateHotObservable(
  693. OnNext(220, 11),
  694. OnNext(250, 12),
  695. OnNext(280, 13),
  696. OnCompleted<int>(310)
  697. );
  698. var map = new Dictionary<int, IObservable<int>>
  699. {
  700. { 1, xs },
  701. { 2, ys }
  702. };
  703. var results = scheduler.Start(() => Observable.Case(() => 2, map, scheduler));
  704. results.Messages.AssertEqual(
  705. OnNext(220, 11),
  706. OnNext(250, 12),
  707. OnNext(280, 13),
  708. OnCompleted<int>(310)
  709. );
  710. xs.Subscriptions.AssertEqual(
  711. );
  712. ys.Subscriptions.AssertEqual(
  713. Subscribe(200, 310)
  714. );
  715. }
  716. [Fact]
  717. public void CaseWithDefault_Three()
  718. {
  719. var scheduler = new TestScheduler();
  720. var xs = scheduler.CreateHotObservable(
  721. OnNext(210, 1),
  722. OnNext(240, 2),
  723. OnNext(270, 3),
  724. OnCompleted<int>(300)
  725. );
  726. var ys = scheduler.CreateHotObservable(
  727. OnNext(220, 11),
  728. OnNext(250, 12),
  729. OnNext(280, 13),
  730. OnCompleted<int>(310)
  731. );
  732. var map = new Dictionary<int, IObservable<int>>
  733. {
  734. { 1, xs },
  735. { 2, ys }
  736. };
  737. var results = scheduler.Start(() => Observable.Case(() => 3, map, scheduler));
  738. results.Messages.AssertEqual(
  739. OnCompleted<int>(201)
  740. );
  741. xs.Subscriptions.AssertEqual(
  742. );
  743. ys.Subscriptions.AssertEqual(
  744. );
  745. }
  746. [Fact]
  747. public void CaseWithDefault_Throw()
  748. {
  749. var scheduler = new TestScheduler();
  750. var xs = scheduler.CreateHotObservable(
  751. OnNext(210, 1),
  752. OnNext(240, 2),
  753. OnNext(270, 3),
  754. OnCompleted<int>(300)
  755. );
  756. var ys = scheduler.CreateHotObservable(
  757. OnNext(220, 11),
  758. OnNext(250, 12),
  759. OnNext(280, 13),
  760. OnCompleted<int>(310)
  761. );
  762. var map = new Dictionary<int, IObservable<int>>
  763. {
  764. { 1, xs },
  765. { 2, ys }
  766. };
  767. var ex = new Exception();
  768. var results = scheduler.Start(() => Observable.Case(() => Throw<int>(ex), map, scheduler));
  769. results.Messages.AssertEqual(
  770. OnError<int>(200, ex)
  771. );
  772. xs.Subscriptions.AssertEqual(
  773. );
  774. ys.Subscriptions.AssertEqual(
  775. );
  776. }
  777. [Fact]
  778. public void CaseWithDefault_CheckDefault()
  779. {
  780. Observable.Case(() => 1, new Dictionary<int, IObservable<int>>(), DefaultScheduler.Instance)
  781. .AssertEqual(Observable.Case(() => 1, new Dictionary<int, IObservable<int>>()));
  782. }
  783. [Fact]
  784. public void Case_Error()
  785. {
  786. var scheduler = new TestScheduler();
  787. var ex = new Exception();
  788. var xs = scheduler.CreateHotObservable(
  789. OnNext(210, 1),
  790. OnNext(240, 2),
  791. OnNext(270, 3),
  792. OnError<int>(300, ex)
  793. );
  794. var ys = scheduler.CreateHotObservable(
  795. OnNext(220, 11),
  796. OnNext(250, 12),
  797. OnNext(280, 13),
  798. OnCompleted<int>(310)
  799. );
  800. var map = new Dictionary<int, IObservable<int>>
  801. {
  802. { 1, xs },
  803. { 2, ys }
  804. };
  805. var results = scheduler.Start(() => Observable.Case(() => 1, map, scheduler));
  806. results.Messages.AssertEqual(
  807. OnNext(210, 1),
  808. OnNext(240, 2),
  809. OnNext(270, 3),
  810. OnError<int>(300, ex)
  811. );
  812. xs.Subscriptions.AssertEqual(
  813. Subscribe(200, 300)
  814. );
  815. ys.Subscriptions.AssertEqual(
  816. );
  817. }
  818. #endregion
  819. #region + DoWhile +
  820. [Fact]
  821. public void DoWhile_ArgumentChecking()
  822. {
  823. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.DoWhile(DummyObservable<int>.Instance, null));
  824. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.DoWhile(default(IObservable<int>), DummyFunc<bool>.Instance));
  825. }
  826. [Fact]
  827. public void DoWhile_AlwaysFalse()
  828. {
  829. var scheduler = new TestScheduler();
  830. var xs = scheduler.CreateColdObservable(
  831. OnNext(50, 1),
  832. OnNext(100, 2),
  833. OnNext(150, 3),
  834. OnNext(200, 4),
  835. OnCompleted<int>(250)
  836. );
  837. var results = scheduler.Start(() => Observable.DoWhile(xs, () => false));
  838. results.Messages.AssertEqual(
  839. OnNext(250, 1),
  840. OnNext(300, 2),
  841. OnNext(350, 3),
  842. OnNext(400, 4),
  843. OnCompleted<int>(450)
  844. );
  845. xs.Subscriptions.AssertEqual(
  846. Subscribe(200, 450)
  847. );
  848. }
  849. [Fact]
  850. public void DoWhile_AlwaysTrue()
  851. {
  852. var scheduler = new TestScheduler();
  853. var xs = scheduler.CreateColdObservable(
  854. OnNext(50, 1),
  855. OnNext(100, 2),
  856. OnNext(150, 3),
  857. OnNext(200, 4),
  858. OnCompleted<int>(250)
  859. );
  860. var results = scheduler.Start(() => Observable.DoWhile(xs, () => true));
  861. results.Messages.AssertEqual(
  862. OnNext(250, 1),
  863. OnNext(300, 2),
  864. OnNext(350, 3),
  865. OnNext(400, 4),
  866. OnNext(500, 1),
  867. OnNext(550, 2),
  868. OnNext(600, 3),
  869. OnNext(650, 4),
  870. OnNext(750, 1),
  871. OnNext(800, 2),
  872. OnNext(850, 3),
  873. OnNext(900, 4)
  874. );
  875. xs.Subscriptions.AssertEqual(
  876. Subscribe(200, 450),
  877. Subscribe(450, 700),
  878. Subscribe(700, 950),
  879. Subscribe(950, 1000)
  880. );
  881. }
  882. [Fact]
  883. public void DoWhile_AlwaysTrue_Throw()
  884. {
  885. var scheduler = new TestScheduler();
  886. var ex = new Exception();
  887. var xs = scheduler.CreateColdObservable(
  888. OnError<int>(50, ex)
  889. );
  890. var results = scheduler.Start(() => Observable.DoWhile(xs, () => true));
  891. results.Messages.AssertEqual(
  892. OnError<int>(250, ex)
  893. );
  894. xs.Subscriptions.AssertEqual(
  895. Subscribe(200, 250)
  896. );
  897. }
  898. [Fact]
  899. public void DoWhile_AlwaysTrue_Infinite()
  900. {
  901. var scheduler = new TestScheduler();
  902. var xs = scheduler.CreateColdObservable(
  903. OnNext(50, 1)
  904. );
  905. var results = scheduler.Start(() => Observable.DoWhile(xs, () => true));
  906. results.Messages.AssertEqual(
  907. OnNext(250, 1)
  908. );
  909. xs.Subscriptions.AssertEqual(
  910. Subscribe(200, 1000)
  911. );
  912. }
  913. [Fact]
  914. public void DoWhile_SometimesTrue()
  915. {
  916. var scheduler = new TestScheduler();
  917. var xs = scheduler.CreateColdObservable(
  918. OnNext(50, 1),
  919. OnNext(100, 2),
  920. OnNext(150, 3),
  921. OnNext(200, 4),
  922. OnCompleted<int>(250)
  923. );
  924. int n = 0;
  925. var results = scheduler.Start(() => Observable.DoWhile(xs, () => ++n < 3));
  926. results.Messages.AssertEqual(
  927. OnNext(250, 1),
  928. OnNext(300, 2),
  929. OnNext(350, 3),
  930. OnNext(400, 4),
  931. OnNext(500, 1),
  932. OnNext(550, 2),
  933. OnNext(600, 3),
  934. OnNext(650, 4),
  935. OnNext(750, 1),
  936. OnNext(800, 2),
  937. OnNext(850, 3),
  938. OnNext(900, 4),
  939. OnCompleted<int>(950)
  940. );
  941. xs.Subscriptions.AssertEqual(
  942. Subscribe(200, 450),
  943. Subscribe(450, 700),
  944. Subscribe(700, 950)
  945. );
  946. }
  947. [Fact]
  948. public void DoWhile_SometimesThrows()
  949. {
  950. var scheduler = new TestScheduler();
  951. var xs = scheduler.CreateColdObservable(
  952. OnNext(50, 1),
  953. OnNext(100, 2),
  954. OnNext(150, 3),
  955. OnNext(200, 4),
  956. OnCompleted<int>(250)
  957. );
  958. int n = 0;
  959. var ex = new Exception();
  960. var results = scheduler.Start(() => Observable.DoWhile(xs, () => ++n < 3 ? true : Throw<bool>(ex)));
  961. results.Messages.AssertEqual(
  962. OnNext(250, 1),
  963. OnNext(300, 2),
  964. OnNext(350, 3),
  965. OnNext(400, 4),
  966. OnNext(500, 1),
  967. OnNext(550, 2),
  968. OnNext(600, 3),
  969. OnNext(650, 4),
  970. OnNext(750, 1),
  971. OnNext(800, 2),
  972. OnNext(850, 3),
  973. OnNext(900, 4),
  974. OnError<int>(950, ex)
  975. );
  976. xs.Subscriptions.AssertEqual(
  977. Subscribe(200, 450),
  978. Subscribe(450, 700),
  979. Subscribe(700, 950)
  980. );
  981. }
  982. #endregion
  983. #region + For +
  984. [Fact]
  985. public void For_ArgumentChecking()
  986. {
  987. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.For(DummyEnumerable<int>.Instance, default(Func<int, IObservable<int>>)));
  988. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.For(null, DummyFunc<int, IObservable<int>>.Instance));
  989. }
  990. [Fact]
  991. public void For_Basic()
  992. {
  993. var scheduler = new TestScheduler();
  994. var results = scheduler.Start(() => Observable.For(new[] { 1, 2, 3 }, x => scheduler.CreateColdObservable(
  995. OnNext<int>((ushort)(x * 100 + 10), x * 10 + 1),
  996. OnNext<int>((ushort)(x * 100 + 20), x * 10 + 2),
  997. OnNext<int>((ushort)(x * 100 + 30), x * 10 + 3),
  998. OnCompleted<int>((ushort)(x * 100 + 40)))));
  999. results.Messages.AssertEqual(
  1000. OnNext(310, 11),
  1001. OnNext(320, 12),
  1002. OnNext(330, 13),
  1003. OnNext(550, 21),
  1004. OnNext(560, 22),
  1005. OnNext(570, 23),
  1006. OnNext(890, 31),
  1007. OnNext(900, 32),
  1008. OnNext(910, 33),
  1009. OnCompleted<int>(920)
  1010. );
  1011. }
  1012. IEnumerable<int> For_Error_Core(Exception ex)
  1013. {
  1014. yield return 1;
  1015. yield return 2;
  1016. yield return 3;
  1017. throw ex;
  1018. }
  1019. [Fact]
  1020. public void For_Error_Iterator()
  1021. {
  1022. var scheduler = new TestScheduler();
  1023. var ex = new Exception();
  1024. var results = scheduler.Start(() => Observable.For(For_Error_Core(ex), x => scheduler.CreateColdObservable(
  1025. OnNext<int>((ushort)(x * 100 + 10), x * 10 + 1),
  1026. OnNext<int>((ushort)(x * 100 + 20), x * 10 + 2),
  1027. OnNext<int>((ushort)(x * 100 + 30), x * 10 + 3),
  1028. OnCompleted<int>((ushort)(x * 100 + 40)))));
  1029. results.Messages.AssertEqual(
  1030. OnNext(310, 11),
  1031. OnNext(320, 12),
  1032. OnNext(330, 13),
  1033. OnNext(550, 21),
  1034. OnNext(560, 22),
  1035. OnNext(570, 23),
  1036. OnNext(890, 31),
  1037. OnNext(900, 32),
  1038. OnNext(910, 33),
  1039. OnError<int>(920, ex)
  1040. );
  1041. }
  1042. [Fact]
  1043. public void For_Error_Source()
  1044. {
  1045. var scheduler = new TestScheduler();
  1046. var ex = new Exception();
  1047. var results = scheduler.Start(() => Observable.For(new[] { 1, 2, 3 }, x => Observable.Throw<int>(ex)));
  1048. results.Messages.AssertEqual(
  1049. OnError<int>(200, ex)
  1050. );
  1051. }
  1052. [Fact]
  1053. public void For_SelectorThrows()
  1054. {
  1055. var scheduler = new TestScheduler();
  1056. var ex = new Exception();
  1057. var results = scheduler.Start(() => Observable.For(new[] { 1, 2, 3 }, x => Throw<IObservable<int>>(ex)));
  1058. results.Messages.AssertEqual(
  1059. OnError<int>(200, ex)
  1060. );
  1061. }
  1062. #endregion
  1063. #region + If +
  1064. [Fact]
  1065. public void If_ArgumentChecking()
  1066. {
  1067. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If(null, DummyObservable<int>.Instance, DummyObservable<int>.Instance));
  1068. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If(DummyFunc<bool>.Instance, null, DummyObservable<int>.Instance));
  1069. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If(DummyFunc<bool>.Instance, DummyObservable<int>.Instance, default(IObservable<int>)));
  1070. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If(null, DummyObservable<int>.Instance, Scheduler.Default));
  1071. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If(DummyFunc<bool>.Instance, default(IObservable<int>), Scheduler.Default));
  1072. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If(DummyFunc<bool>.Instance, DummyObservable<int>.Instance, default(IScheduler)));
  1073. }
  1074. [Fact]
  1075. public void If_True()
  1076. {
  1077. var scheduler = new TestScheduler();
  1078. var xs = scheduler.CreateHotObservable(
  1079. OnNext(210, 1),
  1080. OnNext(250, 2),
  1081. OnCompleted<int>(300)
  1082. );
  1083. var ys = scheduler.CreateHotObservable(
  1084. OnNext(310, 3),
  1085. OnNext(350, 4),
  1086. OnCompleted<int>(400)
  1087. );
  1088. var results = scheduler.Start(() => Observable.If(() => true, xs, ys));
  1089. results.Messages.AssertEqual(
  1090. OnNext(210, 1),
  1091. OnNext(250, 2),
  1092. OnCompleted<int>(300)
  1093. );
  1094. xs.Subscriptions.AssertEqual(
  1095. Subscribe(200, 300)
  1096. );
  1097. ys.Subscriptions.AssertEqual(
  1098. );
  1099. }
  1100. [Fact]
  1101. public void If_False()
  1102. {
  1103. var scheduler = new TestScheduler();
  1104. var xs = scheduler.CreateHotObservable(
  1105. OnNext(210, 1),
  1106. OnNext(250, 2),
  1107. OnCompleted<int>(300)
  1108. );
  1109. var ys = scheduler.CreateHotObservable(
  1110. OnNext(310, 3),
  1111. OnNext(350, 4),
  1112. OnCompleted<int>(400)
  1113. );
  1114. var results = scheduler.Start(() => Observable.If(() => false, xs, ys));
  1115. results.Messages.AssertEqual(
  1116. OnNext(310, 3),
  1117. OnNext(350, 4),
  1118. OnCompleted<int>(400)
  1119. );
  1120. xs.Subscriptions.AssertEqual(
  1121. );
  1122. ys.Subscriptions.AssertEqual(
  1123. Subscribe(200, 400)
  1124. );
  1125. }
  1126. [Fact]
  1127. public void If_Throw()
  1128. {
  1129. var scheduler = new TestScheduler();
  1130. var xs = scheduler.CreateHotObservable(
  1131. OnNext(210, 1),
  1132. OnNext(250, 2),
  1133. OnCompleted<int>(300)
  1134. );
  1135. var ys = scheduler.CreateHotObservable(
  1136. OnNext(310, 3),
  1137. OnNext(350, 4),
  1138. OnCompleted<int>(400)
  1139. );
  1140. var ex = new Exception();
  1141. var results = scheduler.Start(() => Observable.If(() => Throw<bool>(ex), xs, ys));
  1142. results.Messages.AssertEqual(
  1143. OnError<int>(200, ex)
  1144. );
  1145. xs.Subscriptions.AssertEqual(
  1146. );
  1147. ys.Subscriptions.AssertEqual(
  1148. );
  1149. }
  1150. [Fact]
  1151. public void If_Dispose()
  1152. {
  1153. var scheduler = new TestScheduler();
  1154. var xs = scheduler.CreateHotObservable(
  1155. OnNext(210, 1),
  1156. OnNext(250, 2)
  1157. );
  1158. var ys = scheduler.CreateHotObservable(
  1159. OnNext(310, 3),
  1160. OnNext(350, 4),
  1161. OnCompleted<int>(400)
  1162. );
  1163. var results = scheduler.Start(() => Observable.If(() => true, xs, ys));
  1164. results.Messages.AssertEqual(
  1165. OnNext(210, 1),
  1166. OnNext(250, 2)
  1167. );
  1168. xs.Subscriptions.AssertEqual(
  1169. Subscribe(200, 1000)
  1170. );
  1171. ys.Subscriptions.AssertEqual(
  1172. );
  1173. }
  1174. [Fact]
  1175. public void If_Default_ArgumentChecking()
  1176. {
  1177. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If<int>(null, DummyObservable<int>.Instance));
  1178. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.If<int>(DummyFunc<bool>.Instance, null));
  1179. }
  1180. [Fact]
  1181. public void If_Default_Completed()
  1182. {
  1183. var scheduler = new TestScheduler();
  1184. var xs = scheduler.CreateHotObservable(
  1185. OnNext(110, 1),
  1186. OnNext(220, 2),
  1187. OnNext(330, 3),
  1188. OnCompleted<int>(440)
  1189. );
  1190. var b = false;
  1191. scheduler.ScheduleAbsolute(150, () => b = true);
  1192. var results = scheduler.Start(() => Observable.If(() => b, xs));
  1193. results.Messages.AssertEqual(
  1194. OnNext(220, 2),
  1195. OnNext(330, 3),
  1196. OnCompleted<int>(440)
  1197. );
  1198. xs.Subscriptions.AssertEqual(
  1199. Subscribe(200, 440)
  1200. );
  1201. }
  1202. [Fact]
  1203. public void If_Default_Error()
  1204. {
  1205. var scheduler = new TestScheduler();
  1206. var ex = new Exception();
  1207. var xs = scheduler.CreateHotObservable(
  1208. OnNext(110, 1),
  1209. OnNext(220, 2),
  1210. OnNext(330, 3),
  1211. OnError<int>(440, ex)
  1212. );
  1213. var b = false;
  1214. scheduler.ScheduleAbsolute(150, () => b = true);
  1215. var results = scheduler.Start(() => Observable.If(() => b, xs));
  1216. results.Messages.AssertEqual(
  1217. OnNext(220, 2),
  1218. OnNext(330, 3),
  1219. OnError<int>(440, ex)
  1220. );
  1221. xs.Subscriptions.AssertEqual(
  1222. Subscribe(200, 440)
  1223. );
  1224. }
  1225. [Fact]
  1226. public void If_Default_Never()
  1227. {
  1228. var scheduler = new TestScheduler();
  1229. var xs = scheduler.CreateHotObservable(
  1230. OnNext(110, 1),
  1231. OnNext(220, 2),
  1232. OnNext(330, 3)
  1233. );
  1234. var b = false;
  1235. scheduler.ScheduleAbsolute(150, () => b = true);
  1236. var results = scheduler.Start(() => Observable.If(() => b, xs));
  1237. results.Messages.AssertEqual(
  1238. OnNext(220, 2),
  1239. OnNext(330, 3)
  1240. );
  1241. xs.Subscriptions.AssertEqual(
  1242. Subscribe(200, 1000)
  1243. );
  1244. }
  1245. [Fact]
  1246. public void If_Default_Other()
  1247. {
  1248. var scheduler = new TestScheduler();
  1249. var xs = scheduler.CreateHotObservable(
  1250. OnNext(110, 1),
  1251. OnNext(220, 2),
  1252. OnNext(330, 3),
  1253. OnError<int>(440, new Exception())
  1254. );
  1255. var b = true;
  1256. scheduler.ScheduleAbsolute(150, () => b = false);
  1257. var results = scheduler.Start(() => Observable.If(() => b, xs));
  1258. results.Messages.AssertEqual(
  1259. OnCompleted<int>(200)
  1260. );
  1261. xs.Subscriptions.AssertEqual(
  1262. );
  1263. }
  1264. [Fact]
  1265. public void If_Default_Scheduler()
  1266. {
  1267. var scheduler = new TestScheduler();
  1268. var xs = scheduler.CreateHotObservable(
  1269. OnNext(110, 1),
  1270. OnNext(220, 2),
  1271. OnNext(330, 3),
  1272. OnError<int>(440, new Exception())
  1273. );
  1274. var results = scheduler.Start(() => Observable.If(() => false, xs, scheduler));
  1275. results.Messages.AssertEqual(
  1276. OnCompleted<int>(201)
  1277. );
  1278. xs.Subscriptions.AssertEqual(
  1279. );
  1280. }
  1281. #endregion
  1282. #region + While +
  1283. [Fact]
  1284. public void While_ArgumentChecking()
  1285. {
  1286. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.While(default(Func<bool>), DummyObservable<int>.Instance));
  1287. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.While(DummyFunc<bool>.Instance, default(IObservable<int>)));
  1288. }
  1289. [Fact]
  1290. public void While_AlwaysFalse()
  1291. {
  1292. var scheduler = new TestScheduler();
  1293. var xs = scheduler.CreateColdObservable(
  1294. OnNext(50, 1),
  1295. OnNext(100, 2),
  1296. OnNext(150, 3),
  1297. OnNext(200, 4),
  1298. OnCompleted<int>(250)
  1299. );
  1300. var results = scheduler.Start(() => Observable.While(() => false, xs));
  1301. results.Messages.AssertEqual(
  1302. OnCompleted<int>(200)
  1303. );
  1304. xs.Subscriptions.AssertEqual(
  1305. );
  1306. }
  1307. [Fact]
  1308. public void While_AlwaysTrue()
  1309. {
  1310. var scheduler = new TestScheduler();
  1311. var xs = scheduler.CreateColdObservable(
  1312. OnNext(50, 1),
  1313. OnNext(100, 2),
  1314. OnNext(150, 3),
  1315. OnNext(200, 4),
  1316. OnCompleted<int>(250)
  1317. );
  1318. var results = scheduler.Start(() => Observable.While(() => true, xs));
  1319. results.Messages.AssertEqual(
  1320. OnNext(250, 1),
  1321. OnNext(300, 2),
  1322. OnNext(350, 3),
  1323. OnNext(400, 4),
  1324. OnNext(500, 1),
  1325. OnNext(550, 2),
  1326. OnNext(600, 3),
  1327. OnNext(650, 4),
  1328. OnNext(750, 1),
  1329. OnNext(800, 2),
  1330. OnNext(850, 3),
  1331. OnNext(900, 4)
  1332. );
  1333. xs.Subscriptions.AssertEqual(
  1334. Subscribe(200, 450),
  1335. Subscribe(450, 700),
  1336. Subscribe(700, 950),
  1337. Subscribe(950, 1000)
  1338. );
  1339. }
  1340. [Fact]
  1341. public void While_AlwaysTrue_Throw()
  1342. {
  1343. var scheduler = new TestScheduler();
  1344. var ex = new Exception();
  1345. var xs = scheduler.CreateColdObservable(
  1346. OnError<int>(50, ex)
  1347. );
  1348. var results = scheduler.Start(() => Observable.While(() => true, xs));
  1349. results.Messages.AssertEqual(
  1350. OnError<int>(250, ex)
  1351. );
  1352. xs.Subscriptions.AssertEqual(
  1353. Subscribe(200, 250)
  1354. );
  1355. }
  1356. [Fact]
  1357. public void While_AlwaysTrue_Infinite()
  1358. {
  1359. var scheduler = new TestScheduler();
  1360. var xs = scheduler.CreateColdObservable(
  1361. OnNext(50, 1)
  1362. );
  1363. var results = scheduler.Start(() => Observable.While(() => true, xs));
  1364. results.Messages.AssertEqual(
  1365. OnNext(250, 1)
  1366. );
  1367. xs.Subscriptions.AssertEqual(
  1368. Subscribe(200, 1000)
  1369. );
  1370. }
  1371. [Fact]
  1372. public void While_SometimesTrue()
  1373. {
  1374. var scheduler = new TestScheduler();
  1375. var xs = scheduler.CreateColdObservable(
  1376. OnNext(50, 1),
  1377. OnNext(100, 2),
  1378. OnNext(150, 3),
  1379. OnNext(200, 4),
  1380. OnCompleted<int>(250)
  1381. );
  1382. int n = 0;
  1383. var results = scheduler.Start(() => Observable.While(() => ++n < 3, xs));
  1384. results.Messages.AssertEqual(
  1385. OnNext(250, 1),
  1386. OnNext(300, 2),
  1387. OnNext(350, 3),
  1388. OnNext(400, 4),
  1389. OnNext(500, 1),
  1390. OnNext(550, 2),
  1391. OnNext(600, 3),
  1392. OnNext(650, 4),
  1393. OnCompleted<int>(700)
  1394. );
  1395. xs.Subscriptions.AssertEqual(
  1396. Subscribe(200, 450),
  1397. Subscribe(450, 700)
  1398. );
  1399. }
  1400. static T Throw<T>(Exception ex)
  1401. {
  1402. throw ex;
  1403. }
  1404. [Fact]
  1405. public void While_SometimesThrows()
  1406. {
  1407. var scheduler = new TestScheduler();
  1408. var xs = scheduler.CreateColdObservable(
  1409. OnNext(50, 1),
  1410. OnNext(100, 2),
  1411. OnNext(150, 3),
  1412. OnNext(200, 4),
  1413. OnCompleted<int>(250)
  1414. );
  1415. int n = 0;
  1416. var ex = new Exception();
  1417. var results = scheduler.Start(() => Observable.While(() => ++n < 3 ? true : Throw<bool>(ex), xs));
  1418. results.Messages.AssertEqual(
  1419. OnNext(250, 1),
  1420. OnNext(300, 2),
  1421. OnNext(350, 3),
  1422. OnNext(400, 4),
  1423. OnNext(500, 1),
  1424. OnNext(550, 2),
  1425. OnNext(600, 3),
  1426. OnNext(650, 4),
  1427. OnError<int>(700, ex)
  1428. );
  1429. xs.Subscriptions.AssertEqual(
  1430. Subscribe(200, 450),
  1431. Subscribe(450, 700)
  1432. );
  1433. }
  1434. #endregion
  1435. #region General tests for loops
  1436. #if HAS_STACKTRACE
  1437. [Fact]
  1438. public void LoopTest1()
  1439. {
  1440. var loop = Observable.Defer(() =>
  1441. {
  1442. var n = 0;
  1443. return Observable.While(
  1444. () => n++ < 5,
  1445. Observable.Defer(() =>
  1446. {
  1447. return Observable.For(
  1448. Enumerable.Range(0, n),
  1449. x => Observable.Return(x)
  1450. );
  1451. })
  1452. );
  1453. });
  1454. var res = new List<int>();
  1455. var std = new List<int>();
  1456. loop.ForEach(x =>
  1457. {
  1458. res.Add(x);
  1459. std.Add(new System.Diagnostics.StackTrace().FrameCount);
  1460. });
  1461. Assert.True(res.SequenceEqual(new[] { 0, 0, 1, 0, 1, 2, 0, 1, 2, 3, 0, 1, 2, 3, 4 }));
  1462. Assert.True(std.Distinct().Count() == 1);
  1463. }
  1464. [Fact]
  1465. public void LoopTest2()
  1466. {
  1467. var n = 0;
  1468. var loop = default(IObservable<int>);
  1469. loop = Observable.While(
  1470. () => n++ < 10,
  1471. Observable.Concat(
  1472. Observable.Return(42),
  1473. Observable.Defer(() => loop)
  1474. )
  1475. );
  1476. var res = new List<int>();
  1477. var std = new List<int>();
  1478. loop.ForEach(x =>
  1479. {
  1480. res.Add(x);
  1481. std.Add(new System.Diagnostics.StackTrace().FrameCount);
  1482. });
  1483. Assert.True(res.SequenceEqual(Enumerable.Repeat(42, 10)));
  1484. Assert.True(std.Distinct().Count() == 1);
  1485. }
  1486. #endif
  1487. #endregion
  1488. }
  1489. }