ObservableConcurrencyTest.cs 34 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126
  1. // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Reactive;
  6. using System.Reactive.Concurrency;
  7. using System.Reactive.Disposables;
  8. using System.Reactive.Linq;
  9. using System.Reactive.Subjects;
  10. using System.Threading;
  11. #if NET45
  12. using System.Windows.Threading;
  13. #endif
  14. using Microsoft.Reactive.Testing;
  15. using Xunit;
  16. using ReactiveTests.Dummies;
  17. #if HAS_WINFORMS
  18. using System.Windows.Forms;
  19. #endif
  20. #if SILVERLIGHT && !SILVERLIGHTM7
  21. using Microsoft.Silverlight.Testing;
  22. #endif
  23. namespace ReactiveTests.Tests
  24. {
  25. public partial class ObservableConcurrencyTest : TestBase
  26. {
  27. #region + ObserveOn +
  28. [Fact]
  29. public void ObserveOn_ArgumentChecking()
  30. {
  31. var someObservable = Observable.Empty<int>();
  32. #if HAS_WINFORMS
  33. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(default(IObservable<int>), new ControlScheduler(new Label())));
  34. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(someObservable, default(ControlScheduler)));
  35. ReactiveAssert.Throws<ArgumentNullException>(() => ControlObservable.ObserveOn<int>(default(IObservable<int>), new Label()));
  36. ReactiveAssert.Throws<ArgumentNullException>(() => ControlObservable.ObserveOn<int>(someObservable, default(Label)));
  37. #endif
  38. #if HAS_DISPATCHER
  39. #if USE_SL_DISPATCHER
  40. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(default(IObservable<int>), new DispatcherScheduler(System.Windows.Deployment.Current.Dispatcher)));
  41. #else
  42. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(default(IObservable<int>), new DispatcherScheduler(Dispatcher.CurrentDispatcher)));
  43. #endif
  44. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(someObservable, default(DispatcherScheduler)));
  45. #if USE_SL_DISPATCHER
  46. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.ObserveOn<int>(default(IObservable<int>), System.Windows.Deployment.Current.Dispatcher));
  47. #else
  48. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.ObserveOn<int>(default(IObservable<int>), Dispatcher.CurrentDispatcher));
  49. #endif
  50. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.ObserveOn<int>(someObservable, default(Dispatcher)));
  51. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.ObserveOnDispatcher<int>(default(IObservable<int>)));
  52. #endif
  53. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(default(IObservable<int>), new SynchronizationContext()));
  54. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn<int>(someObservable, default(SynchronizationContext)));
  55. }
  56. #if HAS_WINFORMS
  57. [Fact]
  58. public void ObserveOn_Control()
  59. {
  60. var lbl = CreateLabel();
  61. var evt = new ManualResetEvent(false);
  62. bool okay = true;
  63. Observable.Range(0, 10, NewThreadScheduler.Default).ObserveOn(lbl).Subscribe(x =>
  64. {
  65. lbl.Text = x.ToString();
  66. okay &= (SynchronizationContext.Current is System.Windows.Forms.WindowsFormsSynchronizationContext);
  67. }, () => evt.Set());
  68. evt.WaitOne();
  69. Application.Exit();
  70. Assert.True(okay);
  71. }
  72. [Fact]
  73. public void ObserveOn_ControlScheduler()
  74. {
  75. var lbl = CreateLabel();
  76. var evt = new ManualResetEvent(false);
  77. bool okay = true;
  78. Observable.Range(0, 10, NewThreadScheduler.Default).ObserveOn(new ControlScheduler(lbl)).Subscribe(x =>
  79. {
  80. lbl.Text = x.ToString();
  81. okay &= (SynchronizationContext.Current is System.Windows.Forms.WindowsFormsSynchronizationContext);
  82. }, () => evt.Set());
  83. evt.WaitOne();
  84. Application.Exit();
  85. Assert.True(okay);
  86. }
  87. private Label CreateLabel()
  88. {
  89. var loaded = new ManualResetEvent(false);
  90. var lbl = default(Label);
  91. var t = new Thread(() =>
  92. {
  93. lbl = new Label();
  94. var frm = new Form { Controls = { lbl }, Width = 0, Height = 0, FormBorderStyle = FormBorderStyle.None, ShowInTaskbar = false };
  95. frm.Load += (_, __) =>
  96. {
  97. loaded.Set();
  98. };
  99. Application.Run(frm);
  100. });
  101. t.SetApartmentState(ApartmentState.STA);
  102. t.Start();
  103. loaded.WaitOne();
  104. return lbl;
  105. }
  106. #endif
  107. #if HAS_DISPATCHER
  108. [Fact]
  109. [Asynchronous]
  110. public void ObserveOn_Dispatcher()
  111. {
  112. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  113. RunAsync(evt =>
  114. {
  115. bool okay = true;
  116. Observable.Range(0, 10, NewThreadScheduler.Default).ObserveOn(dispatcher).Subscribe(x =>
  117. {
  118. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  119. }, () =>
  120. {
  121. Assert.True(okay);
  122. dispatcher.InvokeShutdown();
  123. evt.Set();
  124. });
  125. });
  126. }
  127. [Fact]
  128. [Asynchronous]
  129. public void ObserveOn_DispatcherScheduler()
  130. {
  131. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  132. RunAsync(evt =>
  133. {
  134. bool okay = true;
  135. Observable.Range(0, 10, NewThreadScheduler.Default).ObserveOn(new DispatcherScheduler(dispatcher)).Subscribe(x =>
  136. {
  137. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  138. }, () =>
  139. {
  140. Assert.True(okay);
  141. dispatcher.InvokeShutdown();
  142. evt.Set();
  143. });
  144. });
  145. }
  146. [Fact]
  147. [Asynchronous]
  148. public void ObserveOn_CurrentDispatcher()
  149. {
  150. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  151. RunAsync(evt =>
  152. {
  153. bool okay = true;
  154. dispatcher.BeginInvoke(new Action(() =>
  155. {
  156. Observable.Range(0, 10, NewThreadScheduler.Default).ObserveOnDispatcher().Subscribe(x =>
  157. {
  158. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  159. }, () =>
  160. {
  161. Assert.True(okay);
  162. dispatcher.InvokeShutdown();
  163. evt.Set();
  164. });
  165. }));
  166. });
  167. }
  168. [Fact]
  169. [Asynchronous]
  170. public void ObserveOn_Error()
  171. {
  172. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  173. RunAsync(evt =>
  174. {
  175. var ex = new Exception();
  176. bool okay = true;
  177. dispatcher.BeginInvoke(new Action(() =>
  178. {
  179. Observable.Throw<int>(ex).ObserveOnDispatcher().Subscribe(x =>
  180. {
  181. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  182. },
  183. e =>
  184. {
  185. Assert.True(okay);
  186. Assert.Same(ex, e);
  187. dispatcher.InvokeShutdown();
  188. evt.Set();
  189. },
  190. () =>
  191. {
  192. Assert.True(false);
  193. dispatcher.InvokeShutdown();
  194. evt.Set();
  195. });
  196. }));
  197. });
  198. }
  199. #endif
  200. #endregion
  201. #region SubscribeOn
  202. [Fact]
  203. public void SubscribeOn_ArgumentChecking()
  204. {
  205. var someObservable = Observable.Empty<int>();
  206. #if HAS_WINFORMS
  207. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(default(IObservable<int>), new ControlScheduler(new Label())));
  208. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(someObservable, default(ControlScheduler)));
  209. ReactiveAssert.Throws<ArgumentNullException>(() => ControlObservable.SubscribeOn<int>(default(IObservable<int>), new Label()));
  210. ReactiveAssert.Throws<ArgumentNullException>(() => ControlObservable.SubscribeOn<int>(someObservable, default(Label)));
  211. #endif
  212. #if HAS_DISPATCHER
  213. #if USE_SL_DISPATCHER
  214. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(default(IObservable<int>), new DispatcherScheduler(System.Windows.Deployment.Current.Dispatcher)));
  215. #else
  216. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(default(IObservable<int>), new DispatcherScheduler(Dispatcher.CurrentDispatcher)));
  217. #endif
  218. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(someObservable, default(DispatcherScheduler)));
  219. #if USE_SL_DISPATCHER
  220. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.SubscribeOn<int>(default(IObservable<int>), System.Windows.Deployment.Current.Dispatcher));
  221. #else
  222. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.SubscribeOn<int>(default(IObservable<int>), Dispatcher.CurrentDispatcher));
  223. #endif
  224. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.SubscribeOn<int>(someObservable, default(Dispatcher)));
  225. ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherObservable.SubscribeOnDispatcher<int>(default(IObservable<int>)));
  226. #endif
  227. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(default(IObservable<int>), new SynchronizationContext()));
  228. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn<int>(someObservable, default(SynchronizationContext)));
  229. }
  230. #if HAS_WINFORMS
  231. [Fact]
  232. public void SubscribeOn_Control()
  233. {
  234. var lbl = CreateLabel();
  235. var evt2 = new ManualResetEvent(false);
  236. var evt = new ManualResetEvent(false);
  237. bool okay = true;
  238. var d = Observable.Create<int>(obs =>
  239. {
  240. lbl.Text = "Subscribe";
  241. okay &= (SynchronizationContext.Current is System.Windows.Forms.WindowsFormsSynchronizationContext);
  242. evt2.Set();
  243. return () =>
  244. {
  245. lbl.Text = "Unsubscribe";
  246. okay &= (SynchronizationContext.Current is System.Windows.Forms.WindowsFormsSynchronizationContext);
  247. evt.Set();
  248. };
  249. })
  250. .SubscribeOn(lbl)
  251. .Subscribe(_ => {});
  252. evt2.WaitOne();
  253. d.Dispose();
  254. evt.WaitOne();
  255. Application.Exit();
  256. Assert.True(okay);
  257. }
  258. [Fact]
  259. public void SubscribeOn_ControlScheduler()
  260. {
  261. var lbl = CreateLabel();
  262. var evt2 = new ManualResetEvent(false);
  263. var evt = new ManualResetEvent(false);
  264. bool okay = true;
  265. var d = Observable.Create<int>(obs =>
  266. {
  267. lbl.Text = "Subscribe";
  268. okay &= (SynchronizationContext.Current is System.Windows.Forms.WindowsFormsSynchronizationContext);
  269. evt2.Set();
  270. return () =>
  271. {
  272. lbl.Text = "Unsubscribe";
  273. okay &= (SynchronizationContext.Current is System.Windows.Forms.WindowsFormsSynchronizationContext);
  274. evt.Set();
  275. };
  276. })
  277. .SubscribeOn(new ControlScheduler(lbl))
  278. .Subscribe(_ => { });
  279. evt2.WaitOne();
  280. d.Dispose();
  281. evt.WaitOne();
  282. Application.Exit();
  283. Assert.True(okay);
  284. }
  285. #endif
  286. #if HAS_DISPATCHER
  287. [Fact]
  288. [Asynchronous]
  289. public void SubscribeOn_Dispatcher()
  290. {
  291. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  292. RunAsync(evt =>
  293. {
  294. var s = new AsyncSubject<Unit>();
  295. bool okay = true;
  296. var d = Observable.Create<int>(obs =>
  297. {
  298. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  299. s.OnNext(Unit.Default);
  300. s.OnCompleted();
  301. return () =>
  302. {
  303. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  304. Assert.True(okay);
  305. dispatcher.InvokeShutdown();
  306. evt.Set();
  307. };
  308. })
  309. .SubscribeOn(dispatcher)
  310. .Subscribe(_ => { });
  311. s.Subscribe(_ => d.Dispose());
  312. });
  313. }
  314. [Fact]
  315. [Asynchronous]
  316. public void SubscribeOn_DispatcherScheduler()
  317. {
  318. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  319. RunAsync(evt =>
  320. {
  321. var s = new AsyncSubject<Unit>();
  322. bool okay = true;
  323. var d = Observable.Create<int>(obs =>
  324. {
  325. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  326. s.OnNext(Unit.Default);
  327. s.OnCompleted();
  328. return () =>
  329. {
  330. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  331. Assert.True(okay);
  332. dispatcher.InvokeShutdown();
  333. evt.Set();
  334. };
  335. })
  336. .SubscribeOn(new DispatcherScheduler(dispatcher))
  337. .Subscribe(_ => { });
  338. s.Subscribe(_ => d.Dispose());
  339. });
  340. }
  341. [Fact]
  342. [Asynchronous]
  343. public void SubscribeOn_CurrentDispatcher()
  344. {
  345. var dispatcher = DispatcherHelpers.EnsureDispatcher();
  346. RunAsync(evt =>
  347. {
  348. var s = new AsyncSubject<Unit>();
  349. bool okay = true;
  350. dispatcher.BeginInvoke(new Action(() =>
  351. {
  352. var d = Observable.Create<int>(obs =>
  353. {
  354. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  355. s.OnNext(Unit.Default);
  356. s.OnCompleted();
  357. return () =>
  358. {
  359. okay &= (SynchronizationContext.Current is System.Windows.Threading.DispatcherSynchronizationContext);
  360. Assert.True(okay);
  361. dispatcher.InvokeShutdown();
  362. evt.Set();
  363. };
  364. })
  365. .SubscribeOnDispatcher()
  366. .Subscribe(_ => { });
  367. s.Subscribe(_ => d.Dispose());
  368. }));
  369. });
  370. }
  371. #endif
  372. #endregion
  373. #region + Synchronize +
  374. [Fact]
  375. public void Synchronize_ArgumentChecking()
  376. {
  377. var someObservable = Observable.Empty<int>();
  378. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Synchronize<int>(default(IObservable<int>)));
  379. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Synchronize<int>(default(IObservable<int>), new object()));
  380. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.Synchronize<int>(someObservable, null));
  381. }
  382. [Fact]
  383. public void Synchronize_Range()
  384. {
  385. int i = 0;
  386. bool outsideLock = true;
  387. var gate = new object();
  388. lock (gate)
  389. {
  390. outsideLock = false;
  391. Observable.Range(0, 100, NewThreadScheduler.Default).Synchronize(gate).Subscribe(x => i++, () => { Assert.True(outsideLock); });
  392. Thread.Sleep(100);
  393. Assert.Equal(0, i);
  394. outsideLock = true;
  395. }
  396. while (i < 100)
  397. {
  398. Thread.Sleep(10);
  399. lock (gate)
  400. {
  401. int start = i;
  402. Thread.Sleep(100);
  403. Assert.Equal(start, i);
  404. }
  405. }
  406. }
  407. [Fact]
  408. public void Synchronize_Throw()
  409. {
  410. var ex = new Exception();
  411. var resLock = new object();
  412. var e = default(Exception);
  413. bool outsideLock = true;
  414. var gate = new object();
  415. lock (gate)
  416. {
  417. outsideLock = false;
  418. Observable.Throw<int>(ex, NewThreadScheduler.Default).Synchronize(gate).Subscribe(x => { Assert.True(false); }, err => { lock (resLock) { e = err; } }, () => { Assert.True(outsideLock); });
  419. Thread.Sleep(100);
  420. Assert.Null(e);
  421. outsideLock = true;
  422. }
  423. while (true)
  424. {
  425. lock (resLock)
  426. {
  427. if (e != null)
  428. break;
  429. }
  430. }
  431. Assert.Same(ex, e);
  432. }
  433. [Fact]
  434. public void Synchronize_BadObservable()
  435. {
  436. var o = Observable.Create<int>(obs =>
  437. {
  438. var t1 = new Thread(() =>
  439. {
  440. for (int i = 0; i < 100; i++)
  441. {
  442. obs.OnNext(i);
  443. }
  444. });
  445. new Thread(() =>
  446. {
  447. t1.Start();
  448. for (int i = 100; i < 200; i++)
  449. {
  450. obs.OnNext(i);
  451. }
  452. t1.Join();
  453. obs.OnCompleted();
  454. }).Start();
  455. return () => { };
  456. });
  457. var evt = new ManualResetEvent(false);
  458. int sum = 0;
  459. o.Synchronize().Subscribe(x => sum += x, () => { evt.Set(); });
  460. evt.WaitOne();
  461. Assert.Equal(Enumerable.Range(0, 200).Sum(), sum);
  462. }
  463. #endregion
  464. }
  465. public class ObservableConcurrencyReactiveTest : ReactiveTest
  466. {
  467. #region + ObserveOn +
  468. [Fact]
  469. public void ObserveOn_Scheduler_ArgumentChecking()
  470. {
  471. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn(default(IObservable<int>), DummyScheduler.Instance));
  472. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn(DummyObservable<int>.Instance, default(IScheduler)));
  473. }
  474. [Fact]
  475. public void ObserveOn_Scheduler_Completed()
  476. {
  477. var scheduler = new TestScheduler();
  478. var xs = scheduler.CreateHotObservable(
  479. OnNext( 90, 1),
  480. OnNext(120, 2),
  481. OnNext(230, 3),
  482. OnNext(240, 4),
  483. OnNext(310, 5),
  484. OnNext(470, 6),
  485. OnCompleted<int>(530)
  486. );
  487. var results = scheduler.Start(() =>
  488. xs.ObserveOn(scheduler)
  489. );
  490. results.Messages.AssertEqual(
  491. OnNext(231, 3),
  492. OnNext(241, 4),
  493. OnNext(311, 5),
  494. OnNext(471, 6),
  495. OnCompleted<int>(531)
  496. );
  497. #if !NO_PERF && !NO_CDS
  498. // BREAKING CHANGE v2 > v1.x -> More aggressive disposal behavior
  499. xs.Subscriptions.AssertEqual(
  500. Subscribe(200, 531)
  501. );
  502. #else
  503. //
  504. // TODO: Check platform discrepancies
  505. //
  506. //xs.Subscriptions.AssertEqual(
  507. // Subscribe(200, 1000)
  508. //);
  509. #endif
  510. }
  511. [Fact]
  512. public void ObserveOn_Scheduler_Error()
  513. {
  514. var scheduler = new TestScheduler();
  515. var ex = new Exception();
  516. var xs = scheduler.CreateHotObservable(
  517. OnNext(90, 1),
  518. OnNext(120, 2),
  519. OnNext(230, 3),
  520. OnNext(240, 4),
  521. OnNext(310, 5),
  522. OnNext(470, 6),
  523. OnError<int>(530, ex)
  524. );
  525. var results = scheduler.Start(() =>
  526. xs.ObserveOn(scheduler)
  527. );
  528. results.Messages.AssertEqual(
  529. OnNext(231, 3),
  530. OnNext(241, 4),
  531. OnNext(311, 5),
  532. OnNext(471, 6),
  533. OnError<int>(531, ex)
  534. );
  535. #if !NO_PERF && !NO_CDS
  536. // BREAKING CHANGE v2 > v1.x -> More aggressive disposal behavior
  537. xs.Subscriptions.AssertEqual(
  538. Subscribe(200, 531)
  539. );
  540. #else
  541. //
  542. // TODO: Check platform discrepancies
  543. //
  544. //xs.Subscriptions.AssertEqual(
  545. // Subscribe(200, 1000)
  546. //);
  547. #endif
  548. }
  549. [Fact]
  550. public void ObserveOn_Scheduler_Dispose()
  551. {
  552. var scheduler = new TestScheduler();
  553. var xs = scheduler.CreateHotObservable(
  554. OnNext(90, 1),
  555. OnNext(120, 2),
  556. OnNext(230, 3),
  557. OnNext(240, 4),
  558. OnNext(310, 5),
  559. OnNext(470, 6)
  560. );
  561. var results = scheduler.Start(() =>
  562. xs.ObserveOn(scheduler)
  563. );
  564. results.Messages.AssertEqual(
  565. OnNext(231, 3),
  566. OnNext(241, 4),
  567. OnNext(311, 5),
  568. OnNext(471, 6)
  569. );
  570. xs.Subscriptions.AssertEqual(
  571. Subscribe(200, 1000)
  572. );
  573. }
  574. [Fact]
  575. public void ObserveOn_Scheduler_SameTime()
  576. {
  577. var scheduler = new TestScheduler();
  578. var xs = scheduler.CreateHotObservable(
  579. OnNext(210, 1),
  580. OnNext(210, 2)
  581. );
  582. var results = scheduler.Start(() =>
  583. xs.ObserveOn(scheduler)
  584. );
  585. results.Messages.AssertEqual(
  586. OnNext(211, 1),
  587. OnNext(212, 2)
  588. );
  589. xs.Subscriptions.AssertEqual(
  590. Subscribe(200, 1000)
  591. );
  592. }
  593. [Fact]
  594. public void ObserveOn_Scheduler_OnNextThrows()
  595. {
  596. var e = new ManualResetEvent(false);
  597. var scheduler = new MyScheduler(e);
  598. Observable.Range(0, 10, Scheduler.Default).ObserveOn(scheduler).Subscribe(
  599. x =>
  600. {
  601. if (x == 5)
  602. throw new Exception();
  603. }
  604. );
  605. e.WaitOne();
  606. Assert.NotNull(scheduler._exception);
  607. }
  608. class MyScheduler : IScheduler
  609. {
  610. internal Exception _exception;
  611. private ManualResetEvent _evt;
  612. public MyScheduler(ManualResetEvent e)
  613. {
  614. _evt = e;
  615. }
  616. public DateTimeOffset Now
  617. {
  618. get { throw new NotImplementedException(); }
  619. }
  620. public IDisposable Schedule<TState>(TState state, Func<IScheduler, TState, IDisposable> action)
  621. {
  622. try
  623. {
  624. return action(this, state);
  625. }
  626. catch (Exception ex)
  627. {
  628. _exception = ex;
  629. _evt.Set();
  630. return Disposable.Empty;
  631. }
  632. }
  633. public IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)
  634. {
  635. throw new NotImplementedException();
  636. }
  637. public IDisposable Schedule<TState>(TState state, DateTimeOffset dueTime, Func<IScheduler, TState, IDisposable> action)
  638. {
  639. throw new NotImplementedException();
  640. }
  641. }
  642. #if !NO_PERF && !NO_CDS
  643. [Fact]
  644. public void ObserveOn_LongRunning_Simple()
  645. {
  646. var started = default(ManualResetEvent);
  647. var stopped = default(ManualResetEvent);
  648. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  649. var s = new Subject<int>();
  650. var end = new ManualResetEvent(false);
  651. var lst = new List<int>();
  652. s.ObserveOn(scheduler).Subscribe(lst.Add, () => end.Set());
  653. s.OnNext(1);
  654. s.OnNext(2);
  655. s.OnNext(3);
  656. s.OnCompleted();
  657. end.WaitOne();
  658. Assert.True(lst.SequenceEqual(new[] { 1, 2, 3 }));
  659. }
  660. [Fact]
  661. public void ObserveOn_LongRunning_Error()
  662. {
  663. var started = default(ManualResetEvent);
  664. var stopped = default(ManualResetEvent);
  665. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  666. var s = new Subject<int>();
  667. var end = new ManualResetEvent(false);
  668. var err = default(Exception);
  669. s.ObserveOn(scheduler).Subscribe(_ => { }, ex => { err = ex; end.Set(); });
  670. s.OnNext(1);
  671. s.OnNext(2);
  672. s.OnNext(3);
  673. var ex_ = new Exception();
  674. s.OnError(ex_);
  675. end.WaitOne();
  676. Assert.Same(ex_, err);
  677. }
  678. [Fact]
  679. public void ObserveOn_LongRunning_TimeVariance()
  680. {
  681. var started = default(ManualResetEvent);
  682. var stopped = default(ManualResetEvent);
  683. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  684. var s = new Subject<int>();
  685. var end = new ManualResetEvent(false);
  686. s.ObserveOn(scheduler).Subscribe(_ => { }, () => end.Set());
  687. s.OnNext(1); // Ensure active
  688. started.WaitOne();
  689. Thread.Sleep(100); // Try to enter the dispatcher event wait state
  690. for (int i = 0; i < 1000; i++)
  691. {
  692. if (i % 100 == 0)
  693. Thread.Sleep(10);
  694. s.OnNext(i);
  695. }
  696. s.OnCompleted();
  697. end.WaitOne();
  698. }
  699. [Fact]
  700. public void ObserveOn_LongRunning_HoldUpDuringDispatchAndFail()
  701. {
  702. var started = default(ManualResetEvent);
  703. var stopped = default(ManualResetEvent);
  704. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  705. var s = new Subject<int>();
  706. var onNext = new ManualResetEvent(false);
  707. var resume = new ManualResetEvent(false);
  708. var lst = new List<int>();
  709. var err = default(Exception);
  710. var end = new ManualResetEvent(false);
  711. s.ObserveOn(scheduler).Subscribe(x => { lst.Add(x); onNext.Set(); resume.WaitOne(); }, ex_ => { err = ex_; end.Set(); });
  712. s.OnNext(1);
  713. onNext.WaitOne();
  714. s.OnNext(2);
  715. s.OnNext(3);
  716. var ex = new Exception();
  717. s.OnError(ex);
  718. resume.Set();
  719. end.WaitOne();
  720. Assert.True(lst.SequenceEqual(new[] { 1, 2, 3 }));
  721. Assert.Same(ex, err);
  722. }
  723. [Fact]
  724. public void ObserveOn_LongRunning_Cancel()
  725. {
  726. var started = default(ManualResetEvent);
  727. var stopped = default(ManualResetEvent);
  728. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  729. var s = new Subject<int>();
  730. var lst = new List<int>();
  731. var end = new ManualResetEvent(false);
  732. var running = new ManualResetEvent(false);
  733. var d = s.ObserveOn(scheduler).Subscribe(x => { lst.Add(x); running.Set(); });
  734. s.OnNext(0);
  735. started.WaitOne();
  736. s.OnNext(1);
  737. s.OnNext(2);
  738. s.OnNext(3);
  739. running.WaitOne();
  740. d.Dispose();
  741. stopped.WaitOne();
  742. s.OnNext(4);
  743. Assert.True(lst.Count > 0 && !lst.Contains(4));
  744. }
  745. [Fact]
  746. public void ObserveOn_LongRunning_OnNextThrows()
  747. {
  748. var started = default(ManualResetEvent);
  749. var stopped = default(ManualResetEvent);
  750. var exception = default(Exception);
  751. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e, ex => exception = ex);
  752. var s = new Subject<int>();
  753. var lst = new List<int>();
  754. var end = new ManualResetEvent(false);
  755. var running = new ManualResetEvent(false);
  756. var d = s.ObserveOn(scheduler).Subscribe(x => { lst.Add(x); running.Set(); if (x == 3) throw new Exception(); });
  757. s.OnNext(0);
  758. started.WaitOne();
  759. s.OnNext(1);
  760. s.OnNext(2);
  761. s.OnNext(3);
  762. running.WaitOne();
  763. s.OnNext(4);
  764. stopped.WaitOne();
  765. Assert.NotNull(exception);
  766. }
  767. #endif
  768. #if !NO_SYNCCTX
  769. [Fact]
  770. public void ObserveOn_SynchronizationContext_Simple()
  771. {
  772. var scheduler = new TestScheduler();
  773. var xs = scheduler.CreateHotObservable(
  774. OnNext(90, 1),
  775. OnNext(120, 2),
  776. OnNext(230, 3),
  777. OnNext(240, 4),
  778. OnNext(310, 5),
  779. OnNext(470, 6),
  780. OnCompleted<int>(530)
  781. );
  782. var results = scheduler.Start(() =>
  783. xs.ObserveOn(new MyCtx(scheduler))
  784. );
  785. results.Messages.AssertEqual(
  786. OnNext(231, 3),
  787. OnNext(241, 4),
  788. OnNext(311, 5),
  789. OnNext(471, 6),
  790. OnCompleted<int>(531)
  791. );
  792. xs.Subscriptions.AssertEqual(
  793. Subscribe(200, 531)
  794. );
  795. }
  796. #endif
  797. #endregion
  798. #region SubscribeOn
  799. [Fact]
  800. public void SubscribeOn_Scheduler_ArgumentChecking()
  801. {
  802. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn(default(IObservable<int>), DummyScheduler.Instance));
  803. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn(DummyObservable<int>.Instance, default(IScheduler)));
  804. }
  805. [Fact]
  806. public void SubscribeOn_Scheduler_Sleep()
  807. {
  808. var scheduler = new TestScheduler();
  809. var s = 0L;
  810. var d = 0L;
  811. var xs = Observable.Create<long>(observer =>
  812. {
  813. s = scheduler.Clock;
  814. return () => d = scheduler.Clock;
  815. });
  816. var results = scheduler.Start(() =>
  817. xs.SubscribeOn(scheduler)
  818. );
  819. results.Messages.AssertEqual(
  820. );
  821. Assert.Equal(201, s);
  822. Assert.Equal(1001, d);
  823. }
  824. [Fact]
  825. public void SubscribeOn_Scheduler_Completed()
  826. {
  827. var scheduler = new TestScheduler();
  828. var xs = scheduler.CreateHotObservable(
  829. OnCompleted<long>(300)
  830. );
  831. var results = scheduler.Start(() =>
  832. xs.SubscribeOn(scheduler)
  833. );
  834. results.Messages.AssertEqual(
  835. OnCompleted<long>(300)
  836. );
  837. xs.Subscriptions.AssertEqual(
  838. Subscribe(201, 301)
  839. );
  840. }
  841. [Fact]
  842. public void SubscribeOn_Scheduler_Error()
  843. {
  844. var scheduler = new TestScheduler();
  845. var ex = new Exception();
  846. var xs = scheduler.CreateHotObservable(
  847. OnError<int>(300, ex)
  848. );
  849. var results = scheduler.Start(() =>
  850. xs.SubscribeOn(scheduler)
  851. );
  852. results.Messages.AssertEqual(
  853. OnError<int>(300, ex)
  854. );
  855. xs.Subscriptions.AssertEqual(
  856. Subscribe(201, 301)
  857. );
  858. }
  859. [Fact]
  860. public void SubscribeOn_Scheduler_Dispose()
  861. {
  862. var scheduler = new TestScheduler();
  863. var xs = scheduler.CreateHotObservable<int>(
  864. );
  865. var results = scheduler.Start(() =>
  866. xs.SubscribeOn(scheduler)
  867. );
  868. results.Messages.AssertEqual(
  869. );
  870. xs.Subscriptions.AssertEqual(
  871. Subscribe(201, 1001)
  872. );
  873. }
  874. #if !NO_SYNCCTX
  875. [Fact]
  876. public void SubscribeOn_SynchronizationContext_Simple()
  877. {
  878. var scheduler = new TestScheduler();
  879. var xs = scheduler.CreateHotObservable(
  880. OnNext(90, 1),
  881. OnNext(120, 2),
  882. OnNext(230, 3),
  883. OnNext(240, 4),
  884. OnNext(310, 5),
  885. OnNext(470, 6),
  886. OnCompleted<int>(530)
  887. );
  888. var results = scheduler.Start(() =>
  889. xs.SubscribeOn(new MyCtx(scheduler))
  890. );
  891. results.Messages.AssertEqual(
  892. OnNext(230, 3),
  893. OnNext(240, 4),
  894. OnNext(310, 5),
  895. OnNext(470, 6),
  896. OnCompleted<int>(530)
  897. );
  898. xs.Subscriptions.AssertEqual(
  899. Subscribe(201, 531)
  900. );
  901. }
  902. #endif
  903. #endregion
  904. #region |> Helpers <|
  905. #if !NO_SYNCCTX
  906. class MyCtx : SynchronizationContext
  907. {
  908. private IScheduler scheduler;
  909. public MyCtx(IScheduler scheduler)
  910. {
  911. this.scheduler = scheduler;
  912. }
  913. public override void Post(SendOrPostCallback d, object state)
  914. {
  915. scheduler.Schedule(state, (self, s) => { d(s); return Disposable.Empty; });
  916. }
  917. }
  918. #endif
  919. #endregion
  920. }
  921. }