// 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.Collections.Generic; using System.Linq; using System.Reactive; using System.Reactive.Concurrency; using System.Reactive.Disposables; using System.Reactive.Joins; using System.Reactive.Linq; using System.Reactive.Subjects; using System.Reflection; using System.Threading; using System.Threading.Tasks; using Xunit; namespace ReactiveTests.Tests { /// /// Check if the Observable operator methods perform the proper /// argument validations en-masse with reflective checks. /// public class ArgumentValidationTest { #region + Default values for the generic types + /// /// Contains a map of various types, represented /// as strings generated via , /// mapped to a value. /// private static Dictionary _defaultValues; /// /// Prepare the default instances for various types used /// throughout Rx.NET. /// static ArgumentValidationTest() { _defaultValues = new Dictionary { { "IObservable`1[Object]", Observable.Return(new object()) }, { "IObservable`1[Int32]", Observable.Return(1) }, { "IObservable`1[Task`1[Int32]]", Observable.Return(Task.FromResult(1)) }, { "IObservable`1[Notification`1[Int32]]", Observable.Return(Notification.CreateOnNext(1)) }, { "IObservable`1[Int64]", Observable.Return(1L) }, { "IObservable`1[Double]", Observable.Return(1.0) }, { "IObservable`1[Single]", Observable.Return(1.0f) }, { "IObservable`1[Decimal]", Observable.Return(1.0m) }, { "IObservable`1[Nullable`1[Int32]]", Observable.Return(1) }, { "IObservable`1[Nullable`1[Int64]]", Observable.Return(1L) }, { "IObservable`1[Nullable`1[Double]]", Observable.Return(1.0) }, { "IObservable`1[Nullable`1[Single]]", Observable.Return(1.0f) }, { "IObservable`1[Nullable`1[Decimal]]", Observable.Return(1.0m) }, { "IObservable`1[IObservable`1[Int32]]", Observable.Return(Observable.Return(1)) }, { "IObservable`1[][Int32]", new[] { Observable.Return(1) } }, { "IConnectableObservable`1[Int32]", Observable.Return(1).Publish() }, { "Int32", 1 }, { "Int64", 1L }, { "IScheduler", Scheduler.Immediate }, { "TimeSpan", TimeSpan.FromMilliseconds(1) }, { "DateTimeOffset", DateTimeOffset.Now }, { "Object", new object() }, { "Exception", new Exception() }, { "String", "String" }, { "IDictionary`2[Int32, IObservable`1[Int32]]", new Dictionary>() }, { "Type", typeof(object) }, { "Int32[]", new[] { 1 } }, { "ISubject`1[Int32]", new Subject() }, { "ISubject`2[Int32, Int32]", new Subject() }, { "IEnumerable`1[Int32]", new[] { 1 } }, { "IEnumerable`1[IObservable`1[Int32]]", new[] { Observable.Return(1) } }, { "SynchronizationContext", SynchronizationContext.Current }, { "IEqualityComparer`1[Int32]", EqualityComparer.Default }, { "IComparer`1[Int32]", Comparer.Default }, { "IObserver`1[Int32]", Observer.Create(v => { }) }, { "CancellationToken", new CancellationToken() }, { "Action", new Action(() => { }) }, { "Action`1[Int32]", new Action(v => { }) }, { "Action`1[Exception]", new Action(v => { }) }, { "Action`1[IDisposable]", new Action(v => { }) }, { "Action`1[EventHandler]", new Action(v => { }) }, { "Action`1[EventHandler`1[Int32]]", new Action>(v => { }) }, { "Action`1[Action`1[Int32]]", new Action>(v => { }) }, { "Action`1[Action]", new Action(v => { }) }, { "Action`1[IAsyncResult]", new Action(v => { }) }, { "Action`2[Int32, Int32]", new Action((v, u) => { }) }, { "Func`1[Boolean]", new Func(() => true) }, { "Func`1[Int32]", new Func(() => 1) }, { "Func`1[IObservable`1[Int32]]", new Func>(() => Observable.Return(1)) }, { "Func`1[ISubject`2[Int32, Int32]]", new Func>(() => new Subject()) }, { "Func`1[Task`1[IObservable`1[Int32]]]", new Func>>(() => Task.FromResult(Observable.Return(1))) }, { "Func`1[IDisposable]", new Func(() => Disposable.Empty) }, { "Func`1[Task]", new Func(() => Task.FromResult(1)) }, { "Func`1[Task`1[Int32]]", new Func>(() => Task.FromResult(1)) }, { "Func`1[IEnumerable`1[IObservable`1[Object]]]", new Func>>(() => new[] { Observable.Return((object)1) }) }, { "Func`2[Int32, IObservable`1[Int32]]", new Func>(v => Observable.Return(v)) }, { "Func`2[Exception, IObservable`1[Int32]]", new Func>(v => Observable.Return(1)) }, { "Func`2[Int32, Task`1[Int32]]", new Func>(v => Task.FromResult(v)) }, { "Func`2[Int32, Int32]", new Func(v => v) }, { "Func`2[Int32, IEnumerable`1[Int32]]", new Func>(v => new[] { v }) }, { "Func`2[Int32, Boolean]", new Func(v => true) }, { "Func`2[Int32, TimeSpan]", new Func(v => TimeSpan.FromMilliseconds(1)) }, { "Func`2[Int32, DateTimeOffset]", new Func(v => DateTimeOffset.Now) }, { "Func`2[IList`1[Int32], Int32]", new Func, int>(v => v.Count) }, { "Func`2[Int32, Nullable`1[Double]]", new Func(v => v) }, { "Func`2[Int32, Nullable`1[Single]]", new Func(v => v) }, { "Func`2[Int32, Nullable`1[Int32]]", new Func(v => v) }, { "Func`2[Int32, Nullable`1[Decimal]]", new Func(v => v) }, { "Func`2[Int32, Nullable`1[Int64]]", new Func(v => v) }, { "Func`2[Int32, Double]", new Func(v => v) }, { "Func`2[Int32, Single]", new Func(v => v) }, { "Func`2[Int32, Decimal]", new Func(v => v) }, { "Func`2[Int32, Int64]", new Func(v => v) }, { "Func`2[IObservable`1[Object], IObservable`1[Int32]]", new Func, IObservable>(v => v.Select(w => 1)) }, { "Func`2[IObservable`1[Exception], IObservable`1[Int32]]", new Func, IObservable>(v => v.Select(w => 1)) }, { "Func`2[IGroupedObservable`2[Int32, Int32], IObservable`1[Int32]]", new Func, IObservable>(v => v) }, { "Func`2[IObservable`1[Int32], IObservable`1[Int32]]", new Func, IObservable>(v => v.Select(w => 1)) }, { "Func`2[CancellationToken, Task`1[IObservable`1[Int32]]]", new Func>>(v => Task.FromResult(Observable.Return(1))) }, { "Func`2[IDisposable, Task`1[IObservable`1[Int32]]]", new Func>>(v => Task.FromResult(Observable.Return(1))) }, { "Func`2[IDisposable, IObservable`1[Int32]]", new Func>(v => Observable.Return(1)) }, { "Func`2[CancellationToken, Task`1[IDisposable]]", new Func>(v => Task.FromResult(Disposable.Empty)) }, { "Func`2[EventHandler`1[Int32], Int32]", new Func, int>(v => 1) }, { "Func`2[Action`1[Int32], Int32]", new Func, int>(v => 1) }, { "Func`2[IObserver`1[Int32], IDisposable]", new Func, IDisposable>(v => Disposable.Empty) }, { "Func`2[IObserver`1[Int32], Action]", new Func, Action>(v => () => { }) }, { "Func`2[IObserver`1[Int32], Task]", new Func, Task>(v => Task.FromResult(1)) }, { "Func`2[IObserver`1[Int32], Task`1[IDisposable]]", new Func, Task>(v => Task.FromResult(Disposable.Empty)) }, { "Func`2[IObserver`1[Int32], Task`1[Action]]", new Func, Task>(v => Task.FromResult(() => { })) }, { "Func`2[CancellationToken, Task]", new Func(v => Task.FromResult(1)) }, { "Func`2[CancellationToken, Task`1[Int32]]", new Func>(v => Task.FromResult(1)) }, { "Func`2[IAsyncResult, Int32]", new Func(v => 1) }, { "Func`2[IObserver`1[Int32], IEnumerable`1[IObservable`1[Object]]]", new Func, IEnumerable>>(v => new[] { Observable.Return((object)1) }) }, { "Func`2[IObservable`1[Int32], Int32]", new Func, int>(v => 1) }, { "Func`3[Int32, Int32, IObservable`1[Int32]]", new Func>((v, u) => Observable.Return(v + u)) }, { "Func`3[Int32, Int32, Task`1[Int32]]", new Func>((v, u) => Task.FromResult(v + u)) }, { "Func`3[Int32, CancellationToken, Task`1[Int32]]", new Func>((v, u) => Task.FromResult(v)) }, { "Func`3[Int32, Int32, Int32]", new Func((v, u) => v + u) }, { "Func`3[Int32, Int32, IEnumerable`1[Int32]]", new Func>((v, u) => new[] { v, u }) }, { "Func`3[Int32, Int32, Boolean]", new Func((v, u) => true) }, { "Func`3[Int32, IObservable`1[Int32], Int32]", new Func, int>((v, u) => v) }, { "Func`3[IDisposable, CancellationToken, Task`1[IObservable`1[Int32]]]", new Func>>((v, u) => Task.FromResult(Observable.Return(1))) }, { "Func`3[IObserver`1[Int32], CancellationToken, Task]", new Func, CancellationToken, Task>((v, w) => Task.FromResult(1)) }, { "Func`3[IObserver`1[Int32], CancellationToken, Task`1[IDisposable]]", new Func, CancellationToken, Task>((v, w) => Task.FromResult(Disposable.Empty)) }, { "Func`3[IObserver`1[Int32], CancellationToken, Task`1[Action]]", new Func, CancellationToken, Task>((v, w) => Task.FromResult(() => { })) }, { "Func`3[AsyncCallback, Object, IAsyncResult]", new Func((v, w) => null) }, { "Func`4[Int32, Int32, CancellationToken, Task`1[Int32]]", new Func>((v, u, w) => Task.FromResult(v)) }, { "Func`4[Int32, Int32, Int32, Int32]", new Func((v1, v2, v3) => v1 + v2 + v3) }, { "Func`4[Int32, AsyncCallback, Object, IAsyncResult]", new Func((v, w, x) => null) }, { "Func`5[Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4) => v1 + v2 + v3 + v4) }, { "Func`6[Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5) => v1 + v2 + v3 + v4 + v5) }, { "Func`7[Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6) => v1 + v2 + v3 + v4 + v5 + v6) }, { "Func`8[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7) => v1 + v2 + v3 + v4 + v5 + v6 + v7) }, { "Func`9[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8) }, { "Func`10[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9) }, { "Func`11[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10) }, { "Func`12[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10 + v11) }, { "Func`13[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10 + v11 + v12) }, { "Func`14[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10 + v11 + v12 + v13) }, { "Func`15[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10 + v11 + v12 + v13 + v14) }, { "Func`16[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10 + v11 + v12 + v13 + v14 + v15) }, { "Func`17[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) => v1 + v2 + v3 + v4 + v5 + v6 + v7 + v8 + v9 + v10 + v11 + v12 + v13 + v14 + v15 + v16) }, { "Plan`1[][Int32]", new Plan[] { Observable.Return(1).Then(v => v) } }, { "IEnumerable`1[Plan`1[Int32]]", new Plan[] { Observable.Return(1).Then(v => v) } }, { "Action`3[Int32, Int32, Int32]", new Action((v1, v2, v3) => { }) }, { "Action`4[Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4) => { }) }, { "Action`5[Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5) => { }) }, { "Action`6[Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6) => { }) }, { "Action`7[Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7) => { }) }, { "Action`8[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8) => { }) }, { "Action`9[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9) => { }) }, { "Action`10[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10) => { }) }, { "Action`11[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) => { }) }, { "Action`12[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12) => { }) }, { "Action`13[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13) => { }) }, { "Action`14[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14) => { }) }, { "Action`15[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15) => { }) }, { "Action`16[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) => { }) }, { "Func`5[Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4) => null) }, { "Func`6[Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5) => null) }, { "Func`7[Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6) => null) }, { "Func`8[Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7) => null) }, { "Func`9[Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8) => null) }, { "Func`10[Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9) => null) }, { "Func`11[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10) => null) }, { "Func`12[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) => null) }, { "Func`13[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12) => null) }, { "Func`14[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13) => null) }, { "Func`15[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14) => null) }, { "Func`16[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15) => null) }, { "Func`17[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) => null) } }; } #endregion [Fact] public void Verify_Observable() { VerifyClass(typeof(Observable)); } [Fact] public void Verify_ObservableEx() { VerifyClass(typeof(ObservableEx)); } #region + Verification method + /// /// Verify that public static members of the class /// check for nulls in their arguments as /// well as when invoking Subscribe with null. /// /// The type to verify. private static void VerifyClass(Type type) { foreach (var method in type.GetMethods()) { // public static only (skip methods like Equals) if (!method.IsPublic || !method.IsStatic) { continue; } var m = default(MethodInfo); // Is this a generic method? if (method.IsGenericMethodDefinition) { // we need to specialize it to concrete types // for the reflective call to work // get the type arguments var ga = method.GetGenericArguments(); var targs = new Type[ga.Length]; // fill in the type arguments for (var k = 0; k < targs.Length; k++) { // watch out for type constrains // the default typeof(int) will not work when // exception or IDisposable is required at minimum var gac = ga[k].GetGenericParameterConstraints(); // no type constraints if (gac.Length == 0) { targs[k] = typeof(int); } else if (gac[0] == typeof(Exception)) { targs[k] = typeof(Exception); } else if (gac[0] == typeof(IDisposable)) { targs[k] = typeof(IDisposable); } else { // If we get here, a new rule should be added above throw new Exception("Unknown constraint: " + gac + "\r\n" + method); } } // generate a specialized method with the concrete generic arguments try { m = method.MakeGenericMethod(targs); } catch (Exception ex) { throw new Exception("MakeGenericMethod threw: " + method, ex); } } else { // non generic method, we can invoke this directly m = method; } var args = m.GetParameters(); // for each parameter of the (generic) method for (var i = 0; i < args.Length; i++) { // prepare a pattern for the method invocation var margs = new object[args.Length]; // some arguments can be null, often indicated with a default == null marker // this tracks this case and forgives for not throwing an ArgumentNullException var argumentCanBeNull = true; // for each argument index // with the loop i, this creates an N x N matrix where in each row, one argument is null for (var j = 0; j < args.Length; j++) { // figure out the type of the argument var pt = args[j].ParameterType; // by using some type naming convention as string var paramTypeName = TypeNameOf(pt); // classes, interfaces, arrays and abstract markers can be null // for the diagonal entries of the test matrix if (j == i && (pt.IsClass || pt.IsInterface || pt.IsArray || pt.IsAbstract)) { margs[j] = null; // check if the argument can be actually argumentCanBeNull = args[j].HasDefaultValue && args[j].DefaultValue == null; } else { // this argument is not tested for null or is not a null type // find the default instance for it if (_defaultValues.ContainsKey(paramTypeName)) { margs[j] = _defaultValues[paramTypeName]; } else { // default values have to be instantiated in _defaultValues for // each possible generic type arguments. // this will indicate what concrete instance value is missing. throw new Exception("Default instance not found for: " + paramTypeName + "\r\n\r\n" + m); } } } // assume it threw var thrown = true; var obj = default(object); try { obj = m.Invoke(null, margs); thrown = false; } catch (ArgumentNullException) { // expected exception, just in case } catch (Exception ex) { // reflection wraps the actual exception, let's unwrap it if (!(ex.InnerException is ArgumentNullException)) { throw new Exception("Method threw: " + method + " @ " + i, ex); } } // if the call didn't throw and the argument being tested isn't defaulted to null, throw if (!thrown && !argumentCanBeNull) { throw new Exception("Should have thrown: " + method + " @ " + i); } // if the call didn't throw and returned a null object, throw // no operators should return null if (obj == null && !thrown) { throw new NullReferenceException("null return: " + method + " @ " + i); } } // Now check the same method with valid arguments but // Subscribe(null) if it returns an IObservable subclass if (m.ReturnType.Name.Equals("IObservable`1") || m.ReturnType.Name.Equals("IConnectableObservable`1")) { // these will fail other argument validation with the defaults, skip them if (m.Name.Equals("FromEventPattern")) { continue; } // prepare method arguments var margs = new object[args.Length]; for (var j = 0; j < args.Length; j++) { var pt = args[j].ParameterType; var paramTypeName = TypeNameOf(pt); if (_defaultValues.ContainsKey(paramTypeName)) { margs[j] = _defaultValues[paramTypeName]; } else { // default values have to be instantiated in _defaultValues for // each possible generic type arguments. // this will indicate what concrete instance value is missing. // // it may fail independently of the null test above because // the particular type is non-nullable thus skipped above // or was the solo argument and it got never tested with a non-null // value throw new Exception("Default instance not found (Subscribe(null) check): " + paramTypeName + "\r\n\r\n" + m); } } // Assume it throws var thrown = true; try { // Should not return null, but would be mistaken for // throwing because of Subscribe(null) if (m.Invoke(null, margs) is IObservable o) { o.Subscribe(null); thrown = false; } } catch (ArgumentNullException) { // expected } catch (Exception ex) { // Unexpected exception // Maybe some other validation failed inside the method call // Consider skipping this method (set) // // Otherwise, the operator may run with the null IObserver // for a while and crash later. throw new Exception("Method threw (Subscribe(null) check): " + m, ex); } // If it didn't throw, report it if (!thrown) { throw new Exception("Should have thrown (Subscribe(null) check): " + m); } } } } /// /// Generate a string representation of a possibly generic type /// that is not verbose (i.e, no "System." everywhere). /// /// The type to get a string representation /// The string representation of a possibly generic type private static string TypeNameOf(Type type) { var ga = type.GetGenericArguments(); if (ga.Length == 0) { return type.Name; } return type.Name + "[" + string.Join(", ", ga.Select(t => TypeNameOf(t)).ToArray()) + "]"; } #endregion } }