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. #if NET_4_5
  39. [Test]
  40. public void Ctor_Invalid ()
  41. {
  42. try {
  43. new CancellationTokenSource (-4);
  44. Assert.Fail ("#1");
  45. } catch (ArgumentException) {
  46. }
  47. }
  48. [Test]
  49. public void Ctor_Timeout ()
  50. {
  51. int called = 0;
  52. var cts = new CancellationTokenSource (TimeSpan.FromMilliseconds (20));
  53. cts.Token.Register (() => called++);
  54. Thread.Sleep (50);
  55. Assert.AreEqual (1, called, "#1");
  56. }
  57. [Test]
  58. public void CancelAfter ()
  59. {
  60. int called = 0;
  61. var cts = new CancellationTokenSource ();
  62. cts.Token.Register (() => called++);
  63. cts.CancelAfter (20);
  64. Thread.Sleep (50);
  65. Assert.AreEqual (1, called, "#1");
  66. }
  67. [Test]
  68. public void CancelAfter_Invalid ()
  69. {
  70. var cts = new CancellationTokenSource ();
  71. try {
  72. cts.CancelAfter (-9);
  73. Assert.Fail ("#1");
  74. } catch (ArgumentException) {
  75. }
  76. }
  77. [Test]
  78. public void CancelAfter_Disposed ()
  79. {
  80. int called = 0;
  81. var cts = new CancellationTokenSource ();
  82. cts.Token.Register (() => called++);
  83. cts.CancelAfter (50);
  84. cts.Dispose ();
  85. Thread.Sleep (100);
  86. Assert.AreEqual (0, called, "#1");
  87. }
  88. #endif
  89. [Test]
  90. public void Token ()
  91. {
  92. CancellationTokenSource cts = new CancellationTokenSource ();
  93. Assert.IsTrue (cts.Token.CanBeCanceled, "#1");
  94. Assert.IsFalse (cts.Token.IsCancellationRequested, "#2");
  95. Assert.IsNotNull (cts.Token.WaitHandle, "#3");
  96. }
  97. [Test]
  98. public void Cancel_NoRegistration ()
  99. {
  100. CancellationTokenSource cts = new CancellationTokenSource ();
  101. cts.Cancel ();
  102. }
  103. [Test]
  104. public void Cancel ()
  105. {
  106. var cts = new CancellationTokenSource ();
  107. int called = 0;
  108. cts.Token.Register (l => { Assert.AreEqual ("v", l); ++called; }, "v");
  109. cts.Cancel ();
  110. Assert.AreEqual (1, called, "#1");
  111. called = 0;
  112. cts.Token.Register (() => { called += 12; });
  113. cts.Cancel ();
  114. Assert.AreEqual (12, called, "#2");
  115. }
  116. [Test]
  117. public void Cancel_Order ()
  118. {
  119. var cts = new CancellationTokenSource ();
  120. var current = 0;
  121. Action<object> a = x => { Assert.AreEqual(current, x); current++; };
  122. cts.Token.Register (a, 2);
  123. cts.Token.Register (a, 1);
  124. cts.Token.Register (a, 0);
  125. cts.Cancel ();
  126. }
  127. [Test]
  128. public void CancelWithDispose ()
  129. {
  130. CancellationTokenSource cts = new CancellationTokenSource ();
  131. CancellationToken c = cts.Token;
  132. c.Register (() => {
  133. cts.Dispose ();
  134. });
  135. int called = 0;
  136. c.Register (() => {
  137. called++;
  138. });
  139. cts.Cancel ();
  140. Assert.AreEqual (1, called, "#1");
  141. }
  142. [Test]
  143. public void Cancel_SingleException ()
  144. {
  145. var cts = new CancellationTokenSource ();
  146. cts.Token.Register (() => { throw new ApplicationException (); });
  147. try {
  148. cts.Cancel ();
  149. Assert.Fail ("#1");
  150. } catch (AggregateException e) {
  151. Assert.AreEqual (1, e.InnerExceptions.Count, "#2");
  152. }
  153. cts.Cancel ();
  154. }
  155. [Test]
  156. public void Cancel_MultipleExceptions ()
  157. {
  158. var cts = new CancellationTokenSource ();
  159. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  160. cts.Token.Register (() => { throw new ApplicationException ("2"); });
  161. cts.Token.Register (() => { throw new ApplicationException ("3"); });
  162. try {
  163. cts.Cancel ();
  164. Assert.Fail ("#1");
  165. } catch (AggregateException e) {
  166. Assert.AreEqual (3, e.InnerExceptions.Count, "#2");
  167. }
  168. cts.Cancel ();
  169. try {
  170. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  171. Assert.Fail ("#11");
  172. } catch (ApplicationException) {
  173. }
  174. cts.Cancel ();
  175. }
  176. [Test]
  177. public void Cancel_ExceptionOrder ()
  178. {
  179. var cts = new CancellationTokenSource ();
  180. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  181. cts.Token.Register (() => { throw new ApplicationException ("2"); });
  182. cts.Token.Register (() => { throw new ApplicationException ("3"); });
  183. try {
  184. cts.Cancel ();
  185. } catch (AggregateException e) {
  186. Assert.AreEqual (3, e.InnerExceptions.Count, "#2");
  187. Assert.AreEqual ("3", e.InnerExceptions[0].Message, "#3");
  188. Assert.AreEqual ("2", e.InnerExceptions[1].Message, "#4");
  189. Assert.AreEqual ("1", e.InnerExceptions[2].Message, "#5");
  190. }
  191. }
  192. [Test]
  193. public void Cancel_MultipleException_Recursive ()
  194. {
  195. CancellationTokenSource cts = new CancellationTokenSource ();
  196. CancellationToken c = cts.Token;
  197. c.Register (() => {
  198. cts.Cancel ();
  199. });
  200. c.Register (() => {
  201. throw new ApplicationException ();
  202. });
  203. c.Register (() => {
  204. throw new NotSupportedException ();
  205. });
  206. try {
  207. cts.Cancel (false);
  208. Assert.Fail ("#1");
  209. } catch (AggregateException e) {
  210. Assert.AreEqual (2, e.InnerExceptions.Count, "#2");
  211. }
  212. }
  213. [Test]
  214. public void Cancel_MultipleExceptionsFirstThrows ()
  215. {
  216. var cts = new CancellationTokenSource ();
  217. cts.Token.Register (() => { throw new ApplicationException ("1"); });
  218. cts.Token.Register (() => { throw new ApplicationException ("2"); });
  219. cts.Token.Register (() => { throw new ApplicationException ("3"); });
  220. try {
  221. cts.Cancel (true);
  222. Assert.Fail ("#1");
  223. } catch (ApplicationException) {
  224. }
  225. cts.Cancel ();
  226. }
  227. [Test]
  228. public void CreateLinkedTokenSource_InvalidArguments ()
  229. {
  230. var cts = new CancellationTokenSource ();
  231. var token = cts.Token;
  232. try {
  233. CancellationTokenSource.CreateLinkedTokenSource (null);
  234. Assert.Fail ("#1");
  235. } catch (ArgumentNullException) {
  236. }
  237. try {
  238. CancellationTokenSource.CreateLinkedTokenSource (new CancellationToken[0]);
  239. Assert.Fail ("#2");
  240. } catch (ArgumentException) {
  241. }
  242. }
  243. [Test]
  244. public void CreateLinkedTokenSource ()
  245. {
  246. var cts = new CancellationTokenSource ();
  247. cts.Cancel ();
  248. var linked = CancellationTokenSource.CreateLinkedTokenSource (cts.Token);
  249. Assert.IsTrue (linked.IsCancellationRequested, "#1");
  250. linked = CancellationTokenSource.CreateLinkedTokenSource (new CancellationToken ());
  251. Assert.IsFalse (linked.IsCancellationRequested, "#2");
  252. }
  253. [Test]
  254. public void Dispose ()
  255. {
  256. var cts = new CancellationTokenSource ();
  257. var token = cts.Token;
  258. cts.Dispose ();
  259. cts.Dispose ();
  260. var b = cts.IsCancellationRequested;
  261. token.ThrowIfCancellationRequested ();
  262. try {
  263. cts.Cancel ();
  264. Assert.Fail ("#1");
  265. } catch (ObjectDisposedException) {
  266. }
  267. try {
  268. var t = cts.Token;
  269. Assert.Fail ("#2");
  270. } catch (ObjectDisposedException) {
  271. }
  272. try {
  273. token.Register (() => { });
  274. Assert.Fail ("#3");
  275. } catch (ObjectDisposedException) {
  276. }
  277. try {
  278. var wh = token.WaitHandle;
  279. Assert.Fail ("#4");
  280. } catch (ObjectDisposedException) {
  281. }
  282. try {
  283. CancellationTokenSource.CreateLinkedTokenSource (token);
  284. Assert.Fail ("#5");
  285. } catch (ObjectDisposedException) {
  286. }
  287. #if NET_4_5
  288. try {
  289. cts.CancelAfter (1);
  290. Assert.Fail ("#6");
  291. } catch (ObjectDisposedException) {
  292. }
  293. #endif
  294. }
  295. [Test]
  296. public void RegisterThenDispose ()
  297. {
  298. var cts1 = new CancellationTokenSource ();
  299. var reg1 = cts1.Token.Register (() => { throw new ApplicationException (); });
  300. var cts2 = new CancellationTokenSource ();
  301. var reg2 = cts2.Token.Register (() => { throw new ApplicationException (); });
  302. Assert.AreNotEqual (cts1, cts2, "#1");
  303. Assert.AreNotSame (cts1, cts2, "#2");
  304. reg1.Dispose ();
  305. cts1.Cancel ();
  306. try {
  307. cts2.Cancel ();
  308. Assert.Fail ("#3");
  309. } catch (AggregateException) {
  310. }
  311. }
  312. [Test]
  313. public void RegisterWhileCancelling ()
  314. {
  315. var cts = new CancellationTokenSource ();
  316. var mre = new ManualResetEvent (false);
  317. var mre2 = new ManualResetEvent (false);
  318. int called = 0;
  319. cts.Token.Register (() => {
  320. Assert.IsTrue (cts.IsCancellationRequested, "#10");
  321. Assert.IsTrue (cts.Token.WaitHandle.WaitOne (0), "#11");
  322. mre2.Set ();
  323. mre.WaitOne (3000);
  324. called += 11;
  325. });
  326. var t = Task.Factory.StartNew (() => { cts.Cancel (); });
  327. Assert.IsTrue (mre2.WaitOne (1000), "#0");
  328. cts.Token.Register (() => { called++; });
  329. Assert.AreEqual (1, called, "#1");
  330. Assert.IsFalse (t.IsCompleted, "#2");
  331. mre.Set ();
  332. Assert.IsTrue (t.Wait (1000), "#3");
  333. Assert.AreEqual (12, called, "#4");
  334. }
  335. [Test]
  336. public void ReEntrantRegistrationTest ()
  337. {
  338. bool unregister = false;
  339. bool register = false;
  340. var source = new CancellationTokenSource ();
  341. var token = source.Token;
  342. Console.WriteLine ("Test1");
  343. var reg = token.Register (() => unregister = true);
  344. token.Register (() => reg.Dispose ());
  345. token.Register (() => { Console.WriteLine ("Gnyah"); token.Register (() => register = true); });
  346. source.Cancel ();
  347. Assert.IsFalse (unregister);
  348. Assert.IsTrue (register);
  349. }
  350. [Test]
  351. public void DisposeAfterRegistrationTest ()
  352. {
  353. var source = new CancellationTokenSource ();
  354. bool ran = false;
  355. var req = source.Token.Register (() => ran = true);
  356. source.Dispose ();
  357. req.Dispose ();
  358. Assert.IsFalse (ran);
  359. }
  360. [Test]
  361. public void CancelLinkedTokenSource ()
  362. {
  363. var cts = new CancellationTokenSource ();
  364. bool canceled = false;
  365. cts.Token.Register (() => canceled = true);
  366. using (var linked = CancellationTokenSource.CreateLinkedTokenSource (cts.Token))
  367. ;
  368. Assert.IsFalse (canceled, "#1");
  369. Assert.IsFalse (cts.IsCancellationRequested, "#2");
  370. cts.Cancel ();
  371. Assert.IsTrue (canceled, "#3");
  372. }
  373. [Category ("NotWorking")] // why would linked token be imune to ObjectDisposedException on Cancel?
  374. [Test]
  375. public void ConcurrentCancelLinkedTokenSourceWhileDisposing ()
  376. {
  377. for (int i = 0, total = 500; i < total; ++i) {
  378. var src = new CancellationTokenSource ();
  379. var linked = CancellationTokenSource.CreateLinkedTokenSource (src.Token);
  380. var cntd = new CountdownEvent (2);
  381. var t1 = new Thread (() => {
  382. if (!cntd.Signal ())
  383. cntd.Wait (200);
  384. src.Cancel ();
  385. });
  386. var t2 = new Thread (() => {
  387. if (!cntd.Signal ())
  388. cntd.Wait (200);
  389. linked.Dispose ();
  390. });
  391. t1.Start ();
  392. t2.Start ();
  393. t1.Join (500);
  394. t2.Join (500);
  395. }
  396. }
  397. #if NET_4_5
  398. [Test]
  399. public void DisposeRace ()
  400. {
  401. for (int i = 0, total = 1000; i < total; ++i) {
  402. var c1 = new CancellationTokenSource ();
  403. var wh = c1.Token.WaitHandle;
  404. c1.CancelAfter (1);
  405. Thread.Sleep (1);
  406. c1.Dispose ();
  407. }
  408. }
  409. #endif
  410. }
  411. }