mirror of
https://github.com/Smaug123/unofficial-nunit-runner
synced 2025-10-24 09:48:40 +00:00
Compare commits
187 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 | ||
|
f264fca446 | ||
|
8dd18603d6 | ||
|
012b4a6e03 | ||
|
51918f6d35 | ||
|
225b2645b0 | ||
|
5c05d29917 | ||
|
33aa337598 | ||
|
daa53a84a5 | ||
|
4c9568819a | ||
|
6ab8316b2b | ||
|
600772a81f | ||
|
c6b735816e | ||
|
29c3c17bc0 | ||
|
512b41d7c1 | ||
|
f920e597bd | ||
|
226da02b1b | ||
|
e9b9366b90 | ||
|
e7c31b5366 | ||
|
9d26610384 | ||
|
61fbb5f55b | ||
|
d9938d96a3 | ||
|
874a367ce3 | ||
|
4300fbe6b8 | ||
|
402b98f85c | ||
|
a795d6222c | ||
|
cf482b677b | ||
|
db27a7acc8 | ||
|
99826df864 | ||
|
1c93b2c4b2 | ||
|
4369b35dd1 | ||
|
a183455f55 | ||
|
de5f5a64ef | ||
|
fdddbf828b | ||
|
67d9d71100 | ||
|
c39745280b | ||
|
26c29d63bc | ||
|
78f7f76074 | ||
|
c2401207c4 | ||
|
46e097a02e | ||
|
7c23c3bb1c | ||
|
ace1417de6 | ||
|
a694637958 | ||
|
8626cc1252 | ||
|
0287a6b7eb | ||
|
8579ee8f8b | ||
|
debda1a557 | ||
|
bf7d846f61 | ||
|
4690bf23ad | ||
|
04283ee961 | ||
|
6c61c2cbf1 | ||
|
2a0cabd3a9 | ||
|
111e019a83 | ||
|
16d353e8df | ||
|
8528da77aa | ||
|
b74ae980d7 | ||
|
318ba70608 | ||
|
1ee38136a1 | ||
|
3d8bb8d0ca | ||
|
2f64191348 | ||
|
86b8c0ec64 | ||
|
e088d2b420 | ||
|
3ea9d94861 | ||
|
e003ef0934 | ||
|
50948e629c | ||
|
b14e5ede40 | ||
|
f256eb7b29 | ||
|
ec7d9187c5 | ||
|
62eb3c5a66 | ||
|
97e7a87e6a | ||
|
14d91840b2 | ||
|
2859a5f6e6 | ||
|
64649b76ce | ||
|
c9f013891b | ||
|
a7660d1c38 | ||
|
d49d36206e | ||
|
881d5227e7 | ||
|
6564835ee4 | ||
|
db2ecdfa43 | ||
|
5b376cc592 | ||
|
4d34382cd3 | ||
|
c26e4f085d | ||
|
b002be5d72 | ||
|
5483184edc | ||
|
3596b8a3a8 | ||
|
68326d7628 | ||
|
f4b0a5457b | ||
|
c4b67304a1 | ||
|
337a0635d2 | ||
|
e8e302db2d | ||
|
1522e3cc9c | ||
|
81c6b584a4 | ||
|
40824e06e7 | ||
|
fb945c04ac | ||
|
85cd116d52 | ||
|
8e7c54cc83 | ||
|
378a0169f8 | ||
|
870804d6ef | ||
|
9f5f22c644 | ||
|
296f230616 | ||
|
56ac203570 | ||
|
e17e769d5a | ||
|
57c34e0c4c | ||
|
7f9464b826 | ||
|
3d04199c56 | ||
|
9d4b893e02 | ||
|
55e9645316 | ||
|
e9dc768449 | ||
|
e0b2d52812 | ||
|
2ed4a04f70 | ||
|
2e066a1a9a | ||
|
6468a301b9 | ||
|
13f636df3d | ||
|
eed076cad5 | ||
|
df64e46079 | ||
|
b3bc0aa4c0 | ||
|
d3f9ee6b02 |
@@ -3,13 +3,13 @@
|
|||||||
"isRoot": true,
|
"isRoot": true,
|
||||||
"tools": {
|
"tools": {
|
||||||
"fantomas": {
|
"fantomas": {
|
||||||
"version": "6.3.9",
|
"version": "7.0.3",
|
||||||
"commands": [
|
"commands": [
|
||||||
"fantomas"
|
"fantomas"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"fsharp-analyzers": {
|
"fsharp-analyzers": {
|
||||||
"version": "0.26.0",
|
"version": "0.33.1",
|
||||||
"commands": [
|
"commands": [
|
||||||
"fsharp-analyzers"
|
"fsharp-analyzers"
|
||||||
]
|
]
|
||||||
|
@@ -1,40 +1,40 @@
|
|||||||
root=true
|
root = true
|
||||||
|
|
||||||
[*]
|
[*]
|
||||||
charset=utf-8
|
charset = utf-8
|
||||||
trim_trailing_whitespace=true
|
trim_trailing_whitespace = true
|
||||||
insert_final_newline=true
|
insert_final_newline = true
|
||||||
indent_style=space
|
indent_style = space
|
||||||
indent_size=4
|
indent_size = 4
|
||||||
|
|
||||||
# ReSharper properties
|
# ReSharper properties
|
||||||
resharper_xml_indent_size=2
|
resharper_xml_indent_size = 2
|
||||||
resharper_xml_max_line_length=100
|
resharper_xml_max_line_length = 100
|
||||||
resharper_xml_tab_width=2
|
resharper_xml_tab_width = 2
|
||||||
|
|
||||||
[*.{csproj,fsproj,sqlproj,targets,props,ts,tsx,css,json}]
|
[*.{csproj,fsproj,sqlproj,targets,props,ts,tsx,css,json}]
|
||||||
indent_style=space
|
indent_style = space
|
||||||
indent_size=2
|
indent_size = 2
|
||||||
|
|
||||||
[*.{fs,fsi}]
|
[*.{fs,fsi}]
|
||||||
fsharp_bar_before_discriminated_union_declaration=true
|
fsharp_bar_before_discriminated_union_declaration = true
|
||||||
fsharp_space_before_uppercase_invocation=true
|
fsharp_space_before_uppercase_invocation = true
|
||||||
fsharp_space_before_class_constructor=true
|
fsharp_space_before_class_constructor = true
|
||||||
fsharp_space_before_member=true
|
fsharp_space_before_member = true
|
||||||
fsharp_space_before_colon=true
|
fsharp_space_before_colon = true
|
||||||
fsharp_space_before_semicolon=true
|
fsharp_space_before_semicolon = true
|
||||||
fsharp_multiline_bracket_style=aligned
|
fsharp_multiline_bracket_style = aligned
|
||||||
fsharp_newline_between_type_definition_and_members=true
|
fsharp_newline_between_type_definition_and_members = true
|
||||||
fsharp_align_function_signature_to_indentation=true
|
fsharp_align_function_signature_to_indentation = true
|
||||||
fsharp_alternative_long_member_definitions=true
|
fsharp_alternative_long_member_definitions = true
|
||||||
fsharp_multi_line_lambda_closing_newline=true
|
fsharp_multi_line_lambda_closing_newline = true
|
||||||
fsharp_experimental_keep_indent_in_branch=true
|
fsharp_experimental_keep_indent_in_branch = true
|
||||||
fsharp_max_value_binding_width=80
|
fsharp_max_value_binding_width = 80
|
||||||
fsharp_max_record_width=0
|
fsharp_max_record_width = 0
|
||||||
max_line_length=120
|
max_line_length = 120
|
||||||
end_of_line=lf
|
end_of_line = lf
|
||||||
|
|
||||||
[*.{appxmanifest,build,dtd,nuspec,xaml,xamlx,xoml,xsd}]
|
[*.{appxmanifest,build,dtd,nuspec,xaml,xamlx,xoml,xsd}]
|
||||||
indent_style=space
|
indent_style = space
|
||||||
indent_size=2
|
indent_size = 2
|
||||||
tab_width=2
|
tab_width = 2
|
||||||
|
23
.envrc
Normal file
23
.envrc
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
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
|
1
.fantomasignore
Normal file
1
.fantomasignore
Normal file
@@ -0,0 +1 @@
|
|||||||
|
.direnv/
|
267
.github/workflows/dotnet.yaml
vendored
267
.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@V27
|
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 }}
|
||||||
@@ -38,7 +38,30 @@ jobs:
|
|||||||
- name: Build
|
- name: Build
|
||||||
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: 'nix develop --command dotnet test --no-build --verbosity normal --configuration ${{matrix.config}}'
|
run: |
|
||||||
|
nix develop --command dotnet test --no-build --verbosity normal --configuration ${{matrix.config}} --filter 'FullyQualifiedName !~ FailingConsumer'
|
||||||
|
|
||||||
|
selftest-intended-failures:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v5
|
||||||
|
with:
|
||||||
|
fetch-depth: 0 # so that NerdBank.GitVersioning has access to history
|
||||||
|
- name: Install Nix
|
||||||
|
uses: cachix/install-nix-action@v31
|
||||||
|
with:
|
||||||
|
extra_nix_config: |
|
||||||
|
access-tokens = github.com=${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Restore dependencies
|
||||||
|
run: nix develop --command dotnet restore
|
||||||
|
- name: Build
|
||||||
|
run: 'nix develop --command dotnet build --no-restore --configuration Release'
|
||||||
|
- name: Test using self
|
||||||
|
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
|
||||||
|
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
|
||||||
|
run: 'actual=$(cat snapshot.txt | sort) expected=$(cat FailingConsumer/expected.txt | sort) [ "$expected" == "$actual" ]'
|
||||||
|
|
||||||
selftest:
|
selftest:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
@@ -58,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@V27
|
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 }}
|
||||||
@@ -71,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/net8.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
|
||||||
@@ -86,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@V27
|
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 }}
|
||||||
@@ -105,22 +128,24 @@ 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@V27
|
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: Build
|
- name: Build
|
||||||
run: nix build
|
run: nix build
|
||||||
|
- name: Reproducibility check
|
||||||
|
run: nix build --rebuild
|
||||||
|
|
||||||
check-dotnet-format:
|
check-dotnet-format:
|
||||||
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@V27
|
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 }}
|
||||||
@@ -131,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@V27
|
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 }}
|
||||||
@@ -146,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@V27
|
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 }}
|
||||||
@@ -159,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@V27
|
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 }}
|
||||||
@@ -169,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@V27
|
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 }}
|
||||||
@@ -199,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
|
||||||
@@ -207,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
|
||||||
@@ -215,59 +240,165 @@ 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-tool -maxdepth 1 -name 'WoofWare.NUnitTestRunner.*.nupkg' -printf c | wc -c) -ne "1" ]]; then exit 1; fi
|
run: if [[ $(find packed-tool -maxdepth 1 -name 'WoofWare.NUnitTestRunner.*.nupkg' -printf c | wc -c) -ne "1" ]]; then exit 1; fi
|
||||||
|
|
||||||
github-release-tool-dry-run:
|
all-required-checks-complete:
|
||||||
needs: [nuget-pack]
|
if: ${{ always() }}
|
||||||
|
needs: [github-release-dry-run, check-dotnet-format, check-nix-format, build, build-nix, linkcheck, flake-check, analyzers, nuget-pack, expected-pack, selftest-intended-failures, selftest]
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: G-Research/common-actions/check-required-lite@2b7dc49cb14f3344fbe6019c14a31165e258c059
|
||||||
- name: Download NuGet artifact (lib)
|
with:
|
||||||
uses: actions/download-artifact@v4
|
needs-context: ${{ toJSON(needs) }}
|
||||||
|
|
||||||
|
attestation-lib:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [all-required-checks-complete]
|
||||||
|
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
attestations: write
|
||||||
|
contents: read
|
||||||
|
steps:
|
||||||
|
- name: Download NuGet artifact
|
||||||
|
uses: actions/download-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: nuget-package-lib
|
name: nuget-package-lib
|
||||||
- name: Download NuGet artifact (tool)
|
path: packed
|
||||||
uses: actions/download-artifact@v4
|
- name: Attest Build Provenance
|
||||||
|
uses: actions/attest-build-provenance@977bb373ede98d70efdf65b84cb5f73e068dcc2a # v3.0.0
|
||||||
|
with:
|
||||||
|
subject-path: "packed/*.nupkg"
|
||||||
|
|
||||||
|
attestation-tool:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [all-required-checks-complete]
|
||||||
|
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
attestations: write
|
||||||
|
contents: read
|
||||||
|
steps:
|
||||||
|
- name: Download NuGet artifact
|
||||||
|
uses: actions/download-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: nuget-package-tool
|
name: nuget-package-tool
|
||||||
- name: Tag and release tool
|
path: packed
|
||||||
env:
|
- name: Attest Build Provenance
|
||||||
DRY_RUN: 1
|
uses: actions/attest-build-provenance@977bb373ede98d70efdf65b84cb5f73e068dcc2a # v3.0.0
|
||||||
GITHUB_TOKEN: mock-token
|
with:
|
||||||
run: sh .github/workflows/tag.sh
|
subject-path: "packed/*.nupkg"
|
||||||
|
|
||||||
all-required-checks-complete:
|
nuget-publish-lib:
|
||||||
needs: [check-dotnet-format, check-nix-format, build, build-nix, linkcheck, flake-check, analyzers, nuget-pack, expected-pack, github-release-tool-dry-run]
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- run: echo "All required checks complete."
|
|
||||||
|
|
||||||
nuget-publish:
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
||||||
needs: [all-required-checks-complete]
|
needs: [all-required-checks-complete]
|
||||||
environment: main-deploy
|
environment: main-deploy
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
attestations: write
|
||||||
|
contents: read
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
- name: Install Nix
|
- name: Install Nix
|
||||||
uses: cachix/install-nix-action@V27
|
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 (lib)
|
- 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-lib
|
path: packed
|
||||||
- name: Publish to NuGet (lib)
|
- name: Identify .NET
|
||||||
run: nix develop --command dotnet nuget push "packed-lib/WoofWare.NUnitTestRunner.Lib.*.nupkg" --api-key ${{ secrets.NUGET_API_KEY }} --source https://api.nuget.org/v3/index.json --skip-duplicate
|
id: identify-dotnet
|
||||||
- name: Download NuGet artifact (tool)
|
run: nix develop --command bash -c "echo dotnet=$(which dotnet) >> $GITHUB_OUTPUT"
|
||||||
uses: actions/download-artifact@v4
|
- name: Obtain NuGet key
|
||||||
|
uses: NuGet/login@d22cc5f58ff5b88bf9bd452535b4335137e24544
|
||||||
|
id: login
|
||||||
|
with:
|
||||||
|
user: ${{ secrets.NUGET_USER }}
|
||||||
|
- name: Publish NuGet package
|
||||||
|
uses: G-Research/common-actions/publish-nuget@2b7dc49cb14f3344fbe6019c14a31165e258c059
|
||||||
|
with:
|
||||||
|
package-name: WoofWare.NUnitTestRunner.Lib
|
||||||
|
nuget-key: ${{ steps.login.outputs.NUGET_API_KEY }}
|
||||||
|
nupkg-dir: packed/
|
||||||
|
dotnet: ${{ steps.identify-dotnet.outputs.dotnet }}
|
||||||
|
|
||||||
|
nuget-publish-tool:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
||||||
|
needs: [all-required-checks-complete]
|
||||||
|
environment: main-deploy
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
attestations: write
|
||||||
|
contents: read
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v5
|
||||||
|
- name: Install Nix
|
||||||
|
uses: cachix/install-nix-action@v31
|
||||||
|
with:
|
||||||
|
extra_nix_config: |
|
||||||
|
access-tokens = github.com=${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Download NuGet artifact
|
||||||
|
uses: actions/download-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: nuget-package-tool
|
name: nuget-package-tool
|
||||||
path: packed-tool
|
path: packed
|
||||||
- name: Publish to NuGet (tool)
|
- name: Identify .NET
|
||||||
run: nix develop --command dotnet nuget push "packed-tool/WoofWare.NUnitTestRunner.*.nupkg" --api-key ${{ secrets.NUGET_API_KEY }} --source https://api.nuget.org/v3/index.json --skip-duplicate
|
id: identify-dotnet
|
||||||
|
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
|
||||||
|
uses: G-Research/common-actions/publish-nuget@2b7dc49cb14f3344fbe6019c14a31165e258c059
|
||||||
|
with:
|
||||||
|
package-name: WoofWare.NUnitTestRunner
|
||||||
|
nuget-key: ${{ steps.login.outputs.NUGET_API_KEY }}
|
||||||
|
nupkg-dir: packed/
|
||||||
|
dotnet: ${{ steps.identify-dotnet.outputs.dotnet }}
|
||||||
|
|
||||||
github-release-tool:
|
github-release-dry-run:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
artifact:
|
||||||
|
- nuget-package-tool
|
||||||
|
- nuget-package-lib
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [nuget-pack]
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v5
|
||||||
|
- name: Download NuGet artifact (tool)
|
||||||
|
uses: actions/download-artifact@v5
|
||||||
|
with:
|
||||||
|
name: ${{ matrix.artifact }}
|
||||||
|
- name: Compute package path
|
||||||
|
id: compute-path
|
||||||
|
run: |
|
||||||
|
find . -maxdepth 1 -type f -name 'WoofWare.NUnitTestRunner.*.nupkg' -exec sh -c 'echo "output=$(basename "$1")" >> $GITHUB_OUTPUT' shell {} \;
|
||||||
|
- name: Compute tag name
|
||||||
|
id: compute-tag
|
||||||
|
env:
|
||||||
|
NUPKG_PATH: ${{ steps.compute-path.outputs.output }}
|
||||||
|
run: echo "output=$(basename "$NUPKG_PATH" .nupkg)" >> $GITHUB_OUTPUT
|
||||||
|
- name: Tag and release
|
||||||
|
uses: G-Research/common-actions/github-release@19d7281a0f9f83e13c78f99a610dbc80fc59ba3b
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
target-commitish: ${{ github.sha }}
|
||||||
|
tag: ${{ steps.compute-tag.outputs.output }}
|
||||||
|
binary-contents: ${{ steps.compute-path.outputs.output }}
|
||||||
|
dry-run: true
|
||||||
|
|
||||||
|
github-release:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
artifact:
|
||||||
|
- nuget-package-tool
|
||||||
|
- nuget-package-lib
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
if: ${{ !github.event.repository.fork && github.ref == 'refs/heads/main' }}
|
||||||
needs: [all-required-checks-complete]
|
needs: [all-required-checks-complete]
|
||||||
@@ -275,16 +406,24 @@ 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: nuget-package-tool
|
name: ${{ matrix.artifact }}
|
||||||
- name: Download NuGet artifact (lib)
|
- name: Compute package path
|
||||||
uses: actions/download-artifact@v4
|
id: compute-path
|
||||||
with:
|
run: |
|
||||||
name: nuget-package-lib
|
find . -maxdepth 1 -type f -name 'WoofWare.NUnitTestRunner.*.nupkg' -exec sh -c 'echo "output=$(basename "$1")" >> $GITHUB_OUTPUT' shell {} \;
|
||||||
- name: Tag and release plugin
|
- name: Compute tag name
|
||||||
|
id: compute-tag
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
NUPKG_PATH: ${{ steps.compute-path.outputs.output }}
|
||||||
run: sh .github/workflows/tag.sh
|
run: echo "output=$(basename "$NUPKG_PATH" .nupkg)" >> $GITHUB_OUTPUT
|
||||||
|
- name: Tag and release
|
||||||
|
uses: G-Research/common-actions/github-release@19d7281a0f9f83e13c78f99a610dbc80fc59ba3b
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
target-commitish: ${{ github.sha }}
|
||||||
|
tag: ${{ steps.compute-tag.outputs.output }}
|
||||||
|
binary-contents: ${{ steps.compute-path.outputs.output }}
|
||||||
|
54
.github/workflows/flake_update.yaml
vendored
Normal file
54
.github/workflows/flake_update.yaml
vendored
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
# yaml-language-server: $schema=https://raw.githubusercontent.com/SchemaStore/schemastore/master/src/schemas/json/github-workflow.json
|
||||||
|
name: Weekly Nix Flake Update
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '0 0 * * 0' # Runs at 00:00 every Sunday
|
||||||
|
workflow_dispatch: # Allows manual triggering
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update-nix-flake:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Check out repository
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Install Nix
|
||||||
|
uses: DeterminateSystems/nix-installer-action@main
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Update Nix flake
|
||||||
|
run: 'nix flake update'
|
||||||
|
|
||||||
|
- name: Build fetch-deps
|
||||||
|
run: 'nix build ".#default.fetch-deps"'
|
||||||
|
|
||||||
|
- name: Run fetch-deps
|
||||||
|
run: ./result nix/deps.json
|
||||||
|
|
||||||
|
- name: Format
|
||||||
|
run: 'nix develop --command alejandra .'
|
||||||
|
|
||||||
|
- name: Create token
|
||||||
|
id: generate-token
|
||||||
|
uses: actions/create-github-app-token@v2
|
||||||
|
with:
|
||||||
|
# https://github.com/actions/create-github-app-token/issues/136
|
||||||
|
app-id: ${{ secrets.APP_ID }}
|
||||||
|
private-key: ${{ secrets.APP_PRIVATE_KEY }}
|
||||||
|
|
||||||
|
- name: Raise pull request
|
||||||
|
uses: Smaug123/commit-action@cc25e6d80a796c49669dda4a0aa36c54c573983d
|
||||||
|
id: cpr
|
||||||
|
with:
|
||||||
|
bearer-token: ${{ steps.generate-token.outputs.token }}
|
||||||
|
pr-title: "Upgrade Nix flake and deps"
|
||||||
|
|
||||||
|
- name: Enable Pull Request Automerge
|
||||||
|
if: ${{ steps.cpr.outputs.pull-request-number }}
|
||||||
|
uses: peter-evans/enable-pull-request-automerge@v3
|
||||||
|
with:
|
||||||
|
token: ${{ steps.generate-token.outputs.token }}
|
||||||
|
pull-request-number: ${{ steps.cpr.outputs.pull-request-number }}
|
||||||
|
merge-method: squash
|
124
.github/workflows/tag.sh
vendored
124
.github/workflows/tag.sh
vendored
@@ -1,124 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
echo "Dry-run? $DRY_RUN!"
|
|
||||||
|
|
||||||
find . -maxdepth 1 -type f ! -name "$(printf "*\n*")" -name '*.nupkg' | while IFS= read -r file
|
|
||||||
do
|
|
||||||
tag=$(basename "$file" .nupkg)
|
|
||||||
git tag "$tag"
|
|
||||||
${DRY_RUN:+echo} git push origin "$tag"
|
|
||||||
done
|
|
||||||
|
|
||||||
export TAG
|
|
||||||
TAG=$(find . -maxdepth 1 -type f -name 'WoofWare.NUnitTestRunner.*.nupkg' -exec sh -c 'basename "$1" .nupkg' shell {} \; | grep -v Lib)
|
|
||||||
|
|
||||||
case "$TAG" in
|
|
||||||
*"
|
|
||||||
"*)
|
|
||||||
echo "Error: TAG contains a newline; multiple tools found."
|
|
||||||
exit 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# target_commitish empty indicates the repo default branch
|
|
||||||
IS_PRERELEASE="false"
|
|
||||||
if [ "${TAG#*prerelease}" != "$TAG" ]; then
|
|
||||||
IS_PRERELEASE="true"
|
|
||||||
fi
|
|
||||||
curl_body='{"tag_name":"'"$TAG"'","target_commitish":"","name":"'"$TAG"'","draft":false,"prerelease":'"$IS_PRERELEASE"',"generate_release_notes":false}'
|
|
||||||
|
|
||||||
echo "cURL body: $curl_body"
|
|
||||||
|
|
||||||
failed_output=$(cat <<'EOF'
|
|
||||||
{
|
|
||||||
"message": "Validation Failed",
|
|
||||||
"errors": [
|
|
||||||
{
|
|
||||||
"resource": "Release",
|
|
||||||
"code": "already_exists",
|
|
||||||
"field": "tag_name"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"documentation_url": "https://docs.github.com/rest/releases/releases#create-a-release"
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
)
|
|
||||||
|
|
||||||
success_output=$(cat <<'EOF'
|
|
||||||
{
|
|
||||||
"url": "https://api.github.com/repos/Smaug123/unofficial-nunit-runner/releases/158152116",
|
|
||||||
"assets_url": "https://api.github.com/repos/Smaug123/unofficial-nunit-runner/releases/158152116/assets",
|
|
||||||
"upload_url": "https://uploads.github.com/repos/Smaug123/unofficial-nunit-runner/releases/158152116/assets{?name,label}",
|
|
||||||
"html_url": "https://github.com/Smaug123/unofficial-nunit-runner/releases/tag/WoofWare.NUnitTestRunner.2.1.30",
|
|
||||||
"id": 158152116,
|
|
||||||
"author": {
|
|
||||||
"login": "github-actions[bot]",
|
|
||||||
"id": 41898282,
|
|
||||||
"node_id": "MDM6Qm90NDE4OTgyODI=",
|
|
||||||
"avatar_url": "https://avatars.githubusercontent.com/in/15368?v=4",
|
|
||||||
"gravatar_id": "",
|
|
||||||
"url": "https://api.github.com/users/github-actions%5Bbot%5D",
|
|
||||||
"html_url": "https://github.com/apps/github-actions",
|
|
||||||
"followers_url": "https://api.github.com/users/github-actions%5Bbot%5D/followers",
|
|
||||||
"following_url": "https://api.github.com/users/github-actions%5Bbot%5D/following{/other_user}",
|
|
||||||
"gists_url": "https://api.github.com/users/github-actions%5Bbot%5D/gists{/gist_id}",
|
|
||||||
"starred_url": "https://api.github.com/users/github-actions%5Bbot%5D/starred{/owner}{/repo}",
|
|
||||||
"subscriptions_url": "https://api.github.com/users/github-actions%5Bbot%5D/subscriptions",
|
|
||||||
"organizations_url": "https://api.github.com/users/github-actions%5Bbot%5D/orgs",
|
|
||||||
"repos_url": "https://api.github.com/users/github-actions%5Bbot%5D/repos",
|
|
||||||
"events_url": "https://api.github.com/users/github-actions%5Bbot%5D/events{/privacy}",
|
|
||||||
"received_events_url": "https://api.github.com/users/github-actions%5Bbot%5D/received_events",
|
|
||||||
"type": "Bot",
|
|
||||||
"site_admin": false
|
|
||||||
},
|
|
||||||
"node_id": "RE_kwDOJfksgc4JbTW0",
|
|
||||||
"tag_name": "WoofWare.NUnitTestRunner.2.1.30",
|
|
||||||
"target_commitish": "main",
|
|
||||||
"name": "WoofWare.NUnitTestRunner.2.1.30",
|
|
||||||
"draft": false,
|
|
||||||
"prerelease": false,
|
|
||||||
"created_at": "2024-05-30T11:00:55Z",
|
|
||||||
"published_at": "2024-05-30T11:03:02Z",
|
|
||||||
"assets": [
|
|
||||||
|
|
||||||
],
|
|
||||||
"tarball_url": "https://api.github.com/repos/Smaug123/unofficial-nunit-runner/tarball/WoofWare.NUnitTestRunner.2.1.30",
|
|
||||||
"zipball_url": "https://api.github.com/repos/Smaug123/unofficial-nunit-runner/zipball/WoofWare.NUnitTestRunner.2.1.30",
|
|
||||||
"body": null
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
)
|
|
||||||
|
|
||||||
HANDLE_OUTPUT=''
|
|
||||||
handle_error() {
|
|
||||||
ERROR_OUTPUT="$1"
|
|
||||||
exit_message=$(echo "$ERROR_OUTPUT" | jq -r --exit-status 'if .errors | length == 1 then .errors[0].code else null end')
|
|
||||||
if [ "$exit_message" = "already_exists" ] ; then
|
|
||||||
HANDLE_OUTPUT="Did not create GitHub release because it already exists at this version."
|
|
||||||
else
|
|
||||||
echo "Unexpected error output from curl: $(cat curl_output.json)"
|
|
||||||
echo "JQ output: $(exit_message)"
|
|
||||||
exit 2
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
run_tests() {
|
|
||||||
handle_error "$failed_output"
|
|
||||||
if [ "$HANDLE_OUTPUT" != "Did not create GitHub release because it already exists at this version." ]; then
|
|
||||||
echo "Bad output from handler: $HANDLE_OUTPUT"
|
|
||||||
exit 3
|
|
||||||
fi
|
|
||||||
HANDLE_OUTPUT=''
|
|
||||||
echo "Tests passed."
|
|
||||||
}
|
|
||||||
|
|
||||||
run_tests
|
|
||||||
|
|
||||||
if [ "$DRY_RUN" != 1 ] ; then
|
|
||||||
if curl --fail-with-body -L -X POST -H "Accept: application/vnd.github+json" -H "Authorization: Bearer $GITHUB_TOKEN" -H "X-GitHub-Api-Version: 2022-11-28" https://api.github.com/repos/Smaug123/unofficial-nunit-runner/releases -d "$curl_body" > curl_output.json; then
|
|
||||||
echo "Curl succeeded."
|
|
||||||
else
|
|
||||||
handle_error "$(cat curl_output.json)"
|
|
||||||
echo "$HANDLE_OUTPUT"
|
|
||||||
fi
|
|
||||||
fi
|
|
@@ -1,8 +1,7 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFrameworks>net9.0</TargetFrameworks>
|
||||||
|
|
||||||
<IsPackable>false</IsPackable>
|
<IsPackable>false</IsPackable>
|
||||||
<IsTestProject>true</IsTestProject>
|
<IsTestProject>true</IsTestProject>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
@@ -10,6 +9,14 @@
|
|||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="NoAttribute.fs" />
|
<Compile Include="NoAttribute.fs" />
|
||||||
<Compile Include="Inconclusive.fs" />
|
<Compile Include="Inconclusive.fs" />
|
||||||
|
<Compile Include="RunSubProcess.fs" />
|
||||||
|
<Compile Include="TestAsync.fs" />
|
||||||
|
<Compile Include="TestExplicit.fs" />
|
||||||
|
<Compile Include="TestNonParallel.fs" />
|
||||||
|
<Compile Include="TestParallel.fs" />
|
||||||
|
<Compile Include="TestParallelIndividualTest.fs" />
|
||||||
|
<Compile Include="TestStdout.fs" />
|
||||||
|
<Compile Include="TestParameterisedFixture.fs" />
|
||||||
<Compile Include="TestSetUp.fs" />
|
<Compile Include="TestSetUp.fs" />
|
||||||
<Compile Include="TestValues.fs" />
|
<Compile Include="TestValues.fs" />
|
||||||
<None Include="some-config.json">
|
<None Include="some-config.json">
|
||||||
@@ -21,10 +28,10 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="FsUnit" Version="6.0.0" />
|
<PackageReference Include="FsUnit" Version="7.1.1" />
|
||||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.10.0"/>
|
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.0"/>
|
||||||
<PackageReference Include="NUnit" Version="4.1.0"/>
|
<PackageReference Include="NUnit" Version="4.3.2"/>
|
||||||
<PackageReference Include="NUnit3TestAdapter" Version="4.5.0"/>
|
<PackageReference Include="NUnit3TestAdapter" Version="5.2.0"/>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
45
Consumer/RunSubProcess.fs
Normal file
45
Consumer/RunSubProcess.fs
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
namespace Consumer
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Diagnostics
|
||||||
|
open System.IO
|
||||||
|
open System.IO.Compression
|
||||||
|
open System.Text
|
||||||
|
open NUnit.Framework
|
||||||
|
open FsUnitTyped
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
module RunSubProcess =
|
||||||
|
[<Test>]
|
||||||
|
let ``Run a subprocess`` () =
|
||||||
|
let exe = "/bin/bash"
|
||||||
|
let args = [ "-c" ; "echo hi >&2 && echo bye" ]
|
||||||
|
let workingDir = None
|
||||||
|
|
||||||
|
let psi =
|
||||||
|
ProcessStartInfo (
|
||||||
|
exe,
|
||||||
|
UseShellExecute = false,
|
||||||
|
RedirectStandardError = true,
|
||||||
|
RedirectStandardOutput = true,
|
||||||
|
WorkingDirectory = Option.toObj workingDir
|
||||||
|
)
|
||||||
|
|
||||||
|
for arg in args do
|
||||||
|
psi.ArgumentList.Add arg
|
||||||
|
|
||||||
|
psi.EnvironmentVariables.Add ("THING", Path.Combine (AppDomain.CurrentDomain.BaseDirectory, "hi"))
|
||||||
|
let stderr = StringBuilder ()
|
||||||
|
use proc = new Process (StartInfo = psi)
|
||||||
|
proc.OutputDataReceived.Add (fun e -> printfn $"%s{e.Data}")
|
||||||
|
|
||||||
|
proc.ErrorDataReceived.Add (fun e ->
|
||||||
|
eprintfn $"%s{e.Data}"
|
||||||
|
stderr.AppendLine e.Data |> ignore
|
||||||
|
)
|
||||||
|
|
||||||
|
proc.Start () |> shouldEqual true
|
||||||
|
proc.BeginOutputReadLine ()
|
||||||
|
proc.BeginErrorReadLine ()
|
||||||
|
|
||||||
|
proc.WaitForExit ()
|
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
|
||||||
|
}
|
@@ -35,6 +35,13 @@ module TestCaseData =
|
|||||||
[<TestCaseSource(nameof optionalRaw)>]
|
[<TestCaseSource(nameof optionalRaw)>]
|
||||||
let ``Consume options, raw`` (s : string option) : unit = s |> shouldEqual s
|
let ``Consume options, raw`` (s : string option) : unit = s |> shouldEqual s
|
||||||
|
|
||||||
|
[<TestCase(30, 15, 44, false)>]
|
||||||
|
let bug66 (i : int, j : int, k : int, l : bool) =
|
||||||
|
i |> shouldEqual 30
|
||||||
|
j |> shouldEqual 15
|
||||||
|
k |> shouldEqual 44
|
||||||
|
l |> shouldEqual false
|
||||||
|
|
||||||
[<OneTimeTearDown>]
|
[<OneTimeTearDown>]
|
||||||
let tearDown () =
|
let tearDown () =
|
||||||
testCasesSeen
|
testCasesSeen
|
||||||
|
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"
|
19
Consumer/TestNonParallel.fs
Normal file
19
Consumer/TestNonParallel.fs
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
namespace Consumer
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Threading
|
||||||
|
open NUnit.Framework
|
||||||
|
open FsUnitTyped
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
[<NonParallelizable>]
|
||||||
|
module TestNonParallel =
|
||||||
|
let defaults = List.init 40 id
|
||||||
|
let lock = ref 0
|
||||||
|
|
||||||
|
[<TestCaseSource(nameof defaults)>]
|
||||||
|
let ``Default thing, but not parallel`` (i : int) =
|
||||||
|
Interlocked.Increment lock |> shouldEqual 1
|
||||||
|
Thread.Sleep (TimeSpan.FromMilliseconds (float i))
|
||||||
|
lock.Value <- 0
|
||||||
|
i |> shouldEqual i
|
66
Consumer/TestParallel.fs
Normal file
66
Consumer/TestParallel.fs
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
namespace Consumer
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Threading
|
||||||
|
open NUnit.Framework
|
||||||
|
open FsUnitTyped
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
[<Parallelizable>]
|
||||||
|
module TestParallelDefault =
|
||||||
|
|
||||||
|
let defaults = List.init 60 id
|
||||||
|
|
||||||
|
[<TestCaseSource(nameof defaults)>]
|
||||||
|
let ``Default thing, no scope`` (i : int) =
|
||||||
|
Console.WriteLine i
|
||||||
|
Thread.Sleep (TimeSpan.FromMilliseconds (float i))
|
||||||
|
i |> shouldEqual i
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
[<Parallelizable(ParallelScope.All)>]
|
||||||
|
module TestParallelAllScope =
|
||||||
|
|
||||||
|
let defaults = List.init 60 id
|
||||||
|
|
||||||
|
[<TestCaseSource(nameof defaults)>]
|
||||||
|
let ``Thing, all scope`` (i : int) =
|
||||||
|
Console.WriteLine i
|
||||||
|
Thread.Sleep (TimeSpan.FromMilliseconds (float i))
|
||||||
|
i |> shouldEqual i
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
[<Parallelizable(ParallelScope.Self)>]
|
||||||
|
module TestParallelSelfScope =
|
||||||
|
|
||||||
|
let defaults = List.init 60 id
|
||||||
|
|
||||||
|
[<TestCaseSource(nameof defaults)>]
|
||||||
|
let ``Thing, self scope`` (i : int) =
|
||||||
|
Console.WriteLine i
|
||||||
|
Thread.Sleep (TimeSpan.FromMilliseconds (float i))
|
||||||
|
i |> shouldEqual i
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
[<Parallelizable(ParallelScope.Children)>]
|
||||||
|
module TestParallelChildrenScope =
|
||||||
|
|
||||||
|
let defaults = List.init 60 id
|
||||||
|
|
||||||
|
[<TestCaseSource(nameof defaults)>]
|
||||||
|
let ``Thing, children scope`` (i : int) =
|
||||||
|
Console.WriteLine i
|
||||||
|
Thread.Sleep (TimeSpan.FromMilliseconds (float i))
|
||||||
|
i |> shouldEqual i
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
[<Parallelizable(ParallelScope.Fixtures)>]
|
||||||
|
module TestParallelFixturesScope =
|
||||||
|
|
||||||
|
let defaults = List.init 60 id
|
||||||
|
|
||||||
|
[<TestCaseSource(nameof defaults)>]
|
||||||
|
let ``Thing, fixtures scope`` (i : int) =
|
||||||
|
Console.WriteLine i
|
||||||
|
Thread.Sleep (TimeSpan.FromMilliseconds (float i))
|
||||||
|
i |> shouldEqual i
|
59
Consumer/TestParallelIndividualTest.fs
Normal file
59
Consumer/TestParallelIndividualTest.fs
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
namespace Consumer
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Collections.Concurrent
|
||||||
|
open System.Threading
|
||||||
|
open NUnit.Framework
|
||||||
|
open FsUnitTyped
|
||||||
|
|
||||||
|
// These tests are flaky if the bug https://github.com/Smaug123/unofficial-nunit-runner/issues/168 is unfixed.
|
||||||
|
[<TestFixture>]
|
||||||
|
module TestParallelIndividualTest =
|
||||||
|
|
||||||
|
type private Transitions =
|
||||||
|
| Started of int
|
||||||
|
| LockAcquired of int
|
||||||
|
| Exited of int
|
||||||
|
|
||||||
|
let locker = obj ()
|
||||||
|
let private sequence = ConcurrentQueue<Transitions> ()
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
[<Parallelizable(ParallelScope.None)>]
|
||||||
|
let ``does not run in parallel`` () =
|
||||||
|
sequence.Enqueue (Transitions.Started 0)
|
||||||
|
let entered = Monitor.TryEnter (locker, TimeSpan.Zero)
|
||||||
|
|
||||||
|
if entered then
|
||||||
|
sequence.Enqueue (Transitions.LockAcquired 0)
|
||||||
|
Monitor.Exit locker
|
||||||
|
sequence.Enqueue (Transitions.Exited 0)
|
||||||
|
else
|
||||||
|
sequence.Enqueue (Transitions.Exited 0)
|
||||||
|
failwith "failed to acquire the lock"
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
let ``unrestricted parallelism`` () =
|
||||||
|
sequence.Enqueue (Transitions.Started 1)
|
||||||
|
let entered = Monitor.TryEnter (locker, TimeSpan.Zero)
|
||||||
|
|
||||||
|
if entered then
|
||||||
|
sequence.Enqueue (Transitions.LockAcquired 1)
|
||||||
|
Monitor.Exit locker
|
||||||
|
sequence.Enqueue (Transitions.Exited 1)
|
||||||
|
else
|
||||||
|
sequence.Enqueue (Transitions.Exited 1)
|
||||||
|
failwith "failed to acquire the lock"
|
||||||
|
|
||||||
|
[<OneTimeTearDown>]
|
||||||
|
let ``It worked`` () =
|
||||||
|
let sequence = sequence |> Seq.toList
|
||||||
|
|
||||||
|
let allowed n =
|
||||||
|
[ Transitions.Started n ; Transitions.LockAcquired n ; Transitions.Exited n ]
|
||||||
|
|
||||||
|
if sequence <> allowed 0 @ allowed 1 && sequence <> allowed 1 @ allowed 0 then
|
||||||
|
let s = sequence |> Seq.map string<Transitions> |> String.concat "\n"
|
||||||
|
failwith $"Unexpected sequence!\n%s{s}"
|
||||||
|
|
||||||
|
()
|
18
Consumer/TestParameterisedFixture.fs
Normal file
18
Consumer/TestParameterisedFixture.fs
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
namespace Consumer
|
||||||
|
|
||||||
|
open NUnit.Framework
|
||||||
|
open FsUnitTyped
|
||||||
|
|
||||||
|
[<TestFixture true>]
|
||||||
|
[<TestFixture false>]
|
||||||
|
type TestParameterisedFixture (v : bool) =
|
||||||
|
[<Test>]
|
||||||
|
member _.Thing () = v |> shouldEqual v
|
||||||
|
|
||||||
|
[<TestFixture(3, true)>]
|
||||||
|
[<TestFixture(6, false)>]
|
||||||
|
type TestParameterisedFixtureMultiple (i : int, v : bool) =
|
||||||
|
[<Test>]
|
||||||
|
member _.Thing () =
|
||||||
|
v |> shouldEqual v
|
||||||
|
i |> shouldEqual i
|
@@ -1,5 +1,6 @@
|
|||||||
namespace Consumer
|
namespace Consumer
|
||||||
|
|
||||||
|
open System
|
||||||
open FsUnitTyped
|
open FsUnitTyped
|
||||||
open System.Threading
|
open System.Threading
|
||||||
open NUnit.Framework
|
open NUnit.Framework
|
||||||
@@ -11,6 +12,8 @@ module TestSetUp =
|
|||||||
|
|
||||||
[<OneTimeSetUp>]
|
[<OneTimeSetUp>]
|
||||||
let oneTimeSetUp () =
|
let oneTimeSetUp () =
|
||||||
|
Console.WriteLine "I'm being set up for the first time!"
|
||||||
|
|
||||||
if Interlocked.Increment haveOneTimeSetUp <> 1 then
|
if Interlocked.Increment haveOneTimeSetUp <> 1 then
|
||||||
failwith "one time setup happened more than once"
|
failwith "one time setup happened more than once"
|
||||||
|
|
||||||
@@ -22,12 +25,14 @@ module TestSetUp =
|
|||||||
|
|
||||||
[<SetUp>]
|
[<SetUp>]
|
||||||
let setUp () =
|
let setUp () =
|
||||||
|
Console.WriteLine "It's a set-up!"
|
||||||
haveOneTimeSetUp.Value |> shouldEqual 1
|
haveOneTimeSetUp.Value |> shouldEqual 1
|
||||||
let newId = Interlocked.Increment setUpTimes
|
let newId = Interlocked.Increment setUpTimes
|
||||||
lock setUpTimesSeen (fun () -> setUpTimesSeen.Add newId)
|
lock setUpTimesSeen (fun () -> setUpTimesSeen.Add newId)
|
||||||
|
|
||||||
[<TearDown>]
|
[<TearDown>]
|
||||||
let tearDown () =
|
let tearDown () =
|
||||||
|
Console.WriteLine "I'm a tear-down!"
|
||||||
let newId = Interlocked.Increment tearDownTimes
|
let newId = Interlocked.Increment tearDownTimes
|
||||||
lock tearDownTimesSeen (fun () -> tearDownTimesSeen.Add newId)
|
lock tearDownTimesSeen (fun () -> tearDownTimesSeen.Add newId)
|
||||||
|
|
||||||
@@ -35,19 +40,23 @@ module TestSetUp =
|
|||||||
|
|
||||||
[<OneTimeTearDown>]
|
[<OneTimeTearDown>]
|
||||||
let oneTimeTearDown () =
|
let oneTimeTearDown () =
|
||||||
|
Console.WriteLine "I'm being torn down, finally!"
|
||||||
|
|
||||||
if Interlocked.Increment haveOneTimeTearDown <> 1 then
|
if Interlocked.Increment haveOneTimeTearDown <> 1 then
|
||||||
failwith "one time tear down happened more than once"
|
failwith "one time tear down happened more than once"
|
||||||
|
|
||||||
setUpTimesSeen
|
setUpTimesSeen
|
||||||
|> Seq.toList
|
|> Seq.toList
|
||||||
|
|> List.sort
|
||||||
// Six tests: one for Test, two for the TestCase, three for the Repeat.
|
// Six tests: one for Test, two for the TestCase, three for the Repeat.
|
||||||
|> shouldEqual [ 1..6 ]
|
|> shouldEqual [ 1..6 ]
|
||||||
|
|
||||||
tearDownTimesSeen |> Seq.toList |> shouldEqual [ 1..6 ]
|
tearDownTimesSeen |> Seq.toList |> List.sort |> shouldEqual [ 1..6 ]
|
||||||
|
|
||||||
[<Test>]
|
[<Test>]
|
||||||
let ``Test 1`` () =
|
let ``Test 1`` () =
|
||||||
haveOneTimeTearDown.Value |> shouldEqual 0
|
haveOneTimeTearDown.Value |> shouldEqual 0
|
||||||
|
Console.WriteLine "By the way, I'm test 1"
|
||||||
1 |> shouldEqual 1
|
1 |> shouldEqual 1
|
||||||
|
|
||||||
[<TestCase "h">]
|
[<TestCase "h">]
|
||||||
|
13
Consumer/TestStdout.fs
Normal file
13
Consumer/TestStdout.fs
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
namespace Consumer
|
||||||
|
|
||||||
|
open System
|
||||||
|
open NUnit.Framework
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
module TestStdout =
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
let ``Stdout is redirected`` () =
|
||||||
|
Console.Out.WriteLine "Hi!"
|
||||||
|
Console.WriteLine "Hi! part 2"
|
||||||
|
Console.Error.WriteLine "Bye!"
|
@@ -86,14 +86,42 @@ module TestValues =
|
|||||||
|
|
||||||
[<OneTimeTearDown>]
|
[<OneTimeTearDown>]
|
||||||
let ``Values are all OK`` () =
|
let ``Values are all OK`` () =
|
||||||
seen1 |> Seq.toList |> shouldEqual [ true ; false ]
|
seen1 |> Seq.toList |> List.sort |> shouldEqual [ false ; true ]
|
||||||
seen2 |> Seq.toList |> shouldEqual [ (true, false) ; (false, true) ]
|
|
||||||
seen3 |> Seq.toList |> shouldEqual [ (88, box 29) ; (31, box 0) ]
|
seen2
|
||||||
seen4 |> Seq.toList |> shouldEqual [ ("hi", box "ohh") ; ("bye", null) ]
|
|> Seq.toList
|
||||||
seen5 |> Seq.toList |> shouldEqual [ (88, box 29) ; (31, box 29) ]
|
|> List.sort
|
||||||
seen6 |> Seq.toList |> shouldEqual [ ("hi", box "ohh") ; ("bye", box "ohh") ]
|
|> shouldEqual [ (false, true) ; (true, false) ]
|
||||||
seen7 |> Seq.toList |> shouldEqual [ (88, box 29) ; (31, box 29) ]
|
|
||||||
seen8 |> Seq.toList |> shouldEqual [ ("hi", box "ohh") ; ("bye", box "ohh") ]
|
seen3
|
||||||
|
|> Seq.toList
|
||||||
|
|> List.sortBy fst
|
||||||
|
|> shouldEqual [ (31, box 0) ; (88, box 29) ]
|
||||||
|
|
||||||
|
seen4
|
||||||
|
|> Seq.toList
|
||||||
|
|> List.sortBy fst
|
||||||
|
|> shouldEqual [ ("bye", null) ; ("hi", box "ohh") ]
|
||||||
|
|
||||||
|
seen5
|
||||||
|
|> Seq.toList
|
||||||
|
|> List.sortBy fst
|
||||||
|
|> shouldEqual [ (31, box 29) ; (88, box 29) ]
|
||||||
|
|
||||||
|
seen6
|
||||||
|
|> Seq.toList
|
||||||
|
|> List.sortBy fst
|
||||||
|
|> shouldEqual [ ("bye", box "ohh") ; ("hi", box "ohh") ]
|
||||||
|
|
||||||
|
seen7
|
||||||
|
|> Seq.toList
|
||||||
|
|> List.sortBy fst
|
||||||
|
|> shouldEqual [ (31, box 29) ; (88, box 29) ]
|
||||||
|
|
||||||
|
seen8
|
||||||
|
|> Seq.toList
|
||||||
|
|> List.sortBy fst
|
||||||
|
|> shouldEqual [ ("bye", box "ohh") ; ("hi", box "ohh") ]
|
||||||
|
|
||||||
seen9
|
seen9
|
||||||
|> Seq.toList
|
|> Seq.toList
|
||||||
|
@@ -10,7 +10,7 @@
|
|||||||
<WarnOn>FS3388,FS3559</WarnOn>
|
<WarnOn>FS3388,FS3559</WarnOn>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Nerdbank.GitVersioning" Version="3.6.139" 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>
|
||||||
|
20
FailingConsumer/FailingConsumer.fsproj
Normal file
20
FailingConsumer/FailingConsumer.fsproj
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
|
<IsPackable>false</IsPackable>
|
||||||
|
<IsTestProject>true</IsTestProject>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Compile Include="TestInsufficientArgs.fs" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="FsUnit" Version="7.1.1" />
|
||||||
|
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.0"/>
|
||||||
|
<PackageReference Include="NUnit" Version="4.3.2"/>
|
||||||
|
<PackageReference Include="NUnit3TestAdapter" Version="5.2.0"/>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
9
FailingConsumer/TestInsufficientArgs.fs
Normal file
9
FailingConsumer/TestInsufficientArgs.fs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
namespace FailingConsumer
|
||||||
|
|
||||||
|
open NUnit.Framework
|
||||||
|
|
||||||
|
[<TestFixture>]
|
||||||
|
module TestInsufficientArgs =
|
||||||
|
|
||||||
|
[<Test>]
|
||||||
|
let foo (_ : int) = ()
|
1
FailingConsumer/expected.txt
Normal file
1
FailingConsumer/expected.txt
Normal file
@@ -0,0 +1 @@
|
|||||||
|
foo: had parameter count mismatch: expected 1, actual 0
|
@@ -8,3 +8,12 @@ To supply special characters in a string, XML-encode them and `"quote"` the stri
|
|||||||
We support at least the [documented `dotnet test` examples](https://learn.microsoft.com/en-us/dotnet/core/testing/selective-unit-tests).
|
We support at least the [documented `dotnet test` examples](https://learn.microsoft.com/en-us/dotnet/core/testing/selective-unit-tests).
|
||||||
However, we would recommend phrasing some of them differently, for maximum peace of mind:
|
However, we would recommend phrasing some of them differently, for maximum peace of mind:
|
||||||
* `FullyQualifiedName=MyNamespace.MyTestsClass<ParameterType1%2CParameterType2>.MyTestMethod`. This would be better phrased with quotes and escaping as `FullyQualifiedName="MyNamespace.MyTestsClass<ParameterType1%2CParameterType2>.MyTestMethod"`
|
* `FullyQualifiedName=MyNamespace.MyTestsClass<ParameterType1%2CParameterType2>.MyTestMethod`. This would be better phrased with quotes and escaping as `FullyQualifiedName="MyNamespace.MyTestsClass<ParameterType1%2CParameterType2>.MyTestMethod"`
|
||||||
|
|
||||||
|
## 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).
|
||||||
|
|
||||||
|
# Licence
|
||||||
|
|
||||||
|
WoofWare.NUnitTestRunner is licensed to you under the MIT licence, a copy of which can be found at [LICENSE](./LICENSE).
|
||||||
|
99
WoofWare.NUnitTestRunner.Lib/Args.fs
Normal file
99
WoofWare.NUnitTestRunner.Lib/Args.fs
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.IO
|
||||||
|
|
||||||
|
[<AutoOpen>]
|
||||||
|
module internal Patterns =
|
||||||
|
let (|Key|_|) (start : string) (s : string) : string option =
|
||||||
|
if s.StartsWith (start + "=", StringComparison.Ordinal) then
|
||||||
|
s.Substring (start.Length + 1) |> Some
|
||||||
|
else
|
||||||
|
None
|
||||||
|
|
||||||
|
/// Represents how verbose the test runner's logging should be.
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
type LogLevel =
|
||||||
|
/// Don't log any information about the test run.
|
||||||
|
| Nothing
|
||||||
|
/// Log as much information as is available about the test run.
|
||||||
|
| Verbose
|
||||||
|
|
||||||
|
/// Arguments controlling the test runner itself (not the tests therein).
|
||||||
|
type Args =
|
||||||
|
{
|
||||||
|
/// The DLL containing the tests we'll reflectively discover and invoke.
|
||||||
|
Dll : FileInfo
|
||||||
|
/// If set, the output file into which we will write a TRX report. (We'll create parent directories as necessary.)
|
||||||
|
Trx : FileInfo option
|
||||||
|
/// Also contains the original string which specified the filter.
|
||||||
|
Filter : (string * Filter) option
|
||||||
|
/// How verbose to be with the test runner's own logging.
|
||||||
|
Logging : LogLevel
|
||||||
|
/// Maximum number of tests which can run concurrently. This setting overrides any LevelOfParallelism reflectively
|
||||||
|
/// extracted from the assembly under test.
|
||||||
|
LevelOfParallelism : int option
|
||||||
|
/// Abort if the test runner is running for longer than this timeout.
|
||||||
|
Timeout : TimeSpan option
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parse `argv` into a structured Args.
|
||||||
|
static member Parse (args : string list) : Args =
|
||||||
|
match args with
|
||||||
|
| [] -> failwith "The first arg must be a positional arg, the DLL to test."
|
||||||
|
| dll :: args ->
|
||||||
|
|
||||||
|
let rec go
|
||||||
|
(trx : FileInfo option)
|
||||||
|
(filter : (string * Filter) option)
|
||||||
|
(logging : LogLevel option)
|
||||||
|
(par : int option)
|
||||||
|
(timeout : TimeSpan option)
|
||||||
|
(args : string list)
|
||||||
|
=
|
||||||
|
match args with
|
||||||
|
| [] ->
|
||||||
|
{
|
||||||
|
Dll = FileInfo dll
|
||||||
|
Trx = trx
|
||||||
|
Filter = filter
|
||||||
|
Logging = logging |> Option.defaultValue LogLevel.Nothing
|
||||||
|
LevelOfParallelism = par
|
||||||
|
Timeout = timeout
|
||||||
|
}
|
||||||
|
| Key "--filter" filterStr :: rest
|
||||||
|
| "--filter" :: filterStr :: rest ->
|
||||||
|
match filter with
|
||||||
|
| Some _ -> failwith "Two conflicting filters; you can only specify --filter once"
|
||||||
|
| None -> go trx (Some (filterStr, Filter.parse filterStr)) logging par timeout rest
|
||||||
|
| Key "--trx" trxStr :: rest
|
||||||
|
| "--trx" :: trxStr :: rest ->
|
||||||
|
match trx with
|
||||||
|
| Some _ -> failwith "Two conflicting TRX outputs; you can only specify --trx once"
|
||||||
|
| None -> go (Some (FileInfo trxStr)) filter logging par timeout rest
|
||||||
|
| Key "--verbose" verboseStr :: rest
|
||||||
|
| "--verbose" :: verboseStr :: rest ->
|
||||||
|
match logging with
|
||||||
|
| Some _ -> failwith "Two conflicting --verbose outputs; you can only specify --verbose once"
|
||||||
|
| None ->
|
||||||
|
let verbose =
|
||||||
|
if Boolean.Parse verboseStr then
|
||||||
|
LogLevel.Verbose
|
||||||
|
else
|
||||||
|
LogLevel.Nothing
|
||||||
|
|
||||||
|
go trx filter (Some verbose) par timeout rest
|
||||||
|
| Key "--parallelism" parStr :: rest
|
||||||
|
| "--parallelism" :: parStr :: rest ->
|
||||||
|
match par with
|
||||||
|
| Some _ -> failwith "Two conflicting --parallelism outputs; you can only specify --parallelism once"
|
||||||
|
| None -> go trx filter logging (Some (Int32.Parse parStr)) timeout rest
|
||||||
|
| Key "--timeout-seconds" timeoutStr :: rest
|
||||||
|
| "--timeout-seconds" :: timeoutStr :: rest ->
|
||||||
|
match timeout with
|
||||||
|
| Some _ ->
|
||||||
|
failwith "Two conflicting --timeout-seconds outputs; you can only specify --timeout-seconds once"
|
||||||
|
| None -> go trx filter logging par (Some (TimeSpan.FromSeconds (Int32.Parse timeoutStr |> float))) rest
|
||||||
|
| k :: _rest -> failwith $"Unrecognised arg %s{k}"
|
||||||
|
|
||||||
|
go None None None None None args
|
62
WoofWare.NUnitTestRunner.Lib/AssemblyLevelAttributes.fs
Normal file
62
WoofWare.NUnitTestRunner.Lib/AssemblyLevelAttributes.fs
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System.Reflection
|
||||||
|
|
||||||
|
/// Attributes at the assembly level which control the behaviour of NUnit.
|
||||||
|
type AssemblyLevelAttributes =
|
||||||
|
{
|
||||||
|
/// How many tests can be running at once, if anything's running in parallel.
|
||||||
|
Parallelism : int option
|
||||||
|
/// Whether the tests in this assembly can be parallelised at all.
|
||||||
|
Parallelizable : Parallelizable<AssemblyParallelScope> option
|
||||||
|
}
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module AssemblyLevelAttributes =
|
||||||
|
|
||||||
|
/// Reflectively obtain the values of any relevant assembly attributes.
|
||||||
|
let get (assy : Assembly) : AssemblyLevelAttributes =
|
||||||
|
((None, None), assy.CustomAttributes)
|
||||||
|
||> Seq.fold (fun (levelPar, par) attr ->
|
||||||
|
match attr.AttributeType.FullName with
|
||||||
|
| "NUnit.Framework.LevelOfParallelismAttribute" ->
|
||||||
|
let arg = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<int>
|
||||||
|
|
||||||
|
match levelPar with
|
||||||
|
| None -> (Some arg, par)
|
||||||
|
| Some existing ->
|
||||||
|
failwith $"Assembly %s{assy.Location} declares parallelism %i{arg} and also %i{existing}"
|
||||||
|
| "NUnit.Framework.NonParallelizableAttribute" ->
|
||||||
|
match levelPar with
|
||||||
|
| None -> (Some 1, par)
|
||||||
|
| Some existing ->
|
||||||
|
failwith
|
||||||
|
$"Assembly %s{assy.Location} declares non-parallelizable and also parallelism %i{existing}"
|
||||||
|
| "NUnit.Framework.ParallelizableAttribute" ->
|
||||||
|
match par with
|
||||||
|
| Some _ -> failwith "Got multiple Parallelize attributes in assembly"
|
||||||
|
| None ->
|
||||||
|
match attr.ConstructorArguments |> Seq.toList with
|
||||||
|
| [] -> levelPar, Some (Parallelizable.Yes AssemblyParallelScope.Fixtures)
|
||||||
|
| [ v ] ->
|
||||||
|
match v.Value with
|
||||||
|
| :? int as v ->
|
||||||
|
match ParallelScope.ofInt v with
|
||||||
|
| ParallelScope.Fixtures ->
|
||||||
|
levelPar, Some (Parallelizable.Yes AssemblyParallelScope.Fixtures)
|
||||||
|
| ParallelScope.Children ->
|
||||||
|
levelPar, Some (Parallelizable.Yes AssemblyParallelScope.Children)
|
||||||
|
| ParallelScope.None -> levelPar, Some Parallelizable.No
|
||||||
|
| ParallelScope.All ->
|
||||||
|
failwith "ParallelScope.All is invalid on assemblies; only Fixtures or Children"
|
||||||
|
| ParallelScope.Self ->
|
||||||
|
failwith "ParallelScope.Self is invalid on assemblies; only Fixtures or Children"
|
||||||
|
| v -> failwith $"Unexpectedly non-int value %O{v} of parallel scope on assembly"
|
||||||
|
| _ -> failwith "unexpectedly got multiple args to Parallelizable on assembly"
|
||||||
|
| _ -> levelPar, par
|
||||||
|
)
|
||||||
|
|> fun (par, canPar) ->
|
||||||
|
{
|
||||||
|
Parallelizable = canPar
|
||||||
|
Parallelism = par
|
||||||
|
}
|
173
WoofWare.NUnitTestRunner.Lib/Context.fs
Normal file
173
WoofWare.NUnitTestRunner.Lib/Context.fs
Normal file
@@ -0,0 +1,173 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Collections.Generic
|
||||||
|
open System.IO
|
||||||
|
open System.Reflection
|
||||||
|
open System.Runtime.Loader
|
||||||
|
open System.Text
|
||||||
|
open System.Threading
|
||||||
|
|
||||||
|
type internal OutputStreamId = | OutputStreamId of Guid
|
||||||
|
|
||||||
|
type private ThreadAwareWriter (local : AsyncLocal<OutputStreamId>, underlying : Dictionary<OutputStreamId, TextWriter>)
|
||||||
|
=
|
||||||
|
inherit TextWriter ()
|
||||||
|
override _.get_Encoding () = Encoding.Default
|
||||||
|
|
||||||
|
override this.Write (v : char) : unit =
|
||||||
|
lock
|
||||||
|
underlying
|
||||||
|
(fun () ->
|
||||||
|
match underlying.TryGetValue local.Value with
|
||||||
|
| true, output -> output.Write v
|
||||||
|
| false, _ ->
|
||||||
|
let wanted =
|
||||||
|
underlying |> Seq.map (fun (KeyValue (a, b)) -> $"%O{a}") |> String.concat "\n"
|
||||||
|
|
||||||
|
failwith $"no such context: %O{local.Value}\nwanted:\n{wanted}"
|
||||||
|
)
|
||||||
|
|
||||||
|
override this.WriteLine (v : string) : unit =
|
||||||
|
lock
|
||||||
|
underlying
|
||||||
|
(fun () ->
|
||||||
|
match underlying.TryGetValue local.Value with
|
||||||
|
| true, output -> output.WriteLine v
|
||||||
|
| false, _ ->
|
||||||
|
let wanted =
|
||||||
|
underlying |> Seq.map (fun (KeyValue (a, b)) -> $"%O{a}") |> String.concat "\n"
|
||||||
|
|
||||||
|
failwith $"no such context: %O{local.Value}\nwanted:\n{wanted}"
|
||||||
|
)
|
||||||
|
|
||||||
|
/// Wraps up the necessary context to intercept global state.
|
||||||
|
[<NoEquality ; NoComparison>]
|
||||||
|
type TestContexts =
|
||||||
|
internal
|
||||||
|
{
|
||||||
|
/// Accesses to this must be locked on StdOutWriters.
|
||||||
|
StdOuts : Dictionary<OutputStreamId, MemoryStream>
|
||||||
|
/// Accesses to this must be locked on StdErrWriters.
|
||||||
|
StdErrs : Dictionary<OutputStreamId, MemoryStream>
|
||||||
|
StdOutWriters : Dictionary<OutputStreamId, TextWriter>
|
||||||
|
StdErrWriters : Dictionary<OutputStreamId, TextWriter>
|
||||||
|
StdOutWriter : TextWriter
|
||||||
|
StdErrWriter : TextWriter
|
||||||
|
AsyncLocal : AsyncLocal<OutputStreamId>
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Call this exactly once.
|
||||||
|
static member Empty () =
|
||||||
|
let stdouts = Dictionary ()
|
||||||
|
let stderrs = Dictionary ()
|
||||||
|
let stdoutWriters = Dictionary ()
|
||||||
|
let stderrWriters = Dictionary ()
|
||||||
|
let local = AsyncLocal ()
|
||||||
|
let stdoutWriter = new ThreadAwareWriter (local, stdoutWriters)
|
||||||
|
let stderrWriter = new ThreadAwareWriter (local, stderrWriters)
|
||||||
|
|
||||||
|
{
|
||||||
|
StdOuts = stdouts
|
||||||
|
StdErrs = stderrs
|
||||||
|
StdOutWriter = stdoutWriter
|
||||||
|
StdErrWriter = stderrWriter
|
||||||
|
StdOutWriters = stdoutWriters
|
||||||
|
StdErrWriters = stderrWriters
|
||||||
|
AsyncLocal = local
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An output stream which will identify the ExecutionContext it's being written to from,
|
||||||
|
/// and will separate that output into its own stream internally.
|
||||||
|
member this.Stdout : TextWriter = this.StdOutWriter
|
||||||
|
|
||||||
|
/// An output stream which will identify the ExecutionContext it's being written to from,
|
||||||
|
/// and will separate that output into its own stream internally.
|
||||||
|
member this.Stderr : TextWriter = this.StdErrWriter
|
||||||
|
|
||||||
|
member internal this.DumpStdout (id : OutputStreamId) : string =
|
||||||
|
lock
|
||||||
|
this.StdOutWriters
|
||||||
|
(fun () ->
|
||||||
|
this.StdOutWriters.[id].Flush ()
|
||||||
|
this.StdOuts.[id].ToArray ()
|
||||||
|
)
|
||||||
|
|> Encoding.Default.GetString
|
||||||
|
|
||||||
|
member internal this.DumpStderr (id : OutputStreamId) : string =
|
||||||
|
lock
|
||||||
|
this.StdErrWriters
|
||||||
|
(fun () ->
|
||||||
|
this.StdErrWriters.[id].Flush ()
|
||||||
|
this.StdErrs.[id].ToArray ()
|
||||||
|
)
|
||||||
|
|> Encoding.Default.GetString
|
||||||
|
|
||||||
|
member internal this.NewOutputs () =
|
||||||
|
let id = Guid.NewGuid () |> OutputStreamId
|
||||||
|
let msOut = new MemoryStream ()
|
||||||
|
let wrOut = new StreamWriter (msOut)
|
||||||
|
let msErr = new MemoryStream ()
|
||||||
|
let wrErr = new StreamWriter (msErr)
|
||||||
|
|
||||||
|
lock
|
||||||
|
this.StdOutWriters
|
||||||
|
(fun () ->
|
||||||
|
this.StdOutWriters.Add (id, wrOut)
|
||||||
|
this.StdOuts.Add (id, msOut)
|
||||||
|
)
|
||||||
|
|
||||||
|
lock
|
||||||
|
this.StdErrWriters
|
||||||
|
(fun () ->
|
||||||
|
this.StdErrWriters.Add (id, wrErr)
|
||||||
|
this.StdErrs.Add (id, msErr)
|
||||||
|
)
|
||||||
|
|
||||||
|
id
|
||||||
|
|
||||||
|
interface IDisposable with
|
||||||
|
member this.Dispose () =
|
||||||
|
// TODO: dispose the streams
|
||||||
|
()
|
||||||
|
|
||||||
|
/// A separate AssemblyLoadContext within which you can run the tests in the given DLL.
|
||||||
|
/// Supply places to find the .NET runtimes.
|
||||||
|
type LoadContext (dll : FileInfo, runtimes : DirectoryInfo list, contexts : TestContexts) =
|
||||||
|
inherit AssemblyLoadContext ()
|
||||||
|
|
||||||
|
/// Load the assembly with the given name into this assembly context.
|
||||||
|
/// This additionally monkey-patches System.Console: it performs SetOut and SetError on them
|
||||||
|
/// so that they redirect their outputs into the given `TestContexts`.
|
||||||
|
override this.Load (target : AssemblyName) : Assembly =
|
||||||
|
let path = Path.Combine (dll.Directory.FullName, $"%s{target.Name}.dll")
|
||||||
|
|
||||||
|
let assy =
|
||||||
|
if File.Exists path then
|
||||||
|
this.LoadFromAssemblyPath path
|
||||||
|
else
|
||||||
|
|
||||||
|
runtimes
|
||||||
|
|> List.tryPick (fun di ->
|
||||||
|
let path = Path.Combine (di.FullName, $"%s{target.Name}.dll")
|
||||||
|
|
||||||
|
if File.Exists path then
|
||||||
|
this.LoadFromAssemblyPath path |> Some
|
||||||
|
else
|
||||||
|
None
|
||||||
|
)
|
||||||
|
|> Option.defaultValue null
|
||||||
|
|
||||||
|
if target.Name = "System.Console" then
|
||||||
|
if isNull assy then
|
||||||
|
failwith "could not monkey-patch System.Console"
|
||||||
|
else
|
||||||
|
let consoleType = assy.GetType "System.Console"
|
||||||
|
let setOut = consoleType.GetMethod "SetOut"
|
||||||
|
setOut.Invoke ((null : obj), [| contexts.Stdout |]) |> unbox<unit>
|
||||||
|
let setErr = consoleType.GetMethod "SetError"
|
||||||
|
setErr.Invoke ((null : obj), [| contexts.Stderr |]) |> unbox<unit>
|
||||||
|
|
||||||
|
assy
|
||||||
|
else
|
||||||
|
assy
|
263
WoofWare.NUnitTestRunner.Lib/CreateTrxReport.fs
Normal file
263
WoofWare.NUnitTestRunner.Lib/CreateTrxReport.fs
Normal file
@@ -0,0 +1,263 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Reflection
|
||||||
|
|
||||||
|
/// Methods for constructing TRX reports.
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module BuildTrxReport =
|
||||||
|
|
||||||
|
/// Build a TRX report from the given results.
|
||||||
|
let build
|
||||||
|
(assy : Assembly)
|
||||||
|
(creationTime : DateTimeOffset)
|
||||||
|
(startTime : DateTimeOffset)
|
||||||
|
(results : FixtureRunResults list)
|
||||||
|
: TrxReport
|
||||||
|
=
|
||||||
|
let finishTime = DateTimeOffset.Now
|
||||||
|
let finishTimeHumanReadable = finishTime.ToString @"yyyy-MM-dd HH:mm:ss"
|
||||||
|
let nowMachine = finishTime.ToString @"yyyy-MM-dd_HH_mm_ss"
|
||||||
|
|
||||||
|
let testListId = Guid.NewGuid ()
|
||||||
|
|
||||||
|
let testDefinitions, testEntries =
|
||||||
|
results
|
||||||
|
|> List.collect (fun results -> results.IndividualTestRunMetadata)
|
||||||
|
|> List.map (fun (data, _) ->
|
||||||
|
let defn =
|
||||||
|
{
|
||||||
|
Name = data.TestName
|
||||||
|
Storage = assy.Location.ToLowerInvariant ()
|
||||||
|
Id = data.TestId
|
||||||
|
Execution =
|
||||||
|
{
|
||||||
|
Id = data.ExecutionId
|
||||||
|
}
|
||||||
|
TestMethod =
|
||||||
|
{
|
||||||
|
CodeBase = assy.Location
|
||||||
|
AdapterTypeName = Uri "executor://woofware/"
|
||||||
|
ClassName = data.ClassName
|
||||||
|
Name = data.TestName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let entry : TrxTestEntry =
|
||||||
|
{
|
||||||
|
TestListId = testListId
|
||||||
|
ExecutionId = data.ExecutionId
|
||||||
|
TestId = data.TestId
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
defn, entry
|
||||||
|
)
|
||||||
|
|> List.unzip
|
||||||
|
|
||||||
|
let hostname = Environment.MachineName
|
||||||
|
|
||||||
|
let settings =
|
||||||
|
{
|
||||||
|
Name = "default"
|
||||||
|
Id = Guid.NewGuid ()
|
||||||
|
Deployment =
|
||||||
|
{
|
||||||
|
RunDeploymentRoot = $"_%s{hostname}_%s{nowMachine}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let testList : TrxTestListEntry =
|
||||||
|
{
|
||||||
|
Id = testListId
|
||||||
|
Name = "All"
|
||||||
|
}
|
||||||
|
|
||||||
|
let counters =
|
||||||
|
(TrxCounters.Zero, results)
|
||||||
|
// TODO: this is woefully inefficient
|
||||||
|
||> List.fold (fun counters results ->
|
||||||
|
let counters =
|
||||||
|
(counters, results.Failed)
|
||||||
|
||> List.fold (fun counters (_, _) ->
|
||||||
|
// TODO: the counters can be more specific about the failure mode
|
||||||
|
counters.AddFailed ()
|
||||||
|
)
|
||||||
|
|
||||||
|
let counters =
|
||||||
|
(counters, results.OtherFailures)
|
||||||
|
||> List.fold (fun counters _ ->
|
||||||
|
// TODO: the counters can be more specific about the failure mode
|
||||||
|
counters.AddFailed ()
|
||||||
|
)
|
||||||
|
|
||||||
|
(counters, results.Success)
|
||||||
|
||> List.fold (fun counters (_, success, _) ->
|
||||||
|
match success with
|
||||||
|
| TestMemberSuccess.Ok -> counters.AddPassed ()
|
||||||
|
| TestMemberSuccess.Ignored _
|
||||||
|
| TestMemberSuccess.Explicit _ -> counters.AddNotExecuted ()
|
||||||
|
| TestMemberSuccess.Inconclusive _ -> counters.AddInconclusive ()
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
// TODO: I'm sure we can do better than this; there's a whole range of possible
|
||||||
|
// states!
|
||||||
|
let outcome =
|
||||||
|
if counters.Failed > 0u then
|
||||||
|
TrxOutcome.Failed
|
||||||
|
else
|
||||||
|
TrxOutcome.Completed
|
||||||
|
|
||||||
|
let resultSummary : TrxResultsSummary =
|
||||||
|
{
|
||||||
|
Outcome = outcome
|
||||||
|
Counters = counters
|
||||||
|
Output =
|
||||||
|
{
|
||||||
|
StdOut = None
|
||||||
|
StdErr = None
|
||||||
|
ErrorInfo = None
|
||||||
|
}
|
||||||
|
RunInfos =
|
||||||
|
[
|
||||||
|
// TODO: capture stdout
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
let times : TrxReportTimes =
|
||||||
|
{
|
||||||
|
Creation = creationTime
|
||||||
|
Queuing = startTime
|
||||||
|
Start = startTime
|
||||||
|
Finish = finishTime
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
let magicGuid = Guid.Parse "13cdc9d9-ddb5-4fa4-a97d-d965ccfc6d4b"
|
||||||
|
|
||||||
|
let results =
|
||||||
|
results
|
||||||
|
|> List.collect (fun results -> results.IndividualTestRunMetadata)
|
||||||
|
|> List.map (fun (i, cause) ->
|
||||||
|
let exc =
|
||||||
|
match cause with
|
||||||
|
| Choice2Of3 _ -> None
|
||||||
|
| Choice1Of3 (TestMemberFailure.Malformed reasons) ->
|
||||||
|
{
|
||||||
|
StackTrace = None
|
||||||
|
Message = reasons |> String.concat "\n" |> Some
|
||||||
|
}
|
||||||
|
|> Some
|
||||||
|
| Choice1Of3 (TestMemberFailure.Failed fail)
|
||||||
|
| Choice1Of3 (TestMemberFailure.Failed fail)
|
||||||
|
| Choice1Of3 (TestMemberFailure.Failed fail) ->
|
||||||
|
((None, None), fail)
|
||||||
|
||> List.fold (fun (stackTrace, message) tf ->
|
||||||
|
match tf with
|
||||||
|
| TestFailure.TestFailed (UserMethodFailure.Threw (_, exc))
|
||||||
|
| TestFailure.SetUpFailed (UserMethodFailure.Threw (_, exc))
|
||||||
|
| TestFailure.TearDownFailed (UserMethodFailure.Threw (_, exc)) ->
|
||||||
|
let stackTrace =
|
||||||
|
match stackTrace with
|
||||||
|
| None -> (exc : Exception).ToString ()
|
||||||
|
| Some s -> s
|
||||||
|
|
||||||
|
(Some stackTrace, message)
|
||||||
|
| TestFailure.TestFailed (UserMethodFailure.BadParameters (_, expected, actual))
|
||||||
|
| TestFailure.SetUpFailed (UserMethodFailure.BadParameters (_, expected, actual))
|
||||||
|
| TestFailure.TearDownFailed (UserMethodFailure.BadParameters (_, expected, actual)) ->
|
||||||
|
let newMessage =
|
||||||
|
$"had parameter count mismatch: expected %i{expected.Length}, actual %i{actual.Length}"
|
||||||
|
|
||||||
|
let message =
|
||||||
|
match message with
|
||||||
|
| None -> newMessage
|
||||||
|
| Some message -> $"%s{message}\n%s{newMessage}"
|
||||||
|
|
||||||
|
(stackTrace, Some message)
|
||||||
|
| TestFailure.TestFailed (UserMethodFailure.ReturnedNonUnit (_, ret))
|
||||||
|
| TestFailure.SetUpFailed (UserMethodFailure.ReturnedNonUnit (_, ret))
|
||||||
|
| TestFailure.TearDownFailed (UserMethodFailure.ReturnedNonUnit (_, ret)) ->
|
||||||
|
let newMessage = $"returned non-unit value %O{ret}"
|
||||||
|
|
||||||
|
let message =
|
||||||
|
match message with
|
||||||
|
| None -> newMessage
|
||||||
|
| Some message -> $"%s{message}\n%s{newMessage}"
|
||||||
|
|
||||||
|
(stackTrace, Some message)
|
||||||
|
)
|
||||||
|
|> fun (stackTrace, message) ->
|
||||||
|
{
|
||||||
|
StackTrace = stackTrace
|
||||||
|
Message = message
|
||||||
|
}
|
||||||
|
|> Some
|
||||||
|
| Choice3Of3 (UserMethodFailure.Threw (_, exc)) ->
|
||||||
|
{
|
||||||
|
StackTrace = (exc : Exception).ToString () |> Some
|
||||||
|
Message = None
|
||||||
|
}
|
||||||
|
|> Some
|
||||||
|
| Choice3Of3 (UserMethodFailure.BadParameters (_, expected, actual)) ->
|
||||||
|
{
|
||||||
|
StackTrace = None
|
||||||
|
Message =
|
||||||
|
$"parameter count mismatch, expected %i{expected.Length}, actual %i{actual.Length}"
|
||||||
|
|> Some
|
||||||
|
}
|
||||||
|
|> Some
|
||||||
|
| Choice3Of3 (UserMethodFailure.ReturnedNonUnit (_, ret)) ->
|
||||||
|
{
|
||||||
|
Message = $"returned non-unit value %O{ret}" |> Some
|
||||||
|
StackTrace = None
|
||||||
|
}
|
||||||
|
|> Some
|
||||||
|
|
||||||
|
let outcome =
|
||||||
|
match cause with
|
||||||
|
| Choice1Of3 _ -> TrxTestOutcome.Failed
|
||||||
|
| Choice2Of3 TestMemberSuccess.Ok -> TrxTestOutcome.Passed
|
||||||
|
| Choice2Of3 (TestMemberSuccess.Inconclusive _) -> TrxTestOutcome.Inconclusive
|
||||||
|
| Choice2Of3 (TestMemberSuccess.Ignored _)
|
||||||
|
| Choice2Of3 (TestMemberSuccess.Explicit _) -> TrxTestOutcome.NotExecuted
|
||||||
|
// TODO: we can totally do better here, more fine-grained classification
|
||||||
|
| Choice3Of3 _ -> TrxTestOutcome.Failed
|
||||||
|
|
||||||
|
{
|
||||||
|
ExecutionId = i.ExecutionId
|
||||||
|
TestId = i.TestId
|
||||||
|
TestName = i.TestName
|
||||||
|
ComputerName = i.ComputerName
|
||||||
|
Duration = i.End - i.Start
|
||||||
|
StartTime = i.Start
|
||||||
|
EndTime = i.End
|
||||||
|
TestType = magicGuid
|
||||||
|
Outcome = outcome
|
||||||
|
TestListId = testListId
|
||||||
|
RelativeResultsDirectory = i.ExecutionId.ToString () // for some reason
|
||||||
|
Output =
|
||||||
|
match i.StdOut, i.StdErr, exc with
|
||||||
|
| None, None, None -> None
|
||||||
|
| stdout, stderr, exc ->
|
||||||
|
Some
|
||||||
|
{
|
||||||
|
TrxOutput.StdOut = stdout
|
||||||
|
StdErr = stderr
|
||||||
|
ErrorInfo = exc
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
{
|
||||||
|
Id = Guid.NewGuid ()
|
||||||
|
Name = $"@%s{hostname} %s{finishTimeHumanReadable}"
|
||||||
|
Times = times
|
||||||
|
Settings = settings
|
||||||
|
Results = results
|
||||||
|
TestDefinitions = testDefinitions
|
||||||
|
TestEntries = testEntries
|
||||||
|
TestLists = [ testList ]
|
||||||
|
ResultsSummary = resultSummary
|
||||||
|
}
|
@@ -1,5 +1,6 @@
|
|||||||
namespace WoofWare.NUnitTestRunner
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System
|
||||||
open System.Reflection
|
open System.Reflection
|
||||||
|
|
||||||
/// A modifier on whether a given test should be run.
|
/// A modifier on whether a given test should be run.
|
||||||
@@ -30,6 +31,54 @@ type Combinatorial =
|
|||||||
/// each", and so on. Spare slots are filled with `Unchecked.defaultof<_>`.
|
/// each", and so on. Spare slots are filled with `Unchecked.defaultof<_>`.
|
||||||
| Sequential
|
| Sequential
|
||||||
|
|
||||||
|
/// Describes the level of parallelism permitted in some context.
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
type ClassParallelScope =
|
||||||
|
/// "I may be run in parallel with other tests, although my children might not be able to run in parallel with each
|
||||||
|
/// other".
|
||||||
|
| Self
|
||||||
|
/// "The set of things I contain may be run in parallel with itself".
|
||||||
|
| Children
|
||||||
|
/// "Fixtures within me may be run in parallel with each other, but the tests within a given fixture might not
|
||||||
|
/// be runnable in parallel with each other".
|
||||||
|
| Fixtures
|
||||||
|
/// "All my descendents are happy to run in parallel with anything else, and also so am I".
|
||||||
|
| All
|
||||||
|
|
||||||
|
/// Describes the level of parallelism permitted within an assembly.
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
type AssemblyParallelScope =
|
||||||
|
/// "The set of things I contain may be run in parallel with itself".
|
||||||
|
| Children
|
||||||
|
/// "Fixtures within me may be run in parallel with each other, but the tests within a given fixture might not
|
||||||
|
/// necessarily be runnable in parallel with each other".
|
||||||
|
| Fixtures
|
||||||
|
|
||||||
|
/// Describes whether a test can be run concurrently with other tests.
|
||||||
|
type Parallelizable<'scope> =
|
||||||
|
/// This test is happy, under some conditions (specified by the scope), to be run alongside other tests.
|
||||||
|
| Yes of 'scope
|
||||||
|
/// This test must always be run on its own.
|
||||||
|
| No
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module Parallelizable =
|
||||||
|
/// Functorial map.
|
||||||
|
let inline map<'a, 'b> ([<InlineIfLambda>] f : 'a -> 'b) (p : Parallelizable<'a>) : Parallelizable<'b> =
|
||||||
|
match p with
|
||||||
|
| Parallelizable.No -> Parallelizable.No
|
||||||
|
| Parallelizable.Yes a -> Parallelizable.Yes (f a)
|
||||||
|
|
||||||
|
/// Functorial bind.
|
||||||
|
let inline bind<'a, 'b>
|
||||||
|
([<InlineIfLambda>] f : 'a -> Parallelizable<'b>)
|
||||||
|
(p : Parallelizable<'a>)
|
||||||
|
: Parallelizable<'b>
|
||||||
|
=
|
||||||
|
match p with
|
||||||
|
| Parallelizable.No -> Parallelizable.No
|
||||||
|
| Parallelizable.Yes a -> f a
|
||||||
|
|
||||||
/// A single method or member which holds some tests. (Often such a member will represent only one test, but e.g.
|
/// A single method or member which holds some tests. (Often such a member will represent only one test, but e.g.
|
||||||
/// if it has [<TestCaseSource>] then it represents multiple tests.)
|
/// if it has [<TestCaseSource>] then it represents multiple tests.)
|
||||||
type SingleTestMethod =
|
type SingleTestMethod =
|
||||||
@@ -48,6 +97,8 @@ type SingleTestMethod =
|
|||||||
/// If this test has data supplied by `[<Value>]` annotations, specifies how those annotations are combined
|
/// If this test has data supplied by `[<Value>]` annotations, specifies how those annotations are combined
|
||||||
/// to produce the complete collection of args.
|
/// to produce the complete collection of args.
|
||||||
Combinatorial : Combinatorial option
|
Combinatorial : Combinatorial option
|
||||||
|
/// If this test has declared a parallelisability, that goes here.
|
||||||
|
Parallelize : Parallelizable<unit> option
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Human-readable name of this test method.
|
/// Human-readable name of this test method.
|
||||||
@@ -55,6 +106,7 @@ type SingleTestMethod =
|
|||||||
|
|
||||||
/// A test fixture (usually represented by the [<TestFixture>]` attribute), which may contain many tests,
|
/// A test fixture (usually represented by the [<TestFixture>]` attribute), which may contain many tests,
|
||||||
/// each of which may run many times.
|
/// each of which may run many times.
|
||||||
|
[<NoComparison>]
|
||||||
type TestFixture =
|
type TestFixture =
|
||||||
{
|
{
|
||||||
/// The assembly which contains this TestFixture, loaded into a separate context.
|
/// The assembly which contains this TestFixture, loaded into a separate context.
|
||||||
@@ -62,6 +114,8 @@ type TestFixture =
|
|||||||
/// Fully-qualified name of this fixture (e.g. MyThing.Test.Foo for `[<TestFixture>] module Foo` in the
|
/// Fully-qualified name of this fixture (e.g. MyThing.Test.Foo for `[<TestFixture>] module Foo` in the
|
||||||
/// `MyThing.Test` assembly).
|
/// `MyThing.Test` assembly).
|
||||||
Name : string
|
Name : string
|
||||||
|
/// The type which is this fixture, containing the tests as members.
|
||||||
|
Type : Type
|
||||||
/// A method which is run once when this test fixture starts, before any other setup logic and before
|
/// A method which is run once when this test fixture starts, before any other setup logic and before
|
||||||
/// any tests run. If this method fails, no tests will run and no per-test setup/teardown logic will run,
|
/// any tests run. If this method fails, no tests will run and no per-test setup/teardown logic will run,
|
||||||
/// but OneTimeTearDown will run.
|
/// but OneTimeTearDown will run.
|
||||||
@@ -77,20 +131,36 @@ type TestFixture =
|
|||||||
/// Methods which are run in some arbitrary order after each individual test, even if the test or its setup
|
/// Methods which are run in some arbitrary order after each individual test, even if the test or its setup
|
||||||
/// failed. If the first TearDown we run fails, we don't define whether the other TearDowns run.
|
/// failed. If the first TearDown we run fails, we don't define whether the other TearDowns run.
|
||||||
TearDown : MethodInfo list
|
TearDown : MethodInfo list
|
||||||
|
/// You might have defined e.g. `[<TestFixture true>] type Foo (v : bool) = ...`. If so, this gives the
|
||||||
|
/// various possible parameters.
|
||||||
|
Parameters : obj list list
|
||||||
/// The individual test methods present within this fixture.
|
/// The individual test methods present within this fixture.
|
||||||
Tests : SingleTestMethod list
|
Tests : SingleTestMethod list
|
||||||
|
/// If this fixture has declared a parallelisability, that goes here.
|
||||||
|
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 (containingAssembly : Assembly) (name : string) =
|
static member Empty
|
||||||
|
(ty : Type)
|
||||||
|
(par : Parallelizable<ClassParallelScope> option)
|
||||||
|
(modifiers : Modifier list)
|
||||||
|
(args : obj list list)
|
||||||
|
=
|
||||||
{
|
{
|
||||||
ContainingAssembly = containingAssembly
|
ContainingAssembly = ty.Assembly
|
||||||
Name = name
|
Type = ty
|
||||||
|
Name = ty.Name
|
||||||
OneTimeSetUp = None
|
OneTimeSetUp = None
|
||||||
OneTimeTearDown = None
|
OneTimeTearDown = None
|
||||||
SetUp = []
|
SetUp = []
|
||||||
TearDown = []
|
TearDown = []
|
||||||
|
Parameters = args
|
||||||
Tests = []
|
Tests = []
|
||||||
|
Parallelize = par
|
||||||
|
Modifiers = modifiers
|
||||||
}
|
}
|
||||||
|
|
||||||
/// User code in the unit under test has failed somehow.
|
/// User code in the unit under test has failed somehow.
|
||||||
@@ -100,6 +170,8 @@ type UserMethodFailure =
|
|||||||
| ReturnedNonUnit of name : string * result : obj
|
| ReturnedNonUnit of name : string * result : obj
|
||||||
/// A method threw.
|
/// A method threw.
|
||||||
| Threw of name : string * exn
|
| Threw of name : string * exn
|
||||||
|
/// Parameter count mismatch.
|
||||||
|
| BadParameters of name : string * expected : Type[] * actual : obj[]
|
||||||
|
|
||||||
/// Human-readable representation of the user failure.
|
/// Human-readable representation of the user failure.
|
||||||
override this.ToString () =
|
override this.ToString () =
|
||||||
@@ -108,12 +180,22 @@ type UserMethodFailure =
|
|||||||
$"User-defined method '%s{method}' returned a non-unit: %O{ret}"
|
$"User-defined method '%s{method}' returned a non-unit: %O{ret}"
|
||||||
| UserMethodFailure.Threw (method, exc) ->
|
| UserMethodFailure.Threw (method, exc) ->
|
||||||
$"User-defined method '%s{method}' threw: %s{exc.Message}\n %s{exc.StackTrace}"
|
$"User-defined method '%s{method}' threw: %s{exc.Message}\n %s{exc.StackTrace}"
|
||||||
|
| UserMethodFailure.BadParameters (method, expected, actual) ->
|
||||||
|
let expectedStr = expected |> Seq.map (fun t -> t.Name) |> String.concat ", "
|
||||||
|
|
||||||
|
let actualStr =
|
||||||
|
actual
|
||||||
|
|> Seq.map (fun s -> if isNull s then "null" else s.ToString ())
|
||||||
|
|> String.concat ", "
|
||||||
|
|
||||||
|
$"User-defined method '%s{method}' had parameter count mismatch. Expected: (%s{expectedStr}) (%i{expected.Length} params). Actual: (%s{actualStr}) (%i{actual.Length} params)"
|
||||||
|
|
||||||
/// Name (not fully-qualified) of the method which failed.
|
/// Name (not fully-qualified) of the method which failed.
|
||||||
member this.Name =
|
member this.Name =
|
||||||
match this with
|
match this with
|
||||||
| UserMethodFailure.Threw (name, _)
|
| UserMethodFailure.Threw (name, _)
|
||||||
| UserMethodFailure.ReturnedNonUnit (name, _) -> name
|
| UserMethodFailure.ReturnedNonUnit (name, _) -> name
|
||||||
|
| UserMethodFailure.BadParameters (name, _, _) -> name
|
||||||
|
|
||||||
/// Represents the failure of a single run of one test. An error signalled this way is a user error: the unit under
|
/// Represents the failure of a single run of one test. An error signalled this way is a user error: the unit under
|
||||||
/// test has misbehaved.
|
/// test has misbehaved.
|
||||||
|
101
WoofWare.NUnitTestRunner.Lib/DotnetRuntime.fs
Normal file
101
WoofWare.NUnitTestRunner.Lib/DotnetRuntime.fs
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.IO
|
||||||
|
open WoofWare.DotnetRuntimeLocator
|
||||||
|
|
||||||
|
/// Functions for locating .NET runtimes.
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module DotnetRuntime =
|
||||||
|
let private selectRuntime (config : RuntimeOptions) (f : DotnetEnvironmentInfo) : DirectoryInfo list =
|
||||||
|
let rollForward =
|
||||||
|
match Environment.GetEnvironmentVariable "DOTNET_ROLL_FORWARD" with
|
||||||
|
| null ->
|
||||||
|
config.RollForward
|
||||||
|
|> Option.map RollForward.Parse
|
||||||
|
|> Option.defaultValue RollForward.Minor
|
||||||
|
| s -> RollForward.Parse s
|
||||||
|
|
||||||
|
if
|
||||||
|
Option.isSome config.IncludedFramework
|
||||||
|
|| Option.isSome config.IncludedFrameworks
|
||||||
|
then
|
||||||
|
// No need for a framework that's anywhere other than the given DLL.
|
||||||
|
[]
|
||||||
|
else
|
||||||
|
|
||||||
|
let desiredVersions =
|
||||||
|
match config.Framework with
|
||||||
|
| Some f -> [ Version f.Version, f.Name ]
|
||||||
|
| None ->
|
||||||
|
|
||||||
|
match config.Frameworks with
|
||||||
|
| Some f -> f |> List.map (fun f -> Version f.Version, f.Name)
|
||||||
|
| None ->
|
||||||
|
failwith
|
||||||
|
"Could not deduce a framework version due to lack of either Framework or Frameworks in runtimeconfig"
|
||||||
|
|
||||||
|
let compatiblyNamedRuntimes =
|
||||||
|
f.Frameworks
|
||||||
|
|> Seq.collect (fun availableFramework ->
|
||||||
|
desiredVersions
|
||||||
|
|> List.choose (fun (desiredVersion, desiredName) ->
|
||||||
|
if desiredName = availableFramework.Name then
|
||||||
|
Some
|
||||||
|
{|
|
||||||
|
Desired = desiredVersion
|
||||||
|
Name = desiredName
|
||||||
|
Installed = availableFramework
|
||||||
|
InstalledVersion = Version availableFramework.Version
|
||||||
|
|}
|
||||||
|
else
|
||||||
|
None
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|> Seq.toList
|
||||||
|
|
||||||
|
match rollForward with
|
||||||
|
| RollForward.Minor ->
|
||||||
|
let available =
|
||||||
|
compatiblyNamedRuntimes
|
||||||
|
|> Seq.filter (fun data ->
|
||||||
|
data.InstalledVersion.Major = data.Desired.Major
|
||||||
|
&& data.InstalledVersion.Minor >= data.Desired.Minor
|
||||||
|
)
|
||||||
|
|> Seq.groupBy (fun data -> data.Name)
|
||||||
|
|> Seq.map (fun (name, data) ->
|
||||||
|
let data =
|
||||||
|
data
|
||||||
|
|> Seq.minBy (fun data -> data.InstalledVersion.Minor, data.InstalledVersion.Build)
|
||||||
|
|
||||||
|
name, data.Installed
|
||||||
|
)
|
||||||
|
|> Seq.toList
|
||||||
|
|
||||||
|
// TODO: maybe we can ask the SDK if we don't have any runtimes.
|
||||||
|
// But we keep on trucking: maybe we're self-contained, and we'll actually find all the runtime next to the
|
||||||
|
// DLL.
|
||||||
|
available
|
||||||
|
|> List.map (fun (_name, runtime) -> DirectoryInfo $"%s{runtime.Path}/%s{runtime.Version}")
|
||||||
|
| _ -> failwith "non-minor RollForward not supported yet; please shout if you want it"
|
||||||
|
|
||||||
|
/// Given an executable DLL, locate the .NET runtime that can best run it.
|
||||||
|
let locate (dll : FileInfo) : DirectoryInfo list =
|
||||||
|
let runtimeConfig =
|
||||||
|
let name =
|
||||||
|
if not (dll.Name.EndsWith (".dll", StringComparison.OrdinalIgnoreCase)) then
|
||||||
|
failwith $"Expected DLL %s{dll.FullName} to end in .dll"
|
||||||
|
|
||||||
|
dll.Name.Substring (0, dll.Name.Length - 4)
|
||||||
|
|
||||||
|
Path.Combine (dll.Directory.FullName, $"%s{name}.runtimeconfig.json")
|
||||||
|
|> File.ReadAllText
|
||||||
|
|> System.Text.Json.Nodes.JsonNode.Parse
|
||||||
|
|> RuntimeConfig.jsonParse
|
||||||
|
|> fun f -> f.RuntimeOptions
|
||||||
|
|
||||||
|
let availableRuntimes = DotnetEnvironmentInfo.Get ()
|
||||||
|
|
||||||
|
let runtime = selectRuntime runtimeConfig availableRuntimes
|
||||||
|
|
||||||
|
dll.Directory :: runtime
|
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"
|
@@ -2,7 +2,7 @@ namespace WoofWare.NUnitTestRunner
|
|||||||
|
|
||||||
open System
|
open System
|
||||||
open System.IO
|
open System.IO
|
||||||
open PrattParser
|
open WoofWare.PrattParser
|
||||||
|
|
||||||
// Documentation:
|
// Documentation:
|
||||||
// https://learn.microsoft.com/en-us/dotnet/core/testing/selective-unit-tests?pivots=mstest
|
// https://learn.microsoft.com/en-us/dotnet/core/testing/selective-unit-tests?pivots=mstest
|
||||||
@@ -155,9 +155,7 @@ module internal Lexer =
|
|||||||
[<RequireQualifiedAccess>]
|
[<RequireQualifiedAccess>]
|
||||||
module internal ParsedFilter =
|
module internal ParsedFilter =
|
||||||
let private unescape (s : string) : string =
|
let private unescape (s : string) : string =
|
||||||
System.Xml.XmlReader
|
System.Xml.XmlReader.Create(new StringReader ("<r>" + s + "</r>")).ReadElementString ()
|
||||||
.Create(new StringReader ("<r>" + s + "</r>"))
|
|
||||||
.ReadElementString ()
|
|
||||||
|
|
||||||
let private atom (inputString : string) (token : Token) : ParsedFilter option =
|
let private atom (inputString : string) (token : Token) : ParsedFilter option =
|
||||||
let start, len = token.Trivia
|
let start, len = token.Trivia
|
||||||
|
477
WoofWare.NUnitTestRunner.Lib/ParallelQueue.fs
Normal file
477
WoofWare.NUnitTestRunner.Lib/ParallelQueue.fs
Normal file
@@ -0,0 +1,477 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
open System
|
||||||
|
open System.Threading
|
||||||
|
open System.Threading.Tasks
|
||||||
|
|
||||||
|
type private AsyncThunkEvaluator<'ret> =
|
||||||
|
abstract Eval<'a> : (unit -> Async<'a>) -> AsyncReplyChannel<Result<'a, exn>> -> 'ret
|
||||||
|
|
||||||
|
type private AsyncThunkCrate =
|
||||||
|
abstract Apply<'ret> : AsyncThunkEvaluator<'ret> -> 'ret
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module private AsyncThunkCrate =
|
||||||
|
let make<'a> (t : unit -> Async<'a>) (rc : AsyncReplyChannel<Result<'a, exn>>) : AsyncThunkCrate =
|
||||||
|
{ new AsyncThunkCrate with
|
||||||
|
member _.Apply e = e.Eval t rc
|
||||||
|
}
|
||||||
|
|
||||||
|
type private FakeUnit = FakeUnit
|
||||||
|
|
||||||
|
/// A handle to a running test fixture.
|
||||||
|
type TestFixtureRunningToken = private | TestFixtureRunningToken of TestFixture
|
||||||
|
|
||||||
|
/// A handle to a test fixture whose setup method has been called.
|
||||||
|
type TestFixtureSetupToken = private | TestFixtureSetupToken of TestFixture
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module private TestFixtureSetupToken =
|
||||||
|
let vouchNoSetupRequired (TestFixtureRunningToken tf) = TestFixtureSetupToken tf
|
||||||
|
|
||||||
|
/// A handle to a test fixture whose setup method has been called.
|
||||||
|
type TestFixtureTearDownToken = private | TestFixtureTearDownToken of TestFixture
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module private TestFixtureTearDownToken =
|
||||||
|
let vouchNoTearDownRequired (TestFixtureSetupToken tf) = TestFixtureTearDownToken tf
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
type private MailboxMessage =
|
||||||
|
| Quit of AsyncReplyChannel<unit>
|
||||||
|
/// Check current state, see if we need to start more tests, etc.
|
||||||
|
| Reconcile
|
||||||
|
| RunTestAsync of
|
||||||
|
within : TestFixture *
|
||||||
|
Parallelizable<unit> option *
|
||||||
|
test : AsyncThunkCrate *
|
||||||
|
context : ExecutionContext
|
||||||
|
| BeginTestFixture of TestFixture * AsyncReplyChannel<TestFixtureRunningToken>
|
||||||
|
| EndTestFixture of TestFixtureTearDownToken * AsyncReplyChannel<unit>
|
||||||
|
|
||||||
|
type private RunningFixture =
|
||||||
|
{
|
||||||
|
Fixture : TestFixture
|
||||||
|
RunningCanParallelize : bool
|
||||||
|
Running : Task list
|
||||||
|
Waiting : ((unit -> Task) * Parallelizable<unit> option) list
|
||||||
|
}
|
||||||
|
|
||||||
|
static member Make (f : TestFixture) =
|
||||||
|
{
|
||||||
|
Fixture = f
|
||||||
|
Running = []
|
||||||
|
RunningCanParallelize = true
|
||||||
|
Waiting = []
|
||||||
|
}
|
||||||
|
|
||||||
|
type private RunningState =
|
||||||
|
{
|
||||||
|
MaxParallelism : int
|
||||||
|
// TODO: make these efficiently look-up-able
|
||||||
|
CurrentlyRunning : RunningFixture list
|
||||||
|
Waiting : (TestFixture * AsyncReplyChannel<TestFixtureRunningToken>) list
|
||||||
|
}
|
||||||
|
|
||||||
|
member this.NewTest (tf : TestFixture) (par : Parallelizable<unit> option) (test : unit -> Task) =
|
||||||
|
{
|
||||||
|
MaxParallelism = this.MaxParallelism
|
||||||
|
Waiting = this.Waiting
|
||||||
|
CurrentlyRunning =
|
||||||
|
let found = ref 0
|
||||||
|
|
||||||
|
this.CurrentlyRunning
|
||||||
|
|> List.map (fun f ->
|
||||||
|
if Object.ReferenceEquals (f.Fixture, tf) then
|
||||||
|
Interlocked.Increment found |> ignore<int>
|
||||||
|
|
||||||
|
{ f with
|
||||||
|
Waiting = (test, par) :: f.Waiting
|
||||||
|
}
|
||||||
|
else
|
||||||
|
f
|
||||||
|
)
|
||||||
|
|> fun l ->
|
||||||
|
match found.Value with
|
||||||
|
| 1 -> l
|
||||||
|
| 0 -> failwith $"Unexpectedly did not find the running test fixture '%s{tf.Name}' to add a test to"
|
||||||
|
| _ -> failwith $"Unexpectedly found the running test fixture '%s{tf.Name}' multiple times in list"
|
||||||
|
}
|
||||||
|
|
||||||
|
member this.CompleteFixture (tf : TestFixture) : RunningState =
|
||||||
|
let rec go (acc : RunningFixture list) (running : RunningFixture list) =
|
||||||
|
match running with
|
||||||
|
| [] -> failwith "Caller has somehow called EndTestFixture while we're not running that test fixture"
|
||||||
|
| runningFixture :: tail ->
|
||||||
|
if Object.ReferenceEquals (runningFixture.Fixture, tf) then
|
||||||
|
match runningFixture.Running, runningFixture.Waiting with
|
||||||
|
| [], [] -> acc @ tail
|
||||||
|
| r, [] ->
|
||||||
|
failwith $"Caller has called EndTestFixture while its tests are still running (%i{r.Length})"
|
||||||
|
| [], r ->
|
||||||
|
failwith $"Caller has called EndTestFixture while it has tests waiting to run (%i{r.Length})"
|
||||||
|
| r, s ->
|
||||||
|
failwith
|
||||||
|
$"Caller has called EndTestFixture while it has tests waiting to run (%i{s.Length}) and test running (%i{r.Length})"
|
||||||
|
else
|
||||||
|
go (runningFixture :: acc) tail
|
||||||
|
|
||||||
|
let currentlyRunning = go [] this.CurrentlyRunning
|
||||||
|
|
||||||
|
{
|
||||||
|
CurrentlyRunning = currentlyRunning
|
||||||
|
Waiting = this.Waiting
|
||||||
|
MaxParallelism = this.MaxParallelism
|
||||||
|
}
|
||||||
|
|
||||||
|
type private MailboxState =
|
||||||
|
| Idle
|
||||||
|
| Running of RunningState
|
||||||
|
|
||||||
|
/// Run some things in parallel.
|
||||||
|
/// TODO: actually implement the parallelism! Right now this just runs everything serially.
|
||||||
|
/// TODO: consume the cancellation token
|
||||||
|
type ParallelQueue
|
||||||
|
(parallelism : int option, _scope : Parallelizable<AssemblyParallelScope> option, ?ct : CancellationToken)
|
||||||
|
=
|
||||||
|
let parallelism =
|
||||||
|
match parallelism with
|
||||||
|
| None -> max (Environment.ProcessorCount / 2) 2
|
||||||
|
| Some p -> p
|
||||||
|
|
||||||
|
let rec processTask (state : MailboxState) (m : MailboxProcessor<MailboxMessage>) =
|
||||||
|
async {
|
||||||
|
let! message = m.Receive ()
|
||||||
|
|
||||||
|
match message with
|
||||||
|
| MailboxMessage.Quit rc -> rc.Reply ()
|
||||||
|
| MailboxMessage.Reconcile ->
|
||||||
|
match state with
|
||||||
|
| Idle -> return! processTask state m
|
||||||
|
| Running r ->
|
||||||
|
|
||||||
|
match r.CurrentlyRunning with
|
||||||
|
| [] ->
|
||||||
|
match r.Waiting with
|
||||||
|
| [] -> return! processTask Idle m
|
||||||
|
| (head, rc) :: tail ->
|
||||||
|
rc.Reply (TestFixtureRunningToken head)
|
||||||
|
|
||||||
|
let newRunning =
|
||||||
|
{
|
||||||
|
Fixture = head
|
||||||
|
Running = []
|
||||||
|
RunningCanParallelize = true
|
||||||
|
Waiting = []
|
||||||
|
}
|
||||||
|
|
||||||
|
let state =
|
||||||
|
{
|
||||||
|
MaxParallelism = r.MaxParallelism
|
||||||
|
CurrentlyRunning = [ newRunning ]
|
||||||
|
Waiting = tail
|
||||||
|
}
|
||||||
|
// For now, we'll just run one fixture at a time. When we run multiple fixtures in parallel,
|
||||||
|
// we probably want to call Reconcile here again.
|
||||||
|
return! processTask (Running state) m
|
||||||
|
| [ currentlyRunning ] ->
|
||||||
|
let currentlyRunningTasks =
|
||||||
|
currentlyRunning.Running |> List.filter (fun t -> not t.IsCompleted)
|
||||||
|
|
||||||
|
let r =
|
||||||
|
{ r with
|
||||||
|
CurrentlyRunning =
|
||||||
|
[
|
||||||
|
{ currentlyRunning with
|
||||||
|
Running = currentlyRunningTasks
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
match currentlyRunningTasks with
|
||||||
|
| [] ->
|
||||||
|
match currentlyRunning.Waiting with
|
||||||
|
| [] ->
|
||||||
|
// Nothing to run yet
|
||||||
|
return! processTask (Running r) m
|
||||||
|
| (head, par) :: tail ->
|
||||||
|
let par =
|
||||||
|
match par with
|
||||||
|
| None -> true
|
||||||
|
| Some Parallelizable.No -> false
|
||||||
|
| Some (Parallelizable.Yes ()) -> true
|
||||||
|
|
||||||
|
let state =
|
||||||
|
{
|
||||||
|
Fixture = currentlyRunning.Fixture
|
||||||
|
RunningCanParallelize = par
|
||||||
|
Waiting = tail
|
||||||
|
Running = [ head () ]
|
||||||
|
}
|
||||||
|
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
|
||||||
|
return!
|
||||||
|
processTask
|
||||||
|
(Running
|
||||||
|
{ r with
|
||||||
|
CurrentlyRunning = [ state ]
|
||||||
|
})
|
||||||
|
m
|
||||||
|
|
||||||
|
| currentlyRunningTasks ->
|
||||||
|
|
||||||
|
if currentlyRunningTasks.Length >= parallelism then
|
||||||
|
return! processTask (Running r) m
|
||||||
|
else
|
||||||
|
|
||||||
|
match currentlyRunning.Waiting, currentlyRunning.RunningCanParallelize with
|
||||||
|
| [], _ ->
|
||||||
|
// No new candidates.
|
||||||
|
return! processTask (Running r) m
|
||||||
|
| _, false ->
|
||||||
|
// The running test(s) can't have others added.
|
||||||
|
return! processTask (Running r) m
|
||||||
|
| (head, par) :: tail, true ->
|
||||||
|
match par with
|
||||||
|
| Some Parallelizable.No -> return! processTask (Running r) m
|
||||||
|
| Some (Parallelizable.Yes ()) ->
|
||||||
|
let state =
|
||||||
|
{
|
||||||
|
RunningState.MaxParallelism = r.MaxParallelism
|
||||||
|
Waiting = r.Waiting
|
||||||
|
CurrentlyRunning =
|
||||||
|
[
|
||||||
|
{
|
||||||
|
Fixture = currentlyRunning.Fixture
|
||||||
|
RunningCanParallelize = true
|
||||||
|
Running = head () :: currentlyRunning.Running
|
||||||
|
Waiting = tail
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
return! processTask (Running state) m
|
||||||
|
| None ->
|
||||||
|
match currentlyRunning.Fixture.Parallelize with
|
||||||
|
| Some Parallelizable.No
|
||||||
|
| Some (Parallelizable.Yes ClassParallelScope.Self)
|
||||||
|
| Some (Parallelizable.Yes ClassParallelScope.Fixtures) ->
|
||||||
|
// Can't add this test to the parallel queue right now
|
||||||
|
return! processTask (Running r) m
|
||||||
|
| None
|
||||||
|
| Some (Parallelizable.Yes ClassParallelScope.All)
|
||||||
|
| Some (Parallelizable.Yes ClassParallelScope.Children) ->
|
||||||
|
let state =
|
||||||
|
{
|
||||||
|
Fixture = currentlyRunning.Fixture
|
||||||
|
RunningCanParallelize = true
|
||||||
|
Waiting = tail
|
||||||
|
Running = (head ()) :: currentlyRunningTasks
|
||||||
|
}
|
||||||
|
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
|
||||||
|
return!
|
||||||
|
processTask
|
||||||
|
(Running
|
||||||
|
{ r with
|
||||||
|
CurrentlyRunning = [ state ]
|
||||||
|
})
|
||||||
|
m
|
||||||
|
| _ -> failwith "Logic error: we currently only run one fixture at a time"
|
||||||
|
| MailboxMessage.BeginTestFixture (tf, rc) ->
|
||||||
|
match state with
|
||||||
|
| Running state ->
|
||||||
|
let state =
|
||||||
|
{
|
||||||
|
MaxParallelism = state.MaxParallelism
|
||||||
|
CurrentlyRunning = state.CurrentlyRunning
|
||||||
|
Waiting = (tf, rc) :: state.Waiting
|
||||||
|
}
|
||||||
|
|> Running
|
||||||
|
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
return! processTask state m
|
||||||
|
| Idle ->
|
||||||
|
let state =
|
||||||
|
{
|
||||||
|
MaxParallelism = parallelism
|
||||||
|
CurrentlyRunning = []
|
||||||
|
Waiting = [ (tf, rc) ]
|
||||||
|
}
|
||||||
|
|> Running
|
||||||
|
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
return! processTask state m
|
||||||
|
| MailboxMessage.EndTestFixture (TestFixtureTearDownToken tf, rc) ->
|
||||||
|
match state with
|
||||||
|
| Idle ->
|
||||||
|
return failwith "Caller has somehow called EndTestFixture while we're not running a test fixture"
|
||||||
|
| Running state ->
|
||||||
|
let state = state.CompleteFixture tf
|
||||||
|
rc.Reply ()
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
return! processTask (Running state) m
|
||||||
|
| MailboxMessage.RunTestAsync (withinFixture, par, message, capturedContext) ->
|
||||||
|
let t () =
|
||||||
|
{ new AsyncThunkEvaluator<_> with
|
||||||
|
member _.Eval<'b> (t : unit -> Async<'b>) rc =
|
||||||
|
let tcs = TaskCompletionSource TaskCreationOptions.RunContinuationsAsynchronously
|
||||||
|
|
||||||
|
fun () ->
|
||||||
|
ExecutionContext.Run (
|
||||||
|
capturedContext,
|
||||||
|
(fun _ ->
|
||||||
|
async {
|
||||||
|
let! result =
|
||||||
|
async {
|
||||||
|
try
|
||||||
|
let! r = t ()
|
||||||
|
return Ok r
|
||||||
|
with e ->
|
||||||
|
return Error e
|
||||||
|
}
|
||||||
|
|
||||||
|
tcs.SetResult ()
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
rc.Reply result
|
||||||
|
}
|
||||||
|
|> Async.StartImmediate
|
||||||
|
),
|
||||||
|
()
|
||||||
|
)
|
||||||
|
|> Task.Factory.StartNew
|
||||||
|
|> ignore<Task>
|
||||||
|
|
||||||
|
tcs.Task
|
||||||
|
}
|
||||||
|
|> message.Apply
|
||||||
|
|
||||||
|
let state =
|
||||||
|
match state with
|
||||||
|
| Idle -> failwith "somehow asked the queue to run tests when there is no active fixture"
|
||||||
|
| Running state -> state.NewTest withinFixture par t |> Running
|
||||||
|
|
||||||
|
m.Post MailboxMessage.Reconcile
|
||||||
|
|
||||||
|
return! processTask state m
|
||||||
|
}
|
||||||
|
|
||||||
|
let mb = new MailboxProcessor<_> (processTask MailboxState.Idle)
|
||||||
|
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.
|
||||||
|
/// The resulting Task will return when the action has completed.
|
||||||
|
member this.Run<'a>
|
||||||
|
(parent : TestFixtureSetupToken)
|
||||||
|
(scope : Parallelizable<unit> option)
|
||||||
|
(action : unit -> 'a)
|
||||||
|
: 'a Task
|
||||||
|
=
|
||||||
|
this.RunAsync parent scope (fun () -> async.Return (action ()))
|
||||||
|
|
||||||
|
/// 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.
|
||||||
|
/// Once you've finished running tests from that fixture, call EndTestFixture.
|
||||||
|
member _.StartTestFixture (tf : TestFixture) : Task<TestFixtureRunningToken> =
|
||||||
|
fun rc -> MailboxMessage.BeginTestFixture (tf, rc)
|
||||||
|
|> mb.PostAndAsyncReply
|
||||||
|
|> Async.StartAsTask
|
||||||
|
|
||||||
|
/// Run the given one-time setup for the test fixture.
|
||||||
|
member _.RunTestSetup (TestFixtureRunningToken parent) (action : unit -> 'a) : ('a * TestFixtureSetupToken) Task =
|
||||||
|
task {
|
||||||
|
let par =
|
||||||
|
parent.Parallelize
|
||||||
|
|> Option.map (fun p ->
|
||||||
|
match p with
|
||||||
|
| Parallelizable.No -> Parallelizable.No
|
||||||
|
| Parallelizable.Yes _ -> Parallelizable.Yes ()
|
||||||
|
)
|
||||||
|
|
||||||
|
let ec = ExecutionContext.Capture ()
|
||||||
|
|
||||||
|
let! response =
|
||||||
|
(fun rc ->
|
||||||
|
MailboxMessage.RunTestAsync (
|
||||||
|
parent,
|
||||||
|
par,
|
||||||
|
AsyncThunkCrate.make (fun () -> async.Return (action ())) rc,
|
||||||
|
ec
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|> mb.PostAndAsyncReply
|
||||||
|
|
||||||
|
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.
|
||||||
|
member _.RunTestTearDown
|
||||||
|
(TestFixtureSetupToken parent)
|
||||||
|
(action : unit -> 'a)
|
||||||
|
: ('a * TestFixtureTearDownToken) Task
|
||||||
|
=
|
||||||
|
task {
|
||||||
|
let par =
|
||||||
|
parent.Parallelize
|
||||||
|
|> Option.map (fun p ->
|
||||||
|
match p with
|
||||||
|
| Parallelizable.No -> Parallelizable.No
|
||||||
|
| Parallelizable.Yes _ -> Parallelizable.Yes ()
|
||||||
|
)
|
||||||
|
|
||||||
|
let ec = ExecutionContext.Capture ()
|
||||||
|
|
||||||
|
let! response =
|
||||||
|
(fun rc ->
|
||||||
|
MailboxMessage.RunTestAsync (
|
||||||
|
parent,
|
||||||
|
par,
|
||||||
|
AsyncThunkCrate.make (fun () -> async.Return (action ())) rc,
|
||||||
|
ec
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|> mb.PostAndAsyncReply
|
||||||
|
|
||||||
|
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.
|
||||||
|
/// You don't need to worry about when the resulting Task returns, but we provide it just in case.
|
||||||
|
member _.EndTestFixture (tf : TestFixtureTearDownToken) : Task<unit> =
|
||||||
|
(fun rc -> MailboxMessage.EndTestFixture (tf, rc))
|
||||||
|
|> mb.PostAndAsyncReply
|
||||||
|
|> Async.StartAsTask
|
||||||
|
|
||||||
|
interface IDisposable with
|
||||||
|
member _.Dispose () =
|
||||||
|
// Still race conditions, of course: people could still be submitting after we finish the sync.
|
||||||
|
mb.PostAndReply MailboxMessage.Quit
|
||||||
|
(mb :> IDisposable).Dispose ()
|
29
WoofWare.NUnitTestRunner.Lib/ParallelScope.fs
Normal file
29
WoofWare.NUnitTestRunner.Lib/ParallelScope.fs
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
|
/// Our own strongly-typed rendering of the NUnit ParallelScope enum.
|
||||||
|
/// This is more tightly modelled by ClassParallelScope and AssemblyParallelScope in our own domain; this type exists
|
||||||
|
/// for the initial interop.
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
type ParallelScope =
|
||||||
|
/// Corresponds to NUnit's ParallelScope.Fixtures.
|
||||||
|
| Fixtures
|
||||||
|
/// Corresponds to NUnit's ParallelScope.Children.
|
||||||
|
| Children
|
||||||
|
/// Corresponds to NUnit's ParallelScope.All.
|
||||||
|
| All
|
||||||
|
/// Corresponds to NUnit's ParallelScope.Self.
|
||||||
|
| Self
|
||||||
|
/// Corresponds to NUnit's ParallelScope.None.
|
||||||
|
| None
|
||||||
|
|
||||||
|
[<RequireQualifiedAccess>]
|
||||||
|
module ParallelScope =
|
||||||
|
/// Convert the weakly-typed C# enum that is NUnit's `ParallelScope` to a strongly-typed representation.
|
||||||
|
let ofInt (n : int) =
|
||||||
|
match n with
|
||||||
|
| 512 -> ParallelScope.Fixtures
|
||||||
|
| 256 -> ParallelScope.Children
|
||||||
|
| 257 -> ParallelScope.All
|
||||||
|
| 1 -> ParallelScope.Self
|
||||||
|
| 2 -> ParallelScope.None
|
||||||
|
| _ -> failwith $"Unrecognised ParallelScope enum: %i{n}"
|
@@ -1,32 +1,33 @@
|
|||||||
namespace WoofWare.NUnitTestRunner
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
open System
|
open System
|
||||||
open WoofWare.Myriad.Plugins
|
|
||||||
|
|
||||||
[<JsonParse>]
|
// Myriad runs the JsonParse generator on this
|
||||||
type FrameworkDescription =
|
type internal FrameworkDescription =
|
||||||
{
|
{
|
||||||
Name : string
|
Name : string
|
||||||
Version : string
|
Version : string
|
||||||
}
|
}
|
||||||
|
|
||||||
[<JsonParse>]
|
// Myriad runs the JsonParse generator on this
|
||||||
type RuntimeOptions =
|
type internal RuntimeOptions =
|
||||||
{
|
{
|
||||||
Tfm : string
|
Tfm : string
|
||||||
Framework : FrameworkDescription option
|
Framework : FrameworkDescription option
|
||||||
Frameworks : FrameworkDescription list option
|
Frameworks : FrameworkDescription list option
|
||||||
|
IncludedFramework : FrameworkDescription option
|
||||||
|
IncludedFrameworks : FrameworkDescription list option
|
||||||
RollForward : string option
|
RollForward : string option
|
||||||
}
|
}
|
||||||
|
|
||||||
[<JsonParse>]
|
// Myriad runs the JsonParse generator on this
|
||||||
type RuntimeConfig =
|
type internal RuntimeConfig =
|
||||||
{
|
{
|
||||||
RuntimeOptions : RuntimeOptions
|
RuntimeOptions : RuntimeOptions
|
||||||
}
|
}
|
||||||
|
|
||||||
[<RequireQualifiedAccess>]
|
[<RequireQualifiedAccess>]
|
||||||
type RollForward =
|
type internal RollForward =
|
||||||
| Minor
|
| Minor
|
||||||
| Major
|
| Major
|
||||||
| LatestPatch
|
| LatestPatch
|
@@ -18,75 +18,115 @@ module SingleTestMethod =
|
|||||||
(attrs : CustomAttributeData list)
|
(attrs : CustomAttributeData list)
|
||||||
: SingleTestMethod option * CustomAttributeData list
|
: SingleTestMethod option * CustomAttributeData list
|
||||||
=
|
=
|
||||||
let remaining, isTest, sources, hasData, modifiers, categories, repeat, comb =
|
let remaining, isTest, sources, hasData, modifiers, categories, repeat, comb, par =
|
||||||
(([], false, [], None, [], [], None, None), attrs)
|
(([], false, [], None, [], [], None, None, None), attrs)
|
||||||
||> List.fold (fun (remaining, isTest, sources, hasData, mods, cats, repeat, comb) attr ->
|
||> List.fold (fun (remaining, isTest, sources, hasData, mods, cats, repeat, comb, par) attr ->
|
||||||
match attr.AttributeType.FullName with
|
match attr.AttributeType.FullName with
|
||||||
| "NUnit.Framework.TestAttribute" ->
|
| "NUnit.Framework.TestAttribute" ->
|
||||||
if attr.ConstructorArguments.Count > 0 then
|
if attr.ConstructorArguments.Count > 0 then
|
||||||
failwith "Unexpectedly got arguments to the Test attribute"
|
failwith "Unexpectedly got arguments to the Test attribute"
|
||||||
|
|
||||||
(remaining, true, sources, hasData, mods, cats, repeat, comb)
|
(remaining, true, sources, hasData, mods, cats, repeat, comb, par)
|
||||||
| "NUnit.Framework.TestCaseAttribute" ->
|
| "NUnit.Framework.TestCaseAttribute" ->
|
||||||
let args = attr.ConstructorArguments |> Seq.map _.Value |> Seq.toList
|
let args = attr.ConstructorArguments |> Seq.map _.Value |> Seq.toList
|
||||||
|
|
||||||
|
let args =
|
||||||
|
match args with
|
||||||
|
| [ :? System.Collections.ICollection as x ] ->
|
||||||
|
x
|
||||||
|
|> Seq.cast<CustomAttributeTypedArgument>
|
||||||
|
|> Seq.map (fun v -> v.Value)
|
||||||
|
|> Seq.toList
|
||||||
|
| _ -> args
|
||||||
|
|
||||||
match hasData with
|
match hasData with
|
||||||
| None -> (remaining, isTest, sources, Some [ List.ofSeq args ], mods, cats, repeat, comb)
|
| None -> (remaining, isTest, sources, Some [ List.ofSeq args ], mods, cats, repeat, comb, par)
|
||||||
| Some existing ->
|
| Some existing ->
|
||||||
(remaining, isTest, sources, Some ((List.ofSeq args) :: existing), mods, cats, repeat, comb)
|
let args = (List.ofSeq args) :: existing |> Some
|
||||||
|
(remaining, isTest, sources, args, mods, cats, repeat, comb, par)
|
||||||
| "NUnit.Framework.TestCaseSourceAttribute" ->
|
| "NUnit.Framework.TestCaseSourceAttribute" ->
|
||||||
let arg = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string>
|
let arg = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string>
|
||||||
|
|
||||||
(remaining, isTest, arg :: sources, hasData, mods, cats, repeat, comb)
|
(remaining, isTest, arg :: sources, hasData, mods, cats, repeat, comb, par)
|
||||||
| "NUnit.Framework.ExplicitAttribute" ->
|
| "NUnit.Framework.ExplicitAttribute" ->
|
||||||
let reason =
|
let reason =
|
||||||
attr.ConstructorArguments
|
attr.ConstructorArguments
|
||||||
|> Seq.tryHead
|
|> Seq.tryHead
|
||||||
|> Option.map (_.Value >> unbox<string>)
|
|> Option.map (_.Value >> unbox<string>)
|
||||||
|
|
||||||
(remaining, isTest, sources, hasData, (Modifier.Explicit reason) :: mods, cats, repeat, comb)
|
(remaining, isTest, sources, hasData, (Modifier.Explicit reason) :: mods, cats, repeat, comb, par)
|
||||||
| "NUnit.Framework.IgnoreAttribute" ->
|
| "NUnit.Framework.IgnoreAttribute" ->
|
||||||
let reason =
|
let reason =
|
||||||
attr.ConstructorArguments
|
attr.ConstructorArguments
|
||||||
|> Seq.tryHead
|
|> Seq.tryHead
|
||||||
|> Option.map (_.Value >> unbox<string>)
|
|> Option.map (_.Value >> unbox<string>)
|
||||||
|
|
||||||
(remaining, isTest, sources, hasData, (Modifier.Ignored reason) :: mods, cats, repeat, comb)
|
(remaining, isTest, sources, hasData, (Modifier.Ignored reason) :: mods, cats, repeat, comb, par)
|
||||||
| "NUnit.Framework.CategoryAttribute" ->
|
| "NUnit.Framework.CategoryAttribute" ->
|
||||||
let category =
|
let category =
|
||||||
attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string>
|
attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string>
|
||||||
|
|
||||||
(remaining, isTest, sources, hasData, mods, category :: cats, repeat, comb)
|
(remaining, isTest, sources, hasData, mods, category :: cats, repeat, comb, par)
|
||||||
| "NUnit.Framework.RepeatAttribute" ->
|
| "NUnit.Framework.RepeatAttribute" ->
|
||||||
match repeat with
|
match repeat with
|
||||||
| Some _ -> failwith $"Got RepeatAttribute multiple times on %s{method.Name}"
|
| Some _ -> failwith $"Got RepeatAttribute multiple times on %s{method.Name}"
|
||||||
| None ->
|
| None ->
|
||||||
|
|
||||||
let repeat = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<int>
|
let repeat = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<int>
|
||||||
(remaining, isTest, sources, hasData, mods, cats, Some repeat, comb)
|
(remaining, isTest, sources, hasData, mods, cats, Some repeat, comb, par)
|
||||||
| "NUnit.Framework.CombinatorialAttribute" ->
|
| "NUnit.Framework.CombinatorialAttribute" ->
|
||||||
match comb with
|
match comb with
|
||||||
| Some _ ->
|
| Some _ ->
|
||||||
failwith $"Got CombinatorialAttribute or SequentialAttribute multiple times on %s{method.Name}"
|
failwith $"Got CombinatorialAttribute or SequentialAttribute multiple times on %s{method.Name}"
|
||||||
| None ->
|
| None ->
|
||||||
(remaining, isTest, sources, hasData, mods, cats, repeat, Some Combinatorial.Combinatorial)
|
(remaining, isTest, sources, hasData, mods, cats, repeat, Some Combinatorial.Combinatorial, par)
|
||||||
| "NUnit.Framework.SequentialAttribute" ->
|
| "NUnit.Framework.SequentialAttribute" ->
|
||||||
match comb with
|
match comb with
|
||||||
| Some _ ->
|
| Some _ ->
|
||||||
failwith $"Got CombinatorialAttribute or SequentialAttribute multiple times on %s{method.Name}"
|
failwith $"Got CombinatorialAttribute or SequentialAttribute multiple times on %s{method.Name}"
|
||||||
| None -> (remaining, isTest, sources, hasData, mods, cats, repeat, Some Combinatorial.Sequential)
|
| None ->
|
||||||
|
(remaining, isTest, sources, hasData, mods, cats, repeat, Some Combinatorial.Sequential, par)
|
||||||
|
| "NUnit.Framework.NonParallelizableAttribute" ->
|
||||||
|
match par with
|
||||||
|
| Some _ -> failwith $"Got a parallelization attribute multiple times on %s{method.Name}"
|
||||||
|
| None -> (remaining, isTest, sources, hasData, mods, cats, repeat, comb, Some Parallelizable.No)
|
||||||
|
| "NUnit.Framework.ParallelizableAttribute" ->
|
||||||
|
match par with
|
||||||
|
| Some _ -> failwith $"Got multiple parallelization attributes on %s{method.Name}"
|
||||||
|
| None ->
|
||||||
|
let arg =
|
||||||
|
match Seq.toList attr.ConstructorArguments with
|
||||||
|
| [] -> Parallelizable.Yes ()
|
||||||
|
| [ x ] ->
|
||||||
|
if x.ArgumentType.Name <> "ParallelScope" then
|
||||||
|
failwith
|
||||||
|
$"Got argument %O{x.Value} of unrecognised type %s{x.ArgumentType.Name} on [<Parallelizable>] attribute; expected ParallelScope"
|
||||||
|
|
||||||
|
match ParallelScope.ofInt (unbox<int> x.Value) with
|
||||||
|
| ParallelScope.Children ->
|
||||||
|
failwith
|
||||||
|
$"Unexpected ParallelScope.Children on test %s{method.Name}; this is not valid on individual tests"
|
||||||
|
| ParallelScope.Fixtures ->
|
||||||
|
failwith
|
||||||
|
$"Unexpected ParallelScope.Children on test %s{method.Name}; this is not valid on individual tests"
|
||||||
|
| ParallelScope.All
|
||||||
|
| ParallelScope.Self -> Parallelizable.Yes ()
|
||||||
|
| ParallelScope.None -> Parallelizable.No
|
||||||
|
| s -> failwith $"Got multiple arguments on a [<Parallelizable>] attribute: %O{s}"
|
||||||
|
|
||||||
|
(remaining, isTest, sources, hasData, mods, cats, repeat, comb, Some arg)
|
||||||
| s when s.StartsWith ("NUnit.Framework", StringComparison.Ordinal) ->
|
| s when s.StartsWith ("NUnit.Framework", StringComparison.Ordinal) ->
|
||||||
failwith $"Unrecognised attribute on function %s{method.Name}: %s{attr.AttributeType.FullName}"
|
failwith $"Unrecognised attribute on function %s{method.Name}: %s{attr.AttributeType.FullName}"
|
||||||
| _ -> (attr :: remaining, isTest, sources, hasData, mods, cats, repeat, comb)
|
| _ -> (attr :: remaining, isTest, sources, hasData, mods, cats, repeat, comb, par)
|
||||||
)
|
)
|
||||||
|
|
||||||
let test =
|
let test =
|
||||||
match isTest, sources, hasData, modifiers, categories, repeat, comb with
|
match isTest, sources, hasData, modifiers, categories, repeat, comb, par with
|
||||||
| _, _ :: _, Some _, _, _, _, _ ->
|
| _, _ :: _, Some _, _, _, _, _, _ ->
|
||||||
failwith
|
failwith
|
||||||
$"Test '%s{method.Name}' unexpectedly has both TestData and TestCaseSource; not currently supported"
|
$"Test '%s{method.Name}' unexpectedly has both TestData and TestCaseSource; not currently supported"
|
||||||
| false, [], None, [], _, _, _ -> None
|
| false, [], None, [], _, _, _, _ -> None
|
||||||
| _, _ :: _, None, mods, categories, repeat, comb ->
|
| _, _ :: _, None, mods, categories, repeat, comb, par ->
|
||||||
{
|
{
|
||||||
Kind = TestKind.Source sources
|
Kind = TestKind.Source sources
|
||||||
Method = method
|
Method = method
|
||||||
@@ -94,9 +134,10 @@ module SingleTestMethod =
|
|||||||
Categories = categories @ parentCategories
|
Categories = categories @ parentCategories
|
||||||
Repeat = repeat
|
Repeat = repeat
|
||||||
Combinatorial = comb
|
Combinatorial = comb
|
||||||
|
Parallelize = par
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| _, [], Some data, mods, categories, repeat, comb ->
|
| _, [], Some data, mods, categories, repeat, comb, par ->
|
||||||
{
|
{
|
||||||
Kind = TestKind.Data data
|
Kind = TestKind.Data data
|
||||||
Method = method
|
Method = method
|
||||||
@@ -104,9 +145,10 @@ module SingleTestMethod =
|
|||||||
Categories = categories @ parentCategories
|
Categories = categories @ parentCategories
|
||||||
Repeat = repeat
|
Repeat = repeat
|
||||||
Combinatorial = comb
|
Combinatorial = comb
|
||||||
|
Parallelize = par
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| true, [], None, mods, categories, repeat, comb ->
|
| true, [], None, mods, categories, repeat, comb, par ->
|
||||||
{
|
{
|
||||||
Kind = TestKind.Single
|
Kind = TestKind.Single
|
||||||
Method = method
|
Method = method
|
||||||
@@ -114,9 +156,10 @@ module SingleTestMethod =
|
|||||||
Categories = categories @ parentCategories
|
Categories = categories @ parentCategories
|
||||||
Repeat = repeat
|
Repeat = repeat
|
||||||
Combinatorial = comb
|
Combinatorial = comb
|
||||||
|
Parallelize = par
|
||||||
}
|
}
|
||||||
|> Some
|
|> Some
|
||||||
| false, [], None, _ :: _, _, _, _ ->
|
| false, [], None, _ :: _, _, _, _, _ ->
|
||||||
failwith
|
failwith
|
||||||
$"Unexpectedly got test modifiers but no test settings on '%s{method.Name}', which you probably didn't intend."
|
$"Unexpectedly got test modifiers but no test settings on '%s{method.Name}', which you probably didn't intend."
|
||||||
|
|
||||||
|
@@ -1,8 +1,76 @@
|
|||||||
|
WoofWare.NUnitTestRunner.Args inherit obj, implements WoofWare.NUnitTestRunner.Args System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
|
WoofWare.NUnitTestRunner.Args..ctor [constructor]: (System.IO.FileInfo, System.IO.FileInfo option, (string * WoofWare.NUnitTestRunner.Filter) option, WoofWare.NUnitTestRunner.LogLevel, int option, System.TimeSpan option)
|
||||||
|
WoofWare.NUnitTestRunner.Args.Dll [property]: [read-only] System.IO.FileInfo
|
||||||
|
WoofWare.NUnitTestRunner.Args.Equals [method]: (WoofWare.NUnitTestRunner.Args, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.Args.Filter [property]: [read-only] (string * WoofWare.NUnitTestRunner.Filter) option
|
||||||
|
WoofWare.NUnitTestRunner.Args.get_Dll [method]: unit -> System.IO.FileInfo
|
||||||
|
WoofWare.NUnitTestRunner.Args.get_Filter [method]: unit -> (string * WoofWare.NUnitTestRunner.Filter) option
|
||||||
|
WoofWare.NUnitTestRunner.Args.get_LevelOfParallelism [method]: unit -> int option
|
||||||
|
WoofWare.NUnitTestRunner.Args.get_Logging [method]: unit -> WoofWare.NUnitTestRunner.LogLevel
|
||||||
|
WoofWare.NUnitTestRunner.Args.get_Timeout [method]: unit -> System.TimeSpan option
|
||||||
|
WoofWare.NUnitTestRunner.Args.get_Trx [method]: unit -> System.IO.FileInfo option
|
||||||
|
WoofWare.NUnitTestRunner.Args.LevelOfParallelism [property]: [read-only] int option
|
||||||
|
WoofWare.NUnitTestRunner.Args.Logging [property]: [read-only] WoofWare.NUnitTestRunner.LogLevel
|
||||||
|
WoofWare.NUnitTestRunner.Args.Parse [static method]: string list -> WoofWare.NUnitTestRunner.Args
|
||||||
|
WoofWare.NUnitTestRunner.Args.Timeout [property]: [read-only] System.TimeSpan option
|
||||||
|
WoofWare.NUnitTestRunner.Args.Trx [property]: [read-only] System.IO.FileInfo option
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes inherit obj, implements WoofWare.NUnitTestRunner.AssemblyLevelAttributes System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.AssemblyLevelAttributes System.IComparable, System.IComparable, System.Collections.IStructuralComparable
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes..ctor [constructor]: (int option, WoofWare.NUnitTestRunner.AssemblyParallelScope WoofWare.NUnitTestRunner.Parallelizable option)
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes.Equals [method]: (WoofWare.NUnitTestRunner.AssemblyLevelAttributes, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes.get_Parallelism [method]: unit -> int option
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes.get_Parallelizable [method]: unit -> WoofWare.NUnitTestRunner.AssemblyParallelScope WoofWare.NUnitTestRunner.Parallelizable option
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes.Parallelism [property]: [read-only] int option
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributes.Parallelizable [property]: [read-only] WoofWare.NUnitTestRunner.AssemblyParallelScope WoofWare.NUnitTestRunner.Parallelizable option
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributesModule inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyLevelAttributesModule.get [static method]: System.Reflection.Assembly -> WoofWare.NUnitTestRunner.AssemblyLevelAttributes
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope inherit obj, implements WoofWare.NUnitTestRunner.AssemblyParallelScope System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.AssemblyParallelScope System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope+Tags inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope+Tags.Children [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope+Tags.Fixtures [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.Children [static property]: [read-only] WoofWare.NUnitTestRunner.AssemblyParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.Equals [method]: (WoofWare.NUnitTestRunner.AssemblyParallelScope, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.Fixtures [static property]: [read-only] WoofWare.NUnitTestRunner.AssemblyParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.get_Children [static method]: unit -> WoofWare.NUnitTestRunner.AssemblyParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.get_Fixtures [static method]: unit -> WoofWare.NUnitTestRunner.AssemblyParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.get_IsChildren [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.get_IsFixtures [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.get_Tag [method]: unit -> int
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.IsChildren [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.IsFixtures [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.AssemblyParallelScope.Tag [property]: [read-only] int
|
||||||
|
WoofWare.NUnitTestRunner.BuildTrxReport inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.BuildTrxReport.build [static method]: System.Reflection.Assembly -> System.DateTimeOffset -> System.DateTimeOffset -> WoofWare.NUnitTestRunner.FixtureRunResults list -> WoofWare.NUnitTestRunner.TrxReport
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope inherit obj, implements WoofWare.NUnitTestRunner.ClassParallelScope System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.ClassParallelScope System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 4 cases
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope+Tags inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope+Tags.All [static field]: int = 3
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope+Tags.Children [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope+Tags.Fixtures [static field]: int = 2
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope+Tags.Self [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.All [static property]: [read-only] WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.Children [static property]: [read-only] WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.Equals [method]: (WoofWare.NUnitTestRunner.ClassParallelScope, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.Fixtures [static property]: [read-only] WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_All [static method]: unit -> WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_Children [static method]: unit -> WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_Fixtures [static method]: unit -> WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_IsAll [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_IsChildren [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_IsFixtures [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_IsSelf [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_Self [static method]: unit -> WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.get_Tag [method]: unit -> int
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.IsAll [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.IsChildren [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.IsFixtures [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.IsSelf [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.Self [static property]: [read-only] WoofWare.NUnitTestRunner.ClassParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ClassParallelScope.Tag [property]: [read-only] int
|
||||||
WoofWare.NUnitTestRunner.Combinatorial inherit obj, implements WoofWare.NUnitTestRunner.Combinatorial System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.Combinatorial System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
WoofWare.NUnitTestRunner.Combinatorial inherit obj, implements WoofWare.NUnitTestRunner.Combinatorial System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.Combinatorial System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
||||||
WoofWare.NUnitTestRunner.Combinatorial+Tags inherit obj
|
WoofWare.NUnitTestRunner.Combinatorial+Tags inherit obj
|
||||||
WoofWare.NUnitTestRunner.Combinatorial+Tags.Combinatorial [static field]: int = 0
|
WoofWare.NUnitTestRunner.Combinatorial+Tags.Combinatorial [static field]: int = 0
|
||||||
WoofWare.NUnitTestRunner.Combinatorial+Tags.Sequential [static field]: int = 1
|
WoofWare.NUnitTestRunner.Combinatorial+Tags.Sequential [static field]: int = 1
|
||||||
WoofWare.NUnitTestRunner.Combinatorial.Combinatorial [static property]: [read-only] WoofWare.NUnitTestRunner.Combinatorial
|
WoofWare.NUnitTestRunner.Combinatorial.Combinatorial [static property]: [read-only] WoofWare.NUnitTestRunner.Combinatorial
|
||||||
|
WoofWare.NUnitTestRunner.Combinatorial.Equals [method]: (WoofWare.NUnitTestRunner.Combinatorial, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.Combinatorial.get_Combinatorial [static method]: unit -> WoofWare.NUnitTestRunner.Combinatorial
|
WoofWare.NUnitTestRunner.Combinatorial.get_Combinatorial [static method]: unit -> WoofWare.NUnitTestRunner.Combinatorial
|
||||||
WoofWare.NUnitTestRunner.Combinatorial.get_IsCombinatorial [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Combinatorial.get_IsCombinatorial [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Combinatorial.get_IsSequential [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Combinatorial.get_IsSequential [method]: unit -> bool
|
||||||
@@ -12,6 +80,8 @@ WoofWare.NUnitTestRunner.Combinatorial.IsCombinatorial [property]: [read-only] b
|
|||||||
WoofWare.NUnitTestRunner.Combinatorial.IsSequential [property]: [read-only] bool
|
WoofWare.NUnitTestRunner.Combinatorial.IsSequential [property]: [read-only] bool
|
||||||
WoofWare.NUnitTestRunner.Combinatorial.Sequential [static property]: [read-only] WoofWare.NUnitTestRunner.Combinatorial
|
WoofWare.NUnitTestRunner.Combinatorial.Sequential [static property]: [read-only] WoofWare.NUnitTestRunner.Combinatorial
|
||||||
WoofWare.NUnitTestRunner.Combinatorial.Tag [property]: [read-only] int
|
WoofWare.NUnitTestRunner.Combinatorial.Tag [property]: [read-only] int
|
||||||
|
WoofWare.NUnitTestRunner.DotnetRuntime inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.DotnetRuntime.locate [static method]: System.IO.FileInfo -> System.IO.DirectoryInfo list
|
||||||
WoofWare.NUnitTestRunner.Filter inherit obj, implements WoofWare.NUnitTestRunner.Filter System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.Filter System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 6 cases
|
WoofWare.NUnitTestRunner.Filter inherit obj, implements WoofWare.NUnitTestRunner.Filter System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.Filter System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 6 cases
|
||||||
WoofWare.NUnitTestRunner.Filter+And inherit WoofWare.NUnitTestRunner.Filter
|
WoofWare.NUnitTestRunner.Filter+And inherit WoofWare.NUnitTestRunner.Filter
|
||||||
WoofWare.NUnitTestRunner.Filter+And.get_Item1 [method]: unit -> WoofWare.NUnitTestRunner.Filter
|
WoofWare.NUnitTestRunner.Filter+And.get_Item1 [method]: unit -> WoofWare.NUnitTestRunner.Filter
|
||||||
@@ -42,6 +112,7 @@ WoofWare.NUnitTestRunner.Filter+Tags.TestCategory [static field]: int = 2
|
|||||||
WoofWare.NUnitTestRunner.Filter+TestCategory inherit WoofWare.NUnitTestRunner.Filter
|
WoofWare.NUnitTestRunner.Filter+TestCategory inherit WoofWare.NUnitTestRunner.Filter
|
||||||
WoofWare.NUnitTestRunner.Filter+TestCategory.get_Item [method]: unit -> WoofWare.NUnitTestRunner.Match
|
WoofWare.NUnitTestRunner.Filter+TestCategory.get_Item [method]: unit -> WoofWare.NUnitTestRunner.Match
|
||||||
WoofWare.NUnitTestRunner.Filter+TestCategory.Item [property]: [read-only] WoofWare.NUnitTestRunner.Match
|
WoofWare.NUnitTestRunner.Filter+TestCategory.Item [property]: [read-only] WoofWare.NUnitTestRunner.Match
|
||||||
|
WoofWare.NUnitTestRunner.Filter.Equals [method]: (WoofWare.NUnitTestRunner.Filter, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.Filter.get_IsAnd [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Filter.get_IsAnd [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Filter.get_IsFullyQualifiedName [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Filter.get_IsFullyQualifiedName [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Filter.get_IsName [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Filter.get_IsName [method]: unit -> bool
|
||||||
@@ -67,6 +138,7 @@ WoofWare.NUnitTestRunner.FilterModule.parse [static method]: string -> WoofWare.
|
|||||||
WoofWare.NUnitTestRunner.FilterModule.shouldRun [static method]: WoofWare.NUnitTestRunner.Filter -> (WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.SingleTestMethod -> bool)
|
WoofWare.NUnitTestRunner.FilterModule.shouldRun [static method]: WoofWare.NUnitTestRunner.Filter -> (WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.SingleTestMethod -> bool)
|
||||||
WoofWare.NUnitTestRunner.FixtureRunResults inherit obj, implements WoofWare.NUnitTestRunner.FixtureRunResults System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.FixtureRunResults inherit obj, implements WoofWare.NUnitTestRunner.FixtureRunResults System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.FixtureRunResults..ctor [constructor]: ((WoofWare.NUnitTestRunner.TestMemberFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list, (WoofWare.NUnitTestRunner.SingleTestMethod * WoofWare.NUnitTestRunner.TestMemberSuccess * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list, (WoofWare.NUnitTestRunner.UserMethodFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list)
|
WoofWare.NUnitTestRunner.FixtureRunResults..ctor [constructor]: ((WoofWare.NUnitTestRunner.TestMemberFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list, (WoofWare.NUnitTestRunner.SingleTestMethod * WoofWare.NUnitTestRunner.TestMemberSuccess * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list, (WoofWare.NUnitTestRunner.UserMethodFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list)
|
||||||
|
WoofWare.NUnitTestRunner.FixtureRunResults.Equals [method]: (WoofWare.NUnitTestRunner.FixtureRunResults, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.FixtureRunResults.Failed [property]: [read-only] (WoofWare.NUnitTestRunner.TestMemberFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list
|
WoofWare.NUnitTestRunner.FixtureRunResults.Failed [property]: [read-only] (WoofWare.NUnitTestRunner.TestMemberFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list
|
||||||
WoofWare.NUnitTestRunner.FixtureRunResults.get_Failed [method]: unit -> (WoofWare.NUnitTestRunner.TestMemberFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list
|
WoofWare.NUnitTestRunner.FixtureRunResults.get_Failed [method]: unit -> (WoofWare.NUnitTestRunner.TestMemberFailure * WoofWare.NUnitTestRunner.IndividualTestRunMetadata) list
|
||||||
WoofWare.NUnitTestRunner.FixtureRunResults.get_IndividualTestRunMetadata [method]: unit -> (WoofWare.NUnitTestRunner.IndividualTestRunMetadata * Microsoft.FSharp.Core.FSharpChoice<WoofWare.NUnitTestRunner.TestMemberFailure, WoofWare.NUnitTestRunner.TestMemberSuccess, WoofWare.NUnitTestRunner.UserMethodFailure>) list
|
WoofWare.NUnitTestRunner.FixtureRunResults.get_IndividualTestRunMetadata [method]: unit -> (WoofWare.NUnitTestRunner.IndividualTestRunMetadata * Microsoft.FSharp.Core.FSharpChoice<WoofWare.NUnitTestRunner.TestMemberFailure, WoofWare.NUnitTestRunner.TestMemberSuccess, WoofWare.NUnitTestRunner.UserMethodFailure>) list
|
||||||
@@ -80,6 +152,7 @@ WoofWare.NUnitTestRunner.IndividualTestRunMetadata..ctor [constructor]: (System.
|
|||||||
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.ClassName [property]: [read-only] string
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.ClassName [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.ComputerName [property]: [read-only] string
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.ComputerName [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.End [property]: [read-only] System.DateTimeOffset
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.End [property]: [read-only] System.DateTimeOffset
|
||||||
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.Equals [method]: (WoofWare.NUnitTestRunner.IndividualTestRunMetadata, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.ExecutionId [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.ExecutionId [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.get_ClassName [method]: unit -> string
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.get_ClassName [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.get_ComputerName [method]: unit -> string
|
WoofWare.NUnitTestRunner.IndividualTestRunMetadata.get_ComputerName [method]: unit -> string
|
||||||
@@ -97,12 +170,30 @@ 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
|
||||||
WoofWare.NUnitTestRunner.ITestProgress.OnTestMemberStart [method]: string -> unit
|
WoofWare.NUnitTestRunner.ITestProgress.OnTestMemberStart [method]: string -> unit
|
||||||
|
WoofWare.NUnitTestRunner.LoadContext inherit System.Runtime.Loader.AssemblyLoadContext
|
||||||
|
WoofWare.NUnitTestRunner.LoadContext..ctor [constructor]: (System.IO.FileInfo, System.IO.DirectoryInfo list, WoofWare.NUnitTestRunner.TestContexts)
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel inherit obj, implements WoofWare.NUnitTestRunner.LogLevel System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.LogLevel System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel+Tags inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel+Tags.Nothing [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel+Tags.Verbose [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.Equals [method]: (WoofWare.NUnitTestRunner.LogLevel, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.get_IsNothing [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.get_IsVerbose [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.get_Nothing [static method]: unit -> WoofWare.NUnitTestRunner.LogLevel
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.get_Tag [method]: unit -> int
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.get_Verbose [static method]: unit -> WoofWare.NUnitTestRunner.LogLevel
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.IsNothing [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.IsVerbose [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.Nothing [static property]: [read-only] WoofWare.NUnitTestRunner.LogLevel
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.Tag [property]: [read-only] int
|
||||||
|
WoofWare.NUnitTestRunner.LogLevel.Verbose [static property]: [read-only] WoofWare.NUnitTestRunner.LogLevel
|
||||||
WoofWare.NUnitTestRunner.Match inherit obj, implements WoofWare.NUnitTestRunner.Match System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.Match System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
WoofWare.NUnitTestRunner.Match inherit obj, implements WoofWare.NUnitTestRunner.Match System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.Match System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
||||||
WoofWare.NUnitTestRunner.Match+Contains inherit WoofWare.NUnitTestRunner.Match
|
WoofWare.NUnitTestRunner.Match+Contains inherit WoofWare.NUnitTestRunner.Match
|
||||||
WoofWare.NUnitTestRunner.Match+Contains.get_Item [method]: unit -> string
|
WoofWare.NUnitTestRunner.Match+Contains.get_Item [method]: unit -> string
|
||||||
@@ -113,6 +204,7 @@ WoofWare.NUnitTestRunner.Match+Exact.Item [property]: [read-only] string
|
|||||||
WoofWare.NUnitTestRunner.Match+Tags inherit obj
|
WoofWare.NUnitTestRunner.Match+Tags inherit obj
|
||||||
WoofWare.NUnitTestRunner.Match+Tags.Contains [static field]: int = 1
|
WoofWare.NUnitTestRunner.Match+Tags.Contains [static field]: int = 1
|
||||||
WoofWare.NUnitTestRunner.Match+Tags.Exact [static field]: int = 0
|
WoofWare.NUnitTestRunner.Match+Tags.Exact [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.Match.Equals [method]: (WoofWare.NUnitTestRunner.Match, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.Match.get_IsContains [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Match.get_IsContains [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Match.get_IsExact [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Match.get_IsExact [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Match.get_Tag [method]: unit -> int
|
WoofWare.NUnitTestRunner.Match.get_Tag [method]: unit -> int
|
||||||
@@ -131,6 +223,7 @@ WoofWare.NUnitTestRunner.Modifier+Ignored.reason [property]: [read-only] string
|
|||||||
WoofWare.NUnitTestRunner.Modifier+Tags inherit obj
|
WoofWare.NUnitTestRunner.Modifier+Tags inherit obj
|
||||||
WoofWare.NUnitTestRunner.Modifier+Tags.Explicit [static field]: int = 0
|
WoofWare.NUnitTestRunner.Modifier+Tags.Explicit [static field]: int = 0
|
||||||
WoofWare.NUnitTestRunner.Modifier+Tags.Ignored [static field]: int = 1
|
WoofWare.NUnitTestRunner.Modifier+Tags.Ignored [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.Modifier.Equals [method]: (WoofWare.NUnitTestRunner.Modifier, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.Modifier.get_IsExplicit [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Modifier.get_IsExplicit [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Modifier.get_IsIgnored [method]: unit -> bool
|
WoofWare.NUnitTestRunner.Modifier.get_IsIgnored [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.Modifier.get_Tag [method]: unit -> int
|
WoofWare.NUnitTestRunner.Modifier.get_Tag [method]: unit -> int
|
||||||
@@ -139,24 +232,93 @@ WoofWare.NUnitTestRunner.Modifier.IsIgnored [property]: [read-only] bool
|
|||||||
WoofWare.NUnitTestRunner.Modifier.NewExplicit [static method]: string option -> WoofWare.NUnitTestRunner.Modifier
|
WoofWare.NUnitTestRunner.Modifier.NewExplicit [static method]: string option -> WoofWare.NUnitTestRunner.Modifier
|
||||||
WoofWare.NUnitTestRunner.Modifier.NewIgnored [static method]: string option -> WoofWare.NUnitTestRunner.Modifier
|
WoofWare.NUnitTestRunner.Modifier.NewIgnored [static method]: string option -> WoofWare.NUnitTestRunner.Modifier
|
||||||
WoofWare.NUnitTestRunner.Modifier.Tag [property]: [read-only] int
|
WoofWare.NUnitTestRunner.Modifier.Tag [property]: [read-only] int
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable.bind [static method]: ('a -> 'b WoofWare.NUnitTestRunner.Parallelizable) -> 'a WoofWare.NUnitTestRunner.Parallelizable -> 'b WoofWare.NUnitTestRunner.Parallelizable
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable.map [static method]: ('a -> 'b) -> 'a WoofWare.NUnitTestRunner.Parallelizable -> 'b WoofWare.NUnitTestRunner.Parallelizable
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1 inherit obj, implements 'scope WoofWare.NUnitTestRunner.Parallelizable System.IEquatable, System.Collections.IStructuralEquatable, 'scope WoofWare.NUnitTestRunner.Parallelizable System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 2 cases
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1+Tags inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1+Tags.No [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1+Tags.Yes [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1+Yes inherit 'scope WoofWare.NUnitTestRunner.Parallelizable
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1+Yes.get_Item [method]: unit -> 'scope
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1+Yes.Item [property]: [read-only] 'scope
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.Equals [method]: ('scope WoofWare.NUnitTestRunner.Parallelizable, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.get_IsNo [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.get_IsYes [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.get_No [static method]: unit -> 'scope WoofWare.NUnitTestRunner.Parallelizable
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.get_Tag [method]: unit -> int
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.IsNo [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.IsYes [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.NewYes [static method]: 'scope -> 'scope WoofWare.NUnitTestRunner.Parallelizable
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.No [static property]: [read-only] 'scope WoofWare.NUnitTestRunner.Parallelizable
|
||||||
|
WoofWare.NUnitTestRunner.Parallelizable`1.Tag [property]: [read-only] int
|
||||||
|
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.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.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.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.ParallelScope inherit obj, implements WoofWare.NUnitTestRunner.ParallelScope System.IEquatable, System.Collections.IStructuralEquatable, WoofWare.NUnitTestRunner.ParallelScope System.IComparable, System.IComparable, System.Collections.IStructuralComparable - union type with 5 cases
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope+Tags inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope+Tags.All [static field]: int = 2
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope+Tags.Children [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope+Tags.Fixtures [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope+Tags.None [static field]: int = 4
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope+Tags.Self [static field]: int = 3
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.All [static property]: [read-only] WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.Children [static property]: [read-only] WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.Equals [method]: (WoofWare.NUnitTestRunner.ParallelScope, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.Fixtures [static property]: [read-only] WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_All [static method]: unit -> WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_Children [static method]: unit -> WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_Fixtures [static method]: unit -> WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_IsAll [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_IsChildren [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_IsFixtures [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_IsNone [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_IsSelf [method]: unit -> bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_None [static method]: unit -> WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_Self [static method]: unit -> WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.get_Tag [method]: unit -> int
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.IsAll [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.IsChildren [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.IsFixtures [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.IsNone [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.IsSelf [property]: [read-only] bool
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.None [static property]: [read-only] WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.Self [static property]: [read-only] WoofWare.NUnitTestRunner.ParallelScope
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScope.Tag [property]: [read-only] int
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScopeModule inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.ParallelScopeModule.ofInt [static method]: int -> WoofWare.NUnitTestRunner.ParallelScope
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod inherit obj, implements WoofWare.NUnitTestRunner.SingleTestMethod System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.SingleTestMethod inherit obj, implements WoofWare.NUnitTestRunner.SingleTestMethod System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod..ctor [constructor]: (System.Reflection.MethodInfo, WoofWare.NUnitTestRunner.TestKind, WoofWare.NUnitTestRunner.Modifier list, string list, int option, WoofWare.NUnitTestRunner.Combinatorial option)
|
WoofWare.NUnitTestRunner.SingleTestMethod..ctor [constructor]: (System.Reflection.MethodInfo, WoofWare.NUnitTestRunner.TestKind, WoofWare.NUnitTestRunner.Modifier list, string list, int option, WoofWare.NUnitTestRunner.Combinatorial option, unit WoofWare.NUnitTestRunner.Parallelizable option)
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Categories [property]: [read-only] string list
|
WoofWare.NUnitTestRunner.SingleTestMethod.Categories [property]: [read-only] string list
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Combinatorial [property]: [read-only] WoofWare.NUnitTestRunner.Combinatorial option
|
WoofWare.NUnitTestRunner.SingleTestMethod.Combinatorial [property]: [read-only] WoofWare.NUnitTestRunner.Combinatorial option
|
||||||
|
WoofWare.NUnitTestRunner.SingleTestMethod.Equals [method]: (WoofWare.NUnitTestRunner.SingleTestMethod, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Categories [method]: unit -> string list
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Categories [method]: unit -> string list
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Combinatorial [method]: unit -> WoofWare.NUnitTestRunner.Combinatorial option
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Combinatorial [method]: unit -> WoofWare.NUnitTestRunner.Combinatorial option
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Kind [method]: unit -> WoofWare.NUnitTestRunner.TestKind
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Kind [method]: unit -> WoofWare.NUnitTestRunner.TestKind
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Method [method]: unit -> System.Reflection.MethodInfo
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Method [method]: unit -> System.Reflection.MethodInfo
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Modifiers [method]: unit -> WoofWare.NUnitTestRunner.Modifier list
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Modifiers [method]: unit -> WoofWare.NUnitTestRunner.Modifier list
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Name [method]: unit -> string
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Name [method]: unit -> string
|
||||||
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Parallelize [method]: unit -> unit WoofWare.NUnitTestRunner.Parallelizable option
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.get_Repeat [method]: unit -> int option
|
WoofWare.NUnitTestRunner.SingleTestMethod.get_Repeat [method]: unit -> int option
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Kind [property]: [read-only] WoofWare.NUnitTestRunner.TestKind
|
WoofWare.NUnitTestRunner.SingleTestMethod.Kind [property]: [read-only] WoofWare.NUnitTestRunner.TestKind
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Method [property]: [read-only] System.Reflection.MethodInfo
|
WoofWare.NUnitTestRunner.SingleTestMethod.Method [property]: [read-only] System.Reflection.MethodInfo
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Modifiers [property]: [read-only] WoofWare.NUnitTestRunner.Modifier list
|
WoofWare.NUnitTestRunner.SingleTestMethod.Modifiers [property]: [read-only] WoofWare.NUnitTestRunner.Modifier list
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Name [property]: [read-only] string
|
WoofWare.NUnitTestRunner.SingleTestMethod.Name [property]: [read-only] string
|
||||||
|
WoofWare.NUnitTestRunner.SingleTestMethod.Parallelize [property]: [read-only] unit WoofWare.NUnitTestRunner.Parallelizable option
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethod.Repeat [property]: [read-only] int option
|
WoofWare.NUnitTestRunner.SingleTestMethod.Repeat [property]: [read-only] int option
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethodModule inherit obj
|
WoofWare.NUnitTestRunner.SingleTestMethodModule inherit obj
|
||||||
WoofWare.NUnitTestRunner.SingleTestMethodModule.parse [static method]: string list -> System.Reflection.MethodInfo -> System.Reflection.CustomAttributeData list -> (WoofWare.NUnitTestRunner.SingleTestMethod option * System.Reflection.CustomAttributeData list)
|
WoofWare.NUnitTestRunner.SingleTestMethodModule.parse [static method]: string list -> System.Reflection.MethodInfo -> System.Reflection.CustomAttributeData list -> (WoofWare.NUnitTestRunner.SingleTestMethod option * System.Reflection.CustomAttributeData list)
|
||||||
|
WoofWare.NUnitTestRunner.TestContexts inherit obj, implements IDisposable
|
||||||
|
WoofWare.NUnitTestRunner.TestContexts.Empty [static method]: unit -> WoofWare.NUnitTestRunner.TestContexts
|
||||||
|
WoofWare.NUnitTestRunner.TestContexts.get_Stderr [method]: unit -> System.IO.TextWriter
|
||||||
|
WoofWare.NUnitTestRunner.TestContexts.get_Stdout [method]: unit -> System.IO.TextWriter
|
||||||
|
WoofWare.NUnitTestRunner.TestContexts.Stderr [property]: [read-only] System.IO.TextWriter
|
||||||
|
WoofWare.NUnitTestRunner.TestContexts.Stdout [property]: [read-only] System.IO.TextWriter
|
||||||
WoofWare.NUnitTestRunner.TestFailure inherit obj, implements WoofWare.NUnitTestRunner.TestFailure System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
WoofWare.NUnitTestRunner.TestFailure inherit obj, implements WoofWare.NUnitTestRunner.TestFailure System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
||||||
WoofWare.NUnitTestRunner.TestFailure+SetUpFailed inherit WoofWare.NUnitTestRunner.TestFailure
|
WoofWare.NUnitTestRunner.TestFailure+SetUpFailed inherit WoofWare.NUnitTestRunner.TestFailure
|
||||||
WoofWare.NUnitTestRunner.TestFailure+SetUpFailed.get_Item [method]: unit -> WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.TestFailure+SetUpFailed.get_Item [method]: unit -> WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
@@ -171,6 +333,7 @@ WoofWare.NUnitTestRunner.TestFailure+TearDownFailed.Item [property]: [read-only]
|
|||||||
WoofWare.NUnitTestRunner.TestFailure+TestFailed inherit WoofWare.NUnitTestRunner.TestFailure
|
WoofWare.NUnitTestRunner.TestFailure+TestFailed inherit WoofWare.NUnitTestRunner.TestFailure
|
||||||
WoofWare.NUnitTestRunner.TestFailure+TestFailed.get_Item [method]: unit -> WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.TestFailure+TestFailed.get_Item [method]: unit -> WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
WoofWare.NUnitTestRunner.TestFailure+TestFailed.Item [property]: [read-only] WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.TestFailure+TestFailed.Item [property]: [read-only] WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
|
WoofWare.NUnitTestRunner.TestFailure.Equals [method]: (WoofWare.NUnitTestRunner.TestFailure, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TestFailure.get_IsSetUpFailed [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestFailure.get_IsSetUpFailed [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestFailure.get_IsTearDownFailed [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestFailure.get_IsTearDownFailed [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestFailure.get_IsTestFailed [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestFailure.get_IsTestFailed [method]: unit -> bool
|
||||||
@@ -185,25 +348,41 @@ 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.Reflection.MethodInfo option, System.Reflection.MethodInfo option, System.Reflection.MethodInfo list, System.Reflection.MethodInfo list, WoofWare.NUnitTestRunner.SingleTestMethod list)
|
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.Reflection.Assembly -> string -> 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.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
|
||||||
|
WoofWare.NUnitTestRunner.TestFixture.get_Parallelize [method]: unit -> WoofWare.NUnitTestRunner.ClassParallelScope WoofWare.NUnitTestRunner.Parallelizable option
|
||||||
|
WoofWare.NUnitTestRunner.TestFixture.get_Parameters [method]: unit -> obj list list
|
||||||
WoofWare.NUnitTestRunner.TestFixture.get_SetUp [method]: unit -> System.Reflection.MethodInfo list
|
WoofWare.NUnitTestRunner.TestFixture.get_SetUp [method]: unit -> System.Reflection.MethodInfo list
|
||||||
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.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
|
||||||
|
WoofWare.NUnitTestRunner.TestFixture.Parallelize [property]: [read-only] WoofWare.NUnitTestRunner.ClassParallelScope WoofWare.NUnitTestRunner.Parallelizable option
|
||||||
|
WoofWare.NUnitTestRunner.TestFixture.Parameters [property]: [read-only] obj list list
|
||||||
WoofWare.NUnitTestRunner.TestFixture.SetUp [property]: [read-only] System.Reflection.MethodInfo list
|
WoofWare.NUnitTestRunner.TestFixture.SetUp [property]: [read-only] System.Reflection.MethodInfo list
|
||||||
WoofWare.NUnitTestRunner.TestFixture.TearDown [property]: [read-only] System.Reflection.MethodInfo list
|
WoofWare.NUnitTestRunner.TestFixture.TearDown [property]: [read-only] System.Reflection.MethodInfo list
|
||||||
WoofWare.NUnitTestRunner.TestFixture.Tests [property]: [read-only] WoofWare.NUnitTestRunner.SingleTestMethod list
|
WoofWare.NUnitTestRunner.TestFixture.Tests [property]: [read-only] WoofWare.NUnitTestRunner.SingleTestMethod list
|
||||||
|
WoofWare.NUnitTestRunner.TestFixture.Type [property]: [read-only] System.Type
|
||||||
WoofWare.NUnitTestRunner.TestFixtureModule inherit obj
|
WoofWare.NUnitTestRunner.TestFixtureModule inherit obj
|
||||||
WoofWare.NUnitTestRunner.TestFixtureModule.parse [static method]: System.Type -> WoofWare.NUnitTestRunner.TestFixture
|
WoofWare.NUnitTestRunner.TestFixtureModule.parse [static method]: System.Type -> WoofWare.NUnitTestRunner.TestFixture
|
||||||
WoofWare.NUnitTestRunner.TestFixtureModule.run [static method]: WoofWare.NUnitTestRunner.ITestProgress -> (WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.SingleTestMethod -> bool) -> WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.FixtureRunResults
|
WoofWare.NUnitTestRunner.TestFixtureModule.run [static method]: WoofWare.NUnitTestRunner.TestContexts -> WoofWare.NUnitTestRunner.ParallelQueue -> WoofWare.NUnitTestRunner.ITestProgress -> (WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.SingleTestMethod -> bool) -> WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.FixtureRunResults list System.Threading.Tasks.Task
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureModule.runOneFixture [static method]: WoofWare.NUnitTestRunner.TestContexts -> WoofWare.NUnitTestRunner.ParallelQueue -> WoofWare.NUnitTestRunner.ITestProgress -> (WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.SingleTestMethod -> bool) -> string -> obj -> WoofWare.NUnitTestRunner.TestFixture -> WoofWare.NUnitTestRunner.FixtureRunResults System.Threading.Tasks.Task
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureRunningToken inherit obj, implements WoofWare.NUnitTestRunner.TestFixtureRunningToken System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureRunningToken.Equals [method]: (WoofWare.NUnitTestRunner.TestFixtureRunningToken, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureSetupToken inherit obj, implements WoofWare.NUnitTestRunner.TestFixtureSetupToken System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureSetupToken.Equals [method]: (WoofWare.NUnitTestRunner.TestFixtureSetupToken, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureTearDownToken inherit obj, implements WoofWare.NUnitTestRunner.TestFixtureTearDownToken System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
|
WoofWare.NUnitTestRunner.TestFixtureTearDownToken.Equals [method]: (WoofWare.NUnitTestRunner.TestFixtureTearDownToken, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TestKind inherit obj, implements WoofWare.NUnitTestRunner.TestKind System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
WoofWare.NUnitTestRunner.TestKind inherit obj, implements WoofWare.NUnitTestRunner.TestKind System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
||||||
WoofWare.NUnitTestRunner.TestKind+Data inherit WoofWare.NUnitTestRunner.TestKind
|
WoofWare.NUnitTestRunner.TestKind+Data inherit WoofWare.NUnitTestRunner.TestKind
|
||||||
WoofWare.NUnitTestRunner.TestKind+Data.get_Item [method]: unit -> obj list list
|
WoofWare.NUnitTestRunner.TestKind+Data.get_Item [method]: unit -> obj list list
|
||||||
@@ -215,6 +394,7 @@ WoofWare.NUnitTestRunner.TestKind+Tags inherit obj
|
|||||||
WoofWare.NUnitTestRunner.TestKind+Tags.Data [static field]: int = 2
|
WoofWare.NUnitTestRunner.TestKind+Tags.Data [static field]: int = 2
|
||||||
WoofWare.NUnitTestRunner.TestKind+Tags.Single [static field]: int = 0
|
WoofWare.NUnitTestRunner.TestKind+Tags.Single [static field]: int = 0
|
||||||
WoofWare.NUnitTestRunner.TestKind+Tags.Source [static field]: int = 1
|
WoofWare.NUnitTestRunner.TestKind+Tags.Source [static field]: int = 1
|
||||||
|
WoofWare.NUnitTestRunner.TestKind.Equals [method]: (WoofWare.NUnitTestRunner.TestKind, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TestKind.get_IsData [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestKind.get_IsData [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestKind.get_IsSingle [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestKind.get_IsSingle [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestKind.get_IsSource [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestKind.get_IsSource [method]: unit -> bool
|
||||||
@@ -237,6 +417,7 @@ WoofWare.NUnitTestRunner.TestMemberFailure+Malformed.reasons [property]: [read-o
|
|||||||
WoofWare.NUnitTestRunner.TestMemberFailure+Tags inherit obj
|
WoofWare.NUnitTestRunner.TestMemberFailure+Tags inherit obj
|
||||||
WoofWare.NUnitTestRunner.TestMemberFailure+Tags.Failed [static field]: int = 1
|
WoofWare.NUnitTestRunner.TestMemberFailure+Tags.Failed [static field]: int = 1
|
||||||
WoofWare.NUnitTestRunner.TestMemberFailure+Tags.Malformed [static field]: int = 0
|
WoofWare.NUnitTestRunner.TestMemberFailure+Tags.Malformed [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.TestMemberFailure.Equals [method]: (WoofWare.NUnitTestRunner.TestMemberFailure, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TestMemberFailure.get_IsFailed [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestMemberFailure.get_IsFailed [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestMemberFailure.get_IsMalformed [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestMemberFailure.get_IsMalformed [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestMemberFailure.get_Tag [method]: unit -> int
|
WoofWare.NUnitTestRunner.TestMemberFailure.get_Tag [method]: unit -> int
|
||||||
@@ -260,6 +441,7 @@ WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Explicit [static field]: int = 2
|
|||||||
WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Ignored [static field]: int = 1
|
WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Ignored [static field]: int = 1
|
||||||
WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Inconclusive [static field]: int = 3
|
WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Inconclusive [static field]: int = 3
|
||||||
WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Ok [static field]: int = 0
|
WoofWare.NUnitTestRunner.TestMemberSuccess+Tags.Ok [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.TestMemberSuccess.Equals [method]: (WoofWare.NUnitTestRunner.TestMemberSuccess, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TestMemberSuccess.get_IsExplicit [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestMemberSuccess.get_IsExplicit [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestMemberSuccess.get_IsIgnored [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestMemberSuccess.get_IsIgnored [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.TestMemberSuccess.get_IsInconclusive [method]: unit -> bool
|
WoofWare.NUnitTestRunner.TestMemberSuccess.get_IsInconclusive [method]: unit -> bool
|
||||||
@@ -277,6 +459,7 @@ WoofWare.NUnitTestRunner.TestMemberSuccess.Ok [static property]: [read-only] Woo
|
|||||||
WoofWare.NUnitTestRunner.TestMemberSuccess.Tag [property]: [read-only] int
|
WoofWare.NUnitTestRunner.TestMemberSuccess.Tag [property]: [read-only] int
|
||||||
WoofWare.NUnitTestRunner.TestProgress inherit obj
|
WoofWare.NUnitTestRunner.TestProgress inherit obj
|
||||||
WoofWare.NUnitTestRunner.TestProgress.toStderr [static method]: unit -> WoofWare.NUnitTestRunner.ITestProgress
|
WoofWare.NUnitTestRunner.TestProgress.toStderr [static method]: unit -> WoofWare.NUnitTestRunner.ITestProgress
|
||||||
|
WoofWare.NUnitTestRunner.TestProgress.toWriter [static method]: System.IO.TextWriter -> WoofWare.NUnitTestRunner.ITestProgress
|
||||||
WoofWare.NUnitTestRunner.TrxCounters inherit obj, implements WoofWare.NUnitTestRunner.TrxCounters System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxCounters inherit obj, implements WoofWare.NUnitTestRunner.TrxCounters System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxCounters..ctor [constructor]: (System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32)
|
WoofWare.NUnitTestRunner.TrxCounters..ctor [constructor]: (System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32, System.UInt32)
|
||||||
WoofWare.NUnitTestRunner.TrxCounters.Aborted [property]: [read-only] System.UInt32
|
WoofWare.NUnitTestRunner.TrxCounters.Aborted [property]: [read-only] System.UInt32
|
||||||
@@ -286,6 +469,7 @@ WoofWare.NUnitTestRunner.TrxCounters.AddNotExecuted [method]: unit -> WoofWare.N
|
|||||||
WoofWare.NUnitTestRunner.TrxCounters.AddPassed [method]: unit -> WoofWare.NUnitTestRunner.TrxCounters
|
WoofWare.NUnitTestRunner.TrxCounters.AddPassed [method]: unit -> WoofWare.NUnitTestRunner.TrxCounters
|
||||||
WoofWare.NUnitTestRunner.TrxCounters.Completed [property]: [read-only] System.UInt32
|
WoofWare.NUnitTestRunner.TrxCounters.Completed [property]: [read-only] System.UInt32
|
||||||
WoofWare.NUnitTestRunner.TrxCounters.Disconnected [property]: [read-only] System.UInt32
|
WoofWare.NUnitTestRunner.TrxCounters.Disconnected [property]: [read-only] System.UInt32
|
||||||
|
WoofWare.NUnitTestRunner.TrxCounters.Equals [method]: (WoofWare.NUnitTestRunner.TrxCounters, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxCounters.Errors [property]: [read-only] System.UInt32
|
WoofWare.NUnitTestRunner.TrxCounters.Errors [property]: [read-only] System.UInt32
|
||||||
WoofWare.NUnitTestRunner.TrxCounters.Executed [property]: [read-only] System.UInt32
|
WoofWare.NUnitTestRunner.TrxCounters.Executed [property]: [read-only] System.UInt32
|
||||||
WoofWare.NUnitTestRunner.TrxCounters.Failed [property]: [read-only] System.UInt32
|
WoofWare.NUnitTestRunner.TrxCounters.Failed [property]: [read-only] System.UInt32
|
||||||
@@ -319,16 +503,19 @@ WoofWare.NUnitTestRunner.TrxCounters.Warning [property]: [read-only] System.UInt
|
|||||||
WoofWare.NUnitTestRunner.TrxCounters.Zero [static property]: [read-only] WoofWare.NUnitTestRunner.TrxCounters
|
WoofWare.NUnitTestRunner.TrxCounters.Zero [static property]: [read-only] WoofWare.NUnitTestRunner.TrxCounters
|
||||||
WoofWare.NUnitTestRunner.TrxDeployment inherit obj, implements WoofWare.NUnitTestRunner.TrxDeployment System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxDeployment inherit obj, implements WoofWare.NUnitTestRunner.TrxDeployment System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxDeployment..ctor [constructor]: string
|
WoofWare.NUnitTestRunner.TrxDeployment..ctor [constructor]: string
|
||||||
|
WoofWare.NUnitTestRunner.TrxDeployment.Equals [method]: (WoofWare.NUnitTestRunner.TrxDeployment, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxDeployment.get_RunDeploymentRoot [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxDeployment.get_RunDeploymentRoot [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.TrxDeployment.RunDeploymentRoot [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxDeployment.RunDeploymentRoot [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.TrxErrorInfo inherit obj, implements WoofWare.NUnitTestRunner.TrxErrorInfo System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxErrorInfo inherit obj, implements WoofWare.NUnitTestRunner.TrxErrorInfo System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxErrorInfo..ctor [constructor]: (string option, string option)
|
WoofWare.NUnitTestRunner.TrxErrorInfo..ctor [constructor]: (string option, string option)
|
||||||
|
WoofWare.NUnitTestRunner.TrxErrorInfo.Equals [method]: (WoofWare.NUnitTestRunner.TrxErrorInfo, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxErrorInfo.get_Message [method]: unit -> string option
|
WoofWare.NUnitTestRunner.TrxErrorInfo.get_Message [method]: unit -> string option
|
||||||
WoofWare.NUnitTestRunner.TrxErrorInfo.get_StackTrace [method]: unit -> string option
|
WoofWare.NUnitTestRunner.TrxErrorInfo.get_StackTrace [method]: unit -> string option
|
||||||
WoofWare.NUnitTestRunner.TrxErrorInfo.Message [property]: [read-only] string option
|
WoofWare.NUnitTestRunner.TrxErrorInfo.Message [property]: [read-only] string option
|
||||||
WoofWare.NUnitTestRunner.TrxErrorInfo.StackTrace [property]: [read-only] string option
|
WoofWare.NUnitTestRunner.TrxErrorInfo.StackTrace [property]: [read-only] string option
|
||||||
WoofWare.NUnitTestRunner.TrxExecution inherit obj, implements WoofWare.NUnitTestRunner.TrxExecution System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxExecution inherit obj, implements WoofWare.NUnitTestRunner.TrxExecution System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxExecution..ctor [constructor]: System.Guid
|
WoofWare.NUnitTestRunner.TrxExecution..ctor [constructor]: System.Guid
|
||||||
|
WoofWare.NUnitTestRunner.TrxExecution.Equals [method]: (WoofWare.NUnitTestRunner.TrxExecution, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxExecution.get_Id [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxExecution.get_Id [method]: unit -> System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxExecution.Id [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxExecution.Id [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome inherit obj, implements WoofWare.NUnitTestRunner.TrxOutcome System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
WoofWare.NUnitTestRunner.TrxOutcome inherit obj, implements WoofWare.NUnitTestRunner.TrxOutcome System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
||||||
@@ -337,6 +524,7 @@ WoofWare.NUnitTestRunner.TrxOutcome+Tags.Completed [static field]: int = 0
|
|||||||
WoofWare.NUnitTestRunner.TrxOutcome+Tags.Failed [static field]: int = 2
|
WoofWare.NUnitTestRunner.TrxOutcome+Tags.Failed [static field]: int = 2
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome+Tags.Warning [static field]: int = 1
|
WoofWare.NUnitTestRunner.TrxOutcome+Tags.Warning [static field]: int = 1
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome.Completed [static property]: [read-only] WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxOutcome.Completed [static property]: [read-only] WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
|
WoofWare.NUnitTestRunner.TrxOutcome.Equals [method]: (WoofWare.NUnitTestRunner.TrxOutcome, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome.Failed [static property]: [read-only] WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxOutcome.Failed [static property]: [read-only] WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome.get_Completed [static method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxOutcome.get_Completed [static method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome.get_Failed [static method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxOutcome.get_Failed [static method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
@@ -352,13 +540,17 @@ WoofWare.NUnitTestRunner.TrxOutcome.Parse [static method]: string -> WoofWare.NU
|
|||||||
WoofWare.NUnitTestRunner.TrxOutcome.Tag [property]: [read-only] int
|
WoofWare.NUnitTestRunner.TrxOutcome.Tag [property]: [read-only] int
|
||||||
WoofWare.NUnitTestRunner.TrxOutcome.Warning [static property]: [read-only] WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxOutcome.Warning [static property]: [read-only] WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxOutput inherit obj, implements WoofWare.NUnitTestRunner.TrxOutput System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxOutput inherit obj, implements WoofWare.NUnitTestRunner.TrxOutput System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxOutput..ctor [constructor]: (string option, WoofWare.NUnitTestRunner.TrxErrorInfo option)
|
WoofWare.NUnitTestRunner.TrxOutput..ctor [constructor]: (string option, string option, WoofWare.NUnitTestRunner.TrxErrorInfo option)
|
||||||
|
WoofWare.NUnitTestRunner.TrxOutput.Equals [method]: (WoofWare.NUnitTestRunner.TrxOutput, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxOutput.ErrorInfo [property]: [read-only] WoofWare.NUnitTestRunner.TrxErrorInfo option
|
WoofWare.NUnitTestRunner.TrxOutput.ErrorInfo [property]: [read-only] WoofWare.NUnitTestRunner.TrxErrorInfo option
|
||||||
WoofWare.NUnitTestRunner.TrxOutput.get_ErrorInfo [method]: unit -> WoofWare.NUnitTestRunner.TrxErrorInfo option
|
WoofWare.NUnitTestRunner.TrxOutput.get_ErrorInfo [method]: unit -> WoofWare.NUnitTestRunner.TrxErrorInfo option
|
||||||
|
WoofWare.NUnitTestRunner.TrxOutput.get_StdErr [method]: unit -> string option
|
||||||
WoofWare.NUnitTestRunner.TrxOutput.get_StdOut [method]: unit -> string option
|
WoofWare.NUnitTestRunner.TrxOutput.get_StdOut [method]: unit -> string option
|
||||||
|
WoofWare.NUnitTestRunner.TrxOutput.StdErr [property]: [read-only] string option
|
||||||
WoofWare.NUnitTestRunner.TrxOutput.StdOut [property]: [read-only] string option
|
WoofWare.NUnitTestRunner.TrxOutput.StdOut [property]: [read-only] string option
|
||||||
WoofWare.NUnitTestRunner.TrxReport inherit obj, implements WoofWare.NUnitTestRunner.TrxReport System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxReport inherit obj, implements WoofWare.NUnitTestRunner.TrxReport System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxReport..ctor [constructor]: (System.Guid, string, WoofWare.NUnitTestRunner.TrxReportTimes, WoofWare.NUnitTestRunner.TrxTestSettings, WoofWare.NUnitTestRunner.TrxUnitTestResult list, WoofWare.NUnitTestRunner.TrxUnitTest list, WoofWare.NUnitTestRunner.TrxTestEntry list, WoofWare.NUnitTestRunner.TrxTestListEntry list, WoofWare.NUnitTestRunner.TrxResultsSummary)
|
WoofWare.NUnitTestRunner.TrxReport..ctor [constructor]: (System.Guid, string, WoofWare.NUnitTestRunner.TrxReportTimes, WoofWare.NUnitTestRunner.TrxTestSettings, WoofWare.NUnitTestRunner.TrxUnitTestResult list, WoofWare.NUnitTestRunner.TrxUnitTest list, WoofWare.NUnitTestRunner.TrxTestEntry list, WoofWare.NUnitTestRunner.TrxTestListEntry list, WoofWare.NUnitTestRunner.TrxResultsSummary)
|
||||||
|
WoofWare.NUnitTestRunner.TrxReport.Equals [method]: (WoofWare.NUnitTestRunner.TrxReport, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxReport.get_Id [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxReport.get_Id [method]: unit -> System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxReport.get_Name [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxReport.get_Name [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.TrxReport.get_Results [method]: unit -> WoofWare.NUnitTestRunner.TrxUnitTestResult list
|
WoofWare.NUnitTestRunner.TrxReport.get_Results [method]: unit -> WoofWare.NUnitTestRunner.TrxUnitTestResult list
|
||||||
@@ -383,6 +575,7 @@ WoofWare.NUnitTestRunner.TrxReportModule.toXml [static method]: WoofWare.NUnitTe
|
|||||||
WoofWare.NUnitTestRunner.TrxReportTimes inherit obj, implements WoofWare.NUnitTestRunner.TrxReportTimes System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxReportTimes inherit obj, implements WoofWare.NUnitTestRunner.TrxReportTimes System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxReportTimes..ctor [constructor]: (System.DateTimeOffset, System.DateTimeOffset, System.DateTimeOffset, System.DateTimeOffset)
|
WoofWare.NUnitTestRunner.TrxReportTimes..ctor [constructor]: (System.DateTimeOffset, System.DateTimeOffset, System.DateTimeOffset, System.DateTimeOffset)
|
||||||
WoofWare.NUnitTestRunner.TrxReportTimes.Creation [property]: [read-only] System.DateTimeOffset
|
WoofWare.NUnitTestRunner.TrxReportTimes.Creation [property]: [read-only] System.DateTimeOffset
|
||||||
|
WoofWare.NUnitTestRunner.TrxReportTimes.Equals [method]: (WoofWare.NUnitTestRunner.TrxReportTimes, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxReportTimes.Finish [property]: [read-only] System.DateTimeOffset
|
WoofWare.NUnitTestRunner.TrxReportTimes.Finish [property]: [read-only] System.DateTimeOffset
|
||||||
WoofWare.NUnitTestRunner.TrxReportTimes.get_Creation [method]: unit -> System.DateTimeOffset
|
WoofWare.NUnitTestRunner.TrxReportTimes.get_Creation [method]: unit -> System.DateTimeOffset
|
||||||
WoofWare.NUnitTestRunner.TrxReportTimes.get_Finish [method]: unit -> System.DateTimeOffset
|
WoofWare.NUnitTestRunner.TrxReportTimes.get_Finish [method]: unit -> System.DateTimeOffset
|
||||||
@@ -393,6 +586,7 @@ WoofWare.NUnitTestRunner.TrxReportTimes.Start [property]: [read-only] System.Dat
|
|||||||
WoofWare.NUnitTestRunner.TrxResultsSummary inherit obj, implements WoofWare.NUnitTestRunner.TrxResultsSummary System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxResultsSummary inherit obj, implements WoofWare.NUnitTestRunner.TrxResultsSummary System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxResultsSummary..ctor [constructor]: (WoofWare.NUnitTestRunner.TrxOutcome, WoofWare.NUnitTestRunner.TrxCounters, WoofWare.NUnitTestRunner.TrxOutput, WoofWare.NUnitTestRunner.TrxRunInfo list)
|
WoofWare.NUnitTestRunner.TrxResultsSummary..ctor [constructor]: (WoofWare.NUnitTestRunner.TrxOutcome, WoofWare.NUnitTestRunner.TrxCounters, WoofWare.NUnitTestRunner.TrxOutput, WoofWare.NUnitTestRunner.TrxRunInfo list)
|
||||||
WoofWare.NUnitTestRunner.TrxResultsSummary.Counters [property]: [read-only] WoofWare.NUnitTestRunner.TrxCounters
|
WoofWare.NUnitTestRunner.TrxResultsSummary.Counters [property]: [read-only] WoofWare.NUnitTestRunner.TrxCounters
|
||||||
|
WoofWare.NUnitTestRunner.TrxResultsSummary.Equals [method]: (WoofWare.NUnitTestRunner.TrxResultsSummary, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxResultsSummary.get_Counters [method]: unit -> WoofWare.NUnitTestRunner.TrxCounters
|
WoofWare.NUnitTestRunner.TrxResultsSummary.get_Counters [method]: unit -> WoofWare.NUnitTestRunner.TrxCounters
|
||||||
WoofWare.NUnitTestRunner.TrxResultsSummary.get_Outcome [method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxResultsSummary.get_Outcome [method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxResultsSummary.get_Output [method]: unit -> WoofWare.NUnitTestRunner.TrxOutput
|
WoofWare.NUnitTestRunner.TrxResultsSummary.get_Output [method]: unit -> WoofWare.NUnitTestRunner.TrxOutput
|
||||||
@@ -403,6 +597,7 @@ WoofWare.NUnitTestRunner.TrxResultsSummary.RunInfos [property]: [read-only] Woof
|
|||||||
WoofWare.NUnitTestRunner.TrxRunInfo inherit obj, implements WoofWare.NUnitTestRunner.TrxRunInfo System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxRunInfo inherit obj, implements WoofWare.NUnitTestRunner.TrxRunInfo System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxRunInfo..ctor [constructor]: (string, WoofWare.NUnitTestRunner.TrxOutcome, System.DateTimeOffset, string)
|
WoofWare.NUnitTestRunner.TrxRunInfo..ctor [constructor]: (string, WoofWare.NUnitTestRunner.TrxOutcome, System.DateTimeOffset, string)
|
||||||
WoofWare.NUnitTestRunner.TrxRunInfo.ComputerName [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxRunInfo.ComputerName [property]: [read-only] string
|
||||||
|
WoofWare.NUnitTestRunner.TrxRunInfo.Equals [method]: (WoofWare.NUnitTestRunner.TrxRunInfo, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxRunInfo.get_ComputerName [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxRunInfo.get_ComputerName [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.TrxRunInfo.get_Outcome [method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
WoofWare.NUnitTestRunner.TrxRunInfo.get_Outcome [method]: unit -> WoofWare.NUnitTestRunner.TrxOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxRunInfo.get_Text [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxRunInfo.get_Text [method]: unit -> string
|
||||||
@@ -412,6 +607,7 @@ WoofWare.NUnitTestRunner.TrxRunInfo.Text [property]: [read-only] string
|
|||||||
WoofWare.NUnitTestRunner.TrxRunInfo.Timestamp [property]: [read-only] System.DateTimeOffset
|
WoofWare.NUnitTestRunner.TrxRunInfo.Timestamp [property]: [read-only] System.DateTimeOffset
|
||||||
WoofWare.NUnitTestRunner.TrxTestEntry inherit obj, implements WoofWare.NUnitTestRunner.TrxTestEntry System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxTestEntry inherit obj, implements WoofWare.NUnitTestRunner.TrxTestEntry System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxTestEntry..ctor [constructor]: (System.Guid, System.Guid, System.Guid)
|
WoofWare.NUnitTestRunner.TrxTestEntry..ctor [constructor]: (System.Guid, System.Guid, System.Guid)
|
||||||
|
WoofWare.NUnitTestRunner.TrxTestEntry.Equals [method]: (WoofWare.NUnitTestRunner.TrxTestEntry, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxTestEntry.ExecutionId [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxTestEntry.ExecutionId [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxTestEntry.get_ExecutionId [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxTestEntry.get_ExecutionId [method]: unit -> System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxTestEntry.get_TestId [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxTestEntry.get_TestId [method]: unit -> System.Guid
|
||||||
@@ -420,6 +616,7 @@ WoofWare.NUnitTestRunner.TrxTestEntry.TestId [property]: [read-only] System.Guid
|
|||||||
WoofWare.NUnitTestRunner.TrxTestEntry.TestListId [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxTestEntry.TestListId [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxTestListEntry inherit obj, implements WoofWare.NUnitTestRunner.TrxTestListEntry System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxTestListEntry inherit obj, implements WoofWare.NUnitTestRunner.TrxTestListEntry System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxTestListEntry..ctor [constructor]: (string, System.Guid)
|
WoofWare.NUnitTestRunner.TrxTestListEntry..ctor [constructor]: (string, System.Guid)
|
||||||
|
WoofWare.NUnitTestRunner.TrxTestListEntry.Equals [method]: (WoofWare.NUnitTestRunner.TrxTestListEntry, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxTestListEntry.get_Id [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxTestListEntry.get_Id [method]: unit -> System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxTestListEntry.get_Name [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxTestListEntry.get_Name [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.TrxTestListEntry.Id [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxTestListEntry.Id [property]: [read-only] System.Guid
|
||||||
@@ -429,6 +626,7 @@ WoofWare.NUnitTestRunner.TrxTestMethod..ctor [constructor]: (string, System.Uri,
|
|||||||
WoofWare.NUnitTestRunner.TrxTestMethod.AdapterTypeName [property]: [read-only] System.Uri
|
WoofWare.NUnitTestRunner.TrxTestMethod.AdapterTypeName [property]: [read-only] System.Uri
|
||||||
WoofWare.NUnitTestRunner.TrxTestMethod.ClassName [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxTestMethod.ClassName [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.TrxTestMethod.CodeBase [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxTestMethod.CodeBase [property]: [read-only] string
|
||||||
|
WoofWare.NUnitTestRunner.TrxTestMethod.Equals [method]: (WoofWare.NUnitTestRunner.TrxTestMethod, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxTestMethod.get_AdapterTypeName [method]: unit -> System.Uri
|
WoofWare.NUnitTestRunner.TrxTestMethod.get_AdapterTypeName [method]: unit -> System.Uri
|
||||||
WoofWare.NUnitTestRunner.TrxTestMethod.get_ClassName [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxTestMethod.get_ClassName [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.TrxTestMethod.get_CodeBase [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxTestMethod.get_CodeBase [method]: unit -> string
|
||||||
@@ -440,6 +638,7 @@ WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.Failed [static field]: int = 1
|
|||||||
WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.Inconclusive [static field]: int = 3
|
WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.Inconclusive [static field]: int = 3
|
||||||
WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.NotExecuted [static field]: int = 2
|
WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.NotExecuted [static field]: int = 2
|
||||||
WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.Passed [static field]: int = 0
|
WoofWare.NUnitTestRunner.TrxTestOutcome+Tags.Passed [static field]: int = 0
|
||||||
|
WoofWare.NUnitTestRunner.TrxTestOutcome.Equals [method]: (WoofWare.NUnitTestRunner.TrxTestOutcome, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxTestOutcome.Failed [static property]: [read-only] WoofWare.NUnitTestRunner.TrxTestOutcome
|
WoofWare.NUnitTestRunner.TrxTestOutcome.Failed [static property]: [read-only] WoofWare.NUnitTestRunner.TrxTestOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxTestOutcome.get_Failed [static method]: unit -> WoofWare.NUnitTestRunner.TrxTestOutcome
|
WoofWare.NUnitTestRunner.TrxTestOutcome.get_Failed [static method]: unit -> WoofWare.NUnitTestRunner.TrxTestOutcome
|
||||||
WoofWare.NUnitTestRunner.TrxTestOutcome.get_Inconclusive [static method]: unit -> WoofWare.NUnitTestRunner.TrxTestOutcome
|
WoofWare.NUnitTestRunner.TrxTestOutcome.get_Inconclusive [static method]: unit -> WoofWare.NUnitTestRunner.TrxTestOutcome
|
||||||
@@ -462,6 +661,7 @@ WoofWare.NUnitTestRunner.TrxTestOutcome.Tag [property]: [read-only] int
|
|||||||
WoofWare.NUnitTestRunner.TrxTestSettings inherit obj, implements WoofWare.NUnitTestRunner.TrxTestSettings System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxTestSettings inherit obj, implements WoofWare.NUnitTestRunner.TrxTestSettings System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxTestSettings..ctor [constructor]: (string, System.Guid, WoofWare.NUnitTestRunner.TrxDeployment)
|
WoofWare.NUnitTestRunner.TrxTestSettings..ctor [constructor]: (string, System.Guid, WoofWare.NUnitTestRunner.TrxDeployment)
|
||||||
WoofWare.NUnitTestRunner.TrxTestSettings.Deployment [property]: [read-only] WoofWare.NUnitTestRunner.TrxDeployment
|
WoofWare.NUnitTestRunner.TrxTestSettings.Deployment [property]: [read-only] WoofWare.NUnitTestRunner.TrxDeployment
|
||||||
|
WoofWare.NUnitTestRunner.TrxTestSettings.Equals [method]: (WoofWare.NUnitTestRunner.TrxTestSettings, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxTestSettings.get_Deployment [method]: unit -> WoofWare.NUnitTestRunner.TrxDeployment
|
WoofWare.NUnitTestRunner.TrxTestSettings.get_Deployment [method]: unit -> WoofWare.NUnitTestRunner.TrxDeployment
|
||||||
WoofWare.NUnitTestRunner.TrxTestSettings.get_Id [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxTestSettings.get_Id [method]: unit -> System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxTestSettings.get_Name [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxTestSettings.get_Name [method]: unit -> string
|
||||||
@@ -469,6 +669,7 @@ WoofWare.NUnitTestRunner.TrxTestSettings.Id [property]: [read-only] System.Guid
|
|||||||
WoofWare.NUnitTestRunner.TrxTestSettings.Name [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxTestSettings.Name [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTest inherit obj, implements WoofWare.NUnitTestRunner.TrxUnitTest System.IEquatable, System.Collections.IStructuralEquatable
|
WoofWare.NUnitTestRunner.TrxUnitTest inherit obj, implements WoofWare.NUnitTestRunner.TrxUnitTest System.IEquatable, System.Collections.IStructuralEquatable
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTest..ctor [constructor]: (string, string, System.Guid, WoofWare.NUnitTestRunner.TrxExecution, WoofWare.NUnitTestRunner.TrxTestMethod)
|
WoofWare.NUnitTestRunner.TrxUnitTest..ctor [constructor]: (string, string, System.Guid, WoofWare.NUnitTestRunner.TrxExecution, WoofWare.NUnitTestRunner.TrxTestMethod)
|
||||||
|
WoofWare.NUnitTestRunner.TrxUnitTest.Equals [method]: (WoofWare.NUnitTestRunner.TrxUnitTest, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTest.Execution [property]: [read-only] WoofWare.NUnitTestRunner.TrxExecution
|
WoofWare.NUnitTestRunner.TrxUnitTest.Execution [property]: [read-only] WoofWare.NUnitTestRunner.TrxExecution
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTest.get_Execution [method]: unit -> WoofWare.NUnitTestRunner.TrxExecution
|
WoofWare.NUnitTestRunner.TrxUnitTest.get_Execution [method]: unit -> WoofWare.NUnitTestRunner.TrxExecution
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTest.get_Id [method]: unit -> System.Guid
|
WoofWare.NUnitTestRunner.TrxUnitTest.get_Id [method]: unit -> System.Guid
|
||||||
@@ -484,6 +685,7 @@ WoofWare.NUnitTestRunner.TrxUnitTestResult..ctor [constructor]: (System.Guid, Sy
|
|||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.ComputerName [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.ComputerName [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.Duration [property]: [read-only] System.TimeSpan
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.Duration [property]: [read-only] System.TimeSpan
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.EndTime [property]: [read-only] System.DateTimeOffset
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.EndTime [property]: [read-only] System.DateTimeOffset
|
||||||
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.Equals [method]: (WoofWare.NUnitTestRunner.TrxUnitTestResult, System.Collections.IEqualityComparer) -> bool
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.ExecutionId [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.ExecutionId [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.get_ComputerName [method]: unit -> string
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.get_ComputerName [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.get_Duration [method]: unit -> System.TimeSpan
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.get_Duration [method]: unit -> System.TimeSpan
|
||||||
@@ -505,13 +707,21 @@ WoofWare.NUnitTestRunner.TrxUnitTestResult.TestId [property]: [read-only] System
|
|||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.TestListId [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.TestListId [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.TestName [property]: [read-only] string
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.TestName [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.TrxUnitTestResult.TestType [property]: [read-only] System.Guid
|
WoofWare.NUnitTestRunner.TrxUnitTestResult.TestType [property]: [read-only] System.Guid
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure inherit obj, implements WoofWare.NUnitTestRunner.UserMethodFailure System.IEquatable, System.Collections.IStructuralEquatable - union type with 2 cases
|
WoofWare.NUnitTestRunner.UserMethodFailure inherit obj, implements WoofWare.NUnitTestRunner.UserMethodFailure System.IEquatable, System.Collections.IStructuralEquatable - union type with 3 cases
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters inherit WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters.actual [property]: [read-only] obj []
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters.expected [property]: [read-only] System.Type []
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters.get_actual [method]: unit -> obj []
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters.get_expected [method]: unit -> System.Type []
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters.get_name [method]: unit -> string
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+BadParameters.name [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit inherit WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit inherit WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.get_name [method]: unit -> string
|
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.get_name [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.get_result [method]: unit -> obj
|
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.get_result [method]: unit -> obj
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.name [property]: [read-only] string
|
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.name [property]: [read-only] string
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.result [property]: [read-only] obj
|
WoofWare.NUnitTestRunner.UserMethodFailure+ReturnedNonUnit.result [property]: [read-only] obj
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Tags inherit obj
|
WoofWare.NUnitTestRunner.UserMethodFailure+Tags inherit obj
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure+Tags.BadParameters [static field]: int = 2
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Tags.ReturnedNonUnit [static field]: int = 0
|
WoofWare.NUnitTestRunner.UserMethodFailure+Tags.ReturnedNonUnit [static field]: int = 0
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Tags.Threw [static field]: int = 1
|
WoofWare.NUnitTestRunner.UserMethodFailure+Tags.Threw [static field]: int = 1
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Threw inherit WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.UserMethodFailure+Threw inherit WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
@@ -519,13 +729,17 @@ WoofWare.NUnitTestRunner.UserMethodFailure+Threw.get_Item2 [method]: unit -> Sys
|
|||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Threw.get_name [method]: unit -> string
|
WoofWare.NUnitTestRunner.UserMethodFailure+Threw.get_name [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Threw.Item2 [property]: [read-only] System.Exception
|
WoofWare.NUnitTestRunner.UserMethodFailure+Threw.Item2 [property]: [read-only] System.Exception
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure+Threw.name [property]: [read-only] string
|
WoofWare.NUnitTestRunner.UserMethodFailure+Threw.name [property]: [read-only] string
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure.Equals [method]: (WoofWare.NUnitTestRunner.UserMethodFailure, System.Collections.IEqualityComparer) -> bool
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure.get_IsBadParameters [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.get_IsReturnedNonUnit [method]: unit -> bool
|
WoofWare.NUnitTestRunner.UserMethodFailure.get_IsReturnedNonUnit [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.get_IsThrew [method]: unit -> bool
|
WoofWare.NUnitTestRunner.UserMethodFailure.get_IsThrew [method]: unit -> bool
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.get_Name [method]: unit -> string
|
WoofWare.NUnitTestRunner.UserMethodFailure.get_Name [method]: unit -> string
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.get_Tag [method]: unit -> int
|
WoofWare.NUnitTestRunner.UserMethodFailure.get_Tag [method]: unit -> int
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure.IsBadParameters [property]: [read-only] bool
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.IsReturnedNonUnit [property]: [read-only] bool
|
WoofWare.NUnitTestRunner.UserMethodFailure.IsReturnedNonUnit [property]: [read-only] bool
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.IsThrew [property]: [read-only] bool
|
WoofWare.NUnitTestRunner.UserMethodFailure.IsThrew [property]: [read-only] bool
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.Name [property]: [read-only] string
|
WoofWare.NUnitTestRunner.UserMethodFailure.Name [property]: [read-only] string
|
||||||
|
WoofWare.NUnitTestRunner.UserMethodFailure.NewBadParameters [static method]: (string, System.Type [], obj []) -> WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.NewReturnedNonUnit [static method]: (string, obj) -> WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.UserMethodFailure.NewReturnedNonUnit [static method]: (string, obj) -> WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.NewThrew [static method]: (string, System.Exception) -> WoofWare.NUnitTestRunner.UserMethodFailure
|
WoofWare.NUnitTestRunner.UserMethodFailure.NewThrew [static method]: (string, System.Exception) -> WoofWare.NUnitTestRunner.UserMethodFailure
|
||||||
WoofWare.NUnitTestRunner.UserMethodFailure.Tag [property]: [read-only] int
|
WoofWare.NUnitTestRunner.UserMethodFailure.Tag [property]: [read-only] int
|
@@ -1,25 +1,14 @@
|
|||||||
namespace WoofWare.NUnitTestRunner
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
open System
|
open System
|
||||||
|
open System.Collections
|
||||||
open System.Diagnostics
|
open System.Diagnostics
|
||||||
open System.IO
|
open System.IO
|
||||||
open System.Reflection
|
open System.Reflection
|
||||||
open System.Threading
|
open System.Threading
|
||||||
|
open System.Threading.Tasks
|
||||||
open Microsoft.FSharp.Core
|
open Microsoft.FSharp.Core
|
||||||
|
|
||||||
type private StdoutSetter (newStdout : StreamWriter, newStderr : StreamWriter) =
|
|
||||||
let oldStdout = Console.Out
|
|
||||||
let oldStderr = Console.Error
|
|
||||||
|
|
||||||
do
|
|
||||||
Console.SetOut newStdout
|
|
||||||
Console.SetError newStderr
|
|
||||||
|
|
||||||
interface IDisposable with
|
|
||||||
member _.Dispose () =
|
|
||||||
Console.SetOut oldStdout
|
|
||||||
Console.SetError oldStderr
|
|
||||||
|
|
||||||
/// Information about the circumstances of a run of a single test.
|
/// Information about the circumstances of a run of a single test.
|
||||||
type IndividualTestRunMetadata =
|
type IndividualTestRunMetadata =
|
||||||
{
|
{
|
||||||
@@ -79,111 +68,176 @@ module TestFixture =
|
|||||||
///
|
///
|
||||||
/// This function does not throw.
|
/// This function does not throw.
|
||||||
let private runOne
|
let private runOne
|
||||||
|
(outputId : OutputStreamId)
|
||||||
|
(contexts : TestContexts)
|
||||||
(setUp : MethodInfo list)
|
(setUp : MethodInfo list)
|
||||||
(tearDown : MethodInfo list)
|
(tearDown : MethodInfo list)
|
||||||
(testId : Guid)
|
(testId : Guid)
|
||||||
(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 :? TargetInvocationException as e ->
|
head.Invoke (containingObject, args) |> Ok
|
||||||
Error (UserMethodFailure.Threw (head.Name, e.InnerException))
|
with
|
||||||
|
| :? TargetInvocationException as e ->
|
||||||
|
Error (UserMethodFailure.Threw (head.Name, e.InnerException))
|
||||||
|
| :? TargetParameterCountException ->
|
||||||
|
UserMethodFailure.BadParameters (
|
||||||
|
head.Name,
|
||||||
|
head.GetParameters () |> Array.map (fun pm -> pm.ParameterType),
|
||||||
|
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
|
||||||
|
|
||||||
use stdOutStream = new MemoryStream ()
|
try
|
||||||
use stdErrStream = new MemoryStream ()
|
do! Async.AwaitTask result
|
||||||
use stdOut = new StreamWriter (stdOutStream)
|
with e ->
|
||||||
use stdErr = new StreamWriter (stdErrStream)
|
exc <- Some e
|
||||||
|
|
||||||
use _ = new StdoutSetter (stdOut, stdErr)
|
match exc with
|
||||||
|
| None -> return! runMethods wrap rest args
|
||||||
|
| Some e -> return Error (UserMethodFailure.Threw (head.Name, e) |> wrap)
|
||||||
|
}
|
||||||
|
// We'd like to do this type-test:
|
||||||
|
// | :? Async<unit> as result ->
|
||||||
|
// 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 ()
|
||||||
|
|
||||||
let sw = Stopwatch.StartNew ()
|
if ty.Namespace = "Microsoft.FSharp.Control" && ty.Name = "FSharpAsync`1" then
|
||||||
|
match ty.GenericTypeArguments |> Array.map (fun t -> t.FullName) with
|
||||||
|
| [| "Microsoft.FSharp.Core.Unit" |] ->
|
||||||
|
let asyncModule = ty.Assembly.GetType ("Microsoft.FSharp.Control.FSharpAsync")
|
||||||
|
// let catch = asyncModule.GetMethod("Catch").MakeGenericMethod [| ty.GenericTypeArguments.[0] |]
|
||||||
|
// let caught = catch.Invoke ((null: obj), [| ret |])
|
||||||
|
let startAsTask =
|
||||||
|
asyncModule.GetMethod("StartAsTask").MakeGenericMethod
|
||||||
|
[| ty.GenericTypeArguments.[0] |]
|
||||||
|
|
||||||
let metadata () =
|
let started =
|
||||||
let name =
|
startAsTask.Invoke ((null : obj), [| ret ; (null : obj) ; (null : obj) |])
|
||||||
if args.Length = 0 then
|
|> unbox<Task>
|
||||||
test.Name
|
|
||||||
else
|
|
||||||
let argsStr = args |> Seq.map string<obj> |> String.concat ","
|
|
||||||
$"%s{test.Name}(%s{argsStr})"
|
|
||||||
|
|
||||||
{
|
async {
|
||||||
End = DateTimeOffset.Now
|
let! res = Async.AwaitTask started |> Async.Catch
|
||||||
Start = start
|
|
||||||
Total = sw.Elapsed
|
|
||||||
ComputerName = Environment.MachineName
|
|
||||||
ExecutionId = Guid.NewGuid ()
|
|
||||||
TestId = testId
|
|
||||||
TestName = name
|
|
||||||
ClassName = test.DeclaringType.FullName
|
|
||||||
StdOut =
|
|
||||||
match stdOutStream.ToArray () with
|
|
||||||
| [||] -> None
|
|
||||||
| arr -> Console.OutputEncoding.GetString arr |> Some
|
|
||||||
StdErr =
|
|
||||||
match stdErrStream.ToArray () with
|
|
||||||
| [||] -> None
|
|
||||||
| arr -> Console.OutputEncoding.GetString arr |> Some
|
|
||||||
}
|
|
||||||
|
|
||||||
let setUpResult = runMethods TestFailure.SetUpFailed setUp [||]
|
match res with
|
||||||
sw.Stop ()
|
| 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)
|
||||||
|
|
||||||
match setUpResult with
|
return result
|
||||||
| Error e -> Error [ e ], metadata ()
|
}
|
||||||
| Ok () ->
|
|
||||||
|
|
||||||
sw.Start ()
|
async {
|
||||||
|
let start = DateTimeOffset.Now
|
||||||
|
|
||||||
let result =
|
let sw = Stopwatch.StartNew ()
|
||||||
let result = runMethods TestFailure.TestFailed [ test ] args
|
|
||||||
|
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 =
|
||||||
@@ -222,11 +276,15 @@ module TestFixture =
|
|||||||
/// This method should never throw: it only throws if there's a critical logic error in the runner.
|
/// This method should never throw: it only throws if there's a critical logic error in the runner.
|
||||||
/// Exceptions from the units under test are wrapped up and passed out.
|
/// Exceptions from the units under test are wrapped up and passed out.
|
||||||
let private runTestsFromMember
|
let private runTestsFromMember
|
||||||
|
(contexts : TestContexts)
|
||||||
|
(par : ParallelQueue)
|
||||||
|
(running : TestFixtureSetupToken)
|
||||||
|
(progress : ITestProgress)
|
||||||
(setUp : MethodInfo list)
|
(setUp : MethodInfo list)
|
||||||
(tearDown : MethodInfo list)
|
(tearDown : MethodInfo list)
|
||||||
(containingObject : obj)
|
(containingObject : obj)
|
||||||
(test : SingleTestMethod)
|
(test : SingleTestMethod)
|
||||||
: (Result<TestMemberSuccess, TestMemberFailure> * IndividualTestRunMetadata) list
|
: (Result<TestMemberSuccess, TestMemberFailure> * IndividualTestRunMetadata) Task list
|
||||||
=
|
=
|
||||||
if test.Method.ContainsGenericParameters then
|
if test.Method.ContainsGenericParameters then
|
||||||
let failureMetadata =
|
let failureMetadata =
|
||||||
@@ -246,7 +304,7 @@ module TestFixture =
|
|||||||
let error =
|
let error =
|
||||||
TestMemberFailure.Malformed [ "Test contained generic parameters; generics are not supported." ]
|
TestMemberFailure.Malformed [ "Test contained generic parameters; generics are not supported." ]
|
||||||
|
|
||||||
(Error error, failureMetadata) |> List.singleton
|
(Error error, failureMetadata) |> Task.FromResult |> List.singleton
|
||||||
else
|
else
|
||||||
|
|
||||||
let resultPreRun =
|
let resultPreRun =
|
||||||
@@ -262,17 +320,12 @@ module TestFixture =
|
|||||||
| Modifier.Ignored reason -> Some (TestMemberSuccess.Ignored reason)
|
| Modifier.Ignored reason -> Some (TestMemberSuccess.Ignored reason)
|
||||||
)
|
)
|
||||||
|
|
||||||
let sw = Stopwatch.StartNew ()
|
|
||||||
let startTime = DateTimeOffset.Now
|
|
||||||
|
|
||||||
match resultPreRun with
|
match resultPreRun with
|
||||||
| Some result ->
|
| Some result ->
|
||||||
sw.Stop ()
|
|
||||||
|
|
||||||
let failureMetadata =
|
let failureMetadata =
|
||||||
{
|
{
|
||||||
Total = sw.Elapsed
|
Total = TimeSpan.Zero
|
||||||
Start = startTime
|
Start = DateTimeOffset.Now
|
||||||
End = DateTimeOffset.Now
|
End = DateTimeOffset.Now
|
||||||
ComputerName = Environment.MachineName
|
ComputerName = Environment.MachineName
|
||||||
ExecutionId = Guid.NewGuid ()
|
ExecutionId = Guid.NewGuid ()
|
||||||
@@ -284,7 +337,7 @@ module TestFixture =
|
|||||||
StdOut = None
|
StdOut = None
|
||||||
}
|
}
|
||||||
|
|
||||||
[ Ok result, failureMetadata ]
|
(Ok result, failureMetadata) |> Task.FromResult |> List.singleton
|
||||||
| None ->
|
| None ->
|
||||||
|
|
||||||
let individualTests =
|
let individualTests =
|
||||||
@@ -347,7 +400,7 @@ module TestFixture =
|
|||||||
|
|
||||||
// Might not be an IEnumerable of a reference type.
|
// Might not be an IEnumerable of a reference type.
|
||||||
// Concretely, `FSharpList<HttpStatusCode> :> IEnumerable<obj>` fails.
|
// Concretely, `FSharpList<HttpStatusCode> :> IEnumerable<obj>` fails.
|
||||||
for arg in args.GetValue (null : obj) :?> System.Collections.IEnumerable do
|
for arg in args.GetValue (null : obj) :?> IEnumerable do
|
||||||
yield
|
yield
|
||||||
Guid.NewGuid (),
|
Guid.NewGuid (),
|
||||||
match arg with
|
match arg with
|
||||||
@@ -370,20 +423,19 @@ module TestFixture =
|
|||||||
if isNull argsMem then
|
if isNull argsMem then
|
||||||
failwith "Unexpectedly could not call `.Arguments` on TestCaseData"
|
failwith "Unexpectedly could not call `.Arguments` on TestCaseData"
|
||||||
|
|
||||||
|
// TODO: need to capture this stdout/stderr
|
||||||
(argsMem.Invoke (arg, [||]) |> unbox<obj[]>)
|
(argsMem.Invoke (arg, [||]) |> unbox<obj[]>)
|
||||||
else
|
else
|
||||||
[| arg |]
|
[| arg |]
|
||||||
]
|
]
|
||||||
|> Ok
|
|> Ok
|
||||||
|
|
||||||
sw.Stop ()
|
|
||||||
|
|
||||||
match individualTests with
|
match individualTests with
|
||||||
| Error e ->
|
| Error e ->
|
||||||
let failureMetadata =
|
let failureMetadata =
|
||||||
{
|
{
|
||||||
Total = sw.Elapsed
|
Total = TimeSpan.Zero
|
||||||
Start = startTime
|
Start = DateTimeOffset.Now
|
||||||
End = DateTimeOffset.Now
|
End = DateTimeOffset.Now
|
||||||
ComputerName = Environment.MachineName
|
ComputerName = Environment.MachineName
|
||||||
ExecutionId = Guid.NewGuid ()
|
ExecutionId = Guid.NewGuid ()
|
||||||
@@ -396,7 +448,7 @@ module TestFixture =
|
|||||||
StdOut = None
|
StdOut = None
|
||||||
}
|
}
|
||||||
|
|
||||||
[ Error e, failureMetadata ]
|
(Error e, failureMetadata) |> Task.FromResult |> List.singleton
|
||||||
| Ok individualTests ->
|
| Ok individualTests ->
|
||||||
|
|
||||||
let count = test.Repeat |> Option.defaultValue 1
|
let count = test.Repeat |> Option.defaultValue 1
|
||||||
@@ -404,149 +456,274 @@ module TestFixture =
|
|||||||
Seq.init count (fun _ -> individualTests)
|
Seq.init count (fun _ -> individualTests)
|
||||||
|> Seq.concat
|
|> Seq.concat
|
||||||
|> Seq.map (fun (testGuid, args) ->
|
|> Seq.map (fun (testGuid, args) ->
|
||||||
let results, summary =
|
task {
|
||||||
runOne setUp tearDown testGuid test.Method containingObject args
|
let runMe () =
|
||||||
|
async {
|
||||||
|
progress.OnTestMemberStart test.Name
|
||||||
|
let oldValue = contexts.AsyncLocal.Value
|
||||||
|
let outputId = contexts.NewOutputs ()
|
||||||
|
contexts.AsyncLocal.Value <- outputId
|
||||||
|
|
||||||
match results with
|
let! result, meta =
|
||||||
| Ok results -> Ok results, summary
|
runOne outputId contexts setUp tearDown testGuid test.Method containingObject args
|
||||||
| Error e -> Error (TestMemberFailure.Failed e), summary
|
|
||||||
|
contexts.AsyncLocal.Value <- oldValue
|
||||||
|
progress.OnTestMemberFinished test.Name
|
||||||
|
|
||||||
|
return result, meta
|
||||||
|
}
|
||||||
|
|
||||||
|
let! results, summary = par.RunAsync running test.Parallelize runMe
|
||||||
|
|
||||||
|
match results with
|
||||||
|
| Ok results -> return Ok results, summary
|
||||||
|
| Error e -> return Error (TestMemberFailure.Failed e), summary
|
||||||
|
}
|
||||||
)
|
)
|
||||||
|> Seq.toList
|
|> Seq.toList
|
||||||
|
|
||||||
/// 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.
|
||||||
let run
|
let runOneFixture
|
||||||
|
(contexts : TestContexts)
|
||||||
|
(par : ParallelQueue)
|
||||||
(progress : ITestProgress)
|
(progress : ITestProgress)
|
||||||
(filter : TestFixture -> SingleTestMethod -> bool)
|
(filter : TestFixture -> SingleTestMethod -> bool)
|
||||||
|
(name : string)
|
||||||
|
(containingObject : obj)
|
||||||
(tests : TestFixture)
|
(tests : TestFixture)
|
||||||
: FixtureRunResults
|
: FixtureRunResults Task
|
||||||
=
|
=
|
||||||
progress.OnTestFixtureStart tests.Name tests.Tests.Length
|
task {
|
||||||
|
let! running = par.StartTestFixture tests
|
||||||
|
progress.OnTestFixtureStart name tests.Tests.Length
|
||||||
|
|
||||||
let containingObject =
|
let oldWorkDir = Environment.CurrentDirectory
|
||||||
let methods =
|
Environment.CurrentDirectory <- FileInfo(tests.ContainingAssembly.Location).Directory.FullName
|
||||||
seq {
|
|
||||||
match tests.OneTimeSetUp with
|
|
||||||
| None -> ()
|
|
||||||
| Some t -> yield t
|
|
||||||
|
|
||||||
match tests.OneTimeTearDown with
|
let sw = Stopwatch.StartNew ()
|
||||||
| None -> ()
|
let startTime = DateTimeOffset.UtcNow
|
||||||
| Some t -> yield t
|
|
||||||
|
|
||||||
yield! tests.Tests |> Seq.map (fun t -> t.Method)
|
let endMetadata (outputId : OutputStreamId) =
|
||||||
|
let stdOut = contexts.DumpStdout outputId
|
||||||
|
let stdErr = contexts.DumpStderr outputId
|
||||||
|
|
||||||
|
{
|
||||||
|
Total = sw.Elapsed
|
||||||
|
Start = startTime
|
||||||
|
End = DateTimeOffset.UtcNow
|
||||||
|
ComputerName = Environment.MachineName
|
||||||
|
ExecutionId = Guid.NewGuid ()
|
||||||
|
TestId = Guid.NewGuid ()
|
||||||
|
// This one is a bit dubious, because we don't actually have a test name at all
|
||||||
|
TestName = name
|
||||||
|
ClassName = tests.Name
|
||||||
|
StdOut = if String.IsNullOrEmpty stdOut then None else Some stdOut
|
||||||
|
StdErr = if String.IsNullOrEmpty stdErr then None else Some stdErr
|
||||||
}
|
}
|
||||||
|
|
||||||
methods
|
let! setupResult, running =
|
||||||
|> Seq.tryPick (fun mi ->
|
match tests.OneTimeSetUp with
|
||||||
if not mi.IsStatic then
|
| Some su ->
|
||||||
Some (Activator.CreateInstance mi.DeclaringType)
|
par.RunTestSetup
|
||||||
else
|
running
|
||||||
None
|
(fun () ->
|
||||||
)
|
let oldValue = contexts.AsyncLocal.Value
|
||||||
|> Option.toObj
|
let newOutputs = contexts.NewOutputs ()
|
||||||
|
contexts.AsyncLocal.Value <- newOutputs
|
||||||
|
|
||||||
let oldWorkDir = Environment.CurrentDirectory
|
let result =
|
||||||
Environment.CurrentDirectory <- FileInfo(tests.ContainingAssembly.Location).Directory.FullName
|
try
|
||||||
|
match su.Invoke (containingObject, [||]) with
|
||||||
|
| :? unit -> None
|
||||||
|
| ret ->
|
||||||
|
Some (UserMethodFailure.ReturnedNonUnit (su.Name, ret), endMetadata newOutputs)
|
||||||
|
with :? TargetInvocationException as e ->
|
||||||
|
Some (UserMethodFailure.Threw (su.Name, e.InnerException), endMetadata newOutputs)
|
||||||
|
|
||||||
let sw = Stopwatch.StartNew ()
|
contexts.AsyncLocal.Value <- oldValue
|
||||||
let startTime = DateTimeOffset.UtcNow
|
|
||||||
|
|
||||||
use stdOutStream = new MemoryStream ()
|
match result with
|
||||||
use stdOut = new StreamWriter (stdOutStream)
|
| None -> Ok (Some newOutputs)
|
||||||
use stdErrStream = new MemoryStream ()
|
| Some err -> Error (err, newOutputs)
|
||||||
use stdErr = new StreamWriter (stdErrStream)
|
)
|
||||||
use _ = new StdoutSetter (stdOut, stdErr)
|
| _ -> Task.FromResult (Ok None, TestFixtureSetupToken.vouchNoSetupRequired running)
|
||||||
|
|
||||||
let endMetadata () =
|
let testFailures = ResizeArray<TestMemberFailure * IndividualTestRunMetadata> ()
|
||||||
let stdOut = stdOutStream.ToArray () |> Console.OutputEncoding.GetString
|
|
||||||
let stdErr = stdErrStream.ToArray () |> Console.OutputEncoding.GetString
|
|
||||||
|
|
||||||
{
|
let successes =
|
||||||
Total = sw.Elapsed
|
ResizeArray<SingleTestMethod * TestMemberSuccess * IndividualTestRunMetadata> ()
|
||||||
Start = startTime
|
|
||||||
End = DateTimeOffset.UtcNow
|
|
||||||
ComputerName = Environment.MachineName
|
|
||||||
ExecutionId = Guid.NewGuid ()
|
|
||||||
TestId = Guid.NewGuid ()
|
|
||||||
// This one is a bit dubious, because we don't actually have a test name at all
|
|
||||||
TestName = tests.Name
|
|
||||||
ClassName = tests.Name
|
|
||||||
StdOut = if String.IsNullOrEmpty stdOut then None else Some stdOut
|
|
||||||
StdErr = if String.IsNullOrEmpty stdErr then None else Some stdErr
|
|
||||||
}
|
|
||||||
|
|
||||||
let setupResult =
|
let testsRun =
|
||||||
match tests.OneTimeSetUp with
|
match setupResult with
|
||||||
| Some su ->
|
| Error _ ->
|
||||||
try
|
// Don't run any tests if setup failed.
|
||||||
match su.Invoke (containingObject, [||]) with
|
Task.FromResult ()
|
||||||
| :? unit -> None
|
| Ok _ ->
|
||||||
| ret -> Some (UserMethodFailure.ReturnedNonUnit (su.Name, ret), endMetadata ())
|
tests.Tests
|
||||||
with :? TargetInvocationException as e ->
|
|> Seq.filter (fun test ->
|
||||||
Some (UserMethodFailure.Threw (su.Name, e.InnerException), endMetadata ())
|
if filter tests test then
|
||||||
| _ -> None
|
true
|
||||||
|
else
|
||||||
|
progress.OnTestMemberSkipped test.Name
|
||||||
|
false
|
||||||
|
)
|
||||||
|
|> Seq.map (fun test ->
|
||||||
|
task {
|
||||||
|
let testSuccess = ref 0
|
||||||
|
|
||||||
let testFailures = ResizeArray<TestMemberFailure * IndividualTestRunMetadata> ()
|
let results =
|
||||||
|
runTestsFromMember
|
||||||
|
contexts
|
||||||
|
par
|
||||||
|
running
|
||||||
|
progress
|
||||||
|
tests.SetUp
|
||||||
|
tests.TearDown
|
||||||
|
containingObject
|
||||||
|
test
|
||||||
|
|
||||||
let successes =
|
let! result =
|
||||||
ResizeArray<SingleTestMethod * TestMemberSuccess * IndividualTestRunMetadata> ()
|
results
|
||||||
|
|> List.map (fun t ->
|
||||||
|
task {
|
||||||
|
let! result, report = t
|
||||||
|
|
||||||
match setupResult with
|
match result with
|
||||||
| Some _ ->
|
| Error failure ->
|
||||||
// Don't run any tests if setup failed.
|
testFailures.Add (failure, report)
|
||||||
()
|
progress.OnTestFailed test.Name failure
|
||||||
| None ->
|
| Ok result ->
|
||||||
for test in tests.Tests do
|
Interlocked.Increment testSuccess |> ignore<int>
|
||||||
if filter tests test then
|
lock successes (fun () -> successes.Add (test, result, report))
|
||||||
progress.OnTestMemberStart test.Name
|
}
|
||||||
let testSuccess = ref 0
|
)
|
||||||
|
|> Task.WhenAll
|
||||||
|
|
||||||
let results = runTestsFromMember tests.SetUp tests.TearDown containingObject test
|
result |> Array.iter id
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|> Task.WhenAll
|
||||||
|
|> fun t ->
|
||||||
|
task {
|
||||||
|
let! t = t
|
||||||
|
return t |> Array.iter id
|
||||||
|
}
|
||||||
|
|
||||||
for result, report in results do
|
do! testsRun
|
||||||
match result with
|
|
||||||
| Error failure ->
|
|
||||||
testFailures.Add (failure, report)
|
|
||||||
progress.OnTestFailed test.Name failure
|
|
||||||
| Ok result ->
|
|
||||||
Interlocked.Increment testSuccess |> ignore<int>
|
|
||||||
lock successes (fun () -> successes.Add (test, result, report))
|
|
||||||
|
|
||||||
progress.OnTestMemberFinished test.Name
|
// Unconditionally run OneTimeTearDown if it exists.
|
||||||
else
|
let! tearDownError, tornDown =
|
||||||
progress.OnTestMemberSkipped test.Name
|
match tests.OneTimeTearDown with
|
||||||
|
| Some td ->
|
||||||
|
par.RunTestTearDown
|
||||||
|
running
|
||||||
|
(fun () ->
|
||||||
|
let oldValue = contexts.AsyncLocal.Value
|
||||||
|
let outputs = contexts.NewOutputs ()
|
||||||
|
contexts.AsyncLocal.Value <- outputs
|
||||||
|
|
||||||
// Unconditionally run OneTimeTearDown if it exists.
|
let result =
|
||||||
let tearDownError =
|
try
|
||||||
match tests.OneTimeTearDown with
|
match td.Invoke (containingObject, [||]) with
|
||||||
| Some td ->
|
| :? unit -> None
|
||||||
try
|
| ret ->
|
||||||
match td.Invoke (containingObject, [||]) with
|
Some (UserMethodFailure.ReturnedNonUnit (td.Name, ret), endMetadata outputs)
|
||||||
| null -> None
|
with :? TargetInvocationException as e ->
|
||||||
| ret -> Some (UserMethodFailure.ReturnedNonUnit (td.Name, ret), endMetadata ())
|
Some (UserMethodFailure.Threw (td.Name, e.InnerException), endMetadata outputs)
|
||||||
with :? TargetInvocationException as e ->
|
|
||||||
Some (UserMethodFailure.Threw (td.Name, e), endMetadata ())
|
|
||||||
| _ -> None
|
|
||||||
|
|
||||||
Environment.CurrentDirectory <- oldWorkDir
|
contexts.AsyncLocal.Value <- oldValue
|
||||||
|
|
||||||
{
|
match result with
|
||||||
Failed = testFailures |> Seq.toList
|
| None -> Ok (Some outputs)
|
||||||
Success = successes |> Seq.toList
|
| Some err -> Error (err, outputs)
|
||||||
OtherFailures = [ tearDownError ; setupResult ] |> List.choose id
|
)
|
||||||
|
| _ -> Task.FromResult (Ok None, TestFixtureTearDownToken.vouchNoTearDownRequired running)
|
||||||
|
|
||||||
|
Environment.CurrentDirectory <- oldWorkDir
|
||||||
|
|
||||||
|
do! par.EndTestFixture tornDown
|
||||||
|
|
||||||
|
// TODO: we have access to stdout/err of OneTimeSetUp and OneTimeTearDown here, but we throw them away.
|
||||||
|
return
|
||||||
|
{
|
||||||
|
Failed = testFailures |> Seq.toList
|
||||||
|
Success = successes |> Seq.toList
|
||||||
|
OtherFailures =
|
||||||
|
[ tearDownError ; setupResult ]
|
||||||
|
|> List.choose (
|
||||||
|
function
|
||||||
|
| Error (e, _) -> Some e
|
||||||
|
| Ok _ -> None
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 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 =
|
let categories, args, mods, par =
|
||||||
parentType.CustomAttributes
|
(([], [], [], None), parentType.CustomAttributes)
|
||||||
|> Seq.filter (fun attr -> attr.AttributeType.FullName = "NUnit.Framework.CategoryAttribute")
|
||> Seq.fold (fun (categories, args, mods, par) attr ->
|
||||||
|> Seq.map (fun attr -> attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string>)
|
match attr.AttributeType.FullName with
|
||||||
|> Seq.toList
|
| "NUnit.Framework.SetUpFixtureAttribute" ->
|
||||||
|
failwith "This test runner does not support SetUpFixture. Please shout if you want this."
|
||||||
|
| "NUnit.Framework.CategoryAttribute" ->
|
||||||
|
let cat = attr.ConstructorArguments |> Seq.exactlyOne |> _.Value |> unbox<string>
|
||||||
|
cat :: categories, args, mods, par
|
||||||
|
| "NUnit.Framework.TestFixtureAttribute" ->
|
||||||
|
let newArgs =
|
||||||
|
match attr.ConstructorArguments |> Seq.map _.Value |> Seq.toList with
|
||||||
|
| [ :? ICollection as x ] ->
|
||||||
|
x |> Seq.cast<CustomAttributeTypedArgument> |> Seq.map _.Value |> Seq.toList
|
||||||
|
| xs -> xs
|
||||||
|
|
||||||
(TestFixture.Empty parentType.Assembly parentType.Name, parentType.GetRuntimeMethods ())
|
categories, newArgs :: args, mods, par
|
||||||
|
| "NUnit.Framework.NonParallelizableAttribute" ->
|
||||||
|
match par with
|
||||||
|
| Some _ -> failwith $"Got multiple parallelism attributes on %s{parentType.FullName}"
|
||||||
|
| None -> categories, args, mods, Some Parallelizable.No
|
||||||
|
| "NUnit.Framework.ParallelizableAttribute" ->
|
||||||
|
match par with
|
||||||
|
| Some _ -> failwith $"Got multiple parallelism attributes on %s{parentType.FullName}"
|
||||||
|
| None ->
|
||||||
|
match attr.ConstructorArguments |> Seq.toList with
|
||||||
|
| [] -> categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Self)
|
||||||
|
| [ v ] ->
|
||||||
|
match v.Value with
|
||||||
|
| :? int as v ->
|
||||||
|
match ParallelScope.ofInt v with
|
||||||
|
| ParallelScope.Fixtures ->
|
||||||
|
categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Fixtures)
|
||||||
|
| ParallelScope.Children ->
|
||||||
|
categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Children)
|
||||||
|
| ParallelScope.All ->
|
||||||
|
categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.All)
|
||||||
|
| ParallelScope.Self ->
|
||||||
|
categories, args, mods, Some (Parallelizable.Yes ClassParallelScope.Self)
|
||||||
|
| ParallelScope.None -> categories, args, mods, Some Parallelizable.No
|
||||||
|
| v ->
|
||||||
|
failwith
|
||||||
|
$"Unexpectedly non-int value %O{v} of parallel scope in %s{parentType.FullName}"
|
||||||
|
| _ -> failwith $"unexpectedly got multiple args to Parallelizable on %s{parentType.FullName}"
|
||||||
|
| "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 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 ->
|
||||||
@@ -616,3 +793,78 @@ module TestFixture =
|
|||||||
|
|
||||||
state
|
state
|
||||||
)
|
)
|
||||||
|
|
||||||
|
/// Run every test (except those which fail the `filter`) in this test fixture, as well as the
|
||||||
|
/// 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
|
||||||
|
(contexts : TestContexts)
|
||||||
|
(par : ParallelQueue)
|
||||||
|
(progress : ITestProgress)
|
||||||
|
(filter : TestFixture -> SingleTestMethod -> bool)
|
||||||
|
(tests : TestFixture)
|
||||||
|
: 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
|
||||||
|
| [] -> [ null ]
|
||||||
|
| args -> args |> List.map List.toArray
|
||||||
|
|> List.map (fun args ->
|
||||||
|
let containingObject =
|
||||||
|
let methods =
|
||||||
|
seq {
|
||||||
|
match tests.OneTimeSetUp with
|
||||||
|
| None -> ()
|
||||||
|
| Some t -> yield t
|
||||||
|
|
||||||
|
match tests.OneTimeTearDown with
|
||||||
|
| None -> ()
|
||||||
|
| Some t -> yield t
|
||||||
|
|
||||||
|
yield! tests.Tests |> Seq.map (fun t -> t.Method)
|
||||||
|
}
|
||||||
|
|
||||||
|
methods
|
||||||
|
|> Seq.tryPick (fun mi ->
|
||||||
|
if not mi.IsStatic then
|
||||||
|
Some (Activator.CreateInstance (mi.DeclaringType, args))
|
||||||
|
else
|
||||||
|
None
|
||||||
|
)
|
||||||
|
|> Option.toObj
|
||||||
|
|
||||||
|
let name =
|
||||||
|
if isNull args then
|
||||||
|
tests.Name
|
||||||
|
else
|
||||||
|
let args = args |> Seq.map (fun o -> o.ToString ()) |> String.concat ","
|
||||||
|
$"%s{tests.Name}(%s{args})"
|
||||||
|
|
||||||
|
runOneFixture contexts par progress filter name containingObject tests
|
||||||
|
)
|
||||||
|
|> Task.WhenAll
|
||||||
|
|> fun t ->
|
||||||
|
task {
|
||||||
|
let! t = t
|
||||||
|
return Array.toList t
|
||||||
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
namespace WoofWare.NUnitTestRunner
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
open System
|
open System
|
||||||
|
open System.IO
|
||||||
|
|
||||||
/// Represents something which knows how to report progress through a test suite.
|
/// Represents something which knows how to report progress through a test suite.
|
||||||
/// Note that we don't guarantee anything about parallelism; you must make sure
|
/// Note that we don't guarantee anything about parallelism; you must make sure
|
||||||
@@ -9,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
|
||||||
@@ -24,22 +27,28 @@ type ITestProgress =
|
|||||||
/// Methods for constructing specific ITestProgress objects.
|
/// Methods for constructing specific ITestProgress objects.
|
||||||
[<RequireQualifiedAccess>]
|
[<RequireQualifiedAccess>]
|
||||||
module TestProgress =
|
module TestProgress =
|
||||||
/// An ITestProgress which logs to stderr.
|
/// An ITestProgress which logs to the given writer.
|
||||||
let toStderr () : ITestProgress =
|
let toWriter (writer : TextWriter) : ITestProgress =
|
||||||
{ new ITestProgress with
|
{ new ITestProgress with
|
||||||
member _.OnTestFixtureStart name testCount =
|
member _.OnTestFixtureStart name testCount =
|
||||||
let plural = if testCount = 1 then "" else "s"
|
let plural = if testCount = 1 then "" else "s"
|
||||||
Console.Error.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 =
|
||||||
Console.Error.WriteLine $"Running test: %s{name}"
|
writer.WriteLine $"Running test: %s{name}"
|
||||||
|
|
||||||
member _.OnTestFailed name failure =
|
member _.OnTestFailed name failure =
|
||||||
Console.Error.WriteLine $"Test failed: %O{failure}"
|
writer.WriteLine $"Test failed: %O{failure}"
|
||||||
|
|
||||||
member _.OnTestMemberFinished name =
|
member _.OnTestMemberFinished name =
|
||||||
Console.Error.WriteLine $"Finished test %s{name}"
|
writer.WriteLine $"Finished test %s{name}"
|
||||||
|
|
||||||
member _.OnTestMemberSkipped name =
|
member _.OnTestMemberSkipped name =
|
||||||
Console.Error.WriteLine $"Skipping test due to filter: %s{name}"
|
writer.WriteLine $"Skipping test due to filter: %s{name}"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// An ITestProgress which logs to stderr.
|
||||||
|
let toStderr () : ITestProgress = toWriter Console.Error
|
||||||
|
@@ -290,6 +290,8 @@ type TrxOutput =
|
|||||||
{
|
{
|
||||||
/// What the entity printed to standard output.
|
/// What the entity printed to standard output.
|
||||||
StdOut : string option
|
StdOut : string option
|
||||||
|
/// What the entity printed to standard error.
|
||||||
|
StdErr : string option
|
||||||
/// Description of any error the entity encountered.
|
/// Description of any error the entity encountered.
|
||||||
ErrorInfo : TrxErrorInfo option
|
ErrorInfo : TrxErrorInfo option
|
||||||
}
|
}
|
||||||
@@ -305,6 +307,14 @@ type TrxOutput =
|
|||||||
childNode.AppendChild text |> ignore<XmlNode>
|
childNode.AppendChild text |> ignore<XmlNode>
|
||||||
node.AppendChild childNode |> ignore<XmlNode>
|
node.AppendChild childNode |> ignore<XmlNode>
|
||||||
|
|
||||||
|
match this.StdErr with
|
||||||
|
| None -> ()
|
||||||
|
| Some stderr ->
|
||||||
|
let text = doc.CreateTextNode stderr
|
||||||
|
let childNode = doc.CreateElement ("StdErr", XmlUtil.NS)
|
||||||
|
childNode.AppendChild text |> ignore<XmlNode>
|
||||||
|
node.AppendChild childNode |> ignore<XmlNode>
|
||||||
|
|
||||||
match this.ErrorInfo with
|
match this.ErrorInfo with
|
||||||
| None -> ()
|
| None -> ()
|
||||||
| Some errInfo -> node.AppendChild (errInfo.toXml doc) |> ignore<XmlNode>
|
| Some errInfo -> node.AppendChild (errInfo.toXml doc) |> ignore<XmlNode>
|
||||||
@@ -317,6 +327,11 @@ type TrxOutput =
|
|||||||
| NodeWithNamedChild "StdOut" (OneChildNode "StdOut" (NoChildrenNode stdout)) -> Some stdout
|
| NodeWithNamedChild "StdOut" (OneChildNode "StdOut" (NoChildrenNode stdout)) -> Some stdout
|
||||||
| _ -> None
|
| _ -> None
|
||||||
|
|
||||||
|
let stderr =
|
||||||
|
match node with
|
||||||
|
| NodeWithNamedChild "StdErr" (OneChildNode "StdErr" (NoChildrenNode stdout)) -> Some stdout
|
||||||
|
| _ -> None
|
||||||
|
|
||||||
let errorInfo =
|
let errorInfo =
|
||||||
match node with
|
match node with
|
||||||
| NodeWithNamedChild "ErrorInfo" node ->
|
| NodeWithNamedChild "ErrorInfo" node ->
|
||||||
@@ -330,6 +345,7 @@ type TrxOutput =
|
|||||||
| Ok errorInfo ->
|
| Ok errorInfo ->
|
||||||
{
|
{
|
||||||
StdOut = stdout
|
StdOut = stdout
|
||||||
|
StdErr = stderr
|
||||||
ErrorInfo = errorInfo
|
ErrorInfo = errorInfo
|
||||||
}
|
}
|
||||||
|> Ok
|
|> Ok
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>netstandard2.1</TargetFramework>
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
||||||
<Authors>Patrick Stevens</Authors>
|
<Authors>Patrick Stevens</Authors>
|
||||||
<Copyright>Copyright (c) Patrick Stevens 2024</Copyright>
|
<Copyright>Copyright (c) Patrick Stevens 2024</Copyright>
|
||||||
@@ -14,20 +14,38 @@
|
|||||||
<PackageId>WoofWare.NUnitTestRunner.Lib</PackageId>
|
<PackageId>WoofWare.NUnitTestRunner.Lib</PackageId>
|
||||||
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
||||||
<WarnOn>FS3559</WarnOn>
|
<WarnOn>FS3559</WarnOn>
|
||||||
|
<WoofWareMyriadPluginVersion>9.0.4</WoofWareMyriadPluginVersion>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="AssemblyInfo.fs" />
|
<Compile Include="AssemblyInfo.fs" />
|
||||||
|
<Compile Include="RuntimeConfig.fs" />
|
||||||
|
<Compile Include="GeneratedRuntimeConfig.fs">
|
||||||
|
<MyriadFile>RuntimeConfig.fs</MyriadFile>
|
||||||
|
<MyriadParams>
|
||||||
|
<RuntimeOptions>JsonParse</RuntimeOptions>
|
||||||
|
<RuntimeConfig>JsonParse</RuntimeConfig>
|
||||||
|
<FrameworkDescription>JsonParse</FrameworkDescription>
|
||||||
|
</MyriadParams>
|
||||||
|
</Compile>
|
||||||
|
<Compile Include="ParallelScope.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" />
|
||||||
<Compile Include="Filter.fs" />
|
<Compile Include="Filter.fs" />
|
||||||
|
<Compile Include="Args.fs" />
|
||||||
|
<Compile Include="ParallelQueue.fs" />
|
||||||
<Compile Include="SingleTestMethod.fs" />
|
<Compile Include="SingleTestMethod.fs" />
|
||||||
<Compile Include="TestProgress.fs" />
|
<Compile Include="TestProgress.fs" />
|
||||||
|
<Compile Include="Context.fs" />
|
||||||
<Compile Include="TestFixture.fs" />
|
<Compile Include="TestFixture.fs" />
|
||||||
<Compile Include="Xml.fs" />
|
<Compile Include="Xml.fs" />
|
||||||
<Compile Include="TrxReport.fs" />
|
<Compile Include="TrxReport.fs" />
|
||||||
|
<Compile Include="CreateTrxReport.fs" />
|
||||||
|
<Compile Include="AssemblyLevelAttributes.fs" />
|
||||||
<None Include="..\README.md">
|
<None Include="..\README.md">
|
||||||
<Pack>True</Pack>
|
<Pack>True</Pack>
|
||||||
<PackagePath>\</PackagePath>
|
<PackagePath>\</PackagePath>
|
||||||
@@ -36,8 +54,15 @@
|
|||||||
<EmbeddedResource Include="version.json" />
|
<EmbeddedResource Include="version.json" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="WoofWare.PrattParser" Version="0.1.2" />
|
<PackageReference Include="WoofWare.PrattParser" Version="0.2.5" />
|
||||||
<PackageReference Update="FSharp.Core" Version="6.0.0" />
|
<PackageReference Update="FSharp.Core" Version="6.0.1" />
|
||||||
|
<PackageReference Include="WoofWare.DotnetRuntimeLocator" Version="0.1.12" />
|
||||||
|
<PackageReference Include="Myriad.SDK" Version="0.8.3" PrivateAssets="all" />
|
||||||
|
<PackageReference Include="WoofWare.Myriad.Plugins" Version="$(WoofWareMyriadPluginVersion)" PrivateAssets="all" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<MyriadSdkGenerator Include="$(NuGetPackageRoot)/woofware.myriad.plugins/$(WoofWareMyriadPluginVersion)/lib/net6.0/WoofWare.Myriad.Plugins.dll" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
0
WoofWare.NUnitTestRunner.Lib/myriad.toml
Normal file
0
WoofWare.NUnitTestRunner.Lib/myriad.toml
Normal file
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"version": "0.8",
|
"version": "0.22",
|
||||||
"publicReleaseRefSpec": [
|
"publicReleaseRefSpec": [
|
||||||
"^refs/heads/main$"
|
"^refs/heads/main$"
|
||||||
],
|
],
|
||||||
|
35
WoofWare.NUnitTestRunner.StartupHook/StartupHook.cs
Normal file
35
WoofWare.NUnitTestRunner.StartupHook/StartupHook.cs
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
using System.Linq;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.Loader;
|
||||||
|
using WoofWare.NUnitTestRunner.StartupHook;
|
||||||
|
|
||||||
|
namespace WoofWare.NUnitTestRunner.StartupHook
|
||||||
|
{
|
||||||
|
internal class StartupAssemblyLoadContext : AssemblyLoadContext
|
||||||
|
{
|
||||||
|
private readonly AssemblyDependencyResolver _resolver;
|
||||||
|
|
||||||
|
public StartupAssemblyLoadContext()
|
||||||
|
{
|
||||||
|
_resolver = new AssemblyDependencyResolver(Assembly.GetExecutingAssembly().Location);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override Assembly Load(AssemblyName assemblyName)
|
||||||
|
{
|
||||||
|
var assemblyPath = _resolver.ResolveAssemblyToPath(assemblyName);
|
||||||
|
|
||||||
|
return assemblyPath != null ? LoadFromAssemblyPath(assemblyPath) : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Must be internal and called `StartupHook`
|
||||||
|
internal class StartupHook
|
||||||
|
{
|
||||||
|
public static void Initialize()
|
||||||
|
{
|
||||||
|
var loadContext = new StartupAssemblyLoadContext();
|
||||||
|
var assembly = loadContext.LoadFromAssemblyName(new AssemblyName("WoofWare.NUnitTestRunner.StartupHookLogic"));
|
||||||
|
assembly.DefinedTypes.First(x => x.Name == "StartupHookLogic").GetDeclaredMethod("DoIt")!.Invoke(null, null);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,17 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<EnableDefaultCompileItems>false</EnableDefaultCompileItems>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\WoofWare.NUnitTestRunner.StartupHookLogic\WoofWare.NUnitTestRunner.StartupHookLogic.csproj" />
|
||||||
|
<ProjectReference Include="..\WoofWare.NUnitTestRunner.Lib\WoofWare.NUnitTestRunner.Lib.fsproj"/>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Compile Include="StartupHook.cs"/>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
102
WoofWare.NUnitTestRunner.StartupHookLogic/StartupHookLogic.cs
Normal file
102
WoofWare.NUnitTestRunner.StartupHookLogic/StartupHookLogic.cs
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
using System.Reflection;
|
||||||
|
using Microsoft.FSharp.Collections;
|
||||||
|
using Microsoft.FSharp.Core;
|
||||||
|
|
||||||
|
namespace WoofWare.NUnitTestRunner.StartupHookLogic;
|
||||||
|
|
||||||
|
public class StartupHookLogic
|
||||||
|
{
|
||||||
|
private static void DoIt()
|
||||||
|
{
|
||||||
|
// Load test runner lib
|
||||||
|
var nunitLib = Environment.GetEnvironmentVariable("WOOFWARE_NUNIT_LIB");
|
||||||
|
if (string.IsNullOrEmpty(nunitLib))
|
||||||
|
{
|
||||||
|
throw new ArgumentException("WoofWare.NUnitTestRunner hook expects to run with WOOFWARE_NUNIT_LIB set to WoofWare.NUnitTestRunner.Lib.dll");
|
||||||
|
}
|
||||||
|
Assembly.LoadFrom(nunitLib);
|
||||||
|
|
||||||
|
var startTime = DateTimeOffset.Now;
|
||||||
|
|
||||||
|
// Arg parsing
|
||||||
|
|
||||||
|
var filterVar = Environment.GetEnvironmentVariable("WOOFWARE_NUNIT_FILTER");
|
||||||
|
static bool NoFilter(TestFixture f, SingleTestMethod g)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
FSharpFunc<TestFixture, FSharpFunc<SingleTestMethod, bool>> filter;
|
||||||
|
if (string.IsNullOrEmpty(filterVar))
|
||||||
|
{
|
||||||
|
filter = FuncConvert.FromFunc<TestFixture, SingleTestMethod, bool>(NoFilter);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
filter = FilterModule.shouldRun(FilterModule.parse(filterVar));
|
||||||
|
}
|
||||||
|
|
||||||
|
var assy = Assembly.GetEntryAssembly()!;
|
||||||
|
|
||||||
|
var attrs = AssemblyLevelAttributesModule.get(assy);
|
||||||
|
|
||||||
|
FSharpOption<int> levelOfParallelism;
|
||||||
|
var parallelismVar = Environment.GetEnvironmentVariable("WOOFWARE_NUNIT_PARALLELISM_LEVEL");
|
||||||
|
if (string.IsNullOrEmpty(parallelismVar))
|
||||||
|
{
|
||||||
|
levelOfParallelism = attrs.Parallelism;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
levelOfParallelism = FSharpOption<int>.Some(Int32.Parse(parallelismVar));
|
||||||
|
}
|
||||||
|
|
||||||
|
var testFixtures = assy.ExportedTypes.Select(TestFixtureModule.parse);
|
||||||
|
using var par =
|
||||||
|
new ParallelQueue(levelOfParallelism, attrs.Parallelizable, FSharpOption<CancellationToken>.None);
|
||||||
|
var creationTime = DateTimeOffset.Now;
|
||||||
|
|
||||||
|
var timeoutVar = Environment.GetEnvironmentVariable("WOOFWARE_NUNIT_TIMEOUT_SECS");
|
||||||
|
TimeSpan timeout;
|
||||||
|
if (string.IsNullOrEmpty(timeoutVar))
|
||||||
|
{
|
||||||
|
timeout = TimeSpan.FromHours(2.0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
timeout = TimeSpan.FromSeconds(Int32.Parse(timeoutVar));
|
||||||
|
}
|
||||||
|
|
||||||
|
var normalErr = Console.Error;
|
||||||
|
using var contexts = TestContexts.Empty();
|
||||||
|
Console.SetOut(contexts.Stdout);
|
||||||
|
Console.SetError(contexts.Stderr);
|
||||||
|
var results =
|
||||||
|
Task.WhenAll(testFixtures.Select(x =>
|
||||||
|
TestFixtureModule.run(contexts, par, TestProgress.toWriter(normalErr), filter, x)));
|
||||||
|
|
||||||
|
if (!results.Wait(timeout))
|
||||||
|
{
|
||||||
|
throw new Exception($"Tests failed to terminate within timeout of {timeout}");
|
||||||
|
}
|
||||||
|
|
||||||
|
var sorted =
|
||||||
|
results.Result.SelectMany(x => x);
|
||||||
|
var report = BuildTrxReport.build(assy, creationTime, startTime, ListModule.OfSeq(sorted));
|
||||||
|
|
||||||
|
var trxFile = Environment.GetEnvironmentVariable("WOOFWARE_NUNIT_GENERATE_TRX");
|
||||||
|
if (!string.IsNullOrEmpty(trxFile))
|
||||||
|
{
|
||||||
|
var contents = TrxReportModule.toXml(report).OuterXml;
|
||||||
|
var trxPath = new FileInfo(trxFile);
|
||||||
|
trxPath.Directory!.Create();
|
||||||
|
File.WriteAllText(trxPath.FullName, contents);
|
||||||
|
normalErr.WriteLine($"Written TRX file: {trxPath.FullName}");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (report.ResultsSummary.Outcome.Equals(TrxOutcome.Completed))
|
||||||
|
Environment.Exit(0);
|
||||||
|
else
|
||||||
|
Environment.Exit(1);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,18 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<EnableDefaultCompileItems>false</EnableDefaultCompileItems>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\WoofWare.NUnitTestRunner.Lib\WoofWare.NUnitTestRunner.Lib.fsproj"/>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Compile Include="StartupHookLogic.cs"/>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
@@ -8,6 +8,12 @@ Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "WoofWare.NUnitTestRunner.Li
|
|||||||
EndProject
|
EndProject
|
||||||
Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "WoofWare.NUnitTestRunner.Test", "WoofWare.NUnitTestRunner\WoofWare.NUnitTestRunner.Test\WoofWare.NUnitTestRunner.Test.fsproj", "{443B01B3-2A8C-45CF-96D6-1D890EEA0533}"
|
Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "WoofWare.NUnitTestRunner.Test", "WoofWare.NUnitTestRunner\WoofWare.NUnitTestRunner.Test\WoofWare.NUnitTestRunner.Test.fsproj", "{443B01B3-2A8C-45CF-96D6-1D890EEA0533}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WoofWare.NUnitTestRunner.StartupHook", "WoofWare.NUnitTestRunner.StartupHook\WoofWare.NUnitTestRunner.StartupHook.csproj", "{E2C73D96-570C-4E3C-B997-707AF8BB0E6A}"
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WoofWare.NUnitTestRunner.StartupHookLogic", "WoofWare.NUnitTestRunner.StartupHookLogic\WoofWare.NUnitTestRunner.StartupHookLogic.csproj", "{A70627C8-9D19-42C2-AFEB-CFBDDDCE045D}"
|
||||||
|
EndProject
|
||||||
|
Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "FailingConsumer", "FailingConsumer\FailingConsumer.fsproj", "{DA7160F5-4C3C-4D2E-918B-7DCBA3F4272E}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
@@ -30,5 +36,17 @@ Global
|
|||||||
{443B01B3-2A8C-45CF-96D6-1D890EEA0533}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{443B01B3-2A8C-45CF-96D6-1D890EEA0533}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{443B01B3-2A8C-45CF-96D6-1D890EEA0533}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{443B01B3-2A8C-45CF-96D6-1D890EEA0533}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{443B01B3-2A8C-45CF-96D6-1D890EEA0533}.Release|Any CPU.Build.0 = Release|Any CPU
|
{443B01B3-2A8C-45CF-96D6-1D890EEA0533}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{E2C73D96-570C-4E3C-B997-707AF8BB0E6A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{E2C73D96-570C-4E3C-B997-707AF8BB0E6A}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{E2C73D96-570C-4E3C-B997-707AF8BB0E6A}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{E2C73D96-570C-4E3C-B997-707AF8BB0E6A}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{A70627C8-9D19-42C2-AFEB-CFBDDDCE045D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{A70627C8-9D19-42C2-AFEB-CFBDDDCE045D}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{A70627C8-9D19-42C2-AFEB-CFBDDDCE045D}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{A70627C8-9D19-42C2-AFEB-CFBDDDCE045D}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{DA7160F5-4C3C-4D2E-918B-7DCBA3F4272E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{DA7160F5-4C3C-4D2E-918B-7DCBA3F4272E}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{DA7160F5-4C3C-4D2E-918B-7DCBA3F4272E}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{DA7160F5-4C3C-4D2E-918B-7DCBA3F4272E}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
EndGlobal
|
EndGlobal
|
||||||
|
@@ -1,402 +1,84 @@
|
|||||||
namespace WoofWare.NUnitTestRunner
|
namespace WoofWare.NUnitTestRunner
|
||||||
|
|
||||||
open System
|
open System
|
||||||
open WoofWare.DotnetRuntimeLocator
|
open System.Diagnostics
|
||||||
open System.IO
|
open System.IO
|
||||||
open System.Reflection
|
open System.Reflection
|
||||||
open System.Runtime.Loader
|
open System.Threading.Tasks
|
||||||
|
open Spectre.Console
|
||||||
// Fix for https://github.com/Smaug123/unofficial-nunit-runner/issues/8
|
|
||||||
// Set AppContext.BaseDirectory to where the test DLL is.
|
|
||||||
// (This tells the DLL loader to look next to the test DLL for dependencies.)
|
|
||||||
type SetBaseDir (testDll : FileInfo) =
|
|
||||||
let oldBaseDir = AppContext.BaseDirectory
|
|
||||||
do AppContext.SetData ("APP_CONTEXT_BASE_DIRECTORY", testDll.Directory.FullName)
|
|
||||||
|
|
||||||
interface IDisposable with
|
|
||||||
member _.Dispose () =
|
|
||||||
AppContext.SetData ("APP_CONTEXT_BASE_DIRECTORY", oldBaseDir)
|
|
||||||
|
|
||||||
|
|
||||||
type Ctx (dll : FileInfo, runtimes : DirectoryInfo list) =
|
|
||||||
inherit AssemblyLoadContext ()
|
|
||||||
|
|
||||||
override this.Load (target : AssemblyName) : Assembly =
|
|
||||||
let path = Path.Combine (dll.Directory.FullName, $"%s{target.Name}.dll")
|
|
||||||
|
|
||||||
if File.Exists path then
|
|
||||||
this.LoadFromAssemblyPath path
|
|
||||||
else
|
|
||||||
|
|
||||||
runtimes
|
|
||||||
|> List.tryPick (fun di ->
|
|
||||||
let path = Path.Combine (di.FullName, $"%s{target.Name}.dll")
|
|
||||||
|
|
||||||
if File.Exists path then
|
|
||||||
this.LoadFromAssemblyPath path |> Some
|
|
||||||
else
|
|
||||||
None
|
|
||||||
)
|
|
||||||
|> Option.defaultValue null
|
|
||||||
|
|
||||||
|
|
||||||
module Program =
|
module Program =
|
||||||
let selectRuntime
|
// This is actually transcribed into C# in WoofWare.NUnitTestRunner.StartupHookLogic.
|
||||||
(config : RuntimeOptions)
|
let execute argv =
|
||||||
(f : DotnetEnvironmentInfo)
|
|
||||||
: Choice<DotnetEnvironmentFrameworkInfo, DotnetEnvironmentSdkInfo> option
|
|
||||||
=
|
|
||||||
let rollForward =
|
|
||||||
match Environment.GetEnvironmentVariable "DOTNET_ROLL_FORWARD" with
|
|
||||||
| null ->
|
|
||||||
config.RollForward
|
|
||||||
|> Option.map RollForward.Parse
|
|
||||||
|> Option.defaultValue RollForward.Minor
|
|
||||||
| s -> RollForward.Parse s
|
|
||||||
|
|
||||||
let desiredVersions =
|
|
||||||
match config.Framework with
|
|
||||||
| Some f -> [ Version f.Version, f.Name ]
|
|
||||||
| None ->
|
|
||||||
|
|
||||||
match config.Frameworks with
|
|
||||||
| Some f -> f |> List.map (fun f -> Version f.Version, f.Name)
|
|
||||||
| None ->
|
|
||||||
failwith
|
|
||||||
"Could not deduce a framework version due to lack of either Framework or Frameworks in runtimeconfig"
|
|
||||||
|
|
||||||
let compatiblyNamedRuntimes =
|
|
||||||
f.Frameworks
|
|
||||||
|> Seq.collect (fun availableFramework ->
|
|
||||||
desiredVersions
|
|
||||||
|> List.choose (fun (desiredVersion, desiredName) ->
|
|
||||||
if desiredName = availableFramework.Name then
|
|
||||||
Some
|
|
||||||
{|
|
|
||||||
Desired = desiredVersion
|
|
||||||
Name = desiredName
|
|
||||||
Installed = availableFramework
|
|
||||||
InstalledVersion = Version availableFramework.Version
|
|
||||||
|}
|
|
||||||
else
|
|
||||||
None
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|> Seq.toList
|
|
||||||
|
|
||||||
match rollForward with
|
|
||||||
| RollForward.Minor ->
|
|
||||||
let available =
|
|
||||||
compatiblyNamedRuntimes
|
|
||||||
|> Seq.filter (fun data ->
|
|
||||||
data.InstalledVersion.Major = data.Desired.Major
|
|
||||||
&& data.InstalledVersion.Minor >= data.Desired.Minor
|
|
||||||
)
|
|
||||||
|> Seq.groupBy (fun data -> data.Name)
|
|
||||||
|> Seq.map (fun (name, data) ->
|
|
||||||
let data =
|
|
||||||
data
|
|
||||||
|> Seq.minBy (fun data -> data.InstalledVersion.Minor, data.InstalledVersion.Build)
|
|
||||||
|
|
||||||
name, data.Installed
|
|
||||||
)
|
|
||||||
// TODO: how do we select between many available frameworks?
|
|
||||||
|> Seq.tryHead
|
|
||||||
|
|
||||||
match available with
|
|
||||||
| Some (_, f) -> Some (Choice1Of2 f)
|
|
||||||
| None ->
|
|
||||||
// TODO: maybe we can ask the SDK. But we keep on trucking: maybe we're self-contained,
|
|
||||||
// and we'll actually find all the runtime next to the DLL.
|
|
||||||
None
|
|
||||||
| _ -> failwith "non-minor RollForward not supported yet; please shout if you want it"
|
|
||||||
|
|
||||||
let locateRuntimes (dll : FileInfo) : DirectoryInfo list =
|
|
||||||
let runtimeConfig =
|
|
||||||
let name =
|
|
||||||
if not (dll.Name.EndsWith (".dll", StringComparison.OrdinalIgnoreCase)) then
|
|
||||||
failwith $"Expected DLL %s{dll.FullName} to end in .dll"
|
|
||||||
|
|
||||||
dll.Name.Substring (0, dll.Name.Length - 4)
|
|
||||||
|
|
||||||
Path.Combine (dll.Directory.FullName, $"%s{name}.runtimeconfig.json")
|
|
||||||
|> File.ReadAllText
|
|
||||||
|> System.Text.Json.Nodes.JsonNode.Parse
|
|
||||||
|> RuntimeConfig.jsonParse
|
|
||||||
|> fun f -> f.RuntimeOptions
|
|
||||||
|
|
||||||
let availableRuntimes = DotnetEnvironmentInfo.Get ()
|
|
||||||
|
|
||||||
let runtime = selectRuntime runtimeConfig availableRuntimes
|
|
||||||
|
|
||||||
match runtime with
|
|
||||||
| None ->
|
|
||||||
// Keep on trucking: let's be optimistic and hope that we're self-contained.
|
|
||||||
[ dll.Directory ]
|
|
||||||
| Some (Choice1Of2 runtime) -> [ dll.Directory ; DirectoryInfo $"%s{runtime.Path}/%s{runtime.Version}" ]
|
|
||||||
| Some (Choice2Of2 sdk) -> [ dll.Directory ; DirectoryInfo sdk.Path ]
|
|
||||||
|
|
||||||
let main argv =
|
|
||||||
let startTime = DateTimeOffset.Now
|
let startTime = DateTimeOffset.Now
|
||||||
|
|
||||||
let testDll, filter, trxPath =
|
let args = argv |> List.ofArray |> Args.Parse
|
||||||
match argv |> List.ofSeq with
|
|
||||||
| [ dll ] -> FileInfo dll, None, None
|
|
||||||
| [ dll ; "--trx" ; trxPath ] -> FileInfo dll, None, Some (FileInfo trxPath)
|
|
||||||
| [ dll ; "--filter" ; filter ] -> FileInfo dll, Some (Filter.parse filter), None
|
|
||||||
| [ dll ; "--trx" ; trxPath ; "--filter" ; filter ] ->
|
|
||||||
FileInfo dll, Some (Filter.parse filter), Some (FileInfo trxPath)
|
|
||||||
| [ dll ; "--filter" ; filter ; "--trx" ; trxPath ] ->
|
|
||||||
FileInfo dll, Some (Filter.parse filter), Some (FileInfo trxPath)
|
|
||||||
| _ ->
|
|
||||||
failwith
|
|
||||||
"provide exactly one arg, a test DLL; then optionally `--filter <filter>` and/or `--trx <output-filename>`."
|
|
||||||
|
|
||||||
let filter =
|
let filter =
|
||||||
match filter with
|
match args.Filter with
|
||||||
| Some filter -> Filter.shouldRun filter
|
| Some (_, filter) -> Filter.shouldRun filter
|
||||||
| None -> fun _ _ -> true
|
| None -> fun _ _ -> true
|
||||||
|
|
||||||
let progress = Progress.spectre ()
|
let stderr =
|
||||||
|
let consoleSettings = AnsiConsoleSettings ()
|
||||||
|
consoleSettings.Out <- AnsiConsoleOutput Console.Error
|
||||||
|
AnsiConsole.Create consoleSettings
|
||||||
|
|
||||||
use _ = new SetBaseDir (testDll)
|
let progress = Progress.spectre stderr
|
||||||
|
|
||||||
let ctx = Ctx (testDll, locateRuntimes testDll)
|
let runtime = DotnetRuntime.locate args.Dll
|
||||||
let assy = ctx.LoadFromAssemblyPath testDll.FullName
|
|
||||||
|
match args.Logging with
|
||||||
|
| LogLevel.Nothing -> ()
|
||||||
|
| LogLevel.Verbose ->
|
||||||
|
for d in runtime do
|
||||||
|
stderr.WriteLine $".NET runtime directory: %s{d.FullName}"
|
||||||
|
|
||||||
|
use contexts = TestContexts.Empty ()
|
||||||
|
|
||||||
|
let ctx = LoadContext (args.Dll, runtime, contexts)
|
||||||
|
let assy = ctx.LoadFromAssemblyPath args.Dll.FullName
|
||||||
|
|
||||||
|
let attrs = AssemblyLevelAttributes.get assy
|
||||||
|
|
||||||
|
let levelOfParallelism =
|
||||||
|
match args.LevelOfParallelism, attrs.Parallelism with
|
||||||
|
| None, None -> None
|
||||||
|
| Some taken, Some ignored ->
|
||||||
|
match args.Logging with
|
||||||
|
| LogLevel.Nothing -> ()
|
||||||
|
| LogLevel.Verbose ->
|
||||||
|
stderr.WriteLine
|
||||||
|
$"Taking parallelism %i{taken} from command line, ignoring value %i{ignored} from assembly"
|
||||||
|
|
||||||
|
Some taken
|
||||||
|
| Some x, None
|
||||||
|
| None, Some x -> Some x
|
||||||
|
|
||||||
let testFixtures = assy.ExportedTypes |> Seq.map TestFixture.parse |> Seq.toList
|
let testFixtures = assy.ExportedTypes |> Seq.map TestFixture.parse |> Seq.toList
|
||||||
|
|
||||||
|
use par = new ParallelQueue (levelOfParallelism, attrs.Parallelizable)
|
||||||
|
|
||||||
let creationTime = DateTimeOffset.Now
|
let creationTime = DateTimeOffset.Now
|
||||||
let results = testFixtures |> List.map (TestFixture.run progress filter)
|
|
||||||
|
|
||||||
let finishTime = DateTimeOffset.Now
|
|
||||||
let finishTimeHumanReadable = finishTime.ToString @"yyyy-MM-dd HH:mm:ss"
|
|
||||||
let nowMachine = finishTime.ToString @"yyyy-MM-dd_HH_mm_ss"
|
|
||||||
|
|
||||||
let testListId = Guid.NewGuid ()
|
|
||||||
|
|
||||||
let testDefinitions, testEntries =
|
|
||||||
results
|
|
||||||
|> List.collect (fun results -> results.IndividualTestRunMetadata)
|
|
||||||
|> List.map (fun (data, _) ->
|
|
||||||
let defn =
|
|
||||||
{
|
|
||||||
Name = data.TestName
|
|
||||||
Storage = assy.Location.ToLowerInvariant ()
|
|
||||||
Id = data.TestId
|
|
||||||
Execution =
|
|
||||||
{
|
|
||||||
Id = data.ExecutionId
|
|
||||||
}
|
|
||||||
TestMethod =
|
|
||||||
{
|
|
||||||
CodeBase = assy.Location
|
|
||||||
AdapterTypeName = Uri "executor://woofware/"
|
|
||||||
ClassName = data.ClassName
|
|
||||||
Name = data.TestName
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let entry : TrxTestEntry =
|
|
||||||
{
|
|
||||||
TestListId = testListId
|
|
||||||
ExecutionId = data.ExecutionId
|
|
||||||
TestId = data.TestId
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
defn, entry
|
|
||||||
)
|
|
||||||
|> List.unzip
|
|
||||||
|
|
||||||
let hostname = Environment.MachineName
|
|
||||||
|
|
||||||
let settings =
|
|
||||||
{
|
|
||||||
Name = "default"
|
|
||||||
Id = Guid.NewGuid ()
|
|
||||||
Deployment =
|
|
||||||
{
|
|
||||||
RunDeploymentRoot = $"_%s{hostname}_%s{nowMachine}"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let testList : TrxTestListEntry =
|
|
||||||
{
|
|
||||||
Id = testListId
|
|
||||||
Name = "All"
|
|
||||||
}
|
|
||||||
|
|
||||||
let counters =
|
|
||||||
(TrxCounters.Zero, results)
|
|
||||||
// TODO: this is woefully inefficient
|
|
||||||
||> List.fold (fun counters results ->
|
|
||||||
let counters =
|
|
||||||
(counters, results.Failed)
|
|
||||||
||> List.fold (fun counters (_, _) ->
|
|
||||||
// TODO: the counters can be more specific about the failure mode
|
|
||||||
counters.AddFailed ()
|
|
||||||
)
|
|
||||||
|
|
||||||
let counters =
|
|
||||||
(counters, results.OtherFailures)
|
|
||||||
||> List.fold (fun counters _ ->
|
|
||||||
// TODO: the counters can be more specific about the failure mode
|
|
||||||
counters.AddFailed ()
|
|
||||||
)
|
|
||||||
|
|
||||||
(counters, results.Success)
|
|
||||||
||> List.fold (fun counters (_, success, _) ->
|
|
||||||
match success with
|
|
||||||
| TestMemberSuccess.Ok -> counters.AddPassed ()
|
|
||||||
| TestMemberSuccess.Ignored _
|
|
||||||
| TestMemberSuccess.Explicit _ -> counters.AddNotExecuted ()
|
|
||||||
| TestMemberSuccess.Inconclusive _ -> counters.AddInconclusive ()
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
// TODO: I'm sure we can do better than this; there's a whole range of possible
|
|
||||||
// states!
|
|
||||||
let outcome =
|
|
||||||
if counters.Failed > 0u then
|
|
||||||
TrxOutcome.Failed
|
|
||||||
else
|
|
||||||
TrxOutcome.Completed
|
|
||||||
|
|
||||||
let resultSummary : TrxResultsSummary =
|
|
||||||
{
|
|
||||||
Outcome = outcome
|
|
||||||
Counters = counters
|
|
||||||
Output =
|
|
||||||
{
|
|
||||||
StdOut = None
|
|
||||||
ErrorInfo = None
|
|
||||||
}
|
|
||||||
RunInfos =
|
|
||||||
[
|
|
||||||
// TODO: capture stdout
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
let times : TrxReportTimes =
|
|
||||||
{
|
|
||||||
Creation = creationTime
|
|
||||||
Queuing = startTime
|
|
||||||
Start = startTime
|
|
||||||
Finish = finishTime
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
let magicGuid = Guid.Parse "13cdc9d9-ddb5-4fa4-a97d-d965ccfc6d4b"
|
|
||||||
|
|
||||||
let results =
|
let results =
|
||||||
results
|
testFixtures
|
||||||
|> List.collect (fun results -> results.IndividualTestRunMetadata)
|
|> List.map (TestFixture.run contexts par progress filter)
|
||||||
|> List.map (fun (i, cause) ->
|
|> Task.WhenAll
|
||||||
let exc =
|
|
||||||
match cause with
|
|
||||||
| Choice2Of3 _ -> None
|
|
||||||
| Choice1Of3 (TestMemberFailure.Malformed reasons) ->
|
|
||||||
{
|
|
||||||
StackTrace = None
|
|
||||||
Message = reasons |> String.concat "\n" |> Some
|
|
||||||
}
|
|
||||||
|> Some
|
|
||||||
| Choice1Of3 (TestMemberFailure.Failed fail)
|
|
||||||
| Choice1Of3 (TestMemberFailure.Failed fail)
|
|
||||||
| Choice1Of3 (TestMemberFailure.Failed fail) ->
|
|
||||||
((None, None), fail)
|
|
||||||
||> List.fold (fun (stackTrace, message) tf ->
|
|
||||||
match tf with
|
|
||||||
| TestFailure.TestFailed (UserMethodFailure.Threw (_, exc))
|
|
||||||
| TestFailure.SetUpFailed (UserMethodFailure.Threw (_, exc))
|
|
||||||
| TestFailure.TearDownFailed (UserMethodFailure.Threw (_, exc)) ->
|
|
||||||
let stackTrace =
|
|
||||||
match stackTrace with
|
|
||||||
| None -> (exc : Exception).ToString ()
|
|
||||||
| Some s -> s
|
|
||||||
|
|
||||||
(Some stackTrace, message)
|
let timeout =
|
||||||
| TestFailure.TestFailed (UserMethodFailure.ReturnedNonUnit (_, ret))
|
match args.Timeout with
|
||||||
| TestFailure.SetUpFailed (UserMethodFailure.ReturnedNonUnit (_, ret))
|
| None -> TimeSpan.FromHours 2.0
|
||||||
| TestFailure.TearDownFailed (UserMethodFailure.ReturnedNonUnit (_, ret)) ->
|
| Some t -> t
|
||||||
let newMessage = $"returned non-unit value %O{ret}"
|
|
||||||
|
|
||||||
let message =
|
if not (results.Wait timeout) then
|
||||||
match message with
|
failwith "Tests failed to terminate within two hours"
|
||||||
| None -> newMessage
|
|
||||||
| Some message -> $"%s{message}\n%s{newMessage}"
|
|
||||||
|
|
||||||
(stackTrace, Some message)
|
let results = results.Result |> Seq.concat |> List.ofSeq
|
||||||
)
|
|
||||||
|> fun (stackTrace, message) ->
|
|
||||||
{
|
|
||||||
StackTrace = stackTrace
|
|
||||||
Message = message
|
|
||||||
}
|
|
||||||
|> Some
|
|
||||||
| Choice3Of3 (UserMethodFailure.Threw (_, exc)) ->
|
|
||||||
{
|
|
||||||
StackTrace = (exc : Exception).ToString () |> Some
|
|
||||||
Message = None
|
|
||||||
}
|
|
||||||
|> Some
|
|
||||||
| Choice3Of3 (UserMethodFailure.ReturnedNonUnit (_, ret)) ->
|
|
||||||
{
|
|
||||||
Message = $"returned non-unit value %O{ret}" |> Some
|
|
||||||
StackTrace = None
|
|
||||||
}
|
|
||||||
|> Some
|
|
||||||
|
|
||||||
let outcome =
|
let report = BuildTrxReport.build assy creationTime startTime results
|
||||||
match cause with
|
|
||||||
| Choice1Of3 _ -> TrxTestOutcome.Failed
|
|
||||||
| Choice2Of3 TestMemberSuccess.Ok -> TrxTestOutcome.Passed
|
|
||||||
| Choice2Of3 (TestMemberSuccess.Inconclusive _) -> TrxTestOutcome.Inconclusive
|
|
||||||
| Choice2Of3 (TestMemberSuccess.Ignored _)
|
|
||||||
| Choice2Of3 (TestMemberSuccess.Explicit _) -> TrxTestOutcome.NotExecuted
|
|
||||||
// TODO: we can totally do better here, more fine-grained classification
|
|
||||||
| Choice3Of3 _ -> TrxTestOutcome.Failed
|
|
||||||
|
|
||||||
{
|
match args.Trx with
|
||||||
ExecutionId = i.ExecutionId
|
|
||||||
TestId = i.TestId
|
|
||||||
TestName = i.TestName
|
|
||||||
ComputerName = i.ComputerName
|
|
||||||
Duration = i.End - i.Start
|
|
||||||
StartTime = i.Start
|
|
||||||
EndTime = i.End
|
|
||||||
TestType = magicGuid
|
|
||||||
Outcome = outcome
|
|
||||||
TestListId = testListId
|
|
||||||
RelativeResultsDirectory = i.ExecutionId.ToString () // for some reason
|
|
||||||
Output =
|
|
||||||
match i.StdOut, i.StdErr, exc with
|
|
||||||
| None, None, None -> None
|
|
||||||
// TODO surely stderr can be emitted
|
|
||||||
| stdout, _stderr, exc ->
|
|
||||||
Some
|
|
||||||
{
|
|
||||||
TrxOutput.StdOut = stdout
|
|
||||||
ErrorInfo = exc
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
let report : TrxReport =
|
|
||||||
{
|
|
||||||
Id = Guid.NewGuid ()
|
|
||||||
Name = $"@%s{hostname} %s{finishTimeHumanReadable}"
|
|
||||||
Times = times
|
|
||||||
Settings = settings
|
|
||||||
Results = results
|
|
||||||
TestDefinitions = testDefinitions
|
|
||||||
TestEntries = testEntries
|
|
||||||
TestLists = [ testList ]
|
|
||||||
ResultsSummary = resultSummary
|
|
||||||
}
|
|
||||||
|
|
||||||
match trxPath with
|
|
||||||
| Some trxPath ->
|
| Some trxPath ->
|
||||||
let contents = TrxReport.toXml report |> fun d -> d.OuterXml
|
let contents = TrxReport.toXml report |> fun d -> d.OuterXml
|
||||||
trxPath.Directory.Create ()
|
trxPath.Directory.Create ()
|
||||||
@@ -404,10 +86,56 @@ module Program =
|
|||||||
Console.Error.WriteLine $"Written TRX file: %s{trxPath.FullName}"
|
Console.Error.WriteLine $"Written TRX file: %s{trxPath.FullName}"
|
||||||
| None -> ()
|
| None -> ()
|
||||||
|
|
||||||
match outcome with
|
match report.ResultsSummary.Outcome with
|
||||||
| TrxOutcome.Completed -> 0
|
| TrxOutcome.Completed -> 0
|
||||||
| _ -> 1
|
| _ -> 1
|
||||||
|
|
||||||
|
let main argv =
|
||||||
|
let args = argv |> List.ofArray |> Args.Parse
|
||||||
|
|
||||||
|
let psi = ProcessStartInfo "dotnet"
|
||||||
|
|
||||||
|
match args.Trx with
|
||||||
|
| None -> ()
|
||||||
|
| Some trx -> psi.EnvironmentVariables.Add ("WOOFWARE_NUNIT_GENERATE_TRX", trx.FullName)
|
||||||
|
|
||||||
|
match args.LevelOfParallelism with
|
||||||
|
| None -> ()
|
||||||
|
| Some par -> psi.EnvironmentVariables.Add ("WOOFWARE_NUNIT_PARALLELISM_LEVEL", string<int> par)
|
||||||
|
|
||||||
|
match args.Filter with
|
||||||
|
| None -> ()
|
||||||
|
| Some (filter, _) -> psi.EnvironmentVariables.Add ("WOOFWARE_NUNIT_FILTER", filter)
|
||||||
|
|
||||||
|
match args.Timeout with
|
||||||
|
| None -> ()
|
||||||
|
| Some timeout ->
|
||||||
|
psi.EnvironmentVariables.Add ("WOOFWARE_NUNIT_TIMEOUT_SECS", string<int> (int<float> timeout.TotalSeconds))
|
||||||
|
|
||||||
|
psi.ArgumentList.Add "exec"
|
||||||
|
psi.ArgumentList.Add args.Dll.FullName
|
||||||
|
|
||||||
|
let us = Assembly.GetExecutingAssembly().Location |> FileInfo
|
||||||
|
|
||||||
|
let startupHook =
|
||||||
|
Path.Combine (us.Directory.FullName, "WoofWare.NUnitTestRunner.StartupHook.dll")
|
||||||
|
|
||||||
|
psi.EnvironmentVariables.Add ("DOTNET_STARTUP_HOOKS", startupHook)
|
||||||
|
|
||||||
|
psi.EnvironmentVariables.Add (
|
||||||
|
"WOOFWARE_NUNIT_LIB",
|
||||||
|
Path.Combine (us.Directory.FullName, "WoofWare.NUnitTestRunner.Lib.dll")
|
||||||
|
)
|
||||||
|
|
||||||
|
use proc = new Process ()
|
||||||
|
proc.StartInfo <- psi
|
||||||
|
|
||||||
|
if not (proc.Start ()) then
|
||||||
|
failwith "Failed to start dotnet"
|
||||||
|
|
||||||
|
proc.WaitForExit ()
|
||||||
|
proc.ExitCode
|
||||||
|
|
||||||
[<EntryPoint>]
|
[<EntryPoint>]
|
||||||
let reallyMain argv =
|
let reallyMain argv =
|
||||||
// Hack to make sure `finally`s get run.
|
// Hack to make sure `finally`s get run.
|
||||||
|
@@ -4,21 +4,25 @@ open Spectre.Console
|
|||||||
|
|
||||||
[<RequireQualifiedAccess>]
|
[<RequireQualifiedAccess>]
|
||||||
module Progress =
|
module Progress =
|
||||||
let spectre () : ITestProgress =
|
let spectre (console : IAnsiConsole) : ITestProgress =
|
||||||
{ new ITestProgress with
|
{ new ITestProgress with
|
||||||
member _.OnTestFailed name failure =
|
member _.OnTestFailed name failure =
|
||||||
AnsiConsole.Console.MarkupLine
|
console.MarkupLine
|
||||||
$"[red]Test '%s{Markup.Escape name}' failed: %s{Markup.Escape (failure.ToString ())}[/]"
|
$"[red]Test '%s{Markup.Escape name}' failed: %s{Markup.Escape (failure.ToString ())}[/]"
|
||||||
|
|
||||||
member _.OnTestFixtureStart name testCount =
|
member _.OnTestFixtureStart name testCount =
|
||||||
AnsiConsole.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 =
|
||||||
AnsiConsole.Console.MarkupLine $"[gray]Finished test: %s{Markup.Escape name}[/]"
|
console.MarkupLine $"[gray]Finished test: %s{Markup.Escape name}[/]"
|
||||||
|
|
||||||
member _.OnTestMemberSkipped name =
|
member _.OnTestMemberSkipped name =
|
||||||
AnsiConsole.Console.MarkupLine $"[yellow]Skipping test due to filter: %s{Markup.Escape name}[/]"
|
console.MarkupLine $"[yellow]Skipping test due to filter: %s{Markup.Escape name}[/]"
|
||||||
|
|
||||||
member _.OnTestMemberStart name =
|
member _.OnTestMemberStart name =
|
||||||
AnsiConsole.Console.MarkupLine $"[white]Running test: %s{Markup.Escape name}[/]"
|
console.MarkupLine $"[white]Running test: %s{Markup.Escape name}[/]"
|
||||||
}
|
}
|
||||||
|
@@ -1,22 +0,0 @@
|
|||||||
namespace WoofWare.NUnitTestRunner
|
|
||||||
|
|
||||||
[<RequireQualifiedAccess>]
|
|
||||||
module internal Seq =
|
|
||||||
|
|
||||||
let tryMinBy (f : 'a -> 'b) (s : 'a seq) : 'a option =
|
|
||||||
use enum = s.GetEnumerator ()
|
|
||||||
|
|
||||||
if not (enum.MoveNext ()) then
|
|
||||||
None
|
|
||||||
else
|
|
||||||
|
|
||||||
let mutable answer = enum.Current
|
|
||||||
let mutable fAnswer = f enum.Current
|
|
||||||
|
|
||||||
while enum.MoveNext () do
|
|
||||||
let fNext = f enum.Current
|
|
||||||
|
|
||||||
if fNext < fAnswer then
|
|
||||||
answer <- enum.Current
|
|
||||||
|
|
||||||
Some answer
|
|
@@ -11,6 +11,8 @@ module TestList =
|
|||||||
[<Test>]
|
[<Test>]
|
||||||
let ``combinations has right size`` () =
|
let ``combinations has right size`` () =
|
||||||
let property (xs : int list list) =
|
let property (xs : int list list) =
|
||||||
|
let xs = if xs.Length > 6 then xs |> List.take 6 else xs
|
||||||
|
let xs = xs |> List.map (fun xs -> if xs.Length > 6 then xs |> List.take 6 else xs)
|
||||||
let combs = List.combinations xs
|
let combs = List.combinations xs
|
||||||
|
|
||||||
combs.Length
|
combs.Length
|
||||||
|
@@ -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
|
||||||
|
}
|
@@ -197,6 +197,7 @@ Running all tests in /Users/patrick/Documents/GitHub/TestRunner/TestRunner/TestR
|
|||||||
Ensure version is monotonic: Not yet published
|
Ensure version is monotonic: Not yet published
|
||||||
NUnit Adapter 4.5.0.0: Test execution complete
|
NUnit Adapter 4.5.0.0: Test execution complete
|
||||||
"""
|
"""
|
||||||
|
StdErr = None
|
||||||
ErrorInfo = None
|
ErrorInfo = None
|
||||||
}
|
}
|
||||||
Outcome = TrxOutcome.Failed
|
Outcome = TrxOutcome.Failed
|
||||||
|
@@ -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.0.41" />
|
<PackageReference Include="ApiSurface" Version="5.0.2" />
|
||||||
<PackageReference Include="FsCheck" Version="3.0.0-rc3" />
|
<PackageReference Include="FsCheck" Version="3.3.1" />
|
||||||
<PackageReference Include="FsUnit" Version="6.0.0" />
|
<PackageReference Include="FsUnit" Version="7.1.1" />
|
||||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.10.0" />
|
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="18.0.0" />
|
||||||
<PackageReference Include="NUnit" Version="4.1.0" />
|
<PackageReference Include="NUnit" Version="4.3.2" />
|
||||||
<PackageReference Include="NUnit3TestAdapter" Version="4.5.0"/>
|
<PackageReference Include="NUnit3TestAdapter" Version="5.2.0"/>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
@@ -2,7 +2,8 @@
|
|||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
|
<RollForward>Major</RollForward>
|
||||||
<PackAsTool>true</PackAsTool>
|
<PackAsTool>true</PackAsTool>
|
||||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
||||||
<Authors>Patrick Stevens</Authors>
|
<Authors>Patrick Stevens</Authors>
|
||||||
@@ -16,15 +17,9 @@
|
|||||||
<PackageId>WoofWare.NUnitTestRunner</PackageId>
|
<PackageId>WoofWare.NUnitTestRunner</PackageId>
|
||||||
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
|
||||||
<WarnOn>FS3559</WarnOn>
|
<WarnOn>FS3559</WarnOn>
|
||||||
<WoofWareMyriadPluginVersion>2.1.42</WoofWareMyriadPluginVersion>
|
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="RuntimeConfig.fs" />
|
|
||||||
<Compile Include="GeneratedRuntimeConfig.fs">
|
|
||||||
<MyriadFile>RuntimeConfig.fs</MyriadFile>
|
|
||||||
</Compile>
|
|
||||||
<Compile Include="Seq.fs" />
|
|
||||||
<Compile Include="Progress.fs" />
|
<Compile Include="Progress.fs" />
|
||||||
<Compile Include="Program.fs" />
|
<Compile Include="Program.fs" />
|
||||||
<None Include="..\README.md">
|
<None Include="..\README.md">
|
||||||
@@ -35,18 +30,12 @@
|
|||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\WoofWare.NUnitTestRunner.Lib\WoofWare.NUnitTestRunner.Lib.fsproj" />
|
<ProjectReference Include="..\WoofWare.NUnitTestRunner.Lib\WoofWare.NUnitTestRunner.Lib.fsproj" />
|
||||||
|
<ProjectReference Include="..\WoofWare.NUnitTestRunner.StartupHookLogic\WoofWare.NUnitTestRunner.StartupHookLogic.csproj" />
|
||||||
|
<ProjectReference Include="..\WoofWare.NUnitTestRunner.StartupHook\WoofWare.NUnitTestRunner.StartupHook.csproj" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Spectre.Console" Version="0.49.1" />
|
<PackageReference Include="Spectre.Console" Version="0.52.0" />
|
||||||
<PackageReference Include="WoofWare.DotnetRuntimeLocator" Version="0.1.4" />
|
|
||||||
<PackageReference Include="WoofWare.Myriad.Plugins.Attributes" Version="3.1.6" />
|
|
||||||
<PackageReference Include="Myriad.SDK" Version="0.8.3" />
|
|
||||||
<PackageReference Include="WoofWare.Myriad.Plugins" Version="$(WoofWareMyriadPluginVersion)" PrivateAssets="all" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<MyriadSdkGenerator Include="$(NuGetPackageRoot)/woofware.myriad.plugins/$(WoofWareMyriadPluginVersion)/lib/net6.0/WoofWare.Myriad.Plugins.dll" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
@@ -1,10 +1,11 @@
|
|||||||
{
|
{
|
||||||
"version": "0.2",
|
"version": "0.3",
|
||||||
"publicReleaseRefSpec": [
|
"publicReleaseRefSpec": [
|
||||||
"^refs/heads/main$"
|
"^refs/heads/main$"
|
||||||
],
|
],
|
||||||
"pathFilters": [
|
"pathFilters": [
|
||||||
"./",
|
"./",
|
||||||
|
":^WoofWare.NUnitTestRunner.Test",
|
||||||
":/WoofWare.NUnitTestRunner.Lib",
|
":/WoofWare.NUnitTestRunner.Lib",
|
||||||
":/Directory.Build.props",
|
":/Directory.Build.props",
|
||||||
":/README.md"
|
":/README.md"
|
||||||
|
@@ -4,13 +4,13 @@
|
|||||||
<IsPackable>false</IsPackable>
|
<IsPackable>false</IsPackable>
|
||||||
<IsPublishable>false</IsPublishable>
|
<IsPublishable>false</IsPublishable>
|
||||||
<RestorePackagesPath>../.analyzerpackages/</RestorePackagesPath>
|
<RestorePackagesPath>../.analyzerpackages/</RestorePackagesPath>
|
||||||
<TargetFramework>net6.0</TargetFramework>
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
<DisableImplicitNuGetFallbackFolder>true</DisableImplicitNuGetFallbackFolder>
|
<DisableImplicitNuGetFallbackFolder>true</DisableImplicitNuGetFallbackFolder>
|
||||||
<AutomaticallyUseReferenceAssemblyPackages>false</AutomaticallyUseReferenceAssemblyPackages> <!-- We don't want to build this project, so we do not need the reference assemblies for the framework we chose.-->
|
<AutomaticallyUseReferenceAssemblyPackages>false</AutomaticallyUseReferenceAssemblyPackages> <!-- We don't want to build this project, so we do not need the reference assemblies for the framework we chose.-->
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageDownload Include="G-Research.FSharp.Analyzers" Version="[0.10.0]" />
|
<PackageDownload Include="G-Research.FSharp.Analyzers" Version="[0.19.0]" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
12
flake.lock
generated
12
flake.lock
generated
@@ -5,11 +5,11 @@
|
|||||||
"systems": "systems"
|
"systems": "systems"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1710146030,
|
"lastModified": 1731533236,
|
||||||
"narHash": "sha256-SZ5L6eA7HJ/nmkzGG7/ISclqe6oZdOZTNoesiInkXPQ=",
|
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
|
||||||
"owner": "numtide",
|
"owner": "numtide",
|
||||||
"repo": "flake-utils",
|
"repo": "flake-utils",
|
||||||
"rev": "b1d9ab70662946ef0850d488da1c9019f3a9752a",
|
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -20,11 +20,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1717399147,
|
"lastModified": 1760596604,
|
||||||
"narHash": "sha256-eCWaE/q1VItpFAxxLVt171MdtDcjEnwi6QB/yuF73JU=",
|
"narHash": "sha256-J/i5K6AAz/y5dBePHQOuzC7MbhyTOKsd/GLezSbEFiM=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "4a4ecb0ab415c9fccfb005567a215e6a9564cdf5",
|
"rev": "3cbe716e2346710d6e1f7c559363d14e11c32a43",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
37
flake.nix
37
flake.nix
@@ -14,10 +14,10 @@
|
|||||||
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.dotnet-sdk_8;
|
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: sha256:
|
dotnetTool = dllOverride: toolName: toolVersion: hash:
|
||||||
pkgs.stdenvNoCC.mkDerivation rec {
|
pkgs.stdenvNoCC.mkDerivation rec {
|
||||||
name = toolName;
|
name = toolName;
|
||||||
version = toolVersion;
|
version = toolVersion;
|
||||||
@@ -25,8 +25,8 @@
|
|||||||
src = pkgs.fetchNuGet {
|
src = pkgs.fetchNuGet {
|
||||||
pname = name;
|
pname = name;
|
||||||
version = version;
|
version = version;
|
||||||
sha256 = sha256;
|
hash = hash;
|
||||||
installPhase = ''mkdir -p $out/bin && cp -r tools/net6.0/any/* $out/bin'';
|
installPhase = ''mkdir -p $out/bin && cp -r tools/net*/any/* $out/bin'';
|
||||||
};
|
};
|
||||||
installPhase = let
|
installPhase = let
|
||||||
dll =
|
dll =
|
||||||
@@ -42,9 +42,11 @@
|
|||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
in {
|
in {
|
||||||
packages = {
|
packages = let
|
||||||
fantomas = dotnetTool null "fantomas" (builtins.fromJSON (builtins.readFile ./.config/dotnet-tools.json)).tools.fantomas.version (builtins.head (builtins.filter (elem: elem.pname == "fantomas") ((import ./nix/deps.nix) {fetchNuGet = x: x;}))).sha256;
|
deps = builtins.fromJSON (builtins.readFile ./nix/deps.json);
|
||||||
fsharp-analyzers = dotnetTool "FSharp.Analyzers.Cli" "fsharp-analyzers" (builtins.fromJSON (builtins.readFile ./.config/dotnet-tools.json)).tools.fsharp-analyzers.version (builtins.head (builtins.filter (elem: elem.pname == "fsharp-analyzers") ((import ./nix/deps.nix) {fetchNuGet = x: x;}))).sha256;
|
in {
|
||||||
|
fantomas = dotnetTool null "fantomas" (builtins.fromJSON (builtins.readFile ./.config/dotnet-tools.json)).tools.fantomas.version (builtins.head (builtins.filter (elem: elem.pname == "fantomas") deps)).hash;
|
||||||
|
fsharp-analyzers = dotnetTool "FSharp.Analyzers.Cli" "fsharp-analyzers" (builtins.fromJSON (builtins.readFile ./.config/dotnet-tools.json)).tools.fsharp-analyzers.version (builtins.head (builtins.filter (elem: elem.pname == "fsharp-analyzers") deps)).hash;
|
||||||
default = pkgs.buildDotnetModule {
|
default = pkgs.buildDotnetModule {
|
||||||
inherit pname version dotnet-sdk dotnet-runtime;
|
inherit pname version dotnet-sdk dotnet-runtime;
|
||||||
name = "unofficial-nunit-runner";
|
name = "unofficial-nunit-runner";
|
||||||
@@ -52,17 +54,20 @@
|
|||||||
projectFile = "./WoofWare.NUnitTestRunner/WoofWare.NUnitTestRunner.fsproj";
|
projectFile = "./WoofWare.NUnitTestRunner/WoofWare.NUnitTestRunner.fsproj";
|
||||||
testProjectFile = "./WoofWare.NUnitTestRunner/WoofWare.NUnitTestRunner.Test/WoofWare.NUnitTestRunner.Test.fsproj";
|
testProjectFile = "./WoofWare.NUnitTestRunner/WoofWare.NUnitTestRunner.Test/WoofWare.NUnitTestRunner.Test.fsproj";
|
||||||
disabledTests = ["WoofWare.NUnitTestRunner.Test.TestSurface.EnsureVersionIsMonotonic"];
|
disabledTests = ["WoofWare.NUnitTestRunner.Test.TestSurface.EnsureVersionIsMonotonic"];
|
||||||
nugetDeps = ./nix/deps.nix; # `nix build .#default.passthru.fetch-deps && ./result` and put the result here
|
nugetDeps = ./nix/deps.json; # `nix build .#default.fetch-deps && ./result nix/deps.json`
|
||||||
doCheck = true;
|
doCheck = true;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
devShell = pkgs.mkShell {
|
devShells = {
|
||||||
buildInputs = [dotnet-sdk];
|
default = pkgs.mkShell {
|
||||||
packages = [
|
packages = [
|
||||||
pkgs.alejandra
|
dotnet-sdk
|
||||||
pkgs.nodePackages.markdown-link-check
|
pkgs.alejandra
|
||||||
pkgs.shellcheck
|
pkgs.nodePackages.markdown-link-check
|
||||||
];
|
pkgs.shellcheck
|
||||||
|
pkgs.xmlstarlet
|
||||||
|
];
|
||||||
|
};
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
372
nix/deps.json
Normal file
372
nix/deps.json
Normal file
@@ -0,0 +1,372 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"pname": "ApiSurface",
|
||||||
|
"version": "5.0.2",
|
||||||
|
"hash": "sha256-zcq1H1ccQzsZQf4kolzoOBSbyz07skihgPAvQ9Jri+E="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "fantomas",
|
||||||
|
"version": "7.0.3",
|
||||||
|
"hash": "sha256-0XlfV7SxXPDnk/CjkUesJSaH0cxlNHJ+Jj86zNUhkNA="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Fantomas.Core",
|
||||||
|
"version": "6.1.1",
|
||||||
|
"hash": "sha256-FcTLHQFvKkQY/kV08jhhy/St/+FmXpp3epp/R3zUXMA="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Fantomas.FCS",
|
||||||
|
"version": "6.1.1",
|
||||||
|
"hash": "sha256-NuZ8msPEHYA8T3EYREB28F1RcNgUU8V54eg2+UttYxw="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "FsCheck",
|
||||||
|
"version": "3.3.1",
|
||||||
|
"hash": "sha256-k65ksdOSOGz+meRUUND+yuqJtm5ChaKuaxmRIdKzx2Y="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "fsharp-analyzers",
|
||||||
|
"version": "0.33.1",
|
||||||
|
"hash": "sha256-vYXvqnf3en487svFv3CmNl24SolwMYzu6zKKGXNxSu8="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "FSharp.Core",
|
||||||
|
"version": "4.3.4",
|
||||||
|
"hash": "sha256-styyo+6mJy+yxE0NZG/b1hxkAjPOnJfMgd9zWzCJ5uk="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "FSharp.Core",
|
||||||
|
"version": "6.0.1",
|
||||||
|
"hash": "sha256-Ehsgt3nCJijpaVuJguC1TPVEKSkJd6PSc07D2ZQSemI="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "FSharp.Core",
|
||||||
|
"version": "9.0.303",
|
||||||
|
"hash": "sha256-AxR6wqodeU23KOTgkUfIgbavgbcSuzD4UBP+tiFydgA="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "FsUnit",
|
||||||
|
"version": "7.1.1",
|
||||||
|
"hash": "sha256-UMCEGKxQ4ytjmPuVpiNaAPbi3RQH9gqa61JJIUS/6hg="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.ApplicationInsights",
|
||||||
|
"version": "2.23.0",
|
||||||
|
"hash": "sha256-5sf3bg7CZZjHseK+F3foOchEhmVeioePxMZVvS6Rjb0="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.AspNetCore.App.Ref",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-9jDkWbjw/nd8yqdzVTagCuqr6owJ/DUMi4BlUZT4hWU="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.AspNetCore.App.Runtime.linux-arm64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-JQULJyF0ivLoUU1JaFfK/HHg+/qzpN7V2RR2Cc+WlQ4="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.AspNetCore.App.Runtime.linux-x64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-zUsVIpV481vMLAXaLEEUpEMA9/f1HGOnvaQnaWdzlyY="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.AspNetCore.App.Runtime.osx-arm64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-2seqZcz0JeUjkzh3QcGa9TcJ4LUafpFjTRk+Nm8T6T0="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.AspNetCore.App.Runtime.osx-x64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-yxLafxiBKkvfkDggPk0P9YZIHBkDJOsFTO7/V9mEHuU="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.CodeCoverage",
|
||||||
|
"version": "18.0.0",
|
||||||
|
"hash": "sha256-1RNxheCYASMusDC48BXtaO3MhnInw15JVfjfLM1VMGA="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NET.Test.Sdk",
|
||||||
|
"version": "18.0.0",
|
||||||
|
"hash": "sha256-9iW+9mvMeZgDXwSoR08bnvRNsN4jT8OVWcjq3lcE+cs="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Host.linux-arm64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-9lC/LYnthYhjkWWz2kkFCvlA5LJOv11jdt59SDnpdy0="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Host.linux-x64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-VFRDzx7LJuvI5yzKdGmw/31NYVbwHWPKQvueQt5xc10="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Host.osx-arm64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-DaSWwYACJGolEBuMhzDVCj/rQTdDt061xCVi+gyQnuo="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Host.osx-x64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-FrRny9EI6HKCKQbu6mcLj5w4ooSRrODD4Vj2ZMGnMd4="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Ref",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-9LZgVoIFF8qNyUu8kdJrYGLutMF/cL2K82HN2ywwlx8="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Runtime.linux-arm64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-k3rxvUhCEU0pVH8KgEMtkPiSOibn+nBh+0zT2xIfId8="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Runtime.linux-x64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-U8wJ2snSDFqeAgDVLXjnniidC7Cr5aJ1/h/BMSlyu0c="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Runtime.osx-arm64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-UfLcrL2Gj/OLz0s92Oo+OCJeDpZFAcQLPLiSNND8D5Y="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.App.Runtime.osx-x64",
|
||||||
|
"version": "6.0.36",
|
||||||
|
"hash": "sha256-0xIJYFzxdMcnCj3wzkFRQZSnQcPHzPHMzePRIOA3oJs="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.Platforms",
|
||||||
|
"version": "1.1.0",
|
||||||
|
"hash": "sha256-FeM40ktcObQJk4nMYShB61H/E8B7tIKfl9ObJ0IOcCM="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.Platforms",
|
||||||
|
"version": "1.1.1",
|
||||||
|
"hash": "sha256-8hLiUKvy/YirCWlFwzdejD2Db3DaXhHxT7GSZx/znJg="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.Targets",
|
||||||
|
"version": "1.1.0",
|
||||||
|
"hash": "sha256-0AqQ2gMS8iNlYkrD+BxtIg7cXMnr9xZHtKAuN4bjfaQ="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.NETCore.Targets",
|
||||||
|
"version": "1.1.3",
|
||||||
|
"hash": "sha256-WLsf1NuUfRWyr7C7Rl9jiua9jximnVvzy6nk2D2bVRc="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.Testing.Extensions.Telemetry",
|
||||||
|
"version": "1.9.0",
|
||||||
|
"hash": "sha256-JT91ThKLEyoRS/8ZJqZwlSTT7ofC2QhNqPFI3pYmMaw="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.Testing.Extensions.TrxReport.Abstractions",
|
||||||
|
"version": "1.9.0",
|
||||||
|
"hash": "sha256-oscZOEKw7gM6eRdDrOS3x+CwqIvXWRmfmi0ugCxBRw0="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.Testing.Extensions.VSTestBridge",
|
||||||
|
"version": "1.9.0",
|
||||||
|
"hash": "sha256-CadXLWD093sUDaWhnppzD9LvpxSRqqt93ZEOFiIAPyw="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.Testing.Platform",
|
||||||
|
"version": "1.9.0",
|
||||||
|
"hash": "sha256-6nzjoYbJOh7v/GB7d+TDuM0l/xglCshFX6KWjg7+cFI="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.Testing.Platform.MSBuild",
|
||||||
|
"version": "1.9.0",
|
||||||
|
"hash": "sha256-/bileP4b+9RZp8yjgS6eynXwc2mohyyzf6p/0LZJd8I="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.TestPlatform.AdapterUtilities",
|
||||||
|
"version": "17.13.0",
|
||||||
|
"hash": "sha256-Vr+3Tad/h/nk7f/5HMExn3HvCGFCarehFAzJSfCBaOc="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.TestPlatform.ObjectModel",
|
||||||
|
"version": "17.13.0",
|
||||||
|
"hash": "sha256-6S0fjfj8vA+h6dJVNwLi6oZhYDO/I/6hBZaq2VTW+Uk="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.TestPlatform.ObjectModel",
|
||||||
|
"version": "18.0.0",
|
||||||
|
"hash": "sha256-O/ivHdoIO+q1n0byJ9OZO4quOqACOD3K3Qm00wfhuZk="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Microsoft.TestPlatform.TestHost",
|
||||||
|
"version": "18.0.0",
|
||||||
|
"hash": "sha256-qAIX2Rqxrnh1xaYRjCbkkvvMm407oyKihqyVjURX5wY="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Myriad.Core",
|
||||||
|
"version": "0.8.3",
|
||||||
|
"hash": "sha256-vBOxfq8QriX/yUtaXN69rEQaY/psRNJWxqATLidrt2g="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Myriad.Sdk",
|
||||||
|
"version": "0.8.3",
|
||||||
|
"hash": "sha256-7O397WKhskKOvE3MkJT37BvxorDWngDR6gTUogtDZ2M="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Nerdbank.GitVersioning",
|
||||||
|
"version": "3.8.118",
|
||||||
|
"hash": "sha256-Hmyy0ZKOmwN4zIhI4+MqoN8geZNc1sd033aZJ6APrO8="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Newtonsoft.Json",
|
||||||
|
"version": "13.0.3",
|
||||||
|
"hash": "sha256-hy/BieY4qxBWVVsDqqOPaLy1QobiIapkbrESm6v2PHc="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NuGet.Common",
|
||||||
|
"version": "6.14.0",
|
||||||
|
"hash": "sha256-jDOwt3veI1GSG8CfBnf2+dJxD3E/Nmlc+vHtD4J76Ms="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NuGet.Configuration",
|
||||||
|
"version": "6.14.0",
|
||||||
|
"hash": "sha256-1PN9s6fhCw3wd2260U6hQ4vG3jIvcG8GIn1oQgxMXA0="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NuGet.Frameworks",
|
||||||
|
"version": "6.14.0",
|
||||||
|
"hash": "sha256-3ViM3R1ucQMEL2hQYsivT86kI6veMQK2xDsiAcFcVQk="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NuGet.Packaging",
|
||||||
|
"version": "6.14.0",
|
||||||
|
"hash": "sha256-Yafbnxs3maj55bJ1oKQiZ0QkntFUzXdhorL94YEUOhY="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NuGet.Protocol",
|
||||||
|
"version": "6.14.0",
|
||||||
|
"hash": "sha256-uLDKfs+QN1MdnuQtTES8qfNzzsmYKM6XB9pwJc4G+eo="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NuGet.Versioning",
|
||||||
|
"version": "6.14.0",
|
||||||
|
"hash": "sha256-DqdOJgsphKxSvqB8b60zNPCaiLfbiu3WnUJ/90feLrY="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NUnit",
|
||||||
|
"version": "4.3.2",
|
||||||
|
"hash": "sha256-0RWe8uFoxYp6qhPlDDEghOMcKJgyw2ybvEoAqBLebeE="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "NUnit3TestAdapter",
|
||||||
|
"version": "5.2.0",
|
||||||
|
"hash": "sha256-ybTutL4VkX/fq61mS+O3Ruh+adic4fpv+MKgQ0IZvGg="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "runtime.any.System.Runtime",
|
||||||
|
"version": "4.3.0",
|
||||||
|
"hash": "sha256-qwhNXBaJ1DtDkuRacgHwnZmOZ1u9q7N8j0cWOLYOELM="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "runtime.native.System",
|
||||||
|
"version": "4.3.0",
|
||||||
|
"hash": "sha256-ZBZaodnjvLXATWpXXakFgcy6P+gjhshFXmglrL5xD5Y="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "runtime.unix.System.Private.Uri",
|
||||||
|
"version": "4.3.0",
|
||||||
|
"hash": "sha256-c5tXWhE/fYbJVl9rXs0uHh3pTsg44YD1dJvyOA0WoMs="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "Spectre.Console",
|
||||||
|
"version": "0.52.0",
|
||||||
|
"hash": "sha256-enGa3do7uHQFJOGha+IJZB/rlYhZDvLYbNYgZ4B5V8g="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Collections.Immutable",
|
||||||
|
"version": "8.0.0",
|
||||||
|
"hash": "sha256-F7OVjKNwpqbUh8lTidbqJWYi476nsq9n+6k0+QVRo3w="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Diagnostics.DiagnosticSource",
|
||||||
|
"version": "5.0.0",
|
||||||
|
"hash": "sha256-6mW3N6FvcdNH/pB58pl+pFSCGWgyaP4hfVtC/SMWDV4="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Diagnostics.DiagnosticSource",
|
||||||
|
"version": "7.0.0",
|
||||||
|
"hash": "sha256-9Wk8cHSkjKtqkN6xW7KnXoQVtF/VNbKeBq79WqDesMs="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Formats.Asn1",
|
||||||
|
"version": "6.0.0",
|
||||||
|
"hash": "sha256-KaMHgIRBF7Nf3VwOo+gJS1DcD+41cJDPWFh+TDQ8ee8="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Memory",
|
||||||
|
"version": "4.5.5",
|
||||||
|
"hash": "sha256-EPQ9o1Kin7KzGI5O3U3PUQAZTItSbk9h/i4rViN3WiI="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Private.Uri",
|
||||||
|
"version": "4.3.0",
|
||||||
|
"hash": "sha256-fVfgcoP4AVN1E5wHZbKBIOPYZ/xBeSIdsNF+bdukIRM="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Reflection.Metadata",
|
||||||
|
"version": "8.0.0",
|
||||||
|
"hash": "sha256-dQGC30JauIDWNWXMrSNOJncVa1umR1sijazYwUDdSIE="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Runtime",
|
||||||
|
"version": "4.3.1",
|
||||||
|
"hash": "sha256-R9T68AzS1PJJ7v6ARz9vo88pKL1dWqLOANg4pkQjkA0="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Runtime.CompilerServices.Unsafe",
|
||||||
|
"version": "6.0.0",
|
||||||
|
"hash": "sha256-bEG1PnDp7uKYz/OgLOWs3RWwQSVYm+AnPwVmAmcgp2I="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Security.Cryptography.Pkcs",
|
||||||
|
"version": "6.0.4",
|
||||||
|
"hash": "sha256-2e0aRybote+OR66bHaNiYpF//4fCiaO3zbR2e9GABUI="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Security.Cryptography.ProtectedData",
|
||||||
|
"version": "4.4.0",
|
||||||
|
"hash": "sha256-Ri53QmFX8I8UH0x4PikQ1ZA07ZSnBUXStd5rBfGWFOE="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "System.Text.Json",
|
||||||
|
"version": "8.0.5",
|
||||||
|
"hash": "sha256-yKxo54w5odWT6nPruUVsaX53oPRe+gKzGvLnnxtwP68="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "TypeEquality",
|
||||||
|
"version": "0.4.2",
|
||||||
|
"hash": "sha256-YxK6BGHjcuP76j5BbTDi814jxGqOevQSgS00IJcjZSA="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "WoofWare.DotnetRuntimeLocator",
|
||||||
|
"version": "0.1.12",
|
||||||
|
"hash": "sha256-6pNZs0/R2LnLKSODq9DyHhGo2C+SDyz9k7D/13/78so="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "WoofWare.Myriad.Plugins",
|
||||||
|
"version": "9.0.4",
|
||||||
|
"hash": "sha256-fVahNM2SOvG159Wz6+uBkrl3+jqVtRUhZsZ2Kl2VCfk="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "WoofWare.Myriad.Plugins.Attributes",
|
||||||
|
"version": "3.7.3",
|
||||||
|
"hash": "sha256-scdokAtktZZ6K8c/eXm2DKtPzQPZrJLJ0cnu652uYuY="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "WoofWare.PrattParser",
|
||||||
|
"version": "0.2.5",
|
||||||
|
"hash": "sha256-6+74AMxVIBa5rYO34Hlm02zPtRSvpcvUA6cqeYB3WoQ="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pname": "WoofWare.Whippet.Fantomas",
|
||||||
|
"version": "0.6.4",
|
||||||
|
"hash": "sha256-ScZ7EEcxLvXyam2ZVqDK58QlK3RcePWghzRvtLLLdZI="
|
||||||
|
}
|
||||||
|
]
|
204
nix/deps.nix
204
nix/deps.nix
@@ -1,204 +0,0 @@
|
|||||||
# This file was automatically generated by passthru.fetch-deps.
|
|
||||||
# Please dont edit it manually, your changes might get overwritten!
|
|
||||||
{fetchNuGet}: [
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "ApiSurface";
|
|
||||||
version = "4.0.41";
|
|
||||||
sha256 = "03kfa5ngmgkik9lc58sp8s9rrh9g40hhgjnrv662ks0d0y2i9i89";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "fantomas";
|
|
||||||
version = "6.3.9";
|
|
||||||
sha256 = "1b34iiiff02bbzjv03zyna8xmrgs6y87zdvp5i5k58fcqpjw44sx";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "FsCheck";
|
|
||||||
version = "3.0.0-rc3";
|
|
||||||
sha256 = "1rn4x9qh479927viwww3dy0mikcdcq3pfqv1hzbbawnwxfzm17z1";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "fsharp-analyzers";
|
|
||||||
version = "0.26.0";
|
|
||||||
sha256 = "0xgv5kvbwfdvcp6s8x7xagbbi4s3mqa4ixni6pazqvyflbgnah7b";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "FSharp.Core";
|
|
||||||
version = "6.0.0";
|
|
||||||
sha256 = "1hjhvr39c1vpgrdmf8xln5q86424fqkvy9nirkr29vl2461d2039";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "FSharp.Core";
|
|
||||||
version = "8.0.300";
|
|
||||||
sha256 = "158xxr9hnhz2ibyzzp2d249angvxfc58ifflm4g3hz8qx9zxaq04";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "FsUnit";
|
|
||||||
version = "6.0.0";
|
|
||||||
sha256 = "18q3p0z155znwj1l0qq3vq9nh9wl2i4mlfx4pmrnia4czr0xdkmb";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Microsoft.CodeCoverage";
|
|
||||||
version = "17.10.0";
|
|
||||||
sha256 = "0s0v7jmrq85n356xv7zixvwa4z94fszjcr5vll8x4im1a2lp00f9";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Microsoft.NET.Test.Sdk";
|
|
||||||
version = "17.10.0";
|
|
||||||
sha256 = "13g8fwl09li8fc71nk13dgkb7gahd4qhamyg2xby7am63nlchhdf";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Microsoft.NETCore.Platforms";
|
|
||||||
version = "2.0.0";
|
|
||||||
sha256 = "1fk2fk2639i7nzy58m9dvpdnzql4vb8yl8vr19r2fp8lmj9w2jr0";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Microsoft.TestPlatform.ObjectModel";
|
|
||||||
version = "17.10.0";
|
|
||||||
sha256 = "07j69cw8r39533w4p39mnj00kahazz38760in3jfc45kmlcdb26x";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Microsoft.TestPlatform.TestHost";
|
|
||||||
version = "17.10.0";
|
|
||||||
sha256 = "1bl471s7fx9jycr0cc8rylwf34mrvlg9qn1an6l86nisavfcyb7v";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Myriad.Sdk";
|
|
||||||
version = "0.8.3";
|
|
||||||
sha256 = "0qv78c5s5m04xb8h17nnn2ig26zcyya91k2dpj745cm1cbnzvvgc";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Nerdbank.GitVersioning";
|
|
||||||
version = "3.6.139";
|
|
||||||
sha256 = "0npcryhq3r0c2zi940jk39h13mzc4hyg7z8gm6jdmxi1aqv1vh8c";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NETStandard.Library.Ref";
|
|
||||||
version = "2.1.0";
|
|
||||||
sha256 = "12n76gymxq715lkrw841vi5r84kx746cxxssp22pd08as75jzsj6";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Newtonsoft.Json";
|
|
||||||
version = "13.0.1";
|
|
||||||
sha256 = "0fijg0w6iwap8gvzyjnndds0q4b8anwxxvik7y8vgq97dram4srb";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Newtonsoft.Json";
|
|
||||||
version = "13.0.3";
|
|
||||||
sha256 = "0xrwysmrn4midrjal8g2hr1bbg38iyisl0svamb11arqws4w2bw7";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NuGet.Common";
|
|
||||||
version = "6.10.0";
|
|
||||||
sha256 = "0nizrnilmlcqbm945293h8q3wfqfchb4xi8g50x4kjn0rbpd1kbh";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NuGet.Configuration";
|
|
||||||
version = "6.10.0";
|
|
||||||
sha256 = "1aqaknaawnqx4mnvx9qw73wvj48jjzv0d78dzwl7m9zjlrl9myhz";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NuGet.Frameworks";
|
|
||||||
version = "6.10.0";
|
|
||||||
sha256 = "0hrd8y31zx9a0wps49czw0qgbrakb49zn3abfgylc9xrq990zkqk";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NuGet.Packaging";
|
|
||||||
version = "6.10.0";
|
|
||||||
sha256 = "18s53cvrf51lihmaqqdf48p2qi6ky1l48jv0hvbp76cxwdg7rba4";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NuGet.Protocol";
|
|
||||||
version = "6.10.0";
|
|
||||||
sha256 = "0hmv4q0ks9i34mfgpb13l01la9v3jjllfh1qd3aqv105xrqrdxac";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NuGet.Versioning";
|
|
||||||
version = "6.10.0";
|
|
||||||
sha256 = "1x19njx4x0sw9fz8y5fibi15xfsrw5avir0cx0599yd7p3ykik5g";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NUnit";
|
|
||||||
version = "4.1.0";
|
|
||||||
sha256 = "0fj6xwgqaxq3mrai86bklclfmjkzf038mrslwfqf4ignaz9f7g5j";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "NUnit3TestAdapter";
|
|
||||||
version = "4.5.0";
|
|
||||||
sha256 = "1srx1629s0k1kmf02nmz251q07vj6pv58mdafcr5dr0bbn1fh78i";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "Spectre.Console";
|
|
||||||
version = "0.49.1";
|
|
||||||
sha256 = "0fhl96p3xjd5k1wwvhs80cp35rrlgnza6mw9vy0knhmf7ji9b95n";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Formats.Asn1";
|
|
||||||
version = "6.0.0";
|
|
||||||
sha256 = "1vvr7hs4qzjqb37r0w1mxq7xql2b17la63jwvmgv65s1hj00g8r9";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.IO.Abstractions";
|
|
||||||
version = "4.2.13";
|
|
||||||
sha256 = "0s784iphsmj4vhkrzq9q3w39vsn76w44zclx3hsygsw458zbyh4y";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.IO.FileSystem.AccessControl";
|
|
||||||
version = "4.5.0";
|
|
||||||
sha256 = "1gq4s8w7ds1sp8f9wqzf8nrzal40q5cd2w4pkf4fscrl2ih3hkkj";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Reflection.Metadata";
|
|
||||||
version = "1.6.0";
|
|
||||||
sha256 = "1wdbavrrkajy7qbdblpbpbalbdl48q3h34cchz24gvdgyrlf15r4";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Security.AccessControl";
|
|
||||||
version = "4.5.0";
|
|
||||||
sha256 = "1wvwanz33fzzbnd2jalar0p0z3x0ba53vzx1kazlskp7pwyhlnq0";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Security.Cryptography.Pkcs";
|
|
||||||
version = "6.0.4";
|
|
||||||
sha256 = "0hh5h38pnxmlrnvs72f2hzzpz4b2caiiv6xf8y7fzdg84r3imvfr";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Security.Cryptography.ProtectedData";
|
|
||||||
version = "4.4.0";
|
|
||||||
sha256 = "1q8ljvqhasyynp94a1d7jknk946m20lkwy2c3wa8zw2pc517fbj6";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Security.Principal.Windows";
|
|
||||||
version = "4.5.0";
|
|
||||||
sha256 = "0rmj89wsl5yzwh0kqjgx45vzf694v9p92r4x4q6yxldk1cv1hi86";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Text.Encodings.Web";
|
|
||||||
version = "7.0.0";
|
|
||||||
sha256 = "1151hbyrcf8kyg1jz8k9awpbic98lwz9x129rg7zk1wrs6vjlpxl";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "System.Text.Json";
|
|
||||||
version = "7.0.3";
|
|
||||||
sha256 = "0zjrnc9lshagm6kdb9bdh45dmlnkpwcpyssa896sda93ngbmj8k9";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "WoofWare.DotnetRuntimeLocator";
|
|
||||||
version = "0.1.4";
|
|
||||||
sha256 = "19pp4qlyf18g704ppbcsm1rhjqjpi84py18yljj9nx70331m8bpg";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "WoofWare.Myriad.Plugins";
|
|
||||||
version = "2.1.42";
|
|
||||||
sha256 = "0px46m734gsn1xa97111v1nwkyc2j52bw7z4bjdljzkmzzmnqa91";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "WoofWare.Myriad.Plugins.Attributes";
|
|
||||||
version = "3.1.6";
|
|
||||||
sha256 = "0786pr1p0nq0854mqi2cddmh185j3jihwn6azz9wiy6nxawjbrd2";
|
|
||||||
})
|
|
||||||
(fetchNuGet {
|
|
||||||
pname = "WoofWare.PrattParser";
|
|
||||||
version = "0.1.2";
|
|
||||||
sha256 = "0spypcwsbn805yrs6grjj68ccva902lhkq93mxy32rdply1xs34q";
|
|
||||||
})
|
|
||||||
]
|
|
Reference in New Issue
Block a user