// 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 TakeUntil(this IAsyncObservable source, IAsyncObservable until) { if (source == null) throw new ArgumentNullException(nameof(source)); if (until == null) throw new ArgumentNullException(nameof(until)); return Create(async observer => { var (sourceObserver, untilObserver) = AsyncObserver.TakeUntil(observer); var sourceTask = source.SubscribeAsync(sourceObserver); var untilTask = until.SubscribeAsync(untilObserver); var d1 = await sourceTask.ConfigureAwait(false); var d2 = await untilTask.ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(d1, d2); }); } public static IAsyncObservable TakeUntil(this IAsyncObservable source, DateTimeOffset endTime) { if (source == null) throw new ArgumentNullException(nameof(source)); return Create(observer => source.SubscribeAsync(AsyncObserver.TakeUntil(observer, endTime))); } public static IAsyncObservable TakeUntil(this IAsyncObservable source, DateTimeOffset endTime, IAsyncScheduler scheduler) { if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => source.SubscribeAsync(AsyncObserver.TakeUntil(observer, endTime, scheduler))); } } partial class AsyncObserver { public static (IAsyncObserver, IAsyncObserver) TakeUntil(IAsyncObserver observer) { if (observer == null) throw new ArgumentNullException(nameof(observer)); var gate = new AsyncLock(); return ( Create( async x => { using (await gate.LockAsync().ConfigureAwait(false)) { 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); } } ), Create( async y => { using (await gate.LockAsync().ConfigureAwait(false)) { await observer.OnCompletedAsync().ConfigureAwait(false); } }, async ex => { using (await gate.LockAsync().ConfigureAwait(false)) { await observer.OnErrorAsync(ex).ConfigureAwait(false); } }, () => Task.CompletedTask ) ); } public static IAsyncObserver TakeUntil(IAsyncObserver observer, DateTimeOffset endTime) { if (observer == null) throw new ArgumentNullException(nameof(observer)); throw new NotImplementedException(); } public static IAsyncObserver TakeUntil(IAsyncObserver observer, DateTimeOffset endTime, IAsyncScheduler scheduler) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); throw new NotImplementedException(); } } }