Phema.Validation
3.1.10
dotnet add package Phema.Validation --version 3.1.10
NuGet\Install-Package Phema.Validation -Version 3.1.10
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Phema.Validation" Version="3.1.10" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Phema.Validation --version 3.1.10
The NuGet Team does not provide support for this client. Please contact its maintainers for support.
#r "nuget: Phema.Validation, 3.1.10"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Phema.Validation as a Cake Addin #addin nuget:?package=Phema.Validation&version=3.1.10 // Install Phema.Validation as a Cake Tool #tool nuget:?package=Phema.Validation&version=3.1.10
The NuGet Team does not provide support for this client. Please contact its maintainers for support.
Phema.Validation
A simple, lightweight, fluent and extensible validation library for .NET Core
Installation
$> dotnet add package Phema.Validation
Core concepts
IValidationContext
- Scoped service to store all validation detailsIValidationCondition
- Contains a validation checks (e.g.Is(() => ...)
)IValidationDetail
- WhenIValidationCondition
is not valid adds toIValidationContext.ValidationDetails
ValidationSeverity
- Validation error level, used inIValidationContext.ValidationSeverity
andIValidationDetail.ValidationSeverity
IValidationScope
- Is a nested validation context with validation path and severity override
Usage (ASP.NET Core, HostedService examples)
// Add `IValidationContext` as scoped service
services.AddValidation(options => ...);
// Get or inject
var validationContext = serviceProvider.GetRequiredService<IValidationContext>();
// Validation key will be `Name` using default validation part provider
validationContext.When(person, p => p.Name)
.Is(name => name == null)
.AddValidationDetail("Name must be set");
// Validation key will be `Address.Locations[0].Latitude` using default validation part provider
validationContext.When(person, p => p.Address.Locations[0].Latitude)
.Is(latitude => ...custom check...)
.AddValidationDetail("Some custom check failed");
Validation conditions
- Monads are not composable, so
Is
andIsNot
,IsNull
andIsNotNull
... duplication
// Check for Phema.Validation.Conditions namespace
validationContext.When(person, p => p.Name)
.IsNullOrWhitespace()
.AddValidationDetail("Name must be set");
// Use multiple conditions (joined with AND)
validationContext.When(person, p => p.Name)
.IsNotNull()
// AND
.HasLengthGreater(20)
// .IsNull()
// .IsEqual()
// .IsNotUrl()
// .IsNotEmail()
// .IsMatch(regex)
.AddValidationDetail("Name should be less than 20");
// DateTime conditions
validationContext.When(task, t => t.DueDate)
.IsNotUtc()
.AddValidationDetail("Due date must be in Utc");
// Type checks
validationContext.When(person, p => p.Car)
.IsType<Ferrari>(typed => typed.Is(ferarriCar => ...Some Ferrari specific checks...))
.AddValidationDetail("You have Ferrari car, but ...");
Validation details
var validationDetails = validationContext
.When(person, p => p.Age)
// Validation check is failed, validation condition is valid
.Is(() => false)
.AddValidationDetail("Age must be set");
// Use deconstruction
var (key, message) = validationContext
.When(person, p => p.Age)
.IsNull()
.AddValidationDetail("Age must be set");
// More deconstruction
var (key, message, isValid) = validationContext
.When(person, p => p.Age)
.IsNull()
.AddValidationDetail("Age must be set");
// Even more deconstruction!
var (key, message, isValid, severity) = validationContext
.When(person, p => p.Age)
.IsNull()
.AddValidationDetail("Age must be set");
Check validation
// Throw exception when details severity greater than ValidationContext.ValidationSeverity
validationContext.When(person, p => p.Address)
.IsNull()
.AddValidationFatal("Address is not presented!!!"); // If invalid throw ValidationConditionException
// Check if context is valid
validationContext.IsValid();
validationContext.EnsureIsValid(); // If invalid throw ValidationContextException
// Check concrete validation details
validationContext.IsValid(person, p => p.Age);
validationContext.IsNotValid(person, p => p.Age);
validationContext.EnsureIsValid(person, p => p.Age);
Compose and reuse validation rules with extensions
- Call is allocation free
- Static checks
// Extensions
public static void ValidateCustomer(this IValidationContext validationContext, Customer customer)
{
// Some checks
}
validationContext.ValidateCustomer(customer);
- Write your own middleware or validation components/validators on top of
IValidationContext
Validation part resolvers
ValidationPartResolver
is a delegate, trying to getstring
valdiation part fromMemberInfo
- Use built in resolvers with
ValidationPartResolvers
static class:Default
,DataMember
,PascalCase
,CamelCase
// Configure DataMember validation part resolver
services.AddValidation(options =>
options.WithValidationPartResolver(ValidationPartResolvers.DataMember));
// Override validation parts with `DataMemberAttribute`
[DataMember(Name = "name")]
public string Name { get; set; }
Validation scopes
- Use scopes when you need to have:
- Same nested validation path multiple times
- Empty validation details collection (syncing with parent context/scope)
- ValidationSeverity override
// Validation key will be `Child.*ValidationPart*`
ValidateChild(validationContext.CreateScope(parent, p => p.Child))
// Validation key will be `Address.Locations[0].*ValidationPart*`
ValidateLocation(validationContext.CreateScope(person, p => p.Address.Locations[0]))
// Override local scope ValidationSeverity
using (var scope = validationContext.CreateScope(person, p => p.Address, ValidationSeverity.Warning))
{
// Some scope validation checks syncing with validationContext
}
High performance with non-expression constructions
validationContext.When("key", value)
.IsNull()
.AddValidationDetail("Value is null");
validationContext.CreateScope("key", ValidationSeverity.Warning);
validationContext.IsValid("key");
validationContext.IsNotValid("key");
validationContext.EnsureIsValid("key");
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. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
-
.NETStandard 2.0
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 3.0.0)
- Microsoft.Extensions.Options (>= 3.0.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
3.1.10 | 1,022 | 10/12/2019 |
3.1.9 | 1,073 | 9/27/2019 |
3.1.8 | 993 | 9/25/2019 |
3.1.7 | 360 | 9/15/2019 |
3.1.6 | 321 | 9/8/2019 |
3.1.5 | 303 | 9/7/2019 |
3.1.4 | 303 | 9/7/2019 |
3.1.3 | 325 | 9/7/2019 |
3.1.2 | 308 | 8/25/2019 |
3.1.1 | 295 | 8/24/2019 |
3.1.0 | 298 | 8/23/2019 |
3.0.9 | 302 | 8/23/2019 |
3.0.8 | 314 | 8/22/2019 |
3.0.7 | 298 | 8/16/2019 |
3.0.6 | 332 | 7/30/2019 |
3.0.5 | 295 | 7/29/2019 |
3.0.4 | 317 | 7/29/2019 |
3.0.3 | 302 | 7/28/2019 |
3.0.2 | 301 | 7/21/2019 |