mirror of
				https://github.com/Smaug123/unofficial-nunit-runner
				synced 2025-10-23 01:08:40 +00:00 
			
		
		
		
	Compare commits
	
		
			71 Commits
		
	
	
		
			WoofWare.N
			...
			WoofWare.N
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 675baef736 | ||
|  | 20c13add32 | ||
|  | 5ab0c060ea | ||
|  | 01aa6445f9 | ||
|  | 26a5871d6b | ||
|  | 6ba06b2d6c | ||
|  | 81aa6832d5 | ||
|  | a20f32de02 | ||
|  | 1e53e72d4a | ||
|  | b38a3fcc02 | ||
|  | 73dc21e11f | ||
|  | 5b54bb256e | ||
|  | b56e1b1542 | ||
|  | ba46b1edb6 | ||
|  | 72674e1711 | ||
|  | c4b862bdd8 | ||
|  | 4c629b1d64 | ||
|  | e67820c56d | ||
|  | 31bff4cb03 | ||
|  | d081cfaafb | ||
|  | c237df3885 | ||
|  | 3b9b9eb4c8 | ||
|  | 12e3fc0e4f | ||
|  | 208b809096 | ||
|  | b4e5baddcf | ||
|  | 5597b3f2f8 | ||
|  | fcfdcef6cf | ||
|  | eeada219f6 | ||
|  | 99e0fdff08 | ||
|  | fda4e7ba60 | ||
|  | dfdfa84733 | ||
|  | c218110749 | ||
|  | 309968721c | ||
|  | 876ca9e625 | ||
|  | 59f9789cdc | ||
|  | c8c28b9a32 | ||
|  | 6d87610017 | ||
|  | 9f28334b7f | ||
|  | 5cf2261d7f | ||
|  | c3589820c3 | ||
|  | 5d0c205f21 | ||
|  | 16135fbd56 | ||
|  | cbc51cde14 | ||
|  | 6a4b900aa9 | ||
|  | 4169289ded | ||
|  | 7df58d5309 | ||
|  | ac4c8c245e | ||
|  | 3e79e79978 | ||
|  | c16e7dd1ee | ||
|  | cd66617ce7 | ||
|  | 144f71a417 | ||
|  | 3673fc56ee | ||
|  | 5f74b41825 | ||
|  | 2068007da0 | ||
|  | 6fd824c065 | ||
|  | 0480d5c151 | ||
|  | 9107b1b502 | ||
|  | 6583b9e025 | ||
|  | 48f7302391 | ||
|  | d1fa66a2e8 | ||
|  | e75c584a43 | ||
|  | d7bdd38253 | ||
|  | b7d87459d9 | ||
|  | 992679d8ca | ||
|  | 011a5129cc | ||
|  | 46a13d1583 | ||
|  | 1a291a2ac7 | ||
|  | fd34215461 | ||
|  | eda120332a | ||
|  | 799e5c8c3a | ||
|  | 172865b2a1 | 
| @@ -3,13 +3,13 @@ | |||||||
|   "isRoot": true, |   "isRoot": true, | ||||||
|   "tools": { |   "tools": { | ||||||
|     "fantomas": { |     "fantomas": { | ||||||
|       "version": "7.0.0", |       "version": "7.0.3", | ||||||
|       "commands": [ |       "commands": [ | ||||||
|         "fantomas" |         "fantomas" | ||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "fsharp-analyzers": { |     "fsharp-analyzers": { | ||||||
|       "version": "0.29.0", |       "version": "0.33.1", | ||||||
|       "commands": [ |       "commands": [ | ||||||
|         "fsharp-analyzers" |         "fsharp-analyzers" | ||||||
|       ] |       ] | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								.envrc
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								.envrc
									
									
									
									
									
								
							| @@ -1 +1,23 @@ | |||||||
| use flake | use flake | ||||||
|  | DOTNET_PATH=$(readlink "$(which dotnet)") | ||||||
|  | SETTINGS_FILE=$(find . -maxdepth 1 -type f -name '*.sln.DotSettings.user') | ||||||
|  | MSBUILD=$(realpath "$(find "$(dirname "$DOTNET_PATH")/../share/dotnet/sdk" -maxdepth 2 -type f -name MSBuild.dll)") | ||||||
|  | if [ -f "$SETTINGS_FILE" ] ; then | ||||||
|  |     xmlstarlet ed --inplace \ | ||||||
|  |     -N wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation" \ | ||||||
|  |     -N x="http://schemas.microsoft.com/winfx/2006/xaml" \ | ||||||
|  |     -N s="clr-namespace:System;assembly=mscorlib" \ | ||||||
|  |     -N ss="urn:shemas-jetbrains-com:settings-storage-xaml" \ | ||||||
|  |     --update "//s:String[@x:Key='/Default/Environment/Hierarchy/Build/BuildTool/DotNetCliExePath/@EntryValue']" \ | ||||||
|  |     --value "$(realpath "$(dirname "$DOTNET_PATH")/../share/dotnet/dotnet")" \ | ||||||
|  |     "$SETTINGS_FILE" | ||||||
|  |  | ||||||
|  |     xmlstarlet ed --inplace \ | ||||||
|  |     -N wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation" \ | ||||||
|  |     -N x="http://schemas.microsoft.com/winfx/2006/xaml" \ | ||||||
|  |     -N s="clr-namespace:System;assembly=mscorlib" \ | ||||||
|  |     -N ss="urn:shemas-jetbrains-com:settings-storage-xaml" \ | ||||||
|  |     --update "//s:String[@x:Key='/Default/Environment/Hierarchy/Build/BuildTool/CustomBuildToolPath/@EntryValue']" \ | ||||||
|  |     --value "$MSBUILD" \ | ||||||
|  |     "$SETTINGS_FILE" | ||||||
|  | fi | ||||||
|   | |||||||
							
								
								
									
										92
									
								
								.github/workflows/dotnet.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										92
									
								
								.github/workflows/dotnet.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -25,11 +25,11 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v5 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history |         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history | ||||||
|     - name: Install Nix |     - name: Install Nix | ||||||
|       uses: cachix/install-nix-action@v30 |       uses: cachix/install-nix-action@v31 | ||||||
|       with: |       with: | ||||||
|         extra_nix_config: | |         extra_nix_config: | | ||||||
|           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -39,16 +39,16 @@ jobs: | |||||||
|       run: 'nix develop --command dotnet build --no-restore --configuration ${{matrix.config}}' |       run: 'nix develop --command dotnet build --no-restore --configuration ${{matrix.config}}' | ||||||
|     - name: Test |     - name: Test | ||||||
|       run: | |       run: | | ||||||
|           nix develop --command dotnet test --no-build --verbosity normal --configuration ${{matrix.config}} --framework net8.0 --filter 'FullyQualifiedName !~ FailingConsumer' |           nix develop --command dotnet test --no-build --verbosity normal --configuration ${{matrix.config}} --filter 'FullyQualifiedName !~ FailingConsumer' | ||||||
|  |  | ||||||
|   selftest-intended-failures: |   selftest-intended-failures: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v5 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history |         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history | ||||||
|     - name: Install Nix |     - name: Install Nix | ||||||
|       uses: cachix/install-nix-action@v30 |       uses: cachix/install-nix-action@v31 | ||||||
|       with: |       with: | ||||||
|         extra_nix_config: | |         extra_nix_config: | | ||||||
|           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -57,7 +57,7 @@ jobs: | |||||||
|     - name: Build |     - name: Build | ||||||
|       run: 'nix develop --command dotnet build --no-restore --configuration Release' |       run: 'nix develop --command dotnet build --no-restore --configuration Release' | ||||||
|     - name: Test using self |     - name: Test using self | ||||||
|       run: 'nix develop --command dotnet exec ./WoofWare.NUnitTestRunner/bin/Release/net6.0/WoofWare.NUnitTestRunner.dll ./FailingConsumer/bin/Release/net8.0/FailingConsumer.dll --trx TrxOut/out.trx || true' |       run: 'nix develop --command dotnet exec ./WoofWare.NUnitTestRunner/bin/Release/net9.0/WoofWare.NUnitTestRunner.dll ./FailingConsumer/bin/Release/net9.0/FailingConsumer.dll --trx TrxOut/out.trx || true' | ||||||
|     - name: Munge output |     - name: Munge output | ||||||
|       run: 'nix develop --command xmlstarlet sel -N x="http://microsoft.com/schemas/VisualStudio/TeamTest/2010" -t -m "//x:UnitTestResult" -v "@testName" -o ": " -v ".//x:ErrorInfo/x:Message" -n TrxOut/out.trx > snapshot.txt' |       run: 'nix develop --command xmlstarlet sel -N x="http://microsoft.com/schemas/VisualStudio/TeamTest/2010" -t -m "//x:UnitTestResult" -v "@testName" -o ": " -v ".//x:ErrorInfo/x:Message" -n TrxOut/out.trx > snapshot.txt' | ||||||
|     - name: Check output matches expected |     - name: Check output matches expected | ||||||
| @@ -81,11 +81,11 @@ jobs: | |||||||
|       statuses: read |       statuses: read | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v5 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history |         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history | ||||||
|     - name: Install Nix |     - name: Install Nix | ||||||
|       uses: cachix/install-nix-action@v30 |       uses: cachix/install-nix-action@v31 | ||||||
|       with: |       with: | ||||||
|         extra_nix_config: | |         extra_nix_config: | | ||||||
|           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -94,10 +94,10 @@ jobs: | |||||||
|     - name: Build |     - name: Build | ||||||
|       run: 'nix develop --command dotnet build --no-restore --configuration Release' |       run: 'nix develop --command dotnet build --no-restore --configuration Release' | ||||||
|     - name: Test using self |     - name: Test using self | ||||||
|       run: 'nix develop --command dotnet exec ./WoofWare.NUnitTestRunner/bin/Release/net6.0/WoofWare.NUnitTestRunner.dll ./Consumer/bin/Release/net8.0/Consumer.dll --trx TrxOut/out.trx' |       run: 'nix develop --command dotnet exec ./WoofWare.NUnitTestRunner/bin/Release/net9.0/WoofWare.NUnitTestRunner.dll ./Consumer/bin/Release/net9.0/Consumer.dll --trx TrxOut/out.trx' | ||||||
|     - name: Parse Trx files |     - name: Parse Trx files | ||||||
|       uses: NasAmin/trx-parser@v0.6.0 |       uses: NasAmin/trx-parser@v0.7.0 | ||||||
|       if: always() |       if: always() && github.ref_name != 'main' | ||||||
|       id: trx-parser |       id: trx-parser | ||||||
|       with: |       with: | ||||||
|         TRX_PATH: ${{ github.workspace }}/TrxOut |         TRX_PATH: ${{ github.workspace }}/TrxOut | ||||||
| @@ -109,11 +109,11 @@ jobs: | |||||||
|       security-events: write |       security-events: write | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|         with: |         with: | ||||||
|           fetch-depth: 0 # so that NerdBank.GitVersioning has access to history |           fetch-depth: 0 # so that NerdBank.GitVersioning has access to history | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -128,9 +128,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -143,9 +143,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -156,9 +156,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -171,7 +171,7 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -184,7 +184,7 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -194,11 +194,11 @@ jobs: | |||||||
|   nuget-pack: |   nuget-pack: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v5 | ||||||
|       with: |       with: | ||||||
|         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history |         fetch-depth: 0 # so that NerdBank.GitVersioning has access to history | ||||||
|     - name: Install Nix |     - name: Install Nix | ||||||
|       uses: cachix/install-nix-action@v30 |       uses: cachix/install-nix-action@v31 | ||||||
|       with: |       with: | ||||||
|         extra_nix_config: | |         extra_nix_config: | | ||||||
|           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |           access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
| @@ -224,7 +224,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Download NuGet artifact (lib) |       - name: Download NuGet artifact (lib) | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: nuget-package-lib |           name: nuget-package-lib | ||||||
|           path: packed-lib |           path: packed-lib | ||||||
| @@ -232,7 +232,7 @@ jobs: | |||||||
|         # Verify that there is exactly one nupkg in the artifact that would be NuGet published |         # Verify that there is exactly one nupkg in the artifact that would be NuGet published | ||||||
|         run: if [[ $(find packed-lib -maxdepth 1 -name 'WoofWare.NUnitTestRunner.Lib.*.nupkg' -printf c | wc -c) -ne "1" ]]; then exit 1; fi |         run: if [[ $(find packed-lib -maxdepth 1 -name 'WoofWare.NUnitTestRunner.Lib.*.nupkg' -printf c | wc -c) -ne "1" ]]; then exit 1; fi | ||||||
|       - name: Download NuGet artifact (tool) |       - name: Download NuGet artifact (tool) | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: nuget-package-tool |           name: nuget-package-tool | ||||||
|           path: packed-tool |           path: packed-tool | ||||||
| @@ -259,12 +259,12 @@ jobs: | |||||||
|       contents: read |       contents: read | ||||||
|     steps: |     steps: | ||||||
|       - name: Download NuGet artifact |       - name: Download NuGet artifact | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: nuget-package-lib |           name: nuget-package-lib | ||||||
|           path: packed |           path: packed | ||||||
|       - name: Attest Build Provenance |       - name: Attest Build Provenance | ||||||
|         uses: actions/attest-build-provenance@520d128f165991a6c774bcb264f323e3d70747f4 # v2.2.0 |         uses: actions/attest-build-provenance@977bb373ede98d70efdf65b84cb5f73e068dcc2a # v3.0.0 | ||||||
|         with: |         with: | ||||||
|           subject-path: "packed/*.nupkg" |           subject-path: "packed/*.nupkg" | ||||||
|  |  | ||||||
| @@ -278,12 +278,12 @@ jobs: | |||||||
|       contents: read |       contents: read | ||||||
|     steps: |     steps: | ||||||
|       - name: Download NuGet artifact |       - name: Download NuGet artifact | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: nuget-package-tool |           name: nuget-package-tool | ||||||
|           path: packed |           path: packed | ||||||
|       - name: Attest Build Provenance |       - name: Attest Build Provenance | ||||||
|         uses: actions/attest-build-provenance@520d128f165991a6c774bcb264f323e3d70747f4 # v2.2.0 |         uses: actions/attest-build-provenance@977bb373ede98d70efdf65b84cb5f73e068dcc2a # v3.0.0 | ||||||
|         with: |         with: | ||||||
|           subject-path: "packed/*.nupkg" |           subject-path: "packed/*.nupkg" | ||||||
|  |  | ||||||
| @@ -297,25 +297,30 @@ jobs: | |||||||
|       attestations: write |       attestations: write | ||||||
|       contents: read |       contents: read | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v5 | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: Download NuGet artifact |       - name: Download NuGet artifact | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: nuget-package-lib |           name: nuget-package-lib | ||||||
|           path: packed |           path: packed | ||||||
|       - name: Identify .NET |       - name: Identify .NET | ||||||
|         id: identify-dotnet |         id: identify-dotnet | ||||||
|         run: nix develop --command bash -c "echo dotnet=$(which dotnet) >> $GITHUB_OUTPUT" |         run: nix develop --command bash -c "echo dotnet=$(which dotnet) >> $GITHUB_OUTPUT" | ||||||
|  |       - name: Obtain NuGet key | ||||||
|  |         uses: NuGet/login@d22cc5f58ff5b88bf9bd452535b4335137e24544 | ||||||
|  |         id: login | ||||||
|  |         with: | ||||||
|  |             user: ${{ secrets.NUGET_USER }} | ||||||
|       - name: Publish NuGet package |       - name: Publish NuGet package | ||||||
|         uses: G-Research/common-actions/publish-nuget@2b7dc49cb14f3344fbe6019c14a31165e258c059 |         uses: G-Research/common-actions/publish-nuget@2b7dc49cb14f3344fbe6019c14a31165e258c059 | ||||||
|         with: |         with: | ||||||
|           package-name: WoofWare.NUnitTestRunner.Lib |           package-name: WoofWare.NUnitTestRunner.Lib | ||||||
|           nuget-key: ${{ secrets.NUGET_API_KEY }} |           nuget-key: ${{ steps.login.outputs.NUGET_API_KEY }} | ||||||
|           nupkg-dir: packed/ |           nupkg-dir: packed/ | ||||||
|           dotnet: ${{ steps.identify-dotnet.outputs.dotnet }} |           dotnet: ${{ steps.identify-dotnet.outputs.dotnet }} | ||||||
|  |  | ||||||
| @@ -329,25 +334,30 @@ jobs: | |||||||
|       attestations: write |       attestations: write | ||||||
|       contents: read |       contents: read | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v5 | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: cachix/install-nix-action@v30 |         uses: cachix/install-nix-action@v31 | ||||||
|         with: |         with: | ||||||
|           extra_nix_config: | |           extra_nix_config: | | ||||||
|             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} |             access-tokens = github.com=${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: Download NuGet artifact |       - name: Download NuGet artifact | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: nuget-package-tool |           name: nuget-package-tool | ||||||
|           path: packed |           path: packed | ||||||
|       - name: Identify .NET |       - name: Identify .NET | ||||||
|         id: identify-dotnet |         id: identify-dotnet | ||||||
|         run: nix develop --command bash -c "echo dotnet=$(which dotnet) >> $GITHUB_OUTPUT" |         run: nix develop --command bash -c "echo dotnet=$(which dotnet) >> $GITHUB_OUTPUT" | ||||||
|  |       - name: Obtain NuGet key | ||||||
|  |         uses: NuGet/login@d22cc5f58ff5b88bf9bd452535b4335137e24544 | ||||||
|  |         id: login | ||||||
|  |         with: | ||||||
|  |             user: ${{ secrets.NUGET_USER }} | ||||||
|       - name: Publish NuGet package |       - name: Publish NuGet package | ||||||
|         uses: G-Research/common-actions/publish-nuget@2b7dc49cb14f3344fbe6019c14a31165e258c059 |         uses: G-Research/common-actions/publish-nuget@2b7dc49cb14f3344fbe6019c14a31165e258c059 | ||||||
|         with: |         with: | ||||||
|           package-name: WoofWare.NUnitTestRunner |           package-name: WoofWare.NUnitTestRunner | ||||||
|           nuget-key: ${{ secrets.NUGET_API_KEY }} |           nuget-key: ${{ steps.login.outputs.NUGET_API_KEY }} | ||||||
|           nupkg-dir: packed/ |           nupkg-dir: packed/ | ||||||
|           dotnet: ${{ steps.identify-dotnet.outputs.dotnet }} |           dotnet: ${{ steps.identify-dotnet.outputs.dotnet }} | ||||||
|  |  | ||||||
| @@ -360,9 +370,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     needs: [nuget-pack] |     needs: [nuget-pack] | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v5 | ||||||
|       - name: Download NuGet artifact (tool) |       - name: Download NuGet artifact (tool) | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: ${{ matrix.artifact }} |           name: ${{ matrix.artifact }} | ||||||
|       - name: Compute package path |       - name: Compute package path | ||||||
| @@ -396,9 +406,9 @@ jobs: | |||||||
|     permissions: |     permissions: | ||||||
|       contents: write |       contents: write | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v5 | ||||||
|       - name: Download NuGet artifact (tool) |       - name: Download NuGet artifact (tool) | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v5 | ||||||
|         with: |         with: | ||||||
|           name: ${{ matrix.artifact }} |           name: ${{ matrix.artifact }} | ||||||
|       - name: Compute package path |       - name: Compute package path | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/workflows/flake_update.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/flake_update.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out repository |       - name: Check out repository | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|  |  | ||||||
|       - name: Install Nix |       - name: Install Nix | ||||||
|         uses: DeterminateSystems/nix-installer-action@main |         uses: DeterminateSystems/nix-installer-action@main | ||||||
| @@ -32,7 +32,7 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Create token |       - name: Create token | ||||||
|         id: generate-token |         id: generate-token | ||||||
|         uses: actions/create-github-app-token@v1 |         uses: actions/create-github-app-token@v2 | ||||||
|         with: |         with: | ||||||
|           # https://github.com/actions/create-github-app-token/issues/136 |           # https://github.com/actions/create-github-app-token/issues/136 | ||||||
|           app-id: ${{ secrets.APP_ID }} |           app-id: ${{ secrets.APP_ID }} | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <Project Sdk="Microsoft.NET.Sdk"> | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |  | ||||||
|     <PropertyGroup> |     <PropertyGroup> | ||||||
|         <TargetFrameworks>net8.0</TargetFrameworks> |         <TargetFrameworks>net9.0</TargetFrameworks> | ||||||
|         <IsPackable>false</IsPackable> |         <IsPackable>false</IsPackable> | ||||||
|         <IsTestProject>true</IsTestProject> |         <IsTestProject>true</IsTestProject> | ||||||
|     </PropertyGroup> |     </PropertyGroup> | ||||||
| @@ -10,6 +10,8 @@ | |||||||
|         <Compile Include="NoAttribute.fs" /> |         <Compile Include="NoAttribute.fs" /> | ||||||
|         <Compile Include="Inconclusive.fs" /> |         <Compile Include="Inconclusive.fs" /> | ||||||
|         <Compile Include="RunSubProcess.fs" /> |         <Compile Include="RunSubProcess.fs" /> | ||||||
|  |         <Compile Include="TestAsync.fs" /> | ||||||
|  |         <Compile Include="TestExplicit.fs" /> | ||||||
|         <Compile Include="TestNonParallel.fs" /> |         <Compile Include="TestNonParallel.fs" /> | ||||||
|         <Compile Include="TestParallel.fs" /> |         <Compile Include="TestParallel.fs" /> | ||||||
|         <Compile Include="TestParallelIndividualTest.fs" /> |         <Compile Include="TestParallelIndividualTest.fs" /> | ||||||
| @@ -26,10 +28,10 @@ | |||||||
|     </ItemGroup> |     </ItemGroup> | ||||||
|  |  | ||||||
|     <ItemGroup> |     <ItemGroup> | ||||||
|       <PackageReference Include="FsUnit" Version="7.0.1" /> |       <PackageReference Include="FsUnit" Version="7.1.1" /> | ||||||
|       <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0"/> |       <PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.0"/> | ||||||
|       <PackageReference Include="NUnit" Version="4.3.2"/> |       <PackageReference Include="NUnit" Version="4.3.2"/> | ||||||
|       <PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/> |       <PackageReference Include="NUnit3TestAdapter" Version="5.2.0"/> | ||||||
|     </ItemGroup> |     </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								Consumer/TestAsync.fs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								Consumer/TestAsync.fs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | namespace Consumer | ||||||
|  |  | ||||||
|  | open System | ||||||
|  | open System.Threading.Tasks | ||||||
|  | open FsUnitTyped | ||||||
|  | open NUnit.Framework | ||||||
|  |  | ||||||
|  | [<TestFixture>] | ||||||
|  | module TestAsync = | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``an async test`` () = | ||||||
|  |         async { | ||||||
|  |             do! Async.Sleep (TimeSpan.FromMilliseconds 20.0) | ||||||
|  |             1 |> shouldEqual 1 | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``an async test, task-based`` () = | ||||||
|  |         task { | ||||||
|  |             do! Task.Delay (TimeSpan.FromMilliseconds 20.0) | ||||||
|  |             1 |> shouldEqual 1 | ||||||
|  |         } | ||||||
							
								
								
									
										24
									
								
								Consumer/TestExplicit.fs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								Consumer/TestExplicit.fs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | namespace Consumer | ||||||
|  |  | ||||||
|  | open NUnit.Framework | ||||||
|  |  | ||||||
|  | [<TestFixture>] | ||||||
|  | module TestExplicitIndividual = | ||||||
|  |  | ||||||
|  |     [<Explicit>] | ||||||
|  |     [<Test>] | ||||||
|  |     let ``This test should not be run`` () = failwith<unit> "should not call" | ||||||
|  |  | ||||||
|  | [<Explicit>] | ||||||
|  | [<TestFixture>] | ||||||
|  | module TestExplicitModule = | ||||||
|  |  | ||||||
|  |     [<OneTimeSetUp>] | ||||||
|  |     let setUp () = failwith<unit> "should not call: setup" | ||||||
|  |  | ||||||
|  |     [<OneTimeTearDown>] | ||||||
|  |     let tearDown () = | ||||||
|  |         failwith<unit> "should not call: teardown" | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``This test should not be run because its module is explicit`` () = failwith<unit> "should not call: test" | ||||||
| @@ -10,7 +10,7 @@ | |||||||
|     <WarnOn>FS3388,FS3559</WarnOn> |     <WarnOn>FS3388,FS3559</WarnOn> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="Nerdbank.GitVersioning" Version="3.7.115" PrivateAssets="all"/> |     <PackageReference Include="Nerdbank.GitVersioning" Version="3.8.118" PrivateAssets="all" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <PropertyGroup Condition="'$(GITHUB_ACTION)' != ''"> |   <PropertyGroup Condition="'$(GITHUB_ACTION)' != ''"> | ||||||
|     <ContinuousIntegrationBuild>true</ContinuousIntegrationBuild> |     <ContinuousIntegrationBuild>true</ContinuousIntegrationBuild> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| <Project Sdk="Microsoft.NET.Sdk"> | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |  | ||||||
|     <PropertyGroup> |     <PropertyGroup> | ||||||
|       <TargetFramework>net8.0</TargetFramework> |       <TargetFramework>net9.0</TargetFramework> | ||||||
|       <IsPackable>false</IsPackable> |       <IsPackable>false</IsPackable> | ||||||
|       <IsTestProject>true</IsTestProject> |       <IsTestProject>true</IsTestProject> | ||||||
|     </PropertyGroup> |     </PropertyGroup> | ||||||
| @@ -11,10 +11,10 @@ | |||||||
|     </ItemGroup> |     </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="FsUnit" Version="7.0.1" /> |     <PackageReference Include="FsUnit" Version="7.1.1" /> | ||||||
|     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0"/> |     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.0"/> | ||||||
|     <PackageReference Include="NUnit" Version="4.3.2"/> |     <PackageReference Include="NUnit" Version="4.3.2"/> | ||||||
|     <PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/> |     <PackageReference Include="NUnit3TestAdapter" Version="5.2.0"/> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
| @@ -13,3 +13,7 @@ However, we would recommend phrasing some of them differently, for maximum peace | |||||||
|  |  | ||||||
| WoofWare.NUnitTestRunner has *limited* support for parallelism. | WoofWare.NUnitTestRunner has *limited* support for parallelism. | ||||||
| By default, we run tests in parallel, taking half the available processors; we may or may not respect the NUnit parallelism attributes to any given extent that they tell us to be *more* parallel (but we will never incorrectly run tests in parallel). | By default, we run tests in parallel, taking half the available processors; we may or may not respect the NUnit parallelism attributes to any given extent that they tell us to be *more* parallel (but we will never incorrectly run tests in parallel). | ||||||
|  |  | ||||||
|  | # Licence | ||||||
|  |  | ||||||
|  | WoofWare.NUnitTestRunner is licensed to you under the MIT licence, a copy of which can be found at [LICENSE](./LICENSE). | ||||||
|   | |||||||
| @@ -10,20 +10,14 @@ open System.Threading | |||||||
|  |  | ||||||
| type internal OutputStreamId = | OutputStreamId of Guid | type internal OutputStreamId = | OutputStreamId of Guid | ||||||
|  |  | ||||||
| type private ThreadAwareWriter | type private ThreadAwareWriter (local : AsyncLocal<OutputStreamId>, underlying : Dictionary<OutputStreamId, TextWriter>) | ||||||
|     ( |  | ||||||
|         local : AsyncLocal<OutputStreamId>, |  | ||||||
|         underlying : Dictionary<OutputStreamId, TextWriter>, |  | ||||||
|         mem : Dictionary<OutputStreamId, MemoryStream> |  | ||||||
|     ) |  | ||||||
|     = |     = | ||||||
|     inherit TextWriter () |     inherit TextWriter () | ||||||
|     override _.get_Encoding () = Encoding.Default |     override _.get_Encoding () = Encoding.Default | ||||||
|  |  | ||||||
|     override this.Write (v : char) : unit = |     override this.Write (v : char) : unit = | ||||||
|         use prev = ExecutionContext.Capture () |         lock | ||||||
|  |             underlying | ||||||
|         (fun _ -> |  | ||||||
|             (fun () -> |             (fun () -> | ||||||
|                 match underlying.TryGetValue local.Value with |                 match underlying.TryGetValue local.Value with | ||||||
|                 | true, output -> output.Write v |                 | true, output -> output.Write v | ||||||
| @@ -31,16 +25,12 @@ type private ThreadAwareWriter | |||||||
|                     let wanted = |                     let wanted = | ||||||
|                         underlying |> Seq.map (fun (KeyValue (a, b)) -> $"%O{a}") |> String.concat "\n" |                         underlying |> Seq.map (fun (KeyValue (a, b)) -> $"%O{a}") |> String.concat "\n" | ||||||
|  |  | ||||||
|                     failwith $"no such context: %O{local.Value}\nwanted:\n" |                     failwith $"no such context: %O{local.Value}\nwanted:\n{wanted}" | ||||||
|             ) |             ) | ||||||
|             |> lock underlying |  | ||||||
|         ) |  | ||||||
|         |> fun action -> ExecutionContext.Run (prev, action, ()) |  | ||||||
|  |  | ||||||
|     override this.WriteLine (v : string) : unit = |     override this.WriteLine (v : string) : unit = | ||||||
|         use prev = ExecutionContext.Capture () |         lock | ||||||
|  |             underlying | ||||||
|         (fun _ -> |  | ||||||
|             (fun () -> |             (fun () -> | ||||||
|                 match underlying.TryGetValue local.Value with |                 match underlying.TryGetValue local.Value with | ||||||
|                 | true, output -> output.WriteLine v |                 | true, output -> output.WriteLine v | ||||||
| @@ -48,16 +38,13 @@ type private ThreadAwareWriter | |||||||
|                     let wanted = |                     let wanted = | ||||||
|                         underlying |> Seq.map (fun (KeyValue (a, b)) -> $"%O{a}") |> String.concat "\n" |                         underlying |> Seq.map (fun (KeyValue (a, b)) -> $"%O{a}") |> String.concat "\n" | ||||||
|  |  | ||||||
|                     failwith $"no such context: %O{local.Value}\nwanted:\n" |                     failwith $"no such context: %O{local.Value}\nwanted:\n{wanted}" | ||||||
|             ) |             ) | ||||||
|             |> lock underlying |  | ||||||
|         ) |  | ||||||
|         |> fun action -> ExecutionContext.Run (prev, action, ()) |  | ||||||
|  |  | ||||||
| /// Wraps up the necessary context to intercept global state. | /// Wraps up the necessary context to intercept global state. | ||||||
| [<NoEquality ; NoComparison>] | [<NoEquality ; NoComparison>] | ||||||
| type TestContexts = | type TestContexts = | ||||||
|     private |     internal | ||||||
|         { |         { | ||||||
|             /// Accesses to this must be locked on StdOutWriters. |             /// Accesses to this must be locked on StdOutWriters. | ||||||
|             StdOuts : Dictionary<OutputStreamId, MemoryStream> |             StdOuts : Dictionary<OutputStreamId, MemoryStream> | ||||||
| @@ -77,8 +64,8 @@ type TestContexts = | |||||||
|         let stdoutWriters = Dictionary () |         let stdoutWriters = Dictionary () | ||||||
|         let stderrWriters = Dictionary () |         let stderrWriters = Dictionary () | ||||||
|         let local = AsyncLocal () |         let local = AsyncLocal () | ||||||
|         let stdoutWriter = new ThreadAwareWriter (local, stdoutWriters, stdouts) |         let stdoutWriter = new ThreadAwareWriter (local, stdoutWriters) | ||||||
|         let stderrWriter = new ThreadAwareWriter (local, stderrWriters, stderrs) |         let stderrWriter = new ThreadAwareWriter (local, stderrWriters) | ||||||
|  |  | ||||||
|         { |         { | ||||||
|             StdOuts = stdouts |             StdOuts = stdouts | ||||||
|   | |||||||
| @@ -138,10 +138,17 @@ type TestFixture = | |||||||
|         Tests : SingleTestMethod list |         Tests : SingleTestMethod list | ||||||
|         /// If this fixture has declared a parallelisability, that goes here. |         /// If this fixture has declared a parallelisability, that goes here. | ||||||
|         Parallelize : Parallelizable<ClassParallelScope> option |         Parallelize : Parallelizable<ClassParallelScope> option | ||||||
|  |         /// It is possible to mark a fixture as "Explicit" or "Ignored", for example. | ||||||
|  |         Modifiers : Modifier list | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// A test fixture about which we know nothing. No tests, no setup/teardown. |     /// A test fixture about which we know nothing. No tests, no setup/teardown. | ||||||
|     static member Empty (ty : Type) (par : Parallelizable<ClassParallelScope> option) (args : obj list list) = |     static member Empty | ||||||
|  |         (ty : Type) | ||||||
|  |         (par : Parallelizable<ClassParallelScope> option) | ||||||
|  |         (modifiers : Modifier list) | ||||||
|  |         (args : obj list list) | ||||||
|  |         = | ||||||
|         { |         { | ||||||
|             ContainingAssembly = ty.Assembly |             ContainingAssembly = ty.Assembly | ||||||
|             Type = ty |             Type = ty | ||||||
| @@ -153,6 +160,7 @@ type TestFixture = | |||||||
|             Parameters = args |             Parameters = args | ||||||
|             Tests = [] |             Tests = [] | ||||||
|             Parallelize = par |             Parallelize = par | ||||||
|  |             Modifiers = modifiers | ||||||
|         } |         } | ||||||
|  |  | ||||||
| /// User code in the unit under test has failed somehow. | /// User code in the unit under test has failed somehow. | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								WoofWare.NUnitTestRunner.Lib/Exception.fs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								WoofWare.NUnitTestRunner.Lib/Exception.fs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | namespace WoofWare.NUnitTestRunner | ||||||
|  |  | ||||||
|  | open System.Runtime.ExceptionServices | ||||||
|  |  | ||||||
|  | [<RequireQualifiedAccess>] | ||||||
|  | module internal Exception = | ||||||
|  |     let reraiseWithOriginalStackTrace<'a> (e : exn) : 'a = | ||||||
|  |         let edi = ExceptionDispatchInfo.Capture e | ||||||
|  |         edi.Throw () | ||||||
|  |         failwith "unreachable" | ||||||
| @@ -4,16 +4,16 @@ open System | |||||||
| open System.Threading | open System.Threading | ||||||
| open System.Threading.Tasks | open System.Threading.Tasks | ||||||
|  |  | ||||||
| type private ThunkEvaluator<'ret> = | type private AsyncThunkEvaluator<'ret> = | ||||||
|     abstract Eval<'a> : (unit -> 'a) -> AsyncReplyChannel<'a> -> 'ret |     abstract Eval<'a> : (unit -> Async<'a>) -> AsyncReplyChannel<Result<'a, exn>> -> 'ret | ||||||
|  |  | ||||||
| type private ThunkCrate = | type private AsyncThunkCrate = | ||||||
|     abstract Apply<'ret> : ThunkEvaluator<'ret> -> 'ret |     abstract Apply<'ret> : AsyncThunkEvaluator<'ret> -> 'ret | ||||||
|  |  | ||||||
| [<RequireQualifiedAccess>] | [<RequireQualifiedAccess>] | ||||||
| module private ThunkCrate = | module private AsyncThunkCrate = | ||||||
|     let make<'a> (t : unit -> 'a) (rc : AsyncReplyChannel<'a>) : ThunkCrate = |     let make<'a> (t : unit -> Async<'a>) (rc : AsyncReplyChannel<Result<'a, exn>>) : AsyncThunkCrate = | ||||||
|         { new ThunkCrate with |         { new AsyncThunkCrate with | ||||||
|             member _.Apply e = e.Eval t rc |             member _.Apply e = e.Eval t rc | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -41,7 +41,11 @@ type private MailboxMessage = | |||||||
|     | Quit of AsyncReplyChannel<unit> |     | Quit of AsyncReplyChannel<unit> | ||||||
|     /// Check current state, see if we need to start more tests, etc. |     /// Check current state, see if we need to start more tests, etc. | ||||||
|     | Reconcile |     | Reconcile | ||||||
|     | RunTest of within : TestFixture * Parallelizable<unit> option * test : ThunkCrate |     | RunTestAsync of | ||||||
|  |         within : TestFixture * | ||||||
|  |         Parallelizable<unit> option * | ||||||
|  |         test : AsyncThunkCrate * | ||||||
|  |         context : ExecutionContext | ||||||
|     | BeginTestFixture of TestFixture * AsyncReplyChannel<TestFixtureRunningToken> |     | BeginTestFixture of TestFixture * AsyncReplyChannel<TestFixtureRunningToken> | ||||||
|     | EndTestFixture of TestFixtureTearDownToken * AsyncReplyChannel<unit> |     | EndTestFixture of TestFixtureTearDownToken * AsyncReplyChannel<unit> | ||||||
|  |  | ||||||
| @@ -310,21 +314,31 @@ type ParallelQueue | |||||||
|                     rc.Reply () |                     rc.Reply () | ||||||
|                     m.Post MailboxMessage.Reconcile |                     m.Post MailboxMessage.Reconcile | ||||||
|                     return! processTask (Running state) m |                     return! processTask (Running state) m | ||||||
|             | MailboxMessage.RunTest (withinFixture, par, message) -> |             | MailboxMessage.RunTestAsync (withinFixture, par, message, capturedContext) -> | ||||||
|                 let t () = |                 let t () = | ||||||
|                     { new ThunkEvaluator<_> with |                     { new AsyncThunkEvaluator<_> with | ||||||
|                         member _.Eval<'b> (t : unit -> 'b) rc = |                         member _.Eval<'b> (t : unit -> Async<'b>) rc = | ||||||
|                             let tcs = TaskCompletionSource TaskCreationOptions.RunContinuationsAsynchronously |                             let tcs = TaskCompletionSource TaskCreationOptions.RunContinuationsAsynchronously | ||||||
|                             use ec = ExecutionContext.Capture () |  | ||||||
|  |  | ||||||
|                             fun () -> |                             fun () -> | ||||||
|                                 ExecutionContext.Run ( |                                 ExecutionContext.Run ( | ||||||
|                                     ec, |                                     capturedContext, | ||||||
|                                     (fun _ -> |                                     (fun _ -> | ||||||
|                                         let result = t () |                                         async { | ||||||
|                                         tcs.SetResult () |                                             let! result = | ||||||
|                                         m.Post MailboxMessage.Reconcile |                                                 async { | ||||||
|                                         rc.Reply result |                                                     try | ||||||
|  |                                                         let! r = t () | ||||||
|  |                                                         return Ok r | ||||||
|  |                                                     with e -> | ||||||
|  |                                                         return Error e | ||||||
|  |                                                 } | ||||||
|  |  | ||||||
|  |                                             tcs.SetResult () | ||||||
|  |                                             m.Post MailboxMessage.Reconcile | ||||||
|  |                                             rc.Reply result | ||||||
|  |                                         } | ||||||
|  |                                         |> Async.StartImmediate | ||||||
|                                     ), |                                     ), | ||||||
|                                     () |                                     () | ||||||
|                                 ) |                                 ) | ||||||
| @@ -348,17 +362,36 @@ type ParallelQueue | |||||||
|     let mb = new MailboxProcessor<_> (processTask MailboxState.Idle) |     let mb = new MailboxProcessor<_> (processTask MailboxState.Idle) | ||||||
|     do mb.Start () |     do mb.Start () | ||||||
|  |  | ||||||
|  |     /// Request to run the given async action, freely in parallel with other running tests. | ||||||
|  |     /// The resulting Task will return when the action has completed. | ||||||
|  |     member _.RunAsync<'a> | ||||||
|  |         (TestFixtureSetupToken parent) | ||||||
|  |         (scope : Parallelizable<unit> option) | ||||||
|  |         (action : unit -> Async<'a>) | ||||||
|  |         : 'a Task | ||||||
|  |         = | ||||||
|  |         let ec = ExecutionContext.Capture () | ||||||
|  |  | ||||||
|  |         task { | ||||||
|  |             let! result = | ||||||
|  |                 (fun rc -> MailboxMessage.RunTestAsync (parent, scope, AsyncThunkCrate.make action rc, ec)) | ||||||
|  |                 |> mb.PostAndAsyncReply | ||||||
|  |                 |> Async.StartAsTask | ||||||
|  |  | ||||||
|  |             match result with | ||||||
|  |             | Ok o -> return o | ||||||
|  |             | Error e -> return Exception.reraiseWithOriginalStackTrace e | ||||||
|  |         } | ||||||
|  |  | ||||||
|     /// Request to run the given action, freely in parallel with other running tests. |     /// Request to run the given action, freely in parallel with other running tests. | ||||||
|     /// The resulting Task will return when the action has completed. |     /// The resulting Task will return when the action has completed. | ||||||
|     member _.Run<'a> |     member this.Run<'a> | ||||||
|         (TestFixtureSetupToken parent) |         (parent : TestFixtureSetupToken) | ||||||
|         (scope : Parallelizable<unit> option) |         (scope : Parallelizable<unit> option) | ||||||
|         (action : unit -> 'a) |         (action : unit -> 'a) | ||||||
|         : 'a Task |         : 'a Task | ||||||
|         = |         = | ||||||
|         (fun rc -> MailboxMessage.RunTest (parent, scope, ThunkCrate.make action rc)) |         this.RunAsync parent scope (fun () -> async.Return (action ())) | ||||||
|         |> mb.PostAndAsyncReply |  | ||||||
|         |> Async.StartAsTask |  | ||||||
|  |  | ||||||
|     /// Declare that we wish to start the given test fixture. The resulting Task will return |     /// Declare that we wish to start the given test fixture. The resulting Task will return | ||||||
|     /// when you are allowed to start running tests from that fixture. |     /// when you are allowed to start running tests from that fixture. | ||||||
| @@ -379,11 +412,22 @@ type ParallelQueue | |||||||
|                     | Parallelizable.Yes _ -> Parallelizable.Yes () |                     | Parallelizable.Yes _ -> Parallelizable.Yes () | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|  |             let ec = ExecutionContext.Capture () | ||||||
|  |  | ||||||
|             let! response = |             let! response = | ||||||
|                 (fun rc -> MailboxMessage.RunTest (parent, par, ThunkCrate.make action rc)) |                 (fun rc -> | ||||||
|  |                     MailboxMessage.RunTestAsync ( | ||||||
|  |                         parent, | ||||||
|  |                         par, | ||||||
|  |                         AsyncThunkCrate.make (fun () -> async.Return (action ())) rc, | ||||||
|  |                         ec | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|                 |> mb.PostAndAsyncReply |                 |> mb.PostAndAsyncReply | ||||||
|  |  | ||||||
|             return response, TestFixtureSetupToken parent |             match response with | ||||||
|  |             | Ok response -> return response, TestFixtureSetupToken parent | ||||||
|  |             | Error e -> return Exception.reraiseWithOriginalStackTrace e | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     /// Run the given one-time tear-down for the test fixture. |     /// Run the given one-time tear-down for the test fixture. | ||||||
| @@ -401,11 +445,22 @@ type ParallelQueue | |||||||
|                     | Parallelizable.Yes _ -> Parallelizable.Yes () |                     | Parallelizable.Yes _ -> Parallelizable.Yes () | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|  |             let ec = ExecutionContext.Capture () | ||||||
|  |  | ||||||
|             let! response = |             let! response = | ||||||
|                 (fun rc -> MailboxMessage.RunTest (parent, par, ThunkCrate.make action rc)) |                 (fun rc -> | ||||||
|  |                     MailboxMessage.RunTestAsync ( | ||||||
|  |                         parent, | ||||||
|  |                         par, | ||||||
|  |                         AsyncThunkCrate.make (fun () -> async.Return (action ())) rc, | ||||||
|  |                         ec | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|                 |> mb.PostAndAsyncReply |                 |> mb.PostAndAsyncReply | ||||||
|  |  | ||||||
|             return response, TestFixtureTearDownToken parent |             match response with | ||||||
|  |             | Ok response -> return response, TestFixtureTearDownToken parent | ||||||
|  |             | Error e -> return Exception.reraiseWithOriginalStackTrace e | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     /// Declare that we have finished submitting requests to run in the given test fixture. |     /// Declare that we have finished submitting requests to run in the given test fixture. | ||||||
|   | |||||||
| @@ -1,16 +1,15 @@ | |||||||
| namespace WoofWare.NUnitTestRunner | namespace WoofWare.NUnitTestRunner | ||||||
|  |  | ||||||
| open System | open System | ||||||
| open WoofWare.Myriad.Plugins |  | ||||||
|  |  | ||||||
| [<JsonParse>] | // Myriad runs the JsonParse generator on this | ||||||
| type internal FrameworkDescription = | type internal FrameworkDescription = | ||||||
|     { |     { | ||||||
|         Name : string |         Name : string | ||||||
|         Version : string |         Version : string | ||||||
|     } |     } | ||||||
|  |  | ||||||
| [<JsonParse>] | // Myriad runs the JsonParse generator on this | ||||||
| type internal RuntimeOptions = | type internal RuntimeOptions = | ||||||
|     { |     { | ||||||
|         Tfm : string |         Tfm : string | ||||||
| @@ -21,7 +20,7 @@ type internal RuntimeOptions = | |||||||
|         RollForward : string option |         RollForward : string option | ||||||
|     } |     } | ||||||
|  |  | ||||||
| [<JsonParse>] | // Myriad runs the JsonParse generator on this | ||||||
| type internal RuntimeConfig = | type internal RuntimeConfig = | ||||||
|     { |     { | ||||||
|         RuntimeOptions : RuntimeOptions |         RuntimeOptions : RuntimeOptions | ||||||
|   | |||||||
| @@ -170,8 +170,9 @@ WoofWare.NUnitTestRunner.IndividualTestRunMetadata.StdOut [property]: [read-only | |||||||
| WoofWare.NUnitTestRunner.IndividualTestRunMetadata.TestId [property]: [read-only] System.Guid | WoofWare.NUnitTestRunner.IndividualTestRunMetadata.TestId [property]: [read-only] System.Guid | ||||||
| WoofWare.NUnitTestRunner.IndividualTestRunMetadata.TestName [property]: [read-only] string | WoofWare.NUnitTestRunner.IndividualTestRunMetadata.TestName [property]: [read-only] string | ||||||
| WoofWare.NUnitTestRunner.IndividualTestRunMetadata.Total [property]: [read-only] System.TimeSpan | WoofWare.NUnitTestRunner.IndividualTestRunMetadata.Total [property]: [read-only] System.TimeSpan | ||||||
| WoofWare.NUnitTestRunner.ITestProgress - interface with 5 member(s) | WoofWare.NUnitTestRunner.ITestProgress - interface with 6 member(s) | ||||||
| WoofWare.NUnitTestRunner.ITestProgress.OnTestFailed [method]: string -> WoofWare.NUnitTestRunner.TestMemberFailure -> unit | WoofWare.NUnitTestRunner.ITestProgress.OnTestFailed [method]: string -> WoofWare.NUnitTestRunner.TestMemberFailure -> unit | ||||||
|  | WoofWare.NUnitTestRunner.ITestProgress.OnTestFixtureSkipped [method]: string -> string -> unit | ||||||
| WoofWare.NUnitTestRunner.ITestProgress.OnTestFixtureStart [method]: string -> int -> unit | WoofWare.NUnitTestRunner.ITestProgress.OnTestFixtureStart [method]: string -> int -> unit | ||||||
| WoofWare.NUnitTestRunner.ITestProgress.OnTestMemberFinished [method]: string -> unit | WoofWare.NUnitTestRunner.ITestProgress.OnTestMemberFinished [method]: string -> unit | ||||||
| WoofWare.NUnitTestRunner.ITestProgress.OnTestMemberSkipped [method]: string -> unit | WoofWare.NUnitTestRunner.ITestProgress.OnTestMemberSkipped [method]: string -> unit | ||||||
| @@ -255,6 +256,7 @@ WoofWare.NUnitTestRunner.ParallelQueue inherit obj, implements IDisposable | |||||||
| WoofWare.NUnitTestRunner.ParallelQueue..ctor [constructor]: (int option, WoofWare.NUnitTestRunner.AssemblyParallelScope WoofWare.NUnitTestRunner.Parallelizable option, System.Threading.CancellationToken option) | WoofWare.NUnitTestRunner.ParallelQueue..ctor [constructor]: (int option, WoofWare.NUnitTestRunner.AssemblyParallelScope WoofWare.NUnitTestRunner.Parallelizable option, System.Threading.CancellationToken option) | ||||||
| WoofWare.NUnitTestRunner.ParallelQueue.EndTestFixture [method]: WoofWare.NUnitTestRunner.TestFixtureTearDownToken -> unit System.Threading.Tasks.Task | WoofWare.NUnitTestRunner.ParallelQueue.EndTestFixture [method]: WoofWare.NUnitTestRunner.TestFixtureTearDownToken -> unit System.Threading.Tasks.Task | ||||||
| WoofWare.NUnitTestRunner.ParallelQueue.Run [method]: WoofWare.NUnitTestRunner.TestFixtureSetupToken -> unit WoofWare.NUnitTestRunner.Parallelizable option -> (unit -> 'a) -> 'a System.Threading.Tasks.Task | WoofWare.NUnitTestRunner.ParallelQueue.Run [method]: WoofWare.NUnitTestRunner.TestFixtureSetupToken -> unit WoofWare.NUnitTestRunner.Parallelizable option -> (unit -> 'a) -> 'a System.Threading.Tasks.Task | ||||||
|  | WoofWare.NUnitTestRunner.ParallelQueue.RunAsync [method]: WoofWare.NUnitTestRunner.TestFixtureSetupToken -> unit WoofWare.NUnitTestRunner.Parallelizable option -> (unit -> 'a Microsoft.FSharp.Control.FSharpAsync) -> 'a System.Threading.Tasks.Task | ||||||
| WoofWare.NUnitTestRunner.ParallelQueue.RunTestSetup [method]: WoofWare.NUnitTestRunner.TestFixtureRunningToken -> (unit -> 'a) -> ('a * WoofWare.NUnitTestRunner.TestFixtureSetupToken) System.Threading.Tasks.Task | WoofWare.NUnitTestRunner.ParallelQueue.RunTestSetup [method]: WoofWare.NUnitTestRunner.TestFixtureRunningToken -> (unit -> 'a) -> ('a * WoofWare.NUnitTestRunner.TestFixtureSetupToken) System.Threading.Tasks.Task | ||||||
| WoofWare.NUnitTestRunner.ParallelQueue.RunTestTearDown [method]: WoofWare.NUnitTestRunner.TestFixtureSetupToken -> (unit -> 'a) -> ('a * WoofWare.NUnitTestRunner.TestFixtureTearDownToken) System.Threading.Tasks.Task | WoofWare.NUnitTestRunner.ParallelQueue.RunTestTearDown [method]: WoofWare.NUnitTestRunner.TestFixtureSetupToken -> (unit -> 'a) -> ('a * WoofWare.NUnitTestRunner.TestFixtureTearDownToken) System.Threading.Tasks.Task | ||||||
| WoofWare.NUnitTestRunner.ParallelQueue.StartTestFixture [method]: WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.TestFixtureRunningToken System.Threading.Tasks.Task | WoofWare.NUnitTestRunner.ParallelQueue.StartTestFixture [method]: WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.TestFixtureRunningToken System.Threading.Tasks.Task | ||||||
| @@ -346,11 +348,12 @@ WoofWare.NUnitTestRunner.TestFailure.NewTearDownFailed [static method]: WoofWare | |||||||
| WoofWare.NUnitTestRunner.TestFailure.NewTestFailed [static method]: WoofWare.NUnitTestRunner.UserMethodFailure -> WoofWare.NUnitTestRunner.TestFailure | WoofWare.NUnitTestRunner.TestFailure.NewTestFailed [static method]: WoofWare.NUnitTestRunner.UserMethodFailure -> WoofWare.NUnitTestRunner.TestFailure | ||||||
| WoofWare.NUnitTestRunner.TestFailure.Tag [property]: [read-only] int | WoofWare.NUnitTestRunner.TestFailure.Tag [property]: [read-only] int | ||||||
| WoofWare.NUnitTestRunner.TestFixture inherit obj, implements WoofWare.NUnitTestRunner.TestFixture System.IEquatable, System.Collections.IStructuralEquatable | WoofWare.NUnitTestRunner.TestFixture inherit obj, implements WoofWare.NUnitTestRunner.TestFixture System.IEquatable, System.Collections.IStructuralEquatable | ||||||
| WoofWare.NUnitTestRunner.TestFixture..ctor [constructor]: (System.Reflection.Assembly, string, System.Type, System.Reflection.MethodInfo option, System.Reflection.MethodInfo option, System.Reflection.MethodInfo list, System.Reflection.MethodInfo list, obj list list, WoofWare.NUnitTestRunner.SingleTestMethod list, WoofWare.NUnitTestRunner.ClassParallelScope WoofWare.NUnitTestRunner.Parallelizable option) | WoofWare.NUnitTestRunner.TestFixture..ctor [constructor]: (System.Reflection.Assembly, string, System.Type, System.Reflection.MethodInfo option, System.Reflection.MethodInfo option, System.Reflection.MethodInfo list, System.Reflection.MethodInfo list, obj list list, WoofWare.NUnitTestRunner.SingleTestMethod list, WoofWare.NUnitTestRunner.ClassParallelScope WoofWare.NUnitTestRunner.Parallelizable option, WoofWare.NUnitTestRunner.Modifier list) | ||||||
| WoofWare.NUnitTestRunner.TestFixture.ContainingAssembly [property]: [read-only] System.Reflection.Assembly | WoofWare.NUnitTestRunner.TestFixture.ContainingAssembly [property]: [read-only] System.Reflection.Assembly | ||||||
| WoofWare.NUnitTestRunner.TestFixture.Empty [static method]: System.Type -> WoofWare.NUnitTestRunner.ClassParallelScope WoofWare.NUnitTestRunner.Parallelizable option -> obj list list -> WoofWare.NUnitTestRunner.TestFixture | WoofWare.NUnitTestRunner.TestFixture.Empty [static method]: System.Type -> WoofWare.NUnitTestRunner.ClassParallelScope WoofWare.NUnitTestRunner.Parallelizable option -> WoofWare.NUnitTestRunner.Modifier list -> obj list list -> WoofWare.NUnitTestRunner.TestFixture | ||||||
| WoofWare.NUnitTestRunner.TestFixture.Equals [method]: (WoofWare.NUnitTestRunner.TestFixture, System.Collections.IEqualityComparer) -> bool | WoofWare.NUnitTestRunner.TestFixture.Equals [method]: (WoofWare.NUnitTestRunner.TestFixture, System.Collections.IEqualityComparer) -> bool | ||||||
| WoofWare.NUnitTestRunner.TestFixture.get_ContainingAssembly [method]: unit -> System.Reflection.Assembly | WoofWare.NUnitTestRunner.TestFixture.get_ContainingAssembly [method]: unit -> System.Reflection.Assembly | ||||||
|  | WoofWare.NUnitTestRunner.TestFixture.get_Modifiers [method]: unit -> WoofWare.NUnitTestRunner.Modifier list | ||||||
| WoofWare.NUnitTestRunner.TestFixture.get_Name [method]: unit -> string | WoofWare.NUnitTestRunner.TestFixture.get_Name [method]: unit -> string | ||||||
| WoofWare.NUnitTestRunner.TestFixture.get_OneTimeSetUp [method]: unit -> System.Reflection.MethodInfo option | WoofWare.NUnitTestRunner.TestFixture.get_OneTimeSetUp [method]: unit -> System.Reflection.MethodInfo option | ||||||
| WoofWare.NUnitTestRunner.TestFixture.get_OneTimeTearDown [method]: unit -> System.Reflection.MethodInfo option | WoofWare.NUnitTestRunner.TestFixture.get_OneTimeTearDown [method]: unit -> System.Reflection.MethodInfo option | ||||||
| @@ -360,6 +363,7 @@ WoofWare.NUnitTestRunner.TestFixture.get_SetUp [method]: unit -> System.Reflecti | |||||||
| WoofWare.NUnitTestRunner.TestFixture.get_TearDown [method]: unit -> System.Reflection.MethodInfo list | WoofWare.NUnitTestRunner.TestFixture.get_TearDown [method]: unit -> System.Reflection.MethodInfo list | ||||||
| WoofWare.NUnitTestRunner.TestFixture.get_Tests [method]: unit -> WoofWare.NUnitTestRunner.SingleTestMethod list | WoofWare.NUnitTestRunner.TestFixture.get_Tests [method]: unit -> WoofWare.NUnitTestRunner.SingleTestMethod list | ||||||
| WoofWare.NUnitTestRunner.TestFixture.get_Type [method]: unit -> System.Type | WoofWare.NUnitTestRunner.TestFixture.get_Type [method]: unit -> System.Type | ||||||
|  | WoofWare.NUnitTestRunner.TestFixture.Modifiers [property]: [read-only] WoofWare.NUnitTestRunner.Modifier list | ||||||
| WoofWare.NUnitTestRunner.TestFixture.Name [property]: [read-only] string | WoofWare.NUnitTestRunner.TestFixture.Name [property]: [read-only] string | ||||||
| WoofWare.NUnitTestRunner.TestFixture.OneTimeSetUp [property]: [read-only] System.Reflection.MethodInfo option | WoofWare.NUnitTestRunner.TestFixture.OneTimeSetUp [property]: [read-only] System.Reflection.MethodInfo option | ||||||
| WoofWare.NUnitTestRunner.TestFixture.OneTimeTearDown [property]: [read-only] System.Reflection.MethodInfo option | WoofWare.NUnitTestRunner.TestFixture.OneTimeTearDown [property]: [read-only] System.Reflection.MethodInfo option | ||||||
|   | |||||||
| @@ -76,105 +76,168 @@ module TestFixture = | |||||||
|         (test : MethodInfo) |         (test : MethodInfo) | ||||||
|         (containingObject : obj) |         (containingObject : obj) | ||||||
|         (args : obj[]) |         (args : obj[]) | ||||||
|         : Result<TestMemberSuccess, TestFailure list> * IndividualTestRunMetadata |         : Async<Result<TestMemberSuccess, TestFailure list> * IndividualTestRunMetadata> | ||||||
|         = |         = | ||||||
|         let rec runMethods |         let rec runMethods | ||||||
|             (wrap : UserMethodFailure -> TestFailure) |             (wrap : UserMethodFailure -> TestFailure) | ||||||
|             (toRun : MethodInfo list) |             (toRun : MethodInfo list) | ||||||
|             (args : obj[]) |             (args : obj[]) | ||||||
|             : Result<unit, _> |             : Result<unit, TestFailure> Async | ||||||
|             = |             = | ||||||
|             match toRun with |             match toRun with | ||||||
|             | [] -> Ok () |             | [] -> async.Return (Ok ()) | ||||||
|             | head :: rest -> |             | head :: rest -> | ||||||
|                 let result = |                 async { | ||||||
|                     try |                     let result = | ||||||
|                         head.Invoke (containingObject, args) |> Ok |                         try | ||||||
|                     with |                             head.Invoke (containingObject, args) |> Ok | ||||||
|                     | :? TargetInvocationException as e -> Error (UserMethodFailure.Threw (head.Name, e.InnerException)) |                         with | ||||||
|                     | :? TargetParameterCountException -> |                         | :? TargetInvocationException as e -> | ||||||
|                         UserMethodFailure.BadParameters ( |                             Error (UserMethodFailure.Threw (head.Name, e.InnerException)) | ||||||
|                             head.Name, |                         | :? TargetParameterCountException -> | ||||||
|                             head.GetParameters () |> Array.map (fun pm -> pm.ParameterType), |                             UserMethodFailure.BadParameters ( | ||||||
|                             args |                                 head.Name, | ||||||
|                         ) |                                 head.GetParameters () |> Array.map (fun pm -> pm.ParameterType), | ||||||
|                         |> Error |                                 args | ||||||
|  |                             ) | ||||||
|  |                             |> Error | ||||||
|  |  | ||||||
|                 match result with |                     let! ct = Async.CancellationToken | ||||||
|                 | Error e -> Error (wrap e) |  | ||||||
|                 | Ok result -> |  | ||||||
|                     match result with |  | ||||||
|                     | :? unit -> runMethods wrap rest args |  | ||||||
|                     | ret -> UserMethodFailure.ReturnedNonUnit (head.Name, ret) |> wrap |> Error |  | ||||||
|  |  | ||||||
|         let start = DateTimeOffset.Now |                     let! result = | ||||||
|  |                         match result with | ||||||
|  |                         | Error e -> async.Return (Error (wrap e)) | ||||||
|  |                         | Ok result -> | ||||||
|  |                             match result with | ||||||
|  |                             | :? unit -> runMethods wrap rest args | ||||||
|  |                             | :? Task as result -> | ||||||
|  |                                 async { | ||||||
|  |                                     let mutable exc = None | ||||||
|  |  | ||||||
|         let sw = Stopwatch.StartNew () |                                     try | ||||||
|  |                                         do! Async.AwaitTask result | ||||||
|  |                                     with e -> | ||||||
|  |                                         exc <- Some e | ||||||
|  |  | ||||||
|         let metadata () = |                                     match exc with | ||||||
|             let name = |                                     | None -> return! runMethods wrap rest args | ||||||
|                 if args.Length = 0 then |                                     | Some e -> return Error (UserMethodFailure.Threw (head.Name, e) |> wrap) | ||||||
|                     test.Name |                                 } | ||||||
|                 else |                             // We'd like to do this type-test: | ||||||
|                     let argsStr = args |> Seq.map string<obj> |> String.concat "," |                             // | :? Async<unit> as result -> | ||||||
|                     $"%s{test.Name}(%s{argsStr})" |                             // but instead we have to do all this reflective nonsense, because FSharpAsync is not part | ||||||
|  |                             // of the .NET runtime, so is instead in a different AssemblyLoadContext to us! | ||||||
|  |                             // It's in the user-code context, not ours. | ||||||
|  |                             | ret -> | ||||||
|  |                                 let ty = ret.GetType () | ||||||
|  |  | ||||||
|             { |                                 if ty.Namespace = "Microsoft.FSharp.Control" && ty.Name = "FSharpAsync`1" then | ||||||
|                 End = DateTimeOffset.Now |                                     match ty.GenericTypeArguments |> Array.map (fun t -> t.FullName) with | ||||||
|                 Start = start |                                     | [| "Microsoft.FSharp.Core.Unit" |] -> | ||||||
|                 Total = sw.Elapsed |                                         let asyncModule = ty.Assembly.GetType ("Microsoft.FSharp.Control.FSharpAsync") | ||||||
|                 ComputerName = Environment.MachineName |                                         // let catch = asyncModule.GetMethod("Catch").MakeGenericMethod [| ty.GenericTypeArguments.[0] |] | ||||||
|                 ExecutionId = Guid.NewGuid () |                                         // let caught = catch.Invoke ((null: obj), [| ret |]) | ||||||
|                 TestId = testId |                                         let startAsTask = | ||||||
|                 TestName = name |                                             asyncModule.GetMethod("StartAsTask").MakeGenericMethod | ||||||
|                 ClassName = test.DeclaringType.FullName |                                                 [| ty.GenericTypeArguments.[0] |] | ||||||
|                 StdOut = |  | ||||||
|                     match contexts.DumpStdout outputId with |  | ||||||
|                     | "" -> None |  | ||||||
|                     | v -> Some v |  | ||||||
|                 StdErr = |  | ||||||
|                     match contexts.DumpStderr outputId with |  | ||||||
|                     | "" -> None |  | ||||||
|                     | v -> Some v |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|         let setUpResult = runMethods TestFailure.SetUpFailed setUp [||] |                                         let started = | ||||||
|         sw.Stop () |                                             startAsTask.Invoke ((null : obj), [| ret ; (null : obj) ; (null : obj) |]) | ||||||
|  |                                             |> unbox<Task> | ||||||
|  |  | ||||||
|         match setUpResult with |                                         async { | ||||||
|         | Error e -> Error [ e ], metadata () |                                             let! res = Async.AwaitTask started |> Async.Catch | ||||||
|         | Ok () -> |  | ||||||
|  |  | ||||||
|         sw.Start () |                                             match res with | ||||||
|  |                                             | Choice1Of2 () -> return! runMethods wrap rest args | ||||||
|  |                                             | Choice2Of2 e -> | ||||||
|  |                                                 return | ||||||
|  |                                                     Error ( | ||||||
|  |                                                         UserMethodFailure.Threw (head.Name, started.Exception) |> wrap | ||||||
|  |                                                     ) | ||||||
|  |                                         } | ||||||
|  |                                     | _ -> | ||||||
|  |                                         UserMethodFailure.ReturnedNonUnit (head.Name, ret) | ||||||
|  |                                         |> wrap | ||||||
|  |                                         |> Error | ||||||
|  |                                         |> async.Return | ||||||
|  |                                 else | ||||||
|  |                                     async.Return (UserMethodFailure.ReturnedNonUnit (head.Name, ret) |> wrap |> Error) | ||||||
|  |  | ||||||
|         let result = |                     return result | ||||||
|             let result = runMethods TestFailure.TestFailed [ test ] args |                 } | ||||||
|  |  | ||||||
|  |         async { | ||||||
|  |             let start = DateTimeOffset.Now | ||||||
|  |  | ||||||
|  |             let sw = Stopwatch.StartNew () | ||||||
|  |  | ||||||
|  |             let metadata () = | ||||||
|  |                 let name = | ||||||
|  |                     if args.Length = 0 then | ||||||
|  |                         test.Name | ||||||
|  |                     else | ||||||
|  |                         let argsStr = args |> Seq.map string<obj> |> String.concat "," | ||||||
|  |                         $"%s{test.Name}(%s{argsStr})" | ||||||
|  |  | ||||||
|  |                 { | ||||||
|  |                     End = DateTimeOffset.Now | ||||||
|  |                     Start = start | ||||||
|  |                     Total = sw.Elapsed | ||||||
|  |                     ComputerName = Environment.MachineName | ||||||
|  |                     ExecutionId = Guid.NewGuid () | ||||||
|  |                     TestId = testId | ||||||
|  |                     TestName = name | ||||||
|  |                     ClassName = test.DeclaringType.FullName | ||||||
|  |                     StdOut = | ||||||
|  |                         match contexts.DumpStdout outputId with | ||||||
|  |                         | "" -> None | ||||||
|  |                         | v -> Some v | ||||||
|  |                     StdErr = | ||||||
|  |                         match contexts.DumpStderr outputId with | ||||||
|  |                         | "" -> None | ||||||
|  |                         | v -> Some v | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |             let! setUpResult = runMethods TestFailure.SetUpFailed setUp [||] | ||||||
|             sw.Stop () |             sw.Stop () | ||||||
|  |  | ||||||
|             match result with |             match setUpResult with | ||||||
|             | Ok () -> Ok None |             | Error e -> return Error [ e ], metadata () | ||||||
|             | Error (TestFailure.TestFailed (UserMethodFailure.Threw (_, exc)) as orig) -> |             | Ok () -> | ||||||
|                 match exc.GetType().FullName with |  | ||||||
|                 | "NUnit.Framework.SuccessException" -> Ok None |  | ||||||
|                 | "NUnit.Framework.IgnoreException" -> Ok (Some (TestMemberSuccess.Ignored (Option.ofObj exc.Message))) |  | ||||||
|                 | "NUnit.Framework.InconclusiveException" -> |  | ||||||
|                     Ok (Some (TestMemberSuccess.Inconclusive (Option.ofObj exc.Message))) |  | ||||||
|                 | _ -> Error orig |  | ||||||
|             | Error orig -> Error orig |  | ||||||
|  |  | ||||||
|         // Unconditionally run TearDown after tests, even if tests failed. |             sw.Start () | ||||||
|         sw.Start () |             let! result = runMethods TestFailure.TestFailed [ test ] args | ||||||
|         let tearDownResult = runMethods TestFailure.TearDownFailed tearDown [||] |             sw.Stop () | ||||||
|         sw.Stop () |  | ||||||
|  |  | ||||||
|         let metadata = metadata () |             let result = | ||||||
|  |                 match result with | ||||||
|  |                 | Ok () -> Ok None | ||||||
|  |                 | Error (TestFailure.TestFailed (UserMethodFailure.Threw (_, exc)) as orig) -> | ||||||
|  |                     match exc.GetType().FullName with | ||||||
|  |                     | "NUnit.Framework.SuccessException" -> Ok None | ||||||
|  |                     | "NUnit.Framework.IgnoreException" -> | ||||||
|  |                         Ok (Some (TestMemberSuccess.Ignored (Option.ofObj exc.Message))) | ||||||
|  |                     | "NUnit.Framework.InconclusiveException" -> | ||||||
|  |                         Ok (Some (TestMemberSuccess.Inconclusive (Option.ofObj exc.Message))) | ||||||
|  |                     | _ -> Error orig | ||||||
|  |                 | Error orig -> Error orig | ||||||
|  |  | ||||||
|         match result, tearDownResult with |             // Unconditionally run TearDown after tests, even if tests failed. | ||||||
|         | Ok None, Ok () -> Ok TestMemberSuccess.Ok, metadata |             sw.Start () | ||||||
|         | Ok (Some s), Ok () -> Ok s, metadata |             let! tearDownResult = runMethods TestFailure.TearDownFailed tearDown [||] | ||||||
|         | Error e, Ok () |             sw.Stop () | ||||||
|         | Ok _, Error e -> Error [ e ], metadata |  | ||||||
|         | Error e1, Error e2 -> Error [ e1 ; e2 ], metadata |             let metadata = metadata () | ||||||
|  |  | ||||||
|  |             return | ||||||
|  |                 match result, tearDownResult with | ||||||
|  |                 | Ok None, Ok () -> Ok TestMemberSuccess.Ok, metadata | ||||||
|  |                 | Ok (Some s), Ok () -> Ok s, metadata | ||||||
|  |                 | Error e, Ok () | ||||||
|  |                 | Ok _, Error e -> Error [ e ], metadata | ||||||
|  |                 | Error e1, Error e2 -> Error [ e1 ; e2 ], metadata | ||||||
|  |         } | ||||||
|  |  | ||||||
|     let private getValues (test : SingleTestMethod) = |     let private getValues (test : SingleTestMethod) = | ||||||
|         let valuesAttrs = |         let valuesAttrs = | ||||||
| @@ -395,20 +458,22 @@ module TestFixture = | |||||||
|         |> Seq.map (fun (testGuid, args) -> |         |> Seq.map (fun (testGuid, args) -> | ||||||
|             task { |             task { | ||||||
|                 let runMe () = |                 let runMe () = | ||||||
|                     progress.OnTestMemberStart test.Name |                     async { | ||||||
|                     let oldValue = contexts.AsyncLocal.Value |                         progress.OnTestMemberStart test.Name | ||||||
|                     let outputId = contexts.NewOutputs () |                         let oldValue = contexts.AsyncLocal.Value | ||||||
|                     contexts.AsyncLocal.Value <- outputId |                         let outputId = contexts.NewOutputs () | ||||||
|  |                         contexts.AsyncLocal.Value <- outputId | ||||||
|  |  | ||||||
|                     let result, meta = |                         let! result, meta = | ||||||
|                         runOne outputId contexts setUp tearDown testGuid test.Method containingObject args |                             runOne outputId contexts setUp tearDown testGuid test.Method containingObject args | ||||||
|  |  | ||||||
|                     contexts.AsyncLocal.Value <- oldValue |                         contexts.AsyncLocal.Value <- oldValue | ||||||
|                     progress.OnTestMemberFinished test.Name |                         progress.OnTestMemberFinished test.Name | ||||||
|  |  | ||||||
|                     result, meta |                         return result, meta | ||||||
|  |                     } | ||||||
|  |  | ||||||
|                 let! results, summary = par.Run running test.Parallelize runMe |                 let! results, summary = par.RunAsync running test.Parallelize runMe | ||||||
|  |  | ||||||
|                 match results with |                 match results with | ||||||
|                 | Ok results -> return Ok results, summary |                 | Ok results -> return Ok results, summary | ||||||
| @@ -597,15 +662,15 @@ module TestFixture = | |||||||
|     /// Interpret this type as a [<TestFixture>], extracting the test members from it and annotating them with all |     /// Interpret this type as a [<TestFixture>], extracting the test members from it and annotating them with all | ||||||
|     /// relevant information about how we should run them. |     /// relevant information about how we should run them. | ||||||
|     let parse (parentType : Type) : TestFixture = |     let parse (parentType : Type) : TestFixture = | ||||||
|         let categories, args, par = |         let categories, args, mods, par = | ||||||
|             (([], [], None), parentType.CustomAttributes) |             (([], [], [], None), parentType.CustomAttributes) | ||||||
|             ||> Seq.fold (fun (categories, args, par) attr -> |             ||> Seq.fold (fun (categories, args, mods, par) attr -> | ||||||
|                 match attr.AttributeType.FullName with |                 match attr.AttributeType.FullName with | ||||||
|                 | "NUnit.Framework.SetUpFixtureAttribute" -> |                 | "NUnit.Framework.SetUpFixtureAttribute" -> | ||||||
|                     failwith "This test runner does not support SetUpFixture. Please shout if you want this." |                     failwith "This test runner does not support SetUpFixture. Please shout if you want this." | ||||||
|                 | "NUnit.Framework.CategoryAttribute" -> |                 | "NUnit.Framework.CategoryAttribute" -> | ||||||
|                     let cat = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string> |                     let cat = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string> | ||||||
|                     cat :: categories, args, par |                     cat :: categories, args, mods, par | ||||||
|                 | "NUnit.Framework.TestFixtureAttribute" -> |                 | "NUnit.Framework.TestFixtureAttribute" -> | ||||||
|                     let newArgs = |                     let newArgs = | ||||||
|                         match attr.ConstructorArguments |> Seq.map _.Value |> Seq.toList with |                         match attr.ConstructorArguments |> Seq.map _.Value |> Seq.toList with | ||||||
| @@ -613,38 +678,52 @@ module TestFixture = | |||||||
|                             x |> Seq.cast<CustomAttributeTypedArgument> |> Seq.map _.Value |> Seq.toList |                             x |> Seq.cast<CustomAttributeTypedArgument> |> Seq.map _.Value |> Seq.toList | ||||||
|                         | xs -> xs |                         | xs -> xs | ||||||
|  |  | ||||||
|                     categories, newArgs :: args, par |                     categories, newArgs :: args, mods, par | ||||||
|                 | "NUnit.Framework.NonParallelizableAttribute" -> |                 | "NUnit.Framework.NonParallelizableAttribute" -> | ||||||
|                     match par with |                     match par with | ||||||
|                     | Some _ -> failwith $"Got multiple parallelism attributes on %s{parentType.FullName}" |                     | Some _ -> failwith $"Got multiple parallelism attributes on %s{parentType.FullName}" | ||||||
|                     | None -> categories, args, Some Parallelizable.No |                     | None -> categories, args, mods, Some Parallelizable.No | ||||||
|                 | "NUnit.Framework.ParallelizableAttribute" -> |                 | "NUnit.Framework.ParallelizableAttribute" -> | ||||||
|                     match par with |                     match par with | ||||||
|                     | Some _ -> failwith $"Got multiple parallelism attributes on %s{parentType.FullName}" |                     | Some _ -> failwith $"Got multiple parallelism attributes on %s{parentType.FullName}" | ||||||
|                     | None -> |                     | None -> | ||||||
|                         match attr.ConstructorArguments |> Seq.toList with |                         match attr.ConstructorArguments |> Seq.toList with | ||||||
|                         | [] -> categories, args, Some (Parallelizable.Yes ClassParallelScope.Self) |                         | [] -> categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Self) | ||||||
|                         | [ v ] -> |                         | [ v ] -> | ||||||
|                             match v.Value with |                             match v.Value with | ||||||
|                             | :? int as v -> |                             | :? int as v -> | ||||||
|                                 match ParallelScope.ofInt v with |                                 match ParallelScope.ofInt v with | ||||||
|                                 | ParallelScope.Fixtures -> |                                 | ParallelScope.Fixtures -> | ||||||
|                                     categories, args, Some (Parallelizable.Yes ClassParallelScope.Fixtures) |                                     categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Fixtures) | ||||||
|                                 | ParallelScope.Children -> |                                 | ParallelScope.Children -> | ||||||
|                                     categories, args, Some (Parallelizable.Yes ClassParallelScope.Children) |                                     categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Children) | ||||||
|                                 | ParallelScope.All -> |                                 | ParallelScope.All -> | ||||||
|                                     categories, args, Some (Parallelizable.Yes ClassParallelScope.All) |                                     categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.All) | ||||||
|                                 | ParallelScope.Self -> |                                 | ParallelScope.Self -> | ||||||
|                                     categories, args, Some (Parallelizable.Yes ClassParallelScope.Self) |                                     categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Self) | ||||||
|                                 | ParallelScope.None -> categories, args, Some Parallelizable.No |                                 | ParallelScope.None -> categories, args, mods, Some Parallelizable.No | ||||||
|                             | v -> |                             | v -> | ||||||
|                                 failwith |                                 failwith | ||||||
|                                     $"Unexpectedly non-int value %O{v} of parallel scope in %s{parentType.FullName}" |                                     $"Unexpectedly non-int value %O{v} of parallel scope in %s{parentType.FullName}" | ||||||
|                         | _ -> failwith $"unexpectedly got multiple args to Parallelizable on %s{parentType.FullName}" |                         | _ -> failwith $"unexpectedly got multiple args to Parallelizable on %s{parentType.FullName}" | ||||||
|                 | _ -> categories, args, par |                 | "NUnit.Framework.ExplicitAttribute" -> | ||||||
|  |                     let reason = | ||||||
|  |                         attr.ConstructorArguments | ||||||
|  |                         |> Seq.tryHead | ||||||
|  |                         |> Option.map (_.Value >> unbox<string>) | ||||||
|  |  | ||||||
|  |                     categories, args, Modifier.Explicit reason :: mods, par | ||||||
|  |                 | "NUnit.Framework.IgnoreAttribute" -> | ||||||
|  |                     let reason = | ||||||
|  |                         attr.ConstructorArguments | ||||||
|  |                         |> Seq.tryHead | ||||||
|  |                         |> Option.map (_.Value >> unbox<string>) | ||||||
|  |  | ||||||
|  |                     categories, args, Modifier.Ignored reason :: mods, par | ||||||
|  |                 | _ -> categories, args, mods, par | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         (TestFixture.Empty parentType par args, parentType.GetRuntimeMethods ()) |         (TestFixture.Empty parentType par mods args, parentType.GetRuntimeMethods ()) | ||||||
|         ||> Seq.fold (fun state mi -> |         ||> Seq.fold (fun state mi -> | ||||||
|             ((state, []), mi.CustomAttributes) |             ((state, []), mi.CustomAttributes) | ||||||
|             ||> Seq.fold (fun (state, unrecognisedAttrs) attr -> |             ||> Seq.fold (fun (state, unrecognisedAttrs) attr -> | ||||||
| @@ -717,6 +796,8 @@ module TestFixture = | |||||||
|  |  | ||||||
|     /// Run every test (except those which fail the `filter`) in this test fixture, as well as the |     /// Run every test (except those which fail the `filter`) in this test fixture, as well as the | ||||||
|     /// appropriate setup and tear-down logic. |     /// appropriate setup and tear-down logic. | ||||||
|  |     /// | ||||||
|  |     /// If the TestFixture has modifiers that specify no tests should be run, we don't run any tests. | ||||||
|     let run |     let run | ||||||
|         (contexts : TestContexts) |         (contexts : TestContexts) | ||||||
|         (par : ParallelQueue) |         (par : ParallelQueue) | ||||||
| @@ -725,6 +806,26 @@ module TestFixture = | |||||||
|         (tests : TestFixture) |         (tests : TestFixture) | ||||||
|         : FixtureRunResults list Task |         : FixtureRunResults list Task | ||||||
|         = |         = | ||||||
|  |         match | ||||||
|  |             tests.Modifiers | ||||||
|  |             |> List.tryFind ( | ||||||
|  |                 function | ||||||
|  |                 | Modifier.Explicit _ | ||||||
|  |                 | Modifier.Ignored _ -> true | ||||||
|  |             ) | ||||||
|  |         with | ||||||
|  |         | Some modifier -> | ||||||
|  |             let reason = | ||||||
|  |                 match modifier with | ||||||
|  |                 | Modifier.Explicit (Some reason) -> reason | ||||||
|  |                 | Modifier.Ignored (Some reason) -> reason | ||||||
|  |                 | Modifier.Ignored None -> "test fixture marked Ignore" | ||||||
|  |                 | Modifier.Explicit None -> "test fixture marked Explicit" | ||||||
|  |  | ||||||
|  |             progress.OnTestFixtureSkipped tests.Name reason | ||||||
|  |             Task.FromResult [] | ||||||
|  |         | None -> | ||||||
|  |  | ||||||
|         match tests.Parameters with |         match tests.Parameters with | ||||||
|         | [] -> [ null ] |         | [] -> [ null ] | ||||||
|         | args -> args |> List.map List.toArray |         | args -> args |> List.map List.toArray | ||||||
|   | |||||||
| @@ -10,6 +10,8 @@ type ITestProgress = | |||||||
|     /// Called just before we start executing the setup logic for the given test fixture. |     /// Called just before we start executing the setup logic for the given test fixture. | ||||||
|     /// We tell you how many test methods there are in the fixture. |     /// We tell you how many test methods there are in the fixture. | ||||||
|     abstract OnTestFixtureStart : name : string -> testCount : int -> unit |     abstract OnTestFixtureStart : name : string -> testCount : int -> unit | ||||||
|  |     /// Called when skipping the test fixture with the given name, e.g. because it's `[<Explicit>]`. | ||||||
|  |     abstract OnTestFixtureSkipped : name : string -> reason : string -> unit | ||||||
|     /// Called just before we start executing the test(s) indicated by a particular method. |     /// Called just before we start executing the test(s) indicated by a particular method. | ||||||
|     abstract OnTestMemberStart : name : string -> unit |     abstract OnTestMemberStart : name : string -> unit | ||||||
|     /// Called when a test fails. (This may be called repeatedly with the same `name`, e.g. if the test |     /// Called when a test fails. (This may be called repeatedly with the same `name`, e.g. if the test | ||||||
| @@ -32,6 +34,9 @@ module TestProgress = | |||||||
|                 let plural = if testCount = 1 then "" else "s" |                 let plural = if testCount = 1 then "" else "s" | ||||||
|                 writer.WriteLine $"Running test fixture: %s{name} (%i{testCount} test%s{plural} to run)" |                 writer.WriteLine $"Running test fixture: %s{name} (%i{testCount} test%s{plural} to run)" | ||||||
|  |  | ||||||
|  |             member _.OnTestFixtureSkipped name reason = | ||||||
|  |                 writer.WriteLine $"Skipping test fixture (%s{reason}): %s{name}" | ||||||
|  |  | ||||||
|             member _.OnTestMemberStart name = |             member _.OnTestMemberStart name = | ||||||
|                 writer.WriteLine $"Running test: %s{name}" |                 writer.WriteLine $"Running test: %s{name}" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ | |||||||
|       <PackageId>WoofWare.NUnitTestRunner.Lib</PackageId> |       <PackageId>WoofWare.NUnitTestRunner.Lib</PackageId> | ||||||
|       <TreatWarningsAsErrors>true</TreatWarningsAsErrors> |       <TreatWarningsAsErrors>true</TreatWarningsAsErrors> | ||||||
|       <WarnOn>FS3559</WarnOn> |       <WarnOn>FS3559</WarnOn> | ||||||
|       <WoofWareMyriadPluginVersion>4.0.12</WoofWareMyriadPluginVersion> |       <WoofWareMyriadPluginVersion>9.0.4</WoofWareMyriadPluginVersion> | ||||||
|     </PropertyGroup> |     </PropertyGroup> | ||||||
|  |  | ||||||
|     <ItemGroup> |     <ItemGroup> | ||||||
| @@ -22,10 +22,16 @@ | |||||||
|       <Compile Include="RuntimeConfig.fs" /> |       <Compile Include="RuntimeConfig.fs" /> | ||||||
|       <Compile Include="GeneratedRuntimeConfig.fs"> |       <Compile Include="GeneratedRuntimeConfig.fs"> | ||||||
|         <MyriadFile>RuntimeConfig.fs</MyriadFile> |         <MyriadFile>RuntimeConfig.fs</MyriadFile> | ||||||
|  |         <MyriadParams> | ||||||
|  |           <RuntimeOptions>JsonParse</RuntimeOptions> | ||||||
|  |           <RuntimeConfig>JsonParse</RuntimeConfig> | ||||||
|  |           <FrameworkDescription>JsonParse</FrameworkDescription> | ||||||
|  |         </MyriadParams> | ||||||
|       </Compile> |       </Compile> | ||||||
|       <Compile Include="ParallelScope.fs" /> |       <Compile Include="ParallelScope.fs" /> | ||||||
|       <Compile Include="DotnetRuntime.fs" /> |       <Compile Include="DotnetRuntime.fs" /> | ||||||
|       <Compile Include="Array.fs" /> |       <Compile Include="Array.fs" /> | ||||||
|  |       <Compile Include="Exception.fs" /> | ||||||
|       <Compile Include="List.fs" /> |       <Compile Include="List.fs" /> | ||||||
|       <Compile Include="Result.fs" /> |       <Compile Include="Result.fs" /> | ||||||
|       <Compile Include="Domain.fs" /> |       <Compile Include="Domain.fs" /> | ||||||
| @@ -48,10 +54,9 @@ | |||||||
|       <EmbeddedResource Include="version.json" /> |       <EmbeddedResource Include="version.json" /> | ||||||
|     </ItemGroup> |     </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="WoofWare.PrattParser" Version="0.2.3" /> |     <PackageReference Include="WoofWare.PrattParser" Version="0.2.5" /> | ||||||
|     <PackageReference Update="FSharp.Core" Version="6.0.1" /> |     <PackageReference Update="FSharp.Core" Version="6.0.1" /> | ||||||
|     <PackageReference Include="WoofWare.DotnetRuntimeLocator" Version="0.1.11" /> |     <PackageReference Include="WoofWare.DotnetRuntimeLocator" Version="0.1.12" /> | ||||||
|     <PackageReference Include="WoofWare.Myriad.Plugins.Attributes" Version="3.6.7" /> |  | ||||||
|     <PackageReference Include="Myriad.SDK" Version="0.8.3" PrivateAssets="all" /> |     <PackageReference Include="Myriad.SDK" Version="0.8.3" PrivateAssets="all" /> | ||||||
|     <PackageReference Include="WoofWare.Myriad.Plugins" Version="$(WoofWareMyriadPluginVersion)" PrivateAssets="all" /> |     <PackageReference Include="WoofWare.Myriad.Plugins" Version="$(WoofWareMyriadPluginVersion)" PrivateAssets="all" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| { | { | ||||||
|   "version": "0.20", |   "version": "0.22", | ||||||
|   "publicReleaseRefSpec": [ |   "publicReleaseRefSpec": [ | ||||||
|     "^refs/heads/main$" |     "^refs/heads/main$" | ||||||
|   ], |   ], | ||||||
|   | |||||||
| @@ -13,6 +13,10 @@ module Progress = | |||||||
|             member _.OnTestFixtureStart name testCount = |             member _.OnTestFixtureStart name testCount = | ||||||
|                 console.MarkupLine $"[white]Running tests: %s{Markup.Escape name}[/]" |                 console.MarkupLine $"[white]Running tests: %s{Markup.Escape name}[/]" | ||||||
|  |  | ||||||
|  |             member _.OnTestFixtureSkipped name reason = | ||||||
|  |                 console.MarkupLine | ||||||
|  |                     $"[yellow]Skipping test fixture (%s{Markup.Escape reason}): %s{Markup.Escape name}[/]" | ||||||
|  |  | ||||||
|             member _.OnTestMemberFinished name = |             member _.OnTestMemberFinished name = | ||||||
|                 console.MarkupLine $"[gray]Finished test: %s{Markup.Escape name}[/]" |                 console.MarkupLine $"[gray]Finished test: %s{Markup.Escape name}[/]" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,396 @@ | |||||||
|  | namespace WoofWare.NUnitTestRunner.Test | ||||||
|  |  | ||||||
|  | open System | ||||||
|  | open System.Text | ||||||
|  | open System.Threading | ||||||
|  | open System.Threading.Tasks | ||||||
|  | open NUnit.Framework | ||||||
|  | open FsUnitTyped | ||||||
|  | open WoofWare.NUnitTestRunner | ||||||
|  |  | ||||||
|  | [<TestFixture>] | ||||||
|  | module TestSynchronizationContext = | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``ExecutionContext flows correctly through synchronous operations`` () = | ||||||
|  |         task { | ||||||
|  |             let dummyFixture = | ||||||
|  |                 TestFixture.Empty typeof<obj> (Some (Parallelizable.Yes ClassParallelScope.All)) [] [] | ||||||
|  |  | ||||||
|  |             use contexts = TestContexts.Empty () | ||||||
|  |             use queue = new ParallelQueue (Some 4, None) | ||||||
|  |  | ||||||
|  |             // Track which context values we see during execution | ||||||
|  |             let contextValues = System.Collections.Concurrent.ConcurrentBag<Guid * Guid> () | ||||||
|  |  | ||||||
|  |             // Start the fixture | ||||||
|  |             let! running = queue.StartTestFixture dummyFixture | ||||||
|  |             let! _, setup = queue.RunTestSetup running (fun () -> ()) | ||||||
|  |  | ||||||
|  |             // Create several synchronous operations with different context values | ||||||
|  |             let tasks = | ||||||
|  |                 [ 1..10 ] | ||||||
|  |                 |> List.map (fun _ -> | ||||||
|  |                     task { | ||||||
|  |                         do! Task.Yield () | ||||||
|  |                         // Set a unique context value | ||||||
|  |                         let outputId = contexts.NewOutputs () | ||||||
|  |                         let (OutputStreamId expectedId) = outputId | ||||||
|  |                         contexts.AsyncLocal.Value <- outputId | ||||||
|  |  | ||||||
|  |                         // Run a synchronous operation that checks the context | ||||||
|  |                         let! actualId = | ||||||
|  |                             queue.Run | ||||||
|  |                                 setup | ||||||
|  |                                 None | ||||||
|  |                                 (fun () -> | ||||||
|  |                                     // Check context immediately | ||||||
|  |                                     let immediate = contexts.AsyncLocal.Value | ||||||
|  |                                     let (OutputStreamId immediateGuid) = immediate | ||||||
|  |                                     contextValues.Add (expectedId, immediateGuid) | ||||||
|  |  | ||||||
|  |                                     // Do some work that might cause context issues | ||||||
|  |                                     Thread.Sleep 10 | ||||||
|  |  | ||||||
|  |                                     // Check context after work | ||||||
|  |                                     let afterWork = contexts.AsyncLocal.Value | ||||||
|  |                                     let (OutputStreamId afterWorkGuid) = afterWork | ||||||
|  |                                     contextValues.Add (expectedId, afterWorkGuid) | ||||||
|  |  | ||||||
|  |                                     // Simulate calling into framework code that might use ExecutionContext | ||||||
|  |                                     let mutable capturedValue = Guid.Empty | ||||||
|  |  | ||||||
|  |                                     ExecutionContext.Run ( | ||||||
|  |                                         ExecutionContext.Capture (), | ||||||
|  |                                         (fun _ -> | ||||||
|  |                                             let current = contexts.AsyncLocal.Value | ||||||
|  |                                             let (OutputStreamId currentGuid) = current | ||||||
|  |                                             capturedValue <- currentGuid | ||||||
|  |                                         ), | ||||||
|  |                                         () | ||||||
|  |                                     ) | ||||||
|  |  | ||||||
|  |                                     contextValues.Add (expectedId, capturedValue) | ||||||
|  |  | ||||||
|  |                                     afterWorkGuid | ||||||
|  |                                 ) | ||||||
|  |  | ||||||
|  |                         // Verify the returned value matches what we set | ||||||
|  |                         actualId |> shouldEqual expectedId | ||||||
|  |                     } | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             // Wait for all tasks | ||||||
|  |             let! results = Task.WhenAll tasks | ||||||
|  |             results |> Array.iter id | ||||||
|  |  | ||||||
|  |             // Verify all context values were correct | ||||||
|  |             let allValues = contextValues |> Seq.toList | ||||||
|  |             allValues |> shouldHaveLength 30 // 3 checks per operation * 10 operations | ||||||
|  |  | ||||||
|  |             // Every captured value should match its expected value | ||||||
|  |             allValues | ||||||
|  |             |> List.iter (fun (expected, actual) -> actual |> shouldEqual expected) | ||||||
|  |  | ||||||
|  |             // Clean up | ||||||
|  |             let! _, teardown = queue.RunTestTearDown setup (fun () -> ()) | ||||||
|  |             do! queue.EndTestFixture teardown | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``ExecutionContext isolation between concurrent synchronous operations`` () = | ||||||
|  |         task { | ||||||
|  |             let dummyFixture = | ||||||
|  |                 TestFixture.Empty typeof<obj> (Some (Parallelizable.Yes ClassParallelScope.All)) [] [] | ||||||
|  |  | ||||||
|  |             use contexts = TestContexts.Empty () | ||||||
|  |             use queue = new ParallelQueue (Some 4, None) | ||||||
|  |  | ||||||
|  |             let! running = queue.StartTestFixture dummyFixture | ||||||
|  |             let! _, setup = queue.RunTestSetup running (fun () -> ()) | ||||||
|  |  | ||||||
|  |             // Use a barrier to ensure operations run concurrently | ||||||
|  |             let barrier = new Barrier (3) | ||||||
|  |             let seenValues = System.Collections.Concurrent.ConcurrentBag<int * Guid> () | ||||||
|  |             let outputIds = System.Collections.Concurrent.ConcurrentBag<OutputStreamId> () | ||||||
|  |  | ||||||
|  |             // Create operations that will definitely run concurrently | ||||||
|  |             let tasks = | ||||||
|  |                 [ 1..3 ] | ||||||
|  |                 |> List.map (fun i -> | ||||||
|  |                     task { | ||||||
|  |                         // Each task sets its own context value | ||||||
|  |                         let outputId = contexts.NewOutputs () | ||||||
|  |                         let (OutputStreamId myId) = outputId | ||||||
|  |                         contexts.AsyncLocal.Value <- outputId | ||||||
|  |                         outputIds.Add outputId | ||||||
|  |  | ||||||
|  |                         let! result = | ||||||
|  |                             queue.Run | ||||||
|  |                                 setup | ||||||
|  |                                 (Some (Parallelizable.Yes ())) | ||||||
|  |                                 (fun () -> | ||||||
|  |                                     // Wait for all tasks to reach this point | ||||||
|  |                                     barrier.SignalAndWait () | ||||||
|  |  | ||||||
|  |                                     // Now check what value we see | ||||||
|  |                                     let currentValue = contexts.AsyncLocal.Value | ||||||
|  |  | ||||||
|  |                                     match currentValue with | ||||||
|  |                                     | OutputStreamId guid -> seenValues.Add (i, guid) | ||||||
|  |  | ||||||
|  |                                     // Do some synchronous work | ||||||
|  |                                     Thread.Sleep 5 | ||||||
|  |  | ||||||
|  |                                     // Check again after work | ||||||
|  |                                     let afterWork = contexts.AsyncLocal.Value | ||||||
|  |  | ||||||
|  |                                     match afterWork with | ||||||
|  |                                     | OutputStreamId guid -> | ||||||
|  |                                         // Also verify we can write to the correct streams | ||||||
|  |                                         contexts.Stdout.WriteLine $"Task %i{i} sees context %O{guid}" | ||||||
|  |                                         guid | ||||||
|  |                                 ) | ||||||
|  |  | ||||||
|  |                         // Each task should see its own value | ||||||
|  |                         result |> shouldEqual myId | ||||||
|  |                     } | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             let! results = Task.WhenAll tasks | ||||||
|  |             results |> Array.iter id | ||||||
|  |  | ||||||
|  |             // Verify we saw 3 different values (one per task) | ||||||
|  |             let values = seenValues |> Seq.toList | ||||||
|  |             values |> shouldHaveLength 3 | ||||||
|  |  | ||||||
|  |             // All seen values should be different (no context bleeding) | ||||||
|  |             let uniqueValues = values |> List.map snd |> List.distinct | ||||||
|  |             uniqueValues |> shouldHaveLength 3 | ||||||
|  |  | ||||||
|  |             let! _, teardown = queue.RunTestTearDown setup (fun () -> ()) | ||||||
|  |             do! queue.EndTestFixture teardown | ||||||
|  |  | ||||||
|  |             // Verify stdout content for each task | ||||||
|  |             let collectedOutputs = outputIds |> Seq.toList | ||||||
|  |             collectedOutputs |> shouldHaveLength 3 | ||||||
|  |  | ||||||
|  |             for outputId in collectedOutputs do | ||||||
|  |                 let content = contexts.DumpStdout outputId | ||||||
|  |                 content |> shouldNotEqual "" | ||||||
|  |                 let (OutputStreamId guid) = outputId | ||||||
|  |                 content |> shouldContainText (guid.ToString ()) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``ExecutionContext flows correctly through nested synchronous operations`` () = | ||||||
|  |         task { | ||||||
|  |             let dummyFixture = | ||||||
|  |                 TestFixture.Empty typeof<obj> (Some (Parallelizable.Yes ClassParallelScope.All)) [] [] | ||||||
|  |  | ||||||
|  |             use contexts = TestContexts.Empty () | ||||||
|  |             use queue = new ParallelQueue (Some 4, None) | ||||||
|  |  | ||||||
|  |             let! running = queue.StartTestFixture dummyFixture | ||||||
|  |             let! _, setup = queue.RunTestSetup running (fun () -> ()) | ||||||
|  |  | ||||||
|  |             // Set an initial context | ||||||
|  |             let outputId = contexts.NewOutputs () | ||||||
|  |             let (OutputStreamId outerGuid) = outputId | ||||||
|  |             contexts.AsyncLocal.Value <- outputId | ||||||
|  |  | ||||||
|  |             let! result = | ||||||
|  |                 queue.Run | ||||||
|  |                     setup | ||||||
|  |                     None | ||||||
|  |                     (fun () -> | ||||||
|  |                         // Check we have the outer context | ||||||
|  |                         let outer = contexts.AsyncLocal.Value | ||||||
|  |                         let (OutputStreamId outerSeen) = outer | ||||||
|  |                         outerSeen |> shouldEqual outerGuid | ||||||
|  |  | ||||||
|  |                         // Now change the context for a nested operation | ||||||
|  |                         let innerOutputId = contexts.NewOutputs () | ||||||
|  |                         let (OutputStreamId innerGuid) = innerOutputId | ||||||
|  |                         contexts.AsyncLocal.Value <- innerOutputId | ||||||
|  |  | ||||||
|  |                         // Use Task.Run to potentially hop threads | ||||||
|  |                         let innerResult = | ||||||
|  |                             Task | ||||||
|  |                                 .Run(fun () -> | ||||||
|  |                                     let inner = contexts.AsyncLocal.Value | ||||||
|  |                                     let (OutputStreamId innerSeen) = inner | ||||||
|  |                                     innerSeen |> shouldEqual innerGuid | ||||||
|  |                                     innerSeen | ||||||
|  |                                 ) | ||||||
|  |                                 .Result | ||||||
|  |  | ||||||
|  |                         // After the nested operation, we should still have our inner context | ||||||
|  |                         let afterNested = contexts.AsyncLocal.Value | ||||||
|  |                         let (OutputStreamId afterNestedGuid) = afterNested | ||||||
|  |                         afterNestedGuid |> shouldEqual innerGuid | ||||||
|  |  | ||||||
|  |                         (outerSeen, innerResult, afterNestedGuid) | ||||||
|  |                     ) | ||||||
|  |  | ||||||
|  |             // Unpack results | ||||||
|  |             let seenOuter, seenInner, seenAfter = result | ||||||
|  |             seenOuter |> shouldEqual outerGuid | ||||||
|  |             seenInner |> shouldNotEqual outerGuid | ||||||
|  |             seenAfter |> shouldEqual seenInner | ||||||
|  |  | ||||||
|  |             let! _, teardown = queue.RunTestTearDown setup (fun () -> ()) | ||||||
|  |             do! queue.EndTestFixture teardown | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``ExecutionContext flows correctly through async operations`` () = | ||||||
|  |         task { | ||||||
|  |             // Create a test fixture | ||||||
|  |             let dummyFixture = | ||||||
|  |                 TestFixture.Empty typeof<obj> (Some (Parallelizable.Yes ClassParallelScope.All)) [] [] | ||||||
|  |  | ||||||
|  |             use contexts = TestContexts.Empty () | ||||||
|  |             use queue = new ParallelQueue (Some 4, None) | ||||||
|  |  | ||||||
|  |             // Track which context values we see during execution | ||||||
|  |             let contextValues = System.Collections.Concurrent.ConcurrentBag<Guid * Guid> () | ||||||
|  |  | ||||||
|  |             // Start the fixture | ||||||
|  |             let! running = queue.StartTestFixture dummyFixture | ||||||
|  |             let! _, setup = queue.RunTestSetup running (fun () -> ()) | ||||||
|  |  | ||||||
|  |             // Create several async operations with different context values | ||||||
|  |             let tasks = | ||||||
|  |                 [ 1..10 ] | ||||||
|  |                 |> List.map (fun i -> | ||||||
|  |                     task { | ||||||
|  |                         // Set a unique context value | ||||||
|  |                         let expectedId = Guid.NewGuid () | ||||||
|  |                         let outputId = OutputStreamId expectedId | ||||||
|  |                         contexts.AsyncLocal.Value <- outputId | ||||||
|  |  | ||||||
|  |                         // Run an async operation that checks the context at multiple points | ||||||
|  |                         let! actualId = | ||||||
|  |                             queue.RunAsync | ||||||
|  |                                 setup | ||||||
|  |                                 None | ||||||
|  |                                 (fun () -> | ||||||
|  |                                     async { | ||||||
|  |                                         // Check context immediately | ||||||
|  |                                         let immediate = contexts.AsyncLocal.Value | ||||||
|  |                                         let (OutputStreamId immediateGuid) = immediate | ||||||
|  |                                         contextValues.Add (expectedId, immediateGuid) | ||||||
|  |  | ||||||
|  |                                         // Yield to allow potential context loss | ||||||
|  |                                         do! Async.Sleep 10 | ||||||
|  |  | ||||||
|  |                                         // Check context after yield | ||||||
|  |                                         let afterYield = contexts.AsyncLocal.Value | ||||||
|  |                                         let (OutputStreamId afterYieldGuid) = afterYield | ||||||
|  |                                         contextValues.Add (expectedId, afterYieldGuid) | ||||||
|  |  | ||||||
|  |                                         // Do some actual async work | ||||||
|  |                                         do! Task.Delay (10) |> Async.AwaitTask | ||||||
|  |  | ||||||
|  |                                         // Check context after task | ||||||
|  |                                         let afterTask = contexts.AsyncLocal.Value | ||||||
|  |                                         let (OutputStreamId afterTaskGuid) = afterTask | ||||||
|  |                                         contextValues.Add (expectedId, afterTaskGuid) | ||||||
|  |  | ||||||
|  |                                         return afterTaskGuid | ||||||
|  |                                     } | ||||||
|  |                                 ) | ||||||
|  |  | ||||||
|  |                         // Verify the returned value matches what we set | ||||||
|  |                         actualId |> shouldEqual expectedId | ||||||
|  |                     } | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             // Wait for all tasks | ||||||
|  |             let! results = Task.WhenAll (tasks) | ||||||
|  |             results |> Array.iter id | ||||||
|  |  | ||||||
|  |             // Verify all context values were correct | ||||||
|  |             let allValues = contextValues |> Seq.toList | ||||||
|  |             allValues |> shouldHaveLength 30 // 3 checks per operation * 10 operations | ||||||
|  |  | ||||||
|  |             // Every captured value should match its expected value | ||||||
|  |             allValues | ||||||
|  |             |> List.iter (fun (expected, actual) -> actual |> shouldEqual expected) | ||||||
|  |  | ||||||
|  |             // Clean up | ||||||
|  |             let! _, teardown = queue.RunTestTearDown setup (fun () -> ()) | ||||||
|  |             do! queue.EndTestFixture teardown | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     [<Test>] | ||||||
|  |     let ``ExecutionContext isolation between concurrent operations`` () = | ||||||
|  |         task { | ||||||
|  |             let dummyFixture = | ||||||
|  |                 TestFixture.Empty typeof<obj> (Some (Parallelizable.Yes ClassParallelScope.All)) [] [] | ||||||
|  |  | ||||||
|  |             use contexts = TestContexts.Empty () | ||||||
|  |             use queue = new ParallelQueue (Some 4, None) | ||||||
|  |  | ||||||
|  |             let! running = queue.StartTestFixture dummyFixture | ||||||
|  |             let! _, setup = queue.RunTestSetup running (fun () -> ()) | ||||||
|  |  | ||||||
|  |             // Use a barrier to ensure operations run concurrently | ||||||
|  |             let barrier = new Barrier (3) | ||||||
|  |             let seenValues = System.Collections.Concurrent.ConcurrentBag<int * Guid> () | ||||||
|  |  | ||||||
|  |             // Create operations that will definitely run concurrently | ||||||
|  |             let tasks = | ||||||
|  |                 [ 1..3 ] | ||||||
|  |                 |> List.map (fun i -> | ||||||
|  |                     task { | ||||||
|  |                         // Each task sets its own context value | ||||||
|  |                         let myId = Guid.NewGuid () | ||||||
|  |                         contexts.AsyncLocal.Value <- OutputStreamId myId | ||||||
|  |  | ||||||
|  |                         let! result = | ||||||
|  |                             queue.RunAsync | ||||||
|  |                                 setup | ||||||
|  |                                 (Some (Parallelizable.Yes ())) | ||||||
|  |                                 (fun () -> | ||||||
|  |                                     async { | ||||||
|  |                                         // Wait for all tasks to reach this point | ||||||
|  |                                         barrier.SignalAndWait () |> ignore | ||||||
|  |  | ||||||
|  |                                         // Now check what value we see | ||||||
|  |                                         let currentValue = contexts.AsyncLocal.Value | ||||||
|  |  | ||||||
|  |                                         match currentValue with | ||||||
|  |                                         | OutputStreamId guid -> seenValues.Add (i, guid) | ||||||
|  |  | ||||||
|  |                                         // Do some async work | ||||||
|  |                                         do! Async.Sleep 5 | ||||||
|  |  | ||||||
|  |                                         // Check again after async work | ||||||
|  |                                         let afterAsync = contexts.AsyncLocal.Value | ||||||
|  |  | ||||||
|  |                                         match afterAsync with | ||||||
|  |                                         | OutputStreamId guid -> return guid | ||||||
|  |                                     } | ||||||
|  |                                 ) | ||||||
|  |  | ||||||
|  |                         // Each task should see its own value | ||||||
|  |                         result |> shouldEqual myId | ||||||
|  |                     } | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             let! results = Task.WhenAll (tasks) | ||||||
|  |             results |> Array.iter id | ||||||
|  |  | ||||||
|  |             // Verify we saw 3 different values (one per task) | ||||||
|  |             let values = seenValues |> Seq.toList | ||||||
|  |             values |> shouldHaveLength 3 | ||||||
|  |  | ||||||
|  |             // All seen values should be different (no context bleeding) | ||||||
|  |             let uniqueValues = values |> List.map snd |> List.distinct | ||||||
|  |             uniqueValues |> shouldHaveLength 3 | ||||||
|  |  | ||||||
|  |             let! _, teardown = queue.RunTestTearDown setup (fun () -> ()) | ||||||
|  |             do! queue.EndTestFixture teardown | ||||||
|  |         } | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| <Project Sdk="Microsoft.NET.Sdk"> | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |  | ||||||
|     <PropertyGroup> |     <PropertyGroup> | ||||||
|         <TargetFramework>net8.0</TargetFramework> |         <TargetFramework>net9.0</TargetFramework> | ||||||
|         <IsPackable>false</IsPackable> |         <IsPackable>false</IsPackable> | ||||||
|         <IsTestProject>true</IsTestProject> |         <IsTestProject>true</IsTestProject> | ||||||
|     </PropertyGroup> |     </PropertyGroup> | ||||||
| @@ -11,17 +11,18 @@ | |||||||
|         <Compile Include="TestFilter.fs" /> |         <Compile Include="TestFilter.fs" /> | ||||||
|         <Compile Include="TestList.fs" /> |         <Compile Include="TestList.fs" /> | ||||||
|         <Compile Include="TestSurface.fs" /> |         <Compile Include="TestSurface.fs" /> | ||||||
|  |         <Compile Include="TestSynchronizationContext.fs" /> | ||||||
|         <Compile Include="TestTrx.fs" /> |         <Compile Include="TestTrx.fs" /> | ||||||
|         <EmbeddedResource Include="Example1.trx" /> |         <EmbeddedResource Include="Example1.trx" /> | ||||||
|     </ItemGroup> |     </ItemGroup> | ||||||
|  |  | ||||||
|     <ItemGroup> |     <ItemGroup> | ||||||
|         <PackageReference Include="ApiSurface" Version="4.1.17" /> |         <PackageReference Include="ApiSurface" Version="5.0.2" /> | ||||||
|         <PackageReference Include="FsCheck" Version="3.1.0" /> |         <PackageReference Include="FsCheck" Version="3.3.1" /> | ||||||
|         <PackageReference Include="FsUnit" Version="7.0.1" /> |         <PackageReference Include="FsUnit" Version="7.1.1" /> | ||||||
|         <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0" /> |         <PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.0" /> | ||||||
|         <PackageReference Include="NUnit" Version="4.3.2" /> |         <PackageReference Include="NUnit" Version="4.3.2" /> | ||||||
|         <PackageReference Include="NUnit3TestAdapter" Version="5.0.0"/> |         <PackageReference Include="NUnit3TestAdapter" Version="5.2.0"/> | ||||||
|     </ItemGroup> |     </ItemGroup> | ||||||
|  |  | ||||||
|     <ItemGroup> |     <ItemGroup> | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
|   <PropertyGroup> |   <PropertyGroup> | ||||||
|     <OutputType>Exe</OutputType> |     <OutputType>Exe</OutputType> | ||||||
|     <TargetFramework>net6.0</TargetFramework> |     <TargetFramework>net9.0</TargetFramework> | ||||||
|     <RollForward>Major</RollForward> |     <RollForward>Major</RollForward> | ||||||
|     <PackAsTool>true</PackAsTool> |     <PackAsTool>true</PackAsTool> | ||||||
|     <GenerateDocumentationFile>true</GenerateDocumentationFile> |     <GenerateDocumentationFile>true</GenerateDocumentationFile> | ||||||
| @@ -35,7 +35,7 @@ | |||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="Spectre.Console" Version="0.49.1" /> |     <PackageReference Include="Spectre.Console" Version="0.52.0" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| { | { | ||||||
|   "version": "0.2", |   "version": "0.3", | ||||||
|   "publicReleaseRefSpec": [ |   "publicReleaseRefSpec": [ | ||||||
|     "^refs/heads/main$" |     "^refs/heads/main$" | ||||||
|   ], |   ], | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ | |||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageDownload Include="G-Research.FSharp.Analyzers" Version="[0.13.0]" /> |     <PackageDownload Include="G-Research.FSharp.Analyzers" Version="[0.19.0]" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										6
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							| @@ -20,11 +20,11 @@ | |||||||
|     }, |     }, | ||||||
|     "nixpkgs": { |     "nixpkgs": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1739451785, |         "lastModified": 1760596604, | ||||||
|         "narHash": "sha256-3ebRdThRic9bHMuNi2IAA/ek9b32bsy8F5R4SvGTIog=", |         "narHash": "sha256-J/i5K6AAz/y5dBePHQOuzC7MbhyTOKsd/GLezSbEFiM=", | ||||||
|         "owner": "NixOS", |         "owner": "NixOS", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "rev": "1128e89fd5e11bb25aedbfc287733c6502202ea9", |         "rev": "3cbe716e2346710d6e1f7c559363d14e11c32a43", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
|   | |||||||
| @@ -14,8 +14,8 @@ | |||||||
|     flake-utils.lib.eachDefaultSystem (system: let |     flake-utils.lib.eachDefaultSystem (system: let | ||||||
|       pkgs = nixpkgs.legacyPackages.${system}; |       pkgs = nixpkgs.legacyPackages.${system}; | ||||||
|       pname = "unofficial-nunit-runner"; |       pname = "unofficial-nunit-runner"; | ||||||
|       dotnet-sdk = pkgs.dotnetCorePackages.sdk_8_0; |       dotnet-sdk = pkgs.dotnetCorePackages.sdk_9_0; | ||||||
|       dotnet-runtime = pkgs.dotnetCorePackages.runtime_8_0; |       dotnet-runtime = pkgs.dotnetCorePackages.runtime_9_0; | ||||||
|       version = "0.1"; |       version = "0.1"; | ||||||
|       dotnetTool = dllOverride: toolName: toolVersion: hash: |       dotnetTool = dllOverride: toolName: toolVersion: hash: | ||||||
|         pkgs.stdenvNoCC.mkDerivation rec { |         pkgs.stdenvNoCC.mkDerivation rec { | ||||||
|   | |||||||
							
								
								
									
										180
									
								
								nix/deps.json
									
									
									
									
									
								
							
							
						
						
									
										180
									
								
								nix/deps.json
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | |||||||
| [ | [ | ||||||
|   { |   { | ||||||
|     "pname": "ApiSurface", |     "pname": "ApiSurface", | ||||||
|     "version": "4.1.17", |     "version": "5.0.2", | ||||||
|     "hash": "sha256-ZYIW++7vXYV/7TTNUkV36QDN6yJbHymsKcxTUscTnvA=" |     "hash": "sha256-zcq1H1ccQzsZQf4kolzoOBSbyz07skihgPAvQ9Jri+E=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "fantomas", |     "pname": "fantomas", | ||||||
|     "version": "7.0.0", |     "version": "7.0.3", | ||||||
|     "hash": "sha256-v4bXmvjZOYxl5RSIHuqVfDzBQdRz5SrmzZtD6SeEYTY=" |     "hash": "sha256-0XlfV7SxXPDnk/CjkUesJSaH0cxlNHJ+Jj86zNUhkNA=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Fantomas.Core", |     "pname": "Fantomas.Core", | ||||||
| @@ -21,13 +21,18 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "FsCheck", |     "pname": "FsCheck", | ||||||
|     "version": "3.1.0", |     "version": "3.3.1", | ||||||
|     "hash": "sha256-u0Ha94EjClJ8evNSHu8d6+Dx9qYM3kV3sl1PEq9gF8s=" |     "hash": "sha256-k65ksdOSOGz+meRUUND+yuqJtm5ChaKuaxmRIdKzx2Y=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "fsharp-analyzers", |     "pname": "fsharp-analyzers", | ||||||
|     "version": "0.29.0", |     "version": "0.33.1", | ||||||
|     "hash": "sha256-e4G7/5DwwSZYvVE9yy+g8n1easPLC6/fDszioIsT7Dw=" |     "hash": "sha256-vYXvqnf3en487svFv3CmNl24SolwMYzu6zKKGXNxSu8=" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     "pname": "FSharp.Core", | ||||||
|  |     "version": "4.3.4", | ||||||
|  |     "hash": "sha256-styyo+6mJy+yxE0NZG/b1hxkAjPOnJfMgd9zWzCJ5uk=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "FSharp.Core", |     "pname": "FSharp.Core", | ||||||
| @@ -36,18 +41,18 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "FSharp.Core", |     "pname": "FSharp.Core", | ||||||
|     "version": "8.0.403", |     "version": "9.0.303", | ||||||
|     "hash": "sha256-3XSQp7JUOU5T6gvSQXNfBF4t4gaX4J4xushH+cfM9mE=" |     "hash": "sha256-AxR6wqodeU23KOTgkUfIgbavgbcSuzD4UBP+tiFydgA=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "FsUnit", |     "pname": "FsUnit", | ||||||
|     "version": "7.0.1", |     "version": "7.1.1", | ||||||
|     "hash": "sha256-K85CIdxMeFSHEKZk6heIXp/oFjWAn7dBILKrw49pJUY=" |     "hash": "sha256-UMCEGKxQ4ytjmPuVpiNaAPbi3RQH9gqa61JJIUS/6hg=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.ApplicationInsights", |     "pname": "Microsoft.ApplicationInsights", | ||||||
|     "version": "2.22.0", |     "version": "2.23.0", | ||||||
|     "hash": "sha256-mUQ63atpT00r49ca50uZu2YCiLg3yd6r3HzTryqcuEA=" |     "hash": "sha256-5sf3bg7CZZjHseK+F3foOchEhmVeioePxMZVvS6Rjb0=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.AspNetCore.App.Ref", |     "pname": "Microsoft.AspNetCore.App.Ref", | ||||||
| @@ -76,13 +81,13 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.CodeCoverage", |     "pname": "Microsoft.CodeCoverage", | ||||||
|     "version": "17.13.0", |     "version": "18.0.0", | ||||||
|     "hash": "sha256-GKrIxeyQo5Az1mztfQgea1kGtJwonnNOrXK/0ULfu8o=" |     "hash": "sha256-1RNxheCYASMusDC48BXtaO3MhnInw15JVfjfLM1VMGA=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.NET.Test.Sdk", |     "pname": "Microsoft.NET.Test.Sdk", | ||||||
|     "version": "17.13.0", |     "version": "18.0.0", | ||||||
|     "hash": "sha256-sc2wvyV8cGm1FrNP2GGHEI584RCvRPu15erYCsgw5QY=" |     "hash": "sha256-9iW+9mvMeZgDXwSoR08bnvRNsN4jT8OVWcjq3lcE+cs=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.NETCore.App.Host.linux-arm64", |     "pname": "Microsoft.NETCore.App.Host.linux-arm64", | ||||||
| @@ -129,15 +134,20 @@ | |||||||
|     "version": "6.0.36", |     "version": "6.0.36", | ||||||
|     "hash": "sha256-0xIJYFzxdMcnCj3wzkFRQZSnQcPHzPHMzePRIOA3oJs=" |     "hash": "sha256-0xIJYFzxdMcnCj3wzkFRQZSnQcPHzPHMzePRIOA3oJs=" | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     "pname": "Microsoft.NETCore.Platforms", | ||||||
|  |     "version": "1.1.0", | ||||||
|  |     "hash": "sha256-FeM40ktcObQJk4nMYShB61H/E8B7tIKfl9ObJ0IOcCM=" | ||||||
|  |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.NETCore.Platforms", |     "pname": "Microsoft.NETCore.Platforms", | ||||||
|     "version": "1.1.1", |     "version": "1.1.1", | ||||||
|     "hash": "sha256-8hLiUKvy/YirCWlFwzdejD2Db3DaXhHxT7GSZx/znJg=" |     "hash": "sha256-8hLiUKvy/YirCWlFwzdejD2Db3DaXhHxT7GSZx/znJg=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.NETCore.Platforms", |     "pname": "Microsoft.NETCore.Targets", | ||||||
|     "version": "2.0.0", |     "version": "1.1.0", | ||||||
|     "hash": "sha256-IEvBk6wUXSdyCnkj6tHahOJv290tVVT8tyemYcR0Yro=" |     "hash": "sha256-0AqQ2gMS8iNlYkrD+BxtIg7cXMnr9xZHtKAuN4bjfaQ=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.NETCore.Targets", |     "pname": "Microsoft.NETCore.Targets", | ||||||
| @@ -146,43 +156,48 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.Testing.Extensions.Telemetry", |     "pname": "Microsoft.Testing.Extensions.Telemetry", | ||||||
|     "version": "1.5.3", |     "version": "1.9.0", | ||||||
|     "hash": "sha256-bIXwPSa3jkr2b6xINOqMUs6/uj/r4oVFM7xq3uVIZDU=" |     "hash": "sha256-JT91ThKLEyoRS/8ZJqZwlSTT7ofC2QhNqPFI3pYmMaw=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.Testing.Extensions.TrxReport.Abstractions", |     "pname": "Microsoft.Testing.Extensions.TrxReport.Abstractions", | ||||||
|     "version": "1.5.3", |     "version": "1.9.0", | ||||||
|     "hash": "sha256-IfMRfcyaIKEMRtx326ICKtinDBEfGw/Sv8ZHawJ96Yc=" |     "hash": "sha256-oscZOEKw7gM6eRdDrOS3x+CwqIvXWRmfmi0ugCxBRw0=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.Testing.Extensions.VSTestBridge", |     "pname": "Microsoft.Testing.Extensions.VSTestBridge", | ||||||
|     "version": "1.5.3", |     "version": "1.9.0", | ||||||
|     "hash": "sha256-XpM/yFjhLSsuzyDV+xKubs4V1zVVYiV05E0+N4S1h0g=" |     "hash": "sha256-CadXLWD093sUDaWhnppzD9LvpxSRqqt93ZEOFiIAPyw=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.Testing.Platform", |     "pname": "Microsoft.Testing.Platform", | ||||||
|     "version": "1.5.3", |     "version": "1.9.0", | ||||||
|     "hash": "sha256-y61Iih6w5D79dmrj2V675mcaeIiHoj1HSa1FRit2BLM=" |     "hash": "sha256-6nzjoYbJOh7v/GB7d+TDuM0l/xglCshFX6KWjg7+cFI=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.Testing.Platform.MSBuild", |     "pname": "Microsoft.Testing.Platform.MSBuild", | ||||||
|     "version": "1.5.3", |     "version": "1.9.0", | ||||||
|     "hash": "sha256-YspvjE5Jfi587TAfsvfDVJXNrFOkx1B3y1CKV6m7YLY=" |     "hash": "sha256-/bileP4b+9RZp8yjgS6eynXwc2mohyyzf6p/0LZJd8I=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.TestPlatform.ObjectModel", |     "pname": "Microsoft.TestPlatform.AdapterUtilities", | ||||||
|     "version": "17.12.0", |     "version": "17.13.0", | ||||||
|     "hash": "sha256-3XBHBSuCxggAIlHXmKNQNlPqMqwFlM952Av6RrLw1/w=" |     "hash": "sha256-Vr+3Tad/h/nk7f/5HMExn3HvCGFCarehFAzJSfCBaOc=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.TestPlatform.ObjectModel", |     "pname": "Microsoft.TestPlatform.ObjectModel", | ||||||
|     "version": "17.13.0", |     "version": "17.13.0", | ||||||
|     "hash": "sha256-6S0fjfj8vA+h6dJVNwLi6oZhYDO/I/6hBZaq2VTW+Uk=" |     "hash": "sha256-6S0fjfj8vA+h6dJVNwLi6oZhYDO/I/6hBZaq2VTW+Uk=" | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     "pname": "Microsoft.TestPlatform.ObjectModel", | ||||||
|  |     "version": "18.0.0", | ||||||
|  |     "hash": "sha256-O/ivHdoIO+q1n0byJ9OZO4quOqACOD3K3Qm00wfhuZk=" | ||||||
|  |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Microsoft.TestPlatform.TestHost", |     "pname": "Microsoft.TestPlatform.TestHost", | ||||||
|     "version": "17.13.0", |     "version": "18.0.0", | ||||||
|     "hash": "sha256-L/CJzou7dhmShUgXq3aXL3CaLTJll17Q+JY2DBdUUpo=" |     "hash": "sha256-qAIX2Rqxrnh1xaYRjCbkkvvMm407oyKihqyVjURX5wY=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Myriad.Core", |     "pname": "Myriad.Core", | ||||||
| @@ -196,13 +211,8 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Nerdbank.GitVersioning", |     "pname": "Nerdbank.GitVersioning", | ||||||
|     "version": "3.7.115", |     "version": "3.8.118", | ||||||
|     "hash": "sha256-sqn+i7vvBgBUtm7j82mH+SpApgI2hsmL5DYfLm1Z7gw=" |     "hash": "sha256-Hmyy0ZKOmwN4zIhI4+MqoN8geZNc1sd033aZJ6APrO8=" | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "pname": "Newtonsoft.Json", |  | ||||||
|     "version": "13.0.1", |  | ||||||
|     "hash": "sha256-K2tSVW4n4beRPzPu3rlVaBEMdGvWSv/3Q1fxaDh4Mjo=" |  | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Newtonsoft.Json", |     "pname": "Newtonsoft.Json", | ||||||
| @@ -211,33 +221,33 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NuGet.Common", |     "pname": "NuGet.Common", | ||||||
|     "version": "6.13.1", |     "version": "6.14.0", | ||||||
|     "hash": "sha256-uDSZzCzMETKJE2pYAh9n110YrLPVs83+cxcyNCv9V/o=" |     "hash": "sha256-jDOwt3veI1GSG8CfBnf2+dJxD3E/Nmlc+vHtD4J76Ms=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NuGet.Configuration", |     "pname": "NuGet.Configuration", | ||||||
|     "version": "6.13.1", |     "version": "6.14.0", | ||||||
|     "hash": "sha256-fjit3Qhqu+egLgxvJRD6zQ6U2H+OxsLoy7HYcbXJgBY=" |     "hash": "sha256-1PN9s6fhCw3wd2260U6hQ4vG3jIvcG8GIn1oQgxMXA0=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NuGet.Frameworks", |     "pname": "NuGet.Frameworks", | ||||||
|     "version": "6.13.1", |     "version": "6.14.0", | ||||||
|     "hash": "sha256-upL6cboghOYed6acdlVljMa9nCSExYoBgQlD5QcJ+ek=" |     "hash": "sha256-3ViM3R1ucQMEL2hQYsivT86kI6veMQK2xDsiAcFcVQk=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NuGet.Packaging", |     "pname": "NuGet.Packaging", | ||||||
|     "version": "6.13.1", |     "version": "6.14.0", | ||||||
|     "hash": "sha256-khq+grMovEsO95wAhCHmauL60LB83NkWnDY5zcImXT4=" |     "hash": "sha256-Yafbnxs3maj55bJ1oKQiZ0QkntFUzXdhorL94YEUOhY=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NuGet.Protocol", |     "pname": "NuGet.Protocol", | ||||||
|     "version": "6.13.1", |     "version": "6.14.0", | ||||||
|     "hash": "sha256-0zqowx5iNMMzTSF0HO2Zzc1qqyCB/nEhNHFH8Z0jxzc=" |     "hash": "sha256-uLDKfs+QN1MdnuQtTES8qfNzzsmYKM6XB9pwJc4G+eo=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NuGet.Versioning", |     "pname": "NuGet.Versioning", | ||||||
|     "version": "6.13.1", |     "version": "6.14.0", | ||||||
|     "hash": "sha256-NymrrcVZUoc78elPHYIoBFG2NpdIEZYEeqk6FTdvZug=" |     "hash": "sha256-DqdOJgsphKxSvqB8b60zNPCaiLfbiu3WnUJ/90feLrY=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NUnit", |     "pname": "NUnit", | ||||||
| @@ -246,8 +256,8 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "NUnit3TestAdapter", |     "pname": "NUnit3TestAdapter", | ||||||
|     "version": "5.0.0", |     "version": "5.2.0", | ||||||
|     "hash": "sha256-7jZM4qAbIzne3AcdFfMbvbgogqpxvVe6q2S7Ls8xQy0=" |     "hash": "sha256-ybTutL4VkX/fq61mS+O3Ruh+adic4fpv+MKgQ0IZvGg=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "runtime.any.System.Runtime", |     "pname": "runtime.any.System.Runtime", | ||||||
| @@ -266,8 +276,8 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "Spectre.Console", |     "pname": "Spectre.Console", | ||||||
|     "version": "0.49.1", |     "version": "0.52.0", | ||||||
|     "hash": "sha256-tqSVojyuQjuB34lXo759NOcyLgNIw815mKXJPq5JFDo=" |     "hash": "sha256-enGa3do7uHQFJOGha+IJZB/rlYhZDvLYbNYgZ4B5V8g=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "System.Collections.Immutable", |     "pname": "System.Collections.Immutable", | ||||||
| @@ -286,18 +296,8 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "System.Formats.Asn1", |     "pname": "System.Formats.Asn1", | ||||||
|     "version": "8.0.1", |     "version": "6.0.0", | ||||||
|     "hash": "sha256-may/Wg+esmm1N14kQTG4ESMBi+GQKPp0ZrrBo/o6OXM=" |     "hash": "sha256-KaMHgIRBF7Nf3VwOo+gJS1DcD+41cJDPWFh+TDQ8ee8=" | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "pname": "System.IO.Abstractions", |  | ||||||
|     "version": "4.2.13", |  | ||||||
|     "hash": "sha256-nkC/PiqE6+c1HJ2yTwg3x+qdBh844Z8n3ERWDW8k6Gg=" |  | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "pname": "System.IO.FileSystem.AccessControl", |  | ||||||
|     "version": "4.5.0", |  | ||||||
|     "hash": "sha256-ck44YBQ0M+2Im5dw0VjBgFD1s0XuY54cujrodjjSBL8=" |  | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "System.Memory", |     "pname": "System.Memory", | ||||||
| @@ -311,8 +311,8 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "System.Reflection.Metadata", |     "pname": "System.Reflection.Metadata", | ||||||
|     "version": "1.6.0", |     "version": "8.0.0", | ||||||
|     "hash": "sha256-JJfgaPav7UfEh4yRAQdGhLZF1brr0tUWPl6qmfNWq/E=" |     "hash": "sha256-dQGC30JauIDWNWXMrSNOJncVa1umR1sijazYwUDdSIE=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "System.Runtime", |     "pname": "System.Runtime", | ||||||
| @@ -324,11 +324,6 @@ | |||||||
|     "version": "6.0.0", |     "version": "6.0.0", | ||||||
|     "hash": "sha256-bEG1PnDp7uKYz/OgLOWs3RWwQSVYm+AnPwVmAmcgp2I=" |     "hash": "sha256-bEG1PnDp7uKYz/OgLOWs3RWwQSVYm+AnPwVmAmcgp2I=" | ||||||
|   }, |   }, | ||||||
|   { |  | ||||||
|     "pname": "System.Security.AccessControl", |  | ||||||
|     "version": "4.5.0", |  | ||||||
|     "hash": "sha256-AFsKPb/nTk2/mqH/PYpaoI8PLsiKKimaXf+7Mb5VfPM=" |  | ||||||
|   }, |  | ||||||
|   { |   { | ||||||
|     "pname": "System.Security.Cryptography.Pkcs", |     "pname": "System.Security.Cryptography.Pkcs", | ||||||
|     "version": "6.0.4", |     "version": "6.0.4", | ||||||
| @@ -339,11 +334,6 @@ | |||||||
|     "version": "4.4.0", |     "version": "4.4.0", | ||||||
|     "hash": "sha256-Ri53QmFX8I8UH0x4PikQ1ZA07ZSnBUXStd5rBfGWFOE=" |     "hash": "sha256-Ri53QmFX8I8UH0x4PikQ1ZA07ZSnBUXStd5rBfGWFOE=" | ||||||
|   }, |   }, | ||||||
|   { |  | ||||||
|     "pname": "System.Security.Principal.Windows", |  | ||||||
|     "version": "4.5.0", |  | ||||||
|     "hash": "sha256-BkUYNguz0e4NJp1kkW7aJBn3dyH9STwB5N8XqnlCsmY=" |  | ||||||
|   }, |  | ||||||
|   { |   { | ||||||
|     "pname": "System.Text.Json", |     "pname": "System.Text.Json", | ||||||
|     "version": "8.0.5", |     "version": "8.0.5", | ||||||
| @@ -351,32 +341,32 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "TypeEquality", |     "pname": "TypeEquality", | ||||||
|     "version": "0.3.0", |     "version": "0.4.2", | ||||||
|     "hash": "sha256-V50xAOzzyUJrY+MYPRxtnqW5MVeATXCes89wPprv1r4=" |     "hash": "sha256-YxK6BGHjcuP76j5BbTDi814jxGqOevQSgS00IJcjZSA=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "WoofWare.DotnetRuntimeLocator", |     "pname": "WoofWare.DotnetRuntimeLocator", | ||||||
|     "version": "0.1.11", |     "version": "0.1.12", | ||||||
|     "hash": "sha256-btWYnXxZzAnEcAE3Ufvz+Z3cJqtW7EGXs5OZMEHIWr8=" |     "hash": "sha256-6pNZs0/R2LnLKSODq9DyHhGo2C+SDyz9k7D/13/78so=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "WoofWare.Myriad.Plugins", |     "pname": "WoofWare.Myriad.Plugins", | ||||||
|     "version": "4.0.12", |     "version": "9.0.4", | ||||||
|     "hash": "sha256-6o8oE5loEpKFVMRXf/rmY0YDW/iRF2Zic1lkfsOXcSo=" |     "hash": "sha256-fVahNM2SOvG159Wz6+uBkrl3+jqVtRUhZsZ2Kl2VCfk=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "WoofWare.Myriad.Plugins.Attributes", |     "pname": "WoofWare.Myriad.Plugins.Attributes", | ||||||
|     "version": "3.6.7", |     "version": "3.7.3", | ||||||
|     "hash": "sha256-YvrXS2ZMoMZkWgGxxcumyoqQkg4cSjGQzCrMbBCCtXM=" |     "hash": "sha256-scdokAtktZZ6K8c/eXm2DKtPzQPZrJLJ0cnu652uYuY=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "WoofWare.PrattParser", |     "pname": "WoofWare.PrattParser", | ||||||
|     "version": "0.2.3", |     "version": "0.2.5", | ||||||
|     "hash": "sha256-PsfU33BFsaA8LwlZMkSwmoA4hLb7Vkm2gYm48MglqxY=" |     "hash": "sha256-6+74AMxVIBa5rYO34Hlm02zPtRSvpcvUA6cqeYB3WoQ=" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "pname": "WoofWare.Whippet.Fantomas", |     "pname": "WoofWare.Whippet.Fantomas", | ||||||
|     "version": "0.3.1", |     "version": "0.6.4", | ||||||
|     "hash": "sha256-i5oiqcrxzM90Ocuq5MIu2Ha5lV0aYu5nCvuwmFqp6NA=" |     "hash": "sha256-ScZ7EEcxLvXyam2ZVqDK58QlK3RcePWghzRvtLLLdZI=" | ||||||
|   } |   } | ||||||
| ] | ] | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user