Beckett 0.14.0

There is a newer version of this package available.
See the version list below for details.
dotnet add package Beckett --version 0.14.0                
NuGet\Install-Package Beckett -Version 0.14.0                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Beckett" Version="0.14.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Beckett --version 0.14.0                
#r "nuget: Beckett, 0.14.0"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Beckett as a Cake Addin
#addin nuget:?package=Beckett&version=0.14.0

// Install Beckett as a Cake Tool
#tool nuget:?package=Beckett&version=0.14.0                

Beckett

Event sourcing is a powerful pattern for building applications but reading and writing events using an event store is only half of the equation. Beckett aims to fill in the gaps:

  • Subscriptions - subscribe to messages and process them in order by stream
    • Projections, read models, event handlers - add asynchronous, event-driven behavior to your applications
    • Horizontal scalability - use auto-scaling to have as many workers as needed processing messages in parallel where the work is distributed automatically across all available nodes without needing to manage the distribution by way of consumer groups or similar mechanisms
    • Retries - built-in retry support for failed messages - since messages are processed in order by stream per subscription, a failed message only blocks a single stream for a subscription at a time and the rest of the streams can continue processing for that subscription
  • Scheduled - schedule messages to be sent at a future time with cancellation support
  • Open Telemetry - built-in support to provide tracing and metrics
  • Dashboard - browse messages, retry failed subscription checkpoints
  • Bring Your Own Event Store - Beckett provides a simple Postgres-based message store or use your own by implementing the IMessageStorage interface

Example

We are building a warehouse management system and we need to allocate inventory to orders. The requirements are that allocation occurs when an item is added to an order:

using Beckett;
using Beckett.Database;

var builder = Host.CreateApplicationBuilder(args);

var connectionString = builder.Configuration.GetConnectionString("InventoryAllocation")!;

//ensure the Beckett database schema is up to date
await BeckettDatabase.Migrate(connectionString);

//configure the data source with support for Beckett
builder.Services.AddNpgsqlDataSource(connectionString, options => options.AddBeckett());

//register the subscription handler in the container
builder.Services.AddTransient<OrderItemAddedHandler>();

//add Beckett support to the host for the InventoryAllocation subscription group
var beckett = builder.AddBeckett(
    options => { options.WithSubscriptionGroup("InventoryAllocation"); }
);

//map message types
beckett.Map<OrderItemAdded>("order_item_added");
beckett.Map<InventoryAllocated>("inventory_allocated");

//add subscription handler
beckett.AddSubscription("order-item-inventory-allocation")
    .Message<OrderItemAdded>()
    .Handler<OrderItemAddedHandler>((handler, message, token) => handler.Handle(message, token));

var host = builder.Build();

host.Run();

public record OrderItemAdded(Guid OrderId, Guid ProductId, int Quantity);

public record InventoryAllocated(Guid ProductId, Guid OrderId, int Quantity);

public class OrderItemAddedHandler(IMessageStore messageStore)
{
    public async Task Handle(OrderItemAdded message, CancellationToken cancellationToken)
    {
        await messageStore.AppendToStream(
            $"inventory-{message.ProductId}",
            ExpectedVersion.Any,
            new InventoryAllocated(message.ProductId, message.OrderId, message.Quantity),
            cancellationToken
        );
    }
}

In this example application we are handling the OrderItemAdded event with the OrderItemAddedHandler class. The host has been configured to use the InventoryAllocation subscription group, and there can be as many instances of this host running as necessary and the work will be divided among them automatically allowing you to take advantage of auto scaling without limits. The handler will receive all OrderItemAdded messages written to the message store since it is subscribed to that type in the AddSubscription call. When a message is received it is dispatched to the handler which then writes an InventoryAllocated event to an Inventory stream to track allocated product inventory.

One of the guiding design principles of Beckett is keeping a minimal footprint - there should be as few references to Beckett-provided types in application code as possible. Subscription handlers are registered as inline delegates that can refer to handler instances that are resolved from the container or static functions. The only type from Beckett used in the application code in this sample is IMessageStore which itself is optional if you're using your own message store.

The call to BeckettDatabase.Migrate in the example is applying any outstanding migrations to the database that are required by Beckett. If you wish to run the migrations separately using Flyway or similar tools then you can use the dump-migrations shell script supplied in the root of the directory to create a single SQL file:

