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
33 using System.Threading;
34 using System.Threading.Tasks;
35 using System.Collections.Generic;
37 using NUnit.Framework;
39 using NUnit.Framework.SyntaxHelpers;
42 namespace MonoTests.System.Threading.Tasks
45 public class TaskFactoryTests
47 class CompletedAsyncResult : IAsyncResult
49 public object AsyncState
51 get { throw new NotImplementedException (); }
54 public WaitHandle AsyncWaitHandle
56 get { throw new NotImplementedException (); }
59 public bool CompletedSynchronously
61 get { throw new NotImplementedException (); }
64 public bool IsCompleted
70 class TestAsyncResult : IAsyncResult
72 WaitHandle wh = new ManualResetEvent (true);
74 public object AsyncState
76 get { throw new NotImplementedException (); }
79 public WaitHandle AsyncWaitHandle
87 public bool CompletedSynchronously
89 get { throw new NotImplementedException (); }
92 public bool IsCompleted
98 class TestScheduler : TaskScheduler
100 public bool ExecutedInline { get; set; }
102 protected override void QueueTask (Task task)
104 throw new NotImplementedException ();
107 protected override bool TryDequeue (Task task)
109 throw new NotImplementedException ();
112 protected override bool TryExecuteTaskInline (Task task, bool taskWasPreviouslyQueued)
114 if (taskWasPreviouslyQueued)
115 throw new ArgumentException ("taskWasPreviouslyQueued");
117 if (task.Status != TaskStatus.WaitingToRun)
118 throw new ArgumentException ("task.Status");
120 ExecutedInline = true;
121 return TryExecuteTask (task);
124 protected override IEnumerable<Task> GetScheduledTasks ()
126 throw new NotImplementedException ();
136 this.factory = Task.Factory;
140 public void StartNewTest ()
143 factory.StartNew (() => result = true).Wait ();
144 Assert.IsTrue (result);
148 public void NoDefaultScheduler ()
150 Assert.IsNull (factory.Scheduler, "#1");
154 public void ContinueWhenAll_Simple ()
156 var mre = new ManualResetEventSlim (false);
158 Task[] tasks = new Task[3];
159 tasks[0] = new Task (() => { Thread.Sleep (0); Assert.IsTrue (mre.Wait (5000)); });
160 tasks[1] = new Task (() => { Assert.IsTrue (mre.Wait (5000)); });
161 tasks[2] = new Task (() => { Assert.IsTrue (mre.Wait (5000)); });
164 Task cont = factory.ContinueWhenAll (tasks, ts => {
165 Assert.AreEqual (tasks, ts, "#0");
169 foreach (Task t in tasks)
174 Assert.IsTrue (cont.Wait (3000), "#1");
175 Assert.IsTrue (ran, "#2");
179 public void ContinueWhenAll_WithMixedCompletionState ()
181 var mre = new ManualResetEventSlim ();
182 var task = Task.Factory.StartNew (() => mre.Wait (1000));
183 var contFailed = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnFaulted);
184 var contCanceled = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnCanceled);
185 var contSuccess = task.ContinueWith (t => {}, TaskContinuationOptions.OnlyOnRanToCompletion);
188 var cont = Task.Factory.ContinueWhenAll (new Task[] { contFailed, contCanceled, contSuccess }, _ => ran = true);
194 Assert.AreEqual (TaskStatus.RanToCompletion, cont.Status);
198 public void ContinueWhenAll_InvalidArguments ()
201 factory.ContinueWhenAll (null, delegate { });
203 } catch (ArgumentNullException) {
207 factory.ContinueWhenAll (new Task[0], delegate { });
209 } catch (ArgumentException) {
213 factory.ContinueWhenAll (new Task[] { null }, delegate { });
215 } catch (ArgumentException) {
218 var tasks = new Task [] {
219 factory.StartNew (delegate {})
223 factory.ContinueWhenAll (tasks, null);
225 } catch (ArgumentNullException) {
229 factory.ContinueWhenAll (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.None, null);
231 } catch (ArgumentNullException) {
235 factory.ContinueWhenAll (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.OnlyOnCanceled, null);
237 } catch (ArgumentNullException) {
238 } catch (ArgumentOutOfRangeException) {
243 public void ContinueWhenAll_WithExceptions ()
245 var t1 = Task.Factory.StartNew (() => { throw new ApplicationException ("Foo"); });
246 var t2 = Task.Factory.StartNew (() => { throw new ApplicationException ("Bar"); });
248 var cont = Task.Factory.ContinueWhenAll (new[] { t1, t2 }, delegate {});
251 Assert.IsTrue (t1.IsFaulted);
252 Assert.IsTrue (t2.IsFaulted);
253 Assert.AreEqual (TaskStatus.RanToCompletion, cont.Status);
257 public void ContinueWhenAny_Simple ()
259 var t1 = new ManualResetEvent (false);
260 var t2 = new ManualResetEvent (false);
262 var tasks = new Task[2] {
263 Task.Factory.StartNew (() => { t1.WaitOne (5000); }),
264 Task.Factory.StartNew (() => { t2.WaitOne (5000); })
268 var ct = new CancellationToken ();
269 Task cont = factory.ContinueWhenAny (tasks, t => {
270 Assert.AreEqual (tasks[0], t, "#1");
274 Assert.AreEqual (TaskStatus.WaitingForActivation, cont.Status, "#2");
278 Assert.IsTrue (cont.Wait (3000), "#10");
279 Assert.IsTrue (ran, "#11");
285 public void ContinueWhenAny_WithResult ()
287 var tcs = new TaskCompletionSource<int>();
289 Task[] tasks = new[] { tcs.Task };
290 var res = Task.Factory.ContinueWhenAny (tasks, l => 4);
291 Assert.AreEqual (4, res.Result);
295 public void ContinueWhenAny_InvalidArguments ()
298 factory.ContinueWhenAny (null, delegate { });
300 } catch (ArgumentNullException) {
304 factory.ContinueWhenAny (new Task[0], delegate { });
306 } catch (ArgumentException) {
310 factory.ContinueWhenAny (new Task[] { null }, delegate { });
312 } catch (ArgumentException) {
315 var tasks = new Task [] {
316 factory.StartNew (delegate {})
320 factory.ContinueWhenAny (tasks, null);
322 } catch (ArgumentNullException) {
326 factory.ContinueWhenAny (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.None, null);
328 } catch (ArgumentNullException) {
332 factory.ContinueWhenAny (tasks, delegate { }, CancellationToken.None, TaskContinuationOptions.OnlyOnCanceled, null);
334 } catch (ArgumentNullException) {
335 } catch (ArgumentOutOfRangeException) {
340 public void FromAsyncBeginInvoke_WithResult ()
344 Func<int, int> func = (i) => {
345 Assert.IsTrue (Thread.CurrentThread.IsThreadPoolThread);
346 result = true; return i + 3;
349 var task = factory.FromAsync<int, int> (func.BeginInvoke, func.EndInvoke, 1, "state", TaskCreationOptions.AttachedToParent);
350 Assert.IsTrue (task.Wait (5000), "#1");
351 Assert.IsTrue (result, "#2");
352 Assert.AreEqual (4, task.Result, "#3");
353 Assert.AreEqual ("state", (string) task.AsyncState, "#4");
354 Assert.AreEqual (TaskCreationOptions.AttachedToParent, task.CreationOptions, "#5");
358 public void FromAsyncBeginMethod_DirectResult ()
361 bool continuationTest = false;
363 Func<int, int> func = (i) => { result = true; return i + 3; };
364 Task<int> task = factory.FromAsync<int> (func.BeginInvoke (1, delegate { }, null), func.EndInvoke);
365 var cont = task.ContinueWith (_ => continuationTest = true, TaskContinuationOptions.ExecuteSynchronously);
369 Assert.IsTrue (result);
370 Assert.IsTrue (continuationTest);
371 Assert.AreEqual (4, task.Result);
375 public void FromAsyncBeginMethod_Exception ()
378 bool continuationTest = false;
380 Func<int, int> func = (i) => { result = true; throw new ApplicationException ("bleh"); };
381 Task<int> task = factory.FromAsync<int, int> (func.BeginInvoke, func.EndInvoke, 1, null);
382 var cont = task.ContinueWith (_ => continuationTest = true, TaskContinuationOptions.ExecuteSynchronously);
386 Assert.IsTrue (cont.Wait (2000), "#1");
388 Assert.IsTrue (result);
389 Assert.IsTrue (continuationTest);
390 Assert.IsNotNull (task.Exception);
391 var agg = task.Exception;
392 Assert.AreEqual (1, agg.InnerExceptions.Count);
393 Assert.That (agg.InnerExceptions[0], Is.TypeOf (typeof (ApplicationException)));
394 Assert.AreEqual (TaskStatus.Faulted, task.Status);
399 } catch (AggregateException) {
404 public void FromAsync_ArgumentsCheck ()
406 var result = new CompletedAsyncResult ();
408 factory.FromAsync (null, l => { });
410 } catch (ArgumentNullException) {
414 factory.FromAsync (result, null);
416 } catch (ArgumentNullException) {
420 factory.FromAsync (result, l => { }, TaskCreationOptions.LongRunning);
422 } catch (ArgumentOutOfRangeException) {
426 factory.FromAsync (result, l => { }, TaskCreationOptions.PreferFairness);
428 } catch (ArgumentOutOfRangeException) {
432 factory.FromAsync (result, l => { }, TaskCreationOptions.None, null);
434 } catch (ArgumentNullException) {
438 factory.FromAsync (null, l => { }, null, TaskCreationOptions.None);
440 } catch (ArgumentNullException) {
444 factory.FromAsync ((a, b) => null, l => { }, null, TaskCreationOptions.LongRunning);
446 } catch (ArgumentOutOfRangeException) {
451 public void FromAsync_Completed ()
453 var completed = new CompletedAsyncResult ();
456 Action<IAsyncResult> end = l => {
457 Assert.IsFalse (Thread.CurrentThread.IsThreadPoolThread, "#2");
458 valid = l == completed;
460 Task task = factory.FromAsync (completed, end);
461 Assert.IsTrue (valid == true, "#1");
465 public void FromAsync_CompletedWithException ()
467 var completed = new CompletedAsyncResult ();
469 Action<IAsyncResult> end = l => {
470 throw new ApplicationException ();
472 Task task = factory.FromAsync (completed, end);
473 Assert.AreEqual (TaskStatus.Faulted, task.Status, "#1");
477 public void FromAsync_CompletedCanceled ()
479 var completed = new CompletedAsyncResult ();
481 Action<IAsyncResult> end = l => {
482 throw new OperationCanceledException ();
484 Task task = factory.FromAsync (completed, end);
485 Assert.AreEqual (TaskStatus.Canceled, task.Status, "#1");
486 Assert.IsNull (task.Exception, "#2");
490 public void FromAsync_SimpleAsyncResult ()
492 var result = new TestAsyncResult ();
495 var task = factory.FromAsync (result, l => {
499 Assert.IsTrue (task.Wait (1000), "#1");
500 Assert.IsTrue (called, "#2");
504 public void FromAsync_ResultException ()
506 var result = new TestAsyncResult ();
508 var task = factory.FromAsync (result, l => {
509 throw new ApplicationException ();
513 Assert.IsFalse (task.Wait (1000), "#1");
514 } catch (AggregateException) {
517 Assert.AreEqual (TaskStatus.Faulted, task.Status, "#2");
521 public void FromAsync_ReturnInt ()
523 var result = new TestAsyncResult ();
526 var task = factory.FromAsync<int> (result, l => {
531 Assert.IsTrue (task.Wait (1000), "#1");
532 Assert.IsTrue (called, "#2");
533 Assert.AreEqual (4, task.Result, "#3");
537 public void FromAsync_Scheduler_Explicit ()
539 var result = new TestAsyncResult ();
541 var scheduler = new TestScheduler ();
543 var task = factory.FromAsync (result, l => {
545 }, TaskCreationOptions.None, scheduler);
547 Assert.IsTrue (task.Wait (5000), "#1");
548 Assert.IsTrue (called, "#2");
549 Assert.IsTrue (scheduler.ExecutedInline, "#3");
553 public void FromAsync_Scheduler_Implicit ()
555 var result = new TestAsyncResult ();
557 var scheduler = new TestScheduler ();
559 factory = new TaskFactory (scheduler);
561 Task task = factory.FromAsync (result, l => {
562 Assert.IsTrue (Thread.CurrentThread.IsThreadPoolThread, "#6");
564 }, TaskCreationOptions.AttachedToParent);
566 Assert.AreEqual (TaskCreationOptions.AttachedToParent, task.CreationOptions, "#1");
567 Assert.IsNull (task.AsyncState, "#2");
568 Assert.IsTrue (task.Wait (5000), "#3");
569 Assert.IsTrue (called, "#4");
570 Assert.IsTrue (scheduler.ExecutedInline, "#5");
574 public void FromAsync_BeginCallback ()
577 bool called2 = false;
579 var task = factory.FromAsync (
584 if ((TaskCreationOptions) c != TaskCreationOptions.AttachedToParent)
587 Assert.IsFalse (Thread.CurrentThread.IsThreadPoolThread, "#12");
596 "h", TaskCreationOptions.AttachedToParent);
598 Assert.AreEqual (TaskCreationOptions.None, task.CreationOptions, "#1");
599 Assert.AreEqual (TaskCreationOptions.AttachedToParent, (TaskCreationOptions) task.AsyncState, "#2");
600 Assert.IsTrue (task.Wait (5000), "#3");
601 Assert.IsTrue (called, "#4");
602 Assert.IsTrue (called2, "#5");
606 public void StartNewCancelled ()
608 var ct = new CancellationToken (true);
610 var task = factory.StartNew (() => Assert.Fail ("Should never be called"), ct);
614 } catch (InvalidOperationException) {
620 } catch (AggregateException e) {
621 Assert.That (e.InnerException, Is.TypeOf (typeof (TaskCanceledException)), "#3");
624 Assert.IsTrue (task.IsCanceled, "#4");