// 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.Collections.Generic; using System.Linq; using System.Reactive.Concurrency; using System.Reactive.Disposables; using System.Threading.Tasks; namespace System.Reactive.Linq { partial class AsyncObservable { public static IAsyncObservable Repeat(TSource value) { return Create(observer => AsyncObserver.Repeat(observer, value)); } public static IAsyncObservable Repeat(TSource value, IAsyncScheduler scheduler) { if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Repeat(observer, value, scheduler)); } public static IAsyncObservable Repeat(TSource value, int repeatCount) { if (repeatCount < 0) throw new ArgumentNullException(nameof(repeatCount)); return Create(observer => AsyncObserver.Repeat(observer, value, repeatCount)); } public static IAsyncObservable Repeat(TSource value, int repeatCount, IAsyncScheduler scheduler) { if (repeatCount < 0) throw new ArgumentNullException(nameof(repeatCount)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Repeat(observer, value, repeatCount, scheduler)); } public static IAsyncObservable Repeat(this IAsyncObservable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Create(observer => AsyncObserver.Repeat(observer, source)); } public static IAsyncObservable Repeat(this IAsyncObservable source, int repeatCount) { if (source == null) throw new ArgumentNullException(nameof(source)); if (repeatCount < 0) throw new ArgumentNullException(nameof(repeatCount)); return Create(observer => AsyncObserver.Repeat(observer, source, repeatCount)); } } partial class AsyncObserver { public static Task Repeat(IAsyncObserver observer, TSource value) { if (observer == null) throw new ArgumentNullException(nameof(observer)); return Repeat(observer, value, TaskPoolAsyncScheduler.Default); } public static Task Repeat(IAsyncObserver observer, TSource value, IAsyncScheduler scheduler) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return scheduler.ScheduleAsync(async ct => { while (!ct.IsCancellationRequested) { await observer.OnNextAsync(value).RendezVous(scheduler, ct); } }); } public static Task Repeat(IAsyncObserver observer, TSource value, int repeatCount) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (repeatCount < 0) throw new ArgumentNullException(nameof(repeatCount)); return Repeat(observer, value, repeatCount, TaskPoolAsyncScheduler.Default); } public static Task Repeat(IAsyncObserver observer, TSource value, int repeatCount, IAsyncScheduler scheduler) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (repeatCount < 0) throw new ArgumentNullException(nameof(repeatCount)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return scheduler.ScheduleAsync(async ct => { var i = 0; while (!ct.IsCancellationRequested && i < repeatCount) { await observer.OnNextAsync(value).RendezVous(scheduler, ct); i++; } if (i == repeatCount) { await observer.OnCompletedAsync().RendezVous(scheduler, ct); } }); } public static Task Repeat(IAsyncObserver observer, IAsyncObservable source) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (source == null) throw new ArgumentNullException(nameof(source)); async Task CoreAsync() { var (sink, inner) = Concat(observer, Repeat(source).GetEnumerator()); var subscription = await source.SubscribeAsync(sink).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(subscription, inner); } return CoreAsync(); } public static Task Repeat(IAsyncObserver observer, IAsyncObservable source, int repeatCount) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (source == null) throw new ArgumentNullException(nameof(source)); if (repeatCount < 0) throw new ArgumentNullException(nameof(repeatCount)); async Task CoreAsync() { var (sink, inner) = Concat(observer, Enumerable.Repeat(source, repeatCount).GetEnumerator()); var subscription = await source.SubscribeAsync(sink).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(subscription, inner); } return CoreAsync(); } private static IEnumerable Repeat(T value) { while (true) { yield return value; } } } }