mirror of
				https://github.com/Tyrrrz/CliFx.git
				synced 2025-10-25 15:19:17 +00:00 
			
		
		
		
	Compare commits
	
		
			6 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9564cd5d30 | ||
|  | ed458c3980 | ||
|  | 25538f99db | ||
|  | 36436e7a4b | ||
|  | a6070332c9 | ||
|  | 25cbfdb4b8 | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -143,6 +143,7 @@ _TeamCity* | ||||
| _NCrunch_* | ||||
| .*crunch*.local.xml | ||||
| nCrunchTemp_* | ||||
| .ncrunchsolution | ||||
|  | ||||
| # MightyMoose | ||||
| *.mm.* | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								.screenshots/help.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								.screenshots/help.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 23 KiB | 
| @@ -2,8 +2,7 @@ | ||||
|  | ||||
|   <PropertyGroup> | ||||
|     <OutputType>Exe</OutputType> | ||||
|     <TargetFramework>netcoreapp2.2</TargetFramework> | ||||
|     <LangVersion>latest</LangVersion> | ||||
|     <TargetFramework>netcoreapp3.0</TargetFramework> | ||||
|   </PropertyGroup> | ||||
|  | ||||
|   <ItemGroup> | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -16,6 +17,6 @@ namespace CliFx.Benchmarks.Commands | ||||
|         [CommandOption("bool", 'b')] | ||||
|         public bool BoolOption { get; set; } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -2,8 +2,7 @@ | ||||
|  | ||||
|   <PropertyGroup> | ||||
|     <OutputType>Exe</OutputType> | ||||
|     <TargetFramework>netcoreapp2.2</TargetFramework> | ||||
|     <LangVersion>latest</LangVersion> | ||||
|     <TargetFramework>netcoreapp3.0</TargetFramework> | ||||
|   </PropertyGroup> | ||||
|  | ||||
|   <ItemGroup> | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Demo.Internal; | ||||
| @@ -31,7 +32,7 @@ namespace CliFx.Demo.Commands | ||||
|             _libraryService = libraryService; | ||||
|         } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             // To make the demo simpler, we will just generate random publish date and ISBN if they were not set | ||||
|             if (Published == default) | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Demo.Internal; | ||||
| using CliFx.Demo.Services; | ||||
| @@ -20,7 +21,7 @@ namespace CliFx.Demo.Commands | ||||
|             _libraryService = libraryService; | ||||
|         } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             var book = _libraryService.GetBook(Title); | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Demo.Internal; | ||||
| using CliFx.Demo.Services; | ||||
| @@ -16,7 +17,7 @@ namespace CliFx.Demo.Commands | ||||
|             _libraryService = libraryService; | ||||
|         } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             var library = _libraryService.GetLibrary(); | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Demo.Services; | ||||
| using CliFx.Exceptions; | ||||
| @@ -19,7 +20,7 @@ namespace CliFx.Demo.Commands | ||||
|             _libraryService = libraryService; | ||||
|         } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             var book = _libraryService.GetBook(Title); | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| using System; | ||||
| using NUnit.Framework; | ||||
| using System; | ||||
| using System.IO; | ||||
| using CliFx.Services; | ||||
| using CliFx.Tests.Stubs; | ||||
| using CliFx.Tests.TestCommands; | ||||
| using NUnit.Framework; | ||||
|  | ||||
| namespace CliFx.Tests | ||||
| { | ||||
| @@ -20,8 +21,8 @@ namespace CliFx.Tests | ||||
|             builder | ||||
|                 .AddCommand(typeof(HelloWorldDefaultCommand)) | ||||
|                 .AddCommandsFrom(typeof(HelloWorldDefaultCommand).Assembly) | ||||
|                 .AddCommands(new[] {typeof(HelloWorldDefaultCommand)}) | ||||
|                 .AddCommandsFrom(new[] {typeof(HelloWorldDefaultCommand).Assembly}) | ||||
|                 .AddCommands(new[] { typeof(HelloWorldDefaultCommand) }) | ||||
|                 .AddCommandsFrom(new[] { typeof(HelloWorldDefaultCommand).Assembly }) | ||||
|                 .AddCommandsFromThisAssembly() | ||||
|                 .AllowDebugMode() | ||||
|                 .AllowPreviewMode() | ||||
| @@ -30,8 +31,9 @@ namespace CliFx.Tests | ||||
|                 .UseVersionText("test") | ||||
|                 .UseDescription("test") | ||||
|                 .UseConsole(new VirtualConsole(TextWriter.Null)) | ||||
|                 .UseCommandFactory(schema => (ICommand) Activator.CreateInstance(schema.Type)) | ||||
|                 .UseCommandFactory(schema => (ICommand)Activator.CreateInstance(schema.Type)) | ||||
|                 .UseCommandOptionInputConverter(new CommandOptionInputConverter()) | ||||
|                 .UseEnvironmentVariablesProvider(new EnvironmentVariablesProviderStub()) | ||||
|                 .Build(); | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -1,11 +1,12 @@ | ||||
| using System; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.IO; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Services; | ||||
| using CliFx.Tests.Stubs; | ||||
| using CliFx.Tests.TestCommands; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
|  | ||||
| namespace CliFx.Tests | ||||
| { | ||||
| @@ -17,104 +18,104 @@ namespace CliFx.Tests | ||||
|         private static IEnumerable<TestCaseData> GetTestCases_RunAsync() | ||||
|         { | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(HelloWorldDefaultCommand)}, | ||||
|                 new[] { typeof(HelloWorldDefaultCommand) }, | ||||
|                 new string[0], | ||||
|                 "Hello world." | ||||
|             ); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"concat", "-i", "foo", "-i", "bar", "-s", " "}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "concat", "-i", "foo", "-i", "bar", "-s", " " }, | ||||
|                 "foo bar" | ||||
|             ); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"concat", "-i", "one", "two", "three", "-s", ", "}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "concat", "-i", "one", "two", "three", "-s", ", " }, | ||||
|                 "one, two, three" | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(DivideCommand)}, | ||||
|                 new[] {"div", "-D", "24", "-d", "8"}, | ||||
|                 new[] { typeof(DivideCommand) }, | ||||
|                 new[] { "div", "-D", "24", "-d", "8" }, | ||||
|                 "3" | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(HelloWorldDefaultCommand)}, | ||||
|                 new[] {"--version"}, | ||||
|                 new[] { typeof(HelloWorldDefaultCommand) }, | ||||
|                 new[] { "--version" }, | ||||
|                 TestVersionText | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"--version"}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "--version" }, | ||||
|                 TestVersionText | ||||
|             ); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(HelloWorldDefaultCommand)}, | ||||
|                 new[] {"-h"}, | ||||
|                 new[] { typeof(HelloWorldDefaultCommand) }, | ||||
|                 new[] { "-h" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(HelloWorldDefaultCommand)}, | ||||
|                 new[] {"--help"}, | ||||
|                 new[] { typeof(HelloWorldDefaultCommand) }, | ||||
|                 new[] { "--help" }, | ||||
|                 null | ||||
|             ); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new string[0], | ||||
|                 null | ||||
|             ); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"-h"}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "-h" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"--help"}, | ||||
|                 null | ||||
|             ); | ||||
|              | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"concat", "-h"}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "--help" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ExceptionCommand)}, | ||||
|                 new[] {"exc", "-h"}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "concat", "-h" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(CommandExceptionCommand)}, | ||||
|                 new[] {"exc", "-h"}, | ||||
|                 new[] { typeof(ExceptionCommand) }, | ||||
|                 new[] { "exc", "-h" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"[preview]"}, | ||||
|                 new[] { typeof(CommandExceptionCommand) }, | ||||
|                 new[] { "exc", "-h" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ExceptionCommand)}, | ||||
|                 new[] {"exc", "[preview]"}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "[preview]" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"concat", "[preview]", "-o", "value"}, | ||||
|                 new[] { typeof(ExceptionCommand) }, | ||||
|                 new[] { "exc", "[preview]" }, | ||||
|                 null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "concat", "[preview]", "-o", "value" }, | ||||
|                 null | ||||
|             ); | ||||
|         } | ||||
| @@ -128,38 +129,38 @@ namespace CliFx.Tests | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ConcatCommand)}, | ||||
|                 new[] {"non-existing"}, | ||||
|                 new[] { typeof(ConcatCommand) }, | ||||
|                 new[] { "non-existing" }, | ||||
|                 null, null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(ExceptionCommand)}, | ||||
|                 new[] {"exc"}, | ||||
|                 new[] { typeof(ExceptionCommand) }, | ||||
|                 new[] { "exc" }, | ||||
|                 null, null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(CommandExceptionCommand)}, | ||||
|                 new[] {"exc"}, | ||||
|                 new[] { typeof(CommandExceptionCommand) }, | ||||
|                 new[] { "exc" }, | ||||
|                 null, null | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(CommandExceptionCommand)}, | ||||
|                 new[] {"exc"}, | ||||
|                 new[] { typeof(CommandExceptionCommand) }, | ||||
|                 new[] { "exc" }, | ||||
|                 null, null | ||||
|             ); | ||||
|           | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(CommandExceptionCommand)}, | ||||
|                 new[] {"exc", "-m", "foo bar"}, | ||||
|                 new[] { typeof(CommandExceptionCommand) }, | ||||
|                 new[] { "exc", "-m", "foo bar" }, | ||||
|                 "foo bar", null | ||||
|             ); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(CommandExceptionCommand)}, | ||||
|                 new[] {"exc", "-m", "foo bar", "-c", "666"}, | ||||
|                 new[] { typeof(CommandExceptionCommand) }, | ||||
|                 new[] { "exc", "-m", "foo bar", "-c", "666" }, | ||||
|                 "foo bar", 666 | ||||
|             ); | ||||
|         } | ||||
| @@ -173,11 +174,13 @@ namespace CliFx.Tests | ||||
|             using (var stdoutStream = new StringWriter()) | ||||
|             { | ||||
|                 var console = new VirtualConsole(stdoutStream); | ||||
|                 var environmentVariablesProvider = new EnvironmentVariablesProviderStub(); | ||||
|  | ||||
|                 var application = new CliApplicationBuilder() | ||||
|                     .AddCommands(commandTypes) | ||||
|                     .UseVersionText(TestVersionText) | ||||
|                     .UseConsole(console) | ||||
|                     .UseEnvironmentVariablesProvider(environmentVariablesProvider) | ||||
|                     .Build(); | ||||
|  | ||||
|                 // Act | ||||
| @@ -203,10 +206,12 @@ namespace CliFx.Tests | ||||
|             using (var stderrStream = new StringWriter()) | ||||
|             { | ||||
|                 var console = new VirtualConsole(TextWriter.Null, stderrStream); | ||||
|                 var environmentVariablesProvider = new EnvironmentVariablesProviderStub(); | ||||
|  | ||||
|                 var application = new CliApplicationBuilder() | ||||
|                     .AddCommands(commandTypes) | ||||
|                     .UseVersionText(TestVersionText) | ||||
|                     .UseEnvironmentVariablesProvider(environmentVariablesProvider) | ||||
|                     .UseConsole(console) | ||||
|                     .Build(); | ||||
|  | ||||
| @@ -219,12 +224,38 @@ namespace CliFx.Tests | ||||
|                     exitCode.Should().Be(expectedExitCode); | ||||
|                 else | ||||
|                     exitCode.Should().NotBe(0); | ||||
|                  | ||||
|  | ||||
|                 if (expectedStdErr != null) | ||||
|                     stderr.Should().Be(expectedStdErr); | ||||
|                 else | ||||
|                     stderr.Should().NotBeNullOrWhiteSpace(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         [Test] | ||||
|         public async Task RunAsync_Cancellation_Test() | ||||
|         { | ||||
|             // Arrange | ||||
|             using (var stdoutStream = new StringWriter()) | ||||
|             { | ||||
|                 var console = new VirtualConsole(stdoutStream); | ||||
|                  | ||||
|                 var application = new CliApplicationBuilder() | ||||
|                     .AddCommand(typeof(CancellableCommand)) | ||||
|                     .UseConsole(console) | ||||
|                     .Build(); | ||||
|                 var args = new[] { "cancel" }; | ||||
|  | ||||
|                 // Act | ||||
|                 var runTask = application.RunAsync(args); | ||||
|                 console.Cancel(); | ||||
|                 var exitCode = await runTask.ConfigureAwait(false); | ||||
|                 var stdOut = stdoutStream.ToString().Trim(); | ||||
|  | ||||
|                 // Assert | ||||
|                 exitCode.Should().Be(-2146233029); | ||||
|                 stdOut.Should().Be("Printed"); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -7,7 +7,6 @@ | ||||
|     <CollectCoverage>true</CollectCoverage> | ||||
|     <CoverletOutputFormat>opencover</CoverletOutputFormat> | ||||
|     <CoverletOutput>bin/$(Configuration)/Coverage.xml</CoverletOutput> | ||||
|     <LangVersion>latest</LangVersion> | ||||
|   </PropertyGroup> | ||||
|  | ||||
|   <ItemGroup> | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| using System; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Linq; | ||||
| using CliFx.Exceptions; | ||||
| using CliFx.Models; | ||||
| using CliFx.Services; | ||||
| using CliFx.Tests.TestCommands; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
| using CliFx.Tests.Stubs; | ||||
|  | ||||
| namespace CliFx.Tests.Services | ||||
| { | ||||
| @@ -14,7 +15,7 @@ namespace CliFx.Tests.Services | ||||
|     public class CommandInitializerTests | ||||
|     { | ||||
|         private static CommandSchema GetCommandSchema(Type commandType) => | ||||
|             new CommandSchemaResolver().GetCommandSchemas(new[] {commandType}).Single(); | ||||
|             new CommandSchemaResolver().GetCommandSchemas(new[] { commandType }).Single(); | ||||
|  | ||||
|         private static IEnumerable<TestCaseData> GetTestCases_InitializeCommand() | ||||
|         { | ||||
| @@ -26,7 +27,7 @@ namespace CliFx.Tests.Services | ||||
|                     new CommandOptionInput("dividend", "13"), | ||||
|                     new CommandOptionInput("divisor", "8") | ||||
|                 }), | ||||
|                 new DivideCommand {Dividend = 13, Divisor = 8} | ||||
|                 new DivideCommand { Dividend = 13, Divisor = 8 } | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
| @@ -37,7 +38,7 @@ namespace CliFx.Tests.Services | ||||
|                     new CommandOptionInput("dividend", "13"), | ||||
|                     new CommandOptionInput("d", "8") | ||||
|                 }), | ||||
|                 new DivideCommand {Dividend = 13, Divisor = 8} | ||||
|                 new DivideCommand { Dividend = 13, Divisor = 8 } | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
| @@ -48,7 +49,7 @@ namespace CliFx.Tests.Services | ||||
|                     new CommandOptionInput("D", "13"), | ||||
|                     new CommandOptionInput("d", "8") | ||||
|                 }), | ||||
|                 new DivideCommand {Dividend = 13, Divisor = 8} | ||||
|                 new DivideCommand { Dividend = 13, Divisor = 8 } | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
| @@ -58,7 +59,7 @@ namespace CliFx.Tests.Services | ||||
|                 { | ||||
|                     new CommandOptionInput("i", new[] {"foo", " ", "bar"}) | ||||
|                 }), | ||||
|                 new ConcatCommand {Inputs = new[] {"foo", " ", "bar"}} | ||||
|                 new ConcatCommand { Inputs = new[] { "foo", " ", "bar" } } | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
| @@ -69,7 +70,43 @@ namespace CliFx.Tests.Services | ||||
|                     new CommandOptionInput("i", new[] {"foo", "bar"}), | ||||
|                     new CommandOptionInput("s", " ") | ||||
|                 }), | ||||
|                 new ConcatCommand {Inputs = new[] {"foo", "bar"}, Separator = " "} | ||||
|                 new ConcatCommand { Inputs = new[] { "foo", "bar" }, Separator = " " } | ||||
|             ); | ||||
|  | ||||
|             //Will read a value from environment variables because none is supplied via CommandInput | ||||
|             yield return new TestCaseData( | ||||
|                 new EnvironmentVariableCommand(), | ||||
|                 GetCommandSchema(typeof(EnvironmentVariableCommand)), | ||||
|                 new CommandInput(null, new CommandOptionInput[0], EnvironmentVariablesProviderStub.EnvironmentVariables), | ||||
|                 new EnvironmentVariableCommand { Option = "A" } | ||||
|             ); | ||||
|  | ||||
|             //Will read multiple values from environment variables because none is supplied via CommandInput | ||||
|             yield return new TestCaseData( | ||||
|                 new EnvironmentVariableWithMultipleValuesCommand(), | ||||
|                 GetCommandSchema(typeof(EnvironmentVariableWithMultipleValuesCommand)), | ||||
|                 new CommandInput(null, new CommandOptionInput[0], EnvironmentVariablesProviderStub.EnvironmentVariables), | ||||
|                 new EnvironmentVariableWithMultipleValuesCommand { Option = new[] { "A", "B", "C" } } | ||||
|             ); | ||||
|  | ||||
|             //Will not read a value from environment variables because one is supplied via CommandInput | ||||
|             yield return new TestCaseData( | ||||
|                 new EnvironmentVariableCommand(), | ||||
|                 GetCommandSchema(typeof(EnvironmentVariableCommand)), | ||||
|                 new CommandInput(null, new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("opt", new[] { "X" }) | ||||
|                 }, | ||||
|                 EnvironmentVariablesProviderStub.EnvironmentVariables), | ||||
|                 new EnvironmentVariableCommand { Option = "X" } | ||||
|             ); | ||||
|  | ||||
|             //Will not split environment variable values because underlying property is not a collection | ||||
|             yield return new TestCaseData( | ||||
|                 new EnvironmentVariableWithoutCollectionPropertyCommand(), | ||||
|                 GetCommandSchema(typeof(EnvironmentVariableWithoutCollectionPropertyCommand)), | ||||
|                 new CommandInput(null, new CommandOptionInput[0], EnvironmentVariablesProviderStub.EnvironmentVariables), | ||||
|                 new EnvironmentVariableWithoutCollectionPropertyCommand { Option = "A;B;C;" } | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| using System.Collections.Generic; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
| using System.Collections.Generic; | ||||
| using CliFx.Models; | ||||
| using CliFx.Services; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
| using CliFx.Tests.Stubs; | ||||
|  | ||||
| namespace CliFx.Tests.Services | ||||
| { | ||||
| @@ -11,203 +12,238 @@ namespace CliFx.Tests.Services | ||||
|     { | ||||
|         private static IEnumerable<TestCaseData> GetTestCases_ParseCommandInput() | ||||
|         { | ||||
|             yield return new TestCaseData(new string[0], CommandInput.Empty); | ||||
|             yield return new TestCaseData(new string[0], CommandInput.Empty, new EmptyEnvironmentVariablesProviderStub()); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--option", "value"}, | ||||
|                 new[] { "--option", "value" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option", "value") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--option1", "value1", "--option2", "value2"}, | ||||
|                 new[] { "--option1", "value1", "--option2", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option1", "value1"), | ||||
|                     new CommandOptionInput("option2", "value2") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--option", "value1", "value2"}, | ||||
|                 new[] { "--option", "value1", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option", new[] {"value1", "value2"}) | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--option", "value1", "--option", "value2"}, | ||||
|                 new[] { "--option", "value1", "--option", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option", new[] {"value1", "value2"}) | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-a", "value"}, | ||||
|                 new[] { "-a", "value" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a", "value") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-a", "value1", "-b", "value2"}, | ||||
|                 new[] { "-a", "value1", "-b", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a", "value1"), | ||||
|                     new CommandOptionInput("b", "value2") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-a", "value1", "value2"}, | ||||
|                 new[] { "-a", "value1", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a", new[] {"value1", "value2"}) | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-a", "value1", "-a", "value2"}, | ||||
|                 new[] { "-a", "value1", "-a", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a", new[] {"value1", "value2"}) | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--option1", "value1", "-b", "value2"}, | ||||
|                 new[] { "--option1", "value1", "-b", "value2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option1", "value1"), | ||||
|                     new CommandOptionInput("b", "value2") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--switch"}, | ||||
|                 new[] { "--switch" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("switch") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"--switch1", "--switch2"}, | ||||
|                 new[] { "--switch1", "--switch2" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("switch1"), | ||||
|                     new CommandOptionInput("switch2") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-s"}, | ||||
|                 new[] { "-s" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("s") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-a", "-b"}, | ||||
|                 new[] { "-a", "-b" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a"), | ||||
|                     new CommandOptionInput("b") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-ab"}, | ||||
|                 new[] { "-ab" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a"), | ||||
|                     new CommandOptionInput("b") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"-ab", "value"}, | ||||
|                 new[] { "-ab", "value" }, | ||||
|                 new CommandInput(new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("a"), | ||||
|                     new CommandOptionInput("b", "value") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"command"}, | ||||
|                 new CommandInput("command") | ||||
|                 new[] { "command" }, | ||||
|                 new CommandInput("command"), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"command", "--option", "value"}, | ||||
|                 new[] { "command", "--option", "value" }, | ||||
|                 new CommandInput("command", new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option", "value") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"long", "command", "name"}, | ||||
|                 new CommandInput("long command name") | ||||
|                 new[] { "long", "command", "name" }, | ||||
|                 new CommandInput("long command name"), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"long", "command", "name", "--option", "value"}, | ||||
|                 new[] { "long", "command", "name", "--option", "value" }, | ||||
|                 new CommandInput("long command name", new[] | ||||
|                 { | ||||
|                     new CommandOptionInput("option", "value") | ||||
|                 }) | ||||
|                 }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"[debug]"}, | ||||
|                 new[] { "[debug]" }, | ||||
|                 new CommandInput(null, | ||||
|                     new[] {"debug"}, | ||||
|                     new CommandOptionInput[0]) | ||||
|                     new[] { "debug" }, | ||||
|                     new CommandOptionInput[0]), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"[debug]", "[preview]"}, | ||||
|                 new[] { "[debug]", "[preview]" }, | ||||
|                 new CommandInput(null, | ||||
|                     new[] {"debug", "preview"}, | ||||
|                     new CommandOptionInput[0]) | ||||
|                     new[] { "debug", "preview" }, | ||||
|                     new CommandOptionInput[0]), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"[debug]", "[preview]", "-o", "value"}, | ||||
|                 new[] { "[debug]", "[preview]", "-o", "value" }, | ||||
|                 new CommandInput(null, | ||||
|                     new[] {"debug", "preview"}, | ||||
|                     new[] { "debug", "preview" }, | ||||
|                     new[] | ||||
|                     { | ||||
|                         new CommandOptionInput("o", "value") | ||||
|                     }) | ||||
|                     }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {"command", "[debug]", "[preview]", "-o", "value"}, | ||||
|                 new[] { "command", "[debug]", "[preview]", "-o", "value" }, | ||||
|                 new CommandInput("command", | ||||
|                     new[] {"debug", "preview"}, | ||||
|                     new[] { "debug", "preview" }, | ||||
|                     new[] | ||||
|                     { | ||||
|                         new CommandOptionInput("o", "value") | ||||
|                     }) | ||||
|                     }), | ||||
|                 new EmptyEnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] { "command", "[debug]", "[preview]", "-o", "value" }, | ||||
|                 new CommandInput("command", | ||||
|                     new[] { "debug", "preview" }, | ||||
|                     new[] | ||||
|                     { | ||||
|                         new CommandOptionInput("o", "value") | ||||
|                     }, | ||||
|                     EnvironmentVariablesProviderStub.EnvironmentVariables), | ||||
|                 new EnvironmentVariablesProviderStub() | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         [Test] | ||||
|         [TestCaseSource(nameof(GetTestCases_ParseCommandInput))] | ||||
|         public void ParseCommandInput_Test(IReadOnlyList<string> commandLineArguments, | ||||
|             CommandInput expectedCommandInput) | ||||
|             CommandInput expectedCommandInput, IEnvironmentVariablesProvider environmentVariablesProvider) | ||||
|         { | ||||
|             // Arrange | ||||
|             var parser = new CommandInputParser(); | ||||
|             var parser = new CommandInputParser(environmentVariablesProvider); | ||||
|  | ||||
|             // Act | ||||
|             var commandInput = parser.ParseCommandInput(commandLineArguments); | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| using System; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using CliFx.Exceptions; | ||||
| using CliFx.Models; | ||||
| using CliFx.Services; | ||||
| using CliFx.Tests.TestCommands; | ||||
| using FluentAssertions; | ||||
| using NUnit.Framework; | ||||
|  | ||||
| namespace CliFx.Tests.Services | ||||
| { | ||||
| @@ -15,30 +15,37 @@ namespace CliFx.Tests.Services | ||||
|         private static IEnumerable<TestCaseData> GetTestCases_GetCommandSchemas() | ||||
|         { | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(DivideCommand), typeof(ConcatCommand)}, | ||||
|                 new[] { typeof(DivideCommand), typeof(ConcatCommand), typeof(EnvironmentVariableCommand) }, | ||||
|                 new[] | ||||
|                 { | ||||
|                     new CommandSchema(typeof(DivideCommand), "div", "Divide one number by another.", | ||||
|                         new[] | ||||
|                         { | ||||
|                             new CommandOptionSchema(typeof(DivideCommand).GetProperty(nameof(DivideCommand.Dividend)), | ||||
|                                 "dividend", 'D', true, "The number to divide."), | ||||
|                                 "dividend", 'D', true, "The number to divide.", null), | ||||
|                             new CommandOptionSchema(typeof(DivideCommand).GetProperty(nameof(DivideCommand.Divisor)), | ||||
|                                 "divisor", 'd', true, "The number to divide by.") | ||||
|                                 "divisor", 'd', true, "The number to divide by.", null) | ||||
|                         }), | ||||
|                     new CommandSchema(typeof(ConcatCommand), "concat", "Concatenate strings.", | ||||
|                         new[] | ||||
|                         { | ||||
|                             new CommandOptionSchema(typeof(ConcatCommand).GetProperty(nameof(ConcatCommand.Inputs)), | ||||
|                                 null, 'i', true, "Input strings."), | ||||
|                                 null, 'i', true, "Input strings.", null), | ||||
|                             new CommandOptionSchema(typeof(ConcatCommand).GetProperty(nameof(ConcatCommand.Separator)), | ||||
|                                 null, 's', false, "String separator.") | ||||
|                         }) | ||||
|                                 null, 's', false, "String separator.", null) | ||||
|                         }), | ||||
|                     new CommandSchema(typeof(EnvironmentVariableCommand), null, "Reads option values from environment variables.", | ||||
|                         new[] | ||||
|                         { | ||||
|                             new CommandOptionSchema(typeof(EnvironmentVariableCommand).GetProperty(nameof(EnvironmentVariableCommand.Option)), | ||||
|                                 "opt", null, false, null, "ENV_SINGLE_VALUE") | ||||
|                         } | ||||
|                     ) | ||||
|                 } | ||||
|             ); | ||||
|  | ||||
|             yield return new TestCaseData( | ||||
|                 new[] {typeof(HelloWorldDefaultCommand)}, | ||||
|                 new[] { typeof(HelloWorldDefaultCommand) }, | ||||
|                 new[] | ||||
|                 { | ||||
|                     new CommandSchema(typeof(HelloWorldDefaultCommand), null, null, new CommandOptionSchema[0]) | ||||
| @@ -62,7 +69,7 @@ namespace CliFx.Tests.Services | ||||
|             { | ||||
|                 new[] {typeof(NonAnnotatedCommand)} | ||||
|             }); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData(new object[] | ||||
|             { | ||||
|                 new[] {typeof(DuplicateOptionNamesCommand)} | ||||
| @@ -72,7 +79,7 @@ namespace CliFx.Tests.Services | ||||
|             { | ||||
|                 new[] {typeof(DuplicateOptionShortNamesCommand)} | ||||
|             }); | ||||
|              | ||||
|  | ||||
|             yield return new TestCaseData(new object[] | ||||
|             { | ||||
|                 new[] {typeof(ExceptionCommand), typeof(CommandExceptionCommand)} | ||||
|   | ||||
							
								
								
									
										10
									
								
								CliFx.Tests/Stubs/EmptyEnvironmentVariablesProviderStub.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								CliFx.Tests/Stubs/EmptyEnvironmentVariablesProviderStub.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| using System.Collections.Generic; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.Stubs | ||||
| { | ||||
|     public class EmptyEnvironmentVariablesProviderStub : IEnvironmentVariablesProvider | ||||
|     { | ||||
|         public IReadOnlyDictionary<string, string> GetEnvironmentVariables() => new Dictionary<string, string>(); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										18
									
								
								CliFx.Tests/Stubs/EnvironmentVariablesProviderStub.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								CliFx.Tests/Stubs/EnvironmentVariablesProviderStub.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| using System.Collections.Generic; | ||||
| using System.IO; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.Stubs | ||||
| { | ||||
|     public class EnvironmentVariablesProviderStub : IEnvironmentVariablesProvider | ||||
|     { | ||||
|         public static readonly Dictionary<string, string> EnvironmentVariables = new Dictionary<string, string> | ||||
|         { | ||||
|             ["ENV_SINGLE_VALUE"] = "A", | ||||
|             ["ENV_MULTIPLE_VALUES"] = $"A{Path.PathSeparator}B{Path.PathSeparator}C{Path.PathSeparator}", | ||||
|             ["ENV_ESCAPED_MULTIPLE_VALUES"] = $"\"A{Path.PathSeparator}B{Path.PathSeparator}C{Path.PathSeparator}\"" | ||||
|         }; | ||||
|  | ||||
|         public IReadOnlyDictionary<string, string> GetEnvironmentVariables() => EnvironmentVariables; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										23
									
								
								CliFx.Tests/TestCommands/CancellableCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								CliFx.Tests/TestCommands/CancellableCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| using System; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.TestCommands | ||||
| { | ||||
|     [Command("cancel")] | ||||
|     public class CancellableCommand : ICommand | ||||
|     { | ||||
|         public async Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             await Task.Yield(); | ||||
|  | ||||
|             console.Output.WriteLine("Printed"); | ||||
|  | ||||
|             await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false); | ||||
|  | ||||
|             console.Output.WriteLine("Never printed"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Exceptions; | ||||
| using CliFx.Services; | ||||
| @@ -14,6 +15,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption("msg", 'm')] | ||||
|         public string Message { get; set; } | ||||
|          | ||||
|         public Task ExecuteAsync(IConsole console) => throw new CommandException(Message, ExitCode); | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => throw new CommandException(Message, ExitCode); | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Collections.Generic; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
| @@ -14,7 +15,7 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption('s', Description = "String separator.")] | ||||
|         public string Separator { get; set; } = "";  | ||||
|          | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             console.Output.WriteLine(string.Join(Separator, Inputs)); | ||||
|             return Task.CompletedTask; | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -16,7 +17,7 @@ namespace CliFx.Tests.TestCommands | ||||
|         // This property should be ignored by resolver | ||||
|         public bool NotAnOption { get; set; } | ||||
|          | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             console.Output.WriteLine(Dividend / Divisor); | ||||
|             return Task.CompletedTask; | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -13,6 +14,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption("fruits")] | ||||
|         public string Oranges { get; set; } | ||||
|          | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -13,6 +14,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption('f')] | ||||
|         public string Oranges { get; set; } | ||||
|          | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										16
									
								
								CliFx.Tests/TestCommands/EnvironmentVariableCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								CliFx.Tests/TestCommands/EnvironmentVariableCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.TestCommands | ||||
| { | ||||
| 	[Command(Description = "Reads option values from environment variables.")] | ||||
| 	public class EnvironmentVariableCommand : ICommand | ||||
| 	{ | ||||
| 		[CommandOption("opt", EnvironmentVariableName = "ENV_SINGLE_VALUE")] | ||||
| 		public string Option { get; set; } | ||||
|  | ||||
| 		public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
| 	} | ||||
| } | ||||
| @@ -0,0 +1,17 @@ | ||||
| using System.Collections.Generic; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.TestCommands | ||||
| { | ||||
| 	[Command(Description = "Reads multiple option values from environment variables.")] | ||||
| 	public class EnvironmentVariableWithMultipleValuesCommand : ICommand | ||||
| 	{ | ||||
| 		[CommandOption("opt", EnvironmentVariableName = "ENV_MULTIPLE_VALUES")] | ||||
| 		public IEnumerable<string> Option { get; set; } | ||||
|  | ||||
| 		public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
| 	} | ||||
| } | ||||
| @@ -0,0 +1,17 @@ | ||||
| using System.Collections.Generic; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.TestCommands | ||||
| { | ||||
|     [Command(Description = "Reads one option value from environment variables because target property is not a collection.")] | ||||
|     public class EnvironmentVariableWithoutCollectionPropertyCommand : ICommand | ||||
|     { | ||||
|         [CommandOption("opt", EnvironmentVariableName = "ENV_MULTIPLE_VALUES")] | ||||
|         public string Option { get; set; } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
| @@ -11,6 +12,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption("msg", 'm')] | ||||
|         public string Message { get; set; } | ||||
|          | ||||
|         public Task ExecuteAsync(IConsole console) => throw new Exception(Message); | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => throw new Exception(Message); | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -7,7 +8,7 @@ namespace CliFx.Tests.TestCommands | ||||
|     [Command] | ||||
|     public class HelloWorldDefaultCommand : ICommand | ||||
|     { | ||||
|         public Task ExecuteAsync(IConsole console) | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) | ||||
|         { | ||||
|             console.Output.WriteLine("Hello world."); | ||||
|             return Task.CompletedTask; | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -13,6 +14,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption("option-b", 'b', Description = "OptionB description.")] | ||||
|         public string OptionB { get; set; } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -13,6 +14,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption("option-d", 'd', Description = "OptionD description.")] | ||||
|         public string OptionD { get; set; } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Attributes; | ||||
| using CliFx.Services; | ||||
|  | ||||
| @@ -10,6 +11,6 @@ namespace CliFx.Tests.TestCommands | ||||
|         [CommandOption("option-e", 'e', Description = "OptionE description.")] | ||||
|         public string OptionE { get; set; } | ||||
|  | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -1,10 +1,11 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx.Tests.TestCommands | ||||
| { | ||||
|     public class NonAnnotatedCommand : ICommand | ||||
|     { | ||||
|         public Task ExecuteAsync(IConsole console) => Task.CompletedTask; | ||||
|         public Task ExecuteAsync(IConsole console, CancellationToken cancellationToken) => Task.CompletedTask; | ||||
|     } | ||||
| } | ||||
| @@ -28,6 +28,11 @@ namespace CliFx.Attributes | ||||
|         /// </summary> | ||||
|         public string Description { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Optional environment variable name that will be used as fallback value if no option value is specified. | ||||
|         /// </summary> | ||||
|         public string EnvironmentVariableName { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandOptionAttribute"/>. | ||||
|         /// </summary> | ||||
| @@ -41,7 +46,7 @@ namespace CliFx.Attributes | ||||
|         /// Initializes an instance of <see cref="CommandOptionAttribute"/>. | ||||
|         /// </summary> | ||||
|         public CommandOptionAttribute(string name, char shortName) | ||||
|             : this(name, (char?) shortName) | ||||
|             : this(name, (char?)shortName) | ||||
|         { | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -171,7 +171,7 @@ namespace CliFx | ||||
|             _commandInitializer.InitializeCommand(command, targetCommandSchema, commandInput); | ||||
|  | ||||
|             // Execute command | ||||
|             await command.ExecuteAsync(_console); | ||||
|             await command.ExecuteAsync(_console, _console.CancellationToken); | ||||
|  | ||||
|             // Finish the chain with exit code 0 | ||||
|             return 0; | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace CliFx | ||||
|         private IConsole _console; | ||||
|         private ICommandFactory _commandFactory; | ||||
|         private ICommandOptionInputConverter _commandOptionInputConverter; | ||||
|         private IEnvironmentVariablesProvider _environmentVariablesProvider; | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public ICliApplicationBuilder AddCommand(Type commandType) | ||||
| @@ -116,6 +117,13 @@ namespace CliFx | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public ICliApplicationBuilder UseEnvironmentVariablesProvider(IEnvironmentVariablesProvider environmentVariablesProvider) | ||||
|         { | ||||
|             _environmentVariablesProvider = environmentVariablesProvider.GuardNotNull(nameof(environmentVariablesProvider)); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public ICliApplication Build() | ||||
|         { | ||||
| @@ -126,14 +134,15 @@ namespace CliFx | ||||
|             _console = _console ?? new SystemConsole(); | ||||
|             _commandFactory = _commandFactory ?? new CommandFactory(); | ||||
|             _commandOptionInputConverter = _commandOptionInputConverter ?? new CommandOptionInputConverter(); | ||||
|             _environmentVariablesProvider = _environmentVariablesProvider ?? new EnvironmentVariablesProvider(); | ||||
|  | ||||
|             // Project parameters to expected types | ||||
|             var metadata = new ApplicationMetadata(_title, _executableName, _versionText, _description); | ||||
|             var configuration = new ApplicationConfiguration(_commandTypes.ToArray(), _isDebugModeAllowed, _isPreviewModeAllowed); | ||||
|  | ||||
|             return new CliApplication(metadata, configuration, | ||||
|                 _console, new CommandInputParser(), new CommandSchemaResolver(), | ||||
|                 _commandFactory, new CommandInitializer(_commandOptionInputConverter), new HelpTextRenderer()); | ||||
|                 _console, new CommandInputParser(_environmentVariablesProvider), new CommandSchemaResolver(), | ||||
|                 _commandFactory, new CommandInitializer(_commandOptionInputConverter, new EnvironmentVariablesParser()), new HelpTextRenderer()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -2,8 +2,7 @@ | ||||
|  | ||||
|   <PropertyGroup> | ||||
|     <TargetFrameworks>net45;netstandard2.0</TargetFrameworks> | ||||
|     <LangVersion>latest</LangVersion> | ||||
|     <Version>0.0.5</Version> | ||||
|     <Version>0.0.6</Version> | ||||
|     <Company>Tyrrrz</Company> | ||||
|     <Authors>$(Company)</Authors> | ||||
|     <Copyright>Copyright (C) Alexey Golub</Copyright> | ||||
| @@ -11,7 +10,7 @@ | ||||
|     <PackageTags>command line executable interface framework parser arguments net core</PackageTags> | ||||
|     <PackageProjectUrl>https://github.com/Tyrrrz/CliFx</PackageProjectUrl> | ||||
|     <PackageReleaseNotes>https://github.com/Tyrrrz/CliFx/blob/master/Changelog.md</PackageReleaseNotes> | ||||
|     <PackageIconUrl>https://raw.githubusercontent.com/Tyrrrz/CliFx/master/favicon.png</PackageIconUrl> | ||||
|     <PackageIcon>favicon.png</PackageIcon> | ||||
|     <PackageLicenseExpression>LGPL-3.0-only</PackageLicenseExpression> | ||||
|     <RepositoryUrl>https://github.com/Tyrrrz/CliFx</RepositoryUrl> | ||||
|     <RepositoryType>git</RepositoryType> | ||||
| @@ -20,4 +19,8 @@ | ||||
|     <DocumentationFile>bin/$(Configuration)/$(TargetFramework)/$(AssemblyName).xml</DocumentationFile> | ||||
|   </PropertyGroup> | ||||
|  | ||||
| </Project> | ||||
|   <ItemGroup> | ||||
|     <None Include="../favicon.png" Pack="True" PackagePath="" /> | ||||
|   </ItemGroup> | ||||
|  | ||||
| </Project> | ||||
|   | ||||
| @@ -64,6 +64,11 @@ namespace CliFx | ||||
|         /// </summary> | ||||
|         ICliApplicationBuilder UseCommandOptionInputConverter(ICommandOptionInputConverter converter); | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Configures application to use specified implementation of <see cref="IEnvironmentVariablesProvider"/>. | ||||
|         /// </summary> | ||||
|         ICliApplicationBuilder UseEnvironmentVariablesProvider(IEnvironmentVariablesProvider environmentVariablesProvider); | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Creates an instance of <see cref="ICliApplication"/> using configured parameters. | ||||
|         /// Default values are used in place of parameters that were not specified. | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.Threading.Tasks; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using CliFx.Services; | ||||
|  | ||||
| namespace CliFx | ||||
| @@ -12,6 +13,6 @@ namespace CliFx | ||||
|         /// Executes command using specified implementation of <see cref="IConsole"/>. | ||||
|         /// This method is called when the command is invoked by a user through command line interface. | ||||
|         /// </summary> | ||||
|         Task ExecuteAsync(IConsole console); | ||||
|         Task ExecuteAsync(IConsole console, CancellationToken cancellationToken); | ||||
|     } | ||||
| } | ||||
| @@ -25,14 +25,36 @@ namespace CliFx.Models | ||||
|         /// </summary> | ||||
|         public IReadOnlyList<CommandOptionInput> Options { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Environment variables available when the command was parsed | ||||
|         /// </summary> | ||||
|         public IReadOnlyDictionary<string, string> EnvironmentVariables { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInput"/>. | ||||
|         /// </summary> | ||||
|         public CommandInput(string commandName, IReadOnlyList<string> directives, IReadOnlyList<CommandOptionInput> options) | ||||
|         public CommandInput(string commandName, IReadOnlyList<string> directives, IReadOnlyList<CommandOptionInput> options, IReadOnlyDictionary<string, string> environmentVariables) | ||||
|         { | ||||
|             CommandName = commandName; // can be null | ||||
|             Directives = directives.GuardNotNull(nameof(directives)); | ||||
|             Options = options.GuardNotNull(nameof(options)); | ||||
|             EnvironmentVariables = environmentVariables.GuardNotNull(nameof(environmentVariables)); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInput"/>. | ||||
|         /// </summary> | ||||
|         public CommandInput(string commandName, IReadOnlyList<string> directives, IReadOnlyList<CommandOptionInput> options) | ||||
|             : this(commandName, directives, options, EmptyEnvironmentVariables) | ||||
|         { | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInput"/>. | ||||
|         /// </summary> | ||||
|         public CommandInput(string commandName, IReadOnlyList<CommandOptionInput> options, IReadOnlyDictionary<string, string> environmentVariables) | ||||
|             : this(commandName, EmptyDirectives, options, environmentVariables) | ||||
|         { | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
| @@ -87,6 +109,7 @@ namespace CliFx.Models | ||||
|     { | ||||
|         private static readonly IReadOnlyList<string> EmptyDirectives = new string[0]; | ||||
|         private static readonly IReadOnlyList<CommandOptionInput> EmptyOptions = new CommandOptionInput[0]; | ||||
|         private static readonly IReadOnlyDictionary<string, string> EmptyEnvironmentVariables = new Dictionary<string, string>(); | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Empty input. | ||||
|   | ||||
| @@ -34,16 +34,22 @@ namespace CliFx.Models | ||||
|         /// </summary> | ||||
|         public string Description { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Optional environment variable name that will be used as fallback value if no option value is specified. | ||||
|         /// </summary> | ||||
|         public string EnvironmentVariableName { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandOptionSchema"/>. | ||||
|         /// </summary> | ||||
|         public CommandOptionSchema(PropertyInfo property, string name, char? shortName, bool isRequired, string description) | ||||
|         public CommandOptionSchema(PropertyInfo property, string name, char? shortName, bool isRequired, string description, string environmentVariableName) | ||||
|         { | ||||
|             Property = property; // can be null | ||||
|             Name = name; // can be null | ||||
|             ShortName = shortName; // can be null | ||||
|             IsRequired = isRequired; | ||||
|             Description = description; // can be null | ||||
|             EnvironmentVariableName = environmentVariableName; //can be null | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
| @@ -75,9 +81,9 @@ namespace CliFx.Models | ||||
|         // ...in CliApplication (when reading) and HelpTextRenderer (when writing). | ||||
|  | ||||
|         internal static CommandOptionSchema HelpOption { get; } = | ||||
|             new CommandOptionSchema(null, "help", 'h', false, "Shows help text."); | ||||
|             new CommandOptionSchema(null, "help", 'h', false, "Shows help text.", null); | ||||
|  | ||||
|         internal static CommandOptionSchema VersionOption { get; } = | ||||
|             new CommandOptionSchema(null, "version", null, false, "Shows version information."); | ||||
|             new CommandOptionSchema(null, "version", null, false, "Shows version information.", null); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| using System; | ||||
| using CliFx.Internal; | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Linq; | ||||
| using CliFx.Internal; | ||||
|  | ||||
| namespace CliFx.Models | ||||
| { | ||||
| @@ -71,16 +71,16 @@ namespace CliFx.Models | ||||
|  | ||||
|             return matchesByName || matchesByShortName; | ||||
|         } | ||||
|          | ||||
|         /// <summary> | ||||
|         /// Finds an option that matches specified alias, or null if not found. | ||||
|         /// </summary> | ||||
|         public static CommandOptionSchema FindByAlias(this IReadOnlyList<CommandOptionSchema> optionSchemas, string alias) | ||||
|         { | ||||
|             optionSchemas.GuardNotNull(nameof(optionSchemas)); | ||||
|             alias.GuardNotNull(nameof(alias)); | ||||
|  | ||||
|             return optionSchemas.FirstOrDefault(o => o.MatchesAlias(alias)); | ||||
|         /// <summary> | ||||
|         /// Finds an option input that matches the option schema specified, or null if not found. | ||||
|         /// </summary> | ||||
|         public static CommandOptionInput FindByOptionSchema(this IReadOnlyList<CommandOptionInput> optionInputs, CommandOptionSchema optionSchema) | ||||
|         { | ||||
|             optionInputs.GuardNotNull(nameof(optionInputs)); | ||||
|             optionSchema.GuardNotNull(nameof(optionSchema)); | ||||
|  | ||||
|             return optionInputs.FirstOrDefault(o => optionSchema.MatchesAlias(o.Alias)); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|   | ||||
| @@ -11,20 +11,30 @@ namespace CliFx.Services | ||||
|     public class CommandInitializer : ICommandInitializer | ||||
|     { | ||||
|         private readonly ICommandOptionInputConverter _commandOptionInputConverter; | ||||
|         private readonly IEnvironmentVariablesParser _environmentVariablesParser; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInitializer"/>. | ||||
|         /// </summary> | ||||
|         public CommandInitializer(ICommandOptionInputConverter commandOptionInputConverter) | ||||
|         public CommandInitializer(ICommandOptionInputConverter commandOptionInputConverter, IEnvironmentVariablesParser environmentVariablesParser) | ||||
|         { | ||||
|             _commandOptionInputConverter = commandOptionInputConverter.GuardNotNull(nameof(commandOptionInputConverter)); | ||||
|             _environmentVariablesParser = environmentVariablesParser.GuardNotNull(nameof(environmentVariablesParser)); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInitializer"/>. | ||||
|         /// </summary> | ||||
|         public CommandInitializer(IEnvironmentVariablesParser environmentVariablesParser) | ||||
|             : this(new CommandOptionInputConverter(), environmentVariablesParser) | ||||
|         { | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInitializer"/>. | ||||
|         /// </summary> | ||||
|         public CommandInitializer() | ||||
|             : this(new CommandOptionInputConverter()) | ||||
|             : this(new CommandOptionInputConverter(), new EnvironmentVariablesParser()) | ||||
|         { | ||||
|         } | ||||
|  | ||||
| @@ -38,15 +48,28 @@ namespace CliFx.Services | ||||
|             // Keep track of unset required options to report an error at a later stage | ||||
|             var unsetRequiredOptions = commandSchema.Options.Where(o => o.IsRequired).ToList(); | ||||
|  | ||||
|             // Set command options | ||||
|             foreach (var optionInput in commandInput.Options) | ||||
|             //Set command options | ||||
|             foreach (var optionSchema in commandSchema.Options) | ||||
|             { | ||||
|                 // Find matching option schema for this option input | ||||
|                 var optionSchema = commandSchema.Options.FindByAlias(optionInput.Alias); | ||||
|                 if (optionSchema == null) | ||||
|                 //Find matching option input | ||||
|                 var optionInput = commandInput.Options.FindByOptionSchema(optionSchema); | ||||
|  | ||||
|                 //If no option input is available fall back to environment variable values | ||||
|                 if (optionInput == null && !optionSchema.EnvironmentVariableName.IsNullOrWhiteSpace()) | ||||
|                 { | ||||
|                     var fallbackEnvironmentVariableExists = commandInput.EnvironmentVariables.ContainsKey(optionSchema.EnvironmentVariableName); | ||||
|  | ||||
|                     //If no environment variable is found or there is no valid value for this option skip it | ||||
|                     if (!fallbackEnvironmentVariableExists || commandInput.EnvironmentVariables[optionSchema.EnvironmentVariableName].IsNullOrWhiteSpace()) | ||||
|                         continue; | ||||
|  | ||||
|                     optionInput = _environmentVariablesParser.GetCommandOptionInputFromEnvironmentVariable(commandInput.EnvironmentVariables[optionSchema.EnvironmentVariableName], optionSchema); | ||||
|                 } | ||||
|  | ||||
|                 //No fallback available and no option input was specified, skip option | ||||
|                 if (optionInput == null) | ||||
|                     continue; | ||||
|  | ||||
|                 // Convert option to the type of the underlying property | ||||
|                 var convertedValue = _commandOptionInputConverter.ConvertOptionInput(optionInput, optionSchema.Property.PropertyType); | ||||
|  | ||||
|                 // Set value of the underlying property | ||||
|   | ||||
| @@ -12,6 +12,26 @@ namespace CliFx.Services | ||||
|     /// </summary> | ||||
|     public class CommandInputParser : ICommandInputParser | ||||
|     { | ||||
|         private readonly IEnvironmentVariablesProvider _environmentVariablesProvider; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInputParser"/> | ||||
|         /// </summary> | ||||
|         public CommandInputParser(IEnvironmentVariablesProvider environmentVariablesProvider) | ||||
|         { | ||||
|             environmentVariablesProvider.GuardNotNull(nameof(environmentVariablesProvider)); | ||||
|  | ||||
|             _environmentVariablesProvider = environmentVariablesProvider; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes an instance of <see cref="CommandInputParser"/> | ||||
|         /// </summary> | ||||
|         public CommandInputParser() | ||||
|             : this(new EnvironmentVariablesProvider()) | ||||
|         { | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public CommandInput ParseCommandInput(IReadOnlyList<string> commandLineArguments) | ||||
|         { | ||||
| @@ -78,7 +98,9 @@ namespace CliFx.Services | ||||
|             var commandName = commandNameBuilder.Length > 0 ? commandNameBuilder.ToString() : null; | ||||
|             var options = optionsDic.Select(p => new CommandOptionInput(p.Key, p.Value)).ToArray(); | ||||
|  | ||||
|             return new CommandInput(commandName, directives, options); | ||||
|             var environmentVariables = _environmentVariablesProvider.GetEnvironmentVariables(); | ||||
|  | ||||
|             return new CommandInput(commandName, directives, options, environmentVariables); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -31,7 +31,8 @@ namespace CliFx.Services | ||||
|                     attribute.Name, | ||||
|                     attribute.ShortName, | ||||
|                     attribute.IsRequired, | ||||
|                     attribute.Description); | ||||
|                     attribute.Description, | ||||
|                     attribute.EnvironmentVariableName); | ||||
|  | ||||
|                 // Make sure there are no other options with the same name | ||||
|                 var existingOptionWithSameName = result | ||||
|   | ||||
							
								
								
									
										30
									
								
								CliFx/Services/EnvironmentVariablesParser.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								CliFx/Services/EnvironmentVariablesParser.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| using System.IO; | ||||
| using System.Linq; | ||||
| using CliFx.Internal; | ||||
| using CliFx.Models; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| { | ||||
|     /// <inheritdoct /> | ||||
|     public class EnvironmentVariablesParser : IEnvironmentVariablesParser | ||||
|     { | ||||
|         /// <inheritdoct /> | ||||
|         public CommandOptionInput GetCommandOptionInputFromEnvironmentVariable(string environmentVariableValue, CommandOptionSchema targetOptionSchema) | ||||
|         { | ||||
|             environmentVariableValue.GuardNotNull(nameof(environmentVariableValue)); | ||||
|             targetOptionSchema.GuardNotNull(nameof(targetOptionSchema)); | ||||
|  | ||||
|             //If the option is not a collection do not split environment variable values | ||||
|             var optionIsCollection = targetOptionSchema.Property.PropertyType.IsCollection(); | ||||
|  | ||||
|             if (!optionIsCollection) return new CommandOptionInput(targetOptionSchema.EnvironmentVariableName, environmentVariableValue); | ||||
|  | ||||
|             //If the option is a collection split the values using System separator, empty values are discarded | ||||
|             var environmentVariableValues = environmentVariableValue.Split(Path.PathSeparator) | ||||
|                 .Where(v => !v.IsNullOrWhiteSpace()) | ||||
|                 .ToList(); | ||||
|  | ||||
|             return new CommandOptionInput(targetOptionSchema.EnvironmentVariableName, environmentVariableValues); | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										36
									
								
								CliFx/Services/EnvironmentVariablesProvider.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								CliFx/Services/EnvironmentVariablesProvider.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| using System; | ||||
| using System.Collections; | ||||
| using System.Collections.Generic; | ||||
| using System.Security; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| { | ||||
|     /// <inheritdoc /> | ||||
|     public class EnvironmentVariablesProvider : IEnvironmentVariablesProvider | ||||
|     { | ||||
|         /// <inheritdoc /> | ||||
|         public IReadOnlyDictionary<string, string> GetEnvironmentVariables() | ||||
|         { | ||||
|             try | ||||
|             { | ||||
|                 var environmentVariables = Environment.GetEnvironmentVariables(); | ||||
|  | ||||
|                 //Constructing the dictionary manually allows to specify a key comparer that ignores case | ||||
|                 //This allows to ignore casing when looking for a fallback environment variable of an option | ||||
|                 var environmentVariablesAsDictionary = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase); | ||||
|  | ||||
|                 //Type DictionaryEntry must be explicitly used otherwise it will enumerate as a collection of objects | ||||
|                 foreach (DictionaryEntry environmentVariable in environmentVariables) | ||||
|                 { | ||||
|                     environmentVariablesAsDictionary.Add(environmentVariable.Key.ToString(), environmentVariable.Value.ToString()); | ||||
|                 } | ||||
|  | ||||
|                 return environmentVariablesAsDictionary; | ||||
|             } | ||||
|             catch (SecurityException) | ||||
|             { | ||||
|                 return new Dictionary<string, string>(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using CliFx.Internal; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| @@ -50,5 +51,25 @@ namespace CliFx.Services | ||||
|  | ||||
|             console.WithForegroundColor(foregroundColor, () => console.WithBackgroundColor(backgroundColor, action)); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets wether a string representing an environment variable value is escaped (i.e.: surrounded by double quotation marks) | ||||
|         /// </summary> | ||||
|         public static bool IsEnvironmentVariableEscaped(this string environmentVariableValue) | ||||
|         { | ||||
|             environmentVariableValue.GuardNotNull(nameof(environmentVariableValue)); | ||||
|  | ||||
|             return environmentVariableValue.StartsWith("\"") && environmentVariableValue.EndsWith("\""); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets wether the <see cref="Type"/> supplied is a collection implementing <see cref="IEnumerable{T}"/> | ||||
|         /// </summary> | ||||
|         public static bool IsCollection(this Type type) | ||||
|         { | ||||
|             type.GuardNotNull(nameof(type)); | ||||
|  | ||||
|             return type != typeof(string) && type.GetEnumerableUnderlyingType() != null; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| using System; | ||||
| using System.IO; | ||||
| using System.Threading; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| { | ||||
| @@ -52,5 +53,10 @@ namespace CliFx.Services | ||||
|         /// Resets foreground and background color to default values. | ||||
|         /// </summary> | ||||
|         void ResetColor(); | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Cancels when soft cancellation requested. | ||||
|         /// </summary> | ||||
|         CancellationToken CancellationToken { get; } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										15
									
								
								CliFx/Services/IEnvironmentVariablesParser.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								CliFx/Services/IEnvironmentVariablesParser.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| using CliFx.Models; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Parses environment variable values | ||||
|     /// </summary> | ||||
|     public interface IEnvironmentVariablesParser | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Parse an environment variable value and converts it to a <see cref="CommandOptionInput"/>  | ||||
|         /// </summary> | ||||
|         CommandOptionInput GetCommandOptionInputFromEnvironmentVariable(string environmentVariableValue, CommandOptionSchema targetOptionSchema); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										16
									
								
								CliFx/Services/IEnvironmentVariablesProvider.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								CliFx/Services/IEnvironmentVariablesProvider.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| using System.Collections.Generic; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Provides environment variable values | ||||
|     /// </summary> | ||||
|     public interface IEnvironmentVariablesProvider | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Returns all the environment variables available. | ||||
|         /// </summary> | ||||
|         /// <remarks>If the User is not allowed to read environment variables it will return an empty dictionary.</remarks> | ||||
|         IReadOnlyDictionary<string, string> GetEnvironmentVariables(); | ||||
|     } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| using System; | ||||
| using System.IO; | ||||
| using System.Threading; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| { | ||||
| @@ -8,6 +9,22 @@ namespace CliFx.Services | ||||
|     /// </summary> | ||||
|     public class SystemConsole : IConsole | ||||
|     { | ||||
|         private readonly CancellationTokenSource _cancellationTokenSource = new CancellationTokenSource(); | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public SystemConsole() | ||||
|         { | ||||
|             // Subscribe to CancelKeyPress event with cancellation token source | ||||
|             // Kills app on second cancellation (hard cancellation) | ||||
|             Console.CancelKeyPress += (_, args) => | ||||
|             { | ||||
|                 if (_cancellationTokenSource.IsCancellationRequested)  | ||||
|                     return; | ||||
|                 args.Cancel = true; | ||||
|                 _cancellationTokenSource.Cancel(); | ||||
|             }; | ||||
|         } | ||||
|          | ||||
|         /// <inheritdoc /> | ||||
|         public TextReader Input => Console.In; | ||||
|  | ||||
| @@ -42,5 +59,8 @@ namespace CliFx.Services | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public void ResetColor() => Console.ResetColor(); | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public CancellationToken CancellationToken => _cancellationTokenSource.Token; | ||||
|     } | ||||
| } | ||||
| @@ -1,5 +1,6 @@ | ||||
| using System; | ||||
| using System.IO; | ||||
| using System.Threading; | ||||
| using CliFx.Internal; | ||||
|  | ||||
| namespace CliFx.Services | ||||
| @@ -11,6 +12,8 @@ namespace CliFx.Services | ||||
|     /// </summary> | ||||
|     public class VirtualConsole : IConsole | ||||
|     { | ||||
|         private readonly CancellationTokenSource _cancellationTokenSource = new CancellationTokenSource(); | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public TextReader Input { get; } | ||||
|  | ||||
| @@ -82,5 +85,16 @@ namespace CliFx.Services | ||||
|             ForegroundColor = ConsoleColor.Gray; | ||||
|             BackgroundColor = ConsoleColor.Black; | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc /> | ||||
|         public CancellationToken CancellationToken => _cancellationTokenSource.Token; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Simulates cancellation. | ||||
|         /// </summary> | ||||
|         public void Cancel() | ||||
|         { | ||||
|             _cancellationTokenSource.Cancel(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -32,7 +32,7 @@ _CliFx is to command line interfaces what ASP.NET Core is to web applications._ | ||||
|  | ||||
| ## Screenshots | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Argument syntax | ||||
|  | ||||
| @@ -42,8 +42,8 @@ The following examples are valid for any application created with CliFx: | ||||
|  | ||||
| - `myapp --foo bar` sets option `"foo"` to value `"bar"` | ||||
| - `myapp -f bar` sets option `'f'` to value `"bar"` | ||||
| - `myapp --switch` sets option `"switch"` to value `true`  | ||||
| - `myapp -s` sets option `'s'` to value `true`  | ||||
| - `myapp --switch` sets option `"switch"` to value `true` | ||||
| - `myapp -s` sets option `'s'` to value `true` | ||||
| - `myapp -abc` sets options `'a'`, `'b'` and `'c'` to value `true` | ||||
| - `myapp -xqf bar` sets options `'x'` and `'q'` to value `true`, and option `'f'` to value `"bar"` | ||||
| - `myapp -i file1.txt file2.txt` sets option `'i'` to a sequence of values `"file1.txt"` and `"file2.txt"` | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| version: '{build}' | ||||
|  | ||||
| image: Visual Studio 2017 | ||||
| image: Visual Studio 2019 | ||||
| configuration: Release | ||||
|  | ||||
| before_build: | ||||
|   | ||||
		Reference in New Issue
	
	Block a user