CancellationTokenSourceTest.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497
  1. //
  2. // CancellationTokenSourceTest.cs
  3. //
  4. // Authors:
  5. // Marek Safar ([email protected])
  6. // Jeremie Laval ([email protected])
  7. //
  8. // Copyright 2011 Xamarin, Inc (http://www.xamarin.com)
  9. //
  10. // Permission is hereby granted, free of charge, to any person obtaining
  11. // a copy of this software and associated documentation files (the
  12. // "Software"), to deal in the Software without restriction, including
  13. // without limitation the rights to use, copy, modify, merge, publish,
  14. // distribute, sublicense, and/or sell copies of the Software, and to
  15. // permit persons to whom the Software is furnished to do so, subject to
  16. // the following conditions:
  17. //
  18. // The above copyright notice and this permission notice shall be
  19. // included in all copies or substantial portions of the Software.
  20. //
  21. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  22. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  23. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  24. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  25. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  26. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  27. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  28. //
  29. using System;
  30. using System.Threading;
  31. using NUnit.Framework;
  32. using System.Threading.Tasks;
  33. namespace MonoTests.System.Threading
  34. {
  35. [TestFixture]
  36. public class CancellationTokenSourceTest
  37. {
  38. [Test]
  39. public void Ctor_Invalid ()
  40. {
  41. try {
  42. new CancellationTokenSource (-4);
  43. Assert.Fail ("#1");
  44. } catch (ArgumentException) {
  45. }
  46. }
  47. [Test]
  48. public void Ctor_Timeout ()
  49. {
  50. int called = 0;
  51. var cts = new CancellationTokenSource (TimeSpan.FromMilliseconds (20));
  52. var mre = new ManualResetEvent (false);
  53. cts.Token.Register (() => { called++; mre.Set (); });
  54. Assert.IsTrue (mre.WaitOne (1000), "Not called in 1000ms");
  55. Assert.AreEqual (1, called, "#1");
  56. }
  57. [Test]
  58. public void CancelAfter ()
  59. {
  60. int called = 0;
  61. var cts = new CancellationTokenSource ();
  62. var mre = new ManualResetEvent(false);
  63. cts.Token.Register (() => { called++; mre.Set (); });
  64. cts.CancelAfter (20);
  65. Assert.IsTrue(mre.WaitOne (1000), "Should be cancelled in ~20ms");
  66. Assert.AreEqual (1, called, "#1");
  67. }
  68. [Test]
  69. public void CancelAfter_Invalid ()
  70. {
  71. var cts = new CancellationTokenSource ();
  72. try {
  73. cts.CancelAfter (-9);
  74. Assert.Fail ("#1");
  75. } catch (ArgumentException) {
  76. }
  77. }
  78. [Test]
  79. public void CancelAfter_Disposed ()
  80. {
  81. int called = 0;
  82. var cts = new CancellationTokenSource ();
  83. var mre = new ManualResetEvent (false);
  84. cts.Token.Register (() => { called++; mre.Set (); });
  85. cts.CancelAfter (50);
  86. cts.Dispose ();
  87. Assert.IsFalse (mre.WaitOne (100), "Shouldn't have been called");
  88. Assert.AreEqual (0, called, "#1");
  89. }
  90. [Test]
  91. public void Token ()
  92. {
  93. CancellationTokenSource cts = new CancellationTokenSource ();
  94. Assert.IsTrue (cts.Token.CanBeCanceled, "#1");
  95. Assert.IsFalse (cts.Token.IsCancellationRequested, "#2");
  96. Assert.IsNotNull (cts.Token.WaitHandle, "#3");
  97. }
  98. [Test]
  99. public void Cancel_NoRegistration ()
  100. {
  101. CancellationTokenSource cts = new CancellationTokenSource ();
  102. cts.Cancel ();
  103. }
  104. [Test]
  105. public void Cancel ()
  106. {
  107. var cts = new CancellationTokenSource ();
  108. int called = 0;
  109. cts.Token.Register (l => { Assert.AreEqual ("v", l); ++called; }, "v");
  110. cts.Cancel ();
  111. Assert.AreEqual (1, called, "#1");
  112. called = 0;
  113. cts.Token.Register (() => { called += 12; });
  114. cts.Cancel ();
  115. Assert.AreEqual (12, called, "#2");
  116. }
  117. [Test]
  118. public void Cancel_Order ()
  119. {
  120. var cts = new CancellationTokenSource ();
  121. var current = 0;
  122. Action<object> a = x => { Assert.AreEqual(current, x); current++; };
  123. cts.Token.Register (a, 2);
  124. cts.Token.Register (a, 1);
  125. cts.Token.Register (a, 0);
  126. cts.Cancel ();
  127. }
  128. [Test]
  129. public void CancelWithDispose ()
  130. {
  131. CancellationTokenSource cts = new CancellationTokenSource ();
  132. CancellationToken c = cts.Token;
  133. c.Register (() => {
  134. cts.Dispose ();
  135. });
  136. int called = 0;
  137. c.Register (() => {
  138. called++;
  139. });
  140. cts.Cancel ();
  141. Assert.AreEqual (1, called, "#1");
  142. }
  143. [Test]
  144. public void Cancel_SingleException ()
  145. {
  146. var cts = new CancellationTokenSource ();
  147. cts.Token.Register (() => { throw new ApplicationException (); });
  148. try {
  149. cts.Cancel ();
  150. Assert.Fail ("#1");
  151. } catch (AggregateException e) {
  152. Assert.AreEqual (1, e.InnerExceptions.Count, "#2");
  153. }
  154. cts.Cancel ();
  155. }
  156. [Test]
  157. public void Cancel_MultipleExceptions ()
  158. {
  159. var cts = new CancellationTokenSource ();
  160. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  161. cts.Token.Register (() => { throw new ApplicationException ("2"); });
  162. cts.Token.Register (() => { throw new ApplicationException ("3"); });
  163. try {
  164. cts.Cancel ();
  165. Assert.Fail ("#1");
  166. } catch (AggregateException e) {
  167. Assert.AreEqual (3, e.InnerExceptions.Count, "#2");
  168. }
  169. cts.Cancel ();
  170. try {
  171. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  172. Assert.Fail ("#11");
  173. } catch (ApplicationException) {
  174. }
  175. cts.Cancel ();
  176. }
  177. [Test]
  178. public void Cancel_ExceptionOrder ()
  179. {
  180. var cts = new CancellationTokenSource ();
  181. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  182. cts.Token.Register (() => { throw new ApplicationException ("2"); });
  183. cts.Token.Register (() => { throw new ApplicationException ("3"); });
  184. try {
  185. cts.Cancel ();
  186. } catch (AggregateException e) {
  187. Assert.AreEqual (3, e.InnerExceptions.Count, "#2");
  188. Assert.AreEqual ("3", e.InnerExceptions[0].Message, "#3");
  189. Assert.AreEqual ("2", e.InnerExceptions[1].Message, "#4");
  190. Assert.AreEqual ("1", e.InnerExceptions[2].Message, "#5");
  191. }
  192. }
  193. [Test]
  194. public void Cancel_MultipleException_Recursive ()
  195. {
  196. CancellationTokenSource cts = new CancellationTokenSource ();
  197. CancellationToken c = cts.Token;
  198. c.Register (() => {
  199. cts.Cancel ();
  200. });
  201. c.Register (() => {
  202. throw new ApplicationException ();
  203. });
  204. c.Register (() => {
  205. throw new NotSupportedException ();
  206. });
  207. try {
  208. cts.Cancel (false);
  209. Assert.Fail ("#1");
  210. } catch (AggregateException e) {
  211. Assert.AreEqual (2, e.InnerExceptions.Count, "#2");
  212. }
  213. }
  214. [Test]
  215. public void Cancel_MultipleExceptionsFirstThrows ()
  216. {
  217. var cts = new CancellationTokenSource ();
  218. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  219. cts.Token.Register (() => { throw new ApplicationException ("2"); });
  220. cts.Token.Register (() => { throw new ApplicationException ("3"); });
  221. try {
  222. cts.Cancel (true);
  223. Assert.Fail ("#1");
  224. } catch (ApplicationException) {
  225. }
  226. cts.Cancel ();
  227. }
  228. [Test]
  229. public void CreateLinkedTokenSource_InvalidArguments ()
  230. {
  231. var cts = new CancellationTokenSource ();
  232. var token = cts.Token;
  233. try {
  234. CancellationTokenSource.CreateLinkedTokenSource (null);
  235. Assert.Fail ("#1");
  236. } catch (ArgumentNullException) {
  237. }
  238. try {
  239. CancellationTokenSource.CreateLinkedTokenSource (new CancellationToken[0]);
  240. Assert.Fail ("#2");
  241. } catch (ArgumentException) {
  242. }
  243. }
  244. [Test]
  245. public void CreateLinkedTokenSource ()
  246. {
  247. var cts = new CancellationTokenSource ();
  248. cts.Cancel ();
  249. var linked = CancellationTokenSource.CreateLinkedTokenSource (cts.Token);
  250. Assert.IsTrue (linked.IsCancellationRequested, "#1");
  251. linked = CancellationTokenSource.CreateLinkedTokenSource (new CancellationToken ());
  252. Assert.IsFalse (linked.IsCancellationRequested, "#2");
  253. }
  254. [Test]
  255. public void Dispose ()
  256. {
  257. var cts = new CancellationTokenSource ();
  258. var token = cts.Token;
  259. cts.Dispose ();
  260. cts.Dispose ();
  261. var b = cts.IsCancellationRequested;
  262. token.ThrowIfCancellationRequested ();
  263. try {
  264. cts.Cancel ();
  265. Assert.Fail ("#1");
  266. } catch (ObjectDisposedException) {
  267. }
  268. try {
  269. var t = cts.Token;
  270. Assert.Fail ("#2");
  271. } catch (ObjectDisposedException) {
  272. }
  273. bool throwOnDispose = false;
  274. AppContext.TryGetSwitch ("Switch.System.Threading.ThrowExceptionIfDisposedCancellationTokenSource", out throwOnDispose);
  275. if (throwOnDispose) {
  276. try {
  277. token.Register (() => { });
  278. Assert.Fail ("#3");
  279. } catch (ObjectDisposedException) {
  280. }
  281. }
  282. try {
  283. var wh = token.WaitHandle;
  284. Assert.Fail ("#4");
  285. } catch (ObjectDisposedException) {
  286. }
  287. if (throwOnDispose) {
  288. try {
  289. CancellationTokenSource.CreateLinkedTokenSource (token);
  290. Assert.Fail ("#5");
  291. } catch (ObjectDisposedException) {
  292. }
  293. }
  294. try {
  295. cts.CancelAfter (1);
  296. Assert.Fail ("#6");
  297. } catch (ObjectDisposedException) {
  298. }
  299. }
  300. [Test]
  301. public void RegisterThenDispose ()
  302. {
  303. var cts1 = new CancellationTokenSource ();
  304. var reg1 = cts1.Token.Register (() => { throw new ApplicationException (); });
  305. var cts2 = new CancellationTokenSource ();
  306. var reg2 = cts2.Token.Register (() => { throw new ApplicationException (); });
  307. Assert.AreNotEqual (cts1, cts2, "#1");
  308. Assert.AreNotSame (cts1, cts2, "#2");
  309. reg1.Dispose ();
  310. cts1.Cancel ();
  311. try {
  312. cts2.Cancel ();
  313. Assert.Fail ("#3");
  314. } catch (AggregateException) {
  315. }
  316. }
  317. [Test]
  318. public void RegisterWhileCancelling ()
  319. {
  320. var cts = new CancellationTokenSource ();
  321. var mre = new ManualResetEvent (false);
  322. var mre2 = new ManualResetEvent (false);
  323. int called = 0;
  324. cts.Token.Register (() => {
  325. Assert.IsTrue (cts.IsCancellationRequested, "#10");
  326. Assert.IsTrue (cts.Token.WaitHandle.WaitOne (0), "#11");
  327. mre2.Set ();
  328. mre.WaitOne (3000);
  329. called += 11;
  330. });
  331. var t = Task.Factory.StartNew (() => { cts.Cancel (); });
  332. Assert.IsTrue (mre2.WaitOne (1000), "#0");
  333. cts.Token.Register (() => { called++; });
  334. Assert.AreEqual (1, called, "#1");
  335. Assert.IsFalse (t.IsCompleted, "#2");
  336. mre.Set ();
  337. Assert.IsTrue (t.Wait (1000), "#3");
  338. Assert.AreEqual (12, called, "#4");
  339. }
  340. [Test]
  341. public void ReEntrantRegistrationTest ()
  342. {
  343. bool unregister = false;
  344. bool register = false;
  345. var source = new CancellationTokenSource ();
  346. var token = source.Token;
  347. Console.WriteLine ("Test1");
  348. var reg = token.Register (() => unregister = true);
  349. token.Register (() => reg.Dispose ());
  350. token.Register (() => { Console.WriteLine ("Gnyah"); token.Register (() => register = true); });
  351. source.Cancel ();
  352. Assert.IsFalse (unregister);
  353. Assert.IsTrue (register);
  354. }
  355. [Test]
  356. public void DisposeAfterRegistrationTest ()
  357. {
  358. var source = new CancellationTokenSource ();
  359. bool ran = false;
  360. var req = source.Token.Register (() => ran = true);
  361. source.Dispose ();
  362. req.Dispose ();
  363. Assert.IsFalse (ran);
  364. }
  365. [Test]
  366. public void CancelLinkedTokenSource ()
  367. {
  368. var cts = new CancellationTokenSource ();
  369. bool canceled = false;
  370. cts.Token.Register (() => canceled = true);
  371. using (var linked = CancellationTokenSource.CreateLinkedTokenSource (cts.Token))
  372. ;
  373. Assert.IsFalse (canceled, "#1");
  374. Assert.IsFalse (cts.IsCancellationRequested, "#2");
  375. cts.Cancel ();
  376. Assert.IsTrue (canceled, "#3");
  377. }
  378. [Category ("NotWorking")] // why would linked token be imune to ObjectDisposedException on Cancel?
  379. [Test]
  380. public void ConcurrentCancelLinkedTokenSourceWhileDisposing ()
  381. {
  382. for (int i = 0, total = 500; i < total; ++i) {
  383. var src = new CancellationTokenSource ();
  384. var linked = CancellationTokenSource.CreateLinkedTokenSource (src.Token);
  385. var cntd = new CountdownEvent (2);
  386. var t1 = new Thread (() => {
  387. if (!cntd.Signal ())
  388. cntd.Wait (200);
  389. src.Cancel ();
  390. });
  391. var t2 = new Thread (() => {
  392. if (!cntd.Signal ())
  393. cntd.Wait (200);
  394. linked.Dispose ();
  395. });
  396. t1.Start ();
  397. t2.Start ();
  398. t1.Join (500);
  399. t2.Join (500);
  400. }
  401. }
  402. [Test]
  403. public void DisposeRace ()
  404. {
  405. for (int i = 0, total = 1000; i < total; ++i) {
  406. var c1 = new CancellationTokenSource ();
  407. var wh = c1.Token.WaitHandle;
  408. c1.CancelAfter (1);
  409. Thread.Sleep (1);
  410. c1.Dispose ();
  411. }
  412. }
  413. }
  414. }