DefaultSchedulerTest.cs 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the MIT 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 Microsoft.VisualStudio.TestTools.UnitTesting;
  9. using Assert = Xunit.Assert;
  10. namespace ReactiveTests.Tests
  11. {
  12. [TestClass]
  13. public class DefaultSchedulerTest
  14. {
  15. [TestMethod]
  16. public void Schedule_ArgumentChecking()
  17. {
  18. ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.Schedule(42, default));
  19. ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.Schedule(42, DateTimeOffset.Now, default));
  20. ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.Schedule(42, TimeSpan.Zero, default));
  21. ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.SchedulePeriodic(42, TimeSpan.FromSeconds(1), default));
  22. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => DefaultScheduler.Instance.SchedulePeriodic(42, TimeSpan.FromSeconds(-1), _ => _));
  23. }
  24. [TestMethod]
  25. public void Get_Now()
  26. {
  27. var res = DefaultScheduler.Instance.Now - DateTime.Now;
  28. Assert.True(res.Seconds < 1);
  29. }
  30. #if !NO_THREAD
  31. [TestMethod]
  32. public void ScheduleAction()
  33. {
  34. var id = Thread.CurrentThread.ManagedThreadId;
  35. var nt = DefaultScheduler.Instance;
  36. var evt = new ManualResetEvent(false);
  37. nt.Schedule(() => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  38. evt.WaitOne();
  39. }
  40. [TestMethod]
  41. public void ScheduleActionDue()
  42. {
  43. var id = Thread.CurrentThread.ManagedThreadId;
  44. var nt = DefaultScheduler.Instance;
  45. var evt = new ManualResetEvent(false);
  46. nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });
  47. evt.WaitOne();
  48. }
  49. [TestMethod]
  50. public void ScheduleActionCancel()
  51. {
  52. var id = Thread.CurrentThread.ManagedThreadId;
  53. var nt = DefaultScheduler.Instance;
  54. var set = false;
  55. var d = nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.True(false); set = true; });
  56. d.Dispose();
  57. Thread.Sleep(400);
  58. Assert.False(set);
  59. }
  60. [TestMethod]
  61. public void Periodic_NonReentrant()
  62. {
  63. var n = 0;
  64. var fail = false;
  65. var d = DefaultScheduler.Instance.SchedulePeriodic(0, TimeSpan.FromMilliseconds(50), x =>
  66. {
  67. try
  68. {
  69. if (Interlocked.Increment(ref n) > 1) // Without an AsyncLock this would fail.
  70. {
  71. fail = true;
  72. }
  73. Thread.Sleep(100);
  74. return x + 1;
  75. }
  76. finally
  77. {
  78. Interlocked.Decrement(ref n);
  79. }
  80. });
  81. Thread.Sleep(500);
  82. d.Dispose();
  83. Assert.False(fail);
  84. }
  85. #endif
  86. #if DESKTOPCLR
  87. [TestCategory("SkipCI")]
  88. [TestMethod]
  89. public void No_ThreadPool_Starvation_Dispose()
  90. {
  91. ThreadPool.GetAvailableThreads(out var bwt, out var bio);
  92. var N = Environment.ProcessorCount * 2;
  93. for (var i = 0; i < N; i++)
  94. {
  95. var e = new ManualResetEvent(false);
  96. var f = new ManualResetEvent(false);
  97. var d = Scheduler.Default.Schedule(TimeSpan.FromMilliseconds(1), () => { e.Set(); f.WaitOne(); });
  98. e.WaitOne();
  99. d.Dispose();
  100. f.Set();
  101. }
  102. ThreadPool.GetAvailableThreads(out var ewt, out var eio);
  103. Assert.False(bwt - ewt >= N);
  104. }
  105. #endif
  106. }
  107. }