Kit.Core/LibExternal/System.Reactive/Linq/Observable/TakeUntil.cs

184 lines
5.8 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.
using System.Reactive.Concurrency;
using System.Reactive.Disposables;
namespace System.Reactive.Linq.ObservableImpl
{
internal sealed class TakeUntil<TSource, TOther> : Producer<TSource, TakeUntil<TSource, TOther>._>
{
private readonly IObservable<TSource> _source;
private readonly IObservable<TOther> _other;
public TakeUntil(IObservable<TSource> source, IObservable<TOther> other)
{
_source = source;
_other = other;
}
protected override _ CreateSink(IObserver<TSource> observer) => new(observer);
protected override void Run(_ sink) => sink.Run(this);
internal sealed class _ : IdentitySink<TSource>
{
private SingleAssignmentDisposableValue _otherDisposable;
private int _halfSerializer;
private Exception? _error;
public _(IObserver<TSource> observer)
: base(observer)
{
}
public void Run(TakeUntil<TSource, TOther> parent)
{
_otherDisposable.Disposable = parent._other.Subscribe(new OtherObserver(this));
Run(parent._source);
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
if (!_otherDisposable.IsDisposed)
{
_otherDisposable.Dispose();
}
}
base.Dispose(disposing);
}
public override void OnNext(TSource value)
{
HalfSerializer.ForwardOnNext(this, value, ref _halfSerializer, ref _error);
}
public override void OnError(Exception ex)
{
HalfSerializer.ForwardOnError(this, ex, ref _halfSerializer, ref _error);
}
public override void OnCompleted()
{
HalfSerializer.ForwardOnCompleted(this, ref _halfSerializer, ref _error);
}
private sealed class OtherObserver : IObserver<TOther>
{
private readonly _ _parent;
public OtherObserver(_ parent)
{
_parent = parent;
}
public void OnCompleted()
{
// Completion doesn't mean termination in Rx.NET for this operator
_parent._otherDisposable.Dispose();
}
public void OnError(Exception error)
{
HalfSerializer.ForwardOnError(_parent, error, ref _parent._halfSerializer, ref _parent._error);
}
public void OnNext(TOther value)
{
HalfSerializer.ForwardOnCompleted(_parent, ref _parent._halfSerializer, ref _parent._error);
}
}
}
}
internal sealed class TakeUntil<TSource> : Producer<TSource, TakeUntil<TSource>._>
{
private readonly IObservable<TSource> _source;
private readonly DateTimeOffset _endTime;
internal readonly IScheduler _scheduler;
public TakeUntil(IObservable<TSource> source, DateTimeOffset endTime, IScheduler scheduler)
{
_source = source;
_endTime = endTime;
_scheduler = scheduler;
}
public IObservable<TSource> Combine(DateTimeOffset endTime)
{
//
// Minimum semantics:
//
// t 0--1--2--3--4--5--6--7-> t 0--1--2--3--4--5--6--7->
//
// xs --o--o--o--o--o--o--| xs --o--o--o--o--o--o--|
// xs.TU(5AM) --o--o--o--o--o| xs.TU(3AM) --o--o--o|
// xs.TU(5AM).TU(3AM) --o--o--o| xs.TU(3AM).TU(5AM) --o--o--o|
//
if (_endTime <= endTime)
{
return this;
}
return new TakeUntil<TSource>(_source, endTime, _scheduler);
}
protected override _ CreateSink(IObserver<TSource> observer) => new(observer);
protected override void Run(_ sink) => sink.Run(this);
internal sealed class _ : IdentitySink<TSource>
{
private SingleAssignmentDisposableValue _timerDisposable;
private int _wip;
private Exception? _error;
public _(IObserver<TSource> observer)
: base(observer)
{
}
public void Run(TakeUntil<TSource> parent)
{
_timerDisposable.Disposable = parent._scheduler.ScheduleAction(this, parent._endTime, state => state.Tick());
Run(parent._source);
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
_timerDisposable.Dispose();
}
base.Dispose(disposing);
}
private void Tick()
{
OnCompleted();
}
public override void OnNext(TSource value)
{
HalfSerializer.ForwardOnNext(this, value, ref _wip, ref _error);
}
public override void OnError(Exception error)
{
HalfSerializer.ForwardOnError(this, error, ref _wip, ref _error);
}
public override void OnCompleted()
{
HalfSerializer.ForwardOnCompleted(this, ref _wip, ref _error);
}
}
}
}