FirstOrDefaultTest.cs 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  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. using System;
  5. using System.Linq;
  6. using System.Reactive.Concurrency;
  7. using System.Reactive.Disposables;
  8. using System.Reactive.Linq;
  9. using System.Threading;
  10. using Microsoft.Reactive.Testing;
  11. using ReactiveTests.Dummies;
  12. using Microsoft.VisualStudio.TestTools.UnitTesting;
  13. using Assert = Xunit.Assert;
  14. namespace ReactiveTests.Tests
  15. {
  16. [TestClass]
  17. public class FirstOrDefaultTest : ReactiveTest
  18. {
  19. [TestMethod]
  20. public void FirstOrDefault_ArgumentChecking()
  21. {
  22. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.FirstOrDefault(default(IObservable<int>)));
  23. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.FirstOrDefault(default(IObservable<int>), _ => true));
  24. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.FirstOrDefault(DummyObservable<int>.Instance, default));
  25. }
  26. [TestMethod]
  27. public void FirstOrDefault_Empty()
  28. {
  29. Assert.Equal(default, Observable.Empty<int>().FirstOrDefault());
  30. }
  31. [TestMethod]
  32. public void FirstOrDefaultPredicate_Empty()
  33. {
  34. Assert.Equal(default, Observable.Empty<int>().FirstOrDefault(_ => true));
  35. }
  36. [TestMethod]
  37. public void FirstOrDefault_Return()
  38. {
  39. var value = 42;
  40. Assert.Equal(value, Observable.Return(value).FirstOrDefault());
  41. }
  42. [TestMethod]
  43. public void FirstOrDefault_Throw()
  44. {
  45. var ex = new Exception();
  46. var xs = Observable.Throw<int>(ex);
  47. ReactiveAssert.Throws(ex, () => xs.FirstOrDefault());
  48. }
  49. [TestMethod]
  50. public void FirstOrDefault_Range()
  51. {
  52. var value = 42;
  53. Assert.Equal(value, Observable.Range(value, 10).FirstOrDefault());
  54. }
  55. [TestMethod]
  56. public void FirstOrDefault_NoDoubleSet()
  57. {
  58. //
  59. // Regression test for a possible race condition caused by Return style operators
  60. // that could trigger two Set calls on a ManualResetEvent, causing it to get
  61. // disposed in between those two calls (cf. FirstOrDefaultInternal). This led
  62. // to an exception will the following stack trace:
  63. //
  64. // System.ObjectDisposedException: Safe handle has been closed
  65. // at System.Runtime.InteropServices.SafeHandle.DangerousAddRef(Boolean& success)
  66. // at System.StubHelpers.StubHelpers.SafeHandleAddRef(SafeHandle pHandle, Boolean& success)
  67. // at Microsoft.Win32.Win32Native.SetEvent(SafeWaitHandle handle)
  68. // at System.Threading.EventWaitHandle.Set()
  69. // at System.Reactive.Linq.QueryLanguage.<>c__DisplayClass458_1`1.<FirstOrDefaultInternal>b__2()
  70. //
  71. var o = new O();
  72. Scheduler.Default.Schedule(() =>
  73. {
  74. var x = o.FirstOrDefault();
  75. });
  76. o.Wait();
  77. o.Next();
  78. Thread.Sleep(100); // enough time to let the ManualResetEvent dispose
  79. o.Done();
  80. }
  81. private class O : IObservable<int>
  82. {
  83. private readonly ManualResetEvent _event = new(false);
  84. private IObserver<int> _observer;
  85. public void Wait()
  86. {
  87. _event.WaitOne();
  88. }
  89. public void Next()
  90. {
  91. _observer.OnNext(42);
  92. }
  93. public void Done()
  94. {
  95. _observer.OnCompleted();
  96. }
  97. public IDisposable Subscribe(IObserver<int> observer)
  98. {
  99. _observer = observer;
  100. _event.Set();
  101. return Disposable.Empty;
  102. }
  103. }
  104. }
  105. }