DDD.NETCore
1.0.0-alpha.7
The framework has been renamed to 'OpenDDD.NET'.
Please use that nuget in your projects instead.
See the version list below for details.
dotnet add package DDD.NETCore --version 1.0.0-alpha.7
NuGet\Install-Package DDD.NETCore -Version 1.0.0-alpha.7
<PackageReference Include="DDD.NETCore" Version="1.0.0-alpha.7" />
paket add DDD.NETCore --version 1.0.0-alpha.7
#r "nuget: DDD.NETCore, 1.0.0-alpha.7"
// Install DDD.NETCore as a Cake Addin #addin nuget:?package=DDD.NETCore&version=1.0.0-alpha.7&prerelease // Install DDD.NETCore as a Cake Tool #tool nuget:?package=DDD.NETCore&version=1.0.0-alpha.7&prerelease
DDD.NETCore
This is a framework for domain-driven design (DDD) development with C# and .NET Core.
Star and/or follow the project to don't miss notifications of upcoming releases.
Goal
The goal with this project is to be able to do DDD with .NET Core.
We couldn't find an existing framework with the features that we required. We wanted to be able to autonomously refine and implement the domain model, without worrying about dependencies on frontend teams and other third-party clients.
Another requirement was that it should be possible to build "near-infinitely scalable" applications, based on the Entity pattern.
We also really like the hexagonal architecture pattern and how a software system becomes easier to understand when it's used.
And so this framework was born.
Key Features
- Domain model versioning.
- Fully autonomous development.
- Auto-generated API documentation.
- Backwards-compatible API support.
- On-the-fly migration.
- Recommended workflow.
Design Patterns
The framework is based on the following design patterns:
- Domain-Driven Design
- Hexagonal Architecture
- Event-Carried State Transfer
- Near-infinite Scalability
- xUnit
- Expand and Contract
- Env files
Credits to Eric Evans for his seminal book on DDD and Vaughn Vernon for his reference implementation of DDD in Java.
Supported versions
- .NET 7 (not tested)
- .NET 6.0
- .NET 5.0 (not tested)
- .NET Core 3.1
Installation
Install-Package DDD.NETCore
Example
These files below are from the WeatherForecast sample project.
// Program.cs
void ConfigureServices(WebApplicationBuilder builder)
{
var services = builder.Services;
// DDD.NETCore
services.AddAccessControl(settings);
services.AddMonitoring(settings);
services.AddPersistence(settings);
services.AddPubSub(settings);
// App
AddDomainServices(services);
AddApplicationService(services);
AddSecondaryAdapters(services);
AddPrimaryAdapters(services);
}
# code removed in favour of brevity ...
services.AddAction<PredictWeatherAction, PredictWeatherCommand>();
services.AddListener<WeatherPredictedListener>();
services.AddRepository<IForecastRepository, PostgresForecastRepository>();
# code removed in favour of brevity ...
// Forecast.cs
namespace Domain.Model.Forecast
{
public class Forecast : Aggregate, IAggregate, IEquatable<Forecast>
{
public ForecastId ForecastId { get; set; }
EntityId IAggregate.Id => ForecastId;
public DateTime Date { get; set; }
public int TemperatureC { get; set; }
public int TemperatureF => 32 + (int)(TemperatureC / 0.5556);
public string Summary { get; set; }
// Public
public static async Task<Forecast> PredictTomorrow(
ForecastId forecastId,
ActionId actionId,
IDomainPublisher domainPublisher,
IInterchangePublisher interchangePublisher,
IIcForecastTranslator icForecastTranslator)
{
var forecast =
new Forecast()
{
DomainModelVersion = Domain.Model.DomainModelVersion.Latest(),
ForecastId = forecastId,
Date = DateTime.Now.AddDays(1),
TemperatureC = Random.Shared.Next(-20, 55),
Summary = Summaries[Random.Shared.Next(Summaries.Length)]
};
forecast.Validate();
await domainPublisher.PublishAsync(new WeatherPredicted(forecast, actionId));
await interchangePublisher.PublishAsync(new IcWeatherPredicted(icForecastTranslator.To(forecast), actionId));
return forecast;
}
# code removed in favour of brevity ...
}
}
# env.local.vs
# Logging
CFG_LOGGING_LEVEL_DOTNET=Information
CFG_LOGGING_LEVEL=Debug
# General
CFG_GENERAL_CONTEXT=Weather
# Auth
CFG_AUTH_ENABLED=false
CFG_AUTH_RBAC_PROVIDER=PowerIAM
CFG_AUTH_RBAC_EXTERNAL_REALM_ID=some-external-id
CFG_AUTH_JWT_TOKEN_PRIVATE_KEY=some-fake-private-key
CFG_AUTH_JWT_TOKEN_NAME=Authorization
CFG_AUTH_JWT_TOKEN_LOCATION=header
CFG_AUTH_JWT_TOKEN_SCHEME=Bearer
# Http Adapter
CFG_HTTP_URLS=http://localhost:9000
CFG_HTTP_CORS_ALLOWED_ORIGINS=https://localhost:5052,http://localhost:5051
CFG_HTTP_DOCS_DEFINITIONS=Public,Public,
CFG_HTTP_DOCS_ENABLED=true
CFG_HTTP_DOCS_HTTP_ENABLED=true
CFG_HTTP_DOCS_HTTPS_ENABLED=false
CFG_HTTP_DOCS_HOSTNAME=localhost:5051
CFG_HTTP_DOCS_AUTH_EXTRA_TOKENS=
CFG_HTTP_DOCS_TITLE=Weather API
# Persistence
CFG_PERSISTENCE_PROVIDER=Postgres
CFG_PERSISTENCE_POOLING_ENABLED=true
CFG_PERSISTENCE_POOLING_MIN_SIZE=0
CFG_PERSISTENCE_POOLING_MAX_SIZE=100
# Postgres
CFG_POSTGRES_CONN_STR="Host=localhost:9092;Username=net60;Password=net60;Database=net60"
# PubSub
CFG_PUBSUB_PROVIDER=Rabbit
CFG_PUBSUB_MAX_DELIVERY_RETRIES=3
CFG_PUBSUB_PUBLISHER_ENABLED=true
# Monitoring
CFG_MONITORING_PROVIDER=AppInsights
# PowerIAM
CFG_POWERIAM_URL=http://localhost:9000
# Service Bus
CFG_SERVICEBUS_CONN_STR=
CFG_SERVICEBUS_SUB_NAME=
# Rabbit
CFG_RABBIT_HOST=localhost
CFG_RABBIT_PORT=5672
CFG_RABBIT_USERNAME=guest
CFG_RABBIT_PASSWORD=guest
# Email
CFG_EMAIL_ENABLED=true
CFG_EMAIL_PROVIDER=smtp
CFG_EMAIL_SMTP_HOST=localhost
CFG_EMAIL_SMTP_PORT=1025
Documentation:
Documentation is coming in v1.0.0 rc.
Semantic versioning
Your primary http adapters effectively provides an "API" for clients of the application.
DDD.NETCore requires you to use SemVer2.0 policy for versioning. This means that backwards compatible changes increments the patch- and minor versions. Backwards incompatible changes thus increments the major version.
See table below for when to increment the numbers:
Code Status | Stage | Rule | Example Version |
---|---|---|---|
First release | New product | Start with 1.0.0 | 1.0.0 |
Backward-compatible bug fixes | Patch release | Increment the third digit | 1.0.1 |
Backward compatible new features | Minor release | Increment the middle digit and reset last digit to zero | 1.1.0 |
Changes that break backward compatibility | Major release | Increment the first digit and reset middle and last digits to zero | 2.0.0 |
Contribution:
If you want to contribute to the code base, create a pull request on the develop branch.
Roadmap v1.0.0:
- GitHub README
- NuGet README
- Visual Studio Project Template .NET 7
- Visual Studio Project Template .NET 6.0
- Visual Studio Project Template .NET 5.0
- Visual Studio Project Template .NET Core 3.1
- Start Context
- Stop Context
- Control
- On-the-fly aggregate migration
- Auto-code Generation from domain.yml File
- Postgres Dead Letter Queue
- Memory Dead Letter Queue
- Dead Letter Queue
- Handle Poisonous Messages
- Re-publish Failed Events
- Postgres Outbox
- Memory Outbox
- Outbox
- Domain Event Publishing
- Integration Event Publishing
- Rabbit Event Adapter
- Memory Event Adapter
- PubSub
- Auth Domain Service
- Auth
- Aggregate
- Entity
- Value Object
- Domain Event
- Integration Event
- Repository
- Building Blocks
- Domain Service
- Infrastructure Service
- Application Service
- Auto-Generated Swagger HTTP Adapter Documentation
- HTTP Adapter
- Email Adapter
- Persistence Service
- Postgres Repository
- Memory Repository
Roadmap Future:
- Full Sample Project
- Quickstart Guide
- Documentation
- Azure Monitoring Adapter.
- Monitoring
- Task: Migrate all aggregate roots
- Tasks
- All-aggregates Migration Task
- Periodic Jobs
- Interval Jobs
- One-off Jobs
- Jobs
- Merge old API versions into current swagger json files.
- CLI Operation: Create action/aggregate/event/...
- CLI Operation: Migrate
- CLI Operation: Create Migration
- CLI Operation: Increment Domain Model Version
- Test Framework
- Tests
- Admin Dashboard
- Admin Tool: Inspect Dead Event
- Admin Tool: Republish Dead Event
- Administration
Release Notes:
1.0.0-alpha.7 - 2023-01-01
- Add credentials support to smtp adapter.
- Use api version 2.0.0 in poweriam adapter.
1.0.0-alpha.6 - 2023-01-01
- Add base class for domain services.
- Use new permissions string format: "<domain>:<permission>". (breaking)
1.0.0-alpha.5 - 2022-12-26
- Refactor to follow semver2.0 strictly in http adapter. (breaking)
- Add support for configuring persistence pooling.
- Add html support to email port. (breaking)
- Fix memory leak where db connections weren't closed.
1.0.0-alpha.4 - 2022-12-10
- Add configuration setting for which server urls to listen to. (breaking)
- Fix concurrency issues with memory repositories.
- Add support for IAM ports.
- Add 'PowerIAM' adapter.
- Add RBAC auth settings. (breaking)
- Add a base 'Migrator' class. (breaking)
1.0.0-alpha.3 - 2022-11-20
- Refactor JwtToken and add IdToken. (breaking)
- Add more tasks to code generation tool.
- Add support for http put methods to code generation tool.
- Add some missing repository method implementations.
- Add GetAsync(IEnumerable<...> ...) to repositories.
- Add convenience methods to ApplicationExtensions.
- Return 400 http status code on domain- and invariant exceptions in primary http adapter.
1.0.0-alpha.2 - 2022-10-09
- Make the hexagonal architecture more represented in the namespaces.
1.0.0-alpha.1 - 2022-10-02
This is the first (alpha) release of the framework. Please try it out and submit tickets or otherwise reach out if you find any issues or have any questions.
0.9.0-alpha7 - 2022-07-31
First alpha release on nuget.org.
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 | netcoreapp3.1 is compatible. |
-
.NETCoreApp 3.1
- Azure.Messaging.ServiceBus (>= 7.5.1)
- CompareNETObjects (>= 4.77.0)
- dotenv.net (>= 3.1.1)
- Microsoft.ApplicationInsights (>= 2.20.0)
- Microsoft.AspNet.WebApi.WebHost (>= 5.2.8)
- Microsoft.AspNetCore.TestHost (>= 3.1.26)
- Microsoft.Extensions.Http (>= 6.0.0)
- Microsoft.Extensions.Logging (>= 6.0.0)
- Microsoft.Extensions.Logging.ApplicationInsights (>= 2.20.0)
- Microsoft.Extensions.Logging.Console (>= 6.0.0)
- Microsoft.Extensions.Options.ConfigurationExtensions (>= 6.0.0)
- Newtonsoft.Json (>= 13.0.1)
- Npgsql (>= 6.0.4)
- NSwag.AspNetCore (>= 13.15.10)
- PowerIAM (>= 2.0.0)
- RabbitMQ.Client (>= 6.2.2)
- System.IdentityModel.Tokens.Jwt (>= 6.17.0)
- WireMock.Net (>= 1.5.0)
- xunit (>= 2.4.1)
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 | |
---|---|---|---|
1.0.0-alpha.10 | 124 | 4/24/2023 | |
1.0.0-alpha.9 | 109 | 4/19/2023 | |
1.0.0-alpha.8 | 116 | 4/11/2023 | |
1.0.0-alpha.7 | 143 | 1/1/2023 | |
1.0.0-alpha.6 | 132 | 1/1/2023 | |
1.0.0-alpha.5 | 140 | 12/26/2022 | |
1.0.0-alpha.4 | 132 | 12/10/2022 | |
1.0.0-alpha.3 | 128 | 11/20/2022 | |
1.0.0-alpha.2 | 132 | 10/5/2022 | |
1.0.0-alpha.1 | 158 | 10/2/2022 | |
0.9.1-alpha.3 | 223 | 8/17/2022 | |
0.9.0-alpha8 | 223 | 8/1/2022 | |
0.9.0-alpha7 | 219 | 7/31/2022 | |
0.9.0-alpha10 | 212 | 8/14/2022 |