mirror of
				https://github.com/Tyrrrz/CliFx.git
				synced 2025-10-25 15:19:17 +00:00 
			
		
		
		
	Compare commits
	
		
			82 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 90a01e729b | ||
|  | ac01c2aecb | ||
|  | 4acffe925c | ||
|  | 18f53eeeef | ||
|  | 03d6942540 | ||
|  | 9be811a89a | ||
|  | f9f5a4696b | ||
|  | d6da687170 | ||
|  | eba66d0878 | ||
|  | 8c682766bd | ||
|  | 39d626c8d8 | ||
|  | a338ac8ce2 | ||
|  | 11637127cb | ||
|  | 4e12aefafb | ||
|  | 144d3592fb | ||
|  | 6f82c2f0f9 | ||
|  | b8c60717d5 | ||
|  | fec6850c39 | ||
|  | 6a378ad946 | ||
|  | 11579f11b1 | ||
|  | 60a3b26fd1 | ||
|  | 3abdfb1acf | ||
|  | 9557d386e2 | ||
|  | d0d024c427 | ||
|  | f765af6061 | ||
|  | 7f2202e869 | ||
|  | 14ad9d5738 | ||
|  | b120138de3 | ||
|  | 8df1d607c1 | ||
|  | c06f2810b9 | ||
|  | d52a205f13 | ||
|  | 0ec12e57c1 | ||
|  | c322b7029c | ||
|  | 6a38c04c11 | ||
|  | 5e53107def | ||
|  | 36cea937de | ||
|  | 438d6b98ac | ||
|  | 8e1488c395 | ||
|  | 65d321b476 | ||
|  | c6d2359d6b | ||
|  | 0d32876bad | ||
|  | c063251d89 | ||
|  | 3831cfc7c0 | ||
|  | b17341b56c | ||
|  | 5bda964fb5 | ||
|  | 432430489a | ||
|  | 9a20101f30 | ||
|  | b491818779 | ||
|  | 69c24c8dfc | ||
|  | 004f906148 | ||
|  | ac83233dc2 | ||
|  | 082910c968 | ||
|  | 11e3e0f85d | ||
|  | 42f4d7d5a7 | ||
|  | bed22b6500 | ||
|  | 17449e0794 | ||
|  | 4732166f5f | ||
|  | f5e37b96fc | ||
|  | 4cef596fe8 | ||
|  | 19b87717c1 | ||
|  | 7e4c6b20ff | ||
|  | fb2071ed2b | ||
|  | 7d2f934310 | ||
|  | 95a00b0952 | ||
|  | cb3fee65f3 | ||
|  | 65628b145a | ||
|  | 802bbfccc6 | ||
|  | 6e7742a4f3 | ||
|  | f6a1a40471 | ||
|  | 33ca4da260 | ||
|  | cbb72b16ae | ||
|  | c58629e999 | ||
|  | 387fb72718 | ||
|  | e04f0da318 | ||
|  | d25873ee10 | ||
|  | a28223fc8b | ||
|  | 1dab27de55 | ||
|  | 698629b153 | ||
|  | 65b66b0d27 | ||
|  | 7d3ba612c4 | ||
|  | 8c3b8d1f49 | ||
|  | fdd39855ad | 
							
								
								
									
										63
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										63
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @@ -1,63 +0,0 @@ | |||||||
| ############################################################################### |  | ||||||
| # Set default behavior to automatically normalize line endings. |  | ||||||
| ############################################################################### |  | ||||||
| * text=auto |  | ||||||
|  |  | ||||||
| ############################################################################### |  | ||||||
| # Set default behavior for command prompt diff. |  | ||||||
| # |  | ||||||
| # This is need for earlier builds of msysgit that does not have it on by |  | ||||||
| # default for csharp files. |  | ||||||
| # Note: This is only used by command line |  | ||||||
| ############################################################################### |  | ||||||
| #*.cs     diff=csharp |  | ||||||
|  |  | ||||||
| ############################################################################### |  | ||||||
| # Set the merge driver for project and solution files |  | ||||||
| # |  | ||||||
| # Merging from the command prompt will add diff markers to the files if there |  | ||||||
| # are conflicts (Merging from VS is not affected by the settings below, in VS |  | ||||||
| # the diff markers are never inserted). Diff markers may cause the following  |  | ||||||
| # file extensions to fail to load in VS. An alternative would be to treat |  | ||||||
| # these files as binary and thus will always conflict and require user |  | ||||||
| # intervention with every merge. To do so, just uncomment the entries below |  | ||||||
| ############################################################################### |  | ||||||
| #*.sln       merge=binary |  | ||||||
| #*.csproj    merge=binary |  | ||||||
| #*.vbproj    merge=binary |  | ||||||
| #*.vcxproj   merge=binary |  | ||||||
| #*.vcproj    merge=binary |  | ||||||
| #*.dbproj    merge=binary |  | ||||||
| #*.fsproj    merge=binary |  | ||||||
| #*.lsproj    merge=binary |  | ||||||
| #*.wixproj   merge=binary |  | ||||||
| #*.modelproj merge=binary |  | ||||||
| #*.sqlproj   merge=binary |  | ||||||
| #*.wwaproj   merge=binary |  | ||||||
|  |  | ||||||
| ############################################################################### |  | ||||||
| # behavior for image files |  | ||||||
| # |  | ||||||
| # image files are treated as binary by default. |  | ||||||
| ############################################################################### |  | ||||||
| #*.jpg   binary |  | ||||||
| #*.png   binary |  | ||||||
| #*.gif   binary |  | ||||||
|  |  | ||||||
| ############################################################################### |  | ||||||
| # diff behavior for common document formats |  | ||||||
| #  |  | ||||||
| # Convert binary document formats to text before diffing them. This feature |  | ||||||
| # is only available from the command line. Turn it on by uncommenting the  |  | ||||||
| # entries below. |  | ||||||
| ############################################################################### |  | ||||||
| #*.doc   diff=astextplain |  | ||||||
| #*.DOC   diff=astextplain |  | ||||||
| #*.docx  diff=astextplain |  | ||||||
| #*.DOCX  diff=astextplain |  | ||||||
| #*.dot   diff=astextplain |  | ||||||
| #*.DOT   diff=astextplain |  | ||||||
| #*.pdf   diff=astextplain |  | ||||||
| #*.PDF   diff=astextplain |  | ||||||
| #*.rtf   diff=astextplain |  | ||||||
| #*.RTF   diff=astextplain |  | ||||||
							
								
								
									
										22
									
								
								.github/workflows/CD.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/CD.yml
									
									
									
									
										vendored
									
									
								
							| @@ -3,23 +3,23 @@ name: CD | |||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     tags: |     tags: | ||||||
