266 lines
12 KiB
C#
266 lines
12 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;
|
|
using System.Collections.Generic;
|
|
using System.Net;
|
|
using System.Net.Http;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using FormatterWebSite;
|
|
using Microsoft.AspNetCore.Http;
|
|
using Microsoft.AspNetCore.Testing.xunit;
|
|
using Newtonsoft.Json;
|
|
using Xunit;
|
|
|
|
namespace Microsoft.AspNetCore.Mvc.FunctionalTests
|
|
{
|
|
public class InputObjectValidationTests : IClassFixture<MvcTestFixture<FormatterWebSite.Startup>>
|
|
{
|
|
public InputObjectValidationTests(MvcTestFixture<FormatterWebSite.Startup> fixture)
|
|
{
|
|
Client = fixture.CreateDefaultClient();
|
|
}
|
|
|
|
public HttpClient Client { get; }
|
|
|
|
// Parameters: Request Content, Expected status code, Expected model state error message
|
|
public static IEnumerable<object[]> SimpleTypePropertiesModelRequestData
|
|
{
|
|
get
|
|
{
|
|
yield return new object[] {
|
|
"{\"ByteProperty\":1, \"NullableByteProperty\":5, \"ByteArrayProperty\":[1,2,3]}",
|
|
StatusCodes.Status400BadRequest,
|
|
"The field ByteProperty must be between 2 and 8."};
|
|
|
|
yield return new object[] {
|
|
"{\"ByteProperty\":8, \"NullableByteProperty\":1, \"ByteArrayProperty\":[1,2,3]}",
|
|
StatusCodes.Status400BadRequest,
|
|
"The field NullableByteProperty must be between 2 and 8."};
|
|
|
|
yield return new object[] {
|
|
"{\"ByteProperty\":8, \"NullableByteProperty\":2, \"ByteArrayProperty\":[1]}",
|
|
StatusCodes.Status400BadRequest,
|
|
"The field ByteArrayProperty must be a string or array type with a minimum length of '2'."};
|
|
}
|
|
}
|
|
|
|
[ConditionalFact]
|
|
// Mono issue - https://github.com/aspnet/External/issues/18
|
|
[FrameworkSkipCondition(RuntimeFrameworks.Mono)]
|
|
public async Task CheckIfObjectIsDeserializedWithoutErrors()
|
|
{
|
|
// Arrange
|
|
var sampleId = 2;
|
|
var sampleName = "SampleUser";
|
|
var sampleAlias = "SampleAlias";
|
|
var sampleDesignation = "HelloWorld";
|
|
var sampleDescription = "sample user";
|
|
var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
|
|
"<User xmlns=\"http://schemas.datacontract.org/2004/07/FormatterWebSite\"><Id>" + sampleId +
|
|
"</Id><Name>" + sampleName + "</Name><Alias>" + sampleAlias + "</Alias>" +
|
|
"<Designation>" + sampleDesignation + "</Designation><description>" +
|
|
sampleDescription + "</description></User>";
|
|
var content = new StringContent(input, Encoding.UTF8, "application/xml");
|
|
|
|
// Act
|
|
var response = await Client.PostAsync("http://localhost/Validation/Index", content);
|
|
|
|
// Assert
|
|
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
|
|
Assert.Equal("User has been registered : " + sampleName,
|
|
await response.Content.ReadAsStringAsync());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task CheckIfObjectIsDeserialized_WithErrors()
|
|
{
|
|
// Arrange
|
|
var sampleId = 0;
|
|
var sampleName = "user";
|
|
var sampleAlias = "a";
|
|
var sampleDesignation = "HelloWorld!";
|
|
var sampleDescription = "sample user";
|
|
var input = "{ Id:" + sampleId + ", Name:'" + sampleName + "', Alias:'" + sampleAlias +
|
|
"' ,Designation:'" + sampleDesignation + "', description:'" + sampleDescription + "'}";
|
|
var content = new StringContent(input, Encoding.UTF8, "application/json");
|
|
|
|
// Act
|
|
var response = await Client.PostAsync("http://localhost/Validation/Index", content);
|
|
|
|
// Assert
|
|
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
|
|
// Mono issue - https://github.com/aspnet/External/issues/29
|
|
Assert.Equal(
|
|
"The field Id must be between 1 and 2000.," +
|
|
"The field Name must be a string or array type with a minimum length of '5'.," +
|
|
"The field Alias must be a string with a minimum length of 3 and a maximum length of 15.," +
|
|
"The field Designation must match the regular expression " +
|
|
"'[0-9a-zA-Z]*'.",
|
|
await response.Content.ReadAsStringAsync());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task CheckIfExcludedFieldsAreNotValidated()
|
|
{
|
|
// Arrange
|
|
var content = new StringContent("{\"Alias\":\"xyz\"}", Encoding.UTF8, "application/json");
|
|
|
|
// Act
|
|
var response = await Client.PostAsync("http://localhost/Validation/GetDeveloperName", content);
|
|
|
|
// Assert
|
|
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
|
|
Assert.Equal("No model validation for developer, even though developer.Name is empty.",
|
|
await response.Content.ReadAsStringAsync());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task ShallowValidation_HappensOnExcluded_ComplexTypeProperties()
|
|
{
|
|
// Arrange
|
|
var requestData = "{\"Name\":\"Library Manager\", \"Suppliers\": [{\"Name\":\"Contoso Corp\"}]}";
|
|
var content = new StringContent(requestData, Encoding.UTF8, "application/json");
|
|
var expectedModelStateErrorMessage
|
|
= "The field Suppliers must be a string or array type with a minimum length of '2'.";
|
|
var shouldNotContainMessage
|
|
= "The field Name must be a string or array type with a maximum length of '5'.";
|
|
|
|
// Act
|
|
var response = await Client.PostAsync("http://localhost/Validation/CreateProject", content);
|
|
|
|
// Assert
|
|
Assert.Equal(StatusCodes.Status400BadRequest, (int)response.StatusCode);
|
|
|
|
var responseContent = await response.Content.ReadAsStringAsync();
|
|
var responseObject = JsonConvert.DeserializeObject<Dictionary<string, string[]>>(responseContent);
|
|
var errorKeyValuePair = Assert.Single(responseObject, keyValuePair => keyValuePair.Value.Length > 0);
|
|
var errorMessage = Assert.Single(errorKeyValuePair.Value);
|
|
Assert.Equal(expectedModelStateErrorMessage, errorMessage);
|
|
|
|
// verifies that the excluded type is not validated
|
|
Assert.NotEqual(shouldNotContainMessage, errorMessage);
|
|
}
|
|
|
|
[Theory]
|
|
[MemberData(nameof(SimpleTypePropertiesModelRequestData))]
|
|
public async Task ShallowValidation_HappensOnExcluded_SimpleTypeProperties(
|
|
string requestContent,
|
|
int expectedStatusCode,
|
|
string expectedModelStateErrorMessage)
|
|
{
|
|
// Arrange
|
|
var content = new StringContent(requestContent, Encoding.UTF8, "application/json");
|
|
|
|
// Act
|
|
var response = await Client.PostAsync(
|
|
"http://localhost/Validation/CreateSimpleTypePropertiesModel",
|
|
content);
|
|
|
|
// Assert
|
|
Assert.Equal(expectedStatusCode, (int)response.StatusCode);
|
|
|
|
var responseContent = await response.Content.ReadAsStringAsync();
|
|
var responseObject = JsonConvert.DeserializeObject<Dictionary<string, string[]>>(responseContent);
|
|
var errorKeyValuePair = Assert.Single(responseObject, keyValuePair => keyValuePair.Value.Length > 0);
|
|
var errorMessage = Assert.Single(errorKeyValuePair.Value);
|
|
Assert.Equal(expectedModelStateErrorMessage, errorMessage);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task CheckIfExcludedField_IsNotValidatedForNonBodyBoundModels()
|
|
{
|
|
// Arrange
|
|
var kvps = new List<KeyValuePair<string, string>>();
|
|
kvps.Add(new KeyValuePair<string, string>("Alias", "xyz"));
|
|
var content = new FormUrlEncodedContent(kvps);
|
|
|
|
// Act
|
|
var response = await Client.PostAsync("http://localhost/Validation/GetDeveloperAlias", content);
|
|
|
|
// Assert
|
|
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
|
|
Assert.Equal("xyz", await response.Content.ReadAsStringAsync());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task ValidationProviderAttribute_WillValidateObject()
|
|
{
|
|
// Arrange
|
|
var invalidRequestData = "{\"FirstName\":\"TestName123\", \"LastName\": \"Test\"}";
|
|
var content = new StringContent(invalidRequestData, Encoding.UTF8, "application/json");
|
|
var expectedErrorMessage =
|
|
"{\"FirstName\":[\"The field FirstName must match the regular expression '[A-Za-z]*'.\"," +
|
|
"\"The field FirstName must be a string with a maximum length of 5.\"]}";
|
|
|
|
// Act
|
|
var response = await Client.PostAsync(
|
|
"http://localhost/Validation/ValidationProviderAttribute", content);
|
|
|
|
// Assert
|
|
Assert.Equal(expected: StatusCodes.Status400BadRequest, actual: (int)response.StatusCode);
|
|
|
|
var responseContent = await response.Content.ReadAsStringAsync();
|
|
Assert.Equal(expectedErrorMessage, actual: responseContent);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task ValidationProviderAttribute_DoesNotInterfere_WithOtherValidationAttributes()
|
|
{
|
|
// Arrange
|
|
var invalidRequestData = "{\"FirstName\":\"Test\", \"LastName\": \"Testsson\"}";
|
|
var content = new StringContent(invalidRequestData, Encoding.UTF8, "application/json");
|
|
var expectedErrorMessage =
|
|
"{\"LastName\":[\"The field LastName must be a string with a maximum length of 5.\"]}";
|
|
|
|
// Act
|
|
var response = await Client.PostAsync(
|
|
"http://localhost/Validation/ValidationProviderAttribute", content);
|
|
|
|
// Assert
|
|
Assert.Equal(expected: StatusCodes.Status400BadRequest, actual: (int)response.StatusCode);
|
|
|
|
var responseContent = await response.Content.ReadAsStringAsync();
|
|
Assert.Equal(expectedErrorMessage, actual: responseContent);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task ValidationProviderAttribute_RequiredAttributeErrorMessage_WillComeFirst()
|
|
{
|
|
// Arrange
|
|
var invalidRequestData = "{\"FirstName\":\"Testname\", \"LastName\": \"\"}";
|
|
var content = new StringContent(invalidRequestData, Encoding.UTF8, "application/json");
|
|
var expectedError =
|
|
"{\"LastName\":[\"The LastName field is required.\"]," +
|
|
"\"FirstName\":[\"The field FirstName must be a string with a maximum length of 5.\"]}";
|
|
|
|
// Act
|
|
var response = await Client.PostAsync(
|
|
"http://localhost/Validation/ValidationProviderAttribute", content);
|
|
|
|
// Assert
|
|
Assert.Equal(expected: StatusCodes.Status400BadRequest, actual: (int)response.StatusCode);
|
|
|
|
var responseContent = await response.Content.ReadAsStringAsync();
|
|
Assert.Equal(expectedError, actual: responseContent);
|
|
}
|
|
|
|
// Test for https://github.com/aspnet/Mvc/issues/7357
|
|
[Fact]
|
|
public async Task ValidationThrowsError_WhenValidationExceedsMaxValidationDepth()
|
|
{
|
|
// Arrange
|
|
var expected = $"ValidationVisitor exceeded the maximum configured validation depth '32' when validating property 'Value' on type '{typeof(RecursiveIdentifier)}'. " +
|
|
"This may indicate a very deep or infinitely recursive object graph. Consider modifying 'MvcOptions.MaxValidationDepth' or suppressing validation on the model type.";
|
|
var requestMessage = new HttpRequestMessage(HttpMethod.Post, "Validation/ValidationThrowsError_WhenValidationExceedsMaxValidationDepth")
|
|
{
|
|
Content = new StringContent(@"{ ""Id"": ""S-1-5-21-1004336348-1177238915-682003330-512"" }", Encoding.UTF8, "application/json"),
|
|
};
|
|
|
|
// Act & Assert
|
|
var ex = await Assert.ThrowsAsync<InvalidOperationException>(() => Client.SendAsync(requestMessage));
|
|
Assert.Equal(expected, ex.Message);
|
|
}
|
|
}
|
|
} |