CancellationTokenSourceTest.cs 11 KB

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