// 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. using System.Reactive.Concurrency; using System.Reactive.Disposables; namespace System.Reactive.Linq.ObservableImpl { internal sealed class Sample : Producer._> { private readonly IObservable _source; private readonly IObservable _sampler; public Sample(IObservable source, IObservable sampler) { _source = source; _sampler = sampler; } protected override _ CreateSink(IObserver observer) => new _(observer); protected override void Run(_ sink) => sink.Run(this); internal sealed class _ : IdentitySink { private readonly object _gate = new object(); public _(IObserver observer) : base(observer) { } private IDisposable _sourceDisposable; private IDisposable _samplerDisposable; private bool _hasValue; private TSource _value; private bool _sourceAtEnd; private bool _samplerAtEnd; public void Run(Sample parent) { Disposable.SetSingle(ref _sourceDisposable, parent._source.SubscribeSafe(this)); Disposable.SetSingle(ref _samplerDisposable, parent._sampler.SubscribeSafe(new SampleObserver(this))); } protected override void Dispose(bool disposing) { if (disposing) { Disposable.TryDispose(ref _sourceDisposable); Disposable.TryDispose(ref _samplerDisposable); } base.Dispose(disposing); } public override void OnNext(TSource value) { lock (_gate) { _hasValue = true; _value = value; } } public override void OnError(Exception error) { lock (_gate) { ForwardOnError(error); } } public override void OnCompleted() { lock (_gate) { _sourceAtEnd = true; if (_samplerAtEnd) ForwardOnCompleted(); else Disposable.TryDispose(ref _sourceDisposable); } } private sealed class SampleObserver : IObserver { private readonly _ _parent; public SampleObserver(_ parent) { _parent = parent; } public void OnNext(TSample value) { lock (_parent._gate) { if (_parent._hasValue) { _parent._hasValue = false; _parent.ForwardOnNext(_parent._value); } if (_parent._sourceAtEnd) { _parent.ForwardOnCompleted(); } } } public void OnError(Exception error) { // BREAKING CHANGE v2 > v1.x - This error used to be swallowed lock (_parent._gate) { _parent.ForwardOnError(error); } } public void OnCompleted() { lock (_parent._gate) { _parent._samplerAtEnd = true; if (_parent._hasValue) { _parent._hasValue = false; _parent.ForwardOnNext(_parent._value); } if (_parent._sourceAtEnd) { _parent.ForwardOnCompleted(); } else Disposable.TryDispose(ref _parent._samplerDisposable); } } } } } internal sealed class Sample : Producer._> { private readonly IObservable _source; private readonly TimeSpan _interval; private readonly IScheduler _scheduler; public Sample(IObservable source, TimeSpan interval, IScheduler scheduler) { _source = source; _interval = interval; _scheduler = scheduler; } protected override _ CreateSink(IObserver observer) => new _(observer); protected override void Run(_ sink) => sink.Run(this); internal sealed class _ : IdentitySink { private object _gate = new object(); public _(IObserver observer) : base(observer) { } private IDisposable _sourceDisposable; private bool _hasValue; private TSource _value; private bool _atEnd; public void Run(Sample parent) { Disposable.SetSingle(ref _sourceDisposable, parent._source.SubscribeSafe(this)); SetUpstream(parent._scheduler.SchedulePeriodic(parent._interval, Tick)); } protected override void Dispose(bool disposing) { if (disposing) { Disposable.TryDispose(ref _sourceDisposable); } base.Dispose(disposing); } private void Tick() { lock (_gate) { if (_hasValue) { _hasValue = false; ForwardOnNext(_value); } if (_atEnd) { ForwardOnCompleted(); } } } public override void OnNext(TSource value) { lock (_gate) { _hasValue = true; _value = value; } } public override void OnError(Exception error) { lock (_gate) { ForwardOnError(error); } } public override void OnCompleted() { lock (_gate) { _atEnd = true; Disposable.TryDispose(ref _sourceDisposable); } } } } }