380 lines
14 KiB
C#
380 lines
14 KiB
C#
// Copyright (c) .NET Foundation. All rights reserved.
|
|
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
|
|
|
|
using System.ComponentModel.DataAnnotations;
|
|
using System.IO;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.AspNet.Mvc.Abstractions;
|
|
using Microsoft.AspNet.Mvc.ModelBinding;
|
|
using Xunit;
|
|
|
|
namespace Microsoft.AspNet.Mvc.IntegrationTests
|
|
{
|
|
public class BodyValidationIntegrationTests
|
|
{
|
|
private class Person
|
|
{
|
|
[FromBody]
|
|
[Required]
|
|
public Address Address { get; set; }
|
|
}
|
|
|
|
private class Address
|
|
{
|
|
public string Street { get; set; }
|
|
}
|
|
|
|
[Fact]
|
|
public async Task FromBodyAndRequiredOnProperty_EmptyBody_AddsModelStateError()
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor()
|
|
{
|
|
Name = "Parameter1",
|
|
BindingInfo = new BindingInfo()
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
},
|
|
ParameterType = typeof(Person)
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes(string.Empty));
|
|
request.ContentType = "application/json";
|
|
});
|
|
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
var boundPerson = Assert.IsType<Person>(modelBindingResult.Model);
|
|
Assert.NotNull(boundPerson);
|
|
var key = Assert.Single(modelState.Keys);
|
|
Assert.Equal("CustomParameter.Address", key);
|
|
Assert.False(modelState.IsValid);
|
|
var error = Assert.Single(modelState[key].Errors);
|
|
// Mono issue - https://github.com/aspnet/External/issues/19
|
|
Assert.Equal(PlatformNormalizer.NormalizeContent("The Address field is required."), error.ErrorMessage);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task FromBodyOnActionParameter_EmptyBody_BindsToNullValue()
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor
|
|
{
|
|
Name = "Parameter1",
|
|
BindingInfo = new BindingInfo
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
BindingSource = BindingSource.Body
|
|
},
|
|
ParameterType = typeof(Person)
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes(string.Empty));
|
|
request.ContentType = "application/json";
|
|
});
|
|
|
|
var httpContext = operationContext.HttpContext;
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
Assert.Null(modelBindingResult.Model);
|
|
|
|
Assert.True(modelState.IsValid);
|
|
var entry = Assert.Single(modelState);
|
|
Assert.Empty(entry.Key);
|
|
Assert.Null(entry.Value.RawValue);
|
|
}
|
|
|
|
private class Person4
|
|
{
|
|
[FromBody]
|
|
[Required]
|
|
public int Address { get; set; }
|
|
}
|
|
|
|
[Fact]
|
|
public async Task FromBodyAndRequiredOnValueTypeProperty_EmptyBody_JsonFormatterAddsModelStateError()
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor
|
|
{
|
|
Name = "Parameter1",
|
|
BindingInfo = new BindingInfo
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
},
|
|
ParameterType = typeof(Person4)
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes(string.Empty));
|
|
request.ContentType = "application/json";
|
|
});
|
|
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
var boundPerson = Assert.IsType<Person4>(modelBindingResult.Model);
|
|
|
|
Assert.False(modelState.IsValid);
|
|
var entry = Assert.Single(modelState);
|
|
Assert.Equal("CustomParameter.Address", entry.Key);
|
|
Assert.Null(entry.Value.AttemptedValue);
|
|
Assert.Null(entry.Value.RawValue);
|
|
var error = Assert.Single(entry.Value.Errors);
|
|
Assert.NotNull(error.Exception);
|
|
|
|
// Json.NET currently throws an exception starting with "No JSON content found and type 'System.Int32' is
|
|
// not nullable." but do not tie test to a particular Json.NET build.
|
|
Assert.NotEmpty(error.Exception.Message);
|
|
}
|
|
|
|
private class Person5
|
|
{
|
|
[FromBody]
|
|
public Address5 Address { get; set; }
|
|
}
|
|
|
|
private class Address5
|
|
{
|
|
public int Number { get; set; }
|
|
|
|
// Required attribute does not cause an error in test scenarios. JSON deserializer ok w/ missing data.
|
|
[Required]
|
|
public int RequiredNumber { get; set; }
|
|
}
|
|
|
|
[Fact]
|
|
public async Task FromBodyAndRequiredOnInnerValueTypeProperty_NotBound_JsonFormatterSuccessful()
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor
|
|
{
|
|
Name = "Parameter1",
|
|
BindingInfo = new BindingInfo
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
},
|
|
ParameterType = typeof(Person5)
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes("{ \"Number\": 5 }"));
|
|
request.ContentType = "application/json";
|
|
});
|
|
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
var boundPerson = Assert.IsType<Person5>(modelBindingResult.Model);
|
|
Assert.NotNull(boundPerson.Address);
|
|
Assert.Equal(5, boundPerson.Address.Number);
|
|
Assert.Equal(0, boundPerson.Address.RequiredNumber);
|
|
|
|
Assert.True(modelState.IsValid);
|
|
var entry = Assert.Single(modelState);
|
|
Assert.Equal("CustomParameter.Address", entry.Key);
|
|
Assert.NotNull(entry.Value);
|
|
Assert.Null(entry.Value.AttemptedValue);
|
|
Assert.Same(boundPerson.Address, entry.Value.RawValue);
|
|
Assert.Empty(entry.Value.Errors);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task FromBodyWithInvalidPropertyData_JsonFormatterAddsModelError()
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor
|
|
{
|
|
Name = "Parameter1",
|
|
BindingInfo = new BindingInfo
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
},
|
|
ParameterType = typeof(Person5)
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes("{ \"Number\": \"not a number\" }"));
|
|
request.ContentType = "application/json";
|
|
});
|
|
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
var boundPerson = Assert.IsType<Person5>(modelBindingResult.Model);
|
|
Assert.Null(boundPerson.Address);
|
|
|
|
Assert.False(modelState.IsValid);
|
|
Assert.Equal(2, modelState.Count);
|
|
Assert.Equal(1, modelState.ErrorCount);
|
|
|
|
var state = modelState["CustomParameter.Address"];
|
|
Assert.NotNull(state);
|
|
Assert.Null(state.AttemptedValue);
|
|
Assert.Null(state.RawValue);
|
|
Assert.Empty(state.Errors);
|
|
|
|
state = modelState["CustomParameter.Address.Number"];
|
|
Assert.NotNull(state);
|
|
Assert.Null(state.AttemptedValue);
|
|
Assert.Null(state.RawValue);
|
|
var error = Assert.Single(state.Errors);
|
|
Assert.NotNull(error.Exception);
|
|
|
|
// Json.NET currently throws an Exception with a Message starting with "Could not convert string to
|
|
// integer: not a number." but do not tie test to a particular Json.NET build.
|
|
Assert.NotEmpty(error.Exception.Message);
|
|
}
|
|
|
|
private class Person2
|
|
{
|
|
[FromBody]
|
|
public Address2 Address { get; set; }
|
|
}
|
|
|
|
private class Address2
|
|
{
|
|
[Required]
|
|
public string Street { get; set; }
|
|
|
|
public int Zip { get; set; }
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData("{ \"Zip\" : 123 }")]
|
|
[InlineData("{}")]
|
|
public async Task FromBodyOnTopLevelProperty_RequiredOnSubProperty_AddsModelStateError(string inputText)
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor
|
|
{
|
|
BindingInfo = new BindingInfo
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
},
|
|
ParameterType = typeof(Person2),
|
|
Name = "param-name",
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes(inputText));
|
|
request.ContentType = "application/json";
|
|
});
|
|
var httpContext = operationContext.HttpContext;
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
var boundPerson = Assert.IsType<Person2>(modelBindingResult.Model);
|
|
Assert.NotNull(boundPerson);
|
|
|
|
Assert.False(modelState.IsValid);
|
|
Assert.Equal(2, modelState.Keys.Count);
|
|
var address = Assert.Single(modelState, kvp => kvp.Key == "CustomParameter.Address").Value;
|
|
Assert.Equal(ModelValidationState.Unvalidated, address.ValidationState);
|
|
|
|
var street = Assert.Single(modelState, kvp => kvp.Key == "CustomParameter.Address.Street").Value;
|
|
Assert.Equal(ModelValidationState.Invalid, street.ValidationState);
|
|
var error = Assert.Single(street.Errors);
|
|
// Mono issue - https://github.com/aspnet/External/issues/19
|
|
Assert.Equal(PlatformNormalizer.NormalizeContent("The Street field is required."), error.ErrorMessage);
|
|
}
|
|
|
|
private class Person3
|
|
{
|
|
[FromBody]
|
|
public Address3 Address { get; set; }
|
|
}
|
|
|
|
private class Address3
|
|
{
|
|
public string Street { get; set; }
|
|
|
|
[Required]
|
|
public int Zip { get; set; }
|
|
}
|
|
|
|
[Theory]
|
|
[InlineData("{ \"Street\" : \"someStreet\" }")]
|
|
[InlineData("{}")]
|
|
public async Task FromBodyOnProperty_Succeeds_IgnoresRequiredOnValueTypeSubProperty(string inputText)
|
|
{
|
|
// Arrange
|
|
var argumentBinder = ModelBindingTestHelper.GetArgumentBinder();
|
|
var parameter = new ParameterDescriptor
|
|
{
|
|
BindingInfo = new BindingInfo
|
|
{
|
|
BinderModelName = "CustomParameter",
|
|
},
|
|
ParameterType = typeof(Person3),
|
|
Name = "param-name",
|
|
};
|
|
|
|
var operationContext = ModelBindingTestHelper.GetOperationBindingContext(
|
|
request =>
|
|
{
|
|
request.Body = new MemoryStream(Encoding.UTF8.GetBytes(inputText));
|
|
request.ContentType = "application/json";
|
|
});
|
|
var modelState = operationContext.ActionContext.ModelState;
|
|
|
|
// Act
|
|
var modelBindingResult = await argumentBinder.BindModelAsync(parameter, operationContext);
|
|
|
|
// Assert
|
|
Assert.True(modelBindingResult.IsModelSet);
|
|
Assert.IsType<Person3>(modelBindingResult.Model);
|
|
|
|
Assert.True(modelState.IsValid);
|
|
var entry = Assert.Single(modelState);
|
|
Assert.Equal("CustomParameter.Address", entry.Key);
|
|
}
|
|
}
|
|
} |