/teaching-simple-validation

For teaching developers how to perform Fluent Validation

Primary LanguageC#

Validating application data with Fluent Validation

Validation is the first and most important step in securing an application. It prevents the application from processing unwanted inputs that may produce unpredictable results. Couple validation with properly transmitting data to a data source.

When validating data should there be instant feedback? This is subjective, instant feedback will be better when there are many inputs so this would be better than waiting to submit their input. In the case of instant feedback there needs to be events triggered to perform validation while on submit there is a central generic method to perform validation.

Let's take another view of validating on submit, with all types of application which can be created with Visual Studio there are methods which permit displaying messages to assist the user to correct mistakes. So back to subjective, it may come down to if code is written in a team or individual in regards to, are there team rules or it's the Wild West.

Preface

  • This repository contains validation code samples for user input, not validation of data coming from a database or other external sources, for working with databases and external sources that deserves it's own article.
  • Focus is on Fluent Validation is a validation library for .NET, used for building strongly typed validation rules for business objects. Fluent validations use a Fluent interface and lambda expressions to build validation rules.
  • For working with Data Annotations see Class/model validation basics for web and desktop which focuses on data annotations with several examples also on Fluent Validation while this article has more advance FluentValidation examples.

Basics

graph LR
    A[Request for work] -->|Get requirements from customer| B(Get approval)
    B --> C{Write out requirements}
    C -->|Task 1| D[Create database]
    C -->|Task 2| E[Create Visual Studio solution]
    C -->|Task 3| F[Plan out work in iterations]
Loading

First figure out models (classes) needed for an application followed by writing out business rules. For instance, for a customer model a simple example of a rule, a property name FirstName is required, must be at least three characters and less than ten characters and a property named BirthDate cannot be less than year 1932 and not greater than 2021 are some examples of rules.

Let's look at FirstName,

Data annotations, the StringLength attribute denotes min and max length while Required attribute indicates that first name can't be empty.

[Required]
[StringLength(10, MinimumLength = 3)]
public string FirstName { get; set; }

Fluent Validation there are no attributes

Fluent Validation home page

public string FirstName { get; set; }

Instead rules are setup in a validator class which inherits AbstractValidator<T>. In the case of our Customer model, FirstName

    public class CustomerValidator : AbstractValidator<Customer>
    {
        public CustomerValidator()
        {
            var settings = SettingsOperations.ReadCustomerSettings();

            RuleFor(customer => customer.FirstName)
                .NotEmpty()
                .MinimumLength(settings.FirstNameSettings.MinimumLength)
                .MaximumLength(settings.FirstNameSettings.MaximumLength)
                .WithName(settings.FirstNameSettings.WithName);

In the above, settings reads min, max length from a json file and could also be from a database while with data annotations this is not possible out of the box.

The json file for above.

{
  "FirstNameSettings": {
    "MinimumLength": 3,
    "MaximumLength": 10,
    "WithName": "First name"
  },
  "LastNameSettings": {
    "MinimumLength": 5,
    "MaximumLength": 30,
    "WithName": "Last name"
  }
}

Using a data source as in the above json, this means min, max length and WithName (allows a developer to use an alternate name for the property name, in this case FirstName and LastName.

Rules

To specify a validation rule for a particular property, call the RuleFor1 method, passing a lambda expression that indicates the property that you wish to validate.

Simple example

public class CustomerValidator : AbstractValidator<Customer>
{
  public CustomerValidator()
  {
    RuleFor(customer => customer.FirstName).NotNull();
  }
}

Here are some of the built in rules.

Figure1

There are cases were these will not suit all business rules so a developer can create their own.

Perhaps a Date can not be on a weekend. In the following, IsNotWeekend() is a language extension.

RuleFor(customer => customer.AppointmentDate)
    .Must(dateTime => dateTime.IsNotWeekend())
    .WithName("Appointment date")
    .WithMessage("We are not open on weekends");

In other cases a custom validator may be needed, in this case to ensure a list property is not null.

public class CountryValidator : AbstractValidator<Country>
{
    public CountryValidator()
    {
        RuleFor(country => country).NotNull();
    }
}

Usage

RuleFor(customer => customer.Country).SetValidator(new CountryValidator());

PreValidation

flowchart LR
    id1["Advance"]
    style id1 fill:#ffff99,stroke:#333,stroke-width:2px
Loading

PreValidate2 Suppose there may be a case where the instance of a model might be null, simply override PreValidate event.

protected override bool PreValidate(ValidationContext<Customer> context, ValidationResult result)
{
    if (context.InstanceToValidate is null)
    {
        result.Errors.Add(new ValidationFailure("", $"Dude, must have a none null instance of {nameof(Customer)}"));
        return false;
    }

    return true;
}

Override property name

In the case of FirstName which great for a developer, not the case for a user so use .WithName("First name")

Dropdown selections

For adding a new instance of your model, many times a developer will have a Select option to have the user pick a selection.

Figure2

Simple create a rule. Yes the NotEqual is hard-wired but as with using a json file for FirstName and LastName we can add this into in the case you don't want to use Select, perhaps 'Choice`.

RuleFor(customer => customer.Country.CountryName)
    .NotEqual("Select")
    .WithMessage("Please select a country");

Windows Forms Playground

There is a Windows form application which will allow testing rules visually.

Figure3

Unit test

Also included is a unit test project to play around and learn working with FluentValidation.

Performing validation

No matter if it's a Windows Form, WPF, ASP.NET or unit test the following shows how to invoke a validator.

_customerValidator = new CustomerValidator();

// perform validation
ValidationResult result = _customerValidator.Validate(customer);

if (result.IsValid)
{
    Operations.UpdateCustomer(customer);
}

If you want to see the error messages if any use PresentErrorMessage and to simply display any error messages to Visual Studio output window use ShowErrorMessage.

_customerValidator = new CustomerValidator();

// perform validation
ValidationResult result = _customerValidator.Validate(customer);

Dialogs.Information(result.PresentErrorMessage());

Dialogs

Rather than use conventional MessageBox.Show, I've used TaskDialog in the following project.

Figure4

The footer can easily be customized or removed.

See the following repository dedicated to TaskDialog.

Related

FluentValidation tips written with Microsoft Visual Studio 2022 17.4.x

Summary

Using FluentValidation is one way to perform validation, may or may not be right for every developer, some may want to use data annotations or a third party library like Postsharp.

Only you can make the choice which to use but choice one of these rather than hand write code to validate each property in a class or worst in a form.

Footnotes

  1. The .RuleFor method on the Validator class is used to build up rule chains for properties on your model.

  2. If you need to run specific code every time a validator is invoked, you can do this by overriding the PreValidate method. This method takes a ValidationContext as well as a ValidationResult, which you can use to customise the validation process.