// 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; using System.Reactive.Concurrency; using System.Reactive.Subjects; using Microsoft.Reactive.Testing; using Microsoft.VisualStudio.TestTools.UnitTesting; using Assert = Xunit.Assert; namespace ReactiveTests.Tests { [TestClass] public partial class BehaviorSubjectTest : ReactiveTest { [TestMethod] public void Subscribe_ArgumentChecking() { ReactiveAssert.Throws(() => new BehaviorSubject(1).Subscribe(null)); } [TestMethod] public void OnError_ArgumentChecking() { ReactiveAssert.Throws(() => new BehaviorSubject(1).OnError(null)); } [TestMethod] public void Infinite() { var scheduler = new TestScheduler(); var xs = scheduler.CreateHotObservable( OnNext(70, 1), OnNext(110, 2), OnNext(220, 3), OnNext(270, 4), OnNext(340, 5), OnNext(410, 6), OnNext(520, 7), OnNext(630, 8), OnNext(710, 9), OnNext(870, 10), OnNext(940, 11), OnNext(1020, 12) ); var subject = default(BehaviorSubject); var subscription = default(IDisposable); var results1 = scheduler.CreateObserver(); var subscription1 = default(IDisposable); var results2 = scheduler.CreateObserver(); var subscription2 = default(IDisposable); var results3 = scheduler.CreateObserver(); var subscription3 = default(IDisposable); scheduler.ScheduleAbsolute(100, () => subject = new BehaviorSubject(100)); scheduler.ScheduleAbsolute(200, () => subscription = xs.Subscribe(subject)); scheduler.ScheduleAbsolute(1000, () => subscription.Dispose()); scheduler.ScheduleAbsolute(300, () => subscription1 = subject.Subscribe(results1)); scheduler.ScheduleAbsolute(400, () => subscription2 = subject.Subscribe(results2)); scheduler.ScheduleAbsolute(900, () => subscription3 = subject.Subscribe(results3)); scheduler.ScheduleAbsolute(600, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(700, () => subscription2.Dispose()); scheduler.ScheduleAbsolute(800, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(950, () => subscription3.Dispose()); scheduler.Start(); results1.Messages.AssertEqual( OnNext(300, 4), OnNext(340, 5), OnNext(410, 6), OnNext(520, 7) ); results2.Messages.AssertEqual( OnNext(400, 5), OnNext(410, 6), OnNext(520, 7), OnNext(630, 8) ); results3.Messages.AssertEqual( OnNext(900, 10), OnNext(940, 11) ); } [TestMethod] public void Finite() { var scheduler = new TestScheduler(); var xs = scheduler.CreateHotObservable( OnNext(70, 1), OnNext(110, 2), OnNext(220, 3), OnNext(270, 4), OnNext(340, 5), OnNext(410, 6), OnNext(520, 7), OnCompleted(630), OnNext(640, 9), OnCompleted(650), OnError(660, new Exception()) ); var subject = default(BehaviorSubject); var subscription = default(IDisposable); var results1 = scheduler.CreateObserver(); var subscription1 = default(IDisposable); var results2 = scheduler.CreateObserver(); var subscription2 = default(IDisposable); var results3 = scheduler.CreateObserver(); var subscription3 = default(IDisposable); scheduler.ScheduleAbsolute(100, () => subject = new BehaviorSubject(100)); scheduler.ScheduleAbsolute(200, () => subscription = xs.Subscribe(subject)); scheduler.ScheduleAbsolute(1000, () => subscription.Dispose()); scheduler.ScheduleAbsolute(300, () => subscription1 = subject.Subscribe(results1)); scheduler.ScheduleAbsolute(400, () => subscription2 = subject.Subscribe(results2)); scheduler.ScheduleAbsolute(900, () => subscription3 = subject.Subscribe(results3)); scheduler.ScheduleAbsolute(600, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(700, () => subscription2.Dispose()); scheduler.ScheduleAbsolute(800, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(950, () => subscription3.Dispose()); scheduler.Start(); results1.Messages.AssertEqual( OnNext(300, 4), OnNext(340, 5), OnNext(410, 6), OnNext(520, 7) ); results2.Messages.AssertEqual( OnNext(400, 5), OnNext(410, 6), OnNext(520, 7), OnCompleted(630) ); results3.Messages.AssertEqual( OnCompleted(900) ); } [TestMethod] public void Error() { var scheduler = new TestScheduler(); var ex = new Exception(); var xs = scheduler.CreateHotObservable( OnNext(70, 1), OnNext(110, 2), OnNext(220, 3), OnNext(270, 4), OnNext(340, 5), OnNext(410, 6), OnNext(520, 7), OnError(630, ex), OnNext(640, 9), OnCompleted(650), OnError(660, new Exception()) ); var subject = default(BehaviorSubject); var subscription = default(IDisposable); var results1 = scheduler.CreateObserver(); var subscription1 = default(IDisposable); var results2 = scheduler.CreateObserver(); var subscription2 = default(IDisposable); var results3 = scheduler.CreateObserver(); var subscription3 = default(IDisposable); scheduler.ScheduleAbsolute(100, () => subject = new BehaviorSubject(100)); scheduler.ScheduleAbsolute(200, () => subscription = xs.Subscribe(subject)); scheduler.ScheduleAbsolute(1000, () => subscription.Dispose()); scheduler.ScheduleAbsolute(300, () => subscription1 = subject.Subscribe(results1)); scheduler.ScheduleAbsolute(400, () => subscription2 = subject.Subscribe(results2)); scheduler.ScheduleAbsolute(900, () => subscription3 = subject.Subscribe(results3)); scheduler.ScheduleAbsolute(600, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(700, () => subscription2.Dispose()); scheduler.ScheduleAbsolute(800, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(950, () => subscription3.Dispose()); scheduler.Start(); results1.Messages.AssertEqual( OnNext(300, 4), OnNext(340, 5), OnNext(410, 6), OnNext(520, 7) ); results2.Messages.AssertEqual( OnNext(400, 5), OnNext(410, 6), OnNext(520, 7), OnError(630, ex) ); results3.Messages.AssertEqual( OnError(900, ex) ); } [TestMethod] public void Canceled() { var scheduler = new TestScheduler(); var xs = scheduler.CreateHotObservable( OnCompleted(630), OnNext(640, 9), OnCompleted(650), OnError(660, new Exception()) ); var subject = default(BehaviorSubject); var subscription = default(IDisposable); var results1 = scheduler.CreateObserver(); var subscription1 = default(IDisposable); var results2 = scheduler.CreateObserver(); var subscription2 = default(IDisposable); var results3 = scheduler.CreateObserver(); var subscription3 = default(IDisposable); scheduler.ScheduleAbsolute(100, () => subject = new BehaviorSubject(100)); scheduler.ScheduleAbsolute(200, () => subscription = xs.Subscribe(subject)); scheduler.ScheduleAbsolute(1000, () => subscription.Dispose()); scheduler.ScheduleAbsolute(300, () => subscription1 = subject.Subscribe(results1)); scheduler.ScheduleAbsolute(400, () => subscription2 = subject.Subscribe(results2)); scheduler.ScheduleAbsolute(900, () => subscription3 = subject.Subscribe(results3)); scheduler.ScheduleAbsolute(600, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(700, () => subscription2.Dispose()); scheduler.ScheduleAbsolute(800, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(950, () => subscription3.Dispose()); scheduler.Start(); results1.Messages.AssertEqual( OnNext(300, 100) ); results2.Messages.AssertEqual( OnNext(400, 100), OnCompleted(630) ); results3.Messages.AssertEqual( OnCompleted(900) ); } [TestMethod] public void SubjectDisposed() { var scheduler = new TestScheduler(); var subject = default(BehaviorSubject); var results1 = scheduler.CreateObserver(); var subscription1 = default(IDisposable); var results2 = scheduler.CreateObserver(); var subscription2 = default(IDisposable); var results3 = scheduler.CreateObserver(); var subscription3 = default(IDisposable); scheduler.ScheduleAbsolute(100, () => subject = new BehaviorSubject(0)); scheduler.ScheduleAbsolute(200, () => subscription1 = subject.Subscribe(results1)); scheduler.ScheduleAbsolute(300, () => subscription2 = subject.Subscribe(results2)); scheduler.ScheduleAbsolute(400, () => subscription3 = subject.Subscribe(results3)); scheduler.ScheduleAbsolute(500, () => subscription1.Dispose()); scheduler.ScheduleAbsolute(600, () => subject.Dispose()); scheduler.ScheduleAbsolute(700, () => subscription2.Dispose()); scheduler.ScheduleAbsolute(800, () => subscription3.Dispose()); scheduler.ScheduleAbsolute(150, () => subject.OnNext(1)); scheduler.ScheduleAbsolute(250, () => subject.OnNext(2)); scheduler.ScheduleAbsolute(350, () => subject.OnNext(3)); scheduler.ScheduleAbsolute(450, () => subject.OnNext(4)); scheduler.ScheduleAbsolute(550, () => subject.OnNext(5)); scheduler.ScheduleAbsolute(650, () => ReactiveAssert.Throws(() => subject.OnNext(6))); scheduler.ScheduleAbsolute(750, () => ReactiveAssert.Throws(() => subject.OnCompleted())); scheduler.ScheduleAbsolute(850, () => ReactiveAssert.Throws(() => subject.OnError(new Exception()))); scheduler.ScheduleAbsolute(950, () => ReactiveAssert.Throws(() => subject.Subscribe())); scheduler.Start(); results1.Messages.AssertEqual( OnNext(200, 1), OnNext(250, 2), OnNext(350, 3), OnNext(450, 4) ); results2.Messages.AssertEqual( OnNext(300, 2), OnNext(350, 3), OnNext(450, 4), OnNext(550, 5) ); results3.Messages.AssertEqual( OnNext(400, 3), OnNext(450, 4), OnNext(550, 5) ); } [TestMethod] public void HasObservers() { var s = new BehaviorSubject(42); Assert.False(s.HasObservers); var d1 = s.Subscribe(_ => { }); Assert.True(s.HasObservers); d1.Dispose(); Assert.False(s.HasObservers); var d2 = s.Subscribe(_ => { }); Assert.True(s.HasObservers); var d3 = s.Subscribe(_ => { }); Assert.True(s.HasObservers); d2.Dispose(); Assert.True(s.HasObservers); d3.Dispose(); Assert.False(s.HasObservers); } [TestMethod] public void HasObservers_Dispose1() { var s = new BehaviorSubject(42); Assert.False(s.HasObservers); Assert.False(s.IsDisposed); var d = s.Subscribe(_ => { }); Assert.True(s.HasObservers); Assert.False(s.IsDisposed); s.Dispose(); Assert.False(s.HasObservers); Assert.True(s.IsDisposed); d.Dispose(); Assert.False(s.HasObservers); Assert.True(s.IsDisposed); } [TestMethod] public void HasObservers_Dispose2() { var s = new BehaviorSubject(42); Assert.False(s.HasObservers); Assert.False(s.IsDisposed); var d = s.Subscribe(_ => { }); Assert.True(s.HasObservers); Assert.False(s.IsDisposed); d.Dispose(); Assert.False(s.HasObservers); Assert.False(s.IsDisposed); s.Dispose(); Assert.False(s.HasObservers); Assert.True(s.IsDisposed); } [TestMethod] public void HasObservers_Dispose3() { var s = new BehaviorSubject(42); Assert.False(s.HasObservers); Assert.False(s.IsDisposed); s.Dispose(); Assert.False(s.HasObservers); Assert.True(s.IsDisposed); } [TestMethod] public void HasObservers_OnCompleted() { var s = new BehaviorSubject(42); Assert.False(s.HasObservers); var d = s.Subscribe(_ => { }); Assert.True(s.HasObservers); s.OnNext(42); Assert.True(s.HasObservers); s.OnCompleted(); Assert.False(s.HasObservers); } [TestMethod] public void HasObservers_OnError() { var s = new BehaviorSubject(42); Assert.False(s.HasObservers); var d = s.Subscribe(_ => { }, ex => { }); Assert.True(s.HasObservers); s.OnNext(42); Assert.True(s.HasObservers); s.OnError(new Exception()); Assert.False(s.HasObservers); } [TestMethod] public void Value_Initial() { var s = new BehaviorSubject(42); Assert.Equal(42, s.Value); Assert.True(s.TryGetValue(out var x)); Assert.Equal(42, x); } [TestMethod] public void Value_First() { var s = new BehaviorSubject(42); Assert.Equal(42, s.Value); Assert.True(s.TryGetValue(out var x)); Assert.Equal(42, x); s.OnNext(43); Assert.Equal(43, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(43, x); } [TestMethod] public void Value_Second() { var s = new BehaviorSubject(42); Assert.Equal(42, s.Value); Assert.True(s.TryGetValue(out var x)); Assert.Equal(42, x); s.OnNext(43); Assert.Equal(43, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(43, x); s.OnNext(44); Assert.Equal(44, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(44, x); } [TestMethod] public void Value_FrozenAfterOnCompleted() { var s = new BehaviorSubject(42); Assert.Equal(42, s.Value); Assert.True(s.TryGetValue(out var x)); Assert.Equal(42, x); s.OnNext(43); Assert.Equal(43, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(43, x); s.OnNext(44); Assert.Equal(44, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(44, x); s.OnCompleted(); Assert.Equal(44, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(44, x); s.OnNext(1234); Assert.Equal(44, s.Value); Assert.True(s.TryGetValue(out x)); Assert.Equal(44, x); } [TestMethod] public void Value_ThrowsAfterOnError() { var s = new BehaviorSubject(42); Assert.Equal(42, s.Value); s.OnError(new InvalidOperationException()); ReactiveAssert.Throws(() => { var ignored = s.Value; }); ReactiveAssert.Throws(() => { s.TryGetValue(out var x); }); } [TestMethod] public void Value_ThrowsOnDispose() { var s = new BehaviorSubject(42); Assert.Equal(42, s.Value); s.Dispose(); ReactiveAssert.Throws(() => { var ignored = s.Value; }); Assert.False(s.TryGetValue(out var x)); } } }