|     - '*' |       - "*" | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout |       - name: Checkout | ||||||
|       uses: actions/checkout@v2 |         uses: actions/checkout@v2.3.3 | ||||||
|  |  | ||||||
|     - name: Install .NET Core |       - name: Install .NET | ||||||
|       uses: actions/setup-dotnet@v1.4.0 |         uses: actions/setup-dotnet@v1.7.2 | ||||||
|       with: |         with: | ||||||
|         dotnet-version: 3.1.100 |           dotnet-version: 5.0.100 | ||||||
|  |  | ||||||
|     - name: Pack |       - name: Pack | ||||||
|       run: dotnet pack CliFx --configuration Release |         run: dotnet pack CliFx --configuration Release | ||||||
|  |  | ||||||
|     - name: Deploy |       - name: Deploy | ||||||
|       run: dotnet nuget push CliFx/bin/Release/*.nupkg -s nuget.org -k ${{secrets.NUGET_TOKEN}} |         run: dotnet nuget push CliFx/bin/Release/*.nupkg -s nuget.org -k ${{ secrets.NUGET_TOKEN }} | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								.github/workflows/CI.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.github/workflows/CI.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,19 +11,18 @@ jobs: | |||||||
|         os: [ubuntu-latest, windows-latest, macos-latest] |         os: [ubuntu-latest, windows-latest, macos-latest] | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout |       - name: Checkout | ||||||
|       uses: actions/checkout@v2 |         uses: actions/checkout@v2.3.3 | ||||||
|  |  | ||||||
|     - name: Install .NET Core |       - name: Install .NET | ||||||
|       uses: actions/setup-dotnet@v1.4.0 |         uses: actions/setup-dotnet@v1.7.2 | ||||||
|       with: |         with: | ||||||
|         dotnet-version: 3.1.100 |           dotnet-version: 5.0.100 | ||||||
|  |  | ||||||
|     - name: Build & test |       - name: Build & test | ||||||
|       run: dotnet test --configuration Release |         run: dotnet test --configuration Release --logger GitHubActions | ||||||
|  |  | ||||||
|     - name: Upload coverage |       - name: Upload coverage | ||||||
|       uses: codecov/codecov-action@v1.0.5 |         uses: codecov/codecov-action@v1.0.5 | ||||||
|       with: |         with: | ||||||
|         token: ${{ secrets.CODECOV_TOKEN }} |           token: ${{ secrets.CODECOV_TOKEN }} | ||||||
|         file: CliFx.Tests/bin/Release/Coverage.xml |  | ||||||
|   | |||||||
							
								
								
									
										332
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										332
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,341 +1,21 @@ | |||||||
| ## Ignore Visual Studio temporary files, build results, and |  | ||||||
| ## files generated by popular Visual Studio add-ons. |  | ||||||
| ## |  | ||||||
| ## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore |  | ||||||
|  |  | ||||||
| # User-specific files | # User-specific files | ||||||
| *.rsuser |  | ||||||
| *.suo | *.suo | ||||||
| *.user | *.user | ||||||
| *.userosscache | *.userosscache | ||||||
| *.sln.docstates | *.sln.docstates | ||||||
|  | .idea/ | ||||||
| # User-specific files (MonoDevelop/Xamarin Studio) |  | ||||||
| *.userprefs |  | ||||||
|  |  | ||||||
| # Build results | # Build results | ||||||
| [Dd]ebug/ | [Dd]ebug/ | ||||||
| [Dd]ebugPublic/ | [Dd]ebugPublic/ | ||||||
| [Rr]elease/ | [Rr]elease/ | ||||||
| [Rr]eleases/ | [Rr]eleases/ | ||||||
| x64/ | [Xx]64/ | ||||||
| x86/ | [Xx]86/ | ||||||
| [Aa][Rr][Mm]/ | [Bb]uild/ | ||||||
| [Aa][Rr][Mm]64/ |  | ||||||
| bld/ | bld/ | ||||||
| [Bb]in/ | [Bb]in/ | ||||||
| [Oo]bj/ | [Oo]bj/ | ||||||
| [Ll]og/ |  | ||||||
|  |  | ||||||
| # Visual Studio 2015/2017 cache/options directory | # Coverage | ||||||
| .vs/ | *.opencover.xml | ||||||
| # Uncomment if you have tasks that create the project's static files in wwwroot |  | ||||||
| #wwwroot/ |  | ||||||
|  |  | ||||||
| # Visual Studio 2017 auto generated files |  | ||||||
| Generated\ Files/ |  | ||||||
|  |  | ||||||
| # MSTest test Results |  | ||||||
| [Tt]est[Rr]esult*/ |  | ||||||
| [Bb]uild[Ll]og.* |  | ||||||
|  |  | ||||||
| # NUNIT |  | ||||||
| *.VisualState.xml |  | ||||||
| TestResult.xml |  | ||||||
|  |  | ||||||
| # Build Results of an ATL Project |  | ||||||
| [Dd]ebugPS/ |  | ||||||
| [Rr]eleasePS/ |  | ||||||
| dlldata.c |  | ||||||
|  |  | ||||||
| # Benchmark Results |  | ||||||
| BenchmarkDotNet.Artifacts/ |  | ||||||
|  |  | ||||||
| # .NET Core |  | ||||||
| project.lock.json |  | ||||||
| project.fragment.lock.json |  | ||||||
| artifacts/ |  | ||||||
|  |  | ||||||
| # StyleCop |  | ||||||
| StyleCopReport.xml |  | ||||||
|  |  | ||||||
| # Files built by Visual Studio |  | ||||||
| *_i.c |  | ||||||
| *_p.c |  | ||||||
| *_h.h |  | ||||||
| *.ilk |  | ||||||
| *.meta |  | ||||||
| *.obj |  | ||||||
| *.iobj |  | ||||||
| *.pch |  | ||||||
| *.pdb |  | ||||||
| *.ipdb |  | ||||||
| *.pgc |  | ||||||
| *.pgd |  | ||||||
| *.rsp |  | ||||||
| *.sbr |  | ||||||
| *.tlb |  | ||||||
| *.tli |  | ||||||
| *.tlh |  | ||||||
| *.tmp |  | ||||||
| *.tmp_proj |  | ||||||
| *_wpftmp.csproj |  | ||||||
| *.log |  | ||||||
| *.vspscc |  | ||||||
| *.vssscc |  | ||||||
| .builds |  | ||||||
| *.pidb |  | ||||||
| *.svclog |  | ||||||
| *.scc |  | ||||||
|  |  | ||||||
| # Chutzpah Test files |  | ||||||
| _Chutzpah* |  | ||||||
|  |  | ||||||
| # Visual C++ cache files |  | ||||||
| ipch/ |  | ||||||
| *.aps |  | ||||||
| *.ncb |  | ||||||
| *.opendb |  | ||||||
| *.opensdf |  | ||||||
| *.sdf |  | ||||||
| *.cachefile |  | ||||||
| *.VC.db |  | ||||||
| *.VC.VC.opendb |  | ||||||
|  |  | ||||||
| # Visual Studio profiler |  | ||||||
| *.psess |  | ||||||
| *.vsp |  | ||||||
| *.vspx |  | ||||||
| *.sap |  | ||||||
|  |  | ||||||
| # Visual Studio Trace Files |  | ||||||
| *.e2e |  | ||||||
|  |  | ||||||
| # TFS 2012 Local Workspace |  | ||||||
| $tf/ |  | ||||||
|  |  | ||||||
| # Guidance Automation Toolkit |  | ||||||
| *.gpState |  | ||||||
|  |  | ||||||
| # ReSharper is a .NET coding add-in |  | ||||||
| _ReSharper*/ |  | ||||||
| *.[Rr]e[Ss]harper |  | ||||||
| *.DotSettings.user |  | ||||||
|  |  | ||||||
| # JustCode is a .NET coding add-in |  | ||||||
| .JustCode |  | ||||||
|  |  | ||||||
| # TeamCity is a build add-in |  | ||||||
| _TeamCity* |  | ||||||
|  |  | ||||||
| # DotCover is a Code Coverage Tool |  | ||||||
| *.dotCover |  | ||||||
|  |  | ||||||
| # AxoCover is a Code Coverage Tool |  | ||||||
| .axoCover/* |  | ||||||
| !.axoCover/settings.json |  | ||||||
|  |  | ||||||
| # Visual Studio code coverage results |  | ||||||
| *.coverage |  | ||||||
| *.coveragexml |  | ||||||
|  |  | ||||||
| # NCrunch |  | ||||||
| _NCrunch_* |  | ||||||
| .*crunch*.local.xml |  | ||||||
| nCrunchTemp_* |  | ||||||
| .ncrunchsolution |  | ||||||
|  |  | ||||||
| # MightyMoose |  | ||||||
| *.mm.* |  | ||||||
| AutoTest.Net/ |  | ||||||
|  |  | ||||||
| # Web workbench (sass) |  | ||||||
| .sass-cache/ |  | ||||||
|  |  | ||||||
| # Installshield output folder |  | ||||||
| [Ee]xpress/ |  | ||||||
|  |  | ||||||
| # DocProject is a documentation generator add-in |  | ||||||
| DocProject/buildhelp/ |  | ||||||
| DocProject/Help/*.HxT |  | ||||||
| DocProject/Help/*.HxC |  | ||||||
| DocProject/Help/*.hhc |  | ||||||
| DocProject/Help/*.hhk |  | ||||||
| DocProject/Help/*.hhp |  | ||||||
| DocProject/Help/Html2 |  | ||||||
| DocProject/Help/html |  | ||||||
|  |  | ||||||
| # Click-Once directory |  | ||||||
| publish/ |  | ||||||
|  |  | ||||||
| # Publish Web Output |  | ||||||
| *.[Pp]ublish.xml |  | ||||||
| *.azurePubxml |  | ||||||
| # Note: Comment the next line if you want to checkin your web deploy settings, |  | ||||||
| # but database connection strings (with potential passwords) will be unencrypted |  | ||||||
| *.pubxml |  | ||||||
| *.publishproj |  | ||||||
|  |  | ||||||
| # Microsoft Azure Web App publish settings. Comment the next line if you want to |  | ||||||
| # checkin your Azure Web App publish settings, but sensitive information contained |  | ||||||
| # in these scripts will be unencrypted |  | ||||||
| PublishScripts/ |  | ||||||
|  |  | ||||||
| # NuGet Packages |  | ||||||
| *.nupkg |  | ||||||
| # The packages folder can be ignored because of Package Restore |  | ||||||
| **/[Pp]ackages/* |  | ||||||
| # except build/, which is used as an MSBuild target. |  | ||||||
| !**/[Pp]ackages/build/ |  | ||||||
| # Uncomment if necessary however generally it will be regenerated when needed |  | ||||||
| #!**/[Pp]ackages/repositories.config |  | ||||||
| # NuGet v3's project.json files produces more ignorable files |  | ||||||
| *.nuget.props |  | ||||||
| *.nuget.targets |  | ||||||
|  |  | ||||||
| # Microsoft Azure Build Output |  | ||||||
| csx/ |  | ||||||
| *.build.csdef |  | ||||||
|  |  | ||||||
| # Microsoft Azure Emulator |  | ||||||
| ecf/ |  | ||||||
| rcf/ |  | ||||||
|  |  | ||||||
| # Windows Store app package directories and files |  | ||||||
| AppPackages/ |  | ||||||
| BundleArtifacts/ |  | ||||||
| Package.StoreAssociation.xml |  | ||||||
| _pkginfo.txt |  | ||||||
| *.appx |  | ||||||
|  |  | ||||||
| # Visual Studio cache files |  | ||||||
| # files ending in .cache can be ignored |  | ||||||
| *.[Cc]ache |  | ||||||
| # but keep track of directories ending in .cache |  | ||||||
| !?*.[Cc]ache/ |  | ||||||
|  |  | ||||||
| # Others |  | ||||||
| ClientBin/ |  | ||||||
| ~$* |  | ||||||
| *~ |  | ||||||
| *.dbmdl |  | ||||||
| *.dbproj.schemaview |  | ||||||
| *.jfm |  | ||||||
| *.pfx |  | ||||||
| *.publishsettings |  | ||||||
| orleans.codegen.cs |  | ||||||
|  |  | ||||||
| # Including strong name files can present a security risk |  | ||||||
| # (https://github.com/github/gitignore/pull/2483#issue-259490424) |  | ||||||
| #*.snk |  | ||||||
|  |  | ||||||
| # Since there are multiple workflows, uncomment next line to ignore bower_components |  | ||||||
| # (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) |  | ||||||
| #bower_components/ |  | ||||||
|  |  | ||||||
| # RIA/Silverlight projects |  | ||||||
| Generated_Code/ |  | ||||||
|  |  | ||||||
| # Backup & report files from converting an old project file |  | ||||||
| # to a newer Visual Studio version. Backup files are not needed, |  | ||||||
| # because we have git ;-) |  | ||||||
| _UpgradeReport_Files/ |  | ||||||
| Backup*/ |  | ||||||
| UpgradeLog*.XML |  | ||||||
| UpgradeLog*.htm |  | ||||||
| ServiceFabricBackup/ |  | ||||||
| *.rptproj.bak |  | ||||||
|  |  | ||||||
| # SQL Server files |  | ||||||
| *.mdf |  | ||||||
| *.ldf |  | ||||||
| *.ndf |  | ||||||
|  |  | ||||||
| # Business Intelligence projects |  | ||||||
| *.rdl.data |  | ||||||
| *.bim.layout |  | ||||||
| *.bim_*.settings |  | ||||||
| *.rptproj.rsuser |  | ||||||
| *- Backup*.rdl |  | ||||||
|  |  | ||||||
| # Microsoft Fakes |  | ||||||
| FakesAssemblies/ |  | ||||||
|  |  | ||||||
| # GhostDoc plugin setting file |  | ||||||
| *.GhostDoc.xml |  | ||||||
|  |  | ||||||
| # Node.js Tools for Visual Studio |  | ||||||
| .ntvs_analysis.dat |  | ||||||
| node_modules/ |  | ||||||
|  |  | ||||||
| # Visual Studio 6 build log |  | ||||||
| *.plg |  | ||||||
|  |  | ||||||
| # Visual Studio 6 workspace options file |  | ||||||
| *.opt |  | ||||||
|  |  | ||||||
| # Visual Studio 6 auto-generated workspace file (contains which files were open etc.) |  | ||||||
| *.vbw |  | ||||||
|  |  | ||||||
| # Visual Studio LightSwitch build output |  | ||||||
| **/*.HTMLClient/GeneratedArtifacts |  | ||||||
| **/*.DesktopClient/GeneratedArtifacts |  | ||||||
| **/*.DesktopClient/ModelManifest.xml |  | ||||||
| **/*.Server/GeneratedArtifacts |  | ||||||
| **/*.Server/ModelManifest.xml |  | ||||||
| _Pvt_Extensions |  | ||||||
|  |  | ||||||
| # Paket dependency manager |  | ||||||
| .paket/paket.exe |  | ||||||
| paket-files/ |  | ||||||
|  |  | ||||||
| # FAKE - F# Make |  | ||||||
| .fake/ |  | ||||||
|  |  | ||||||
| # JetBrains Rider |  | ||||||
| .idea/ |  | ||||||
| *.sln.iml |  | ||||||
|  |  | ||||||
| # CodeRush personal settings |  | ||||||
| .cr/personal |  | ||||||
|  |  | ||||||
| # Python Tools for Visual Studio (PTVS) |  | ||||||
| __pycache__/ |  | ||||||
| *.pyc |  | ||||||
|  |  | ||||||
| # Cake - Uncomment if you are using it |  | ||||||
| # tools/** |  | ||||||
| # !tools/packages.config |  | ||||||
|  |  | ||||||
| # Tabs Studio |  | ||||||
| *.tss |  | ||||||
|  |  | ||||||
| # Telerik's JustMock configuration file |  | ||||||
| *.jmconfig |  | ||||||
|  |  | ||||||
| # BizTalk build output |  | ||||||
| *.btp.cs |  | ||||||
| *.btm.cs |  | ||||||
| *.odx.cs |  | ||||||
| *.xsd.cs |  | ||||||
|  |  | ||||||
| # OpenCover UI analysis results |  | ||||||
| OpenCover/ |  | ||||||
|  |  | ||||||
| # Azure Stream Analytics local run output |  | ||||||
| ASALocalRun/ |  | ||||||
|  |  | ||||||
| # MSBuild Binary and Structured Log |  | ||||||
| *.binlog |  | ||||||
|  |  | ||||||
| # NVidia Nsight GPU debugger configuration file |  | ||||||
| *.nvuser |  | ||||||
|  |  | ||||||
| # MFractors (Xamarin productivity tool) working folder |  | ||||||
| .mfractor/ |  | ||||||
|  |  | ||||||
| # Local History for Visual Studio |  | ||||||
| .localhistory/ |  | ||||||
|  |  | ||||||
| # BeatPulse healthcheck temp database |  | ||||||
| healthchecksdb |  | ||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 24 KiB | 
							
								
								
									
										46
									
								
								Changelog.md
									
									
									
									
									
								
							
							
						
						
									
										46
									
								
								Changelog.md
									
									
									
									
									
								
							| @@ -1,3 +1,49 @@ | |||||||
|  | ### v1.6 (06-Dec-2020) | ||||||
|  |  | ||||||
|  | - Added support for custom value validators. You can now create a type that inherits from `CliFx.ArgumentValueValidator<T>` to implement reusable validation logic for command arguments. To use a validator, include it in the `Validators` property on the `CommandOption` or `CommandParameter` attribute. (Thanks [@Oleksandr Shustov](https://github.com/AlexandrShustov)) | ||||||
|  | - Added `CliFx.ArgumentValueConverter<T>` class that you can inherit from to implement custom value converters. `CliFx.IArgumentValueConverter` interface is still available, but it is recommended to inherit from the generic class instead, due to the type safety it provides. The interface may become internal or get removed in one of the future major versions. | ||||||
|  | - Updated requirements for option names and short names: short names now must be letter characters (lowercase or uppercase), while names must now start with a letter character. This means option names can no longer start with a digit or a special character. This change makes it possible to pass negative number values without the need to quote them, i.e. `--my-number -5`. | ||||||
|  |  | ||||||
|  | ### v1.5 (23-Oct-2020) | ||||||
|  |  | ||||||
|  | - Added pretty-printing for unhandled exceptions thrown from within the application. This makes the errors easier to parse visually and should help in troubleshooting. This change does not affect `CommandException`, as it already has special treatment. (Thanks [@Mårten Åsberg](https://github.com/89netraM)) | ||||||
|  | - Added support for custom value converters. You can now create a type that implements `CliFx.IArgumentValueConverter` and specify it as a converter for your parameters or options via the `Converter` named property. This should enable conversion between raw argument values and custom types which are not string-initializable. (Thanks [@Oleksandr Shustov](https://github.com/AlexandrShustov)) | ||||||
|  | - Improved help text so that it also shows minimal usage examples for child and descendant commands, besides the actual command it was requested on. This should improve user experience for applications with many nested commands. (Thanks [@Nikiforov Alexey](https://github.com/NikiforovAll)) | ||||||
|  |  | ||||||
|  | ### v1.4 (20-Aug-2020) | ||||||
|  |  | ||||||
|  | - Added `VirtualConsole.CreateBuffered()` method to simplify test setup when using in-memory backing stores for output and error streams. Please refer to the readme for updated recommendations on how to test applications built with CliFx. | ||||||
|  | - Added generic `CliApplicationBuilder.AddCommand<TCommand>()`. This overload simplifies adding commands one-by-one as it also checks that the type implements `ICommand`. | ||||||
|  |  | ||||||
|  | ### v1.3.2 (31-Jul-2020) | ||||||
|  |  | ||||||
|  | - Fixed an issue where a command was incorrectly allowed to execute when the user did not specify any value for a non-scalar parameter. Since they are always required, a parameter needs to be bound to (at least) one value. (Thanks [@Daniel Hix](https://github.com/ADustyOldMuffin)) | ||||||
|  | - Fixed an issue where `CliApplication.RunAsync(...)` threw `ArgumentException` if there were two environment variables, whose names differed only in case. Environment variable names are now treated case-sensitively. (Thanks [@Ron Myers](https://github.com/ron-myers)) | ||||||
|  |  | ||||||
|  | ### v1.3.1 (19-Jul-2020) | ||||||
|  |  | ||||||
|  | - Running the application with the debug directive (`myapp [debug]`) will now also try to launch a debugger instance. In most cases it will save time as you won't need to attach the debugger manually. (Thanks [@Volodymyr Shkolka](https://github.com/BlackGad)) | ||||||
|  | - Fixed an issue where unhandled generic exceptions (i.e. not `CommandException`) sometimes caused the application to incorrectly return successful exit code due to an overflow issue on Unix systems. Starting from this version, all unhandled generic exceptions will produce `1` as the exit code when thrown. Instances of `CommandException` can still be configured to return any specified exit code, but it's recommended to constrain the values between `1` and `255` to avoid overflow issues. (Thanks [@Ihor Nechyporuk](https://github.com/inech)) | ||||||
|  |  | ||||||
|  | ### v1.3 (23-May-2020) | ||||||
|  |  | ||||||
|  | - Changed analyzers to report errors instead of warnings. If you find that some analyzer works incorrectly, please report it on GitHub. You can also configure inspection severity overrides in your project if you need to. | ||||||
|  | - Improved help text by showing default values for non-required options. This only works on types that have a custom override for `ToString()` method. Additionally, if the type implements `IFormattable`, the overload with a format provider will be used instead. (Thanks [@Domn Werner](https://github.com/domn1995)) | ||||||
|  | - Changed default version text to only show 3 version components instead of 4, if the last component (revision) is not specified or is zero. This makes the default version text compliant with semantic versioning. | ||||||
|  | - Fixed an issue where it was possible to define a command with an option that has the same name or short name as built-in help or version options. Previously it would lead to the user-defined option being ignored in favor of the built-in option. Now this will throw an exception instead. | ||||||
|  | - Changed the underlying representation of `StreamReader`/`StreamWriter` objects used in `SystemConsole` and `VirtualConsole` to be thread-safe. | ||||||
|  |  | ||||||
|  | ### v1.2 (11-May-2020) | ||||||
|  |  | ||||||
|  | - Added built-in Roslyn analyzers that help catch incorrect usage of the library. Currently, all analyzers report issues as warnings so as to not prevent the project from building. In the future that may change. | ||||||
|  | - Added an optional parameter to `new CommandException(...)` called `showHelp` which can be used to instruct CliFx to show help for the current command after printing the error. (Thanks [@Domn Werner](https://github.com/domn1995)) | ||||||
|  | - Improved help text shown for enum options and parameters by providing the list of valid values that the enum can accept. (Thanks [@Domn Werner](https://github.com/domn1995)) | ||||||
|  | - Fixed an issue where it was possible to set an option without providing a value, while the option was marked as required. | ||||||
|  | - Fixed an issue where it was possible to configure an option with an empty name or a name consisting of a single character. If you want to use a single character as a name, you should set the option's short name instead. | ||||||
|  | - Added `CursorLeft` and `CursorTop` properties to `IConsole` and its implementations. In `VirtualConsole`, these are just auto-properties. | ||||||
|  | - Improved exception messages. | ||||||
|  | - Improved exceptions related to user input by also showing help text after the error message. (Thanks [@Domn Werner](https://github.com/domn1995)) | ||||||
|  |  | ||||||
| ### v1.1 (16-Mar-2020) | ### v1.1 (16-Mar-2020) | ||||||
|  |  | ||||||
| - Changed `IConsole` interface (and as a result, `SystemConsole` and `VirtualConsole`) to support writing binary data. Instead of `TextReader`/`TextWriter` instances, the streams are now exposed as `StreamReader`/`StreamWriter` which provide the `BaseStream` property that allows raw access. Existing usages inside commands should remain the same because `StreamReader`/`StreamWriter` are compatible with their base classes `TextReader`/`TextWriter`, but if you were using `VirtualConsole` in tests, you may have to update it to the new API. Refer to the readme for more info. | - Changed `IConsole` interface (and as a result, `SystemConsole` and `VirtualConsole`) to support writing binary data. Instead of `TextReader`/`TextWriter` instances, the streams are now exposed as `StreamReader`/`StreamWriter` which provide the `BaseStream` property that allows raw access. Existing usages inside commands should remain the same because `StreamReader`/`StreamWriter` are compatible with their base classes `TextReader`/`TextWriter`, but if you were using `VirtualConsole` in tests, you may have to update it to the new API. Refer to the readme for more info. | ||||||
|   | |||||||
							
								
								
									
										43
									
								
								CliFx.Analyzers.Tests/AnalyzerTestCase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								CliFx.Analyzers.Tests/AnalyzerTestCase.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers.Tests | ||||||
|  | { | ||||||
|  |     public class AnalyzerTestCase | ||||||
|  |     { | ||||||
|  |         public string Name { get; } | ||||||
|  |  | ||||||
|  |         public IReadOnlyList<DiagnosticDescriptor> TestedDiagnostics { get; } | ||||||
|  |  | ||||||
|  |         public IReadOnlyList<string> SourceCodes { get; } | ||||||
|  |  | ||||||
|  |         public AnalyzerTestCase( | ||||||
|  |             string name, | ||||||
|  |             IReadOnlyList<DiagnosticDescriptor> testedDiagnostics, | ||||||
|  |             IReadOnlyList<string> sourceCodes) | ||||||
|  |         { | ||||||
|  |             Name = name; | ||||||
|  |             TestedDiagnostics = testedDiagnostics; | ||||||
|  |             SourceCodes = sourceCodes; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public AnalyzerTestCase( | ||||||
|  |             string name, | ||||||
|  |             IReadOnlyList<DiagnosticDescriptor> testedDiagnostics, | ||||||
|  |             string sourceCode) | ||||||
|  |             : this(name, testedDiagnostics, new[] {sourceCode}) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public AnalyzerTestCase( | ||||||
|  |             string name, | ||||||
|  |             DiagnosticDescriptor testedDiagnostic, | ||||||
|  |             string sourceCode) | ||||||
|  |             : this(name, new[] {testedDiagnostic}, sourceCode) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public override string ToString() => $"{Name} [{string.Join(", ", TestedDiagnostics.Select(d => d.Id))}]"; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								CliFx.Analyzers.Tests/CliFx.Analyzers.Tests.csproj
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								CliFx.Analyzers.Tests/CliFx.Analyzers.Tests.csproj
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |   <Import Project="../CliFx.props" /> | ||||||
|  |  | ||||||
|  |   <PropertyGroup> | ||||||
|  |     <TargetFramework>net5.0</TargetFramework> | ||||||
|  |     <IsPackable>false</IsPackable> | ||||||
|  |     <IsTestProject>true</IsTestProject> | ||||||
|  |     <CollectCoverage>true</CollectCoverage> | ||||||
|  |     <CoverletOutputFormat>opencover</CoverletOutputFormat> | ||||||
|  |   </PropertyGroup> | ||||||
|  |  | ||||||
|  |   <ItemGroup> | ||||||
|  |     <PackageReference Include="Gu.Roslyn.Asserts" Version="3.3.1" /> | ||||||
|  |     <PackageReference Include="GitHubActionsTestLogger" Version="1.1.2" /> | ||||||
|  |     <PackageReference Include="FluentAssertions" Version="5.10.3" /> | ||||||
|  |     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.7.1" /> | ||||||
|  |     <PackageReference Include="Microsoft.CodeAnalysis.CSharp.Workspaces" Version="3.4.0" /> | ||||||
|  |     <PackageReference Include="xunit" Version="2.4.0" /> | ||||||
|  |     <PackageReference Include="xunit.runner.visualstudio" Version="2.4.3" PrivateAssets="all" /> | ||||||
|  |     <PackageReference Include="coverlet.msbuild" Version="2.9.0" PrivateAssets="all" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |  | ||||||
|  |   <ItemGroup> | ||||||
|  |     <ProjectReference Include="..\CliFx.Analyzers\CliFx.Analyzers.csproj" /> | ||||||
|  |     <ProjectReference Include="..\CliFx\CliFx.csproj" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |  | ||||||
|  | </Project> | ||||||
							
								
								
									
										719
									
								
								CliFx.Analyzers.Tests/CommandSchemaAnalyzerTests.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										719
									
								
								CliFx.Analyzers.Tests/CommandSchemaAnalyzerTests.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,719 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Analyzers.Tests.Internal; | ||||||
|  | using Microsoft.CodeAnalysis.Diagnostics; | ||||||
|  | using Xunit; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers.Tests | ||||||
|  | { | ||||||
|  |     public class CommandSchemaAnalyzerTests | ||||||
|  |     { | ||||||
|  |         private static DiagnosticAnalyzer Analyzer { get; } = new CommandSchemaAnalyzer(); | ||||||
|  |  | ||||||
|  |         public static IEnumerable<object[]> GetValidCases() | ||||||
|  |         { | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Non-command type", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class Foo | ||||||
|  | { | ||||||
|  |     public int Bar { get; set; } = 5; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Command implements interface and has attribute", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Command doesn't have an attribute but is an abstract type", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public abstract class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameters with unique order", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(13)] | ||||||
|  |     public string ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(15)] | ||||||
|  |     public string ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameters with unique names", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(13, Name = ""foo"")] | ||||||
|  |     public string ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(15, Name = ""bar"")] | ||||||
|  |     public string ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Single non-scalar parameter", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(1)] | ||||||
|  |     public string ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(2)] | ||||||
|  |     public HashSet<string> ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Non-scalar parameter is last in order", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(1)] | ||||||
|  |     public string ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(2)] | ||||||
|  |     public IReadOnlyList<string> ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameter with valid converter", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyConverter : ArgumentValueConverter<string> | ||||||
|  | { | ||||||
|  |     public string ConvertFrom(string value) => value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(0, Converter = typeof(MyConverter))] | ||||||
|  |     public string Param { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameter with valid validator", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyValidator : ArgumentValueValidator<string> | ||||||
|  | { | ||||||
|  |     public ValidationResult Validate(string value) => ValidationResult.Ok(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(0, Validators = new[] {typeof(MyValidator)})] | ||||||
|  |     public string Param { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with a proper name", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption(""foo"")] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with a proper name and short name", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption(""foo"", 'f')] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Options with unique names", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption(""foo"")] | ||||||
|  |     public string OptionA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandOption(""bar"")] | ||||||
|  |     public string OptionB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Options with unique short names", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('f')] | ||||||
|  |     public string OptionA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandOption('x')] | ||||||
|  |     public string OptionB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Options with unique environment variable names", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('a', EnvironmentVariableName = ""env_var_a"")] | ||||||
|  |     public string OptionA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandOption('b', EnvironmentVariableName = ""env_var_b"")] | ||||||
|  |     public string OptionB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with valid converter", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyConverter : ArgumentValueConverter<string> | ||||||
|  | { | ||||||
|  |     public string ConvertFrom(string value) => value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('o', Converter = typeof(MyConverter))] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with valid validator", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyValidator : ArgumentValueValidator<string> | ||||||
|  | { | ||||||
|  |     public ValidationResult Validate(string value) => ValidationResult.Ok(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('o', Validators = new[] {typeof(MyValidator)})] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public static IEnumerable<object[]> GetInvalidCases() | ||||||
|  |         { | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Command is missing the attribute", | ||||||
|  |                     DiagnosticDescriptors.CliFx0002, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Command doesn't implement the interface", | ||||||
|  |                     DiagnosticDescriptors.CliFx0001, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameters with duplicate order", | ||||||
|  |                     DiagnosticDescriptors.CliFx0021, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(13)] | ||||||
|  |     public string ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(13)] | ||||||
|  |     public string ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameters with duplicate names", | ||||||
|  |                     DiagnosticDescriptors.CliFx0022, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(13, Name = ""foo"")] | ||||||
|  |     public string ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(15, Name = ""foo"")] | ||||||
|  |     public string ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Multiple non-scalar parameters", | ||||||
|  |                     DiagnosticDescriptors.CliFx0023, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(1)] | ||||||
|  |     public IReadOnlyList<string> ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(2)] | ||||||
|  |     public HashSet<string> ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Non-last non-scalar parameter", | ||||||
|  |                     DiagnosticDescriptors.CliFx0024, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(1)] | ||||||
|  |     public IReadOnlyList<string> ParamA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandParameter(2)] | ||||||
|  |     public string ParamB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameter with invalid converter", | ||||||
|  |                     DiagnosticDescriptors.CliFx0025, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyConverter | ||||||
|  | { | ||||||
|  |     public object ConvertFrom(string value) => value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(0, Converter = typeof(MyConverter))] | ||||||
|  |     public string Param { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Parameter with invalid validator", | ||||||
|  |                     DiagnosticDescriptors.CliFx0026, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyValidator | ||||||
|  | { | ||||||
|  |     public ValidationResult Validate(string value) => ValidationResult.Ok(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandParameter(0, Validators = new[] {typeof(MyValidator)})] | ||||||
|  |     public string Param { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with an empty name", | ||||||
|  |                     DiagnosticDescriptors.CliFx0041, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption("""")] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with a name which is too short", | ||||||
|  |                     DiagnosticDescriptors.CliFx0042, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption(""a"")] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Options with duplicate names", | ||||||
|  |                     DiagnosticDescriptors.CliFx0043, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption(""foo"")] | ||||||
|  |     public string OptionA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandOption(""foo"")] | ||||||
|  |     public string OptionB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Options with duplicate short names", | ||||||
|  |                     DiagnosticDescriptors.CliFx0044, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('f')] | ||||||
|  |     public string OptionA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandOption('f')] | ||||||
|  |     public string OptionB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Options with duplicate environment variable names", | ||||||
|  |                     DiagnosticDescriptors.CliFx0045, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('a', EnvironmentVariableName = ""env_var"")] | ||||||
|  |     public string OptionA { get; set; } | ||||||
|  |      | ||||||
|  |     [CommandOption('b', EnvironmentVariableName = ""env_var"")] | ||||||
|  |     public string OptionB { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with invalid converter", | ||||||
|  |                     DiagnosticDescriptors.CliFx0046, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyConverter | ||||||
|  | { | ||||||
|  |     public object ConvertFrom(string value) => value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('o', Converter = typeof(MyConverter))] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with invalid validator", | ||||||
|  |                     DiagnosticDescriptors.CliFx0047, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | public class MyValidator | ||||||
|  | { | ||||||
|  |     public ValidationResult Validate(string value) => ValidationResult.Ok(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('o', Validators = new[] {typeof(MyValidator)})] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |      | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with a name that doesn't start with a letter character", | ||||||
|  |                     DiagnosticDescriptors.CliFx0048, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption(""0foo"")] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Option with a short name that isn't a letter character", | ||||||
|  |                     DiagnosticDescriptors.CliFx0049, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     [CommandOption('0')] | ||||||
|  |     public string Option { get; set; } | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Theory] | ||||||
|  |         [MemberData(nameof(GetValidCases))] | ||||||
|  |         public void Valid(AnalyzerTestCase testCase) => | ||||||
|  |             Analyzer.Should().NotProduceDiagnostics(testCase); | ||||||
|  |  | ||||||
|  |         [Theory] | ||||||
|  |         [MemberData(nameof(GetInvalidCases))] | ||||||
|  |         public void Invalid(AnalyzerTestCase testCase) => | ||||||
|  |             Analyzer.Should().ProduceDiagnostics(testCase); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										144
									
								
								CliFx.Analyzers.Tests/ConsoleUsageAnalyzerTests.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								CliFx.Analyzers.Tests/ConsoleUsageAnalyzerTests.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,144 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Analyzers.Tests.Internal; | ||||||
|  | using Microsoft.CodeAnalysis.Diagnostics; | ||||||
|  | using Xunit; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers.Tests | ||||||
|  | { | ||||||
|  |     public class ConsoleUsageAnalyzerTests | ||||||
|  |     { | ||||||
|  |         private static DiagnosticAnalyzer Analyzer { get; } = new ConsoleUsageAnalyzer(); | ||||||
|  |  | ||||||
|  |         public static IEnumerable<object[]> GetValidCases() | ||||||
|  |         { | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Using console abstraction", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |     { | ||||||
|  |         console.Output.WriteLine(""Hello world""); | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Console abstraction is not available in scope", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public void SomeOtherMethod() => Console.WriteLine(""Test""); | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public static IEnumerable<object[]> GetInvalidCases() | ||||||
|  |         { | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Not using available console abstraction in the ExecuteAsync method", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |     { | ||||||
|  |         Console.WriteLine(""Hello world""); | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Not using available console abstraction in the ExecuteAsync method when writing stderr", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |     { | ||||||
|  |         Console.Error.WriteLine(""Hello world""); | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Not using available console abstraction while referencing System.Console by full name", | ||||||
|  |                     Analyzer.SupportedDiagnostics, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |     { | ||||||
|  |         System.Console.Error.WriteLine(""Hello world""); | ||||||
|  |         return default; | ||||||
|  |     } | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             yield return new object[] | ||||||
|  |             { | ||||||
|  |                 new AnalyzerTestCase( | ||||||
|  |                     "Not using available console abstraction in another method", | ||||||
|  |                     DiagnosticDescriptors.CliFx0100, | ||||||
|  |  | ||||||
|  |                     // language=cs | ||||||
|  |                     @" | ||||||
|  | [Command] | ||||||
|  | public class MyCommand : ICommand | ||||||
|  | { | ||||||
|  |     public void SomeOtherMethod(IConsole console) => Console.WriteLine(""Test""); | ||||||
|  |  | ||||||
|  |     public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  | }" | ||||||
|  |                 ) | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Theory] | ||||||
|  |         [MemberData(nameof(GetValidCases))] | ||||||
|  |         public void Valid(AnalyzerTestCase testCase) => | ||||||
|  |             Analyzer.Should().NotProduceDiagnostics(testCase); | ||||||
|  |  | ||||||
|  |         [Theory] | ||||||
|  |         [MemberData(nameof(GetInvalidCases))] | ||||||
|  |         public void Invalid(AnalyzerTestCase testCase) => | ||||||
|  |             Analyzer.Should().ProduceDiagnostics(testCase); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										107
									
								
								CliFx.Analyzers.Tests/Internal/AnalyzerAssertions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								CliFx.Analyzers.Tests/Internal/AnalyzerAssertions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using FluentAssertions.Execution; | ||||||
|  | using FluentAssertions.Primitives; | ||||||
|  | using Gu.Roslyn.Asserts; | ||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  | using Microsoft.CodeAnalysis.CSharp; | ||||||
|  | using Microsoft.CodeAnalysis.Diagnostics; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers.Tests.Internal | ||||||
|  | { | ||||||
|  |     internal partial class AnalyzerAssertions : ReferenceTypeAssertions<DiagnosticAnalyzer, AnalyzerAssertions> | ||||||
|  |     { | ||||||
|  |         protected override string Identifier { get; } = "analyzer"; | ||||||
|  |  | ||||||
|  |         public AnalyzerAssertions(DiagnosticAnalyzer analyzer) | ||||||
|  |             : base(analyzer) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public void ProduceDiagnostics( | ||||||
|  |             IReadOnlyList<DiagnosticDescriptor> diagnostics, | ||||||
|  |             IReadOnlyList<string> sourceCodes) | ||||||
|  |         { | ||||||
|  |             var producedDiagnostics = GetProducedDiagnostics(Subject, sourceCodes); | ||||||
|  |  | ||||||
|  |             var expectedIds = diagnostics.Select(d => d.Id).Distinct().OrderBy(d => d).ToArray(); | ||||||
|  |             var producedIds = producedDiagnostics.Select(d => d.Id).Distinct().OrderBy(d => d).ToArray(); | ||||||
|  |  | ||||||
|  |             var result = expectedIds.Intersect(producedIds).Count() == expectedIds.Length; | ||||||
|  |  | ||||||
|  |             Execute.Assertion.ForCondition(result).FailWith($@" | ||||||
|  | Expected and produced diagnostics do not match. | ||||||
|  |  | ||||||
|  | Expected: {string.Join(", ", expectedIds)} | ||||||
|  | Produced: {(producedIds.Any() ? string.Join(", ", producedIds) : "<none>")} | ||||||
|  | ".Trim()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public void ProduceDiagnostics(AnalyzerTestCase testCase) => | ||||||
|  |             ProduceDiagnostics(testCase.TestedDiagnostics, testCase.SourceCodes); | ||||||
|  |  | ||||||
|  |         public void NotProduceDiagnostics( | ||||||
|  |             IReadOnlyList<DiagnosticDescriptor> diagnostics, | ||||||
|  |             IReadOnlyList<string> sourceCodes) | ||||||
|  |         { | ||||||
|  |             var producedDiagnostics = GetProducedDiagnostics(Subject, sourceCodes); | ||||||
|  |  | ||||||
|  |             var expectedIds = diagnostics.Select(d => d.Id).Distinct().OrderBy(d => d).ToArray(); | ||||||
|  |             var producedIds = producedDiagnostics.Select(d => d.Id).Distinct().OrderBy(d => d).ToArray(); | ||||||
|  |  | ||||||
|  |             var result = !expectedIds.Intersect(producedIds).Any(); | ||||||
|  |  | ||||||
|  |             Execute.Assertion.ForCondition(result).FailWith($@" | ||||||
|  | Expected no produced diagnostics. | ||||||
|  |  | ||||||
|  | Produced: {string.Join(", ", producedIds)} | ||||||
|  | ".Trim()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public void NotProduceDiagnostics(AnalyzerTestCase testCase) => | ||||||
|  |             NotProduceDiagnostics(testCase.TestedDiagnostics, testCase.SourceCodes); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     internal partial class AnalyzerAssertions | ||||||
|  |     { | ||||||
|  |         private static IReadOnlyList<MetadataReference> DefaultMetadataReferences { get; } = | ||||||
|  |             MetadataReferences.Transitive(typeof(CliApplication).Assembly).ToArray(); | ||||||
|  |  | ||||||
|  |         private static string WrapCodeWithUsingDirectives(string code) | ||||||
|  |         { | ||||||
|  |             var usingDirectives = new[] | ||||||
|  |             { | ||||||
|  |                 "using System;", | ||||||
|  |                 "using System.Collections.Generic;", | ||||||
|  |                 "using System.Threading.Tasks;", | ||||||
|  |                 "using CliFx;", | ||||||
|  |                 "using CliFx.Attributes;", | ||||||
|  |                 "using CliFx.Exceptions;", | ||||||
|  |                 "using CliFx.Utilities;" | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             return | ||||||
|  |                 string.Join(Environment.NewLine, usingDirectives) + | ||||||
|  |                 Environment.NewLine + | ||||||
|  |                 code; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static IReadOnlyList<Diagnostic> GetProducedDiagnostics( | ||||||
|  |             DiagnosticAnalyzer analyzer, | ||||||
|  |             IReadOnlyList<string> sourceCodes) | ||||||
|  |         { | ||||||
|  |             var compilationOptions = new CSharpCompilationOptions(OutputKind.ConsoleApplication); | ||||||
|  |             var wrappedSourceCodes = sourceCodes.Select(WrapCodeWithUsingDirectives).ToArray(); | ||||||
|  |  | ||||||
|  |             return Analyze.GetDiagnostics(analyzer, wrappedSourceCodes, compilationOptions, DefaultMetadataReferences) | ||||||
|  |                 .SelectMany(d => d) | ||||||
|  |                 .ToArray(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     internal static class AnalyzerAssertionsExtensions | ||||||
|  |     { | ||||||
|  |         public static AnalyzerAssertions Should(this DiagnosticAnalyzer analyzer) => new AnalyzerAssertions(analyzer); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								CliFx.Analyzers/CliFx.Analyzers.csproj
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								CliFx.Analyzers/CliFx.Analyzers.csproj
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |   <Import Project="../CliFx.props" /> | ||||||
|  |  | ||||||
|  |   <PropertyGroup> | ||||||
|  |     <TargetFramework>netstandard2.0</TargetFramework> | ||||||
|  |     <Nullable>annotations</Nullable> | ||||||
|  |   </PropertyGroup> | ||||||
|  |  | ||||||
|  |   <ItemGroup> | ||||||
|  |     <PackageReference Include="Microsoft.CodeAnalysis.CSharp.Workspaces" Version="3.4.0" PrivateAssets="all" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |  | ||||||
|  | </Project> | ||||||
							
								
								
									
										423
									
								
								CliFx.Analyzers/CommandSchemaAnalyzer.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										423
									
								
								CliFx.Analyzers/CommandSchemaAnalyzer.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,423 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Collections.Immutable; | ||||||
|  | using System.Linq; | ||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  | using Microsoft.CodeAnalysis.Diagnostics; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers | ||||||
|  | { | ||||||
|  |     // TODO: split into multiple analyzers | ||||||
|  |     [DiagnosticAnalyzer(LanguageNames.CSharp)] | ||||||
|  |     public class CommandSchemaAnalyzer : DiagnosticAnalyzer | ||||||
|  |     { | ||||||
|  |         public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics { get; } = ImmutableArray.Create( | ||||||
|  |             DiagnosticDescriptors.CliFx0001, | ||||||
|  |             DiagnosticDescriptors.CliFx0002, | ||||||
|  |             DiagnosticDescriptors.CliFx0021, | ||||||
|  |             DiagnosticDescriptors.CliFx0022, | ||||||
|  |             DiagnosticDescriptors.CliFx0023, | ||||||
|  |             DiagnosticDescriptors.CliFx0024, | ||||||
|  |             DiagnosticDescriptors.CliFx0025, | ||||||
|  |             DiagnosticDescriptors.CliFx0026, | ||||||
|  |             DiagnosticDescriptors.CliFx0041, | ||||||
|  |             DiagnosticDescriptors.CliFx0042, | ||||||
|  |             DiagnosticDescriptors.CliFx0043, | ||||||
|  |             DiagnosticDescriptors.CliFx0044, | ||||||
|  |             DiagnosticDescriptors.CliFx0045, | ||||||
|  |             DiagnosticDescriptors.CliFx0046, | ||||||
|  |             DiagnosticDescriptors.CliFx0047, | ||||||
|  |             DiagnosticDescriptors.CliFx0048, | ||||||
|  |             DiagnosticDescriptors.CliFx0049 | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         private static bool IsScalarType(ITypeSymbol typeSymbol) => | ||||||
|  |             KnownSymbols.IsSystemString(typeSymbol) || | ||||||
|  |             !typeSymbol.AllInterfaces.Select(i => i.ConstructedFrom) | ||||||
|  |                 .Any(KnownSymbols.IsSystemCollectionsGenericIEnumerable); | ||||||
|  |  | ||||||
|  |         private static void CheckCommandParameterProperties( | ||||||
|  |             SymbolAnalysisContext context, | ||||||
|  |             IReadOnlyList<IPropertySymbol> properties) | ||||||
|  |         { | ||||||
|  |             var parameters = properties | ||||||
|  |                 .Select(p => | ||||||
|  |                 { | ||||||
|  |                     var attribute = p | ||||||
|  |                         .GetAttributes() | ||||||
|  |                         .First(a => KnownSymbols.IsCommandParameterAttribute(a.AttributeClass)); | ||||||
|  |  | ||||||
|  |                     var order = attribute | ||||||
|  |                         .ConstructorArguments | ||||||
|  |                         .Select(a => a.Value) | ||||||
|  |                         .FirstOrDefault() as int?; | ||||||
|  |  | ||||||
|  |                     var name = attribute | ||||||
|  |                         .NamedArguments | ||||||
|  |                         .Where(a => a.Key == "Name") | ||||||
|  |                         .Select(a => a.Value.Value) | ||||||
|  |                         .FirstOrDefault() as string; | ||||||
|  |  | ||||||
|  |                     var converter = attribute | ||||||
|  |                         .NamedArguments | ||||||
|  |                         .Where(a => a.Key == "Converter") | ||||||
|  |                         .Select(a => a.Value.Value) | ||||||
|  |                         .Cast<ITypeSymbol?>() | ||||||
|  |                         .FirstOrDefault(); | ||||||
|  |  | ||||||
|  |                     var validators = attribute | ||||||
|  |                         .NamedArguments | ||||||
|  |                         .Where(a => a.Key == "Validators") | ||||||
|  |                         .SelectMany(a => a.Value.Values) | ||||||
|  |                         .Select(c => c.Value) | ||||||
|  |                         .Cast<ITypeSymbol>() | ||||||
|  |                         .ToArray(); | ||||||
|  |  | ||||||
|  |                     return new | ||||||
|  |                     { | ||||||
|  |                         Property = p, | ||||||
|  |                         Order = order, | ||||||
|  |                         Name = name, | ||||||
|  |                         Converter = converter, | ||||||
|  |                         Validators = validators | ||||||
|  |                     }; | ||||||
|  |                 }) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             // Duplicate order | ||||||
|  |             var duplicateOrderParameters = parameters | ||||||
|  |                 .Where(p => p.Order != null) | ||||||
|  |                 .GroupBy(p => p.Order) | ||||||
|  |                 .Where(g => g.Count() > 1) | ||||||
|  |                 .SelectMany(g => g.AsEnumerable()) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var parameter in duplicateOrderParameters) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0021, parameter.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Duplicate name | ||||||
|  |             var duplicateNameParameters = parameters | ||||||
|  |                 .Where(p => !string.IsNullOrWhiteSpace(p.Name)) | ||||||
|  |                 .GroupBy(p => p.Name, StringComparer.OrdinalIgnoreCase) | ||||||
|  |                 .Where(g => g.Count() > 1) | ||||||
|  |                 .SelectMany(g => g.AsEnumerable()) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var parameter in duplicateNameParameters) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0022, parameter.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Multiple non-scalar | ||||||
|  |             var nonScalarParameters = parameters | ||||||
|  |                 .Where(p => !IsScalarType(p.Property.Type)) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             if (nonScalarParameters.Length > 1) | ||||||
|  |             { | ||||||
|  |                 foreach (var parameter in nonScalarParameters) | ||||||
|  |                 { | ||||||
|  |                     context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                         DiagnosticDescriptors.CliFx0023, parameter.Property.Locations.First() | ||||||
|  |                     )); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Non-last non-scalar | ||||||
|  |             var nonLastNonScalarParameter = parameters | ||||||
|  |                 .OrderByDescending(a => a.Order) | ||||||
|  |                 .Skip(1) | ||||||
|  |                 .LastOrDefault(p => !IsScalarType(p.Property.Type)); | ||||||
|  |  | ||||||
|  |             if (nonLastNonScalarParameter != null) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0024, nonLastNonScalarParameter.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Invalid converter | ||||||
|  |             var invalidConverterParameters = parameters | ||||||
|  |                 .Where(p => | ||||||
|  |                     p.Converter != null && | ||||||
|  |                     !p.Converter.AllInterfaces.Any(KnownSymbols.IsArgumentValueConverterInterface)) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var parameter in invalidConverterParameters) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0025, parameter.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Invalid validators | ||||||
|  |             var invalidValidatorsParameters = parameters | ||||||
|  |                 .Where(p => !p.Validators.All(v => v.AllInterfaces.Any(KnownSymbols.IsArgumentValueValidatorInterface))) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var parameter in invalidValidatorsParameters) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0026, parameter.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static void CheckCommandOptionProperties( | ||||||
|  |             SymbolAnalysisContext context, | ||||||
|  |             IReadOnlyList<IPropertySymbol> properties) | ||||||
|  |         { | ||||||
|  |             var options = properties | ||||||
|  |                 .Select(p => | ||||||
|  |                 { | ||||||
|  |                     var attribute = p | ||||||
|  |                         .GetAttributes() | ||||||
|  |                         .First(a => KnownSymbols.IsCommandOptionAttribute(a.AttributeClass)); | ||||||
|  |  | ||||||
|  |                     var name = attribute | ||||||
|  |                         .ConstructorArguments | ||||||
|  |                         .Where(a => KnownSymbols.IsSystemString(a.Type)) | ||||||
|  |                         .Select(a => a.Value) | ||||||
|  |                         .FirstOrDefault() as string; | ||||||
|  |  | ||||||
|  |                     var shortName = attribute | ||||||
|  |                         .ConstructorArguments | ||||||
|  |                         .Where(a => KnownSymbols.IsSystemChar(a.Type)) | ||||||
|  |                         .Select(a => a.Value) | ||||||
|  |                         .FirstOrDefault() as char?; | ||||||
|  |  | ||||||
|  |                     var envVarName = attribute | ||||||
|  |                         .NamedArguments | ||||||
|  |                         .Where(a => a.Key == "EnvironmentVariableName") | ||||||
|  |                         .Select(a => a.Value.Value) | ||||||
|  |                         .FirstOrDefault() as string; | ||||||
|  |  | ||||||
|  |                     var converter = attribute | ||||||
|  |                         .NamedArguments | ||||||
|  |                         .Where(a => a.Key == "Converter") | ||||||
|  |                         .Select(a => a.Value.Value) | ||||||
|  |                         .Cast<ITypeSymbol>() | ||||||
|  |                         .FirstOrDefault(); | ||||||
|  |  | ||||||
|  |                     var validators = attribute | ||||||
|  |                         .NamedArguments | ||||||
|  |                         .Where(a => a.Key == "Validators") | ||||||
|  |                         .SelectMany(a => a.Value.Values) | ||||||
|  |                         .Select(c => c.Value) | ||||||
|  |                         .Cast<ITypeSymbol>() | ||||||
|  |                         .ToArray(); | ||||||
|  |  | ||||||
|  |                     return new | ||||||
|  |                     { | ||||||
|  |                         Property = p, | ||||||
|  |                         Name = name, | ||||||
|  |                         ShortName = shortName, | ||||||
|  |                         EnvironmentVariableName = envVarName, | ||||||
|  |                         Converter = converter, | ||||||
|  |                         Validators = validators | ||||||
|  |                     }; | ||||||
|  |                 }) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             // No name | ||||||
|  |             var noNameOptions = options | ||||||
|  |                 .Where(o => string.IsNullOrWhiteSpace(o.Name) && o.ShortName == null) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in noNameOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0041, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Too short name | ||||||
|  |             var invalidNameLengthOptions = options | ||||||
|  |                 .Where(o => !string.IsNullOrWhiteSpace(o.Name) && o.Name.Length <= 1) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in invalidNameLengthOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0042, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Duplicate name | ||||||
|  |             var duplicateNameOptions = options | ||||||
|  |                 .Where(p => !string.IsNullOrWhiteSpace(p.Name)) | ||||||
|  |                 .GroupBy(p => p.Name, StringComparer.OrdinalIgnoreCase) | ||||||
|  |                 .Where(g => g.Count() > 1) | ||||||
|  |                 .SelectMany(g => g.AsEnumerable()) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in duplicateNameOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0043, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Duplicate name | ||||||
|  |             var duplicateShortNameOptions = options | ||||||
|  |                 .Where(p => p.ShortName != null) | ||||||
|  |                 .GroupBy(p => p.ShortName) | ||||||
|  |                 .Where(g => g.Count() > 1) | ||||||
|  |                 .SelectMany(g => g.AsEnumerable()) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in duplicateShortNameOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0044, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Duplicate environment variable name | ||||||
|  |             var duplicateEnvironmentVariableNameOptions = options | ||||||
|  |                 .Where(p => !string.IsNullOrWhiteSpace(p.EnvironmentVariableName)) | ||||||
|  |                 .GroupBy(p => p.EnvironmentVariableName, StringComparer.Ordinal) | ||||||
|  |                 .Where(g => g.Count() > 1) | ||||||
|  |                 .SelectMany(g => g.AsEnumerable()) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in duplicateEnvironmentVariableNameOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0045, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Invalid converter | ||||||
|  |             var invalidConverterOptions = options | ||||||
|  |                 .Where(o => | ||||||
|  |                     o.Converter != null && | ||||||
|  |                     !o.Converter.AllInterfaces.Any(KnownSymbols.IsArgumentValueConverterInterface)) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in invalidConverterOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0046, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Invalid validators | ||||||
|  |             var invalidValidatorsOptions = options | ||||||
|  |                 .Where(o => !o.Validators.All(v => v.AllInterfaces.Any(KnownSymbols.IsArgumentValueValidatorInterface))) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in invalidValidatorsOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0047, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Non-letter first character in name | ||||||
|  |             var nonLetterFirstCharacterInNameOptions = options | ||||||
|  |                 .Where(o => !string.IsNullOrWhiteSpace(o.Name) && !char.IsLetter(o.Name[0])) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in nonLetterFirstCharacterInNameOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0048, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Non-letter short name | ||||||
|  |             var nonLetterShortNameOptions = options | ||||||
|  |                 .Where(o => o.ShortName != null && !char.IsLetter(o.ShortName.Value)) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             foreach (var option in nonLetterShortNameOptions) | ||||||
|  |             { | ||||||
|  |                 context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                     DiagnosticDescriptors.CliFx0049, option.Property.Locations.First() | ||||||
|  |                 )); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static void CheckCommandType(SymbolAnalysisContext context) | ||||||
|  |         { | ||||||
|  |             // Named type: MyCommand | ||||||
|  |             if (!(context.Symbol is INamedTypeSymbol namedTypeSymbol) || | ||||||
|  |                 namedTypeSymbol.TypeKind != TypeKind.Class) | ||||||
|  |                 return; | ||||||
|  |  | ||||||
|  |             // Implements ICommand? | ||||||
|  |             var implementsCommandInterface = namedTypeSymbol | ||||||
|  |                 .AllInterfaces | ||||||
|  |                 .Any(KnownSymbols.IsCommandInterface); | ||||||
|  |  | ||||||
|  |             // Has CommandAttribute? | ||||||
|  |             var hasCommandAttribute = namedTypeSymbol | ||||||
|  |                 .GetAttributes() | ||||||
|  |                 .Select(a => a.AttributeClass) | ||||||
|  |                 .Any(KnownSymbols.IsCommandAttribute); | ||||||
|  |  | ||||||
|  |             var isValidCommandType = | ||||||
|  |                 // implements interface | ||||||
|  |                 implementsCommandInterface && ( | ||||||
|  |                     // and either abstract class or has attribute | ||||||
|  |                     namedTypeSymbol.IsAbstract || hasCommandAttribute | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |             if (!isValidCommandType) | ||||||
|  |             { | ||||||
|  |                 // See if this was meant to be a command type (either interface or attribute present) | ||||||
|  |                 var isAlmostValidCommandType = implementsCommandInterface ^ hasCommandAttribute; | ||||||
|  |  | ||||||
|  |                 if (isAlmostValidCommandType && !implementsCommandInterface) | ||||||
|  |                     context.ReportDiagnostic(Diagnostic.Create(DiagnosticDescriptors.CliFx0001, | ||||||
|  |                         namedTypeSymbol.Locations.First())); | ||||||
|  |  | ||||||
|  |                 if (isAlmostValidCommandType && !hasCommandAttribute) | ||||||
|  |                     context.ReportDiagnostic(Diagnostic.Create(DiagnosticDescriptors.CliFx0002, | ||||||
|  |                         namedTypeSymbol.Locations.First())); | ||||||
|  |  | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             var properties = namedTypeSymbol | ||||||
|  |                 .GetMembers() | ||||||
|  |                 .Where(m => m.Kind == SymbolKind.Property) | ||||||
|  |                 .OfType<IPropertySymbol>().ToArray(); | ||||||
|  |  | ||||||
|  |             // Check parameters | ||||||
|  |             var parameterProperties = properties | ||||||
|  |                 .Where(p => p | ||||||
|  |                     .GetAttributes() | ||||||
|  |                     .Select(a => a.AttributeClass) | ||||||
|  |                     .Any(KnownSymbols.IsCommandParameterAttribute)) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             CheckCommandParameterProperties(context, parameterProperties); | ||||||
|  |  | ||||||
|  |             // Check options | ||||||
|  |             var optionsProperties = properties | ||||||
|  |                 .Where(p => p | ||||||
|  |                     .GetAttributes() | ||||||
|  |                     .Select(a => a.AttributeClass) | ||||||
|  |                     .Any(KnownSymbols.IsCommandOptionAttribute)) | ||||||
|  |                 .ToArray(); | ||||||
|  |  | ||||||
|  |             CheckCommandOptionProperties(context, optionsProperties); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public override void Initialize(AnalysisContext context) | ||||||
|  |         { | ||||||
|  |             context.EnableConcurrentExecution(); | ||||||
|  |             context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None); | ||||||
|  |  | ||||||
|  |             context.RegisterSymbolAction(CheckCommandType, SymbolKind.NamedType); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										74
									
								
								CliFx.Analyzers/ConsoleUsageAnalyzer.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								CliFx.Analyzers/ConsoleUsageAnalyzer.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | |||||||
|  | using System.Collections.Immutable; | ||||||
|  | using System.Linq; | ||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  | using Microsoft.CodeAnalysis.CSharp; | ||||||
|  | using Microsoft.CodeAnalysis.CSharp.Syntax; | ||||||
|  | using Microsoft.CodeAnalysis.Diagnostics; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers | ||||||
|  | { | ||||||
|  |     [DiagnosticAnalyzer(LanguageNames.CSharp)] | ||||||
|  |     public class ConsoleUsageAnalyzer : DiagnosticAnalyzer | ||||||
|  |     { | ||||||
|  |         public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics { get; } = ImmutableArray.Create( | ||||||
|  |             DiagnosticDescriptors.CliFx0100 | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         private static bool IsSystemConsoleInvocation( | ||||||
|  |             SyntaxNodeAnalysisContext context, | ||||||
|  |             InvocationExpressionSyntax invocationSyntax) | ||||||
|  |         { | ||||||
|  |             if (invocationSyntax.Expression is MemberAccessExpressionSyntax memberAccessSyntax && | ||||||
|  |                 context.SemanticModel.GetSymbolInfo(memberAccessSyntax).Symbol is IMethodSymbol methodSymbol) | ||||||
|  |             { | ||||||
|  |                 // Direct call to System.Console (e.g. System.Console.WriteLine()) | ||||||
|  |                 if (KnownSymbols.IsSystemConsole(methodSymbol.ContainingType)) | ||||||
|  |                 { | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // Indirect call to System.Console (e.g. System.Console.Error.WriteLine()) | ||||||
|  |                 if (memberAccessSyntax.Expression is MemberAccessExpressionSyntax parentMemberAccessSyntax && | ||||||
|  |                     context.SemanticModel.GetSymbolInfo(parentMemberAccessSyntax).Symbol is IPropertySymbol propertySymbol) | ||||||
|  |                 { | ||||||
|  |                     return KnownSymbols.IsSystemConsole(propertySymbol.ContainingType); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static void CheckSystemConsoleUsage(SyntaxNodeAnalysisContext context) | ||||||
|  |         { | ||||||
|  |             if (context.Node is InvocationExpressionSyntax invocationSyntax && | ||||||
|  |                 IsSystemConsoleInvocation(context, invocationSyntax)) | ||||||
|  |             { | ||||||
|  |                 // Check if IConsole is available in scope as alternative to System.Console | ||||||
|  |                 var isConsoleInterfaceAvailable = invocationSyntax | ||||||
|  |                     .Ancestors() | ||||||
|  |                     .OfType<MethodDeclarationSyntax>() | ||||||
|  |                     .SelectMany(m => m.ParameterList.Parameters) | ||||||
|  |                     .Select(p => p.Type) | ||||||
|  |                     .Select(t => context.SemanticModel.GetSymbolInfo(t).Symbol) | ||||||
|  |                     .Where(s => s != null) | ||||||
|  |                     .Any(KnownSymbols.IsConsoleInterface!); | ||||||
|  |  | ||||||
|  |                 if (isConsoleInterfaceAvailable) | ||||||
|  |                 { | ||||||
|  |                     context.ReportDiagnostic(Diagnostic.Create( | ||||||
|  |                         DiagnosticDescriptors.CliFx0100, | ||||||
|  |                         invocationSyntax.GetLocation() | ||||||
|  |                     )); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public override void Initialize(AnalysisContext context) | ||||||
|  |         { | ||||||
|  |             context.EnableConcurrentExecution(); | ||||||
|  |             context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None); | ||||||
|  |  | ||||||
|  |             context.RegisterSyntaxNodeAction(CheckSystemConsoleUsage, SyntaxKind.InvocationExpression); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										133
									
								
								CliFx.Analyzers/DiagnosticDescriptors.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								CliFx.Analyzers/DiagnosticDescriptors.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,133 @@ | |||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers | ||||||
|  | { | ||||||
|  |     public static class DiagnosticDescriptors | ||||||
|  |     { | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0001 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0001), | ||||||
|  |                 "Type must implement the 'CliFx.ICommand' interface in order to be a valid command", | ||||||
|  |                 "Type must implement the 'CliFx.ICommand' interface in order to be a valid command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0002 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0002), | ||||||
|  |                 "Type must be annotated with the 'CliFx.Attributes.CommandAttribute' in order to be a valid command", | ||||||
|  |                 "Type must be annotated with the 'CliFx.Attributes.CommandAttribute' in order to be a valid command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0021 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0021), | ||||||
|  |                 "Parameter order must be unique within its command", | ||||||
|  |                 "Parameter order must be unique within its command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0022 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0022), | ||||||
|  |                 "Parameter order must have unique name within its command", | ||||||
|  |                 "Parameter order must have unique name within its command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0023 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0023), | ||||||
|  |                 "Only one non-scalar parameter per command is allowed", | ||||||
|  |                 "Only one non-scalar parameter per command is allowed", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0024 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0024), | ||||||
|  |                 "Non-scalar parameter must be last in order", | ||||||
|  |                 "Non-scalar parameter must be last in order", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0025 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0025), | ||||||
|  |                 "Parameter converter must implement 'CliFx.IArgumentValueConverter'", | ||||||
|  |                 "Parameter converter must implement 'CliFx.IArgumentValueConverter'", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0026 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0026), | ||||||
|  |                 "Parameter validator must implement 'CliFx.ArgumentValueValidator<T>'", | ||||||
|  |                 "Parameter validator must implement 'CliFx.ArgumentValueValidator<T>'", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0041 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0041), | ||||||
|  |                 "Option must have a name or short name specified", | ||||||
|  |                 "Option must have a name or short name specified", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0042 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0042), | ||||||
|  |                 "Option name must be at least 2 characters long", | ||||||
|  |                 "Option name must be at least 2 characters long", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0043 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0043), | ||||||
|  |                 "Option name must be unique within its command", | ||||||
|  |                 "Option name must be unique within its command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0044 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0044), | ||||||
|  |                 "Option short name must be unique within its command", | ||||||
|  |                 "Option short name must be unique within its command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0045 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0045), | ||||||
|  |                 "Option environment variable name must be unique within its command", | ||||||
|  |                 "Option environment variable name must be unique within its command", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0046 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0046), | ||||||
|  |                 "Option converter must implement 'CliFx.IArgumentValueConverter'", | ||||||
|  |                 "Option converter must implement 'CliFx.IArgumentValueConverter'", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0047 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0047), | ||||||
|  |                 "Option validator must implement 'CliFx.ArgumentValueValidator<T>'", | ||||||
|  |                 "Option validator must implement 'CliFx.ArgumentValueValidator<T>'", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0048 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0048), | ||||||
|  |                 "Option name must begin with a letter character.", | ||||||
|  |                 "Option name must begin with a letter character.", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0049 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0049), | ||||||
|  |                 "Option short name must be a letter character.", | ||||||
|  |                 "Option short name must be a letter character.", | ||||||
|  |                 "Usage", DiagnosticSeverity.Error, true | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |         public static readonly DiagnosticDescriptor CliFx0100 = | ||||||
|  |             new DiagnosticDescriptor(nameof(CliFx0100), | ||||||
|  |                 "Use the provided IConsole abstraction instead of System.Console to ensure that the command can be tested in isolation", | ||||||
|  |                 "Use the provided IConsole abstraction instead of System.Console to ensure that the command can be tested in isolation", | ||||||
|  |                 "Usage", DiagnosticSeverity.Warning, true | ||||||
|  |             ); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								CliFx.Analyzers/Internal/RoslynExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								CliFx.Analyzers/Internal/RoslynExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | using System; | ||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers.Internal | ||||||
|  | { | ||||||
|  |     internal static class RoslynExtensions | ||||||
|  |     { | ||||||
|  |         public static bool DisplayNameMatches(this ISymbol symbol, string name) => | ||||||
|  |             string.Equals(symbol.ToDisplayString(), name, StringComparison.Ordinal); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										43
									
								
								CliFx.Analyzers/KnownSymbols.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								CliFx.Analyzers/KnownSymbols.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | using CliFx.Analyzers.Internal; | ||||||
|  | using Microsoft.CodeAnalysis; | ||||||
|  |  | ||||||
|  | namespace CliFx.Analyzers | ||||||
|  | { | ||||||
|  |     internal static class KnownSymbols | ||||||
|  |     { | ||||||
|  |         public static bool IsSystemString(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("string") || | ||||||
|  |             symbol.DisplayNameMatches("System.String"); | ||||||
|  |  | ||||||
|  |         public static bool IsSystemChar(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("char") || | ||||||
|  |             symbol.DisplayNameMatches("System.Char"); | ||||||
|  |  | ||||||
|  |         public static bool IsSystemCollectionsGenericIEnumerable(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("System.Collections.Generic.IEnumerable<T>"); | ||||||
|  |  | ||||||
|  |         public static bool IsSystemConsole(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("System.Console"); | ||||||
|  |  | ||||||
|  |         public static bool IsConsoleInterface(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.IConsole"); | ||||||
|  |  | ||||||
|  |         public static bool IsCommandInterface(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.ICommand"); | ||||||
|  |  | ||||||
|  |         public static bool IsArgumentValueConverterInterface(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.IArgumentValueConverter"); | ||||||
|  |  | ||||||
|  |         public static bool IsArgumentValueValidatorInterface(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.IArgumentValueValidator"); | ||||||
|  |  | ||||||
|  |         public static bool IsCommandAttribute(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.Attributes.CommandAttribute"); | ||||||
|  |  | ||||||
|  |         public static bool IsCommandParameterAttribute(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.Attributes.CommandParameterAttribute"); | ||||||
|  |  | ||||||
|  |         public static bool IsCommandOptionAttribute(ISymbol symbol) => | ||||||
|  |             symbol.DisplayNameMatches("CliFx.Attributes.CommandOptionAttribute"); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -18,7 +18,7 @@ namespace CliFx.Benchmarks | |||||||
|  |  | ||||||
|         [Benchmark(Description = "CliFx", Baseline = true)] |         [Benchmark(Description = "CliFx", Baseline = true)] | ||||||
|         public async ValueTask<int> ExecuteWithCliFx() => |         public async ValueTask<int> ExecuteWithCliFx() => | ||||||
|             await new CliApplicationBuilder().AddCommand(typeof(CliFxCommand)).Build().RunAsync(Arguments, new Dictionary<string, string>()); |             await new CliApplicationBuilder().AddCommand<CliFxCommand>().Build().RunAsync(Arguments, new Dictionary<string, string>()); | ||||||
|  |  | ||||||
|         [Benchmark(Description = "System.CommandLine")] |         [Benchmark(Description = "System.CommandLine")] | ||||||
|         public async Task<int> ExecuteWithSystemCommandLine() => |         public async Task<int> ExecuteWithSystemCommandLine() => | ||||||
|   | |||||||
| @@ -3,15 +3,15 @@ | |||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <OutputType>Exe</OutputType> |     <OutputType>Exe</OutputType> | ||||||
|     <TargetFramework>netcoreapp3.1</TargetFramework> |     <TargetFramework>net5.0</TargetFramework> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="BenchmarkDotNet" Version="0.12.0" /> |     <PackageReference Include="BenchmarkDotNet" Version="0.12.0" /> | ||||||
|     <PackageReference Include="clipr" Version="1.6.1" /> |     <PackageReference Include="clipr" Version="1.6.1" /> | ||||||
|     <PackageReference Include="Cocona" Version="1.3.0" /> |     <PackageReference Include="Cocona" Version="1.5.0" /> | ||||||
|     <PackageReference Include="CommandLineParser" Version="2.7.82" /> |     <PackageReference Include="CommandLineParser" Version="2.8.0" /> | ||||||
|     <PackageReference Include="McMaster.Extensions.CommandLineUtils" Version="2.6.0" /> |     <PackageReference Include="McMaster.Extensions.CommandLineUtils" Version="3.0.0" /> | ||||||
|     <PackageReference Include="PowerArgs" Version="3.6.0" /> |     <PackageReference Include="PowerArgs" Version="3.6.0" /> | ||||||
|     <PackageReference Include="System.CommandLine.Experimental" Version="0.3.0-alpha.19317.1" /> |     <PackageReference Include="System.CommandLine.Experimental" Version="0.3.0-alpha.19317.1" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   | |||||||
| @@ -3,16 +3,17 @@ | |||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <OutputType>Exe</OutputType> |     <OutputType>Exe</OutputType> | ||||||
|     <TargetFramework>netcoreapp3.1</TargetFramework> |     <TargetFramework>net5.0</TargetFramework> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="3.1.1" /> |     <PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="3.1.9" /> | ||||||
|     <PackageReference Include="Newtonsoft.Json" Version="12.0.3" /> |     <PackageReference Include="Newtonsoft.Json" Version="12.0.3" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ProjectReference Include="..\CliFx\CliFx.csproj" /> |     <ProjectReference Include="..\CliFx\CliFx.csproj" /> | ||||||
|  |     <ProjectReference Include="..\CliFx.Analyzers\CliFx.Analyzers.csproj" ReferenceOutputAssembly="false" OutputItemType="Analyzer" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
| @@ -28,7 +28,7 @@ namespace CliFx.Demo | |||||||
|         public static async Task<int> Main() => |         public static async Task<int> Main() => | ||||||
|             await new CliApplicationBuilder() |             await new CliApplicationBuilder() | ||||||
|                 .AddCommandsFromThisAssembly() |                 .AddCommandsFromThisAssembly() | ||||||
|                 .UseTypeActivator(GetServiceProvider().GetService) |                 .UseTypeActivator(GetServiceProvider().GetRequiredService) | ||||||
|                 .Build() |                 .Build() | ||||||
|                 .RunAsync(); |                 .RunAsync(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -3,11 +3,12 @@ | |||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <OutputType>Exe</OutputType> |     <OutputType>Exe</OutputType> | ||||||
|     <TargetFramework>netcoreapp3.1</TargetFramework> |     <TargetFramework>net5.0</TargetFramework> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ProjectReference Include="..\CliFx\CliFx.csproj" /> |     <ProjectReference Include="..\CliFx\CliFx.csproj" /> | ||||||
|  |     <ProjectReference Include="..\CliFx.Analyzers\CliFx.Analyzers.csproj" ReferenceOutputAssembly="false" OutputItemType="Analyzer" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
| @@ -1,137 +0,0 @@ | |||||||
| using System.Collections.Generic; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class ApplicationSpecs |  | ||||||
|     { |  | ||||||
|         [Command] |  | ||||||
|         private class NonImplementedCommand |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class NonAnnotatedCommand : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("dup")] |  | ||||||
|         private class DuplicateNameCommandA : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("dup")] |  | ||||||
|         private class DuplicateNameCommandB : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class DuplicateParameterOrderCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(13)] |  | ||||||
|             public string? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(13)] |  | ||||||
|             public string? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class DuplicateParameterNameCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0, Name = "param")] |  | ||||||
|             public string? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(1, Name = "param")] |  | ||||||
|             public string? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class MultipleNonScalarParametersCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0)] |  | ||||||
|             public IReadOnlyList<string>? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(1)] |  | ||||||
|             public IReadOnlyList<string>? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class NonLastNonScalarParameterCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0)] |  | ||||||
|             public IReadOnlyList<string>? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(1)] |  | ||||||
|             public string? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class DuplicateOptionNamesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("fruits")] |  | ||||||
|             public string? Apples { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("fruits")] |  | ||||||
|             public string? Oranges { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class DuplicateOptionShortNamesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption('x')] |  | ||||||
|             public string? OptionA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption('x')] |  | ||||||
|             public string? OptionB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class DuplicateOptionEnvironmentVariableNamesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("option-a", EnvironmentVariableName = "ENV_VAR")] |  | ||||||
|             public string? OptionA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-b", EnvironmentVariableName = "ENV_VAR")] |  | ||||||
|             public string? OptionB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class ValidCommand : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("hidden", Description = "Description")] |  | ||||||
|         private class HiddenPropertiesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(13, Name = "param", Description = "Param description")] |  | ||||||
|             public string? Parameter { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option", 'o', Description = "Option description", EnvironmentVariableName = "ENV")] |  | ||||||
|             public string? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public string? HiddenA { get; set; } |  | ||||||
|  |  | ||||||
|             public bool? HiddenB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,14 +1,20 @@ | |||||||
| using System; | using System; | ||||||
| using System.IO; | using System.IO; | ||||||
| using CliFx.Domain; | using System.Threading.Tasks; | ||||||
| using CliFx.Exceptions; | using CliFx.Tests.Commands; | ||||||
|  | using CliFx.Tests.Commands.Invalid; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class ApplicationSpecs |     public class ApplicationSpecs | ||||||
|     { |     { | ||||||
|  |         private readonly ITestOutputHelper _output; | ||||||
|  |  | ||||||
|  |         public ApplicationSpecs(ITestOutputHelper output) => _output = output; | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public void Application_can_be_created_with_a_default_configuration() |         public void Application_can_be_created_with_a_default_configuration() | ||||||
|         { |         { | ||||||
| @@ -26,10 +32,10 @@ namespace CliFx.Tests | |||||||
|         { |         { | ||||||
|             // Act |             // Act | ||||||
|             var app = new CliApplicationBuilder() |             var app = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(ValidCommand)) |                 .AddCommand<DefaultCommand>() | ||||||
|                 .AddCommandsFrom(typeof(ValidCommand).Assembly) |                 .AddCommandsFrom(typeof(DefaultCommand).Assembly) | ||||||
|                 .AddCommands(new[] {typeof(ValidCommand)}) |                 .AddCommands(new[] {typeof(DefaultCommand)}) | ||||||
|                 .AddCommandsFrom(new[] {typeof(ValidCommand).Assembly}) |                 .AddCommandsFrom(new[] {typeof(DefaultCommand).Assembly}) | ||||||
|                 .AddCommandsFromThisAssembly() |                 .AddCommandsFromThisAssembly() | ||||||
|                 .AllowDebugMode() |                 .AllowDebugMode() | ||||||
|                 .AllowPreviewMode() |                 .AllowPreviewMode() | ||||||
| @@ -38,7 +44,7 @@ namespace CliFx.Tests | |||||||
|                 .UseVersionText("test") |                 .UseVersionText("test") | ||||||
|                 .UseDescription("test") |                 .UseDescription("test") | ||||||
|                 .UseConsole(new VirtualConsole(Stream.Null)) |                 .UseConsole(new VirtualConsole(Stream.Null)) | ||||||
|                 .UseTypeActivator(Activator.CreateInstance) |                 .UseTypeActivator(Activator.CreateInstance!) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
| @@ -46,152 +52,444 @@ namespace CliFx.Tests | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public void At_least_one_command_must_be_defined_in_an_application() |         public async Task At_least_one_command_must_be_defined_in_an_application() | ||||||
|         { |         { | ||||||
|             // Arrange |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|             var commandTypes = Array.Empty<Type>(); |  | ||||||
|  |  | ||||||
|             // Act & assert |             var application = new CliApplicationBuilder() | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |                 .UseConsole(console) | ||||||
|         } |                 .Build(); | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Commands_must_implement_the_corresponding_interface() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(NonImplementedCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Commands_must_be_annotated_by_an_attribute() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(NonAnnotatedCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Commands_must_have_unique_names() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(DuplicateNameCommandA), typeof(DuplicateNameCommandB)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_parameters_must_have_unique_order() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(DuplicateParameterOrderCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_parameters_must_have_unique_names() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(DuplicateParameterNameCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_parameter_can_be_non_scalar_only_if_no_other_such_parameter_is_present() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(MultipleNonScalarParametersCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_parameter_can_be_non_scalar_only_if_it_is_the_last_in_order() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(NonLastNonScalarParameterCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_options_must_have_unique_names() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(DuplicateOptionNamesCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_options_must_have_unique_short_names() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(DuplicateOptionShortNamesCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_options_must_have_unique_environment_variable_names() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(DuplicateOptionEnvironmentVariableNamesCommand)}; |  | ||||||
|  |  | ||||||
|             // Act & assert |  | ||||||
|             Assert.Throws<CliFxException>(() => ApplicationSchema.Resolve(commandTypes)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public void Command_options_and_parameters_must_be_annotated_by_corresponding_attributes() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var commandTypes = new[] {typeof(HiddenPropertiesCommand)}; |  | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var schema = ApplicationSchema.Resolve(commandTypes); |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             schema.Should().BeEquivalentTo(new ApplicationSchema(new[] |             exitCode.Should().NotBe(0); | ||||||
|             { |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|                 new CommandSchema( |  | ||||||
|                     typeof(HiddenPropertiesCommand), |  | ||||||
|                     "hidden", |  | ||||||
|                     "Description", |  | ||||||
|                     new[] |  | ||||||
|                     { |  | ||||||
|                         new CommandParameterSchema( |  | ||||||
|                             typeof(HiddenPropertiesCommand).GetProperty(nameof(HiddenPropertiesCommand.Parameter)), |  | ||||||
|                             13, |  | ||||||
|                             "param", |  | ||||||
|                             "Param description") |  | ||||||
|                     }, |  | ||||||
|                     new[] |  | ||||||
|                     { |  | ||||||
|                         new CommandOptionSchema( |  | ||||||
|                             typeof(HiddenPropertiesCommand).GetProperty(nameof(HiddenPropertiesCommand.Option)), |  | ||||||
|                             "option", |  | ||||||
|                             'o', |  | ||||||
|                             "ENV", |  | ||||||
|                             false, |  | ||||||
|                             "Option description") |  | ||||||
|                     }) |  | ||||||
|             })); |  | ||||||
|  |  | ||||||
|             schema.ToString().Should().NotBeNullOrWhiteSpace(); // this is only for coverage, I'm sorry |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Commands_must_implement_the_corresponding_interface() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand(typeof(NonImplementedCommand)) | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Commands_must_be_annotated_by_an_attribute() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<NonAnnotatedCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Commands_must_have_unique_names() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<GenericExceptionCommand>() | ||||||
|  |                 .AddCommand<CommandExceptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_can_be_default_but_only_if_it_is_the_only_such_command() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<OtherDefaultCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_parameters_must_have_unique_order() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DuplicateParameterOrderCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_parameters_must_have_unique_names() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DuplicateParameterNameCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_parameter_can_be_non_scalar_only_if_no_other_such_parameter_is_present() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<MultipleNonScalarParametersCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_parameter_can_be_non_scalar_only_if_it_is_the_last_in_order() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<NonLastNonScalarParameterCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_parameter_custom_converter_must_implement_the_corresponding_interface() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<InvalidCustomConverterParameterCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_parameter_custom_validator_must_implement_the_corresponding_interface() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<InvalidCustomValidatorParameterCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_names_that_are_not_empty() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<EmptyOptionNameCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_names_that_are_longer_than_one_character() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<SingleCharacterOptionNameCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_unique_names() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DuplicateOptionNamesCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_unique_short_names() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DuplicateOptionShortNamesCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_unique_environment_variable_names() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DuplicateOptionEnvironmentVariableNamesCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_not_have_conflicts_with_the_implicit_help_option() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<ConflictWithHelpOptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_not_have_conflicts_with_the_implicit_version_option() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<ConflictWithVersionOptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_option_custom_converter_must_implement_the_corresponding_interface() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<InvalidCustomConverterOptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_option_custom_validator_must_implement_the_corresponding_interface() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<InvalidCustomValidatorOptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_names_that_start_with_a_letter_character() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<NonLetterCharacterNameCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_options_must_have_short_names_that_are_letter_characters() | ||||||
|  |         { | ||||||
|  |             var (console, _, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<NonLetterCharacterShortNameCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,191 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Collections; |  | ||||||
| using System.Collections.Generic; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class ArgumentBindingSpecs |  | ||||||
|     { |  | ||||||
|         [Command] |  | ||||||
|         private class AllSupportedTypesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption(nameof(Object))] |  | ||||||
|             public object? Object { get; set; } = 42; |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(String))] |  | ||||||
|             public string? String { get; set; } = "foo bar"; |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Bool))] |  | ||||||
|             public bool Bool { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Char))] |  | ||||||
|             public char Char { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Sbyte))] |  | ||||||
|             public sbyte Sbyte { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Byte))] |  | ||||||
|             public byte Byte { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Short))] |  | ||||||
|             public short Short { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Ushort))] |  | ||||||
|             public ushort Ushort { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Int))] |  | ||||||
|             public int Int { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Uint))] |  | ||||||
|             public uint Uint { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Long))] |  | ||||||
|             public long Long { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Ulong))] |  | ||||||
|             public ulong Ulong { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Float))] |  | ||||||
|             public float Float { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Double))] |  | ||||||
|             public double Double { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Decimal))] |  | ||||||
|             public decimal Decimal { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(DateTime))] |  | ||||||
|             public DateTime DateTime { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(DateTimeOffset))] |  | ||||||
|             public DateTimeOffset DateTimeOffset { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(TimeSpan))] |  | ||||||
|             public TimeSpan TimeSpan { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(CustomEnum))] |  | ||||||
|             public CustomEnum CustomEnum { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(IntNullable))] |  | ||||||
|             public int? IntNullable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(CustomEnumNullable))] |  | ||||||
|             public CustomEnum? CustomEnumNullable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(TimeSpanNullable))] |  | ||||||
|             public TimeSpan? TimeSpanNullable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(TestStringConstructable))] |  | ||||||
|             public StringConstructable? TestStringConstructable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(TestStringParseable))] |  | ||||||
|             public StringParseable? TestStringParseable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(TestStringParseableWithFormatProvider))] |  | ||||||
|             public StringParseableWithFormatProvider? TestStringParseableWithFormatProvider { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(ObjectArray))] |  | ||||||
|             public object[]? ObjectArray { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(StringArray))] |  | ||||||
|             public string[]? StringArray { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(IntArray))] |  | ||||||
|             public int[]? IntArray { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(CustomEnumArray))] |  | ||||||
|             public CustomEnum[]? CustomEnumArray { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(IntNullableArray))] |  | ||||||
|             public int?[]? IntNullableArray { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(TestStringConstructableArray))] |  | ||||||
|             public StringConstructable[]? TestStringConstructableArray { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(Enumerable))] |  | ||||||
|             public IEnumerable? Enumerable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(StringEnumerable))] |  | ||||||
|             public IEnumerable<string>? StringEnumerable { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(StringReadOnlyList))] |  | ||||||
|             public IReadOnlyList<string>? StringReadOnlyList { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(StringList))] |  | ||||||
|             public List<string>? StringList { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(StringHashSet))] |  | ||||||
|             public HashSet<string>? StringHashSet { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class ArrayOptionCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("option", 'o')] |  | ||||||
|             public IReadOnlyList<string>? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class RequiredOptionCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption(nameof(OptionA))] |  | ||||||
|             public string? OptionA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(OptionB), IsRequired = true)] |  | ||||||
|             public string? OptionB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class ParametersCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0)] |  | ||||||
|             public string? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(1)] |  | ||||||
|             public string? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(2)] |  | ||||||
|             public IReadOnlyList<string>? ParameterC { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class UnsupportedPropertyTypeCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption(nameof(Option))] |  | ||||||
|             public DummyType? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class UnsupportedEnumerablePropertyTypeCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption(nameof(Option))] |  | ||||||
|             public CustomEnumerable<string>? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class NoParameterCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption(nameof(OptionA))] |  | ||||||
|             public string? OptionA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption(nameof(OptionB))] |  | ||||||
|             public string? OptionB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,64 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Collections; |  | ||||||
| using System.Collections.Generic; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class ArgumentBindingSpecs |  | ||||||
|     { |  | ||||||
|         private enum CustomEnum |  | ||||||
|         { |  | ||||||
|             Value1 = 1, |  | ||||||
|             Value2 = 2, |  | ||||||
|             Value3 = 3 |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class StringConstructable |  | ||||||
|         { |  | ||||||
|             public string Value { get; } |  | ||||||
|  |  | ||||||
|             public StringConstructable(string value) |  | ||||||
|             { |  | ||||||
|                 Value = value; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class StringParseable |  | ||||||
|         { |  | ||||||
|             public string Value { get; } |  | ||||||
|  |  | ||||||
|             private StringParseable(string value) |  | ||||||
|             { |  | ||||||
|                 Value = value; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             public static StringParseable Parse(string value) => new StringParseable(value); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class StringParseableWithFormatProvider |  | ||||||
|         { |  | ||||||
|             public string Value { get; } |  | ||||||
|  |  | ||||||
|             private StringParseableWithFormatProvider(string value) |  | ||||||
|             { |  | ||||||
|                 Value = value; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             public static StringParseableWithFormatProvider Parse(string value, IFormatProvider formatProvider) => |  | ||||||
|                 new StringParseableWithFormatProvider(value + " " + formatProvider); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class DummyType |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public class CustomEnumerable<T> : IEnumerable<T> |  | ||||||
|         { |  | ||||||
|             private readonly T[] _arr = new T[0]; |  | ||||||
|  |  | ||||||
|             public IEnumerator<T> GetEnumerator() => ((IEnumerable<T>) _arr).GetEnumerator(); |  | ||||||
|  |  | ||||||
|             IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1441
									
								
								CliFx.Tests/ArgumentConversionSpecs.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1441
									
								
								CliFx.Tests/ArgumentConversionSpecs.cs
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,315 +0,0 @@ | |||||||
| using System; |  | ||||||
| using CliFx.Domain; |  | ||||||
| using FluentAssertions; |  | ||||||
| using Xunit; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public class ArgumentSyntaxSpecs |  | ||||||
|     { |  | ||||||
|         [Fact] |  | ||||||
|         public void Input_is_empty_if_no_arguments_are_provided() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             var args = Array.Empty<string>(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             var input = CommandLineInput.Parse(args); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             input.Should().BeEquivalentTo(CommandLineInput.Empty); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public static object[][] DirectivesTestData => new[] |  | ||||||
|         { |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"[preview]"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddDirective("preview") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"[preview]", "[debug]"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddDirective("preview") |  | ||||||
|                     .AddDirective("debug") |  | ||||||
|                     .Build() |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         [Theory] |  | ||||||
|         [MemberData(nameof(DirectivesTestData))] |  | ||||||
|         internal void Directive_can_be_enabled_by_specifying_its_name_in_square_brackets(string[] arguments, CommandLineInput expectedInput) |  | ||||||
|         { |  | ||||||
|             // Act |  | ||||||
|             var input = CommandLineInput.Parse(arguments); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             input.Should().BeEquivalentTo(expectedInput); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public static object[][] OptionsTestData => new[] |  | ||||||
|         { |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option", "value"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option", "value") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option", "value1", "value2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option", "value1", "value2") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option", "same value"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option", "same value") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option1", "--option2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option1") |  | ||||||
|                     .AddOption("option2") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option1", "value1", "--option2", "value2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option1", "value1") |  | ||||||
|                     .AddOption("option2", "value2") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option1", "value1", "value2", "--option2", "value3", "value4"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option1", "value1", "value2") |  | ||||||
|                     .AddOption("option2", "value3", "value4") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"--option1", "value1", "value2", "--option2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("option1", "value1", "value2") |  | ||||||
|                     .AddOption("option2") |  | ||||||
|                     .Build() |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         [Theory] |  | ||||||
|         [MemberData(nameof(OptionsTestData))] |  | ||||||
|         internal void Option_can_be_set_by_specifying_its_name_after_two_dashes(string[] arguments, CommandLineInput expectedInput) |  | ||||||
|         { |  | ||||||
|             // Act |  | ||||||
|             var input = CommandLineInput.Parse(arguments); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             input.Should().BeEquivalentTo(expectedInput); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public static object[][] ShortOptionsTestData => new[] |  | ||||||
|         { |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-o"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("o") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-o", "value"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("o", "value") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-o", "value1", "value2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("o", "value1", "value2") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-o", "same value"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("o", "same value") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-a", "-b"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-a", "value1", "-b", "value2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a", "value1") |  | ||||||
|                     .AddOption("b", "value2") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-a", "value1", "value2", "-b", "value3", "value4"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a", "value1", "value2") |  | ||||||
|                     .AddOption("b", "value3", "value4") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-a", "value1", "value2", "-b"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a", "value1", "value2") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-abc"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .AddOption("c") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-abc", "value"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .AddOption("c", "value") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"-abc", "value1", "value2"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddOption("a") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .AddOption("c", "value1", "value2") |  | ||||||
|                     .Build() |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         [Theory] |  | ||||||
|         [MemberData(nameof(ShortOptionsTestData))] |  | ||||||
|         internal void Option_can_be_set_by_specifying_its_short_name_after_a_single_dash(string[] arguments, CommandLineInput expectedInput) |  | ||||||
|         { |  | ||||||
|             // Act |  | ||||||
|             var input = CommandLineInput.Parse(arguments); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             input.Should().BeEquivalentTo(expectedInput); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         public static object[][] UnboundArgumentsTestData => new[] |  | ||||||
|         { |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"foo"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddUnboundArgument("foo") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"foo", "bar"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddUnboundArgument("foo") |  | ||||||
|                     .AddUnboundArgument("bar") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"[preview]", "foo"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddDirective("preview") |  | ||||||
|                     .AddUnboundArgument("foo") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"foo", "--option", "value", "-abc"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddUnboundArgument("foo") |  | ||||||
|                     .AddOption("option", "value") |  | ||||||
|                     .AddOption("a") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .AddOption("c") |  | ||||||
|                     .Build() |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             new object[] |  | ||||||
|             { |  | ||||||
|                 new[] {"[preview]", "[debug]", "foo", "bar", "--option", "value", "-abc"}, |  | ||||||
|                 new CommandLineInputBuilder() |  | ||||||
|                     .AddDirective("preview") |  | ||||||
|                     .AddDirective("debug") |  | ||||||
|                     .AddUnboundArgument("foo") |  | ||||||
|                     .AddUnboundArgument("bar") |  | ||||||
|                     .AddOption("option", "value") |  | ||||||
|                     .AddOption("a") |  | ||||||
|                     .AddOption("b") |  | ||||||
|                     .AddOption("c") |  | ||||||
|                     .Build() |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         [Theory] |  | ||||||
|         [MemberData(nameof(UnboundArgumentsTestData))] |  | ||||||
|         internal void Any_remaining_arguments_are_treated_as_unbound_arguments(string[] arguments, CommandLineInput expectedInput) |  | ||||||
|         { |  | ||||||
|             // Act |  | ||||||
|             var input = CommandLineInput.Parse(arguments); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             input.Should().BeEquivalentTo(expectedInput); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class CancellationSpecs |  | ||||||
|     { |  | ||||||
|         [Command("cancel")] |  | ||||||
|         private class CancellableCommand : ICommand |  | ||||||
|         { |  | ||||||
|             public async ValueTask ExecuteAsync(IConsole console) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     await Task.Delay(TimeSpan.FromSeconds(3), console.GetCancellationToken()); |  | ||||||
|                     console.Output.WriteLine("Never printed"); |  | ||||||
|                 } |  | ||||||
|                 catch (OperationCanceledException) |  | ||||||
|                 { |  | ||||||
|                     console.Output.WriteLine("Cancellation requested"); |  | ||||||
|                     throw; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,41 +1,36 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; |  | ||||||
| using System.IO; |  | ||||||
| using System.Threading; | using System.Threading; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
|  | using CliFx.Tests.Commands; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class CancellationSpecs |     public class CancellationSpecs | ||||||
|     { |     { | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Command_can_perform_additional_cleanup_if_cancellation_is_requested() |         public async Task Command_can_perform_additional_cleanup_if_cancellation_is_requested() | ||||||
|         { |         { | ||||||
|  |             // Can't test it with a real console because CliWrap can't send Ctrl+C | ||||||
|  |  | ||||||
|             // Arrange |             // Arrange | ||||||
|             using var cts = new CancellationTokenSource(); |             using var cts = new CancellationTokenSource(); | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(cts.Token); | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut, cancellationToken: cts.Token); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(CancellableCommand)) |                 .AddCommand<CancellableCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             cts.CancelAfter(TimeSpan.FromSeconds(0.2)); |             cts.CancelAfter(TimeSpan.FromSeconds(0.2)); | ||||||
|  |  | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync(new[] {"cmd"}); | ||||||
|                 new[] {"cancel"}, |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().NotBe(0); |             exitCode.Should().NotBe(0); | ||||||
|             stdOutData.Should().Be("Cancellation requested"); |             stdOut.GetString().Trim().Should().Be(CancellableCommand.CancellationOutputText); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -2,12 +2,11 @@ | |||||||
|   <Import Project="../CliFx.props" /> |   <Import Project="../CliFx.props" /> | ||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <TargetFramework>netcoreapp3.1</TargetFramework> |     <TargetFramework>net5.0</TargetFramework> | ||||||
|     <IsPackable>false</IsPackable> |     <IsPackable>false</IsPackable> | ||||||
|     <IsTestProject>true</IsTestProject> |     <IsTestProject>true</IsTestProject> | ||||||
|     <CollectCoverage>true</CollectCoverage> |     <CollectCoverage>true</CollectCoverage> | ||||||
|     <CoverletOutputFormat>opencover</CoverletOutputFormat> |     <CoverletOutputFormat>opencover</CoverletOutputFormat> | ||||||
|     <CoverletOutput>bin/$(Configuration)/Coverage.xml</CoverletOutput> |  | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
| @@ -15,12 +14,14 @@ | |||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="CliWrap" Version="3.0.0" /> |     <PackageReference Include="CliWrap" Version="3.2.2" /> | ||||||
|     <PackageReference Include="FluentAssertions" Version="5.10.2" /> |     <PackageReference Include="FluentAssertions" Version="5.10.3" /> | ||||||
|     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.5.0" /> |     <PackageReference Include="GitHubActionsTestLogger" Version="1.1.2" /> | ||||||
|     <PackageReference Include="coverlet.msbuild" Version="2.8.0" PrivateAssets="all" /> |     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.7.1" /> | ||||||
|  |     <PackageReference Include="Newtonsoft.Json" Version="12.0.3" /> | ||||||
|     <PackageReference Include="xunit" Version="2.4.1" /> |     <PackageReference Include="xunit" Version="2.4.1" /> | ||||||
|     <PackageReference Include="xunit.runner.visualstudio" Version="2.4.1" /> |     <PackageReference Include="xunit.runner.visualstudio" Version="2.4.3" PrivateAssets="all" /> | ||||||
|  |     <PackageReference Include="coverlet.msbuild" Version="2.9.0" PrivateAssets="all" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|   | |||||||
							
								
								
									
										31
									
								
								CliFx.Tests/Commands/CancellableCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								CliFx.Tests/Commands/CancellableCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | using System; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class CancellableCommand : ICommand | ||||||
|  |     { | ||||||
|  |         public const string CompletionOutputText = "Finished"; | ||||||
|  |         public const string CancellationOutputText = "Canceled"; | ||||||
|  |  | ||||||
|  |         public async ValueTask ExecuteAsync(IConsole console) | ||||||
|  |         { | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 await Task.Delay( | ||||||
|  |                     TimeSpan.FromSeconds(3), | ||||||
|  |                     console.GetCancellationToken() | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |                 console.Output.WriteLine(CompletionOutputText); | ||||||
|  |             } | ||||||
|  |             catch (OperationCanceledException) | ||||||
|  |             { | ||||||
|  |                 console.Output.WriteLine(CancellationOutputText); | ||||||
|  |                 throw; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										21
									
								
								CliFx.Tests/Commands/CommandExceptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								CliFx.Tests/Commands/CommandExceptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  | using CliFx.Exceptions; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class CommandExceptionCommand : ICommand | ||||||
|  |     { | ||||||
|  |         [CommandOption("code", 'c')] | ||||||
|  |         public int ExitCode { get; set; } = 133; | ||||||
|  |  | ||||||
|  |         [CommandOption("msg", 'm')] | ||||||
|  |         public string? Message { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("show-help")] | ||||||
|  |         public bool ShowHelp { get; set; } | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) => throw new CommandException(Message, ExitCode, ShowHelp); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								CliFx.Tests/Commands/DefaultCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								CliFx.Tests/Commands/DefaultCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command(Description = "Default command description")] | ||||||
|  |     public class DefaultCommand : ICommand | ||||||
|  |     { | ||||||
|  |         public const string ExpectedOutputText = nameof(DefaultCommand); | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |         { | ||||||
|  |             console.Output.WriteLine(ExpectedOutputText); | ||||||
|  |             return default; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								CliFx.Tests/Commands/GenericExceptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								CliFx.Tests/Commands/GenericExceptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | using System; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class GenericExceptionCommand : ICommand | ||||||
|  |     { | ||||||
|  |         [CommandOption("msg", 'm')] | ||||||
|  |         public string? Message { get; set; } | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) => throw new Exception(Message); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										19
									
								
								CliFx.Tests/Commands/GenericInnerExceptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								CliFx.Tests/Commands/GenericInnerExceptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | using System; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class GenericInnerExceptionCommand : ICommand | ||||||
|  |     { | ||||||
|  |         [CommandOption("msg", 'm')] | ||||||
|  |         public string? Message { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("inner-msg", 'i')] | ||||||
|  |         public string? InnerMessage { get; set; } | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) => | ||||||
|  |             throw new Exception(Message, new Exception(InnerMessage)); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class ConflictWithHelpOptionCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("option-h", 'h')] | ||||||
|  |         public string? OptionH { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     // Must be default because version option is available only on default commands | ||||||
|  |     [Command] | ||||||
|  |     public class ConflictWithVersionOptionCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("version")] | ||||||
|  |         public string? Version { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class DuplicateOptionEnvironmentVariableNamesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("option-a", EnvironmentVariableName = "ENV_VAR")] | ||||||
|  |         public string? OptionA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("option-b", EnvironmentVariableName = "ENV_VAR")] | ||||||
|  |         public string? OptionB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								CliFx.Tests/Commands/Invalid/DuplicateOptionNamesCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								CliFx.Tests/Commands/Invalid/DuplicateOptionNamesCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class DuplicateOptionNamesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("fruits")] | ||||||
|  |         public string? Apples { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("fruits")] | ||||||
|  |         public string? Oranges { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class DuplicateOptionShortNamesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption('x')] | ||||||
|  |         public string? OptionA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption('x')] | ||||||
|  |         public string? OptionB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class DuplicateParameterNameCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0, Name = "param")] | ||||||
|  |         public string? ParamA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandParameter(1, Name = "param")] | ||||||
|  |         public string? ParamB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class DuplicateParameterOrderCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(13)] | ||||||
|  |         public string? ParamA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandParameter(13)] | ||||||
|  |         public string? ParamB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								CliFx.Tests/Commands/Invalid/EmptyOptionNameCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								CliFx.Tests/Commands/Invalid/EmptyOptionNameCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class EmptyOptionNameCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("")] | ||||||
|  |         public string? Apples { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,16 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command] | ||||||
|  |     public class InvalidCustomConverterOptionCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption('f', Converter = typeof(Converter))] | ||||||
|  |         public string? Option { get; set; } | ||||||
|  |  | ||||||
|  |         public class Converter | ||||||
|  |         { | ||||||
|  |             public object ConvertFrom(string value) => value; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,16 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command] | ||||||
|  |     public class InvalidCustomConverterParameterCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0, Converter = typeof(Converter))] | ||||||
|  |         public string? Param { get; set; } | ||||||
|  |  | ||||||
|  |         public class Converter | ||||||
|  |         { | ||||||
|  |             public object ConvertFrom(string value) => value; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,16 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command] | ||||||
|  |     public class InvalidCustomValidatorOptionCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption('f', Validators = new[] { typeof(Validator) })] | ||||||
|  |         public string? Option { get; set; } | ||||||
|  |  | ||||||
|  |         public class Validator | ||||||
|  |         { | ||||||
|  |             public ValidationResult Validate(string value) => ValidationResult.Ok(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,16 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command] | ||||||
|  |     public class InvalidCustomValidatorParameterCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0, Validators = new[] { typeof(Validator) })] | ||||||
|  |         public string? Param { get; set; } | ||||||
|  |  | ||||||
|  |         public class Validator | ||||||
|  |         { | ||||||
|  |             public ValidationResult Validate(string value) => ValidationResult.Ok(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class MultipleNonScalarParametersCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0)] | ||||||
|  |         public IReadOnlyList<string>? ParamA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandParameter(1)] | ||||||
|  |         public IReadOnlyList<string>? ParamB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								CliFx.Tests/Commands/Invalid/NonAnnotatedCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								CliFx.Tests/Commands/Invalid/NonAnnotatedCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     public class NonAnnotatedCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								CliFx.Tests/Commands/Invalid/NonImplementedCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								CliFx.Tests/Commands/Invalid/NonImplementedCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command] | ||||||
|  |     public class NonImplementedCommand | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class NonLastNonScalarParameterCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0)] | ||||||
|  |         public IReadOnlyList<string>? ParamA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandParameter(1)] | ||||||
|  |         public string? ParamB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class NonLetterCharacterNameCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("0foo")] | ||||||
|  |         public string? Apples { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class NonLetterCharacterShortNameCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption('0')] | ||||||
|  |         public string? Apples { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								CliFx.Tests/Commands/Invalid/OtherDefaultCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								CliFx.Tests/Commands/Invalid/OtherDefaultCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command] | ||||||
|  |     public class OtherDefaultCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands.Invalid | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class SingleCharacterOptionNameCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("a")] | ||||||
|  |         public string? Apples { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								CliFx.Tests/Commands/NamedCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								CliFx.Tests/Commands/NamedCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("named", Description = "Named command description")] | ||||||
|  |     public class NamedCommand : ICommand | ||||||
|  |     { | ||||||
|  |         public const string ExpectedOutputText = nameof(NamedCommand); | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |         { | ||||||
|  |             console.Output.WriteLine(ExpectedOutputText); | ||||||
|  |             return default; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								CliFx.Tests/Commands/NamedSubCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								CliFx.Tests/Commands/NamedSubCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("named sub", Description = "Named sub command description")] | ||||||
|  |     public class NamedSubCommand : ICommand | ||||||
|  |     { | ||||||
|  |         public const string ExpectedOutputText = nameof(NamedSubCommand); | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |         { | ||||||
|  |             console.Output.WriteLine(ExpectedOutputText); | ||||||
|  |             return default; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								CliFx.Tests/Commands/SelfSerializeCommandBase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								CliFx.Tests/Commands/SelfSerializeCommandBase.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  | using Newtonsoft.Json; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     public abstract class SelfSerializeCommandBase : ICommand | ||||||
|  |     { | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) | ||||||
|  |         { | ||||||
|  |             console.Output.WriteLine(JsonConvert.SerializeObject(this)); | ||||||
|  |             return default; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										174
									
								
								CliFx.Tests/Commands/SupportedArgumentTypesCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								CliFx.Tests/Commands/SupportedArgumentTypesCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Globalization; | ||||||
|  | using CliFx.Attributes; | ||||||
|  | using Newtonsoft.Json; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public partial class SupportedArgumentTypesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("obj")] | ||||||
|  |         public object? Object { get; set; } = 42; | ||||||
|  |  | ||||||
|  |         [CommandOption("str")] | ||||||
|  |         public string? String { get; set; } = "foo bar"; | ||||||
|  |  | ||||||
|  |         [CommandOption("bool")] | ||||||
|  |         public bool Bool { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("char")] | ||||||
|  |         public char Char { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("sbyte")] | ||||||
|  |         public sbyte Sbyte { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("byte")] | ||||||
|  |         public byte Byte { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("short")] | ||||||
|  |         public short Short { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("ushort")] | ||||||
|  |         public ushort Ushort { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("int")] | ||||||
|  |         public int Int { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("uint")] | ||||||
|  |         public uint Uint { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("long")] | ||||||
|  |         public long Long { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("ulong")] | ||||||
|  |         public ulong Ulong { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("float")] | ||||||
|  |         public float Float { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("double")] | ||||||
|  |         public double Double { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("decimal")] | ||||||
|  |         public decimal Decimal { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("datetime")] | ||||||
|  |         public DateTime DateTime { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("datetime-offset")] | ||||||
|  |         public DateTimeOffset DateTimeOffset { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("timespan")] | ||||||
|  |         public TimeSpan TimeSpan { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("enum")] | ||||||
|  |         public CustomEnum Enum { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("int-nullable")] | ||||||
|  |         public int? IntNullable { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("enum-nullable")] | ||||||
|  |         public CustomEnum? EnumNullable { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("timespan-nullable")] | ||||||
|  |         public TimeSpan? TimeSpanNullable { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-constructible")] | ||||||
|  |         public CustomStringConstructible? StringConstructible { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-parseable")] | ||||||
|  |         public CustomStringParseable? StringParseable { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-parseable-format")] | ||||||
|  |         public CustomStringParseableWithFormatProvider? StringParseableWithFormatProvider { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("convertible", Converter = typeof(CustomConvertibleConverter))] | ||||||
|  |         public CustomConvertible? Convertible { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("obj-array")] | ||||||
|  |         public object[]? ObjectArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-array")] | ||||||
|  |         public string[]? StringArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("int-array")] | ||||||
|  |         public int[]? IntArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("enum-array")] | ||||||
|  |         public CustomEnum[]? EnumArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("int-nullable-array")] | ||||||
|  |         public int?[]? IntNullableArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-constructible-array")] | ||||||
|  |         public CustomStringConstructible[]? StringConstructibleArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("convertible-array", Converter = typeof(CustomConvertibleConverter))] | ||||||
|  |         public CustomConvertible[]? ConvertibleArray { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-enumerable")] | ||||||
|  |         public IEnumerable<string>? StringEnumerable { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-read-only-list")] | ||||||
|  |         public IReadOnlyList<string>? StringReadOnlyList { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-list")] | ||||||
|  |         public List<string>? StringList { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("str-set")] | ||||||
|  |         public HashSet<string>? StringHashSet { get; set; } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public partial class SupportedArgumentTypesCommand | ||||||
|  |     { | ||||||
|  |         public enum CustomEnum | ||||||
|  |         { | ||||||
|  |             Value1 = 1, | ||||||
|  |             Value2 = 2, | ||||||
|  |             Value3 = 3 | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class CustomStringConstructible | ||||||
|  |         { | ||||||
|  |             public string Value { get; } | ||||||
|  |  | ||||||
|  |             public CustomStringConstructible(string value) => Value = value; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class CustomStringParseable | ||||||
|  |         { | ||||||
|  |             public string Value { get; } | ||||||
|  |  | ||||||
|  |             [JsonConstructor] | ||||||
|  |             private CustomStringParseable(string value) => Value = value; | ||||||
|  |  | ||||||
|  |             public static CustomStringParseable Parse(string value) => new CustomStringParseable(value); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class CustomStringParseableWithFormatProvider | ||||||
|  |         { | ||||||
|  |             public string Value { get; } | ||||||
|  |  | ||||||
|  |             [JsonConstructor] | ||||||
|  |             private CustomStringParseableWithFormatProvider(string value) => Value = value; | ||||||
|  |  | ||||||
|  |             public static CustomStringParseableWithFormatProvider Parse(string value, IFormatProvider formatProvider) => | ||||||
|  |                 new CustomStringParseableWithFormatProvider(value + " " + formatProvider); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class CustomConvertible | ||||||
|  |         { | ||||||
|  |             public int Value { get; } | ||||||
|  |  | ||||||
|  |             public CustomConvertible(int value) => Value = value; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class CustomConvertibleConverter : ArgumentValueConverter<CustomConvertible> | ||||||
|  |         { | ||||||
|  |             public override CustomConvertible ConvertFrom(string value) => | ||||||
|  |                 new CustomConvertible(int.Parse(value, CultureInfo.InvariantCulture)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										31
									
								
								CliFx.Tests/Commands/UnsupportedArgumentTypesCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								CliFx.Tests/Commands/UnsupportedArgumentTypesCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public partial class UnsupportedArgumentTypesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("custom")] | ||||||
|  |         public CustomType? CustomNonConvertible { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("custom-enumerable")] | ||||||
|  |         public CustomEnumerable<string>? CustomEnumerableNonConvertible { get; set; } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public partial class UnsupportedArgumentTypesCommand | ||||||
|  |     { | ||||||
|  |         public class CustomType | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class CustomEnumerable<T> : IEnumerable<T> | ||||||
|  |         { | ||||||
|  |             public IEnumerator<T> GetEnumerator() => ((IEnumerable<T>) Array.Empty<T>()).GetEnumerator(); | ||||||
|  |  | ||||||
|  |             IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										44
									
								
								CliFx.Tests/Commands/WithDefaultValuesCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								CliFx.Tests/Commands/WithDefaultValuesCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | using System; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithDefaultValuesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         public enum CustomEnum { Value1, Value2, Value3 }; | ||||||
|  |  | ||||||
|  |         [CommandOption("obj")] | ||||||
|  |         public object? Object { get; set; } = 42; | ||||||
|  |  | ||||||
|  |         [CommandOption("str")] | ||||||
|  |         public string? String { get; set; } = "foo"; | ||||||
|  |  | ||||||
|  |         [CommandOption("str-empty")] | ||||||
|  |         public string StringEmpty { get; set; } = ""; | ||||||
|  |  | ||||||
|  |         [CommandOption("str-array")] | ||||||
|  |         public string[]? StringArray { get; set; } = { "foo", "bar", "baz" }; | ||||||
|  |  | ||||||
|  |         [CommandOption("bool")] | ||||||
|  |         public bool Bool { get; set; } = true; | ||||||
|  |  | ||||||
|  |         [CommandOption("char")] | ||||||
|  |         public char Char { get; set; } = 't'; | ||||||
|  |  | ||||||
|  |         [CommandOption("int")] | ||||||
|  |         public int Int { get; set; } = 1337; | ||||||
|  |  | ||||||
|  |         [CommandOption("int-nullable")] | ||||||
|  |         public int? IntNullable { get; set; } = 1337; | ||||||
|  |  | ||||||
|  |         [CommandOption("int-array")] | ||||||
|  |         public int[]? IntArray { get; set; } = { 1, 2, 3 }; | ||||||
|  |  | ||||||
|  |         [CommandOption("timespan")] | ||||||
|  |         public TimeSpan TimeSpan { get; set; } = TimeSpan.FromMinutes(123); | ||||||
|  |  | ||||||
|  |         [CommandOption("enum")] | ||||||
|  |         public CustomEnum Enum { get; set; } = CustomEnum.Value2; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								CliFx.Tests/Commands/WithDependenciesCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								CliFx.Tests/Commands/WithDependenciesCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithDependenciesCommand : ICommand | ||||||
|  |     { | ||||||
|  |         public class DependencyA | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public class DependencyB | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private readonly DependencyA _dependencyA; | ||||||
|  |         private readonly DependencyB _dependencyB; | ||||||
|  |  | ||||||
|  |         public WithDependenciesCommand(DependencyA dependencyA, DependencyB dependencyB) | ||||||
|  |         { | ||||||
|  |             _dependencyA = dependencyA; | ||||||
|  |             _dependencyB = dependencyB; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										19
									
								
								CliFx.Tests/Commands/WithEnumArgumentsCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								CliFx.Tests/Commands/WithEnumArgumentsCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithEnumArgumentsCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         public enum CustomEnum { Value1, Value2, Value3 }; | ||||||
|  |  | ||||||
|  |         [CommandParameter(0, Name = "enum")] | ||||||
|  |         public CustomEnum EnumParameter { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("enum")] | ||||||
|  |         public CustomEnum? EnumOption { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("required-enum", IsRequired = true)] | ||||||
|  |         public CustomEnum RequiredEnumOption { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								CliFx.Tests/Commands/WithEnvironmentVariablesCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								CliFx.Tests/Commands/WithEnvironmentVariablesCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithEnvironmentVariablesCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("opt-a", 'a', EnvironmentVariableName = "ENV_OPT_A")] | ||||||
|  |         public string? OptA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("opt-b", 'b', EnvironmentVariableName = "ENV_OPT_B")] | ||||||
|  |         public IReadOnlyList<string>? OptB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								CliFx.Tests/Commands/WithParametersCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								CliFx.Tests/Commands/WithParametersCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithParametersCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0)] | ||||||
|  |         public string? ParamA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandParameter(1)] | ||||||
|  |         public int? ParamB { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandParameter(2)] | ||||||
|  |         public IReadOnlyList<string>? ParamC { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								CliFx.Tests/Commands/WithRequiredOptionsCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								CliFx.Tests/Commands/WithRequiredOptionsCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithRequiredOptionsCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("opt-a", 'a', IsRequired = true)] | ||||||
|  |         public string? OptA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("opt-b", 'b')] | ||||||
|  |         public int? OptB { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("opt-c", 'c', IsRequired = true)] | ||||||
|  |         public IReadOnlyList<char>? OptC { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								CliFx.Tests/Commands/WithSingleParameterCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								CliFx.Tests/Commands/WithSingleParameterCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithSingleParameterCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandParameter(0)] | ||||||
|  |         public string? ParamA { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								CliFx.Tests/Commands/WithSingleRequiredOptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								CliFx.Tests/Commands/WithSingleRequiredOptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithSingleRequiredOptionCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("opt-a")] | ||||||
|  |         public string? OptA { get; set; } | ||||||
|  |  | ||||||
|  |         [CommandOption("opt-b", IsRequired = true)] | ||||||
|  |         public string? OptB { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								CliFx.Tests/Commands/WithStringArrayOptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								CliFx.Tests/Commands/WithStringArrayOptionCommand.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | using CliFx.Attributes; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Commands | ||||||
|  | { | ||||||
|  |     [Command("cmd")] | ||||||
|  |     public class WithStringArrayOptionCommand : SelfSerializeCommandBase | ||||||
|  |     { | ||||||
|  |         [CommandOption("opt", 'o')] | ||||||
|  |         public IReadOnlyList<string>? Opt { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -38,7 +38,8 @@ namespace CliFx.Tests | |||||||
|             var console = new VirtualConsole( |             var console = new VirtualConsole( | ||||||
|                 input: stdIn, |                 input: stdIn, | ||||||
|                 output: stdOut, |                 output: stdOut, | ||||||
|                 error: stdErr); |                 error: stdErr | ||||||
|  |             ); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             console.Output.Write("output"); |             console.Output.Write("output"); | ||||||
| @@ -51,6 +52,8 @@ namespace CliFx.Tests | |||||||
|             console.ResetColor(); |             console.ResetColor(); | ||||||
|             console.ForegroundColor = ConsoleColor.DarkMagenta; |             console.ForegroundColor = ConsoleColor.DarkMagenta; | ||||||
|             console.BackgroundColor = ConsoleColor.DarkMagenta; |             console.BackgroundColor = ConsoleColor.DarkMagenta; | ||||||
|  |             console.CursorLeft = 42; | ||||||
|  |             console.CursorTop = 24; | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdInData.Should().Be("input"); |             stdInData.Should().Be("input"); | ||||||
|   | |||||||
| @@ -1,37 +0,0 @@ | |||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class DependencyInjectionSpecs |  | ||||||
|     { |  | ||||||
|         [Command] |  | ||||||
|         private class WithoutDependenciesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class DependencyA |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private class DependencyB |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class WithDependenciesCommand : ICommand |  | ||||||
|         { |  | ||||||
|             private readonly DependencyA _dependencyA; |  | ||||||
|             private readonly DependencyB _dependencyB; |  | ||||||
|  |  | ||||||
|             public WithDependenciesCommand(DependencyA dependencyA, DependencyB dependencyB) |  | ||||||
|             { |  | ||||||
|                 _dependencyA = dependencyA; |  | ||||||
|                 _dependencyB = dependencyB; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,11 +1,17 @@ | |||||||
| using CliFx.Exceptions; | using CliFx.Exceptions; | ||||||
|  | using CliFx.Tests.Commands; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class DependencyInjectionSpecs |     public class DependencyInjectionSpecs | ||||||
|     { |     { | ||||||
|  |         private readonly ITestOutputHelper _output; | ||||||
|  |  | ||||||
|  |         public DependencyInjectionSpecs(ITestOutputHelper output) => _output = output; | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public void Default_type_activator_can_initialize_a_command_if_it_has_a_parameterless_constructor() |         public void Default_type_activator_can_initialize_a_command_if_it_has_a_parameterless_constructor() | ||||||
|         { |         { | ||||||
| @@ -13,10 +19,10 @@ namespace CliFx.Tests | |||||||
|             var activator = new DefaultTypeActivator(); |             var activator = new DefaultTypeActivator(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var obj = activator.CreateInstance(typeof(WithoutDependenciesCommand)); |             var obj = activator.CreateInstance(typeof(DefaultCommand)); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             obj.Should().BeOfType<WithoutDependenciesCommand>(); |             obj.Should().BeOfType<DefaultCommand>(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
| @@ -26,8 +32,8 @@ namespace CliFx.Tests | |||||||
|             var activator = new DefaultTypeActivator(); |             var activator = new DefaultTypeActivator(); | ||||||
|  |  | ||||||
|             // Act & assert |             // Act & assert | ||||||
|             Assert.Throws<CliFxException>(() => |             var ex = Assert.Throws<CliFxException>(() => activator.CreateInstance(typeof(WithDependenciesCommand))); | ||||||
|                 activator.CreateInstance(typeof(WithDependenciesCommand))); |             _output.WriteLine(ex.Message); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
| @@ -35,7 +41,10 @@ namespace CliFx.Tests | |||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             var activator = new DelegateTypeActivator(_ => |             var activator = new DelegateTypeActivator(_ => | ||||||
|                 new WithDependenciesCommand(new DependencyA(), new DependencyB())); |                 new WithDependenciesCommand( | ||||||
|  |                     new WithDependenciesCommand.DependencyA(), | ||||||
|  |                     new WithDependenciesCommand.DependencyB()) | ||||||
|  |             ); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var obj = activator.CreateInstance(typeof(WithDependenciesCommand)); |             var obj = activator.CreateInstance(typeof(WithDependenciesCommand)); | ||||||
| @@ -48,11 +57,11 @@ namespace CliFx.Tests | |||||||
|         public void Delegate_type_activator_throws_if_the_underlying_function_returns_null() |         public void Delegate_type_activator_throws_if_the_underlying_function_returns_null() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             var activator = new DelegateTypeActivator(_ => null); |             var activator = new DelegateTypeActivator(_ => null!); | ||||||
|  |  | ||||||
|             // Act & assert |             // Act & assert | ||||||
|             Assert.Throws<CliFxException>(() => |             var ex = Assert.Throws<CliFxException>(() => activator.CreateInstance(typeof(WithDependenciesCommand))); | ||||||
|                 activator.CreateInstance(typeof(WithDependenciesCommand))); |             _output.WriteLine(ex.Message); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class DirectivesSpecs |  | ||||||
|     { |  | ||||||
|         [Command("cmd")] |  | ||||||
|         private class NamedCommand : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,36 +1,44 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.IO; | using System.IO; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
|  | using CliFx.Tests.Commands; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class DirectivesSpecs |     public class DirectivesSpecs | ||||||
|     { |     { | ||||||
|  |         private readonly ITestOutputHelper _output; | ||||||
|  |  | ||||||
|  |         public DirectivesSpecs(ITestOutputHelper output) => _output = output; | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Preview_directive_can_be_enabled_to_print_provided_arguments_as_they_were_parsed() |         public async Task Preview_directive_can_be_specified_to_print_provided_arguments_as_they_were_parsed() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdOut = new MemoryStream(); |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(NamedCommand)) |                 .AddCommand<NamedCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .AllowPreviewMode() |                 .AllowPreviewMode() | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync( | ||||||
|                 new[] {"[preview]", "cmd", "param", "-abc", "--option", "foo"}, |                 new[] {"[preview]", "named", "param", "-abc", "--option", "foo"}, | ||||||
|                 new Dictionary<string, string>()); |                 new Dictionary<string, string>() | ||||||
|  |             ); | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().Be(0); |             exitCode.Should().Be(0); | ||||||
|             stdOutData.Should().ContainAll("cmd", "<param>", "[-a]", "[-b]", "[-c]", "[--option foo]"); |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "named", "<param>", "[-a]", "[-b]", "[-c]", "[--option \"foo\"]" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| using System.Collections.Generic; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class EnvironmentVariablesSpecs |  | ||||||
|     { |  | ||||||
|         [Command] |  | ||||||
|         private class EnvironmentVariableCollectionCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("opt", EnvironmentVariableName = "ENV_OPT")] |  | ||||||
|             public IReadOnlyList<string>? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command] |  | ||||||
|         private class EnvironmentVariableCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("opt", EnvironmentVariableName = "ENV_OPT")] |  | ||||||
|             public string? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,7 +1,8 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.IO; | using System.IO; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using CliFx.Domain; | using CliFx.Tests.Commands; | ||||||
|  | using CliFx.Tests.Internal; | ||||||
| using CliWrap; | using CliWrap; | ||||||
| using CliWrap.Buffered; | using CliWrap.Buffered; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| @@ -9,11 +10,11 @@ using Xunit; | |||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class EnvironmentVariablesSpecs |     public class EnvironmentVariablesSpecs | ||||||
|     { |     { | ||||||
|         // This test uses a real application to make sure environment variables are actually read correctly |         // This test uses a real application to make sure environment variables are actually read correctly | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Option_can_use_a_specific_environment_variable_as_fallback() |         public async Task Option_can_use_an_environment_variable_as_fallback() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             var command = Cli.Wrap("dotnet") |             var command = Cli.Wrap("dotnet") | ||||||
| @@ -26,12 +27,12 @@ namespace CliFx.Tests | |||||||
|             var stdOut = await command.ExecuteBufferedAsync().Select(r => r.StandardOutput); |             var stdOut = await command.ExecuteBufferedAsync().Select(r => r.StandardOutput); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOut.TrimEnd().Should().Be("Hello Mars!"); |             stdOut.Trim().Should().Be("Hello Mars!"); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // This test uses a real application to make sure environment variables are actually read correctly |         // This test uses a real application to make sure environment variables are actually read correctly | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Option_only_uses_environment_variable_as_fallback_if_the_value_was_not_directly_provided() |         public async Task Option_only_uses_an_environment_variable_as_fallback_if_the_value_is_not_directly_provided() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             var command = Cli.Wrap("dotnet") |             var command = Cli.Wrap("dotnet") | ||||||
| @@ -46,50 +47,94 @@ namespace CliFx.Tests | |||||||
|             var stdOut = await command.ExecuteBufferedAsync().Select(r => r.StandardOutput); |             var stdOut = await command.ExecuteBufferedAsync().Select(r => r.StandardOutput); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOut.TrimEnd().Should().Be("Hello Jupiter!"); |             stdOut.Trim().Should().Be("Hello Jupiter!"); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public void Option_of_non_scalar_type_can_take_multiple_separated_values_from_an_environment_variable() |         public async Task Option_only_uses_an_environment_variable_as_fallback_if_the_name_matches_case_sensitively() | ||||||
|         { |         { | ||||||
|             // Arrange |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var schema = ApplicationSchema.Resolve(new[] {typeof(EnvironmentVariableCollectionCommand)}); |  | ||||||
|  |  | ||||||
|             var input = CommandLineInput.Empty; |             var application = new CliApplicationBuilder() | ||||||
|             var envVars = new Dictionary<string, string> |                 .AddCommand<WithEnvironmentVariablesCommand>() | ||||||
|             { |                 .UseConsole(console) | ||||||
|                 ["ENV_OPT"] = $"foo{Path.PathSeparator}bar" |                 .Build(); | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var command = schema.InitializeEntryPoint(input, envVars); |             var exitCode = await application.RunAsync( | ||||||
|  |                 new[] {"cmd"}, | ||||||
|  |                 new Dictionary<string, string> | ||||||
|  |                 { | ||||||
|  |                     ["ENV_opt_A"] = "incorrect", | ||||||
|  |                     ["ENV_OPT_A"] = "correct" | ||||||
|  |                 } | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             var commandInstance = stdOut.GetString().DeserializeJson<WithEnvironmentVariablesCommand>(); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             command.Should().BeEquivalentTo(new EnvironmentVariableCollectionCommand |             exitCode.Should().Be(0); | ||||||
|  |             commandInstance.Should().BeEquivalentTo(new WithEnvironmentVariablesCommand | ||||||
|             { |             { | ||||||
|                 Option = new[] {"foo", "bar"} |                 OptA = "correct" | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public void Option_of_scalar_type_can_only_take_a_single_value_from_an_environment_variable_even_if_it_contains_separators() |         public async Task Option_of_non_scalar_type_can_use_an_environment_variable_as_fallback_and_extract_multiple_values() | ||||||
|         { |         { | ||||||
|             // Arrange |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var schema = ApplicationSchema.Resolve(new[] {typeof(EnvironmentVariableCommand)}); |  | ||||||
|  |  | ||||||
|             var input = CommandLineInput.Empty; |             var application = new CliApplicationBuilder() | ||||||
|             var envVars = new Dictionary<string, string> |                 .AddCommand<WithEnvironmentVariablesCommand>() | ||||||
|             { |                 .UseConsole(console) | ||||||
|                 ["ENV_OPT"] = $"foo{Path.PathSeparator}bar" |                 .Build(); | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var command = schema.InitializeEntryPoint(input, envVars); |             var exitCode = await application.RunAsync( | ||||||
|  |                 new[] {"cmd"}, | ||||||
|  |                 new Dictionary<string, string> | ||||||
|  |                 { | ||||||
|  |                     ["ENV_OPT_B"] = $"foo{Path.PathSeparator}bar" | ||||||
|  |                 } | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             var commandInstance = stdOut.GetString().DeserializeJson<WithEnvironmentVariablesCommand>(); | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             command.Should().BeEquivalentTo(new EnvironmentVariableCommand |             exitCode.Should().Be(0); | ||||||
|  |             commandInstance.Should().BeEquivalentTo(new WithEnvironmentVariablesCommand | ||||||
|             { |             { | ||||||
|                 Option = $"foo{Path.PathSeparator}bar" |                 OptB = new[] {"foo", "bar"} | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Option_of_scalar_type_can_use_an_environment_variable_as_fallback_regardless_of_separators() | ||||||
|  |         { | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<WithEnvironmentVariablesCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync( | ||||||
|  |                 new[] {"cmd"}, | ||||||
|  |                 new Dictionary<string, string> | ||||||
|  |                 { | ||||||
|  |                     ["ENV_OPT_A"] = $"foo{Path.PathSeparator}bar" | ||||||
|  |                 } | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             var commandInstance = stdOut.GetString().DeserializeJson<WithEnvironmentVariablesCommand>(); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             commandInstance.Should().BeEquivalentTo(new WithEnvironmentVariablesCommand | ||||||
|  |             { | ||||||
|  |                 OptA = $"foo{Path.PathSeparator}bar" | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,31 +0,0 @@ | |||||||
| using System; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
| using CliFx.Exceptions; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class ErrorReportingSpecs |  | ||||||
|     { |  | ||||||
|         [Command("exc")] |  | ||||||
|         private class GenericExceptionCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("msg", 'm')] |  | ||||||
|             public string? Message { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => throw new Exception(Message); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("exc")] |  | ||||||
|         private class CommandExceptionCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("code", 'c')] |  | ||||||
|             public int ExitCode { get; set; } = 1337; |  | ||||||
|  |  | ||||||
|             [CommandOption("msg", 'm')] |  | ||||||
|             public string? Message { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => throw new CommandException(Message, ExitCode); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,84 +1,174 @@ | |||||||
| using System.Collections.Generic; | using System.Threading.Tasks; | ||||||
| using System.IO; | using CliFx.Tests.Commands; | ||||||
| using System.Threading.Tasks; |  | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class ErrorReportingSpecs |     public class ErrorReportingSpecs | ||||||
|     { |     { | ||||||
|  |         private readonly ITestOutputHelper _output; | ||||||
|  |  | ||||||
|  |         public ErrorReportingSpecs(ITestOutputHelper output) => _output = output; | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Command_may_throw_a_generic_exception_which_exits_and_prints_full_error_details() |         public async Task Command_may_throw_a_generic_exception_which_exits_and_prints_error_message_and_stack_trace() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdErr = new MemoryStream(); |             var (console, stdOut, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(error: stdErr); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(GenericExceptionCommand)) |                 .AddCommand<GenericExceptionCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync(new[] {"cmd", "-m", "Kaput"}); | ||||||
|                 new[] {"exc", "-m", "Kaput"}, |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdErrData = console.Error.Encoding.GetString(stdErr.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().NotBe(0); |             exitCode.Should().NotBe(0); | ||||||
|             stdErrData.Should().Contain("Kaput"); |             stdOut.GetString().Should().BeEmpty(); | ||||||
|             stdErrData.Length.Should().BeGreaterThan("Kaput".Length); |             stdErr.GetString().Should().ContainAll( | ||||||
|  |                 "System.Exception:", | ||||||
|  |                 "Kaput", "at", | ||||||
|  |                 "CliFx.Tests" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_may_throw_a_generic_exception_with_inner_exception_which_exits_and_prints_error_message_and_stack_trace() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<GenericInnerExceptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"cmd", "-m", "Kaput", "-i", "FooBar"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdOut.GetString().Should().BeEmpty(); | ||||||
|  |             stdErr.GetString().Should().ContainAll( | ||||||
|  |                 "System.Exception:", | ||||||
|  |                 "FooBar", | ||||||
|  |                 "Kaput", "at", | ||||||
|  |                 "CliFx.Tests" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Command_may_throw_a_specialized_exception_which_exits_with_custom_code_and_prints_minimal_error_details() |         public async Task Command_may_throw_a_specialized_exception_which_exits_with_custom_code_and_prints_minimal_error_details() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdErr = new MemoryStream(); |             var (console, stdOut, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(error: stdErr); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(CommandExceptionCommand)) |                 .AddCommand<CommandExceptionCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync(new[] {"cmd", "-m", "Kaput", "-c", "69"}); | ||||||
|                 new[] {"exc", "-m", "Kaput", "-c", "69"}, |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdErrData = console.Error.Encoding.GetString(stdErr.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().Be(69); |             exitCode.Should().Be(69); | ||||||
|             stdErrData.Should().Be("Kaput"); |             stdOut.GetString().Should().BeEmpty(); | ||||||
|  |             stdErr.GetString().Trim().Should().Be("Kaput"); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Command_may_throw_a_specialized_exception_without_error_message_which_exits_and_prints_full_error_details() |         public async Task Command_may_throw_a_specialized_exception_without_error_message_which_exits_and_prints_full_error_details() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdErr = new MemoryStream(); |             var (console, stdOut, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(error: stdErr); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(CommandExceptionCommand)) |                 .AddCommand<CommandExceptionCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync(new[] {"cmd"}); | ||||||
|                 new[] {"exc", "-m", "Kaput"}, |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdErrData = console.Error.Encoding.GetString(stdErr.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().NotBe(0); |             exitCode.Should().NotBe(0); | ||||||
|             stdErrData.Should().NotBeEmpty(); |             stdOut.GetString().Should().BeEmpty(); | ||||||
|  |             stdErr.GetString().Should().ContainAll( | ||||||
|  |                 "CliFx.Exceptions.CommandException:", | ||||||
|  |                 "at", | ||||||
|  |                 "CliFx.Tests" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_may_throw_a_specialized_exception_which_exits_and_prints_help_text() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<CommandExceptionCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"cmd", "-m", "Kaput", "--show-help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Usage", | ||||||
|  |                 "Options", | ||||||
|  |                 "-h|--help" | ||||||
|  |             ); | ||||||
|  |             stdErr.GetString().Trim().Should().Be("Kaput"); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Command_shows_help_text_on_invalid_user_input() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, stdErr) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"not-a-valid-command", "-r", "foo"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().NotBe(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Usage", | ||||||
|  |                 "Options", | ||||||
|  |                 "-h|--help" | ||||||
|  |             ); | ||||||
|  |             stdErr.GetString().Should().NotBeNullOrWhiteSpace(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |             _output.WriteLine(stdErr.GetString()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,96 +0,0 @@ | |||||||
| using System.Collections.Generic; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class HelpTextSpecs |  | ||||||
|     { |  | ||||||
|         [Command(Description = "DefaultCommand description.")] |  | ||||||
|         private class DefaultCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("option-a", 'a', Description = "OptionA description.")] |  | ||||||
|             public string? OptionA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-b", 'b', Description = "OptionB description.")] |  | ||||||
|             public string? OptionB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("cmd", Description = "NamedCommand description.")] |  | ||||||
|         private class NamedCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0, Name = "param-a", Description = "ParameterA description.")] |  | ||||||
|             public string? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-c", 'c', Description = "OptionC description.")] |  | ||||||
|             public string? OptionC { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-d", 'd', Description = "OptionD description.")] |  | ||||||
|             public string? OptionD { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("cmd sub", Description = "NamedSubCommand description.")] |  | ||||||
|         private class NamedSubCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0, Name = "param-b", Description = "ParameterB description.")] |  | ||||||
|             public string? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(1, Name = "param-c", Description = "ParameterC description.")] |  | ||||||
|             public string? ParameterC { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-e", 'e', Description = "OptionE description.")] |  | ||||||
|             public string? OptionE { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("cmd-with-params")] |  | ||||||
|         private class ParametersCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandParameter(0, Name = "first")] |  | ||||||
|             public string? ParameterA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(10)] |  | ||||||
|             public int? ParameterB { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandParameter(20, Description = "A list of numbers", Name = "third list")] |  | ||||||
|             public IEnumerable<int>? ParameterC { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option", 'o')] |  | ||||||
|             public string? Option { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("cmd-with-req-opts")] |  | ||||||
|         private class RequiredOptionsCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("option-f", 'f', IsRequired = true)] |  | ||||||
|             public string? OptionF { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-g", 'g', IsRequired = true)] |  | ||||||
|             public IEnumerable<int>? OptionG { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-h", 'h')] |  | ||||||
|             public string? OptionH { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("cmd-with-env-vars")] |  | ||||||
|         private class EnvironmentVariableCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("option-a", 'a', IsRequired = true, EnvironmentVariableName = "ENV_OPT_A")] |  | ||||||
|             public string? OptionA { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("option-b", 'b', EnvironmentVariableName = "ENV_OPT_B")] |  | ||||||
|             public string? OptionB { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) => default; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,246 +1,180 @@ | |||||||
| using System.IO; | using System.Threading.Tasks; | ||||||
| using System.Threading.Tasks; | using CliFx.Tests.Commands; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class HelpTextSpecs |     public class HelpTextSpecs | ||||||
|     { |     { | ||||||
|         [Fact] |         private readonly ITestOutputHelper _output; | ||||||
|         public async Task Version_information_can_be_requested_by_providing_the_version_option_without_other_arguments() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |         public HelpTextSpecs(ITestOutputHelper output) => _output = output; | ||||||
|                 .AddCommand(typeof(DefaultCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedSubCommand)) |  | ||||||
|                 .UseVersionText("v6.9") |  | ||||||
|                 .UseConsole(console) |  | ||||||
|                 .Build(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             var exitCode = await application.RunAsync(new[] {"--version"}); |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             exitCode.Should().Be(0); |  | ||||||
|             stdOutData.Should().Be("v6.9"); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public async Task Help_text_can_be_requested_by_providing_the_help_option() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |  | ||||||
|                 .AddCommand(typeof(DefaultCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedSubCommand)) |  | ||||||
|                 .UseTitle("AppTitle") |  | ||||||
|                 .UseVersionText("AppVer") |  | ||||||
|                 .UseDescription("AppDesc") |  | ||||||
|                 .UseExecutableName("AppExe") |  | ||||||
|                 .UseConsole(console) |  | ||||||
|                 .Build(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             await application.RunAsync(new[] {"--help"}); |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             stdOutData.Should().ContainAll( |  | ||||||
|                 "AppTitle", "AppVer", |  | ||||||
|                 "AppDesc", |  | ||||||
|                 "Usage", |  | ||||||
|                 "AppExe", "[command]", "[options]", |  | ||||||
|                 "Options", |  | ||||||
|                 "-a|--option-a", "OptionA description.", |  | ||||||
|                 "-b|--option-b", "OptionB description.", |  | ||||||
|                 "-h|--help", "Shows help text.", |  | ||||||
|                 "--version", "Shows version information.", |  | ||||||
|                 "Commands", |  | ||||||
|                 "cmd", "NamedCommand description.", |  | ||||||
|                 "You can run", "to show help on a specific command." |  | ||||||
|             ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public async Task Help_text_can_be_requested_on_a_specific_named_command() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |  | ||||||
|                 .AddCommand(typeof(DefaultCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedSubCommand)) |  | ||||||
|                 .UseConsole(console) |  | ||||||
|                 .Build(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             await application.RunAsync(new[] {"cmd", "--help"}); |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             stdOutData.Should().ContainAll( |  | ||||||
|                 "Description", |  | ||||||
|                 "NamedCommand description.", |  | ||||||
|                 "Usage", |  | ||||||
|                 "cmd", "[command]", "<param-a>", "[options]", |  | ||||||
|                 "Parameters", |  | ||||||
|                 "* param-a", "ParameterA description.", |  | ||||||
|                 "Options", |  | ||||||
|                 "-c|--option-c", "OptionC description.", |  | ||||||
|                 "-d|--option-d", "OptionD description.", |  | ||||||
|                 "-h|--help", "Shows help text.", |  | ||||||
|                 "Commands", |  | ||||||
|                 "sub", "SubCommand description.", |  | ||||||
|                 "You can run", "to show help on a specific command." |  | ||||||
|             ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public async Task Help_text_can_be_requested_on_a_specific_named_sub_command() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |  | ||||||
|                 .AddCommand(typeof(DefaultCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedSubCommand)) |  | ||||||
|                 .UseConsole(console) |  | ||||||
|                 .Build(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             await application.RunAsync(new[] {"cmd", "sub", "--help"}); |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             stdOutData.Should().ContainAll( |  | ||||||
|                 "Description", |  | ||||||
|                 "SubCommand description.", |  | ||||||
|                 "Usage", |  | ||||||
|                 "cmd sub", "<param-b>", "<param-c>", "[options]", |  | ||||||
|                 "Parameters", |  | ||||||
|                 "* param-b", "ParameterB description.", |  | ||||||
|                 "* param-c", "ParameterC description.", |  | ||||||
|                 "Options", |  | ||||||
|                 "-e|--option-e", "OptionE description.", |  | ||||||
|                 "-h|--help", "Shows help text." |  | ||||||
|             ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |  | ||||||
|         public async Task Help_text_can_be_requested_without_specifying_command_even_if_default_command_is_not_defined() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |  | ||||||
|                 .AddCommand(typeof(NamedCommand)) |  | ||||||
|                 .AddCommand(typeof(NamedSubCommand)) |  | ||||||
|                 .UseConsole(console) |  | ||||||
|                 .Build(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             await application.RunAsync(new[] {"--help"}); |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             stdOutData.Should().ContainAll( |  | ||||||
|                 "Usage", |  | ||||||
|                 "[command]", |  | ||||||
|                 "Options", |  | ||||||
|                 "-h|--help", "Shows help text.", |  | ||||||
|                 "--version", "Shows version information.", |  | ||||||
|                 "Commands", |  | ||||||
|                 "cmd", "NamedCommand description.", |  | ||||||
|                 "You can run", "to show help on a specific command." |  | ||||||
|             ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Help_text_shows_usage_format_which_lists_all_parameters() |         public async Task Help_text_shows_usage_format_which_lists_all_parameters() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdOut = new MemoryStream(); |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(ParametersCommand)) |                 .AddCommand<WithParametersCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             await application.RunAsync(new[] {"cmd-with-params", "--help"}); |             var exitCode = await application.RunAsync(new[] {"cmd", "--help"}); | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOutData.Should().ContainAll( |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|                 "Usage", |                 "Usage", | ||||||
|                 "cmd-with-params", "<first>", "<parameterb>", "<third list...>", "[options]" |                 "cmd", "<parama>", "<paramb>", "<paramc...>" | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Help_text_shows_usage_format_which_lists_all_required_options() |         public async Task Help_text_shows_usage_format_which_lists_all_required_options() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdOut = new MemoryStream(); |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(RequiredOptionsCommand)) |                 .AddCommand<WithRequiredOptionsCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             await application.RunAsync(new[] {"cmd-with-req-opts", "--help"}); |             var exitCode = await application.RunAsync(new[] {"cmd", "--help"}); | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOutData.Should().ContainAll( |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|                 "Usage", |                 "Usage", | ||||||
|                 "cmd-with-req-opts", "--option-f <value>", "--option-g <values...>", "[options]", |                 "cmd", "--opt-a <value>", "--opt-c <values...>", "[options]", | ||||||
|                 "Options", |                 "Options", | ||||||
|                 "* -f|--option-f", |                 "* -a|--opt-a", | ||||||
|                 "* -g|--option-g", |                 "-b|--opt-b", | ||||||
|                 "-h|--option-h" |                 "* -c|--opt-c" | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Help_text_lists_environment_variable_names_for_options_that_have_them_defined() |         public async Task Help_text_shows_usage_format_which_lists_available_sub_commands() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdOut = new MemoryStream(); |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(EnvironmentVariableCommand)) |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             await application.RunAsync(new[] {"cmd-with-env-vars", "--help"}); |             var exitCode = await application.RunAsync(new[] {"--help"}); | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOutData.Should().ContainAll( |             exitCode.Should().Be(0); | ||||||
|                 "Options", |             stdOut.GetString().Should().ContainAll( | ||||||
|                 "* -a|--option-a", "Environment variable:", "ENV_OPT_A", |                 "Usage", | ||||||
|                 "-b|--option-b", "Environment variable:", "ENV_OPT_B" |                 "... named", | ||||||
|  |                 "... named sub" | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_shows_all_valid_values_for_enum_arguments() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<WithEnumArgumentsCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"cmd", "--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Parameters", | ||||||
|  |                 "enum", "Valid values: \"Value1\", \"Value2\", \"Value3\".", | ||||||
|  |                 "Options", | ||||||
|  |                 "--enum", "Valid values: \"Value1\", \"Value2\", \"Value3\".", | ||||||
|  |                 "* --required-enum", "Valid values: \"Value1\", \"Value2\", \"Value3\"." | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_shows_environment_variable_names_for_options_that_have_them_defined() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<WithEnvironmentVariablesCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"cmd", "--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Options", | ||||||
|  |                 "-a|--opt-a", "Environment variable:", "ENV_OPT_A", | ||||||
|  |                 "-b|--opt-b", "Environment variable:", "ENV_OPT_B" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_shows_default_values_for_non_required_options() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<WithDefaultValuesCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"cmd", "--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Options", | ||||||
|  |                 "--obj", "Default: \"42\"", | ||||||
|  |                 "--str", "Default: \"foo\"", | ||||||
|  |                 "--str-empty", "Default: \"\"", | ||||||
|  |                 "--str-array", "Default: \"foo\" \"bar\" \"baz\"", | ||||||
|  |                 "--bool", "Default: \"True\"", | ||||||
|  |                 "--char", "Default: \"t\"", | ||||||
|  |                 "--int", "Default: \"1337\"", | ||||||
|  |                 "--int-nullable", "Default: \"1337\"", | ||||||
|  |                 "--int-array", "Default: \"1\" \"2\" \"3\"", | ||||||
|  |                 "--timespan", "Default: \"02:03:00\"", | ||||||
|  |                 "--enum", "Default: \"Value2\"" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
							
								
								
									
										10
									
								
								CliFx.Tests/Internal/JsonExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								CliFx.Tests/Internal/JsonExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | using Newtonsoft.Json; | ||||||
|  |  | ||||||
|  | namespace CliFx.Tests.Internal | ||||||
|  | { | ||||||
|  |     internal static class JsonExtensions | ||||||
|  |     { | ||||||
|  |         public static T DeserializeJson<T>(this string json) => | ||||||
|  |             JsonConvert.DeserializeObject<T>(json); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,51 +0,0 @@ | |||||||
| using System.Collections.Generic; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| using CliFx.Attributes; |  | ||||||
|  |  | ||||||
| namespace CliFx.Tests |  | ||||||
| { |  | ||||||
|     public partial class RoutingSpecs |  | ||||||
|     { |  | ||||||
|         [Command] |  | ||||||
|         private class DefaultCommand : ICommand |  | ||||||
|         { |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) |  | ||||||
|             { |  | ||||||
|                 console.Output.WriteLine("Hello world!"); |  | ||||||
|                 return default; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("concat", Description = "Concatenate strings.")] |  | ||||||
|         private class ConcatCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption('i', IsRequired = true, Description = "Input strings.")] |  | ||||||
|             public IReadOnlyList<string> Inputs { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption('s', Description = "String separator.")] |  | ||||||
|             public string Separator { get; set; } = ""; |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) |  | ||||||
|             { |  | ||||||
|                 console.Output.WriteLine(string.Join(Separator, Inputs)); |  | ||||||
|                 return default; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Command("div", Description = "Divide one number by another.")] |  | ||||||
|         private class DivideCommand : ICommand |  | ||||||
|         { |  | ||||||
|             [CommandOption("dividend", 'D', IsRequired = true, Description = "The number to divide.")] |  | ||||||
|             public double Dividend { get; set; } |  | ||||||
|  |  | ||||||
|             [CommandOption("divisor", 'd', IsRequired = true, Description = "The number to divide by.")] |  | ||||||
|             public double Divisor { get; set; } |  | ||||||
|  |  | ||||||
|             public ValueTask ExecuteAsync(IConsole console) |  | ||||||
|             { |  | ||||||
|                 console.Output.WriteLine(Dividend / Divisor); |  | ||||||
|                 return default; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,90 +1,235 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections.Generic; |  | ||||||
| using System.IO; |  | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
|  | using CliFx.Tests.Commands; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public partial class RoutingSpecs |     public class RoutingSpecs | ||||||
|     { |     { | ||||||
|  |         private readonly ITestOutputHelper _output; | ||||||
|  |  | ||||||
|  |         public RoutingSpecs(ITestOutputHelper testOutput) => _output = testOutput; | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Default_command_is_executed_if_provided_arguments_do_not_match_any_named_command() |         public async Task Default_command_is_executed_if_provided_arguments_do_not_match_any_named_command() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdOut = new MemoryStream(); |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(DefaultCommand)) |                 .AddCommand<DefaultCommand>() | ||||||
|                 .AddCommand(typeof(ConcatCommand)) |                 .AddCommand<NamedCommand>() | ||||||
|                 .AddCommand(typeof(DivideCommand)) |                 .AddCommand<NamedSubCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|                 Array.Empty<string>(), |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().Be(0); |             exitCode.Should().Be(0); | ||||||
|             stdOutData.Should().Be("Hello world!"); |             stdOut.GetString().Trim().Should().Be(DefaultCommand.ExpectedOutputText); | ||||||
|         } |  | ||||||
|  |  | ||||||
|         [Fact] |             _output.WriteLine(stdOut.GetString()); | ||||||
|         public async Task Help_text_is_printed_if_no_arguments_were_provided_and_default_command_is_not_defined() |  | ||||||
|         { |  | ||||||
|             // Arrange |  | ||||||
|             await using var stdOut = new MemoryStream(); |  | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |  | ||||||
|                 .AddCommand(typeof(ConcatCommand)) |  | ||||||
|                 .AddCommand(typeof(DivideCommand)) |  | ||||||
|                 .UseConsole(console) |  | ||||||
|                 .UseDescription("This will be visible in help") |  | ||||||
|                 .Build(); |  | ||||||
|  |  | ||||||
|             // Act |  | ||||||
|             var exitCode = await application.RunAsync( |  | ||||||
|                 Array.Empty<string>(), |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |  | ||||||
|             exitCode.Should().Be(0); |  | ||||||
|             stdOutData.Should().Contain("This will be visible in help"); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public async Task Specific_named_command_is_executed_if_provided_arguments_match_its_name() |         public async Task Specific_named_command_is_executed_if_provided_arguments_match_its_name() | ||||||
|         { |         { | ||||||
|             // Arrange |             // Arrange | ||||||
|             await using var stdOut = new MemoryStream(); |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|             var console = new VirtualConsole(output: stdOut); |  | ||||||
|  |  | ||||||
|             var application = new CliApplicationBuilder() |             var application = new CliApplicationBuilder() | ||||||
|                 .AddCommand(typeof(DefaultCommand)) |                 .AddCommand<DefaultCommand>() | ||||||
|                 .AddCommand(typeof(ConcatCommand)) |                 .AddCommand<NamedCommand>() | ||||||
|                 .AddCommand(typeof(DivideCommand)) |                 .AddCommand<NamedSubCommand>() | ||||||
|                 .UseConsole(console) |                 .UseConsole(console) | ||||||
|                 .Build(); |                 .Build(); | ||||||
|  |  | ||||||
|             // Act |             // Act | ||||||
|             var exitCode = await application.RunAsync( |             var exitCode = await application.RunAsync(new[] {"named"}); | ||||||
|                 new[] {"concat", "-i", "foo", "bar", "-s", ", "}, |  | ||||||
|                 new Dictionary<string, string>()); |  | ||||||
|  |  | ||||||
|             var stdOutData = console.Output.Encoding.GetString(stdOut.ToArray()).TrimEnd(); |  | ||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             exitCode.Should().Be(0); |             exitCode.Should().Be(0); | ||||||
|             stdOutData.Should().Be("foo, bar"); |             stdOut.GetString().Trim().Should().Be(NamedCommand.ExpectedOutputText); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Specific_named_sub_command_is_executed_if_provided_arguments_match_its_name() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"named", "sub"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Trim().Should().Be(NamedSubCommand.ExpectedOutputText); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_is_printed_if_no_arguments_were_provided_and_default_command_is_not_defined() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .UseDescription("This will be visible in help") | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(Array.Empty<string>()); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().Contain("This will be visible in help"); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_is_printed_if_provided_arguments_contain_the_help_option() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Default command description", | ||||||
|  |                 "Usage" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_is_printed_if_provided_arguments_contain_the_help_option_even_if_default_command_is_not_defined() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseDescription("This will be visible in help") | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().Contain("This will be visible in help"); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_for_a_specific_named_command_is_printed_if_provided_arguments_match_its_name_and_contain_the_help_option() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"named", "--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Named command description", | ||||||
|  |                 "Usage", | ||||||
|  |                 "named" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Help_text_for_a_specific_named_sub_command_is_printed_if_provided_arguments_match_its_name_and_contain_the_help_option() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"named", "sub", "--help"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Should().ContainAll( | ||||||
|  |                 "Named sub command description", | ||||||
|  |                 "Usage", | ||||||
|  |                 "named", "sub" | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         [Fact] | ||||||
|  |         public async Task Version_is_printed_if_the_only_provided_argument_is_the_version_option() | ||||||
|  |         { | ||||||
|  |             // Arrange | ||||||
|  |             var (console, stdOut, _) = VirtualConsole.CreateBuffered(); | ||||||
|  |  | ||||||
|  |             var application = new CliApplicationBuilder() | ||||||
|  |                 .AddCommand<DefaultCommand>() | ||||||
|  |                 .AddCommand<NamedCommand>() | ||||||
|  |                 .AddCommand<NamedSubCommand>() | ||||||
|  |                 .UseVersionText("v6.9") | ||||||
|  |                 .UseConsole(console) | ||||||
|  |                 .Build(); | ||||||
|  |  | ||||||
|  |             // Act | ||||||
|  |             var exitCode = await application.RunAsync(new[] {"--version"}); | ||||||
|  |  | ||||||
|  |             // Assert | ||||||
|  |             exitCode.Should().Be(0); | ||||||
|  |             stdOut.GetString().Trim().Should().Be("v6.9"); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOut.GetString()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -3,11 +3,16 @@ using System.Linq; | |||||||
| using CliFx.Utilities; | using CliFx.Utilities; | ||||||
| using FluentAssertions; | using FluentAssertions; | ||||||
| using Xunit; | using Xunit; | ||||||
|  | using Xunit.Abstractions; | ||||||
|  |  | ||||||
| namespace CliFx.Tests | namespace CliFx.Tests | ||||||
| { | { | ||||||
|     public class UtilitiesSpecs |     public class UtilitiesSpecs | ||||||
|     { |     { | ||||||
|  |         private readonly ITestOutputHelper _output; | ||||||
|  |  | ||||||
|  |         public UtilitiesSpecs(ITestOutputHelper output) => _output = output; | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
|         public void Progress_ticker_can_be_used_to_report_progress_to_console() |         public void Progress_ticker_can_be_used_to_report_progress_to_console() | ||||||
|         { |         { | ||||||
| @@ -28,6 +33,8 @@ namespace CliFx.Tests | |||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOutData.Should().ContainAll(progressStringValues); |             stdOutData.Should().ContainAll(progressStringValues); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOutData); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         [Fact] |         [Fact] | ||||||
| @@ -49,6 +56,8 @@ namespace CliFx.Tests | |||||||
|  |  | ||||||
|             // Assert |             // Assert | ||||||
|             stdOutData.Should().BeEmpty(); |             stdOutData.Should().BeEmpty(); | ||||||
|  |  | ||||||
|  |             _output.WriteLine(stdOutData); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,11 +1,12 @@ | |||||||
| <Project> | <Project> | ||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <Version>1.1</Version> |     <Version>1.6</Version> | ||||||
|     <Company>Tyrrrz</Company> |     <Company>Tyrrrz</Company> | ||||||
|     <Copyright>Copyright (C) Alexey Golub</Copyright> |     <Copyright>Copyright (C) Alexey Golub</Copyright> | ||||||
|     <LangVersion>latest</LangVersion> |     <LangVersion>latest</LangVersion> | ||||||
|     <Nullable>enable</Nullable> |     <Nullable>enable</Nullable> | ||||||
|  |     <WarningsAsErrors>nullable</WarningsAsErrors> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
							
								
								
									
										32
									
								
								CliFx.sln
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								CliFx.sln
									
									
									
									
									
								
							| @@ -10,16 +10,20 @@ EndProject | |||||||
| Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{3AAE8166-BB8E-49DA-844C-3A0EE6BD40A0}" | Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{3AAE8166-BB8E-49DA-844C-3A0EE6BD40A0}" | ||||||
| 	ProjectSection(SolutionItems) = preProject | 	ProjectSection(SolutionItems) = preProject | ||||||
| 		Changelog.md = Changelog.md | 		Changelog.md = Changelog.md | ||||||
|  | 		CliFx.props = CliFx.props | ||||||
| 		License.txt = License.txt | 		License.txt = License.txt | ||||||
| 		Readme.md = Readme.md | 		Readme.md = Readme.md | ||||||
| 		CliFx.props = CliFx.props |  | ||||||
| 	EndProjectSection | 	EndProjectSection | ||||||
| EndProject | EndProject | ||||||
| Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CliFx.Benchmarks", "CliFx.Benchmarks\CliFx.Benchmarks.csproj", "{8ACD6DC2-D768-4850-9223-5B7C83A78513}" | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CliFx.Benchmarks", "CliFx.Benchmarks\CliFx.Benchmarks.csproj", "{8ACD6DC2-D768-4850-9223-5B7C83A78513}" | ||||||
| EndProject | EndProject | ||||||
| Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CliFx.Demo", "CliFx.Demo\CliFx.Demo.csproj", "{AAB6844C-BF71-448F-A11B-89AEE459AB15}" | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CliFx.Demo", "CliFx.Demo\CliFx.Demo.csproj", "{AAB6844C-BF71-448F-A11B-89AEE459AB15}" | ||||||
| EndProject | EndProject | ||||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CliFx.Tests.Dummy", "CliFx.Tests.Dummy\CliFx.Tests.Dummy.csproj", "{F717347D-8656-44DA-A4A2-BE515E8C4655}" | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CliFx.Tests.Dummy", "CliFx.Tests.Dummy\CliFx.Tests.Dummy.csproj", "{F717347D-8656-44DA-A4A2-BE515E8C4655}" | ||||||
|  | EndProject | ||||||
|  | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CliFx.Analyzers", "CliFx.Analyzers\CliFx.Analyzers.csproj", "{F8460D69-F8CF-405C-A6ED-BED02A21DB42}" | ||||||
|  | EndProject | ||||||
|  | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CliFx.Analyzers.Tests", "CliFx.Analyzers.Tests\CliFx.Analyzers.Tests.csproj", "{49878E75-2097-4C79-9151-B98A28FBB973}" | ||||||
| EndProject | EndProject | ||||||
| Global | Global | ||||||
| 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | ||||||
| @@ -91,6 +95,30 @@ Global | |||||||
| 		{F717347D-8656-44DA-A4A2-BE515E8C4655}.Release|x64.Build.0 = Release|Any CPU | 		{F717347D-8656-44DA-A4A2-BE515E8C4655}.Release|x64.Build.0 = Release|Any CPU | ||||||
| 		{F717347D-8656-44DA-A4A2-BE515E8C4655}.Release|x86.ActiveCfg = Release|Any CPU | 		{F717347D-8656-44DA-A4A2-BE515E8C4655}.Release|x86.ActiveCfg = Release|Any CPU | ||||||
| 		{F717347D-8656-44DA-A4A2-BE515E8C4655}.Release|x86.Build.0 = Release|Any CPU | 		{F717347D-8656-44DA-A4A2-BE515E8C4655}.Release|x86.Build.0 = Release|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Debug|x64.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Debug|x64.Build.0 = Debug|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Debug|x86.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Debug|x86.Build.0 = Debug|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Release|Any CPU.Build.0 = Release|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Release|x64.ActiveCfg = Release|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Release|x64.Build.0 = Release|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Release|x86.ActiveCfg = Release|Any CPU | ||||||
|  | 		{F8460D69-F8CF-405C-A6ED-BED02A21DB42}.Release|x86.Build.0 = Release|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Debug|x64.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Debug|x64.Build.0 = Debug|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Debug|x86.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Debug|x86.Build.0 = Debug|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Release|Any CPU.Build.0 = Release|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Release|x64.ActiveCfg = Release|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Release|x64.Build.0 = Release|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Release|x86.ActiveCfg = Release|Any CPU | ||||||
|  | 		{49878E75-2097-4C79-9151-B98A28FBB973}.Release|x86.Build.0 = Release|Any CPU | ||||||
| 	EndGlobalSection | 	EndGlobalSection | ||||||
| 	GlobalSection(SolutionProperties) = preSolution | 	GlobalSection(SolutionProperties) = preSolution | ||||||
| 		HideSolutionNode = FALSE | 		HideSolutionNode = FALSE | ||||||
|   | |||||||
| @@ -28,7 +28,8 @@ namespace CliFx | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         public ApplicationConfiguration( |         public ApplicationConfiguration( | ||||||
|             IReadOnlyList<Type> commandTypes, |             IReadOnlyList<Type> commandTypes, | ||||||
|             bool isDebugModeAllowed, bool isPreviewModeAllowed) |             bool isDebugModeAllowed, | ||||||
|  |             bool isPreviewModeAllowed) | ||||||
|         { |         { | ||||||
|             CommandTypes = commandTypes; |             CommandTypes = commandTypes; | ||||||
|             IsDebugModeAllowed = isDebugModeAllowed; |             IsDebugModeAllowed = isDebugModeAllowed; | ||||||
|   | |||||||
| @@ -28,7 +28,11 @@ | |||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Initializes an instance of <see cref="ApplicationMetadata"/>. |         /// Initializes an instance of <see cref="ApplicationMetadata"/>. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|         public ApplicationMetadata(string title, string executableName, string versionText, string? description) |         public ApplicationMetadata( | ||||||
|  |             string title, | ||||||
|  |             string executableName, | ||||||
|  |             string versionText, | ||||||
|  |             string? description) | ||||||
|         { |         { | ||||||
|             Title = title; |             Title = title; | ||||||
|             ExecutableName = executableName; |             ExecutableName = executableName; | ||||||
|   | |||||||
							
								
								
									
										30
									
								
								CliFx/ArgumentValueConverter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								CliFx/ArgumentValueConverter.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | namespace CliFx | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Implements custom conversion logic that maps an argument value to a domain type. | ||||||
|  |     /// </summary> | ||||||
|  |     /// <remarks> | ||||||
|  |     /// This type is public for legacy reasons. | ||||||
|  |     /// Please derive from <see cref="ArgumentValueConverter{T}"/> instead. | ||||||
|  |     /// </remarks> | ||||||
|  |     public interface IArgumentValueConverter | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Converts an input value to object of required type. | ||||||
|  |         /// </summary> | ||||||
|  |         public object ConvertFrom(string value); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// A base type for custom argument converters. | ||||||
|  |     /// </summary> | ||||||
|  |     public abstract class ArgumentValueConverter<T> : IArgumentValueConverter | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Converts an input value to object of required type. | ||||||
|  |         /// </summary> | ||||||
|  |         public abstract T ConvertFrom(string value); | ||||||
|  |  | ||||||
|  |         object IArgumentValueConverter.ConvertFrom(string value) => ConvertFrom(value)!; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								CliFx/ArgumentValueValidator.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								CliFx/ArgumentValueValidator.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | namespace CliFx | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Represents a result of a validation. | ||||||
|  |     /// </summary> | ||||||
|  |     public partial class ValidationResult | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Whether validation was successful. | ||||||
|  |         /// </summary> | ||||||
|  |         public bool IsValid => ErrorMessage == null; | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// If validation has failed, contains the associated error, otherwise null. | ||||||
|  |         /// </summary> | ||||||
|  |         public string? ErrorMessage { get; } | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes an instance of <see cref="ValidationResult"/>. | ||||||
|  |         /// </summary> | ||||||
|  |         public ValidationResult(string? errorMessage = null) => | ||||||
|  |             ErrorMessage = errorMessage; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public partial class ValidationResult | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Creates successful result, meaning that the validation has passed. | ||||||
|  |         /// </summary> | ||||||
|  |         public static ValidationResult Ok() => new ValidationResult(); | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Creates an error result, meaning that the validation has failed. | ||||||
|  |         /// </summary> | ||||||
|  |         public static ValidationResult Error(string message) => new ValidationResult(message); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     internal interface IArgumentValueValidator | ||||||
|  |     { | ||||||
|  |         ValidationResult Validate(object value); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// A base type for custom argument validators. | ||||||
|  |     /// </summary> | ||||||
|  |     public abstract class ArgumentValueValidator<T> : IArgumentValueValidator | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates the value. | ||||||
|  |         /// </summary> | ||||||
|  |         public abstract ValidationResult Validate(T value); | ||||||
|  |  | ||||||
|  |         ValidationResult IArgumentValueValidator.Validate(object value) => Validate((T) value); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								CliFx/Attributes/CommandArgumentAttribute.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								CliFx/Attributes/CommandArgumentAttribute.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | using System; | ||||||
|  |  | ||||||
|  | namespace CliFx.Attributes | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Properties shared by parameter and option arguments. | ||||||
|  |     /// </summary> | ||||||
|  |     [AttributeUsage(AttributeTargets.Property)] | ||||||
|  |     public abstract class CommandArgumentAttribute : Attribute | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Option description, which is used in help text. | ||||||
|  |         /// </summary> | ||||||
|  |         public string? Description { get; set; } | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Type of converter to use when mapping the argument value. | ||||||
|  |         /// Converter must derive from <see cref="ArgumentValueConverter{T}"/>. | ||||||
|  |         /// </summary> | ||||||
|  |         public Type? Converter { get; set; } | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Types of validators to use when mapping the argument value. | ||||||
|  |         /// Validators must derive from <see cref="ArgumentValueValidator{T}"/>. | ||||||
|  |         /// </summary> | ||||||
|  |         public Type[] Validators { get; set; } = Array.Empty<Type>(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -6,17 +6,17 @@ namespace CliFx.Attributes | |||||||
|     /// Annotates a property that defines a command option. |     /// Annotates a property that defines a command option. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     [AttributeUsage(AttributeTargets.Property)] |     [AttributeUsage(AttributeTargets.Property)] | ||||||
|     public class CommandOptionAttribute : Attribute |     public class CommandOptionAttribute : CommandArgumentAttribute | ||||||
|     { |     { | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Option name. |         /// Option name (must be longer than a single character). | ||||||
|         /// Either <see cref="Name"/> or <see cref="ShortName"/> must be set. |         /// Either <see cref="Name"/> or <see cref="ShortName"/> must be set. | ||||||
|         /// All options in a command must have different names (comparison is not case-sensitive). |         /// All options in a command must have different names (comparison is not case-sensitive). | ||||||
|         /// </summary> |         /// </summary> | ||||||
|         public string? Name { get; } |         public string? Name { get; } | ||||||
|  |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Option short name. |         /// Option short name (single character). | ||||||
|         /// Either <see cref="Name"/> or <see cref="ShortName"/> must be set. |         /// Either <see cref="Name"/> or <see cref="ShortName"/> must be set. | ||||||
|         /// All options in a command must have different short names (comparison is case-sensitive). |         /// All options in a command must have different short names (comparison is case-sensitive). | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @@ -27,11 +27,6 @@ namespace CliFx.Attributes | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         public bool IsRequired { get; set; } |         public bool IsRequired { get; set; } | ||||||
|  |  | ||||||
|         /// <summary> |  | ||||||
|         /// Option description, which is used in help text. |  | ||||||
|         /// </summary> |  | ||||||
|         public string? Description { get; set; } |  | ||||||
|  |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Environment variable that will be used as fallback if no option value is specified. |         /// Environment variable that will be used as fallback if no option value is specified. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ namespace CliFx.Attributes | |||||||
|     /// Annotates a property that defines a command parameter. |     /// Annotates a property that defines a command parameter. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     [AttributeUsage(AttributeTargets.Property)] |     [AttributeUsage(AttributeTargets.Property)] | ||||||
|     public class CommandParameterAttribute : Attribute |     public class CommandParameterAttribute : CommandArgumentAttribute | ||||||
|     { |     { | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Order of this parameter compared to other parameters. |         /// Order of this parameter compared to other parameters. | ||||||
| @@ -21,11 +21,6 @@ namespace CliFx.Attributes | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         public string? Name { get; set; } |         public string? Name { get; set; } | ||||||
|  |  | ||||||
|         /// <summary> |  | ||||||
|         /// Parameter description, which is used in help text. |  | ||||||
|         /// </summary> |  | ||||||
|         public string? Description { get; set; } |  | ||||||
|  |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Initializes an instance of <see cref="CommandParameterAttribute"/>. |         /// Initializes an instance of <see cref="CommandParameterAttribute"/>. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|   | |||||||
| @@ -1,18 +1,21 @@ | |||||||
| using System; | using System; | ||||||
| using System.Collections; | using System.Collections; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Diagnostics; | using System.Diagnostics; | ||||||
|  | using System.Diagnostics.CodeAnalysis; | ||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
|  | using CliFx.Attributes; | ||||||
| using CliFx.Domain; | using CliFx.Domain; | ||||||
| using CliFx.Exceptions; | using CliFx.Exceptions; | ||||||
|  | using CliFx.Internal; | ||||||
|  |  | ||||||
| namespace CliFx | namespace CliFx | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Command line application facade. |     /// Command line application facade. | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class CliApplication |     public partial class CliApplication | ||||||
|     { |     { | ||||||
|         private readonly ApplicationMetadata _metadata; |         private readonly ApplicationMetadata _metadata; | ||||||
|         private readonly ApplicationConfiguration _configuration; |         private readonly ApplicationConfiguration _configuration; | ||||||
| @@ -36,42 +39,34 @@ namespace CliFx | |||||||
|             _helpTextWriter = new HelpTextWriter(metadata, console); |             _helpTextWriter = new HelpTextWriter(metadata, console); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private async ValueTask<int?> HandleDebugDirectiveAsync(CommandLineInput commandLineInput) |         private async ValueTask LaunchAndWaitForDebuggerAsync() | ||||||
|         { |         { | ||||||
|             var isDebugMode = _configuration.IsDebugModeAllowed && commandLineInput.IsDebugDirectiveSpecified; |             var processId = ProcessEx.GetCurrentProcessId(); | ||||||
|             if (!isDebugMode) |  | ||||||
|                 return null; |  | ||||||
|  |  | ||||||
|             _console.WithForegroundColor(ConsoleColor.Green, () => |             _console.WithForegroundColor(ConsoleColor.Green, () => | ||||||
|                 _console.Output.WriteLine($"Attach debugger to PID {Process.GetCurrentProcess().Id} to continue.")); |                 _console.Output.WriteLine($"Attach debugger to PID {processId} to continue.")); | ||||||
|  |  | ||||||
|  |             Debugger.Launch(); | ||||||
|  |  | ||||||
|             while (!Debugger.IsAttached) |             while (!Debugger.IsAttached) | ||||||
|  |             { | ||||||
|                 await Task.Delay(100); |                 await Task.Delay(100); | ||||||
|  |             } | ||||||
|             return null; |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private int? HandlePreviewDirective(ApplicationSchema applicationSchema, CommandLineInput commandLineInput) |         private void WriteCommandLineInput(CommandInput input) | ||||||
|         { |         { | ||||||
|             var isPreviewMode = _configuration.IsPreviewModeAllowed && commandLineInput.IsPreviewDirectiveSpecified; |  | ||||||
|             if (!isPreviewMode) |  | ||||||
|                 return null; |  | ||||||
|  |  | ||||||
|             var commandSchema = applicationSchema.TryFindCommand(commandLineInput, out var argumentOffset); |  | ||||||
|  |  | ||||||
|             _console.Output.WriteLine("Parser preview:"); |  | ||||||
|  |  | ||||||
|             // Command name |             // Command name | ||||||
|             if (commandSchema != null && argumentOffset > 0) |             if (!string.IsNullOrWhiteSpace(input.CommandName)) | ||||||
|             { |             { | ||||||
|                 _console.WithForegroundColor(ConsoleColor.Cyan, () => |                 _console.WithForegroundColor(ConsoleColor.Cyan, () => | ||||||
|                     _console.Output.Write(commandSchema.Name)); |                     _console.Output.Write(input.CommandName)); | ||||||
|  |  | ||||||
|                 _console.Output.Write(' '); |                 _console.Output.Write(' '); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             // Parameters |             // Parameters | ||||||
|             foreach (var parameter in commandLineInput.UnboundArguments.Skip(argumentOffset)) |             foreach (var parameter in input.Parameters) | ||||||
|             { |             { | ||||||
|                 _console.Output.Write('<'); |                 _console.Output.Write('<'); | ||||||
|  |  | ||||||
| @@ -83,100 +78,145 @@ namespace CliFx | |||||||
|             } |             } | ||||||
|  |  | ||||||
|             // Options |             // Options | ||||||
|             foreach (var option in commandLineInput.Options) |             foreach (var option in input.Options) | ||||||
|             { |             { | ||||||
|                 _console.Output.Write('['); |                 _console.Output.Write('['); | ||||||
|  |  | ||||||
|                 _console.WithForegroundColor(ConsoleColor.White, () => |                 _console.WithForegroundColor(ConsoleColor.White, () => | ||||||
|                     _console.Output.Write(option)); |                 { | ||||||
|  |                     // Alias | ||||||
|  |                     _console.Output.Write(option.GetRawAlias()); | ||||||
|  |  | ||||||
|  |                     // Values | ||||||
|  |                     if (option.Values.Any()) | ||||||
|  |                     { | ||||||
|  |                         _console.Output.Write(' '); | ||||||
|  |                         _console.Output.Write(option.GetRawValues()); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |  | ||||||
|                 _console.Output.Write(']'); |                 _console.Output.Write(']'); | ||||||
|                 _console.Output.Write(' '); |                 _console.Output.Write(' '); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             _console.Output.WriteLine(); |             _console.Output.WriteLine(); | ||||||
|  |  | ||||||
|             return 0; |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private int? HandleVersionOption(CommandLineInput commandLineInput) |         private ICommand GetCommandInstance(CommandSchema command) => | ||||||
|         { |             command != FallbackDefaultCommand.Schema | ||||||
|             // Version option is available only on the default command (i.e. when arguments are not specified) |                 ? (ICommand) _typeActivator.CreateInstance(command.Type) | ||||||
|             var shouldRenderVersion = !commandLineInput.UnboundArguments.Any() && commandLineInput.IsVersionOptionSpecified; |                 : new FallbackDefaultCommand(); | ||||||
|             if (!shouldRenderVersion) |  | ||||||
|                 return null; |  | ||||||
|  |  | ||||||
|             _console.Output.WriteLine(_metadata.VersionText); |  | ||||||
|  |  | ||||||
|             return 0; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private int? HandleHelpOption(ApplicationSchema applicationSchema, CommandLineInput commandLineInput) |  | ||||||
|         { |  | ||||||
|             // Help is rendered either when it's requested or when the user provides no arguments and there is no default command |  | ||||||
|             var shouldRenderHelp = |  | ||||||
|                 commandLineInput.IsHelpOptionSpecified || |  | ||||||
|                 !applicationSchema.Commands.Any(c => c.IsDefault) && !commandLineInput.UnboundArguments.Any() && !commandLineInput.Options.Any(); |  | ||||||
|  |  | ||||||
|             if (!shouldRenderHelp) |  | ||||||
|                 return null; |  | ||||||
|  |  | ||||||
|             // Get the command schema that matches the input or use a dummy default command as a fallback |  | ||||||
|             var commandSchema = |  | ||||||
|                 applicationSchema.TryFindCommand(commandLineInput) ?? |  | ||||||
|                 CommandSchema.StubDefaultCommand; |  | ||||||
|  |  | ||||||
|             _helpTextWriter.Write(applicationSchema, commandSchema); |  | ||||||
|  |  | ||||||
|             return 0; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private async ValueTask<int> HandleCommandExecutionAsync( |  | ||||||
|             ApplicationSchema applicationSchema, |  | ||||||
|             CommandLineInput commandLineInput, |  | ||||||
|             IReadOnlyDictionary<string, string> environmentVariables) |  | ||||||
|         { |  | ||||||
|             await applicationSchema |  | ||||||
|                 .InitializeEntryPoint(commandLineInput, environmentVariables, _typeActivator) |  | ||||||
|                 .ExecuteAsync(_console); |  | ||||||
|  |  | ||||||
|             return 0; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Runs the application with specified command line arguments and environment variables, and returns the exit code. |         /// Runs the application with specified command line arguments and environment variables, and returns the exit code. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  |         /// <remarks> | ||||||
|  |         /// If a <see cref="CommandException"/> is thrown during command execution, it will be handled and routed to the console. | ||||||
|  |         /// Additionally, if the debugger is not attached (i.e. the app is running in production), all other exceptions thrown within | ||||||
|  |         /// this method will be handled and routed to the console as well. | ||||||
|  |         /// </remarks> | ||||||
|         public async ValueTask<int> RunAsync( |         public async ValueTask<int> RunAsync( | ||||||
|             IReadOnlyList<string> commandLineArguments, |             IReadOnlyList<string> commandLineArguments, | ||||||
|             IReadOnlyDictionary<string, string> environmentVariables) |             IReadOnlyDictionary<string, string> environmentVariables) | ||||||
|         { |         { | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 var applicationSchema = ApplicationSchema.Resolve(_configuration.CommandTypes); |                 var root = RootSchema.Resolve(_configuration.CommandTypes); | ||||||
|                 var commandLineInput = CommandLineInput.Parse(commandLineArguments); |                 var input = CommandInput.Parse(commandLineArguments, root.GetCommandNames()); | ||||||
|  |  | ||||||
|                 return |                 // Debug mode | ||||||
|                     await HandleDebugDirectiveAsync(commandLineInput) ?? |                 if (_configuration.IsDebugModeAllowed && input.IsDebugDirectiveSpecified) | ||||||
|                     HandlePreviewDirective(applicationSchema, commandLineInput) ?? |                 { | ||||||
|                     HandleVersionOption(commandLineInput) ?? |                     await LaunchAndWaitForDebuggerAsync(); | ||||||
|                     HandleHelpOption(applicationSchema, commandLineInput) ?? |                 } | ||||||
|                     await HandleCommandExecutionAsync(applicationSchema, commandLineInput, environmentVariables); |  | ||||||
|  |                 // Preview mode | ||||||
|  |                 if (_configuration.IsPreviewModeAllowed && input.IsPreviewDirectiveSpecified) | ||||||
|  |                 { | ||||||
|  |                     WriteCommandLineInput(input); | ||||||
|  |                     return ExitCode.Success; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // Try to get the command matching the input or fallback to default | ||||||
|  |                 var command = | ||||||
|  |                     root.TryFindCommand(input.CommandName) ?? | ||||||
|  |                     root.TryFindDefaultCommand() ?? | ||||||
|  |                     FallbackDefaultCommand.Schema; | ||||||
|  |  | ||||||
|  |                 // Version option | ||||||
|  |                 if (command.IsVersionOptionAvailable && input.IsVersionOptionSpecified) | ||||||
|  |                 { | ||||||
|  |                     _console.Output.WriteLine(_metadata.VersionText); | ||||||
|  |                     return ExitCode.Success; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // Get command instance (also used in help text) | ||||||
|  |                 var instance = GetCommandInstance(command); | ||||||
|  |  | ||||||
|  |                 // To avoid instantiating the command twice, we need to get default values | ||||||
|  |                 // before the arguments are bound to the properties | ||||||
|  |                 var defaultValues = command.GetArgumentValues(instance); | ||||||
|  |  | ||||||
|  |                 // Help option | ||||||
|  |                 if (command.IsHelpOptionAvailable && input.IsHelpOptionSpecified || | ||||||
|  |                     command == FallbackDefaultCommand.Schema && !input.Parameters.Any() && !input.Options.Any()) | ||||||
|  |                 { | ||||||
|  |                     _helpTextWriter.Write(root, command, defaultValues); | ||||||
|  |                     return ExitCode.Success; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // Bind arguments | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     command.Bind(instance, input, environmentVariables); | ||||||
|  |                 } | ||||||
|  |                 // This may throw exceptions which are useful only to the end-user | ||||||
|  |                 catch (CliFxException ex) | ||||||
|  |                 { | ||||||
|  |                     _console.WithForegroundColor(ConsoleColor.Red, () => | ||||||
|  |                         _console.Error.WriteLine(ex.ToString()) | ||||||
|  |                     ); | ||||||
|  |  | ||||||
|  |                     _helpTextWriter.Write(root, command, defaultValues); | ||||||
|  |  | ||||||
|  |                     return ExitCode.FromException(ex); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // Execute the command | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await instance.ExecuteAsync(_console); | ||||||
|  |                     return ExitCode.Success; | ||||||
|  |                 } | ||||||
|  |                 // Swallow command exceptions and route them to the console | ||||||
|  |                 catch (CommandException ex) | ||||||
|  |                 { | ||||||
|  |                     _console.WithForegroundColor(ConsoleColor.Red, () => | ||||||
|  |                         _console.Error.WriteLine(ex.ToString()) | ||||||
|  |                     ); | ||||||
|  |  | ||||||
|  |                     if (ex.ShowHelp) | ||||||
|  |                     { | ||||||
|  |                         _helpTextWriter.Write(root, command, defaultValues); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     return ex.ExitCode; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             catch (Exception ex) |             // To prevent the app from showing the annoying Windows troubleshooting dialog, | ||||||
|  |             // we handle all exceptions and route them to the console nicely. | ||||||
|  |             // However, we don't want to swallow unhandled exceptions when the debugger is attached, | ||||||
|  |             // because we still want the IDE to show them to the developer. | ||||||
|  |             catch (Exception ex) when (!Debugger.IsAttached) | ||||||
|             { |             { | ||||||
|                 // We want to catch exceptions in order to print errors and return correct exit codes. |                 _console.WithColors(ConsoleColor.White, ConsoleColor.DarkRed, () => | ||||||
|                 // Doing this also gets rid of the annoying Windows troubleshooting dialog that shows up on unhandled exceptions. |                     _console.Error.Write("ERROR:") | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|                 // Prefer showing message without stack trace on exceptions coming from CliFx or on CommandException |                 _console.Error.Write(" "); | ||||||
|                 var errorMessage = !string.IsNullOrWhiteSpace(ex.Message) && (ex is CliFxException || ex is CommandException) |                 _console.WriteException(ex); | ||||||
|                     ? ex.Message |  | ||||||
|                     : ex.ToString(); |  | ||||||
|  |  | ||||||
|                 _console.WithForegroundColor(ConsoleColor.Red, () => _console.Error.WriteLine(errorMessage)); |                 return ExitCode.FromException(ex); | ||||||
|  |  | ||||||
|                 return ex is CommandException commandException |  | ||||||
|                     ? commandException.ExitCode |  | ||||||
|                     : ex.HResult; |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -184,11 +224,17 @@ namespace CliFx | |||||||
|         /// Runs the application with specified command line arguments and returns the exit code. |         /// Runs the application with specified command line arguments and returns the exit code. | ||||||
|         /// Environment variables are retrieved automatically. |         /// Environment variables are retrieved automatically. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  |         /// <remarks> | ||||||
|  |         /// If a <see cref="CommandException"/> is thrown during command execution, it will be handled and routed to the console. | ||||||
|  |         /// Additionally, if the debugger is not attached (i.e. the app is running in production), all other exceptions thrown within | ||||||
|  |         /// this method will be handled and routed to the console as well. | ||||||
|  |         /// </remarks> | ||||||
|         public async ValueTask<int> RunAsync(IReadOnlyList<string> commandLineArguments) |         public async ValueTask<int> RunAsync(IReadOnlyList<string> commandLineArguments) | ||||||
|         { |         { | ||||||
|  |             // Environment variable names are case-insensitive on Windows but are case-sensitive on Linux and macOS | ||||||
|             var environmentVariables = Environment.GetEnvironmentVariables() |             var environmentVariables = Environment.GetEnvironmentVariables() | ||||||
|                 .Cast<DictionaryEntry>() |                 .Cast<DictionaryEntry>() | ||||||
|                 .ToDictionary(e => (string) e.Key, e => (string) e.Value, StringComparer.OrdinalIgnoreCase); |                 .ToDictionary(e => (string) e.Key, e => (string) e.Value, StringComparer.Ordinal); | ||||||
|  |  | ||||||
|             return await RunAsync(commandLineArguments, environmentVariables); |             return await RunAsync(commandLineArguments, environmentVariables); | ||||||
|         } |         } | ||||||
| @@ -197,6 +243,11 @@ namespace CliFx | |||||||
|         /// Runs the application and returns the exit code. |         /// Runs the application and returns the exit code. | ||||||
|         /// Command line arguments and environment variables are retrieved automatically. |         /// Command line arguments and environment variables are retrieved automatically. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  |         /// <remarks> | ||||||
|  |         /// If a <see cref="CommandException"/> is thrown during command execution, it will be handled and routed to the console. | ||||||
|  |         /// Additionally, if the debugger is not attached (i.e. the app is running in production), all other exceptions thrown within | ||||||
|  |         /// this method will be handled and routed to the console as well. | ||||||
|  |         /// </remarks> | ||||||
|         public async ValueTask<int> RunAsync() |         public async ValueTask<int> RunAsync() | ||||||
|         { |         { | ||||||
|             var commandLineArguments = Environment.GetCommandLineArgs() |             var commandLineArguments = Environment.GetCommandLineArgs() | ||||||
| @@ -206,4 +257,28 @@ namespace CliFx | |||||||
|             return await RunAsync(commandLineArguments); |             return await RunAsync(commandLineArguments); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public partial class CliApplication | ||||||
|  |     { | ||||||
|  |         private static class ExitCode | ||||||
|  |         { | ||||||
|  |             public const int Success = 0; | ||||||
|  |  | ||||||
|  |             public static int FromException(Exception ex) => | ||||||
|  |                 ex is CommandException cmdEx | ||||||
|  |                     ? cmdEx.ExitCode | ||||||
|  |                     : 1; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Fallback default command used when none is defined in the application | ||||||
|  |         [Command] | ||||||
|  |         private class FallbackDefaultCommand : ICommand | ||||||
|  |         { | ||||||
|  |             public static CommandSchema Schema { get; } = CommandSchema.TryResolve(typeof(FallbackDefaultCommand))!; | ||||||
|  |  | ||||||
|  |             // Never actually executed | ||||||
|  |             [ExcludeFromCodeCoverage] | ||||||
|  |             public ValueTask ExecuteAsync(IConsole console) => default; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -4,6 +4,7 @@ using System.IO; | |||||||
| using System.Linq; | using System.Linq; | ||||||
| using System.Reflection; | using System.Reflection; | ||||||
| using CliFx.Domain; | using CliFx.Domain; | ||||||
|  | using CliFx.Internal.Extensions; | ||||||
|  |  | ||||||
| namespace CliFx | namespace CliFx | ||||||
| { | { | ||||||
| @@ -33,6 +34,12 @@ namespace CliFx | |||||||
|             return this; |             return this; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Adds a command of specified type to the application. | ||||||
|  |         /// </summary> | ||||||
|  |         public CliApplicationBuilder AddCommand<TCommand>() where TCommand : ICommand => | ||||||
|  |             AddCommand(typeof(TCommand)); | ||||||
|  |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Adds multiple commands to the application. |         /// Adds multiple commands to the application. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @@ -158,9 +165,9 @@ namespace CliFx | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         public CliApplication Build() |         public CliApplication Build() | ||||||
|         { |         { | ||||||
|             _title ??= GetDefaultTitle() ?? "App"; |             _title ??= GetDefaultTitle(); | ||||||
|             _executableName ??= GetDefaultExecutableName() ?? "app"; |             _executableName ??= GetDefaultExecutableName(); | ||||||
|             _versionText ??= GetDefaultVersionText() ?? "v1.0"; |             _versionText ??= GetDefaultVersionText(); | ||||||
|             _console ??= new SystemConsole(); |             _console ??= new SystemConsole(); | ||||||
|             _typeActivator ??= new DefaultTypeActivator(); |             _typeActivator ??= new DefaultTypeActivator(); | ||||||
|  |  | ||||||
| @@ -173,30 +180,34 @@ namespace CliFx | |||||||
|  |  | ||||||
|     public partial class CliApplicationBuilder |     public partial class CliApplicationBuilder | ||||||
|     { |     { | ||||||
|         private static readonly Lazy<Assembly> LazyEntryAssembly = new Lazy<Assembly>(Assembly.GetEntryAssembly); |         private static readonly Lazy<Assembly?> LazyEntryAssembly = new Lazy<Assembly?>(Assembly.GetEntryAssembly); | ||||||
|  |  | ||||||
|         // Entry assembly is null in tests |         // Entry assembly is null in tests | ||||||
|         private static Assembly EntryAssembly => LazyEntryAssembly.Value; |         private static Assembly? EntryAssembly => LazyEntryAssembly.Value; | ||||||
|  |  | ||||||
|         private static string? GetDefaultTitle() => EntryAssembly?.GetName().Name; |         private static string GetDefaultTitle() => EntryAssembly?.GetName().Name?? "App"; | ||||||
|  |  | ||||||
|         private static string? GetDefaultExecutableName() |         private static string GetDefaultExecutableName() | ||||||
|         { |         { | ||||||
|             var entryAssemblyLocation = EntryAssembly?.Location; |             var entryAssemblyLocation = EntryAssembly?.Location; | ||||||
|  |  | ||||||
|             // If it's a .dll assembly, prepend 'dotnet' and keep the file extension |             // The assembly can be an executable or a dll, depending on how it was packaged | ||||||
|             if (string.Equals(Path.GetExtension(entryAssemblyLocation), ".dll", StringComparison.OrdinalIgnoreCase)) |             var isDll = string.Equals( | ||||||
|             { |                 Path.GetExtension(entryAssemblyLocation), | ||||||
|                 return "dotnet " + Path.GetFileName(entryAssemblyLocation); |                 ".dll", | ||||||
|             } |                 StringComparison.OrdinalIgnoreCase | ||||||
|  |             ); | ||||||
|  |  | ||||||
|             // Otherwise just use assembly file name without extension |             var name = isDll | ||||||
|             return Path.GetFileNameWithoutExtension(entryAssemblyLocation); |                 ? "dotnet " + Path.GetFileName(entryAssemblyLocation) | ||||||
|  |                 : Path.GetFileNameWithoutExtension(entryAssemblyLocation); | ||||||
|  |  | ||||||
|  |             return name ?? "app"; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private static string? GetDefaultVersionText() => |         private static string GetDefaultVersionText() => | ||||||
|             EntryAssembly != null |             EntryAssembly != null | ||||||
|                 ? $"v{EntryAssembly.GetName().Version}" |                 ? $"v{EntryAssembly.GetName().Version.ToSemanticString()}" | ||||||
|                 : null; |                 : "v1.0"; | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -2,7 +2,7 @@ | |||||||
|   <Import Project="../CliFx.props" /> |   <Import Project="../CliFx.props" /> | ||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <TargetFrameworks>netstandard2.1;netstandard2.0;net45</TargetFrameworks> |     <TargetFrameworks>netstandard2.1;netstandard2.0</TargetFrameworks> | ||||||
|     <Authors>$(Company)</Authors> |     <Authors>$(Company)</Authors> | ||||||
|     <Description>Declarative framework for CLI applications</Description> |     <Description>Declarative framework for CLI applications</Description> | ||||||
|     <PackageTags>command line executable interface framework parser arguments net core</PackageTags> |     <PackageTags>command line executable interface framework parser arguments net core</PackageTags> | ||||||
| @@ -10,35 +10,41 @@ | |||||||
|     <PackageReleaseNotes>https://github.com/Tyrrrz/CliFx/blob/master/Changelog.md</PackageReleaseNotes> |     <PackageReleaseNotes>https://github.com/Tyrrrz/CliFx/blob/master/Changelog.md</PackageReleaseNotes> | ||||||
|     <PackageIcon>favicon.png</PackageIcon> |     <PackageIcon>favicon.png</PackageIcon> | ||||||
|     <PackageLicenseExpression>MIT</PackageLicenseExpression> |     <PackageLicenseExpression>MIT</PackageLicenseExpression> | ||||||
|     <PackageRequireLicenseAcceptance>True</PackageRequireLicenseAcceptance> |     <GenerateDocumentationFile>true</GenerateDocumentationFile> | ||||||
|     <GenerateDocumentationFile>True</GenerateDocumentationFile> |     <PublishRepositoryUrl>true</PublishRepositoryUrl> | ||||||
|     <PublishRepositoryUrl>True</PublishRepositoryUrl> |     <EmbedUntrackedSources>true</EmbedUntrackedSources> | ||||||
|     <EmbedUntrackedSources>True</EmbedUntrackedSources> |     <IncludeSymbols>true</IncludeSymbols> | ||||||
|     <IncludeSymbols>True</IncludeSymbols> |  | ||||||
|     <SymbolPackageFormat>snupkg</SymbolPackageFormat> |     <SymbolPackageFormat>snupkg</SymbolPackageFormat> | ||||||
|  |     <TargetsForTfmSpecificContentInPackage>$(TargetsForTfmSpecificContentInPackage);CopyAnalyzerToPackage</TargetsForTfmSpecificContentInPackage> | ||||||
|  |   </PropertyGroup> | ||||||
|  |  | ||||||
|  |   <!-- Disable nullability warnings on older frameworks because there is no nullability info for BCL --> | ||||||
|  |   <PropertyGroup Condition="'$(TargetFramework)' == 'netstandard2.0'"> | ||||||
|  |     <Nullable>annotations</Nullable> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleToAttribute"> |     <None Include="../favicon.png" Pack="true" PackagePath="" /> | ||||||
|       <_Parameter1>$(AssemblyName).Tests</_Parameter1> |  | ||||||
|     </AssemblyAttribute> |  | ||||||
|     <AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleToAttribute"> |  | ||||||
|       <_Parameter1>$(AssemblyName).Analyzers</_Parameter1> |  | ||||||
|     </AssemblyAttribute> |  | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <None Include="../favicon.png" Pack="True" PackagePath="" /> |  | ||||||
|   </ItemGroup> |  | ||||||
|  |  | ||||||
|   <ItemGroup> |  | ||||||
|     <PackageReference Include="Microsoft.NETFramework.ReferenceAssemblies" Version="1.0.0" PrivateAssets="all" /> |  | ||||||
|     <PackageReference Include="Microsoft.SourceLink.GitHub" Version="1.0.0" PrivateAssets="all" /> |     <PackageReference Include="Microsoft.SourceLink.GitHub" Version="1.0.0" PrivateAssets="all" /> | ||||||
|     <PackageReference Include="Nullable" Version="1.2.1" PrivateAssets="all" /> |     <PackageReference Include="Nullable" Version="1.3.0" PrivateAssets="all" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup Condition="'$(TargetFramework)' == 'netstandard2.0' or '$(TargetFramework)' == 'net45'"> |   <ItemGroup Condition="'$(TargetFramework)' == 'netstandard2.0'"> | ||||||
|     <PackageReference Include="System.Threading.Tasks.Extensions" Version="4.5.3" /> |     <PackageReference Include="System.Threading.Tasks.Extensions" Version="4.5.4" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|  |   <!-- The following item group and target ensure that the analyzer project is copied into the output NuGet package --> | ||||||
|  |   <ItemGroup> | ||||||
|  |     <ProjectReference Include="..\CliFx.Analyzers\CliFx.Analyzers.csproj" ReferenceOutputAssembly="true" IncludeAssets="CliFx.Analyzers.dll" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |  | ||||||
|  |   <Target Name="CopyAnalyzerToPackage"> | ||||||
|  |     <ItemGroup> | ||||||
|  |       <TfmSpecificPackageFile Include="$(OutDir)/CliFx.Analyzers.dll" PackagePath="analyzers/dotnet/cs" BuildAction="none" /> | ||||||
|  |     </ItemGroup> | ||||||
|  |   </Target> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| using System; | using System; | ||||||
| using System.Text; |  | ||||||
| using CliFx.Exceptions; | using CliFx.Exceptions; | ||||||
|  | using CliFx.Internal.Extensions; | ||||||
|  |  | ||||||
| namespace CliFx | namespace CliFx | ||||||
| { | { | ||||||
| @@ -14,15 +14,11 @@ namespace CliFx | |||||||
|         { |         { | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 return Activator.CreateInstance(type); |                 return type.CreateInstance(); | ||||||
|             } |             } | ||||||
|             catch (Exception ex) |             catch (Exception ex) | ||||||
|             { |             { | ||||||
|                 throw new CliFxException(new StringBuilder() |                 throw CliFxException.DefaultActivatorFailed(type, ex); | ||||||
|                     .Append($"Failed to create an instance of {type.FullName}.").Append(" ") |  | ||||||
|                     .AppendLine("The type must have a public parameter-less constructor in order to be instantiated by the default activator.") |  | ||||||
|                     .Append($"To supply a custom activator (for example when using dependency injection), call {nameof(CliApplicationBuilder)}.{nameof(CliApplicationBuilder.UseTypeActivator)}(...).") |  | ||||||
|                     .ToString(), ex); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| using System; | using System; | ||||||
| using System.Text; |  | ||||||
| using CliFx.Exceptions; | using CliFx.Exceptions; | ||||||
|  |  | ||||||
| namespace CliFx | namespace CliFx | ||||||
| @@ -18,10 +17,6 @@ namespace CliFx | |||||||
|  |  | ||||||
|         /// <inheritdoc /> |         /// <inheritdoc /> | ||||||
|         public object CreateInstance(Type type) => |         public object CreateInstance(Type type) => | ||||||
|             _func(type) ?? throw new CliFxException(new StringBuilder() |             _func(type) ?? throw CliFxException.DelegateActivatorReturnedNull(type); | ||||||
|                 .Append($"Failed to create an instance of type {type.FullName}, received <null> instead.").Append(" ") |  | ||||||
|                 .Append("Make sure that the provided type activator was configured correctly.").Append(" ") |  | ||||||
|                 .Append("If you are using a dependency container, make sure that this type is registered.") |  | ||||||
|                 .ToString()); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user