1
0

RepeatWhen.cs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the MIT License.
  3. // See the LICENSE file in the project root for more information.
  4. #nullable disable
  5. using System.Reactive.Disposables;
  6. using System.Reactive.Subjects;
  7. using System.Threading;
  8. namespace System.Reactive.Linq.ObservableImpl
  9. {
  10. internal sealed class RepeatWhen<T, U> : IObservable<T>
  11. {
  12. private readonly IObservable<T> _source;
  13. private readonly Func<IObservable<object>, IObservable<U>> _handler;
  14. internal RepeatWhen(IObservable<T> source, Func<IObservable<object>, IObservable<U>> handler)
  15. {
  16. _source = source;
  17. _handler = handler;
  18. }
  19. public IDisposable Subscribe(IObserver<T> observer)
  20. {
  21. if (observer == null)
  22. {
  23. throw new ArgumentNullException(nameof(observer));
  24. }
  25. var completeSignals = new Subject<object>();
  26. IObservable<U> redo;
  27. try
  28. {
  29. redo = _handler(completeSignals);
  30. if (redo == null)
  31. {
  32. throw new NullReferenceException("The handler returned a null IObservable");
  33. }
  34. }
  35. catch (Exception ex)
  36. {
  37. observer.OnError(ex);
  38. return Disposable.Empty;
  39. }
  40. var parent = new MainObserver(observer, _source, new RedoSerializedObserver<object>(completeSignals));
  41. var d = redo.SubscribeSafe(parent.HandlerConsumer);
  42. Disposable.SetSingle(ref parent.HandlerUpstream, d);
  43. parent.HandlerNext();
  44. return parent;
  45. }
  46. private sealed class MainObserver : Sink<T>, IObserver<T>
  47. {
  48. private readonly IObserver<Exception> _errorSignal;
  49. internal readonly HandlerObserver HandlerConsumer;
  50. private readonly IObservable<T> _source;
  51. private IDisposable _upstream;
  52. internal IDisposable HandlerUpstream;
  53. private int _trampoline;
  54. private int _halfSerializer;
  55. private Exception _error;
  56. internal MainObserver(IObserver<T> downstream, IObservable<T> source, IObserver<Exception> errorSignal) : base(downstream)
  57. {
  58. _source = source;
  59. _errorSignal = errorSignal;
  60. HandlerConsumer = new HandlerObserver(this);
  61. }
  62. protected override void Dispose(bool disposing)
  63. {
  64. if (disposing)
  65. {
  66. Disposable.Dispose(ref _upstream);
  67. Disposable.Dispose(ref HandlerUpstream);
  68. }
  69. base.Dispose(disposing);
  70. }
  71. public void OnCompleted()
  72. {
  73. if (Disposable.TrySetSerial(ref _upstream, null))
  74. {
  75. _errorSignal.OnNext(null);
  76. }
  77. }
  78. public void OnError(Exception error)
  79. {
  80. HalfSerializer.ForwardOnError(this, error, ref _halfSerializer, ref _error);
  81. }
  82. public void OnNext(T value)
  83. {
  84. HalfSerializer.ForwardOnNext(this, value, ref _halfSerializer, ref _error);
  85. }
  86. private void HandlerError(Exception error)
  87. {
  88. HalfSerializer.ForwardOnError(this, error, ref _halfSerializer, ref _error);
  89. }
  90. private void HandlerComplete()
  91. {
  92. HalfSerializer.ForwardOnCompleted(this, ref _halfSerializer, ref _error);
  93. }
  94. internal void HandlerNext()
  95. {
  96. if (Interlocked.Increment(ref _trampoline) == 1)
  97. {
  98. do
  99. {
  100. var sad = new SingleAssignmentDisposable();
  101. if (Interlocked.CompareExchange(ref _upstream, sad, null) != null)
  102. {
  103. return;
  104. }
  105. sad.Disposable = _source.SubscribeSafe(this);
  106. }
  107. while (Interlocked.Decrement(ref _trampoline) != 0);
  108. }
  109. }
  110. internal sealed class HandlerObserver : IObserver<U>
  111. {
  112. private readonly MainObserver _main;
  113. internal HandlerObserver(MainObserver main)
  114. {
  115. _main = main;
  116. }
  117. public void OnCompleted()
  118. {
  119. _main.HandlerComplete();
  120. }
  121. public void OnError(Exception error)
  122. {
  123. _main.HandlerError(error);
  124. }
  125. public void OnNext(U value)
  126. {
  127. _main.HandlerNext();
  128. }
  129. }
  130. }
  131. }
  132. }