Timer.cs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  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.Reactive.Concurrency;
  5. using System.Threading.Tasks;
  6. namespace System.Reactive.Linq
  7. {
  8. partial class AsyncObservable
  9. {
  10. public static IAsyncObservable<long> Timer(TimeSpan dueTime)
  11. {
  12. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime));
  13. }
  14. public static IAsyncObservable<long> Timer(TimeSpan dueTime, IAsyncScheduler scheduler)
  15. {
  16. if (scheduler == null)
  17. throw new ArgumentNullException(nameof(scheduler));
  18. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime, scheduler));
  19. }
  20. public static IAsyncObservable<long> Timer(DateTimeOffset dueTime)
  21. {
  22. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime));
  23. }
  24. public static IAsyncObservable<long> Timer(DateTimeOffset dueTime, IAsyncScheduler scheduler)
  25. {
  26. if (scheduler == null)
  27. throw new ArgumentNullException(nameof(scheduler));
  28. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime, scheduler));
  29. }
  30. public static IAsyncObservable<long> Timer(TimeSpan dueTime, TimeSpan period)
  31. {
  32. if (period < TimeSpan.Zero)
  33. throw new ArgumentOutOfRangeException(nameof(period));
  34. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime, period));
  35. }
  36. public static IAsyncObservable<long> Timer(TimeSpan dueTime, TimeSpan period, IAsyncScheduler scheduler)
  37. {
  38. if (period < TimeSpan.Zero)
  39. throw new ArgumentOutOfRangeException(nameof(period));
  40. if (scheduler == null)
  41. throw new ArgumentNullException(nameof(scheduler));
  42. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime, period, scheduler));
  43. }
  44. public static IAsyncObservable<long> Timer(DateTimeOffset dueTime, TimeSpan period)
  45. {
  46. if (period < TimeSpan.Zero)
  47. throw new ArgumentOutOfRangeException(nameof(period));
  48. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime, period));
  49. }
  50. public static IAsyncObservable<long> Timer(DateTimeOffset dueTime, TimeSpan period, IAsyncScheduler scheduler)
  51. {
  52. if (period < TimeSpan.Zero)
  53. throw new ArgumentOutOfRangeException(nameof(period));
  54. if (scheduler == null)
  55. throw new ArgumentNullException(nameof(scheduler));
  56. return Create<long>(observer => AsyncObserver.Timer(observer, dueTime, period, scheduler));
  57. }
  58. }
  59. partial class AsyncObserver
  60. {
  61. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, TimeSpan dueTime) => Timer(observer, dueTime, TaskPoolAsyncScheduler.Default);
  62. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, TimeSpan dueTime, IAsyncScheduler scheduler)
  63. {
  64. if (scheduler == null)
  65. throw new ArgumentNullException(nameof(scheduler));
  66. return scheduler.ScheduleAsync(async ct =>
  67. {
  68. ct.ThrowIfCancellationRequested();
  69. await observer.OnNextAsync(0L).RendezVous(scheduler);
  70. ct.ThrowIfCancellationRequested();
  71. await observer.OnCompletedAsync().RendezVous(scheduler);
  72. }, dueTime);
  73. }
  74. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime) => Timer(observer, dueTime, TaskPoolAsyncScheduler.Default);
  75. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime, IAsyncScheduler scheduler)
  76. {
  77. if (scheduler == null)
  78. throw new ArgumentNullException(nameof(scheduler));
  79. return scheduler.ScheduleAsync(async ct =>
  80. {
  81. ct.ThrowIfCancellationRequested();
  82. await observer.OnNextAsync(0L).RendezVous(scheduler);
  83. ct.ThrowIfCancellationRequested();
  84. await observer.OnCompletedAsync().RendezVous(scheduler);
  85. }, dueTime);
  86. }
  87. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, TimeSpan dueTime, TimeSpan period) => Timer(observer, dueTime, period, TaskPoolAsyncScheduler.Default);
  88. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, TimeSpan dueTime, TimeSpan period, IAsyncScheduler scheduler)
  89. {
  90. if (period < TimeSpan.Zero)
  91. throw new ArgumentOutOfRangeException(nameof(period));
  92. if (scheduler == null)
  93. throw new ArgumentNullException(nameof(scheduler));
  94. var tick = 0L;
  95. return scheduler.ScheduleAsync(async ct =>
  96. {
  97. ct.ThrowIfCancellationRequested();
  98. // TODO: Compensate for drift by adding stopwatch functionality.
  99. do
  100. {
  101. await observer.OnNextAsync(tick++).RendezVous(scheduler);
  102. await scheduler.Delay(period, ct).RendezVous(scheduler);
  103. } while (!ct.IsCancellationRequested);
  104. ct.ThrowIfCancellationRequested();
  105. }, dueTime);
  106. }
  107. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime, TimeSpan period) => Timer(observer, dueTime, period, TaskPoolAsyncScheduler.Default);
  108. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime, TimeSpan period, IAsyncScheduler scheduler)
  109. {
  110. if (period < TimeSpan.Zero)
  111. throw new ArgumentOutOfRangeException(nameof(period));
  112. if (scheduler == null)
  113. throw new ArgumentNullException(nameof(scheduler));
  114. var tick = 0L;
  115. return scheduler.ScheduleAsync(async ct =>
  116. {
  117. ct.ThrowIfCancellationRequested();
  118. // TODO: Compensate for drift by adding stopwatch functionality.
  119. do
  120. {
  121. await observer.OnNextAsync(tick++).RendezVous(scheduler);
  122. await scheduler.Delay(period, ct).RendezVous(scheduler);
  123. } while (!ct.IsCancellationRequested);
  124. ct.ThrowIfCancellationRequested();
  125. }, dueTime);
  126. }
  127. }
  128. }