AsyncSchedulerBase.cs 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  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.Threading;
  5. using System.Threading.Tasks;
  6. namespace System.Reactive.Concurrency
  7. {
  8. public abstract class AsyncSchedulerBase : IAsyncScheduler
  9. {
  10. public virtual DateTimeOffset Now => DateTimeOffset.Now;
  11. public virtual ValueTask<IAsyncDisposable> ScheduleAsync(Func<CancellationToken, ValueTask> action)
  12. {
  13. if (action == null)
  14. throw new ArgumentNullException(nameof(action));
  15. return ScheduleAsyncCore(action);
  16. }
  17. public virtual ValueTask<IAsyncDisposable> ScheduleAsync(Func<CancellationToken, ValueTask> action, TimeSpan dueTime)
  18. {
  19. if (action == null)
  20. throw new ArgumentNullException(nameof(action));
  21. var dueTimeRelative = Normalize(dueTime);
  22. return ScheduleAsyncCore(async ct =>
  23. {
  24. await Delay(dueTimeRelative, ct); // NB: Honor SynchronizationContext to stay on scheduler.
  25. await action(ct);
  26. });
  27. }
  28. public virtual ValueTask<IAsyncDisposable> ScheduleAsync(Func<CancellationToken, ValueTask> action, DateTimeOffset dueTime)
  29. {
  30. if (action == null)
  31. throw new ArgumentNullException(nameof(action));
  32. return ScheduleAsyncCore(async ct =>
  33. {
  34. var dueTimeRelative = Normalize(dueTime - Now); // TODO: Support clock drift and clock changes.
  35. await Delay(dueTimeRelative, ct); // NB: Honor SynchronizationContext to stay on scheduler.
  36. await action(ct);
  37. });
  38. }
  39. protected virtual async ValueTask<IAsyncDisposable> ScheduleAsyncCore(Func<CancellationToken, ValueTask> action)
  40. {
  41. var cad = new CancellationAsyncDisposable();
  42. await ScheduleAsyncCore(action, cad.Token).ConfigureAwait(false);
  43. return cad;
  44. }
  45. protected abstract ValueTask ScheduleAsyncCore(Func<CancellationToken, ValueTask> action, CancellationToken token);
  46. protected abstract ValueTask Delay(TimeSpan dueTime, CancellationToken token);
  47. protected static TimeSpan Normalize(TimeSpan timeSpan) => timeSpan < TimeSpan.Zero ? TimeSpan.Zero : timeSpan;
  48. private sealed class CancellationAsyncDisposable : IAsyncDisposable
  49. {
  50. private readonly CancellationTokenSource _cts = new CancellationTokenSource();
  51. public CancellationToken Token => _cts.Token;
  52. public ValueTask DisposeAsync()
  53. {
  54. _cts.Cancel();
  55. return default;
  56. }
  57. }
  58. }
  59. }