// 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.Reactive.Disposables; using System.Threading; using System.Threading.Tasks; namespace System.Reactive.Linq { partial class AsyncObservable { public static IAsyncObservable Skip(this IAsyncObservable source, int count) { if (source == null) throw new ArgumentNullException(nameof(source)); if (count < 0) throw new ArgumentOutOfRangeException(nameof(count)); if (count == 0) { return source; } return Create(observer => source.SubscribeSafeAsync(AsyncObserver.Skip(observer, count))); } public static IAsyncObservable Skip(this IAsyncObservable source, TimeSpan duration) { if (source == null) throw new ArgumentNullException(nameof(source)); if (duration < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(duration)); if (duration == TimeSpan.Zero) { return source; } // REVIEW: May be easier to just use SkipUntil with a Timer parameter. Do we want Skip on the observer? return Create(async observer => { var (sourceObserver, timer) = await AsyncObserver.Skip(observer, duration); var subscription = await source.SubscribeSafeAsync(sourceObserver).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(subscription, timer); }); } public static IAsyncObservable Skip(this IAsyncObservable source, TimeSpan duration, IAsyncScheduler scheduler) { if (source == null) throw new ArgumentNullException(nameof(source)); if (duration < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(duration)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); if (duration == TimeSpan.Zero) { return source; } // REVIEW: May be easier to just use SkipUntil with a Timer parameter. Do we want Skip on the observer? return Create(async observer => { var (sourceObserver, timer) = await AsyncObserver.Skip(observer, duration); var subscription = await source.SubscribeSafeAsync(sourceObserver).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(subscription, timer); }); } } partial class AsyncObserver { public static IAsyncObserver Skip(IAsyncObserver observer, int count) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (count <= 0) throw new ArgumentOutOfRangeException(nameof(count)); return Create( async x => { if (count == 0) { await observer.OnNextAsync(x).ConfigureAwait(false); } else { --count; } }, observer.OnErrorAsync, observer.OnCompletedAsync ); } public static Task<(IAsyncObserver, IAsyncDisposable)> Skip(IAsyncObserver observer, TimeSpan duration) => Skip(observer, duration, TaskPoolAsyncScheduler.Default); public static Task<(IAsyncObserver, IAsyncDisposable)> Skip(IAsyncObserver observer, TimeSpan duration, IAsyncScheduler scheduler) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (duration < TimeSpan.Zero) throw new ArgumentOutOfRangeException(nameof(duration)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return CoreAsync(); async Task<(IAsyncObserver, IAsyncDisposable)> CoreAsync() { // REVIEW: May be easier to just use SkipUntil with a Timer parameter. Do we want Skip on the observer? // DESIGN: It seems that if an observer would be an IAsyncDisposable, this could get a bit easier ("inject" the inner disposable). var gate = new AsyncLock(); var open = false; return ( Create( async x => { using (await gate.LockAsync().ConfigureAwait(false)) { if (open) { await observer.OnNextAsync(x).ConfigureAwait(false); } } }, async ex => { using (await gate.LockAsync().ConfigureAwait(false)) { await observer.OnErrorAsync(ex).ConfigureAwait(false); } }, async () => { using (await gate.LockAsync().ConfigureAwait(false)) { await observer.OnCompletedAsync().ConfigureAwait(false); } } ), await scheduler.ScheduleAsync(async ct => { ct.ThrowIfCancellationRequested(); using (await gate.LockAsync().RendezVous(scheduler)) { open = true; } }, duration) ); } } } }