DisposableTests.cs 27 KB

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