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 using NUnit.Framework.SyntaxHelpers;
43 namespace MonoTests.System.Threading.Tasks
46 public class TaskFactoryTests
48 class CompletedAsyncResult : IAsyncResult
50 public object AsyncState
52 get { throw new NotImplementedException (); }
55 public WaitHandle AsyncWaitHandle
57 get { throw new NotImplementedException (); }
60 public bool CompletedSynchronously
62 get { throw new NotImplementedException (); }
65 public bool IsCompleted
71 class TestAsyncResult : IAsyncResult
73 WaitHandle wh = new ManualResetEvent (true);
75 public object AsyncState
77 get { throw new NotImplementedException (); }
80 public WaitHandle AsyncWaitHandle
88 public bool CompletedSynchronously
90 get { throw new NotImplementedException (); }
93 public bool IsCompleted
99 class TestScheduler : TaskScheduler
101 public bool ExecutedInline { get; set; }
103 protected override void QueueTask (Task task)
105 throw new NotImplementedException ();
108 protected override bool TryDequeue (Task task)
110 throw new NotImplementedException ();
113 protected override bool TryExecuteTaskInline (Task task, bool taskWasPreviouslyQueued)
115 if (taskWasPreviouslyQueued)
116 throw new ArgumentException ("taskWasPreviouslyQueued");
118 if (task.Status != TaskStatus.WaitingToRun)
119 throw new ArgumentException ("task.Status");
121 ExecutedInline = true;
122 return TryExecuteTask (task);
125 protected override IEnumerable<Task> GetScheduledTasks ()
127 throw new NotImplementedException ();
137 this.factory = Task.Factory;
141 public void StartNewTest ()
144 factory.StartNew (() => result = true).Wait ();
145 Assert.IsTrue (result);
149 public void NoDefaultScheduler ()
151 Assert.IsNull (factory.Scheduler, "#1");
155 public void ContinueWhenAll_Simple ()
157 var mre = new ManualResetEventSlim (false);
159 Task[] tasks = new Task[3];
160 tasks[0] = new Task (() => { Thread.Sleep (0); Assert.IsTrue (mre.Wait (5000)); });
161 tasks[1] = new Task (() => { Assert.IsTrue (mre.Wait (5000)); });
162 tasks[2] = new Task (() => { Assert.IsTrue (mre.Wait (5000)); });
165 Task cont = factory.ContinueWhenAll (tasks, ts => {
166 Assert.AreEqual (tasks, ts, "#0");
170 foreach (Task t in tasks)
175 Assert.IsTrue (cont.Wait (3000), "#1");
176 Assert.IsTrue (ran, "#2");
180 public void ContinueWhenAll_WithMixedCompletionState ()
182 var mre = new ManualResetEventSlim ();
183 var task = Task.Factory.StartNew (() => mre.Wait (1000));
184 var contFailed = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnFaulted);
185 var contCanceled = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnCanceled);
186 var contSuccess = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnRanToCompletion);
189 var cont = Task.Factory.ContinueWhenAll (new Task[] { contFailed, contCanceled, contSuccess }, _ => ran = true);
195 Assert.AreEqual (TaskStatus.RanToCompletion, cont.Status);
199 public void ContinueWhenAll_InvalidArguments ()
202 factory.ContinueWhenAll (null, delegate { });
204 } catch (ArgumentNullException) {
208 factory.ContinueWhenAll (new Task[0], delegate { });
210 } catch (ArgumentException) {
214 factory.ContinueWhenAll (new Task[] { null }, delegate { });
216 } catch (ArgumentException) {
219 var tasks = new Task [] {
220 factory.StartNew (delegate {})
224 factory.ContinueWhenAll (tasks, null);
226 } catch (ArgumentNullException) {
230 factory.ContinueWhenAll (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.None, null);
232 } catch (ArgumentNullException) {
236 factory.ContinueWhenAll (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.OnlyOnCanceled, null);
238 } catch (ArgumentNullException) {
239 } catch (ArgumentOutOfRangeException) {
244 public void ContinueWhenAll_WithExceptions ()
246 var t1 = Task.Factory.StartNew (() => { throw new ApplicationException ("Foo"); });
247 var t2 = Task.Factory.StartNew (() => { throw new ApplicationException ("Bar"); });
249 var cont = Task.Factory.ContinueWhenAll (new[] { t1, t2 }, delegate {});
252 Assert.IsTrue (t1.IsFaulted);
253 Assert.IsTrue (t2.IsFaulted);
254 Assert.AreEqual (TaskStatus.RanToCompletion, cont.Status);
258 public void ContinueWhenAny_Simple ()
260 var t1 = new ManualResetEvent (false);
261 var t2 = new ManualResetEvent (false);
263 var tasks = new Task[2] {
264 Task.Factory.StartNew (() => { t1.WaitOne (5000); }),
265 Task.Factory.StartNew (() => { t2.WaitOne (5000); })
269 var ct = new CancellationToken ();
270 Task cont = factory.ContinueWhenAny (tasks, t => {
271 Assert.AreEqual (tasks[0], t, "#1");
275 Assert.AreEqual (TaskStatus.WaitingForActivation, cont.Status, "#2");
279 Assert.IsTrue (cont.Wait (3000), "#10");
280 Assert.IsTrue (ran, "#11");
286 public void ContinueWhenAny_WithResult ()
288 var tcs = new TaskCompletionSource<int>();
290 Task[] tasks = new[] { tcs.Task };
291 var res = Task.Factory.ContinueWhenAny (tasks, l => 4);
292 Assert.AreEqual (4, res.Result);
296 public void ContinueWhenAny_InvalidArguments ()
299 factory.ContinueWhenAny (null, delegate { });
301 } catch (ArgumentNullException) {
305 factory.ContinueWhenAny (new Task[0], delegate { });
307 } catch (ArgumentException) {
311 factory.ContinueWhenAny (new Task[] { null }, delegate { });
313 } catch (ArgumentException) {
316 var tasks = new Task [] {
317 factory.StartNew (delegate {})
321 factory.ContinueWhenAny (tasks, null);
323 } catch (ArgumentNullException) {
327 factory.ContinueWhenAny (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.None, null);
329 } catch (ArgumentNullException) {
333 factory.ContinueWhenAny (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.OnlyOnCanceled, null);
335 } catch (ArgumentNullException) {
336 } catch (ArgumentOutOfRangeException) {
341 public void FromAsyncBeginInvoke_WithResult ()
345 Func<int, int> func = (i) => {
346 Assert.IsTrue (Thread.CurrentThread.IsThreadPoolThread);
347 result = true; return i + 3;
350 var task = factory.FromAsync<int, int> (func.BeginInvoke, func.EndInvoke, 1, "state", TaskCreationOptions.AttachedToParent);
351 Assert.IsTrue (task.Wait (5000), "#1");
352 Assert.IsTrue (result, "#2");
353 Assert.AreEqual (4, task.Result, "#3");
354 Assert.AreEqual ("state", (string) task.AsyncState, "#4");
355 Assert.AreEqual (TaskCreationOptions.AttachedToParent, task.CreationOptions, "#5");
359 public void FromAsyncBeginMethod_DirectResult ()
362 bool continuationTest = false;
364 Func<int, int> func = (i) => { result = true; return i + 3; };
365 Task<int> task = factory.FromAsync<int> (func.BeginInvoke (1, delegate { }, null), func.EndInvoke);
366 var cont = task.ContinueWith (_ => continuationTest = true, TaskContinuationOptions.ExecuteSynchronously);
370 Assert.IsTrue (result);
371 Assert.IsTrue (continuationTest);
372 Assert.AreEqual (4, task.Result);
376 public void FromAsyncBeginMethod_Exception ()
379 bool continuationTest = false;
381 Func<int, int> func = (i) => { result = true; throw new ApplicationException ("bleh"); };
382 Task<int> task = factory.FromAsync<int, int> (func.BeginInvoke, func.EndInvoke, 1, null);
383 var cont = task.ContinueWith (_ => continuationTest = true, TaskContinuationOptions.ExecuteSynchronously);
387 Assert.IsTrue (cont.Wait (2000), "#1");
389 Assert.IsTrue (result);
390 Assert.IsTrue (continuationTest);
391 Assert.IsNotNull (task.Exception);
392 var agg = task.Exception;
393 Assert.AreEqual (1, agg.InnerExceptions.Count);
394 Assert.That (agg.InnerExceptions[0], Is.TypeOf (typeof (ApplicationException)));
395 Assert.AreEqual (TaskStatus.Faulted, task.Status);
400 } catch (AggregateException) {
405 public void FromAsync_ArgumentsCheck ()
407 var result = new CompletedAsyncResult ();
409 factory.FromAsync (null, l => { });
411 } catch (ArgumentNullException) {
415 factory.FromAsync (result, null);
417 } catch (ArgumentNullException) {
421 factory.FromAsync (result, l => { }, TaskCreationOptions.LongRunning);
423 } catch (ArgumentOutOfRangeException) {
427 factory.FromAsync (result, l => { }, TaskCreationOptions.PreferFairness);
429 } catch (ArgumentOutOfRangeException) {
433 factory.FromAsync (result, l => { }, TaskCreationOptions.None, null);
435 } catch (ArgumentNullException) {
439 factory.FromAsync (null, l => { }, null, TaskCreationOptions.None);
441 } catch (ArgumentNullException) {
445 factory.FromAsync ((a, b) => null, l => { }, null, TaskCreationOptions.LongRunning);
447 } catch (ArgumentOutOfRangeException) {
452 public void FromAsync_Completed ()
454 var completed = new CompletedAsyncResult ();
457 Action<IAsyncResult> end = l => {
458 Assert.IsFalse (Thread.CurrentThread.IsThreadPoolThread, "#2");
459 valid = l == completed;
461 Task task = factory.FromAsync (completed, end);
462 Assert.IsTrue (valid == true, "#1");
466 public void FromAsync_CompletedWithException ()
468 var completed = new CompletedAsyncResult ();
470 Action<IAsyncResult> end = l => {
471 throw new ApplicationException ();
473 Task task = factory.FromAsync (completed, end);
474 Assert.AreEqual (TaskStatus.Faulted, task.Status, "#1");
478 public void FromAsync_CompletedCanceled ()
480 var completed = new CompletedAsyncResult ();
482 Action<IAsyncResult> end = l => {
483 throw new OperationCanceledException ();
485 Task task = factory.FromAsync (completed, end);
486 Assert.AreEqual (TaskStatus.Canceled, task.Status, "#1");
487 Assert.IsNull (task.Exception, "#2");
491 public void FromAsync_SimpleAsyncResult ()
493 var result = new TestAsyncResult ();
496 var task = factory.FromAsync (result, l => {
500 Assert.IsTrue (task.Wait (1000), "#1");
501 Assert.IsTrue (called, "#2");
505 public void FromAsync_ResultException ()
507 var result = new TestAsyncResult ();
509 var task = factory.FromAsync (result, l => {
510 throw new ApplicationException ();
514 Assert.IsFalse (task.Wait (1000), "#1");
515 } catch (AggregateException) {
518 Assert.AreEqual (TaskStatus.Faulted, task.Status, "#2");
522 public void FromAsync_ReturnInt ()
524 var result = new TestAsyncResult ();
527 var task = factory.FromAsync<int> (result, l => {
532 Assert.IsTrue (task.Wait (1000), "#1");
533 Assert.IsTrue (called, "#2");
534 Assert.AreEqual (4, task.Result, "#3");
538 public void FromAsync_Scheduler_Explicit ()
540 var result = new TestAsyncResult ();
542 var scheduler = new TestScheduler ();
544 var task = factory.FromAsync (result, l => {
546 }, TaskCreationOptions.None, scheduler);
548 Assert.IsTrue (task.Wait (5000), "#1");
549 Assert.IsTrue (called, "#2");
550 Assert.IsTrue (scheduler.ExecutedInline, "#3");
554 public void FromAsync_Scheduler_Implicit ()
556 var result = new TestAsyncResult ();
558 var scheduler = new TestScheduler ();
560 factory = new TaskFactory (scheduler);
562 Task task = factory.FromAsync (result, l => {
563 Assert.IsTrue (Thread.CurrentThread.IsThreadPoolThread, "#6");
565 }, TaskCreationOptions.AttachedToParent);
567 Assert.AreEqual (TaskCreationOptions.AttachedToParent, task.CreationOptions, "#1");
568 Assert.IsNull (task.AsyncState, "#2");
569 Assert.IsTrue (task.Wait (5000), "#3");
570 Assert.IsTrue (called, "#4");
571 Assert.IsTrue (scheduler.ExecutedInline, "#5");
575 public void FromAsync_BeginCallback ()
578 bool called2 = false;
580 var task = factory.FromAsync (
585 if ((TaskCreationOptions) c != TaskCreationOptions.AttachedToParent)
588 Assert.IsFalse (Thread.CurrentThread.IsThreadPoolThread, "#12");
597 "h", TaskCreationOptions.AttachedToParent);
599 Assert.AreEqual (TaskCreationOptions.None, task.CreationOptions, "#1");
600 Assert.AreEqual (TaskCreationOptions.AttachedToParent, (TaskCreationOptions) task.AsyncState, "#2");
601 Assert.IsTrue (task.Wait (5000), "#3");
602 Assert.IsTrue (called, "#4");
603 Assert.IsTrue (called2, "#5");
607 public void StartNewCancelled ()
609 var ct = new CancellationToken (true);
611 var task = factory.StartNew (() => Assert.Fail ("Should never be called"), ct);
615 } catch (InvalidOperationException) {
621 } catch (AggregateException e) {
622 Assert.That (e.InnerException, Is.TypeOf (typeof (TaskCanceledException)), "#3");
625 Assert.IsTrue (task.IsCanceled, "#4");