Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Unit tests to illustrate issue #39 #40

Merged
merged 3 commits into from
Sep 28, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
117 changes: 117 additions & 0 deletions src/Namotion.Reflection.Tests/ObjectExtensionsTests.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
using System;
using System.Collections.Generic;
using Newtonsoft.Json;
using Xunit;

Expand Down Expand Up @@ -74,6 +75,122 @@ public void When_object_has_no_null_properties_then_errors_is_empty()
Assert.Empty(errors);
}

///// <summary>
///// One of the reproduce cases for issue 39 - should throw InvalidOperationException as 'nullable enable' is set for this file
///// </summary>
//[Fact]
//public void When_array_of_non_nullable_strings_contains_null_then_EnsureValidNullability_should_throw_InvalidOperationException()
//{
// //// Arrange
// var arrayOfNonNulllableStrings = JsonConvert.DeserializeObject<string[]>("[ null ]");

// //// Act and assert
// Assert.Throws<InvalidOperationException>(() => arrayOfNonNulllableStrings.EnsureValidNullability());

// TODO: This here is not possible because string[] (without property or parameter) has no context and nullability information.
//}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a non-null string value is being deserialised into an array
/// </summary>
[Fact]
public void When_array_of_non_nullable_strings_contains_single_non_null_string_then_EnsureValidNullability_should_not_throw()
{
//// Arrange
var arrayOfNonNulllableStrings = JsonConvert.DeserializeObject<string[]>("[ \"ok\" ]");

//// Act and assert
arrayOfNonNulllableStrings.EnsureValidNullability();
}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a null string value is being deserialised into an array of strings that are allowed to be null
/// </summary>
[Fact]
public void When_array_of_nullable_strings_contains_null_then_EnsureValidNullability_should_not_throw()
{
//// Arrange
var arrayOfNonNulllableStrings = JsonConvert.DeserializeObject<string?[]>("[ null ]");

//// Act and assert
arrayOfNonNulllableStrings.EnsureValidNullability();
}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a non-null string value is being deserialised into an array of strings (that are allowed to be null)
/// </summary>
[Fact]
public void When_array_of_nullable_strings_contains_single_non_null_string_then_EnsureValidNullability_should_not_throw()
{
//// Arrange
var arrayOfNonNulllableStrings = JsonConvert.DeserializeObject<string?[]>("[ \"ok\" ]");

//// Act and assert
arrayOfNonNulllableStrings.EnsureValidNullability();
}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a non-null string value is being deserialised into a property that is a list of non-nullable strings
/// </summary>
[Fact]
public void When_property_list_of_non_nullable_strings_contains_single_non_null_string_then_EnsureValidNullability_should_not_throw()
{
//// Arrange
var objectWithItemsListContainingSingleNonNullEntry = JsonConvert.DeserializeObject<SomethingWithListOfNonNullableStringItems>("{ \"Items\": [ \"ok\" ] }");

//// Act and assert
objectWithItemsListContainingSingleNonNullEntry.EnsureValidNullability();
}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a non-null string value is being deserialised into a property that is a list of non-nullable strings
/// </summary>
[Fact]
public void When_property_list_of_non_nullable_strings_contains_null_string_then_EnsureValidNullability_should_throw_InvalidOperationException()
{
//// Arrange
var objectWithItemsListContainingSingleNullEntry = JsonConvert.DeserializeObject<SomethingWithListOfNonNullableStringItems>("{ \"Items\": [ null ] }");

//// Act and assert
Assert.Throws<InvalidOperationException>(() => objectWithItemsListContainingSingleNullEntry.EnsureValidNullability());
}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a non-null string value is being deserialised into a property that is a list of (non-nullable) strings
/// </summary>
[Fact]
public void When_property_list_of_nullable_strings_contains_single_non_null_string_then_EnsureValidNullability_should_not_throw()
{
//// Arrange
var objectWithItemsListContainingSingleNonNullEntry = JsonConvert.DeserializeObject<SomethingWithListOfNullableStringItems>("{ \"Items\": [ \"ok\" ] }");

//// Act and assert
objectWithItemsListContainingSingleNonNullEntry.EnsureValidNullability();
}

