// 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. 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 IDisposable Run(IObserver observer, IDisposable cancel, Action setSink) { var sink = new _(_accumulator, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } private sealed class _ : Sink, IObserver { private readonly Func _accumulator; private TSource _accumulation; private bool _hasAccumulation; public _(Func accumulator, IObserver observer, IDisposable cancel) : base(observer, cancel) { _accumulator = accumulator; _accumulation = default(TSource); _hasAccumulation = false; } public void OnNext(TSource value) { if (!_hasAccumulation) { _accumulation = value; _hasAccumulation = true; } else { try { _accumulation = _accumulator(_accumulation, value); } catch (Exception exception) { base._observer.OnError(exception); base.Dispose(); } } } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { if (!_hasAccumulation) { base._observer.OnError(new InvalidOperationException(Strings_Linq.NO_ELEMENTS)); base.Dispose(); } else { base._observer.OnNext(_accumulation); base._observer.OnCompleted(); base.Dispose(); } } } } 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 IDisposable Run(IObserver observer, IDisposable cancel, Action setSink) { var sink = new _(_seed, _accumulator, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } private sealed class _ : Sink, IObserver { private readonly Func _accumulator; private TAccumulate _accumulation; public _(TAccumulate seed, Func accumulator, IObserver observer, IDisposable cancel) : base(observer, cancel) { _accumulator = accumulator; _accumulation = seed; } public void OnNext(TSource value) { try { _accumulation = _accumulator(_accumulation, value); } catch (Exception exception) { base._observer.OnError(exception); base.Dispose(); } } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { base._observer.OnNext(_accumulation); base._observer.OnCompleted(); base.Dispose(); } } } 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 IDisposable Run(IObserver observer, IDisposable cancel, Action setSink) { var sink = new _(this, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } private sealed class _ : Sink, IObserver { // CONSIDER: This sink has a parent reference that can be considered for removal. private readonly Aggregate _parent; private TAccumulate _accumulation; public _(Aggregate parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { _parent = parent; _accumulation = _parent._seed; } public void OnNext(TSource value) { try { _accumulation = _parent._accumulator(_accumulation, value); } catch (Exception exception) { base._observer.OnError(exception); base.Dispose(); } } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { var result = default(TResult); try { result = _parent._resultSelector(_accumulation); } catch (Exception exception) { base._observer.OnError(exception); base.Dispose(); return; } base._observer.OnNext(result); base._observer.OnCompleted(); base.Dispose(); } } } }