Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Deprecate ClusterSingleton.Init() and add missing singleton feature to ClusterSingletonSettings #7387

Merged
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
37 changes: 37 additions & 0 deletions docs/community/whats-new/akkadotnet-v1.5-upgrade-advisories.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,43 @@ This document contains specific upgrade suggestions, warnings, and notices that
<iframe width="560" height="315" src="https://www.youtube.com/embed/-UPestlIw4k" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" allowfullscreen></iframe>
<!-- markdownlint-enable MD033 -->

## Upgrading to Akka.NET v1.5.32

### Breaking change in `Akka.Cluster.Tools`

The method `ClusterSingleton.Init()` will be removed in future v1.6, if you're using this method, you need to convert it to use `ClusterSingletonManager.Props` and `ClusterSingletonProxy.Props` instead.

In order to preserve backward compatibility within your cluster, if you're using this convention:
```csharp
var settings = ClusterSingletonSettings.Create(system);
var singletonActor = SingletonActor.Create(Counter.Props, "GlobalCounter")
.WithStopMessage(MyStopMessage.Instance)
.WithSettings(settings);
var proxy = singleton.Init(singletonActor);
```

You will need to convert it to:
```csharp
var managerSettings = ClusterSingletonManagerSettings.Create(system)
.WithSingletonName("GlobalCounter");
system.ActorOf(
props: ClusterSingletonManager.Props(
singletonProps: Counter.Props,
terminationMessage: MyStopMessage.Instance,
settings: managerSettings),
name: "singletonManagerGlobalCounter");

var proxySettings = ClusterSingletonProxySettings.Create(system)
.WithSingletonName("GlobalCounter");
var proxy = system.ActorOf(
props: ClusterSingletonProxy.Props(
singletonManagerPath: "/user/singletonManagerGlobalCounter",
settings: proxySettings),
name: "singletonProxyGlobalCounter");
```

Note that to preserve backward compatibility between cluster nodes, the singleton manager actor name **MUST** be in the `$"singletonManager{singletonName}"` format.

## Upgrading to Akka.NET v1.5.31

Akka.NET v1.5.31 introduces a breaking behavior change to actor `Stash`. In previous behavior, `Stash` will filter out any messages that are identical (see explanation below) when it is prepended with another. It will not do so now, which is the actual intended behavior.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
using System.Collections.Concurrent;
using Akka.Actor;
using Akka.Annotations;
using Akka.Configuration;
using Akka.Util;

namespace Akka.Cluster.Tools.Singleton
Expand All @@ -22,6 +23,18 @@ public class ClusterSingleton : IExtension
{
private readonly ActorSystem _system;
private readonly Lazy<Cluster> _cluster;

/// <summary>
/// Returns default HOCON configuration for the cluster singleton.
/// </summary>
/// <returns>TBD</returns>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove the TBD

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

public static Config DefaultConfig()
{
return ConfigurationFactory.FromResource<ClusterSingleton>(
"Akka.Cluster.Tools.Singleton.reference.conf");
}

// Cache for singleton proxies, remove in v1.6
private readonly ConcurrentDictionary<string, IActorRef> _proxies = new();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah I agree we don't really need a cache for this - that's a little too clever for its own good.


public static ClusterSingleton Get(ActorSystem system) =>
Expand All @@ -30,6 +43,7 @@ public static ClusterSingleton Get(ActorSystem system) =>
public ClusterSingleton(ExtendedActorSystem system)
{
_system = system;
_system.Settings.InjectTopLevelFallback(DefaultConfig());
_cluster = new Lazy<Cluster>(() => Cluster.Get(system));
}

Expand All @@ -40,6 +54,10 @@ public ClusterSingleton(ExtendedActorSystem system)
/// <para>If there already is a proxy running for the given `singletonName` on this node, an <see cref="IActorRef"/> to that is returned.</para>
/// </summary>
/// <returns>A proxy actor that can be used to communicate with the singleton in the cluster</returns>
[Obsolete("This convenience method is deprecated and will be removed in v1.6, " +
"please use ClusterSingletonManager.Props and ClusterSingletonProxy.Props directly instead. " +
"See https://getakka.net/community/whats-new/akkadotnet-v1.5-upgrade-advisories.html#upgrading-to-akkanet-v1532. " +
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

"Since 1.5.32.")]
public IActorRef Init(SingletonActor singleton)
{
var settings = singleton.Settings.GetOrElse(ClusterSingletonSettings.Create(_system));
Expand All @@ -63,21 +81,23 @@ public IActorRef Init(SingletonActor singleton)
return GetProxy(singleton.Name, settings);
}

[Obsolete("Deprecated, remove in v1.6")]
private IActorRef GetProxy(string name, ClusterSingletonSettings settings)
{
IActorRef ProxyCreator()
{
var proxyName = $"singletonProxy{name}";
return _system.ActorOf(ClusterSingletonProxy.Props(
singletonManagerPath: $"/user/{ManagerNameFor(name)}",
settings: settings.ToProxySettings(name)),
proxyName);
return _system.ActorOf(
props: ClusterSingletonProxy.Props(
singletonManagerPath: $"/user/{ManagerNameFor(name)}",
settings: settings.ToProxySettings(name)),
name: proxyName);
}

return _proxies.GetOrAdd(name, _ => ProxyCreator());
}


[Obsolete("Deprecated, remove in v1.6")]
private string ManagerNameFor(string singletonName) => $"singletonManager{singletonName}";
}