/// <summary>
/// One of the reproduce cases for issue 39 - should not throw anything because a non-null string value is being deserialised into a property that is a list of non-nullable strings
/// </summary>
[Fact]
public void When_property_list_of_nullable_strings_contains_null_string_then_EnsureValidNullability_should_not_throw()
{
//// Arrange
var objectWithItemsListContainingSingleNullEntry = JsonConvert.DeserializeObject<SomethingWithListOfNullableStringItems>("{ \"Items\": [ null ] }");

//// Act and assert
objectWithItemsListContainingSingleNullEntry.EnsureValidNullability();
}

private sealed class SomethingWithListOfNonNullableStringItems
{
public List<string> Items { get; set; } = new List<string>();
}

private sealed class SomethingWithListOfNullableStringItems
{
public List<string?> Items { get; set; } = new List<string?>();
}

/// <summary>
/// This is a reproduce case for Issue 24
/// </summary>
Expand Down
29 changes: 20 additions & 9 deletions src/Namotion.Reflection/ObjectExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,7 @@ public static bool HasValidNullability(this object obj, bool checkChildren = tru
/// <returns>The result.</returns>
public static void EnsureValidNullability(this object obj, bool checkChildren = true)
{
ValidateNullability(obj, checkChildren ? new HashSet<object>() : null, null, false);
ValidateNullability(obj, obj?.GetType().ToContextualType(), checkChildren ? new HashSet<object>() : null, null, false);
}

/// <summary>Checks which non nullable properties are null (invalid).</summary>
Expand All @@ -69,11 +69,11 @@ public static void EnsureValidNullability(this object obj, bool checkChildren =
public static IEnumerable<string> ValidateNullability(this object obj, bool checkChildren = true)
{
var errors = new List<string>();
ValidateNullability(obj, checkChildren ? new HashSet<object>() : null, errors, false);
ValidateNullability(obj, obj?.GetType().ToContextualType(), checkChildren ? new HashSet<object>() : null, errors, false);
return errors;
}

private static void ValidateNullability(object obj, HashSet<object> checkedObjects, List<string> errors, bool stopFirstFail)
private static void ValidateNullability(object obj, ContextualType type, HashSet<object> checkedObjects, List<string> errors, bool stopFirstFail)
{
if (DisableNullabilityValidation)
{
Expand All @@ -97,25 +97,36 @@ private static void ValidateNullability(object obj, HashSet<object> checkedObjec
}
}

var type = obj.GetType();
if (checkedObjects != null && obj is IDictionary dictionary)
{
foreach (var item in dictionary.Keys.Cast<object>()
.Concat(dictionary.Values.Cast<object>()))
{
ValidateNullability(item, checkedObjects, errors, stopFirstFail);
ValidateNullability(item, type.GenericArguments[1], checkedObjects, errors, stopFirstFail);
}
}
else if (checkedObjects != null && obj is IEnumerable enumerable && !(obj is string))
{
var itemType = type.ElementType ?? type.GenericArguments[0];
foreach (var item in enumerable.Cast<object>())
{
ValidateNullability(item, checkedObjects, errors, stopFirstFail);
if (item == null)
{
if (itemType.Nullability == Nullability.NotNullable)
{
throw new InvalidOperationException(
"The object's nullability is invalid, item in enumerable.");
}
}
else
{
ValidateNullability(item, itemType, checkedObjects, errors, stopFirstFail);
}
}
}
else if (!type.ToCachedType().TypeInfo.IsValueType)
else if (!type.TypeInfo.IsValueType)
{
var properties = type.GetContextualProperties();
var properties = type.Type.GetContextualProperties();
for (int i = 0; i < properties.Length; i++)
{
var property = properties[i];
Expand Down Expand Up @@ -143,7 +154,7 @@ private static void ValidateNullability(object obj, HashSet<object> checkedObjec
}
else if (checkedObjects != null)
{
ValidateNullability(value, checkedObjects, errors, stopFirstFail);
ValidateNullability(value, property.MemberInfo.ToContextualMember(), checkedObjects, errors, stopFirstFail);
}
}
}
Expand Down