// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the Apache 2.0 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.Text;
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.
///
static Dictionary _defaultValues;
///
/// Prepare the default instances for various types used
/// throughout Rx.NET.
///
static ArgumentValidationTest()
{
_defaultValues = new Dictionary();
_defaultValues.Add("IObservable`1[Object]", Observable.Return(new object()));
_defaultValues.Add("IObservable`1[Int32]", Observable.Return(1));
_defaultValues.Add("IObservable`1[Task`1[Int32]]", Observable.Return(Task.FromResult(1)));
_defaultValues.Add("IObservable`1[Notification`1[Int32]]", Observable.Return(Notification.CreateOnNext(1)));
_defaultValues.Add("IObservable`1[Int64]", Observable.Return(1L));
_defaultValues.Add("IObservable`1[Double]", Observable.Return(1.0));
_defaultValues.Add("IObservable`1[Single]", Observable.Return(1.0f));
_defaultValues.Add("IObservable`1[Decimal]", Observable.Return(1.0m));
_defaultValues.Add("IObservable`1[Nullable`1[Int32]]", Observable.Return(1));
_defaultValues.Add("IObservable`1[Nullable`1[Int64]]", Observable.Return(1L));
_defaultValues.Add("IObservable`1[Nullable`1[Double]]", Observable.Return(1.0));
_defaultValues.Add("IObservable`1[Nullable`1[Single]]", Observable.Return(1.0f));
_defaultValues.Add("IObservable`1[Nullable`1[Decimal]]", Observable.Return(1.0m));
_defaultValues.Add("IObservable`1[IObservable`1[Int32]]", Observable.Return(Observable.Return(1)));
_defaultValues.Add("IObservable`1[][Int32]", new[] { Observable.Return(1) });
_defaultValues.Add("IConnectableObservable`1[Int32]", Observable.Return(1).Publish());
_defaultValues.Add("Int32", 1);
_defaultValues.Add("Int64", 1L);
_defaultValues.Add("IScheduler", Scheduler.Immediate);
_defaultValues.Add("TimeSpan", TimeSpan.FromMilliseconds(1));
_defaultValues.Add("DateTimeOffset", DateTimeOffset.Now);
_defaultValues.Add("Object", new object());
_defaultValues.Add("Exception", new Exception());
_defaultValues.Add("String", "String");
_defaultValues.Add("IDictionary`2[Int32, IObservable`1[Int32]]", new Dictionary>());
_defaultValues.Add("Type", typeof(object));
_defaultValues.Add("Int32[]", new[] { 1 });
_defaultValues.Add("ISubject`1[Int32]", new Subject());
_defaultValues.Add("ISubject`2[Int32, Int32]", new Subject());
_defaultValues.Add("IEnumerable`1[Int32]", new[] { 1 });
_defaultValues.Add("IEnumerable`1[IObservable`1[Int32]]", new[] { Observable.Return(1) });
_defaultValues.Add("SynchronizationContext", SynchronizationContext.Current);
_defaultValues.Add("IEqualityComparer`1[Int32]", EqualityComparer.Default);
_defaultValues.Add("IComparer`1[Int32]", Comparer.Default);
_defaultValues.Add("IObserver`1[Int32]", Observer.Create(v => { }));
_defaultValues.Add("CancellationToken", new CancellationToken());
_defaultValues.Add("Action", new Action(() => { }));
_defaultValues.Add("Action`1[Int32]", new Action(v => { }));
_defaultValues.Add("Action`1[Exception]", new Action(v => { }));
_defaultValues.Add("Action`1[IDisposable]", new Action(v => { }));
_defaultValues.Add("Action`1[EventHandler]", new Action(v => { }));
_defaultValues.Add("Action`1[EventHandler`1[Int32]]", new Action>(v => { }));
_defaultValues.Add("Action`1[Action`1[Int32]]", new Action>(v => { }));
_defaultValues.Add("Action`1[Action]", new Action(v => { }));
_defaultValues.Add("Action`1[IAsyncResult]", new Action(v => { }));
_defaultValues.Add("Action`2[Int32, Int32]", new Action((v, u) => { }));
_defaultValues.Add("Func`1[Boolean]", new Func(() => true));
_defaultValues.Add("Func`1[Int32]", new Func(() => 1));
_defaultValues.Add("Func`1[IObservable`1[Int32]]", new Func>(() => Observable.Return(1)));
_defaultValues.Add("Func`1[ISubject`2[Int32, Int32]]", new Func>(() => new Subject()));
_defaultValues.Add("Func`1[Task`1[IObservable`1[Int32]]]", new Func>>(() => Task.FromResult(Observable.Return(1))));
_defaultValues.Add("Func`1[IDisposable]", new Func(() => Disposable.Empty));
_defaultValues.Add("Func`1[Task]", new Func(() => Task.FromResult(1)));
_defaultValues.Add("Func`1[Task`1[Int32]]", new Func>(() => Task.FromResult(1)));
_defaultValues.Add("Func`1[IEnumerable`1[IObservable`1[Object]]]", new Func>>(() => new[] { Observable.Return((object)1) }));
_defaultValues.Add("Func`2[Int32, IObservable`1[Int32]]", new Func>(v => Observable.Return(v)));
_defaultValues.Add("Func`2[Exception, IObservable`1[Int32]]", new Func>(v => Observable.Return(1)));
_defaultValues.Add("Func`2[Int32, Task`1[Int32]]", new Func>(v => Task.FromResult(v)));
_defaultValues.Add("Func`2[Int32, Int32]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, IEnumerable`1[Int32]]", new Func>(v => new[] { v }));
_defaultValues.Add("Func`2[Int32, Boolean]", new Func(v => true));
_defaultValues.Add("Func`2[Int32, TimeSpan]", new Func(v => TimeSpan.FromMilliseconds(1)));
_defaultValues.Add("Func`2[Int32, DateTimeOffset]", new Func(v => DateTimeOffset.Now));
_defaultValues.Add("Func`2[IList`1[Int32], Int32]", new Func, int>(v => v.Count));
_defaultValues.Add("Func`2[Int32, Nullable`1[Double]]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Nullable`1[Single]]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Nullable`1[Int32]]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Nullable`1[Decimal]]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Nullable`1[Int64]]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Double]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Single]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Decimal]", new Func(v => v));
_defaultValues.Add("Func`2[Int32, Int64]", new Func(v => v));
_defaultValues.Add("Func`2[IObservable`1[Object], IObservable`1[Int32]]", new Func, IObservable>(v => v.Select(w => 1)));
_defaultValues.Add("Func`2[IObservable`1[Exception], IObservable`1[Int32]]", new Func, IObservable>(v => v.Select(w => 1)));
_defaultValues.Add("Func`2[IGroupedObservable`2[Int32, Int32], IObservable`1[Int32]]", new Func, IObservable>(v => v));
_defaultValues.Add("Func`2[IObservable`1[Int32], IObservable`1[Int32]]", new Func, IObservable>(v => v.Select(w => 1)));
_defaultValues.Add("Func`2[CancellationToken, Task`1[IObservable`1[Int32]]]", new Func>>(v => Task.FromResult(Observable.Return(1))));
_defaultValues.Add("Func`2[IDisposable, Task`1[IObservable`1[Int32]]]", new Func>>(v => Task.FromResult(Observable.Return(1))));
_defaultValues.Add("Func`2[IDisposable, IObservable`1[Int32]]", new Func>(v => Observable.Return(1)));
_defaultValues.Add("Func`2[CancellationToken, Task`1[IDisposable]]", new Func>(v => Task.FromResult(Disposable.Empty)));
_defaultValues.Add("Func`2[EventHandler`1[Int32], Int32]", new Func, int>(v => 1));
_defaultValues.Add("Func`2[Action`1[Int32], Int32]", new Func, int>(v => 1));
_defaultValues.Add("Func`2[IObserver`1[Int32], IDisposable]", new Func, IDisposable>(v => Disposable.Empty));
_defaultValues.Add("Func`2[IObserver`1[Int32], Action]", new Func, Action>(v => () => { }));
_defaultValues.Add("Func`2[IObserver`1[Int32], Task]", new Func, Task>(v => Task.FromResult(1)));
_defaultValues.Add("Func`2[IObserver`1[Int32], Task`1[IDisposable]]", new Func, Task>(v => Task.FromResult(Disposable.Empty)));
_defaultValues.Add("Func`2[IObserver`1[Int32], Task`1[Action]]", new Func, Task>(v => Task.FromResult(() => { })));
_defaultValues.Add("Func`2[CancellationToken, Task]", new Func(v => Task.FromResult(1)));
_defaultValues.Add("Func`2[CancellationToken, Task`1[Int32]]", new Func>(v => Task.FromResult(1)));
_defaultValues.Add("Func`2[IAsyncResult, Int32]", new Func(v => 1));
_defaultValues.Add("Func`2[IObserver`1[Int32], IEnumerable`1[IObservable`1[Object]]]", new Func, IEnumerable>>(v => new[] { Observable.Return((object)1) }));
_defaultValues.Add("Func`2[IObservable`1[Int32], Int32]", new Func, int>(v => 1));
_defaultValues.Add("Func`3[Int32, Int32, IObservable`1[Int32]]", new Func>((v, u) => Observable.Return(v + u)));
_defaultValues.Add("Func`3[Int32, Int32, Task`1[Int32]]", new Func>((v, u) => Task.FromResult(v + u)));
_defaultValues.Add("Func`3[Int32, CancellationToken, Task`1[Int32]]", new Func>((v, u) => Task.FromResult(v)));
_defaultValues.Add("Func`3[Int32, Int32, Int32]", new Func((v, u) => v + u));
_defaultValues.Add("Func`3[Int32, Int32, IEnumerable`1[Int32]]", new Func>((v, u) => new[] { v, u }));
_defaultValues.Add("Func`3[Int32, Int32, Boolean]", new Func((v, u) => true));
_defaultValues.Add("Func`3[Int32, IObservable`1[Int32], Int32]", new Func, int>((v, u) => v));
_defaultValues.Add("Func`3[IDisposable, CancellationToken, Task`1[IObservable`1[Int32]]]", new Func>>((v, u) => Task.FromResult(Observable.Return(1))));
_defaultValues.Add("Func`3[IObserver`1[Int32], CancellationToken, Task]", new Func, CancellationToken, Task>((v, w) => Task.FromResult(1)));
_defaultValues.Add("Func`3[IObserver`1[Int32], CancellationToken, Task`1[IDisposable]]", new Func, CancellationToken, Task>((v, w) => Task.FromResult(Disposable.Empty)));
_defaultValues.Add("Func`3[IObserver`1[Int32], CancellationToken, Task`1[Action]]", new Func, CancellationToken, Task>((v, w) => Task.FromResult(() => { })));
_defaultValues.Add("Func`3[AsyncCallback, Object, IAsyncResult]", new Func((v, w) => null));
_defaultValues.Add("Func`4[Int32, Int32, CancellationToken, Task`1[Int32]]", new Func>((v, u, w) => Task.FromResult(v)));
_defaultValues.Add("Func`4[Int32, Int32, Int32, Int32]", new Func((v1, v2, v3) => v1 + v2 + v3));
_defaultValues.Add("Func`4[Int32, AsyncCallback, Object, IAsyncResult]", new Func((v, w, x) => null));
_defaultValues.Add("Func`5[Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4) => v1 + v2 + v3 + v4));
_defaultValues.Add("Func`6[Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5) => v1 + v2 + v3 + v4 + v5));
_defaultValues.Add("Func`7[Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Func((v1, v2, v3, v4, v5, v6) => v1 + v2 + v3 + v4 + v5 + v6));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("Plan`1[][Int32]", new Plan[] {
Observable.Return(1).Then(v => v)
});
_defaultValues.Add("IEnumerable`1[Plan`1[Int32]]", new Plan[] {
Observable.Return(1).Then(v => v)
});
_defaultValues.Add("Action`3[Int32, Int32, Int32]", new Action((v1, v2, v3) => { }));
_defaultValues.Add("Action`4[Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4) => { }));
_defaultValues.Add("Action`5[Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5) => { }));
_defaultValues.Add("Action`6[Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6) => { }));
_defaultValues.Add("Action`7[Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7) => { }));
_defaultValues.Add("Action`8[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8) => { }));
_defaultValues.Add("Action`9[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9) => { }));
_defaultValues.Add("Action`10[Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32, Int32]", new Action((v1, v2, v3, v4, v5, v6, v7, v8, v9, v10) => { }));
_defaultValues.Add("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) => { }));
_defaultValues.Add("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) => { }));
_defaultValues.Add("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) => { }));
_defaultValues.Add("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) => { }));
_defaultValues.Add("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) => { }));
_defaultValues.Add("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) => { }));
_defaultValues.Add("Func`5[Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4) => null));
_defaultValues.Add("Func`6[Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5) => null));
_defaultValues.Add("Func`7[Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6) => null));
_defaultValues.Add("Func`8[Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7) => null));
_defaultValues.Add("Func`9[Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8) => null));
_defaultValues.Add("Func`10[Int32, Int32, Int32, Int32, Int32, Int32, Int32, AsyncCallback, Object, IAsyncResult]", new Func((v1, v2, v3, v4, v5, v6, v7, v8, v9) => null));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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));
_defaultValues.Add("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.
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
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
}
}