Skip to content

openapi: support validation attributes #1477

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

Merged
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
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
Original file line number Diff line number Diff line change
@@ -0,0 +1,358 @@
using FluentAssertions;
using FluentAssertions.Specialized;
using JsonApiDotNetCore.OpenApi.Client.NSwag;
using Newtonsoft.Json;
using OpenApiNSwagEndToEndTests.ModelStateValidation.GeneratedCode;
using OpenApiTests;
using OpenApiTests.ModelStateValidation;
using TestBuildingBlocks;
using Xunit;
using Xunit.Abstractions;

namespace OpenApiNSwagEndToEndTests.ModelStateValidation;

public sealed class ModelStateValidationTests : IClassFixture<IntegrationTestContext<OpenApiStartup<ModelStateValidationDbContext>, ModelStateValidationDbContext>>
{
private readonly IntegrationTestContext<OpenApiStartup<ModelStateValidationDbContext>, ModelStateValidationDbContext> _testContext;
private readonly XUnitLogHttpMessageHandler _logHttpMessageHandler;
private readonly ModelStateValidationFakers _fakers = new();

public ModelStateValidationTests(IntegrationTestContext<OpenApiStartup<ModelStateValidationDbContext>, ModelStateValidationDbContext> testContext, ITestOutputHelper testOutputHelper)
{
_testContext = testContext;
_logHttpMessageHandler = new XUnitLogHttpMessageHandler(testOutputHelper);

testContext.UseController<SocialMediaAccountsController>();
}

[Fact]
public async Task xxx()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest()
}
});

// Assert
ExceptionAssertions<JsonSerializationException> assertion = await action.Should().ThrowExactlyAsync<JsonSerializationException>();
assertion.Which.Message.Should().Be("Cannot write a null value for property 'lastName'. Property requires a value. Path 'data.attributes'.");
}

[Theory]
[InlineData("ab")]
[InlineData("abcdefghijklmnopqrs")]
public async Task Cannot_exceed_length_constraint(string userName)
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
UserName = userName
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("The field UserName must be a string with a minimum length of 3 and a maximum length of 18.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/userName");
}

[Fact]
public async Task Cannot_violate_regular_expression_constraint()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
UserName = "aB1"
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("Only letters are allowed.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/userName");
}

[Fact]
public async Task Cannot_use_invalid_credit_card()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
CreditCard = "123-456"
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("The CreditCard field is not a valid credit card number.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/creditCard");
}

[Fact]
public async Task Cannot_use_invalid_email()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
Email = "abc"
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("The Email field is not a valid e-mail address.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/email");
}

[Theory]
[InlineData(-1)]
[InlineData(-0.56)]
[InlineData(123.98)]
[InlineData(124)]
public async Task Cannot_use_double_outside_of_valid_range(int age)
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
Age = age
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("The field Age must be between 0 and 123.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/age");
}

[Fact]
public async Task Cannot_use_invalid_url()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
BackgroundPicture = new Uri("/justapath", UriKind.Relative)
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("The BackgroundPicture field is not a valid fully-qualified http, https, or ftp URL.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/backgroundPicture");
}

[Fact]
public async Task Cannot_use_TimeSpan_outside_of_valid_range()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
NextRevalidation = "00:00:01"
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Input validation failed.");
errorObject.Detail.Should().Be("The field NextRevalidation must be between 01:00:00 and 05:00:00.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/nextRevalidation");
}

[Fact]
public async Task Cannot_use_invalid_TimeOnly()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
LastName = socialMediaAccount.LastName,
ValidatedAtTime = TimeSpan.FromSeconds(-1)
}
}
});

// Assert
ErrorResponseDocument document = (await action.Should().ThrowExactlyAsync<ApiException<ErrorResponseDocument>>()).Which.Result;
document.Errors.ShouldHaveCount(1);

ErrorObject errorObject = document.Errors.First();
errorObject.Title.Should().Be("Failed to deserialize request body: Incompatible attribute value found.");
errorObject.Detail.Should().Be("Failed to convert attribute 'validatedAtTime' with value '-00:00:01' of type 'String' to type 'Nullable<TimeOnly>'.");
errorObject.Source.ShouldNotBeNull();
errorObject.Source.Pointer.Should().Be("/data/attributes/validatedAtTime");
}

[Fact]
public async Task Can_create_resource_with_valid_properties()
{
// Arrange
SocialMediaAccount socialMediaAccount = _fakers.SocialMediaAccount.Generate();

using HttpClient httpClient = _testContext.Factory.CreateDefaultClient(_logHttpMessageHandler);
ModelStateValidationClient apiClient = new(httpClient);

// Act
Func<Task<SocialMediaAccountPrimaryResponseDocument>> action = () => apiClient.PostSocialMediaAccountAsync(new SocialMediaAccountPostRequestDocument
{
Data = new SocialMediaAccountDataInPostRequest
{
Attributes = new SocialMediaAccountAttributesInPostRequest
{
FirstName = socialMediaAccount.FirstName,
GivenName = socialMediaAccount.GivenName,
LastName = socialMediaAccount.LastName,
UserName = socialMediaAccount.UserName,
CreditCard = socialMediaAccount.CreditCard,
Email = socialMediaAccount.Email,
Phone = socialMediaAccount.Phone,
Age = socialMediaAccount.Age,
ProfilePicture = socialMediaAccount.ProfilePicture,
BackgroundPicture = new Uri(socialMediaAccount.BackgroundPicture!),
NextRevalidation = "02:00:00",
ValidatedAt = socialMediaAccount.ValidatedAt!,
ValidatedAtDate = new DateTimeOffset(socialMediaAccount.ValidatedAtDate!.Value.ToDateTime(new TimeOnly()).ToUniversalTime()),
ValidatedAtTime = socialMediaAccount.ValidatedAtTime!.Value.ToTimeSpan()
}
}
});

// Assert
await action.Should().NotThrowAsync();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,13 @@
<CodeGenerator>NSwagCSharp</CodeGenerator>
<Options>/ClientClassAccessModifier:internal /GenerateExceptionClasses:false /AdditionalNamespaceUsages:JsonApiDotNetCore.OpenApi.Client.NSwag /GenerateNullableReferenceTypes:true</Options>
</OpenApiReference>
<OpenApiReference Include="..\OpenApiTests\ModelStateValidation\GeneratedSwagger\$(TargetFramework)\swagger.g.json">
<Namespace>OpenApiNSwagEndToEndTests.ModelStateValidation.GeneratedCode</Namespace>
<ClassName>ModelStateValidationClient</ClassName>
<OutputPath>ModelStateValidationClient.cs</OutputPath>
<CodeGenerator>NSwagCSharp</CodeGenerator>
<Options>/ClientClassAccessModifier:internal /GenerateExceptionClasses:false /AdditionalNamespaceUsages:JsonApiDotNetCore.OpenApi.Client.NSwag /GenerateNullableReferenceTypes:true /GenerateOptionalParameters:true</Options>
</OpenApiReference>
<OpenApiReference Include="..\OpenApiTests\Headers\GeneratedSwagger\swagger.g.json">
<Namespace>OpenApiNSwagEndToEndTests.Headers.GeneratedCode</Namespace>
<ClassName>HeadersClient</ClassName>
Expand Down
Loading
Loading