This is a template for creating your own application using Clean Architecture, utilizing building blocks from Domain-Driven Design. Read and write operations have been separated according to the CQRS pattern. System observability is ensured by the implementation of OpenTelemetry and the Aspire Dashboard. Additionally, you'll find implementations of design patterns such as mediator, factory, strategy, and several others.
The implementation of business domain logic is kept to a minimum. Selected use cases were implemented to demonstrate communication between layers and the use of domain and integration events.
This project is undergoing rapid development, with new features being added frequently. Stay updated and click Watch button, click ⭐ if you find it useful.
- Table of contents
- 1. Installation
- 2. Introduction
- 3. Domain
- 4. Architecture
- 4.5 Caching
- 5. Observability
- 6 Design patterns implemented in this project
- 7. Tests
- 8. Build with
I do recommend using CLI instead Visual Studnio Wizard when creating your application based on this template because Vistual Studio doesn't keep folders structure correctly.
First and foremost, install this template on your system to be able to create applications based on it.
dotnet new install CA.And.DDD.Template
Once installed type. (Replace the phrase 'MyDreamProject' with the name of your project.)
dotnet new ca-and-ddd -o MyDreamProject
The template uses MSSQL as a database provider. Migrations will be applied automatically during project startup, so you don't have to do anything.
As a result of running command from step 1.1 all files and folders will be created. Among them you will find docker-compose.yaml. Simply run the command 'docker-compose up' to create required containers. docker-compose.yaml provides instances of: MSSQL, Redis, RabbitMQ, and Aspire Dashboard.
In case you need to add a new migration, just navigate to the src/ directory and run. (Please remember to replace "Clean.Architecture.And.DDD.Template" with you project name).
dotnet ef migrations add "Migration name" --context "AppDbContext" --project .\CA.And.DDD.Template.Infrastructure\ --startup-project .\CA.And.DDD.Template.WebApi\
I couldn't find any repository that met the following criteria:
- Implemented Mediator pattern using MassTransit library instead of MediatR with added message interception support (similar to IPipelineBehaviour in MediatR).
- Implemented system observability using Open Telemetry.
- Implemented Domain Events as part of Eventual Consistency, so Domain Events are not published in the same transaction as saving/updating Aggregate.
Even when encountering projects that fulfilled one of these points, they often conflicted with others or omitted them entirely. Therefore, I decided to create a project that meets the above criteria. I am sharing it in case someone else is looking for something similar.
The e-commerce domain was deliberately chosen because it is widely known and understood. This template consists of two aggregates: Customer and Order. I decided to extend the domain just enough to utilize all the building blocks, but nothing more. The goal of this repository is to create a template that provides an example implementation. Keeping things simple in the Domain layer allows you to focus on understanding complex topics, such as the building blocks of the Domain-Driven Design approach.
There are two aggregates in the Domain Layer. These are:
Customer – This is the most important aggregate in our layer because it is responsible for placing orders. In other words, this is the entity that starts the process of placing orders. Besides placing orders, the customer can also change their e-mail address and verify it.
Order – This represents a particular order. It has a list of order items, along with their price, quantity, etc.
The Policy pattern allows encapsulating domain logic into a separate class. This is very useful for testing and modifying it. In our case, there is a policy called AmountBasedDiscountPolicy, which is responsible for applying a discount to an order. As you can see below, the discount amount depends on how much the user spent last month, if not on the value of their current order.
Code
public class AmountBasedDiscountPolicy
{
public decimal CalculateDiscount(IReadOnlyCollection<OrderItem> orderItems)
{
var currentTotalAmount = orderItems.Sum(x => x.Quantity * x.Price.Amount);
if (currentTotalAmount > 800)
{
return 0.05m;
}
else if (currentTotalAmount > 600)
{
return 0.025m;
}
else
{
return 0.00m;
}
}
}
Domain services are building blocks that help encapsulate business logic that does not belong to any specific entity or aggregate.
In our example, we use it to calculate a discount for the order. We retrieve the total amount of money spent in the last month and pass it to the domain service named OrderDomainService. There, we calculate the discount using the TotalSpentMoneyInLast31DaysDiscountPolicy class.
Whenever possible, you should avoid creating domain services. Relying on them shifts the business logic from aggregates to these services, making the aggregates more anemic.
Domain events allow informing other parts of the application about changes that have occurred within our domain. They are an excellent way to implement business processes in a loosely coupled manner. Each domain event represents a specific action adhering to the (Single Responsibility Principle, SRP) that has happened in the system; therefore, event names are represented in the past tense, e.g., OrderCreatedDomainEvent. When a domain event is published, all interested parties can react to it. As the number of "interested parties" for a given domain event grows, all that needs to be done is to create an additional handler to perform extra logic. This approach also aligns with the Open/Closed Principle.
In our solution, domain events are stored in the database within the same transaction as the aggregate's save operation. This means they are not dispatched immediately but only after the aggregate has been successfully saved. Once this happens, the DomainEventsProcessor retrieves the events from the database and publishes them.
For example: When a Customer places an order, an event called OrderCreatedDomainEvent is stored in the database and then retrieved and published by the DomainEventsProcessor. An event handler, OrderCreatedDomainEventHandler, listens for this event and handles sending an email to the customer who placed the order. Implementing the business process in this way allows for leveraging the eventual consistency approach.
The diagram above shows all implemented Domain Events and their respective handlers.
Dispatching domain events within the same transaction or outside of it is a sticking point in the community. You can find various implementations on the web.
Outside transaction: ardalis/CleanArchitecture
During transaction: jasontaylordev/CleanArchitecture
By applying the Clean Architecture approach, we achieved a separation of the application into layers that are independent of each other, making them easy to test. The project consists of four layers, as shown in the image below.
The presentation layer (in our case, simply a RESTful API) is responsible for interacting with users by receiving and processing HTTP requests. It performs validation of commands and queries, and then communicates directly with the application layer to handle the request.
The Infrastructure layer is responsible for implementing technical aspects such as database access and broad integrations with external systems. This layer is also responsible for implementing interfaces defined in the domain and application layers. The Infrastructure layer references the Application layer.
This layer orchestrates processes in the application. The Application Layer only references the Domain Layer. Most of the application logic resides in command handlers, where you can implement specific scenarios (use cases).
This is the most important layer, as it contains business logic and implements business processes. It consists of aggregates, value objects, domain services, and other domain-related elements. This layer is the heart of the whole system. The domain layer is completely independent; it does not reference other layers and does not depend on any external libraries, making it very easy to test and maintain.
Saving and publishing domain events only after the aggregate has been saved means that we must use the approach known as eventual consistency.
In our case: When a Customer places an order, an event called OrderCreatedDomainEvent is stored in the database and then retrieved and published by the DomainEventsProcessor. An event handler, OrderCreatedDomainEventHandler, listens for this event and handles sending an email to the customer who placed the order.
If we decided to publish the OrderCreatedDomainEvent just before saving the aggregate to the database, there could be a situation where we send an email to the customer, but the save operation fails, for example, due to network issues.
Integration events are used to notify other modules or systems about important events that occurred in our system. Through these events, we can achieve consistency at the level of components or microservices by synchronizing data. In our solution, domain events are mapped to integration events and are only published after the aggregate has been successfully saved. The IntegrationEventsProcessor is responsible for fetching and publishing integration events from the database.
In our solution, we have only one handler, which was included purely for demonstration purposes. The CustomerCreatedIntegrationEvent is sent to a RabbitMq queue. http://localhost:15672/ login: guest, password: guest.
public class CustomerCreatedIntegrationEventHandler : IConsumer<CustomerCreatedIntegrationEvent>
{
public Task Consume(ConsumeContext<CustomerCreatedIntegrationEvent> context)
{
//This handler is being triggered as a result of mapping CustomerCreatedDomainEvent to CustomerCreatedIntegrationEvent.
//Integration events are the way to notify other modules / microservices about changes in our domain.
//This particular integration event is sent to RabbitMQ by IntegrationEventsProcessor and received here.
//This handler should never have been placed here; it should be placed in another module or microservice.
//However, I decided to leave that implementation here just to demonstrate how to register this handler
//in the IoC container and generally how to use it if needed in the future.
return Task.CompletedTask;
}
}
This handler should be placed in a separate module or another microservice.
Implementation of CQRS involves separating write and read requests. Commands are responsible for changing the state of aggregates (saving and updating them), and such operations are available only through specific repositories, e.g., ICustomerRepository.
public interface ICustomerRepository
{
Task AddAsync(Customer customer, CancellationToken cancellationToken = default);
Task<Customer?> GetAsync(string email, CancellationToken cancellationToken = default);
Task UpdateAsync(Customer customer, CancellationToken cancellationToken = default);
}
Queries are responsible for retrieving entities from the database. Queries are placed in the infrastructure layer to directly utilize the DbContext and avoid creating unnecessary abstractions.
Please take a look at the example: GetCustomerQueryHandler
public GetCustomerQueryHandler(ICacheService cacheService, ICustomerRepository customerRepository)
{
_cacheService = cacheService;
_customerRepository = customerRepository;
}
/// <summary>
/// This handler demonstrates the usage of the Cache Aside Pattern.
/// First, we check if the data is available in the cache (Redis). If not,
/// we retrieve the data from the database and store it in the cache.
/// </summary>
/// <param name="query"></param>
/// <returns></returns>
/// <exception cref="CustomerNotFoundApplicationException"></exception>
public async Task Consume(ConsumeContext<GetCustomerQuery> query)
{
var cachedCustomerDto = await _cacheService.GetAsync<CustomerDto>(CacheKeyBuilder.GetCustomerKey(query.Message.Email));
if (cachedCustomerDto is { })
{
await query.RespondAsync(cachedCustomerDto);
return;
}
var email = query.Message.Email;
var customer = (await _customerRepository.GetAsync(email))!.ToDto();
await _cacheService.SetAsync(CacheKeyBuilder.GetCustomerKey(query.Message.Email), customer);
await query.RespondAsync(customer);
}
Cross-cutting concerns are implemented using MassTransit filters. This is a very convenient and elegant approach to achieving these kinds of tasks. MassTransit filters follow the same concept as IPipelineBehavior<TRequest, TResponse> in the MediatR library. There are several filters implemented in our code:
cfg.ConfigureMediator((context, cfg) =>
{
cfg.UseConsumeFilter(typeof(ValidationFilter<>), context, x => x.Include(type => !type.HasInterface<IDomainEvent>()));
cfg.UseConsumeFilter(typeof(LoggingFilter<>), context, x => x.Include(type => !type.HasInterface<IDomainEvent>()));
cfg.UseConsumeFilter(typeof(RedisFilter<>), context, x => x.Include(type => !type.HasInterface<IDomainEvent>()));
cfg.UseConsumeFilter(typeof(EventsFilter<>), context, x => x.Include(type => !type.HasInterface<IDomainEvent>()));
cfg.UseConsumeFilter(typeof(HtmlSanitizerFilter<>), context, x => x.Include(type => !type.HasInterface<IDomainEvent>()));
});
ValidationFilter – is responsible for performing validation of commands and queries sent by the user. The validations are done using the FluentValidation library. If a validation error occurs, processing is interrupted, and a response is sent to the user with details about what went wrong.
Code
public class ValidationFilter<T> : IFilter<ConsumeContext<T>> where T : class
{
private readonly IEnumerable<IValidator<T>> _validators;
public ValidationFilter(IEnumerable<IValidator<T>> validators)
{
_validators = validators;
}
public async Task Send(ConsumeContext<T> context, IPipe<ConsumeContext<T>> next)
{
var _context = new ValidationContext<T>(context.Message);
var validationFailures = await Task.WhenAll(
_validators.Select(v => v.ValidateAsync(context.Message)));
if (validationFailures.Any(x => x.Errors.Any()))
{
var groupedErrors = validationFailures.SelectMany(x => x.Errors).GroupBy(x => x.PropertyName)
.ToDictionary(g => g.Key, g => g.Select(a => a.ErrorMessage).ToArray());
throw new CommandValidationException(String.Empty, groupedErrors);
}
await next.Send(context);
}
public void Probe(ProbeContext context) { }
}
LoggingFilter - is responsible for logging requests along with their total duration and payload. The current implementation logs all requests; however, you could, for example, detect only long-running requests and log them.
Code
public class LoggingFilter<T> : IFilter<ConsumeContext<T>> where T : class
{
private readonly ILogger<LoggingFilter<T>> _logger;
public LoggingFilter(ILogger<LoggingFilter<T>> logger)
{
_logger = logger;
}
public async Task Send(ConsumeContext<T> context, IPipe<ConsumeContext<T>> next)
{
Stopwatch stopwatch = new Stopwatch();
stopwatch.Start();
try
{
await next.Send(context);
}
finally
{
stopwatch.Stop();
}
_logger.LogTrace($"Operation duration: {stopwatch.Elapsed.TotalMilliseconds} ms", context);
}
public void Probe(ProbeContext context) { }
}
EventsFilter - is responsible for saving Domain Events and Integration Events to the database.
HtmlSanitizerFilter - is responsible for cleaning HTML that can lead to XSS attacks.
Caching was provided through the implementation of the Cache Aside Pattern. It is a simple pattern that can be described in three steps:
1. When the application requests data, it first checks if they are available in the cache.
2. If they are available, they are returned.
3. If they are not available, the data is retrieved from the database, stored in the cache, and then returned.
In our case, the implementation looks like this:
Code
/// <summary>
/// This handler demonstrates the usage of the Cache Aside Pattern.
/// First, we check if the data is available in the cache (Redis). If not,
/// we retrieve the data from the database and store it in the cache.
/// </summary>
/// <param name="query"></param>
/// <returns></returns>
/// <exception cref="OrderNotFoundApplicationException"></exception>
public async Task Consume(ConsumeContext<GetOrderQuery> query)
{
var cachedOder = await _cacheService.GetAsync<OrderDto>(CacheKeyBuilder.GetOrderKey(query.Message.Id));
if (cachedOder is { })
{
await query.RespondAsync(cachedOder);
}
var id = query.Message.Id;
var order = await _appDbContext
.Set<Order>()
.AsNoTracking()
.AsSplitQuery()
.Include(x => x.OrderItems)
.Where(x => ((Guid)x.OrderId) == id)
.FirstOrDefaultAsync();
if (order == null)
{
throw new OrderNotFoundApplicationException(id);
}
await _cacheService.SetAsync(CacheKeyBuilder.GetOrderKey(query.Message.Id), order);
await query.RespondAsync(
new OrderDto(
order.OrderId.Value,
order.OrderItems.MapToOrderItemDto()
)
);
}
Cache invalidation can be implemented in various ways. I decided to invalidate the cache as a result of processing commands, because they produce domain events. When a customer changes their email, an event called CustomerEmailChangedDomainEvent is emitted, and we can react to it by invalidating the cache.
Code
public class CustomerEmailChangedDomainEventHandler : IConsumer<CustomerEmailChangedDomainEvent>
{
private readonly ICacheService _cacheService;
public CustomerEmailChangedDomainEventHandler(ICacheService cacheService)
{
_cacheService = cacheService;
}
public async Task Consume(ConsumeContext<CustomerEmailChangedDomainEvent> context)
{
//Here, you could send an emails to old and new e-email addresses
//informing about the correct change of the email address.
// You could also include other logic here that should be part
// of the eventual consistency pattern.
var customerDto = await _cacheService.GetAsync<CustomerDto>(CacheKeyBuilder.GetCustomerKey(context.Message.OldEmailAddress));
if(customerDto is { })
{
await _cacheService.RemoveAsync(CacheKeyBuilder.GetCustomerKey(context.Message.OldEmailAddress));
}
}
}
The observability of the system has been ensured through the use of OpenTelemetry. Telemetry data is sent to the Aspire Dashboard collector and visualized there. With this approach, we can check how long an HTTP request took, how much time was spent communicating with the MSSQL database, and how much with Redis. Event logs are linked to requests, making it easy to navigate between them.
Aspire Dashboard is avaiable at: http://localhost:18888 once docker-compose has been launched.
Here is an example of creating a customer, which results in adding a new record in the Aspire Dashboard.
And here is the code responsible for setting up Open Telemetry.
Code
public static void InstallTelemetry(this WebApplicationBuilder builder, IConfiguration configuration, ConnectionMultiplexer redisConnection)
{
var telemetrySettings = builder.Configuration.GetSection(nameof(AppSettings)).Get<AppSettings>().Telemetry;
var url = $"{telemetrySettings.Host}:{telemetrySettings.Port}";
builder.Services.AddOpenTelemetry()
.ConfigureResource(resource => resource.AddService(telemetrySettings.Name, serviceInstanceId: Environment.MachineName))
.WithMetrics(metrics =>
{
metrics
.AddAspNetCoreInstrumentation()
.AddHttpClientInstrumentation();
metrics.AddOtlpExporter(options =>
{
if (!string.IsNullOrEmpty(url))
{
options.Endpoint = new Uri(url);
}
});
})
.WithTracing(tracing =>
{
tracing
.AddAspNetCoreInstrumentation()
.AddHttpClientInstrumentation()
.AddRedisInstrumentation(redisConnection, opt => opt.FlushInterval = TimeSpan.FromSeconds(1))
.AddEntityFrameworkCoreInstrumentation(options =>
{
options.EnrichWithIDbCommand = (activity, command) =>
{
var stateDisplayName = $"{command.CommandType} {command.CommandText} Database: {command.Connection?.Database}";
activity.DisplayName = stateDisplayName;
activity.SetTag("db.name", stateDisplayName);
};
});
tracing.AddOtlpExporter(options =>
{
if (!string.IsNullOrWhiteSpace(url))
{
options.Endpoint = new Uri(url);
}
});
});
builder.Logging.AddOpenTelemetry(logging =>
{
if (!string.IsNullOrEmpty(url))
{
logging.AddOtlpExporter(options => options.Endpoint = new Uri(url));
}
});
}
The Mediator from the MassTransit library was chosen because it doesn't require the implementation of any interfaces, unlike the MediatR library. If we were to use the Mediator from MediatR instead of MassTransit, our domain event would look like this:
public sealed record CustomerCreatedDomainEvent(Guid CustomerId) : INotification
INotification intefrace comes from MediatR library. However our domain events look like this:
public sealed record CustomerCreatedDomainEvent(Guid CustomerId) : IDomainEvent;
IDomainEvent is just a marker interface that we keep in our Domain Layer.
This is particularly important because domain events located in the domain layer can remain free of any dependencies. I believe that the domain layer should be free from all libraries, making it easier to test.
https://masstransit.io/documentation/concepts/mediator
Domain Events are mapped to Integration Events thanks to EventMapperFactory class.
public class EventMapperFactory
{
private readonly Dictionary<Type, IEventMapper> _mappers;
public EventMapperFactory(Dictionary<Type, IEventMapper> mappers)
{
_mappers = mappers;
}
public IEventMapper GetMapper(IDomainEvent domainEvent)
{
if (_mappers.TryGetValue(domainEvent.GetType(), out var mapper))
{
return mapper;
}
return null;
}
}
The registration of new mappers is moved to the DependencyInjectionInstaller file presented below. As you can see, currently, we have only one mapper registered for the CustomerCreatedDomainEvent. To add another mapper, simply register it here. This approach supports the Open/Closed Principle.
builder.Services.AddSingleton<EventMapperFactory>(provider =>
{
var mappers = new Dictionary<Type, IEventMapper>
{
{ typeof(CustomerCreatedDomainEvent), provider.GetRequiredService<CustomerCreatedEventMapper>() },
};
return new EventMapperFactory(mappers);
});
The Strategy pattern is used to obtain the appropriate mapper for mapping Domain Events to Integration Events. Each mapper must implement the IEventMapper interface, which allows you to dynamically apply the correct mapper at runtime.
In the project, tests were implemented for the domain and application layers.
Test Explorer
Thanks to separating domain logic from other layers, we are able to easily test our code. Below is a unit test responsible for creating a customer.
public class CustomerTests
{
[Theory]
[InlineData("incomplete-email@")]
[InlineData("sample.email")]
internal void Should_Throw_Invalid_Email_Domain_Exception_For_Invalid_Email(string invalidEmail)
{
Assert.Throws<InvalidEmailDomainException>(() =>
{
new Email(invalidEmail);
});
}
[Fact]
internal void Should_Create_Customer_For_Valid_Input_Data()
{
// Arrange
var customerId = new CustomerId(Guid.NewGuid());
var fullName = new FullName("Mikolaj Jankowski");
var age = new Age(DateTime.UtcNow.AddYears(-20));
var email = new Email("my-email@yahoo.com");
var address = new Address("Fifth Avenue", "10A", "1", "USA", "10037");
// Act
var customer = CA.And.DDD.Template.Domain.Customers.Customer.CreateCustomer(
customerId,
fullName,
age,
email,
address);
// Assert
var domainEvents = customer.DomainEvents;
Assert.NotNull(domainEvents);
Assert.Single(domainEvents);
var domainEvent = domainEvents.FirstOrDefault();
Assert.NotNull(domainEvent);
Assert.IsType<CustomerCreatedDomainEvent>(domainEvent);
}
}
Testing the application layer essentially comes down to testing handlers. Below are selected implemented test cases.
[Fact]
public async Task Should_Change_Email_When_Customer_Exists()
{
// Arrange
var oldEmail = "old@email.com";
var newEmail = "new@email.com";
var customer = CA.And.DDD.Template.Domain.Customers.Customer.CreateCustomer(
new CustomerId(Guid.NewGuid()),
new FullName("Mikolaj"),
new Age(DateTime.Now.AddYears(-30)),
new Email("email@email.com"),
new Address("Fifth Avenue", "10A", "1", "PL", "10037"));
_customerRepositoryMock.Setup(repo => repo.GetAsync(oldEmail, default))
.ReturnsAsync(customer);
var command = new ChangeEmailCommand(oldEmail, newEmail);
var consumeContextMock = Mock.Of<ConsumeContext<ChangeEmailCommand>>(c => c.Message == command);
// Act
await _handler.Consume(consumeContextMock);
// Assert
Assert.Equal(newEmail, customer.Email.Value);
_customerRepositoryMock.Verify(repo => repo.GetAsync(It.IsAny<string>(), default), Times.Exactly(1));
}
[Fact]
public async Task Should_Throw_CustomerNotFoundApplicationException_When_Customer_Does_Not_Exist()
{
// Arrange
var oldEmail = "nonexistent@example.com";
var newEmail = "new@example.com";
_customerRepositoryMock.Setup(repo => repo.GetAsync(oldEmail, default)).ReturnsAsync((Customer?)null);
var command = new ChangeEmailCommand(oldEmail, newEmail);
var consumeContextMock = Mock.Of<ConsumeContext<ChangeEmailCommand>>(c => c.Message == command);
// Act & Assert
var exception = await Assert.ThrowsAsync<CustomerNotFoundApplicationException>(() => _handler.Consume(consumeContextMock));
_customerRepositoryMock.Verify(repo => repo.GetAsync(It.IsAny<string>(), default), Times.Exactly(1));
}