FirstOrDefaultTest.cs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the Apache 2.0 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 Xunit;
  13. namespace ReactiveTests.Tests
  14. {
  15. public class FirstOrDefaultTest : ReactiveTest
  16. {
  17. [Fact]
  18. public void FirstOrDefault_ArgumentChecking()
  19. {
  20. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.FirstOrDefault(default(IObservable<int>)));
  21. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.FirstOrDefault(default(IObservable<int>), _ => true));
  22. ReactiveAssert.Throws<ArgumentNullException>(() => Observable.FirstOrDefault(DummyObservable<int>.Instance, default));
  23. }
  24. [Fact]
  25. public void FirstOrDefault_Empty()
  26. {
  27. Assert.Equal(default, Observable.Empty<int>().FirstOrDefault());
  28. }
  29. [Fact]
  30. public void FirstOrDefaultPredicate_Empty()
  31. {
  32. Assert.Equal(default, Observable.Empty<int>().FirstOrDefault(_ => true));
  33. }
  34. [Fact]
  35. public void FirstOrDefault_Return()
  36. {
  37. var value = 42;
  38. Assert.Equal(value, Observable.Return(value).FirstOrDefault());
  39. }
  40. [Fact]
  41. public void FirstOrDefault_Throw()
  42. {
  43. var ex = new Exception();
  44. var xs = Observable.Throw<int>(ex);
  45. ReactiveAssert.Throws(ex, () => xs.FirstOrDefault());
  46. }
  47. [Fact]
  48. public void FirstOrDefault_Range()
  49. {
  50. var value = 42;
  51. Assert.Equal(value, Observable.Range(value, 10).FirstOrDefault());
  52. }
  53. #if !NO_THREAD
  54. [Fact]
  55. public void FirstOrDefault_NoDoubleSet()
  56. {
  57. //
  58. // Regression test for a possible race condition caused by Return style operators
  59. // that could trigger two Set calls on a ManualResetEvent, causing it to get
  60. // disposed in between those two calls (cf. FirstOrDefaultInternal). This led
  61. // to an exception will the following stack trace:
  62. //
  63. // System.ObjectDisposedException: Safe handle has been closed
  64. // at System.Runtime.InteropServices.SafeHandle.DangerousAddRef(Boolean& success)
  65. // at System.StubHelpers.StubHelpers.SafeHandleAddRef(SafeHandle pHandle, Boolean& success)
  66. // at Microsoft.Win32.Win32Native.SetEvent(SafeWaitHandle handle)
  67. // at System.Threading.EventWaitHandle.Set()
  68. // at System.Reactive.Linq.QueryLanguage.<>c__DisplayClass458_1`1.<FirstOrDefaultInternal>b__2()
  69. //
  70. var o = new O();
  71. Scheduler.Default.Schedule(() =>
  72. {
  73. var x = o.FirstOrDefault();
  74. });
  75. o.Wait();
  76. o.Next();
  77. Thread.Sleep(100); // enough time to let the ManualResetEvent dispose
  78. o.Done();
  79. }
  80. #endif
  81. private class O : IObservable<int>
  82. {
  83. private readonly ManualResetEvent _event = new ManualResetEvent(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. }