Rocket.Libraries.Validation
1.1.4
See the version list below for details.
dotnet add package Rocket.Libraries.Validation --version 1.1.4
NuGet\Install-Package Rocket.Libraries.Validation -Version 1.1.4
<PackageReference Include="Rocket.Libraries.Validation" Version="1.1.4" />
paket add Rocket.Libraries.Validation --version 1.1.4
#r "nuget: Rocket.Libraries.Validation, 1.1.4"
// Install Rocket.Libraries.Validation as a Cake Addin #addin nuget:?package=Rocket.Libraries.Validation&version=1.1.4 // Install Rocket.Libraries.Validation as a Cake Tool #tool nuget:?package=Rocket.Libraries.Validation&version=1.1.4
Rocket.Libraries.Validation
A Functional C# validation library
What It Is
This is a simple library that allows you to organize validation rules and then evaluate the rules during runtime with an exception getting thrown if any of the rules, fails.
Why Use a Library?
I found myself writing a forest of if-elseif-else or switch blocks to handle validation.
While these conditional blocks work, they tend to get more unreadable and more unmaintainable with the number of checks required.
Worse still while working in a team, there wasn't a consistent way of that team members were using to validate and hence checks were peppered all over code in various flavours.
What this validation library allows you to do, is to collapse what would be branch statements into inline statements, for more manageable code.
Package
Installation, grab the package from nuget https://www.nuget.org/packages/Rocket.Libraries.Validation/
Example Usage
Single Condition Evaluation
You may for example wish to check that an object isn't null before you proceed to use it. In such cases, it is common to write code similar to below.
public void DoStuff(User user)
{
if(user == null)
{
throw new Exception("User was not supplied");
}
else
{
//Do stuff with the user object
}
}
With Rocket Validation, you can reduce the complexity of the above code by re-writing it as below.
private DataValidator _validator = new DataValidator();
public void DoStuff(User user)
{
_validator
.EvaluateImmediate(() => user == null, "User was not supplied");
//Do stuff with the user object
}
The inline EvaluateImmediate method takes in a boolean function as its first parameter, which if evaluated to true, causes an exception to be thrown. The thrown exceptions's message is the string passed as the second parameter to EvaluateImmediate
The second code snippet that uses the library is more readable, enforces more structure and introduces less code complexity.
Multiple/Chained Condition Evaluation
In this example, we'll check that a User object is not null and that the property Username is not empty and that the length of the Username property is not less than 6 characters. Without the library, we can handle that validation as shown below.
public bool ValidateUserName(User user)
{
const int minLength = 6;
if(user != null)
{
if(!string.IsNullOrEmpty(user.Username))
{
if(user.Username.length < minLength)
{
throw new Exception($"Username '{user.Username}' is too short. At least {minLength} characters are required");
}
else
{
return true;
}
}
else
{
throw new Exception("Username was not supplied. Usernames are mandatory");
}
}
else
{
throw new Exception("No user was supplied");
}
}
}
With the validation library, we're able collapse the above code to something similar to the following:
private DataValidator _validator = new DataValidator();
public bool ValidateUserName(User user)
{
const int minLength = 6;
_validator
.AddFailureCondition(() => user == null, "User was not supplied", true)
.AddFailureCondition(() => string.IsNullOrEmpty(user.Username), "Username was not supplied. Usernames are mandatory", false)
.AddFailureCondition(() => user.Username.Length < minLength, $"Username '{user.Username}' is too short. At least {minLength} characters are required", false)
.ThrowExceptionOnInvalidRules();
}
The AddFailureCondition method also take in a Func<bool>
as its first parameter, which should evaluate to true to indicate a failing validation. As a second paramter, it takes in a string
which serves as the error message for the failing condition, and finally it takes a bool
parameter to indicate whether or not a rule is critical, i.e does the failing of a rule make validation of follow-up rules unnecessary.
An example of a critical rule is whether our user
object is null, as if the object is indeed null, then it follows that trying to check whether user.Username
is empty would result in a NullPointerException
being thrown by the runtime.
Again, using the library greatly reduces code complexity and also as an bonus inlining allows error messages to live next to the rule they belong to.
In opposition to inlining, the snippet without the library has rule in the if
part of the clause and the error message in the else
. A practise that makes it harder to follow what the code is doing as rules increase especially if the if/else
blocks get nested too deeply.
Not only does the Rocket Validation Library reduce the lines of code required, but it enforces structured validation, and also increases code readabililty which will result in easier to troubleshoot code, thats more maintainable.
Validation Results
IncorrectDataException
ONLY in cases where a validation rules fails, whether it be via the single rule method EvaluateImmediate or the chained rule method AddFailureCondition (or its async relative AddFailureConditionAsync) is an exception of type IncorrectDataException is thrown. If no rule(s) fail, then no exception is thrown.
The IncorrectDataException is a simple sub-class of the Exception object, with the main addition being the property Errors, which is a list that contains all the error messages from the failing rule(s).
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.2 is compatible. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
-
.NETCoreApp 2.2
- No dependencies.
NuGet packages (2)
Showing the top 2 NuGet packages that depend on Rocket.Libraries.Validation:
Package | Downloads |
---|---|
Rocket.Libraries.QBuilder
Helper library for querying databases from C# |
|
Rocket.Libraries.Emailing
Package Description |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
2.0.2 | 1,181 | 2/28/2020 |
2.0.0-beta05 | 1,934 | 10/29/2019 |
2.0.0-beta02 | 350 | 9/4/2019 |
2.0.0-beta01 | 350 | 9/3/2019 |
1.1.8 | 530 | 9/2/2019 |
1.1.7 | 472 | 9/2/2019 |
1.1.5-beta10 | 362 | 7/30/2019 |
1.1.5-beta02 | 373 | 7/24/2019 |
1.1.4 | 481 | 7/21/2019 |