// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT License. // See the LICENSE file in the project root for more information. using System.Collections.Generic; using System.Reactive.Concurrency; using System.Reactive.Disposables; namespace System.Reactive.Linq { public partial class AsyncObservable { public static IAsyncObservable Append(this IAsyncObservable source, TSource value) { if (source == null) throw new ArgumentNullException(nameof(source)); return CreateAsyncObservable.From( source, value, static(source, value, observer) => source.SubscribeSafeAsync(AsyncObserver.Append(observer, value))); } public static IAsyncObservable Append(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( source, (value, scheduler), static async (source, state, observer) => { var d = new CompositeAsyncDisposable(); var (sink, disposable) = AsyncObserver.Append(observer, state.value, state.scheduler); await d.AddAsync(disposable).ConfigureAwait(false); var subscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false); await d.AddAsync(subscription).ConfigureAwait(false); return d; }); } public static IAsyncObservable Append(this IAsyncObservable source, params TSource[] values) { if (source == null) throw new ArgumentNullException(nameof(source)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create( source, values, static (source, values, observer) => source.SubscribeSafeAsync(AsyncObserver.Append(observer, values))); } public static IAsyncObservable Append(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( source, (scheduler, values), static async (source, state, observer) => { var d = new CompositeAsyncDisposable(); var (sink, disposable) = AsyncObserver.Append(observer, state.scheduler, state.values); await d.AddAsync(disposable).ConfigureAwait(false); var subscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false); await d.AddAsync(subscription).ConfigureAwait(false); return d; }); } public static IAsyncObservable Append(this IAsyncObservable source, IEnumerable values) { if (source == null) throw new ArgumentNullException(nameof(source)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create( source, values, static (source, values, observer) => source.SubscribeSafeAsync(AsyncObserver.Append(observer, values))); } public static IAsyncObservable Append(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( source, (scheduler, values), static async (source, state, observer) => { var d = new CompositeAsyncDisposable(); var (sink, disposable) = AsyncObserver.Append(observer, state.scheduler, state.values); await d.AddAsync(disposable).ConfigureAwait(false); var subscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false); await d.AddAsync(subscription).ConfigureAwait(false); return d; }); } } public partial class AsyncObserver { public static IAsyncObserver Append(IAsyncObserver observer, TSource value) { if (observer == null) throw new ArgumentNullException(nameof(observer)); return Create( observer.OnNextAsync, observer.OnErrorAsync, async () => { await observer.OnNextAsync(value).ConfigureAwait(false); await observer.OnCompletedAsync().ConfigureAwait(false); } ); } public static (IAsyncObserver, IAsyncDisposable) Append(IAsyncObserver observer, TSource value, IAsyncScheduler scheduler) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); var d = new SingleAssignmentAsyncDisposable(); return ( Create( observer.OnNextAsync, observer.OnErrorAsync, async () => { var task = await scheduler.ScheduleAsync(async ct => { if (!ct.IsCancellationRequested) { await observer.OnNextAsync(value).RendezVous(scheduler, ct); await observer.OnCompletedAsync().RendezVous(scheduler, ct); } }).ConfigureAwait(false); await d.AssignAsync(task).ConfigureAwait(false); } ), d ); } public static IAsyncObserver Append(IAsyncObserver observer, params TSource[] values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create( observer.OnNextAsync, observer.OnErrorAsync, async () => { foreach (var value in values) { await observer.OnNextAsync(value).ConfigureAwait(false); } await observer.OnCompletedAsync().ConfigureAwait(false); } ); } public static (IAsyncObserver, IAsyncDisposable) Append(IAsyncObserver observer, IAsyncScheduler scheduler, params TSource[] values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (values == null) throw new ArgumentNullException(nameof(values)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); var d = new SingleAssignmentAsyncDisposable(); return ( Create( observer.OnNextAsync, observer.OnErrorAsync, async () => { var task = await scheduler.ScheduleAsync(async ct => { for (var i = 0; i < values.Length && !ct.IsCancellationRequested; i++) { await observer.OnNextAsync(values[i]).RendezVous(scheduler, ct); } await observer.OnCompletedAsync().RendezVous(scheduler, ct); }).ConfigureAwait(false); await d.AssignAsync(task).ConfigureAwait(false); } ), d ); } public static IAsyncObserver Append(IAsyncObserver observer, IEnumerable values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (values == null) throw new ArgumentNullException(nameof(values)); return Create( observer.OnNextAsync, observer.OnErrorAsync, async () => { foreach (var value in values) { await observer.OnNextAsync(value).ConfigureAwait(false); } await observer.OnCompletedAsync().ConfigureAwait(false); } ); } public static (IAsyncObserver, IAsyncDisposable) Append(IAsyncObserver observer, IAsyncScheduler scheduler, IEnumerable values) { if (observer == null) throw new ArgumentNullException(nameof(observer)); if (values == null) throw new ArgumentNullException(nameof(values)); if (scheduler == null) throw new ArgumentNullException(nameof(scheduler)); var d = new SingleAssignmentAsyncDisposable(); return ( Create( observer.OnNextAsync, observer.OnErrorAsync, async () => { var task = await scheduler.ScheduleAsync(async ct => { 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; } } } await observer.OnCompletedAsync().RendezVous(scheduler, ct); }).ConfigureAwait(false); await d.AssignAsync(task).ConfigureAwait(false); } ), d ); } } }