ObservableConcurrencyTest.cs 34 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129
  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 || DNX451
  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. #if !NO_THREAD
  383. [Fact]
  384. public void Synchronize_Range()
  385. {
  386. int i = 0;
  387. bool outsideLock = true;
  388. var gate = new object();
  389. lock (gate)
  390. {
  391. outsideLock = false;
  392. Observable.Range(0, 100, NewThreadScheduler.Default).Synchronize(gate).Subscribe(x => i++, () => { Assert.True(outsideLock); });
  393. Thread.Sleep(100);
  394. Assert.Equal(0, i);
  395. outsideLock = true;
  396. }
  397. while (i < 100)
  398. {
  399. Thread.Sleep(10);
  400. lock (gate)
  401. {
  402. int start = i;
  403. Thread.Sleep(100);
  404. Assert.Equal(start, i);
  405. }
  406. }
  407. }
  408. [Fact]
  409. public void Synchronize_Throw()
  410. {
  411. var ex = new Exception();
  412. var resLock = new object();
  413. var e = default(Exception);
  414. bool outsideLock = true;
  415. var gate = new object();
  416. lock (gate)
  417. {
  418. outsideLock = false;
  419. Observable.Throw<int>(ex, NewThreadScheduler.Default).Synchronize(gate).Subscribe(x => { Assert.True(false); }, err => { lock (resLock) { e = err; } }, () => { Assert.True(outsideLock); });
  420. Thread.Sleep(100);
  421. Assert.Null(e);
  422. outsideLock = true;
  423. }
  424. while (true)
  425. {
  426. lock (resLock)
  427. {
  428. if (e != null)
  429. break;
  430. }
  431. }
  432. Assert.Same(ex, e);
  433. }
  434. [Fact]
  435. public void Synchronize_BadObservable()
  436. {
  437. var o = Observable.Create<int>(obs =>
  438. {
  439. var t1 = new Thread(() =>
  440. {
  441. for (int i = 0; i < 100; i++)
  442. {
  443. obs.OnNext(i);
  444. }
  445. });
  446. new Thread(() =>
  447. {
  448. t1.Start();
  449. for (int i = 100; i < 200; i++)
  450. {
  451. obs.OnNext(i);
  452. }
  453. t1.Join();
  454. obs.OnCompleted();
  455. }).Start();
  456. return () => { };
  457. });
  458. var evt = new ManualResetEvent(false);
  459. int sum = 0;
  460. o.Synchronize().Subscribe(x => sum += x, () => { evt.Set(); });
  461. evt.WaitOne();
  462. Assert.Equal(Enumerable.Range(0, 200).Sum(), sum);
  463. }
  464. #endif
  465. #endregion
  466. }
  467. public class ObservableConcurrencyReactiveTest : ReactiveTest
  468. {
  469. #region + ObserveOn +
  470. [Fact]
  471. public void ObserveOn_Scheduler_ArgumentChecking()
  472. {
  473. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn(default(IObservable<int>), DummyScheduler.Instance));
  474. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.ObserveOn(DummyObservable<int>.Instance, default(IScheduler)));
  475. }
  476. [Fact]
  477. public void ObserveOn_Scheduler_Completed()
  478. {
  479. var scheduler = new TestScheduler();
  480. var xs = scheduler.CreateHotObservable(
  481. OnNext( 90, 1),
  482. OnNext(120, 2),
  483. OnNext(230, 3),
  484. OnNext(240, 4),
  485. OnNext(310, 5),
  486. OnNext(470, 6),
  487. OnCompleted<int>(530)
  488. );
  489. var results = scheduler.Start(() =>
  490. xs.ObserveOn(scheduler)
  491. );
  492. results.Messages.AssertEqual(
  493. OnNext(231, 3),
  494. OnNext(241, 4),
  495. OnNext(311, 5),
  496. OnNext(471, 6),
  497. OnCompleted<int>(531)
  498. );
  499. #if !NO_PERF && !NO_CDS
  500. // BREAKING CHANGE v2 > v1.x -> More aggressive disposal behavior
  501. xs.Subscriptions.AssertEqual(
  502. Subscribe(200, 531)
  503. );
  504. #else
  505. //
  506. // TODO: Check platform discrepancies
  507. //
  508. //xs.Subscriptions.AssertEqual(
  509. // Subscribe(200, 1000)
  510. //);
  511. #endif
  512. }
  513. [Fact]
  514. public void ObserveOn_Scheduler_Error()
  515. {
  516. var scheduler = new TestScheduler();
  517. var ex = new Exception();
  518. var xs = scheduler.CreateHotObservable(
  519. OnNext(90, 1),
  520. OnNext(120, 2),
  521. OnNext(230, 3),
  522. OnNext(240, 4),
  523. OnNext(310, 5),
  524. OnNext(470, 6),
  525. OnError<int>(530, ex)
  526. );
  527. var results = scheduler.Start(() =>
  528. xs.ObserveOn(scheduler)
  529. );
  530. results.Messages.AssertEqual(
  531. OnNext(231, 3),
  532. OnNext(241, 4),
  533. OnNext(311, 5),
  534. OnNext(471, 6),
  535. OnError<int>(531, ex)
  536. );
  537. #if !NO_PERF && !NO_CDS
  538. // BREAKING CHANGE v2 > v1.x -> More aggressive disposal behavior
  539. xs.Subscriptions.AssertEqual(
  540. Subscribe(200, 531)
  541. );
  542. #else
  543. //
  544. // TODO: Check platform discrepancies
  545. //
  546. //xs.Subscriptions.AssertEqual(
  547. // Subscribe(200, 1000)
  548. //);
  549. #endif
  550. }
  551. [Fact]
  552. public void ObserveOn_Scheduler_Dispose()
  553. {
  554. var scheduler = new TestScheduler();
  555. var xs = scheduler.CreateHotObservable(
  556. OnNext(90, 1),
  557. OnNext(120, 2),
  558. OnNext(230, 3),
  559. OnNext(240, 4),
  560. OnNext(310, 5),
  561. OnNext(470, 6)
  562. );
  563. var results = scheduler.Start(() =>
  564. xs.ObserveOn(scheduler)
  565. );
  566. results.Messages.AssertEqual(
  567. OnNext(231, 3),
  568. OnNext(241, 4),
  569. OnNext(311, 5),
  570. OnNext(471, 6)
  571. );
  572. xs.Subscriptions.AssertEqual(
  573. Subscribe(200, 1000)
  574. );
  575. }
  576. [Fact]
  577. public void ObserveOn_Scheduler_SameTime()
  578. {
  579. var scheduler = new TestScheduler();
  580. var xs = scheduler.CreateHotObservable(
  581. OnNext(210, 1),
  582. OnNext(210, 2)
  583. );
  584. var results = scheduler.Start(() =>
  585. xs.ObserveOn(scheduler)
  586. );
  587. results.Messages.AssertEqual(
  588. OnNext(211, 1),
  589. OnNext(212, 2)
  590. );
  591. xs.Subscriptions.AssertEqual(
  592. Subscribe(200, 1000)
  593. );
  594. }
  595. [Fact]
  596. public void ObserveOn_Scheduler_OnNextThrows()
  597. {
  598. var e = new ManualResetEvent(false);
  599. var scheduler = new MyScheduler(e);
  600. Observable.Range(0, 10, Scheduler.Default).ObserveOn(scheduler).Subscribe(
  601. x =>
  602. {
  603. if (x == 5)
  604. throw new Exception();
  605. }
  606. );
  607. e.WaitOne();
  608. Assert.NotNull(scheduler._exception);
  609. }
  610. class MyScheduler : IScheduler
  611. {
  612. internal Exception _exception;
  613. private ManualResetEvent _evt;
  614. public MyScheduler(ManualResetEvent e)
  615. {
  616. _evt = e;
  617. }
  618. public DateTimeOffset Now
  619. {
  620. get { throw new NotImplementedException(); }
  621. }
  622. public IDisposable Schedule<TState>(TState state, Func<IScheduler, TState, IDisposable> action)
  623. {
  624. try
  625. {
  626. return action(this, state);
  627. }
  628. catch (Exception ex)
  629. {
  630. _exception = ex;
  631. _evt.Set();
  632. return Disposable.Empty;
  633. }
  634. }
  635. public IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)
  636. {
  637. throw new NotImplementedException();
  638. }
  639. public IDisposable Schedule<TState>(TState state, DateTimeOffset dueTime, Func<IScheduler, TState, IDisposable> action)
  640. {
  641. throw new NotImplementedException();
  642. }
  643. }
  644. #if !NO_PERF && !NO_CDS
  645. [Fact]
  646. public void ObserveOn_LongRunning_Simple()
  647. {
  648. var started = default(ManualResetEvent);
  649. var stopped = default(ManualResetEvent);
  650. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  651. var s = new Subject<int>();
  652. var end = new ManualResetEvent(false);
  653. var lst = new List<int>();
  654. s.ObserveOn(scheduler).Subscribe(lst.Add, () => end.Set());
  655. s.OnNext(1);
  656. s.OnNext(2);
  657. s.OnNext(3);
  658. s.OnCompleted();
  659. end.WaitOne();
  660. Assert.True(lst.SequenceEqual(new[] { 1, 2, 3 }));
  661. }
  662. [Fact]
  663. public void ObserveOn_LongRunning_Error()
  664. {
  665. var started = default(ManualResetEvent);
  666. var stopped = default(ManualResetEvent);
  667. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  668. var s = new Subject<int>();
  669. var end = new ManualResetEvent(false);
  670. var err = default(Exception);
  671. s.ObserveOn(scheduler).Subscribe(_ => { }, ex => { err = ex; end.Set(); });
  672. s.OnNext(1);
  673. s.OnNext(2);
  674. s.OnNext(3);
  675. var ex_ = new Exception();
  676. s.OnError(ex_);
  677. end.WaitOne();
  678. Assert.Same(ex_, err);
  679. }
  680. #if !NO_THREAD
  681. [Fact]
  682. public void ObserveOn_LongRunning_TimeVariance()
  683. {
  684. var started = default(ManualResetEvent);
  685. var stopped = default(ManualResetEvent);
  686. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  687. var s = new Subject<int>();
  688. var end = new ManualResetEvent(false);
  689. s.ObserveOn(scheduler).Subscribe(_ => { }, () => end.Set());
  690. s.OnNext(1); // Ensure active
  691. started.WaitOne();
  692. Thread.Sleep(100); // Try to enter the dispatcher event wait state
  693. for (int i = 0; i < 1000; i++)
  694. {
  695. if (i % 100 == 0)
  696. Thread.Sleep(10);
  697. s.OnNext(i);
  698. }
  699. s.OnCompleted();
  700. end.WaitOne();
  701. }
  702. #endif
  703. [Fact]
  704. public void ObserveOn_LongRunning_HoldUpDuringDispatchAndFail()
  705. {
  706. var started = default(ManualResetEvent);
  707. var stopped = default(ManualResetEvent);
  708. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  709. var s = new Subject<int>();
  710. var onNext = new ManualResetEvent(false);
  711. var resume = new ManualResetEvent(false);
  712. var lst = new List<int>();
  713. var err = default(Exception);
  714. var end = new ManualResetEvent(false);
  715. s.ObserveOn(scheduler).Subscribe(x => { lst.Add(x); onNext.Set(); resume.WaitOne(); }, ex_ => { err = ex_; end.Set(); });
  716. s.OnNext(1);
  717. onNext.WaitOne();
  718. s.OnNext(2);
  719. s.OnNext(3);
  720. var ex = new Exception();
  721. s.OnError(ex);
  722. resume.Set();
  723. end.WaitOne();
  724. Assert.True(lst.SequenceEqual(new[] { 1, 2, 3 }));
  725. Assert.Same(ex, err);
  726. }
  727. [Fact]
  728. public void ObserveOn_LongRunning_Cancel()
  729. {
  730. var started = default(ManualResetEvent);
  731. var stopped = default(ManualResetEvent);
  732. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e);
  733. var s = new Subject<int>();
  734. var lst = new List<int>();
  735. var end = new ManualResetEvent(false);
  736. var running = new ManualResetEvent(false);
  737. var d = s.ObserveOn(scheduler).Subscribe(x => { lst.Add(x); running.Set(); });
  738. s.OnNext(0);
  739. started.WaitOne();
  740. s.OnNext(1);
  741. s.OnNext(2);
  742. s.OnNext(3);
  743. running.WaitOne();
  744. d.Dispose();
  745. stopped.WaitOne();
  746. s.OnNext(4);
  747. Assert.True(lst.Count > 0 && !lst.Contains(4));
  748. }
  749. [Fact]
  750. public void ObserveOn_LongRunning_OnNextThrows()
  751. {
  752. var started = default(ManualResetEvent);
  753. var stopped = default(ManualResetEvent);
  754. var exception = default(Exception);
  755. var scheduler = new TestLongRunningScheduler(e => started = e, e => stopped = e, ex => exception = ex);
  756. var s = new Subject<int>();
  757. var lst = new List<int>();
  758. var end = new ManualResetEvent(false);
  759. var running = new ManualResetEvent(false);
  760. var d = s.ObserveOn(scheduler).Subscribe(x => { lst.Add(x); running.Set(); if (x == 3) throw new Exception(); });
  761. s.OnNext(0);
  762. started.WaitOne();
  763. s.OnNext(1);
  764. s.OnNext(2);
  765. s.OnNext(3);
  766. running.WaitOne();
  767. s.OnNext(4);
  768. stopped.WaitOne();
  769. Assert.NotNull(exception);
  770. }
  771. #endif
  772. #if !NO_SYNCCTX
  773. [Fact]
  774. public void ObserveOn_SynchronizationContext_Simple()
  775. {
  776. var scheduler = new TestScheduler();
  777. var xs = scheduler.CreateHotObservable(
  778. OnNext(90, 1),
  779. OnNext(120, 2),
  780. OnNext(230, 3),
  781. OnNext(240, 4),
  782. OnNext(310, 5),
  783. OnNext(470, 6),
  784. OnCompleted<int>(530)
  785. );
  786. var results = scheduler.Start(() =>
  787. xs.ObserveOn(new MyCtx(scheduler))
  788. );
  789. results.Messages.AssertEqual(
  790. OnNext(231, 3),
  791. OnNext(241, 4),
  792. OnNext(311, 5),
  793. OnNext(471, 6),
  794. OnCompleted<int>(531)
  795. );
  796. xs.Subscriptions.AssertEqual(
  797. Subscribe(200, 531)
  798. );
  799. }
  800. #endif
  801. #endregion
  802. #region SubscribeOn
  803. [Fact]
  804. public void SubscribeOn_Scheduler_ArgumentChecking()
  805. {
  806. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn(default(IObservable<int>), DummyScheduler.Instance));
  807. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.SubscribeOn(DummyObservable<int>.Instance, default(IScheduler)));
  808. }
  809. [Fact]
  810. public void SubscribeOn_Scheduler_Sleep()
  811. {
  812. var scheduler = new TestScheduler();
  813. var s = 0L;
  814. var d = 0L;
  815. var xs = Observable.Create<long>(observer =>
  816. {
  817. s = scheduler.Clock;
  818. return () => d = scheduler.Clock;
  819. });
  820. var results = scheduler.Start(() =>
  821. xs.SubscribeOn(scheduler)
  822. );
  823. results.Messages.AssertEqual(
  824. );
  825. Assert.Equal(201, s);
  826. Assert.Equal(1001, d);
  827. }
  828. [Fact]
  829. public void SubscribeOn_Scheduler_Completed()
  830. {
  831. var scheduler = new TestScheduler();
  832. var xs = scheduler.CreateHotObservable(
  833. OnCompleted<long>(300)
  834. );
  835. var results = scheduler.Start(() =>
  836. xs.SubscribeOn(scheduler)
  837. );
  838. results.Messages.AssertEqual(
  839. OnCompleted<long>(300)
  840. );
  841. xs.Subscriptions.AssertEqual(
  842. Subscribe(201, 301)
  843. );
  844. }
  845. [Fact]
  846. public void SubscribeOn_Scheduler_Error()
  847. {
  848. var scheduler = new TestScheduler();
  849. var ex = new Exception();
  850. var xs = scheduler.CreateHotObservable(
  851. OnError<int>(300, ex)
  852. );
  853. var results = scheduler.Start(() =>
  854. xs.SubscribeOn(scheduler)
  855. );
  856. results.Messages.AssertEqual(
  857. OnError<int>(300, ex)
  858. );
  859. xs.Subscriptions.AssertEqual(
  860. Subscribe(201, 301)
  861. );
  862. }
  863. [Fact]
  864. public void SubscribeOn_Scheduler_Dispose()
  865. {
  866. var scheduler = new TestScheduler();
  867. var xs = scheduler.CreateHotObservable<int>(
  868. );
  869. var results = scheduler.Start(() =>
  870. xs.SubscribeOn(scheduler)
  871. );
  872. results.Messages.AssertEqual(
  873. );
  874. xs.Subscriptions.AssertEqual(
  875. Subscribe(201, 1001)
  876. );
  877. }
  878. #if !NO_SYNCCTX
  879. [Fact]
  880. public void SubscribeOn_SynchronizationContext_Simple()
  881. {
  882. var scheduler = new TestScheduler();
  883. var xs = scheduler.CreateHotObservable(
  884. OnNext(90, 1),
  885. OnNext(120, 2),
  886. OnNext(230, 3),
  887. OnNext(240, 4),
  888. OnNext(310, 5),
  889. OnNext(470, 6),
  890. OnCompleted<int>(530)
  891. );
  892. var results = scheduler.Start(() =>
  893. xs.SubscribeOn(new MyCtx(scheduler))
  894. );
  895. results.Messages.AssertEqual(
  896. OnNext(230, 3),
  897. OnNext(240, 4),
  898. OnNext(310, 5),
  899. OnNext(470, 6),
  900. OnCompleted<int>(530)
  901. );
  902. xs.Subscriptions.AssertEqual(
  903. Subscribe(201, 531)
  904. );
  905. }
  906. #endif
  907. #endregion
  908. #region |> Helpers <|
  909. #if !NO_SYNCCTX
  910. class MyCtx : SynchronizationContext
  911. {
  912. private IScheduler scheduler;
  913. public MyCtx(IScheduler scheduler)
  914. {
  915. this.scheduler = scheduler;
  916. }
  917. public override void Post(SendOrPostCallback d, object state)
  918. {
  919. scheduler.Schedule(state, (self, s) => { d(s); return Disposable.Empty; });
  920. }
  921. }
  922. #endif
  923. #endregion
  924. }
  925. }