// 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. namespace System.Reactive.Linq.ObservableImpl { internal sealed class Aggregate : Producer._> { private readonly IObservable _source; private readonly Func _accumulator; public Aggregate(IObservable source, Func accumulator) { _source = source; _accumulator = accumulator; } protected override _ CreateSink(IObserver observer) => new(_accumulator, observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private readonly Func _accumulator; private TSource? _accumulation; private bool _hasAccumulation; public _(Func accumulator, IObserver observer) : base(observer) { _accumulator = accumulator; } public override void OnNext(TSource value) { if (!_hasAccumulation) { _accumulation = value; _hasAccumulation = true; } else { try { _accumulation = _accumulator(_accumulation!, value); } catch (Exception exception) { _accumulation = default; ForwardOnError(exception); } } } public override void OnError(Exception error) { _accumulation = default; ForwardOnError(error); } public override void OnCompleted() { if (!_hasAccumulation) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { var accumulation = _accumulation!; _accumulation = default; ForwardOnNext(accumulation); ForwardOnCompleted(); } } } } internal sealed class Aggregate : Producer._> { private readonly IObservable _source; private readonly TAccumulate _seed; private readonly Func _accumulator; public Aggregate(IObservable source, TAccumulate seed, Func accumulator) { _source = source; _seed = seed; _accumulator = accumulator; } protected override _ CreateSink(IObserver observer) => new(_seed, _accumulator, observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : Sink { private readonly Func _accumulator; private TAccumulate? _accumulation; public _(TAccumulate seed, Func accumulator, IObserver observer) : base(observer) { _accumulator = accumulator; _accumulation = seed; } public override void OnNext(TSource value) { try { _accumulation = _accumulator(_accumulation!, value); } catch (Exception exception) { _accumulation = default; ForwardOnError(exception); } } public override void OnError(Exception error) { _accumulation = default; ForwardOnError(error); } public override void OnCompleted() { var accumulation = _accumulation!; _accumulation = default; ForwardOnNext(accumulation); ForwardOnCompleted(); } } } internal sealed class Aggregate : Producer._> { private readonly IObservable _source; private readonly TAccumulate _seed; private readonly Func _accumulator; private readonly Func _resultSelector; public Aggregate(IObservable source, TAccumulate seed, Func accumulator, Func resultSelector) { _source = source; _seed = seed; _accumulator = accumulator; _resultSelector = resultSelector; } protected override _ CreateSink(IObserver observer) => new(this, observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : Sink { private readonly Func _accumulator; private readonly Func _resultSelector; private TAccumulate? _accumulation; public _(Aggregate parent, IObserver observer) : base(observer) { _accumulator = parent._accumulator; _resultSelector = parent._resultSelector; _accumulation = parent._seed; } public override void OnNext(TSource value) { try { _accumulation = _accumulator(_accumulation!, value); } catch (Exception exception) { _accumulation = default; ForwardOnError(exception); } } public override void OnError(Exception error) { _accumulation = default; ForwardOnError(error); } public override void OnCompleted() { var accumulation = _accumulation!; _accumulation = default; TResult result; try { result = _resultSelector(accumulation); } catch (Exception exception) { ForwardOnError(exception); return; } ForwardOnNext(result); ForwardOnCompleted(); } } } }