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

feat: Add HiddenAttribute to hide specific command/parameter #171

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
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
7 changes: 7 additions & 0 deletions ReadMe.md
Original file line number Diff line number Diff line change
Expand Up @@ -697,6 +697,13 @@ The field secondArg must be between 0 and 2.

By default, the ExitCode is set to 1 in this case.

Hide command/parameter help
---
`ConsoleAppFramework` supports `HiddenAttribute` which is used to hide specific help for a command/parameter.

- When`HiddenAttribute` is set to command, it hides command from command list.
- When`HiddenAttribute` is set to parameter, it hides parameter from command help.

Filter(Middleware) Pipeline / ConsoleAppContext
---
Filters are provided as a mechanism to hook into the execution before and after. To use filters, define an `internal class` that implements `ConsoleAppFilter`.
Expand Down
2 changes: 2 additions & 0 deletions src/ConsoleAppFramework/Command.cs
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ public record class Command
{
public required bool IsAsync { get; init; } // Task or Task<int>
public required bool IsVoid { get; init; } // void or int
public required bool IsHidden { get; init; } // Hide help from command list

public bool IsRootCommand => Name == "";
public required string Name { get; init; }
Expand Down Expand Up @@ -153,6 +154,7 @@ public record class CommandParameter
public required IgnoreEquality<WellKnownTypes> WellKnownTypes { get; init; }
public required bool IsNullableReference { get; init; }
public required bool IsParams { get; init; }
public required bool IsHidden { get; init; } // Hide command parameter help
public required string Name { get; init; }
public required string OriginalParameterName { get; init; }
public required bool HasDefaultValue { get; init; }
Expand Down
15 changes: 10 additions & 5 deletions src/ConsoleAppFramework/CommandHelpBuilder.cs
Original file line number Diff line number Diff line change
Expand Up @@ -64,15 +64,15 @@ static string BuildHelpMessageCore(Command command, bool showCommandName, bool s
if (definition.Options.Any())
{
var hasArgument = definition.Options.Any(x => x.Index.HasValue);
var hasOptions = definition.Options.Any(x => !x.Index.HasValue);
var hasNoHiddenOptions = definition.Options.Any(x => !x.Index.HasValue && !x.IsHidden);

if (hasArgument)
{
sb.AppendLine();
sb.AppendLine(BuildArgumentsMessage(definition));
}

if (hasOptions)
if (hasNoHiddenOptions)
{
sb.AppendLine();
sb.AppendLine(BuildOptionsMessage(definition));
Expand Down Expand Up @@ -102,7 +102,7 @@ static string BuildUsageMessage(CommandHelpDefinition definition, bool showComma
sb.Append(" [arguments...]");
}

if (definition.Options.Any(x => !x.Index.HasValue))
if (definition.Options.Any(x => !x.Index.HasValue && !x.IsHidden))
{
sb.Append(" [options...]");
}
Expand Down Expand Up @@ -160,6 +160,7 @@ static string BuildOptionsMessage(CommandHelpDefinition definition)
{
var optionsFormatted = definition.Options
.Where(x => !x.Index.HasValue)
.Where(x => !x.IsHidden)
.Select(x => (Options: string.Join("|", x.Options) + (x.IsFlag ? string.Empty : $" {x.FormattedValueTypeName}{(x.IsParams ? "..." : "")}"), x.Description, x.IsRequired, x.IsFlag, x.DefaultValue))
.ToArray();

Expand Down Expand Up @@ -215,6 +216,7 @@ static string BuildOptionsMessage(CommandHelpDefinition definition)
static string BuildMethodListMessage(IEnumerable<Command> commands, out int maxWidth)
{
var formatted = commands
.Where(x => !x.IsHidden)
.Select(x =>
{
return (Command: x.Name, x.Description);
Expand Down Expand Up @@ -279,6 +281,7 @@ static CommandHelpDefinition CreateCommandHelpDefinition(Command descriptor)
var description = item.Description;
var isFlag = item.Type.SpecialType == Microsoft.CodeAnalysis.SpecialType.System_Boolean;
var isParams = item.IsParams;
var isHidden = item.IsHidden;

var defaultValue = default(string);
if (item.HasDefaultValue)
Expand All @@ -300,7 +303,7 @@ static CommandHelpDefinition CreateCommandHelpDefinition(Command descriptor)
}

var paramTypeName = item.ToTypeShortString();
parameterDefinitions.Add(new CommandOptionHelpDefinition(options.Distinct().ToArray(), description, paramTypeName, defaultValue, index, isFlag, isParams));
parameterDefinitions.Add(new CommandOptionHelpDefinition(options.Distinct().ToArray(), description, paramTypeName, defaultValue, index, isFlag, isParams, isHidden));
}

var commandName = descriptor.Name;
Expand Down Expand Up @@ -336,9 +339,10 @@ class CommandOptionHelpDefinition
public bool IsRequired => DefaultValue == null && !IsParams;
public bool IsFlag { get; }
public bool IsParams { get; }
public bool IsHidden { get; }
public string FormattedValueTypeName => "<" + ValueTypeName + ">";

public CommandOptionHelpDefinition(string[] options, string description, string valueTypeName, string? defaultValue, int? index, bool isFlag, bool isParams)
public CommandOptionHelpDefinition(string[] options, string description, string valueTypeName, string? defaultValue, int? index, bool isFlag, bool isParams, bool isHidden)
{
Options = options;
Description = description;
Expand All @@ -347,6 +351,7 @@ public CommandOptionHelpDefinition(string[] options, string description, string
Index = index;
IsFlag = isFlag;
IsParams = isParams;
IsHidden = isHidden;
}
}
}
5 changes: 5 additions & 0 deletions src/ConsoleAppFramework/ConsoleAppBaseCode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -122,6 +122,11 @@ public CommandAttribute(string command)
}
}

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method | AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
internal sealed class HiddenAttribute : Attribute
{
}

[AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false)]
internal sealed class RegisterCommandsAttribute : Attribute
{
Expand Down
13 changes: 12 additions & 1 deletion src/ConsoleAppFramework/Parser.cs
Original file line number Diff line number Diff line change
Expand Up @@ -283,7 +283,11 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
}
}

var hasParams = x.Modifiers.Any(x => x.IsKind(SyntaxKind.ParamsKeyword));
var hasParams = x.Modifiers.Any(x => x.IsKind(SyntaxKind.ParamsKeyword));

var isHidden = x.AttributeLists
.SelectMany(x => x.Attributes)
.Any(x => model.GetTypeInfo(x).Type?.Name == "HiddenAttribute");

var customParserType = x.AttributeLists.SelectMany(x => x.Attributes)
.Select(x =>
Expand Down Expand Up @@ -360,6 +364,7 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
IsNullableReference = isNullableReference,
IsConsoleAppContext = isConsoleAppContext,
IsParams = hasParams,
IsHidden = isHidden,
Type = new EquatableTypeSymbol(type.Type!),
Location = x.GetLocation(),
HasDefaultValue = hasDefault,
Expand All @@ -381,6 +386,7 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
Name = commandName,
IsAsync = isAsync,
IsVoid = isVoid,
IsHidden = false, // Anonymous lambda don't support attribute.
Parameters = parameters,
MethodKind = MethodKind.Lambda,
Description = "",
Expand Down Expand Up @@ -472,6 +478,8 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
return null;
}

var isHiddenCommand = methodSymbol.GetAttributes().Any(x => x.AttributeClass?.Name == "HiddenAttribute");

var methodFilters = methodSymbol.GetAttributes()
.Where(x => x.AttributeClass?.Name == "ConsoleAppFilterAttribute")
.Select(x =>
Expand Down Expand Up @@ -516,6 +524,7 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
var hasValidation = x.GetAttributes().Any(x => x.AttributeClass?.GetBaseTypes().Any(y => y.Name == "ValidationAttribute") ?? false);
var isCancellationToken = SymbolEqualityComparer.Default.Equals(x.Type, wellKnownTypes.CancellationToken);
var isConsoleAppContext = x.Type!.Name == "ConsoleAppContext";
var isHiddenParameter = x.GetAttributes().Any(x => x.AttributeClass?.Name == "HiddenAttribute");

string description = "";
string[] aliases = [];
Expand Down Expand Up @@ -547,6 +556,7 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
IsNullableReference = isNullableReference,
IsConsoleAppContext = isConsoleAppContext,
IsParams = x.IsParams,
IsHidden = isHiddenParameter,
Location = x.DeclaringSyntaxReferences[0].GetSyntax().GetLocation(),
Type = new EquatableTypeSymbol(x.Type),
HasDefaultValue = x.HasExplicitDefaultValue,
Expand All @@ -567,6 +577,7 @@ internal class Parser(ConsoleAppFrameworkGeneratorOptions generatorOptions, Diag
Name = commandName,
IsAsync = isAsync,
IsVoid = isVoid,
IsHidden = isHiddenCommand,
Parameters = parameters,
MethodKind = addressOf ? MethodKind.FunctionPointer : MethodKind.Method,
Description = summary,
Expand Down
121 changes: 121 additions & 0 deletions tests/ConsoleAppFramework.GeneratorTests/HiddenAttributeTest.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
namespace ConsoleAppFramework.GeneratorTests;

public class HiddenAtttributeTest(ITestOutputHelper output)
{
VerifyHelper verifier = new(output, "CAF");

[Fact]
public void VerifyHiddenOptions_Lambda()
{
var code =
"""
ConsoleApp.Run(args, (int x, [Hidden]int y) => { });
""";

// Verify Hidden options is not shown on command help.
verifier.Execute(code, args: "--help", expected:
"""
Usage: [options...] [-h|--help] [--version]

Options:
--x <int> (Required)

""");
}

[Fact]
public void VerifyHiddenCommands_Class()
{
var code =
"""
var builder = ConsoleApp.Create();
builder.Add<Commands>();
await builder.RunAsync(args);

public class Commands
{
[Hidden]
public void Command1() { Console.Write("command1"); }

public void Command2() { Console.Write("command2"); }

[Hidden]
public void Command3(int x, [Hidden]int y) { Console.Write($"command3: x={x} y={y}"); }
}
""";

// Verify hidden command is not shown on root help commands.
verifier.Execute(code, args: "--help", expected:
"""
Usage: [command] [-h|--help] [--version]

Commands:
command2

""");

// Verify Hidden command help is shown when explicitly specify command name.
verifier.Execute(code, args: "command1 --help", expected:
"""
Usage: command1 [-h|--help] [--version]

""");

verifier.Execute(code, args: "command2 --help", expected:
"""
Usage: command2 [-h|--help] [--version]

""");

verifier.Execute(code, args: "command3 --help", expected:
"""
Usage: command3 [options...] [-h|--help] [--version]

Options:
--x <int> (Required)

""");

// Verify commands involations
verifier.Execute(code, args: "command1", "command1");
verifier.Execute(code, args: "command2", "command2");
verifier.Execute(code, args: "command3 --x 1 --y 2", expected: "command3: x=1 y=2");
}

[Fact]
public void VerifyHiddenCommands_LocalFunctions()
{
var code =
"""
var builder = ConsoleApp.Create();

builder.Add("", () => { Console.Write("root"); });
builder.Add("command1", Command1);
builder.Add("command2", Command2);
builder.Add("command3", Command3);
builder.Run(args);

[Hidden]
static void Command1() { Console.Write("command1"); }

static void Command2() { Console.Write("command2"); }

[Hidden]
static void Command3(int x, [Hidden]int y) { Console.Write($"command3: x={x} y={y}"); }
""";

verifier.Execute(code, args: "--help", expected:
"""
Usage: [command] [-h|--help] [--version]

Commands:
command2

""");

// Verify commands can be invoked.
verifier.Execute(code, args: "command1", expected: "command1");
verifier.Execute(code, args: "command2", expected: "command2");
verifier.Execute(code, args: "command3 --x 1 --y 2", expected: "command3: x=1 y=2");
}
}