VirtualSchedulerTest.cs 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  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.Collections.Generic;
  6. using System.Diagnostics;
  7. using System.Reactive.Concurrency;
  8. using System.Reactive.Linq;
  9. using System.Threading;
  10. using Microsoft.Reactive.Testing;
  11. using Xunit;
  12. namespace ReactiveTests.Tests
  13. {
  14. public class VirtualSchedulerTest
  15. {
  16. class VirtualSchedulerTestScheduler : VirtualTimeScheduler<string, char>
  17. {
  18. public VirtualSchedulerTestScheduler()
  19. {
  20. }
  21. public VirtualSchedulerTestScheduler(string initialClock, IComparer<string> comparer)
  22. : base(initialClock, comparer)
  23. {
  24. }
  25. protected override string Add(string absolute, char relative)
  26. {
  27. return (absolute ?? string.Empty) + relative;
  28. }
  29. protected override DateTimeOffset ToDateTimeOffset(string absolute)
  30. {
  31. return new DateTimeOffset((absolute ?? string.Empty).Length, TimeSpan.Zero);
  32. }
  33. protected override char ToRelative(TimeSpan timeSpan)
  34. {
  35. return (char)(timeSpan.Ticks % char.MaxValue);
  36. }
  37. }
  38. [Fact]
  39. public void Virtual_Now()
  40. {
  41. var res = new VirtualSchedulerTestScheduler().Now - DateTime.Now;
  42. Assert.True(res.Seconds < 1);
  43. }
  44. #if !NO_THREAD
  45. [Fact]
  46. public void Virtual_ScheduleAction()
  47. {
  48. var id = Thread.CurrentThread.ManagedThreadId;
  49. var ran = false;
  50. var scheduler = new VirtualSchedulerTestScheduler();
  51. scheduler.Schedule(() => { Assert.Equal(id, Thread.CurrentThread.ManagedThreadId); ran = true; });
  52. scheduler.Start();
  53. Assert.True(ran);
  54. }
  55. #endif
  56. [Fact]
  57. public void Virtual_ScheduleActionError()
  58. {
  59. var ex = new Exception();
  60. try
  61. {
  62. var scheduler = new VirtualSchedulerTestScheduler();
  63. scheduler.Schedule(() => { throw ex; });
  64. scheduler.Start();
  65. Assert.True(false);
  66. }
  67. catch (Exception e)
  68. {
  69. Assert.Same(e, ex);
  70. }
  71. }
  72. [Fact]
  73. public void Virtual_InitialAndComparer_Now()
  74. {
  75. var s = new VirtualSchedulerTestScheduler("Bar", Comparer<string>.Default);
  76. Assert.Equal(3, s.Now.Ticks);
  77. }
  78. [Fact]
  79. public void Virtual_ArgumentChecking()
  80. {
  81. ReactiveAssert.Throws<ArgumentNullException>(() => new VirtualSchedulerTestScheduler("", null));
  82. ReactiveAssert.Throws<ArgumentNullException>(() => new VirtualSchedulerTestScheduler().ScheduleRelative(0, 'a', null));
  83. ReactiveAssert.Throws<ArgumentNullException>(() => new VirtualSchedulerTestScheduler().ScheduleAbsolute(0, "", null));
  84. ReactiveAssert.Throws<ArgumentNullException>(() => new VirtualSchedulerTestScheduler().Schedule(0, default(Func<IScheduler, int, IDisposable>)));
  85. ReactiveAssert.Throws<ArgumentNullException>(() => new VirtualSchedulerTestScheduler().Schedule(0, TimeSpan.Zero, default(Func<IScheduler, int, IDisposable>)));
  86. ReactiveAssert.Throws<ArgumentNullException>(() => new VirtualSchedulerTestScheduler().Schedule(0, DateTimeOffset.UtcNow, default(Func<IScheduler, int, IDisposable>)));
  87. ReactiveAssert.Throws<ArgumentNullException>(() => VirtualTimeSchedulerExtensions.ScheduleAbsolute(default(VirtualSchedulerTestScheduler), "", () => {}));
  88. ReactiveAssert.Throws<ArgumentNullException>(() => VirtualTimeSchedulerExtensions.ScheduleAbsolute(new VirtualSchedulerTestScheduler(), "", default(Action)));
  89. ReactiveAssert.Throws<ArgumentNullException>(() => VirtualTimeSchedulerExtensions.ScheduleRelative(default(VirtualSchedulerTestScheduler), 'a', () => { }));
  90. ReactiveAssert.Throws<ArgumentNullException>(() => VirtualTimeSchedulerExtensions.ScheduleRelative(new VirtualSchedulerTestScheduler(), 'a', default(Action)));
  91. }
  92. [Fact]
  93. public void Historical_ArgumentChecking()
  94. {
  95. ReactiveAssert.Throws<ArgumentNullException>(() => new HistoricalScheduler(DateTime.Now, default(IComparer<DateTimeOffset>)));
  96. ReactiveAssert.Throws<ArgumentNullException>(() => new HistoricalScheduler().ScheduleAbsolute(42, DateTime.Now, default(Func<IScheduler, int, IDisposable>)));
  97. ReactiveAssert.Throws<ArgumentNullException>(() => new HistoricalScheduler().ScheduleRelative(42, TimeSpan.FromSeconds(1), default(Func<IScheduler, int, IDisposable>)));
  98. }
  99. #if !SILVERLIGHT && !NO_THREAD
  100. [Fact(Skip = "Ignored")]
  101. public void Virtual_ScheduleActionDue()
  102. {
  103. var id = Thread.CurrentThread.ManagedThreadId;
  104. var ran = false;
  105. var sw = new Stopwatch();
  106. sw.Start();
  107. var scheduler = new VirtualSchedulerTestScheduler();
  108. scheduler.Schedule(TimeSpan.FromSeconds(0.2), () => { sw.Stop(); Assert.Equal(id, Thread.CurrentThread.ManagedThreadId); ran = true; });
  109. scheduler.Start();
  110. Assert.True(ran, "ran");
  111. Assert.True(sw.ElapsedMilliseconds > 180, "due " + sw.ElapsedMilliseconds);
  112. }
  113. #endif
  114. #if DESKTOPCLR
  115. [Fact]
  116. public void Virtual_ThreadSafety()
  117. {
  118. for (var i = 0; i < 10; i++)
  119. {
  120. var scheduler = new TestScheduler();
  121. var seq = Observable.Never<string>();
  122. ThreadPool.QueueUserWorkItem(_ =>
  123. {
  124. Thread.Sleep(50);
  125. seq.Timeout(TimeSpan.FromSeconds(10), scheduler).Subscribe(s => { });
  126. });
  127. var watch = scheduler.StartStopwatch();
  128. try
  129. {
  130. while (watch.Elapsed < TimeSpan.FromSeconds(20))
  131. {
  132. scheduler.AdvanceBy(10);
  133. }
  134. }
  135. catch (TimeoutException)
  136. {
  137. }
  138. catch (Exception ex)
  139. {
  140. Assert.True(false, string.Format("Virtual time {0}, exception {1}", watch.Elapsed, ex));
  141. }
  142. }
  143. }
  144. #endif
  145. }
  146. }