DisposableTests.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921
  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;
  6. using System.Collections.Generic;
  7. using System.Linq;
  8. using System.Reactive.Concurrency;
  9. using System.Reactive.Disposables;
  10. using System.Reactive.Disposables.Fluent;
  11. using System.Threading;
  12. using Microsoft.Reactive.Testing;
  13. using Microsoft.VisualStudio.TestTools.UnitTesting;
  14. using Assert = Xunit.Assert;
  15. namespace ReactiveTests.Tests
  16. {
  17. [TestClass]
  18. public class DisposableTests
  19. {
  20. [TestMethod]
  21. public void AnonymousDisposable_Create()
  22. {
  23. var d = Disposable.Create(() => { });
  24. Assert.NotNull(d);
  25. }
  26. [TestMethod]
  27. public void AnonymousDisposable_CreateNull()
  28. {
  29. Assert.Throws<ArgumentNullException>(() => Disposable.Create(null));
  30. }
  31. [TestMethod]
  32. public void AnonymousDisposable_Dispose()
  33. {
  34. var disposed = false;
  35. var d = Disposable.Create(() => { disposed = true; });
  36. Assert.False(disposed);
  37. d.Dispose();
  38. Assert.True(disposed);
  39. var c = d as ICancelable;
  40. Assert.NotNull(c);
  41. Assert.True(c.IsDisposed);
  42. }
  43. [TestMethod]
  44. public void EmptyDisposable()
  45. {
  46. var d = Disposable.Empty;
  47. Assert.NotNull(d);
  48. d.Dispose();
  49. }
  50. [TestMethod]
  51. public void BooleanDisposable()
  52. {
  53. var d = new BooleanDisposable();
  54. Assert.False(d.IsDisposed);
  55. d.Dispose();
  56. Assert.True(d.IsDisposed);
  57. d.Dispose();
  58. Assert.True(d.IsDisposed);
  59. }
  60. [TestMethod]
  61. public void SingleAssignmentDisposable_SetNull()
  62. {
  63. _ = new SingleAssignmentDisposable
  64. {
  65. Disposable = null
  66. };
  67. }
  68. [TestMethod]
  69. public void SingleAssignmentDisposable_DisposeAfterSet()
  70. {
  71. var disposed = false;
  72. var d = new SingleAssignmentDisposable();
  73. var dd = Disposable.Create(() => { disposed = true; });
  74. d.Disposable = dd;
  75. Assert.Same(dd, d.Disposable);
  76. Assert.False(disposed);
  77. d.Dispose();
  78. Assert.True(disposed);
  79. d.Dispose();
  80. Assert.True(disposed);
  81. Assert.True(d.IsDisposed);
  82. }
  83. [TestMethod]
  84. public void SingleAssignmentDisposable_DisposeBeforeSet()
  85. {
  86. var disposed = false;
  87. var d = new SingleAssignmentDisposable();
  88. var dd = Disposable.Create(() => { disposed = true; });
  89. Assert.False(disposed);
  90. d.Dispose();
  91. Assert.False(disposed);
  92. Assert.True(d.IsDisposed);
  93. d.Disposable = dd;
  94. Assert.True(disposed);
  95. //Assert.Null(d.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  96. d.Disposable.Dispose(); // This should be a nop.
  97. d.Dispose();
  98. Assert.True(disposed);
  99. }
  100. [TestMethod]
  101. public void SingleAssignmentDisposable_SetMultipleTimes()
  102. {
  103. var d = new SingleAssignmentDisposable
  104. {
  105. Disposable = Disposable.Empty
  106. };
  107. ReactiveAssert.Throws<InvalidOperationException>(() => { d.Disposable = Disposable.Empty; });
  108. }
  109. [TestMethod]
  110. public void CompositeDisposable_ArgumentChecking()
  111. {
  112. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  113. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable(default(IDisposable[])));
  114. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable(default(IEnumerable<IDisposable>)));
  115. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => new CompositeDisposable(-1));
  116. #pragma warning restore CA1806
  117. }
  118. [TestMethod]
  119. public void CompositeDisposable_Contains()
  120. {
  121. var d1 = Disposable.Create(() => { });
  122. var d2 = Disposable.Create(() => { });
  123. var g = new CompositeDisposable(d1, d2);
  124. Assert.Equal(2, g.Count);
  125. Assert.True(g.Contains(d1));
  126. Assert.True(g.Contains(d2));
  127. ReactiveAssert.Throws<ArgumentNullException>(() => g.Contains(null));
  128. }
  129. [TestMethod]
  130. public void CompositeDisposable_IsReadOnly()
  131. {
  132. Assert.False(new CompositeDisposable().IsReadOnly);
  133. }
  134. [TestMethod]
  135. public void CompositeDisposable_CopyTo_Null()
  136. {
  137. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable().CopyTo(null, 0));
  138. }
  139. [TestMethod]
  140. public void CompositeDisposable_CopyTo_Negative()
  141. {
  142. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => new CompositeDisposable().CopyTo(new IDisposable[2], -1));
  143. }
  144. [TestMethod]
  145. public void CompositeDisposable_CopyTo_BeyondEnd()
  146. {
  147. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => new CompositeDisposable().CopyTo(new IDisposable[2], 2));
  148. }
  149. [TestMethod]
  150. public void CompositeDisposable_CopyTo()
  151. {
  152. var d1 = Disposable.Create(() => { });
  153. var d2 = Disposable.Create(() => { });
  154. var g = new CompositeDisposable(new List<IDisposable> { d1, d2 });
  155. var d = new IDisposable[3];
  156. g.CopyTo(d, 1);
  157. Assert.Same(d1, d[1]);
  158. Assert.Same(d2, d[2]);
  159. }
  160. [TestMethod]
  161. public void CompositeDisposable_ToArray()
  162. {
  163. var d1 = Disposable.Create(() => { });
  164. var d2 = Disposable.Create(() => { });
  165. var g = new CompositeDisposable(d1, d2);
  166. Assert.Equal(2, g.Count);
  167. var x = Enumerable.ToArray(g);
  168. Assert.True(g.ToArray().SequenceEqual([d1, d2]));
  169. }
  170. [TestMethod]
  171. public void CompositeDisposable_GetEnumerator()
  172. {
  173. var d1 = Disposable.Create(() => { });
  174. var d2 = Disposable.Create(() => { });
  175. var g = new CompositeDisposable(d1, d2);
  176. var lst = new List<IDisposable>();
  177. foreach (var x in g)
  178. {
  179. lst.Add(x);
  180. }
  181. Assert.True(lst.SequenceEqual([d1, d2]));
  182. }
  183. [TestMethod]
  184. public void CompositeDisposable_GetEnumeratorNonGeneric()
  185. {
  186. var d1 = Disposable.Create(() => { });
  187. var d2 = Disposable.Create(() => { });
  188. var g = new CompositeDisposable(d1, d2);
  189. var lst = new List<IDisposable>();
  190. foreach (IDisposable x in (IEnumerable)g)
  191. {
  192. lst.Add(x);
  193. }
  194. Assert.True(lst.SequenceEqual([d1, d2]));
  195. }
  196. [TestMethod]
  197. public void CompositeDisposable_CollectionInitializer()
  198. {
  199. var d1 = Disposable.Create(() => { });
  200. var d2 = Disposable.Create(() => { });
  201. var g = new CompositeDisposable { d1, d2 };
  202. Assert.Equal(2, g.Count);
  203. Assert.True(g.Contains(d1));
  204. Assert.True(g.Contains(d2));
  205. }
  206. [TestMethod]
  207. public void CompositeDisposable_AddNull_via_params_ctor()
  208. {
  209. IDisposable d1 = null;
  210. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  211. ReactiveAssert.Throws<ArgumentException>(() => new CompositeDisposable(d1));
  212. #pragma warning restore CA1806
  213. }
  214. [TestMethod]
  215. public void CompositeDisposable_AddNull_via_IEnum_ctor()
  216. {
  217. IEnumerable<IDisposable> values = [null];
  218. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  219. ReactiveAssert.Throws<ArgumentException>(() => new CompositeDisposable(values));
  220. #pragma warning restore CA1806
  221. }
  222. [TestMethod]
  223. public void CompositeDisposable_AddNull()
  224. {
  225. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable().Add(null));
  226. }
  227. [TestMethod]
  228. public void CompositeDisposable_Add()
  229. {
  230. var d1 = Disposable.Create(() => { });
  231. var d2 = Disposable.Create(() => { });
  232. var g = new CompositeDisposable(d1);
  233. Assert.Equal(1, g.Count);
  234. Assert.True(g.Contains(d1));
  235. g.Add(d2);
  236. Assert.Equal(2, g.Count);
  237. Assert.True(g.Contains(d2));
  238. }
  239. [TestMethod]
  240. public void CompositeDisposable_AddAfterDispose()
  241. {
  242. var disp1 = false;
  243. var disp2 = false;
  244. var d1 = Disposable.Create(() => { disp1 = true; });
  245. var d2 = Disposable.Create(() => { disp2 = true; });
  246. var g = new CompositeDisposable(d1);
  247. Assert.Equal(1, g.Count);
  248. g.Dispose();
  249. Assert.True(disp1);
  250. Assert.Equal(0, g.Count); // CHECK
  251. g.Add(d2);
  252. Assert.True(disp2);
  253. Assert.Equal(0, g.Count); // CHECK
  254. Assert.True(g.IsDisposed);
  255. }
  256. [TestMethod]
  257. public void CompositeDisposable_Remove()
  258. {
  259. var disp1 = false;
  260. var disp2 = false;
  261. var d1 = Disposable.Create(() => { disp1 = true; });
  262. var d2 = Disposable.Create(() => { disp2 = true; });
  263. var g = new CompositeDisposable(d1, d2);
  264. Assert.Equal(2, g.Count);
  265. Assert.True(g.Contains(d1));
  266. Assert.True(g.Contains(d2));
  267. Assert.True(g.Remove(d1));
  268. Assert.Equal(1, g.Count);
  269. Assert.False(g.Contains(d1));
  270. Assert.True(g.Contains(d2));
  271. Assert.True(disp1);
  272. Assert.True(g.Remove(d2));
  273. Assert.False(g.Contains(d1));
  274. Assert.False(g.Contains(d2));
  275. Assert.True(disp2);
  276. var disp3 = false;
  277. var d3 = Disposable.Create(() => { disp3 = true; });
  278. Assert.False(g.Remove(d3));
  279. Assert.False(disp3);
  280. }
  281. [TestMethod]
  282. public void CompositeDisposable_Clear()
  283. {
  284. var disp1 = false;
  285. var disp2 = false;
  286. var d1 = Disposable.Create(() => { disp1 = true; });
  287. var d2 = Disposable.Create(() => { disp2 = true; });
  288. var g = new CompositeDisposable(d1, d2);
  289. Assert.Equal(2, g.Count);
  290. g.Clear();
  291. Assert.True(disp1);
  292. Assert.True(disp2);
  293. Assert.Equal(0, g.Count);
  294. var disp3 = false;
  295. var d3 = Disposable.Create(() => { disp3 = true; });
  296. g.Add(d3);
  297. Assert.False(disp3);
  298. Assert.Equal(1, g.Count);
  299. }
  300. [TestMethod]
  301. public void CompositeDisposable_RemoveOptimizationBehavior()
  302. {
  303. var g = new CompositeDisposable();
  304. var m = new Dictionary<int, IDisposable>();
  305. var r = new List<int>();
  306. var N = 100;
  307. for (var i = 0; i < N; i++)
  308. {
  309. var j = i;
  310. var d = Disposable.Create(() => r.Add(j));
  311. m[j] = d;
  312. g.Add(d);
  313. }
  314. var d1 = Enumerable.Range(0, N).Where(i => i % 2 == 0).ToArray();
  315. foreach (var i in d1)
  316. {
  317. g.Remove(m[i]);
  318. }
  319. Assert.True(r.SequenceEqual(d1));
  320. var d2 = Enumerable.Range(0, N).Where(i => i % 3 == 0).ToArray();
  321. foreach (var i in d2)
  322. {
  323. g.Remove(m[i]);
  324. }
  325. Assert.True(r.SequenceEqual(d1.Concat(d2.Where(x => !d1.Any(y => x == y)))));
  326. var d3 = Enumerable.Range(0, N).Where(i => i % 5 == 0).ToArray();
  327. foreach (var i in d3)
  328. {
  329. g.Remove(m[i]);
  330. }
  331. Assert.True(r.SequenceEqual(d1.Concat(d2.Where(x => !d1.Any(y => x == y))).Concat(d3.Where(x => !d1.Any(y => x == y) && !d2.Any(y => x == y)))));
  332. g.Dispose();
  333. var z = r.Except(d1.Union(d2).Union(d3)).ToArray();
  334. Assert.True(z.SequenceEqual(Enumerable.Range(0, N).Where(i => !(i % 2 == 0 || i % 3 == 0 || i % 5 == 0))));
  335. }
  336. [TestMethod]
  337. public void CompositeDisposable_RemoveNull()
  338. {
  339. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable().Remove(null));
  340. }
  341. [TestMethod]
  342. public void CompositeDisposable_Empty_GetEnumerator()
  343. {
  344. var composite = new CompositeDisposable();
  345. Assert.False(composite.GetEnumerator().MoveNext());
  346. }
  347. [TestMethod]
  348. public void CompositeDisposable_DisposeWith()
  349. {
  350. var c = new CompositeDisposable();
  351. var d = new BooleanDisposable();
  352. d.DisposeWith(c);
  353. Assert.True(c.Contains(d));
  354. c.Dispose();
  355. Assert.True(d.IsDisposed);
  356. Assert.True(c.IsDisposed);
  357. }
  358. [TestMethod]
  359. public void CompositeDisposable_NonCollection_Enumerable_Init()
  360. {
  361. var d = new BooleanDisposable();
  362. var composite = new CompositeDisposable(Just(d));
  363. composite.Dispose();
  364. Assert.True(d.IsDisposed);
  365. }
  366. private static IEnumerable<IDisposable> Just(IDisposable d)
  367. {
  368. yield return d;
  369. }
  370. [TestMethod]
  371. public void CompositeDisposable_Disposed_Is_NoOp()
  372. {
  373. var d = new BooleanDisposable();
  374. var composite = new CompositeDisposable(d);
  375. composite.Dispose();
  376. composite.Clear();
  377. Assert.False(composite.Contains(d));
  378. var array = new IDisposable[1];
  379. composite.CopyTo(array, 0);
  380. Assert.Null(array[0]);
  381. }
  382. [TestMethod]
  383. public void CompositeDisposable_CopyTo_Index_Out_Of_Range()
  384. {
  385. var d1 = new BooleanDisposable();
  386. var d2 = new BooleanDisposable();
  387. var composite = new CompositeDisposable(d1, d2);
  388. var array = new IDisposable[2];
  389. try
  390. {
  391. composite.CopyTo(array, 1);
  392. Assert.False(true, "Should have thrown!");
  393. }
  394. catch (ArgumentOutOfRangeException)
  395. {
  396. // expected
  397. }
  398. }
  399. [TestMethod]
  400. public void CompositeDisposable_GetEnumerator_Reset()
  401. {
  402. var d = new BooleanDisposable();
  403. var composite = new CompositeDisposable(d);
  404. var enumerator = composite.GetEnumerator();
  405. Assert.True(enumerator.MoveNext());
  406. Assert.Equal(d, enumerator.Current);
  407. Assert.False(enumerator.MoveNext());
  408. enumerator.Reset();
  409. Assert.True(enumerator.MoveNext());
  410. Assert.Equal(d, enumerator.Current);
  411. }
  412. [TestMethod]
  413. public void CompositeDisposable_GetEnumerator_Disposed_Entries()
  414. {
  415. var d1 = new BooleanDisposable();
  416. var d2 = new BooleanDisposable();
  417. var d3 = new BooleanDisposable();
  418. var composite = new CompositeDisposable(d1, d2, d3);
  419. composite.Remove(d2);
  420. var enumerator = composite.GetEnumerator();
  421. Assert.True(enumerator.MoveNext());
  422. Assert.Equal(d1, enumerator.Current);
  423. Assert.True(enumerator.MoveNext());
  424. Assert.Equal(d3, enumerator.Current);
  425. Assert.False(enumerator.MoveNext());
  426. }
  427. [TestMethod]
  428. public void CancellationDisposable_Ctor_Null()
  429. {
  430. Assert.Throws<ArgumentNullException>(() => new CancellationDisposable(null));
  431. }
  432. [TestMethod]
  433. public void CancellationDisposable_DefaultCtor()
  434. {
  435. var c = new CancellationDisposable();
  436. Assert.NotNull(c.Token);
  437. Assert.False(c.Token.IsCancellationRequested);
  438. Assert.True(c.Token.CanBeCanceled);
  439. c.Dispose();
  440. Assert.True(c.IsDisposed);
  441. Assert.True(c.Token.IsCancellationRequested);
  442. }
  443. [TestMethod]
  444. public void CancellationDisposable_TokenCtor()
  445. {
  446. var t = new CancellationTokenSource();
  447. var c = new CancellationDisposable(t);
  448. Assert.True(t.Token == c.Token);
  449. Assert.False(c.Token.IsCancellationRequested);
  450. Assert.True(c.Token.CanBeCanceled);
  451. c.Dispose();
  452. Assert.True(c.IsDisposed);
  453. Assert.True(c.Token.IsCancellationRequested);
  454. }
  455. [TestMethod]
  456. public void ContextDisposable_CreateNullContext()
  457. {
  458. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  459. ReactiveAssert.Throws<ArgumentNullException>(() => new ContextDisposable(null, Disposable.Empty));
  460. #pragma warning restore CA1806
  461. }
  462. [TestMethod]
  463. public void ContextDisposable_CreateNullDisposable()
  464. {
  465. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  466. ReactiveAssert.Throws<ArgumentNullException>(() => new ContextDisposable(new SynchronizationContext(), null));
  467. #pragma warning restore CA1806
  468. }
  469. [TestMethod]
  470. public void ContextDisposable()
  471. {
  472. var disp = false;
  473. var m = new MySync();
  474. var c = new ContextDisposable(m, Disposable.Create(() => { disp = true; }));
  475. Assert.Same(m, c.Context);
  476. Assert.False(m._disposed);
  477. Assert.False(disp);
  478. c.Dispose();
  479. Assert.True(c.IsDisposed);
  480. Assert.True(m._disposed);
  481. Assert.True(disp);
  482. }
  483. private class MySync : SynchronizationContext
  484. {
  485. internal bool _disposed = false;
  486. public override void Post(SendOrPostCallback d, object state)
  487. {
  488. d(state);
  489. _disposed = true;
  490. }
  491. }
  492. [TestMethod]
  493. public void SerialDisposable_Ctor_Prop()
  494. {
  495. var m = new SerialDisposable();
  496. Assert.Null(m.Disposable);
  497. }
  498. [TestMethod]
  499. public void SerialDisposable_ReplaceBeforeDispose()
  500. {
  501. var disp1 = false;
  502. var disp2 = false;
  503. var m = new SerialDisposable();
  504. var d1 = Disposable.Create(() => { disp1 = true; });
  505. m.Disposable = d1;
  506. Assert.Same(d1, m.Disposable);
  507. Assert.False(disp1);
  508. var d2 = Disposable.Create(() => { disp2 = true; });
  509. m.Disposable = d2;
  510. Assert.Same(d2, m.Disposable);
  511. Assert.True(disp1);
  512. Assert.False(disp2);
  513. }
  514. [TestMethod]
  515. public void SerialDisposable_ReplaceAfterDispose()
  516. {
  517. var disp1 = false;
  518. var disp2 = false;
  519. var m = new SerialDisposable();
  520. m.Dispose();
  521. Assert.True(m.IsDisposed);
  522. var d1 = Disposable.Create(() => { disp1 = true; });
  523. m.Disposable = d1;
  524. Assert.Null(m.Disposable); // CHECK
  525. Assert.True(disp1);
  526. var d2 = Disposable.Create(() => { disp2 = true; });
  527. m.Disposable = d2;
  528. Assert.Null(m.Disposable); // CHECK
  529. Assert.True(disp2);
  530. }
  531. [TestMethod]
  532. public void SerialDisposable_Dispose()
  533. {
  534. var disp = false;
  535. var m = new SerialDisposable();
  536. var d = Disposable.Create(() => { disp = true; });
  537. m.Disposable = d;
  538. Assert.Same(d, m.Disposable);
  539. Assert.False(disp);
  540. m.Dispose();
  541. Assert.True(m.IsDisposed);
  542. Assert.True(disp);
  543. //Assert.Null(m.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  544. }
  545. [TestMethod]
  546. public void RefCountDisposable_Ctor_Null()
  547. {
  548. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  549. ReactiveAssert.Throws<ArgumentNullException>(() => new RefCountDisposable(null));
  550. #pragma warning restore CA1806
  551. }
  552. [TestMethod]
  553. public void RefCountDisposable_SingleReference()
  554. {
  555. var d = new BooleanDisposable();
  556. var r = new RefCountDisposable(d);
  557. Assert.False(d.IsDisposed);
  558. r.Dispose();
  559. Assert.True(d.IsDisposed);
  560. r.Dispose();
  561. Assert.True(d.IsDisposed);
  562. }
  563. [TestMethod]
  564. public void RefCountDisposable_RefCounting()
  565. {
  566. var d = new BooleanDisposable();
  567. var r = new RefCountDisposable(d);
  568. Assert.False(d.IsDisposed);
  569. var d1 = r.GetDisposable();
  570. var d2 = r.GetDisposable();
  571. Assert.False(d.IsDisposed);
  572. d1.Dispose();
  573. Assert.False(d.IsDisposed);
  574. d2.Dispose();
  575. Assert.False(d.IsDisposed); // CHECK
  576. r.Dispose();
  577. Assert.True(d.IsDisposed);
  578. Assert.True(r.IsDisposed);
  579. var d3 = r.GetDisposable(); // CHECK
  580. d3.Dispose();
  581. }
  582. [TestMethod]
  583. public void RefCountDisposable_PrimaryDisposesFirst()
  584. {
  585. var d = new BooleanDisposable();
  586. var r = new RefCountDisposable(d);
  587. Assert.False(d.IsDisposed);
  588. var d1 = r.GetDisposable();
  589. var d2 = r.GetDisposable();
  590. Assert.False(d.IsDisposed);
  591. d1.Dispose();
  592. Assert.False(d.IsDisposed);
  593. r.Dispose();
  594. Assert.False(d.IsDisposed);
  595. d2.Dispose();
  596. Assert.True(d.IsDisposed);
  597. }
  598. [TestMethod]
  599. public void RefCountDisposable_Throw_If_Disposed()
  600. {
  601. var d = new BooleanDisposable();
  602. var r = new RefCountDisposable(d, true);
  603. r.Dispose();
  604. Assert.True(d.IsDisposed);
  605. ReactiveAssert.Throws<ObjectDisposedException>(() => { r.GetDisposable(); });
  606. }
  607. [TestMethod]
  608. public void ScheduledDisposable_Null()
  609. {
  610. #pragma warning disable CA1806 // (Unused new instance.) We expect the constructor to throw.
  611. ReactiveAssert.Throws<ArgumentNullException>(() => new ScheduledDisposable(null, Disposable.Empty));
  612. ReactiveAssert.Throws<ArgumentNullException>(() => new ScheduledDisposable(Scheduler.Immediate, null));
  613. #pragma warning restore CA1806
  614. }
  615. [TestMethod]
  616. public void ScheduledDisposable()
  617. {
  618. var d = new BooleanDisposable();
  619. var s = new ScheduledDisposable(Scheduler.Immediate, d);
  620. Assert.False(d.IsDisposed);
  621. Assert.Same(Scheduler.Immediate, s.Scheduler);
  622. Assert.Same(d, s.Disposable);
  623. s.Dispose();
  624. Assert.True(d.IsDisposed);
  625. Assert.True(s.IsDisposed);
  626. Assert.Same(Scheduler.Immediate, s.Scheduler);
  627. //Assert.Same(d, s.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  628. s.Disposable.Dispose(); // This should be a nop.
  629. }
  630. [TestMethod]
  631. public void MultipleAssignmentDisposable()
  632. {
  633. var m = new MultipleAssignmentDisposable();
  634. var disp1 = false;
  635. var d1 = Disposable.Create(() => { disp1 = true; });
  636. m.Disposable = d1;
  637. Assert.Same(d1, m.Disposable);
  638. Assert.False(m.IsDisposed);
  639. var disp2 = false;
  640. var d2 = Disposable.Create(() => { disp2 = true; });
  641. m.Disposable = d2;
  642. Assert.Same(d2, m.Disposable);
  643. Assert.False(m.IsDisposed);
  644. Assert.False(disp1);
  645. m.Dispose();
  646. Assert.True(disp2);
  647. Assert.True(m.IsDisposed);
  648. //Assert.Null(m.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  649. m.Disposable.Dispose(); // This should be a nop.
  650. var disp3 = false;
  651. var d3 = Disposable.Create(() => { disp3 = true; });
  652. m.Disposable = d3;
  653. Assert.True(disp3);
  654. //Assert.Null(m.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  655. m.Disposable.Dispose(); // This should be a nop.
  656. Assert.True(m.IsDisposed);
  657. }
  658. [TestMethod]
  659. public void StableCompositeDisposable_ArgumentChecking()
  660. {
  661. var d = Disposable.Empty;
  662. ReactiveAssert.Throws<ArgumentNullException>(() => StableCompositeDisposable.Create(null, d));
  663. ReactiveAssert.Throws<ArgumentNullException>(() => StableCompositeDisposable.Create(d, null));
  664. ReactiveAssert.Throws<ArgumentNullException>(() => StableCompositeDisposable.Create(default));
  665. ReactiveAssert.Throws<ArgumentNullException>(() => StableCompositeDisposable.Create(default(IEnumerable<IDisposable>)));
  666. ReactiveAssert.Throws<ArgumentException>(() => StableCompositeDisposable.Create(null, d, d));
  667. ReactiveAssert.Throws<ArgumentException>(() => StableCompositeDisposable.Create(d, null, d));
  668. ReactiveAssert.Throws<ArgumentException>(() => StableCompositeDisposable.Create(d, d, null));
  669. }
  670. [TestMethod]
  671. public void StableCompositeDisposable_Binary()
  672. {
  673. var disp1 = false;
  674. var d1 = Disposable.Create(() => { Assert.False(disp1); disp1 = true; });
  675. var disp2 = false;
  676. var d2 = Disposable.Create(() => { Assert.False(disp2); disp2 = true; });
  677. var d = StableCompositeDisposable.Create(d1, d2);
  678. Assert.False(disp1);
  679. Assert.False(disp2);
  680. Assert.False(d.IsDisposed);
  681. d.Dispose();
  682. Assert.True(disp1);
  683. Assert.True(disp2);
  684. Assert.True(d.IsDisposed);
  685. d.Dispose();
  686. Assert.True(disp1);
  687. Assert.True(disp2);
  688. Assert.True(d.IsDisposed);
  689. }
  690. [TestMethod]
  691. public void StableCompositeDisposable_Nary1()
  692. {
  693. var disp1 = false;
  694. var d1 = Disposable.Create(() => { Assert.False(disp1); disp1 = true; });
  695. var disp2 = false;
  696. var d2 = Disposable.Create(() => { Assert.False(disp2); disp2 = true; });
  697. var disp3 = false;
  698. var d3 = Disposable.Create(() => { Assert.False(disp3); disp3 = true; });
  699. var d = StableCompositeDisposable.Create(d1, d2, d3);
  700. Assert.False(disp1);
  701. Assert.False(disp2);
  702. Assert.False(disp3);
  703. Assert.False(d.IsDisposed);
  704. d.Dispose();
  705. Assert.True(disp1);
  706. Assert.True(disp2);
  707. Assert.True(disp3);
  708. Assert.True(d.IsDisposed);
  709. d.Dispose();
  710. Assert.True(disp1);
  711. Assert.True(disp2);
  712. Assert.True(disp3);
  713. Assert.True(d.IsDisposed);
  714. }
  715. [TestMethod]
  716. public void StableCompositeDisposable_Nary2()
  717. {
  718. var disp1 = false;
  719. var d1 = Disposable.Create(() => { Assert.False(disp1); disp1 = true; });
  720. var disp2 = false;
  721. var d2 = Disposable.Create(() => { Assert.False(disp2); disp2 = true; });
  722. var disp3 = false;
  723. var d3 = Disposable.Create(() => { Assert.False(disp3); disp3 = true; });
  724. var d = StableCompositeDisposable.Create(new List<IDisposable>([d1, d2, d3]));
  725. Assert.False(disp1);
  726. Assert.False(disp2);
  727. Assert.False(disp3);
  728. Assert.False(d.IsDisposed);
  729. d.Dispose();
  730. Assert.True(disp1);
  731. Assert.True(disp2);
  732. Assert.True(disp3);
  733. Assert.True(d.IsDisposed);
  734. d.Dispose();
  735. Assert.True(disp1);
  736. Assert.True(disp2);
  737. Assert.True(disp3);
  738. Assert.True(d.IsDisposed);
  739. }
  740. }
  741. }