From 80d46a67221bb1deb79a058e2c81f8a84ad1a7c6 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 14:03:35 +0100 Subject: [PATCH 1/9] Created tests for the first to features in the README. --- Blazored.FluentValidation.sln | 17 +++ README.md | 2 +- .../AssemblyScanning/Component.razor | 34 ++++++ .../AssemblyScanning/Readme.md | 7 ++ .../AssemblyScanning/Tests.cs | 64 ++++++++++ .../BasicValidation/Component.razor | 44 +++++++ .../BasicValidation/Readme.md | 7 ++ .../BasicValidation/Tests.cs | 110 ++++++++++++++++++ .../Blazored.FluentValidation.Tests.csproj | 41 +++++++ .../GlobalUsings.cs | 3 + .../Model/Address.cs | 27 +++++ .../Model/Person.cs | 58 +++++++++ .../Model/ValidationResultType.cs | 8 ++ .../_Imports.razor | 14 +++ 14 files changed, 435 insertions(+), 1 deletion(-) create mode 100644 tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor create mode 100644 tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md create mode 100644 tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs create mode 100644 tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor create mode 100644 tests/Blazored.FluentValidation.Tests/BasicValidation/Readme.md create mode 100644 tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs create mode 100644 tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj create mode 100644 tests/Blazored.FluentValidation.Tests/GlobalUsings.cs create mode 100644 tests/Blazored.FluentValidation.Tests/Model/Address.cs create mode 100644 tests/Blazored.FluentValidation.Tests/Model/Person.cs create mode 100644 tests/Blazored.FluentValidation.Tests/Model/ValidationResultType.cs create mode 100644 tests/Blazored.FluentValidation.Tests/_Imports.razor diff --git a/Blazored.FluentValidation.sln b/Blazored.FluentValidation.sln index 5b8ce3e..bc1abab 100644 --- a/Blazored.FluentValidation.sln +++ b/Blazored.FluentValidation.sln @@ -20,6 +20,10 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "BlazorServer", "samples\Bla EndProject Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "SharedModels", "samples\Shared\SharedModels\SharedModels.csproj", "{42276235-5139-41D6-923D-18B7EB5E3E44}" EndProject +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "tests", "tests", "{DACAA0DB-2B93-4FE1-9D21-F45A4E63A640}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Blazored.FluentValidation.Tests", "tests\Blazored.FluentValidation.Tests\Blazored.FluentValidation.Tests.csproj", "{C92DF59B-B760-4FCC-A34C-A4007529BCC5}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -78,6 +82,18 @@ Global {42276235-5139-41D6-923D-18B7EB5E3E44}.Release|x64.Build.0 = Release|Any CPU {42276235-5139-41D6-923D-18B7EB5E3E44}.Release|x86.ActiveCfg = Release|Any CPU {42276235-5139-41D6-923D-18B7EB5E3E44}.Release|x86.Build.0 = Release|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Debug|Any CPU.Build.0 = Debug|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Debug|x64.ActiveCfg = Debug|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Debug|x64.Build.0 = Debug|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Debug|x86.ActiveCfg = Debug|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Debug|x86.Build.0 = Debug|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Release|Any CPU.ActiveCfg = Release|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Release|Any CPU.Build.0 = Release|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Release|x64.ActiveCfg = Release|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Release|x64.Build.0 = Release|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Release|x86.ActiveCfg = Release|Any CPU + {C92DF59B-B760-4FCC-A34C-A4007529BCC5}.Release|x86.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -86,6 +102,7 @@ Global {8BC1065A-A71E-4568-8A67-9C3AF039F73A} = {D5C6DCA9-C2BD-4117-BCCC-19E36E8406AB} {2459CF4B-6548-4031-B784-43E943E270A9} = {D5C6DCA9-C2BD-4117-BCCC-19E36E8406AB} {42276235-5139-41D6-923D-18B7EB5E3E44} = {D5C6DCA9-C2BD-4117-BCCC-19E36E8406AB} + {C92DF59B-B760-4FCC-A34C-A4007529BCC5} = {DACAA0DB-2B93-4FE1-9D21-F45A4E63A640} EndGlobalSection GlobalSection(ExtensibilityGlobals) = postSolution SolutionGuid = {42B22D99-6E59-4B30-88AD-B9CC07E0DA49} diff --git a/README.md b/README.md index 02b1e46..1ce13f1 100644 --- a/README.md +++ b/README.md @@ -132,7 +132,7 @@ The second is when manually validating the model using the `Validate` or `Valida ``` ## Access to full `ValidationFailure` -If you need details about the specifics of a validation result (e.g. its `Severity), you can access the result of the +If you need details about the specifics of a validation result (e.g. its `Severity`), you can access the result of the last validation by calling the `GetFailuresFromLastValidation` method on the `FluentValidationValidator` component. ```razor diff --git a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor new file mode 100644 index 0000000..476cd48 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor @@ -0,0 +1,34 @@ +@using Blazored.FluentValidation.Tests.Model + + + @if (DisableAssemblyScanning is null) + { + + } + else + { + + } + + + +

+ + +

