Delay.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  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.Collections.Generic;
  5. using System.Reactive.Concurrency;
  6. using System.Reactive.Disposables;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. namespace System.Reactive.Linq
  10. {
  11. // TODO: Add overloads with DateTimeOffset and with duration selector.
  12. partial class AsyncObservable
  13. {
  14. public static IAsyncObservable<TSource> Delay<TSource>(this IAsyncObservable<TSource> source, TimeSpan dueTime)
  15. {
  16. if (source == null)
  17. throw new ArgumentNullException(nameof(source));
  18. return Create<TSource>(async observer =>
  19. {
  20. var (sink, drain) = await AsyncObserver.Delay(observer, dueTime).ConfigureAwait(false);
  21. var subscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false);
  22. return StableCompositeAsyncDisposable.Create(subscription, drain);
  23. });
  24. }
  25. public static IAsyncObservable<TSource> Delay<TSource>(this IAsyncObservable<TSource> source, TimeSpan dueTime, IAsyncScheduler scheduler)
  26. {
  27. if (source == null)
  28. throw new ArgumentNullException(nameof(source));
  29. if (scheduler == null)
  30. throw new ArgumentNullException(nameof(scheduler));
  31. return Create<TSource>(async observer =>
  32. {
  33. var (sink, drain) = await AsyncObserver.Delay(observer, dueTime, scheduler).ConfigureAwait(false);
  34. var subscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false);
  35. return StableCompositeAsyncDisposable.Create(subscription, drain);
  36. });
  37. }
  38. }
  39. partial class AsyncObserver
  40. {
  41. public static Task<(IAsyncObserver<TSource>, IAsyncDisposable)> Delay<TSource>(this IAsyncObserver<TSource> observer, TimeSpan dueTime)
  42. {
  43. if (observer == null)
  44. throw new ArgumentNullException(nameof(observer));
  45. return Delay(observer, dueTime, TaskPoolAsyncScheduler.Default);
  46. }
  47. public static async Task<(IAsyncObserver<TSource>, IAsyncDisposable)> Delay<TSource>(this IAsyncObserver<TSource> observer, TimeSpan dueTime, IAsyncScheduler scheduler)
  48. {
  49. if (observer == null)
  50. throw new ArgumentNullException(nameof(observer));
  51. if (scheduler == null)
  52. throw new ArgumentNullException(nameof(scheduler));
  53. // TODO: Use stopwatch functionality.
  54. var start = scheduler.Now;
  55. var semaphore = new SemaphoreSlim(0);
  56. var gate = new AsyncLock();
  57. var queue = new Queue<TimeInterval<TSource>>();
  58. var isDone = false;
  59. var drain = await scheduler.ScheduleAsync(async ct =>
  60. {
  61. while (!ct.IsCancellationRequested)
  62. {
  63. await semaphore.WaitAsync(ct).RendezVous(scheduler, ct);
  64. if (queue.Count > 0)
  65. {
  66. var next = queue.Dequeue();
  67. var nextDueTime = start + next.Interval + dueTime;
  68. var delay = nextDueTime - scheduler.Now;
  69. await scheduler.Delay(delay, ct).RendezVous(scheduler, ct);
  70. await observer.OnNextAsync(next.Value).RendezVous(scheduler, ct);
  71. }
  72. if (queue.Count == 0 && isDone)
  73. {
  74. await observer.OnCompletedAsync().RendezVous(scheduler, ct);
  75. break;
  76. }
  77. }
  78. }).ConfigureAwait(false);
  79. var sink = Create<TSource>(
  80. async x =>
  81. {
  82. var time = scheduler.Now;
  83. var delay = time - start;
  84. using (await gate.LockAsync().ConfigureAwait(false))
  85. {
  86. queue.Enqueue(new TimeInterval<TSource>(x, delay));
  87. }
  88. semaphore.Release(1);
  89. },
  90. async ex =>
  91. {
  92. using (await gate.LockAsync().ConfigureAwait(false))
  93. {
  94. await observer.OnErrorAsync(ex).ConfigureAwait(false);
  95. }
  96. },
  97. async () =>
  98. {
  99. using (await gate.LockAsync().ConfigureAwait(false))
  100. {
  101. isDone = true;
  102. }
  103. semaphore.Release(1);
  104. }
  105. );
  106. return (sink, drain);
  107. }
  108. }
  109. }