Timer.cs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  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. if (ct.IsCancellationRequested)
  69. return;
  70. await observer.OnNextAsync(0L).RendezVous(scheduler, ct);
  71. if (ct.IsCancellationRequested)
  72. return;
  73. await observer.OnCompletedAsync().RendezVous(scheduler, ct);
  74. }, dueTime);
  75. }
  76. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime) => Timer(observer, dueTime, TaskPoolAsyncScheduler.Default);
  77. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime, IAsyncScheduler scheduler)
  78. {
  79. if (scheduler == null)
  80. throw new ArgumentNullException(nameof(scheduler));
  81. return scheduler.ScheduleAsync(async ct =>
  82. {
  83. if (ct.IsCancellationRequested)
  84. return;
  85. await observer.OnNextAsync(0L).RendezVous(scheduler, ct);
  86. if (ct.IsCancellationRequested)
  87. return;
  88. await observer.OnCompletedAsync().RendezVous(scheduler, ct);
  89. }, dueTime);
  90. }
  91. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, TimeSpan dueTime, TimeSpan period) => Timer(observer, dueTime, period, TaskPoolAsyncScheduler.Default);
  92. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, TimeSpan dueTime, TimeSpan period, IAsyncScheduler scheduler)
  93. {
  94. if (period < TimeSpan.Zero)
  95. throw new ArgumentOutOfRangeException(nameof(period));
  96. if (scheduler == null)
  97. throw new ArgumentNullException(nameof(scheduler));
  98. var tick = 0L;
  99. return scheduler.ScheduleAsync(async ct =>
  100. {
  101. if (ct.IsCancellationRequested)
  102. return;
  103. // TODO: Compensate for drift by adding stopwatch functionality.
  104. do
  105. {
  106. await observer.OnNextAsync(tick++).RendezVous(scheduler, ct);
  107. await scheduler.Delay(period, ct).RendezVous(scheduler, ct);
  108. } while (!ct.IsCancellationRequested);
  109. }, dueTime);
  110. }
  111. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime, TimeSpan period) => Timer(observer, dueTime, period, TaskPoolAsyncScheduler.Default);
  112. public static Task<IAsyncDisposable> Timer(IAsyncObserver<long> observer, DateTimeOffset dueTime, TimeSpan period, IAsyncScheduler scheduler)
  113. {
  114. if (period < TimeSpan.Zero)
  115. throw new ArgumentOutOfRangeException(nameof(period));
  116. if (scheduler == null)
  117. throw new ArgumentNullException(nameof(scheduler));
  118. var tick = 0L;
  119. return scheduler.ScheduleAsync(async ct =>
  120. {
  121. if (ct.IsCancellationRequested)
  122. return;
  123. // TODO: Compensate for drift by adding stopwatch functionality.
  124. do
  125. {
  126. await observer.OnNextAsync(tick++).RendezVous(scheduler, ct);
  127. await scheduler.Delay(period, ct).RendezVous(scheduler, ct);
  128. } while (!ct.IsCancellationRequested);
  129. }, dueTime);
  130. }
  131. }
  132. }