RefCountTest.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the MIT License.
  3. // See the LICENSE file in the project root for more information.
  4. using System;
  5. using System.Collections.Generic;
  6. using System.Reactive;
  7. using System.Reactive.Concurrency;
  8. using System.Reactive.Disposables;
  9. using System.Reactive.Linq;
  10. using System.Reactive.Subjects;
  11. using Microsoft.Reactive.Testing;
  12. using Microsoft.VisualStudio.TestTools.UnitTesting;
  13. using Assert = Xunit.Assert;
  14. namespace ReactiveTests.Tests
  15. {
  16. [TestClass]
  17. public class RefCountTest : ReactiveTest
  18. {
  19. private sealed class DematerializingConnectableObservable<T> : IConnectableObservable<T>
  20. {
  21. private readonly IConnectableObservable<Notification<T>> _subject;
  22. public DematerializingConnectableObservable(IConnectableObservable<Notification<T>> subject)
  23. {
  24. _subject = subject;
  25. }
  26. public IDisposable Subscribe(IObserver<T> observer)
  27. {
  28. return _subject.Dematerialize().Subscribe(observer);
  29. }
  30. public IDisposable Connect()
  31. {
  32. return _subject.Connect();
  33. }
  34. }
  35. [TestMethod]
  36. public void RefCount_ArgumentChecking()
  37. {
  38. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount<int>(null));
  39. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount<int>(null, 2));
  40. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), 0));
  41. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), -1));
  42. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), -2));
  43. }
  44. [TestMethod]
  45. public void RefCount_ConnectsOnFirst()
  46. {
  47. var scheduler = new TestScheduler();
  48. var xs = scheduler.CreateHotObservable(
  49. OnNext(210, 1),
  50. OnNext(220, 2),
  51. OnNext(230, 3),
  52. OnNext(240, 4),
  53. OnCompleted<int>(250)
  54. );
  55. var subject = new MySubject();
  56. var conn = new ConnectableObservable<int>(xs, subject);
  57. var res = scheduler.Start(() =>
  58. conn.RefCount()
  59. );
  60. res.Messages.AssertEqual(
  61. OnNext(210, 1),
  62. OnNext(220, 2),
  63. OnNext(230, 3),
  64. OnNext(240, 4),
  65. OnCompleted<int>(250)
  66. );
  67. Assert.True(subject.Disposed);
  68. }
  69. [TestMethod]
  70. public void RefCount_NotConnected()
  71. {
  72. var disconnected = false;
  73. var count = 0;
  74. var xs = Observable.Defer(() =>
  75. {
  76. count++;
  77. return Observable.Create<int>(obs =>
  78. {
  79. return () => { disconnected = true; };
  80. });
  81. });
  82. var subject = new MySubject();
  83. var conn = new ConnectableObservable<int>(xs, subject);
  84. var refd = conn.RefCount();
  85. var dis1 = refd.Subscribe();
  86. Assert.Equal(1, count);
  87. Assert.Equal(1, subject.SubscribeCount);
  88. Assert.False(disconnected);
  89. var dis2 = refd.Subscribe();
  90. Assert.Equal(1, count);
  91. Assert.Equal(2, subject.SubscribeCount);
  92. Assert.False(disconnected);
  93. dis1.Dispose();
  94. Assert.False(disconnected);
  95. dis2.Dispose();
  96. Assert.True(disconnected);
  97. disconnected = false;
  98. var dis3 = refd.Subscribe();
  99. Assert.Equal(2, count);
  100. Assert.Equal(3, subject.SubscribeCount);
  101. Assert.False(disconnected);
  102. dis3.Dispose();
  103. Assert.True(disconnected);
  104. }
  105. [TestMethod]
  106. public void RefCount_OnError()
  107. {
  108. var ex = new Exception();
  109. var xs = Observable.Throw<int>(ex, Scheduler.Immediate);
  110. var res = xs.Publish().RefCount();
  111. res.Subscribe(_ => { Assert.True(false); }, ex_ => { Assert.Same(ex, ex_); }, () => { Assert.True(false); });
  112. res.Subscribe(_ => { Assert.True(false); }, ex_ => { Assert.Same(ex, ex_); }, () => { Assert.True(false); });
  113. }
  114. [TestMethod]
  115. public void RefCount_Publish()
  116. {
  117. var scheduler = new TestScheduler();
  118. var xs = scheduler.CreateHotObservable(
  119. OnNext(210, 1),
  120. OnNext(220, 2),
  121. OnNext(230, 3),
  122. OnNext(240, 4),
  123. OnNext(250, 5),
  124. OnNext(260, 6),
  125. OnNext(270, 7),
  126. OnNext(280, 8),
  127. OnNext(290, 9),
  128. OnCompleted<int>(300)
  129. );
  130. var res = xs.Publish().RefCount();
  131. var d1 = default(IDisposable);
  132. var o1 = scheduler.CreateObserver<int>();
  133. scheduler.ScheduleAbsolute(215, () => { d1 = res.Subscribe(o1); });
  134. scheduler.ScheduleAbsolute(235, () => { d1.Dispose(); });
  135. var d2 = default(IDisposable);
  136. var o2 = scheduler.CreateObserver<int>();
  137. scheduler.ScheduleAbsolute(225, () => { d2 = res.Subscribe(o2); });
  138. scheduler.ScheduleAbsolute(275, () => { d2.Dispose(); });
  139. var d3 = default(IDisposable);
  140. var o3 = scheduler.CreateObserver<int>();
  141. scheduler.ScheduleAbsolute(255, () => { d3 = res.Subscribe(o3); });
  142. scheduler.ScheduleAbsolute(265, () => { d3.Dispose(); });
  143. var d4 = default(IDisposable);
  144. var o4 = scheduler.CreateObserver<int>();
  145. scheduler.ScheduleAbsolute(285, () => { d4 = res.Subscribe(o4); });
  146. scheduler.ScheduleAbsolute(320, () => { d4.Dispose(); });
  147. scheduler.Start();
  148. o1.Messages.AssertEqual(
  149. OnNext(220, 2),
  150. OnNext(230, 3)
  151. );
  152. o2.Messages.AssertEqual(
  153. OnNext(230, 3),
  154. OnNext(240, 4),
  155. OnNext(250, 5),
  156. OnNext(260, 6),
  157. OnNext(270, 7)
  158. );
  159. o3.Messages.AssertEqual(
  160. OnNext(260, 6)
  161. );
  162. o4.Messages.AssertEqual(
  163. OnNext(290, 9),
  164. OnCompleted<int>(300)
  165. );
  166. xs.Subscriptions.AssertEqual(
  167. Subscribe(215, 275),
  168. Subscribe(285, 300)
  169. );
  170. }
  171. [TestMethod]
  172. public void RefCount_can_connect_again_if_previous_subscription_terminated_synchronously()
  173. {
  174. var seen = 0;
  175. var terminated = false;
  176. var subject = new ReplaySubject<Notification<int>>(1);
  177. var connectable = new DematerializingConnectableObservable<int>(subject.Publish());
  178. var refCount = connectable.RefCount();
  179. subject.OnNext(Notification.CreateOnNext(36));
  180. using (refCount.Subscribe(value => seen = value, () => terminated = true))
  181. {
  182. Assert.Equal(36, seen);
  183. }
  184. seen = 0;
  185. terminated = false;
  186. subject.OnNext(Notification.CreateOnCompleted<int>());
  187. using (refCount.Subscribe(value => seen = value, () => terminated = true))
  188. {
  189. Assert.Equal(0, seen);
  190. Assert.True(terminated);
  191. }
  192. seen = 0;
  193. terminated = false;
  194. subject.OnNext(Notification.CreateOnNext(36));
  195. using (refCount.Subscribe(value => seen = value, () => terminated = true))
  196. {
  197. Assert.Equal(36, seen);
  198. Assert.False(terminated);
  199. }
  200. }
  201. [TestMethod]
  202. public void LazyRefCount_ArgumentChecking()
  203. {
  204. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount<int>(null, TimeSpan.FromSeconds(2)));
  205. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount<int>(null, TimeSpan.FromSeconds(2), Scheduler.Default));
  206. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount<int>(null, 2, TimeSpan.FromSeconds(2)));
  207. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount<int>(null, 2, TimeSpan.FromSeconds(2)));
  208. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount(Observable.Never<int>().Publish(), TimeSpan.FromSeconds(2), null));
  209. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), 0, TimeSpan.FromSeconds(2)));
  210. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), -1, TimeSpan.FromSeconds(2)));
  211. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.RefCount(Observable.Never<int>().Publish(), 2, TimeSpan.FromSeconds(2), null));
  212. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), 0, TimeSpan.FromSeconds(2), Scheduler.Default));
  213. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Observable.RefCount(Observable.Never<int>().Publish(), -1, TimeSpan.FromSeconds(2), Scheduler.Default));
  214. }
  215. [TestMethod]
  216. public void LazyRefCount_ConnectsOnFirst()
  217. {
  218. var scheduler = new TestScheduler();
  219. var xs = scheduler.CreateHotObservable(
  220. OnNext(210, 1),
  221. OnNext(220, 2),
  222. OnNext(230, 3),
  223. OnNext(240, 4),
  224. OnCompleted<int>(250)
  225. );
  226. var subject = new MySubject();
  227. var conn = new ConnectableObservable<int>(xs, subject);
  228. var res = scheduler.Start(() =>
  229. conn.RefCount(TimeSpan.FromSeconds(2))
  230. );
  231. res.Messages.AssertEqual(
  232. OnNext(210, 1),
  233. OnNext(220, 2),
  234. OnNext(230, 3),
  235. OnNext(240, 4),
  236. OnCompleted<int>(250)
  237. );
  238. Assert.True(subject.Disposed);
  239. }
  240. [TestMethod]
  241. public void LazyRefCount_NotConnected()
  242. {
  243. var scheduler = new TestScheduler();
  244. var disconnected = false;
  245. var count = 0;
  246. var xs = Observable.Defer(() =>
  247. {
  248. count++;
  249. return Observable.Create<int>(obs =>
  250. {
  251. return () => { disconnected = true; };
  252. });
  253. });
  254. var subject = new MySubject();
  255. var conn = new ConnectableObservable<int>(xs, subject);
  256. var refd = conn.RefCount(TimeSpan.FromTicks(20), scheduler);
  257. var dis1 = refd.Subscribe();
  258. Assert.Equal(1, count);
  259. Assert.Equal(1, subject.SubscribeCount);
  260. Assert.False(disconnected);
  261. var dis2 = refd.Subscribe();
  262. Assert.Equal(1, count);
  263. Assert.Equal(2, subject.SubscribeCount);
  264. Assert.False(disconnected);
  265. dis1.Dispose();
  266. Assert.False(disconnected);
  267. dis2.Dispose();
  268. Assert.False(disconnected);
  269. scheduler.AdvanceBy(19);
  270. Assert.False(disconnected);
  271. scheduler.AdvanceBy(1);
  272. Assert.True(disconnected);
  273. disconnected = false;
  274. var dis3 = refd.Subscribe();
  275. Assert.Equal(2, count);
  276. Assert.Equal(3, subject.SubscribeCount);
  277. Assert.False(disconnected);
  278. dis3.Dispose();
  279. scheduler.AdvanceBy(20);
  280. Assert.True(disconnected);
  281. }
  282. [TestMethod]
  283. public void LazyRefCount_OnError()
  284. {
  285. var ex = new Exception();
  286. var xs = Observable.Throw<int>(ex, Scheduler.Immediate);
  287. var res = xs.Publish().RefCount(TimeSpan.FromSeconds(2));
  288. res.Subscribe(_ => throw new Exception(), ex_ => { Assert.Same(ex, ex_); }, () => throw new Exception());
  289. res.Subscribe(_ => throw new Exception(), ex_ => { Assert.Same(ex, ex_); }, () => throw new Exception());
  290. }
  291. [TestMethod]
  292. public void LazyRefCount_Publish()
  293. {
  294. var scheduler = new TestScheduler();
  295. var xs = scheduler.CreateHotObservable(
  296. OnNext(210, 1),
  297. OnNext(220, 2),
  298. OnNext(230, 3),
  299. OnNext(240, 4),
  300. OnNext(250, 5),
  301. OnNext(260, 6),
  302. OnNext(270, 7),
  303. OnNext(280, 8),
  304. OnNext(290, 9),
  305. OnCompleted<int>(300)
  306. );
  307. var res = xs.Publish().RefCount(TimeSpan.FromTicks(9), scheduler);
  308. var d1 = default(IDisposable);
  309. var o1 = scheduler.CreateObserver<int>();
  310. scheduler.ScheduleAbsolute(215, () => { d1 = res.Subscribe(o1); });
  311. scheduler.ScheduleAbsolute(235, () => { d1.Dispose(); });
  312. var d2 = default(IDisposable);
  313. var o2 = scheduler.CreateObserver<int>();
  314. scheduler.ScheduleAbsolute(225, () => { d2 = res.Subscribe(o2); });
  315. scheduler.ScheduleAbsolute(275, () =>
  316. {
  317. d2.Dispose();
  318. });
  319. var d3 = default(IDisposable);
  320. var o3 = scheduler.CreateObserver<int>();
  321. scheduler.ScheduleAbsolute(255, () => { d3 = res.Subscribe(o3); });
  322. scheduler.ScheduleAbsolute(265, () => { d3.Dispose(); });
  323. var d4 = default(IDisposable);
  324. var o4 = scheduler.CreateObserver<int>();
  325. scheduler.ScheduleAbsolute(285, () => { d4 = res.Subscribe(o4); });
  326. scheduler.ScheduleAbsolute(320, () => { d4.Dispose(); });
  327. scheduler.Start();
  328. o1.Messages.AssertEqual(
  329. OnNext(220, 2),
  330. OnNext(230, 3)
  331. );
  332. o2.Messages.AssertEqual(
  333. OnNext(230, 3),
  334. OnNext(240, 4),
  335. OnNext(250, 5),
  336. OnNext(260, 6),
  337. OnNext(270, 7)
  338. );
  339. o3.Messages.AssertEqual(
  340. OnNext(260, 6)
  341. );
  342. o4.Messages.AssertEqual(
  343. OnNext(290, 9),
  344. OnCompleted<int>(300)
  345. );
  346. xs.Subscriptions.AssertEqual(
  347. Subscribe(215, 284),
  348. Subscribe(285, 300)
  349. );
  350. }
  351. [TestMethod]
  352. public void RefCount_source_already_completed_synchronously()
  353. {
  354. var subscribed = 0;
  355. var unsubscribed = 0;
  356. var o1 = Observable.Create<string>(observer =>
  357. {
  358. subscribed++;
  359. observer.OnCompleted();
  360. return Disposable.Create(() => unsubscribed++);
  361. });
  362. var o2 = o1.Publish().RefCount();
  363. var s1 = o2.Subscribe();
  364. Assert.Equal(1, subscribed);
  365. Assert.Equal(1, unsubscribed);
  366. var s2 = o2.Subscribe();
  367. Assert.Equal(1, subscribed);
  368. Assert.Equal(1, unsubscribed);
  369. }
  370. [TestMethod]
  371. public void RefCount_minObservers_not_connected_Eager()
  372. {
  373. int connected = 0;
  374. var source = Observable.Defer(() =>
  375. {
  376. connected++;
  377. return Observable.Never<int>();
  378. })
  379. .Publish()
  380. .RefCount(2);
  381. Assert.Equal(0, connected);
  382. source.Subscribe();
  383. Assert.Equal(0, connected);
  384. }
  385. [TestMethod]
  386. public void RefCount_minObservers_connected_Eager()
  387. {
  388. var connected = 0;
  389. var source = Observable.Defer(() =>
  390. {
  391. connected++;
  392. return Observable.Range(1, 5);
  393. })
  394. .Publish()
  395. .RefCount(2);
  396. Assert.Equal(0, connected);
  397. var list1 = new List<int>();
  398. source.Subscribe(list1.Add);
  399. Assert.Equal(0, connected);
  400. Assert.Empty(list1);
  401. var list2 = new List<int>();
  402. source.Subscribe(list2.Add);
  403. Assert.Equal(1, connected);
  404. var expected = new List<int>(new[] { 1, 2, 3, 4, 5 });
  405. Assert.Equal(expected, list1);
  406. Assert.Equal(expected, list2);
  407. }
  408. [TestMethod]
  409. public void RefCount_minObservers_not_connected_Lazy()
  410. {
  411. int connected = 0;
  412. var source = Observable.Defer(() =>
  413. {
  414. connected++;
  415. return Observable.Never<int>();
  416. })
  417. .Publish()
  418. .RefCount(2, TimeSpan.FromMinutes(1));
  419. Assert.Equal(0, connected);
  420. source.Subscribe();
  421. Assert.Equal(0, connected);
  422. }
  423. [TestMethod]
  424. public void RefCount_minObservers_connected_Lazy()
  425. {
  426. var connected = 0;
  427. var source = Observable.Defer(() =>
  428. {
  429. connected++;
  430. return Observable.Range(1, 5);
  431. })
  432. .Publish()
  433. .RefCount(2, TimeSpan.FromMinutes(1));
  434. Assert.Equal(0, connected);
  435. var list1 = new List<int>();
  436. source.Subscribe(list1.Add);
  437. Assert.Equal(0, connected);
  438. Assert.Empty(list1);
  439. var list2 = new List<int>();
  440. source.Subscribe(list2.Add);
  441. Assert.Equal(1, connected);
  442. var expected = new List<int>(new[] { 1, 2, 3, 4, 5 });
  443. Assert.Equal(expected, list1);
  444. Assert.Equal(expected, list2);
  445. }
  446. }
  447. }