![]() |
VOOZH | about |
dotnet add package Akka.Persistence.Redis --version 1.5.68
NuGet\Install-Package Akka.Persistence.Redis -Version 1.5.68
<PackageReference Include="Akka.Persistence.Redis" Version="1.5.68" />
<PackageVersion Include="Akka.Persistence.Redis" Version="1.5.68" />Directory.Packages.props
<PackageReference Include="Akka.Persistence.Redis" />Project file
paket add Akka.Persistence.Redis --version 1.5.68
#r "nuget: Akka.Persistence.Redis, 1.5.68"
#:package Akka.Persistence.Redis@1.5.68
#addin nuget:?package=Akka.Persistence.Redis&version=1.5.68Install as a Cake Addin
#tool nuget:?package=Akka.Persistence.Redis&version=1.5.68Install as a Cake Tool
Akka Persistence Redis Plugin is a plugin for Akka persistence that provides several components:
NOTE: in Akka.Persistence.Redis v1.4.16 we removed Akka.Persistence.Query support. Please read more about that decision and comment here: https://github.com/akkadotnet/Akka.Persistence.Redis/issues/126
This plugin stores data in a redis database and based on Stackexchange.Redis library.
From Nuget Package Manager
Install-Package Akka.Persistence.Redis
From .NET CLI
dotnet add package Akka.Persistence.Redis
To activate the journal plugin, add the following line to your HOCON config:
akka.persistence.journal.plugin = "akka.persistence.journal.redis"
This will run the journal with its default settings. The default settings can be changed with the configuration properties defined in your HOCON config:
akka.persistence.journal.redis {
# qualified type name of the Redis persistence journal actor
class = "Akka.Persistence.Redis.Journal.RedisJournal, Akka.Persistence.Redis"
# connection string, as described here: https://stackexchange.github.io/StackExchange.Redis/Configuration#basic-configuration-strings
configuration-string = ""
# Redis journals key prefixes. Leave it for default or change it to appropriate value. WARNING: don't change it on production instances.
key-prefix = ""
}
configuration-string - connection string, as described here: https://stackexchange.github.io/StackExchange.Redis/Configuration#basic-configuration-stringskey-prefix - Redis journals key prefixes. Leave it for default or change it to customized value. WARNING: don't change this value after you've started persisting data in production.database - Set the Redis default database to use. If you added defaultDatabase to the connection-strings, you have to set database to the value of defaultDatabase.use-database-number-from-connection-string - determines redis database precedence when a user adds defaultDatabase to the connection-strings. For Redis Cluster, the defaultDatabase is 0! See below:NOTE: Redis Standalone supports deploying multiple instances, but Redis cluster does not. The default database with Redis Cluster is always 0 - If you are deploying Redis Cluster, you don't need to add the defaultDatabase to the connection-string'! cluster-spec
To activate the snapshot plugin, add the following line to your HOCON config:
akka.persistence.snapshot-store.plugin = "akka.persistence.snapshot-store.redis"
This will run the snapshot-store with its default settings. The default settings can be changed with the configuration properties defined in your HOCON config:
akka.persistence.snapshot-store.redis {
# qualified type name of the Redis persistence journal actor
class = "Akka.Persistence.Redis.Snapshot.RedisSnapshotStore, Akka.Persistence.Redis"
# connection string, as described here: https://stackexchange.github.io/StackExchange.Redis/Configuration#basic-configuration-strings
configuration-string = ""
# Redis journals key prefixes. Leave it for default or change it to appropriate value. WARNING: don't change it on production instances.
key-prefix = ""
}
configuration-string - connection string, as described here: https://stackexchange.github.io/StackExchange.Redis/Configuration#basic-configuration-stringskey-prefix - Redis journals key prefixes. Leave it for default or change it to appropriate value. WARNING: don't change it on production instances.You can secure the Redis server Akka.Persistence.Redis connects to by leveraging Redis ACL and requiring users to use AUTH to connect to the Redis server.
Redis SSL/TLS You can use redis-cli to enable SSL/TLS feature in Redis.
StackExchange.Redis Connection string To connect to ACL enabled Redis server, you will need to set the user and password option in the connection string: "myServer.net:6380,user=<username>,password=<password>"
All of these features are supported via StackExchange.Redis, which Akka.Persistence.Redis uses internally, and you only need to customize your akka.persistence.journal.redis.configuration-string and akka.persistence.snapshot-store.redis.configuration-string values to customize it.
For instance, if you want to enable TLS on your Akka.Persistence.Redis instance:
akka.persistence.journal.redis.configuration-string = "contoso5.redis.cache.windows.net,ssl=true,password=..."
Or if you need to connect to multiple redis instances in a cluster:
akka.persistence.journal.redis.configuration-string = "contoso5.redis.cache.windows.net, contoso4.redis.cache.windows.net,ssl=true,password=..."
To connect to your redis instance with access control (ACL) support for Akka.Persistence.Redis, all you need to do is specify the user name and password in your connection string and this will restrict the StackExchange.Redis client used internally by Akka.Persistence.Redis to whatever permissions you specified in your cluster:
akka.persistence.journal.redis.configuration-string = "contoso5.redis.cache.windows.net, contoso4.redis.cache.windows.net,user=akka-persistence,password=..."
These are the minimum Redis commands that are needed by Akka.Persistence.Redis to work properly.
| Redis Command | StackExchange.Redis Command |
|---|---|
| MULTI | Transaction |
| EXEC | |
| DISCARD | |
| SET | String Set |
| SETNX | |
| SETEX | |
| GET | String Get |
| LLEN | List Length |
| LRANGE | List Range |
| RPUSH | List Right Push |
| RPUSHX | |
| LLEN | |
| ZADD | Sorted Set Add |
| ZREMRANGEBYSCORE | Delete Sorted Set by Score Range |
| ZREVRANGEBYSCORE | Get Sorted Set by Score Range |
| ZRANGEBYSCORE | |
| WITHSCORES | |
| LIMIT | |
| SSCAN | Scan |
| SMEMBERS | |
| PUBSUB | Pub/Sub |
| PING | |
| UNSUBSCRIBE | |
| SUBSCRIBE | |
| PSUBSCRIBE | |
| PUNSUBSCRIBE | |
| PUBLISH | Pub/Sub Publish |
Redis Cluster failovers shift slot ownership between nodes. During the brief window between a node being demoted to replica and the SE.Redis client refreshing its slot-map cache, writes can be routed to a node that now refuses them (Command cannot be issued to a replica). Production deployments need three layers of mitigation, two of which are user-configured and the third of which the plugin does for you automatically.
{addresses},abortConnect=false,connectRetry=5,configCheckSeconds=10,syncTimeout=5000,asyncTimeout=5000
| Setting | Why |
|---|---|
abortConnect=false |
Without it, initial connection failures can leave the multiplexer permanently unhealthy. |
connectRetry=5 |
Number of times SE.Redis retries the initial connection attempt. Helps during transient failover. |
configCheckSeconds=10 |
How often SE.Redis re-queries cluster topology (CLUSTER NODES/CLUSTER SLOTS). Default is 60s β cutting to 10s shrinks the stale-topology window after a failover. |
syncTimeout / asyncTimeout (5000 ms) |
Per-operation timeouts at the SE.Redis layer. Should be lower than the Akka circuit-breaker call-timeout so the breaker trips on real hangs rather than waiting on SE.Redis's own timeout. |
Do not add
allowAdmin=trueunless you genuinely need admin commands (FLUSHDB,CONFIG, etc.). It enables dangerous operations and is not needed for persistence.
akka.persistence {
max-concurrent-recoveries = 64 # or 128 for large sharded deployments β NOT higher
journal.redis {
circuit-breaker {
call-timeout = 10s # do NOT set to 120s
reset-timeout = 30s
max-failures = 10
}
}
snapshot-store.redis {
circuit-breaker {
call-timeout = 10s
reset-timeout = 30s
max-failures = 10
}
}
}
max-concurrent-recoveries = 64β128 β higher values turn a cluster blip into a recovery storm.call-timeout = 10s β values like 120s are an amplifier, not a safety margin. They turn a 30-second cluster failover into a 30-minute outage by holding hundreds of in-flight recoveries open for two minutes each.Ask timeouts must be larger than call-timeout, never smaller. Otherwise the Ask fails before the journal has any chance to fail fast its own way, and upstream consumers (RabbitMQ requeues, HTTP retries, etc.) hammer the system while persistence is still on its first attempt. Rule of thumb: Ask timeout β₯ call-timeout Γ max-failures Γ 1.5.Two failure modes need different handling:
MOVED redirects β handled by StackExchange.Redis itself (since 2.6.86). It proactively refreshes its slot map on a 5-second debounce when it sees a MOVED response. No plugin involvement.Command cannot be issued to a replica β runtime refusal from a node that thinks it's a replica. SE.Redis does NOT auto-handle this. The plugin catches it inside the journal and snapshot-store write paths, fires IConnectionMultiplexer.ConfigureAsync() in the background to force a topology refresh, and rethrows so the journal's circuit breaker handles retry timing. By the time reset-timeout (e.g. 30s) elapses, the refresh has completed and the next attempt sees fresh topology.No manual restart or topology poke is required.
ConfigurationOptions control via WithRedisPersistenceThe connection string above expresses most of what you need, but some tunings (e.g. an ExponentialRetry ReconnectRetryPolicy, custom SocketManager, explicit EndPoints) can only be set programmatically. For those, supply a pre-configured IConnectionMultiplexer β see Supplying a Pre-Configured IConnectionMultiplexer (Non-Azure) below. The plugin's auto-topology-refresh works identically whether you let the plugin own the connection or supply your own.
Akka.Persistence.Redis.Hosting provides a set of extension methods for integrating Akka.Persistence.Redis with Akka.Hosting, making it easy to configure Redis persistence and health checks using Microsoft's dependency injection and hosting model.
From Nuget Package Manager
Install-Package Akka.Persistence.Redis.Hosting
From .NET CLI
dotnet add package Akka.Persistence.Redis.Hosting
using var host = new HostBuilder()
.ConfigureServices((context, services) =>
{
services.AddAkka("redisDemo", (builder, provider) =>
{
builder
.WithRedisPersistence("your-redis-connection-string");
});
}).Build();
await host.RunAsync();
For Azure Managed Redis (*.redis.azure.net) or Azure Cache for Redis (*.redis.cache.windows.net) deployments, use WithAzureRedisPersistence(...). It auto-detects the Azure host suffix, configures TLS + RESP3, and authenticates via Microsoft.Azure.StackExchangeRedis's ConfigureForAzureWithTokenCredentialAsync using a TokenCredential from Azure.Identity (defaulting to ManagedIdentityCredential). For non-Azure hosts (e.g. local-dev localhost:6379) it falls through to the plain HOCON connection-string path.
// One-liner β system-assigned managed identity is detected automatically.
builder.WithAzureRedisPersistence("your-redis.swedencentral.redis.azure.net:10000");
// Pin a user-assigned managed identity by client id.
builder.WithAzureRedisPersistence(
"your-redis.swedencentral.redis.azure.net:10000",
credential: new ManagedIdentityCredential("your-client-id"));
// Local development β no Azure suffix, plain HOCON connection-string auth.
builder.WithAzureRedisPersistence("localhost:6379");
Add the Akka.Persistence.Redis.Hosting package and using Azure.Identity; (the package transitively pulls in Microsoft.Azure.StackExchangeRedis and Azure.Identity for you).
IConnectionMultiplexer (Non-Azure)For deployments that aren't Azure-managed but still need an IConnectionMultiplexer authored programmatically β Redis Sentinel, a custom ReconnectRetryPolicy (e.g. ExponentialRetry), a tighter ConfigCheckSeconds for clustered Redis, or any other ConfigurationOptions knob that does not have a connection-string equivalent β pass the multiplexer or factory directly to WithRedisPersistence.
WithRedisPersistence has three connection-source overloads. Each takes its connection source as a non-optional positional argument, so the compiler enforces that you always provide one:
// 1) HOCON connection string. The plugin opens, owns, and disposes the multiplexer.
builder.WithRedisPersistence("your-redis-connection-string");
// 2) Pre-built IConnectionMultiplexer. Caller-owned by default β the plugin will not
// dispose it on actor shutdown. Use PluginOwned to hand off disposal.
var multiplexer = await ConnectionMultiplexer.ConnectAsync(new ConfigurationOptions
{
EndPoints = { { "your-redis-host", 6380 } },
Ssl = true,
AbortOnConnectFail = false,
ConfigCheckSeconds = 10,
ReconnectRetryPolicy = new ExponentialRetry(deltaBackOffMilliseconds: 1000),
});
builder.WithRedisPersistence(multiplexer);
// 3) Async factory. Useful when construction itself is async (custom auth handshakes,
// deferred bootstrap, etc.). Cache the result in your closure if you want sharing
// across plugins; an uncached factory gives every plugin its own multiplexer.
var shared = new Lazy<Task<IConnectionMultiplexer>>(
() => ConnectionMultiplexer.ConnectAsync(configurationOptions),
LazyThreadSafetyMode.ExecutionAndPublication);
builder.WithRedisPersistence(multiplexerFactory: () => shared.Value);
When the journal and snapshot store should point at different Redis instances β journal events on a write-tuned cluster, snapshots on a separate durable store, or cluster-sharding regions backed by different persistence β give each plugin its own HOCON connection string with a distinct pluginIdentifier. Each plugin opens its own multiplexer:
builder
.WithRedisPersistence("redis-events.example.com:6379",
pluginIdentifier: "events", isDefaultPlugin: false, mode: PersistenceMode.Journal)
.WithRedisPersistence("redis-snapshots.example.com:6379",
pluginIdentifier: "snapshots", isDefaultPlugin: false, mode: PersistenceMode.SnapshotStore);
Connection injection is plugin-scoped. A source registered for akka.persistence.journal.events does not affect akka.persistence.journal.redis or any other Redis plugin unless that plugin id is explicitly registered to the same source. Plugins without a registered source fall back to their own HOCON connection string.
The hosting extension carries multiplexer factories through Akka.NET's typed ActorSystemSetup container. The multiplexer: and multiplexerFactory: overloads register entries in RedisConnectionMultiplexerSetup for the journal and/or snapshot plugin ids configured by the call. The journal and snapshot store actors read their plugin-specific entry once at construction. When no entry exists for that plugin id, the plugin opens its own multiplexer from the HOCON connection string and disposes it on PostStop. The connectivity health check resolves the same plugin-specific setup entry at probe time, so a single source of configuration drives both the plugin and its liveness probe.
If you bootstrap the ActorSystem directly via ActorSystem.Create(...), build the setup yourself and pass it to the system:
var multiplexer = await ConnectionMultiplexer.ConnectAsync(configurationOptions);
var setup = ActorSystemSetup.Create(
BootstrapSetup.Create().WithConfig(redisHocon),
new RedisConnectionMultiplexerSetup()
.Add(
"akka.persistence.journal.redis",
() => Task.FromResult<IConnectionMultiplexer>(multiplexer),
RedisConnectionOwnership.CallerOwned)
.Add(
"akka.persistence.snapshot-store.redis",
() => Task.FromResult<IConnectionMultiplexer>(multiplexer),
RedisConnectionOwnership.CallerOwned));
var system = ActorSystem.Create("my-system", setup);
RedisConnectionOwnership.CallerOwned means the plugin will not dispose the multiplexer. Use RedisConnectionOwnership.PluginOwned when the factory is purely a construction hook and each plugin should dispose the connection it receives.
The Hosting package includes built-in connectivity health check support for verifying Redis availability and accessibility. These liveness checks proactively verify that your Redis instance is accessible and responsive by performing PING commands against the configured Redis instance.
Enable connectivity health checks by calling WithHealthCheck() on the journal and/or snapshot builder:
builder
.WithRedisPersistence(
journalOptions: new RedisJournalOptions
{
ConfigurationString = "your-redis-connection-string",
},
snapshotOptions: new RedisSnapshotOptions
{
ConfigurationString = "your-redis-connection-string",
},
journalBuilder: journal => journal.WithHealthCheck(HealthStatus.Degraded),
snapshotBuilder: snapshot => snapshot.WithHealthCheck(HealthStatus.Degraded));
When enabled, the connectivity health checks will:
Healthy when the Redis instance is accessibleDegraded or Unhealthy (configurable) when the instance is unreachable or unresponsiveHealth checks are tagged with akka, persistence, and redis for easy filtering and organization in your health check endpoints.
For ASP.NET Core applications, you can expose these health checks via an endpoint:
var builder = WebApplication.CreateBuilder(args);
// Add health checks service
builder.Services.AddHealthChecks();
builder.Services.AddAkka("redisDemo", (configBuilder, provider) =>
{
configBuilder
.WithRedisPersistence(
journalOptions: new RedisJournalOptions { ConfigurationString = "your-redis-connection-string" },
snapshotOptions: new RedisSnapshotOptions { ConfigurationString = "your-redis-connection-string" },
journalBuilder: journal => journal.WithHealthCheck(),
snapshotBuilder: snapshot => snapshot.WithHealthCheck());
});
var app = builder.Build();
// Map health check endpoint
app.MapHealthChecks("/healthz");
app.Run();
You can customize the tags applied to health checks by providing an IEnumerable<string> to the WithHealthCheck() method:
journalBuilder: journal => journal.WithHealthCheck(
unHealthyStatus: HealthStatus.Degraded,
name: "redis-journal",
tags: new[] { "backend", "database", "redis" }),
snapshotBuilder: snapshot => snapshot.WithHealthCheck(
unHealthyStatus: HealthStatus.Degraded,
name: "redis-snapshot",
tags: new[] { "backend", "database", "redis" })
When tags are not specified, the default tags are used: ["akka", "persistence", "redis"] for both journals and snapshot stores.
Akka Persistence provided serializers wrap the user payload in an envelope containing all persistence-relevant information. Redis Journal uses provided Protobuf serializers for the wrapper types (e.g. IPersistentRepresentation), then the payload will be serialized using the user configured serializer.
The payload will be serialized using Akka.NET's serialization bindings for your events and snapshot objects. By default, all objects that do not have a specified serializer will use Newtonsoft.Json polymorphic serialization (your CLR types β-> JSON.)
This is fine for testing and initial phases of your development (while youβre still figuring out things and the data will not need to stay persisted forever). However, once you move to production you should really pick a different serializer for your payloads.
We highly recommend creating schema-based serialization definitions using MsgPack, Google.Protobuf, or something similar and configuring serialization bindings for those in your configuration: https://getakka.net/articles/networking/serialization.html#usage
Serialization of snapshots and payloads of Persistent messages is configurable with Akkaβs Serialization infrastructure. For example, if an application wants to serialize
MyPayload with a custom MyPayloadSerializer andMySnapshot with a custom MySnapshotSerializer
it must addakka.actor {
serializers {
redis = "Akka.Serialization.YourOwnSerializer, YourOwnSerializer"
}
serialization-bindings {
"Akka.Persistence.Redis.Journal.JournalEntry, Akka.Persistence.Redis" = redis
"Akka.Persistence.Redis.Snapshot.SnapshotEntry, Akka.Persistence.Redis" = redis
}
}
The test suites stand up Redis via Testcontainers for .NET, so a working Docker host is required (Docker Desktop, Rancher Desktop, Colima, or Podman all work β Testcontainers auto-detects the socket).
Akka.Persistence.Redis.Tests spins up two redis:latest containers and connects to them through a comma-separated host list, exercising the standalone-Redis code paths.
Akka.Persistence.Redis.Cluster.Tests runs the cluster suite against the grokzen/redis-cluster:6.0.13 image. The image is one Docker container running six redis-server processes under supervisord β three masters + three replicas, on consecutive ports starting from a randomly-chosen base port. The fixture publishes those six ports 1:1 to the host so the cluster's gossiped node addresses match what the SE.Redis client connects to, then probes CLUSTER INFO on every endpoint until each node reports cluster_state:ok with full 16384-slot coverage before any test runs.
One implication of the single-container topology: integration tests cannot trigger a real per-node failover by stopping a Docker container, because killing the container kills the whole cluster. Anything that needs to exercise a single primary failover today has to either docker exec into the container and signal one of the redis processes (or supervisorctl stop redis-N), or use IServer.Shutdown(...) against a specific endpoint. A multi-container cluster fixture is the longer-term answer when richer failover coverage is required.
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net5.0 net5.0 was computed. net5.0-windows net5.0-windows was computed. net6.0 net6.0 is compatible. net6.0-android net6.0-android was computed. net6.0-ios net6.0-ios was computed. net6.0-maccatalyst net6.0-maccatalyst was computed. net6.0-macos net6.0-macos was computed. net6.0-tvos net6.0-tvos was computed. net6.0-windows net6.0-windows was computed. net7.0 net7.0 was computed. net7.0-android net7.0-android was computed. net7.0-ios net7.0-ios was computed. net7.0-maccatalyst net7.0-maccatalyst was computed. net7.0-macos net7.0-macos was computed. net7.0-tvos net7.0-tvos was computed. net7.0-windows net7.0-windows was computed. net8.0 net8.0 was computed. net8.0-android net8.0-android was computed. net8.0-browser net8.0-browser was computed. net8.0-ios net8.0-ios was computed. net8.0-maccatalyst net8.0-maccatalyst was computed. net8.0-macos net8.0-macos was computed. net8.0-tvos net8.0-tvos was computed. net8.0-windows net8.0-windows was computed. net9.0 net9.0 was computed. net9.0-android net9.0-android was computed. net9.0-browser net9.0-browser was computed. net9.0-ios net9.0-ios was computed. net9.0-maccatalyst net9.0-maccatalyst was computed. net9.0-macos net9.0-macos was computed. net9.0-tvos net9.0-tvos was computed. net9.0-windows net9.0-windows was computed. net10.0 net10.0 was computed. net10.0-android net10.0-android was computed. net10.0-browser net10.0-browser was computed. net10.0-ios net10.0-ios was computed. net10.0-maccatalyst net10.0-maccatalyst was computed. net10.0-macos net10.0-macos was computed. net10.0-tvos net10.0-tvos was computed. net10.0-windows net10.0-windows was computed. |
| .NET Core | netcoreapp2.0 netcoreapp2.0 was computed. netcoreapp2.1 netcoreapp2.1 was computed. netcoreapp2.2 netcoreapp2.2 was computed. netcoreapp3.0 netcoreapp3.0 was computed. netcoreapp3.1 netcoreapp3.1 was computed. |
| .NET Standard | netstandard2.0 netstandard2.0 is compatible. netstandard2.1 netstandard2.1 was computed. |
| .NET Framework | net461 net461 was computed. net462 net462 was computed. net463 net463 was computed. net47 net47 was computed. net471 net471 was computed. net472 net472 was computed. net48 net48 was computed. net481 net481 was computed. |
| MonoAndroid | monoandroid monoandroid was computed. |
| MonoMac | monomac monomac was computed. |
| MonoTouch | monotouch monotouch was computed. |
| Tizen | tizen40 tizen40 was computed. tizen60 tizen60 was computed. |
| Xamarin.iOS | xamarinios xamarinios was computed. |
| Xamarin.Mac | xamarinmac xamarinmac was computed. |
| Xamarin.TVOS | xamarintvos xamarintvos was computed. |
| Xamarin.WatchOS | xamarinwatchos xamarinwatchos was computed. |
Showing the top 1 NuGet packages that depend on Akka.Persistence.Redis:
| Package | Downloads |
|---|---|
|
Akka.Persistence.Redis.Hosting
Akka.NET Persistence journal and snapshot store backed by Redis. |
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 1.5.68 | 174 | 6/1/2026 |
| 1.5.68-beta1 | 140 | 5/13/2026 |
| 1.5.67 | 1,819 | 4/28/2026 |
| 1.5.59 | 1,488 | 1/27/2026 |
| 1.5.55.1 | 2,556 | 10/29/2025 |
| 1.5.55 | 1,276 | 10/27/2025 |
| 1.5.53 | 10,683 | 10/15/2025 |
| 1.5.42 | 7,978 | 5/22/2025 |
| 1.5.37 | 3,993 | 1/23/2025 |
| 1.5.30 | 6,658 | 10/4/2024 |
| 1.5.29 | 2,246 | 10/2/2024 |
| 1.5.24 | 7,157 | 6/11/2024 |
| 1.5.13 | 11,502 | 10/6/2023 |
| 1.5.0 | 13,751 | 3/7/2023 |
| 1.4.35 | 12,572 | 3/24/2022 |
| 1.4.31 | 4,493 | 12/21/2021 |
| 1.4.25 | 17,435 | 9/9/2021 |
| 1.4.20 | 7,738 | 5/14/2021 |
| 1.4.17 | 3,975 | 3/17/2021 |
| 1.4.16 | 6,026 | 2/6/2021 |
This is the stable release of the post-1.5.67 Redis hardening work.
**Behavior Changes / Compatibility Notes**
* Removed the legacy query subscriber notification protocol (`ISubscriptionCommand`, `SubscribeNewEvents`, and `NewEventAppended`). Live persistence queries now rely exclusively on polling via `akka.persistence.query.journal.redis.refresh-interval`. This removes a dead actor-local optimization that could leak subscribers and race with journal writes. Users that need lower live-query latency can reduce the Redis query `refresh-interval`.
* Redis journal and snapshot-store connections are now created during plugin actor construction instead of lazily on first database access. This fixes a `ConnectionMultiplexer` lifecycle leak and allows plugin-owned connections to be disposed when the actor stops, but connection failures may now surface earlier during plugin startup.
* Akka.Hosting Redis configuration now validates missing connection sources earlier. Each configured Redis journal or snapshot store must have either a HOCON connection string or a matching `RedisConnectionMultiplexerSetup` entry.
**Improvements**
* Added `WithAzureRedisPersistence(...)` for Azure Managed Redis and Azure Cache for Redis using Entra ID / Managed Identity authentication. The helper configures TLS, RESP3, `SslHost`, and token authentication via `Microsoft.Azure.StackExchangeRedis`.
* Added plugin-scoped `RedisConnectionMultiplexerSetup` support for caller-supplied multiplexers and multiplexer factories. Setup entries are keyed by Redis plugin id, so one Redis plugin's injected connection source does not override another plugin's HOCON connection string.
* Added explicit Redis connection ownership semantics via `RedisConnectionOwnership`: caller-owned, plugin-owned, and actor-system-owned.
* Updated Redis health checks so setup-backed plugins reuse the configured connection source, while HOCON-only plugins open a temporary multiplexer for the probe and dispose it after `PING`.
* Fixed snapshot-only Akka.Hosting configuration so Redis default HOCON is still loaded when only the snapshot store is configured.
* Fixed `DeferAsync` with an empty write batch, which previously could throw from `ContinueWhenAll`.
* Added `CommandFlags.DemandMaster` to Redis write operations as defense-in-depth for primary-only writes.
* Removed the `ConnectionMultiplexer` leak in journal and snapshot actors by tracking connection ownership and disposing plugin-owned connections from `PostStop`.
* Reworked Redis test fixtures to use Testcontainers and hardened cluster readiness / CI failure behavior.
* Added automatic Redis Cluster topology refresh on the "Command cannot be issued to a replica" error that surfaces after a failover demotes a primary. The journal and snapshot store catch this case, fire `IConnectionMultiplexer.ConfigureAsync()` in the background, and rethrow so the existing circuit breaker handles retry timing. Detection uses `IServer.IsReplica` against the endpoint in `Exception.Data["redis-server"]`, with the literal message as a fallback when `ConfigurationOptions.IncludeDetailInExceptions = false`.
* Added a new "Running against Redis Cluster" section to `README.md` covering the recommended SE.Redis connection string, Akka.Persistence circuit-breaker tuning, the `Ask` timeout vs `call-timeout` rule, and which cluster-failover failure modes are handled by StackExchange.Redis vs the plugin.
* Added commented circuit-breaker tuning guidance under the journal and snapshot-store sections of `reference.conf`. No HOCON defaults change.
**Dependencies**
* Upgraded `StackExchange.Redis` to 2.12.14.
* Added `Microsoft.Azure.StackExchangeRedis` 3.3.1 and `Azure.Identity` 1.17.1 to `Akka.Persistence.Redis.Hosting`.
* Updated test/build dependencies including `Microsoft.NET.Test.Sdk`, `coverlet.collector`, `Testcontainers`, and `Microsoft.SourceLink.GitHub`.