DisposableTests.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626
  1. // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
  2. using System;
  3. using System.Collections;
  4. using System.Collections.Generic;
  5. using System.Linq;
  6. using System.Reactive.Concurrency;
  7. using System.Reactive.Disposables;
  8. using System.Threading;
  9. using Microsoft.Reactive.Testing;
  10. using Microsoft.VisualStudio.TestTools.UnitTesting;
  11. namespace ReactiveTests.Tests
  12. {
  13. [TestClass]
  14. public class DisposableTests
  15. {
  16. [TestMethod]
  17. public void AnonymousDisposable_Create()
  18. {
  19. var d = Disposable.Create(() => { });
  20. Assert.IsNotNull(d);
  21. }
  22. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  23. public void AnonymousDisposable_CreateNull()
  24. {
  25. Disposable.Create(null);
  26. }
  27. [TestMethod]
  28. public void AnonymousDisposable_Dispose()
  29. {
  30. var disposed = false;
  31. var d = Disposable.Create(() => { disposed = true; });
  32. Assert.IsFalse(disposed);
  33. d.Dispose();
  34. Assert.IsTrue(disposed);
  35. var c = d as ICancelable;
  36. Assert.IsNotNull(c);
  37. Assert.IsTrue(c.IsDisposed);
  38. }
  39. [TestMethod]
  40. public void EmptyDisposable()
  41. {
  42. var d = Disposable.Empty;
  43. Assert.IsNotNull(d);
  44. d.Dispose();
  45. }
  46. [TestMethod]
  47. public void BooleanDisposable()
  48. {
  49. var d = new BooleanDisposable();
  50. Assert.IsFalse(d.IsDisposed);
  51. d.Dispose();
  52. Assert.IsTrue(d.IsDisposed);
  53. d.Dispose();
  54. Assert.IsTrue(d.IsDisposed);
  55. }
  56. [TestMethod]
  57. public void SingleAssignmentDisposable_SetNull()
  58. {
  59. var d = new SingleAssignmentDisposable();
  60. d.Disposable = null;
  61. }
  62. [TestMethod]
  63. public void SingleAssignmentDisposable_DisposeAfterSet()
  64. {
  65. var disposed = false;
  66. var d = new SingleAssignmentDisposable();
  67. var dd = Disposable.Create(() => { disposed = true; });
  68. d.Disposable = dd;
  69. Assert.AreSame(dd, d.Disposable);
  70. Assert.IsFalse(disposed);
  71. d.Dispose();
  72. Assert.IsTrue(disposed);
  73. d.Dispose();
  74. Assert.IsTrue(disposed);
  75. Assert.IsTrue(d.IsDisposed);
  76. }
  77. [TestMethod]
  78. public void SingleAssignmentDisposable_DisposeBeforeSet()
  79. {
  80. var disposed = false;
  81. var d = new SingleAssignmentDisposable();
  82. var dd = Disposable.Create(() => { disposed = true; });
  83. Assert.IsFalse(disposed);
  84. d.Dispose();
  85. Assert.IsFalse(disposed);
  86. Assert.IsTrue(d.IsDisposed);
  87. d.Disposable = dd;
  88. Assert.IsTrue(disposed);
  89. //Assert.IsNull(d.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  90. d.Disposable.Dispose(); // This should be a nop.
  91. d.Dispose();
  92. Assert.IsTrue(disposed);
  93. }
  94. [TestMethod]
  95. public void SingleAssignmentDisposable_SetMultipleTimes()
  96. {
  97. var d = new SingleAssignmentDisposable();
  98. d.Disposable = Disposable.Empty;
  99. ReactiveAssert.Throws<InvalidOperationException>(() => { d.Disposable = Disposable.Empty; });
  100. }
  101. [TestMethod]
  102. public void CompositeDisposable_ArgumentChecking()
  103. {
  104. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable(default(IDisposable[])));
  105. ReactiveAssert.Throws<ArgumentNullException>(() => new CompositeDisposable(default(IEnumerable<IDisposable>)));
  106. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => new CompositeDisposable(-1));
  107. }
  108. [TestMethod]
  109. public void CompositeDisposable_Contains()
  110. {
  111. var d1 = Disposable.Create(() => {} );
  112. var d2 = Disposable.Create(() => { });
  113. var g = new CompositeDisposable(d1, d2);
  114. Assert.AreEqual(2, g.Count);
  115. Assert.IsTrue(g.Contains(d1));
  116. Assert.IsTrue(g.Contains(d2));
  117. ReactiveAssert.Throws<ArgumentNullException>(() => g.Contains(null));
  118. }
  119. [TestMethod]
  120. public void CompositeDisposable_IsReadOnly()
  121. {
  122. Assert.IsFalse(new CompositeDisposable().IsReadOnly);
  123. }
  124. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  125. public void CompositeDisposable_CopyTo_Null()
  126. {
  127. new CompositeDisposable().CopyTo(null, 0);
  128. }
  129. [TestMethod, ExpectedException(typeof(ArgumentOutOfRangeException))]
  130. public void CompositeDisposable_CopyTo_Negative()
  131. {
  132. new CompositeDisposable().CopyTo(new IDisposable[2], -1);
  133. }
  134. [TestMethod, ExpectedException(typeof(ArgumentOutOfRangeException))]
  135. public void CompositeDisposable_CopyTo_BeyondEnd()
  136. {
  137. new CompositeDisposable().CopyTo(new IDisposable[2], 2);
  138. }
  139. [TestMethod]
  140. public void CompositeDisposable_CopyTo()
  141. {
  142. var d1 = Disposable.Create(() => { });
  143. var d2 = Disposable.Create(() => { });
  144. var g = new CompositeDisposable(new List<IDisposable> { d1, d2 });
  145. var d = new IDisposable[3];
  146. g.CopyTo(d, 1);
  147. Assert.AreSame(d1, d[1]);
  148. Assert.AreSame(d2, d[2]);
  149. }
  150. [TestMethod]
  151. public void CompositeDisposable_ToArray()
  152. {
  153. var d1 = Disposable.Create(() => { });
  154. var d2 = Disposable.Create(() => { });
  155. var g = new CompositeDisposable(d1, d2);
  156. Assert.AreEqual(2, g.Count);
  157. var x = Enumerable.ToArray(g);
  158. Assert.IsTrue(g.ToArray().SequenceEqual(new[] { d1, d2 }));
  159. }
  160. [TestMethod]
  161. public void CompositeDisposable_GetEnumerator()
  162. {
  163. var d1 = Disposable.Create(() => { });
  164. var d2 = Disposable.Create(() => { });
  165. var g = new CompositeDisposable(d1, d2);
  166. var lst = new List<IDisposable>();
  167. foreach (var x in g)
  168. lst.Add(x);
  169. Assert.IsTrue(lst.SequenceEqual(new[] { d1, d2 }));
  170. }
  171. [TestMethod]
  172. public void CompositeDisposable_GetEnumeratorNonGeneric()
  173. {
  174. var d1 = Disposable.Create(() => { });
  175. var d2 = Disposable.Create(() => { });
  176. var g = new CompositeDisposable(d1, d2);
  177. var lst = new List<IDisposable>();
  178. foreach (IDisposable x in (IEnumerable)g)
  179. lst.Add(x);
  180. Assert.IsTrue(lst.SequenceEqual(new[] { d1, d2 }));
  181. }
  182. [TestMethod]
  183. public void CompositeDisposable_CollectionInitializer()
  184. {
  185. var d1 = Disposable.Create(() => { });
  186. var d2 = Disposable.Create(() => { });
  187. var g = new CompositeDisposable { d1, d2 };
  188. Assert.AreEqual(2, g.Count);
  189. Assert.IsTrue(g.Contains(d1));
  190. Assert.IsTrue(g.Contains(d2));
  191. }
  192. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  193. public void CompositeDisposable_AddNull()
  194. {
  195. new CompositeDisposable().Add(null);
  196. }
  197. [TestMethod]
  198. public void CompositeDisposable_Add()
  199. {
  200. var d1 = Disposable.Create(() => { });
  201. var d2 = Disposable.Create(() => { });
  202. var g = new CompositeDisposable(d1);
  203. Assert.AreEqual(1, g.Count);
  204. Assert.IsTrue(g.Contains(d1));
  205. g.Add(d2);
  206. Assert.AreEqual(2, g.Count);
  207. Assert.IsTrue(g.Contains(d2));
  208. }
  209. [TestMethod]
  210. public void CompositeDisposable_AddAfterDispose()
  211. {
  212. var disp1 = false;
  213. var disp2 = false;
  214. var d1 = Disposable.Create(() => { disp1 = true; });
  215. var d2 = Disposable.Create(() => { disp2 = true; });
  216. var g = new CompositeDisposable(d1);
  217. Assert.AreEqual(1, g.Count);
  218. g.Dispose();
  219. Assert.IsTrue(disp1);
  220. Assert.AreEqual(0, g.Count); // CHECK
  221. g.Add(d2);
  222. Assert.IsTrue(disp2);
  223. Assert.AreEqual(0, g.Count); // CHECK
  224. Assert.IsTrue(g.IsDisposed);
  225. }
  226. [TestMethod]
  227. public void CompositeDisposable_Remove()
  228. {
  229. var disp1 = false;
  230. var disp2 = false;
  231. var d1 = Disposable.Create(() => { disp1 = true; });
  232. var d2 = Disposable.Create(() => { disp2 = true; });
  233. var g = new CompositeDisposable(d1, d2);
  234. Assert.AreEqual(2, g.Count);
  235. Assert.IsTrue(g.Contains(d1));
  236. Assert.IsTrue(g.Contains(d2));
  237. Assert.IsTrue(g.Remove(d1));
  238. Assert.AreEqual(1, g.Count);
  239. Assert.IsFalse(g.Contains(d1));
  240. Assert.IsTrue(g.Contains(d2));
  241. Assert.IsTrue(disp1);
  242. Assert.IsTrue(g.Remove(d2));
  243. Assert.IsFalse(g.Contains(d1));
  244. Assert.IsFalse(g.Contains(d2));
  245. Assert.IsTrue(disp2);
  246. var disp3 = false;
  247. var d3 = Disposable.Create(() => { disp3 = true; });
  248. Assert.IsFalse(g.Remove(d3));
  249. Assert.IsFalse(disp3);
  250. }
  251. [TestMethod]
  252. public void CompositeDisposable_Clear()
  253. {
  254. var disp1 = false;
  255. var disp2 = false;
  256. var d1 = Disposable.Create(() => { disp1 = true; });
  257. var d2 = Disposable.Create(() => { disp2 = true; });
  258. var g = new CompositeDisposable(d1, d2);
  259. Assert.AreEqual(2, g.Count);
  260. g.Clear();
  261. Assert.IsTrue(disp1);
  262. Assert.IsTrue(disp2);
  263. Assert.AreEqual(0, g.Count);
  264. var disp3 = false;
  265. var d3 = Disposable.Create(() => { disp3 = true; });
  266. g.Add(d3);
  267. Assert.IsFalse(disp3);
  268. Assert.AreEqual(1, g.Count);
  269. }
  270. [TestMethod]
  271. public void CompositeDisposable_RemoveOptimizationBehavior()
  272. {
  273. var g = new CompositeDisposable();
  274. var m = new Dictionary<int, IDisposable>();
  275. var r = new List<int>();
  276. var N = 100;
  277. for (int i = 0; i < N; i++)
  278. {
  279. var j = i;
  280. var d = Disposable.Create(() => r.Add(j));
  281. m[j] = d;
  282. g.Add(d);
  283. }
  284. var d1 = Enumerable.Range(0, N).Where(i => i % 2 == 0).ToArray();
  285. foreach (var i in d1)
  286. g.Remove(m[i]);
  287. Assert.IsTrue(r.SequenceEqual(d1));
  288. var d2 = Enumerable.Range(0, N).Where(i => i % 3 == 0).ToArray();
  289. foreach (var i in d2)
  290. g.Remove(m[i]);
  291. Assert.IsTrue(r.SequenceEqual(d1.Concat(d2.Where(x => !d1.Any(y => x == y)))));
  292. var d3 = Enumerable.Range(0, N).Where(i => i % 5 == 0).ToArray();
  293. foreach (var i in d3)
  294. g.Remove(m[i]);
  295. Assert.IsTrue(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)))));
  296. g.Dispose();
  297. var z = r.Except(d1.Union(d2).Union(d3)).ToArray();
  298. Assert.IsTrue(z.SequenceEqual(Enumerable.Range(0, N).Where(i => !(i % 2 == 0 || i % 3 == 0 || i % 5 == 0))));
  299. }
  300. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  301. public void CompositeDisposable_RemoveNull()
  302. {
  303. new CompositeDisposable().Remove(null);
  304. }
  305. #if DESKTOPCLR40 || DESKTOPCLR45
  306. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  307. public void CancellationDisposable_Ctor_Null()
  308. {
  309. new CancellationDisposable(null);
  310. }
  311. [TestMethod]
  312. public void CancellationDisposable_DefaultCtor()
  313. {
  314. var c = new CancellationDisposable();
  315. Assert.IsNotNull(c.Token);
  316. Assert.IsFalse(c.Token.IsCancellationRequested);
  317. Assert.IsTrue(c.Token.CanBeCanceled);
  318. c.Dispose();
  319. Assert.IsTrue(c.IsDisposed);
  320. Assert.IsTrue(c.Token.IsCancellationRequested);
  321. }
  322. [TestMethod]
  323. public void CancellationDisposable_TokenCtor()
  324. {
  325. var t = new CancellationTokenSource();
  326. var c = new CancellationDisposable(t);
  327. Assert.IsTrue(t.Token == c.Token);
  328. Assert.IsFalse(c.Token.IsCancellationRequested);
  329. Assert.IsTrue(c.Token.CanBeCanceled);
  330. c.Dispose();
  331. Assert.IsTrue(c.IsDisposed);
  332. Assert.IsTrue(c.Token.IsCancellationRequested);
  333. }
  334. #endif
  335. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  336. public void ContextDisposable_CreateNullContext()
  337. {
  338. new ContextDisposable(null, Disposable.Empty);
  339. }
  340. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  341. public void ContextDisposable_CreateNullDisposable()
  342. {
  343. new ContextDisposable(new SynchronizationContext(), null);
  344. }
  345. [TestMethod]
  346. public void ContextDisposable()
  347. {
  348. var disp = false;
  349. var m = new MySync();
  350. var c = new ContextDisposable(m, Disposable.Create(() => { disp = true; }));
  351. Assert.AreSame(m, c.Context);
  352. Assert.IsFalse(m._disposed);
  353. Assert.IsFalse(disp);
  354. c.Dispose();
  355. Assert.IsTrue(c.IsDisposed);
  356. Assert.IsTrue(m._disposed);
  357. Assert.IsTrue(disp);
  358. }
  359. class MySync : SynchronizationContext
  360. {
  361. internal bool _disposed = false;
  362. public override void Post(SendOrPostCallback d, object state)
  363. {
  364. d(state);
  365. _disposed = true;
  366. }
  367. }
  368. [TestMethod]
  369. public void SerialDisposable_Ctor_Prop()
  370. {
  371. var m = new SerialDisposable();
  372. Assert.IsNull(m.Disposable);
  373. }
  374. [TestMethod]
  375. public void SerialDisposable_ReplaceBeforeDispose()
  376. {
  377. var disp1 = false;
  378. var disp2 = false;
  379. var m = new SerialDisposable();
  380. var d1 = Disposable.Create(() => { disp1 = true; });
  381. m.Disposable = d1;
  382. Assert.AreSame(d1, m.Disposable);
  383. Assert.IsFalse(disp1);
  384. var d2 = Disposable.Create(() => { disp2 = true; });
  385. m.Disposable = d2;
  386. Assert.AreSame(d2, m.Disposable);
  387. Assert.IsTrue(disp1);
  388. Assert.IsFalse(disp2);
  389. }
  390. [TestMethod]
  391. public void SerialDisposable_ReplaceAfterDispose()
  392. {
  393. var disp1 = false;
  394. var disp2 = false;
  395. var m = new SerialDisposable();
  396. m.Dispose();
  397. Assert.IsTrue(m.IsDisposed);
  398. var d1 = Disposable.Create(() => { disp1 = true; });
  399. m.Disposable = d1;
  400. Assert.IsNull(m.Disposable); // CHECK
  401. Assert.IsTrue(disp1);
  402. var d2 = Disposable.Create(() => { disp2 = true; });
  403. m.Disposable = d2;
  404. Assert.IsNull(m.Disposable); // CHECK
  405. Assert.IsTrue(disp2);
  406. }
  407. [TestMethod]
  408. public void SerialDisposable_Dispose()
  409. {
  410. var disp = false;
  411. var m = new SerialDisposable();
  412. var d = Disposable.Create(() => { disp = true; });
  413. m.Disposable = d;
  414. Assert.AreSame(d, m.Disposable);
  415. Assert.IsFalse(disp);
  416. m.Dispose();
  417. Assert.IsTrue(m.IsDisposed);
  418. Assert.IsTrue(disp);
  419. //Assert.IsNull(m.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  420. }
  421. [TestMethod, ExpectedException(typeof(ArgumentNullException))]
  422. public void RefCountDisposable_Ctor_Null()
  423. {
  424. new RefCountDisposable(null);
  425. }
  426. [TestMethod]
  427. public void RefCountDisposable_SingleReference()
  428. {
  429. var d = new BooleanDisposable();
  430. var r = new RefCountDisposable(d);
  431. Assert.IsFalse(d.IsDisposed);
  432. r.Dispose();
  433. Assert.IsTrue(d.IsDisposed);
  434. r.Dispose();
  435. Assert.IsTrue(d.IsDisposed);
  436. }
  437. [TestMethod]
  438. public void RefCountDisposable_RefCounting()
  439. {
  440. var d = new BooleanDisposable();
  441. var r = new RefCountDisposable(d);
  442. Assert.IsFalse(d.IsDisposed);
  443. var d1 = r.GetDisposable();
  444. var d2 = r.GetDisposable();
  445. Assert.IsFalse(d.IsDisposed);
  446. d1.Dispose();
  447. Assert.IsFalse(d.IsDisposed);
  448. d2.Dispose();
  449. Assert.IsFalse(d.IsDisposed); // CHECK
  450. r.Dispose();
  451. Assert.IsTrue(d.IsDisposed);
  452. Assert.IsTrue(r.IsDisposed);
  453. var d3 = r.GetDisposable(); // CHECK
  454. d3.Dispose();
  455. }
  456. [TestMethod]
  457. public void RefCountDisposable_PrimaryDisposesFirst()
  458. {
  459. var d = new BooleanDisposable();
  460. var r = new RefCountDisposable(d);
  461. Assert.IsFalse(d.IsDisposed);
  462. var d1 = r.GetDisposable();
  463. var d2 = r.GetDisposable();
  464. Assert.IsFalse(d.IsDisposed);
  465. d1.Dispose();
  466. Assert.IsFalse(d.IsDisposed);
  467. r.Dispose();
  468. Assert.IsFalse(d.IsDisposed);
  469. d2.Dispose();
  470. Assert.IsTrue(d.IsDisposed);
  471. }
  472. [TestMethod]
  473. public void ScheduledDisposable_Null()
  474. {
  475. ReactiveAssert.Throws<ArgumentNullException>(() => new ScheduledDisposable(null, Disposable.Empty));
  476. ReactiveAssert.Throws<ArgumentNullException>(() => new ScheduledDisposable(Scheduler.Immediate, null));
  477. }
  478. [TestMethod]
  479. public void ScheduledDisposable()
  480. {
  481. var d = new BooleanDisposable();
  482. var s = new ScheduledDisposable(Scheduler.Immediate, d);
  483. Assert.IsFalse(d.IsDisposed);
  484. Assert.AreSame(Scheduler.Immediate, s.Scheduler);
  485. Assert.AreSame(d, s.Disposable);
  486. s.Dispose();
  487. Assert.IsTrue(d.IsDisposed);
  488. Assert.IsTrue(s.IsDisposed);
  489. Assert.AreSame(Scheduler.Immediate, s.Scheduler);
  490. //Assert.AreSame(d, s.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  491. s.Disposable.Dispose(); // This should be a nop.
  492. }
  493. [TestMethod]
  494. public void MultipleAssignmentDisposable()
  495. {
  496. var m = new MultipleAssignmentDisposable();
  497. var disp1 = false;
  498. var d1 = Disposable.Create(() => { disp1 = true; });
  499. m.Disposable = d1;
  500. Assert.AreSame(d1, m.Disposable);
  501. Assert.IsFalse(m.IsDisposed);
  502. var disp2 = false;
  503. var d2 = Disposable.Create(() => { disp2 = true; });
  504. m.Disposable = d2;
  505. Assert.AreSame(d2, m.Disposable);
  506. Assert.IsFalse(m.IsDisposed);
  507. Assert.IsFalse(disp1);
  508. m.Dispose();
  509. Assert.IsTrue(disp2);
  510. Assert.IsTrue(m.IsDisposed);
  511. //Assert.IsNull(m.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  512. m.Disposable.Dispose(); // This should be a nop.
  513. var disp3 = false;
  514. var d3 = Disposable.Create(() => { disp3 = true; });
  515. m.Disposable = d3;
  516. Assert.IsTrue(disp3);
  517. //Assert.IsNull(m.Disposable); // BREAKING CHANGE v2 > v1.x - Undefined behavior after disposal.
  518. m.Disposable.Dispose(); // This should be a nop.
  519. Assert.IsTrue(m.IsDisposed);
  520. }
  521. }
  522. }