5 // Jérémie "Garuma" Laval <jeremie.laval@gmail.com>
6 // Marek Safar <marek.safar@gmail.com>
8 // Copyright (c) 2010 Jérémie "Garuma" Laval
9 // Copyright 2011 Xamarin, Inc (http://www.xamarin.com)
11 // Permission is hereby granted, free of charge, to any person obtaining a copy
12 // of this software and associated documentation files (the "Software"), to deal
13 // in the Software without restriction, including without limitation the rights
14 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
15 // copies of the Software, and to permit persons to whom the Software is
16 // furnished to do so, subject to the following conditions:
18 // The above copyright notice and this permission notice shall be included in
19 // all copies or substantial portions of the Software.
21 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
24 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
26 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
34 using System.Threading;
35 using System.Threading.Tasks;
36 using System.Collections.Generic;
38 using NUnit.Framework;
40 namespace MonoTests.System.Threading.Tasks
43 public class TaskFactoryTests
45 class CompletedAsyncResult : IAsyncResult
47 public object AsyncState
49 get { throw new NotImplementedException (); }
52 public WaitHandle AsyncWaitHandle
54 get { throw new NotImplementedException (); }
57 public bool CompletedSynchronously
59 get { throw new NotImplementedException (); }
62 public bool IsCompleted
68 class TestAsyncResult : IAsyncResult
70 WaitHandle wh = new ManualResetEvent (true);
72 public object AsyncState
74 get { throw new NotImplementedException (); }
77 public WaitHandle AsyncWaitHandle
85 public bool CompletedSynchronously
87 get { throw new NotImplementedException (); }
90 public bool IsCompleted
96 class TestScheduler : TaskScheduler
98 public bool ExecutedInline { get; set; }
100 protected override void QueueTask (Task task)
102 throw new NotImplementedException ();
105 protected override bool TryDequeue (Task task)
107 throw new NotImplementedException ();
110 protected override bool TryExecuteTaskInline (Task task, bool taskWasPreviouslyQueued)
112 if (taskWasPreviouslyQueued)
113 throw new ArgumentException ("taskWasPreviouslyQueued");
115 if (task.Status != TaskStatus.WaitingToRun)
116 throw new ArgumentException ("task.Status");
118 ExecutedInline = true;
119 return TryExecuteTask (task);
122 protected override IEnumerable<Task> GetScheduledTasks ()
124 throw new NotImplementedException ();
134 this.factory = Task.Factory;
138 public void StartNewTest ()
141 factory.StartNew (() => result = true).Wait ();
142 Assert.IsTrue (result);
146 public void NoDefaultScheduler ()
148 Assert.IsNull (factory.Scheduler, "#1");
152 public void ContinueWhenAll_Simple ()
154 var mre = new ManualResetEventSlim (false);
156 Task[] tasks = new Task[3];
157 tasks[0] = new Task (() => { Thread.Sleep (0); Assert.IsTrue (mre.Wait (3000)); });
158 tasks[1] = new Task (() => { Assert.IsTrue (mre.Wait (3000)); });
159 tasks[2] = new Task (() => { Assert.IsTrue (mre.Wait (3000)); });
162 Task cont = factory.ContinueWhenAll (tasks, ts => {
163 Assert.AreEqual (tasks, ts, "#0");
167 foreach (Task t in tasks)
172 Assert.IsTrue (cont.Wait (1000), "#1");
173 Assert.IsTrue (ran, "#2");
177 public void ContinueWhenAll_WithMixedCompletionState ()
179 var mre = new ManualResetEventSlim ();
180 var task = Task.Factory.StartNew (() => mre.Wait (200));
181 var contFailed = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnFaulted);
182 var contCanceled = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnCanceled);
183 var contSuccess = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnRanToCompletion);
186 var cont = Task.Factory.ContinueWhenAll (new Task[] { contFailed, contCanceled, contSuccess }, _ => ran = true);
192 Assert.AreEqual (TaskStatus.RanToCompletion, cont.Status);
196 public void ContinueWhenAll_InvalidArguments ()
199 factory.ContinueWhenAll (null, delegate { });
201 } catch (ArgumentNullException) {
205 factory.ContinueWhenAll (new Task[0], delegate { });
207 } catch (ArgumentException) {
211 factory.ContinueWhenAll (new Task[] { null }, delegate { });
213 } catch (ArgumentException) {
216 var tasks = new Task [] {
217 factory.StartNew (delegate {})
221 factory.ContinueWhenAll (tasks, null);
223 } catch (ArgumentException) {
227 factory.ContinueWhenAll (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.None, null);
229 } catch (ArgumentException) {
233 factory.ContinueWhenAll (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.OnlyOnCanceled, null);
235 } catch (ArgumentException) {
240 public void ContinueWhenAll_WithExceptions ()
242 var t1 = Task.Factory.StartNew (() => { throw new ApplicationException ("Foo"); });
243 var t2 = Task.Factory.StartNew (() => { throw new ApplicationException ("Bar"); });
245 var cont = Task.Factory.ContinueWhenAll (new[] { t1, t2 }, delegate {});
248 Assert.IsTrue (t1.IsFaulted);
249 Assert.IsTrue (t2.IsFaulted);
250 Assert.AreEqual (TaskStatus.RanToCompletion, cont.Status);
254 public void ContinueWhenAny_Simple ()
256 var t1 = new ManualResetEvent (false);
257 var t2 = new ManualResetEvent (false);
259 var tasks = new Task[2] {
260 Task.Factory.StartNew (() => { t1.WaitOne (3000); }),
261 Task.Factory.StartNew (() => { t2.WaitOne (3000); })
265 var ct = new CancellationToken ();
266 Task cont = factory.ContinueWhenAny (tasks, t => {
267 Assert.AreEqual (tasks[0], t, "#1");
271 Assert.AreEqual (TaskStatus.WaitingForActivation, cont.Status, "#2");
275 Assert.IsTrue (cont.Wait (2000), "#10");
276 Assert.IsTrue (ran, "#11");
282 public void ContinueWhenAny_InvalidArguments ()
285 factory.ContinueWhenAny (null, delegate { });
287 } catch (ArgumentNullException) {
291 factory.ContinueWhenAny (new Task[0], delegate { });
293 } catch (ArgumentException) {
297 factory.ContinueWhenAny (new Task[] { null }, delegate { });
299 } catch (ArgumentException) {
302 var tasks = new Task [] {
303 factory.StartNew (delegate {})
307 factory.ContinueWhenAny (tasks, null);
309 } catch (ArgumentException) {
313 factory.ContinueWhenAny (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.None, null);
315 } catch (ArgumentException) {
319 factory.ContinueWhenAny (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.OnlyOnCanceled, null);
321 } catch (ArgumentException) {
326 public void FromAsyncBeginInvoke_WithResult ()
330 Func<int, int> func = (i) => {
331 Assert.IsTrue (Thread.CurrentThread.IsThreadPoolThread);
332 result = true; return i + 3;
335 var task = factory.FromAsync<int, int> (func.BeginInvoke, func.EndInvoke, 1, "state", TaskCreationOptions.AttachedToParent);
336 Assert.IsTrue (task.Wait (5000), "#1");
337 Assert.IsTrue (result, "#2");
338 Assert.AreEqual (4, task.Result, "#3");
339 Assert.AreEqual ("state", (string) task.AsyncState, "#4");
340 Assert.AreEqual (TaskCreationOptions.AttachedToParent, task.CreationOptions, "#5");
344 public void FromAsyncBeginMethod_DirectResult ()
347 bool continuationTest = false;
349 Func<int, int> func = (i) => { result = true; return i + 3; };
350 Task<int> task = factory.FromAsync<int> (func.BeginInvoke (1, delegate { }, null), func.EndInvoke);
351 var cont = task.ContinueWith (_ => continuationTest = true, TaskContinuationOptions.ExecuteSynchronously);
355 Assert.IsTrue (result);
356 Assert.IsTrue (continuationTest);
357 Assert.AreEqual (4, task.Result);
361 public void FromAsyncBeginMethod_Exception ()
364 bool continuationTest = false;
366 Func<int, int> func = (i) => { result = true; throw new ApplicationException ("bleh"); };
367 Task<int> task = factory.FromAsync<int, int> (func.BeginInvoke, func.EndInvoke, 1, null);
368 var cont = task.ContinueWith (_ => continuationTest = true, TaskContinuationOptions.ExecuteSynchronously);
374 Assert.IsTrue (result);
375 Assert.IsTrue (continuationTest);
376 Assert.IsNotNull (task.Exception);
377 var agg = task.Exception;
378 Assert.AreEqual (1, agg.InnerExceptions.Count);
379 Assert.IsInstanceOfType (typeof (ApplicationException), agg.InnerExceptions[0]);
380 Assert.AreEqual (TaskStatus.Faulted, task.Status);
385 } catch (AggregateException) {
390 public void FromAsync_ArgumentsCheck ()
392 var result = new CompletedAsyncResult ();
394 factory.FromAsync (null, l => { });
396 } catch (ArgumentNullException) {
400 factory.FromAsync (result, null);
402 } catch (ArgumentNullException) {
406 factory.FromAsync (result, l => { }, TaskCreationOptions.LongRunning);
408 } catch (ArgumentOutOfRangeException) {
412 factory.FromAsync (result, l => { }, TaskCreationOptions.PreferFairness);
414 } catch (ArgumentOutOfRangeException) {
418 factory.FromAsync (result, l => { }, TaskCreationOptions.None, null);
420 } catch (ArgumentNullException) {
424 factory.FromAsync (null, l => { }, null, TaskCreationOptions.None);
426 } catch (ArgumentNullException) {
430 factory.FromAsync ((a, b) => null, l => { }, null, TaskCreationOptions.LongRunning);
432 } catch (ArgumentOutOfRangeException) {
437 public void FromAsync_Completed ()
439 var completed = new CompletedAsyncResult ();
442 Action<IAsyncResult> end = l => {
443 Assert.IsFalse (Thread.CurrentThread.IsThreadPoolThread, "#2");
444 valid = l == completed;
446 Task task = factory.FromAsync (completed, end);
447 Assert.IsTrue (valid == true, "#1");
451 public void FromAsync_CompletedWithException ()
453 var completed = new CompletedAsyncResult ();
455 Action<IAsyncResult> end = l => {
456 throw new ApplicationException ();
458 Task task = factory.FromAsync (completed, end);
459 Assert.AreEqual (TaskStatus.Faulted, task.Status, "#1");
463 public void FromAsync_CompletedCanceled ()
465 var completed = new CompletedAsyncResult ();
467 Action<IAsyncResult> end = l => {
468 throw new OperationCanceledException ();
470 Task task = factory.FromAsync (completed, end);
471 Assert.AreEqual (TaskStatus.Canceled, task.Status, "#1");
472 Assert.IsNull (task.Exception, "#2");
476 public void FromAsync_SimpleAsyncResult ()
478 var result = new TestAsyncResult ();
481 var task = factory.FromAsync (result, l => {
485 Assert.IsTrue (task.Wait (1000), "#1");
486 Assert.IsTrue (called, "#2");
490 public void FromAsync_ResultException ()
492 var result = new TestAsyncResult ();
494 var task = factory.FromAsync (result, l => {
495 throw new ApplicationException ();
499 Assert.IsFalse (task.Wait (1000), "#1");
500 } catch (AggregateException) {
503 Assert.AreEqual (TaskStatus.Faulted, task.Status, "#2");
507 public void FromAsync_ReturnInt ()
509 var result = new TestAsyncResult ();
512 var task = factory.FromAsync<int> (result, l => {
517 Assert.IsTrue (task.Wait (1000), "#1");
518 Assert.IsTrue (called, "#2");
519 Assert.AreEqual (4, task.Result, "#3");
523 public void FromAsync_Scheduler_Explicit ()
525 var result = new TestAsyncResult ();
527 var scheduler = new TestScheduler ();
529 var task = factory.FromAsync (result, l => {
531 }, TaskCreationOptions.None, scheduler);
533 Assert.IsTrue (task.Wait (5000), "#1");
534 Assert.IsTrue (called, "#2");
535 Assert.IsTrue (scheduler.ExecutedInline, "#3");
539 public void FromAsync_Scheduler_Implicit ()
541 var result = new TestAsyncResult ();
543 var scheduler = new TestScheduler ();
545 factory = new TaskFactory (scheduler);
547 Task task = factory.FromAsync (result, l => {
548 Assert.IsTrue (Thread.CurrentThread.IsThreadPoolThread, "#6");
550 }, TaskCreationOptions.AttachedToParent);
552 Assert.AreEqual (TaskCreationOptions.AttachedToParent, task.CreationOptions, "#1");
553 Assert.IsNull (task.AsyncState, "#2");
554 Assert.IsTrue (task.Wait (5000), "#3");
555 Assert.IsTrue (called, "#4");
556 Assert.IsTrue (scheduler.ExecutedInline, "#5");
560 public void FromAsync_BeginCallback ()
563 bool called2 = false;
565 var task = factory.FromAsync (
570 if ((TaskCreationOptions) c != TaskCreationOptions.AttachedToParent)
573 Assert.IsFalse (Thread.CurrentThread.IsThreadPoolThread, "#12");
582 "h", TaskCreationOptions.AttachedToParent);
584 Assert.AreEqual (TaskCreationOptions.None, task.CreationOptions, "#1");
585 Assert.AreEqual (TaskCreationOptions.AttachedToParent, (TaskCreationOptions) task.AsyncState, "#2");
586 Assert.IsTrue (task.Wait (5000), "#3");
587 Assert.IsTrue (called, "#4");
588 Assert.IsTrue (called2, "#5");
592 public void StartNewCancelled ()
594 var cts = new CancellationTokenSource ();
597 var task = factory.StartNew (() => Assert.Fail ("Should never be called"), cts.Token);
600 } catch (InvalidOperationException) {
603 Assert.IsTrue (task.IsCanceled, "#2");