Akka.Persistence.EventStore 1.5.30

Prefix Reserved
dotnet add package Akka.Persistence.EventStore --version 1.5.30                
NuGet\Install-Package Akka.Persistence.EventStore -Version 1.5.30                
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="Akka.Persistence.EventStore" Version="1.5.30" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Akka.Persistence.EventStore --version 1.5.30                
#r "nuget: Akka.Persistence.EventStore, 1.5.30"                
#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 Akka.Persistence.EventStore as a Cake Addin
#addin nuget:?package=Akka.Persistence.EventStore&version=1.5.30

// Install Akka.Persistence.EventStore as a Cake Tool
#tool nuget:?package=Akka.Persistence.EventStore&version=1.5.30                

Akka.Persistence.EventStore

NuGet Version

Akka Persistence EventStore Plugin is a plugin for Akka Persistence that provides components:

This plugin stores data in a EventStoreDB database and based on EventStore.Client.Grpc client library.

Getting started

The Easy Way, Using Akka.Hosting

var host = new HostBuilder()
    .ConfigureServices((context, services) => {
        services.AddAkka("my-system-name", (builder, provider) =>
        {
            builder.WithEventStorePersistence(connectionString: _myConnectionString)
        });
    })

The Classic Way, Using HOCON

These are the minimum HOCON configuration you need to start using Akka.Persistence.EventStore:

akka.persistence {
    journal {
      plugin = "akka.persistence.journal.eventstore"
      
      eventstore {
            class = "Akka.Persistence.EventStore.Journal.EventStoreJournal, Akka.Persistence.EventStore"
            connection-string = "{database-connection-string}"
        }
    }
  
    query.journal.eventstore {
        class = "Akka.Persistence.EventStore.Query.EventStoreReadJournalProvider, Akka.Persistence.EventStore"
        write-plugin = "akka.persistence.journal.eventstore"
    }
  
    snapshot-store {
        plugin = "akka.persistence.snapshot-store.eventstore"
        
        eventstore {
            class = "Akka.Persistence.EventStore.Snapshot.EventStoreSnapshotStore, Akka.Persistence.EventStore"
            connection-string = "{database-connection-string}"
        }
    }
}

Configuration options

Journal

  • connection-string - Connection string, as described here: https://developers.eventstore.com/clients/grpc/#connection-string.
  • materializer-dispatcher - Dispatcher used to drive journal actor
  • adapter - Controls how the event data and metadata is stored and retrieved. See Adapter section below for more information.
  • auto-initialize - Whether or not the plugin should create projections to support read journal on startup. See Projections section below for more information.
  • prefix - A optional prefix that will be added to streams.
  • tenant - A optional tenant that should be used to support multi-tenant environments.
  • tagged-stream-name-pattern - A pattern used when creating a stream name for a tags-stream. The name [[TAG]] will be replaced by the actual tag used.
  • persistence-ids-stream-name - A name for the stream that stores all persistence id's (to support read journal).
  • persisted-events-stream-name - A name for the stream that stores all events (to support read journal).

Snapshot store

  • connection-string - Connection string, as described here: https://developers.eventstore.com/clients/grpc/#connection-string.
  • materializer-dispatcher - Dispatcher used to drive journal actor
  • adapter - Controls how the event data and metadata is stored and retrieved. See Adapter section below for more information.- prefix - A optional prefix that will be added to streams.
  • tenant - A optional tenant that should be used to support multi-tenant environments.

Query journal

  • write-plugin - Absolute path to the write journal plugin configuration entry that this query journal will connect to.

Adapter

Akka Persistence EventStore Plugin supports changing how data is stored and retrieved. By default, it will serialize the data using the configured serializer in Akka, and populate the Metadata with the following information:

{
  "persistenceId": "p-14",
  "occurredOn": "2018-05-03T10:28:06.3437687-06:00",
  "manifest": "",
  "sender": "",
  "sequenceNr": 5,
  "writerGuid": "f8706bba-52a7-4326-a760-990c7f657c46",
  "journalType": "WriteJournal",
  "timestamp": 123456789,
  "tenant": "",
  "tags": []
}

If you are happy with the default serialization and metadata, but want to just augment the metadata or data, or do any of the following:

  • Inspect event to add metadata
  • Encrypt data
  • Change the "type" stored in event store

You can inherit from DefaultAdapter and override the Serialize, DeSerialize, GetEventMetadata, GetSnapshotMetadata , GetEventMetadataFrom and GetSnapshotMetadataFrom methods

You also have the option of creating a new implemenation of Akka.Persistence.EventStore.Serialization.IMessageAdapter. Everything is DIY in this case, including correct handling of internal Akka types if they appear in events. Make use of the supplied Akka.Serialization.Serialization to help with this.

