219 lines
7.2 KiB
C#
219 lines
7.2 KiB
C#
// 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<TSource> : Producer<TSource, Aggregate<TSource>._>
|
|
{
|
|
private readonly IObservable<TSource> _source;
|
|
private readonly Func<TSource, TSource, TSource> _accumulator;
|
|
|
|
public Aggregate(IObservable<TSource> source, Func<TSource, TSource, TSource> accumulator)
|
|
{
|
|
_source = source;
|
|
_accumulator = accumulator;
|
|
}
|
|
|
|
protected override _ CreateSink(IObserver<TSource> observer) => new(_accumulator, observer);
|
|
|
|
protected override void Run(_ sink) => sink.Run(_source);
|
|
|
|
internal sealed class _ : IdentitySink<TSource>
|
|
{
|
|
private readonly Func<TSource, TSource, TSource> _accumulator;
|
|
private TSource? _accumulation;
|
|
private bool _hasAccumulation;
|
|
|
|
public _(Func<TSource, TSource, TSource> accumulator, IObserver<TSource> 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<TSource, TAccumulate> : Producer<TAccumulate, Aggregate<TSource, TAccumulate>._>
|
|
{
|
|
private readonly IObservable<TSource> _source;
|
|
private readonly TAccumulate _seed;
|
|
private readonly Func<TAccumulate, TSource, TAccumulate> _accumulator;
|
|
|
|
public Aggregate(IObservable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator)
|
|
{
|
|
_source = source;
|
|
_seed = seed;
|
|
_accumulator = accumulator;
|
|
}
|
|
|
|
protected override _ CreateSink(IObserver<TAccumulate> observer) => new(_seed, _accumulator, observer);
|
|
|
|
protected override void Run(_ sink) => sink.Run(_source);
|
|
|
|
internal sealed class _ : Sink<TSource, TAccumulate>
|
|
{
|
|
private readonly Func<TAccumulate, TSource, TAccumulate> _accumulator;
|
|
private TAccumulate? _accumulation;
|
|
|
|
public _(TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, IObserver<TAccumulate> 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<TSource, TAccumulate, TResult> : Producer<TResult, Aggregate<TSource, TAccumulate, TResult>._>
|
|
{
|
|
private readonly IObservable<TSource> _source;
|
|
private readonly TAccumulate _seed;
|
|
private readonly Func<TAccumulate, TSource, TAccumulate> _accumulator;
|
|
private readonly Func<TAccumulate, TResult> _resultSelector;
|
|
|
|
public Aggregate(IObservable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, Func<TAccumulate, TResult> resultSelector)
|
|
{
|
|
_source = source;
|
|
_seed = seed;
|
|
_accumulator = accumulator;
|
|
_resultSelector = resultSelector;
|
|
}
|
|
|
|
protected override _ CreateSink(IObserver<TResult> observer) => new(this, observer);
|
|
|
|
protected override void Run(_ sink) => sink.Run(_source);
|
|
|
|
internal sealed class _ : Sink<TSource, TResult>
|
|
{
|
|
private readonly Func<TAccumulate, TSource, TAccumulate> _accumulator;
|
|
private readonly Func<TAccumulate, TResult> _resultSelector;
|
|
|
|
private TAccumulate? _accumulation;
|
|
|
|
public _(Aggregate<TSource, TAccumulate, TResult> parent, IObserver<TResult> 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();
|
|
}
|
|
}
|
|
}
|
|
}
|