+ + +
+ +@code { + [Parameter] public bool? DisableAssemblyScanning { get; set; } + private readonly Person _person = new(); + + internal ValidationResultType Result { get; private set; } = ValidationResultType.Valid; + + private void ValidSubmit() => Result = ValidationResultType.Valid; + private void InvalidSubmit() => Result = ValidationResultType.Error; +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md new file mode 100644 index 0000000..a332f37 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md @@ -0,0 +1,7 @@ +### What does this test? +This test checks if the assembly scanning works. It leverages, that this test +assembly does not register any `AbstractValidator` by default. + + - Setting the `DisableAssemblyScanning` to `true` should not find any validators and ignore errors. + - Setting the `DisableAssemblyScanning` to `false` or not setting the attribute at all, should + find the validators in the assembly and validate normally. diff --git a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs new file mode 100644 index 0000000..7f4ba4c --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs @@ -0,0 +1,64 @@ +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.AssemblyScanning; + +public class Tests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void DisableAssemblyScanning_SetToTrue_NoValidationHappens() + { + // Arrange + var cut = RenderComponent(p => p.Add(c => c.DisableAssemblyScanning, true)); + var person = _fixture.InvalidPerson(); + + // Act + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + + [Fact] + public void DisableAssemblyScanning_SetToFalse_ValidationHappens() + { + // Arrange + var cut = RenderComponent(p => p.Add(c => c.DisableAssemblyScanning, false)); + var person = _fixture.InvalidPerson(); + + // Act + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void DisableAssemblyScanning_NotSet_ValidationHappens() + { + // Arrange + var cut = RenderComponent(p => p.Add(c => c.DisableAssemblyScanning, null)); + var person = _fixture.InvalidPerson(); + + // Act + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + private class Fixture + { + public Person InvalidPerson() => new() + { + FirstName = "", + LastName = "Doe", + EmailAddress = "john.doe@blazored.org", + Age = 30 + }; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor new file mode 100644 index 0000000..7a12405 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor @@ -0,0 +1,44 @@ +@using Blazored.FluentValidation.Tests.Model + + + + + +

+ + +

+ +

+ + +

+ +

+ + +

+ +

+ + +

+ +

+ + +

+ + +
+ +@code { + private readonly Person _person = new(); + internal ValidationResultType Result { get; private set; } = ValidationResultType.Valid; + + private void ValidSubmit() => Result = ValidationResultType.Valid; + private void InvalidSubmit() => Result = ValidationResultType.Error; +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/BasicValidation/Readme.md b/tests/Blazored.FluentValidation.Tests/BasicValidation/Readme.md new file mode 100644 index 0000000..4ab205e --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/BasicValidation/Readme.md @@ -0,0 +1,7 @@ +### What does this test? +This test checks if the basic validation works. + + - Does a valid model pass the validation? + - Do basic validation rules get picked up? + - Do validation errors get displayed correctly in the UI? + - Are nested rules validated correctly? diff --git a/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs b/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs new file mode 100644 index 0000000..ae461af --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs @@ -0,0 +1,110 @@ +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.BasicValidation; + +public class Tests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void Validate_DataIsValid_ValidSubmit() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + + [Fact] + public void Validate_FirstNameMissing_InvalidSubmit() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with { FirstName = string.Empty }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void Validate_FirstNameMissing_ValidationErrorsPresent() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with { FirstName = string.Empty }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameRequired); + cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameRequired); + } + + [Fact] + public void Validate_AgeTooOld_ValidationErrorsPresent() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with { Age = 250 }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.AgeMax); + } + + [Fact] + public void Validate_AddressLine1Missing_ValidationErrorsPresent() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with { Address = new() { Line1 = string.Empty } }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(AddressValidator.Line1Required); + } + + private static void FillForm(IRenderedComponent cut, Person person) + { + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find($"input[name={nameof(Person.LastName)}]").Change(person.LastName); + cut.Find($"input[name={nameof(Person.EmailAddress)}]").Change(person.EmailAddress); + cut.Find($"input[name={nameof(Person.Age)}]").Change(person.Age.ToString()); + cut.Find($"input[name={nameof(Person.Address.Line1)}]").Change(person.Address.Line1); + } + + private class Fixture + { + public Person ValidPerson() => new() + { + FirstName = "John", + LastName = "Doe", + EmailAddress = "john.doe@blazored.org", + Age = 30, + Address = new() + { + Line1 = "123 Main St", + Town = "Springfield", + Postcode = "12345" + } + }; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj new file mode 100644 index 0000000..8269342 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj @@ -0,0 +1,41 @@ + + + + net7.0 + enable + false + + + + + + + + + + + + + + + + runtime; build; native; contentfiles; analyzers; buildtransitive + all + + + + + + + all + runtime; build; native; contentfiles; analyzers + + + + + + + + + + diff --git a/tests/Blazored.FluentValidation.Tests/GlobalUsings.cs b/tests/Blazored.FluentValidation.Tests/GlobalUsings.cs new file mode 100644 index 0000000..6b9b9b4 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/GlobalUsings.cs @@ -0,0 +1,3 @@ +global using System.Threading.Tasks; +global using FluentAssertions; +global using FluentValidation; \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/Model/Address.cs b/tests/Blazored.FluentValidation.Tests/Model/Address.cs new file mode 100644 index 0000000..d7a5332 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/Model/Address.cs @@ -0,0 +1,27 @@ +namespace Blazored.FluentValidation.Tests.Model +{ + public record Address + { + public string? Line1 { get; set; } + public string? Line2 { get; set; } + public string? Town { get; set; } + public string? County { get; set; } + public string? Postcode { get; set; } + } + + public class AddressValidator : AbstractValidator
+ { + public const string Line1Required = "You must enter Line 1"; + public const string TownRequired = "You must enter a town"; + public const string CountyRequired = "You must enter a county"; + public const string PostcodeRequired = "You must enter a postcode"; + + public AddressValidator() + { + RuleFor(p => p.Line1).NotEmpty().WithMessage(Line1Required); + RuleFor(p => p.Town).NotEmpty().WithMessage(TownRequired); + RuleFor(p => p.County).NotEmpty().WithMessage(CountyRequired); + RuleFor(p => p.Postcode).NotEmpty().WithMessage(PostcodeRequired); + } + } +} diff --git a/tests/Blazored.FluentValidation.Tests/Model/Person.cs b/tests/Blazored.FluentValidation.Tests/Model/Person.cs new file mode 100644 index 0000000..e84b1b7 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/Model/Person.cs @@ -0,0 +1,58 @@ +namespace Blazored.FluentValidation.Tests.Model +{ + public record Person + { + public string? FirstName { get; set; } + public string? LastName { get; set; } + public int? Age { get; set; } + public string? EmailAddress { get; set; } + public Address Address { get; set; } = new(); + } + + public class PersonValidator : AbstractValidator + { + public const string FirstNameRequired = "You must enter your first name"; + public const string FirstNameMaxLength = "First name cannot be longer than 50 characters"; + public const string LastNameRequired = "You must enter your last name"; + public const string LastNameMaxLength = "Last name cannot be longer than 50 characters"; + public const string AgeRequired = "You must enter your age"; + public const string AgeMin = "Age must be greater than 0"; + public const string AgeMax = "Age cannot be greater than 150"; + public const string EmailRequired = "You must enter an email address"; + public const string EmailValid = "You must provide a valid email address"; + public const string EmailUnique = "Email address must be unique"; + public const string DuplicateEmail = "mail@my.com"; + + public PersonValidator() + { + RuleSet("Names", () => + { + RuleFor(p => p.FirstName) + .NotEmpty().WithMessage(FirstNameRequired) + .MaximumLength(50).WithMessage(FirstNameMaxLength); + + RuleFor(p => p.LastName) + .NotEmpty().WithMessage(LastNameRequired) + .MaximumLength(50).WithMessage(LastNameMaxLength); + }); + + RuleFor(p => p.Age) + .NotNull().WithMessage(AgeRequired) + .GreaterThanOrEqualTo(0).WithMessage(AgeMin) + .LessThan(150).WithMessage(AgeMax); + + RuleFor(p => p.EmailAddress) + .NotEmpty().WithMessage(EmailRequired) + .EmailAddress().WithMessage(EmailValid) + .MustAsync(async (email, _) => await IsUniqueAsync(email)).WithMessage(EmailUnique); + + RuleFor(p => p.Address).SetValidator(new AddressValidator()); + } + + private static async Task IsUniqueAsync(string? email) + { + await Task.Delay(300); + return email?.ToLower() != DuplicateEmail; + } + } +} diff --git a/tests/Blazored.FluentValidation.Tests/Model/ValidationResultType.cs b/tests/Blazored.FluentValidation.Tests/Model/ValidationResultType.cs new file mode 100644 index 0000000..e35c667 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/Model/ValidationResultType.cs @@ -0,0 +1,8 @@ +namespace Blazored.FluentValidation.Tests.Model; + +public enum ValidationResultType +{ + Valid, + Warning, + Error +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/_Imports.razor b/tests/Blazored.FluentValidation.Tests/_Imports.razor new file mode 100644 index 0000000..6d4953f --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/_Imports.razor @@ -0,0 +1,14 @@ +@using Blazored.FluentValidation + +@using Bunit +@using Bunit.TestDoubles + +@using Microsoft.AspNetCore.Components.Forms +@using Microsoft.AspNetCore.Components.Routing +@using Microsoft.AspNetCore.Components.Web +@using Microsoft.Extensions.DependencyInjection +@using Microsoft.JSInterop +@using System.Net.Http +@using System.Net.Http.Json + +@using Xunit \ No newline at end of file From 8071461b0b5a3376ae7a4d5859ff6060d5a77629 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 14:11:12 +0100 Subject: [PATCH 2/9] Added IoC test case to assembly scanning tests. --- .../AssemblyScanning/Readme.md | 2 ++ .../AssemblyScanning/Tests.cs | 16 ++++++++++++++++ .../Blazored.FluentValidation.Tests.csproj | 4 ++++ 3 files changed, 22 insertions(+) diff --git a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md index a332f37..4540b8d 100644 --- a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md +++ b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Readme.md @@ -5,3 +5,5 @@ assembly does not register any `AbstractValidator` by default. - Setting the `DisableAssemblyScanning` to `true` should not find any validators and ignore errors. - Setting the `DisableAssemblyScanning` to `false` or not setting the attribute at all, should find the validators in the assembly and validate normally. + - Setting the `DisableAssemblyScanning` to `true` and registering the validators manually should + find the validators and validate normally. diff --git a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs index 7f4ba4c..32be18a 100644 --- a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs +++ b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Tests.cs @@ -51,6 +51,22 @@ public void DisableAssemblyScanning_NotSet_ValidationHappens() cut.Instance.Result.Should().Be(ValidationResultType.Error); } + [Fact] + public void DisableAssemblyScanning_SetToTrueButValidatorsRegistered_ValidationHappens() + { + // Arrange + Services.AddTransient, PersonValidator>(); + var cut = RenderComponent(p => p.Add(c => c.DisableAssemblyScanning, null)); + var person = _fixture.InvalidPerson(); + + // Act + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + private class Fixture { public Person InvalidPerson() => new() diff --git a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj index 8269342..b52f9e7 100644 --- a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj +++ b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj @@ -36,6 +36,10 @@ + + + + From 12eb3d2fc59d31d46eadd700d26e9782c0941681 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 14:28:13 +0100 Subject: [PATCH 3/9] Added some `ValidateAsync` tests, one is failing for some reason, need to investigate. --- .../AssemblyScanning/Component.razor | 3 +- .../AsyncValidation/Component.razor | 41 ++++++++++ .../AsyncValidation/Readme.md | 1 + .../AsyncValidation/Tests.cs | 80 +++++++++++++++++++ .../BasicValidation/Component.razor | 3 +- .../Blazored.FluentValidation.Tests.csproj | 4 - .../_Imports.razor | 1 + 7 files changed, 125 insertions(+), 8 deletions(-) create mode 100644 tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor create mode 100644 tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md create mode 100644 tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs diff --git a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor index 476cd48..e501004 100644 --- a/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/AssemblyScanning/Component.razor @@ -1,5 +1,4 @@ -@using Blazored.FluentValidation.Tests.Model - diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor new file mode 100644 index 0000000..9889587 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor @@ -0,0 +1,41 @@ + + + + +

+ + +

+ +

+ + +

+ +

+ + +

+ +

+ + +

+ + + +
+ +@code { + private readonly Person _person = new(); + private FluentValidationValidator? _fluentValidationValidator; + + public ValidationResultType Result { get; private set; } + + private async Task SubmitFormAsync() + { + var result = await _fluentValidationValidator!.ValidateAsync(); + Result = result ? ValidationResultType.Valid : ValidationResultType.Error; + } + +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md new file mode 100644 index 0000000..3e49f9c --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md @@ -0,0 +1 @@ +### What does this test? diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs new file mode 100644 index 0000000..1556ac1 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs @@ -0,0 +1,80 @@ +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.AsyncValidation; + +public class Tests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void AsyncValidate_PersonIsValid_ResultIsValid() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + + [Fact] + public void AsyncValidate_FirstNameTooLong_ResultIsError() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with + { + FirstName = "This is a very long first name that is over 50 characters long" + }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void AsyncValidate_FirstNameTooLong_ValidationMessagesPresent() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with + { + FirstName = "This is a very long first name that is over 50 characters long", + LastName = "", + }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); + cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); + } + + private void FillForm(IRenderedComponent cut, Person person) + { + cut.Find("input[name=FirstName]").Change(person.FirstName); + cut.Find("input[name=LastName]").Change(person.LastName); + cut.Find("input[name=Age]").Change(person.Age.ToString()); + cut.Find("input[name=EmailAddress]").Change(person.EmailAddress); + } + + private class Fixture + { + public Person ValidPerson() => new() + { + FirstName = "John", + LastName = "Doe", + Age = 30, + EmailAddress = "john.doe@blazored.com" + }; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor index 7a12405..82093d8 100644 --- a/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor @@ -1,5 +1,4 @@ -@using Blazored.FluentValidation.Tests.Model - diff --git a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj index b52f9e7..8269342 100644 --- a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj +++ b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj @@ -36,10 +36,6 @@ - - - - diff --git a/tests/Blazored.FluentValidation.Tests/_Imports.razor b/tests/Blazored.FluentValidation.Tests/_Imports.razor index 6d4953f..b5e1119 100644 --- a/tests/Blazored.FluentValidation.Tests/_Imports.razor +++ b/tests/Blazored.FluentValidation.Tests/_Imports.razor @@ -1,4 +1,5 @@ @using Blazored.FluentValidation +@using Blazored.FluentValidation.Tests.Model @using Bunit @using Bunit.TestDoubles From 1e11c18cc90285f760b5b31bccbdb1c46a529270 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 14:53:46 +0100 Subject: [PATCH 4/9] Test now passes, however the behavior is strange: When doing normal validation, all rules are automatically included, but when doing it with `ValidateAsync`, they need to be included manually. --- .../AsyncValidation/Component.razor | 3 +- .../AsyncValidation/Tests.cs | 29 ++++++++----------- .../BasicValidation/Component.razor | 4 +-- .../BasicValidation/Tests.cs | 2 +- .../Model/Person.cs | 6 ++-- 5 files changed, 20 insertions(+), 24 deletions(-) diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor index 9889587..ff4f9eb 100644 --- a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor @@ -30,12 +30,11 @@ private readonly Person _person = new(); private FluentValidationValidator? _fluentValidationValidator; - public ValidationResultType Result { get; private set; } + public ValidationResultType? Result { get; private set; } private async Task SubmitFormAsync() { var result = await _fluentValidationValidator!.ValidateAsync(); Result = result ? ValidationResultType.Valid : ValidationResultType.Error; } - } \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs index 1556ac1..453cec4 100644 --- a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs @@ -1,11 +1,10 @@ using Blazored.FluentValidation.Tests.Model; - namespace Blazored.FluentValidation.Tests.AsyncValidation; public class Tests : TestContext { private readonly Fixture _fixture = new(); - + [Fact] public void AsyncValidate_PersonIsValid_ResultIsValid() { @@ -16,47 +15,43 @@ public void AsyncValidate_PersonIsValid_ResultIsValid() // Act FillForm(cut, person); cut.Find("button").Click(); + cut.WaitForState(() => cut.Instance.Result is not null); // Assert cut.Instance.Result.Should().Be(ValidationResultType.Valid); } - + [Fact] - public void AsyncValidate_FirstNameTooLong_ResultIsError() + public void AsyncValidate_AgeNegative_ResultIsError() { // Arrange var cut = RenderComponent(); - var person = _fixture.ValidPerson() with - { - FirstName = "This is a very long first name that is over 50 characters long" - }; + var person = _fixture.ValidPerson() with { Age = -5 }; // Act FillForm(cut, person); cut.Find("button").Click(); + cut.WaitForState(() => cut.Instance.Result is not null); // Assert cut.Instance.Result.Should().Be(ValidationResultType.Error); } - + [Fact] - public void AsyncValidate_FirstNameTooLong_ValidationMessagesPresent() + public void AsyncValidate_AgeNegative_ValidationMessagesPresent() { // Arrange var cut = RenderComponent(); - var person = _fixture.ValidPerson() with - { - FirstName = "This is a very long first name that is over 50 characters long", - LastName = "", - }; + var person = _fixture.ValidPerson() with { Age = -5 }; // Act FillForm(cut, person); cut.Find("button").Click(); + cut.WaitForState(() => cut.Instance.Result is not null); // Assert - cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); - cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.AgeMin); + cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.AgeMin); } private void FillForm(IRenderedComponent cut, Person person) diff --git a/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor index 82093d8..17b2f0f 100644 --- a/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/BasicValidation/Component.razor @@ -28,14 +28,14 @@

- +

@code { - private readonly Person _person = new(); + private readonly Person _person = new() { Address = new() }; internal ValidationResultType Result { get; private set; } = ValidationResultType.Valid; private void ValidSubmit() => Result = ValidationResultType.Valid; diff --git a/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs b/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs index ae461af..fdcef88 100644 --- a/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs +++ b/tests/Blazored.FluentValidation.Tests/BasicValidation/Tests.cs @@ -88,7 +88,7 @@ private static void FillForm(IRenderedComponent cut, Person person) cut.Find($"input[name={nameof(Person.LastName)}]").Change(person.LastName); cut.Find($"input[name={nameof(Person.EmailAddress)}]").Change(person.EmailAddress); cut.Find($"input[name={nameof(Person.Age)}]").Change(person.Age.ToString()); - cut.Find($"input[name={nameof(Person.Address.Line1)}]").Change(person.Address.Line1); + cut.Find($"input[name={nameof(Person.Address.Line1)}]").Change(person.Address!.Line1); } private class Fixture diff --git a/tests/Blazored.FluentValidation.Tests/Model/Person.cs b/tests/Blazored.FluentValidation.Tests/Model/Person.cs index e84b1b7..93a61a1 100644 --- a/tests/Blazored.FluentValidation.Tests/Model/Person.cs +++ b/tests/Blazored.FluentValidation.Tests/Model/Person.cs @@ -6,7 +6,7 @@ public record Person public string? LastName { get; set; } public int? Age { get; set; } public string? EmailAddress { get; set; } - public Address Address { get; set; } = new(); + public Address? Address { get; set; } } public class PersonValidator : AbstractValidator @@ -46,7 +46,9 @@ public PersonValidator() .EmailAddress().WithMessage(EmailValid) .MustAsync(async (email, _) => await IsUniqueAsync(email)).WithMessage(EmailUnique); - RuleFor(p => p.Address).SetValidator(new AddressValidator()); + RuleFor(p => p.Address!) + .SetValidator(new AddressValidator()) + .When(p => p.Address is not null); } private static async Task IsUniqueAsync(string? email) From 0771d422e885a15b5bafdc8bd083764ca9aad80b Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 14:58:18 +0100 Subject: [PATCH 5/9] Forgot Readme.md. --- .../Blazored.FluentValidation.Tests/AsyncValidation/Readme.md | 3 +++ .../Blazored.FluentValidation.Tests.csproj | 4 ++++ 2 files changed, 7 insertions(+) diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md index 3e49f9c..e1363a0 100644 --- a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md @@ -1 +1,4 @@ ### What does this test? +This test checks if the `ValidateAsync` method works correctly, +specifically that the `bool` returned is `false` when validation fails, +and `true` otherwise. \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj index 8269342..dbfbe06 100644 --- a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj +++ b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj @@ -36,6 +36,10 @@ + + + + From 7d2df1b7702112cbdc27f765beb17452d8b11587 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 15:46:09 +0100 Subject: [PATCH 6/9] Tests for rule sets (.razor variant bugged?) --- .../AsyncValidation/Component.razor | 1 - .../Blazored.FluentValidation.Tests.csproj | 4 -- .../RuleSets/Component.razor | 37 ++++++++++++++ .../RuleSets/Tests.cs | 50 +++++++++++++++++++ 4 files changed, 87 insertions(+), 5 deletions(-) create mode 100644 tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor create mode 100644 tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor index ff4f9eb..2acb1a0 100644 --- a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor @@ -23,7 +23,6 @@

-
@code { diff --git a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj index dbfbe06..8269342 100644 --- a/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj +++ b/tests/Blazored.FluentValidation.Tests/Blazored.FluentValidation.Tests.csproj @@ -36,10 +36,6 @@ - - - - diff --git a/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor b/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor new file mode 100644 index 0000000..e8b633b --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor @@ -0,0 +1,37 @@ + + + @if (IncludeWithAttribute) + { + @*

Can't implement.. bugged?

*@ + } + else + { + + } + + +

+ + +

+ + +
+ +@code { + [Parameter] public bool IncludeWithAttribute { get; set; } + [Parameter] public bool IncludeWithCode { get; set; } + + private readonly Person _person = new(); + private FluentValidationValidator? _fluentValidationValidator; + + public ValidationResultType? Result { get; private set; } + + protected void Submit() + { + var result = _fluentValidationValidator!.Validate(o => o.IncludeRuleSets("Names")); + Result = result ? ValidationResultType.Valid : ValidationResultType.Error; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs b/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs new file mode 100644 index 0000000..dc8d5c7 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs @@ -0,0 +1,50 @@ +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.RuleSets; + +public class Tests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void AddedByAttribute_PersonFirstNameTooLong_ValidationFails() + { + // Arrange + var person = _fixture.ValidPerson() with + { + FirstName = "This name is clearly longer than 50 characters and thus should fail." + }; + var cut = RenderComponent(p => p.Add(c => c.IncludeWithCode, true)); + + // Act + cut.Find("input[name=FirstName]").Change(person.FirstName); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void AddedByAttribute_PersonFirstNameTooLong_ValidationMessagesPresent() + { + // Arrange + var person = _fixture.ValidPerson() with + { + FirstName = "This name is clearly longer than 50 characters and thus should fail." + }; + var cut = RenderComponent(p => p.Add(c => c.IncludeWithCode, true)); + + // Act + cut.Find("input[name=FirstName]").Change(person.FirstName); + cut.Find("button").Click(); + + // Assert + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); + cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); + } + + private class Fixture + { + public Person ValidPerson() => new() { FirstName = "John" }; + } +} \ No newline at end of file From 7f4c4eb8536f52f429b12680b1390d62a4737288 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 15:57:14 +0100 Subject: [PATCH 7/9] Re-added code I destroyed. --- .../RuleSets/Component.razor | 22 +++++++-- .../RuleSets/Tests.cs | 45 +++++++++++++++---- 2 files changed, 56 insertions(+), 11 deletions(-) diff --git a/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor b/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor index e8b633b..fa209c7 100644 --- a/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/RuleSets/Component.razor @@ -1,7 +1,7 @@  - + @if (IncludeWithAttribute) { @*

Can't implement.. bugged?

*@ @@ -17,21 +17,37 @@

+

+ + +

+ +

+ + +

+ +

+ + +

+
@code { [Parameter] public bool IncludeWithAttribute { get; set; } [Parameter] public bool IncludeWithCode { get; set; } - + private readonly Person _person = new(); private FluentValidationValidator? _fluentValidationValidator; public ValidationResultType? Result { get; private set; } - + protected void Submit() { var result = _fluentValidationValidator!.Validate(o => o.IncludeRuleSets("Names")); Result = result ? ValidationResultType.Valid : ValidationResultType.Error; } + } \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs b/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs index dc8d5c7..d0f7ab5 100644 --- a/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs +++ b/tests/Blazored.FluentValidation.Tests/RuleSets/Tests.cs @@ -6,6 +6,21 @@ public class Tests : TestContext { private readonly Fixture _fixture = new(); + [Fact] + public void AddedByAttribute_PersonValid_ValidationPasses() + { + // Arrange + var person = _fixture.ValidPerson(); + var cut = RenderComponent(p => p.Add(c => c.IncludeWithCode, true)); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + [Fact] public void AddedByAttribute_PersonFirstNameTooLong_ValidationFails() { @@ -15,15 +30,15 @@ public void AddedByAttribute_PersonFirstNameTooLong_ValidationFails() FirstName = "This name is clearly longer than 50 characters and thus should fail." }; var cut = RenderComponent(p => p.Add(c => c.IncludeWithCode, true)); - + // Act - cut.Find("input[name=FirstName]").Change(person.FirstName); + FillForm(cut, person); cut.Find("button").Click(); - + // Assert cut.Instance.Result.Should().Be(ValidationResultType.Error); } - + [Fact] public void AddedByAttribute_PersonFirstNameTooLong_ValidationMessagesPresent() { @@ -33,18 +48,32 @@ public void AddedByAttribute_PersonFirstNameTooLong_ValidationMessagesPresent() FirstName = "This name is clearly longer than 50 characters and thus should fail." }; var cut = RenderComponent(p => p.Add(c => c.IncludeWithCode, true)); - + // Act - cut.Find("input[name=FirstName]").Change(person.FirstName); + FillForm(cut, person); cut.Find("button").Click(); - + // Assert cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.FirstNameMaxLength); } + private static void FillForm(IRenderedComponent cut, Person person) + { + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find($"input[name={nameof(Person.LastName)}]").Change(person.LastName); + cut.Find($"input[name={nameof(Person.EmailAddress)}]").Change(person.EmailAddress); + cut.Find($"input[name={nameof(Person.Age)}]").Change(person.Age.ToString()); + } + private class Fixture { - public Person ValidPerson() => new() { FirstName = "John" }; + public Person ValidPerson() => new() + { + FirstName = "John", + LastName = "Doe", + EmailAddress = "john.doe@blazored.com", + Age = 30 + }; } } \ No newline at end of file From d472eaa9673feaa18042e2a2660dca9035ca4d91 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 16:27:14 +0100 Subject: [PATCH 8/9] Created readme for rule sets. Added tests for validation failure access. --- .../FullFailureAccess/AsyncComponent.razor | 55 +++++++++++++ .../FullFailureAccess/AsyncTests.cs | 77 +++++++++++++++++++ .../FullFailureAccess/Readme.md | 5 ++ .../FullFailureAccess/SyncComponent.razor | 54 +++++++++++++ .../FullFailureAccess/SyncTests.cs | 74 ++++++++++++++++++ .../Model/Person.cs | 6 ++ .../RuleSets/Readme.md | 17 ++++ 7 files changed, 288 insertions(+) create mode 100644 tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncComponent.razor create mode 100644 tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncTests.cs create mode 100644 tests/Blazored.FluentValidation.Tests/FullFailureAccess/Readme.md create mode 100644 tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncComponent.razor create mode 100644 tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncTests.cs create mode 100644 tests/Blazored.FluentValidation.Tests/RuleSets/Readme.md diff --git a/tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncComponent.razor b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncComponent.razor new file mode 100644 index 0000000..144144b --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncComponent.razor @@ -0,0 +1,55 @@ + + + + + +

+ + +

+ +

+ + +

+ +

+ + +

+ +

+ + +

+ + +
+ +@code { + private readonly Person _person = new(); + private FluentValidationValidator? _fluentValidationValidator; + + public ValidationResultType? Result { get; private set; } + + protected async Task Submit() + { + await _fluentValidationValidator!.ValidateAsync(); + var lastResult = _fluentValidationValidator!.GetFailuresFromLastValidation(); + if (!lastResult.Any()) + { + Result = ValidationResultType.Valid; + } + else if (lastResult.Any(failure => failure.Severity == Severity.Error)) + { + Result = ValidationResultType.Error; + } + else + { + Result = ValidationResultType.Warning; + } + } + +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncTests.cs b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncTests.cs new file mode 100644 index 0000000..8491b24 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/AsyncTests.cs @@ -0,0 +1,77 @@ +using System; +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.FullFailureAccess; + +public class AsyncTests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void GetFailuresFromLastValidation_PersonValid_ResultIsValid() + { + // Arrange + var person = _fixture.ValidPerson(); + var cut = RenderComponent(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + cut.WaitForState(() => cut.Instance.Result is not null); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + + [Fact] + public void GetFailuresFromLastValidation_EmailInvalid_ResultIsError() + { + // Arrange + var person = _fixture.ValidPerson() with { EmailAddress = "invalid-email" }; + var cut = RenderComponent(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + cut.WaitForState(() => cut.Instance.Result is not null); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void GetFailuresFromLastValidation_AgeSuspect_ResultIsWarning() + { + // Arrange + var person = _fixture.ValidPerson() with { Age = 69 }; + var cut = RenderComponent(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + cut.WaitForState(() => cut.Instance.Result is not null); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Warning); + } + + + private static void FillForm(IRenderedComponent cut, Person person) + { + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find($"input[name={nameof(Person.LastName)}]").Change(person.LastName); + cut.Find($"input[name={nameof(Person.EmailAddress)}]").Change(person.EmailAddress); + cut.Find($"input[name={nameof(Person.Age)}]").Change(person.Age.ToString()); + } + + private class Fixture + { + public Person ValidPerson() => new() + { + FirstName = "John", + LastName = "Doe", + EmailAddress = "john.doe@blazored.com", + Age = 30 + }; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/FullFailureAccess/Readme.md b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/Readme.md new file mode 100644 index 0000000..aff880a --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/Readme.md @@ -0,0 +1,5 @@ +### What does this test? +This test checks if the `GetFailuresFromLastValidation` method works correctly. It does so by both using `Validate` +and `ValidateAsync`. The failures given back are then checked for severity. + +To test warnings, the age of a person can be set to 69. diff --git a/tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncComponent.razor b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncComponent.razor new file mode 100644 index 0000000..4958393 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncComponent.razor @@ -0,0 +1,54 @@ + + + + + +

+ + +

+ +

+ + +

+ +

+ + +

+ +

+ + +

+ + +
+ +@code { + private readonly Person _person = new(); + private FluentValidationValidator? _fluentValidationValidator; + + public ValidationResultType? Result { get; private set; } + + protected void Submit() + { + _fluentValidationValidator!.Validate(); + var lastResult = _fluentValidationValidator!.GetFailuresFromLastValidation(); + if (!lastResult.Any()) + { + Result = ValidationResultType.Valid; + } + else if (lastResult.Any(failure => failure.Severity == Severity.Error)) + { + Result = ValidationResultType.Error; + } + else + { + Result = ValidationResultType.Warning; + } + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncTests.cs b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncTests.cs new file mode 100644 index 0000000..78abfe9 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/FullFailureAccess/SyncTests.cs @@ -0,0 +1,74 @@ +using System; +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.FullFailureAccess; + +public class SyncTests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void GetFailuresFromLastValidation_PersonValid_ResultIsValid() + { + // Arrange + var person = _fixture.ValidPerson(); + var cut = RenderComponent(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + + [Fact] + public void GetFailuresFromLastValidation_EmailInvalid_ResultIsError() + { + // Arrange + var person = _fixture.ValidPerson() with { EmailAddress = "invalid-email" }; + var cut = RenderComponent(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void GetFailuresFromLastValidation_AgeSuspect_ResultIsWarning() + { + // Arrange + var person = _fixture.ValidPerson() with { Age = 69 }; + var cut = RenderComponent(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Warning); + } + + + private static void FillForm(IRenderedComponent cut, Person person) + { + cut.Find($"input[name={nameof(Person.FirstName)}]").Change(person.FirstName); + cut.Find($"input[name={nameof(Person.LastName)}]").Change(person.LastName); + cut.Find($"input[name={nameof(Person.EmailAddress)}]").Change(person.EmailAddress); + cut.Find($"input[name={nameof(Person.Age)}]").Change(person.Age.ToString()); + } + + private class Fixture + { + public Person ValidPerson() => new() + { + FirstName = "John", + LastName = "Doe", + EmailAddress = "john.doe@blazored.com", + Age = 30 + }; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/Model/Person.cs b/tests/Blazored.FluentValidation.Tests/Model/Person.cs index 93a61a1..fc8c0f5 100644 --- a/tests/Blazored.FluentValidation.Tests/Model/Person.cs +++ b/tests/Blazored.FluentValidation.Tests/Model/Person.cs @@ -18,6 +18,7 @@ public class PersonValidator : AbstractValidator public const string AgeRequired = "You must enter your age"; public const string AgeMin = "Age must be greater than 0"; public const string AgeMax = "Age cannot be greater than 150"; + public const string AgeSuspect = "Age is suspect. Troll?"; public const string EmailRequired = "You must enter an email address"; public const string EmailValid = "You must provide a valid email address"; public const string EmailUnique = "Email address must be unique"; @@ -49,6 +50,11 @@ public PersonValidator() RuleFor(p => p.Address!) .SetValidator(new AddressValidator()) .When(p => p.Address is not null); + + RuleFor(p => p.Age) + .NotEqual(69) + .WithMessage(AgeSuspect) + .WithSeverity(Severity.Warning); } private static async Task IsUniqueAsync(string? email) diff --git a/tests/Blazored.FluentValidation.Tests/RuleSets/Readme.md b/tests/Blazored.FluentValidation.Tests/RuleSets/Readme.md new file mode 100644 index 0000000..afc393f --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/RuleSets/Readme.md @@ -0,0 +1,17 @@ +### What does this test? +This test checks if using the `IncludeRuleSets..` method work, once by attribute + +```html + +``` + +and once by code + +```csharp +@code { + private FluentValidationValidator? _fluentValidationValidator; + + private void PartialValidate() + => _fluentValidationValidator?.Validate(options => options.IncludeRuleSets("Names")); +} +``` \ No newline at end of file From 898e51b11eeee7b0cc2f2d086ed2b4b7d56b2d13 Mon Sep 17 00:00:00 2001 From: luetm Date: Wed, 21 Feb 2024 16:32:35 +0100 Subject: [PATCH 9/9] Tested `Validate` in addition to `ValidateAsync` --- .../AsyncComponent.razor} | 4 +- .../AsyncTests.cs} | 19 ++--- .../Readme.md | 0 .../DirectValidation/SyncComponent.razor | 39 ++++++++++ .../DirectValidation/SyncTests.cs | 73 +++++++++++++++++++ 5 files changed, 124 insertions(+), 11 deletions(-) rename tests/Blazored.FluentValidation.Tests/{AsyncValidation/Component.razor => DirectValidation/AsyncComponent.razor} (91%) rename tests/Blazored.FluentValidation.Tests/{AsyncValidation/Tests.cs => DirectValidation/AsyncTests.cs} (77%) rename tests/Blazored.FluentValidation.Tests/{AsyncValidation => DirectValidation}/Readme.md (100%) create mode 100644 tests/Blazored.FluentValidation.Tests/DirectValidation/SyncComponent.razor create mode 100644 tests/Blazored.FluentValidation.Tests/DirectValidation/SyncTests.cs diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor b/tests/Blazored.FluentValidation.Tests/DirectValidation/AsyncComponent.razor similarity index 91% rename from tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor rename to tests/Blazored.FluentValidation.Tests/DirectValidation/AsyncComponent.razor index 2acb1a0..c0cafe7 100644 --- a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Component.razor +++ b/tests/Blazored.FluentValidation.Tests/DirectValidation/AsyncComponent.razor @@ -1,4 +1,4 @@ - + @@ -31,7 +31,7 @@ public ValidationResultType? Result { get; private set; } - private async Task SubmitFormAsync() + private async Task SubmitAsync() { var result = await _fluentValidationValidator!.ValidateAsync(); Result = result ? ValidationResultType.Valid : ValidationResultType.Error; diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs b/tests/Blazored.FluentValidation.Tests/DirectValidation/AsyncTests.cs similarity index 77% rename from tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs rename to tests/Blazored.FluentValidation.Tests/DirectValidation/AsyncTests.cs index 453cec4..634efe3 100644 --- a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Tests.cs +++ b/tests/Blazored.FluentValidation.Tests/DirectValidation/AsyncTests.cs @@ -1,15 +1,16 @@ using Blazored.FluentValidation.Tests.Model; -namespace Blazored.FluentValidation.Tests.AsyncValidation; -public class Tests : TestContext +namespace Blazored.FluentValidation.Tests.DirectValidation; + +public class AsyncTests : TestContext { private readonly Fixture _fixture = new(); [Fact] - public void AsyncValidate_PersonIsValid_ResultIsValid() + public void ValidateAsync_PersonIsValid_ResultIsValid() { // Arrange - var cut = RenderComponent(); + var cut = RenderComponent(); var person = _fixture.ValidPerson(); // Act @@ -22,10 +23,10 @@ public void AsyncValidate_PersonIsValid_ResultIsValid() } [Fact] - public void AsyncValidate_AgeNegative_ResultIsError() + public void ValidateAsync_AgeNegative_ResultIsError() { // Arrange - var cut = RenderComponent(); + var cut = RenderComponent(); var person = _fixture.ValidPerson() with { Age = -5 }; // Act @@ -38,10 +39,10 @@ public void AsyncValidate_AgeNegative_ResultIsError() } [Fact] - public void AsyncValidate_AgeNegative_ValidationMessagesPresent() + public void ValidateAsync_AgeNegative_ValidationMessagesPresent() { // Arrange - var cut = RenderComponent(); + var cut = RenderComponent(); var person = _fixture.ValidPerson() with { Age = -5 }; // Act @@ -54,7 +55,7 @@ public void AsyncValidate_AgeNegative_ValidationMessagesPresent() cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.AgeMin); } - private void FillForm(IRenderedComponent cut, Person person) + private void FillForm(IRenderedComponent cut, Person person) { cut.Find("input[name=FirstName]").Change(person.FirstName); cut.Find("input[name=LastName]").Change(person.LastName); diff --git a/tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md b/tests/Blazored.FluentValidation.Tests/DirectValidation/Readme.md similarity index 100% rename from tests/Blazored.FluentValidation.Tests/AsyncValidation/Readme.md rename to tests/Blazored.FluentValidation.Tests/DirectValidation/Readme.md diff --git a/tests/Blazored.FluentValidation.Tests/DirectValidation/SyncComponent.razor b/tests/Blazored.FluentValidation.Tests/DirectValidation/SyncComponent.razor new file mode 100644 index 0000000..00a73e9 --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/DirectValidation/SyncComponent.razor @@ -0,0 +1,39 @@ + + + + +

+ + +

+ +

+ + +

+ +

+ + +

+ +

+ + +

+ + +
+ +@code { + private readonly Person _person = new(); + private FluentValidationValidator? _fluentValidationValidator; + + public ValidationResultType? Result { get; private set; } + + private void Submit() + { + var result = _fluentValidationValidator!.Validate(); + Result = result ? ValidationResultType.Valid : ValidationResultType.Error; + } +} \ No newline at end of file diff --git a/tests/Blazored.FluentValidation.Tests/DirectValidation/SyncTests.cs b/tests/Blazored.FluentValidation.Tests/DirectValidation/SyncTests.cs new file mode 100644 index 0000000..cd48fcb --- /dev/null +++ b/tests/Blazored.FluentValidation.Tests/DirectValidation/SyncTests.cs @@ -0,0 +1,73 @@ +using Blazored.FluentValidation.Tests.Model; + +namespace Blazored.FluentValidation.Tests.DirectValidation; + +public class SyncTests : TestContext +{ + private readonly Fixture _fixture = new(); + + [Fact] + public void Validate_PersonIsValid_ResultIsValid() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson(); + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Valid); + } + + [Fact] + public void Validate_AgeNegative_ResultIsError() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with { Age = -5 }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Instance.Result.Should().Be(ValidationResultType.Error); + } + + [Fact] + public void Validate_AgeNegative_ValidationMessagesPresent() + { + // Arrange + var cut = RenderComponent(); + var person = _fixture.ValidPerson() with { Age = -5 }; + + // Act + FillForm(cut, person); + cut.Find("button").Click(); + + // Assert + cut.Find(".validation-errors>.validation-message").TextContent.Should().Contain(PersonValidator.AgeMin); + cut.Find("li.validation-message").TextContent.Should().Contain(PersonValidator.AgeMin); + } + + private void FillForm(IRenderedComponent cut, Person person) + { + cut.Find("input[name=FirstName]").Change(person.FirstName); + cut.Find("input[name=LastName]").Change(person.LastName); + cut.Find("input[name=Age]").Change(person.Age.ToString()); + cut.Find("input[name=EmailAddress]").Change(person.EmailAddress); + } + + private class Fixture + { + public Person ValidPerson() => new() + { + FirstName = "John", + LastName = "Doe", + Age = 30, + EmailAddress = "john.doe@blazored.com" + }; + } +} \ No newline at end of file