DisposableTests.cs 28 KB

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