From 04610cf492296277c415dc6a48d19a43178bec12 Mon Sep 17 00:00:00 2001 From: Ilya Hryapko <35522908+ilyahryapko@users.noreply.github.com> Date: Thu, 9 Feb 2023 17:26:06 +0300 Subject: [PATCH] Alias for command branches (#411) --- .../ConfiguratorExtensions.cs | 16 +- .../IBranchConfigurator.cs | 14 + src/Spectre.Console.Cli/IConfigurator.cs | 3 +- src/Spectre.Console.Cli/IConfiguratorOfT.cs | 3 +- .../Configuration/BranchConfigurator.cs | 17 + .../Internal/Configuration/Configurator.cs | 10 +- .../Internal/Configuration/ConfiguratorOfT.cs | 10 +- .../Unsafe/IUnsafeConfigurator.cs | 3 +- .../Unit/CommandAppTests.cs | 401 ++++++++++-------- 9 files changed, 288 insertions(+), 189 deletions(-) create mode 100644 src/Spectre.Console.Cli/IBranchConfigurator.cs create mode 100644 src/Spectre.Console.Cli/Internal/Configuration/BranchConfigurator.cs diff --git a/src/Spectre.Console.Cli/ConfiguratorExtensions.cs b/src/Spectre.Console.Cli/ConfiguratorExtensions.cs index e4a234fc1..467930203 100644 --- a/src/Spectre.Console.Cli/ConfiguratorExtensions.cs +++ b/src/Spectre.Console.Cli/ConfiguratorExtensions.cs @@ -88,12 +88,12 @@ public static IConfigurator UseStrictParsing(this IConfigurator configurator) configurator.Settings.StrictParsing = true; return configurator; - } - + } + /// /// Tells the help writer whether or not to trim trailing period. /// - /// The configurator. + /// The configurator. /// True to trim trailing period (default), false to not. /// A configurator that can be used to configure the application further. public static IConfigurator TrimTrailingPeriods(this IConfigurator configurator, bool trimTrailingPeriods) @@ -181,7 +181,8 @@ public static IConfigurator SetInterceptor(this IConfigurator configurator, ICom /// The configurator. /// The name of the command branch. /// The command branch configuration. - public static void AddBranch( + /// A branch configurator that can be used to configure the branch further. + public static IBranchConfigurator AddBranch( this IConfigurator configurator, string name, Action> action) @@ -191,7 +192,7 @@ public static void AddBranch( throw new ArgumentNullException(nameof(configurator)); } - configurator.AddBranch(name, action); + return configurator.AddBranch(name, action); } /// @@ -201,7 +202,8 @@ public static void AddBranch( /// The configurator. /// The name of the command branch. /// The command branch configuration. - public static void AddBranch( + /// A branch configurator that can be used to configure the branch further. + public static IBranchConfigurator AddBranch( this IConfigurator configurator, string name, Action> action) @@ -212,7 +214,7 @@ public static void AddBranch( throw new ArgumentNullException(nameof(configurator)); } - configurator.AddBranch(name, action); + return configurator.AddBranch(name, action); } /// diff --git a/src/Spectre.Console.Cli/IBranchConfigurator.cs b/src/Spectre.Console.Cli/IBranchConfigurator.cs new file mode 100644 index 000000000..2b8a44bdf --- /dev/null +++ b/src/Spectre.Console.Cli/IBranchConfigurator.cs @@ -0,0 +1,14 @@ +namespace Spectre.Console.Cli; + +/// +/// Represents a branch configurator. +/// +public interface IBranchConfigurator +{ + /// + /// Adds an alias (an alternative name) to the branch being configured. + /// + /// The alias to add to the branch being configured. + /// The same instance so that multiple calls can be chained. + IBranchConfigurator WithAlias(string name); +} \ No newline at end of file diff --git a/src/Spectre.Console.Cli/IConfigurator.cs b/src/Spectre.Console.Cli/IConfigurator.cs index 565a4f322..f6108b654 100644 --- a/src/Spectre.Console.Cli/IConfigurator.cs +++ b/src/Spectre.Console.Cli/IConfigurator.cs @@ -41,6 +41,7 @@ ICommandConfigurator AddDelegate(string name, FuncThe command setting type. /// The name of the command branch. /// The command branch configurator. - void AddBranch(string name, Action> action) + /// A branch configurator that can be used to configure the branch further. + IBranchConfigurator AddBranch(string name, Action> action) where TSettings : CommandSettings; } \ No newline at end of file diff --git a/src/Spectre.Console.Cli/IConfiguratorOfT.cs b/src/Spectre.Console.Cli/IConfiguratorOfT.cs index 7852aafce..1dabf0c38 100644 --- a/src/Spectre.Console.Cli/IConfiguratorOfT.cs +++ b/src/Spectre.Console.Cli/IConfiguratorOfT.cs @@ -51,6 +51,7 @@ ICommandConfigurator AddDelegate(string name, FuncThe derived command setting type. /// The name of the command branch. /// The command branch configuration. - void AddBranch(string name, Action> action) + /// A branch configurator that can be used to configure the branch further. + IBranchConfigurator AddBranch(string name, Action> action) where TDerivedSettings : TSettings; } \ No newline at end of file diff --git a/src/Spectre.Console.Cli/Internal/Configuration/BranchConfigurator.cs b/src/Spectre.Console.Cli/Internal/Configuration/BranchConfigurator.cs new file mode 100644 index 000000000..ba26f088c --- /dev/null +++ b/src/Spectre.Console.Cli/Internal/Configuration/BranchConfigurator.cs @@ -0,0 +1,17 @@ +namespace Spectre.Console.Cli; + +internal sealed class BranchConfigurator : IBranchConfigurator +{ + public ConfiguredCommand Command { get; } + + public BranchConfigurator(ConfiguredCommand command) + { + Command = command; + } + + public IBranchConfigurator WithAlias(string alias) + { + Command.Aliases.Add(alias); + return this; + } +} \ No newline at end of file diff --git a/src/Spectre.Console.Cli/Internal/Configuration/Configurator.cs b/src/Spectre.Console.Cli/Internal/Configuration/Configurator.cs index cd1b42d77..e4e503633 100644 --- a/src/Spectre.Console.Cli/Internal/Configuration/Configurator.cs +++ b/src/Spectre.Console.Cli/Internal/Configuration/Configurator.cs @@ -48,12 +48,13 @@ public ICommandConfigurator AddDelegate(string name, Func(string name, Action> action) + public IBranchConfigurator AddBranch(string name, Action> action) where TSettings : CommandSettings { var command = ConfiguredCommand.FromBranch(name); action(new Configurator(command, _registrar)); - Commands.Add(command); + var added = Commands.AddAndReturn(command); + return new BranchConfigurator(added); } ICommandConfigurator IUnsafeConfigurator.AddCommand(string name, Type command) @@ -74,7 +75,7 @@ ICommandConfigurator IUnsafeConfigurator.AddCommand(string name, Type command) return result; } - void IUnsafeConfigurator.AddBranch(string name, Type settings, Action action) + IBranchConfigurator IUnsafeConfigurator.AddBranch(string name, Type settings, Action action) { var command = ConfiguredCommand.FromBranch(settings, name); @@ -86,6 +87,7 @@ void IUnsafeConfigurator.AddBranch(string name, Type settings, Action(string name, Func(string name, Action> action) + public IBranchConfigurator AddBranch(string name, Action> action) where TDerivedSettings : TSettings { var command = ConfiguredCommand.FromBranch(name); action(new Configurator(command, _registrar)); - _command.Children.Add(command); + var added = _command.Children.AddAndReturn(command); + return new BranchConfigurator(added); } ICommandConfigurator IUnsafeConfigurator.AddCommand(string name, Type command) @@ -73,7 +74,7 @@ ICommandConfigurator IUnsafeConfigurator.AddCommand(string name, Type command) return result; } - void IUnsafeConfigurator.AddBranch(string name, Type settings, Action action) + IBranchConfigurator IUnsafeConfigurator.AddBranch(string name, Type settings, Action action) { var command = ConfiguredCommand.FromBranch(settings, name); @@ -85,6 +86,7 @@ void IUnsafeConfigurator.AddBranch(string name, Type settings, ActionThe name of the command branch. /// The command setting type. /// The command branch configurator. - void AddBranch(string name, Type settings, Action action); + /// A branch configurator that can be used to configure the branch further. + IBranchConfigurator AddBranch(string name, Type settings, Action action); } \ No newline at end of file diff --git a/test/Spectre.Console.Cli.Tests/Unit/CommandAppTests.cs b/test/Spectre.Console.Cli.Tests/Unit/CommandAppTests.cs index 2f69a6bf9..65c152517 100644 --- a/test/Spectre.Console.Cli.Tests/Unit/CommandAppTests.cs +++ b/test/Spectre.Console.Cli.Tests/Unit/CommandAppTests.cs @@ -132,8 +132,8 @@ public void Should_Pass_Case_4() dog.IsAlive.ShouldBe(false); dog.Name.ShouldBe("Rufus"); }); - } - + } + [Fact] public void Should_Pass_Case_5() { @@ -165,7 +165,7 @@ public void Should_Pass_Case_5() dog.IsAlive.ShouldBe(true); dog.Name.ShouldBe("Rufus"); }); - } + } [Fact] public void Should_Pass_Case_6() @@ -237,10 +237,10 @@ public void Should_Preserve_Quotes_Hyphen_Delimiters_Spaces() var result = app.Run(new[] { "dog", "12", "4", - "--name=\" -Rufus --' ", - "--", - "--order-by", "\"-size\"", - "--order-by", " ", + "--name=\" -Rufus --' ", + "--", + "--order-by", "\"-size\"", + "--order-by", " ", "--order-by", string.Empty, }); @@ -249,8 +249,8 @@ public void Should_Preserve_Quotes_Hyphen_Delimiters_Spaces() result.Settings.ShouldBeOfType().And(dog => { dog.Name.ShouldBe("\" -Rufus --' "); - }); - result.Context.Remaining.Parsed.Count.ShouldBe(1); + }); + result.Context.Remaining.Parsed.Count.ShouldBe(1); result.Context.ShouldHaveRemainingArgument("order-by", values: new[] { "\"-size\"", " ", string.Empty }); } @@ -280,6 +280,65 @@ public void Should_Be_Able_To_Use_Command_Alias() }); } + [Fact] + public void Should_Be_Able_To_Use_Branch_Alias() + { + // Given + var app = new CommandAppTester(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddBranch("animal", animal => + { + animal.AddCommand("dog"); + animal.AddCommand("horse"); + }).WithAlias("a"); + }); + + // When + var result = app.Run(new[] + { + "a", "dog", "12", "--good-boy", + "--name", "Rufus", + }); + + // Then + result.ExitCode.ShouldBe(0); + result.Settings.ShouldBeOfType().And(dog => + { + dog.Age.ShouldBe(12); + dog.GoodBoy.ShouldBe(true); + dog.Name.ShouldBe("Rufus"); + dog.IsAlive.ShouldBe(false); + }); + } + + [Fact] + public void Should_Throw_If_Branch_Alias_Conflicts_With_Another_Command() + { + // Given + var app = new CommandApp(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddCommand("a"); + config.AddBranch("animal", animal => + { + animal.AddCommand("dog"); + animal.AddCommand("horse"); + }).WithAlias("a"); + }); + + // When + var result = Record.Exception(() => app.Run(new[] { "a", "0", "12" })); + + // Then + result.ShouldBeOfType().And(ex => + { + ex.Message.ShouldBe("The alias 'a' for 'animal' conflicts with another command."); + }); + } + [Fact] public void Should_Assign_Default_Value_To_Optional_Argument() { @@ -554,181 +613,181 @@ public void Should_Accept_Explicit_Boolan_Flag(string value, bool expected) { dog.IsAlive.ShouldBe(expected); }); - } - - [Fact] - public void Should_Set_Short_Option_Before_Argument() - { - // Given - var app = new CommandAppTester(); - app.Configure(config => - { - config.PropagateExceptions(); - config.AddCommand("dog"); - }); - - // When - var result = app.Run(new[] { "dog", "-a", "-n=Rufus", "4", "12", }); - - // Then - result.ExitCode.ShouldBe(0); - result.Settings.ShouldBeOfType().And(settings => - { - settings.IsAlive.ShouldBeTrue(); - settings.Name.ShouldBe("Rufus"); - settings.Legs.ShouldBe(4); - settings.Age.ShouldBe(12); - }); - } - + } + + [Fact] + public void Should_Set_Short_Option_Before_Argument() + { + // Given + var app = new CommandAppTester(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddCommand("dog"); + }); + + // When + var result = app.Run(new[] { "dog", "-a", "-n=Rufus", "4", "12", }); + + // Then + result.ExitCode.ShouldBe(0); + result.Settings.ShouldBeOfType().And(settings => + { + settings.IsAlive.ShouldBeTrue(); + settings.Name.ShouldBe("Rufus"); + settings.Legs.ShouldBe(4); + settings.Age.ShouldBe(12); + }); + } + [Theory] [InlineData("true", true)] [InlineData("True", true)] [InlineData("false", false)] - [InlineData("False", false)] - public void Should_Set_Short_Option_With_Explicit_Boolan_Flag_Before_Argument(string value, bool expected) - { - // Given - var app = new CommandAppTester(); - app.Configure(config => - { - config.PropagateExceptions(); - config.AddCommand("dog"); - }); - - // When - var result = app.Run(new[] { "dog", "-a", value, "4", "12", }); - - // Then - result.ExitCode.ShouldBe(0); - result.Settings.ShouldBeOfType().And(settings => - { - settings.IsAlive.ShouldBe(expected); - settings.Legs.ShouldBe(4); - settings.Age.ShouldBe(12); - }); - } - - [Fact] - public void Should_Set_Long_Option_Before_Argument() - { - // Given - var app = new CommandAppTester(); - app.Configure(config => - { - config.PropagateExceptions(); - config.AddCommand("dog"); - }); - - // When - var result = app.Run(new[] { "dog", "--alive", "--name=Rufus", "4", "12" }); - - // Then - result.ExitCode.ShouldBe(0); - result.Settings.ShouldBeOfType().And(settings => - { - settings.IsAlive.ShouldBeTrue(); - settings.Name.ShouldBe("Rufus"); - settings.Legs.ShouldBe(4); - settings.Age.ShouldBe(12); - }); - } - + [InlineData("False", false)] + public void Should_Set_Short_Option_With_Explicit_Boolan_Flag_Before_Argument(string value, bool expected) + { + // Given + var app = new CommandAppTester(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddCommand("dog"); + }); + + // When + var result = app.Run(new[] { "dog", "-a", value, "4", "12", }); + + // Then + result.ExitCode.ShouldBe(0); + result.Settings.ShouldBeOfType().And(settings => + { + settings.IsAlive.ShouldBe(expected); + settings.Legs.ShouldBe(4); + settings.Age.ShouldBe(12); + }); + } + + [Fact] + public void Should_Set_Long_Option_Before_Argument() + { + // Given + var app = new CommandAppTester(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddCommand("dog"); + }); + + // When + var result = app.Run(new[] { "dog", "--alive", "--name=Rufus", "4", "12" }); + + // Then + result.ExitCode.ShouldBe(0); + result.Settings.ShouldBeOfType().And(settings => + { + settings.IsAlive.ShouldBeTrue(); + settings.Name.ShouldBe("Rufus"); + settings.Legs.ShouldBe(4); + settings.Age.ShouldBe(12); + }); + } + [Theory] [InlineData("true", true)] [InlineData("True", true)] [InlineData("false", false)] - [InlineData("False", false)] - public void Should_Set_Long_Option_With_Explicit_Boolan_Flag_Before_Argument(string value, bool expected) - { - // Given - var app = new CommandAppTester(); - app.Configure(config => - { - config.PropagateExceptions(); - config.AddCommand("dog"); - }); - - // When - var result = app.Run(new[] { "dog", "--alive", value, "4", "12", }); - - // Then - result.ExitCode.ShouldBe(0); - result.Settings.ShouldBeOfType().And(settings => - { - settings.IsAlive.ShouldBe(expected); - settings.Legs.ShouldBe(4); - settings.Age.ShouldBe(12); - }); - } - - [Theory] - + [InlineData("False", false)] + public void Should_Set_Long_Option_With_Explicit_Boolan_Flag_Before_Argument(string value, bool expected) + { + // Given + var app = new CommandAppTester(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddCommand("dog"); + }); + + // When + var result = app.Run(new[] { "dog", "--alive", value, "4", "12", }); + + // Then + result.ExitCode.ShouldBe(0); + result.Settings.ShouldBeOfType().And(settings => + { + settings.IsAlive.ShouldBe(expected); + settings.Legs.ShouldBe(4); + settings.Age.ShouldBe(12); + }); + } + + [Theory] + // Long options - [InlineData("dog --alive 4 12 --name Rufus", 4, 12, false, true, "Rufus")] - [InlineData("dog --alive=true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] - [InlineData("dog --alive:true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] - [InlineData("dog --alive --good-boy 4 12 --name Rufus", 4, 12, true, true, "Rufus")] - [InlineData("dog --alive=true --good-boy=true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] - [InlineData("dog --alive:true --good-boy:true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] - [InlineData("dog --alive --good-boy --name Rufus 4 12", 4, 12, true, true, "Rufus")] - [InlineData("dog --alive=true --good-boy=true --name Rufus 4 12", 4, 12, true, true, "Rufus")] - [InlineData("dog --alive:true --good-boy:true --name Rufus 4 12", 4, 12, true, true, "Rufus")] - - // Short options - [InlineData("dog -a 4 12 --name Rufus", 4, 12, false, true, "Rufus")] - [InlineData("dog -a=true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] - [InlineData("dog -a:true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] - [InlineData("dog -a --good-boy 4 12 --name Rufus", 4, 12, true, true, "Rufus")] - [InlineData("dog -a=true -g=true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] - [InlineData("dog -a:true -g:true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] - [InlineData("dog -a -g --name Rufus 4 12", 4, 12, true, true, "Rufus")] - [InlineData("dog -a=true -g=true --name Rufus 4 12", 4, 12, true, true, "Rufus")] - [InlineData("dog -a:true -g:true --name Rufus 4 12", 4, 12, true, true, "Rufus")] - - // Switch around ordering of the options - [InlineData("dog --good-boy:true --name Rufus --alive:true 4 12", 4, 12, true, true, "Rufus")] - [InlineData("dog --name Rufus --alive:true --good-boy:true 4 12", 4, 12, true, true, "Rufus")] - [InlineData("dog --name Rufus --good-boy:true --alive:true 4 12", 4, 12, true, true, "Rufus")] - - // Inject the command arguments in between the options - [InlineData("dog 4 12 --good-boy:true --name Rufus --alive:true", 4, 12, true, true, "Rufus")] - [InlineData("dog 4 --good-boy:true 12 --name Rufus --alive:true", 4, 12, true, true, "Rufus")] - [InlineData("dog --good-boy:true 4 12 --name Rufus --alive:true", 4, 12, true, true, "Rufus")] - [InlineData("dog --good-boy:true 4 --name Rufus 12 --alive:true", 4, 12, true, true, "Rufus")] - [InlineData("dog --name Rufus --alive:true 4 12 --good-boy:true", 4, 12, true, true, "Rufus")] - [InlineData("dog --name Rufus --alive:true 4 --good-boy:true 12", 4, 12, true, true, "Rufus")] - - // Inject the command arguments in between the options (all flag values set to false) - [InlineData("dog 4 12 --good-boy:false --name Rufus --alive:false", 4, 12, false, false, "Rufus")] - [InlineData("dog 4 --good-boy:false 12 --name Rufus --alive:false", 4, 12, false, false, "Rufus")] - [InlineData("dog --good-boy:false 4 12 --name Rufus --alive:false", 4, 12, false, false, "Rufus")] - [InlineData("dog --good-boy:false 4 --name Rufus 12 --alive:false", 4, 12, false, false, "Rufus")] - [InlineData("dog --name Rufus --alive:false 4 12 --good-boy:false", 4, 12, false, false, "Rufus")] - [InlineData("dog --name Rufus --alive:false 4 --good-boy:false 12", 4, 12, false, false, "Rufus")] - public void Should_Set_Option_Before_Argument(string arguments, int legs, int age, bool goodBoy, bool isAlive, string name) - { - // Given - var app = new CommandAppTester(); - app.Configure(config => - { - config.PropagateExceptions(); - config.AddCommand("dog"); - }); - - // When - var result = app.Run(arguments.Split(' ')); - - // Then - result.ExitCode.ShouldBe(0); - result.Settings.ShouldBeOfType().And(settings => - { + [InlineData("dog --alive 4 12 --name Rufus", 4, 12, false, true, "Rufus")] + [InlineData("dog --alive=true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] + [InlineData("dog --alive:true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] + [InlineData("dog --alive --good-boy 4 12 --name Rufus", 4, 12, true, true, "Rufus")] + [InlineData("dog --alive=true --good-boy=true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] + [InlineData("dog --alive:true --good-boy:true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] + [InlineData("dog --alive --good-boy --name Rufus 4 12", 4, 12, true, true, "Rufus")] + [InlineData("dog --alive=true --good-boy=true --name Rufus 4 12", 4, 12, true, true, "Rufus")] + [InlineData("dog --alive:true --good-boy:true --name Rufus 4 12", 4, 12, true, true, "Rufus")] + + // Short options + [InlineData("dog -a 4 12 --name Rufus", 4, 12, false, true, "Rufus")] + [InlineData("dog -a=true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] + [InlineData("dog -a:true 4 12 --name Rufus", 4, 12, false, true, "Rufus")] + [InlineData("dog -a --good-boy 4 12 --name Rufus", 4, 12, true, true, "Rufus")] + [InlineData("dog -a=true -g=true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] + [InlineData("dog -a:true -g:true 4 12 --name Rufus", 4, 12, true, true, "Rufus")] + [InlineData("dog -a -g --name Rufus 4 12", 4, 12, true, true, "Rufus")] + [InlineData("dog -a=true -g=true --name Rufus 4 12", 4, 12, true, true, "Rufus")] + [InlineData("dog -a:true -g:true --name Rufus 4 12", 4, 12, true, true, "Rufus")] + + // Switch around ordering of the options + [InlineData("dog --good-boy:true --name Rufus --alive:true 4 12", 4, 12, true, true, "Rufus")] + [InlineData("dog --name Rufus --alive:true --good-boy:true 4 12", 4, 12, true, true, "Rufus")] + [InlineData("dog --name Rufus --good-boy:true --alive:true 4 12", 4, 12, true, true, "Rufus")] + + // Inject the command arguments in between the options + [InlineData("dog 4 12 --good-boy:true --name Rufus --alive:true", 4, 12, true, true, "Rufus")] + [InlineData("dog 4 --good-boy:true 12 --name Rufus --alive:true", 4, 12, true, true, "Rufus")] + [InlineData("dog --good-boy:true 4 12 --name Rufus --alive:true", 4, 12, true, true, "Rufus")] + [InlineData("dog --good-boy:true 4 --name Rufus 12 --alive:true", 4, 12, true, true, "Rufus")] + [InlineData("dog --name Rufus --alive:true 4 12 --good-boy:true", 4, 12, true, true, "Rufus")] + [InlineData("dog --name Rufus --alive:true 4 --good-boy:true 12", 4, 12, true, true, "Rufus")] + + // Inject the command arguments in between the options (all flag values set to false) + [InlineData("dog 4 12 --good-boy:false --name Rufus --alive:false", 4, 12, false, false, "Rufus")] + [InlineData("dog 4 --good-boy:false 12 --name Rufus --alive:false", 4, 12, false, false, "Rufus")] + [InlineData("dog --good-boy:false 4 12 --name Rufus --alive:false", 4, 12, false, false, "Rufus")] + [InlineData("dog --good-boy:false 4 --name Rufus 12 --alive:false", 4, 12, false, false, "Rufus")] + [InlineData("dog --name Rufus --alive:false 4 12 --good-boy:false", 4, 12, false, false, "Rufus")] + [InlineData("dog --name Rufus --alive:false 4 --good-boy:false 12", 4, 12, false, false, "Rufus")] + public void Should_Set_Option_Before_Argument(string arguments, int legs, int age, bool goodBoy, bool isAlive, string name) + { + // Given + var app = new CommandAppTester(); + app.Configure(config => + { + config.PropagateExceptions(); + config.AddCommand("dog"); + }); + + // When + var result = app.Run(arguments.Split(' ')); + + // Then + result.ExitCode.ShouldBe(0); + result.Settings.ShouldBeOfType().And(settings => + { settings.Legs.ShouldBe(legs); settings.Age.ShouldBe(age); settings.GoodBoy.ShouldBe(goodBoy); settings.IsAlive.ShouldBe(isAlive); - settings.Name.ShouldBe(name); - }); + settings.Name.ShouldBe(name); + }); } [Fact]