CancellationTokenSourceTest.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491
  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. try {
  274. token.Register (() => { });
  275. Assert.Fail ("#3");
  276. } catch (ObjectDisposedException) {
  277. }
  278. try {
  279. var wh = token.WaitHandle;
  280. Assert.Fail ("#4");
  281. } catch (ObjectDisposedException) {
  282. }
  283. try {
  284. CancellationTokenSource.CreateLinkedTokenSource (token);
  285. Assert.Fail ("#5");
  286. } catch (ObjectDisposedException) {
  287. }
  288. try {
  289. cts.CancelAfter (1);
  290. Assert.Fail ("#6");
  291. } catch (ObjectDisposedException) {
  292. }
  293. }
  294. [Test]
  295. public void RegisterThenDispose ()
  296. {
  297. var cts1 = new CancellationTokenSource ();
  298. var reg1 = cts1.Token.Register (() => { throw new ApplicationException (); });
  299. var cts2 = new CancellationTokenSource ();
  300. var reg2 = cts2.Token.Register (() => { throw new ApplicationException (); });
  301. Assert.AreNotEqual (cts1, cts2, "#1");
  302. Assert.AreNotSame (cts1, cts2, "#2");
  303. reg1.Dispose ();
  304. cts1.Cancel ();
  305. try {
  306. cts2.Cancel ();
  307. Assert.Fail ("#3");
  308. } catch (AggregateException) {
  309. }
  310. }
  311. [Test]
  312. public void RegisterWhileCancelling ()
  313. {
  314. var cts = new CancellationTokenSource ();
  315. var mre = new ManualResetEvent (false);
  316. var mre2 = new ManualResetEvent (false);
  317. int called = 0;
  318. cts.Token.Register (() => {
  319. Assert.IsTrue (cts.IsCancellationRequested, "#10");
  320. Assert.IsTrue (cts.Token.WaitHandle.WaitOne (0), "#11");
  321. mre2.Set ();
  322. mre.WaitOne (3000);
  323. called += 11;
  324. });
  325. var t = Task.Factory.StartNew (() => { cts.Cancel (); });
  326. Assert.IsTrue (mre2.WaitOne (1000), "#0");
  327. cts.Token.Register (() => { called++; });
  328. Assert.AreEqual (1, called, "#1");
  329. Assert.IsFalse (t.IsCompleted, "#2");
  330. mre.Set ();
  331. Assert.IsTrue (t.Wait (1000), "#3");
  332. Assert.AreEqual (12, called, "#4");
  333. }
  334. [Test]
  335. public void ReEntrantRegistrationTest ()
  336. {
  337. bool unregister = false;
  338. bool register = false;
  339. var source = new CancellationTokenSource ();
  340. var token = source.Token;
  341. Console.WriteLine ("Test1");
  342. var reg = token.Register (() => unregister = true);
  343. token.Register (() => reg.Dispose ());
  344. token.Register (() => { Console.WriteLine ("Gnyah"); token.Register (() => register = true); });
  345. source.Cancel ();
  346. Assert.IsFalse (unregister);
  347. Assert.IsTrue (register);
  348. }
  349. [Test]
  350. public void DisposeAfterRegistrationTest ()
  351. {
  352. var source = new CancellationTokenSource ();
  353. bool ran = false;
  354. var req = source.Token.Register (() => ran = true);
  355. source.Dispose ();
  356. req.Dispose ();
  357. Assert.IsFalse (ran);
  358. }
  359. [Test]
  360. public void CancelLinkedTokenSource ()
  361. {
  362. var cts = new CancellationTokenSource ();
  363. bool canceled = false;
  364. cts.Token.Register (() => canceled = true);
  365. using (var linked = CancellationTokenSource.CreateLinkedTokenSource (cts.Token))
  366. ;
  367. Assert.IsFalse (canceled, "#1");
  368. Assert.IsFalse (cts.IsCancellationRequested, "#2");
  369. cts.Cancel ();
  370. Assert.IsTrue (canceled, "#3");
  371. }
  372. [Category ("NotWorking")] // why would linked token be imune to ObjectDisposedException on Cancel?
  373. [Test]
  374. public void ConcurrentCancelLinkedTokenSourceWhileDisposing ()
  375. {
  376. for (int i = 0, total = 500; i < total; ++i) {
  377. var src = new CancellationTokenSource ();
  378. var linked = CancellationTokenSource.CreateLinkedTokenSource (src.Token);
  379. var cntd = new CountdownEvent (2);
  380. var t1 = new Thread (() => {
  381. if (!cntd.Signal ())
  382. cntd.Wait (200);
  383. src.Cancel ();
  384. });
  385. var t2 = new Thread (() => {
  386. if (!cntd.Signal ())
  387. cntd.Wait (200);
  388. linked.Dispose ();
  389. });
  390. t1.Start ();
  391. t2.Start ();
  392. t1.Join (500);
  393. t2.Join (500);
  394. }
  395. }
  396. [Test]
  397. public void DisposeRace ()
  398. {
  399. for (int i = 0, total = 1000; i < total; ++i) {
  400. var c1 = new CancellationTokenSource ();
  401. var wh = c1.Token.WaitHandle;
  402. c1.CancelAfter (1);
  403. Thread.Sleep (1);
  404. c1.Dispose ();
  405. }
  406. }
  407. }
  408. }