123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292 |
- // 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;
- using System.Reactive.Concurrency;
- using System.Reactive.Linq;
- using System.Threading;
- using Microsoft.Reactive.Testing;
- using Microsoft.VisualStudio.TestTools.UnitTesting;
- using Assert = Xunit.Assert;
- namespace ReactiveTests.Tests
- {
- [TestClass]
- public partial class ObservableSafetyTest : ReactiveTest
- {
- [TestMethod]
- public void SubscribeSafe_ArgumentChecking()
- {
- ReactiveAssert.Throws<ArgumentNullException>(() => ObservableExtensions.SubscribeSafe(default, Observer.Create<int>(_ => { })));
- ReactiveAssert.Throws<ArgumentNullException>(() => ObservableExtensions.SubscribeSafe(Observable.Return(42), default));
- }
- [TestMethod]
- public void Safety_Subscription1()
- {
- var ex = new Exception();
- var xs = new RogueObservable(ex);
- var res = xs.Where(x => true).Select(x => x);
- var err = default(Exception);
- var d = res.Subscribe(x => { Assert.True(false); }, ex_ => { err = ex_; }, () => { Assert.True(false); });
- Assert.Same(ex, err);
- d.Dispose();
- }
- [TestMethod]
- public void Safety_Subscription2()
- {
- var ex = new Exception();
- var scheduler = new TestScheduler();
- var xs = scheduler.CreateHotObservable(
- OnNext(210, 42),
- OnNext(220, 43),
- OnNext(230, 44),
- OnNext(240, 45),
- OnCompleted<int>(250)
- );
- var ys = new RogueObservable(ex);
- var res = scheduler.Start(() =>
- xs.Merge(ys)
- );
- res.Messages.AssertEqual(
- OnError<int>(200, ex)
- );
- xs.Subscriptions.AssertEqual(
- Subscribe(200, 200)
- );
- }
- private class RogueObservable : IObservable<int>
- {
- private readonly Exception _ex;
- public RogueObservable(Exception ex)
- {
- _ex = ex;
- }
- public IDisposable Subscribe(IObserver<int> observer)
- {
- throw _ex;
- }
- }
- [TestMethod]
- public void ObservableBase_ObserverThrows()
- {
- var ex = new Exception();
- var failed = new ManualResetEvent(false);
- var disposed = new ManualResetEvent(false);
- var err = default(Exception);
- var xs = Observable.Create<int>(observer =>
- {
- Scheduler.Default.Schedule(() =>
- {
- try
- {
- observer.OnNext(42);
- }
- catch (Exception ex_)
- {
- err = ex_;
- failed.Set();
- }
- });
- return () => { disposed.Set(); };
- });
- xs.Subscribe(x =>
- {
- throw ex;
- });
- // Can't use WaitAll - we're on an STA thread.
- disposed.WaitOne();
- failed.WaitOne();
- Assert.Same(ex, err);
- }
- [TestMethod]
- public void ObservableBase_ObserverThrows_CustomObserver()
- {
- var ex = new Exception();
- var failed = new ManualResetEvent(false);
- var disposed = new ManualResetEvent(false);
- var err = default(Exception);
- var xs = Observable.Create<int>(observer =>
- {
- Scheduler.Default.Schedule(() =>
- {
- try
- {
- observer.OnNext(42);
- }
- catch (Exception ex_)
- {
- err = ex_;
- failed.Set();
- }
- });
- return () => { disposed.Set(); };
- });
- xs.Subscribe(new MyObserver(_ => true, ex));
- // Can't use WaitAll - we're on an STA thread.
- disposed.WaitOne();
- failed.WaitOne();
- Assert.Same(ex, err);
- }
- [TestMethod]
- public void Producer_ObserverThrows()
- {
- var ex = new Exception();
- var scheduler = new TestScheduler();
- var xs = scheduler.CreateHotObservable(
- OnNext(210, 1),
- OnNext(220, 2),
- OnNext(230, 3)
- );
- var ys = scheduler.CreateHotObservable(
- OnNext(215, 1),
- OnNext(225, 2),
- OnNext(235, 3)
- );
- var res = xs.CombineLatest(ys, (x, y) => x + y); // This creates a Producer object
- scheduler.ScheduleAbsolute(200, () =>
- {
- res.Subscribe(z =>
- {
- if (z == 4)
- {
- throw ex;
- }
- });
- });
- try
- {
- scheduler.Start();
- Assert.True(false);
- }
- catch (Exception err)
- {
- Assert.Same(ex, err);
- }
- Assert.Equal(225, scheduler.Clock);
- xs.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- ys.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- }
- [TestMethod]
- public void Producer_ObserverThrows_CustomObserver()
- {
- var ex = new Exception();
- var scheduler = new TestScheduler();
- var xs = scheduler.CreateHotObservable(
- OnNext(210, 1),
- OnNext(220, 2),
- OnNext(230, 3)
- );
- var ys = scheduler.CreateHotObservable(
- OnNext(215, 1),
- OnNext(225, 2),
- OnNext(235, 3)
- );
- var res = xs.CombineLatest(ys, (x, y) => x + y); // This creates a Producer object
- scheduler.ScheduleAbsolute(200, () =>
- {
- res.Subscribe(new MyObserver(x => x == 4, ex));
- });
- try
- {
- scheduler.Start();
- Assert.True(false);
- }
- catch (Exception err)
- {
- Assert.Same(ex, err);
- }
- Assert.Equal(225, scheduler.Clock);
- xs.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- ys.Subscriptions.AssertEqual(
- Subscribe(200, 225)
- );
- }
- private class MyObserver : ObserverBase<int>
- {
- private readonly Func<int, bool> _predicate;
- private readonly Exception _exception;
- public MyObserver(Func<int, bool> predicate, Exception exception)
- {
- _predicate = predicate;
- _exception = exception;
- }
- protected override void OnNextCore(int value)
- {
- if (_predicate(value))
- {
- throw _exception;
- }
- }
- protected override void OnErrorCore(Exception error)
- {
- }
- protected override void OnCompletedCore()
- {
- }
- }
- }
- }
|