fix(cli): resolve CLI-001 — honor SCADALINK_FORMAT and config-file format precedence
This commit is contained in:
@@ -8,7 +8,7 @@
|
|||||||
| Last reviewed | 2026-05-16 |
|
| Last reviewed | 2026-05-16 |
|
||||||
| Reviewer | claude-agent |
|
| Reviewer | claude-agent |
|
||||||
| Commit reviewed | `9c60592` |
|
| Commit reviewed | `9c60592` |
|
||||||
| Open findings | 13 |
|
| Open findings | 12 |
|
||||||
|
|
||||||
## Summary
|
## Summary
|
||||||
|
|
||||||
@@ -54,7 +54,7 @@ argument parsing, and the command-tree wiring are untested.
|
|||||||
|--|--|
|
|--|--|
|
||||||
| Severity | High |
|
| Severity | High |
|
||||||
| Category | Correctness & logic bugs |
|
| Category | Correctness & logic bugs |
|
||||||
| Status | Open |
|
| Status | Resolved |
|
||||||
| Location | `src/ScadaLink.CLI/Commands/CommandHelpers.cs:18`, `src/ScadaLink.CLI/Commands/DebugCommands.cs:45`, `src/ScadaLink.CLI/CliConfig.cs:37-39` |
|
| Location | `src/ScadaLink.CLI/Commands/CommandHelpers.cs:18`, `src/ScadaLink.CLI/Commands/DebugCommands.cs:45`, `src/ScadaLink.CLI/CliConfig.cs:37-39` |
|
||||||
|
|
||||||
**Description**
|
**Description**
|
||||||
@@ -79,7 +79,12 @@ only then override the config value. Apply the same fix to `DebugCommands.BuildS
|
|||||||
|
|
||||||
**Resolution**
|
**Resolution**
|
||||||
|
|
||||||
_Unresolved._
|
Resolved 2026-05-16 (commit `<pending>`). Removed the `--format` option's
|
||||||
|
`DefaultValueFactory` in `Program.cs` and added `CommandHelpers.ResolveFormat`, which uses
|
||||||
|
`ParseResult.GetResult(formatOption)` to detect an explicitly supplied flag and resolves
|
||||||
|
precedence explicitly: explicit `--format` → `CliConfig.DefaultFormat` (env var / config
|
||||||
|
file) → `"json"`. Both `CommandHelpers.ExecuteCommandAsync` and `DebugCommands.BuildStream`
|
||||||
|
now call `ResolveFormat`. Regression tests added in `FormatResolutionTests`.
|
||||||
|
|
||||||
### CLI-002 — Empty success body crashes table rendering with an unhandled exception
|
### CLI-002 — Empty success body crashes table rendering with an unhandled exception
|
||||||
|
|
||||||
|
|||||||
@@ -15,8 +15,8 @@ internal static class CommandHelpers
|
|||||||
Option<string> passwordOption,
|
Option<string> passwordOption,
|
||||||
object command)
|
object command)
|
||||||
{
|
{
|
||||||
var format = result.GetValue(formatOption) ?? "json";
|
|
||||||
var config = CliConfig.Load();
|
var config = CliConfig.Load();
|
||||||
|
var format = ResolveFormat(result, formatOption, config);
|
||||||
|
|
||||||
// Resolve management URL
|
// Resolve management URL
|
||||||
var url = result.GetValue(urlOption);
|
var url = result.GetValue(urlOption);
|
||||||
@@ -53,6 +53,27 @@ internal static class CommandHelpers
|
|||||||
return HandleResponse(response, format);
|
return HandleResponse(response, format);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Resolves the output format using the documented precedence chain:
|
||||||
|
/// an explicitly supplied <c>--format</c> option wins, otherwise the
|
||||||
|
/// config-file / environment-variable default (<see cref="CliConfig.DefaultFormat"/>)
|
||||||
|
/// is used, otherwise <c>json</c>. The <c>--format</c> option must not declare a
|
||||||
|
/// <c>DefaultValueFactory</c> — that would mask whether the flag was supplied.
|
||||||
|
/// </summary>
|
||||||
|
internal static string ResolveFormat(ParseResult result, Option<string> formatOption, CliConfig config)
|
||||||
|
{
|
||||||
|
// GetResult returns non-null only when the option was actually present on the
|
||||||
|
// command line, letting an explicit --format override the config default.
|
||||||
|
if (result.GetResult(formatOption) != null)
|
||||||
|
{
|
||||||
|
var explicitValue = result.GetValue(formatOption);
|
||||||
|
if (!string.IsNullOrWhiteSpace(explicitValue))
|
||||||
|
return explicitValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
return string.IsNullOrWhiteSpace(config.DefaultFormat) ? "json" : config.DefaultFormat;
|
||||||
|
}
|
||||||
|
|
||||||
internal static int HandleResponse(ManagementResponse response, string format)
|
internal static int HandleResponse(ManagementResponse response, string format)
|
||||||
{
|
{
|
||||||
if (response.JsonData != null)
|
if (response.JsonData != null)
|
||||||
|
|||||||
@@ -42,8 +42,8 @@ public static class DebugCommands
|
|||||||
cmd.SetAction(async (ParseResult result) =>
|
cmd.SetAction(async (ParseResult result) =>
|
||||||
{
|
{
|
||||||
var instanceId = result.GetValue(idOption);
|
var instanceId = result.GetValue(idOption);
|
||||||
var format = result.GetValue(formatOption) ?? "json";
|
|
||||||
var config = CliConfig.Load();
|
var config = CliConfig.Load();
|
||||||
|
var format = CommandHelpers.ResolveFormat(result, formatOption, config);
|
||||||
|
|
||||||
var url = result.GetValue(urlOption);
|
var url = result.GetValue(urlOption);
|
||||||
if (string.IsNullOrWhiteSpace(url))
|
if (string.IsNullOrWhiteSpace(url))
|
||||||
|
|||||||
@@ -7,8 +7,9 @@ var rootCommand = new RootCommand("ScadaLink CLI — manage the ScadaLink SCADA
|
|||||||
var urlOption = new Option<string>("--url") { Description = "Management API URL", Recursive = true };
|
var urlOption = new Option<string>("--url") { Description = "Management API URL", Recursive = true };
|
||||||
var usernameOption = new Option<string>("--username") { Description = "LDAP username", Recursive = true };
|
var usernameOption = new Option<string>("--username") { Description = "LDAP username", Recursive = true };
|
||||||
var passwordOption = new Option<string>("--password") { Description = "LDAP password", Recursive = true };
|
var passwordOption = new Option<string>("--password") { Description = "LDAP password", Recursive = true };
|
||||||
|
// No DefaultValueFactory: format precedence (explicit --format -> config/env -> "json")
|
||||||
|
// is resolved by CommandHelpers.ResolveFormat, which needs to distinguish an absent flag.
|
||||||
var formatOption = new Option<string>("--format") { Description = "Output format (json or table)", Recursive = true };
|
var formatOption = new Option<string>("--format") { Description = "Output format (json or table)", Recursive = true };
|
||||||
formatOption.DefaultValueFactory = _ => "json";
|
|
||||||
|
|
||||||
rootCommand.Add(urlOption);
|
rootCommand.Add(urlOption);
|
||||||
rootCommand.Add(usernameOption);
|
rootCommand.Add(usernameOption);
|
||||||
|
|||||||
54
tests/ScadaLink.CLI.Tests/FormatResolutionTests.cs
Normal file
54
tests/ScadaLink.CLI.Tests/FormatResolutionTests.cs
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
using System.CommandLine;
|
||||||
|
using ScadaLink.CLI;
|
||||||
|
using ScadaLink.CLI.Commands;
|
||||||
|
|
||||||
|
namespace ScadaLink.CLI.Tests;
|
||||||
|
|
||||||
|
public class FormatResolutionTests
|
||||||
|
{
|
||||||
|
private static (Option<string> formatOption, RootCommand root) BuildHarness()
|
||||||
|
{
|
||||||
|
var formatOption = new Option<string>("--format") { Recursive = true };
|
||||||
|
var root = new RootCommand();
|
||||||
|
root.Add(formatOption);
|
||||||
|
return (formatOption, root);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ResolveFormat_ExplicitFlag_OverridesConfig()
|
||||||
|
{
|
||||||
|
var (formatOption, root) = BuildHarness();
|
||||||
|
var result = root.Parse(new[] { "--format", "table" });
|
||||||
|
var config = new CliConfig { DefaultFormat = "json" };
|
||||||
|
|
||||||
|
var format = CommandHelpers.ResolveFormat(result, formatOption, config);
|
||||||
|
|
||||||
|
Assert.Equal("table", format);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ResolveFormat_FlagAbsent_UsesConfigDefaultFormat()
|
||||||
|
{
|
||||||
|
// Regression for CLI-001: when --format is not supplied, the config-file /
|
||||||
|
// env-var DefaultFormat must be honoured instead of always falling back to "json".
|
||||||
|
var (formatOption, root) = BuildHarness();
|
||||||
|
var result = root.Parse(Array.Empty<string>());
|
||||||
|
var config = new CliConfig { DefaultFormat = "table" };
|
||||||
|
|
||||||
|
var format = CommandHelpers.ResolveFormat(result, formatOption, config);
|
||||||
|
|
||||||
|
Assert.Equal("table", format);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ResolveFormat_FlagAbsent_AndNoConfig_DefaultsToJson()
|
||||||
|
{
|
||||||
|
var (formatOption, root) = BuildHarness();
|
||||||
|
var result = root.Parse(Array.Empty<string>());
|
||||||
|
var config = new CliConfig { DefaultFormat = "json" };
|
||||||
|
|
||||||
|
var format = CommandHelpers.ResolveFormat(result, formatOption, config);
|
||||||
|
|
||||||
|
Assert.Equal("json", format);
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user