feat: add HTTP Management API, migrate CLI from Akka ClusterClient to HTTP
Replace the CLI's Akka.NET ClusterClient transport with a simple HTTP client targeting a new POST /management endpoint on the Central Host. The endpoint handles Basic Auth, LDAP authentication, role resolution, and ManagementActor dispatch in a single round-trip — eliminating the CLI's Akka, LDAP, and Security dependencies. Also fixes DCL ReSubscribeAll losing subscriptions on repeated reconnect by deriving the tag list from _subscriptionsByInstance instead of _subscriptionIds.
This commit is contained in:
@@ -1,17 +1,17 @@
|
||||
using ScadaLink.CLI;
|
||||
using ScadaLink.CLI.Commands;
|
||||
using ScadaLink.Commons.Messages.Management;
|
||||
|
||||
namespace ScadaLink.CLI.Tests;
|
||||
|
||||
public class CommandHelpersTests
|
||||
{
|
||||
[Fact]
|
||||
public void HandleResponse_ManagementSuccess_JsonFormat_ReturnsZero()
|
||||
public void HandleResponse_Success_JsonFormat_ReturnsZero()
|
||||
{
|
||||
var writer = new StringWriter();
|
||||
Console.SetOut(writer);
|
||||
|
||||
var response = new ManagementSuccess("corr-1", "{\"id\":1,\"name\":\"test\"}");
|
||||
var response = new ManagementResponse(200, "{\"id\":1,\"name\":\"test\"}", null, null);
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "json");
|
||||
|
||||
Assert.Equal(0, exitCode);
|
||||
@@ -21,13 +21,13 @@ public class CommandHelpersTests
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_ManagementSuccess_TableFormat_ArrayData_ReturnsZero()
|
||||
public void HandleResponse_Success_TableFormat_ArrayData_ReturnsZero()
|
||||
{
|
||||
var writer = new StringWriter();
|
||||
Console.SetOut(writer);
|
||||
|
||||
var json = "[{\"Id\":1,\"Name\":\"Alpha\"},{\"Id\":2,\"Name\":\"Beta\"}]";
|
||||
var response = new ManagementSuccess("corr-1", json);
|
||||
var response = new ManagementResponse(200, json, null, null);
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "table");
|
||||
|
||||
Assert.Equal(0, exitCode);
|
||||
@@ -41,13 +41,13 @@ public class CommandHelpersTests
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_ManagementSuccess_TableFormat_ObjectData_ReturnsZero()
|
||||
public void HandleResponse_Success_TableFormat_ObjectData_ReturnsZero()
|
||||
{
|
||||
var writer = new StringWriter();
|
||||
Console.SetOut(writer);
|
||||
|
||||
var json = "{\"Id\":1,\"Name\":\"Alpha\",\"Status\":\"Active\"}";
|
||||
var response = new ManagementSuccess("corr-1", json);
|
||||
var response = new ManagementResponse(200, json, null, null);
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "table");
|
||||
|
||||
Assert.Equal(0, exitCode);
|
||||
@@ -61,12 +61,12 @@ public class CommandHelpersTests
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_ManagementSuccess_TableFormat_EmptyArray_ShowsNoResults()
|
||||
public void HandleResponse_Success_TableFormat_EmptyArray_ShowsNoResults()
|
||||
{
|
||||
var writer = new StringWriter();
|
||||
Console.SetOut(writer);
|
||||
|
||||
var response = new ManagementSuccess("corr-1", "[]");
|
||||
var response = new ManagementResponse(200, "[]", null, null);
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "table");
|
||||
|
||||
Assert.Equal(0, exitCode);
|
||||
@@ -76,12 +76,12 @@ public class CommandHelpersTests
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_ManagementError_ReturnsOne()
|
||||
public void HandleResponse_Error_ReturnsOne()
|
||||
{
|
||||
var errWriter = new StringWriter();
|
||||
Console.SetError(errWriter);
|
||||
|
||||
var response = new ManagementError("corr-1", "Something failed", "FAIL_CODE");
|
||||
var response = new ManagementResponse(400, null, "Something failed", "FAIL_CODE");
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "json");
|
||||
|
||||
Assert.Equal(1, exitCode);
|
||||
@@ -91,12 +91,12 @@ public class CommandHelpersTests
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_ManagementUnauthorized_ReturnsTwo()
|
||||
public void HandleResponse_Unauthorized_ReturnsTwo()
|
||||
{
|
||||
var errWriter = new StringWriter();
|
||||
Console.SetError(errWriter);
|
||||
|
||||
var response = new ManagementUnauthorized("corr-1", "Access denied");
|
||||
var response = new ManagementResponse(403, null, "Access denied", "UNAUTHORIZED");
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "json");
|
||||
|
||||
Assert.Equal(2, exitCode);
|
||||
@@ -106,26 +106,47 @@ public class CommandHelpersTests
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_UnexpectedType_ReturnsOne()
|
||||
public void HandleResponse_AuthFailure_ReturnsOne()
|
||||
{
|
||||
var errWriter = new StringWriter();
|
||||
Console.SetError(errWriter);
|
||||
|
||||
var exitCode = CommandHelpers.HandleResponse("unexpected", "json");
|
||||
var response = new ManagementResponse(401, null, "Invalid credentials", "AUTH_FAILED");
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "json");
|
||||
|
||||
Assert.Equal(1, exitCode);
|
||||
Assert.Contains("Unexpected response type", errWriter.ToString());
|
||||
Assert.Contains("Invalid credentials", errWriter.ToString());
|
||||
|
||||
Console.SetError(new StreamWriter(Console.OpenStandardError()) { AutoFlush = true });
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void NewCorrelationId_ReturnsNonEmpty32CharHex()
|
||||
public void HandleResponse_ConnectionFailure_ReturnsOne()
|
||||
{
|
||||
var id = CommandHelpers.NewCorrelationId();
|
||||
var errWriter = new StringWriter();
|
||||
Console.SetError(errWriter);
|
||||
|
||||
Assert.NotNull(id);
|
||||
Assert.Equal(32, id.Length);
|
||||
Assert.True(id.All(c => "0123456789abcdef".Contains(c)));
|
||||
var response = new ManagementResponse(0, null, "Connection failed: No such host", "CONNECTION_FAILED");
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "json");
|
||||
|
||||
Assert.Equal(1, exitCode);
|
||||
Assert.Contains("Connection failed", errWriter.ToString());
|
||||
|
||||
Console.SetError(new StreamWriter(Console.OpenStandardError()) { AutoFlush = true });
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void HandleResponse_Timeout_ReturnsOne()
|
||||
{
|
||||
var errWriter = new StringWriter();
|
||||
Console.SetError(errWriter);
|
||||
|
||||
var response = new ManagementResponse(504, null, "Request timed out.", "TIMEOUT");
|
||||
var exitCode = CommandHelpers.HandleResponse(response, "json");
|
||||
|
||||
Assert.Equal(1, exitCode);
|
||||
Assert.Contains("timed out", errWriter.ToString());
|
||||
|
||||
Console.SetError(new StreamWriter(Console.OpenStandardError()) { AutoFlush = true });
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user