mirror of
				https://github.com/actions/setup-dotnet.git
				synced 2025-10-31 02:53:46 +00:00 
			
		
		
		
	Add and configure ESLint and update configuration for Prettier (#391)
* Apply ESLint config and update Prettier * Update dependencies and rebuild * Update docs
This commit is contained in:
		
							parent
							
								
									926f442022
								
							
						
					
					
						commit
						c82240598b
					
				
							
								
								
									
										6
									
								
								.eslintignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.eslintignore
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | ||||
| # Ignore list | ||||
| /* | ||||
| 
 | ||||
| # Do not ignore these folders: | ||||
| !__tests__/ | ||||
| !src/ | ||||
							
								
								
									
										49
									
								
								.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,49 @@ | ||||
| module.exports = { | ||||
|   extends: [ | ||||
|     'eslint:recommended', | ||||
|     'plugin:@typescript-eslint/recommended', | ||||
|     'plugin:eslint-plugin-jest/recommended', | ||||
|     'eslint-config-prettier' | ||||
|   ], | ||||
|   parser: '@typescript-eslint/parser', | ||||
|   plugins: ['@typescript-eslint', 'eslint-plugin-jest'], | ||||
|   rules: { | ||||
|     '@typescript-eslint/no-require-imports': 'error', | ||||
|     '@typescript-eslint/no-non-null-assertion': 'off', | ||||
|     '@typescript-eslint/no-explicit-any': 'off', | ||||
|     '@typescript-eslint/no-empty-function': 'off', | ||||
|     '@typescript-eslint/ban-ts-comment': [ | ||||
|       'error', | ||||
|       { | ||||
|         'ts-ignore': 'allow-with-description' | ||||
|       } | ||||
|     ], | ||||
|     'no-console': 'error', | ||||
|     'yoda': 'error', | ||||
|     'prefer-const': [ | ||||
|       'error', | ||||
|       { | ||||
|         destructuring: 'all' | ||||
|       } | ||||
|     ], | ||||
|     'no-control-regex': 'off', | ||||
|     'no-constant-condition': ['error', {checkLoops: false}] | ||||
|   }, | ||||
|   overrides: [ | ||||
|     { | ||||
|       files: ['**/*{test,spec}.ts'], | ||||
|       rules: { | ||||
|         '@typescript-eslint/no-unused-vars': 'off', | ||||
|         'jest/no-standalone-expect': 'off', | ||||
|         'jest/no-conditional-expect': 'off', | ||||
|         'no-console': 'off', | ||||
| 
 | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   env: { | ||||
|     node: true, | ||||
|     es6: true, | ||||
|     'jest/globals': true | ||||
|   } | ||||
| }; | ||||
							
								
								
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @ -1 +1,2 @@ | ||||
| * text=auto eol=lf | ||||
| .licenses/** -diff linguist-generated=true | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/basic-validation.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/basic-validation.yml
									
									
									
									
										vendored
									
									
								
							| @ -14,4 +14,4 @@ on: | ||||
| jobs: | ||||
|   call-basic-validation: | ||||
|     name: Basic validation | ||||
|     uses: actions/reusable-workflows/.github/workflows/basic-validation.yml@main | ||||
|     uses: actions/reusable-workflows/.github/workflows/basic-validation.yml@main | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/check-dist.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/check-dist.yml
									
									
									
									
										vendored
									
									
								
							| @ -14,4 +14,4 @@ on: | ||||
| jobs: | ||||
|   call-check-dist: | ||||
|     name: Check dist/ | ||||
|     uses: actions/reusable-workflows/.github/workflows/check-dist.yml@main | ||||
|     uses: actions/reusable-workflows/.github/workflows/check-dist.yml@main | ||||
|  | ||||
							
								
								
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @ -2,13 +2,13 @@ name: CodeQL analysis | ||||
| 
 | ||||
| on: | ||||
|   push: | ||||
|     branches: [ main ] | ||||
|     branches: [main] | ||||
|   pull_request: | ||||
|     branches: [ main ] | ||||
|     branches: [main] | ||||
|   schedule: | ||||
|     - cron: '0 3 * * 0' | ||||
| 
 | ||||
| jobs: | ||||
|   call-codeQL-analysis: | ||||
|     name: CodeQL analysis  | ||||
|     uses: actions/reusable-workflows/.github/workflows/codeql-analysis.yml@main | ||||
|     name: CodeQL analysis | ||||
|     uses: actions/reusable-workflows/.github/workflows/codeql-analysis.yml@main | ||||
|  | ||||
							
								
								
									
										148
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										148
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							| @ -192,83 +192,83 @@ jobs: | ||||
|         run: __tests__/verify-dotnet.ps1 2.2 3.1 | ||||
| 
 | ||||
|   test-setup-with-dotnet-quality: | ||||
|       runs-on: ${{ matrix.operating-system }} | ||||
|       strategy: | ||||
|         fail-fast: false | ||||
|         matrix: | ||||
|           operating-system: [ubuntu-latest, windows-latest, macOS-latest] | ||||
|       steps: | ||||
|         - name: Checkout | ||||
|           uses: actions/checkout@v3 | ||||
|         - name: Clear toolcache | ||||
|           shell: pwsh | ||||
|           run: __tests__/clear-toolcache.ps1 ${{ runner.os }} | ||||
|                 | ||||
|         - name: Setup dotnet 7.0 with preview quality | ||||
|           uses: ./ | ||||
|           with: | ||||
|             dotnet-version: "7.0" | ||||
|             dotnet-quality: "preview" | ||||
|         - name: Verify preview version | ||||
|           shell: pwsh | ||||
|           run: | | ||||
|             $version = & dotnet --version | ||||
|             Write-Host "Installed version: $version" | ||||
|             if (-not ($version.Contains("preview") -or $version.Contains("rc"))) { throw "Unexpected version" } | ||||
|    | ||||
|     runs-on: ${{ matrix.operating-system }} | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         operating-system: [ubuntu-latest, windows-latest, macOS-latest] | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Clear toolcache | ||||
|         shell: pwsh | ||||
|         run: __tests__/clear-toolcache.ps1 ${{ runner.os }} | ||||
| 
 | ||||
|       - name: Setup dotnet 7.0 with preview quality | ||||
|         uses: ./ | ||||
|         with: | ||||
|           dotnet-version: '7.0' | ||||
|           dotnet-quality: 'preview' | ||||
|       - name: Verify preview version | ||||
|         shell: pwsh | ||||
|         run: | | ||||
|           $version = & dotnet --version | ||||
|           Write-Host "Installed version: $version" | ||||
|           if (-not ($version.Contains("preview") -or $version.Contains("rc"))) { throw "Unexpected version" } | ||||
| 
 | ||||
|   test-dotnet-version-output-during-single-version-installation: | ||||
|       runs-on: ${{ matrix.operating-system }} | ||||
|       strategy: | ||||
|         fail-fast: false | ||||
|         matrix: | ||||
|           operating-system: [ubuntu-latest, windows-latest, macOS-latest] | ||||
|       steps: | ||||
|         - name: Checkout | ||||
|           uses: actions/checkout@v3 | ||||
|         - name: Clear toolcache | ||||
|           shell: pwsh | ||||
|           run: __tests__/clear-toolcache.ps1 ${{ runner.os }} | ||||
|                 | ||||
|         - name: Setup dotnet 6.0.401 | ||||
|           uses: ./ | ||||
|           id: step1 | ||||
|           with: | ||||
|             dotnet-version: "6.0.401" | ||||
|     runs-on: ${{ matrix.operating-system }} | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         operating-system: [ubuntu-latest, windows-latest, macOS-latest] | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Clear toolcache | ||||
|         shell: pwsh | ||||
|         run: __tests__/clear-toolcache.ps1 ${{ runner.os }} | ||||
| 
 | ||||
|       - name: Setup dotnet 6.0.401 | ||||
|         uses: ./ | ||||
|         id: step1 | ||||
|         with: | ||||
|           dotnet-version: '6.0.401' | ||||
| 
 | ||||
|       - name: Verify value of the dotnet-version output | ||||
|         shell: pwsh | ||||
|         run: | | ||||
|           $version = & dotnet --version | ||||
|           Write-Host "Installed version: $version" | ||||
|           if (-not ($version -eq '${{steps.step1.outputs.dotnet-version}}')) { throw "Unexpected output value" } | ||||
| 
 | ||||
|         - name: Verify value of the dotnet-version output | ||||
|           shell: pwsh | ||||
|           run: | | ||||
|             $version = & dotnet --version | ||||
|             Write-Host "Installed version: $version" | ||||
|             if (-not ($version -eq '${{steps.step1.outputs.dotnet-version}}')) { throw "Unexpected output value" } | ||||
|    | ||||
|   test-dotnet-version-output-during-multiple-version-installation: | ||||
|       runs-on: ${{ matrix.operating-system }} | ||||
|       strategy: | ||||
|         fail-fast: false | ||||
|         matrix: | ||||
|           operating-system: [ubuntu-latest, windows-latest, macOS-latest] | ||||
|       steps: | ||||
|         - name: Checkout | ||||
|           uses: actions/checkout@v3 | ||||
|         - name: Clear toolcache | ||||
|           shell: pwsh | ||||
|           run: __tests__/clear-toolcache.ps1 ${{ runner.os }} | ||||
|                 | ||||
|         - name: Setup dotnet 6.0.401, 5.0.408, 7.0.100-rc.1.22431.12 | ||||
|           uses: ./ | ||||
|           id: step2 | ||||
|           with: | ||||
|             dotnet-version: | | ||||
|               7.0.100-rc.1.22431.12 | ||||
|               6.0.401 | ||||
|               5.0.408 | ||||
|     runs-on: ${{ matrix.operating-system }} | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         operating-system: [ubuntu-latest, windows-latest, macOS-latest] | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Clear toolcache | ||||
|         shell: pwsh | ||||
|         run: __tests__/clear-toolcache.ps1 ${{ runner.os }} | ||||
| 
 | ||||
|         - name: Verify value of the dotnet-version output | ||||
|           shell: pwsh | ||||
|           run: | | ||||
|             $version = "7.0.100-rc.1.22431.12" | ||||
|             if (-not ($version -eq '${{steps.step2.outputs.dotnet-version}}')) { throw "Unexpected output value" } | ||||
|       - name: Setup dotnet 6.0.401, 5.0.408, 7.0.100-rc.1.22431.12 | ||||
|         uses: ./ | ||||
|         id: step2 | ||||
|         with: | ||||
|           dotnet-version: | | ||||
|             7.0.100-rc.1.22431.12 | ||||
|             6.0.401 | ||||
|             5.0.408 | ||||
| 
 | ||||
|       - name: Verify value of the dotnet-version output | ||||
|         shell: pwsh | ||||
|         run: | | ||||
|           $version = "7.0.100-rc.1.22431.12" | ||||
|           if (-not ($version -eq '${{steps.step2.outputs.dotnet-version}}')) { throw "Unexpected output value" } | ||||
| 
 | ||||
|   test-proxy: | ||||
|     runs-on: ubuntu-latest | ||||
| @ -320,4 +320,4 @@ jobs: | ||||
|         env: | ||||
|           NUGET_AUTH_TOKEN: NOTATOKEN | ||||
|       - name: Verify dotnet | ||||
|         run: __tests__/verify-dotnet.sh 3.1.201 | ||||
|         run: __tests__/verify-dotnet.sh 3.1.201 | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/licensed.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/licensed.yml
									
									
									
									
										vendored
									
									
								
							| @ -12,4 +12,4 @@ on: | ||||
| jobs: | ||||
|   call-licensed: | ||||
|     name: Licensed | ||||
|     uses: actions/reusable-workflows/.github/workflows/licensed.yml@main | ||||
|     uses: actions/reusable-workflows/.github/workflows/licensed.yml@main | ||||
|  | ||||
							
								
								
									
										12
									
								
								.github/workflows/release-new-action-version.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/release-new-action-version.yml
									
									
									
									
										vendored
									
									
								
							| @ -21,9 +21,9 @@ jobs: | ||||
|       name: releaseNewActionVersion | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: Update the ${{ env.TAG_NAME }} tag | ||||
|       id: update-major-tag | ||||
|       uses: actions/publish-action@v0.2.2 | ||||
|       with: | ||||
|         source-tag: ${{ env.TAG_NAME }} | ||||
|         slack-webhook: ${{ secrets.SLACK_WEBHOOK }} | ||||
|       - name: Update the ${{ env.TAG_NAME }} tag | ||||
|         id: update-major-tag | ||||
|         uses: actions/publish-action@v0.2.2 | ||||
|         with: | ||||
|           source-tag: ${{ env.TAG_NAME }} | ||||
|           slack-webhook: ${{ secrets.SLACK_WEBHOOK }} | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/test-dotnet.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/test-dotnet.yml
									
									
									
									
										vendored
									
									
								
							| @ -34,4 +34,4 @@ jobs: | ||||
|         run: | | ||||
|           $version = & dotnet --version | ||||
|           Write-Host "Installed version: $version" | ||||
|           if (-not $version.StartsWith("${{ matrix.dotnet-version }}")) { throw "Unexpected version" } | ||||
|           if (-not $version.StartsWith("${{ matrix.dotnet-version }}")) { throw "Unexpected version" } | ||||
|  | ||||
| @ -2,3 +2,4 @@ | ||||
| . "$(dirname -- "$0")/_/husky.sh" | ||||
| 
 | ||||
| npm run format | ||||
| npm run lint:fix | ||||
|  | ||||
| @ -3,3 +3,4 @@ | ||||
| 
 | ||||
| # Tests are not run at push time since they can take 2-4 minutes to complete | ||||
| npm run format-check | ||||
| npm run lint | ||||
|  | ||||
							
								
								
									
										7
									
								
								.prettierignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.prettierignore
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | ||||
| # Ignore list | ||||
| /* | ||||
| 
 | ||||
| # Do not ignore these folders: | ||||
| !__tests__/ | ||||
| !.github/ | ||||
| !src/ | ||||
							
								
								
									
										10
									
								
								.prettierrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.prettierrc.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| module.exports = { | ||||
|   printWidth: 80, | ||||
|   tabWidth: 2, | ||||
|   useTabs: false, | ||||
|   semi: true, | ||||
|   singleQuote: true, | ||||
|   trailingComma: 'none', | ||||
|   bracketSpacing: false, | ||||
|   arrowParens: 'avoid' | ||||
| }; | ||||
| @ -1,12 +0,0 @@ | ||||
| { | ||||
|     "printWidth": 80, | ||||
|     "tabWidth": 2, | ||||
|     "useTabs": false, | ||||
|     "semi": true, | ||||
|     "singleQuote": true, | ||||
|     "trailingComma": "none", | ||||
|     "bracketSpacing": false, | ||||
|     "arrowParens": "avoid", | ||||
|     "parser": "typescript", | ||||
|     "endOfLine": "auto" | ||||
|   } | ||||
| @ -1,336 +1,336 @@ | ||||
| import * as io from '@actions/io'; | ||||
| import fs from 'fs'; | ||||
| import path from 'path'; | ||||
| 
 | ||||
| const fakeSourcesDirForTesting = path.join( | ||||
|   __dirname, | ||||
|   'runner', | ||||
|   path.join(Math.random().toString(36).substring(7)), | ||||
|   's' | ||||
| ); | ||||
| 
 | ||||
| const invalidNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>`; | ||||
| 
 | ||||
| const emptyNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const nugetorgNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const gprnugetorgNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|     <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const gprNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const twogprNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR-GitHub" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|     <add key="GPR-Actions" value="https://nuget.pkg.github.com/actions/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const spaceNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR GitHub" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const azureartifactsNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="AzureArtifacts" value="https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const azureartifactsnugetorgNuGetConfig: string = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="AzureArtifacts" value="https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json" protocolVersion="3" /> | ||||
|     <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| // We want a NuGet.config one level above the sources directory, so it doesn't trample a user's NuGet.config but is still picked up by NuGet/dotnet.
 | ||||
| const nugetConfigFile = path.join(fakeSourcesDirForTesting, '../nuget.config'); | ||||
| 
 | ||||
| process.env['GITHUB_REPOSITORY'] = 'OwnerName/repo'; | ||||
| import * as auth from '../src/authutil'; | ||||
| 
 | ||||
| describe('authutil tests', () => { | ||||
|   beforeEach(async () => { | ||||
|     await io.rmRF(fakeSourcesDirForTesting); | ||||
|     await io.mkdirP(fakeSourcesDirForTesting); | ||||
|   }, 30000); | ||||
| 
 | ||||
|   afterAll(async () => { | ||||
|     await io.rmRF(fakeSourcesDirForTesting); | ||||
|   }, 30000); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     if (fs.existsSync(nugetConfigFile)) { | ||||
|       fs.unlinkSync(nugetConfigFile); | ||||
|     } | ||||
|     process.env['INPUT_OWNER'] = ''; | ||||
|     process.env['NUGET_AUTH_TOKEN'] = ''; | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, sets up a full NuGet.config with URL and user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, auth token environment variable not provided, throws', async () => { | ||||
|     let thrown = false; | ||||
|     try { | ||||
|       await auth.configAuthentication( | ||||
|         'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|         '', | ||||
|         fakeSourcesDirForTesting | ||||
|       ); | ||||
|     } catch { | ||||
|       thrown = true; | ||||
|     } | ||||
|     expect(thrown).toBe(true); | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, sets up a full NuGet.config with URL and other owner/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     process.env['INPUT_OWNER'] = 'otherorg'; | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/otherorg/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config (invalid), tries to parse an invalid NuGet.config and throws', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, invalidNuGetConfig); | ||||
|     let thrown = false; | ||||
|     try { | ||||
|       await auth.configAuthentication( | ||||
|         'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|         '', | ||||
|         fakeSourcesDirForTesting | ||||
|       ); | ||||
|     } catch { | ||||
|       thrown = true; | ||||
|     } | ||||
|     expect(thrown).toBe(true); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ no sources, sets up a full NuGet.config with URL and user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, emptyNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ no GPR sources, sets up a full NuGet.config with URL and user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, nugetorgNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ only GPR source, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, gprNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ GPR source and NuGet.org, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, gprnugetorgNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ two GPR sources, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, twogprNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ spaces in key, throws for now', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, spaceNuGetConfig); | ||||
|     let thrown = false; | ||||
|     try { | ||||
|       await auth.configAuthentication( | ||||
|         'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|         '', | ||||
|         fakeSourcesDirForTesting | ||||
|       ); | ||||
|     } catch { | ||||
|       thrown = true; | ||||
|     } | ||||
|     expect(thrown).toBe(true); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config not in repo root, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigDirectory: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'subfolder' | ||||
|     ); | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       inputNuGetConfigDirectory, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.mkdirSync(inputNuGetConfigDirectory, {recursive: true}); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, gprNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       'subfolder/nuget.config', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ only Azure Artifacts source, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, azureartifactsNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ Azure Artifacts source and NuGet.org, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, azureartifactsnugetorgNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, sets up a full NuGet.config with URL and token for other source', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     await auth.configAuthentication( | ||||
|       'https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| }); | ||||
| import * as io from '@actions/io'; | ||||
| import fs from 'fs'; | ||||
| import path from 'path'; | ||||
| 
 | ||||
| const fakeSourcesDirForTesting = path.join( | ||||
|   __dirname, | ||||
|   'runner', | ||||
|   path.join(Math.random().toString(36).substring(7)), | ||||
|   's' | ||||
| ); | ||||
| 
 | ||||
| const invalidNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>`; | ||||
| 
 | ||||
| const emptyNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const nugetorgNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const gprnugetorgNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|     <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const gprNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const twogprNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR-GitHub" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|     <add key="GPR-Actions" value="https://nuget.pkg.github.com/actions/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const spaceNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="GPR GitHub" value="https://nuget.pkg.github.com/OwnerName/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const azureartifactsNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="AzureArtifacts" value="https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| const azureartifactsnugetorgNuGetConfig = `<?xml version="1.0" encoding="utf-8"?>
 | ||||
| <configuration> | ||||
|   <packageSources> | ||||
|     <add key="AzureArtifacts" value="https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json" protocolVersion="3" /> | ||||
|     <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" /> | ||||
|   </packageSources> | ||||
| </configuration>`;
 | ||||
| 
 | ||||
| // We want a NuGet.config one level above the sources directory, so it doesn't trample a user's NuGet.config but is still picked up by NuGet/dotnet.
 | ||||
| const nugetConfigFile = path.join(fakeSourcesDirForTesting, '../nuget.config'); | ||||
| 
 | ||||
| process.env['GITHUB_REPOSITORY'] = 'OwnerName/repo'; | ||||
| import * as auth from '../src/authutil'; | ||||
| 
 | ||||
| describe('authutil tests', () => { | ||||
|   beforeEach(async () => { | ||||
|     await io.rmRF(fakeSourcesDirForTesting); | ||||
|     await io.mkdirP(fakeSourcesDirForTesting); | ||||
|   }, 30000); | ||||
| 
 | ||||
|   afterAll(async () => { | ||||
|     await io.rmRF(fakeSourcesDirForTesting); | ||||
|   }, 30000); | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     if (fs.existsSync(nugetConfigFile)) { | ||||
|       fs.unlinkSync(nugetConfigFile); | ||||
|     } | ||||
|     process.env['INPUT_OWNER'] = ''; | ||||
|     process.env['NUGET_AUTH_TOKEN'] = ''; | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, sets up a full NuGet.config with URL and user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, auth token environment variable not provided, throws', async () => { | ||||
|     let thrown = false; | ||||
|     try { | ||||
|       await auth.configAuthentication( | ||||
|         'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|         '', | ||||
|         fakeSourcesDirForTesting | ||||
|       ); | ||||
|     } catch { | ||||
|       thrown = true; | ||||
|     } | ||||
|     expect(thrown).toBe(true); | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, sets up a full NuGet.config with URL and other owner/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     process.env['INPUT_OWNER'] = 'otherorg'; | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/otherorg/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config (invalid), tries to parse an invalid NuGet.config and throws', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, invalidNuGetConfig); | ||||
|     let thrown = false; | ||||
|     try { | ||||
|       await auth.configAuthentication( | ||||
|         'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|         '', | ||||
|         fakeSourcesDirForTesting | ||||
|       ); | ||||
|     } catch { | ||||
|       thrown = true; | ||||
|     } | ||||
|     expect(thrown).toBe(true); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ no sources, sets up a full NuGet.config with URL and user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, emptyNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ no GPR sources, sets up a full NuGet.config with URL and user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, nugetorgNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ only GPR source, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, gprNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ GPR source and NuGet.org, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, gprnugetorgNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ two GPR sources, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, twogprNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ spaces in key, throws for now', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, spaceNuGetConfig); | ||||
|     let thrown = false; | ||||
|     try { | ||||
|       await auth.configAuthentication( | ||||
|         'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|         '', | ||||
|         fakeSourcesDirForTesting | ||||
|       ); | ||||
|     } catch { | ||||
|       thrown = true; | ||||
|     } | ||||
|     expect(thrown).toBe(true); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config not in repo root, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigDirectory: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'subfolder' | ||||
|     ); | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       inputNuGetConfigDirectory, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.mkdirSync(inputNuGetConfigDirectory, {recursive: true}); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, gprNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://nuget.pkg.github.com/OwnerName/index.json', | ||||
|       'subfolder/nuget.config', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ only Azure Artifacts source, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, azureartifactsNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('Existing config w/ Azure Artifacts source and NuGet.org, sets up a partial NuGet.config user/PAT for GPR', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     const inputNuGetConfigPath: string = path.join( | ||||
|       fakeSourcesDirForTesting, | ||||
|       'nuget.config' | ||||
|     ); | ||||
|     fs.writeFileSync(inputNuGetConfigPath, azureartifactsnugetorgNuGetConfig); | ||||
|     await auth.configAuthentication( | ||||
|       'https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| 
 | ||||
|   it('No existing config, sets up a full NuGet.config with URL and token for other source', async () => { | ||||
|     process.env['NUGET_AUTH_TOKEN'] = 'TEST_FAKE_AUTH_TOKEN'; | ||||
|     await auth.configAuthentication( | ||||
|       'https://pkgs.dev.azure.com/amullans/_packaging/GitHubBuilds/nuget/v3/index.json', | ||||
|       '', | ||||
|       fakeSourcesDirForTesting | ||||
|     ); | ||||
|     expect(fs.existsSync(nugetConfigFile)).toBe(true); | ||||
|     expect( | ||||
|       fs.readFileSync(nugetConfigFile, {encoding: 'utf8'}) | ||||
|     ).toMatchSnapshot(); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
| @ -1,19 +1,19 @@ | ||||
| import cscFile from '../.github/csc.json'; | ||||
| describe('csc tests', () => { | ||||
|   it('Valid regular expression', async () => { | ||||
|     var cscFile = require('../.github/csc.json'); | ||||
|     var regex = cscFile['problemMatcher'][0]['pattern'][0]['regexp']; | ||||
|     const regex = cscFile['problemMatcher'][0]['pattern'][0]['regexp']; | ||||
| 
 | ||||
|     console.log(regex); | ||||
|     var re = new RegExp(regex); | ||||
|     const re = new RegExp(regex); | ||||
| 
 | ||||
|     // Ideally we would verify that this
 | ||||
|     var stringsToMatch = [ | ||||
|     const stringsToMatch = [ | ||||
|       'Program.cs(10,79): error CS1002: ; expected [/Users/zacharyeisinger/Documents/repo/setup-dotnet/__tests__/sample-broken-csproj/sample.csproj]', | ||||
|       "S:\\Msbuild\\src\\Build\\Evaluation\\ExpressionShredder.cs(33,7): error CS1003: Syntax error, ',' expected [S:\\msbuild\\src\\Build\\Microsoft.Build.csproj > Properties:prop]" | ||||
|     ]; | ||||
| 
 | ||||
|     stringsToMatch.forEach(string => { | ||||
|       var matchStr = string.match(re); | ||||
|       const matchStr = string.match(re); | ||||
|       console.log(matchStr); | ||||
|       expect(matchStr).toEqual(expect.anything()); | ||||
|     }); | ||||
|  | ||||
| @ -68,8 +68,8 @@ describe('DotnetCoreInstaller tests', () => { | ||||
|       expect(fs.existsSync(path.join(toolDir, 'dotnet'))).toBe(true); | ||||
|     } | ||||
| 
 | ||||
|     expect(process.env.DOTNET_ROOT).toBeDefined; | ||||
|     expect(process.env.PATH).toBeDefined; | ||||
|     expect(process.env.DOTNET_ROOT).toBeDefined(); | ||||
|     expect(process.env.PATH).toBeDefined(); | ||||
|     expect(process.env.DOTNET_ROOT).toBe(toolDir); | ||||
|     expect(process.env.PATH?.startsWith(toolDir)).toBe(true); | ||||
|   }, 600000); | ||||
| @ -83,15 +83,15 @@ describe('DotnetCoreInstaller tests', () => { | ||||
|       expect(fs.existsSync(path.join(toolDir, 'dotnet'))).toBe(true); | ||||
|     } | ||||
| 
 | ||||
|     expect(process.env.DOTNET_ROOT).toBeDefined; | ||||
|     expect(process.env.PATH).toBeDefined; | ||||
|     expect(process.env.DOTNET_ROOT).toBeDefined(); | ||||
|     expect(process.env.PATH).toBeDefined(); | ||||
|     expect(process.env.DOTNET_ROOT).toBe(toolDir); | ||||
|     expect(process.env.PATH?.startsWith(toolDir)).toBe(true); | ||||
|   }, 600000); //This needs some time to download on "slower" internet connections
 | ||||
| 
 | ||||
|   it('Acquires generic version of dotnet if no matching version is installed', async () => { | ||||
|     await getDotnet('3.1'); | ||||
|     var directory = fs | ||||
|     const directory = fs | ||||
|       .readdirSync(path.join(toolDir, 'sdk')) | ||||
|       .filter(fn => fn.startsWith('3.1.')); | ||||
|     expect(directory.length > 0).toBe(true); | ||||
| @ -101,17 +101,16 @@ describe('DotnetCoreInstaller tests', () => { | ||||
|       expect(fs.existsSync(path.join(toolDir, 'dotnet'))).toBe(true); | ||||
|     } | ||||
| 
 | ||||
|     expect(process.env.DOTNET_ROOT).toBeDefined; | ||||
|     expect(process.env.PATH).toBeDefined; | ||||
|     expect(process.env.DOTNET_ROOT).toBeDefined(); | ||||
|     expect(process.env.PATH).toBeDefined(); | ||||
|     expect(process.env.DOTNET_ROOT).toBe(toolDir); | ||||
|     expect(process.env.PATH?.startsWith(toolDir)).toBe(true); | ||||
|   }, 600000); //This needs some time to download on "slower" internet connections
 | ||||
| 
 | ||||
|   it('Returns string with installed SDK version', async () => { | ||||
|     const version = '3.1.120'; | ||||
|     let installedVersion: string; | ||||
| 
 | ||||
|     installedVersion = await getDotnet(version); | ||||
|     const installedVersion = await getDotnet(version); | ||||
| 
 | ||||
|     expect(installedVersion).toBe('3.1.120'); | ||||
|   }, 600000); | ||||
| @ -143,7 +142,7 @@ describe('DotnetCoreInstaller tests', () => { | ||||
|   }, 30000); | ||||
| 
 | ||||
|   it('Uses an up to date powershell download script', async () => { | ||||
|     var httpCallbackClient = new hc.HttpClient('setup-dotnet-test', [], { | ||||
|     const httpCallbackClient = new hc.HttpClient('setup-dotnet-test', [], { | ||||
|       allowRetries: true, | ||||
|       maxRetries: 3 | ||||
|     }); | ||||
| @ -180,7 +179,7 @@ describe('DotnetVersionResolver tests', () => { | ||||
|       ); | ||||
|       const versionObject = await dotnetVersionResolver.createDotNetVersion(); | ||||
| 
 | ||||
|       expect(!!versionObject.value).toBeTruthy; | ||||
|       expect(!!versionObject.value).toBe(true); | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
| @ -223,15 +222,27 @@ describe('DotnetVersionResolver tests', () => { | ||||
|   ); | ||||
| 
 | ||||
|   each(['3.1', '3.1.x', '3.1.*', '3.1.X']).test( | ||||
|     "if version: '%s' that can be resolved to 'channel' option is supplied, it should set quality flag to 'true' and type to 'channel' in version object", | ||||
|     "if version: '%s' that can be resolved to 'channel' option is supplied, it should set type to 'channel' in version object", | ||||
|     async version => { | ||||
|       const dotnetVersionResolver = new installer.DotnetVersionResolver( | ||||
|         version | ||||
|       ); | ||||
|       const versionObject = await dotnetVersionResolver.createDotNetVersion(); | ||||
| 
 | ||||
|       expect(versionObject.type.toLowerCase().includes('channel')).toBeTruthy; | ||||
|       expect(versionObject.qualityFlag).toBeTruthy; | ||||
|       expect(versionObject.type.toLowerCase().includes('channel')).toBe(true); | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
|   each(['6.0', '6.0.x', '6.0.*', '6.0.X']).test( | ||||
|     "if version: '%s' that can be resolved to 'channel' option is supplied and its major tag is >= 6, it should set type to 'channel' and qualityFlag to 'true' in version object", | ||||
|     async version => { | ||||
|       const dotnetVersionResolver = new installer.DotnetVersionResolver( | ||||
|         version | ||||
|       ); | ||||
|       const versionObject = await dotnetVersionResolver.createDotNetVersion(); | ||||
| 
 | ||||
|       expect(versionObject.type.toLowerCase().includes('channel')).toBe(true); | ||||
|       expect(versionObject.qualityFlag).toBe(true); | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
| @ -243,8 +254,8 @@ describe('DotnetVersionResolver tests', () => { | ||||
|       ); | ||||
|       const versionObject = await dotnetVersionResolver.createDotNetVersion(); | ||||
| 
 | ||||
|       expect(versionObject.type.toLowerCase().includes('version')).toBeTruthy; | ||||
|       expect(versionObject.qualityFlag).toBeFalsy; | ||||
|       expect(versionObject.type.toLowerCase().includes('version')).toBe(true); | ||||
|       expect(versionObject.qualityFlag).toBe(false); | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
| @ -259,11 +270,11 @@ describe('DotnetVersionResolver tests', () => { | ||||
|       const nonWindowsRegEx = new RegExp(/^--[vc]/); | ||||
| 
 | ||||
|       if (IS_WINDOWS) { | ||||
|         expect(windowsRegEx.test(versionObject.type)).toBeTruthy; | ||||
|         expect(nonWindowsRegEx.test(versionObject.type)).toBeFalsy; | ||||
|         expect(windowsRegEx.test(versionObject.type)).toBe(true); | ||||
|         expect(nonWindowsRegEx.test(versionObject.type)).toBe(false); | ||||
|       } else { | ||||
|         expect(nonWindowsRegEx.test(versionObject.type)).toBeTruthy; | ||||
|         expect(windowsRegEx.test(versionObject.type)).toBeFalsy; | ||||
|         expect(nonWindowsRegEx.test(versionObject.type)).toBe(true); | ||||
|         expect(windowsRegEx.test(versionObject.type)).toBe(false); | ||||
|       } | ||||
|     } | ||||
|   ); | ||||
| @ -276,10 +287,7 @@ function normalizeFileContents(contents: string): string { | ||||
|     .replace(new RegExp('\r', 'g'), '\n'); | ||||
| } | ||||
| 
 | ||||
| async function getDotnet( | ||||
|   version: string, | ||||
|   quality: string = '' | ||||
| ): Promise<string> { | ||||
| async function getDotnet(version: string, quality = ''): Promise<string> { | ||||
|   const dotnetInstaller = new installer.DotnetCoreInstaller( | ||||
|     version, | ||||
|     quality as QualityOptions | ||||
|  | ||||
| @ -21,11 +21,11 @@ if (IS_WINDOWS) { | ||||
| const tempDir = path.join(__dirname, 'runner', 'temp2'); | ||||
| 
 | ||||
| describe('setup-dotnet tests', () => { | ||||
|   let getInputSpy = jest.spyOn(core, 'getInput'); | ||||
|   let getMultilineInputSpy = jest.spyOn(core, 'getMultilineInput'); | ||||
|   let setOutputSpy = jest.spyOn(core, 'setOutput'); | ||||
|   const getInputSpy = jest.spyOn(core, 'getInput'); | ||||
|   const getMultilineInputSpy = jest.spyOn(core, 'getMultilineInput'); | ||||
|   const setOutputSpy = jest.spyOn(core, 'setOutput'); | ||||
| 
 | ||||
|   let inputs = {} as any; | ||||
|   const inputs = {} as any; | ||||
| 
 | ||||
|   beforeAll(async () => { | ||||
|     process.env.RUNNER_TOOL_CACHE = toolDir; | ||||
| @ -74,7 +74,7 @@ describe('setup-dotnet tests', () => { | ||||
| 
 | ||||
|     await setup.run(); | ||||
| 
 | ||||
|     expect(setOutputSpy).toBeCalledWith('dotnet-version', '6.0.401'); | ||||
|     expect(setOutputSpy).toHaveBeenCalledWith('dotnet-version', '6.0.401'); | ||||
|   }, 400000); | ||||
| 
 | ||||
|   it("Sets output with the version specified in global.json, if it's present", async () => { | ||||
| @ -93,6 +93,6 @@ describe('setup-dotnet tests', () => { | ||||
| 
 | ||||
|     await setup.run(); | ||||
| 
 | ||||
|     expect(setOutputSpy).toBeCalledWith('dotnet-version', '3.0.103'); | ||||
|     expect(setOutputSpy).toHaveBeenCalledWith('dotnet-version', '3.0.103'); | ||||
|   }, 400000); | ||||
| }); | ||||
|  | ||||
							
								
								
									
										18
									
								
								dist/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								dist/index.js
									
									
									
									
										vendored
									
									
								
							| @ -45,7 +45,7 @@ function configAuthentication(feedUrl, existingFileLocation = '', processRoot = | ||||
| } | ||||
| exports.configAuthentication = configAuthentication; | ||||
| function isValidKey(key) { | ||||
|     return /^[\w\-\.]+$/i.test(key); | ||||
|     return /^[\w\-.]+$/i.test(key); | ||||
| } | ||||
| function getExistingNugetConfig(processRoot) { | ||||
|     const defaultConfigName = 'nuget.config'; | ||||
| @ -60,9 +60,9 @@ function getExistingNugetConfig(processRoot) { | ||||
| function writeFeedToFile(feedUrl, existingFileLocation, tempFileLocation) { | ||||
|     var _a, _b; | ||||
|     core.info(`dotnet-auth: Finding any source references in ${existingFileLocation}, writing a new temporary configuration file with credentials to ${tempFileLocation}`); | ||||
|     let sourceKeys = []; | ||||
|     const sourceKeys = []; | ||||
|     let owner = core.getInput('owner'); | ||||
|     let sourceUrl = feedUrl; | ||||
|     const sourceUrl = feedUrl; | ||||
|     if (!owner) { | ||||
|         owner = github.context.repo.owner; | ||||
|     } | ||||
| @ -132,7 +132,7 @@ function writeFeedToFile(feedUrl, existingFileLocation, tempFileLocation) { | ||||
|         } | ||||
|     ]; | ||||
|     if (!sourceKeys.length) { | ||||
|         let keystring = 'Source'; | ||||
|         const keystring = 'Source'; | ||||
|         xmlSource[1].configuration.push({ | ||||
|             packageSources: [ | ||||
|                 { | ||||
| @ -300,9 +300,9 @@ class DotnetVersionResolver { | ||||
|         return __awaiter(this, void 0, void 0, function* () { | ||||
|             const response = yield httpClient.getJson(DotnetVersionResolver.DotNetCoreIndexUrl); | ||||
|             const result = response.result || {}; | ||||
|             let releasesInfo = result['releases-index']; | ||||
|             let releaseInfo = releasesInfo.find(info => { | ||||
|                 let sdkParts = info['channel-version'].split('.'); | ||||
|             const releasesInfo = result['releases-index']; | ||||
|             const releaseInfo = releasesInfo.find(info => { | ||||
|                 const sdkParts = info['channel-version'].split('.'); | ||||
|                 return sdkParts[0] === versionParts[0]; | ||||
|             }); | ||||
|             if (!releaseInfo) { | ||||
| @ -408,8 +408,8 @@ class DotnetCoreInstaller { | ||||
|     outputDotnetVersion(version) { | ||||
|         return __awaiter(this, void 0, void 0, function* () { | ||||
|             const installationPath = process.env['DOTNET_INSTALL_DIR']; | ||||
|             let versionsOnRunner = yield (0, promises_1.readdir)(path_1.default.join(installationPath.replace(/'/g, ''), 'sdk')); | ||||
|             let installedVersion = semver_1.default.maxSatisfying(versionsOnRunner, version, { | ||||
|             const versionsOnRunner = yield (0, promises_1.readdir)(path_1.default.join(installationPath.replace(/'/g, ''), 'sdk')); | ||||
|             const installedVersion = semver_1.default.maxSatisfying(versionsOnRunner, version, { | ||||
|                 includePrerelease: true | ||||
|             }); | ||||
|             return installedVersion; | ||||
|  | ||||
| @ -61,6 +61,7 @@ Pull requests are the easiest way to contribute changes to git repos at GitHub. | ||||
| 
 | ||||
| - To implement new features or fix bugs, you need to make changes to the `.ts` files, which are located in the `src` folder | ||||
| - To comply with the code style, **you need to run the `format` script** | ||||
| - To lint the code, **you need to run the `lint:fix` script** | ||||
| - To transpile source code to `javascript` we use [NCC](https://github.com/vercel/ncc). **It is very important to run the `build` script after making changes**, otherwise your changes will not get into the final `javascript` build | ||||
| 
 | ||||
| **Learn more about how to implement tests:** | ||||
|  | ||||
							
								
								
									
										2137
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2137
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										14
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								package.json
									
									
									
									
									
								
							| @ -6,9 +6,10 @@ | ||||
|   "main": "lib/setup-dotnet.js", | ||||
|   "scripts": { | ||||
|     "build": "tsc && ncc build", | ||||
|     "format": "prettier --write **/*.ts", | ||||
|     "format-check": "prettier --check **/*.ts", | ||||
|     "lint": "echo \"Fake command that does nothing. It is used in reusable workflows\"", | ||||
|     "format": "prettier --no-error-on-unmatched-pattern --config ./.prettierrc.js --write **/*.{ts,yml,yaml}", | ||||
|     "format-check": "prettier --no-error-on-unmatched-pattern --config ./.prettierrc.js --check **/*.{ts,yml,yaml}", | ||||
|     "lint": "eslint --config ./.eslintrc.js **/*.ts", | ||||
|     "lint:fix": "eslint --config ./.eslintrc.js **/*.ts --fix", | ||||
|     "prepare": "husky install", | ||||
|     "test": "jest --coverage --config ./jest.config.js", | ||||
|     "update-installers": "nwget https://dot.net/v1/dotnet-install.ps1 -O externals/install-dotnet.ps1 && nwget https://dot.net/v1/dotnet-install.sh -O externals/install-dotnet.sh" | ||||
| @ -37,11 +38,16 @@ | ||||
|     "@types/jest": "^27.0.2", | ||||
|     "@types/node": "^16.11.25", | ||||
|     "@types/semver": "^6.2.2", | ||||
|     "@typescript-eslint/eslint-plugin": "^5.54.0", | ||||
|     "@typescript-eslint/parser": "^5.54.0", | ||||
|     "@vercel/ncc": "^0.33.4", | ||||
|     "eslint": "^8.35.0", | ||||
|     "eslint-config-prettier": "^8.6.0", | ||||
|     "eslint-plugin-jest": "^27.2.1", | ||||
|     "husky": "^8.0.1", | ||||
|     "jest": "^27.2.5", | ||||
|     "jest-circus": "^27.2.5", | ||||
|     "prettier": "^2.7.1", | ||||
|     "prettier": "^2.8.4", | ||||
|     "ts-jest": "^27.0.5", | ||||
|     "typescript": "^4.8.4", | ||||
|     "wget-improved": "^3.2.1" | ||||
|  | ||||
							
								
								
									
										392
									
								
								src/authutil.ts
									
									
									
									
									
								
							
							
						
						
									
										392
									
								
								src/authutil.ts
									
									
									
									
									
								
							| @ -1,196 +1,196 @@ | ||||
| import * as fs from 'fs'; | ||||
| import * as path from 'path'; | ||||
| import * as core from '@actions/core'; | ||||
| import * as github from '@actions/github'; | ||||
| import {XMLParser, XMLBuilder} from 'fast-xml-parser'; | ||||
| 
 | ||||
| export function configAuthentication( | ||||
|   feedUrl: string, | ||||
|   existingFileLocation: string = '', | ||||
|   processRoot: string = process.cwd() | ||||
| ) { | ||||
|   const existingNuGetConfig: string = path.resolve( | ||||
|     processRoot, | ||||
|     existingFileLocation === '' | ||||
|       ? getExistingNugetConfig(processRoot) | ||||
|       : existingFileLocation | ||||
|   ); | ||||
| 
 | ||||
|   const tempNuGetConfig: string = path.resolve( | ||||
|     processRoot, | ||||
|     '../', | ||||
|     'nuget.config' | ||||
|   ); | ||||
| 
 | ||||
|   writeFeedToFile(feedUrl, existingNuGetConfig, tempNuGetConfig); | ||||
| } | ||||
| 
 | ||||
| function isValidKey(key: string): boolean { | ||||
|   return /^[\w\-\.]+$/i.test(key); | ||||
| } | ||||
| 
 | ||||
| function getExistingNugetConfig(processRoot: string) { | ||||
|   const defaultConfigName = 'nuget.config'; | ||||
|   const configFileNames = fs | ||||
|     .readdirSync(processRoot) | ||||
|     .filter(filename => filename.toLowerCase() === defaultConfigName); | ||||
|   if (configFileNames.length) { | ||||
|     return configFileNames[0]; | ||||
|   } | ||||
|   return defaultConfigName; | ||||
| } | ||||
| 
 | ||||
| function writeFeedToFile( | ||||
|   feedUrl: string, | ||||
|   existingFileLocation: string, | ||||
|   tempFileLocation: string | ||||
| ) { | ||||
|   core.info( | ||||
|     `dotnet-auth: Finding any source references in ${existingFileLocation}, writing a new temporary configuration file with credentials to ${tempFileLocation}` | ||||
|   ); | ||||
|   let sourceKeys: string[] = []; | ||||
|   let owner: string = core.getInput('owner'); | ||||
|   let sourceUrl: string = feedUrl; | ||||
|   if (!owner) { | ||||
|     owner = github.context.repo.owner; | ||||
|   } | ||||
| 
 | ||||
|   if (!process.env.NUGET_AUTH_TOKEN) { | ||||
|     throw new Error( | ||||
|       'The NUGET_AUTH_TOKEN environment variable was not provided. In this step, add the following: \r\nenv:\r\n  NUGET_AUTH_TOKEN: ${{secrets.GITHUB_TOKEN}}' | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   if (fs.existsSync(existingFileLocation)) { | ||||
|     // get key from existing NuGet.config so NuGet/dotnet can match credentials
 | ||||
|     const curContents: string = fs.readFileSync(existingFileLocation, 'utf8'); | ||||
| 
 | ||||
|     const parserOptions = { | ||||
|       ignoreAttributes: false | ||||
|     }; | ||||
|     const parser = new XMLParser(parserOptions); | ||||
|     const json = parser.parse(curContents); | ||||
| 
 | ||||
|     if (typeof json.configuration === 'undefined') { | ||||
|       throw new Error(`The provided NuGet.config seems invalid.`); | ||||
|     } | ||||
|     if (json.configuration?.packageSources?.add) { | ||||
|       const packageSources = json.configuration.packageSources.add; | ||||
| 
 | ||||
|       if (Array.isArray(packageSources)) { | ||||
|         packageSources.forEach(source => { | ||||
|           const value = source['@_value']; | ||||
|           core.debug(`source '${value}'`); | ||||
|           if (value.toLowerCase().includes(feedUrl.toLowerCase())) { | ||||
|             const key = source['@_key']; | ||||
|             sourceKeys.push(key); | ||||
|             core.debug(`Found a URL with key ${key}`); | ||||
|           } | ||||
|         }); | ||||
|       } else { | ||||
|         if ( | ||||
|           packageSources['@_value'] | ||||
|             .toLowerCase() | ||||
|             .includes(feedUrl.toLowerCase()) | ||||
|         ) { | ||||
|           const key = packageSources['@_key']; | ||||
|           sourceKeys.push(key); | ||||
|           core.debug(`Found a URL with key ${key}`); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const xmlSource: any[] = [ | ||||
|     { | ||||
|       '?xml': [ | ||||
|         { | ||||
|           '#text': '' | ||||
|         } | ||||
|       ], | ||||
|       ':@': { | ||||
|         '@_version': '1.0' | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       configuration: [ | ||||
|         { | ||||
|           config: [ | ||||
|             { | ||||
|               add: [], | ||||
|               ':@': { | ||||
|                 '@_key': 'defaultPushSource', | ||||
|                 '@_value': sourceUrl | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   ]; | ||||
| 
 | ||||
|   if (!sourceKeys.length) { | ||||
|     let keystring = 'Source'; | ||||
| 
 | ||||
|     xmlSource[1].configuration.push({ | ||||
|       packageSources: [ | ||||
|         { | ||||
|           add: [], | ||||
|           ':@': { | ||||
|             '@_key': keystring, | ||||
|             '@_value': sourceUrl | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }); | ||||
| 
 | ||||
|     sourceKeys.push(keystring); | ||||
|   } | ||||
| 
 | ||||
|   const packageSourceCredentials: any[] = []; | ||||
|   sourceKeys.forEach(key => { | ||||
|     if (!isValidKey(key)) { | ||||
|       throw new Error( | ||||
|         "Source name can contain letters, numbers, and '-', '_', '.' symbols only. Please, fix source name in NuGet.config and try again." | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     packageSourceCredentials.push({ | ||||
|       [key]: [ | ||||
|         { | ||||
|           add: [], | ||||
|           ':@': { | ||||
|             '@_key': 'Username', | ||||
|             '@_value': owner | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           add: [], | ||||
|           ':@': { | ||||
|             '@_key': 'ClearTextPassword', | ||||
|             '@_value': process.env.NUGET_AUTH_TOKEN | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   xmlSource[1].configuration.push({ | ||||
|     packageSourceCredentials | ||||
|   }); | ||||
| 
 | ||||
|   const xmlBuilderOptions = { | ||||
|     format: true, | ||||
|     ignoreAttributes: false, | ||||
|     preserveOrder: true, | ||||
|     allowBooleanAttributes: true, | ||||
|     suppressBooleanAttributes: true, | ||||
|     suppressEmptyNode: true | ||||
|   }; | ||||
| 
 | ||||
|   const builder = new XMLBuilder(xmlBuilderOptions); | ||||
| 
 | ||||
|   const output = builder.build(xmlSource).trim(); | ||||
| 
 | ||||
|   fs.writeFileSync(tempFileLocation, output); | ||||
| } | ||||
| import * as fs from 'fs'; | ||||
| import * as path from 'path'; | ||||
| import * as core from '@actions/core'; | ||||
| import * as github from '@actions/github'; | ||||
| import {XMLParser, XMLBuilder} from 'fast-xml-parser'; | ||||
| 
 | ||||
| export function configAuthentication( | ||||
|   feedUrl: string, | ||||
|   existingFileLocation = '', | ||||
|   processRoot: string = process.cwd() | ||||
| ) { | ||||
|   const existingNuGetConfig: string = path.resolve( | ||||
|     processRoot, | ||||
|     existingFileLocation === '' | ||||
|       ? getExistingNugetConfig(processRoot) | ||||
|       : existingFileLocation | ||||
|   ); | ||||
| 
 | ||||
|   const tempNuGetConfig: string = path.resolve( | ||||
|     processRoot, | ||||
|     '../', | ||||
|     'nuget.config' | ||||
|   ); | ||||
| 
 | ||||
|   writeFeedToFile(feedUrl, existingNuGetConfig, tempNuGetConfig); | ||||
| } | ||||
| 
 | ||||
| function isValidKey(key: string): boolean { | ||||
|   return /^[\w\-.]+$/i.test(key); | ||||
| } | ||||
| 
 | ||||
| function getExistingNugetConfig(processRoot: string) { | ||||
|   const defaultConfigName = 'nuget.config'; | ||||
|   const configFileNames = fs | ||||
|     .readdirSync(processRoot) | ||||
|     .filter(filename => filename.toLowerCase() === defaultConfigName); | ||||
|   if (configFileNames.length) { | ||||
|     return configFileNames[0]; | ||||
|   } | ||||
|   return defaultConfigName; | ||||
| } | ||||
| 
 | ||||
| function writeFeedToFile( | ||||
|   feedUrl: string, | ||||
|   existingFileLocation: string, | ||||
|   tempFileLocation: string | ||||
| ) { | ||||
|   core.info( | ||||
|     `dotnet-auth: Finding any source references in ${existingFileLocation}, writing a new temporary configuration file with credentials to ${tempFileLocation}` | ||||
|   ); | ||||
|   const sourceKeys: string[] = []; | ||||
|   let owner: string = core.getInput('owner'); | ||||
|   const sourceUrl: string = feedUrl; | ||||
|   if (!owner) { | ||||
|     owner = github.context.repo.owner; | ||||
|   } | ||||
| 
 | ||||
|   if (!process.env.NUGET_AUTH_TOKEN) { | ||||
|     throw new Error( | ||||
|       'The NUGET_AUTH_TOKEN environment variable was not provided. In this step, add the following: \r\nenv:\r\n  NUGET_AUTH_TOKEN: ${{secrets.GITHUB_TOKEN}}' | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   if (fs.existsSync(existingFileLocation)) { | ||||
|     // get key from existing NuGet.config so NuGet/dotnet can match credentials
 | ||||
|     const curContents: string = fs.readFileSync(existingFileLocation, 'utf8'); | ||||
| 
 | ||||
|     const parserOptions = { | ||||
|       ignoreAttributes: false | ||||
|     }; | ||||
|     const parser = new XMLParser(parserOptions); | ||||
|     const json = parser.parse(curContents); | ||||
| 
 | ||||
|     if (typeof json.configuration === 'undefined') { | ||||
|       throw new Error(`The provided NuGet.config seems invalid.`); | ||||
|     } | ||||
|     if (json.configuration?.packageSources?.add) { | ||||
|       const packageSources = json.configuration.packageSources.add; | ||||
| 
 | ||||
|       if (Array.isArray(packageSources)) { | ||||
|         packageSources.forEach(source => { | ||||
|           const value = source['@_value']; | ||||
|           core.debug(`source '${value}'`); | ||||
|           if (value.toLowerCase().includes(feedUrl.toLowerCase())) { | ||||
|             const key = source['@_key']; | ||||
|             sourceKeys.push(key); | ||||
|             core.debug(`Found a URL with key ${key}`); | ||||
|           } | ||||
|         }); | ||||
|       } else { | ||||
|         if ( | ||||
|           packageSources['@_value'] | ||||
|             .toLowerCase() | ||||
|             .includes(feedUrl.toLowerCase()) | ||||
|         ) { | ||||
|           const key = packageSources['@_key']; | ||||
|           sourceKeys.push(key); | ||||
|           core.debug(`Found a URL with key ${key}`); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const xmlSource: any[] = [ | ||||
|     { | ||||
|       '?xml': [ | ||||
|         { | ||||
|           '#text': '' | ||||
|         } | ||||
|       ], | ||||
|       ':@': { | ||||
|         '@_version': '1.0' | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       configuration: [ | ||||
|         { | ||||
|           config: [ | ||||
|             { | ||||
|               add: [], | ||||
|               ':@': { | ||||
|                 '@_key': 'defaultPushSource', | ||||
|                 '@_value': sourceUrl | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   ]; | ||||
| 
 | ||||
|   if (!sourceKeys.length) { | ||||
|     const keystring = 'Source'; | ||||
| 
 | ||||
|     xmlSource[1].configuration.push({ | ||||
|       packageSources: [ | ||||
|         { | ||||
|           add: [], | ||||
|           ':@': { | ||||
|             '@_key': keystring, | ||||
|             '@_value': sourceUrl | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }); | ||||
| 
 | ||||
|     sourceKeys.push(keystring); | ||||
|   } | ||||
| 
 | ||||
|   const packageSourceCredentials: any[] = []; | ||||
|   sourceKeys.forEach(key => { | ||||
|     if (!isValidKey(key)) { | ||||
|       throw new Error( | ||||
|         "Source name can contain letters, numbers, and '-', '_', '.' symbols only. Please, fix source name in NuGet.config and try again." | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     packageSourceCredentials.push({ | ||||
|       [key]: [ | ||||
|         { | ||||
|           add: [], | ||||
|           ':@': { | ||||
|             '@_key': 'Username', | ||||
|             '@_value': owner | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           add: [], | ||||
|           ':@': { | ||||
|             '@_key': 'ClearTextPassword', | ||||
|             '@_value': process.env.NUGET_AUTH_TOKEN | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   xmlSource[1].configuration.push({ | ||||
|     packageSourceCredentials | ||||
|   }); | ||||
| 
 | ||||
|   const xmlBuilderOptions = { | ||||
|     format: true, | ||||
|     ignoreAttributes: false, | ||||
|     preserveOrder: true, | ||||
|     allowBooleanAttributes: true, | ||||
|     suppressBooleanAttributes: true, | ||||
|     suppressEmptyNode: true | ||||
|   }; | ||||
| 
 | ||||
|   const builder = new XMLBuilder(xmlBuilderOptions); | ||||
| 
 | ||||
|   const output = builder.build(xmlSource).trim(); | ||||
| 
 | ||||
|   fs.writeFileSync(tempFileLocation, output); | ||||
| } | ||||
|  | ||||
							
								
								
									
										530
									
								
								src/installer.ts
									
									
									
									
									
								
							
							
						
						
									
										530
									
								
								src/installer.ts
									
									
									
									
									
								
							| @ -1,265 +1,265 @@ | ||||
| // Load tempDirectory before it gets wiped by tool-cache
 | ||||
| import * as core from '@actions/core'; | ||||
| import * as exec from '@actions/exec'; | ||||
| import * as io from '@actions/io'; | ||||
| import * as hc from '@actions/http-client'; | ||||
| import {chmodSync} from 'fs'; | ||||
| import {readdir} from 'fs/promises'; | ||||
| import path from 'path'; | ||||
| import os from 'os'; | ||||
| import semver from 'semver'; | ||||
| import {IS_LINUX, IS_WINDOWS} from './utils'; | ||||
| import {QualityOptions} from './setup-dotnet'; | ||||
| 
 | ||||
| export interface DotnetVersion { | ||||
|   type: string; | ||||
|   value: string; | ||||
|   qualityFlag: boolean; | ||||
| } | ||||
| 
 | ||||
| export class DotnetVersionResolver { | ||||
|   private inputVersion: string; | ||||
|   private resolvedArgument: DotnetVersion; | ||||
| 
 | ||||
|   constructor(version: string) { | ||||
|     this.inputVersion = version.trim(); | ||||
|     this.resolvedArgument = {type: '', value: '', qualityFlag: false}; | ||||
|   } | ||||
| 
 | ||||
|   private async resolveVersionInput(): Promise<void> { | ||||
|     if (!semver.validRange(this.inputVersion)) { | ||||
|       throw new Error( | ||||
|         `'dotnet-version' was supplied in invalid format: ${this.inputVersion}! Supported syntax: A.B.C, A.B, A.B.x, A, A.x` | ||||
|       ); | ||||
|     } | ||||
|     if (semver.valid(this.inputVersion)) { | ||||
|       this.resolvedArgument.type = 'version'; | ||||
|       this.resolvedArgument.value = this.inputVersion; | ||||
|     } else { | ||||
|       const [major, minor] = this.inputVersion.split('.'); | ||||
| 
 | ||||
|       if (this.isNumericTag(major)) { | ||||
|         this.resolvedArgument.type = 'channel'; | ||||
|         if (this.isNumericTag(minor)) { | ||||
|           this.resolvedArgument.value = `${major}.${minor}`; | ||||
|         } else { | ||||
|           const httpClient = new hc.HttpClient('actions/setup-dotnet', [], { | ||||
|             allowRetries: true, | ||||
|             maxRetries: 3 | ||||
|           }); | ||||
|           this.resolvedArgument.value = await this.getLatestVersion( | ||||
|             httpClient, | ||||
|             [major, minor] | ||||
|           ); | ||||
|         } | ||||
|       } | ||||
|       this.resolvedArgument.qualityFlag = +major >= 6 ? true : false; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   private isNumericTag(versionTag): boolean { | ||||
|     return /^\d+$/.test(versionTag); | ||||
|   } | ||||
| 
 | ||||
|   public async createDotNetVersion(): Promise<{ | ||||
|     type: string; | ||||
|     value: string; | ||||
|     qualityFlag: boolean; | ||||
|   }> { | ||||
|     await this.resolveVersionInput(); | ||||
|     if (!this.resolvedArgument.type) { | ||||
|       return this.resolvedArgument; | ||||
|     } | ||||
|     if (IS_WINDOWS) { | ||||
|       this.resolvedArgument.type = | ||||
|         this.resolvedArgument.type === 'channel' ? '-Channel' : '-Version'; | ||||
|     } else { | ||||
|       this.resolvedArgument.type = | ||||
|         this.resolvedArgument.type === 'channel' ? '--channel' : '--version'; | ||||
|     } | ||||
|     return this.resolvedArgument; | ||||
|   } | ||||
| 
 | ||||
|   private async getLatestVersion( | ||||
|     httpClient: hc.HttpClient, | ||||
|     versionParts: string[] | ||||
|   ): Promise<string> { | ||||
|     const response = await httpClient.getJson<any>( | ||||
|       DotnetVersionResolver.DotNetCoreIndexUrl | ||||
|     ); | ||||
|     const result = response.result || {}; | ||||
|     let releasesInfo: any[] = result['releases-index']; | ||||
| 
 | ||||
|     let releaseInfo = releasesInfo.find(info => { | ||||
|       let sdkParts: string[] = info['channel-version'].split('.'); | ||||
|       return sdkParts[0] === versionParts[0]; | ||||
|     }); | ||||
| 
 | ||||
|     if (!releaseInfo) { | ||||
|       throw new Error( | ||||
|         `Could not find info for version ${versionParts.join('.')} at ${ | ||||
|           DotnetVersionResolver.DotNetCoreIndexUrl | ||||
|         }` | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return releaseInfo['channel-version']; | ||||
|   } | ||||
| 
 | ||||
|   static DotNetCoreIndexUrl: string = | ||||
|     'https://dotnetcli.azureedge.net/dotnet/release-metadata/releases-index.json'; | ||||
| } | ||||
| 
 | ||||
| export class DotnetCoreInstaller { | ||||
|   private version: string; | ||||
|   private quality: QualityOptions; | ||||
| 
 | ||||
|   static { | ||||
|     const installationDirectoryWindows = path.join( | ||||
|       process.env['PROGRAMFILES'] + '', | ||||
|       'dotnet' | ||||
|     ); | ||||
|     const installationDirectoryLinux = '/usr/share/dotnet'; | ||||
|     const installationDirectoryMac = path.join( | ||||
|       process.env['HOME'] + '', | ||||
|       '.dotnet' | ||||
|     ); | ||||
|     const dotnetInstallDir: string | undefined = | ||||
|       process.env['DOTNET_INSTALL_DIR']; | ||||
|     if (dotnetInstallDir) { | ||||
|       process.env['DOTNET_INSTALL_DIR'] = | ||||
|         this.convertInstallPathToAbsolute(dotnetInstallDir); | ||||
|     } else { | ||||
|       if (IS_WINDOWS) { | ||||
|         process.env['DOTNET_INSTALL_DIR'] = installationDirectoryWindows; | ||||
|       } else { | ||||
|         process.env['DOTNET_INSTALL_DIR'] = IS_LINUX | ||||
|           ? installationDirectoryLinux | ||||
|           : installationDirectoryMac; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   constructor(version: string, quality: QualityOptions) { | ||||
|     this.version = version; | ||||
|     this.quality = quality; | ||||
|   } | ||||
| 
 | ||||
|   private static convertInstallPathToAbsolute(installDir: string): string { | ||||
|     let transformedPath; | ||||
|     if (path.isAbsolute(installDir)) { | ||||
|       transformedPath = installDir; | ||||
|     } else { | ||||
|       transformedPath = installDir.startsWith('~') | ||||
|         ? path.join(os.homedir(), installDir.slice(1)) | ||||
|         : (transformedPath = path.join(process.cwd(), installDir)); | ||||
|     } | ||||
|     return path.normalize(transformedPath); | ||||
|   } | ||||
| 
 | ||||
|   static addToPath() { | ||||
|     core.addPath(process.env['DOTNET_INSTALL_DIR']!); | ||||
|     core.exportVariable('DOTNET_ROOT', process.env['DOTNET_INSTALL_DIR']); | ||||
|   } | ||||
| 
 | ||||
|   private setQuality( | ||||
|     dotnetVersion: DotnetVersion, | ||||
|     scriptArguments: string[] | ||||
|   ): void { | ||||
|     const option = IS_WINDOWS ? '-Quality' : '--quality'; | ||||
|     if (dotnetVersion.qualityFlag) { | ||||
|       scriptArguments.push(option, this.quality); | ||||
|     } else { | ||||
|       core.warning( | ||||
|         `'dotnet-quality' input can be used only with .NET SDK version in A.B, A.B.x, A and A.x formats where the major tag is higher than 5. You specified: ${this.version}. 'dotnet-quality' input is ignored.` | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public async installDotnet(): Promise<string> { | ||||
|     const windowsDefaultOptions = [ | ||||
|       '-NoLogo', | ||||
|       '-Sta', | ||||
|       '-NoProfile', | ||||
|       '-NonInteractive', | ||||
|       '-ExecutionPolicy', | ||||
|       'Unrestricted', | ||||
|       '-Command' | ||||
|     ]; | ||||
|     const scriptName = IS_WINDOWS ? 'install-dotnet.ps1' : 'install-dotnet.sh'; | ||||
|     const escapedScript = path | ||||
|       .join(__dirname, '..', 'externals', scriptName) | ||||
|       .replace(/'/g, "''"); | ||||
|     let scriptArguments: string[]; | ||||
|     let scriptPath = ''; | ||||
| 
 | ||||
|     const versionResolver = new DotnetVersionResolver(this.version); | ||||
|     const dotnetVersion = await versionResolver.createDotNetVersion(); | ||||
| 
 | ||||
|     if (IS_WINDOWS) { | ||||
|       scriptArguments = ['&', `'${escapedScript}'`]; | ||||
| 
 | ||||
|       if (dotnetVersion.type) { | ||||
|         scriptArguments.push(dotnetVersion.type, dotnetVersion.value); | ||||
|       } | ||||
| 
 | ||||
|       if (this.quality) { | ||||
|         this.setQuality(dotnetVersion, scriptArguments); | ||||
|       } | ||||
| 
 | ||||
|       if (process.env['https_proxy'] != null) { | ||||
|         scriptArguments.push(`-ProxyAddress ${process.env['https_proxy']}`); | ||||
|       } | ||||
|       // This is not currently an option
 | ||||
|       if (process.env['no_proxy'] != null) { | ||||
|         scriptArguments.push(`-ProxyBypassList ${process.env['no_proxy']}`); | ||||
|       } | ||||
| 
 | ||||
|       scriptPath = | ||||
|         (await io.which('pwsh', false)) || (await io.which('powershell', true)); | ||||
|       scriptArguments = windowsDefaultOptions.concat(scriptArguments); | ||||
|     } else { | ||||
|       chmodSync(escapedScript, '777'); | ||||
|       scriptPath = await io.which(escapedScript, true); | ||||
|       scriptArguments = []; | ||||
| 
 | ||||
|       if (dotnetVersion.type) { | ||||
|         scriptArguments.push(dotnetVersion.type, dotnetVersion.value); | ||||
|       } | ||||
| 
 | ||||
|       if (this.quality) { | ||||
|         this.setQuality(dotnetVersion, scriptArguments); | ||||
|       } | ||||
|     } | ||||
|     // process.env must be explicitly passed in for DOTNET_INSTALL_DIR to be used
 | ||||
|     const getExecOutputOptions = { | ||||
|       ignoreReturnCode: true, | ||||
|       env: process.env as {string: string} | ||||
|     }; | ||||
|     const {exitCode, stderr} = await exec.getExecOutput( | ||||
|       `"${scriptPath}"`, | ||||
|       scriptArguments, | ||||
|       getExecOutputOptions | ||||
|     ); | ||||
|     if (exitCode) { | ||||
|       throw new Error( | ||||
|         `Failed to install dotnet, exit code: ${exitCode}. ${stderr}` | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return this.outputDotnetVersion(dotnetVersion.value); | ||||
|   } | ||||
| 
 | ||||
|   private async outputDotnetVersion(version): Promise<string> { | ||||
|     const installationPath = process.env['DOTNET_INSTALL_DIR']!; | ||||
|     let versionsOnRunner: string[] = await readdir( | ||||
|       path.join(installationPath.replace(/'/g, ''), 'sdk') | ||||
|     ); | ||||
| 
 | ||||
|     let installedVersion = semver.maxSatisfying(versionsOnRunner, version, { | ||||
|       includePrerelease: true | ||||
|     })!; | ||||
| 
 | ||||
|     return installedVersion; | ||||
|   } | ||||
| } | ||||
| // Load tempDirectory before it gets wiped by tool-cache
 | ||||
| import * as core from '@actions/core'; | ||||
| import * as exec from '@actions/exec'; | ||||
| import * as io from '@actions/io'; | ||||
| import * as hc from '@actions/http-client'; | ||||
| import {chmodSync} from 'fs'; | ||||
| import {readdir} from 'fs/promises'; | ||||
| import path from 'path'; | ||||
| import os from 'os'; | ||||
| import semver from 'semver'; | ||||
| import {IS_LINUX, IS_WINDOWS} from './utils'; | ||||
| import {QualityOptions} from './setup-dotnet'; | ||||
| 
 | ||||
| export interface DotnetVersion { | ||||
|   type: string; | ||||
|   value: string; | ||||
|   qualityFlag: boolean; | ||||
| } | ||||
| 
 | ||||
| export class DotnetVersionResolver { | ||||
|   private inputVersion: string; | ||||
|   private resolvedArgument: DotnetVersion; | ||||
| 
 | ||||
|   constructor(version: string) { | ||||
|     this.inputVersion = version.trim(); | ||||
|     this.resolvedArgument = {type: '', value: '', qualityFlag: false}; | ||||
|   } | ||||
| 
 | ||||
|   private async resolveVersionInput(): Promise<void> { | ||||
|     if (!semver.validRange(this.inputVersion)) { | ||||
|       throw new Error( | ||||
|         `'dotnet-version' was supplied in invalid format: ${this.inputVersion}! Supported syntax: A.B.C, A.B, A.B.x, A, A.x` | ||||
|       ); | ||||
|     } | ||||
|     if (semver.valid(this.inputVersion)) { | ||||
|       this.resolvedArgument.type = 'version'; | ||||
|       this.resolvedArgument.value = this.inputVersion; | ||||
|     } else { | ||||
|       const [major, minor] = this.inputVersion.split('.'); | ||||
| 
 | ||||
|       if (this.isNumericTag(major)) { | ||||
|         this.resolvedArgument.type = 'channel'; | ||||
|         if (this.isNumericTag(minor)) { | ||||
|           this.resolvedArgument.value = `${major}.${minor}`; | ||||
|         } else { | ||||
|           const httpClient = new hc.HttpClient('actions/setup-dotnet', [], { | ||||
|             allowRetries: true, | ||||
|             maxRetries: 3 | ||||
|           }); | ||||
|           this.resolvedArgument.value = await this.getLatestVersion( | ||||
|             httpClient, | ||||
|             [major, minor] | ||||
|           ); | ||||
|         } | ||||
|       } | ||||
|       this.resolvedArgument.qualityFlag = +major >= 6 ? true : false; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   private isNumericTag(versionTag): boolean { | ||||
|     return /^\d+$/.test(versionTag); | ||||
|   } | ||||
| 
 | ||||
|   public async createDotNetVersion(): Promise<{ | ||||
|     type: string; | ||||
|     value: string; | ||||
|     qualityFlag: boolean; | ||||
|   }> { | ||||
|     await this.resolveVersionInput(); | ||||
|     if (!this.resolvedArgument.type) { | ||||
|       return this.resolvedArgument; | ||||
|     } | ||||
|     if (IS_WINDOWS) { | ||||
|       this.resolvedArgument.type = | ||||
|         this.resolvedArgument.type === 'channel' ? '-Channel' : '-Version'; | ||||
|     } else { | ||||
|       this.resolvedArgument.type = | ||||
|         this.resolvedArgument.type === 'channel' ? '--channel' : '--version'; | ||||
|     } | ||||
|     return this.resolvedArgument; | ||||
|   } | ||||
| 
 | ||||
|   private async getLatestVersion( | ||||
|     httpClient: hc.HttpClient, | ||||
|     versionParts: string[] | ||||
|   ): Promise<string> { | ||||
|     const response = await httpClient.getJson<any>( | ||||
|       DotnetVersionResolver.DotNetCoreIndexUrl | ||||
|     ); | ||||
|     const result = response.result || {}; | ||||
|     const releasesInfo: any[] = result['releases-index']; | ||||
| 
 | ||||
|     const releaseInfo = releasesInfo.find(info => { | ||||
|       const sdkParts: string[] = info['channel-version'].split('.'); | ||||
|       return sdkParts[0] === versionParts[0]; | ||||
|     }); | ||||
| 
 | ||||
|     if (!releaseInfo) { | ||||
|       throw new Error( | ||||
|         `Could not find info for version ${versionParts.join('.')} at ${ | ||||
|           DotnetVersionResolver.DotNetCoreIndexUrl | ||||
|         }` | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return releaseInfo['channel-version']; | ||||
|   } | ||||
| 
 | ||||
|   static DotNetCoreIndexUrl = | ||||
|     'https://dotnetcli.azureedge.net/dotnet/release-metadata/releases-index.json'; | ||||
| } | ||||
| 
 | ||||
| export class DotnetCoreInstaller { | ||||
|   private version: string; | ||||
|   private quality: QualityOptions; | ||||
| 
 | ||||
|   static { | ||||
|     const installationDirectoryWindows = path.join( | ||||
|       process.env['PROGRAMFILES'] + '', | ||||
|       'dotnet' | ||||
|     ); | ||||
|     const installationDirectoryLinux = '/usr/share/dotnet'; | ||||
|     const installationDirectoryMac = path.join( | ||||
|       process.env['HOME'] + '', | ||||
|       '.dotnet' | ||||
|     ); | ||||
|     const dotnetInstallDir: string | undefined = | ||||
|       process.env['DOTNET_INSTALL_DIR']; | ||||
|     if (dotnetInstallDir) { | ||||
|       process.env['DOTNET_INSTALL_DIR'] = | ||||
|         this.convertInstallPathToAbsolute(dotnetInstallDir); | ||||
|     } else { | ||||
|       if (IS_WINDOWS) { | ||||
|         process.env['DOTNET_INSTALL_DIR'] = installationDirectoryWindows; | ||||
|       } else { | ||||
|         process.env['DOTNET_INSTALL_DIR'] = IS_LINUX | ||||
|           ? installationDirectoryLinux | ||||
|           : installationDirectoryMac; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   constructor(version: string, quality: QualityOptions) { | ||||
|     this.version = version; | ||||
|     this.quality = quality; | ||||
|   } | ||||
| 
 | ||||
|   private static convertInstallPathToAbsolute(installDir: string): string { | ||||
|     let transformedPath; | ||||
|     if (path.isAbsolute(installDir)) { | ||||
|       transformedPath = installDir; | ||||
|     } else { | ||||
|       transformedPath = installDir.startsWith('~') | ||||
|         ? path.join(os.homedir(), installDir.slice(1)) | ||||
|         : (transformedPath = path.join(process.cwd(), installDir)); | ||||
|     } | ||||
|     return path.normalize(transformedPath); | ||||
|   } | ||||
| 
 | ||||
|   static addToPath() { | ||||
|     core.addPath(process.env['DOTNET_INSTALL_DIR']!); | ||||
|     core.exportVariable('DOTNET_ROOT', process.env['DOTNET_INSTALL_DIR']); | ||||
|   } | ||||
| 
 | ||||
|   private setQuality( | ||||
|     dotnetVersion: DotnetVersion, | ||||
|     scriptArguments: string[] | ||||
|   ): void { | ||||
|     const option = IS_WINDOWS ? '-Quality' : '--quality'; | ||||
|     if (dotnetVersion.qualityFlag) { | ||||
|       scriptArguments.push(option, this.quality); | ||||
|     } else { | ||||
|       core.warning( | ||||
|         `'dotnet-quality' input can be used only with .NET SDK version in A.B, A.B.x, A and A.x formats where the major tag is higher than 5. You specified: ${this.version}. 'dotnet-quality' input is ignored.` | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public async installDotnet(): Promise<string> { | ||||
|     const windowsDefaultOptions = [ | ||||
|       '-NoLogo', | ||||
|       '-Sta', | ||||
|       '-NoProfile', | ||||
|       '-NonInteractive', | ||||
|       '-ExecutionPolicy', | ||||
|       'Unrestricted', | ||||
|       '-Command' | ||||
|     ]; | ||||
|     const scriptName = IS_WINDOWS ? 'install-dotnet.ps1' : 'install-dotnet.sh'; | ||||
|     const escapedScript = path | ||||
|       .join(__dirname, '..', 'externals', scriptName) | ||||
|       .replace(/'/g, "''"); | ||||
|     let scriptArguments: string[]; | ||||
|     let scriptPath = ''; | ||||
| 
 | ||||
|     const versionResolver = new DotnetVersionResolver(this.version); | ||||
|     const dotnetVersion = await versionResolver.createDotNetVersion(); | ||||
| 
 | ||||
|     if (IS_WINDOWS) { | ||||
|       scriptArguments = ['&', `'${escapedScript}'`]; | ||||
| 
 | ||||
|       if (dotnetVersion.type) { | ||||
|         scriptArguments.push(dotnetVersion.type, dotnetVersion.value); | ||||
|       } | ||||
| 
 | ||||
|       if (this.quality) { | ||||
|         this.setQuality(dotnetVersion, scriptArguments); | ||||
|       } | ||||
| 
 | ||||
|       if (process.env['https_proxy'] != null) { | ||||
|         scriptArguments.push(`-ProxyAddress ${process.env['https_proxy']}`); | ||||
|       } | ||||
|       // This is not currently an option
 | ||||
|       if (process.env['no_proxy'] != null) { | ||||
|         scriptArguments.push(`-ProxyBypassList ${process.env['no_proxy']}`); | ||||
|       } | ||||
| 
 | ||||
|       scriptPath = | ||||
|         (await io.which('pwsh', false)) || (await io.which('powershell', true)); | ||||
|       scriptArguments = windowsDefaultOptions.concat(scriptArguments); | ||||
|     } else { | ||||
|       chmodSync(escapedScript, '777'); | ||||
|       scriptPath = await io.which(escapedScript, true); | ||||
|       scriptArguments = []; | ||||
| 
 | ||||
|       if (dotnetVersion.type) { | ||||
|         scriptArguments.push(dotnetVersion.type, dotnetVersion.value); | ||||
|       } | ||||
| 
 | ||||
|       if (this.quality) { | ||||
|         this.setQuality(dotnetVersion, scriptArguments); | ||||
|       } | ||||
|     } | ||||
|     // process.env must be explicitly passed in for DOTNET_INSTALL_DIR to be used
 | ||||
|     const getExecOutputOptions = { | ||||
|       ignoreReturnCode: true, | ||||
|       env: process.env as {string: string} | ||||
|     }; | ||||
|     const {exitCode, stderr} = await exec.getExecOutput( | ||||
|       `"${scriptPath}"`, | ||||
|       scriptArguments, | ||||
|       getExecOutputOptions | ||||
|     ); | ||||
|     if (exitCode) { | ||||
|       throw new Error( | ||||
|         `Failed to install dotnet, exit code: ${exitCode}. ${stderr}` | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return this.outputDotnetVersion(dotnetVersion.value); | ||||
|   } | ||||
| 
 | ||||
|   private async outputDotnetVersion(version): Promise<string> { | ||||
|     const installationPath = process.env['DOTNET_INSTALL_DIR']!; | ||||
|     const versionsOnRunner: string[] = await readdir( | ||||
|       path.join(installationPath.replace(/'/g, ''), 'sdk') | ||||
|     ); | ||||
| 
 | ||||
|     const installedVersion = semver.maxSatisfying(versionsOnRunner, version, { | ||||
|       includePrerelease: true | ||||
|     })!; | ||||
| 
 | ||||
|     return installedVersion; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -13,7 +13,7 @@ const qualityOptions = [ | ||||
|   'ga' | ||||
| ] as const; | ||||
| 
 | ||||
| export type QualityOptions = typeof qualityOptions[number]; | ||||
| export type QualityOptions = (typeof qualityOptions)[number]; | ||||
| 
 | ||||
| export async function run() { | ||||
|   try { | ||||
| @ -100,7 +100,7 @@ export async function run() { | ||||
| } | ||||
| 
 | ||||
| function getVersionFromGlobalJson(globalJsonPath: string): string { | ||||
|   let version: string = ''; | ||||
|   let version = ''; | ||||
|   const globalJson = JSON.parse( | ||||
|     // .trim() is necessary to strip BOM https://github.com/nodejs/node/issues/20649
 | ||||
|     fs.readFileSync(globalJsonPath, {encoding: 'utf8'}).trim() | ||||
|  | ||||
| @ -49,7 +49,8 @@ | ||||
|     // "typeRoots": [],                       /* List of folders to include type definitions from. */ | ||||
|     // "types": [],                           /* Type declaration files to be included in compilation. */ | ||||
|     // "allowSyntheticDefaultImports": true,  /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */ | ||||
|     "esModuleInterop": true                   /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */ | ||||
|     "esModuleInterop": true,                   /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */ | ||||
|     "resolveJsonModule": true,                /* Allows importing modules with a '.json' extension, which is a common practice in node projects. */ | ||||
|     // "preserveSymlinks": true,              /* Do not resolve the real path of symlinks. */ | ||||
|     // "allowUmdGlobalAccess": true,          /* Allow accessing UMD globals from modules. */ | ||||
| 
 | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user