./dump-migrations.sh beckett 001.sql

In this case beckett is the schema you'd like to use in your database for the tables, functions, and types that Beckett uses and 001.sql is the path of the file you'd like to create.

Dashboard

Beckett comes batteries-included with a dashboard to provide visibility into your system while it's running, retry failed checkpoints, and so on:

<img width="1575" alt="Beckett Dashboard" src="https://github.com/user-attachments/assets/0dc5a445-111b-4552-a639-36b37779d094">

Adding the Beckett dashboard to an ASP.NET Core application is simple:

var builder = WebApplication.CreateBuilder(args);

var app = builder.Build();

app.MapBeckettDashboard("/beckett");

app.Run();

In this example, the dashboard will be available at http://localhost:<port>/beckett and can be further configured using standard ASP.NET Core route group configuration options - authorization, etc...

Product 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.  net9.0 is compatible.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (2)

Showing the top 2 NuGet packages that depend on Beckett:

Package Downloads
Beckett.Dashboard

Messaging and event sourcing library

Beckett.Testing

Messaging and event sourcing library

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
0.16.7 0 1/15/2025
0.16.6 0 1/14/2025
0.16.5 90 1/10/2025
0.16.4 83 1/10/2025
0.16.3 197 1/4/2025
0.16.2 105 1/4/2025
0.16.1 187 1/1/2025
0.16.0 112 1/1/2025
0.15.0 151 12/31/2024
0.14.4 89 12/17/2024
0.14.3 94 12/16/2024
0.14.2 92 12/16/2024
0.14.1 95 12/12/2024
0.14.0 96 12/12/2024
0.13.1 956 11/15/2024
0.13.0 105 11/8/2024
0.12.6 464 11/1/2024
0.12.5 137 10/31/2024
0.12.4 190 10/30/2024
0.12.3 125 10/29/2024
0.12.2 139 10/29/2024
0.12.1 707 10/28/2024
0.12.0 118 10/28/2024
0.11.12 142 10/25/2024
0.11.11 215 10/22/2024
0.11.10 342 10/18/2024
0.11.9 150 10/18/2024
0.11.8 164 10/18/2024
0.11.7 149 10/18/2024
0.11.6 169 10/17/2024
0.11.5 107 10/17/2024
0.11.4 103 10/17/2024
0.11.3 103 10/17/2024
0.11.2 236 10/13/2024
0.11.1 135 10/12/2024
0.11.0 137 10/11/2024
0.10.4 108 10/8/2024
0.10.3 119 10/7/2024
0.10.2 88 10/7/2024
0.10.1 91 10/7/2024
0.10.0 91 10/7/2024
0.9.17 1,739 9/14/2024
0.9.16 135 9/14/2024
0.9.15 218 9/13/2024
0.9.14 120 9/13/2024
0.9.13 298 9/7/2024
0.9.12 110 9/6/2024
0.9.11 276 9/5/2024
0.9.10 357 9/4/2024
0.9.9 404 8/29/2024
0.9.8 99 8/29/2024
0.9.7 238 8/28/2024
0.9.6 203 8/27/2024
0.9.5 108 8/27/2024
0.9.4 578 8/16/2024
0.9.3 592 8/14/2024
0.9.2 412 8/12/2024
0.9.1 271 8/8/2024
0.9.0 304 8/5/2024
0.8.13 71 8/2/2024
0.8.12 72 8/2/2024
0.8.11 442 7/26/2024
0.8.10 112 7/25/2024
0.8.9 111 7/25/2024
0.8.8 291 7/21/2024
0.8.7 151 7/19/2024
0.8.6 97 7/19/2024
0.8.5 193 7/16/2024
0.8.4 122 7/16/2024
0.8.3 114 7/16/2024
0.8.2 198 7/13/2024
0.8.1 108 7/13/2024
0.8.0 102 7/13/2024
0.7.9 123 7/12/2024
0.7.8 108 7/12/2024
0.7.7 123 7/11/2024
0.7.6 109 7/10/2024
0.7.5 106 7/10/2024
0.7.4 128 7/10/2024
0.7.3 112 7/10/2024
0.7.2 100 7/9/2024
0.7.1 115 7/9/2024
0.7.0 109 7/9/2024