// 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.Reactive.Concurrency; using System.Reactive.Disposables; using System.Threading.Tasks; namespace System.Reactive.Linq { partial class AsyncObservable { public static IAsyncObservable StartWith(this IAsyncObservable source, params TSource[] values) => Prepend(source, values); public static IAsyncObservable StartWith(this IAsyncObservable source, IEnumerable values) => Prepend(source, values); public static IAsyncObservable StartWith(this IAsyncObservable source, IAsyncScheduler scheduler, params TSource[] values) => Prepend(source, scheduler, values); public static IAsyncObservable StartWith(this IAsyncObservable source, IAsyncScheduler scheduler, IEnumerable values) => Prepend(source, scheduler, values); public static IAsyncObservable Prepend(this IAsyncObservable source, TSource value) { if (source == null) throw new ArgumentNullException(nameof(source)); return Create(async observer => await source.SubscribeSafeAsync(await AsyncObserver.Prepend(observer, value).ConfigureAwait(false)).ConfigureAwait(false)); } public static IAsyncObservable Prepend(this IAsyncObservable source, TSource value, IAsyncScheduler scheduler) { if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Create(observer => AsyncObserver.Prepend(observer, source, value, scheduler)); } public static IAsyncObservable Prepend(this IAsyncObservable source, params TSource[] values) { if (source == null) throw new ArgumentNullException(nameof(source)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create(async observer => await source.SubscribeSafeAsync(await AsyncObserver.Prepend(observer, values).ConfigureAwait(false)).ConfigureAwait(false)); } public static IAsyncObservable Prepend(this IAsyncObservable source, IAsyncScheduler scheduler, params TSource[] values) { if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create(observer => AsyncObserver.Prepend(observer, source, scheduler, values)); } public static IAsyncObservable Prepend(this IAsyncObservable source, IEnumerable values) { if (source == null) throw new ArgumentNullException(nameof(source)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create(async observer => await source.SubscribeSafeAsync(await AsyncObserver.Prepend(observer, values).ConfigureAwait(false)).ConfigureAwait(false)); } public static IAsyncObservable Prepend(this IAsyncObservable source, IAsyncScheduler scheduler, IEnumerable values) { if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create(observer => AsyncObserver.Prepend(observer, source, scheduler, values)); } } partial class AsyncObserver { // REVIEW: There's some asymmetry on these overloads. Should standardize to Concat style. public static Task> Prepend(IAsyncObserver observer, TSource value) { if (observer == null) throw new ArgumentNullException(nameof(observer)); return Core(); async Task> Core() { await observer.OnNextAsync(value).ConfigureAwait(false); return observer; } } public static Task Prepend(IAsyncObserver observer, IAsyncObservable source, TSource value, IAsyncScheduler scheduler) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); return Core(); async Task Core() { var subscription = new SingleAssignmentAsyncDisposable(); var task = await scheduler.ScheduleAsync(async ct => { if (ct.IsCancellationRequested) return; await observer.OnNextAsync(value).RendezVous(scheduler, ct); if (ct.IsCancellationRequested) return; var inner = await source.SubscribeSafeAsync(observer).ConfigureAwait(false); await subscription.AssignAsync(inner).RendezVous(scheduler, ct); }).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(task, subscription); } } public static Task> Prepend(IAsyncObserver observer, params TSource[] values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (values == null) throw new ArgumentNullException(nameof(values)); return Core(); async Task> Core() { foreach (var value in values) { await observer.OnNextAsync(value).ConfigureAwait(false); } return observer; } } public static Task Prepend(IAsyncObserver observer, IAsyncObservable source, IAsyncScheduler scheduler, params TSource[] values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); if (values == null) throw new ArgumentNullException(nameof(values)); return Core(); async Task Core() { var subscription = new SingleAssignmentAsyncDisposable(); var task = await scheduler.ScheduleAsync(async ct => { if (ct.IsCancellationRequested) return; for (var i = 0; i < values.Length && !ct.IsCancellationRequested; i++) { await observer.OnNextAsync(values[i]).RendezVous(scheduler, ct); } if (ct.IsCancellationRequested) return; var inner = await source.SubscribeSafeAsync(observer).ConfigureAwait(false); await subscription.AssignAsync(inner).RendezVous(scheduler, ct); }).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(task, subscription); } } public static Task> Prepend(IAsyncObserver observer, IEnumerable values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (values == null) throw new ArgumentNullException(nameof(values)); return Core(); async Task> Core() { foreach (var value in values) { await observer.OnNextAsync(value).ConfigureAwait(false); } return observer; } } public static Task Prepend(IAsyncObserver observer, IAsyncObservable source, IAsyncScheduler scheduler, IEnumerable values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (source == null) throw new ArgumentNullException(nameof(source)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); if (values == null) throw new ArgumentNullException(nameof(values)); return Core(); async Task Core() { var subscription = new SingleAssignmentAsyncDisposable(); var task = await scheduler.ScheduleAsync(async ct => { if (ct.IsCancellationRequested) return; var e = default(IEnumerator); try { e = values.GetEnumerator(); } catch (Exception ex) { await observer.OnErrorAsync(ex).RendezVous(scheduler, ct); return; } using (e) { while (!ct.IsCancellationRequested) { var b = default(bool); var value = default(TSource); try { b = e.MoveNext(); if (b) { value = e.Current; } } catch (Exception ex) { await observer.OnErrorAsync(ex).RendezVous(scheduler, ct); return; } if (b) { await observer.OnNextAsync(value).RendezVous(scheduler, ct); } else { break; } } } if (ct.IsCancellationRequested) return; var inner = await source.SubscribeSafeAsync(observer).ConfigureAwait(false); await subscription.AssignAsync(inner).RendezVous(scheduler, ct); }).ConfigureAwait(false); return StableCompositeAsyncDisposable.Create(task, subscription); } } } }