TaskPoolSchedulerTest.cs 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
  2. #if !NO_TPL
  3. using System;
  4. using System.Reactive.Concurrency;
  5. using System.Threading;
  6. using Microsoft.Reactive.Testing;
  7. using Xunit;
  8. namespace ReactiveTests.Tests
  9. {
  10. public class TaskPoolSchedulerTest
  11. {
  12. [Fact]
  13. public void TaskPool_ArgumentChecking()
  14. {
  15. ReactiveAssert.Throws<ArgumentNullException>(() => new TaskPoolScheduler(null));
  16. ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule<int>(42, default(Func<IScheduler, int, IDisposable>)));
  17. ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule<int>(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));
  18. ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule<int>(42, TimeSpan.Zero, default(Func<IScheduler, int, IDisposable>)));
  19. ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.SchedulePeriodic(42, TimeSpan.FromSeconds(1), default(Func<int, int>)));
  20. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => TaskPoolScheduler.Default.SchedulePeriodic(42, TimeSpan.FromSeconds(-1), _ => _));
  21. }
  22. [Fact]
  23. public void TaskPool_Now()
  24. {
  25. var res = TaskPoolScheduler.Default.Now - DateTime.Now;
  26. Assert.True(res.Seconds < 1);
  27. }
  28. #if !NO_THREAD
  29. [Fact]
  30. public void TaskPool_ScheduleAction()
  31. {
  32. var id = Thread.CurrentThread.ManagedThreadId;
  33. var nt = TaskPoolScheduler.Default;
  34. var evt = new ManualResetEvent(false);
  35. nt.Schedule(() => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  36. evt.WaitOne();
  37. }
  38. [Fact]
  39. public void TaskPool_ScheduleActionDueNow()
  40. {
  41. var id = Thread.CurrentThread.ManagedThreadId;
  42. var nt = TaskPoolScheduler.Default;
  43. var evt = new ManualResetEvent(false);
  44. nt.Schedule(TimeSpan.Zero, () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  45. evt.WaitOne();
  46. }
  47. [Fact]
  48. public void TaskPool_ScheduleActionDue()
  49. {
  50. var id = Thread.CurrentThread.ManagedThreadId;
  51. var nt = TaskPoolScheduler.Default;
  52. var evt = new ManualResetEvent(false);
  53. nt.Schedule(TimeSpan.FromMilliseconds(1), () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  54. evt.WaitOne();
  55. }
  56. [Fact]
  57. public void TaskPool_ScheduleActionCancel()
  58. {
  59. var id = Thread.CurrentThread.ManagedThreadId;
  60. var nt = TaskPoolScheduler.Default;
  61. var set = false;
  62. var d = nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.True(false); set = true; });
  63. d.Dispose();
  64. Thread.Sleep(400);
  65. Assert.False(set);
  66. }
  67. #if !NO_PERF
  68. [Fact]
  69. public void TaskPool_ScheduleLongRunning()
  70. {
  71. var n = 0;
  72. var e = new ManualResetEvent(false);
  73. var gate = new object();
  74. var d = TaskPoolScheduler.Default.ScheduleLongRunning(42, (x, cancel) =>
  75. {
  76. while (!cancel.IsDisposed)
  77. lock (gate)
  78. n++;
  79. e.Set();
  80. });
  81. while (true)
  82. {
  83. lock (gate)
  84. if (n >= 10)
  85. break;
  86. Thread.Sleep(10);
  87. }
  88. d.Dispose();
  89. e.WaitOne();
  90. Assert.True(n >= 0);
  91. }
  92. #endif
  93. #endif
  94. #if !NO_PERF
  95. #if !NO_STOPWATCH
  96. [Fact]
  97. public void Stopwatch()
  98. {
  99. StopwatchTest.Run(TaskPoolScheduler.Default);
  100. }
  101. #endif
  102. #endif
  103. [Fact]
  104. public void TaskPool_Periodic()
  105. {
  106. var n = 0;
  107. var e = new ManualResetEvent(false);
  108. var d = TaskPoolScheduler.Default.SchedulePeriodic(TimeSpan.FromMilliseconds(25), () =>
  109. {
  110. if (Interlocked.Increment(ref n) == 10)
  111. e.Set();
  112. });
  113. if (!e.WaitOne(10000))
  114. Assert.True(false);
  115. d.Dispose();
  116. }
  117. #if !NO_THREAD
  118. [Fact]
  119. public void TaskPool_Periodic_NonReentrant()
  120. {
  121. var n = 0;
  122. var fail = false;
  123. var d = TaskPoolScheduler.Default.SchedulePeriodic(0, TimeSpan.FromMilliseconds(50), x =>
  124. {
  125. try
  126. {
  127. if (Interlocked.Increment(ref n) > 1) // Without an AsyncLock this would fail.
  128. fail = true;
  129. Thread.Sleep(100);
  130. return x + 1;
  131. }
  132. finally
  133. {
  134. Interlocked.Decrement(ref n);
  135. }
  136. });
  137. Thread.Sleep(500);
  138. d.Dispose();
  139. Assert.False(fail);
  140. }
  141. #endif
  142. [Fact]
  143. public void TaskPool_Delay_LargerThanIntMaxValue()
  144. {
  145. var dueTime = TimeSpan.FromMilliseconds((double)int.MaxValue + 1);
  146. // Just ensuring the call to Schedule does not throw.
  147. var d = TaskPoolScheduler.Default.Schedule(dueTime, () => { });
  148. d.Dispose();
  149. }
  150. }
  151. }
  152. #endif