Xpandables.Net.AspNetCore
8.1.2
See the version list below for details.
dotnet add package Xpandables.Net.AspNetCore --version 8.1.2
NuGet\Install-Package Xpandables.Net.AspNetCore -Version 8.1.2
<PackageReference Include="Xpandables.Net.AspNetCore" Version="8.1.2" />
paket add Xpandables.Net.AspNetCore --version 8.1.2
#r "nuget: Xpandables.Net.AspNetCore, 8.1.2"
// Install Xpandables.Net.AspNetCore as a Cake Addin #addin nuget:?package=Xpandables.Net.AspNetCore&version=8.1.2 // Install Xpandables.Net.AspNetCore as a Cake Tool #tool nuget:?package=Xpandables.Net.AspNetCore&version=8.1.2
Introduction
Provides with useful interfaces contracts in .Net 8.0 and some implementations mostly following the spirit of SOLID principles, Commands... The library is strongly-typed, which means it should be hard to make invalid requests and it also makes it easy to discover available methods and properties though IntelliSense.
Feel free to fork this project, make your own changes and create a pull request.
Getting Started
Optional
The Optional< T> type allows to create a named value or a variable that contains a value or not. An Optional has an underlying type and can hold a value of that type, or it might not have a value.
- Is a struct, immutable, a generic type, so it can hold a value of any type.
- Is a monad, implements IEnumerable< T>, so it can be used in a foreach loop.
Creating an Optional : Some or Empty
var optional = Optional.Some("Hello World");
// optional : Optional<string> = [Some("Hello World")]
Apply a function to the value of an Optional : Map
var optional = Optional.Some("Hello World");
var result = optional.Map(value => value.ToUpper());
// result : Optional<string> = [Some("HELLO WORLD")]
Change the type of an Optional : Bind
var optional = Optional.Some("Hello World");
var result = optional.Bind(value => value.Length);
// result : Optional<int> = [Some(11)]
Change the return value of a function : Empty
public string GetName()
{
Optional<Name> optional = function call;
return optional
.Empty("No Name");
// If the optional has a value, the function value will be returned.
// If the optional is empty, the Empty value will be returned.
}
Apply serialization to an Optional : Serialize/Deserialize
// value type
readonly record struct Name(string Value);
var optional = Optional.Some(new Name("Hello World"));
var result = JsonSerializer.Serialize(optional);
// result : Value = "Hello World"
// reference type
sealed record class Name(string Value);
var optional = Optional.Some(new Name("Hello World"));
var result = JsonSerializer.Serialize(optional);
// result : string = "{\"Value\":\"Hello World\"}"
// anonymous type
var optional = Optional.Some(new { Name = "Hello World" });
var result = JsonSerializer.Serialize(optional);
// result : string = "{\"Name\":\"Hello World\"}"
var deserialized = result.DeserializeAnonymousType(Optional.Some(new { Name = string.Empty }));
// or you can use the same anonymous variable
// var deserialized = result.DeserializeAnonymousType(optional);
// DeserializeAnonymousType is an extension method that allows to deserialize an anonymous type
// the method is defined in the Xpandables.Net.Primitives.Text namespace
IOperationResult
Allows to create methods that return the status of an execution.
This interface contains all properties according to the result of the method execution. Some of those properties let you determine for example if the result instance is generic, a collection of errors, the status code or the value of the execution. The status code here is the one from the System.Net.HttpStatusCode. It contains methods to convert from non-generic type to generic and vis-versa. The type is useful if you want to return a result that can be analyzed even in a web environment by using some extensions that can automatically convert an OperationResult to IResult.
The non generic type has the following properties :
- An optional object Result, a property that qualifies or contains information about an operation return if available.
- An optional Uri LocationUrl, a property that contains the URL mostly used with the status code Created in the web environment.
- An ElementCollection Headers property that contains a collection of headers if available. ElementCollection is a predefined record struct that contains a collection of ElementEntry with useful methods.
- An ElementCollection Extensions property that contains a collection of extensions if available, that will be returned with the response.
- An ElementCollection Errors property that stores errors. Each error is a predefined ElementEntry struct which contains the error key and the error message and/or exceptions.
- A HttpStatusCode StatusCode property that contains the status code of the execution. The status code from the System.Net.HttpStatusCode.
- A boolean IsGeneric to determine whether or not the current instance is generic.
- A boolean IsSuccess and IsFailure to determine whether or not the operation is a success or a failure according to System.Net.HttpStatusCode.
- An optional string Title that contains the operation summary problem from the execution operation.
- An optional string Detail that contains he operation explanation specific to the execution operation.
The generic type overrides the object Result to TResponse type.
Create a method that returns an IOperationResult :
public IOperationResult CheckThatValueMatchCriteria(string? value)
{
if(string.IsNullOrEmpty(value))
{
return OperationResults
.BadRequest()
.WithError(nameof(value), "value can not be null")
.Build();
}
return OperationResults
.Ok()
.Build();
}
The method returns an IOperationResult* implementation type. To do so, you can use the specific extension method according to your needs :
- OperationResults which is a factory to create specifics results from Ok to InternalServerError.
- build your owns using the IOperationResult.ISuccessBuilder or IOperationResult.IFailureBuilder
Each extension method allows you to add errors, headers, Uri or a value to the target operation result. The key here in error can be the name of the member that has the error. The caller of this method can check if the return operation is a success or a failure result.
When used in an Asp.Net Core application, you will need to add the : Xpandables.Net.AspNetCore NuGet package that will provides helpers to automatically manage IResult responses. It also provides with a middleware that will automatically convert a failure OperationResult to ValidationProblem, Problem or result, according to the StatusCode.
// Minimal Api
// You need to the following code in the Program.cs file
builder.Services
.AddXOperationResultMiddleware()
.AddXOperationResultResponseBuilder()
.AddXOperationResultSerializationConfigureOptions();
...
app.UseXOperationResultMiddleware();
app.MapGet("/api/users", (string name) =>
{
if(CheckThatValueIsNotNull(name) is { IsFailure : true} failure)
return failure;
// ...get the user
IOperationResult<User> resultUser = DoGetUser(...);
// return the result
return result;
})
.WithXOperationResultFilter()
.WithXValidatorFilter();
// WithXValidatorFilter is an extension method that allows to use the validator filter
// to automatically validate the request according to the specified type.
// and allows you to use custom validation implementation using **IValidator** interface.
// WithXOperationResultFilter is an extension method that allows to use the operation result filter
// to automatically convert the result to a specific response according to the status code.
// You can implement the **IOperationResultResponseBuilder** interface to create a custom response.
// Controller
// You need to the following code in the Program.cs file
builder.Services
.AddXOperationResultMiddleware()
.AddXOperationResultResponseBuilder()
.AddXOperationResultConfigureMvcOptions();
...
app.UseXOperationResultMiddleware();
[HttpGet]
public object GetUserByName(string name)
{
if(CheckThatValueIsNotNull(name) is { isFailure : true} failure)
return failure;
// ...get the user
IOperationResult<User> resultUser = DoGetUser(...);
return result;
}
In the Minimal Api case, if the name is null, the operation result from the method will be converted to an implementation of IResult, that will produce a perfect response with all needed information.
You can also use the OperationResultException to throw a specific exception that contains a failure IOperationResult when you are not able to return an IOperationResult instance. All the operation result instances are serializable with a specific case for Asp.Net Core application, the produced response Content will contains the serialized Result property value if available in the operation result. You will find the same behavior for all the interfaces that use the IOperationResult in their method as return value such as : IRequestHandler< TRequest >, IRequestHandler< TRequest, TResponse >, ...*
Decorator pattern
You can use the extension methods to apply the decorator pattern to your types.
This method and its extensions ensure that the supplied TDecorator" decorator is returned, wrapping the original registered "TService", by injecting that service type into the constructor of the supplied "TDecorator". Multiple decorators may be applied to the same "TService". By default, a new "TDecorator" instance will be returned on each request, independently of the lifestyle of the wrapped service. Multiple decorators can be applied to the same service type. The order in which they are registered is the order they get applied in. This means that the decorator that gets registered first, gets applied first, which means that the next registered decorator, will wrap the first decorator, which wraps the original service type.
services.XTryDecorate<TService, TDecorator, TMarker>();
Suppose you have a request and a request handler defined like this :
public sealed record AddPersonRequest : IRequest;
public sealed class AddPersonRequestHandler : IRequestHandler<AddPersonRequest>
{
public Task<IOperationResult> HandleAsync(
AddPersonRequest request,
CancellationToken cancellationToken = default)
{
// your code ...
return OperationResults.Ok().Build();
}
}
Suppose you want to add logging for the AddPersonRequestHandler, you just need to define the decorator class that will use the logger and the handler.
```c#
public sealed class AddPersonRequestHandlerLoggingDecorator :
IRequestHandler<AddPersonRequest>
{
private readonly IRequestHandler<AddPersonRequest> _decoratee;
private readonly ILogger<AddPersonRequestHandler> _logger;
public AddPersonRequestHandlerLoggingDecorator(
ILogger<AddPersonRequestHandler> logger,
IRequestHandler<AddPersonRequest> decoratee)
=> (_logger, _decoratee) = (logger, decoratee);
public async Task<IOperationResult> HandleAsync(
AddPersonRequest request,
CancellationToken cancellationToken = default)
{
_logger.Information(...);
var response = await _decoratee
.HandleAsync(request, cancellationToken)
.configureAwait(false);
_logger.Information(...)
return response;
}
}
And to register the decorator, you just need to call the specific extension method :
services
.AddXRequestHandlers()
.XTryDecorate<AddPersonRequestHandler, AddPersonRequestHandlerLoggingDecorator>();
Sometimes you want to use a generic decorator. You can do so for all requests that implement IRequest interface or something else. ILoggerDecorator is a marker interface that allows to apply the logger decorator to the request.
public sealed record AddPersonRequest : IRequest, ILoggerDecorator;
public sealed class RequestLoggingDecorator<TRequest> : IRequestHandler<TRequest>
where TRequest : notnull, IRequest, ILoggerDecorator // you can add more constraints
{
private readonly IRequestHandler<TRequest> _ decoratee;
private readonly ILogger<TRequest> _logger;
public RequestLoggingDecorator(
ILogger<TRequest> logger,
IRequestHandler<TRequest> decoratee)
=> (_logger, _ decoratee) = (logger, decoratee);
public async Task<IOperationResult> HandleAsync(
TRequest request,
CancellationToken cancellationToken = default)
{
_logger.Information(...);
var response = await _decoratee
.HandleAsync(request, cancellationToken)
.configureAwait(false);
_logger.Information(...)
return response;
}
}
And for registration the RequestLoggingDecorator will be applied to all request handlers whose requests meet the decorator's constraints : To be a notnull and implement IRequest interface.
services
.AddXRequestHandlers()
.XTryDecorate(typeof(IRequestHandler<>), typeof(RequestLoggingDecorator<>), typeof(ILoggerDecorator));
CQRS Pattern
Stands for Command and Query Responsibility Segregation, a pattern that separates read and update operations for a data store.
The following interfaces are used to apply request operations :
public interface IRequest<TResponse> {}
public interface IAsyncRequest<TResponse> {}
public interface IRequest {}
public interface IRequestHandler<TRequest, TResponse>
where TRequest : notnull, IRequest<TResponse>
{
Task<IOperationResult<TResponse>> HandleAsync(
TRequest request,
CancellationToken cancellationToken = default);
}
public interface IAsyncQueryHandler<TRequest, TResponse>
where TRequest : notnull, IAsyncRequest<TResponse>
{
IAsyncEnumerable<TResponse> HandleAsync(
TRequest request,
CancellationToken cancellationToken = default);
}
public interface IRequestHandler<TRequest>
where TRequest : notnull, IRequest
{
Task<IOperationResult> HandleAsync(
TRequest request,
CancellationToken cancellationToken = default);
}
So let's create a request and its handler. A request to add a new product for example.
public sealed record AddProductRequest(
[property : Required] Guid Id,
[property : StringLength(byte.MaxValue, MinimumLength = 3)] string Name,
[property : StringLength(short.MaxValue, MinimumLength = 3)] string Description) :
IRequest, IPersistenceDecorator;
IPersistenceDecorator interface is to allow the request to be persisted at the end of the control flow when there is no exception. Entity is a base class that contains common properties for entities.
public sealed class AddProductRequestHandler : IRequestHandler<AddProductRequest>
{
private readonly ProductContext _uow;
public AddProductRequestHandler(ProductContext uow) => _uow = uow;
public async Task<IOperationResult> HandleAsync(
AddProductRequest request,
CancellationToken cancellationToken)
{
// create the new product instance : 'With' is static method to build a product
var product = Product.With(request.Id, request.Name, request.Description);
// insert the new product in the collection of products
await _uow.Products
.AddAsync(product, cancellationToken)
.ConfigureAwait(false);
//...
// you can return the product id or the product itself
return OperationResults
.Ok(product)
.Build();
// or
return OperationResults.Ok().Build();
}
}
The validation of the request, the validation of request duplication and persistence will happen during the control flow using decorators.
public sealed class AddProductRequestValidator<AddProductRequest> :
Validator<AddProductRequest>
{
private readonly ProductContext _uow;
public AddProductRequestValidator(ProductContext uow, IServiceProvider sp)
:base(sp) => _uow = uow;
public async Task<IOperationResult> ValidateAsync(AddProductRequest argument)
{
// validate the request using attributes
if(Validate(request) is { isFailure : true } failure)
return failure;
// check for duplication
// You can stop here because if a duplication error occurs while saving,
// the final operation result will contain this error.
// this is just for demo
// we just need to know if a record with
// the specified id already exist
var isFound = await _uow.Products
.CountAsync(x => x.Id == request.Id, cancellationToken)
.ConfigureAwait(false) > 0;
if( isFound ) // duplicate
{
// the result can directly be used in a web environment
return OperationResults
.Conflict()
.WithError(nameof(request.Id), $"Command identifier '{request.Id}' already exist")
.Build();
}
return OperationResults.Ok().Build();
}
}
And now let's create a request and its handler to request a product.
public readonly record struct ProductDTO(string Id, string Name, string Description);
public sealed record GetProductRequest(Guid Id) : IRequest<ProductDTO?>;
// You can use a class and apply a filter directly on that class :
public sealed record GetProductRequest(Guid Id) :
QueryExpression<Product>, IRequest<ProductDTO?>
{
public override Expression<Func<Product, bool>> GetExpression()
=> x => x.Id == Id;
}
public sealed class GetProductRequestHandler :
IRequestHandler<GetProductRequest, ProductDTO?>
{
private readonly ProductContext _uow;
public GetProductRequestHandler(ProductContext uow) => _uow = uow;
public async Task<IOperationResult<ProductDTO?>> HandleAsync(
GetProductRequest request,
CancellationToken cancellationToken = default)
{
// You can make a search using a request or the key
if( await _uow.Products
.Where(request)
.OrderBy(o => o.Id)
.Select(s => new ProductDTO(x.Id, x.Name, x.Description))
.FirstOrDefaultAsync(canellationToken)
.ConfigureAwait(false)
is { } productDTO)
{
return OperationResults
.Ok<ProductDTO?>(productDTO)
.Build();
}
// create a key for search --------------
var key = ProductId.With(request.Id);
if(await _uow.Products
.FindAsync(key, cancellationToken)
.ConfigureAwait(false) is { } product)
{
ProductDTO productDTO = new(product.Id, product.Name, product.Description);
return OperationResults
.OkResult<ProductDTO?>()
.WithResult(productDTO)
.Build();
}
return OperationResults
.NotFound<ProductDTO?>()
.Build();
}
}
Finally, we need to use the dependency injection to put it all together :
var serviceProvider = new ServiceCollection()
.AddXDataContext<ProductContext>(define options)
.AddXAllRequestHandlers(
options => options.UsePersistence().UseValidator())
.AddXDistributor()
.BuildServiceprovider();
// Add a product
var distributor = serviceProvider.GetRequiredService<IDistributor>();
var addProduct = new AddProductRequest("Xpandables 8", "Xpandables.Net Library");
IOperationResult result = await distributor
.SendAsync(addProduct)
.ConfigureAwait(false);
// check the result
...
The AddXDataContext registers the specified data context using the options provided.
The AddXRequestHandlers registers all handlers found in the executing application, and apply persistence decorator and validation decorator to all the requests according to the constraints.
The AddXDistributor registers the internal implementation of IDistributor to resolve handlers.
Features
Usually, when registering types, we are forced to reference the libraries concerned and we end up with a very coupled set. To avoid this, you can register these types by calling an export extension method, which uses MEF: Managed Extensibility Framework.
In your api program class
AddXServiceExport(IConfiguration, Action{ExportServiceOptions}) adds and configures registration of services using the IAddServiceExport interface implementation found in the target libraries according to the export options. You can use configuration file to set up the libraries to be scanned.
....
builder.Services
.AddXServiceExport(
Configuration,
options => options.SearchPattern = "your-search-pattern-dll");
...
In the library you want types to be registered
[Export(typeof(IAddServiceExport))]
public sealed class RegisterServiceExport : IAddServiceExport
{
public void AddServices(IServiceCollection services, IConfiguration configuration)
{
// you can register your services here
....
}
}
IAggregate
Libraries also provide with DDD model implementation IAggregate using event sourcing and out-box pattern.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. 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. |
-
net8.0
- Xpandables.Net (>= 8.1.2)
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 |
---|---|---|
9.0.0-rc.1 | 45 | 10/26/2024 |
8.1.2 | 114 | 9/12/2024 |
8.0.8 | 112 | 6/21/2024 |
8.0.6 | 98 | 5/25/2024 |
8.0.5 | 107 | 5/18/2024 |
8.0.1 | 299 | 2/11/2024 |
8.0.0 | 441 | 12/3/2023 |
8.0.0-rc.2.1.1 | 94 | 11/12/2023 |
8.0.0-rc.2.1 | 79 | 11/6/2023 |
8.0.0-rc.2.0 | 73 | 11/5/2023 |
7.3.3 | 494 | 5/9/2023 |
7.1.4 | 561 | 2/26/2023 |
7.1.3 | 605 | 2/19/2023 |
7.0.0 | 663 | 11/9/2022 |
7.0.0-rc2.0.1 | 102 | 10/12/2022 |
7.0.0-rc1.0.0 | 146 | 9/26/2022 |
6.1.1 | 746 | 8/6/2022 |
6.0.9 | 754 | 7/9/2022 |
6.0.8 | 781 | 6/27/2022 |
6.0.4 | 799 | 3/15/2022 |
6.0.3 | 726 | 2/22/2022 |
6.0.2 | 568 | 1/4/2022 |
6.0.1 | 558 | 12/4/2021 |
6.0.0 | 618 | 11/8/2021 |
6.0.0-rc.4.3 | 152 | 11/3/2021 |
6.0.0-rc.3.1 | 165 | 10/15/2021 |
6.0.0-rc.3 | 152 | 10/14/2021 |
6.0.0-rc.2 | 157 | 9/21/2021 |
6.0.0-preview.5 | 169 | 8/26/2021 |
5.6.1 | 668 | 6/30/2021 |
5.6.0 | 673 | 6/9/2021 |
5.5.1 | 633 | 5/26/2021 |
5.4.4 | 633 | 4/12/2021 |
Add Decider pattern to Aggregate