skills/09-domain-events-generator/SKILL.md
Generates Domain Events and their handlers following DDD patterns. Implements event raising in entities, MediatR notification handlers, and the Outbox pattern for reliable event processing.
npx skillsauth add ronnythedev/dotnet-clean-architecture-skills domain-events-generatorInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Domain Events capture something significant that happened in the domain:
| Component | Purpose | Location |
|-----------|---------|----------|
| IDomainEvent | Marker interface | Domain/Abstractions |
| {Entity}{Action}DomainEvent | Event record | Domain/{Aggregate}/Events |
| {Event}DomainEventHandler | Event handler | Application/{Feature} |
| OutboxMessage | Persisted event | Infrastructure/Outbox |
/Domain/
├── Abstractions/
│ └── IDomainEvent.cs
└── {Aggregate}/
└── Events/
├── {Entity}CreatedDomainEvent.cs
├── {Entity}UpdatedDomainEvent.cs
└── ...
/Application/
└── {Feature}/
└── EventHandlers/
├── {Event}Handler.cs
└── ...
/Infrastructure/
└── Outbox/
├── OutboxMessage.cs
├── OutboxMessageConfiguration.cs
└── ProcessOutboxMessagesJob.cs
// src/{name}.domain/Abstractions/IDomainEvent.cs
using MediatR;
namespace {name}.domain.abstractions;
/// <summary>
/// Marker interface for domain events.
/// Domain events represent something significant that happened in the domain.
/// </summary>
public interface IDomainEvent : INotification
{
/// <summary>
/// Unique identifier for this event instance
/// </summary>
Guid Id { get; }
/// <summary>
/// When the event occurred
/// </summary>
DateTime OccurredOnUtc { get; }
}
// src/{name}.domain/Abstractions/DomainEvent.cs
namespace {name}.domain.abstractions;
/// <summary>
/// Base record for domain events with common properties
/// </summary>
public abstract record DomainEvent : IDomainEvent
{
public Guid Id { get; init; } = Guid.NewGuid();
public DateTime OccurredOnUtc { get; init; } = DateTime.UtcNow;
}
// src/{name}.domain/{Aggregate}/Events/{Entity}CreatedDomainEvent.cs
using {name}.domain.abstractions;
namespace {name}.domain.{aggregate}.events;
/// <summary>
/// Raised when a new {Entity} is created
/// </summary>
public sealed record {Entity}CreatedDomainEvent(
Guid {Entity}Id) : DomainEvent;
// src/{name}.domain/{Aggregate}/Events/{Entity}UpdatedDomainEvent.cs
/// <summary>
/// Raised when a {Entity} is updated
/// </summary>
public sealed record {Entity}UpdatedDomainEvent(
Guid {Entity}Id,
string PropertyName,
string? OldValue,
string? NewValue) : DomainEvent;
// src/{name}.domain/{Aggregate}/Events/{Entity}DeactivatedDomainEvent.cs
/// <summary>
/// Raised when a {Entity} is deactivated
/// </summary>
public sealed record {Entity}DeactivatedDomainEvent(
Guid {Entity}Id,
string Reason) : DomainEvent;
// src/{name}.domain/{Aggregate}/Events/{Entity}DeletedDomainEvent.cs
/// <summary>
/// Raised when a {Entity} is deleted
/// </summary>
public sealed record {Entity}DeletedDomainEvent(
Guid {Entity}Id) : DomainEvent;
// src/{name}.domain/Users/Events/UserRegisteredDomainEvent.cs
using {name}.domain.abstractions;
namespace {name}.domain.users.events;
/// <summary>
/// Raised when a new user registers
/// </summary>
public sealed record UserRegisteredDomainEvent : DomainEvent
{
public Guid UserId { get; init; }
public string Email { get; init; } = string.Empty;
public string Name { get; init; } = string.Empty;
public Guid OrganizationId { get; init; }
public UserRegisteredDomainEvent(
Guid userId,
string email,
string name,
Guid organizationId)
{
UserId = userId;
Email = email;
Name = name;
OrganizationId = organizationId;
}
}
// src/{name}.domain/Assessments/Events/AssessmentCompletedDomainEvent.cs
/// <summary>
/// Raised when a user completes an assessment
/// </summary>
public sealed record AssessmentCompletedDomainEvent : DomainEvent
{
public Guid AssessmentId { get; init; }
public Guid UserId { get; init; }
public Guid OrganizationId { get; init; }
public string AssessmentType { get; init; } = string.Empty;
public decimal Score { get; init; }
public DateTime CompletedAt { get; init; }
public AssessmentCompletedDomainEvent(
Guid assessmentId,
Guid userId,
Guid organizationId,
string assessmentType,
decimal score,
DateTime completedAt)
{
AssessmentId = assessmentId;
UserId = userId;
OrganizationId = organizationId;
AssessmentType = assessmentType;
Score = score;
CompletedAt = completedAt;
}
}
// src/{name}.domain/{Aggregate}/{Entity}.cs
using {name}.domain.abstractions;
using {name}.domain.{aggregate}.events;
namespace {name}.domain.{aggregate};
public sealed class {Entity} : Entity
{
// ... properties
private {Entity}(
Guid id,
string name,
Guid organizationId,
DateTime createdAt)
: base(id)
{
Name = name;
OrganizationId = organizationId;
CreatedAt = createdAt;
}
/// <summary>
/// Factory method - raises Created event
/// </summary>
public static Result<{Entity}> Create(
string name,
Guid organizationId,
DateTime createdAt)
{
// Validation...
var {entity} = new {Entity}(
Guid.NewGuid(),
name,
organizationId,
createdAt);
// Raise domain event
{entity}.RaiseDomainEvent(new {Entity}CreatedDomainEvent({entity}.Id));
return {entity};
}
/// <summary>
/// Update method - raises Updated event
/// </summary>
public Result Update(string name, DateTime updatedAt)
{
if (string.IsNullOrWhiteSpace(name))
{
return Result.Failure({Entity}Errors.NameRequired);
}
var oldName = Name;
Name = name;
UpdatedAt = updatedAt;
// Raise domain event with change details
RaiseDomainEvent(new {Entity}UpdatedDomainEvent(
Id,
nameof(Name),
oldName,
name));
return Result.Success();
}
/// <summary>
/// Deactivate method - raises Deactivated event
/// </summary>
public Result Deactivate(string reason, DateTime deactivatedAt)
{
if (!IsActive)
{
return Result.Failure({Entity}Errors.AlreadyDeactivated);
}
IsActive = false;
UpdatedAt = deactivatedAt;
RaiseDomainEvent(new {Entity}DeactivatedDomainEvent(Id, reason));
return Result.Success();
}
}
// src/{name}.application/{Feature}/EventHandlers/{Entity}CreatedDomainEventHandler.cs
using MediatR;
using Microsoft.Extensions.Logging;
using {name}.domain.{aggregate}.events;
namespace {name}.application.{feature}.eventhandlers;
/// <summary>
/// Handles {Entity}CreatedDomainEvent
/// </summary>
internal sealed class {Entity}CreatedDomainEventHandler
: INotificationHandler<{Entity}CreatedDomainEvent>
{
private readonly ILogger<{Entity}CreatedDomainEventHandler> _logger;
public {Entity}CreatedDomainEventHandler(
ILogger<{Entity}CreatedDomainEventHandler> logger)
{
_logger = logger;
}
public Task Handle(
{Entity}CreatedDomainEvent notification,
CancellationToken cancellationToken)
{
_logger.LogInformation(
"{Entity} created: {EntityId} at {OccurredOn}",
notification.{Entity}Id,
notification.OccurredOnUtc);
// Add any side effects here:
// - Send notifications
// - Update read models
// - Trigger workflows
// - Publish to external systems
return Task.CompletedTask;
}
}
// src/{name}.application/Users/EventHandlers/UserRegisteredDomainEventHandler.cs
using MediatR;
using Microsoft.Extensions.Logging;
using {name}.application.abstractions.email;
using {name}.domain.users.events;
namespace {name}.application.users.eventhandlers;
/// <summary>
/// Sends welcome email when user registers
/// </summary>
internal sealed class UserRegisteredSendWelcomeEmailHandler
: INotificationHandler<UserRegisteredDomainEvent>
{
private readonly IEmailService _emailService;
private readonly ILogger<UserRegisteredSendWelcomeEmailHandler> _logger;
public UserRegisteredSendWelcomeEmailHandler(
IEmailService emailService,
ILogger<UserRegisteredSendWelcomeEmailHandler> logger)
{
_emailService = emailService;
_logger = logger;
}
public async Task Handle(
UserRegisteredDomainEvent notification,
CancellationToken cancellationToken)
{
_logger.LogInformation(
"Sending welcome email to user {UserId}",
notification.UserId);
await _emailService.SendWelcomeEmailAsync(
notification.Email,
notification.Name,
cancellationToken);
}
}
/// <summary>
/// Creates default settings when user registers
/// </summary>
internal sealed class UserRegisteredCreateDefaultSettingsHandler
: INotificationHandler<UserRegisteredDomainEvent>
{
private readonly IUserSettingsRepository _settingsRepository;
private readonly IUnitOfWork _unitOfWork;
public UserRegisteredCreateDefaultSettingsHandler(
IUserSettingsRepository settingsRepository,
IUnitOfWork unitOfWork)
{
_settingsRepository = settingsRepository;
_unitOfWork = unitOfWork;
}
public async Task Handle(
UserRegisteredDomainEvent notification,
CancellationToken cancellationToken)
{
var settings = UserSettings.CreateDefault(notification.UserId);
_settingsRepository.Add(settings);
await _unitOfWork.SaveChangesAsync(cancellationToken);
}
}
// src/{name}.infrastructure/Outbox/OutboxMessage.cs
namespace {name}.infrastructure.outbox;
/// <summary>
/// Represents a domain event stored for reliable delivery
/// </summary>
public sealed class OutboxMessage
{
public Guid Id { get; set; }
public string Type { get; set; } = string.Empty;
public string Content { get; set; } = string.Empty;
public DateTime OccurredOnUtc { get; set; }
public DateTime? ProcessedOnUtc { get; set; }
public string? Error { get; set; }
}
// src/{name}.infrastructure/Configurations/OutboxMessageConfiguration.cs
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
using {name}.infrastructure.outbox;
namespace {name}.infrastructure.configurations;
internal sealed class OutboxMessageConfiguration
: IEntityTypeConfiguration<OutboxMessage>
{
public void Configure(EntityTypeBuilder<OutboxMessage> builder)
{
builder.ToTable("outbox_message");
builder.HasKey(o => o.Id);
builder.Property(o => o.Type)
.HasMaxLength(500)
.IsRequired();
builder.Property(o => o.Content)
.HasColumnType("jsonb")
.IsRequired();
builder.Property(o => o.OccurredOnUtc)
.IsRequired();
builder.Property(o => o.ProcessedOnUtc);
builder.Property(o => o.Error)
.HasColumnType("text");
// Index for efficient querying of unprocessed messages
builder.HasIndex(o => o.ProcessedOnUtc)
.HasFilter("processed_on_utc IS NULL");
}
}
// src/{name}.infrastructure/ApplicationDbContext.cs
using System.Text.Json;
using Microsoft.EntityFrameworkCore;
using {name}.domain.abstractions;
using {name}.infrastructure.outbox;
namespace {name}.infrastructure;
public sealed class ApplicationDbContext : DbContext, IUnitOfWork
{
private static readonly JsonSerializerOptions JsonOptions = new()
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase
};
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
: base(options)
{
}
public DbSet<OutboxMessage> OutboxMessages => Set<OutboxMessage>();
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.ApplyConfigurationsFromAssembly(typeof(ApplicationDbContext).Assembly);
base.OnModelCreating(modelBuilder);
}
public override async Task<int> SaveChangesAsync(CancellationToken cancellationToken = default)
{
// Convert domain events to outbox messages before saving
AddDomainEventsAsOutboxMessages();
return await base.SaveChangesAsync(cancellationToken);
}
private void AddDomainEventsAsOutboxMessages()
{
var entities = ChangeTracker
.Entries<Entity>()
.Where(e => e.Entity.GetDomainEvents().Any())
.Select(e => e.Entity)
.ToList();
var domainEvents = entities
.SelectMany(e => e.GetDomainEvents())
.ToList();
foreach (var domainEvent in domainEvents)
{
var outboxMessage = new OutboxMessage
{
Id = Guid.NewGuid(),
Type = domainEvent.GetType().AssemblyQualifiedName!,
Content = JsonSerializer.Serialize(
domainEvent,
domainEvent.GetType(),
JsonOptions),
OccurredOnUtc = DateTime.UtcNow
};
OutboxMessages.Add(outboxMessage);
}
foreach (var entity in entities)
{
entity.ClearDomainEvents();
}
}
}
// src/{name}.infrastructure/Outbox/ProcessOutboxMessagesJob.cs
using System.Text.Json;
using MediatR;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Logging;
using Quartz;
using {name}.domain.abstractions;
namespace {name}.infrastructure.outbox;
[DisallowConcurrentExecution]
internal sealed class ProcessOutboxMessagesJob : IJob
{
private static readonly JsonSerializerOptions JsonOptions = new()
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase
};
private readonly ApplicationDbContext _dbContext;
private readonly IPublisher _publisher;
private readonly ILogger<ProcessOutboxMessagesJob> _logger;
public ProcessOutboxMessagesJob(
ApplicationDbContext dbContext,
IPublisher publisher,
ILogger<ProcessOutboxMessagesJob> logger)
{
_dbContext = dbContext;
_publisher = publisher;
_logger = logger;
}
public async Task Execute(IJobExecutionContext context)
{
_logger.LogInformation("Processing outbox messages...");
var messages = await _dbContext
.OutboxMessages
.Where(m => m.ProcessedOnUtc == null)
.OrderBy(m => m.OccurredOnUtc)
.Take(20)
.ToListAsync(context.CancellationToken);
foreach (var message in messages)
{
try
{
var type = Type.GetType(message.Type);
if (type is null)
{
_logger.LogWarning(
"Could not resolve type {Type} for outbox message {MessageId}",
message.Type,
message.Id);
message.Error = $"Could not resolve type: {message.Type}";
message.ProcessedOnUtc = DateTime.UtcNow;
continue;
}
var domainEvent = JsonSerializer.Deserialize(
message.Content,
type,
JsonOptions) as IDomainEvent;
if (domainEvent is null)
{
_logger.LogWarning(
"Could not deserialize outbox message {MessageId}",
message.Id);
message.Error = "Could not deserialize message content";
message.ProcessedOnUtc = DateTime.UtcNow;
continue;
}
await _publisher.Publish(domainEvent, context.CancellationToken);
message.ProcessedOnUtc = DateTime.UtcNow;
_logger.LogInformation(
"Processed outbox message {MessageId} of type {Type}",
message.Id,
message.Type);
}
catch (Exception ex)
{
_logger.LogError(
ex,
"Error processing outbox message {MessageId}",
message.Id);
message.Error = ex.ToString();
}
}
await _dbContext.SaveChangesAsync(context.CancellationToken);
}
}
// src/{name}.infrastructure/DependencyInjection.cs
private static void AddBackgroundJobs(IServiceCollection services)
{
services.AddQuartz(configure =>
{
var jobKey = new JobKey(nameof(ProcessOutboxMessagesJob));
configure
.AddJob<ProcessOutboxMessagesJob>(jobKey)
.AddTrigger(trigger =>
trigger
.ForJob(jobKey)
.WithSimpleSchedule(schedule =>
schedule.WithIntervalInSeconds(10).RepeatForever()));
});
services.AddQuartzHostedService();
}
| Event Type | Naming Pattern | Example |
|------------|----------------|---------|
| Created | {Entity}CreatedDomainEvent | UserCreatedDomainEvent |
| Updated | {Entity}UpdatedDomainEvent | UserUpdatedDomainEvent |
| Deleted | {Entity}DeletedDomainEvent | UserDeletedDomainEvent |
| Status Change | {Entity}{Status}DomainEvent | OrderShippedDomainEvent |
| Action | {Entity}{Action}DomainEvent | PaymentProcessedDomainEvent |
record types// ❌ WRONG: Events with behavior
public record UserCreatedEvent
{
public void SendEmail() { } // Events should be data only!
}
// ✅ CORRECT: Pure data event
public record UserCreatedDomainEvent(Guid UserId, string Email) : DomainEvent;
// ❌ WRONG: Raising events in handler
internal sealed class CreateUserHandler : ICommandHandler<CreateUser, Guid>
{
public async Task<Result<Guid>> Handle(...)
{
// Don't raise events here!
await _publisher.Publish(new UserCreatedEvent(user.Id));
}
}
// ✅ CORRECT: Raise events in entity
public static Result<User> Create(...)
{
var user = new User(...);
user.RaiseDomainEvent(new UserCreatedDomainEvent(user.Id, user.Email));
return user;
}
// ❌ WRONG: Handler depends on other handler's result
internal sealed class Handler1 : INotificationHandler<Event>
{
public async Task Handle(Event e, CancellationToken ct)
{
// Waiting for Handler2 to complete - bad!
while (!await _service.IsHandler2Complete()) { }
}
}
// ✅ CORRECT: Handlers are independent
internal sealed class Handler1 : INotificationHandler<Event>
{
public Task Handle(Event e, CancellationToken ct)
{
// Does its own work, doesn't care about other handlers
return DoWork(e, ct);
}
}
domain-entity-generator - Entities that raise eventspipeline-behaviors - Event publishing behaviordotnet-clean-architecture - Infrastructure layer setupcqrs-command-generator - Commands that trigger eventstools
Implements the Options pattern for strongly-typed configuration in .NET. Covers IOptions<T>, IOptionsSnapshot<T>, and IOptionsMonitor<T> with validation and reload support.
tools
SQL Server database design best practices, naming conventions, indexing strategies, and performance optimization for .NET applications using Microsoft.Data.SqlClient and EF Core.
data-ai
PostgreSQL database design best practices, naming conventions, indexing strategies, and performance optimization for .NET applications using Npgsql and EF Core.
development
Implements ASP.NET Core rate limiting middleware for API protection. Covers fixed window, sliding window, token bucket, and concurrency limiters with custom policies.