public class CustomAdapter : IMessageAdapter
{
    public CustomAdapter(Akka.Serialization.Serialization serialization, ISettingsWithAdapter settings)
    {
    }

    public Task<EventData> Adapt(IPersistentRepresentation persistentMessage)
    {
        // Implement
    }

    public Task<EventData> Adapt(SnapshotMetadata snapshotMetadata, object snapshot)
    {
        // Implement
    }
    
    public Task<IPersistentRepresentation?> AdaptEvent(ResolvedEvent evnt)
    {
    	// Implement
    }
    
    public Task<SelectedSnapshot?> AdaptSnapshot(ResolvedEvent evnt)
    {
    	// Implement
    }
    
    public string GetManifest(Type type)
    {
    	// Implement
    }
}

Whichever direction you go, you will need to override the configuration

Using Akka Hosting:

var host = new HostBuilder()
    .ConfigureServices((context, services) => {
        services.AddAkka("my-system-name", (builder, provider) =>
        {
            builder.WithEventStorePersistence(
                connectionString: _myConnectionString,
                adapter: "Your.Namespace.YourAdapter, Your.Assembly")
        });
    })

Using Hocon:

akka.persistence {
    journal {
        plugin = "akka.persistence.journal.eventstore""
        eventstore {
            class = "Akka.Persistence.EventStore.Journal.EventStoreJournal, Akka.Persistence.EventStore"
            connection-string = "esdb://admin:changeit@localhost:2113"
            adapter = "Your.Namespace.YourAdapter, Your.Assembly"
        }
    }
}

There is also an implementation using System.Text.Json in the plugin that can be used. That can be configured like this: Using Akka Hosting:

var host = new HostBuilder()
    .ConfigureServices((context, services) => {
        services.AddAkka("my-system-name", (builder, provider) =>
        {
            builder.WithEventStorePersistence(
                connectionString: _myConnectionString,
                adapter: "system-text-json")
        });
    })

Using Hocon:

akka.persistence {
    journal {
        plugin = "akka.persistence.journal.eventstore""
        eventstore {
            class = "Akka.Persistence.EventStore.Journal.EventStoreJournal, Akka.Persistence.EventStore"
            connection-string = "esdb://admin:changeit@localhost:2113"
            adapter = "system-text-json"
        }
    }
}

Projections

To support the Read Journal the plugin takes advantage of the projections feature of EventStoreDB. If you setup auto-initialize on the Journal the required projections will be created for you on startup. You can also use Akka.Persistence.EventStore.Projections.EventStoreProjectionsSetup to create the projections yourself if you want.

Persistent subscriptions

Persistent subscriptions can be used to subscribe to events stored in EventStoreDb and let the database handle offsets. This plugin gives you a Akka streams source to make it easy to work with within Akka.net.

var clientSettings = EventStoreClientSettings.Create(eventStoreContainer.ConnectionString ?? "");
        
var subscriptionClient = new EventStorePersistentSubscriptionsClient(clientSettings);

EventStoreSource
    .ForPersistentSubscription(
        subscriptionClient,
        "your-stream-name",
        "your-subscriptions-group-name",
        keepReconnecting: true) //true if you want the client to reconnect if it's disconnected, otherwise false (default). 
    .RunForeach(x =>
    {
        Console.WriteLine(x.Event.Event.EventType);

        x.Ack();
    }, _actorSystem.Materializer());

Release Notes, Version Numbers, Etc

This project will automatically populate its release notes in all of its modules via the entries written inside RELEASE_NOTES.md and will automatically update the versions of all assemblies and NuGet packages via the metadata included inside Directory.Build.props.

Breaking Changes in 1.5

This is a complete rewrite of the plugin to use EventStore.Client.Grpc. This means that the plugin is not compatible with previous versions.

Maintainer

Product Compatible and additional computed target framework versions.
.NET net6.0 is compatible.  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.  net9.0 was computed.  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 (3)

Showing the top 3 NuGet packages that depend on Akka.Persistence.EventStore:

Package Downloads
Akka.Persistence.EventStore.Query

Akka.NET Persistence read journal backed by EventStore.

Akka.Persistence.EventStore.Hosting

Akka.Persistence.EventStore Microsoft.Extensions.Hosting support.

Akka.Persistence.EventStore.Benchmarks

Package Description

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.5.30 416 10/14/2024
1.5.26 1,059 6/28/2024
1.5.25 149 6/18/2024
1.5.20 204 5/9/2024
1.5.18 156 4/9/2024
1.3.0 14,186 10/28/2018
1.0.0 854 10/18/2018

Upgrade Akka to version 1.5.30
Upgrade EventStore to version 23.3.5
[Added a message adapter for System.Text.Json](https://github.com/akkadotnet/Akka.Persistence.EventStore/pull/54)
[Added support for a writer uuid](https://github.com/akkadotnet/Akka.Persistence.EventStore/pull/55)