---
name: csharp-wolverinefx
description: Build .NET applications with WolverineFX for messaging, HTTP services, and event sourcing. Use when implementing command handlers, message handlers, HTTP endpoints with WolverineFx.HTTP, transactional outbox patterns, event sourcing with Marten, CQRS architectures, cascading messages, batch message processing, or configuring transports like RabbitMQ, Azure Service Bus, or Amazon SQS.
---
# WolverineFX for .NET
## When to Use This Skill
Use this skill when:
- Building message handlers or command handlers with Wolverine
- Creating HTTP endpoints with WolverineFx.HTTP (alternative to Minimal API/MVC)
- Implementing event sourcing with Marten and Wolverine
- Setting up transactional outbox pattern for reliable messaging
- Configuring message transports (RabbitMQ, Azure Service Bus, Amazon SQS, TCP)
- Implementing CQRS with event sourcing
- Processing messages in batches
- Using cascading messages for testable, pure function handlers
- Configuring error handling and retry policies
- Pre-generating code for optimized cold starts
## Related Skills
- **`efcore-patterns`** - Entity Framework Core patterns for data access
- **`csharp-coding-standards`** - Modern C# patterns (records, pattern matching)
- **`http-client-resilience`** - Polly resilience patterns (complementary)
- **`background-services`** - Hosted services and background job patterns
- **`aspire-configuration`** - .NET Aspire orchestration
## Core Principles
1. **Low Ceremony Code** - Pure functions, method injection, minimal boilerplate
2. **Cascading Messages** - Return messages from handlers instead of injecting IMessageBus
3. **Transactional Outbox** - Guaranteed message delivery with database transactions
4. **Code Generation** - Runtime or pre-generated code for optimal performance
5. **Vertical Slice Architecture** - Organize code by feature, not technical layers
6. **Pure Functions for Business Logic** - Isolate infrastructure from business logic
## Required NuGet Packages
### Core Messaging
```xml
```
### Persistence Integration
```xml
```
### Transports
```xml
```
## Basic Setup
### Program.cs (ASP.NET Core)
```csharp
using JasperFx;
using Wolverine;
using Wolverine.Http;
var builder = WebApplication.CreateBuilder(args);
builder.Host.UseWolverine(opts =>
{
opts.Policies.AutoApplyTransactions();
opts.Policies.UseDurableLocalQueues();
});
builder.Services.AddWolverineHttp();
var app = builder.Build();
app.MapWolverineEndpoints();
return await app.RunJasperFxCommands(args);
```
## Message Handlers
### Simple Message Handler
```csharp
public record DebitAccount(long AccountId, decimal Amount);
public static class DebitAccountHandler
{
public static void Handle(DebitAccount command, IAccountRepository repository)
{
repository.Debit(command.AccountId, command.Amount);
}
}
```
### Handler with Cascading Messages
```csharp
public record CreateOrder(Guid OrderId, string[] Items);
public record OrderCreated(Guid OrderId);
public static class CreateOrderHandler
{
public static (OrderCreated, ShipOrder) Handle(
CreateOrder command,
IDocumentSession session)
{
var order = new Order { Id = command.OrderId, Items = command.Items };
session.Store(order);
return (
new OrderCreated(command.OrderId),
new ShipOrder(command.OrderId)
);
}
}
```
### Using OutgoingMessages for Multiple Messages
```csharp
public static OutgoingMessages Handle(ProcessOrder command)
{
var messages = new OutgoingMessages
{
new OrderProcessed(command.OrderId),
new SendEmail(command.CustomerEmail, "Order processed"),
new UpdateInventory(command.Items)
};
messages.Delay(new CleanupOrder(command.OrderId), 5.Minutes());
return messages;
}
```
## HTTP Endpoints (WolverineFx.HTTP)
### Basic GET Endpoint
```csharp
[WolverineGet("/users/{id}")]
public static Task GetUser(int id, IQuerySession session)
=> session.LoadAsync(id);
```
### POST with Message Publishing
```csharp
[WolverinePost("/orders")]
public static async Task CreateOrder(
CreateOrderRequest request,
IDocumentSession session,
IMessageBus bus)
{
var order = new Order { Id = Guid.NewGuid(), Items = request.Items };
session.Store(order);
await bus.PublishAsync(new OrderCreated(order.Id));
return Results.Created($"/orders/{order.Id}", order);
}
```
### Compound Handler (Load/Validate/Handle)
```csharp
public static class UpdateOrderEndpoint
{
public static async Task<(Order?, IResult)> LoadAsync(
UpdateOrder command,
IDocumentSession session)
{
var order = await session.LoadAsync(command.OrderId);
return order != null
? (order, new WolverineContinue())
: (order, Results.NotFound());
}
[WolverinePut("/orders")]
public static void Handle(UpdateOrder command, Order order, IDocumentSession session)
{
order.Items = command.Items;
session.Store(order);
}
}
```
## Event Sourcing with Marten
### Aggregate Handler Workflow
```csharp
public class Order
{
public Guid Id { get; set; }
public int Version { get; set; }
public Dictionary Items { get; set; } = new();
public DateTimeOffset? Shipped { get; private set; }
public void Apply(ItemReady ready) => Items[ready.Name].Ready = true;
public void Apply(IEvent shipped) => Shipped = shipped.Timestamp;
public bool IsReadyToShip() => Shipped == null && Items.Values.All(x => x.Ready);
}
public record MarkItemReady(Guid OrderId, string ItemName, int Version);
[AggregateHandler]
public static IEnumerable