Expand All @@ -86,6 +106,10 @@ public class ClusterSingletonProvider : ExtensionIdProvider<ClusterSingleton>
public override ClusterSingleton CreateExtension(ExtendedActorSystem system) => new(system);
}

[Obsolete("This setting class is deprecated and will be removed in v1.6, " +
"please use ClusterSingletonManager.Props and ClusterSingletonProxy.Props directly instead. " +
"See https://getakka.net/community/whats-new/akkadotnet-v1.5-upgrade-advisories.html#upgrading-to-akkanet-v1532. " +
"Since 1.5.32.")]
public class SingletonActor
{
public string Name { get; }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -535,6 +535,7 @@ public sealed class ClusterSingletonManager : FSM<ClusterSingletonState, ICluste
/// Returns default HOCON configuration for the cluster singleton.
/// </summary>
/// <returns>TBD</returns>
[Obsolete("Deprecated and will be removed in v1.6, please use ClusterSingleton.DefaultConfig() instead. Since 1.5.32.")]
public static Config DefaultConfig()
{
return ConfigurationFactory.FromResource<ClusterSingletonManager>(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,10 @@ namespace Akka.Cluster.Tools.Singleton
/// The settings used for the <see cref="ClusterSingleton"/>
/// </summary>
[Serializable]
[Obsolete("This setting class is deprecated and will be removed in v1.6, " +
"please use ClusterSingletonManager.Props and ClusterSingletonProxy.Props directly instead. " +
"See https://getakka.net/community/whats-new/akkadotnet-v1.5-upgrade-advisories.html#upgrading-to-akkanet-v1532. " +
"Since 1.5.32.")]
public class ClusterSingletonSettings : INoSerializationVerificationNeeded
{
/// <summary>
Expand Down Expand Up @@ -64,6 +68,16 @@ public class ClusterSingletonSettings : INoSerializationVerificationNeeded
/// </summary>
public bool ConsiderAppVersion { get; }

/// <summary>
/// Should the singleton proxy publish a warning if no singleton actor were found after a period of time
/// </summary>
public bool LogSingletonIdentificationFailure { get; }

/// <summary>
/// The period the proxy will wait until it logs a missing singleton warning, defaults to 1 minute
/// </summary>
public TimeSpan SingletonIdentificationFailurePeriod { get; }

/// <summary>
/// Create settings from the default configuration `akka.cluster`.
/// </summary>
Expand All @@ -88,7 +102,9 @@ public static ClusterSingletonSettings Create(Config config)
mgrSettings.HandOverRetryInterval,
proxySettings.BufferSize,
mgrSettings.LeaseSettings,
false);
false,
proxySettings.LogSingletonIdentificationFailure,
proxySettings.SingletonIdentificationFailurePeriod);
}

private ClusterSingletonSettings(
Expand All @@ -98,7 +114,9 @@ private ClusterSingletonSettings(
TimeSpan handOverRetryInterval,
int bufferSize,
LeaseUsageSettings leaseSettings,
bool considerAppVersion)
bool considerAppVersion,
bool logSingletonIdentificationFailure,
TimeSpan singletonIdentificationFailurePeriod)
{
if (singletonIdentificationInterval == TimeSpan.Zero)
throw new ArgumentException("singletonIdentificationInterval must be positive", nameof(singletonIdentificationInterval));
Expand All @@ -119,24 +137,39 @@ private ClusterSingletonSettings(
BufferSize = bufferSize;
LeaseSettings = leaseSettings;
ConsiderAppVersion = considerAppVersion;
LogSingletonIdentificationFailure = logSingletonIdentificationFailure;
SingletonIdentificationFailurePeriod = singletonIdentificationFailurePeriod;
}

public ClusterSingletonSettings WithRole(string role) => Copy(role: role);

public ClusterSingletonSettings WithSingletonIdentificationInterval(TimeSpan singletonIdentificationInterval)
=> Copy(singletonIdentificationInterval: singletonIdentificationInterval);

public ClusterSingletonSettings WithRemovalMargin(TimeSpan removalMargin) => Copy(removalMargin: removalMargin);

public ClusterSingletonSettings WithHandOverRetryInterval(TimeSpan handOverRetryInterval) => Copy(handOverRetryInterval: handOverRetryInterval);

public ClusterSingletonSettings WithBufferSize(int bufferSize) => Copy(bufferSize: bufferSize);

public ClusterSingletonSettings WithLeaseSettings(LeaseUsageSettings leaseSettings) => Copy(leaseSettings: leaseSettings);

public ClusterSingletonSettings WithLogSingletonIdentificationFailure(bool logSingletonIdentificationFailure)
=> Copy(logSingletonIdentificationFailure: logSingletonIdentificationFailure);

public ClusterSingletonSettings WithSingletonIdentificationFailurePeriod(TimeSpan singletonIdentificationFailurePeriod)
=> Copy(singletonIdentificationFailurePeriod: singletonIdentificationFailurePeriod);

private ClusterSingletonSettings Copy(
Option<string> role = default,
TimeSpan? singletonIdentificationInterval = null,
TimeSpan? removalMargin = null,
TimeSpan? handOverRetryInterval = null,
int? bufferSize = null,
Option<LeaseUsageSettings> leaseSettings = default,
bool? considerAppVersion = null)
bool? considerAppVersion = null,
bool? logSingletonIdentificationFailure = null,
TimeSpan? singletonIdentificationFailurePeriod = null)
{
return new ClusterSingletonSettings(
role: role.HasValue ? role.Value : Role,
Expand All @@ -145,16 +178,18 @@ private ClusterSingletonSettings Copy(
handOverRetryInterval: handOverRetryInterval ?? HandOverRetryInterval,
bufferSize: bufferSize ?? BufferSize,
leaseSettings: leaseSettings.HasValue ? leaseSettings.Value : LeaseSettings,
considerAppVersion: considerAppVersion ?? ConsiderAppVersion);
considerAppVersion: considerAppVersion ?? ConsiderAppVersion,
logSingletonIdentificationFailure: logSingletonIdentificationFailure ?? LogSingletonIdentificationFailure,
singletonIdentificationFailurePeriod: singletonIdentificationFailurePeriod ?? SingletonIdentificationFailurePeriod);
}

[InternalApi]
internal ClusterSingletonManagerSettings ToManagerSettings(string singletonName) =>
new(singletonName, Role, RemovalMargin, HandOverRetryInterval, LeaseSettings, ConsiderAppVersion);
new(singletonName, Role, RemovalMargin, HandOverRetryInterval, LeaseSettings, false);

[InternalApi]
internal ClusterSingletonProxySettings ToProxySettings(string singletonName) =>
new(singletonName, Role, SingletonIdentificationInterval, BufferSize, ConsiderAppVersion, true, TimeSpan.FromSeconds(30));
new(singletonName, Role, SingletonIdentificationInterval, BufferSize, false, LogSingletonIdentificationFailure, SingletonIdentificationFailurePeriod);

[InternalApi]
internal bool ShouldRunManager(Cluster cluster) => string.IsNullOrEmpty(Role) || cluster.SelfMember.Roles.Contains(Role);
Expand Down
Loading