// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. using System.Reactive.Concurrency; using System.Threading.Tasks; namespace System.Reactive.Linq { partial class AsyncObservable { public static IAsyncObservable Timer(TimeSpan dueTime) { return Create(observer => AsyncObserver.Timer(observer, dueTime)); } public static IAsyncObservable Timer(TimeSpan dueTime, IAsyncScheduler scheduler) { if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Timer(observer, dueTime, scheduler)); } public static IAsyncObservable Timer(DateTimeOffset dueTime) { return Create(observer => AsyncObserver.Timer(observer, dueTime)); } public static IAsyncObservable Timer(DateTimeOffset dueTime, IAsyncScheduler scheduler) { if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Timer(observer, dueTime, scheduler)); } public static IAsyncObservable Timer(TimeSpan dueTime, TimeSpan period) { if (period < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(period)); return Create(observer => AsyncObserver.Timer(observer, dueTime, period)); } public static IAsyncObservable Timer(TimeSpan dueTime, TimeSpan period, IAsyncScheduler scheduler) { if (period < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(period)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Timer(observer, dueTime, period, scheduler)); } public static IAsyncObservable Timer(DateTimeOffset dueTime, TimeSpan period) { if (period < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(period)); return Create(observer => AsyncObserver.Timer(observer, dueTime, period)); } public static IAsyncObservable Timer(DateTimeOffset dueTime, TimeSpan period, IAsyncScheduler scheduler) { if (period < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(period)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Timer(observer, dueTime, period, scheduler)); } } partial class AsyncObserver { public static Task Timer(IAsyncObserver observer, TimeSpan dueTime) => Timer(observer, dueTime, TaskPoolAsyncScheduler.Default); public static Task Timer(IAsyncObserver observer, TimeSpan dueTime, IAsyncScheduler scheduler) { if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return scheduler.ScheduleAsync(async ct => { ct.ThrowIfCancellationRequested(); await observer.OnNextAsync(0L).RendezVous(scheduler); ct.ThrowIfCancellationRequested(); await observer.OnCompletedAsync().RendezVous(scheduler); }, dueTime); } public static Task Timer(IAsyncObserver observer, DateTimeOffset dueTime) => Timer(observer, dueTime, TaskPoolAsyncScheduler.Default); public static Task Timer(IAsyncObserver observer, DateTimeOffset dueTime, IAsyncScheduler scheduler) { if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return scheduler.ScheduleAsync(async ct => { ct.ThrowIfCancellationRequested(); await observer.OnNextAsync(0L).RendezVous(scheduler); ct.ThrowIfCancellationRequested(); await observer.OnCompletedAsync().RendezVous(scheduler); }, dueTime); } public static Task Timer(IAsyncObserver observer, TimeSpan dueTime, TimeSpan period) => Timer(observer, dueTime, period, TaskPoolAsyncScheduler.Default); public static Task Timer(IAsyncObserver observer, TimeSpan dueTime, TimeSpan period, IAsyncScheduler scheduler) { if (period < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(period)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); var tick = 0L; return scheduler.ScheduleAsync(async ct => { ct.ThrowIfCancellationRequested(); // TODO: Compensate for drift by adding stopwatch functionality. do { await observer.OnNextAsync(tick++).RendezVous(scheduler); await scheduler.Delay(period, ct).RendezVous(scheduler); } while (!ct.IsCancellationRequested); ct.ThrowIfCancellationRequested(); }, dueTime); } public static Task Timer(IAsyncObserver observer, DateTimeOffset dueTime, TimeSpan period) => Timer(observer, dueTime, period, TaskPoolAsyncScheduler.Default); public static Task Timer(IAsyncObserver observer, DateTimeOffset dueTime, TimeSpan period, IAsyncScheduler scheduler) { if (period < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(period)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); var tick = 0L; return scheduler.ScheduleAsync(async ct => { ct.ThrowIfCancellationRequested(); // TODO: Compensate for drift by adding stopwatch functionality. do { await observer.OnNextAsync(tick++).RendezVous(scheduler); await scheduler.Delay(period, ct).RendezVous(scheduler); } while (!ct.IsCancellationRequested); ct.ThrowIfCancellationRequested(); }, dueTime); } } }