TaskPoolSchedulerTest.cs 5.7 KB

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